123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488 |
- #include "scripts.h"
- #include "logging.h"
- ScriptTrader::ScriptTrader(Director &d) : Dispatch(d) {
- BUGZ_LOG(fatal) << "ScriptTrader()";
- state = 0;
- };
- ScriptTrader::~ScriptTrader() { BUGZ_LOG(fatal) << "~ScriptTrader()"; }
- void ScriptTrader::activate(void) {
- // ok, lookup port1 port2
- BUGZ_LOG(fatal) << "ScriptTrader::activate " << port[0] << " & " << port[1];
- auto port_info = director.galaxy.ports.find(port[0]);
- int port0_type = port_info->second.type;
- port_buysell[0] = get_buysell(port0_type);
- // Special case - we just want to buy resources
- if (port[1] != 0) {
- port_info = director.galaxy.ports.find(port[1]);
- int port1_type = port_info->second.type;
- port_buysell[1] = get_buysell(port1_type);
- BUGZ_LOG(fatal) << port0_type << " and " << port1_type;
- } else {
- BUGZ_LOG(fatal) << "Just buy from " << port[0];
- }
- /*
- auto ttr = director.galaxy.trade_type_info(port0_type, port1_type);
- trades = ttr.trades;
- */
- /*
- if (trades.foe[0] && trades.foe[1] && trades.foe[2]) {
- // it has all three -- use the last 2.
- trades.foe[0] = false;
- }
- */
- // Ok, what do we do first here?
- // I - Info
- state = 1;
- percent = 5.0;
- to_server("I");
- director.galaxy.meta["help"]["stop_percent"] =
- "ScriptTrader stop trading if below this percent.";
- if (director.galaxy.config["stop_percent"]) {
- stop_percent = director.galaxy.config["stop_percent"].as<int>();
- } else {
- stop_percent = 20;
- director.galaxy.config["stop_percent"] = stop_percent;
- }
- director.galaxy.meta["help"]["trade_end_empty"] =
- "ScriptTrader end trades with empty holds? Y/N";
- if (director.galaxy.config["trade_end_empty"]) {
- std::string tee =
- director.galaxy.config["trade_end_empty"].as<std::string>();
- if ((toupper(tee[0]) == 'Y') || (toupper(tee[0]) == 'T')) {
- trade_end_empty = true;
- } else {
- trade_end_empty = false;
- // director.galaxy.config["trade_end_empty"] = "N";
- }
- } else {
- trade_end_empty = false;
- director.galaxy.config["trade_end_empty"] = "N";
- }
- }
- void ScriptTrader::deactivate(void) { notify(); }
- void ScriptTrader::server_line(const std::string &line,
- const std::string &raw_line) {
- // FUTURE: powering up weapons check
- // Show what's going on...
- if (state > 1) {
- std::string temp = raw_line;
- temp.append("\n\r");
- to_client(temp);
- }
- if (line == "Docking...") {
- last_offer = 0;
- final_offer = 0;
- initial_offer = 0;
- }
- static std::set<std::string> success_lines = {
- "If only more honest traders would port here, we'll take them though.",
- "You will put me out of business, I'll take your offer.",
- "FINE, we'll take them, just leave!",
- "Agreed, and a pleasure doing business with you!",
- "You are a rogue! We'll take them anyway.",
- "You insult my intelligence, but we'll buy them anyway.",
- "Very well, we'll take that offer.",
- "You drive a hard bargain, but we'll take them.",
- "Done, we'll take the lot.",
- "I hate haggling, they're all yours.",
- "You are robbing me, but we'll buy them anyway.",
- "SOLD! Come back anytime!",
- "Cheapskate. Here, take them and leave me alone.",
- "Very well, we'll buy them.",
- "You are a shrewd trader, they're all yours.",
- "I could have twice that much in the Androcan Empire, but they're yours.",
- "Oh well, maybe I can sell these to some other fool, we'll take them.",
- "I PAID more than that! But we'll sell them to you anyway.",
- "(Sigh) Very well, pay up and take them away.",
- "Agreed! We'll purchase them!"};
- if (success_lines.find(line) != success_lines.end()) {
- BUGZ_LOG(fatal) << "Success " << buying << " " << initial_offer << " : "
- << last_offer;
- // calculate % ?
- BUGZ_LOG(fatal) << "% " << (float)initial_offer / (float)last_offer * 100.0;
- BUGZ_LOG(fatal) << "meta trade setting: " << percent << " for "
- << active_port << " " << product;
- director.galaxy.meta["trade"][active_port][product] = percent;
- }
- // <P-Probe estimates your offer was
- if (startswith(line, "Agreed, ")) {
- last_offer = 0;
- final_offer = 0;
- if (director.galaxy.meta["trade"][active_port][product]) {
- percent = director.galaxy.meta["trade"][active_port][product].as<float>();
- percent += 1.0;
- BUGZ_LOG(fatal) << "Percent for " << active_port << " now " << percent;
- } else {
- BUGZ_LOG(fatal) << "using default for " << active_port;
- percent = 5.0; // check meta for past trades information
- }
- }
- if (startswith(line, "We'll buy them for ")) {
- // I need the initial offer!
- std::string offer = line.substr(19);
- replace(offer, ",", "");
- initial_offer = stoi(offer);
- BUGZ_LOG(fatal) << "Buying, initial: " << initial_offer;
- buying = true; // Port is buying, we are selling.
- }
- if (startswith(line, "We'll sell them for ")) {
- // I need the initial offer!
- std::string offer = line.substr(20);
- replace(offer, ",", "");
- initial_offer = stoi(offer);
- BUGZ_LOG(fatal) << "Selling, initial: " << initial_offer;
- buying = false; // Port is selling, we are buying.
- }
- // SL: [Our final offer is 1,263 credits.]
- if (startswith(line, "Our final offer is ")) {
- // Well snap!
- std::string offer = line.substr(19);
- replace(offer, ",", "");
- final_offer = stoi(offer);
- BUGZ_LOG(fatal) << "Final offer: " << final_offer;
- }
- // SL: [You have 16,767 credits and 0 empty cargo holds.]
- // trade accepted. if not 0 empty cargo holds -- we failed!
- // SL: [<P-Probe estimates your offer was 91.83% of best price>]
- // SL: [You have 4,046 credits and 0 empty cargo holds.]
- // this shows up at the initial docking of the port.
- if (startswith(line, "You have ")) {
- if (initial_offer != 0) {
- // Ok, the offer was possibly accepted.
- int success;
- if (buying)
- success = 0;
- else
- success = director.galaxy.meta["ship"]["holds"]["total"].as<int>();
- std::string text = std::to_string(success);
- text.append(" empty cargo holds.");
- if (endswith(line, text)) {
- BUGZ_LOG(fatal) << "Trade SUCCESS!";
- // record this action somewhere in meta.
- }
- }
- }
- }
- void ScriptTrader::server_prompt(const std::string &prompt) {
- // FUTURE: check for "Surrender/Attack"
- if (at_command_prompt(prompt)) {
- if (state == 1) {
- // Ok, decision time!
- if (director.galaxy.meta["ship"]["holds"]["c"]) {
- // holds contain colonists
- to_client("ScriptTrader FAIL: holds contain colonists.");
- deactivate();
- return;
- }
- // Which port to trade with first? examine trades
- BUGZ_LOG(fatal) << "trades: " << trades;
- BUGZ_LOG(fatal) << "port0:" << text_from_buysell(port_buysell[0]);
- if (port[1] != 0)
- BUGZ_LOG(fatal) << "port1:" << text_from_buysell(port_buysell[1]);
- // Ok, I might still need this (so I know what port to start with)
- // which is selling?
- // must set active port!
- bool all_holds_empty = false;
- active_port = 0;
- // check the ship and holds here. (MAYBE)
- int holds = director.galaxy.meta["ship"]["holds"]["total"].as<int>();
- if (director.galaxy.meta["ship"]["holds"]["empty"]) {
- if (holds == director.galaxy.meta["ship"]["holds"]["empty"].as<int>())
- all_holds_empty = true;
- }
- if (port[1] == 0) {
- active_port = port[0];
- } else {
- if (!all_holds_empty) {
- for (int x = 0; x < 3; ++x) {
- if (director.galaxy.meta["ship"]["holds"][foe[x]]) {
- if (port_buysell[0].foe[x]) {
- active_port = port[0];
- break;
- }
- if (port_buysell[1].foe[x]) {
- active_port = port[1];
- }
- }
- }
- if (active_port == 0) {
- to_client(
- "I don't see any ports that are buying what we have in our "
- "holds.\n\r");
- deactivate();
- return;
- };
- } else {
- // all holds empty, find selling port
- for (int x = 0; x < 3; ++x) {
- if (trades.foe[x]) {
- if (port_buysell[0].foe[x]) {
- active_port = port[0];
- break;
- }
- if (port_buysell[1].foe[x]) {
- active_port = port[1];
- break;
- }
- }
- }
- }
- }
- #ifdef NO_JUST_TRADES
- // Do we have what they are buying?
- bool have_buy = false;
- int active_buy = 0;
- int active_sell = 0;
- for (int x = 0; x < 3; ++x) {
- if (trades.foe[x]) {
- // this is what they will be trading...
- if (director.galaxy.meta["ship"]["holds"][foe[x]]) {
- // key exists ...
- have_buy = true;
- // which port is buying?
- if (port_buysell[0].foe[x]) {
- active_buy = 0;
- have_buy = true;
- } else if (port_buysell[1].foe[x]) {
- active_buy = 1;
- have_buy = true;
- }
- }
- if (!port_buysell[0].foe[x]) {
- active_sell = 0;
- } else {
- active_sell = 1;
- }
- }
- }
- if (have_buy) {
- BUGZ_LOG(fatal) << "have_buy: port " << active_buy;
- active_port = port[active_buy];
- } else {
- // which port is selling?
- // if they aren't buying what I have in my holds, should I check to see
- // if my holds are full? This could be the "not buying" what I'm
- // setting bug!
- BUGZ_LOG(fatal) << "!have_buy: port " << active_sell;
- active_port = port[active_sell];
- // yes, this is the bug alright.
- // Ok, this shows up all the time. I need to look at my holds!
- if (!all_holds_empty) {
- to_client(
- "I don't see any ports that are buying what we have in our "
- "holds!\n\r");
- deactivate();
- return;
- }
- }
- #endif
- state = 2;
- if (director.current_sector == active_port) {
- // begin state 3
- state = 3;
- to_client("Trading...\n\r");
- to_server("PT");
- return;
- } else {
- // initiate move
- std::string move = std::to_string(active_port);
- to_client("Moving...\n\r");
- move.append("\r");
- to_server(move);
- return;
- }
- }
- if (state == 2) {
- if (director.current_sector == active_port) {
- // We're here
- state = 3;
- to_client("Trading...\n\r");
- to_server("PT");
- return;
- } else {
- // we failed to move to where we wanted to go?!
- BUGZ_LOG(fatal) << "Expecting: " << active_port << " but got "
- << director.current_sector;
- deactivate();
- return;
- }
- }
- }
- if (state == 3) {
- if (startswith(prompt, "How many holds of ")) {
- char selling = tolower(prompt[18]);
- for (int x = 0; x < 3; ++x) {
- if (foe[x] == selling) product = x;
- }
- if (in(prompt, " to sell ")) {
- // always sell everything
- to_server("\r");
- return;
- }
- if (in(prompt, " to buy ")) {
- bool buy_ok = true;
- if (trade_end_empty) {
- // Ok, we want to end with empty holds...
- int other_port;
- if (active_port == port[0])
- other_port = port[1];
- else
- other_port = port[0];
- // Is target port burnt?
- auto pos = director.galaxy.ports.find(other_port);
- bool burnt = false;
- if (pos != director.galaxy.ports.end()) {
- // We'll find the port. Really.
- if (!pos->second.unknown()) {
- // port isn't unknown, so check to see if it's burnt
- for (int x = 0; x < 3; ++x) {
- if (trades.foe[x]) {
- if (pos->second.percent[x] < stop_percent) burnt = true;
- }
- }
- }
- }
- if (burnt) {
- buy_ok = false;
- }
- }
- // Ok, what are they selling?
- // char selling = tolower(prompt[18]);
- BUGZ_LOG(fatal) << "Selling: " << selling;
- if (!buy_ok) {
- // no!
- to_server("0\r");
- } else
- for (int x = 0; x < 3; ++x) {
- // if (foe[x] == selling) {
- // We found the item ... is it something that we're trading?
- if (foe[x] == selling) {
- if (trades.foe[x]) {
- // Yes!
- to_server("\r");
- product = x;
- } else {
- // No!
- to_server("0\r");
- }
- }
- }
- // }
- }
- }
- if (startswith(prompt, "Your offer [") && endswith(prompt, " ? ")) {
- // Ok, things get weird here. We also need to look for final offer.
- if (last_offer != 0) percent -= 1.0;
- if (buying)
- last_offer = (int)(initial_offer * (100 + percent) / 100.0);
- else
- last_offer = (int)(initial_offer * (100 - percent) / 100.0);
- BUGZ_LOG(fatal) << "Offer: " << buying << " offer " << last_offer << " % "
- << percent;
- std::string text = std::to_string(last_offer);
- text.append("\r");
- to_server(text);
- }
- if (at_command_prompt(prompt)) {
- // we're done trading...
- // do we carry on, or stop?
- // 1.) CHECK TURNS // need turn tracking
- // 2.) PORTS BURNT?
- if (active_port == port[0]) {
- if (port[0] == 0) {
- deactivate();
- return;
- }
- active_port = port[1];
- }
- else
- active_port = port[0];
- // Is target port burnt?
- auto pos = director.galaxy.ports.find(active_port);
- bool burnt = false;
- if (pos != director.galaxy.ports.end()) {
- // We'll find the port. Really.
- if (!pos->second.unknown()) {
- // port isn't unknown, check to see if burnt
- for (int x = 0; x < 3; ++x) {
- if (trades.foe[x]) {
- BUGZ_LOG(fatal) << x << " % " << (int)pos->second.percent[x]
- << " " << stop_percent;
- if (pos->second.percent[x] < stop_percent) burnt = true;
- }
- }
- }
- }
- if (burnt) {
- to_client("Ports burnt.\n\r");
- deactivate();
- return;
- }
- std::string move = std::to_string(active_port);
- to_client("Moving...\n\r");
- move.append("\r");
- to_server(move);
- state = 2;
- }
- }
- }
- void ScriptTrader::client_input(const std::string &cinput) { deactivate(); };
|