dispatchers.cpp 33 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224
  1. #include "dispatchers.h"
  2. #include <boost/format.hpp>
  3. #include <cctype>
  4. #include "boxes.h"
  5. #include "logging.h"
  6. #include "utils.h"
  7. Dispatch::Dispatch(Director &d) : director{d} {};
  8. Dispatch::~Dispatch(){};
  9. void Dispatch::to_server(const std::string &send) { director.to_server(send); }
  10. void Dispatch::to_client(const std::string &send) { director.to_client(send); }
  11. const std::string &Dispatch::get_prompt(void) {
  12. return director.current_prompt;
  13. }
  14. void Dispatch::setNotify(notifyFunc nf) { notify_ = nf; }
  15. void Dispatch::notify(void) {
  16. if (director.post) {
  17. director.post(notify_);
  18. }
  19. }
  20. void Dispatch::chain_client_input(const std::string &input) {
  21. if (chain) {
  22. chain->chain_client_input(input);
  23. } else {
  24. client_input(input);
  25. }
  26. }
  27. void Dispatch::chain_server_line(const std::string &line,
  28. const std::string &raw_line) {
  29. if (chain) {
  30. chain->chain_server_line(line, raw_line);
  31. } else {
  32. server_line(line, raw_line);
  33. }
  34. }
  35. void Dispatch::chain_server_prompt(const std::string &prompt) {
  36. if (chain) {
  37. chain->chain_server_prompt(prompt);
  38. } else {
  39. server_prompt(prompt);
  40. }
  41. }
  42. void Dispatch::server_line(const std::string &line,
  43. const std::string &raw_line) {}
  44. void Dispatch::server_prompt(const std::string &prompt) {}
  45. void Dispatch::client_input(const std::string &input) {}
  46. #ifdef NOMORE
  47. MainDispatch::MainDispatch(Director &d) : Dispatch{d}, id{d}, md{d} {
  48. BUGZ_LOG(warning) << "MainDispatch()";
  49. }
  50. MainDispatch::~MainDispatch() { BUGZ_LOG(warning) << "~MainDispatch()"; }
  51. void MainDispatch::activate(void) {
  52. count = 0;
  53. old_prompt = get_prompt();
  54. /*
  55. ╔══════════════════════════════╗
  56. ║ TradeWars Proxy Active ║
  57. ╚══════════════════════════════╝
  58. -=>
  59. */
  60. Boxes box(30, 1, true);
  61. box.boxcolor = "\x1b[1;33;44m";
  62. box.textcolor = "\x1b[1;33;44m";
  63. to_client("\n\r");
  64. to_client(box.top());
  65. std::string output = " TradeWars Proxy \x1b[5mActive\x1b[0;1;33;44m ";
  66. to_client(box.row(output));
  67. to_client(box.bottom());
  68. // to_client("\n\r\x1b[1;34mWELCOME! You are now in the proxy zone...\n\r");
  69. id.prompt = "\x1b[0m \x1b[1;33;44m-=>\x1b[0m \x1b[1;37;44m";
  70. id.max_length = 15;
  71. id.setNotify([this]() { this->have_input(); });
  72. id.activate();
  73. }
  74. void MainDispatch::have_input(void) {
  75. ++count;
  76. std::string output =
  77. str(boost::format("Your Input (%2%): [%1%]\n\r") % id.input % count);
  78. to_client("\x1b[0m");
  79. to_client(output);
  80. if (id.input == "?") {
  81. // Maybe? Maybe not.
  82. }
  83. if (id.input == "menu") {
  84. md.menu_box_color = "\x1b[1;33;44m";
  85. md.menu_text_color = "\x1b[1;37;44m";
  86. md.menu_title = "Proxy Menu";
  87. md.menu_options_color = "\x1b[1;36;40m";
  88. // md.menu_prompt = " --==>> ";
  89. // bold white to white --- black to green
  90. // md.menu_prompt = "\x1b[1;37;47m\xdb\xb2\xb1\xb0\x1b[0;30;47m RED GREEN
  91. // \x1b[30;42m\xdb\xb2\xb1\xb0\x1b[0m : ";
  92. // md.menu_prompt = "\x1b[0;31;47m\xdb\xb2\xb1\xb0\x1b[0;30;47m RED
  93. // GREEN\x1b[37;42m\xdb\xb2\xb1\xb0\x1b[0m : ";
  94. const char *CP437_GRADIENT = "\xdb\xb2\xb1\xb0 "; // 100, 75, 50, 25, 0
  95. md.menu_prompt =
  96. "\x1b[0;31;40m\xdb\xb2\xb1\xb0 \x1b[31;40mRED "
  97. "\x1b[32;40mGREEN\x1b[30;42m\xdb\xb2\xb1\xb0 \x1b[0m : ";
  98. md.lazy = true;
  99. md.menu = {{"A", "Apple"}, {"B", "Blue"}, {"R", "Rabbit"}, {"Z", "ZOOO!"}};
  100. md.setNotify([this]() { this->menu_choice(); });
  101. md.activate();
  102. return;
  103. }
  104. if (id.input == "menu2") {
  105. md.lazy = false;
  106. md.setNotify([this]() { this->menu_choice(); });
  107. md.activate();
  108. return;
  109. }
  110. if (id.input.empty()) {
  111. // if (count >= 5) {
  112. auto lines = Boxes::alert(" Returning you to the game... ", "",
  113. "\x1b[1;32m", 30, 1, true);
  114. // I'm not setting the box color, so the last color bleeds over.
  115. to_client("\x1b[0m");
  116. for (auto line : lines) {
  117. to_client(line);
  118. };
  119. to_client("Returning you to the game...\n\r");
  120. deactivate();
  121. } else {
  122. // prompt it again, sam.
  123. id.setNotify([this]() { this->have_input(); });
  124. id.activate();
  125. }
  126. }
  127. void MainDispatch::menu_choice(void) {
  128. if (md.input.empty()) {
  129. to_client("Menu abort.\n\r");
  130. } else {
  131. std::string text = "Back from menu [";
  132. text.append(md.input);
  133. text.append("] was your selection.\n\r");
  134. to_client(text);
  135. }
  136. id.max_length = 15;
  137. id.setNotify([this]() { this->have_input(); });
  138. id.activate();
  139. }
  140. void MainDispatch::deactivate(void) {
  141. // Since we're the main thing there --
  142. // sess->show_client = true;
  143. // sess->talk_direct = true;
  144. notify();
  145. }
  146. void MainDispatch::server_line(const std::string &line,
  147. const std::string &raw_line) {
  148. BUGZ_LOG(info) << "MDSL: " << line;
  149. to_client("SL: ");
  150. to_client(line);
  151. to_client("\n\r");
  152. }
  153. void MainDispatch::server_prompt(const std::string &prompt) {
  154. BUGZ_LOG(info) << "MDSP: " << prompt;
  155. }
  156. #ifdef NEVERMORE
  157. void MainDispatch::client_input(const std::string &input) {
  158. // I don't care what the old prompt looked liked at this point.
  159. BUGZ_LOG(warning) << "Got: " << input; // << " prompt=" << get_prompt();
  160. // Serious problem when the input = "\x1b" ESC. The output gets gummed/locked
  161. // up.
  162. if (input == "\x1b") {
  163. return;
  164. }
  165. ++count;
  166. std::string output = str(boost::format("MSG %1%: [%2%]\n\r") % count % input);
  167. to_client(output);
  168. if (count >= 5) {
  169. to_client("And we're outta here!\n\r");
  170. deactivate();
  171. }
  172. }
  173. #endif
  174. #endif
  175. InputDispatch::InputDispatch(Director &d) : Dispatch(d) {
  176. BUGZ_LOG(warning) << "InputDispatch()";
  177. }
  178. InputDispatch::~InputDispatch() { BUGZ_LOG(warning) << "~InputDispatch()"; }
  179. void InputDispatch::activate(void) {
  180. BUGZ_LOG(warning) << "InputDispatch::activate() " << max_length;
  181. input.clear();
  182. to_client(prompt);
  183. }
  184. void InputDispatch::deactivate(void) { notify(); }
  185. void InputDispatch::server_line(const std::string &line,
  186. const std::string &raw_line) {
  187. if (line.empty()) return;
  188. std::string temp = repr(raw_line);
  189. BUGZ_LOG(fatal) << "Input:SL(" << temp << ")";
  190. if (startswith(line, "Command [TL=")) {
  191. return;
  192. }
  193. /*
  194. temp = raw_line;
  195. clean_string(temp);
  196. BUGZ_LOG(fatal) << "InputDispatch::server_line(" << temp << ")";
  197. */
  198. temp = prompt;
  199. ansi_clean(temp);
  200. size_t total = temp.length() + input.length();
  201. to_client("\x1b[0m"); // reset colors
  202. while (total > 0) {
  203. to_client("\b \b");
  204. --total;
  205. }
  206. // Lines line "\[[1A\[[1;36mphil \[[0;32mwarps into the sector.\[[0m"
  207. temp = raw_line;
  208. replace(temp, "\x1b[1A", "");
  209. // replace(temp, "\x1[2J", "");
  210. to_client(temp);
  211. to_client("\n\r");
  212. // Doesn't matter if it is one or two calls.
  213. temp = prompt;
  214. temp.append(input);
  215. to_client(temp);
  216. // to_client(prompt);
  217. // to_client(input);
  218. }
  219. // void InputDispatch::server_prompt(const std::string &prompt) {}
  220. void InputDispatch::client_input(const std::string &cinput) {
  221. // BUGZ_LOG(info) << "InputDispatch::client_input(" << cinput << ")";
  222. for (const char ch : cinput) {
  223. if (isprint(ch)) {
  224. // Ok!
  225. if (input.length() < max_length) {
  226. to_client(std::string(1, ch));
  227. input += ch;
  228. }
  229. } else if ((ch == '\b') || (ch == 0x7f)) {
  230. // Backspace or rubout
  231. if (input.length() > 0) {
  232. to_client("\b \b");
  233. input.erase(input.size() - 1);
  234. }
  235. } else if (ch == '\r') {
  236. // Ok, we're done!
  237. BUGZ_LOG(info) << "InputDispatch done: " << input;
  238. to_client("\x1b[0m\n\r");
  239. deactivate();
  240. }
  241. }
  242. }
  243. /**
  244. * Menu Dispatch
  245. *
  246. * Two types of menus:
  247. * lazy: display the menu name and show prompt. ? shows menu.
  248. * non-lazy: displays menu + prompts.
  249. *
  250. *
  251. */
  252. MenuDispatch::MenuDispatch(Director &d) : Dispatch(d) {
  253. BUGZ_LOG(warning) << "MenuDispatch()";
  254. }
  255. MenuDispatch::~MenuDispatch() { BUGZ_LOG(warning) << "~MenuDispatch()"; }
  256. void MenuDispatch::activate(void) {
  257. calculate_widths();
  258. input.clear();
  259. BUGZ_LOG(warning) << "MenuDispatch::activate() " << max_width << ", "
  260. << max_option_width;
  261. if (lazy)
  262. menubox();
  263. else
  264. help();
  265. to_client(menu_prompt);
  266. }
  267. void MenuDispatch::deactivate(void) { notify(); }
  268. void MenuDispatch::help(void) {
  269. size_t max = max_width;
  270. Boxes mbox(max, 1, true);
  271. mbox.boxcolor = menu_box_color;
  272. if (lazy) {
  273. // just the menu
  274. mbox.textcolor = menu_options_color;
  275. to_client(mbox.top());
  276. for (auto const &menu_item : menu) {
  277. std::string text = " ";
  278. text.append(menu_item.first);
  279. text.append(" - ");
  280. text.append(menu_item.second);
  281. while (text.length() < max) text.append(1, ' ');
  282. to_client(mbox.row(text));
  283. }
  284. to_client(mbox.bottom());
  285. } else {
  286. // full menu
  287. mbox.textcolor = menu_text_color;
  288. to_client(mbox.top());
  289. std::string title = centered(max, menu_title);
  290. BUGZ_LOG(debug) << "help max=" << max << " [" << title << "]";
  291. to_client(mbox.row(title));
  292. to_client(mbox.middle());
  293. mbox.textcolor = menu_options_color;
  294. for (auto const &menu_item : menu) {
  295. std::string text = " ";
  296. text.append(menu_item.first);
  297. text.append(" - ");
  298. text.append(menu_item.second);
  299. while (text.length() < max) text.append(1, ' ');
  300. to_client(mbox.row(text));
  301. }
  302. to_client(mbox.bottom());
  303. }
  304. }
  305. std::string MenuDispatch::centered(int length, const std::string &s) {
  306. std::string text = s;
  307. size_t leftovers = length - text.length();
  308. int count = leftovers / 2;
  309. if (count > 0) {
  310. text.insert(0, count, ' ');
  311. text.append(count, ' ');
  312. }
  313. if (leftovers % 1 == 1) text.append(1, ' ');
  314. return text;
  315. }
  316. void MenuDispatch::menubox(void) {
  317. // just the menu box
  318. std::string title = centered(max_width, menu_title);
  319. /*
  320. int leftovers = max - menu_title.length();
  321. while (leftovers > 2) {
  322. title.insert(0, 1, " ");
  323. title.append(1, " ");
  324. leftovers -= 2;
  325. };
  326. if (leftovers == 1)
  327. title.append(1, " ");
  328. */
  329. BUGZ_LOG(debug) << "menubox max=" << max_width << " [" << title << "]";
  330. auto abox =
  331. Boxes::alert(title, menu_box_color, menu_text_color, max_width, 1, true);
  332. for (auto line : abox) {
  333. to_client(line);
  334. }
  335. }
  336. void MenuDispatch::calculate_widths(void) {
  337. max_width = menu_title.length() + 2;
  338. max_option_width = 0;
  339. for (auto key : menu) {
  340. size_t menu_line_length =
  341. 1 + key.first.length() + 3 + key.second.length() + 1;
  342. if (menu_line_length > max_width) max_width = menu_line_length;
  343. if (key.first.length() > max_option_width)
  344. max_option_width = key.first.length();
  345. }
  346. instant = max_option_width == 1;
  347. }
  348. void MenuDispatch::server_line(const std::string &line,
  349. const std::string &raw_line) {
  350. // TODO:
  351. // Clear prompt, display raw server line, restore prompt.
  352. if (line.empty()) return;
  353. std::string temp = repr(raw_line);
  354. BUGZ_LOG(fatal) << "Input:SL(" << temp << ")";
  355. if (startswith(line, "Command [TL=")) {
  356. return;
  357. }
  358. /*
  359. temp = raw_line;
  360. clean_string(temp);
  361. BUGZ_LOG(fatal) << "InputDispatch::server_line(" << temp << ")";
  362. */
  363. temp = menu_prompt;
  364. ansi_clean(temp);
  365. size_t total = temp.length() + input.length();
  366. to_client("\x1b[0m"); // reset colors
  367. while (total > 0) {
  368. to_client("\b \b");
  369. --total;
  370. }
  371. // Lines line "\[[1A\[[1;36mphil \[[0;32mwarps into the sector.\[[0m"
  372. temp = raw_line;
  373. replace(temp, "\x1b[1A", "");
  374. // replace(temp, "\x1[2J", "");
  375. to_client(temp);
  376. to_client("\n\r");
  377. // Doesn't matter if it is one or two calls.
  378. temp = menu_prompt;
  379. temp.append(input);
  380. to_client(temp);
  381. // to_client(prompt);
  382. // to_client(input);
  383. }
  384. void MenuDispatch::client_input(const std::string &cinput) {
  385. for (auto const ch : cinput) {
  386. // not likely that we'd have more then one,
  387. // but deal with it correctly.
  388. if (ch == '\r') {
  389. // enter
  390. if (instant) return;
  391. for (auto const &mnu : menu) {
  392. if (mnu.first == input) {
  393. to_client("\x1b[0m\n\r");
  394. deactivate();
  395. return;
  396. }
  397. }
  398. // input wasn't found ?
  399. while (input.length() > 0) {
  400. to_client("\b \b");
  401. input.erase(input.length() - 1);
  402. }
  403. return; // don't continue ...
  404. }
  405. if (ch == '\x1b') {
  406. // [ESC] - erase the input string
  407. while (input.length() > 0) {
  408. to_client("\b \b");
  409. input.erase(input.length() - 1);
  410. }
  411. // Exit - allow escape from menu
  412. deactivate();
  413. return;
  414. }
  415. if (ch == '\b') {
  416. if (input.length() > 0) {
  417. to_client("\b \b");
  418. input.erase(input.length() - 1);
  419. }
  420. }
  421. if (ch == '?') {
  422. to_client(cinput); // display what they entered.
  423. to_client("\x1b[0m\n\r");
  424. help();
  425. to_client(menu_prompt);
  426. return;
  427. }
  428. if (isprint(ch)) {
  429. char c = ch;
  430. if (!case_sensitive) c = toupper(ch);
  431. // ok, it's a printable character
  432. if (input.length() < max_option_width) {
  433. // ok, there's room.
  434. to_client(std::string(1, c));
  435. input.append(1, c);
  436. }
  437. if (instant) {
  438. for (auto const &mnu : menu) {
  439. if (mnu.first == input) {
  440. to_client("\x1b[0m\n\r");
  441. deactivate();
  442. return;
  443. }
  444. }
  445. // Erase last character, wasn't an option.
  446. to_client("\b \b");
  447. input.erase(input.length() - 1);
  448. }
  449. }
  450. }
  451. }
  452. CIMDispatch::CIMDispatch(Director &d) : Dispatch(d) {}
  453. void CIMDispatch::activate(void) { count = 0; }
  454. void CIMDispatch::deactivate(void) { notify(); }
  455. void CIMDispatch::server_line(const std::string &line,
  456. const std::string &raw_line) {
  457. if (!((line.empty() || startswith(line, ": ") ||
  458. startswith(line, "Command [")))) {
  459. count++;
  460. if (count % 100 == 0) {
  461. std::string message = str(boost::format("\r%1%") % count);
  462. to_client(message);
  463. }
  464. }
  465. if (line == ": ENDINTERROG") {
  466. std::string message = str(boost::format("\r%1%\n\r") % count);
  467. to_client(message);
  468. deactivate();
  469. }
  470. }
  471. MoveDispatch::MoveDispatch(Director &d) : Dispatch(d) {
  472. BUGZ_LOG(warning) << "MoveDispatch()";
  473. }
  474. // sector_type move_to;
  475. void MoveDispatch::activate(void) {
  476. starting = director.current_sector;
  477. BUGZ_LOG(warning) << "Moving from " << starting << " to " << move_to;
  478. // Start with density scan
  479. to_server("SD");
  480. state = 1;
  481. warp_lane.clear();
  482. warp_pos = 0;
  483. use_express = false;
  484. // build final string to match against
  485. at_destination = "Auto Warping to sector ";
  486. at_destination.append(std::to_string(move_to));
  487. }
  488. void MoveDispatch::deactivate(void) { notify(); }
  489. // optional here
  490. void MoveDispatch::server_line(const std::string &line,
  491. const std::string &raw_line) {
  492. if (state == 1) {
  493. if (endswith(line, "Relative Density Scan")) {
  494. state = 2;
  495. }
  496. if (line == "You don't have a long range scanner.") {
  497. // Ok, we'll have to express move our way there then.
  498. use_express = true;
  499. // state = 2;
  500. std::string command = str(boost::format("M%1%\r") % move_to);
  501. to_server(command);
  502. state = 3;
  503. }
  504. }
  505. if ((state != 2) && (state != 5)) {
  506. // hide the density scan part
  507. std::string temp = raw_line;
  508. if (line == "<Auto Pilot Engaging>") {
  509. // trim out the stray cursor home
  510. replace(temp, "\x1b[H", "");
  511. }
  512. // Replace progress bar with something else.
  513. if (startswith(temp, "\x1b[1;33m\xb3"))
  514. temp = "\x1b[1A\x1b[1;33m** SLOW MOVE **";
  515. /*
  516. if (in(temp, "\xb3")) {
  517. std::string debug = repr(temp);
  518. BUGZ_LOG(fatal) << "MOVE LINE: " << debug;
  519. }
  520. */
  521. // Don't output raw_lines that move the cursor up 3.
  522. if (!in(temp, "\x1b[3A")) {
  523. temp.append("\n\r");
  524. to_client(temp);
  525. }
  526. }
  527. if (state == 3) {
  528. if (line == "You are already in that sector!") {
  529. success = true;
  530. deactivate();
  531. return;
  532. }
  533. if (line == "That Warp Lane is not adjacent.") {
  534. // ok! Parse out the path that we need to take...
  535. }
  536. // [611 > 612 > 577 > 543 > 162 > 947 > 185 > 720 > 894 > 3 > 1]
  537. // multiple lines possible here? Yes.
  538. // [344 > 23328 > 2981 > 10465 > 14016 > 8979 > 1916 > 18734 > 5477 > 131 >
  539. // 27464 >] watch for <Move> it contains >
  540. if ((line != "<Move>") && in(line, " > ")) {
  541. bool more = false;
  542. std::string work = line;
  543. if (endswith(work, " >")) {
  544. more = true;
  545. work = work.substr(0, work.length() - 2);
  546. }
  547. replace(work, " > ", " ");
  548. replace(work, "(", "");
  549. replace(work, ")", "");
  550. auto warps = split(work);
  551. for (auto const &w : warps) {
  552. BUGZ_LOG(fatal) << "lane: " << w;
  553. warp_lane.push_back(stoi(w));
  554. }
  555. if (!more) state = 4;
  556. }
  557. }
  558. if (state == 4) {
  559. if (line == at_destination) {
  560. // [Auto Warping to sector 1]
  561. state = 6;
  562. }
  563. }
  564. }
  565. bool MoveDispatch::density_clear(density d) { // int sector, int density) {
  566. if (d.sector == 0) return false;
  567. switch (d.density) {
  568. case 0:
  569. case 1:
  570. case 100:
  571. case 101:
  572. return true;
  573. }
  574. // special case for sector 1:
  575. if ((d.sector == 1) && (d.density == 601)) return true;
  576. return false;
  577. }
  578. void MoveDispatch::server_prompt(const std::string &prompt) {
  579. BUGZ_LOG(fatal) << "server_prompt: " << prompt;
  580. if (state == 2) {
  581. if (at_command_prompt(prompt)) {
  582. // Ok, density is done
  583. // BUG: If the sector is adjacent, this doesn't check density!
  584. density d = director.galaxy.dscan.find(move_to);
  585. if (d.sector == move_to) {
  586. // Yes! we found the sector in the scan!
  587. if (!density_clear(d)) {
  588. BUGZ_LOG(fatal) << "Failed density check on single move.";
  589. success = false;
  590. deactivate();
  591. return;
  592. }
  593. }
  594. std::string command = str(boost::format("M%1%\r") % move_to);
  595. to_server(command);
  596. state = 3;
  597. }
  598. } else if (state == 3) {
  599. if (at_command_prompt(prompt)) {
  600. // this happens when the sector is adject to current_sector.
  601. BUGZ_LOG(fatal) << "Are we there yet?";
  602. success = true;
  603. deactivate();
  604. return;
  605. }
  606. } else if (state == 4) {
  607. if (prompt == "Engage the Autopilot? (Y/N/Single step/Express) [Y] ") {
  608. if (use_express) {
  609. BUGZ_LOG(fatal) << "Using Express";
  610. to_server("E");
  611. } else {
  612. int to_check = warp_lane[warp_pos + 1];
  613. // check density scan
  614. density d = director.galaxy.dscan.find(to_check);
  615. /*
  616. int density =
  617. director.galaxy.meta["density"][to_check]["density"].as<int>();
  618. */
  619. if (density_clear(d)) { // to_check, density)) {
  620. to_server("S");
  621. ++warp_pos;
  622. } else {
  623. to_server("N");
  624. BUGZ_LOG(fatal) << "density_clear(" << to_check << ") : false";
  625. success = false;
  626. deactivate();
  627. }
  628. }
  629. }
  630. if (prompt == "Stop in this sector (Y,N,E,I,R,S,D,P,?) (?=Help) [N] ? ") {
  631. state = 5;
  632. to_server("SD");
  633. }
  634. } else if (state == 5) {
  635. // finished scan
  636. if (prompt == "Stop in this sector (Y,N,E,I,R,S,D,P,?) (?=Help) [N] ? ") {
  637. int to_check = warp_lane[warp_pos + 1];
  638. density d = director.galaxy.dscan.find(to_check);
  639. /*
  640. int density =
  641. director.galaxy.meta["density"][to_check]["density"].as<int>();
  642. */
  643. if (density_clear(d)) {
  644. to_server("N");
  645. ++warp_pos;
  646. state = 4;
  647. } else {
  648. to_server("Y");
  649. BUGZ_LOG(fatal) << "Stopped by density: " << to_check;
  650. success = 0;
  651. deactivate();
  652. }
  653. }
  654. } else if (state == 6) {
  655. if (at_command_prompt(prompt)) {
  656. // We're done!
  657. success = 1;
  658. deactivate();
  659. }
  660. }
  661. }
  662. void MoveDispatch::client_input(const std::string &input) {
  663. // This exits us out quickly -- should I stop at a better spot?
  664. // As in answer "Y" to Stop in this sector?
  665. success = 0;
  666. deactivate();
  667. }
  668. TraderDispatch::TraderDispatch(Director &d) : Dispatch(d) {
  669. BUGZ_LOG(fatal) << "TraderDispatch()";
  670. state = 0;
  671. };
  672. TraderDispatch::~TraderDispatch() { BUGZ_LOG(fatal) << "~TraderDispatch()"; }
  673. void TraderDispatch::activate(void) {
  674. // ok, lookup port1 port2
  675. BUGZ_LOG(fatal) << "STraderDispatch::activate " << port[0] << " & " << port[1];
  676. auto port_info = director.galaxy.ports.find(port[0]);
  677. int port0_type = port_info->second.type;
  678. port_buysell[0] = get_buysell(port0_type);
  679. // Special case - we just want to buy resources
  680. if (port[1] != 0) {
  681. port_info = director.galaxy.ports.find(port[1]);
  682. int port1_type = port_info->second.type;
  683. port_buysell[1] = get_buysell(port1_type);
  684. BUGZ_LOG(fatal) << port0_type << " and " << port1_type;
  685. } else {
  686. BUGZ_LOG(fatal) << "Just buy from " << port[0];
  687. }
  688. /*
  689. auto ttr = director.galaxy.trade_type_info(port0_type, port1_type);
  690. trades = ttr.trades;
  691. */
  692. /*
  693. if (trades.foe[0] && trades.foe[1] && trades.foe[2]) {
  694. // it has all three -- use the last 2.
  695. trades.foe[0] = false;
  696. }
  697. */
  698. // Ok, what do we do first here?
  699. // I - Info
  700. state = 1;
  701. percent = 5.0;
  702. to_server("I");
  703. director.galaxy.meta["help"]["stop_percent"] =
  704. "ScriptTrader stop trading if below this percent.";
  705. if (director.galaxy.config["stop_percent"]) {
  706. stop_percent = director.galaxy.config["stop_percent"].as<int>();
  707. } else {
  708. stop_percent = 20;
  709. director.galaxy.config["stop_percent"] = stop_percent;
  710. }
  711. director.galaxy.meta["help"]["trade_end_empty"] =
  712. "ScriptTrader end trades with empty holds? Y/N";
  713. if (director.galaxy.config["trade_end_empty"]) {
  714. std::string tee =
  715. director.galaxy.config["trade_end_empty"].as<std::string>();
  716. if ((toupper(tee[0]) == 'Y') || (toupper(tee[0]) == 'T')) {
  717. trade_end_empty = true;
  718. } else {
  719. trade_end_empty = false;
  720. // director.galaxy.config["trade_end_empty"] = "N";
  721. }
  722. } else {
  723. trade_end_empty = false;
  724. director.galaxy.config["trade_end_empty"] = "N";
  725. }
  726. }
  727. void TraderDispatch::deactivate(void) { notify(); }
  728. void TraderDispatch::server_line(const std::string &line,
  729. const std::string &raw_line) {
  730. // FUTURE: powering up weapons check
  731. // Show what's going on...
  732. if (state > 1) {
  733. std::string temp = raw_line;
  734. temp.append("\n\r");
  735. to_client(temp);
  736. }
  737. if (line == "Docking...") {
  738. last_offer = 0;
  739. final_offer = 0;
  740. initial_offer = 0;
  741. }
  742. static std::set<std::string> success_lines = {
  743. "If only more honest traders would port here, we'll take them though.",
  744. "You will put me out of business, I'll take your offer.",
  745. "FINE, we'll take them, just leave!",
  746. "Agreed, and a pleasure doing business with you!",
  747. "You are a rogue! We'll take them anyway.",
  748. "You insult my intelligence, but we'll buy them anyway.",
  749. "Very well, we'll take that offer.",
  750. "You drive a hard bargain, but we'll take them.",
  751. "Done, we'll take the lot.",
  752. "I hate haggling, they're all yours.",
  753. "You are robbing me, but we'll buy them anyway.",
  754. "SOLD! Come back anytime!",
  755. "Cheapskate. Here, take them and leave me alone.",
  756. "Very well, we'll buy them.",
  757. "You are a shrewd trader, they're all yours.",
  758. "I could have twice that much in the Androcan Empire, but they're yours.",
  759. "Oh well, maybe I can sell these to some other fool, we'll take them.",
  760. "I PAID more than that! But we'll sell them to you anyway.",
  761. "(Sigh) Very well, pay up and take them away.",
  762. "Agreed! We'll purchase them!"};
  763. if (success_lines.find(line) != success_lines.end()) {
  764. BUGZ_LOG(fatal) << "Success " << buying << " " << initial_offer << " : "
  765. << last_offer;
  766. // calculate % ?
  767. BUGZ_LOG(fatal) << "% " << (float)initial_offer / (float)last_offer * 100.0;
  768. BUGZ_LOG(fatal) << "meta trade setting: " << percent << " for "
  769. << active_port << " " << product;
  770. director.galaxy.meta["trade"][active_port][product] = percent;
  771. }
  772. // <P-Probe estimates your offer was
  773. if (startswith(line, "Agreed, ")) {
  774. last_offer = 0;
  775. final_offer = 0;
  776. if (director.galaxy.meta["trade"][active_port][product]) {
  777. percent = director.galaxy.meta["trade"][active_port][product].as<float>();
  778. percent += 1.0;
  779. BUGZ_LOG(fatal) << "Percent for " << active_port << " now " << percent;
  780. } else {
  781. BUGZ_LOG(fatal) << "using default for " << active_port;
  782. percent = 5.0; // check meta for past trades information
  783. }
  784. }
  785. if (startswith(line, "We'll buy them for ")) {
  786. // I need the initial offer!
  787. std::string offer = line.substr(19);
  788. replace(offer, ",", "");
  789. initial_offer = stoi(offer);
  790. BUGZ_LOG(fatal) << "Buying, initial: " << initial_offer;
  791. buying = true; // Port is buying, we are selling.
  792. }
  793. if (startswith(line, "We'll sell them for ")) {
  794. // I need the initial offer!
  795. std::string offer = line.substr(20);
  796. replace(offer, ",", "");
  797. initial_offer = stoi(offer);
  798. BUGZ_LOG(fatal) << "Selling, initial: " << initial_offer;
  799. buying = false; // Port is selling, we are buying.
  800. }
  801. // SL: [Our final offer is 1,263 credits.]
  802. if (startswith(line, "Our final offer is ")) {
  803. // Well snap!
  804. std::string offer = line.substr(19);
  805. replace(offer, ",", "");
  806. final_offer = stoi(offer);
  807. BUGZ_LOG(fatal) << "Final offer: " << final_offer;
  808. }
  809. // SL: [You have 16,767 credits and 0 empty cargo holds.]
  810. // trade accepted. if not 0 empty cargo holds -- we failed!
  811. // SL: [<P-Probe estimates your offer was 91.83% of best price>]
  812. // SL: [You have 4,046 credits and 0 empty cargo holds.]
  813. // this shows up at the initial docking of the port.
  814. if (startswith(line, "You have ")) {
  815. if (initial_offer != 0) {
  816. // Ok, the offer was possibly accepted.
  817. int success;
  818. if (buying)
  819. success = 0;
  820. else
  821. success = director.galaxy.meta["ship"]["holds"]["total"].as<int>();
  822. std::string text = std::to_string(success);
  823. text.append(" empty cargo holds.");
  824. if (endswith(line, text)) {
  825. BUGZ_LOG(fatal) << "Trade SUCCESS!";
  826. // record this action somewhere in meta.
  827. }
  828. }
  829. }
  830. }
  831. void TraderDispatch::server_prompt(const std::string &prompt) {
  832. // FUTURE: check for "Surrender/Attack"
  833. if (at_command_prompt(prompt)) {
  834. if (state == 1) {
  835. // Ok, decision time!
  836. if (director.galaxy.meta["ship"]["holds"]["c"]) {
  837. // holds contain colonists
  838. to_client("ScriptTrader FAIL: holds contain colonists.");
  839. deactivate();
  840. return;
  841. }
  842. // Which port to trade with first? examine trades
  843. BUGZ_LOG(fatal) << "trades: " << trades;
  844. BUGZ_LOG(fatal) << "port0:" << text_from_buysell(port_buysell[0]);
  845. if (port[1] != 0)
  846. BUGZ_LOG(fatal) << "port1:" << text_from_buysell(port_buysell[1]);
  847. // Ok, I might still need this (so I know what port to start with)
  848. // which is selling?
  849. // must set active port!
  850. bool all_holds_empty = false;
  851. active_port = 0;
  852. // check the ship and holds here. (MAYBE)
  853. int holds = director.galaxy.meta["ship"]["holds"]["total"].as<int>();
  854. if (director.galaxy.meta["ship"]["holds"]["empty"]) {
  855. if (holds == director.galaxy.meta["ship"]["holds"]["empty"].as<int>())
  856. all_holds_empty = true;
  857. }
  858. if (port[1] == 0) {
  859. active_port = port[0];
  860. } else {
  861. if (!all_holds_empty) {
  862. for (int x = 0; x < 3; ++x) {
  863. if (director.galaxy.meta["ship"]["holds"][foe[x]]) {
  864. if (port_buysell[0].foe[x]) {
  865. active_port = port[0];
  866. break;
  867. }
  868. if (port_buysell[1].foe[x]) {
  869. active_port = port[1];
  870. }
  871. }
  872. }
  873. if (active_port == 0) {
  874. to_client(
  875. "I don't see any ports that are buying what we have in our "
  876. "holds.\n\r");
  877. deactivate();
  878. return;
  879. };
  880. } else {
  881. // all holds empty, find selling port
  882. for (int x = 0; x < 3; ++x) {
  883. if (trades.foe[x]) {
  884. if (port_buysell[0].foe[x]) {
  885. active_port = port[0];
  886. break;
  887. }
  888. if (port_buysell[1].foe[x]) {
  889. active_port = port[1];
  890. break;
  891. }
  892. }
  893. }
  894. }
  895. }
  896. state = 2;
  897. if (director.current_sector == active_port) {
  898. // begin state 3
  899. state = 3;
  900. to_client("Trading...\n\r");
  901. to_server("PT");
  902. return;
  903. } else {
  904. // initiate move
  905. std::string move = std::to_string(active_port);
  906. to_client("Moving...\n\r");
  907. move.append("\r");
  908. to_server(move);
  909. return;
  910. }
  911. }
  912. if (state == 2) {
  913. if (director.current_sector == active_port) {
  914. // We're here
  915. state = 3;
  916. to_client("Trading...\n\r");
  917. to_server("PT");
  918. return;
  919. } else {
  920. // we failed to move to where we wanted to go?!
  921. BUGZ_LOG(fatal) << "Expecting: " << active_port << " but got "
  922. << director.current_sector;
  923. deactivate();
  924. return;
  925. }
  926. }
  927. }
  928. if (state == 3) {
  929. if (startswith(prompt, "How many holds of ") && endswith(prompt, "]? ")) {
  930. char selling = tolower(prompt[18]);
  931. for (int x = 0; x < 3; ++x) {
  932. if (foe[x] == selling) product = x;
  933. }
  934. if (in(prompt, " to sell ")) {
  935. // always sell everything
  936. to_server("\r");
  937. return;
  938. }
  939. if (in(prompt, " to buy ")) {
  940. bool buy_ok = true;
  941. if (trade_end_empty) {
  942. // Ok, we want to end with empty holds...
  943. int other_port;
  944. if (active_port == port[0])
  945. other_port = port[1];
  946. else
  947. other_port = port[0];
  948. // Is target port burnt?
  949. auto pos = director.galaxy.ports.find(other_port);
  950. bool burnt = false;
  951. if (pos != director.galaxy.ports.end()) {
  952. // We'll find the port. Really.
  953. if (!pos->second.unknown()) {
  954. // port isn't unknown, so check to see if it's burnt
  955. for (int x = 0; x < 3; ++x) {
  956. if (trades.foe[x]) {
  957. if (pos->second.percent[x] < stop_percent) burnt = true;
  958. }
  959. }
  960. }
  961. }
  962. if (burnt) {
  963. buy_ok = false;
  964. }
  965. }
  966. // Ok, what are they selling?
  967. // char selling = tolower(prompt[18]);
  968. BUGZ_LOG(fatal) << "Selling: " << selling;
  969. if (!buy_ok) {
  970. // no!
  971. to_server("0\r");
  972. } else
  973. for (int x = 0; x < 3; ++x) {
  974. // if (foe[x] == selling) {
  975. // We found the item ... is it something that we're trading?
  976. if (foe[x] == selling) {
  977. if (trades.foe[x]) {
  978. // Yes!
  979. to_server("\r");
  980. product = x;
  981. } else {
  982. // No!
  983. to_server("0\r");
  984. }
  985. }
  986. }
  987. // }
  988. }
  989. }
  990. if (startswith(prompt, "Your offer [") && endswith(prompt, " ? ")) {
  991. // Ok, things get weird here. We also need to look for final offer.
  992. if (last_offer != 0) percent -= 1.0;
  993. if (buying)
  994. last_offer = (int)(initial_offer * (100 + percent) / 100.0);
  995. else
  996. last_offer = (int)(initial_offer * (100 - percent) / 100.0);
  997. BUGZ_LOG(fatal) << "Offer: " << buying << " offer " << last_offer << " % "
  998. << percent;
  999. std::string text = std::to_string(last_offer);
  1000. text.append("\r");
  1001. to_server(text);
  1002. }
  1003. if (at_command_prompt(prompt)) {
  1004. // we're done trading...
  1005. // do we carry on, or stop?
  1006. // 1.) CHECK TURNS // need turn tracking
  1007. // 2.) PORTS BURNT?
  1008. if (active_port == port[0]) {
  1009. if (port[0] == 0) {
  1010. deactivate();
  1011. return;
  1012. }
  1013. active_port = port[1];
  1014. } else
  1015. active_port = port[0];
  1016. // Is target port burnt?
  1017. auto pos = director.galaxy.ports.find(active_port);
  1018. bool burnt = false;
  1019. if (pos != director.galaxy.ports.end()) {
  1020. // We'll find the port. Really.
  1021. if (!pos->second.unknown()) {
  1022. // port isn't unknown, check to see if burnt
  1023. for (int x = 0; x < 3; ++x) {
  1024. if (trades.foe[x]) {
  1025. BUGZ_LOG(fatal) << x << " % " << (int)pos->second.percent[x]
  1026. << " " << stop_percent;
  1027. if (pos->second.percent[x] < stop_percent) burnt = true;
  1028. }
  1029. }
  1030. }
  1031. }
  1032. if (burnt) {
  1033. to_client("Ports burnt.\n\r");
  1034. deactivate();
  1035. return;
  1036. }
  1037. std::string move = std::to_string(active_port);
  1038. to_client("Moving...\n\r");
  1039. move.append("\r");
  1040. to_server(move);
  1041. state = 2;
  1042. }
  1043. }
  1044. }
  1045. void TraderDispatch::client_input(const std::string &cinput) { deactivate(); };
  1046. /*
  1047. * CoreDispatch: This is an example class that does dispatch.
  1048. * Copy this and make changes from there...
  1049. */
  1050. CoreDispatch::CoreDispatch(Director &d) : Dispatch(d) {
  1051. BUGZ_LOG(warning) << "CoreDispatch()";
  1052. }
  1053. void CoreDispatch::activate(void) {
  1054. // save things, set things
  1055. }
  1056. void CoreDispatch::deactivate(void) {
  1057. // restore things
  1058. notify();
  1059. }
  1060. void CoreDispatch::server_line(const std::string &line,
  1061. const std::string &raw_line) {}
  1062. void CoreDispatch::server_prompt(const std::string &prompt) {}
  1063. void CoreDispatch::client_input(const std::string &input) {
  1064. BUGZ_LOG(warning) << "Got: " << input << " prompt=" << get_prompt();
  1065. }