Browse Source

Working trades. (again). But better.

We can now burn a port, then continue to trade using other
trades.  It's working like I want now.
Steve Thielemann 3 years ago
parent
commit
066c6478d5
6 changed files with 245 additions and 145 deletions
  1. 1 0
      director.cpp
  2. 170 135
      galaxy.cpp
  3. 15 3
      galaxy.h
  4. 54 7
      scripts.cpp
  5. 2 0
      scripts.h
  6. 3 0
      test-galaxy.cpp

+ 1 - 0
director.cpp

@@ -521,6 +521,7 @@ void Director::scripts_done(void) {
           ts->port[0] = found[0].s1;
           ts->port[1] = found[0].s2;
           ts->type = found[0].type;
+          ts->trades = found[0].trades;
           chain = script;
           chain->activate();
           return;

+ 170 - 135
galaxy.cpp

@@ -26,6 +26,17 @@ std::ostream &operator<<(std::ostream &os, const port &p) {
   return os;
 }
 
+/**
+ * Is port unknown?
+ *
+ * As in we haven't visited it, we don't know what it has?
+ * We were checking percent to 0, but we've seen valid ports with 0 percent.
+ * We now check amount.  If this becomes an issue, we'll change to an unknown
+ * flag.
+ *
+ * @return true
+ * @return false
+ */
 bool port::unknown(void) {
   for (int x = 0; x < 3; ++x) {
     if (amount[x] != 0) return false;
@@ -71,62 +82,6 @@ bool operator==(const density lhs, const density rhs) {
           (lhs.warps == rhs.warps) && (lhs.known == rhs.known));
 }
 
-trade_type_result trade_type_info(port_type port1, port_type port2) {
-  // This only gives us one trade_type per pair.  There actually
-  // should be multiple values returned here!
-  // Like in case of BBB/SSS:  return 3, 4 and 5.
-
-  // NONE = 0
-  // GOOD = 1 = OE PAIR
-  // OK   = 2 = ?? Pair
-  // FAIR = 3 = B/S E
-  //        4 = B/S O
-  //        5 = B/S F
-
-  buysell p1 = get_buysell(port1);
-  buysell p2 = get_buysell(port2);
-
-  buysell inv2 = invert_buysell(p2);
-  int matches = 0;  // or pos.size();
-  std::vector<int> pos;
-
-  // find which FOE are flipped.  Save index pos.
-  for (int x = 0; x < 3; ++x) {
-    inv2.foe[x] = (p1.foe[x] == inv2.foe[x]);
-    if (inv2.foe[x]) {
-      matches++;
-      pos.push_back(x);
-    }
-  }
-
-  if (matches > 1) {
-    // O != E for both ports, and O != O
-    if ((p1.foe[ORG] != p1.foe[EQU]) && (p2.foe[ORG] != p2.foe[EQU]) &&
-        (p1.foe[ORG] != p2.foe[ORG])) {
-      return trade_type_result{1, inv2};
-    }
-
-    // at least 2 matches.  but are they trade pairs?
-    // I can tell by comparing the last two positions in the same port.
-    if (p1.foe[pos[matches - 1]] == p1.foe[pos[matches - 2]]) {
-      // they are NOT.
-      return trade_type_result{3, inv2};
-    }
-    return trade_type_result{2, inv2};
-  }
-
-  if (matches == 1) {
-    if (inv2.foe[FUEL]) return trade_type_result{4, inv2};
-    return trade_type_result{3, inv2};
-  }
-  return trade_type_result{0, inv2};
-}
-
-int trade_type(port_type port1, port_type port2) {
-  trade_type_result r = trade_type_info(port1, port2);
-  return r.type;
-}
-
 sector_warps::sector_warps() { sector = 0; }
 
 void sector_warps::add(sector_type new_sector) {
@@ -533,68 +488,6 @@ void Galaxy::save(void) {
     }
   }
   */
-
-#ifdef YAML_NODE_SLOW_OUTPUT
-  YAML::Node data;
-  // add some information to meta before saving.
-  meta["save_to"] = filename;
-  std::chrono::_V2::system_clock::time_point now =
-      std::chrono::system_clock::now();
-  meta["save_time"] = std::chrono::system_clock::to_time_t(now);  // time_t
-
-  data["meta"] = meta;
-  // data["meta"].SetStyle(YAML::EmitterStyle::Flow);
-  BUGZ_LOG(fatal) << "YAML config: " << config.size();
-  data["config"] = config;
-  data["config"].SetStyle(YAML::EmitterStyle::Flow);
-
-  /*
-  for (auto const &config_iter : config) {
-    data["config"][config_iter.first] = config_iter.second;
-  }
-  */
-  BUGZ_LOG(fatal) << "YAML warps: " << warps.size();
-  for (auto const &warp : warps) {
-    for (auto const &sector : warp.second.warps) {
-      data["warps"][warp.first].push_back(sector);
-    }
-    data["warps"][warp.first].SetStyle(YAML::EmitterStyle::Flow);
-    /*
-    for (int x = 0; x < MAX_WARPS; ++x) {
-      if (warp.second.warps[x] == 0) break;
-      data["warps"][warp.first].push_back(warp.second.warps[x]);
-    }
-    */
-  }
-  BUGZ_LOG(fatal) << "YAML ports: " << ports.size();
-  /*
-  When saving to yaml, my sector_type is like char.  So, it wants
-  to save the values as a character.  Cast to int.
-   */
-  for (auto const &port : ports) {
-    data["ports"][port.second.sector]["class"] = (int)port.second.type;
-    if (port.second.type == 0) {
-      data["ports"][port.second.sector].SetStyle(YAML::EmitterStyle::Flow);
-    } else {
-      // nothing to save for type = 0
-      for (int x = 0; x < 3; x++) {
-        data["ports"][port.second.sector]["amount"].push_back(
-            (int)port.second.amount[x]);
-        data["ports"][port.second.sector]["pct"].push_back(
-            (int)port.second.percent[x]);
-      }
-      data["ports"][port.second.sector]["amount"].SetStyle(
-          YAML::EmitterStyle::Flow);
-      data["ports"][port.second.sector]["pct"].SetStyle(
-          YAML::EmitterStyle::Flow);
-      data["ports"][port.second.sector].SetStyle(YAML::EmitterStyle::Flow);
-    }
-  }
-
-  std::ofstream fout(filename);
-  fout << data << std::endl;
-  BUGZ_LOG(fatal) << "YAML: " << filename;
-#endif
 }
 
 std::vector<port_pair_type> Galaxy::find_trades(sector_type sector,
@@ -634,23 +527,11 @@ std::vector<port_pair_type> Galaxy::find_trades(sector_type sector,
     BUGZ_LOG(trace) << "find_trades: Port " << sector << ","
                     << (int)port->second.type << " " << s
                     << (int)possible_port->second.type;
-    trade_type_result ttr =
-        trade_type_info(port->second.type, possible_port->second.type);
-    if ((ttr.type == 0) || (ttr.type == 4)) continue;
-
-    bool burnt = false;
-    for (int x = 0; x < 3; ++x) {
-      if (ttr.trades.foe[x]) {
-        // if port isn't unknown, check to see if it's burnt out.
-        if (!possible_port->second.unknown())
-          if (possible_port->second.percent[x] < burnt_percent) burnt = true;
-        if (!port->second.unknown())
-          if (port->second.percent[x] < burnt_percent) burnt = true;
-      }
-    }
-    if (burnt) continue;
+    trade_type_result ttr = trade_type_info(
+        sector, s);  // port->second.type, possible_port->second.type);
+    if ((ttr.type == NONE) || (ttr.type == FAIR_F)) continue;
 
-    pptv.push_back(port_pair_type{ttr.type, sector, s});
+    pptv.push_back(port_pair_type{ttr.type, ttr.trades, sector, s});
     BUGZ_LOG(trace) << "sector: " << sector << " and " << s
                     << " tt:" << ttr.type;
   }
@@ -758,4 +639,158 @@ port_pair_type Galaxy::find_closest(int sector) {
   port_pair_type ppt;
   ppt.type = 0;
   return ppt;
-}
+}
+
+trade_type_result Galaxy::trade_type_info(sector_type port1, sector_type port2,
+                                          int burnt_percent) {
+  BUGZ_LOG(fatal) << "Trade_type_info(" << port1 << "," << port2 << ")";
+  // This only gives us one trade_type per pair.  There actually
+  // should be multiple values returned here!
+  // Like in case of BBB/SSS:  return 3, 4 and 5.
+
+  // NONE = 0
+  // GOOD = 1 = OE PAIR
+  // OK   = 2 = ?? Pair
+  // FAIR = 3 = B/S E
+  //        4 = B/S O
+  //        5 = B/S F
+  buysell inv2;
+
+  auto p1 = ports.find(port1);
+  if (p1 == ports.end()) {
+    BUGZ_LOG(fatal) << "Can't find port 1: " << (int)port1;
+    return {NONE, inv2};
+  }
+  BUGZ_LOG(fatal) << "port 1: " << p1->first << " " << p1->second.sector << ", "
+                  << (int)p1->second.type;
+  auto p2 = ports.find(port2);
+
+  if (p2 == ports.end()) {
+    BUGZ_LOG(fatal) << "Can't find port 2: " << (int)port2;
+    return {NONE, inv2};
+  }
+  BUGZ_LOG(fatal) << "port 2: " << p2->first << " " << p2->second.sector << ", "
+                  << (int)p2->second.type;
+
+  buysell bsp1 = get_buysell(p1->second.type);
+  buysell bsp2 = get_buysell(p2->second.type);
+
+  inv2 = invert_buysell(bsp2);
+  int matches = 0;  // or pos.size();
+  std::vector<int> pos;
+
+  // find which FOE are flipped.  Save index pos.
+  for (int x = 0; x < 3; ++x) {
+    inv2.foe[x] = (bsp1.foe[x] == inv2.foe[x]);
+    // Ok, these are possible trades (B->S or S->B)
+
+    // If known, check for burnt
+    if (!p1->second.unknown()) {
+      if (p1->second.percent[x] < burnt_percent) {
+        BUGZ_LOG(fatal) << "Marking Port 1: " << x << " (burnt)";
+        inv2.foe[x] = false;
+      }
+    } else {
+      BUGZ_LOG(fatal) << "Port 1 : unknown / skip burnt checks";
+    }
+    if (!p2->second.unknown()) {
+      if (p2->second.percent[x] < burnt_percent) {
+        BUGZ_LOG(fatal) << "Marking Port 2: " << x << " (burnt)";
+        inv2.foe[x] = false;
+      }
+    } else {
+      BUGZ_LOG(fatal) << "Port 2 : unknown / skip burnt checks";
+    }
+
+    if (inv2.foe[x]) {
+      matches++;
+      pos.push_back(x);
+    }
+  }
+
+  BUGZ_LOG(fatal) << "Matches: " << matches << " inv: " << inv2;
+
+  if (matches > 1) {
+    // Check for BEST
+    // O != E for both ports, and O != O, and ORG/EQU in inv2 list
+    if (inv2.foe[ORG] && inv2.foe[EQU] && (bsp1.foe[ORG] != bsp1.foe[EQU]) &&
+        (bsp2.foe[ORG] != bsp2.foe[EQU]) && (bsp1.foe[ORG] != bsp2.foe[ORG])) {
+      // verify that fuel isn't set.
+      inv2.foe[FUEL] = false;
+      BUGZ_LOG(fatal) << "result: " << BEST << " " << inv2;
+      return trade_type_result{BEST, inv2};
+    }
+
+    if (matches == 3) {
+      // This could be SBB / BSS (it's a pair, but not BEST)
+      // Is it FO or FE ?
+
+      if (bsp1.foe[FUEL] != bsp2.foe[EQU]) {
+        // OK, FE
+        inv2.foe[ORG] = false;
+        BUGZ_LOG(fatal) << "result: " << OK << " " << inv2;
+        return trade_type_result{OK, inv2};
+      }
+
+      if (bsp1.foe[FUEL] != bsp2.foe[ORG]) {
+        // OK, FO
+        inv2.foe[EQU] = false;
+        BUGZ_LOG(fatal) << "result: " << OK << " " << inv2;
+        return trade_type_result{OK, inv2};
+      }
+
+      // Ok, take the highest (EQU)
+      inv2.foe[FUEL] = false;
+      inv2.foe[ORG] = false;
+      BUGZ_LOG(fatal) << "result: " << FAIR_E << " " << inv2;
+      return trade_type_result{FAIR_E, inv2};
+    }
+
+    // 2 matches.  but are they trade pairs?
+    if (bsp1.foe[pos[matches - 1]] != bsp1.foe[pos[matches - 2]]) {
+      // yes!
+      BUGZ_LOG(fatal) << "result: " << OK << " " << inv2;
+      return trade_type_result{OK, inv2};
+    } else {
+      // they are NOT.  Use highest one.  clear the lower flag
+
+      inv2.foe[pos[0]] = false;
+
+      switch (pos[1]) {
+        case 0:
+          BUGZ_LOG(fatal) << "result: " << FAIR_F << " " << inv2;
+          return trade_type_result{FAIR_F, inv2};
+          break;
+        case 1:
+          BUGZ_LOG(fatal) << "result: " << FAIR_O << " " << inv2;
+          return trade_type_result{FAIR_O, inv2};
+          break;
+        case 2:
+          BUGZ_LOG(fatal) << "result: " << FAIR_E << " " << inv2;
+          return trade_type_result{FAIR_E, inv2};
+          break;
+      }
+    }
+  }
+
+  if (matches == 1) {
+    switch (pos[0]) {
+      case 0:
+        BUGZ_LOG(fatal) << "result: " << FAIR_F << " " << inv2;
+        return trade_type_result{FAIR_F, inv2};
+        break;
+      case 1:
+        BUGZ_LOG(fatal) << "result: " << FAIR_O << " " << inv2;
+        return trade_type_result{FAIR_O, inv2};
+        break;
+      case 2:
+        BUGZ_LOG(fatal) << "result: " << FAIR_E << " " << inv2;
+        return trade_type_result{FAIR_E, inv2};
+        break;
+    }
+  }
+
+  // no matches.
+  BUGZ_LOG(fatal) << "result: " << NONE << " " << inv2;
+  return trade_type_result{NONE, inv2};
+}

+ 15 - 3
galaxy.h

@@ -49,15 +49,26 @@ struct sector_warps {
  * @param port2
  * @return int
  */
-int trade_type(port_type port1, port_type port2);
+// int trade_type(port_type port1, port_type port2);
+
+enum trade_types {
+  NONE, 
+  BEST,   // OE PAIR
+  OK,     // PAIR
+  FAIR_E, // BS E
+  FAIR_O, // BS O
+  FAIR_F, // BS F
+};
+
 struct trade_type_result {
-  int type;
+  trade_types type;
   buysell trades;
 };
-trade_type_result trade_type_info(port_type port1, port_type port2);
+// trade_type_result trade_type_info(port_type port1, port_type port2);
 
 struct port_pair_type {
   int type;
+  buysell trades;
   sector_type s1, s2;
 };
 
@@ -130,6 +141,7 @@ class Galaxy {
   std::vector<port_pair_type> find_best_trades(void);
   std::vector<port_pair_type> find_trades(sector_type sector,
                                           bool highest = true);
+  trade_type_result trade_type_info(sector_type port1, sector_type port2, int burnt_percent = 20);
   void sort_port_pair_type(std::vector<port_pair_type>& pptv);
   port_pair_type find_closest(int sector);
 

+ 54 - 7
scripts.cpp

@@ -19,13 +19,17 @@ void ScriptTrader::activate(void) {
   int port1_type = port_info->second.type;
   port_buysell[1] = get_buysell(port1_type);
   BUGZ_LOG(fatal) << port0_type << " and " << port1_type;
-  auto ttr = trade_type_info(port0_type, port1_type);
+  /*
+  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
@@ -196,12 +200,12 @@ void ScriptTrader::server_prompt(const std::string &prompt) {
       BUGZ_LOG(fatal) << "port0:" << text_from_buysell(port_buysell[0]);
       BUGZ_LOG(fatal) << "port1:" << text_from_buysell(port_buysell[1]);
 
-      // Do we have what they are buying?
-      bool have_buy = false;
-      int active_buy = 0;
-      int active_sell = 0;
-      bool all_holds_empty = false;
+      // 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"]) {
@@ -209,6 +213,46 @@ void ScriptTrader::server_prompt(const std::string &prompt) {
           all_holds_empty = true;
       }
 
+      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...
@@ -256,6 +300,7 @@ void ScriptTrader::server_prompt(const std::string &prompt) {
           return;
         }
       }
+#endif
 
       state = 2;
       if (director.current_sector == active_port) {
@@ -348,8 +393,9 @@ void ScriptTrader::server_prompt(const std::string &prompt) {
           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) {
-              // We found the item ... is it something that we're trading?
               if (trades.foe[x]) {
                 // Yes!
                 to_server("\r");
@@ -360,6 +406,7 @@ void ScriptTrader::server_prompt(const std::string &prompt) {
               }
             }
           }
+        // }
       }
     }
 

+ 2 - 0
scripts.h

@@ -38,7 +38,9 @@ class ScriptTrader : public Dispatch {
   // information from the find_best_trades function + others.
   int port[2];
   int active;
+  // I probably don't care about trade type, juse
   int type;
+  
   buysell trades;
   buysell port_buysell[2];
 

+ 3 - 0
test-galaxy.cpp

@@ -139,6 +139,7 @@ TEST(ports, trade_types) {
   GTEST_COUT << "Done!" << std::endl;
   */
 
+#ifdef NOT_ANYMORE
   for (int p1 = 1; p1 <= 8; ++p1) {
     for (int p2 = p1; p2 <= 8; ++p2) {
       // GTEST_COUT << "Type " << p1 << " and " << p2 << std::endl;
@@ -156,6 +157,8 @@ TEST(ports, trade_types) {
       }
     }
   }
+#endif
+  
 }
 
 TEST(ports, parse_portcim) {