main.cpp 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169
  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 << std::endl;
  67. fout << config << std::endl;
  68. fout << "# end yaml config" << std::endl;
  69. }
  70. // configure
  71. irc.nick = door.handle;
  72. irc.realname = config["realname"].as<std::string>();
  73. irc.hostname = config["hostname"].as<std::string>();
  74. irc.port = config["port"].as<std::string>();
  75. irc.username = config["username"].as<std::string>();
  76. irc.autojoin = config["autojoin"].as<std::string>();
  77. // set the delay between irc updates
  78. ms_input_delay = config["input_delay"].as<int>();
  79. timestamp_format = config["timestamp_format"].as<std::string>();
  80. if (config["log"]) {
  81. irc.debug_output = config["log"].as<std::string>();
  82. door << "irc debug logfile = " << config["log"].as<std::string>()
  83. << door::nl;
  84. }
  85. if (config["allow_join"].as<int>() == 1) {
  86. allow_part = true;
  87. allow_join = true;
  88. }
  89. irc.begin(); // start the initial request so io_context has work to do
  90. // boost::thread thread(boost::bind(&boost::asio::io_service::run,
  91. // &io_context));
  92. // thread Thread(boost::bind(&boost::asio::io_service::run, &io_context));
  93. thread Thread([&io_context]() -> void { io_context.run(); });
  94. door << "Welcome to the IRC chat door." << door::nl;
  95. bool in_door = true;
  96. while (in_door) {
  97. // the main loop
  98. // custom input routine goes here
  99. check_for_input(door, irc);
  100. boost::optional<message_stamp> msg;
  101. bool input_cleared = false;
  102. do {
  103. msg = irc.message_pop();
  104. if (msg) {
  105. if (!input_cleared) {
  106. input_cleared = true;
  107. clear_input(door);
  108. }
  109. render(*msg, door, irc);
  110. }
  111. } while (msg);
  112. if (input_cleared)
  113. restore_input(door);
  114. // sleep is done in the check_for_input
  115. // std::this_thread::sleep_for(200ms);
  116. if (irc.shutdown)
  117. in_door = false;
  118. }
  119. // Store error messages into door log!
  120. while (!irc.errors.empty()) {
  121. door.log() << "ERROR: " << irc.errors.front() << std::endl;
  122. irc.errors.erase(irc.errors.begin());
  123. }
  124. io_context.stop();
  125. Thread.join();
  126. // disable the global logging std::function
  127. get_logger = nullptr;
  128. door << "Returning to the BBS..." << door::nl;
  129. return 0;
  130. }