scripts.cpp 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251
  1. #include "scripts.h"
  2. #include "logging.h"
  3. ScriptTrader::ScriptTrader(Director &d) : Dispatch(d) {
  4. BUGZ_LOG(fatal) << "ScriptTrader()";
  5. };
  6. ScriptTrader::~ScriptTrader() { BUGZ_LOG(fatal) << "~ScriptTrader()"; }
  7. void ScriptTrader::activate(void) {
  8. // ok, lookup port1 port2
  9. BUGZ_LOG(fatal) << "ScriptTrader::activate " << port[0] << " & " << port[1];
  10. auto port_info = director.galaxy.ports.find(port[0]);
  11. int port0_type = port_info->second.type;
  12. port_buysell[0] = get_buysell(port0_type);
  13. port_info = director.galaxy.ports.find(port[1]);
  14. int port1_type = port_info->second.type;
  15. port_buysell[1] = get_buysell(port1_type);
  16. BUGZ_LOG(fatal) << port0_type << " and " << port1_type;
  17. auto ttr = trade_type_info(port0_type, port1_type);
  18. trades = ttr.trades;
  19. // Ok, what do we do first here?
  20. // I - Info
  21. state = 1;
  22. percent = 5.0; // check meta for past trades information
  23. to_server("I");
  24. }
  25. void ScriptTrader::deactivate(void) { notify(); }
  26. void ScriptTrader::server_line(const std::string &line,
  27. const std::string &raw_line) {
  28. // FUTURE: powering up weapons check
  29. if (line == "Docking...") {
  30. last_offer = 0;
  31. final_offer = 0;
  32. initial_offer = 0;
  33. }
  34. if (startswith(line, "We'll buy them for ")) {
  35. // I need the initial offer!
  36. std::string offer = line.substr(19);
  37. replace(offer, ",", "");
  38. initial_offer = stoi(offer);
  39. BUGZ_LOG(fatal) << "Buying, initial: " << initial_offer;
  40. buying = true;
  41. last_offer = 0;
  42. final_offer = 0;
  43. }
  44. if (startswith(line, "We'll sell them for ")) {
  45. // I need the initial offer!
  46. std::string offer = line.substr(20);
  47. replace(offer, ",", "");
  48. initial_offer = stoi(offer);
  49. BUGZ_LOG(fatal) << "Selling, initial: " << initial_offer;
  50. buying = false;
  51. last_offer = 0;
  52. final_offer = 0;
  53. }
  54. // SL: [Our final offer is 1,263 credits.]
  55. if (startswith(line, "Our final offer is ")) {
  56. // Well snap!
  57. std::string offer = line.substr(20);
  58. replace(offer, ",", "");
  59. final_offer = stoi(offer);
  60. BUGZ_LOG(fatal) << "Final offer: " << final_offer;
  61. }
  62. // SL: [You have 16,767 credits and 0 empty cargo holds.]
  63. // trade accepted. if not 0 empty cargo holds -- we failed!
  64. // SL: [<P-Probe estimates your offer was 91.83% of best price>]
  65. // SL: [You have 4,046 credits and 0 empty cargo holds.]
  66. // this shows up at the initial docking of the port.
  67. if (startswith(line, "You have ")) {
  68. if (initial_offer != 0) {
  69. // Ok, the offer was possibly accepted.
  70. int success;
  71. if (buying)
  72. success = 0;
  73. else
  74. success = director.galaxy.meta["ship"]["holds"]["total"].as<int>();
  75. std::string text = std::to_string(success);
  76. text.append(" empty cargo holds.");
  77. if (endswith(line, text)) {
  78. BUGZ_LOG(fatal) << "Trade SUCCESS!";
  79. // record this action somewhere in meta.
  80. }
  81. }
  82. }
  83. }
  84. void ScriptTrader::server_prompt(const std::string &prompt) {
  85. // FUTURE: check for "Surrender/Attack"
  86. if (at_command_prompt(prompt)) {
  87. if (state == 1) {
  88. // Ok, decision time!
  89. if (director.galaxy.meta["ship"]["holds"]["c"]) {
  90. // holds contain colonists
  91. to_client("ScriptTrader FAIL: holds contain colonists.");
  92. deactivate();
  93. return;
  94. }
  95. // Which port to trade with first? examine trades
  96. BUGZ_LOG(fatal) << "trades: " << trades;
  97. BUGZ_LOG(fatal) << "port0:" << text_from_buysell(port_buysell[0]);
  98. BUGZ_LOG(fatal) << "port1:" << text_from_buysell(port_buysell[1]);
  99. // Do we have what they are buying?
  100. bool have_buy = false;
  101. int active_buy = 0;
  102. int active_sell = 0;
  103. for (int x = 0; x < 3; ++x) {
  104. if (trades.foe[x]) {
  105. // this is what they will be trading...
  106. if (director.galaxy.meta["ship"]["holds"][foe[x]]) {
  107. // key exists ...
  108. have_buy = true;
  109. // which port is buying?
  110. if (port_buysell[0].foe[x]) {
  111. active_buy = 0;
  112. } else {
  113. active_buy = 1;
  114. }
  115. }
  116. if (!port_buysell[0].foe[x]) {
  117. active_sell = 0;
  118. } else {
  119. active_sell = 1;
  120. }
  121. }
  122. }
  123. if (have_buy) {
  124. BUGZ_LOG(fatal) << "have_buy: port " << active_buy;
  125. active_port = port[active_buy];
  126. } else {
  127. // which port is selling?
  128. // if they aren't buying what I have in my holds, should I check to see
  129. // if my holds are full? This could be the "not buying" what I'm
  130. // setting bug!
  131. BUGZ_LOG(fatal) << "!have_buy: port " << active_sell;
  132. active_port = port[active_sell];
  133. }
  134. state = 2;
  135. if (director.current_sector == active_port) {
  136. // begin state 3
  137. state = 3;
  138. to_client("Trade...\n\r");
  139. to_server("PT");
  140. return;
  141. } else {
  142. // initiate move
  143. std::string move = std::to_string(active_port);
  144. to_client("Moving...\n\r");
  145. move.append("\r");
  146. to_server(move);
  147. return;
  148. }
  149. // for now...
  150. deactivate();
  151. }
  152. if (state == 2) {
  153. if (director.current_sector == active_port) {
  154. // We're here
  155. state = 3;
  156. to_server("PT");
  157. return;
  158. } else {
  159. // we failed to move to where we wanted to go?!
  160. BUGZ_LOG(fatal) << "Expecting: " << active_port << " but got "
  161. << director.current_sector;
  162. deactivate();
  163. return;
  164. }
  165. }
  166. }
  167. if (state == 3) {
  168. if (in(prompt, " to sell ")) {
  169. // always sell everything
  170. to_server("\r");
  171. return;
  172. }
  173. if (in(prompt, " to buy ") && startswith(prompt, "How many holds of ")) {
  174. // Ok, what are they selling?
  175. char selling = tolower(prompt[18]);
  176. BUGZ_LOG(fatal) << "Selling: " << selling;
  177. for (int x = 0; x < 3; ++x) {
  178. if (foe[x] == selling) {
  179. // We found the item ... is it something that we're trading?
  180. if (trades.foe[x]) {
  181. // Yes!
  182. to_server("\r");
  183. } else {
  184. // No!
  185. to_server("0\r");
  186. }
  187. }
  188. }
  189. }
  190. if (startswith(prompt, "Your offer [") && endswith(prompt, " ? ")) {
  191. // Ok, things get weird here. We also need to look for final offer.
  192. if (last_offer != 0) percent -= 1.0;
  193. if (buying)
  194. last_offer = (int)(initial_offer * (100 - percent) / 100.0);
  195. else
  196. last_offer = (int)(initial_offer * (100 + percent) / 100.0);
  197. BUGZ_LOG(fatal) << "Offer: " << last_offer << " % " << percent;
  198. std::string text = std::to_string(last_offer);
  199. text.append("\r");
  200. to_server(text);
  201. }
  202. if (at_command_prompt(prompt)) {
  203. // we're done trading...
  204. // do we carry on, or stop?
  205. // 1.) CHECK TURNS // need turn tracking
  206. // 2.) PORTS BURNT?
  207. if (active_port == port[0])
  208. active_port = port[1];
  209. else
  210. active_port = port[0];
  211. std::string move = std::to_string(active_port);
  212. to_client("Moving...\n\r");
  213. move.append("\r");
  214. to_server(move);
  215. state = 2;
  216. }
  217. }
  218. }
  219. void ScriptTrader::client_input(const std::string &cinput) { deactivate(); };