scripts.cpp 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642
  1. #include "scripts.h"
  2. #include <boost/format.hpp>
  3. #include "logging.h"
  4. #ifdef DEPRECATED_SEE_TRADER_DISPATCH
  5. ScriptTrader::ScriptTrader(Director &d) : Dispatch(d) {
  6. BUGZ_LOG(fatal) << "ScriptTrader()";
  7. state = 0;
  8. };
  9. ScriptTrader::~ScriptTrader() { BUGZ_LOG(fatal) << "~ScriptTrader()"; }
  10. void ScriptTrader::activate(void) {
  11. // ok, lookup port1 port2
  12. BUGZ_LOG(fatal) << "ScriptTrader::activate " << port[0] << " & " << port[1];
  13. auto port_info = director.galaxy.ports.find(port[0]);
  14. int port0_type = port_info->second.type;
  15. port_buysell[0] = get_buysell(port0_type);
  16. // Special case - we just want to buy resources
  17. if (port[1] != 0) {
  18. port_info = director.galaxy.ports.find(port[1]);
  19. int port1_type = port_info->second.type;
  20. port_buysell[1] = get_buysell(port1_type);
  21. BUGZ_LOG(fatal) << port0_type << " and " << port1_type;
  22. } else {
  23. BUGZ_LOG(fatal) << "Just buy from " << port[0];
  24. }
  25. /*
  26. auto ttr = director.galaxy.trade_type_info(port0_type, port1_type);
  27. trades = ttr.trades;
  28. */
  29. /*
  30. if (trades.foe[0] && trades.foe[1] && trades.foe[2]) {
  31. // it has all three -- use the last 2.
  32. trades.foe[0] = false;
  33. }
  34. */
  35. // Ok, what do we do first here?
  36. // I - Info
  37. state = 1;
  38. percent = 5.0;
  39. to_server("I");
  40. director.galaxy.meta["help"]["stop_percent"] =
  41. "ScriptTrader stop trading if below this percent.";
  42. if (director.galaxy.config["stop_percent"]) {
  43. stop_percent = director.galaxy.config["stop_percent"].as<int>();
  44. } else {
  45. stop_percent = 20;
  46. director.galaxy.config["stop_percent"] = stop_percent;
  47. }
  48. director.galaxy.meta["help"]["trade_end_empty"] =
  49. "ScriptTrader end trades with empty holds? Y/N";
  50. if (director.galaxy.config["trade_end_empty"]) {
  51. std::string tee =
  52. director.galaxy.config["trade_end_empty"].as<std::string>();
  53. if ((toupper(tee[0]) == 'Y') || (toupper(tee[0]) == 'T')) {
  54. trade_end_empty = true;
  55. } else {
  56. trade_end_empty = false;
  57. // director.galaxy.config["trade_end_empty"] = "N";
  58. }
  59. } else {
  60. trade_end_empty = false;
  61. director.galaxy.config["trade_end_empty"] = "N";
  62. }
  63. }
  64. void ScriptTrader::deactivate(void) { notify(); }
  65. void ScriptTrader::server_line(const std::string &line,
  66. const std::string &raw_line) {
  67. // FUTURE: powering up weapons check
  68. // Show what's going on...
  69. if (state > 1) {
  70. std::string temp = raw_line;
  71. temp.append("\n\r");
  72. to_client(temp);
  73. }
  74. if (line == "Docking...") {
  75. last_offer = 0;
  76. final_offer = 0;
  77. initial_offer = 0;
  78. }
  79. static std::set<std::string> success_lines = {
  80. "If only more honest traders would port here, we'll take them though.",
  81. "You will put me out of business, I'll take your offer.",
  82. "FINE, we'll take them, just leave!",
  83. "Agreed, and a pleasure doing business with you!",
  84. "You are a rogue! We'll take them anyway.",
  85. "You insult my intelligence, but we'll buy them anyway.",
  86. "Very well, we'll take that offer.",
  87. "You drive a hard bargain, but we'll take them.",
  88. "Done, we'll take the lot.",
  89. "I hate haggling, they're all yours.",
  90. "You are robbing me, but we'll buy them anyway.",
  91. "SOLD! Come back anytime!",
  92. "Cheapskate. Here, take them and leave me alone.",
  93. "Very well, we'll buy them.",
  94. "You are a shrewd trader, they're all yours.",
  95. "I could have twice that much in the Androcan Empire, but they're yours.",
  96. "Oh well, maybe I can sell these to some other fool, we'll take them.",
  97. "I PAID more than that! But we'll sell them to you anyway.",
  98. "(Sigh) Very well, pay up and take them away.",
  99. "Agreed! We'll purchase them!"};
  100. if (success_lines.find(line) != success_lines.end()) {
  101. BUGZ_LOG(fatal) << "Success " << buying << " " << initial_offer << " : "
  102. << last_offer;
  103. // calculate % ?
  104. BUGZ_LOG(fatal) << "% " << (float)initial_offer / (float)last_offer * 100.0;
  105. BUGZ_LOG(fatal) << "meta trade setting: " << percent << " for "
  106. << active_port << " " << product;
  107. director.galaxy.meta["trade"][active_port][product] = percent;
  108. }
  109. // <P-Probe estimates your offer was
  110. if (startswith(line, "Agreed, ")) {
  111. last_offer = 0;
  112. final_offer = 0;
  113. if (director.galaxy.meta["trade"][active_port][product]) {
  114. percent = director.galaxy.meta["trade"][active_port][product].as<float>();
  115. percent += 1.0;
  116. BUGZ_LOG(fatal) << "Percent for " << active_port << " now " << percent;
  117. } else {
  118. BUGZ_LOG(fatal) << "using default for " << active_port;
  119. percent = 5.0; // check meta for past trades information
  120. }
  121. }
  122. if (startswith(line, "We'll buy them for ")) {
  123. // I need the initial offer!
  124. std::string offer = line.substr(19);
  125. replace(offer, ",", "");
  126. initial_offer = stoi(offer);
  127. BUGZ_LOG(fatal) << "Buying, initial: " << initial_offer;
  128. buying = true; // Port is buying, we are selling.
  129. }
  130. if (startswith(line, "We'll sell them for ")) {
  131. // I need the initial offer!
  132. std::string offer = line.substr(20);
  133. replace(offer, ",", "");
  134. initial_offer = stoi(offer);
  135. BUGZ_LOG(fatal) << "Selling, initial: " << initial_offer;
  136. buying = false; // Port is selling, we are buying.
  137. }
  138. // SL: [Our final offer is 1,263 credits.]
  139. if (startswith(line, "Our final offer is ")) {
  140. // Well snap!
  141. std::string offer = line.substr(19);
  142. replace(offer, ",", "");
  143. final_offer = stoi(offer);
  144. BUGZ_LOG(fatal) << "Final offer: " << final_offer;
  145. }
  146. // SL: [You have 16,767 credits and 0 empty cargo holds.]
  147. // trade accepted. if not 0 empty cargo holds -- we failed!
  148. // SL: [<P-Probe estimates your offer was 91.83% of best price>]
  149. // SL: [You have 4,046 credits and 0 empty cargo holds.]
  150. // this shows up at the initial docking of the port.
  151. if (startswith(line, "You have ")) {
  152. if (initial_offer != 0) {
  153. // Ok, the offer was possibly accepted.
  154. int success;
  155. if (buying)
  156. success = 0;
  157. else
  158. success = director.galaxy.meta["ship"]["holds"]["total"].as<int>();
  159. std::string text = std::to_string(success);
  160. text.append(" empty cargo holds.");
  161. if (endswith(line, text)) {
  162. BUGZ_LOG(fatal) << "Trade SUCCESS!";
  163. // record this action somewhere in meta.
  164. }
  165. }
  166. }
  167. }
  168. void ScriptTrader::server_prompt(const std::string &prompt) {
  169. // FUTURE: check for "Surrender/Attack"
  170. if (at_command_prompt(prompt)) {
  171. if (state == 1) {
  172. // Ok, decision time!
  173. if (director.galaxy.meta["ship"]["holds"]["c"]) {
  174. // holds contain colonists
  175. to_client("ScriptTrader FAIL: holds contain colonists.");
  176. deactivate();
  177. return;
  178. }
  179. // Which port to trade with first? examine trades
  180. BUGZ_LOG(fatal) << "trades: " << trades;
  181. BUGZ_LOG(fatal) << "port0:" << text_from_buysell(port_buysell[0]);
  182. if (port[1] != 0)
  183. BUGZ_LOG(fatal) << "port1:" << text_from_buysell(port_buysell[1]);
  184. // Ok, I might still need this (so I know what port to start with)
  185. // which is selling?
  186. // must set active port!
  187. bool all_holds_empty = false;
  188. active_port = 0;
  189. // check the ship and holds here. (MAYBE)
  190. int holds = director.galaxy.meta["ship"]["holds"]["total"].as<int>();
  191. if (director.galaxy.meta["ship"]["holds"]["empty"]) {
  192. if (holds == director.galaxy.meta["ship"]["holds"]["empty"].as<int>())
  193. all_holds_empty = true;
  194. }
  195. if (port[1] == 0) {
  196. active_port = port[0];
  197. } else {
  198. if (!all_holds_empty) {
  199. for (int x = 0; x < 3; ++x) {
  200. if (director.galaxy.meta["ship"]["holds"][foe[x]]) {
  201. if (port_buysell[0].foe[x]) {
  202. active_port = port[0];
  203. break;
  204. }
  205. if (port_buysell[1].foe[x]) {
  206. active_port = port[1];
  207. }
  208. }
  209. }
  210. if (active_port == 0) {
  211. to_client(
  212. "I don't see any ports that are buying what we have in our "
  213. "holds.\n\r");
  214. deactivate();
  215. return;
  216. };
  217. } else {
  218. // all holds empty, find selling port
  219. for (int x = 0; x < 3; ++x) {
  220. if (trades.foe[x]) {
  221. if (port_buysell[0].foe[x]) {
  222. active_port = port[0];
  223. break;
  224. }
  225. if (port_buysell[1].foe[x]) {
  226. active_port = port[1];
  227. break;
  228. }
  229. }
  230. }
  231. }
  232. }
  233. #ifdef NO_JUST_TRADES
  234. // Do we have what they are buying?
  235. bool have_buy = false;
  236. int active_buy = 0;
  237. int active_sell = 0;
  238. for (int x = 0; x < 3; ++x) {
  239. if (trades.foe[x]) {
  240. // this is what they will be trading...
  241. if (director.galaxy.meta["ship"]["holds"][foe[x]]) {
  242. // key exists ...
  243. have_buy = true;
  244. // which port is buying?
  245. if (port_buysell[0].foe[x]) {
  246. active_buy = 0;
  247. have_buy = true;
  248. } else if (port_buysell[1].foe[x]) {
  249. active_buy = 1;
  250. have_buy = true;
  251. }
  252. }
  253. if (!port_buysell[0].foe[x]) {
  254. active_sell = 0;
  255. } else {
  256. active_sell = 1;
  257. }
  258. }
  259. }
  260. if (have_buy) {
  261. BUGZ_LOG(fatal) << "have_buy: port " << active_buy;
  262. active_port = port[active_buy];
  263. } else {
  264. // which port is selling?
  265. // if they aren't buying what I have in my holds, should I check to see
  266. // if my holds are full? This could be the "not buying" what I'm
  267. // setting bug!
  268. BUGZ_LOG(fatal) << "!have_buy: port " << active_sell;
  269. active_port = port[active_sell];
  270. // yes, this is the bug alright.
  271. // Ok, this shows up all the time. I need to look at my holds!
  272. if (!all_holds_empty) {
  273. to_client(
  274. "I don't see any ports that are buying what we have in our "
  275. "holds!\n\r");
  276. deactivate();
  277. return;
  278. }
  279. }
  280. #endif
  281. state = 2;
  282. if (director.current_sector == active_port) {
  283. // begin state 3
  284. state = 3;
  285. to_client("Trading...\n\r");
  286. to_server("PT");
  287. return;
  288. } else {
  289. // initiate move
  290. std::string move = std::to_string(active_port);
  291. to_client("Moving...\n\r");
  292. move.append("\r");
  293. to_server(move);
  294. return;
  295. }
  296. }
  297. if (state == 2) {
  298. if (director.current_sector == active_port) {
  299. // We're here
  300. state = 3;
  301. to_client("Trading...\n\r");
  302. to_server("PT");
  303. return;
  304. } else {
  305. // we failed to move to where we wanted to go?!
  306. BUGZ_LOG(fatal) << "Expecting: " << active_port << " but got "
  307. << director.current_sector;
  308. deactivate();
  309. return;
  310. }
  311. }
  312. }
  313. if (state == 3) {
  314. if (startswith(prompt, "How many holds of ")) {
  315. char selling = tolower(prompt[18]);
  316. for (int x = 0; x < 3; ++x) {
  317. if (foe[x] == selling) product = x;
  318. }
  319. if (in(prompt, " to sell ")) {
  320. // always sell everything
  321. to_server("\r");
  322. return;
  323. }
  324. if (in(prompt, " to buy ")) {
  325. bool buy_ok = true;
  326. if (trade_end_empty) {
  327. // Ok, we want to end with empty holds...
  328. int other_port;
  329. if (active_port == port[0])
  330. other_port = port[1];
  331. else
  332. other_port = port[0];
  333. // Is target port burnt?
  334. auto pos = director.galaxy.ports.find(other_port);
  335. bool burnt = false;
  336. if (pos != director.galaxy.ports.end()) {
  337. // We'll find the port. Really.
  338. if (!pos->second.unknown()) {
  339. // port isn't unknown, so check to see if it's burnt
  340. for (int x = 0; x < 3; ++x) {
  341. if (trades.foe[x]) {
  342. if (pos->second.percent[x] < stop_percent) burnt = true;
  343. }
  344. }
  345. }
  346. }
  347. if (burnt) {
  348. buy_ok = false;
  349. }
  350. }
  351. // Ok, what are they selling?
  352. // char selling = tolower(prompt[18]);
  353. BUGZ_LOG(fatal) << "Selling: " << selling;
  354. if (!buy_ok) {
  355. // no!
  356. to_server("0\r");
  357. } else
  358. for (int x = 0; x < 3; ++x) {
  359. // if (foe[x] == selling) {
  360. // We found the item ... is it something that we're trading?
  361. if (foe[x] == selling) {
  362. if (trades.foe[x]) {
  363. // Yes!
  364. to_server("\r");
  365. product = x;
  366. } else {
  367. // No!
  368. to_server("0\r");
  369. }
  370. }
  371. }
  372. // }
  373. }
  374. }
  375. if (startswith(prompt, "Your offer [") && endswith(prompt, " ? ")) {
  376. // Ok, things get weird here. We also need to look for final offer.
  377. if (last_offer != 0) percent -= 1.0;
  378. if (buying)
  379. last_offer = (int)(initial_offer * (100 + percent) / 100.0);
  380. else
  381. last_offer = (int)(initial_offer * (100 - percent) / 100.0);
  382. BUGZ_LOG(fatal) << "Offer: " << buying << " offer " << last_offer << " % "
  383. << percent;
  384. std::string text = std::to_string(last_offer);
  385. text.append("\r");
  386. to_server(text);
  387. }
  388. if (at_command_prompt(prompt)) {
  389. // we're done trading...
  390. // do we carry on, or stop?
  391. // 1.) CHECK TURNS // need turn tracking
  392. // 2.) PORTS BURNT?
  393. if (active_port == port[0]) {
  394. if (port[0] == 0) {
  395. deactivate();
  396. return;
  397. }
  398. active_port = port[1];
  399. } else
  400. active_port = port[0];
  401. // Is target port burnt?
  402. auto pos = director.galaxy.ports.find(active_port);
  403. bool burnt = false;
  404. if (pos != director.galaxy.ports.end()) {
  405. // We'll find the port. Really.
  406. if (!pos->second.unknown()) {
  407. // port isn't unknown, check to see if burnt
  408. for (int x = 0; x < 3; ++x) {
  409. if (trades.foe[x]) {
  410. BUGZ_LOG(fatal) << x << " % " << (int)pos->second.percent[x]
  411. << " " << stop_percent;
  412. if (pos->second.percent[x] < stop_percent) burnt = true;
  413. }
  414. }
  415. }
  416. }
  417. if (burnt) {
  418. to_client("Ports burnt.\n\r");
  419. deactivate();
  420. return;
  421. }
  422. std::string move = std::to_string(active_port);
  423. to_client("Moving...\n\r");
  424. move.append("\r");
  425. to_server(move);
  426. state = 2;
  427. }
  428. }
  429. }
  430. void ScriptTrader::client_input(const std::string &cinput) { deactivate(); };
  431. #endif
  432. ScriptTerror::ScriptTerror(Director &d) : Dispatch(d) {
  433. BUGZ_LOG(warning) << "ScriptTerror()";
  434. init();
  435. }
  436. ScriptTerror::~ScriptTerror() { BUGZ_LOG(warning) << "~ScriptTerror()"; }
  437. void ScriptTerror::init(void) {
  438. BUGZ_LOG(fatal) << "ScriptTerror::init()";
  439. move = std::make_shared<MoveDispatch>(director);
  440. md = static_cast<MoveDispatch *>(&(*move));
  441. // setup notify functions for results/completion.
  442. md->setNotify([this]() { this->move_notify(); });
  443. input = std::make_shared<InputDispatch>(director);
  444. id = static_cast<InputDispatch *>(&(*input));
  445. id->prompt = "Number of loops: ";
  446. id->max_length = 4;
  447. id->numeric = true;
  448. id->setNotify([this]() { this->input_notify(); });
  449. trader = std::make_shared<TraderDispatch>(director);
  450. td = static_cast<TraderDispatch *>(&(*trader));
  451. td->setNotify([this]() { this->trade_notify(); });
  452. BUGZ_LOG(fatal) << "ScriptTerror::init() completed.";
  453. }
  454. void ScriptTerror::activate(void) {
  455. BUGZ_LOG(warning) << "ScriptTerror::activate()";
  456. // Need: InputDispatch, MoveDispatch, ScriptTrader
  457. // Save the trade_end_empty, and set to Y
  458. if (director.galaxy.config["trade_end_empty"]) {
  459. old_trade_end_empty =
  460. director.galaxy.config["trade_end_empty"].as<std::string>();
  461. } else {
  462. old_trade_end_empty = "Y";
  463. }
  464. director.galaxy.config["trade_end_empty"] = "Y";
  465. // Step 1: Get number of loops of terror
  466. director.chain = input;
  467. input->activate();
  468. // Step 2: Look for closest trades, try ScriptTrade until none < some
  469. // level. Step 3: Move on, unless out of loops (or low on turns)
  470. // deactivate();
  471. }
  472. void ScriptTerror::deactivate(void) {
  473. BUGZ_LOG(warning) << "ScriptTerror::deactivate()";
  474. // restore the original value.
  475. director.galaxy.config["trade_end_empty"] = old_trade_end_empty;
  476. notify();
  477. }
  478. void ScriptTerror::input_notify(void) {
  479. if (id->input.empty()) {
  480. deactivate();
  481. return;
  482. }
  483. max_loops = sstoi(id->input, -1);
  484. if (max_loops == -1) {
  485. deactivate();
  486. return;
  487. }
  488. id->input.clear();
  489. BUGZ_LOG(warning) << "Loops of terror: " << max_loops;
  490. loops = max_loops;
  491. // find nearest
  492. ppt = director.galaxy.find_closest_trade(director.current_sector, 3);
  493. if (ppt.type == 0) {
  494. to_client("No trades found! You've burnt the galaxy!\n\r");
  495. deactivate();
  496. return;
  497. }
  498. // ok, step 2: move!
  499. // md->setNotify([this]() { this->proxy_deactivate(); });
  500. BUGZ_LOG(fatal) << "Moving to: " << ppt.s1;
  501. md->move_to = ppt.s1;
  502. director.chain = move;
  503. director.chain->activate();
  504. return;
  505. }
  506. void ScriptTerror::move_notify(void) {
  507. BUGZ_LOG(fatal) << "move_notify()";
  508. // Check for success, and start trading!
  509. if (md->success) {
  510. to_client("We're here, get trading!\n\r");
  511. td->port[0] = ppt.s1;
  512. td->port[1] = ppt.s2;
  513. td->trades = ppt.trades;
  514. td->type = ppt.type;
  515. director.chain = trader;
  516. director.chain->activate();
  517. return;
  518. } else {
  519. to_client("Move FAILED.\n\r");
  520. deactivate();
  521. }
  522. }
  523. void ScriptTerror::trade_notify(void) {
  524. // Done trading -- maybe! :P
  525. if (td->success) {
  526. // success!
  527. ppt = director.galaxy.find_closest_trade(director.current_sector, 3);
  528. if (ppt.type == 0) {
  529. to_client("No trades found! You've burnt the galaxy!\n\r");
  530. deactivate();
  531. return;
  532. }
  533. if ((director.current_sector == ppt.s1) ||
  534. (director.current_sector == ppt.s2)) {
  535. // We're still here...
  536. BUGZ_LOG(fatal) << "Trade it again, Sam.";
  537. to_client("Keep trading.\n\r");
  538. td->port[0] = ppt.s1;
  539. td->port[1] = ppt.s2;
  540. td->trades = ppt.trades;
  541. td->type = ppt.type;
  542. director.chain = trader;
  543. director.chain->activate();
  544. return;
  545. }
  546. // Ok, this isn't a local trade.
  547. if (loops == 0) {
  548. to_client("We're done terrorizing, for now...\n\r");
  549. deactivate();
  550. return;
  551. }
  552. --loops;
  553. // Move to our next target
  554. BUGZ_LOG(fatal) << "Moving to: " << ppt.s1;
  555. md->move_to = ppt.s1;
  556. director.chain = move;
  557. director.chain->activate();
  558. return;
  559. }
  560. to_client("Ok, trade is done.\n\r");
  561. deactivate();
  562. }