director.cpp 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336
  1. #include "director.h"
  2. #include <boost/format.hpp>
  3. #include "boxes.h"
  4. #include "galaxy.h"
  5. #include "logging.h"
  6. #include "utils.h"
  7. Director::Director() {
  8. BUGZ_LOG(warning) << "Director::Director()";
  9. // active = false;
  10. game = 0; // not in a game
  11. // do everything proxy_deactivate does ...
  12. // proxy_deactivate();
  13. active = false;
  14. // reset everything back to good state
  15. talk_direct = true;
  16. show_client = true;
  17. count = 0;
  18. }
  19. Director::~Director() { BUGZ_LOG(warning) << "Director::~Director()"; }
  20. void Director::client_input(const std::string &input) {
  21. // If we're already active, don't try to activate.
  22. if (chain) {
  23. chain->client_input(input);
  24. return;
  25. }
  26. if (active) {
  27. if (input == "Q" || input == "q") proxy_deactivate();
  28. return;
  29. } else if (input == "\x1b" || input == "~") {
  30. std::string &prompt = current_prompt;
  31. BUGZ_LOG(trace) << "CI: ACTIVATE prompt shows: [" << prompt << "]";
  32. if (prompt == "Selection (? for menu): ") {
  33. to_client(
  34. "\n\rThere's not much we can do here. Activate in-game at a "
  35. "Command prompt.\n\r");
  36. to_client(current_raw_prompt);
  37. return;
  38. }
  39. // easter-eggs:
  40. if (prompt == "Enter your choice: ") {
  41. to_client(
  42. "\n\r\x1b[1;36mI'd choose \x1b[1;37m`T`\x1b[1;36m, but "
  43. "that's how I was coded.\n\r");
  44. to_client(current_raw_prompt);
  45. return;
  46. }
  47. // easter-egg
  48. if (prompt == "[Pause]") {
  49. to_client(" \x1b[1;36mPAWS\x1b[0m\n\r");
  50. to_client(current_raw_prompt);
  51. return;
  52. }
  53. if (prompt == "Planet command (?=help) [D] ") {
  54. // future: Activate at planet menu ?
  55. return;
  56. }
  57. //
  58. // The command prompt that we're looking for:
  59. //
  60. // "Command [TL=00:00:00]:[242] (?=Help)? : "
  61. // the time, and the sector number vary...
  62. if (prompt.substr(0, 9) == "Command [") {
  63. int len = prompt.length();
  64. if (prompt.substr(len - 14) == "] (?=Help)? : ") {
  65. proxy_activate();
  66. /*
  67. to_client("\n\r\x1b[1;34mWELCOME! This is where the proxy would "
  68. "activate.\n\r");
  69. // active = true;
  70. // show_client = true; // because if something comes (unexpected)
  71. // from the server? talk_direct = false;
  72. // but we aren't activating (NNY)
  73. to_client(get_prompt());
  74. */
  75. return;
  76. }
  77. }
  78. }
  79. // Ok...
  80. if (talk_direct) to_server(input);
  81. /*
  82. if (emit_client_input)
  83. emit_client_input(line);
  84. */
  85. }
  86. void Director::server_line(const std::string &line, const std::string &raw_line) {
  87. // check for if we entered game/left game
  88. if (line.find("TradeWars Game Server ") != std::string::npos) {
  89. to_client("\rTradeWars Proxy v2++ READY (~ or ESC to activate)\n\r");
  90. game = 0;
  91. // reset "active game" -- we're at the TWGS main menu
  92. }
  93. if (line.find("Selection (? for menu): ") != std::string::npos) {
  94. char ch = line[line.length() - 1];
  95. if (ch >= 'A' && ch < 'Q') {
  96. game = ch;
  97. BUGZ_LOG(warning) << "GAME " << game << " activated!";
  98. }
  99. // not needed (handled by above Game Server check).
  100. if (ch == 'Q') game = 0;
  101. }
  102. if (game) {
  103. // in-game parsing here.
  104. }
  105. /*
  106. if (emit_server_line) {
  107. emit_server_line(line);
  108. }
  109. */
  110. if (chain) {
  111. chain->server_line(line, raw_line);
  112. }
  113. }
  114. void Director::server_prompt(const std::string &prompt,
  115. const std::string &raw_prompt) {
  116. current_prompt = prompt;
  117. current_raw_prompt = raw_prompt;
  118. /*
  119. if (emit_server_prompt)
  120. emit_server_prompt(prompt);
  121. */
  122. if (chain) chain->server_prompt(prompt);
  123. }
  124. void Director::proxy_activate(void) {
  125. active = true; // yes, set keep-alive timer.
  126. to_server(" "); // start keep-alive timer.
  127. // set other values we need
  128. talk_direct = false;
  129. show_client = false;
  130. /*
  131. Wait a minute .. this might be confusing.
  132. Shouldn't I send them the current prompt?
  133. Just in case we abort in the middle of something?!?
  134. */
  135. old_prompt = current_prompt;
  136. old_raw_prompt = current_raw_prompt;
  137. to_client("\x1b[0m\n\r");
  138. /*
  139. ╔══════════════════════════════╗
  140. ║ TradeWars Proxy Active ║
  141. ╚══════════════════════════════╝
  142. -=>
  143. */
  144. Boxes box(30, 1, true);
  145. box.boxcolor = "\x1b[1;33;44m";
  146. box.textcolor = "\x1b[1;33;44m";
  147. to_client(box.top());
  148. std::string output = " TradeWars Proxy \x1b[5mActive\x1b[0;1;33;44m ";
  149. to_client(box.row(output));
  150. to_client(box.bottom());
  151. std::shared_ptr<Dispatch> menu = std::make_shared<MenuDispatch>(*this);
  152. chain = menu;
  153. MenuDispatch *md = static_cast<MenuDispatch *>(&(*menu));
  154. md->menu_box_color = "\x1b[1;33;44m";
  155. md->menu_text_color = "\x1b[1;37;44m";
  156. md->menu_title = "Proxy Menu";
  157. md->menu_options_color = "\x1b[1;36;40m";
  158. md->menu_prompt =
  159. "\x1b[0;31;40m\xdb\xb2\xb1\xb0 \x1b[31;40mRED "
  160. "\x1b[32;40mGREEN\x1b[30;42m\xdb\xb2\xb1\xb0 \x1b[0m : ";
  161. md->lazy = true;
  162. md->menu = {{"A", "Apple"}, {"B", "Blue"}, {"R", "Rabbit"}, {"Z", "ZOOO!"}};
  163. md->setNotify([this]() { this->menu_choice(); });
  164. menu->activate();
  165. /*
  166. // Using InputDispatch -- and see have_input
  167. std::shared_ptr<Dispatch> readline = std::make_shared<InputDispatch>(*this);
  168. chain = readline;
  169. InputDispatch *id = static_cast<InputDispatch *>(&(*readline));
  170. id->prompt = "\x1b[0m \x1b[1;33;44m-=>\x1b[0m \x1b[1;37;44m";
  171. id->max_length = 15;
  172. id->setNotify([this]() { this->have_input(); });
  173. readline->activate();
  174. */
  175. }
  176. void Director::menu_choice(void) {
  177. MenuDispatch * md = dynamic_cast<MenuDispatch *>(&(*chain));
  178. if (md) {
  179. if (md->input.empty()) {
  180. to_client("Menu aborted.\n\r");
  181. proxy_deactivate();
  182. return;
  183. } else {
  184. std::string text = str(boost::format( "Back from Menu [%1%] was selected.\n\r") % md->input);
  185. to_client(text);
  186. md->activate();
  187. }
  188. }
  189. }
  190. void Director::have_input(void) {
  191. ++count;
  192. InputDispatch *id = dynamic_cast<InputDispatch *>(&(*chain));
  193. if (id) {
  194. std::string output =
  195. str(boost::format("Your Input (%2%): [%1%]\n\r") % id->input % count);
  196. to_client("\x1b[0m");
  197. to_client(output);
  198. } else {
  199. proxy_deactivate();
  200. return;
  201. }
  202. if (count > 3) {
  203. proxy_deactivate();
  204. } else {
  205. chain->activate();
  206. }
  207. }
  208. void Director::proxy_deactivate(void) {
  209. active = false;
  210. // reset everything back to good state
  211. talk_direct = true;
  212. show_client = true;
  213. /*
  214. current_prompt = old_prompt;
  215. current_raw_prompt = old_raw_prompt;
  216. */
  217. chain.reset();
  218. to_client("\n\r");
  219. to_client(current_raw_prompt);
  220. }
  221. void Director::SL_cimline(const std::string &line) {
  222. if (line == ": ENDINTERROG") {
  223. SL_parser = nullptr;
  224. return;
  225. }
  226. if (line == ": ") {
  227. // do I need to do anything special here for this?
  228. return;
  229. }
  230. if (line.empty()) {
  231. SL_parser = nullptr;
  232. return;
  233. }
  234. // parse cimline
  235. size_t pos = line.find('%');
  236. std::string work = line;
  237. if (pos == line.npos) {
  238. // warpcim
  239. BUGZ_LOG(fatal) << "warpcim: [" << line << "]";
  240. auto warps = split(line);
  241. sector_warps sw;
  242. for (auto const &w : warps) {
  243. if (sw.sector == 0) {
  244. sw.sector = stoi(w);
  245. } else {
  246. sw.add(stoi(w));
  247. }
  248. }
  249. BUGZ_LOG(fatal) << "warpcim: " << sw;
  250. } else {
  251. // portcim
  252. struct port p = parse_portcim(line);
  253. if (p.sector == 0)
  254. BUGZ_LOG(fatal) << "portcim: [" << line << "]";
  255. else
  256. BUGZ_LOG(fatal) << "portcim: " << p;
  257. }
  258. }
  259. void Director::SL_thiefline(const std::string &line) {
  260. size_t pos = line.find("Suddenly you're Busted!");
  261. bool busted = pos != line.npos;
  262. if (busted) {
  263. BUGZ_LOG(fatal) << "set bust";
  264. SL_parser = nullptr;
  265. } else {
  266. pos = line.find("(You realize the guards saw you last time!)");
  267. if (pos != line.npos) SL_parser = nullptr;
  268. }
  269. // Are those the two ways to exit from this state?
  270. }
  271. void Director::SL_sectorline(const std::string &line) {
  272. BUGZ_LOG(fatal) << "sectorline: [" << line << "]";
  273. }
  274. void Director::SL_portline(const std::string &line) {
  275. if (line.empty()) {
  276. SL_parser = nullptr;
  277. return;
  278. }
  279. BUGZ_LOG(info) << "portline : " << line;
  280. size_t pos = line.find('%');
  281. if (pos != line.npos) {
  282. // Ok, this is a valid portline
  283. std::string work = line;
  284. replace(work, "Fuel Ore", "Fuel");
  285. BUGZ_LOG(fatal) << "re.split? : [" << work << "]";
  286. }
  287. }
  288. void Director::SL_warpline(const std::string &line) {
  289. if (line.empty()) {
  290. SL_parser = nullptr;
  291. return;
  292. }
  293. // process warp line
  294. BUGZ_LOG(fatal) << "warpline: [" << line << "]";
  295. }