main.cpp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467
  1. #include <chrono>
  2. #include <iostream>
  3. #include <string>
  4. #include <thread> // sleep_for
  5. #include "door.h"
  6. #include "irc.h"
  7. #include "yaml-cpp/yaml.h"
  8. #include <boost/asio.hpp>
  9. // #include <boost/thread.hpp>
  10. YAML::Node config;
  11. std::function<std::ofstream &(void)> get_logger;
  12. bool file_exists(const std::string name) {
  13. std::ifstream f(name.c_str());
  14. return f.good();
  15. }
  16. const int ms_input_delay = 50;
  17. std::string input;
  18. std::string prompt; // mostly for length to erase/restore properly
  19. int max_input = 100;
  20. door::ANSIColor prompt_color{door::COLOR::YELLOW, door::COLOR::BLUE,
  21. door::ATTR::BOLD};
  22. door::ANSIColor input_color{door::COLOR::WHITE}; // , door::COLOR::BLUE};
  23. void erase(door::Door &d, int count) {
  24. d << door::reset;
  25. for (int x = 0; x < count; ++x) {
  26. d << "\x08 \x08";
  27. }
  28. }
  29. void clear_input(door::Door &d) {
  30. if (prompt.empty())
  31. return;
  32. erase(d, input.size());
  33. erase(d, prompt.size() + 1);
  34. }
  35. void restore_input(door::Door &d) {
  36. if (prompt.empty())
  37. return;
  38. d << prompt_color << prompt << input_color << " " << input;
  39. }
  40. /*
  41. commands:
  42. /h /help /?
  43. /t /talk /talkto [TARGET]
  44. /msg [TO] [message]
  45. /me [message]
  46. /quit [message, maybe]
  47. /join [TARGET]
  48. /part [TARGET]
  49. future:
  50. /list ?
  51. /version ?
  52. */
  53. bool check_for_input(door::Door &d, ircClient &irc) {
  54. int c;
  55. // return true when we have input and is "valid" // ready
  56. if (prompt.empty()) {
  57. // ok, nothing has been displayed at this time.
  58. if (d.haskey()) {
  59. // something to do.
  60. prompt = "[" + irc.talkto() + "]";
  61. d << prompt_color << prompt << input_color << " ";
  62. c = d.sleep_key(1);
  63. if (c < 0) {
  64. // handle timeout/hangup/out of time
  65. return false;
  66. }
  67. if (c > 0x1000)
  68. return false;
  69. if (isprint(c)) {
  70. d << (char)c;
  71. input.append(1, c);
  72. }
  73. }
  74. return false;
  75. } else {
  76. // continue on with what we have displayed.
  77. c = d.sleep_ms_key(ms_input_delay);
  78. if (c != -1) {
  79. /*
  80. c = d.sleep_key(1);
  81. if (c < 0) {
  82. // handle error
  83. return false;
  84. }
  85. */
  86. if (c > 0x1000)
  87. return false;
  88. if (isprint(c)) {
  89. d << (char)c;
  90. input.append(1, c);
  91. // hot-keys
  92. if (input[0] == '/') {
  93. if (input.size() == 2) {
  94. switch (input[1]) {
  95. case 'j':
  96. case 'J':
  97. erase(d, input.size());
  98. input = "/join ";
  99. d << input;
  100. break;
  101. case 'p':
  102. case 'P':
  103. erase(d, input.size());
  104. input = "/part ";
  105. d << input;
  106. break;
  107. case 't':
  108. case 'T':
  109. erase(d, input.size());
  110. input = "/talkto ";
  111. d << input;
  112. break;
  113. case 'h':
  114. case 'H':
  115. case '?':
  116. erase(d, input.size());
  117. input = "/help";
  118. d << input;
  119. break;
  120. case 'q':
  121. case 'Q':
  122. erase(d, input.size());
  123. input = "/quit";
  124. d << input;
  125. }
  126. }
  127. }
  128. }
  129. if ((c == 0x08) or (c == 0x7f)) {
  130. // hot-keys
  131. if (input[0] == '/') {
  132. if ((input == "/help") or (input == "/talkto ") or
  133. (input == "/join ") or (input == "/part") or (input == "/quit")) {
  134. erase(d, input.size());
  135. erase(d, prompt.size());
  136. input.clear();
  137. prompt.clear();
  138. return false;
  139. }
  140. }
  141. if (input.size() > 1) {
  142. erase(d, 1);
  143. d << input_color;
  144. input.erase(input.length() - 1);
  145. } else {
  146. // erasing the last character
  147. erase(d, 1);
  148. input.clear();
  149. erase(d, prompt.size() + 1);
  150. prompt.clear();
  151. return false;
  152. }
  153. }
  154. if (c == 0x0d) {
  155. clear_input(d);
  156. prompt.clear();
  157. return true;
  158. }
  159. }
  160. return false;
  161. }
  162. }
  163. int main(int argc, char *argv[]) {
  164. using namespace std::chrono_literals;
  165. boost::asio::io_context io_context;
  166. ircClient irc(io_context);
  167. door::Door door("irc-door", argc, argv);
  168. get_logger = [&door]() -> ofstream & { return door.log(); };
  169. if (file_exists("irc-door.yaml")) {
  170. config = YAML::LoadFile("irc-door.yaml");
  171. }
  172. bool update_config = false;
  173. if (!config["hostname"]) {
  174. config["hostname"] = "127.0.0.1";
  175. update_config = true;
  176. }
  177. if (!config["port"]) {
  178. config["port"] = "6697";
  179. update_config = true;
  180. }
  181. if (!config["autojoin"]) {
  182. config["autojoin"] = "#bugz";
  183. update_config = true;
  184. }
  185. if (!config["realname"]) {
  186. config["realname"] = "A poor soul on BZBZ BBS...";
  187. update_config = true;
  188. }
  189. if (!config["username"]) {
  190. config["username"] = "bzbz";
  191. update_config = true;
  192. }
  193. if (update_config) {
  194. std::ofstream fout("irc-door.yaml");
  195. fout << config << std::endl;
  196. }
  197. // configure
  198. irc.nick = door.handle;
  199. irc.realname = config["realname"].as<std::string>();
  200. irc.hostname = config["hostname"].as<std::string>();
  201. irc.port = config["port"].as<std::string>();
  202. irc.username = config["username"].as<std::string>();
  203. irc.autojoin = config["autojoin"].as<std::string>();
  204. irc.debug_output = "irc.log";
  205. irc.begin(); // start the initial request so io_context has work to do
  206. // boost::thread thread(boost::bind(&boost::asio::io_service::run,
  207. // &io_context));
  208. // thread Thread(boost::bind(&boost::asio::io_service::run, &io_context));
  209. thread Thread([&io_context]() -> void { io_context.run(); });
  210. door << "Welcome to the IRC chat door." << door::nl;
  211. // main "loop" -- not the proper way to do it.
  212. bool in_door = true;
  213. while (in_door) {
  214. // the main loop
  215. // custom input routine goes here
  216. if (check_for_input(door, irc)) {
  217. // yes, we have something
  218. if (input[0] == '/') {
  219. // command given
  220. std::vector<std::string> cmd = split_limit(input, 3);
  221. if (cmd[0] == "/quit") {
  222. irc.write("QUIT");
  223. }
  224. if (cmd[0] == "/talkto") {
  225. irc.talkto(cmd[1]);
  226. door << "[talkto = " << cmd[1] << "]" << door::nl;
  227. }
  228. if (cmd[0] == "/join") {
  229. std::string tmp = "JOIN " + cmd[1];
  230. irc.write(tmp);
  231. }
  232. if (cmd[0] == "/part") {
  233. std::string tmp = "PART " + cmd[1];
  234. irc.write(tmp);
  235. }
  236. if (cmd[0] == "/me") {
  237. cmd = split_limit(input, 2);
  238. std::string tmp = "PRIVMSG " + irc.talkto() + " :\x01" + "ACTION " +
  239. cmd[1] + "\x01";
  240. irc.write(tmp);
  241. door << "* " << irc.nick << " " << cmd[1] << door::nl;
  242. }
  243. if (cmd[0] == "/info") {
  244. irc.channels_lock.lock();
  245. for (auto c : irc.channels) {
  246. door << "CH " << c.first << " ";
  247. for (auto s : c.second) {
  248. door << s << " ";
  249. }
  250. door << door::nl;
  251. }
  252. irc.channels_lock.unlock();
  253. }
  254. /*
  255. if (std::toupper(input[1]) == 'Q') {
  256. irc.write("QUIT");
  257. } else {
  258. // for now, just output whatever they gave us.
  259. input.erase(0, 1);
  260. irc.write(input);
  261. }
  262. */
  263. } else {
  264. std::string target = irc.talkto();
  265. std::string output = "PRIVMSG " + target + " :" + input;
  266. // I need to display something here to show we've said something (and
  267. // where we've said it)
  268. door::ANSIColor nick_color{door::COLOR::WHITE, door::COLOR::BLUE};
  269. if (target[0] == '#') {
  270. door::ANSIColor channel_color = door::ANSIColor{
  271. door::COLOR::YELLOW, door::COLOR::BLUE, door::ATTR::BOLD};
  272. door << nick_color << irc.nick << door::ANSIColor(door::COLOR::CYAN)
  273. << "/" << channel_color << target << door::reset << " " << input
  274. << door::nl;
  275. } else {
  276. door << nick_color << irc.nick << "/" << target << door::reset << " "
  277. << input << door::nl;
  278. }
  279. irc.write(output);
  280. };
  281. input.clear();
  282. }
  283. /*
  284. if (door.haskey()) {
  285. door << ">> ";
  286. std::string input = door.input_string(100);
  287. door << door::nl;
  288. if (!input.empty()) {
  289. if (input == "/q") {
  290. irc.write("QUIT :What other doors on BZ&BZ BBS work...");
  291. in_door = false;
  292. }
  293. if (input[0] == '/') {
  294. input.erase(0, 1);
  295. irc.write(input);
  296. } else {
  297. if (irc.talkto.empty()) {
  298. door << " talkto is empty? Whaat?" << door::nl;
  299. } else {
  300. std::string output = "PRIVMSG " + irc.talkto + " :" + input;
  301. irc.write(output);
  302. }
  303. }
  304. }
  305. }
  306. */
  307. boost::optional<std::vector<std::string>> msg;
  308. // hold list of users -- until end names received.
  309. // std::vector<std::string> names;
  310. bool input_cleared = false;
  311. do {
  312. msg = irc.buffer_maybe_pop();
  313. if (msg) {
  314. if (!input_cleared) {
  315. input_cleared = true;
  316. clear_input(door);
  317. }
  318. std::vector<std::string> m = *msg;
  319. if (m.size() == 1) {
  320. // system message
  321. door << "(" << m[0] << ")" << door::nl;
  322. continue;
  323. }
  324. std::string cmd = m[1];
  325. if (cmd == "366") {
  326. // end of names, output and clear
  327. std::string channel = split_limit(m[3], 2)[0];
  328. irc.channels_lock.lock();
  329. door << "* users on " << channel << " : ";
  330. for (auto name : irc.channels[channel]) {
  331. door << name << " ";
  332. }
  333. irc.channels_lock.unlock();
  334. door << door::nl;
  335. // names.clear();
  336. }
  337. if (cmd == "372") {
  338. // MOTD
  339. std::string temp = m[3];
  340. temp.erase(0, 1);
  341. door << "* " << temp << door::nl;
  342. }
  343. // 400 and 500 are errors? should show those.
  344. if ((cmd[0] == '4') or (cmd[0] == '5')) {
  345. std::string tmp = m[3];
  346. tmp.erase(0, 1);
  347. door << "* " << tmp << door::nl;
  348. }
  349. if (cmd == "NOTICE") {
  350. std::string tmp = m[3];
  351. tmp.erase(0, 1);
  352. door << parse_nick(m[0]) << " NOTICE " << tmp << door::nl;
  353. }
  354. if (cmd == "ACTION") {
  355. if (m[2][0] == '#') {
  356. door << "* " << parse_nick(m[0]) << "/" << m[2] << " " << m[3]
  357. << door::nl;
  358. } else {
  359. door << "* " << parse_nick(m[0]) << " " << m[3] << door::nl;
  360. }
  361. }
  362. if (cmd == "TOPIC") {
  363. std::string tmp = m[3];
  364. tmp.erase(0, 1);
  365. door << parse_nick(m[0]) << " set topic of " << m[2] << " to " << tmp
  366. << door::nl;
  367. }
  368. if (cmd == "PRIVMSG") {
  369. door::ANSIColor nick_color{door::COLOR::WHITE, door::COLOR::BLUE};
  370. if (m[2][0] == '#') {
  371. std::string tmp = m[3];
  372. tmp.erase(0, 1);
  373. door::ANSIColor channel_color{door::COLOR::WHITE,
  374. door::COLOR::BLUE};
  375. if (m[2] == irc.talkto()) {
  376. channel_color = door::ANSIColor{
  377. door::COLOR::YELLOW, door::COLOR::BLUE, door::ATTR::BOLD};
  378. }
  379. door << nick_color << parse_nick(m[0])
  380. << door::ANSIColor(door::COLOR::CYAN) << "/" << channel_color
  381. << m[2] << door::reset << " " << tmp << door::nl;
  382. } else {
  383. std::string tmp = m[3];
  384. tmp.erase(0, 1);
  385. door << nick_color << parse_nick(m[0]) << door::reset << " " << tmp
  386. << door::nl;
  387. }
  388. }
  389. }
  390. } while (msg);
  391. if (input_cleared)
  392. restore_input(door);
  393. // std::this_thread::sleep_for(200ms);
  394. if (irc.shutdown)
  395. in_door = false;
  396. }
  397. io_context.stop();
  398. Thread.join();
  399. door << "Returning to the BBS..." << door::nl;
  400. // std::this_thread::sleep_for(2s);
  401. return 0;
  402. }