main.cpp 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171
  1. #include <chrono>
  2. #include <iostream>
  3. #include <string>
  4. #include <thread> // sleep_for
  5. #include "door.h"
  6. #include "input.h"
  7. #include "irc.h"
  8. #include "render.h"
  9. #include "yaml-cpp/yaml.h"
  10. #include <boost/asio.hpp>
  11. // #include <boost/thread.hpp>
  12. YAML::Node config;
  13. std::function<std::ofstream &(void)> get_logger;
  14. bool file_exists(const std::string name) {
  15. std::ifstream f(name.c_str());
  16. return f.good();
  17. }
  18. int main(int argc, char *argv[]) {
  19. using namespace std::chrono_literals;
  20. boost::asio::io_context io_context;
  21. ircClient irc(io_context);
  22. door::Door door("irc-door", argc, argv);
  23. get_logger = [&door]() -> ofstream & { return door.log(); };
  24. if (file_exists("irc-door.yaml")) {
  25. config = YAML::LoadFile("irc-door.yaml");
  26. }
  27. bool update_config = false;
  28. if (!config["hostname"]) {
  29. config["hostname"] = "127.0.0.1";
  30. update_config = true;
  31. }
  32. if (!config["port"]) {
  33. config["port"] = "6697";
  34. update_config = true;
  35. }
  36. if (!config["allow_join"]) {
  37. config["allow_join"] = "0";
  38. update_config = true;
  39. }
  40. if (!config["autojoin"]) {
  41. config["autojoin"] = "#bugz";
  42. update_config = true;
  43. }
  44. if (!config["realname"]) {
  45. config["realname"] = "A poor soul on BZBZ BBS...";
  46. update_config = true;
  47. }
  48. if (!config["username"]) {
  49. config["username"] = "bzbz";
  50. update_config = true;
  51. }
  52. if (!config["input_delay"]) {
  53. config["input_delay"] = "500";
  54. update_config = true;
  55. }
  56. if (!config["timestamp_format"]) {
  57. config["timestamp_format"] = "%T";
  58. update_config = true;
  59. }
  60. if (update_config) {
  61. std::ofstream fout("irc-door.yaml");
  62. fout << "# IRC Chat Door configuration" << std::endl;
  63. fout << "# to add comments (that don't get destroyed)" << std::endl;
  64. fout << "# Add comments as key: values, like:" << std::endl;
  65. fout << "# _comment: This will survive the test of time." << std::endl;
  66. fout << "# %r AM/PM, %T 24 hour time" << std::endl;
  67. fout << std::endl;
  68. fout << config << std::endl;
  69. fout << "# end yaml config" << std::endl;
  70. }
  71. // configure
  72. irc.nick = door.handle;
  73. irc.realname = config["realname"].as<std::string>();
  74. irc.hostname = config["hostname"].as<std::string>();
  75. irc.port = config["port"].as<std::string>();
  76. irc.username = config["username"].as<std::string>();
  77. irc.autojoin = config["autojoin"].as<std::string>();
  78. // set the delay between irc updates
  79. ms_input_delay = config["input_delay"].as<int>();
  80. timestamp_format = config["timestamp_format"].as<std::string>();
  81. if (config["log"]) {
  82. irc.debug_output = config["log"].as<std::string>();
  83. door << "irc debug logfile = " << config["log"].as<std::string>()
  84. << door::nl;
  85. }
  86. if (config["allow_join"].as<int>() == 1) {
  87. allow_part = true;
  88. allow_join = true;
  89. }
  90. irc.begin(); // start the initial request so io_context has work to do
  91. // boost::thread thread(boost::bind(&boost::asio::io_service::run,
  92. // &io_context));
  93. // thread Thread(boost::bind(&boost::asio::io_service::run, &io_context));
  94. thread Thread([&io_context]() -> void { io_context.run(); });
  95. door << "Welcome to the IRC chat door." << door::nl;
  96. bool in_door = true;
  97. while (in_door) {
  98. // the main loop
  99. // custom input routine goes here
  100. check_for_input(door, irc);
  101. boost::optional<message_stamp> msg;
  102. bool input_cleared = false;
  103. if (irc.channels_updated)
  104. do {
  105. msg = irc.message_pop();
  106. if (msg) {
  107. if (!input_cleared) {
  108. input_cleared = true;
  109. clear_input(door);
  110. }
  111. render(*msg, door, irc);
  112. }
  113. } while (msg);
  114. if (input_cleared)
  115. restore_input(door);
  116. // sleep is done in the check_for_input
  117. // std::this_thread::sleep_for(200ms);
  118. if (irc.shutdown)
  119. in_door = false;
  120. }
  121. // Store error messages into door log!
  122. while (!irc.errors.empty()) {
  123. door.log() << "ERROR: " << irc.errors.front() << std::endl;
  124. irc.errors.erase(irc.errors.begin());
  125. }
  126. io_context.stop();
  127. Thread.join();
  128. // disable the global logging std::function
  129. get_logger = nullptr;
  130. door << "Returning to the BBS..." << door::nl;
  131. return 0;
  132. }