director.cpp 35 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213
  1. #include "director.h"
  2. #include <boost/format.hpp>
  3. #include <cctype>
  4. #include "ansicolor.h"
  5. #include "boxes.h"
  6. #include "galaxy.h"
  7. #include "logging.h"
  8. #include "scripts.h"
  9. #include "utils.h"
  10. Director::Director() {
  11. BUGZ_LOG(warning) << "Director::Director()";
  12. // active = false;
  13. game = 0; // not in a game
  14. galaxy.reset();
  15. // do everything proxy_deactivate does ...
  16. // proxy_deactivate();
  17. active = false;
  18. // reset everything back to good state
  19. talk_direct = true;
  20. show_client = true;
  21. count = 0;
  22. /*
  23. Setup StringFunc for SL_parser:
  24. Construct these once, rather then every single time we need them.
  25. */
  26. SF_cimline = [this](const std::string &s) { this->SL_cimline(s); };
  27. SF_sectorline = [this](const std::string &s) { this->SL_sectorline(s); };
  28. SF_portline = [this](const std::string &s) { this->SL_portline(s); };
  29. SF_warpline = [this](const std::string &s) { this->SL_warpline(s); };
  30. SF_infoline = [this](const std::string &s) { this->SL_infoline(s); };
  31. SF_densityline = [this](const std::string &s) { this->SL_densityline(s); };
  32. SF_computer_portline = [this](const std::string &s) {
  33. this->SL_computer_portline(s);
  34. };
  35. build_menu();
  36. }
  37. Director::~Director() { BUGZ_LOG(warning) << "Director::~Director()"; }
  38. void Director::client_input(const std::string &input) {
  39. // If we're already active, don't try to activate.
  40. if (chain) {
  41. chain->client_input(input);
  42. return;
  43. }
  44. if (active) {
  45. if (input == "Q" || input == "q") proxy_deactivate();
  46. return;
  47. } else if (input == "\x1b" || input == "~") {
  48. std::string &prompt = current_prompt;
  49. BUGZ_LOG(trace) << "ACTIVATE prompt shows: [" << prompt << "]";
  50. if (prompt == "Selection (? for menu): ") {
  51. to_client(
  52. "\n\rThere's not much we can do here. Activate in-game at a "
  53. "Command prompt.\n\r");
  54. to_client(current_raw_prompt);
  55. return;
  56. }
  57. // easter-eggs:
  58. if (prompt == "Enter your choice: ") {
  59. ANSIColor c1(COLOR::CYAN, ATTR::BOLD);
  60. ANSIColor c2(COLOR::WHITE, ATTR::BOLD);
  61. to_client(std::string("\n\r") + c1() + "I'd choose " + c2() + "`T`" +
  62. c1() + ", but that's how I was coded.\n\r");
  63. // "\n\r\x1b[1;36mI'd choose \x1b[1;37m`T`\x1b[1;36m, but "
  64. // "that's how I was coded.\n\r");
  65. to_client(current_raw_prompt);
  66. return;
  67. }
  68. // easter-egg
  69. if (prompt == "[Pause]") {
  70. ANSIColor c1(COLOR::CYAN, ATTR::BOLD);
  71. to_client(std::string(" ") + c1() + "PAWS" + reset() + "\n\r");
  72. to_client(current_raw_prompt);
  73. return;
  74. }
  75. if (at_planet_prompt(prompt)) {
  76. // future: If activated at planet menu, activate the planet upgrade
  77. // script!
  78. to_client("\n\r\x1b[0mFUTURE: Activate the planet upgrade script.\n\r");
  79. to_client(current_raw_prompt);
  80. return;
  81. }
  82. if (at_command_prompt(prompt)) {
  83. proxy_activate();
  84. return;
  85. }
  86. }
  87. // Ok...
  88. if (talk_direct) to_server(input);
  89. }
  90. void Director::server_line(const std::string &line,
  91. const std::string &raw_line) {
  92. // check for if we entered game/left game
  93. if (line.find("TradeWars Game Server ") != std::string::npos) {
  94. to_client("\rTradeWars Proxy v2++ READY (~ or ESC to activate)\n\r");
  95. /*
  96. There's a delay here when I save the game data.
  97. I've moved it futher down. Hide it at a prompt, so it isn't so
  98. noticeable.
  99. */
  100. // reset "active game" -- we're at the TWGS main menu
  101. }
  102. if (line.find("Selection (? for menu): ") != std::string::npos) {
  103. char ch = line[line.length() - 1];
  104. if (ch >= 'A' && ch < 'Q') {
  105. if ((game) && (game != ch)) {
  106. galaxy.save();
  107. }
  108. game = ch;
  109. BUGZ_LOG(warning) << "GAME " << game << " activated!";
  110. // TODO: Load game data
  111. galaxy.game = game;
  112. galaxy.username = username;
  113. galaxy.load();
  114. // YAML loaded, set sensible default config values (if missing).
  115. if (!galaxy.config["display_lines"]) {
  116. galaxy.config["display_lines"] = 20;
  117. }
  118. galaxy.meta["help"]["display_lines"] =
  119. "Number of report lines to display";
  120. if (!galaxy.config["burnt_percent"]) {
  121. galaxy.config["burnt_percent"] = 40;
  122. }
  123. galaxy.meta["help"]["burnt_percent"] =
  124. "Don't display ports in report below this percent";
  125. }
  126. // not needed (handled by above Game Server check).
  127. if (ch == 'Q') {
  128. if (game) {
  129. // TODO: Save galaxy data
  130. galaxy.save();
  131. }
  132. game = 0;
  133. galaxy.reset();
  134. }
  135. }
  136. if (game) {
  137. // in-game parsing here.
  138. /*
  139. ____ _ _
  140. / ___| ___ _ ____ _____ _ __ | | (_)_ __ ___
  141. \___ \ / _ \ '__\ \ / / _ \ '__| | | | | '_ \ / _ \
  142. ___) | __/ | \ V / __/ | | |___| | | | | __/
  143. |____/ \___|_| \_/ \___|_| |_____|_|_| |_|\___|
  144. ____ _
  145. | _ \ __ _ _ __ ___(_)_ __ __ _
  146. | |_) / _` | '__/ __| | '_ \ / _` |
  147. | __/ (_| | | \__ \ | | | | (_| |
  148. |_| \__,_|_| |___/_|_| |_|\__, |
  149. |___/
  150. This is where all of the server lines are gleaned for all the
  151. information that we can get out of them.
  152. // When activating the computer
  153. SP: [Command [TL=00:00:00]:[926] (?=Help)? : ]
  154. Sector: 926
  155. CI: [c]
  156. SL: [Command [TL=00:00:00]:[926] (?=Help)? : C]
  157. SL: [<Computer>]
  158. SL: []
  159. SL: [<Computer activated>]
  160. SL: []
  161. SP: [Computer command [TL=00:00:00]:[926] (?=Help)? ]
  162. // deactivating the computer
  163. SL: [Computer command [TL=00:00:00]:[926] (?=Help)? Q]
  164. SL: []
  165. SL: [<Computer deactivated>]
  166. SL: []
  167. */
  168. if (line == "<Port>") {
  169. SL_parser = SF_portline;
  170. }
  171. /*
  172. if (startswith(line, " Items Status Trading % of max OnBoard"))
  173. SL_parser = SF_portline;
  174. */
  175. if (endswith(line, "Relative Density Scan")) {
  176. galaxy.dscan.reset(current_sector);
  177. SL_parser = SF_densityline;
  178. }
  179. if (startswith(line, "Sector : ")) SL_parser = SF_sectorline;
  180. if (line == ": ") SL_parser = SF_cimline;
  181. if (line == "<Info>") SL_parser = SF_infoline;
  182. if (startswith(line, "What sector is the port in? [")) {
  183. // Computer Port Report
  184. // SL: [What sector is the port in? [611] 4]
  185. size_t pos = line.rfind(' ');
  186. computer_port_sector = stoi(line.substr(pos));
  187. SL_parser = SF_computer_portline;
  188. }
  189. }
  190. if (SL_parser) {
  191. SL_parser(line);
  192. }
  193. if (chain) {
  194. chain->server_line(line, raw_line);
  195. }
  196. }
  197. void Director::server_prompt(const std::string &prompt,
  198. const std::string &raw_prompt) {
  199. current_prompt = prompt;
  200. current_raw_prompt = raw_prompt;
  201. if (game) {
  202. if (prompt == "Selection (? for menu): ") {
  203. galaxy.save();
  204. game = 0;
  205. galaxy.reset();
  206. }
  207. // in-game parsing here.
  208. if (startswith(prompt, "Command [") && endswith(prompt, "] (?=Help)? : ")) {
  209. std::string sector_text;
  210. size_t before, after;
  211. before = prompt.find("]:[") + 3;
  212. after = prompt.find("] (?=Help)");
  213. sector_text = prompt.substr(before, after - before);
  214. current_sector = stoi(sector_text);
  215. BUGZ_LOG(info) << "current_sector = " << current_sector;
  216. }
  217. }
  218. /*
  219. if (emit_server_prompt)
  220. emit_server_prompt(prompt);
  221. */
  222. if (chain) chain->server_prompt(prompt);
  223. }
  224. void Director::build_menu(void) {
  225. main_menu = std::make_shared<MenuDispatch>(*this);
  226. MenuDispatch *md = static_cast<MenuDispatch *>(&(*main_menu));
  227. ANSIColor bcolor(COLOR::YELLOW, COLOR::BLUE, ATTR::BOLD);
  228. ANSIColor tcolor(COLOR::WHITE, COLOR::BLUE, ATTR::BOLD);
  229. ANSIColor mocolor(COLOR::CYAN, ATTR::BOLD);
  230. md->menu_box_color = bcolor(); // "\x1b[1;33;44m";
  231. md->menu_text_color = tcolor(); // "\x1b[1;37;44m";
  232. md->menu_title = "Proxy Menu";
  233. md->menu_options_color = mocolor(); // "\x1b[1;36;40m";
  234. ANSIColor by{1, 33};
  235. ANSIColor cyan{36};
  236. ANSIColor bg{1, 32};
  237. std::string prompt = by() + "M" + cyan() + "ain " + by() + "P" + cyan() +
  238. "roxy " + bg() + "=>" + reset() + " ";
  239. md->menu_prompt = prompt; // "Main Proxy => ";
  240. // "\x1b[0;31;40m\xdb\xb2\xb1\xb0 \x1b[31;40mRED "
  241. // "\x1b[32;40mGREEN\x1b[30;42m\xdb\xb2\xb1\xb0 \x1b[0m : ";
  242. md->lazy = true;
  243. md->menu = {{"C", "Configure"},
  244. {"D", "Display Report"},
  245. {"E", "Export Data/Save"},
  246. {"I", "Information"},
  247. {"P", "Port CIM"},
  248. {"W", "Warp CIM"},
  249. {"T", "Trading Report (same as D)"},
  250. {"S", "Scripts"},
  251. {"X", "eXit"}};
  252. md->setNotify([this]() { this->menu_choice(); });
  253. cim = std::make_shared<CIMDispatch>(*this);
  254. cim->setNotify([this]() { this->cim_done(); });
  255. }
  256. void Director::proxy_activate(void) {
  257. active = true; // yes, set keep-alive timer.
  258. to_server(" "); // start keep-alive timer.
  259. // set other values we need
  260. talk_direct = false;
  261. show_client = false;
  262. /*
  263. Wait a minute .. this might be confusing.
  264. Shouldn't I send them the current prompt?
  265. Just in case we abort in the middle of something?!?
  266. */
  267. old_prompt = current_prompt;
  268. old_raw_prompt = current_raw_prompt;
  269. to_client("\x1b[0m\n\r");
  270. /*
  271. ╔══════════════════════════════╗
  272. ║ TradeWars Proxy Active ║
  273. ╚══════════════════════════════╝
  274. -=>
  275. */
  276. Boxes box(30, 1, true);
  277. box.boxcolor = "\x1b[1;33;44m";
  278. box.textcolor = "\x1b[1;33;44m";
  279. to_client(box.top());
  280. std::string output = " TradeWars Proxy \x1b[5mActive\x1b[0;1;33;44m ";
  281. to_client(box.row(output));
  282. to_client(box.bottom());
  283. chain = main_menu;
  284. main_menu->activate();
  285. /*
  286. // Using InputDispatch -- and see have_input
  287. std::shared_ptr<Dispatch> readline = std::make_shared<InputDispatch>(*this);
  288. chain = readline;
  289. InputDispatch *id = static_cast<InputDispatch *>(&(*readline));
  290. id->prompt = "\x1b[0m\x1b[1;33;44m-=>\x1b[0m \x1b[1;37;44m";
  291. id->max_length = 15;
  292. id->setNotify([this]() { this->have_input(); });
  293. readline->activate();
  294. */
  295. }
  296. void Director::menu_choice(void) {
  297. MenuDispatch *md = dynamic_cast<MenuDispatch *>(&(*chain));
  298. if (md) {
  299. if (md->input.empty()) {
  300. to_client("Menu aborted.\n\r");
  301. proxy_deactivate();
  302. return;
  303. } else {
  304. switch (md->input[0]) {
  305. case 'C': // configure
  306. config_edit();
  307. return;
  308. break;
  309. case 'D':
  310. case 'T': // display trading report
  311. {
  312. auto pptv = galaxy.find_best_trades();
  313. std::string output;
  314. galaxy.sort_port_pair_type(pptv);
  315. int max_display = 20;
  316. if (galaxy.config["display_lines"])
  317. max_display = galaxy.config["display_lines"].as<int>();
  318. else
  319. galaxy.config["display_lines"] = max_display;
  320. if ((max_display <= 0) || (max_display > 255)) {
  321. max_display = 255;
  322. galaxy.config["display_lines"] = 255;
  323. }
  324. const int per_line = 5;
  325. int count = 0;
  326. int line = 0;
  327. std::string display_line;
  328. ANSIColor by{1, 33};
  329. ANSIColor cyan{36};
  330. ANSIColor bg{1, 32};
  331. ANSIColor bb{1, 34};
  332. for (auto const &ppt : pptv) {
  333. output =
  334. str(boost::format("%1%%2$5d%3%:%4%%5$-5d%3%(%6%%7$d%3%) ") %
  335. by() % ppt.s1 % cyan() % bg() % ppt.s2 % bb() % ppt.type);
  336. display_line.append(output);
  337. ++count;
  338. if (count == per_line) {
  339. count = 0;
  340. display_line.append("\n\r");
  341. to_client(display_line);
  342. display_line.clear();
  343. ++line;
  344. }
  345. if (line == max_display) break;
  346. }
  347. if (count != 0) {
  348. display_line.append("\n\r");
  349. to_client(display_line);
  350. display_line.clear();
  351. }
  352. // We got < 5 lines, and max_display is > 5. Offer suggestion:
  353. if ((line < 5) && (max_display > 5)) {
  354. // suggestion:
  355. to_client(
  356. "HINT: For more lines, try reducing the burnt_percent?\n\r");
  357. }
  358. } break;
  359. case 'E': // Export Data/Save
  360. to_client("Saving...");
  361. galaxy.save();
  362. to_client("\rSaved....\n\r");
  363. break;
  364. case 'I': // Information
  365. information();
  366. break;
  367. case 'P': // Port CIM
  368. // Since we're adding/updating, we don't lose our
  369. // type 0 ports. Type 9 stays at 9.
  370. chain = cim;
  371. to_server("^RQ");
  372. {
  373. std::string text = str(boost::format("Port CIM Report (%1%)\n\r") %
  374. galaxy.ports.size());
  375. // to_client("Port CIM Report\n\r");
  376. to_client(text);
  377. }
  378. chain->activate();
  379. return;
  380. break;
  381. case 'W': // Warp CIM
  382. chain = cim;
  383. to_server("^IQ");
  384. {
  385. std::string text = str(boost::format("Warp CIM Report (%1%)\n\r") %
  386. galaxy.warps.size());
  387. // to_client("Warp CIM Report\n\r");
  388. to_client(text);
  389. }
  390. chain->activate();
  391. return;
  392. break;
  393. // case 'T': // Trading Report
  394. // break;
  395. case 'S': // Scripts
  396. {
  397. init_scripts_menu();
  398. chain = scripts_menu;
  399. chain->activate();
  400. return;
  401. } break;
  402. case 'X': // Exit
  403. proxy_deactivate();
  404. return;
  405. }
  406. /*
  407. std::string text = str(
  408. boost::format("Back from Menu [%1%] was selected.\n\r") %
  409. md->input);
  410. to_client(text);
  411. */
  412. md->activate();
  413. }
  414. }
  415. }
  416. MenuDispatch *Director::init_scripts_menu(void) {
  417. MenuDispatch *md;
  418. if (scripts_menu) {
  419. md = dynamic_cast<MenuDispatch *>(&(*scripts_menu));
  420. return md;
  421. } else {
  422. scripts_menu = std::make_shared<MenuDispatch>(*this);
  423. md = static_cast<MenuDispatch *>(&(*scripts_menu));
  424. md->menu_box_color = "\x1b[0;32;40m";
  425. md->menu_text_color = "\x1b[1;32;40m";
  426. md->menu_title = "Scripts Menu";
  427. md->menu_options_color = "\x1b[1;32;40m";
  428. md->lazy = false;
  429. ANSIColor by{1, 33};
  430. ANSIColor cyan{36};
  431. ANSIColor bg{1, 32};
  432. std::string prompt = by() + "S" + cyan() + "cript " + by() + "M" + cyan() +
  433. "enu " + bg() + "=>" + reset() + " ";
  434. md->menu_prompt = prompt; // " SCRIPT : ";
  435. md->menu = {{"!", "Terror"},
  436. {"T", "Trade"},
  437. {"S", "Safe Move"},
  438. {"C", "Closest Trade"},
  439. {"U", "Upgrade Planet Pants"},
  440. {"X", "Exit Scripts"}};
  441. md->setNotify([this]() { this->scripts_done(); });
  442. return md;
  443. }
  444. }
  445. void Director::scripts_done(void) {
  446. // Was script selected? If so, run it!
  447. // otherwise, back to the menu we go...
  448. MenuDispatch *md = dynamic_cast<MenuDispatch *>(&(*scripts_menu));
  449. if (md) {
  450. if (md->input.empty()) {
  451. to_client("Scripts aborted.\n\r");
  452. scripts_menu.reset();
  453. proxy_deactivate();
  454. return;
  455. } else {
  456. switch (md->input[0]) {
  457. case 'T': // Trade
  458. {
  459. script = std::make_shared<TraderDispatch>(*this);
  460. TraderDispatch *ts = static_cast<TraderDispatch *>(&((*script)));
  461. ts->setNotify([this]() { this->proxy_deactivate(); });
  462. // Locate best trades
  463. auto found = galaxy.find_trades(current_sector, false);
  464. if (found.empty()) {
  465. to_client(
  466. "No Trades found. Port burnt (CONFIG: lower burnt_percent?) "
  467. "or no ports around.\n\r");
  468. proxy_deactivate();
  469. return;
  470. }
  471. // sort first?
  472. galaxy.sort_port_pair_type(found);
  473. BUGZ_LOG(fatal) << "Found " << found.size() << " possible trade(s).";
  474. BUGZ_LOG(fatal) << "Best: " << found[0].s1 << "," << found[0].s2
  475. << " : " << found[0].type;
  476. // Set parameters
  477. ts->port[0] = found[0].s1;
  478. ts->port[1] = found[0].s2;
  479. ts->type = found[0].type;
  480. ts->trades = found[0].trades;
  481. chain = script;
  482. chain->activate();
  483. return;
  484. } break;
  485. case 'S': {
  486. script = std::make_shared<MoveDispatch>(*this);
  487. MoveDispatch *md = static_cast<MoveDispatch *>(&((*script)));
  488. md->setNotify([this]() { this->proxy_deactivate(); });
  489. md->move_to = 1;
  490. chain = script;
  491. chain->activate();
  492. return;
  493. } break;
  494. case '!': {
  495. script = std::make_shared<ScriptTerror>(*this);
  496. ScriptTerror *st = static_cast<ScriptTerror *>(&(*script));
  497. st->setNotify([this]() {
  498. script.reset();
  499. this->proxy_deactivate();
  500. });
  501. chain = script;
  502. chain->activate();
  503. return;
  504. } break;
  505. // }
  506. case 'C': {
  507. auto best = galaxy.find_closest(current_sector);
  508. if (best.type != 0) {
  509. std::string text =
  510. str(boost::format("Best/Closest: %1% with %2% & %3%\n\r") %
  511. best.type % best.s1 % best.s2);
  512. to_client(text);
  513. } else {
  514. to_client("I don't see any best trades.\n\r");
  515. }
  516. } break;
  517. case 'Q':
  518. chain = main_menu;
  519. main_menu->activate();
  520. return;
  521. break;
  522. }
  523. }
  524. }
  525. proxy_activate();
  526. // And to end scripts, we do .. what exactly?
  527. // DEBUG: Ok, why does everything work OK if I reset the scripts_menu
  528. // here?? probably do want to destroy scripts here. ;)
  529. // scripts_menu.reset();
  530. // proxy_deactivate();
  531. }
  532. /**
  533. * @brief Setup Config Input
  534. *
  535. * @return DispatchInput*
  536. */
  537. InputDispatch *Director::init_config_input(void) {
  538. InputDispatch *id;
  539. if (config_input) {
  540. // Yes, it has been setup before.
  541. id = dynamic_cast<InputDispatch *>(&(*config_input));
  542. ANSIColor by{1, 33};
  543. ANSIColor cyan{36};
  544. ANSIColor bg{1, 32};
  545. std::string prompt =
  546. by() + "C" + cyan() + "onfig " + bg() + "=>" + reset() + " ";
  547. id->prompt = prompt; // "Config => ";
  548. id->numeric = true;
  549. id->max_length = 3;
  550. config_item.clear();
  551. return id;
  552. } else {
  553. // set it up
  554. config_input = std::make_shared<InputDispatch>(*this);
  555. id = static_cast<InputDispatch *>(&(*config_input));
  556. ANSIColor by{1, 33};
  557. ANSIColor cyan{36};
  558. ANSIColor bg{1, 32};
  559. std::string prompt =
  560. by() + "C" + cyan() + "onfig " + bg() + "=>" + reset() + " ";
  561. id->prompt = prompt; // "Config => ";
  562. id->numeric = true;
  563. id->max_length = 3;
  564. id->setNotify([this]() { this->config_have_input(); });
  565. config_item.clear();
  566. return id;
  567. }
  568. }
  569. void Director::config_edit(void) {
  570. // display current config
  571. std::string menu_box_color = "\x1b[1;33;44m";
  572. std::string menu_text_color = "\x1b[1;37;44m";
  573. auto abox = Boxes::alert(" Configuration: ", menu_box_color,
  574. menu_text_color, 20, 1, true);
  575. for (auto line : abox) {
  576. to_client(line);
  577. }
  578. // to_client("Configuration:\n\r");
  579. int item = 1;
  580. ANSIColor number(COLOR::CYAN);
  581. ANSIColor key(COLOR::GREEN, ATTR::BOLD);
  582. ANSIColor value(COLOR::BLUE, ATTR::BOLD);
  583. for (auto const &cfg : galaxy.config) {
  584. std::string output =
  585. str(boost::format("%1%%2$2d %3%%4$20s: %5%%6$s%7%\n\r") % number() %
  586. item % key() % cfg.first % value() % cfg.second % reset());
  587. to_client(output);
  588. ++item;
  589. }
  590. std::string message =
  591. number() + "Enter number to edit, " + key() + "blank to exit.\n\r";
  592. // to_client("Enter number to edit, blank to exit.\n\r");
  593. to_client(message);
  594. // setup call to config_input:
  595. InputDispatch *id = init_config_input();
  596. chain = config_input;
  597. id->activate();
  598. // to return to the menu:
  599. // MenuDispatch *md = dynamic_cast<MenuDispatch *>(&(*chain));
  600. // md->activate();
  601. }
  602. void Director::config_have_input(void) {
  603. InputDispatch *id = dynamic_cast<InputDispatch *>(&(*config_input));
  604. if (config_item.empty()) {
  605. // This is a config menu selection
  606. if (id->input.empty()) {
  607. // We're done here. Return to menu.
  608. chain = main_menu;
  609. MenuDispatch *md = dynamic_cast<MenuDispatch *>(&(*chain));
  610. md->activate();
  611. // destroy the input? yes.
  612. config_input.reset();
  613. return;
  614. } else {
  615. int item = sstoi(id->input);
  616. if ((item < 1) || (item > (int)galaxy.config.size())) {
  617. // selection out of range - redisplay config menu
  618. to_client("What? I didn't see that item.\n\r");
  619. config_edit();
  620. return;
  621. } else {
  622. int pos = 1;
  623. const YAML::Node &config = galaxy.config;
  624. for (auto const &c : config) {
  625. if (pos == item) {
  626. // got it!
  627. ANSIColor key(COLOR::GREEN, ATTR::BOLD);
  628. ANSIColor value(COLOR::BLUE, ATTR::BOLD);
  629. config_item = c.first.as<std::string>();
  630. std::string output =
  631. str(boost::format("%1%%2% : %3%%4%\n\r") % key() % config_item %
  632. value() % galaxy.meta["help"][config_item]);
  633. to_client(output);
  634. id->max_length = 30;
  635. id->numeric = false;
  636. ANSIColor by{1, 33};
  637. ANSIColor cyan{36};
  638. ANSIColor bg{1, 32};
  639. std::string prompt =
  640. by() + "C" + cyan() + "hange to " + bg() + "=>" + reset() + " ";
  641. id->prompt = prompt;
  642. id->activate();
  643. return;
  644. };
  645. ++pos;
  646. }
  647. to_client("What? I didn't find that item?\n\r");
  648. config_edit();
  649. return;
  650. }
  651. }
  652. } else {
  653. // This is a config item edit
  654. if (id->input.empty()) {
  655. to_client("No change.\n\r");
  656. config_edit();
  657. return;
  658. } else {
  659. BUGZ_LOG(fatal) << "Config EDIT: " << config_item << " = " << id->input;
  660. galaxy.config[config_item] = id->input;
  661. config_edit();
  662. return;
  663. }
  664. }
  665. }
  666. void Director::have_input(void) {
  667. ++count;
  668. InputDispatch *id = dynamic_cast<InputDispatch *>(&(*chain));
  669. if (id) {
  670. std::string output =
  671. str(boost::format("Your Input (%2%): [%1%]\n\r") % id->input % count);
  672. to_client("\x1b[0m");
  673. to_client(output);
  674. } else {
  675. proxy_deactivate();
  676. return;
  677. }
  678. if (count > 3) {
  679. proxy_deactivate();
  680. } else {
  681. chain->activate();
  682. }
  683. }
  684. void Director::cim_done(void) {
  685. BUGZ_LOG(info) << "CIM done";
  686. chain = main_menu;
  687. main_menu->activate();
  688. }
  689. void Director::information(void) {
  690. std::string output;
  691. to_client("I currently know the following:\n\r");
  692. output = str(
  693. boost::format("Ports: %1%, Sectors: %2%, Config: %3%, Meta: %4%\n\r") %
  694. galaxy.ports.size() % galaxy.warps.size() % galaxy.config.size() %
  695. galaxy.meta.size());
  696. to_client(output);
  697. }
  698. void Director::proxy_deactivate(void) {
  699. active = false;
  700. // reset everything back to good state
  701. talk_direct = true;
  702. show_client = true;
  703. chain.reset();
  704. to_client("\n\r");
  705. to_client(current_raw_prompt);
  706. }
  707. /*
  708. Server Line Parsing Routines
  709. */
  710. void Director::SL_cimline(const std::string &line) {
  711. if (line == ": ENDINTERROG") {
  712. SL_parser = nullptr;
  713. return;
  714. }
  715. if (line == ": ") {
  716. // do I need to do anything special here for this?
  717. // Maybe -- We would save special ports that don't show up
  718. // (StarDock/Special) before. We don't know (at this point) if this is
  719. // warps or ports.
  720. return;
  721. }
  722. if (line.empty()) {
  723. SL_parser = nullptr;
  724. return;
  725. }
  726. // parse cimline
  727. // size_t pos = line.find('%');
  728. // std::string work = line;
  729. // if (pos == line.npos) {
  730. if (in(line, "%")) {
  731. // portcim
  732. struct port p = parse_portcim(line);
  733. if (p.sector == 0)
  734. BUGZ_LOG(fatal) << "portcim: FAIL [" << line << "]";
  735. else
  736. BUGZ_LOG(trace) << "portcim: " << p;
  737. galaxy.add_port(p);
  738. } else {
  739. // warpcim
  740. // BUGZ_LOG(fatal) << "warpcim: [" << line << "]";
  741. auto warps = split(line);
  742. sector_warps sw;
  743. for (auto const &w : warps) {
  744. if (sw.sector == 0) {
  745. sw.sector = stoi(w);
  746. } else {
  747. sw.add(stoi(w));
  748. }
  749. }
  750. BUGZ_LOG(trace) << "warpcim: " << sw;
  751. galaxy.add_warp(sw);
  752. }
  753. }
  754. void Director::SL_thiefline(const std::string &line) {
  755. size_t pos = line.find("Suddenly you're Busted!");
  756. bool busted = pos != line.npos;
  757. if (busted) {
  758. BUGZ_LOG(fatal) << "set bust";
  759. SL_parser = nullptr;
  760. } else {
  761. pos = line.find("(You realize the guards saw you last time!)");
  762. if (pos != line.npos) SL_parser = nullptr;
  763. }
  764. // Are those the two ways to exit from this state?
  765. }
  766. void Director::SL_sectorline(const std::string &line) {
  767. BUGZ_LOG(fatal) << "sectorline: [" << line << "]";
  768. if (line.empty()) {
  769. SL_parser = nullptr;
  770. } else {
  771. /*
  772. sectorline: [Sector : 926 in The Federation.]
  773. sectorline: [Beacon : FedSpace, FedLaw Enforced]
  774. sectorline: [Ports : Stargate Alpha I, Class 9 (Special) (StarDock)]
  775. sectorline: [Traders : Civilian phil, w/ 30 ftrs,]
  776. sectorline: [ in Star Stomper (Sverdlov Merchant Cruiser)]
  777. sectorline: [Warps to Sector(s) : 70 - 441 - 575 - 600 - 629 - 711]
  778. sectorline: [Warps to Sector(s) : 70 - (475) - 569]
  779. What can we get from Traders : line? Can we get if they are hostile
  780. to us? We need to respond to "is powering up weapons" ... We can
  781. react faster then a person can!
  782. "phil is powering up weapons systems!"
  783. Also the auto-attack ones Ferrengi -- we need to auto-respond Retreat.
  784. */
  785. if (in(line, "Sector :")) {
  786. current_sector = stoi(line.substr(10));
  787. BUGZ_LOG(warning) << "SECTOR: " << current_sector;
  788. }
  789. if (in(line, "Ports :")) {
  790. std::string port_class;
  791. size_t pos = line.find(", Class ");
  792. if (pos != std::string::npos) {
  793. pos += 8;
  794. int class_ = stoi(line.substr(pos));
  795. BUGZ_LOG(fatal) << "PORT: " << class_;
  796. galaxy.add_port(current_sector, class_);
  797. }
  798. }
  799. if (in(line, "Warps to Sector(s) :")) {
  800. std::string temp = line.substr(20);
  801. replace(temp, " - ", " ");
  802. // unexplored sectors ()
  803. // Should I track these?
  804. replace(temp, "(", "");
  805. replace(temp, ")", "");
  806. sector_warps sw;
  807. auto warps = split(temp);
  808. sw.sector = current_sector;
  809. // what if there is only one warp?
  810. for (auto const &w : warps) {
  811. sw.add(stoi(w));
  812. }
  813. BUGZ_LOG(fatal) << "WARPS: " << sw;
  814. galaxy.add_warp(sw);
  815. }
  816. }
  817. }
  818. void Director::SL_densityline(const std::string &line) {
  819. BUGZ_LOG(fatal) << "densityline: [" << line << "]";
  820. if (line.empty()) {
  821. SL_parser = nullptr;
  822. return;
  823. }
  824. /*
  825. // Ensure this really is a density scan and not something else
  826. if (!in(line, "Sector") || !in(line, "Warps") || !in(line, "NavHaz") ||
  827. !in(line, "Anom")) {
  828. BUGZ_LOG(fatal) << "densityline: Invalid line.";
  829. SL_parser = nullptr;
  830. return;
  831. }
  832. if (not galaxy.meta["density"]) {
  833. galaxy.meta["density"] = YAML::Node();
  834. }
  835. */
  836. /*
  837. 0 1 2 3 4 5 6 7 8 9 10 11 12
  838. "Sector 55 ==> 0 Warps : 4 NavHaz : 0% Anom : No"
  839. "Sector ( 223) ==> 0 Warps : 3 NavHaz : 0% Anom : No"
  840. // Cleaned up line
  841. 0 1 2 3 4 5 6 7 8 9
  842. "Sector 55 ==> 0 Warps 4 NavHaz 0 Anom No"
  843. "Sector 223 ==> 0 Warps 3 NavHaz 0 Anom No"
  844. */
  845. if (in(line, "==>")) {
  846. std::string work = line;
  847. replace(work, ":", "");
  848. bool known = !in(work, "(");
  849. replace(work, "(", "");
  850. replace(work, ")", "");
  851. replace(work, "%", "");
  852. auto dense = split(work);
  853. // Parse our data
  854. sector_type sector = std::stoi(dense.at(1));
  855. uint16_t density = std::stoi(dense.at(3));
  856. uint16_t warps = std::stoi(dense.at(5));
  857. uint16_t navhaz = std::stoi(dense.at(7));
  858. bool anom = in(dense.at(9), "Yes");
  859. struct density d = {sector, density, warps, navhaz, anom, known};
  860. galaxy.dscan.add_scan(d);
  861. // Commit data
  862. BUGZ_LOG(warning) << "densityline: {sector=" << sector
  863. << " density=" << density << " warps=" << warps
  864. << " navhaz=" << navhaz << " anom=" << anom
  865. << " known=" << known << "}";
  866. /*
  867. if (galaxy.meta["density"][sector]) {
  868. galaxy.meta["density"][sector] = YAML::Node();
  869. }
  870. */
  871. galaxy.meta["density"][sector]["density"] = density;
  872. galaxy.meta["density"][sector]["warps"] = warps;
  873. galaxy.meta["density"][sector]["navhaz"] = navhaz;
  874. galaxy.meta["density"][sector]["anom"] = anom;
  875. galaxy.meta["density"][sector]["known"] = known;
  876. // Add a check to see if density is greater than 500
  877. // Add datetime stamp
  878. }
  879. }
  880. void Director::SL_portline(const std::string &line) {
  881. /*
  882. We take blank lines because we start at <Port>.
  883. Otherwise, we trigger on computer port requests.
  884. if (line.empty()) {
  885. SL_parser = nullptr;
  886. return;
  887. }
  888. */
  889. /*
  890. SL: [ Items Status Trading % of max OnBoard]
  891. SL: [ ----- ------ ------- -------- -------]
  892. SL: [Fuel Ore Buying 3000 100% 0]
  893. SL: [Organics Buying 3000 100% 0]
  894. SL: [Equipment Buying 3000 100% 0]
  895. SL: []
  896. SL: [Commerce report for: 03:51:56 PM Mon Oct 24, 2033 You can buy:]
  897. SL: [A Cargo holds : 650 credits / next hold 0]
  898. SL: [B Fighters : 233 credits per fighter 75]
  899. SL: [C Shield Points : 116 credits per point 100]
  900. SL: []
  901. */
  902. // BUGZ_LOG(info) << "portline : " << line;
  903. if (in(line, "%")) {
  904. // size_t pos = line.find('%');
  905. // if (pos != line.npos) {
  906. // Ok, this is a valid portline
  907. std::string work = line;
  908. replace(work, "Fuel Ore", "Fuel");
  909. auto parts = split(work);
  910. if (parts[0] == "Items") return;
  911. char c = tolower(parts[0][0]);
  912. int pos;
  913. char foe[4] = "foe";
  914. for (pos = 0; pos < 3; ++pos) {
  915. if (c == foe[pos]) break;
  916. }
  917. int amount = stoi(parts[2]);
  918. int percent = stoi(parts[3]);
  919. // update port
  920. auto port = galaxy.ports.find(current_sector);
  921. if (port != galaxy.ports.end()) {
  922. port->second.amount[pos] = amount;
  923. port->second.percent[pos] = percent;
  924. }
  925. /*
  926. BUGZ_LOG(fatal) << "portline split:";
  927. for (auto const p : parts) {
  928. BUGZ_LOG(fatal) << p;
  929. }
  930. */
  931. // Here's the end:
  932. if (parts[0] == "Equipment") SL_parser = nullptr;
  933. // BUGZ_LOG(fatal) << "portline split : [" << parts << "]";
  934. }
  935. }
  936. void Director::SL_warpline(const std::string &line) {
  937. if (line.empty()) {
  938. SL_parser = nullptr;
  939. return;
  940. }
  941. // process warp line
  942. BUGZ_LOG(fatal) << "warpline: [" << line << "]";
  943. }
  944. void Director::SL_infoline(const std::string &line) {
  945. static int state;
  946. if (line == "<Info>") {
  947. state = 0;
  948. galaxy.meta["info"] = YAML::Node();
  949. }
  950. if (line.empty()) {
  951. ++state;
  952. if (state == 2) {
  953. SL_parser = nullptr;
  954. // clear out the existing ship data
  955. galaxy.meta["ship"] = YAML::Node();
  956. // process the parsed information in meta["info"]
  957. if (galaxy.meta["info"]["Total Holds"]) {
  958. std::string work = galaxy.meta["info"]["Total Holds"].as<std::string>();
  959. replace(work, "Fuel Ore", "Fuel");
  960. auto parts = split(work, " - ");
  961. int total_holds = stoi(parts[0]);
  962. BUGZ_LOG(fatal) << "total holds: " << total_holds;
  963. auto contents = split(parts[1]);
  964. for (auto const &hold : contents) {
  965. auto hold_amount = split(hold, "=");
  966. BUGZ_LOG(fatal) << hold_amount[0] << " with " << hold_amount[1];
  967. std::string key = hold_amount[0];
  968. str_tolower(key);
  969. // equipment = e
  970. // organics = o
  971. // fuel = f
  972. // colonists = c
  973. // empty = empty
  974. if (key != "empty") {
  975. key = key[0];
  976. }
  977. galaxy.meta["ship"]["holds"][key] = stoi(hold_amount[1]);
  978. }
  979. galaxy.meta["ship"]["holds"]["total"] = total_holds;
  980. }
  981. if (galaxy.meta["info"]["Turns to Warp"]) {
  982. int warp_turns = galaxy.meta["info"]["Turns to Warp"].as<int>();
  983. BUGZ_LOG(fatal) << "Turns to Warp: " << warp_turns;
  984. galaxy.meta["ship"]["warp_turns"] = warp_turns;
  985. }
  986. if (galaxy.meta["info"]["LongRange Scan"]) {
  987. std::string scanner_text =
  988. galaxy.meta["info"]["LongRange Scan"].as<std::string>();
  989. char scanner = scanner_text[0];
  990. BUGZ_LOG(fatal) << "Scanner: " << scanner;
  991. galaxy.meta["ship"]["scanner"] = scanner;
  992. }
  993. // turns isn't ship specific
  994. if (galaxy.meta["info"]["Turns left"]) {
  995. // OR this could be "Unlimited" !!!
  996. std::string text = galaxy.meta["info"]["Turns left"].as<std::string>();
  997. if (text == "Unlimited") {
  998. galaxy.meta["turns"] = -1;
  999. } else {
  1000. int turns =
  1001. stoi(text); // galaxy.meta["info"]["Turns left"].as<int>();
  1002. BUGZ_LOG(fatal) << "Turns left: " << turns;
  1003. galaxy.meta["turns"] = turns;
  1004. }
  1005. }
  1006. if (galaxy.meta["info"]["Current Sector"]) {
  1007. int sector = galaxy.meta["info"]["Current Sector"].as<int>();
  1008. BUGZ_LOG(fatal) << "Sector: " << sector;
  1009. // it should already be sector ...
  1010. current_sector = sector;
  1011. }
  1012. if (galaxy.meta["info"]["Credits"]) {
  1013. std::string credit_text =
  1014. galaxy.meta["info"]["Credits"].as<std::string>();
  1015. replace(credit_text, ",", "");
  1016. int credits = stoi(credit_text);
  1017. galaxy.meta["credits"] = credits;
  1018. BUGZ_LOG(fatal) << "Credits: " << credits;
  1019. }
  1020. }
  1021. return;
  1022. }
  1023. // info to parse:
  1024. size_t pos = line.find(" : ");
  1025. if ((!endswith(line, " : ")) && (pos != line.npos)) {
  1026. std::string key = line.substr(0, pos);
  1027. // Ferrengi ships don't have date built
  1028. std::string value = line.substr(pos + 3);
  1029. trim(key);
  1030. trim(value);
  1031. galaxy.meta["info"][key] = value;
  1032. BUGZ_LOG(fatal) << "Info: " << key << " : " << value;
  1033. }
  1034. }
  1035. void Director::SL_computer_portline(const std::string &line) {
  1036. if (startswith(line, "What sector is the port in?"))
  1037. computer_port_done = false;
  1038. if (line == "I have no information about a port in that sector.") {
  1039. computer_port_sector = 0;
  1040. SL_parser = nullptr;
  1041. return;
  1042. }
  1043. if (!computer_port_done) {
  1044. if (in(line, "Fuel Ore")) computer_port_done = true;
  1045. if (in(line, "Cargo holds")) {
  1046. // If I want to scan the class type 0 ports:
  1047. // computer_port_done = true;
  1048. // otherwise:
  1049. SL_parser = nullptr;
  1050. return;
  1051. }
  1052. }
  1053. if (computer_port_done) {
  1054. if (line.empty()) {
  1055. SL_parser = nullptr;
  1056. return;
  1057. }
  1058. // scan for items of interest
  1059. // SL: [Fuel Ore Buying 810 100% 0]
  1060. // SL: [Organics Buying 856 57% 0]
  1061. // SL: [Equipment Selling 922 44% 0]
  1062. std::string work = line;
  1063. replace(work, "Fuel Ore", "Fuel");
  1064. replace(work, "%", "");
  1065. auto parts = split(work);
  1066. char c = tolower(parts[0][0]);
  1067. int pos;
  1068. char foe[4] = "foe";
  1069. for (pos = 0; pos < 3; ++pos) {
  1070. if (c == foe[pos]) break;
  1071. }
  1072. int amount = stoi(parts[2]);
  1073. int percent = stoi(parts[3]);
  1074. // update port
  1075. auto port = galaxy.ports.find(computer_port_sector);
  1076. if (port != galaxy.ports.end()) {
  1077. BUGZ_LOG(info) << "COM PORT " << computer_port_sector << " " << c << " "
  1078. << amount << " " << percent;
  1079. port->second.amount[pos] = amount;
  1080. port->second.percent[pos] = percent;
  1081. }
  1082. }
  1083. }