galaxy.cpp 22 KB

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