123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467 |
- #include <chrono>
- #include <iostream>
- #include <string>
- #include <thread> // sleep_for
- #include "door.h"
- #include "irc.h"
- #include "yaml-cpp/yaml.h"
- #include <boost/asio.hpp>
- // #include <boost/thread.hpp>
- YAML::Node config;
- std::function<std::ofstream &(void)> get_logger;
- bool file_exists(const std::string name) {
- std::ifstream f(name.c_str());
- return f.good();
- }
- const int ms_input_delay = 50;
- std::string input;
- std::string prompt; // mostly for length to erase/restore properly
- int max_input = 100;
- door::ANSIColor prompt_color{door::COLOR::YELLOW, door::COLOR::BLUE,
- door::ATTR::BOLD};
- door::ANSIColor input_color{door::COLOR::WHITE}; // , door::COLOR::BLUE};
- void erase(door::Door &d, int count) {
- d << door::reset;
- for (int x = 0; x < count; ++x) {
- d << "\x08 \x08";
- }
- }
- void clear_input(door::Door &d) {
- if (prompt.empty())
- return;
- erase(d, input.size());
- erase(d, prompt.size() + 1);
- }
- void restore_input(door::Door &d) {
- if (prompt.empty())
- return;
- d << prompt_color << prompt << input_color << " " << input;
- }
- /*
- commands:
- /h /help /?
- /t /talk /talkto [TARGET]
- /msg [TO] [message]
- /me [message]
- /quit [message, maybe]
- /join [TARGET]
- /part [TARGET]
- future:
- /list ?
- /version ?
- */
- bool check_for_input(door::Door &d, ircClient &irc) {
- int c;
- // return true when we have input and is "valid" // ready
- if (prompt.empty()) {
- // ok, nothing has been displayed at this time.
- if (d.haskey()) {
- // something to do.
- prompt = "[" + irc.talkto() + "]";
- d << prompt_color << prompt << input_color << " ";
- c = d.sleep_key(1);
- if (c < 0) {
- // handle timeout/hangup/out of time
- return false;
- }
- if (c > 0x1000)
- return false;
- if (isprint(c)) {
- d << (char)c;
- input.append(1, c);
- }
- }
- return false;
- } else {
- // continue on with what we have displayed.
- c = d.sleep_ms_key(ms_input_delay);
- if (c != -1) {
- /*
- c = d.sleep_key(1);
- if (c < 0) {
- // handle error
- return false;
- }
- */
- if (c > 0x1000)
- return false;
- if (isprint(c)) {
- d << (char)c;
- input.append(1, c);
- // hot-keys
- if (input[0] == '/') {
- if (input.size() == 2) {
- switch (input[1]) {
- case 'j':
- case 'J':
- erase(d, input.size());
- input = "/join ";
- d << input;
- break;
- case 'p':
- case 'P':
- erase(d, input.size());
- input = "/part ";
- d << input;
- break;
- case 't':
- case 'T':
- erase(d, input.size());
- input = "/talkto ";
- d << input;
- break;
- case 'h':
- case 'H':
- case '?':
- erase(d, input.size());
- input = "/help";
- d << input;
- break;
- case 'q':
- case 'Q':
- erase(d, input.size());
- input = "/quit";
- d << input;
- }
- }
- }
- }
- if ((c == 0x08) or (c == 0x7f)) {
- // hot-keys
- if (input[0] == '/') {
- if ((input == "/help") or (input == "/talkto ") or
- (input == "/join ") or (input == "/part") or (input == "/quit")) {
- erase(d, input.size());
- erase(d, prompt.size());
- input.clear();
- prompt.clear();
- return false;
- }
- }
- if (input.size() > 1) {
- erase(d, 1);
- d << input_color;
- input.erase(input.length() - 1);
- } else {
- // erasing the last character
- erase(d, 1);
- input.clear();
- erase(d, prompt.size() + 1);
- prompt.clear();
- return false;
- }
- }
- if (c == 0x0d) {
- clear_input(d);
- prompt.clear();
- return true;
- }
- }
- return false;
- }
- }
- int main(int argc, char *argv[]) {
- using namespace std::chrono_literals;
- boost::asio::io_context io_context;
- ircClient irc(io_context);
- door::Door door("irc-door", argc, argv);
- get_logger = [&door]() -> ofstream & { return door.log(); };
- if (file_exists("irc-door.yaml")) {
- config = YAML::LoadFile("irc-door.yaml");
- }
- bool update_config = false;
- if (!config["hostname"]) {
- config["hostname"] = "127.0.0.1";
- update_config = true;
- }
- if (!config["port"]) {
- config["port"] = "6697";
- update_config = true;
- }
- if (!config["autojoin"]) {
- config["autojoin"] = "#bugz";
- update_config = true;
- }
- if (!config["realname"]) {
- config["realname"] = "A poor soul on BZBZ BBS...";
- update_config = true;
- }
- if (!config["username"]) {
- config["username"] = "bzbz";
- update_config = true;
- }
- if (update_config) {
- std::ofstream fout("irc-door.yaml");
- fout << config << std::endl;
- }
- // configure
- irc.nick = door.handle;
- irc.realname = config["realname"].as<std::string>();
- irc.hostname = config["hostname"].as<std::string>();
- irc.port = config["port"].as<std::string>();
- irc.username = config["username"].as<std::string>();
- irc.autojoin = config["autojoin"].as<std::string>();
- irc.debug_output = "irc.log";
- irc.begin(); // start the initial request so io_context has work to do
- // boost::thread thread(boost::bind(&boost::asio::io_service::run,
- // &io_context));
- // thread Thread(boost::bind(&boost::asio::io_service::run, &io_context));
- thread Thread([&io_context]() -> void { io_context.run(); });
- door << "Welcome to the IRC chat door." << door::nl;
- // main "loop" -- not the proper way to do it.
- bool in_door = true;
- while (in_door) {
- // the main loop
- // custom input routine goes here
- if (check_for_input(door, irc)) {
- // yes, we have something
- if (input[0] == '/') {
- // command given
- std::vector<std::string> cmd = split_limit(input, 3);
- if (cmd[0] == "/quit") {
- irc.write("QUIT");
- }
- if (cmd[0] == "/talkto") {
- irc.talkto(cmd[1]);
- door << "[talkto = " << cmd[1] << "]" << door::nl;
- }
- if (cmd[0] == "/join") {
- std::string tmp = "JOIN " + cmd[1];
- irc.write(tmp);
- }
- if (cmd[0] == "/part") {
- std::string tmp = "PART " + cmd[1];
- irc.write(tmp);
- }
- if (cmd[0] == "/me") {
- cmd = split_limit(input, 2);
- std::string tmp = "PRIVMSG " + irc.talkto() + " :\x01" + "ACTION " +
- cmd[1] + "\x01";
- irc.write(tmp);
- door << "* " << irc.nick << " " << cmd[1] << door::nl;
- }
- if (cmd[0] == "/info") {
- irc.channels_lock.lock();
- for (auto c : irc.channels) {
- door << "CH " << c.first << " ";
- for (auto s : c.second) {
- door << s << " ";
- }
- door << door::nl;
- }
- irc.channels_lock.unlock();
- }
- /*
- if (std::toupper(input[1]) == 'Q') {
- irc.write("QUIT");
- } else {
- // for now, just output whatever they gave us.
- input.erase(0, 1);
- irc.write(input);
- }
- */
- } else {
- std::string target = irc.talkto();
- std::string output = "PRIVMSG " + target + " :" + input;
- // I need to display something here to show we've said something (and
- // where we've said it)
- door::ANSIColor nick_color{door::COLOR::WHITE, door::COLOR::BLUE};
- if (target[0] == '#') {
- door::ANSIColor channel_color = door::ANSIColor{
- door::COLOR::YELLOW, door::COLOR::BLUE, door::ATTR::BOLD};
- door << nick_color << irc.nick << door::ANSIColor(door::COLOR::CYAN)
- << "/" << channel_color << target << door::reset << " " << input
- << door::nl;
- } else {
- door << nick_color << irc.nick << "/" << target << door::reset << " "
- << input << door::nl;
- }
- irc.write(output);
- };
- input.clear();
- }
- /*
- if (door.haskey()) {
- door << ">> ";
- std::string input = door.input_string(100);
- door << door::nl;
- if (!input.empty()) {
- if (input == "/q") {
- irc.write("QUIT :What other doors on BZ&BZ BBS work...");
- in_door = false;
- }
- if (input[0] == '/') {
- input.erase(0, 1);
- irc.write(input);
- } else {
- if (irc.talkto.empty()) {
- door << " talkto is empty? Whaat?" << door::nl;
- } else {
- std::string output = "PRIVMSG " + irc.talkto + " :" + input;
- irc.write(output);
- }
- }
- }
- }
- */
- boost::optional<std::vector<std::string>> msg;
- // hold list of users -- until end names received.
- // std::vector<std::string> names;
- bool input_cleared = false;
- do {
- msg = irc.buffer_maybe_pop();
- if (msg) {
- if (!input_cleared) {
- input_cleared = true;
- clear_input(door);
- }
- std::vector<std::string> m = *msg;
- if (m.size() == 1) {
- // system message
- door << "(" << m[0] << ")" << door::nl;
- continue;
- }
- std::string cmd = m[1];
- if (cmd == "366") {
- // end of names, output and clear
- std::string channel = split_limit(m[3], 2)[0];
- irc.channels_lock.lock();
- door << "* users on " << channel << " : ";
- for (auto name : irc.channels[channel]) {
- door << name << " ";
- }
- irc.channels_lock.unlock();
- door << door::nl;
- // names.clear();
- }
- if (cmd == "372") {
- // MOTD
- std::string temp = m[3];
- temp.erase(0, 1);
- door << "* " << temp << door::nl;
- }
- // 400 and 500 are errors? should show those.
- if ((cmd[0] == '4') or (cmd[0] == '5')) {
- std::string tmp = m[3];
- tmp.erase(0, 1);
- door << "* " << tmp << door::nl;
- }
- if (cmd == "NOTICE") {
- std::string tmp = m[3];
- tmp.erase(0, 1);
- door << parse_nick(m[0]) << " NOTICE " << tmp << door::nl;
- }
- if (cmd == "ACTION") {
- if (m[2][0] == '#') {
- door << "* " << parse_nick(m[0]) << "/" << m[2] << " " << m[3]
- << door::nl;
- } else {
- door << "* " << parse_nick(m[0]) << " " << m[3] << door::nl;
- }
- }
- if (cmd == "TOPIC") {
- std::string tmp = m[3];
- tmp.erase(0, 1);
- door << parse_nick(m[0]) << " set topic of " << m[2] << " to " << tmp
- << door::nl;
- }
- if (cmd == "PRIVMSG") {
- door::ANSIColor nick_color{door::COLOR::WHITE, door::COLOR::BLUE};
- if (m[2][0] == '#') {
- std::string tmp = m[3];
- tmp.erase(0, 1);
- door::ANSIColor channel_color{door::COLOR::WHITE,
- door::COLOR::BLUE};
- if (m[2] == irc.talkto()) {
- channel_color = door::ANSIColor{
- door::COLOR::YELLOW, door::COLOR::BLUE, door::ATTR::BOLD};
- }
- door << nick_color << parse_nick(m[0])
- << door::ANSIColor(door::COLOR::CYAN) << "/" << channel_color
- << m[2] << door::reset << " " << tmp << door::nl;
- } else {
- std::string tmp = m[3];
- tmp.erase(0, 1);
- door << nick_color << parse_nick(m[0]) << door::reset << " " << tmp
- << door::nl;
- }
- }
- }
- } while (msg);
- if (input_cleared)
- restore_input(door);
- // std::this_thread::sleep_for(200ms);
- if (irc.shutdown)
- in_door = false;
- }
- io_context.stop();
- Thread.join();
- door << "Returning to the BBS..." << door::nl;
- // std::this_thread::sleep_for(2s);
- return 0;
- }
|