galaxy.cpp 28 KB

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