director.cpp 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301
  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> readline = std::make_shared<InputDispatch>(*this);
  152. chain = readline;
  153. InputDispatch *id = static_cast<InputDispatch *>(&(*readline));
  154. id->prompt = "\x1b[0m \x1b[1;33;44m-=>\x1b[0m \x1b[1;37;44m";
  155. id->max_length = 15;
  156. id->setNotify([this]() { this->have_input(); });
  157. readline->activate();
  158. }
  159. void Director::have_input(void) {
  160. ++count;
  161. InputDispatch *id = dynamic_cast<InputDispatch *>(&(*chain));
  162. if (id) {
  163. std::string output =
  164. str(boost::format("Your Input (%2%): [%1%]\n\r") % id->input % count);
  165. to_client("\x1b[0m");
  166. to_client(output);
  167. } else {
  168. proxy_deactivate();
  169. return;
  170. }
  171. if (count > 3) {
  172. proxy_deactivate();
  173. } else {
  174. chain->activate();
  175. }
  176. }
  177. void Director::proxy_deactivate(void) {
  178. active = false;
  179. // reset everything back to good state
  180. talk_direct = true;
  181. show_client = true;
  182. /*
  183. current_prompt = old_prompt;
  184. current_raw_prompt = old_raw_prompt;
  185. */
  186. chain.reset();
  187. to_client("\n\r");
  188. to_client(current_raw_prompt);
  189. }
  190. void Director::SL_cimline(const std::string &line) {
  191. if (line == ": ENDINTERROG") {
  192. SL_parser = nullptr;
  193. return;
  194. }
  195. if (line == ": ") {
  196. // do I need to do anything special here for this?
  197. return;
  198. }
  199. if (line.empty()) {
  200. SL_parser = nullptr;
  201. return;
  202. }
  203. // parse cimline
  204. size_t pos = line.find('%');
  205. std::string work = line;
  206. if (pos == line.npos) {
  207. // warpcim
  208. BUGZ_LOG(fatal) << "warpcim: [" << line << "]";
  209. auto warps = split(line);
  210. sector_warps sw;
  211. for (auto const &w : warps) {
  212. if (sw.sector == 0) {
  213. sw.sector = stoi(w);
  214. } else {
  215. sw.add(stoi(w));
  216. }
  217. }
  218. BUGZ_LOG(fatal) << "warpcim: " << sw;
  219. } else {
  220. // portcim
  221. struct port p = parse_portcim(line);
  222. if (p.sector == 0)
  223. BUGZ_LOG(fatal) << "portcim: [" << line << "]";
  224. else
  225. BUGZ_LOG(fatal) << "portcim: " << p;
  226. }
  227. }
  228. void Director::SL_thiefline(const std::string &line) {
  229. size_t pos = line.find("Suddenly you're Busted!");
  230. bool busted = pos != line.npos;
  231. if (busted) {
  232. BUGZ_LOG(fatal) << "set bust";
  233. SL_parser = nullptr;
  234. } else {
  235. pos = line.find("(You realize the guards saw you last time!)");
  236. if (pos != line.npos) SL_parser = nullptr;
  237. }
  238. // Are those the two ways to exit from this state?
  239. }
  240. void Director::SL_sectorline(const std::string &line) {
  241. BUGZ_LOG(fatal) << "sectorline: [" << line << "]";
  242. }
  243. void Director::SL_portline(const std::string &line) {
  244. if (line.empty()) {
  245. SL_parser = nullptr;
  246. return;
  247. }
  248. BUGZ_LOG(info) << "portline : " << line;
  249. size_t pos = line.find('%');
  250. if (pos != line.npos) {
  251. // Ok, this is a valid portline
  252. std::string work = line;
  253. replace(work, "Fuel Ore", "Fuel");
  254. BUGZ_LOG(fatal) << "re.split? : [" << work << "]";
  255. }
  256. }
  257. void Director::SL_warpline(const std::string &line) {
  258. if (line.empty()) {
  259. SL_parser = nullptr;
  260. return;
  261. }
  262. // process warp line
  263. BUGZ_LOG(fatal) << "warpline: [" << line << "]";
  264. }