dispatchers.cpp 33 KB

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