galaxy.cpp 29 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043
  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 "json.hpp"
  12. #include "logging.h"
  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. if (data.contains("meta")) {
  273. meta = data["meta"];
  274. } else {
  275. BUGZ_LOG(fatal) << "Missing meta data.";
  276. }
  277. meta["load_from"] = filename;
  278. meta["load_time"] = time_t_now(); // time_t
  279. if (data.contains("config")) {
  280. config = data["config"];
  281. } else {
  282. BUGZ_LOG(fatal) << "Missing config section.";
  283. }
  284. if (data.contains("ports")) {
  285. // const json &ports = data["ports"];
  286. for (auto const &port_iter : data["ports"].items()) {
  287. port p;
  288. p.sector = sstoi(port_iter.key()); // first.as<int>();
  289. p.type = port_iter.value()["class"].get<int>();
  290. int x;
  291. if (p.type == 0) {
  292. // nothing to load for type = 0, set defaults.
  293. for (x = 0; x < 3; ++x) {
  294. p.amount[x] = 0;
  295. p.percent[x] = 0;
  296. }
  297. } else {
  298. x = 0;
  299. for (auto const &amount : port_iter.value()["amount"].items()) {
  300. p.amount[x] = amount.value().get<int>();
  301. ++x;
  302. }
  303. x = 0;
  304. for (auto const &pct : port_iter.value()["pct"].items()) {
  305. p.percent[x] = pct.value().get<int>();
  306. ++x;
  307. }
  308. }
  309. add_port(p);
  310. }
  311. } else {
  312. BUGZ_LOG(fatal) << "Missing ports section.";
  313. }
  314. if (data.contains("warps")) {
  315. // const json &warps = data["warps"];
  316. // if (warps.IsMap()) {
  317. for (auto const warp_iter : data["warps"].items()) {
  318. sector_warps sw;
  319. sw.sector = sstoi(warp_iter.key());
  320. for (auto const sector_iter : warp_iter.value().items()) {
  321. sw.add(sector_iter.value().get<int>());
  322. }
  323. // BUGZ_LOG(trace) << "YAML warp: " << sw;
  324. add_warp(sw);
  325. }
  326. // }
  327. } else {
  328. BUGZ_LOG(fatal) << "Missing warps section.";
  329. }
  330. BUGZ_LOG(trace) << "JSON: meta keys: " << meta.size();
  331. BUGZ_LOG(trace) << "JSON: config keys: " << config.size();
  332. BUGZ_LOG(trace) << "JSON: warp keys: " << warps.size();
  333. BUGZ_LOG(trace) << "JSON: port keys: " << ports.size();
  334. } else {
  335. BUGZ_LOG(fatal) << "Missing: " << filename;
  336. }
  337. }
  338. void Galaxy::save(void) {
  339. std::string basename = "galaxy";
  340. if (CONFIG.contains("basename")) {
  341. basename = CONFIG["basename"];
  342. }
  343. std::string filename =
  344. str(boost::format("%1%-%2%-%3%.json") % basename % game % username);
  345. std::ofstream fout(filename);
  346. BUGZ_LOG(fatal) << "save: " << filename;
  347. // int depth = 0;
  348. // std::string depth_spacer;
  349. meta["save_to"] = filename;
  350. meta["save_time"] = time_t_now(); // time_t
  351. // For now, we'll delete this manually.
  352. // FUTURE: delete all keys starting with _.
  353. if (meta.contains("density")) {
  354. meta.erase("density");
  355. }
  356. // clean meta data
  357. {
  358. // build vector of things to delete, don't invalidate your iterator. (UB)
  359. std::vector<std::string> to_delete;
  360. for (auto m : meta.items()) {
  361. if (m.key()[0] == '_') {
  362. to_delete.push_back(m.key());
  363. }
  364. }
  365. for (auto const &key : to_delete) {
  366. BUGZ_LOG(fatal) << "erasing meta key: [" << key << "]";
  367. meta.erase(key);
  368. }
  369. }
  370. /* // testing sequence code
  371. meta["sequence"]["part"].push_back(1);
  372. meta["sequence"]["part"].push_back(2);
  373. meta["sequence"]["part"].push_back(3);
  374. meta["sequence"]["smeg"].push_back(0);
  375. */
  376. json output;
  377. output["meta"] = meta;
  378. output["config"] = config;
  379. BUGZ_LOG(trace) << "JSON: meta: " << meta.size();
  380. // yaml_out(fout, depth, meta);
  381. BUGZ_LOG(trace) << "JSON: config: " << config.size();
  382. BUGZ_LOG(trace) << "JSON: warps: " << warps.size();
  383. // in config, I usually switch to doing flow instead. I'll keep this like
  384. // this for now.
  385. // yaml_out(fout, depth, config);
  386. for (auto const &warp : warps) {
  387. // fout << depth_spacer << warp.first << ": [";
  388. // bool first = true;
  389. std::string warp_text = std::to_string(warp.first);
  390. // output["warps"][warp.first] = json::array();
  391. for (auto const &sector : warp.second.warps) {
  392. output["warps"][warp_text].push_back(sector);
  393. /*
  394. if (!first) {
  395. fout << ", ";
  396. } else
  397. first = false;
  398. fout << sector;
  399. */
  400. }
  401. // fout << "]" << std::endl;
  402. }
  403. BUGZ_LOG(trace) << "JSON: ports: " << ports.size();
  404. // fout << "ports:" << std::endl;
  405. for (auto const &port : ports) {
  406. std::string port_text = std::to_string(port.first);
  407. output["ports"][port_text]["class"] = (int)port.second.type;
  408. if (port.second.type != 0) {
  409. // write out the rest of the information
  410. // output["ports"][port.first]["amount"] =
  411. // json::array{port.second.amount[0], port.second.amount[1],
  412. // port.second.aount[2]};
  413. for (int x = 0; x < 3; x++) {
  414. // output["ports"][port.first]["amount"][x] = port.second.amount[x];
  415. output["ports"][port_text]["amount"].push_back(port.second.amount[x]);
  416. }
  417. // output["ports"][port.first]["pct"] =
  418. // json::array{port.second.percent[0], port.second.percent[1],
  419. // port.second.percent[2]};
  420. for (int x = 0; x < 3; x++) {
  421. // output["ports"][port.first]["pct"][x] = port.second.percent[x];
  422. output["ports"][port_text]["pct"].push_back(port.second.percent[x]);
  423. }
  424. }
  425. }
  426. fout << output;
  427. }
  428. std::vector<port_pair_type> Galaxy::find_trades(sector_type sector,
  429. bool highest) {
  430. std::vector<port_pair_type> pptv;
  431. // Does this sector have a port?
  432. auto port = ports.find(sector);
  433. if (port == ports.end()) return pptv;
  434. auto port_warps = warps.find(sector);
  435. // warps not found for port sector?
  436. if (port_warps == warps.end()) return pptv;
  437. for (auto const &s : port_warps->second.warps) {
  438. // only count the ports > our sector number -- so we don't have dups
  439. if (highest && (s < sector)) continue;
  440. // verify we have a way back
  441. {
  442. auto warpback = warps.find(s);
  443. // can we find that warp?
  444. if (warpback == warps.end()) continue;
  445. // does it link back to the port's sector?
  446. if (warpback->second.warps.find(sector) == warpback->second.warps.end())
  447. continue;
  448. }
  449. // Does this sector have a port?
  450. auto possible_port = ports.find(s);
  451. if (possible_port == ports.end()) continue;
  452. if (possible_port->second.type == 0) continue;
  453. // calculate trade type
  454. // int t = trade_type(port->second.type, possible_port->second.type);
  455. BUGZ_LOG(trace) << "find_trades: Port " << sector << ","
  456. << (int)port->second.type << " " << s
  457. << (int)possible_port->second.type;
  458. trade_type_result ttr = trade_type_info(
  459. sector, s); // port->second.type, possible_port->second.type);
  460. if ((ttr.type == NONE) || (ttr.type == FAIR_F)) continue;
  461. pptv.push_back(port_pair_type{ttr.type, ttr.trades, sector, s});
  462. BUGZ_LOG(trace) << "sector: " << sector << " and " << s
  463. << " tt:" << ttr.type;
  464. }
  465. return pptv;
  466. }
  467. bool compare_port_pair(const port_pair_type &ppt1, const port_pair_type &ppt2) {
  468. if (ppt1.type == ppt2.type) {
  469. return ppt1.s1 < ppt2.s1;
  470. } else {
  471. return (ppt1.type < ppt2.type);
  472. }
  473. }
  474. void Galaxy::sort_port_pair_type(std::vector<port_pair_type> &pptv) {
  475. sort(pptv.begin(), pptv.end(), compare_port_pair);
  476. }
  477. std::vector<port_pair_type> Galaxy::find_best_trades(void) {
  478. std::vector<port_pair_type> pptv;
  479. burnt_percent = json_int(config["burnt_percent"]);
  480. if (burnt_percent > 90) {
  481. burnt_percent = 90;
  482. config["burnt_percent"] = 90;
  483. }
  484. if (burnt_percent < 10) {
  485. burnt_percent = 10;
  486. config["burnt_percent"] = 10;
  487. }
  488. for (auto const &pi : ports) {
  489. if (pi.second.type == 0) continue;
  490. sector_type sector = pi.second.sector;
  491. auto port_warps = warps.find(sector);
  492. if (port_warps == warps.end()) continue;
  493. std::vector<port_pair_type> ppt_sector = find_trades(sector, true);
  494. if (ppt_sector.empty()) continue;
  495. pptv.insert(pptv.end(), ppt_sector.begin(), ppt_sector.end());
  496. }
  497. return pptv;
  498. }
  499. /**
  500. * Find_closest trade
  501. *
  502. * This works by getting all of the ports we know of.
  503. * Then, we find the nearest.
  504. *
  505. * @param sector
  506. * @return port_pair_type
  507. */
  508. port_pair_type Galaxy::find_closest(int sector) {
  509. auto trades = find_best_trades();
  510. // int type, sector_type s1, s2;
  511. std::set<sector_type> seen;
  512. std::set<sector_type> current;
  513. current.insert(sector);
  514. bool found = false;
  515. bool added_new = false;
  516. int depth = 0;
  517. while (!found) {
  518. // is current list in any of the trades ?
  519. for (auto const &t : trades) {
  520. if (t.type > 2) continue;
  521. if (current.find(t.s1) != current.end()) {
  522. // found one!
  523. return t;
  524. }
  525. if (current.find(t.s2) != current.end()) {
  526. return t;
  527. }
  528. }
  529. ++depth;
  530. BUGZ_LOG(info) << "depth: " << depth << " current:" << current.size()
  531. << " seen: " << seen.size();
  532. added_new = false;
  533. if (!found) {
  534. // update the seen
  535. seen.insert(current.begin(), current.end());
  536. auto look_in = current;
  537. current.clear();
  538. for (auto const &li : look_in) {
  539. auto wi = warps.find(li);
  540. if (wi == warps.end()) continue;
  541. for (auto const &w : wi->second.warps) {
  542. // have we already seen this sector?
  543. if (seen.find(w) != seen.end()) continue;
  544. // does it have a warp back to the original sector?
  545. auto warp_back = warps.find(w);
  546. if (warp_back != warps.end()) {
  547. if (warp_back->second.warps.find(li) !=
  548. warp_back->second.warps.end()) {
  549. // Ok, this links back to the original sector...
  550. current.insert(w);
  551. added_new = true;
  552. }
  553. }
  554. }
  555. }
  556. if (!added_new) {
  557. BUGZ_LOG(warning) << "No new sectors added. We're done!";
  558. port_pair_type ppt;
  559. ppt.type = 0;
  560. return ppt;
  561. }
  562. }
  563. }
  564. port_pair_type ppt;
  565. ppt.type = 0;
  566. return ppt;
  567. }
  568. /**
  569. * Find closest and best trade
  570. *
  571. * @param sector
  572. * @param lowest_trade_type
  573. * @return port_pair_type
  574. */
  575. port_pair_type Galaxy::find_closest_trade(int sector, int lowest_trade_type,
  576. int burnt_percent) {
  577. // int type, sector_type s1, s2;
  578. BUGZ_LOG(fatal) << "find_closest_trade(" << sector << ")";
  579. std::vector<port_pair_type> vppt;
  580. std::set<sector_type> seen;
  581. std::set<sector_type> current;
  582. current.insert(sector);
  583. bool found = false;
  584. bool added_new = false;
  585. int depth = 0;
  586. while (!found) {
  587. ++depth;
  588. BUGZ_LOG(info) << "depth: " << depth << " current:" << current.size()
  589. << " seen: " << seen.size();
  590. // search current for trades
  591. for (auto const &c : current) {
  592. auto port = ports.find(c);
  593. if (port == ports.end()) continue;
  594. if (port->second.type == 0) continue;
  595. auto port_warps = warps.find(c);
  596. if (port_warps == warps.end()) continue;
  597. for (auto const &s : port_warps->second.warps) {
  598. // verify we have a way back
  599. {
  600. auto warpback = warps.find(s);
  601. if (warpback == warps.end()) continue;
  602. if (warpback->second.warps.find(c) == warpback->second.warps.end())
  603. continue;
  604. }
  605. auto possible_port = ports.find(s);
  606. if (possible_port == ports.end()) continue;
  607. if (possible_port->second.type == 0) continue;
  608. trade_type_result ttr = trade_type_info(c, s, burnt_percent);
  609. if ((ttr.type == NONE) || (ttr.type > lowest_trade_type)) continue;
  610. // Ok! we found a trade that fits the criteria!
  611. vppt.push_back(port_pair_type{ttr.type, ttr.trades, c, s});
  612. found = true;
  613. }
  614. }
  615. added_new = false;
  616. if (found) {
  617. // ok, we've found some trades, sort and return the best
  618. sort_port_pair_type(vppt);
  619. return port_pair_type{vppt[0]};
  620. } else {
  621. // update the seen
  622. seen.insert(current.begin(), current.end());
  623. auto look_in = current;
  624. current.clear();
  625. for (auto const &li : look_in) {
  626. auto wi = warps.find(li);
  627. if (wi == warps.end()) continue;
  628. for (auto const &w : wi->second.warps) {
  629. // have we already seen this sector?
  630. if (seen.find(w) != seen.end()) continue;
  631. // does it have a warp back to the original sector?
  632. auto warp_back = warps.find(w);
  633. if (warp_back != warps.end()) {
  634. if (warp_back->second.warps.find(li) !=
  635. warp_back->second.warps.end()) {
  636. // Ok, this links back to the original sector...
  637. current.insert(w);
  638. added_new = true;
  639. }
  640. }
  641. }
  642. }
  643. if (!added_new) {
  644. BUGZ_LOG(warning) << "No new sectors added. We're done!";
  645. port_pair_type ppt;
  646. ppt.type = 0;
  647. return ppt;
  648. }
  649. }
  650. }
  651. port_pair_type ppt;
  652. ppt.type = 0;
  653. return ppt;
  654. }
  655. sector_type Galaxy::find_nearest_unexplored(sector_type sector) {
  656. // search the galaxy for unexplored
  657. std::set<sector_type> seen;
  658. std::set<sector_type> current;
  659. current.insert(sector);
  660. bool found = false;
  661. bool added_new = false;
  662. int depth = 0;
  663. while (!found) {
  664. ++depth;
  665. BUGZ_LOG(info) << "depth: " << depth << " current:" << current.size()
  666. << " seen: " << seen.size();
  667. // search for warps
  668. for (auto const &c : current) {
  669. auto port_warps = warps.find(c);
  670. if (port_warps == warps.end()) return c;
  671. }
  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()) return li;
  679. for (auto const &w : wi->second.warps) {
  680. // have we already seen this sector?
  681. if (seen.find(w) != seen.end()) continue;
  682. current.insert(w);
  683. added_new = true;
  684. }
  685. }
  686. if (!added_new) {
  687. BUGZ_LOG(warning) << "No new sectors added. We're done!";
  688. found = false;
  689. }
  690. }
  691. return 0;
  692. }
  693. trade_type_result Galaxy::trade_type_info(sector_type port1, sector_type port2,
  694. int burnt_percent) {
  695. BUGZ_LOG(fatal) << "Trade_type_info(" << port1 << "," << port2 << ")";
  696. // This only gives us one trade_type per pair. There actually
  697. // should be multiple values returned here!
  698. // Like in case of BBB/SSS: return 3, 4 and 5.
  699. // NONE = 0
  700. // GOOD = 1 = OE PAIR
  701. // OK = 2 = ?? Pair
  702. // FAIR = 3 = B/S E
  703. // 4 = B/S O
  704. // 5 = B/S F
  705. buysell inv2;
  706. auto p1 = ports.find(port1);
  707. if (p1 == ports.end()) {
  708. BUGZ_LOG(fatal) << "Can't find port 1: " << (int)port1;
  709. return {NONE, inv2};
  710. }
  711. BUGZ_LOG(fatal) << "port 1: " << p1->first << " " << p1->second.sector << ", "
  712. << (int)p1->second.type;
  713. auto p2 = ports.find(port2);
  714. if (p2 == ports.end()) {
  715. BUGZ_LOG(fatal) << "Can't find port 2: " << (int)port2;
  716. return {NONE, inv2};
  717. }
  718. BUGZ_LOG(fatal) << "port 2: " << p2->first << " " << p2->second.sector << ", "
  719. << (int)p2->second.type;
  720. buysell bsp1 = get_buysell(p1->second.type);
  721. buysell bsp2 = get_buysell(p2->second.type);
  722. inv2 = invert_buysell(bsp2);
  723. int matches = 0; // or pos.size();
  724. std::vector<int> pos;
  725. // If we don't know how many holds the ship has, default to 300.
  726. int max_holds = 300;
  727. if (meta["ship"]["holds"].contains("total")) {
  728. max_holds = json_int(meta["ship"]["holds"]["total"]);
  729. }
  730. // find which FOE are flipped. Save index pos.
  731. for (int x = 0; x < 3; ++x) {
  732. inv2.foe[x] = (bsp1.foe[x] == inv2.foe[x]);
  733. // Ok, these are possible trades (B->S or S->B)
  734. // If known, check for burnt
  735. if (!p1->second.unknown()) {
  736. if (p1->second.percent[x] < burnt_percent) {
  737. BUGZ_LOG(fatal) << "Marking Port 1: " << x << " (burnt)";
  738. inv2.foe[x] = false;
  739. }
  740. if (p1->second.amount[x] < max_holds) {
  741. BUGZ_LOG(fatal) << "Marking Port 1: " << x << " (burnt/amount)";
  742. inv2.foe[x] = false;
  743. }
  744. } else {
  745. BUGZ_LOG(fatal) << "Port 1 : unknown / skip burnt checks";
  746. }
  747. if (!p2->second.unknown()) {
  748. if (p2->second.percent[x] < burnt_percent) {
  749. BUGZ_LOG(fatal) << "Marking Port 2: " << x << " (burnt)";
  750. inv2.foe[x] = false;
  751. }
  752. if (p2->second.amount[x] < max_holds) {
  753. BUGZ_LOG(fatal) << "Marking Port 2: " << x << " (burnt/amount)";
  754. inv2.foe[x] = false;
  755. }
  756. } else {
  757. BUGZ_LOG(fatal) << "Port 2 : unknown / skip burnt checks";
  758. }
  759. if (inv2.foe[x]) {
  760. matches++;
  761. pos.push_back(x);
  762. }
  763. }
  764. BUGZ_LOG(fatal) << "Matches: " << matches << " inv: " << inv2;
  765. if (matches > 1) {
  766. // Check for BEST
  767. // O != E for both ports, and O != O, and ORG/EQU in inv2 list
  768. if (inv2.foe[ORG] && inv2.foe[EQU] && (bsp1.foe[ORG] != bsp1.foe[EQU]) &&
  769. (bsp2.foe[ORG] != bsp2.foe[EQU]) && (bsp1.foe[ORG] != bsp2.foe[ORG])) {
  770. // verify that fuel isn't set.
  771. inv2.foe[FUEL] = false;
  772. BUGZ_LOG(fatal) << "result: " << BEST << " " << inv2;
  773. return trade_type_result{BEST, inv2};
  774. }
  775. if (matches == 3) {
  776. // This could be SBB / BSS (it's a pair, but not BEST)
  777. // Is it FO or FE ?
  778. if (bsp1.foe[FUEL] != bsp2.foe[EQU]) {
  779. // OK, FE
  780. inv2.foe[ORG] = false;
  781. BUGZ_LOG(fatal) << "result: " << OK << " " << inv2;
  782. return trade_type_result{OK, inv2};
  783. }
  784. if (bsp1.foe[FUEL] != bsp2.foe[ORG]) {
  785. // OK, FO
  786. inv2.foe[EQU] = false;
  787. BUGZ_LOG(fatal) << "result: " << OK << " " << inv2;
  788. return trade_type_result{OK, inv2};
  789. }
  790. // Ok, take the highest (EQU)
  791. inv2.foe[FUEL] = false;
  792. inv2.foe[ORG] = false;
  793. BUGZ_LOG(fatal) << "result: " << FAIR_E << " " << inv2;
  794. return trade_type_result{FAIR_E, inv2};
  795. }
  796. // 2 matches. but are they trade pairs?
  797. if (bsp1.foe[pos[matches - 1]] != bsp1.foe[pos[matches - 2]]) {
  798. // yes!
  799. BUGZ_LOG(fatal) << "result: " << OK << " " << inv2;
  800. return trade_type_result{OK, inv2};
  801. } else {
  802. // they are NOT. Use highest one. clear the lower flag
  803. inv2.foe[pos[0]] = false;
  804. switch (pos[1]) {
  805. case 0:
  806. BUGZ_LOG(fatal) << "result: " << FAIR_F << " " << inv2;
  807. return trade_type_result{FAIR_F, inv2};
  808. break;
  809. case 1:
  810. BUGZ_LOG(fatal) << "result: " << FAIR_O << " " << inv2;
  811. return trade_type_result{FAIR_O, inv2};
  812. break;
  813. case 2:
  814. BUGZ_LOG(fatal) << "result: " << FAIR_E << " " << inv2;
  815. return trade_type_result{FAIR_E, inv2};
  816. break;
  817. }
  818. }
  819. }
  820. if (matches == 1) {
  821. switch (pos[0]) {
  822. case 0:
  823. BUGZ_LOG(fatal) << "result: " << FAIR_F << " " << inv2;
  824. return trade_type_result{FAIR_F, inv2};
  825. break;
  826. case 1:
  827. BUGZ_LOG(fatal) << "result: " << FAIR_O << " " << inv2;
  828. return trade_type_result{FAIR_O, inv2};
  829. break;
  830. case 2:
  831. BUGZ_LOG(fatal) << "result: " << FAIR_E << " " << inv2;
  832. return trade_type_result{FAIR_E, inv2};
  833. break;
  834. }
  835. }
  836. // no matches.
  837. BUGZ_LOG(fatal) << "result: " << NONE << " " << inv2;
  838. return trade_type_result{NONE, inv2};
  839. }
  840. sector_type Galaxy::find_nearest_selling(int sector, int product, int selling) {
  841. BUGZ_LOG(fatal) << "find_nearest_selling(" << sector << ") " << product;
  842. std::set<sector_type> seen;
  843. std::set<sector_type> current;
  844. current.insert(sector);
  845. bool found = false;
  846. bool added_new = false;
  847. int depth = 0;
  848. // check current sector
  849. auto port = ports.find(sector);
  850. if (port != ports.end()) {
  851. // ok, port exists in starting sector
  852. // are they selling?
  853. int t = port->second.type;
  854. if (t != 0) {
  855. buysell bs = get_buysell(t);
  856. if (bs.foe[product] == false) {
  857. if (port->second.amount[product] >= selling) {
  858. BUGZ_LOG(fatal) << "Found port: " << port->first;
  859. return port->first;
  860. }
  861. }
  862. }
  863. }
  864. while (!found) {
  865. ++depth;
  866. BUGZ_LOG(info) << "depth: " << depth << " current:" << current.size()
  867. << " seen: " << seen.size();
  868. // search current for trades
  869. for (auto const &c : current) {
  870. auto port_warps = warps.find(c);
  871. if (port_warps == warps.end()) continue;
  872. for (auto const &s : port_warps->second.warps) {
  873. // I don't care if there's a way back the way we came
  874. auto possible_port = ports.find(s);
  875. if (possible_port == ports.end()) continue;
  876. if (possible_port->second.type == 0) continue;
  877. // are they selling?
  878. int t = possible_port->second.type;
  879. buysell bs = get_buysell(t);
  880. if (bs.foe[product] == true) continue;
  881. // check this port out
  882. if (possible_port->second.amount[product] >= selling) {
  883. BUGZ_LOG(fatal) << "Found port: " << possible_port->first;
  884. return possible_port->first;
  885. }
  886. }
  887. }
  888. added_new = false;
  889. // update the seen
  890. seen.insert(current.begin(), current.end());
  891. auto look_in = current;
  892. current.clear();
  893. for (auto const &li : look_in) {
  894. auto wi = warps.find(li);
  895. if (wi == warps.end()) continue;
  896. for (auto const &w : wi->second.warps) {
  897. // have we already seen this sector?
  898. if (seen.find(w) != seen.end()) continue;
  899. // I don't care if it has a warp back or not.
  900. current.insert(w);
  901. added_new = true;
  902. }
  903. }
  904. if (!added_new) {
  905. BUGZ_LOG(warning) << "No new sectors added. We're done!";
  906. return 0;
  907. }
  908. }
  909. return 0;
  910. };
  911. std::vector<sector_type> Galaxy::find_safe(void) {
  912. std::vector<sector_type> vst;
  913. for( auto const & warp : warps ) {
  914. if (warp.second.warps.size() == 1) {
  915. vst.push_back(warp.first);
  916. }
  917. }
  918. return vst;
  919. }