galaxy.cpp 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796
  1. #include "galaxy.h"
  2. #include <algorithm> // sort
  3. #include <boost/format.hpp>
  4. #include <chrono>
  5. #include <exception>
  6. #include <fstream>
  7. #include <ostream>
  8. #include <set>
  9. #include <string>
  10. #include "config.h"
  11. #include "logging.h"
  12. #include "yaml-cpp/yaml.h"
  13. // c++ default exceptions list
  14. // https://en.cppreference.com/w/cpp/error/exception
  15. std::ostream &operator<<(std::ostream &os, const port &p) {
  16. if (p.type == 0) {
  17. os << p.sector << ": " << (int)p.type;
  18. } else {
  19. os << p.sector << ": " << (int)p.type << " " << text_from_type(p.type)
  20. << " " << p.amount[0] << "," << p.amount[1] << "," << p.amount[2];
  21. }
  22. return os;
  23. }
  24. /**
  25. * Is port unknown?
  26. *
  27. * As in we haven't visited it, we don't know what it has?
  28. * We were checking percent to 0, but we've seen valid ports with 0 percent.
  29. * We now check amount. If this becomes an issue, we'll change to an unknown
  30. * flag.
  31. *
  32. * @return true
  33. * @return false
  34. */
  35. bool port::unknown(void) {
  36. for (int x = 0; x < 3; ++x) {
  37. if (amount[x] != 0) return false;
  38. }
  39. return true;
  40. }
  41. density_scan::density_scan() { reset(0); }
  42. void density_scan::reset(sector_type s) {
  43. sector = s;
  44. pos = 0;
  45. for (int x = 0; x < (int)d.size(); ++x) {
  46. d[x].sector = 0;
  47. }
  48. }
  49. void density_scan::add_scan(density den) {
  50. if (pos > (int)d.size()) {
  51. std::string message("density_scan::add_scan() exceeded max size ");
  52. message.append(std::to_string(d.size()));
  53. throw std::out_of_range(message);
  54. }
  55. d[pos] = den;
  56. ++pos;
  57. }
  58. density density_scan::find(sector_type sector) {
  59. for (int x = 0; x < pos; ++x) {
  60. if (d[x].sector == sector) return d[x];
  61. }
  62. BUGZ_LOG(fatal) << "density_scan::find failed: " << sector << " we have "
  63. << pos << " scans.";
  64. density den;
  65. den.sector = 0;
  66. return den;
  67. }
  68. bool operator==(const density lhs, const density rhs) {
  69. return ((lhs.sector == rhs.sector) && (lhs.density == rhs.density) &&
  70. (lhs.navhaz == rhs.navhaz) && (lhs.anomaly == rhs.anomaly) &&
  71. (lhs.warps == rhs.warps) && (lhs.known == rhs.known));
  72. }
  73. sector_warps::sector_warps() { sector = 0; }
  74. void sector_warps::add(sector_type new_sector) {
  75. warps.insert(new_sector);
  76. /*
  77. for (int x = 0; x < MAX_WARPS; ++x) {
  78. if (warps[x] == new_sector) return;
  79. if (warps[x] == 0) {
  80. warps[x] = new_sector;
  81. return;
  82. }
  83. }
  84. std::string message = str(boost::format("More then MAX %1% sectors for %2%") %
  85. MAX_WARPS % (int)sector);
  86. throw std::out_of_range(message);
  87. */
  88. }
  89. std::ostream &operator<<(std::ostream &os, const sector_warps &warps) {
  90. os << "Sector: " << warps.sector << " ";
  91. bool comma = false;
  92. for (auto const &warp : warps.warps) {
  93. if (comma)
  94. os << ",";
  95. else
  96. comma = true;
  97. os << warp;
  98. }
  99. return os;
  100. }
  101. #define GTEST_COUT std::cerr << "[ ] [ INFO ]"
  102. // #define GTEST_DEBUG
  103. // TODO: fix this. I want some trace output, but I don't want
  104. // my logs flooded ...
  105. struct port parse_portcim(const std::string line) {
  106. struct port p;
  107. p.sector = std::stoi(line);
  108. // 20 - 1708 97% - 710 56% 287 15%
  109. static std::regex portrx(
  110. "[ ]*([0-9]+) (.)[ ]+([0-9]+)[ ]+([0-9]+%) (.)[ "
  111. "]+([0-9]+)[ ]+([0-9]+%) (.)[ ]+([0-9]+)[ ]+([0-9]+%)[ ]*",
  112. std::regex_constants::ECMAScript);
  113. // does it not understand {3} ??
  114. // NO, it does not, from regex101.com:
  115. // A repeated capturing group will only capture the last iteration. Put a
  116. // capturing group around the repeated group to capture all iterations or use
  117. // a non-capturing group instead if you're not interested in the data
  118. //
  119. // static std::regex portrx("[ ]*([0-9]+)( (.)[ ]+([0-9]+)[ ]+([0-9]+%)){3}[
  120. // ]*",
  121. // std::regex_constants::ECMAScript);
  122. // sector + amount pct + amount pct + amount pct
  123. // 1 2 3 4 5 6 7 8 9 10
  124. #ifdef GTEST_DEBUG
  125. GTEST_COUT << "Sector: " << p.sector << std::endl;
  126. GTEST_COUT << "Line: [" << line << "]" << std::endl;
  127. #endif
  128. buysell port_buysell;
  129. std::smatch matches;
  130. if (std::regex_match(line, matches, portrx)) {
  131. #ifdef GTEST_DEBUG
  132. for (size_t x = 1; x < matches.size(); ++x) {
  133. GTEST_COUT << x << " : " << matches[x] << std::endl;
  134. }
  135. #endif
  136. if (matches.size() != 11) {
  137. #ifdef GTEST_DEBUG
  138. GTEST_COUT << "Now you have 101 problems." << std::endl;
  139. #endif
  140. p.sector = 0;
  141. p.type = 0;
  142. return p;
  143. }
  144. // GTEST_COUT << "matches: " << matches.size() << std::endl;
  145. p.sector = stoi(matches[1]);
  146. // GTEST_COUT << "sector: " << matches[1] << std::endl;
  147. // for (int x = 1; x < 11; ++x) {
  148. // GTEST_COUT << x << " : " << matches[x] << std::endl;
  149. // }
  150. for (int x = 0; x < 3; ++x) {
  151. int pos = x * 3;
  152. port_buysell.foe[x] = matches[pos + 2] == "-";
  153. p.amount[x] = stoi(matches[pos + 3]);
  154. p.percent[x] = stoi(matches[pos + 4]);
  155. }
  156. p.type = type_from_buysell(port_buysell);
  157. #ifdef GTEST_DEBUG
  158. GTEST_COUT << "port is type " << (int)p.type << std::endl;
  159. #endif
  160. return p;
  161. } else {
  162. #ifdef GTEST_DEBUG
  163. GTEST_COUT << "regex_match failed." << std::endl;
  164. #endif
  165. p.type = 0;
  166. p.sector = 0;
  167. return p;
  168. }
  169. }
  170. Galaxy::Galaxy() { burnt_percent = 40; }
  171. Galaxy::~Galaxy() { BUGZ_LOG(fatal) << "Galaxy::~Galaxy()"; }
  172. void Galaxy::reset(void) {
  173. meta = YAML::Node();
  174. config = YAML::Node();
  175. ports.clear();
  176. warps.clear();
  177. }
  178. void Galaxy::add_warp(sector_warps sw) {
  179. auto pos = warps.find(sw.sector);
  180. if (pos == warps.end()) {
  181. // not found
  182. // sw.sort();
  183. warps[sw.sector] = sw;
  184. // BUGZ_LOG(info) << "add_warp NEW " << sw.sector;
  185. } else {
  186. // found!
  187. if (pos->second.warps == sw.warps) {
  188. // BUGZ_LOG(trace) << "add_warp: Yup, I already know about " << sw.sector;
  189. } else {
  190. BUGZ_LOG(info) << "add_warp: Warps don't match! Updating...";
  191. BUGZ_LOG(warning) << "Have: " << pos->second;
  192. BUGZ_LOG(warning) << "Got : " << sw;
  193. warps[sw.sector] = sw;
  194. }
  195. }
  196. }
  197. void Galaxy::add_port(sector_type sector, int port_type) {
  198. auto pos = ports.find(sector);
  199. if (pos == ports.end()) {
  200. // no such port.
  201. port p;
  202. p.sector = sector;
  203. p.type = port_type;
  204. for (int x = 0; x < 3; x++) {
  205. p.amount[x] = 0;
  206. p.percent[x] = 0;
  207. }
  208. // BUGZ_LOG(trace) << "add_port: " << sector << ", " << port_type << " : "
  209. // << p;
  210. ports[sector] = p;
  211. } else {
  212. // port was found, so:
  213. if (pos->second.type == port_type) {
  214. // BUGZ_LOG(trace) << "add_port: Yup, port " << sector << " is class " <<
  215. // port_type;
  216. } else {
  217. BUGZ_LOG(fatal) << "add_port: " << sector << " shows " << pos->second.type
  218. << " >> set to " << port_type;
  219. pos->second.type = port_type;
  220. }
  221. }
  222. }
  223. void Galaxy::add_port(port p) {
  224. auto pos = ports.find(p.sector);
  225. if (pos == ports.end()) {
  226. // BUGZ_LOG(trace) << "add_port: NEW " << p;
  227. ports[p.sector] = p;
  228. } else {
  229. if (pos->second.type != p.type) {
  230. if ((pos->second.type == 9) && (p.type == 8)) {
  231. BUGZ_LOG(trace) << "add_port: StarDock " << p.sector;
  232. p.type = 9;
  233. ports[p.sector] = p;
  234. } else {
  235. BUGZ_LOG(fatal) << "add_port: " << pos->second << " NEW : " << p;
  236. ports[p.sector] = p;
  237. }
  238. } else {
  239. if (pos->second.amount != p.amount) {
  240. // BUGZ_LOG(info) << "add_port: UPDATE " << p.sector;
  241. pos->second = p;
  242. } else {
  243. // BUGZ_LOG(info) << "add_port: Yup " << p.sector;
  244. }
  245. }
  246. }
  247. }
  248. void Galaxy::load(void) {
  249. std::string basename = "galaxy";
  250. if (CONFIG["basename"]) {
  251. basename = CONFIG["basename"].as<std::string>();
  252. }
  253. std::string filename =
  254. str(boost::format("%1%-%2%-%3%.yaml") % basename % game % username);
  255. // reset ?
  256. meta = YAML::Node();
  257. config = YAML::Node();
  258. ports.clear();
  259. warps.clear();
  260. if (file_exists(filename)) {
  261. YAML::Node data = YAML::LoadFile(filename);
  262. if (config["meta"]) meta = config["meta"];
  263. meta["load_from"] = filename;
  264. std::chrono::_V2::system_clock::time_point now =
  265. std::chrono::system_clock::now();
  266. meta["load_time"] = std::chrono::system_clock::to_time_t(now); // time_t
  267. if (data["config"]) {
  268. config = data["config"];
  269. } else {
  270. BUGZ_LOG(fatal) << "YAML Missing config section.";
  271. }
  272. if (data["ports"]) {
  273. const YAML::Node ports = data["ports"];
  274. for (auto const &port_iter : ports) {
  275. port p;
  276. p.sector = port_iter.first.as<int>();
  277. p.type = port_iter.second["class"].as<int>();
  278. int x;
  279. if (p.type == 0) {
  280. // nothing to load for type = 0, set defaults.
  281. for (x = 0; x < 3; ++x) {
  282. p.amount[x] = 0;
  283. p.percent[x] = 0;
  284. }
  285. } else {
  286. x = 0;
  287. for (auto const &amount : port_iter.second["amount"]) {
  288. p.amount[x] = amount.as<int>();
  289. ++x;
  290. }
  291. x = 0;
  292. for (auto const &pct : port_iter.second["pct"]) {
  293. p.percent[x] = pct.as<int>();
  294. ++x;
  295. }
  296. }
  297. add_port(p);
  298. }
  299. } else {
  300. BUGZ_LOG(fatal) << "YAML Missing ports section.";
  301. }
  302. if (data["warps"]) {
  303. const YAML::Node &warps = data["warps"];
  304. // if (warps.IsMap()) {
  305. for (auto const warp_iter : warps) {
  306. sector_warps sw;
  307. sw.sector = warp_iter.first.as<int>();
  308. for (auto const sector_iter : warp_iter.second) {
  309. sw.add(sector_iter.as<int>());
  310. }
  311. // BUGZ_LOG(trace) << "YAML warp: " << sw;
  312. add_warp(sw);
  313. }
  314. // }
  315. } else {
  316. BUGZ_LOG(fatal) << "YAML Missing warps section.";
  317. }
  318. BUGZ_LOG(fatal) << "YAML: config keys: " << config.size();
  319. BUGZ_LOG(fatal) << "YAML: warp keys: " << warps.size();
  320. BUGZ_LOG(fatal) << "YAML: port keys: " << ports.size();
  321. } else {
  322. BUGZ_LOG(fatal) << "Missing YAML: " << filename;
  323. }
  324. }
  325. void Galaxy::save(void) {
  326. std::string basename = "galaxy";
  327. if (CONFIG["basename"]) {
  328. basename = CONFIG["basename"].as<std::string>();
  329. }
  330. std::string filename =
  331. str(boost::format("%1%-%2%-%3%.yaml") % basename % game % username);
  332. std::ofstream fout(filename);
  333. fout << "%YAML 1.2" << std::endl << "---" << std::endl;
  334. BUGZ_LOG(fatal) << "YAML: " << filename;
  335. int depth = 0;
  336. std::string depth_spacer;
  337. meta["save_to"] = filename;
  338. std::chrono::_V2::system_clock::time_point now =
  339. std::chrono::system_clock::now();
  340. meta["save_time"] = std::chrono::system_clock::to_time_t(now); // time_t
  341. // meta:
  342. fout << "meta:" << std::endl;
  343. ++depth;
  344. depth_spacer.assign(depth * 2, ' ');
  345. std::function<void(std::ofstream & of, int depth, const YAML::Node &n)>
  346. yaml_out;
  347. yaml_out = [&yaml_out](std::ofstream &of, int yaml_depth,
  348. const YAML::Node &n) {
  349. std::string yaml_spacer;
  350. yaml_spacer.assign(yaml_depth * 2, ' ');
  351. for (auto const &data : n) {
  352. if (data.second.Type() == YAML::NodeType::Scalar) {
  353. of << yaml_spacer << data.first << ": " << data.second << std::endl;
  354. } else if (data.second.Type() == YAML::NodeType::Map) {
  355. // nested
  356. of << yaml_spacer << data.first << ":" << std::endl;
  357. yaml_out(of, yaml_depth + 1, data.second);
  358. } else if (data.second.Type() == YAML::NodeType::Sequence) {
  359. // sequence
  360. BUGZ_LOG(fatal) << "Ignoring Sequence... " << data.first;
  361. } else {
  362. BUGZ_LOG(fatal) << "Unsupported NodeType: " << data.second.Type();
  363. }
  364. }
  365. };
  366. yaml_out(fout, depth, meta);
  367. BUGZ_LOG(fatal) << "YAML config: " << config.size();
  368. fout << "config:" << std::endl;
  369. // in config, I usually switch to doing flow instead. I'll keep this like
  370. // this for now.
  371. yaml_out(fout, depth, config);
  372. BUGZ_LOG(fatal) << "YAML warps: " << warps.size();
  373. fout << "warps:" << std::endl;
  374. for (auto const &warp : warps) {
  375. fout << depth_spacer << warp.first << ": [";
  376. bool first = true;
  377. for (auto const &sector : warp.second.warps) {
  378. if (!first) {
  379. fout << ", ";
  380. } else
  381. first = false;
  382. fout << sector;
  383. }
  384. fout << "]" << std::endl;
  385. }
  386. BUGZ_LOG(fatal) << "YAML ports: " << ports.size();
  387. fout << "ports:" << std::endl;
  388. /*
  389. When saving to yaml, my sector_type is like char. So, it wants
  390. to save the values as a character. Cast to int.
  391. */
  392. for (auto const &port : ports) {
  393. fout << depth_spacer << port.second.sector
  394. << ": {class: " << (int)port.second.type;
  395. if (port.second.type == 0) {
  396. fout << "}" << std::endl;
  397. } else {
  398. // write out the rest of the information
  399. fout << ", amount: [";
  400. for (int x = 0; x < 3; x++) {
  401. fout << port.second.amount[x];
  402. if (x != 2) fout << ", ";
  403. }
  404. fout << "], pct: [";
  405. for (int x = 0; x < 3; x++) {
  406. fout << (int)port.second.percent[x];
  407. if (x != 2) fout << ", ";
  408. }
  409. fout << "]}" << std::endl;
  410. }
  411. }
  412. // the big data structures later on are the ones I really need to optimize
  413. // here. Writing out 20 lines isn't what is killing us!
  414. /*
  415. for (auto const & meta_data : meta ) {
  416. std::cerr << "key: " << meta_data.first << std::endl;
  417. if (meta_data.second.Type() == YAML::NodeType::Scalar) {
  418. fout << depth_spacer << meta_data.first << ": " << meta_data.second <<
  419. std::endl; } else {
  420. // nested structure
  421. std::cerr << "Nested: " << meta_data.first << std::endl;
  422. fout << depth_spacer << meta_data.first << ":" << std::endl;
  423. ++depth;
  424. depth_spacer.assign(depth *2, ' ');
  425. for (auto const &nested : meta_data.second) {
  426. if (nested.second.Type() == YAML::NodeType::Scalar) {
  427. fout << depth_spacer << nested.first << ": " << nested.second <<
  428. std::endl; } else { std::cerr << "double-Nested meta structure under key: " <<
  429. nested.first << std::endl;
  430. }
  431. }
  432. --depth;
  433. depth_spacer.assign(depth * 2, ' ');
  434. }
  435. }
  436. */
  437. }
  438. std::vector<port_pair_type> Galaxy::find_trades(sector_type sector,
  439. bool highest) {
  440. std::vector<port_pair_type> pptv;
  441. // Does this sector have a port?
  442. auto port = ports.find(sector);
  443. if (port == ports.end()) return pptv;
  444. auto port_warps = warps.find(sector);
  445. // warps not found for port sector?
  446. if (port_warps == warps.end()) return pptv;
  447. for (auto const &s : port_warps->second.warps) {
  448. // only count the ports > our sector number -- so we don't have dups
  449. if (highest && (s < sector)) continue;
  450. // verify we have a way back
  451. {
  452. auto warpback = warps.find(s);
  453. // can we find that warp?
  454. if (warpback == warps.end()) continue;
  455. // does it link back to the port's sector?
  456. if (warpback->second.warps.find(sector) == warpback->second.warps.end())
  457. continue;
  458. }
  459. // Does this sector have a port?
  460. auto possible_port = ports.find(s);
  461. if (possible_port == ports.end()) continue;
  462. if (possible_port->second.type == 0) continue;
  463. // calculate trade type
  464. // int t = trade_type(port->second.type, possible_port->second.type);
  465. BUGZ_LOG(trace) << "find_trades: Port " << sector << ","
  466. << (int)port->second.type << " " << s
  467. << (int)possible_port->second.type;
  468. trade_type_result ttr = trade_type_info(
  469. sector, s); // port->second.type, possible_port->second.type);
  470. if ((ttr.type == NONE) || (ttr.type == FAIR_F)) continue;
  471. pptv.push_back(port_pair_type{ttr.type, ttr.trades, sector, s});
  472. BUGZ_LOG(trace) << "sector: " << sector << " and " << s
  473. << " tt:" << ttr.type;
  474. }
  475. return pptv;
  476. }
  477. bool compare_port_pair(const port_pair_type &ppt1, const port_pair_type &ppt2) {
  478. if (ppt1.type == ppt2.type) {
  479. return ppt1.s1 < ppt2.s1;
  480. } else {
  481. return (ppt1.type < ppt2.type);
  482. }
  483. }
  484. void Galaxy::sort_port_pair_type(std::vector<port_pair_type> &pptv) {
  485. sort(pptv.begin(), pptv.end(), compare_port_pair);
  486. }
  487. std::vector<port_pair_type> Galaxy::find_best_trades(void) {
  488. std::vector<port_pair_type> pptv;
  489. burnt_percent = config["burnt_percent"].as<int>();
  490. if (burnt_percent > 90) {
  491. burnt_percent = 90;
  492. config["burnt_percent"] = 90;
  493. }
  494. if (burnt_percent < 10) {
  495. burnt_percent = 10;
  496. config["burnt_percent"] = 10;
  497. }
  498. for (auto const &pi : ports) {
  499. if (pi.second.type == 0) continue;
  500. sector_type sector = pi.second.sector;
  501. auto port_warps = warps.find(sector);
  502. if (port_warps == warps.end()) continue;
  503. std::vector<port_pair_type> ppt_sector = find_trades(sector, true);
  504. if (ppt_sector.empty()) continue;
  505. pptv.insert(pptv.end(), ppt_sector.begin(), ppt_sector.end());
  506. }
  507. return pptv;
  508. }
  509. port_pair_type Galaxy::find_closest(int sector) {
  510. auto trades = find_best_trades();
  511. // int type, sector_type s1, s2;
  512. std::set<sector_type> seen;
  513. std::set<sector_type> current;
  514. current.insert(sector);
  515. bool found = false;
  516. bool added_new = false;
  517. int depth = 0;
  518. while (!found) {
  519. // is current list in any of the trades ?
  520. for (auto const &t : trades) {
  521. if (t.type > 2) continue;
  522. if (current.find(t.s1) != current.end()) {
  523. // found one!
  524. return t;
  525. }
  526. if (current.find(t.s2) != current.end()) {
  527. return t;
  528. }
  529. }
  530. ++depth;
  531. BUGZ_LOG(info) << "depth: " << depth << " current:" << current.size()
  532. << " seen: " << seen.size();
  533. added_new = false;
  534. if (!found) {
  535. // update the seen
  536. seen.insert(current.begin(), current.end());
  537. auto look_in = current;
  538. current.clear();
  539. for (auto const &li : look_in) {
  540. auto wi = warps.find(li);
  541. if (wi == warps.end()) continue;
  542. for (auto const &w : wi->second.warps) {
  543. // have we already seen this sector?
  544. if (seen.find(w) != seen.end()) continue;
  545. // does it have a warp back to the original sector?
  546. auto warp_back = warps.find(w);
  547. if (warp_back != warps.end()) {
  548. if (warp_back->second.warps.find(li) !=
  549. warp_back->second.warps.end()) {
  550. // Ok, this links back to the original sector...
  551. current.insert(w);
  552. added_new = true;
  553. }
  554. }
  555. }
  556. }
  557. if (!added_new) {
  558. BUGZ_LOG(warning) << "No new sectors added. We're done!";
  559. port_pair_type ppt;
  560. ppt.type = 0;
  561. return ppt;
  562. }
  563. }
  564. }
  565. port_pair_type ppt;
  566. ppt.type = 0;
  567. return ppt;
  568. }
  569. trade_type_result Galaxy::trade_type_info(sector_type port1, sector_type port2,
  570. int burnt_percent) {
  571. BUGZ_LOG(fatal) << "Trade_type_info(" << port1 << "," << port2 << ")";
  572. // This only gives us one trade_type per pair. There actually
  573. // should be multiple values returned here!
  574. // Like in case of BBB/SSS: return 3, 4 and 5.
  575. // NONE = 0
  576. // GOOD = 1 = OE PAIR
  577. // OK = 2 = ?? Pair
  578. // FAIR = 3 = B/S E
  579. // 4 = B/S O
  580. // 5 = B/S F
  581. buysell inv2;
  582. auto p1 = ports.find(port1);
  583. if (p1 == ports.end()) {
  584. BUGZ_LOG(fatal) << "Can't find port 1: " << (int)port1;
  585. return {NONE, inv2};
  586. }
  587. BUGZ_LOG(fatal) << "port 1: " << p1->first << " " << p1->second.sector << ", "
  588. << (int)p1->second.type;
  589. auto p2 = ports.find(port2);
  590. if (p2 == ports.end()) {
  591. BUGZ_LOG(fatal) << "Can't find port 2: " << (int)port2;
  592. return {NONE, inv2};
  593. }
  594. BUGZ_LOG(fatal) << "port 2: " << p2->first << " " << p2->second.sector << ", "
  595. << (int)p2->second.type;
  596. buysell bsp1 = get_buysell(p1->second.type);
  597. buysell bsp2 = get_buysell(p2->second.type);
  598. inv2 = invert_buysell(bsp2);
  599. int matches = 0; // or pos.size();
  600. std::vector<int> pos;
  601. // find which FOE are flipped. Save index pos.
  602. for (int x = 0; x < 3; ++x) {
  603. inv2.foe[x] = (bsp1.foe[x] == inv2.foe[x]);
  604. // Ok, these are possible trades (B->S or S->B)
  605. // If known, check for burnt
  606. if (!p1->second.unknown()) {
  607. if (p1->second.percent[x] < burnt_percent) {
  608. BUGZ_LOG(fatal) << "Marking Port 1: " << x << " (burnt)";
  609. inv2.foe[x] = false;
  610. }
  611. } else {
  612. BUGZ_LOG(fatal) << "Port 1 : unknown / skip burnt checks";
  613. }
  614. if (!p2->second.unknown()) {
  615. if (p2->second.percent[x] < burnt_percent) {
  616. BUGZ_LOG(fatal) << "Marking Port 2: " << x << " (burnt)";
  617. inv2.foe[x] = false;
  618. }
  619. } else {
  620. BUGZ_LOG(fatal) << "Port 2 : unknown / skip burnt checks";
  621. }
  622. if (inv2.foe[x]) {
  623. matches++;
  624. pos.push_back(x);
  625. }
  626. }
  627. BUGZ_LOG(fatal) << "Matches: " << matches << " inv: " << inv2;
  628. if (matches > 1) {
  629. // Check for BEST
  630. // O != E for both ports, and O != O, and ORG/EQU in inv2 list
  631. if (inv2.foe[ORG] && inv2.foe[EQU] && (bsp1.foe[ORG] != bsp1.foe[EQU]) &&
  632. (bsp2.foe[ORG] != bsp2.foe[EQU]) && (bsp1.foe[ORG] != bsp2.foe[ORG])) {
  633. // verify that fuel isn't set.
  634. inv2.foe[FUEL] = false;
  635. BUGZ_LOG(fatal) << "result: " << BEST << " " << inv2;
  636. return trade_type_result{BEST, inv2};
  637. }
  638. if (matches == 3) {
  639. // This could be SBB / BSS (it's a pair, but not BEST)
  640. // Is it FO or FE ?
  641. if (bsp1.foe[FUEL] != bsp2.foe[EQU]) {
  642. // OK, FE
  643. inv2.foe[ORG] = false;
  644. BUGZ_LOG(fatal) << "result: " << OK << " " << inv2;
  645. return trade_type_result{OK, inv2};
  646. }
  647. if (bsp1.foe[FUEL] != bsp2.foe[ORG]) {
  648. // OK, FO
  649. inv2.foe[EQU] = false;
  650. BUGZ_LOG(fatal) << "result: " << OK << " " << inv2;
  651. return trade_type_result{OK, inv2};
  652. }
  653. // Ok, take the highest (EQU)
  654. inv2.foe[FUEL] = false;
  655. inv2.foe[ORG] = false;
  656. BUGZ_LOG(fatal) << "result: " << FAIR_E << " " << inv2;
  657. return trade_type_result{FAIR_E, inv2};
  658. }
  659. // 2 matches. but are they trade pairs?
  660. if (bsp1.foe[pos[matches - 1]] != bsp1.foe[pos[matches - 2]]) {
  661. // yes!
  662. BUGZ_LOG(fatal) << "result: " << OK << " " << inv2;
  663. return trade_type_result{OK, inv2};
  664. } else {
  665. // they are NOT. Use highest one. clear the lower flag
  666. inv2.foe[pos[0]] = false;
  667. switch (pos[1]) {
  668. case 0:
  669. BUGZ_LOG(fatal) << "result: " << FAIR_F << " " << inv2;
  670. return trade_type_result{FAIR_F, inv2};
  671. break;
  672. case 1:
  673. BUGZ_LOG(fatal) << "result: " << FAIR_O << " " << inv2;
  674. return trade_type_result{FAIR_O, inv2};
  675. break;
  676. case 2:
  677. BUGZ_LOG(fatal) << "result: " << FAIR_E << " " << inv2;
  678. return trade_type_result{FAIR_E, inv2};
  679. break;
  680. }
  681. }
  682. }
  683. if (matches == 1) {
  684. switch (pos[0]) {
  685. case 0:
  686. BUGZ_LOG(fatal) << "result: " << FAIR_F << " " << inv2;
  687. return trade_type_result{FAIR_F, inv2};
  688. break;
  689. case 1:
  690. BUGZ_LOG(fatal) << "result: " << FAIR_O << " " << inv2;
  691. return trade_type_result{FAIR_O, inv2};
  692. break;
  693. case 2:
  694. BUGZ_LOG(fatal) << "result: " << FAIR_E << " " << inv2;
  695. return trade_type_result{FAIR_E, inv2};
  696. break;
  697. }
  698. }
  699. // no matches.
  700. BUGZ_LOG(fatal) << "result: " << NONE << " " << inv2;
  701. return trade_type_result{NONE, inv2};
  702. }