| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937 | #include "director.h"#include <boost/format.hpp>#include "boxes.h"#include "galaxy.h"#include "logging.h"#include "scripts.h"#include "utils.h"Director::Director() {  BUGZ_LOG(warning) << "Director::Director()";  // active = false;  game = 0;  // not in a game  galaxy.reset();  // do everything proxy_deactivate does ...  // proxy_deactivate();  active = false;  // reset everything back to good state  talk_direct = true;  show_client = true;  count = 0;  /*  Setup StringFunc for SL_parser:  Construct these once, rather then every single time we need them.  */  SF_cimline = [this](const std::string &s) { this->SL_cimline(s); };  SF_sectorline = [this](const std::string &s) { this->SL_sectorline(s); };  SF_portline = [this](const std::string &s) { this->SL_portline(s); };  SF_warpline = [this](const std::string &s) { this->SL_warpline(s); };  SF_infoline = [this](const std::string &s) { this->SL_infoline(s); };  build_menu();}Director::~Director() { BUGZ_LOG(warning) << "Director::~Director()"; }void Director::client_input(const std::string &input) {  // If we're already active, don't try to activate.  if (chain) {    chain->client_input(input);    return;  }  if (active) {    if (input == "Q" || input == "q") proxy_deactivate();    return;  } else if (input == "\x1b" || input == "~") {    std::string &prompt = current_prompt;    BUGZ_LOG(trace) << "CI: ACTIVATE prompt shows: [" << prompt << "]";    if (prompt == "Selection (? for menu): ") {      to_client(          "\n\rThere's not much we can do here.  Activate in-game at a "          "Command prompt.\n\r");      to_client(current_raw_prompt);      return;    }    // easter-eggs:    if (prompt == "Enter your choice: ") {      to_client(          "\n\r\x1b[1;36mI'd choose \x1b[1;37m`T`\x1b[1;36m, but "          "that's how I was coded.\n\r");      to_client(current_raw_prompt);      return;    }    // easter-egg    if (prompt == "[Pause]") {      to_client(" \x1b[1;36mPAWS\x1b[0m\n\r");      to_client(current_raw_prompt);      return;    }    if (prompt == "Planet command (?=help) [D] ") {      // future:  If activated at planet menu, activate the planet upgrade      // script!      to_client("\n\r\x1b[0mFUTURE:  Activate the planet upgrade script.\n\r");      to_client(current_raw_prompt);      return;    }    //    // The command prompt that we're looking for:    //    // "Command [TL=00:00:00]:[242] (?=Help)? : "    // the time, and the sector number vary...    /*        if (startswith(prompt, "Command [")) {          // if (prompt.substr(0, 9) == "Command [") {          //  int len = prompt.length();          if (endswith(prompt, "] (?=Help)? : ")) {            // if (prompt.substr(len - 14) == "] (?=Help)? : ") {            proxy_activate();            return;          }        }      }    */    if (at_command_prompt(prompt)) {      proxy_activate();      return;    }  }  // Ok...  if (talk_direct) to_server(input);  /*  if (emit_client_input)    emit_client_input(line);  */}void Director::server_line(const std::string &line,                           const std::string &raw_line) {  // check for if we entered game/left game  if (line.find("TradeWars Game Server   ") != std::string::npos) {    to_client("\rTradeWars Proxy v2++ READY (~ or ESC to activate)\n\r");    /*    There's a delay here when I save the game data.    I've moved it futher down.  Hide it at a prompt, so it isn't so    noticeable.     */    // reset "active game" -- we're at the TWGS main menu  }  if (line.find("Selection (? for menu): ") != std::string::npos) {    char ch = line[line.length() - 1];    if (ch >= 'A' && ch < 'Q') {      if ((game) && (game != ch)) {        galaxy.save();      }      game = ch;      BUGZ_LOG(warning) << "GAME " << game << " activated!";      // TODO:  Load game data      galaxy.game = game;      galaxy.username = username;      galaxy.load();      // YAML loaded, set sensible default config values (if missing).      if (!galaxy.config["display_lines"]) {        galaxy.config["display_lines"] = 20;      }      galaxy.meta["help"]["display_lines"] = "Number of lines to display";      if (!galaxy.config["burnt_percent"]) {        galaxy.config["burnt_percent"] = 40;      }      galaxy.meta["help"]["burnt_percent"] = "Ignore ports below this %";    }    // not needed (handled by above Game Server check).    if (ch == 'Q') {      if (game) {        // TODO:  Save galaxy data        galaxy.save();      }      game = 0;      galaxy.reset();    }  }  if (game) {    // in-game parsing here.    /*      ____                             _     _     / ___|  ___ _ ____   _____ _ __  | |   (_)_ __   ___     \___ \ / _ \ '__\ \ / / _ \ '__| | |   | | '_ \ / _ \      ___) |  __/ |   \ V /  __/ |    | |___| | | | |  __/     |____/ \___|_|    \_/ \___|_|    |_____|_|_| |_|\___|      ____                _     |  _ \ __ _ _ __ ___(_)_ __   __ _     | |_) / _` | '__/ __| | '_ \ / _` |     |  __/ (_| | |  \__ \ | | | | (_| |     |_|   \__,_|_|  |___/_|_| |_|\__, |                                  |___/   This is where all of the server lines are gleaned for all the   information that we can get out of them.    // When activating the computer    SP: [Command [TL=00:00:00]:[926] (?=Help)? : ]    Sector: 926    CI: [c]    SL: [Command [TL=00:00:00]:[926] (?=Help)? : C]    SL: [<Computer>]    SL: []    SL: [<Computer activated>]    SL: []    SP: [Computer command [TL=00:00:00]:[926] (?=Help)? ]    // deactivating the computer    SL: [Computer command [TL=00:00:00]:[926] (?=Help)? Q]    SL: []    SL: [<Computer deactivated>]    SL: []    */    if (startswith(line, " Items     Status  Trading % of max OnBoard"))      SL_parser = SF_portline;    if (startswith(line, "Sector  : ")) SL_parser = SF_sectorline;    if (line == ": ") SL_parser = SF_cimline;    if (line == "<Info>") SL_parser = SF_infoline;  }  if (SL_parser) {    SL_parser(line);  }  /*  if (emit_server_line) {    emit_server_line(line);  }   */  if (chain) {    chain->server_line(line, raw_line);  }}void Director::server_prompt(const std::string &prompt,                             const std::string &raw_prompt) {  current_prompt = prompt;  current_raw_prompt = raw_prompt;  if (game) {    if (prompt == "Selection (? for menu): ") {      galaxy.save();      game = 0;      galaxy.reset();    }    // in-game parsing here.    if (startswith(prompt, "Command [") && endswith(prompt, "] (?=Help)? : ")) {      std::string sector_text;      size_t before, after;      before = prompt.find("]:[") + 3;      after = prompt.find("] (?=Help)");      sector_text = prompt.substr(before, after - before);      current_sector = stoi(sector_text);      BUGZ_LOG(fatal) << "Sector: " << sector_text;    }  }  /*  if (emit_server_prompt)    emit_server_prompt(prompt);   */  if (chain) chain->server_prompt(prompt);}void Director::build_menu(void) {  main_menu = std::make_shared<MenuDispatch>(*this);  MenuDispatch *md = static_cast<MenuDispatch *>(&(*main_menu));  md->menu_box_color = "\x1b[1;33;44m";  md->menu_text_color = "\x1b[1;37;44m";  md->menu_title = "Proxy Menu";  md->menu_options_color = "\x1b[1;36;40m";  md->menu_prompt =      "\x1b[0;31;40m\xdb\xb2\xb1\xb0 \x1b[31;40mRED "      "\x1b[32;40mGREEN\x1b[30;42m\xdb\xb2\xb1\xb0 \x1b[0m : ";  md->lazy = true;  md->menu = {{"C", "Configure"},              {"D", "Display Report"},              {"E", "Export Data/Save"},              {"I", "Information"},              {"P", "Port CIM"},              {"W", "Warp CIM"},              {"T", "Trading Report (same as D)"},              {"S", "Scripts"},              {"X", "eXit"}};  md->setNotify([this]() { this->menu_choice(); });  cim = std::make_shared<CIMDispatch>(*this);  cim->setNotify([this]() { this->cim_done(); });}void Director::proxy_activate(void) {  active = true;   // yes, set keep-alive timer.  to_server(" ");  // start keep-alive timer.  // set other values we need  talk_direct = false;  show_client = false;  /*  Wait a minute .. this might be confusing.  Shouldn't I send them the current prompt?  Just in case we abort in the middle of something?!?  */  old_prompt = current_prompt;  old_raw_prompt = current_raw_prompt;  to_client("\x1b[0m\n\r");  /*  ╔══════════════════════════════╗  ║    TradeWars Proxy Active    ║  ╚══════════════════════════════╝     -=>  */  Boxes box(30, 1, true);  box.boxcolor = "\x1b[1;33;44m";  box.textcolor = "\x1b[1;33;44m";  to_client(box.top());  std::string output = "    TradeWars Proxy \x1b[5mActive\x1b[0;1;33;44m    ";  to_client(box.row(output));  to_client(box.bottom());  chain = main_menu;  main_menu->activate();  /*    // Using InputDispatch  -- and see have_input    std::shared_ptr<Dispatch> readline = std::make_shared<InputDispatch>(*this);    chain = readline;    InputDispatch *id = static_cast<InputDispatch *>(&(*readline));    id->prompt = "\x1b[0m\x1b[1;33;44m-=>\x1b[0m \x1b[1;37;44m";    id->max_length = 15;    id->setNotify([this]() { this->have_input(); });    readline->activate();  */}void Director::menu_choice(void) {  MenuDispatch *md = dynamic_cast<MenuDispatch *>(&(*chain));  if (md) {    if (md->input.empty()) {      to_client("Menu aborted.\n\r");      proxy_deactivate();      return;    } else {      switch (md->input[0]) {        case 'C':  // configure          config_edit();          return;          break;        case 'D':        case 'T':  // display trading report        {          auto pptv = galaxy.find_best_trades();          std::string output;          galaxy.sort_port_pair_type(pptv);          int max_display = 20;          if (galaxy.config["display_lines"])            max_display = galaxy.config["display_lines"].as<int>();          else            galaxy.config["display_lines"] = max_display;          if ((max_display <= 0) || (max_display > 255)) {            max_display = 255;            galaxy.config["display_lines"] = 255;          }          const int per_line = 5;          int count = 0;          int line = 0;          for (auto const &ppt : pptv) {            output = str(boost::format("%1$5d:%2$-5d = %3$d") % ppt.s1 %                         ppt.s2 % ppt.type);            to_client(output);            ++count;            if (count == per_line) {              count = 0;              to_client("\n\r");              ++line;            }            if (line == max_display) break;          }          if (count != 0) to_client("\n\r");          // We got < 5 lines, and max_display is > 5.  Offer suggestion:          if ((line < 5) && (max_display > 5)) {            // suggestion:            to_client(                "HINT: For more lines, try reducing the burnt_percent?\n\r");          }        } break;        case 'E':  // Export Data/Save          to_client("Saving...");          galaxy.save();          to_client("\rSaved....\n\r");          break;        case 'I':  // Information          information();          break;        case 'P':  // Port CIM          // Since we're adding/updating, we don't lose our          // type 0 ports.  Type 9 stays at 9.          chain = cim;          to_server("^RQ");          to_client("Port CIM Report\n\r");          chain->activate();          return;          break;        case 'W':  // Warp CIM          chain = cim;          to_server("^IQ");          to_client("Warp CIM Report\n\r");          chain->activate();          return;          break;        // case 'T':  // Trading Report        //  break;        case 'S':  // Scripts        {          init_scripts_menu();          chain = scripts_menu;          chain->activate();          return;        } break;        case 'X':  // Exit          proxy_deactivate();          return;      }      /*      std::string text = str(          boost::format("Back from Menu [%1%] was selected.\n\r") %      md->input);      to_client(text);      */      md->activate();    }  }}MenuDispatch *Director::init_scripts_menu(void) {  MenuDispatch *md;  if (scripts_menu) {    md = dynamic_cast<MenuDispatch *>(&(*scripts_menu));    return md;  } else {    scripts_menu = std::make_shared<MenuDispatch>(*this);    md = static_cast<MenuDispatch *>(&(*scripts_menu));    md->menu_box_color = "\x1b[0;32;40m";    md->menu_text_color = "\x1b[1;32;40m";    md->menu_title = "Scripts Menu";    md->menu_options_color = "\x1b[1;32;40m";    md->lazy = false;    md->menu_prompt = " SCRIPT : ";    md->menu = {{"!", "Terror"}, {"T", "Trade"}, {"U", "Upgrade Planet Pants"}};    md->setNotify([this]() { this->scripts_done(); });    return md;  }}void Director::scripts_done(void) {  // Was script selected?  If so, run it!  // otherwise, back to the menu we go...  MenuDispatch *md = dynamic_cast<MenuDispatch *>(&(*scripts_menu));  if (md) {    if (md->input.empty()) {      to_client("Scripts aborted.\n\r");      scripts_menu.reset();      proxy_deactivate();      return;    } else {      switch (md->input[0]) {        case 'T':  // Trade          script = std::make_shared<ScriptTrader>(*this);          ScriptTrader *ts = static_cast<ScriptTrader *>(&((*script)));          ts->setNotify([this]() { this->proxy_deactivate(); });          // Set parameters          auto found = galaxy.find_trades(current_sector, false);          if (found.empty()) {            to_client(                "No Trades found.  Port burnt (CONFIG: lower burnt_percent?) "                "or no ports around.\n\r");            proxy_deactivate();            return;          }          // sort first?          galaxy.sort_port_pair_type(found);          BUGZ_LOG(fatal) << "Found " << found.size() << " possible trade(s).";          BUGZ_LOG(fatal) << found[0].s1 << "," << found[0].s2 << " : "                          << found[0].type;          ts->port[0] = found[0].s1;          ts->port[1] = found[0].s2;          ts->type = found[0].type;          chain = script;          chain->activate();          return;          break;      }    }  }  proxy_activate();  // And to end scripts, we do .. what exactly?  // DEBUG:  Ok, why does everything work OK if I reset the scripts_menu  // here?? probably do want to destroy scripts here.  ;)  // scripts_menu.reset();  // proxy_deactivate();}/** * @brief Setup Config Input * * @return DispatchInput* */InputDispatch *Director::init_config_input(void) {  InputDispatch *id;  if (config_input) {    // Yes, it has been setup before.    id = dynamic_cast<InputDispatch *>(&(*config_input));    id->prompt = "Config => ";    id->max_length = 3;    config_item.clear();    return id;  } else {    // set it up    config_input = std::make_shared<InputDispatch>(*this);    id = static_cast<InputDispatch *>(&(*config_input));    id->prompt = "Config => ";    id->max_length = 3;    id->setNotify([this]() { this->config_have_input(); });    config_item.clear();    return id;  }}void Director::config_edit(void) {  // display current config  std::string menu_box_color = "\x1b[1;33;44m";  std::string menu_text_color = "\x1b[1;37;44m";  auto abox = Boxes::alert("   Configuration:   ", menu_box_color,                           menu_text_color, 20, 1, true);  for (auto line : abox) {    to_client(line);  }  // to_client("Configuration:\n\r");  int item = 1;  for (auto const &cfg : galaxy.config) {    std::string output = str(boost::format("%1$2d %2$20s: %3$s\n\r") % item %                             cfg.first % cfg.second);    to_client(output);    ++item;  }  to_client("Enter number to edit, blank to exit.\n\r");  // setup call to config_input:  InputDispatch *id = init_config_input();  chain = config_input;  id->activate();  // to return to the menu:  // MenuDispatch *md = dynamic_cast<MenuDispatch *>(&(*chain));  // md->activate();}void Director::config_have_input(void) {  InputDispatch *id = dynamic_cast<InputDispatch *>(&(*config_input));  if (config_item.empty()) {    // This is a config menu selection    if (id->input.empty()) {      // We're done here.  Return to menu.      chain = main_menu;      MenuDispatch *md = dynamic_cast<MenuDispatch *>(&(*chain));      md->activate();      // destroy the input?  yes.      config_input.reset();      return;    } else {      int item;      try {        item = stoi(id->input);      } catch (const std::invalid_argument &e) {        BUGZ_LOG(fatal) << e.what();        item = 0;      } catch (const std::out_of_range &e) {        BUGZ_LOG(fatal) << e.what();        item = 0;      }      if ((item < 1) || (item > (int)galaxy.config.size())) {        // selection out of range - redisplay config menu        config_edit();        return;      } else {        int pos = 1;        const YAML::Node &config = galaxy.config;        for (auto const &c : config) {          if (pos == item) {            // got it!            config_item = c.first.as<std::string>();            std::string output =                str(boost::format("%1% : %2%\n\r") % config_item %                    galaxy.meta["help"][config_item]);            to_client(output);            id->max_length = 30;            id->prompt = "Change to => ";            id->activate();            return;          };          ++pos;        }        to_client("What?  I didn't find that item?\n\r");        config_edit();        return;      }    }  } else {    // This is a config item edit    if (id->input.empty()) {      to_client("No change.\n\r");      config_edit();      return;    } else {      BUGZ_LOG(fatal) << "Config EDIT: " << config_item << " = " << id->input;      galaxy.config[config_item] = id->input;      config_edit();      return;    }  }}void Director::have_input(void) {  ++count;  InputDispatch *id = dynamic_cast<InputDispatch *>(&(*chain));  if (id) {    std::string output =        str(boost::format("Your Input (%2%): [%1%]\n\r") % id->input % count);    to_client("\x1b[0m");    to_client(output);  } else {    proxy_deactivate();    return;  }  if (count > 3) {    proxy_deactivate();  } else {    chain->activate();  }}void Director::cim_done(void) {  BUGZ_LOG(info) << "CIM done";  chain = main_menu;  main_menu->activate();}void Director::information(void) {  std::string output;  to_client("I currently know the following:\n\r");  output = str(      boost::format("Ports: %1%, Sectors: %2%, Config: %3%, Meta: %4%\n\r") %      galaxy.ports.size() % galaxy.warps.size() % galaxy.config.size() %      galaxy.meta.size());  to_client(output);}void Director::proxy_deactivate(void) {  active = false;  // reset everything back to good state  talk_direct = true;  show_client = true;  chain.reset();  to_client("\n\r");  to_client(current_raw_prompt);}/*Server Line Parsing Routines */void Director::SL_cimline(const std::string &line) {  if (line == ": ENDINTERROG") {    SL_parser = nullptr;    return;  }  if (line == ": ") {    // do I need to do anything special here for this?    // Maybe -- We would save special ports that don't show up    // (StarDock/Special) before. We don't know (at this point) if this is    // warps or ports.    return;  }  if (line.empty()) {    SL_parser = nullptr;    return;  }  // parse cimline  // size_t pos = line.find('%');  // std::string work = line;  // if (pos == line.npos) {  if (in(line, "%")) {    // portcim    struct port p = parse_portcim(line);    if (p.sector == 0)      BUGZ_LOG(fatal) << "portcim:  FAIL [" << line << "]";    else      BUGZ_LOG(trace) << "portcim: " << p;    galaxy.add_port(p);  } else {    // warpcim    // BUGZ_LOG(fatal) << "warpcim: [" << line << "]";    auto warps = split(line);    sector_warps sw;    for (auto const &w : warps) {      if (sw.sector == 0) {        sw.sector = stoi(w);      } else {        sw.add(stoi(w));      }    }    BUGZ_LOG(trace) << "warpcim: " << sw;    galaxy.add_warp(sw);  }}void Director::SL_thiefline(const std::string &line) {  size_t pos = line.find("Suddenly you're Busted!");  bool busted = pos != line.npos;  if (busted) {    BUGZ_LOG(fatal) << "set bust";    SL_parser = nullptr;  } else {    pos = line.find("(You realize the guards saw you last time!)");    if (pos != line.npos) SL_parser = nullptr;  }  // Are those the two ways to exit from this state?}void Director::SL_sectorline(const std::string &line) {  BUGZ_LOG(fatal) << "sectorline: [" << line << "]";  if (line.empty()) {    SL_parser = nullptr;  } else {    /*    sectorline: [Sector  : 926 in The Federation.]    sectorline: [Beacon  : FedSpace, FedLaw Enforced]    sectorline: [Ports   : Stargate Alpha I, Class 9 (Special) (StarDock)]    sectorline: [Traders : Civilian phil, w/ 30 ftrs,]    sectorline: [           in Star Stomper (Sverdlov Merchant Cruiser)]    sectorline: [Warps to Sector(s) :  70 - 441 - 575 - 600 - 629 - 711]    sectorline: [Warps to Sector(s) :  70 - (475) - 569]    What can we get from Traders : line?  Can we get if they are hostile    to us?  We need to respond to "is powering up weapons" ... We can    react faster then a person can!    "phil is powering up weapons systems!"    Also the auto-attack ones Ferrengi -- we need to auto-respond Retreat.    */    if (in(line, "Sector  :")) {      current_sector = stoi(line.substr(10));      BUGZ_LOG(warning) << "SECTOR: " << current_sector;    }    if (in(line, "Ports   :")) {      std::string port_class;      size_t pos = line.find(", Class ");      if (pos != std::string::npos) {        pos += 8;        int class_ = stoi(line.substr(pos));        BUGZ_LOG(fatal) << "PORT: " << class_;        galaxy.add_port(current_sector, class_);      }    }    if (in(line, "Warps to Sector(s) :")) {      std::string temp = line.substr(20);      replace(temp, " - ", " ");      // unexplored sectors ()      // Should I track these?      replace(temp, "(", "");      replace(temp, ")", "");      sector_warps sw;      auto warps = split(temp);      sw.sector = current_sector;      for (auto const &w : warps) {        sw.add(stoi(w));      }      BUGZ_LOG(fatal) << "WARPS: " << sw;      galaxy.add_warp(sw);    }  }}void Director::SL_portline(const std::string &line) {  if (line.empty()) {    SL_parser = nullptr;    return;  }  /*  SL: [ Items     Status  Trading % of max OnBoard]  SL: [ -----     ------  ------- -------- -------]  SL: [Fuel Ore   Buying    3000    100%       0]  SL: [Organics   Buying    3000    100%       0]  SL: [Equipment  Buying    3000    100%       0]  SL: []  SL: [Commerce report for: 03:51:56 PM Mon Oct 24, 2033     You can buy:]  SL: [A  Cargo holds     :    650 credits / next hold                0]  SL: [B  Fighters        :    233 credits per fighter               75]  SL: [C  Shield Points   :    116 credits per point                100]  SL: []   */  // BUGZ_LOG(info) << "portline : " << line;  if (in(line, "%")) {    // size_t pos = line.find('%');    // if (pos != line.npos) {    // Ok, this is a valid portline    std::string work = line;    replace(work, "Fuel Ore", "Fuel");    auto parts = split(work);    /*    BUGZ_LOG(fatal) << "portline split:";    for (auto const p : parts) {      BUGZ_LOG(fatal) << p;    }    */    // BUGZ_LOG(fatal) << "portline split : [" << parts << "]";  }}void Director::SL_warpline(const std::string &line) {  if (line.empty()) {    SL_parser = nullptr;    return;  }  // process warp line  BUGZ_LOG(fatal) << "warpline: [" << line << "]";}void Director::SL_infoline(const std::string &line) {  static int state;  if (line == "<Info>") {    state = 0;    galaxy.meta["info"] = YAML::Node();  }  if (line.empty()) {    ++state;    if (state == 2) {      SL_parser = nullptr;      // clear out the existing ship data      galaxy.meta["ship"] = YAML::Node();      // process the parsed information in meta["info"]      if (galaxy.meta["info"]["Total Holds"]) {        std::string work = galaxy.meta["info"]["Total Holds"].as<std::string>();        replace(work, "Fuel Ore", "Fuel");        auto parts = split(work, " - ");        int total_holds = stoi(parts[0]);        BUGZ_LOG(fatal) << "total holds: " << total_holds;        auto contents = split(parts[1]);        for (auto const &hold : contents) {          auto hold_amount = split(hold, "=");          BUGZ_LOG(fatal) << hold_amount[0] << " with " << hold_amount[1];          std::string key = hold_amount[0];          str_tolower(key);          // equipment = e          // organics = o          // fuel = f          // colonists = c          // empty = empty          if (key != "empty") {            key = key[0];          }          galaxy.meta["ship"]["holds"][key] = stoi(hold_amount[1]);        }        galaxy.meta["ship"]["holds"]["total"] = total_holds;      }      if (galaxy.meta["info"]["Turns to Warp"]) {        int warp_turns = galaxy.meta["info"]["Turns to Warp"].as<int>();        BUGZ_LOG(fatal) << "Turns to Warp: " << warp_turns;        galaxy.meta["ship"]["warp_turns"] = warp_turns;      }      if (galaxy.meta["info"]["LongRange Scan"]) {        std::string scanner_text =            galaxy.meta["info"]["LongRange Scan"].as<std::string>();        char scanner = scanner_text[0];        BUGZ_LOG(fatal) << "Scanner: " << scanner;        galaxy.meta["ship"]["scanner"] = scanner;      }      // turns isn't ship specific      if (galaxy.meta["info"]["Turns left"]) {        // OR this could be "Unlimited" !!!        std::string text = galaxy.meta["info"]["Turns left"].as<std::string>();        if (text == "Unlimited") {          galaxy.meta["turns"] = -1;        } else {          int turns =              stoi(text);  // galaxy.meta["info"]["Turns left"].as<int>();          BUGZ_LOG(fatal) << "Turns left: " << turns;          galaxy.meta["turns"] = turns;        }      }      if (galaxy.meta["info"]["Current Sector"]) {        int sector = galaxy.meta["info"]["Current Sector"].as<int>();        BUGZ_LOG(fatal) << "Sector: " << sector;        // it should already be sector ...        current_sector = sector;      }      if (galaxy.meta["info"]["Credits"]) {        std::string credit_text =            galaxy.meta["info"]["Credits"].as<std::string>();        replace(credit_text, ",", "");        int credits = stoi(credit_text);        galaxy.meta["credits"] = credits;        BUGZ_LOG(fatal) << "Credits: " << credits;      }    }    return;  }  // info to parse:  size_t pos = line.find(" : ");  if (pos != line.npos) {    std::string key = line.substr(0, pos);    std::string value = line.substr(pos + 3);    trim(key);    trim(value);    galaxy.meta["info"][key] = value;    BUGZ_LOG(fatal) << "Info: " << key << " : " << value;  }}
 |