|
@@ -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 §or : 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};
|
|
|
+}
|