director.cpp 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299
  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; // sets Session keep-alive timer.
  126. // set other values we need
  127. talk_direct = false;
  128. show_client = false;
  129. /*
  130. Wait a minute .. this might be confusing.
  131. Shouldn't I send them the current prompt?
  132. Just in case we abort in the middle of something?!?
  133. */
  134. old_prompt = current_prompt;
  135. old_raw_prompt = current_raw_prompt;
  136. to_client("\x1b[0m\n\r");
  137. /*
  138. ╔══════════════════════════════╗
  139. ║ TradeWars Proxy Active ║
  140. ╚══════════════════════════════╝
  141. -=>
  142. */
  143. Boxes box(30, 1, true);
  144. box.boxcolor = "\x1b[1;33;44m";
  145. box.textcolor = "\x1b[1;33;44m";
  146. to_client(box.top());
  147. std::string output = " TradeWars Proxy \x1b[5mActive\x1b[0;1;33;44m ";
  148. to_client(box.row(output));
  149. to_client(box.bottom());
  150. std::shared_ptr<Dispatch> readline = std::make_shared<InputDispatch>(*this);
  151. chain = readline;
  152. InputDispatch *id = static_cast<InputDispatch *>(&(*readline));
  153. id->prompt = "\x1b[0m \x1b[1;33;44m-=>\x1b[0m \x1b[1;37;44m";
  154. id->max_length = 15;
  155. id->setNotify([this]() { this->have_input(); });
  156. readline->activate();
  157. }
  158. void Director::have_input(void) {
  159. ++count;
  160. InputDispatch *id = dynamic_cast<InputDispatch *>(&(*chain));
  161. if (id) {
  162. std::string output =
  163. str(boost::format("Your Input (%2%): [%1%]\n\r") % id->input % count);
  164. to_client("\x1b[0m");
  165. to_client(output);
  166. } else {
  167. proxy_deactivate();
  168. return;
  169. }
  170. if (count > 3) {
  171. proxy_deactivate();
  172. } else {
  173. chain->activate();
  174. }
  175. }
  176. void Director::proxy_deactivate(void) {
  177. active = false;
  178. // reset everything back to good state
  179. talk_direct = true;
  180. show_client = true;
  181. /*
  182. current_prompt = old_prompt;
  183. current_raw_prompt = old_raw_prompt;
  184. */
  185. chain.reset();
  186. to_client("\n\r");
  187. to_client(current_raw_prompt);
  188. }
  189. void Director::SL_cimline(const std::string &line) {
  190. if (line == ": ENDINTERROG") {
  191. SL_parser = nullptr;
  192. return;
  193. }
  194. if (line == ": ") {
  195. // do I need to do anything special here for this?
  196. return;
  197. }
  198. if (line.empty()) {
  199. SL_parser = nullptr;
  200. return;
  201. }
  202. // parse cimline
  203. size_t pos = line.find('%');
  204. std::string work = line;
  205. if (pos == line.npos) {
  206. // warpcim
  207. BUGZ_LOG(fatal) << "warpcim: [" << line << "]";
  208. auto warps = split(line);
  209. sector_warps sw;
  210. for (auto const &w : warps) {
  211. if (sw.sector == 0) {
  212. sw.sector = stoi(w);
  213. } else {
  214. sw.add(stoi(w));
  215. }
  216. }
  217. BUGZ_LOG(fatal) << "warpcim: " << sw;
  218. } else {
  219. // portcim
  220. struct port p = parse_portcim(line);
  221. if (p.sector == 0)
  222. BUGZ_LOG(fatal) << "portcim: [" << line << "]";
  223. else
  224. BUGZ_LOG(fatal) << "portcim: " << p;
  225. }
  226. }
  227. void Director::SL_thiefline(const std::string &line) {
  228. size_t pos = line.find("Suddenly you're Busted!");
  229. bool busted = pos != line.npos;
  230. if (busted) {
  231. BUGZ_LOG(fatal) << "set bust";
  232. SL_parser = nullptr;
  233. } else {
  234. pos = line.find("(You realize the guards saw you last time!)");
  235. if (pos != line.npos) SL_parser = nullptr;
  236. }
  237. // Are those the two ways to exit from this state?
  238. }
  239. void Director::SL_sectorline(const std::string &line) {
  240. BUGZ_LOG(fatal) << "sectorline: [" << line << "]";
  241. }
  242. void Director::SL_portline(const std::string &line) {
  243. if (line.empty()) {
  244. SL_parser = nullptr;
  245. return;
  246. }
  247. BUGZ_LOG(info) << "portline : " << line;
  248. size_t pos = line.find('%');
  249. if (pos != line.npos) {
  250. // Ok, this is a valid portline
  251. std::string work = line;
  252. replace(work, "Fuel Ore", "Fuel");
  253. BUGZ_LOG(fatal) << "re.split? : [" << work << "]";
  254. }
  255. }
  256. void Director::SL_warpline(const std::string &line) {
  257. if (line.empty()) {
  258. SL_parser = nullptr;
  259. return;
  260. }
  261. // process warp line
  262. BUGZ_LOG(fatal) << "warpline: [" << line << "]";
  263. }