galaxy.cpp 28 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000
  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. sector_warps::sector_warps(sector_type s, std::set<sector_type> w) {
  75. sector = s;
  76. warps = w;
  77. }
  78. void sector_warps::add(sector_type new_sector) {
  79. warps.insert(new_sector);
  80. /*
  81. for (int x = 0; x < MAX_WARPS; ++x) {
  82. if (warps[x] == new_sector) return;
  83. if (warps[x] == 0) {
  84. warps[x] = new_sector;
  85. return;
  86. }
  87. }
  88. std::string message = str(boost::format("More then MAX %1% sectors for %2%") %
  89. MAX_WARPS % (int)sector);
  90. throw std::out_of_range(message);
  91. */
  92. }
  93. std::ostream &operator<<(std::ostream &os, const sector_warps &warps) {
  94. os << "Sector: " << warps.sector << " ";
  95. bool comma = false;
  96. for (auto const &warp : warps.warps) {
  97. if (comma)
  98. os << ",";
  99. else
  100. comma = true;
  101. os << warp;
  102. }
  103. return os;
  104. }
  105. #define GTEST_COUT std::cerr << "[ ] [ INFO ]"
  106. // #define GTEST_DEBUG
  107. // TODO: fix this. I want some trace output, but I don't want
  108. // my logs flooded ...
  109. struct port parse_portcim(const std::string line) {
  110. struct port p;
  111. p.sector = std::stoi(line);
  112. // 20 - 1708 97% - 710 56% 287 15%
  113. static std::regex portrx(
  114. "[ ]*([0-9]+) (.)[ ]+([0-9]+)[ ]+([0-9]+%) (.)[ "
  115. "]+([0-9]+)[ ]+([0-9]+%) (.)[ ]+([0-9]+)[ ]+([0-9]+%)[ ]*",
  116. std::regex_constants::ECMAScript);
  117. // does it not understand {3} ??
  118. // NO, it does not, from regex101.com:
  119. // A repeated capturing group will only capture the last iteration. Put a
  120. // capturing group around the repeated group to capture all iterations or use
  121. // a non-capturing group instead if you're not interested in the data
  122. //
  123. // static std::regex portrx("[ ]*([0-9]+)( (.)[ ]+([0-9]+)[ ]+([0-9]+%)){3}[
  124. // ]*",
  125. // std::regex_constants::ECMAScript);
  126. // sector + amount pct + amount pct + amount pct
  127. // 1 2 3 4 5 6 7 8 9 10
  128. #ifdef GTEST_DEBUG
  129. GTEST_COUT << "Sector: " << p.sector << std::endl;
  130. GTEST_COUT << "Line: [" << line << "]" << std::endl;
  131. #endif
  132. buysell port_buysell;
  133. std::smatch matches;
  134. if (std::regex_match(line, matches, portrx)) {
  135. #ifdef GTEST_DEBUG
  136. for (size_t x = 1; x < matches.size(); ++x) {
  137. GTEST_COUT << x << " : " << matches[x] << std::endl;
  138. }
  139. #endif
  140. if (matches.size() != 11) {
  141. #ifdef GTEST_DEBUG
  142. GTEST_COUT << "Now you have 101 problems." << std::endl;
  143. #endif
  144. p.sector = 0;
  145. p.type = 0;
  146. return p;
  147. }
  148. // GTEST_COUT << "matches: " << matches.size() << std::endl;
  149. p.sector = stoi(matches[1]);
  150. // GTEST_COUT << "sector: " << matches[1] << std::endl;
  151. // for (int x = 1; x < 11; ++x) {
  152. // GTEST_COUT << x << " : " << matches[x] << std::endl;
  153. // }
  154. for (int x = 0; x < 3; ++x) {
  155. int pos = x * 3;
  156. port_buysell.foe[x] = matches[pos + 2] == "-";
  157. p.amount[x] = stoi(matches[pos + 3]);
  158. p.percent[x] = stoi(matches[pos + 4]);
  159. }
  160. p.type = type_from_buysell(port_buysell);
  161. #ifdef GTEST_DEBUG
  162. GTEST_COUT << "port is type " << (int)p.type << std::endl;
  163. #endif
  164. return p;
  165. } else {
  166. #ifdef GTEST_DEBUG
  167. GTEST_COUT << "regex_match failed." << std::endl;
  168. #endif
  169. p.type = 0;
  170. p.sector = 0;
  171. return p;
  172. }
  173. }
  174. Galaxy::Galaxy() { burnt_percent = 40; }
  175. Galaxy::~Galaxy() { BUGZ_LOG(fatal) << "Galaxy::~Galaxy()"; }
  176. void Galaxy::reset(void) {
  177. meta = YAML::Node();
  178. config = YAML::Node();
  179. ports.clear();
  180. warps.clear();
  181. }
  182. void Galaxy::add_warp(sector_warps sw) {
  183. auto pos = warps.find(sw.sector);
  184. if (pos == warps.end()) {
  185. // not found
  186. // sw.sort();
  187. warps[sw.sector] = sw;
  188. // BUGZ_LOG(info) << "add_warp NEW " << sw.sector;
  189. } else {
  190. // found!
  191. if (pos->second.warps == sw.warps) {
  192. // BUGZ_LOG(trace) << "add_warp: Yup, I already know about " << sw.sector;
  193. } else {
  194. BUGZ_LOG(info) << "add_warp: Warps don't match! Updating...";
  195. BUGZ_LOG(warning) << "Have: " << pos->second;
  196. BUGZ_LOG(warning) << "Got : " << sw;
  197. warps[sw.sector] = sw;
  198. }
  199. }
  200. }
  201. void Galaxy::add_port(sector_type sector, int port_type) {
  202. auto pos = ports.find(sector);
  203. if (pos == ports.end()) {
  204. // no such port.
  205. port p;
  206. p.sector = sector;
  207. p.type = port_type;
  208. for (int x = 0; x < 3; x++) {
  209. p.amount[x] = 0;
  210. p.percent[x] = 0;
  211. }
  212. // BUGZ_LOG(trace) << "add_port: " << sector << ", " << port_type << " : "
  213. // << p;
  214. ports[sector] = p;
  215. } else {
  216. // port was found, so:
  217. if (pos->second.type == port_type) {
  218. // BUGZ_LOG(trace) << "add_port: Yup, port " << sector << " is class " <<
  219. // port_type;
  220. } else {
  221. BUGZ_LOG(fatal) << "add_port: " << sector << " shows " << pos->second.type
  222. << " >> set to " << port_type;
  223. pos->second.type = port_type;
  224. }
  225. }
  226. }
  227. void Galaxy::add_port(port p) {
  228. auto pos = ports.find(p.sector);
  229. if (pos == ports.end()) {
  230. // BUGZ_LOG(trace) << "add_port: NEW " << p;
  231. ports[p.sector] = p;
  232. } else {
  233. if (pos->second.type != p.type) {
  234. if ((pos->second.type == 9) && (p.type == 8)) {
  235. BUGZ_LOG(trace) << "add_port: StarDock " << p.sector;
  236. p.type = 9;
  237. ports[p.sector] = p;
  238. } else {
  239. BUGZ_LOG(fatal) << "add_port: " << pos->second << " NEW : " << p;
  240. ports[p.sector] = p;
  241. }
  242. } else {
  243. if (pos->second.amount != p.amount) {
  244. // BUGZ_LOG(info) << "add_port: UPDATE " << p.sector;
  245. pos->second = p;
  246. } else {
  247. // BUGZ_LOG(info) << "add_port: Yup " << p.sector;
  248. }
  249. }
  250. }
  251. }
  252. void Galaxy::load(void) {
  253. std::string basename = "galaxy";
  254. if (CONFIG["basename"]) {
  255. basename = CONFIG["basename"].as<std::string>();
  256. }
  257. std::string filename =
  258. str(boost::format("%1%-%2%-%3%.yaml") % basename % game % username);
  259. // reset ?
  260. meta = YAML::Node();
  261. config = YAML::Node();
  262. ports.clear();
  263. warps.clear();
  264. if (file_exists(filename)) {
  265. BUGZ_LOG(fatal) << "Galaxy::load( " << filename << " )";
  266. YAML::Node data = YAML::LoadFile(filename);
  267. if (data["meta"]) {
  268. meta = data["meta"];
  269. } else {
  270. BUGZ_LOG(fatal) << "Missing meta data.";
  271. }
  272. meta["load_from"] = filename;
  273. meta["load_time"] = time_t_now(); // time_t
  274. if (data["config"]) {
  275. config = data["config"];
  276. } else {
  277. BUGZ_LOG(fatal) << "YAML Missing config section.";
  278. }
  279. if (data["ports"]) {
  280. const YAML::Node ports = data["ports"];
  281. for (auto const &port_iter : ports) {
  282. port p;
  283. p.sector = port_iter.first.as<int>();
  284. p.type = port_iter.second["class"].as<int>();
  285. int x;
  286. if (p.type == 0) {
  287. // nothing to load for type = 0, set defaults.
  288. for (x = 0; x < 3; ++x) {
  289. p.amount[x] = 0;
  290. p.percent[x] = 0;
  291. }
  292. } else {
  293. x = 0;
  294. for (auto const &amount : port_iter.second["amount"]) {
  295. p.amount[x] = amount.as<int>();
  296. ++x;
  297. }
  298. x = 0;
  299. for (auto const &pct : port_iter.second["pct"]) {
  300. p.percent[x] = pct.as<int>();
  301. ++x;
  302. }
  303. }
  304. add_port(p);
  305. }
  306. } else {
  307. BUGZ_LOG(fatal) << "YAML Missing ports section.";
  308. }
  309. if (data["warps"]) {
  310. const YAML::Node &warps = data["warps"];
  311. // if (warps.IsMap()) {
  312. for (auto const warp_iter : warps) {
  313. sector_warps sw;
  314. sw.sector = warp_iter.first.as<int>();
  315. for (auto const sector_iter : warp_iter.second) {
  316. sw.add(sector_iter.as<int>());
  317. }
  318. // BUGZ_LOG(trace) << "YAML warp: " << sw;
  319. add_warp(sw);
  320. }
  321. // }
  322. } else {
  323. BUGZ_LOG(fatal) << "YAML Missing warps section.";
  324. }
  325. BUGZ_LOG(trace) << "YAML: meta keys: " << meta.size();
  326. BUGZ_LOG(trace) << "YAML: config keys: " << config.size();
  327. BUGZ_LOG(trace) << "YAML: warp keys: " << warps.size();
  328. BUGZ_LOG(trace) << "YAML: port keys: " << ports.size();
  329. } else {
  330. BUGZ_LOG(fatal) << "Missing YAML: " << filename;
  331. }
  332. }
  333. void Galaxy::save(void) {
  334. std::string basename = "galaxy";
  335. if (CONFIG["basename"]) {
  336. basename = CONFIG["basename"].as<std::string>();
  337. }
  338. std::string filename =
  339. str(boost::format("%1%-%2%-%3%.yaml") % basename % game % username);
  340. std::ofstream fout(filename);
  341. // fout << "%YAML 1.2" << std::endl << "---" << std::endl;
  342. fout << "---" << std::endl;
  343. BUGZ_LOG(fatal) << "YAML: " << filename;
  344. int depth = 0;
  345. std::string depth_spacer;
  346. meta["save_to"] = filename;
  347. meta["save_time"] = time_t_now(); // time_t
  348. /* // testing sequence code
  349. meta["sequence"]["part"].push_back(1);
  350. meta["sequence"]["part"].push_back(2);
  351. meta["sequence"]["part"].push_back(3);
  352. meta["sequence"]["smeg"].push_back(0);
  353. */
  354. // meta:
  355. fout << "meta:" << std::endl;
  356. ++depth;
  357. depth_spacer.assign(depth * 2, ' ');
  358. std::function<void(std::ofstream & of, int depth, const YAML::Node &n)>
  359. yaml_out;
  360. yaml_out = [&yaml_out](std::ofstream &of, int yaml_depth,
  361. const YAML::Node &n) {
  362. std::string yaml_spacer;
  363. yaml_spacer.assign(yaml_depth * 2, ' ');
  364. for (auto const &data : n) {
  365. std::string key = data.first.as<std::string>();
  366. if (key[0] == '_') {
  367. BUGZ_LOG(fatal) << "Skip key: " << key;
  368. continue;
  369. }
  370. if (data.second.Type() == YAML::NodeType::Scalar) {
  371. of << yaml_spacer << data.first << ": " << data.second << std::endl;
  372. } else if (data.second.Type() == YAML::NodeType::Map) {
  373. // nested
  374. of << yaml_spacer << data.first << ":" << std::endl;
  375. yaml_out(of, yaml_depth + 1, data.second);
  376. } else if (data.second.Type() == YAML::NodeType::Sequence) {
  377. // sequence
  378. BUGZ_LOG(fatal) << "Outputting Sequence... " << data.first;
  379. of << yaml_spacer << data.first << ": [";
  380. bool first = true;
  381. for (auto const &seq : data.second) {
  382. if (!first)
  383. of << ", ";
  384. else
  385. first = false;
  386. of << seq;
  387. }
  388. of << "]" << std::endl;
  389. } else {
  390. BUGZ_LOG(fatal) << "Unsupported NodeType: " << data.second.Type();
  391. }
  392. }
  393. };
  394. BUGZ_LOG(trace) << "YAML meta: " << meta.size();
  395. yaml_out(fout, depth, meta);
  396. BUGZ_LOG(trace) << "YAML config: " << config.size();
  397. fout << "config:" << std::endl;
  398. // in config, I usually switch to doing flow instead. I'll keep this like
  399. // this for now.
  400. yaml_out(fout, depth, config);
  401. BUGZ_LOG(trace) << "YAML warps: " << warps.size();
  402. fout << "warps:" << std::endl;
  403. for (auto const &warp : warps) {
  404. fout << depth_spacer << warp.first << ": [";
  405. bool first = true;
  406. for (auto const &sector : warp.second.warps) {
  407. if (!first) {
  408. fout << ", ";
  409. } else
  410. first = false;
  411. fout << sector;
  412. }
  413. fout << "]" << std::endl;
  414. }
  415. BUGZ_LOG(trace) << "YAML ports: " << ports.size();
  416. fout << "ports:" << std::endl;
  417. /*
  418. When saving to yaml, my sector_type is like char. So, it wants
  419. to save the values as a character. Cast to int.
  420. */
  421. for (auto const &port : ports) {
  422. fout << depth_spacer << port.second.sector
  423. << ": {class: " << (int)port.second.type;
  424. if (port.second.type == 0) {
  425. fout << "}" << std::endl;
  426. } else {
  427. // write out the rest of the information
  428. fout << ", amount: [";
  429. for (int x = 0; x < 3; x++) {
  430. fout << port.second.amount[x];
  431. if (x != 2) fout << ", ";
  432. }
  433. fout << "], pct: [";
  434. for (int x = 0; x < 3; x++) {
  435. fout << (int)port.second.percent[x];
  436. if (x != 2) fout << ", ";
  437. }
  438. fout << "]}" << std::endl;
  439. }
  440. }
  441. // the big data structures later on are the ones I really need to optimize
  442. // here. Writing out 20 lines isn't what is killing us!
  443. /*
  444. for (auto const & meta_data : meta ) {
  445. std::cerr << "key: " << meta_data.first << std::endl;
  446. if (meta_data.second.Type() == YAML::NodeType::Scalar) {
  447. fout << depth_spacer << meta_data.first << ": " << meta_data.second <<
  448. std::endl; } else {
  449. // nested structure
  450. std::cerr << "Nested: " << meta_data.first << std::endl;
  451. fout << depth_spacer << meta_data.first << ":" << std::endl;
  452. ++depth;
  453. depth_spacer.assign(depth *2, ' ');
  454. for (auto const &nested : meta_data.second) {
  455. if (nested.second.Type() == YAML::NodeType::Scalar) {
  456. fout << depth_spacer << nested.first << ": " << nested.second <<
  457. std::endl; } else { std::cerr << "double-Nested meta structure under key: " <<
  458. nested.first << std::endl;
  459. }
  460. }
  461. --depth;
  462. depth_spacer.assign(depth * 2, ' ');
  463. }
  464. }
  465. */
  466. }
  467. std::vector<port_pair_type> Galaxy::find_trades(sector_type sector,
  468. bool highest) {
  469. std::vector<port_pair_type> pptv;
  470. // Does this sector have a port?
  471. auto port = ports.find(sector);
  472. if (port == ports.end()) return pptv;
  473. auto port_warps = warps.find(sector);
  474. // warps not found for port sector?
  475. if (port_warps == warps.end()) return pptv;
  476. for (auto const &s : port_warps->second.warps) {
  477. // only count the ports > our sector number -- so we don't have dups
  478. if (highest && (s < sector)) continue;
  479. // verify we have a way back
  480. {
  481. auto warpback = warps.find(s);
  482. // can we find that warp?
  483. if (warpback == warps.end()) continue;
  484. // does it link back to the port's sector?
  485. if (warpback->second.warps.find(sector) == warpback->second.warps.end())
  486. continue;
  487. }
  488. // Does this sector have a port?
  489. auto possible_port = ports.find(s);
  490. if (possible_port == ports.end()) continue;
  491. if (possible_port->second.type == 0) continue;
  492. // calculate trade type
  493. // int t = trade_type(port->second.type, possible_port->second.type);
  494. BUGZ_LOG(trace) << "find_trades: Port " << sector << ","
  495. << (int)port->second.type << " " << s
  496. << (int)possible_port->second.type;
  497. trade_type_result ttr = trade_type_info(
  498. sector, s); // port->second.type, possible_port->second.type);
  499. if ((ttr.type == NONE) || (ttr.type == FAIR_F)) continue;
  500. pptv.push_back(port_pair_type{ttr.type, ttr.trades, sector, s});
  501. BUGZ_LOG(trace) << "sector: " << sector << " and " << s
  502. << " tt:" << ttr.type;
  503. }
  504. return pptv;
  505. }
  506. bool compare_port_pair(const port_pair_type &ppt1, const port_pair_type &ppt2) {
  507. if (ppt1.type == ppt2.type) {
  508. return ppt1.s1 < ppt2.s1;
  509. } else {
  510. return (ppt1.type < ppt2.type);
  511. }
  512. }
  513. void Galaxy::sort_port_pair_type(std::vector<port_pair_type> &pptv) {
  514. sort(pptv.begin(), pptv.end(), compare_port_pair);
  515. }
  516. std::vector<port_pair_type> Galaxy::find_best_trades(void) {
  517. std::vector<port_pair_type> pptv;
  518. burnt_percent = config["burnt_percent"].as<int>();
  519. if (burnt_percent > 90) {
  520. burnt_percent = 90;
  521. config["burnt_percent"] = 90;
  522. }
  523. if (burnt_percent < 10) {
  524. burnt_percent = 10;
  525. config["burnt_percent"] = 10;
  526. }
  527. for (auto const &pi : ports) {
  528. if (pi.second.type == 0) continue;
  529. sector_type sector = pi.second.sector;
  530. auto port_warps = warps.find(sector);
  531. if (port_warps == warps.end()) continue;
  532. std::vector<port_pair_type> ppt_sector = find_trades(sector, true);
  533. if (ppt_sector.empty()) continue;
  534. pptv.insert(pptv.end(), ppt_sector.begin(), ppt_sector.end());
  535. }
  536. return pptv;
  537. }
  538. /**
  539. * Find_closest trade
  540. *
  541. * This works by getting all of the ports we know of.
  542. * Then, we find the nearest.
  543. *
  544. * @param sector
  545. * @return port_pair_type
  546. */
  547. port_pair_type Galaxy::find_closest(int sector) {
  548. auto trades = find_best_trades();
  549. // int type, sector_type s1, s2;
  550. std::set<sector_type> seen;
  551. std::set<sector_type> current;
  552. current.insert(sector);
  553. bool found = false;
  554. bool added_new = false;
  555. int depth = 0;
  556. while (!found) {
  557. // is current list in any of the trades ?
  558. for (auto const &t : trades) {
  559. if (t.type > 2) continue;
  560. if (current.find(t.s1) != current.end()) {
  561. // found one!
  562. return t;
  563. }
  564. if (current.find(t.s2) != current.end()) {
  565. return t;
  566. }
  567. }
  568. ++depth;
  569. BUGZ_LOG(info) << "depth: " << depth << " current:" << current.size()
  570. << " seen: " << seen.size();
  571. added_new = false;
  572. if (!found) {
  573. // update the seen
  574. seen.insert(current.begin(), current.end());
  575. auto look_in = current;
  576. current.clear();
  577. for (auto const &li : look_in) {
  578. auto wi = warps.find(li);
  579. if (wi == warps.end()) continue;
  580. for (auto const &w : wi->second.warps) {
  581. // have we already seen this sector?
  582. if (seen.find(w) != seen.end()) continue;
  583. // does it have a warp back to the original sector?
  584. auto warp_back = warps.find(w);
  585. if (warp_back != warps.end()) {
  586. if (warp_back->second.warps.find(li) !=
  587. warp_back->second.warps.end()) {
  588. // Ok, this links back to the original sector...
  589. current.insert(w);
  590. added_new = true;
  591. }
  592. }
  593. }
  594. }
  595. if (!added_new) {
  596. BUGZ_LOG(warning) << "No new sectors added. We're done!";
  597. port_pair_type ppt;
  598. ppt.type = 0;
  599. return ppt;
  600. }
  601. }
  602. }
  603. port_pair_type ppt;
  604. ppt.type = 0;
  605. return ppt;
  606. }
  607. /**
  608. * Find closest and best trade
  609. *
  610. * @param sector
  611. * @param lowest_trade_type
  612. * @return port_pair_type
  613. */
  614. port_pair_type Galaxy::find_closest_trade(int sector, int lowest_trade_type,
  615. int burnt_percent) {
  616. // int type, sector_type s1, s2;
  617. BUGZ_LOG(fatal) << "find_closest_trade(" << sector << ")";
  618. std::vector<port_pair_type> vppt;
  619. std::set<sector_type> seen;
  620. std::set<sector_type> current;
  621. current.insert(sector);
  622. bool found = false;
  623. bool added_new = false;
  624. int depth = 0;
  625. while (!found) {
  626. /*
  627. for (auto const &t : trades) {
  628. if (t.type > 2) continue;
  629. if (current.find(t.s1) != current.end()) {
  630. // found one!
  631. return t;
  632. }
  633. if (current.find(t.s2) != current.end()) {
  634. return t;
  635. }
  636. }
  637. */
  638. ++depth;
  639. BUGZ_LOG(info) << "depth: " << depth << " current:" << current.size()
  640. << " seen: " << seen.size();
  641. // search current for trades
  642. for (auto const &c : current) {
  643. auto port = ports.find(c);
  644. if (port == ports.end()) continue;
  645. if (port->second.type == 0) continue;
  646. auto port_warps = warps.find(c);
  647. if (port_warps == warps.end()) continue;
  648. for (auto const &s : port_warps->second.warps) {
  649. // verify we have a way back
  650. {
  651. auto warpback = warps.find(s);
  652. if (warpback == warps.end()) continue;
  653. if (warpback->second.warps.find(c) == warpback->second.warps.end())
  654. continue;
  655. }
  656. auto possible_port = ports.find(s);
  657. if (possible_port == ports.end()) continue;
  658. if (possible_port->second.type == 0) continue;
  659. trade_type_result ttr = trade_type_info(c, s, burnt_percent);
  660. if ((ttr.type == NONE) || (ttr.type > lowest_trade_type)) continue;
  661. // Ok! we found a trade that fits the criteria!
  662. vppt.push_back(port_pair_type{ttr.type, ttr.trades, c, s});
  663. found = true;
  664. }
  665. }
  666. added_new = false;
  667. if (found) {
  668. // ok, we've found some trades, sort and return the best
  669. sort_port_pair_type(vppt);
  670. return port_pair_type{vppt[0]};
  671. } else {
  672. // update the seen
  673. seen.insert(current.begin(), current.end());
  674. auto look_in = current;
  675. current.clear();
  676. for (auto const &li : look_in) {
  677. auto wi = warps.find(li);
  678. if (wi == warps.end()) continue;
  679. for (auto const &w : wi->second.warps) {
  680. // have we already seen this sector?
  681. if (seen.find(w) != seen.end()) continue;
  682. // does it have a warp back to the original sector?
  683. auto warp_back = warps.find(w);
  684. if (warp_back != warps.end()) {
  685. if (warp_back->second.warps.find(li) !=
  686. warp_back->second.warps.end()) {
  687. // Ok, this links back to the original sector...
  688. current.insert(w);
  689. added_new = true;
  690. }
  691. }
  692. }
  693. }
  694. if (!added_new) {
  695. BUGZ_LOG(warning) << "No new sectors added. We're done!";
  696. port_pair_type ppt;
  697. ppt.type = 0;
  698. return ppt;
  699. }
  700. }
  701. }
  702. port_pair_type ppt;
  703. ppt.type = 0;
  704. return ppt;
  705. }
  706. sector_type Galaxy::find_nearest_unexplored(sector_type sector) {
  707. // search the galaxy for unexplored
  708. std::set<sector_type> seen;
  709. std::set<sector_type> current;
  710. current.insert(sector);
  711. bool found = false;
  712. bool added_new = false;
  713. int depth = 0;
  714. while (!found) {
  715. ++depth;
  716. BUGZ_LOG(info) << "depth: " << depth << " current:" << current.size()
  717. << " seen: " << seen.size();
  718. // search for warps
  719. for (auto const &c : current) {
  720. auto port_warps = warps.find(c);
  721. if (port_warps == warps.end()) return c;
  722. }
  723. // update the seen
  724. seen.insert(current.begin(), current.end());
  725. auto look_in = current;
  726. current.clear();
  727. for (auto const &li : look_in) {
  728. auto wi = warps.find(li);
  729. if (wi == warps.end()) return li;
  730. for (auto const &w : wi->second.warps) {
  731. // have we already seen this sector?
  732. if (seen.find(w) != seen.end()) continue;
  733. current.insert(w);
  734. added_new = true;
  735. }
  736. }
  737. if (!added_new) {
  738. BUGZ_LOG(warning) << "No new sectors added. We're done!";
  739. found = false;
  740. }
  741. }
  742. return 0;
  743. }
  744. trade_type_result Galaxy::trade_type_info(sector_type port1, sector_type port2,
  745. int burnt_percent) {
  746. BUGZ_LOG(fatal) << "Trade_type_info(" << port1 << "," << port2 << ")";
  747. // This only gives us one trade_type per pair. There actually
  748. // should be multiple values returned here!
  749. // Like in case of BBB/SSS: return 3, 4 and 5.
  750. // NONE = 0
  751. // GOOD = 1 = OE PAIR
  752. // OK = 2 = ?? Pair
  753. // FAIR = 3 = B/S E
  754. // 4 = B/S O
  755. // 5 = B/S F
  756. buysell inv2;
  757. auto p1 = ports.find(port1);
  758. if (p1 == ports.end()) {
  759. BUGZ_LOG(fatal) << "Can't find port 1: " << (int)port1;
  760. return {NONE, inv2};
  761. }
  762. BUGZ_LOG(fatal) << "port 1: " << p1->first << " " << p1->second.sector << ", "
  763. << (int)p1->second.type;
  764. auto p2 = ports.find(port2);
  765. if (p2 == ports.end()) {
  766. BUGZ_LOG(fatal) << "Can't find port 2: " << (int)port2;
  767. return {NONE, inv2};
  768. }
  769. BUGZ_LOG(fatal) << "port 2: " << p2->first << " " << p2->second.sector << ", "
  770. << (int)p2->second.type;
  771. buysell bsp1 = get_buysell(p1->second.type);
  772. buysell bsp2 = get_buysell(p2->second.type);
  773. inv2 = invert_buysell(bsp2);
  774. int matches = 0; // or pos.size();
  775. std::vector<int> pos;
  776. // If we don't know how many holds the ship has, default to 300.
  777. int max_holds = 300;
  778. if (meta["ship"]["holds"]["total"]) {
  779. max_holds = meta["ship"]["holds"]["total"].as<int>();
  780. }
  781. // find which FOE are flipped. Save index pos.
  782. for (int x = 0; x < 3; ++x) {
  783. inv2.foe[x] = (bsp1.foe[x] == inv2.foe[x]);
  784. // Ok, these are possible trades (B->S or S->B)
  785. // If known, check for burnt
  786. if (!p1->second.unknown()) {
  787. if (p1->second.percent[x] < burnt_percent) {
  788. BUGZ_LOG(fatal) << "Marking Port 1: " << x << " (burnt)";
  789. inv2.foe[x] = false;
  790. }
  791. if (p1->second.amount[x] < max_holds) {
  792. BUGZ_LOG(fatal) << "Marking Port 1: " << x << " (burnt/amount)";
  793. inv2.foe[x] = false;
  794. }
  795. } else {
  796. BUGZ_LOG(fatal) << "Port 1 : unknown / skip burnt checks";
  797. }
  798. if (!p2->second.unknown()) {
  799. if (p2->second.percent[x] < burnt_percent) {
  800. BUGZ_LOG(fatal) << "Marking Port 2: " << x << " (burnt)";
  801. inv2.foe[x] = false;
  802. }
  803. if (p2->second.amount[x] < max_holds) {
  804. BUGZ_LOG(fatal) << "Marking Port 2: " << x << " (burnt/amount)";
  805. inv2.foe[x] = false;
  806. }
  807. } else {
  808. BUGZ_LOG(fatal) << "Port 2 : unknown / skip burnt checks";
  809. }
  810. if (inv2.foe[x]) {
  811. matches++;
  812. pos.push_back(x);
  813. }
  814. }
  815. BUGZ_LOG(fatal) << "Matches: " << matches << " inv: " << inv2;
  816. if (matches > 1) {
  817. // Check for BEST
  818. // O != E for both ports, and O != O, and ORG/EQU in inv2 list
  819. if (inv2.foe[ORG] && inv2.foe[EQU] && (bsp1.foe[ORG] != bsp1.foe[EQU]) &&
  820. (bsp2.foe[ORG] != bsp2.foe[EQU]) && (bsp1.foe[ORG] != bsp2.foe[ORG])) {
  821. // verify that fuel isn't set.
  822. inv2.foe[FUEL] = false;
  823. BUGZ_LOG(fatal) << "result: " << BEST << " " << inv2;
  824. return trade_type_result{BEST, inv2};
  825. }
  826. if (matches == 3) {
  827. // This could be SBB / BSS (it's a pair, but not BEST)
  828. // Is it FO or FE ?
  829. if (bsp1.foe[FUEL] != bsp2.foe[EQU]) {
  830. // OK, FE
  831. inv2.foe[ORG] = false;
  832. BUGZ_LOG(fatal) << "result: " << OK << " " << inv2;
  833. return trade_type_result{OK, inv2};
  834. }
  835. if (bsp1.foe[FUEL] != bsp2.foe[ORG]) {
  836. // OK, FO
  837. inv2.foe[EQU] = false;
  838. BUGZ_LOG(fatal) << "result: " << OK << " " << inv2;
  839. return trade_type_result{OK, inv2};
  840. }
  841. // Ok, take the highest (EQU)
  842. inv2.foe[FUEL] = false;
  843. inv2.foe[ORG] = false;
  844. BUGZ_LOG(fatal) << "result: " << FAIR_E << " " << inv2;
  845. return trade_type_result{FAIR_E, inv2};
  846. }
  847. // 2 matches. but are they trade pairs?
  848. if (bsp1.foe[pos[matches - 1]] != bsp1.foe[pos[matches - 2]]) {
  849. // yes!
  850. BUGZ_LOG(fatal) << "result: " << OK << " " << inv2;
  851. return trade_type_result{OK, inv2};
  852. } else {
  853. // they are NOT. Use highest one. clear the lower flag
  854. inv2.foe[pos[0]] = false;
  855. switch (pos[1]) {
  856. case 0:
  857. BUGZ_LOG(fatal) << "result: " << FAIR_F << " " << inv2;
  858. return trade_type_result{FAIR_F, inv2};
  859. break;
  860. case 1:
  861. BUGZ_LOG(fatal) << "result: " << FAIR_O << " " << inv2;
  862. return trade_type_result{FAIR_O, inv2};
  863. break;
  864. case 2:
  865. BUGZ_LOG(fatal) << "result: " << FAIR_E << " " << inv2;
  866. return trade_type_result{FAIR_E, inv2};
  867. break;
  868. }
  869. }
  870. }
  871. if (matches == 1) {
  872. switch (pos[0]) {
  873. case 0:
  874. BUGZ_LOG(fatal) << "result: " << FAIR_F << " " << inv2;
  875. return trade_type_result{FAIR_F, inv2};
  876. break;
  877. case 1:
  878. BUGZ_LOG(fatal) << "result: " << FAIR_O << " " << inv2;
  879. return trade_type_result{FAIR_O, inv2};
  880. break;
  881. case 2:
  882. BUGZ_LOG(fatal) << "result: " << FAIR_E << " " << inv2;
  883. return trade_type_result{FAIR_E, inv2};
  884. break;
  885. }
  886. }
  887. // no matches.
  888. BUGZ_LOG(fatal) << "result: " << NONE << " " << inv2;
  889. return trade_type_result{NONE, inv2};
  890. }