galaxy.cpp 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535
  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 "logging.h"
  11. #include "yaml-cpp/yaml.h"
  12. // c++ default exceptions list
  13. // https://en.cppreference.com/w/cpp/error/exception
  14. std::ostream &operator<<(std::ostream &os, const port &p) {
  15. if (p.type == 0) {
  16. os << p.sector << ": " << (int)p.type;
  17. } else {
  18. os << p.sector << ": " << (int)p.type << " " << text_from_type(p.type)
  19. << " " << p.amount[0] << "," << p.amount[1] << "," << p.amount[2];
  20. }
  21. return os;
  22. }
  23. bool port::unknown(void) {
  24. for (int x = 0; x < 3; ++x) {
  25. if (percent[x] != 0) return false;
  26. }
  27. return true;
  28. }
  29. trade_type_result trade_type_info(port_type port1, port_type port2) {
  30. // NONE = 0
  31. // GOOD = 1 = OE PAIR
  32. // OK = 2 = ?? Pair
  33. // FAIR = 3 = B / S
  34. buysell p1 = get_buysell(port1);
  35. buysell p2 = get_buysell(port2);
  36. buysell inv2 = invert_buysell(p2);
  37. int matches = 0; // or pos.size();
  38. std::vector<int> pos;
  39. // find which FOE are flipped. Save index pos.
  40. for (int x = 0; x < 3; ++x) {
  41. inv2.foe[x] = (p1.foe[x] == inv2.foe[x]);
  42. if (inv2.foe[x]) {
  43. matches++;
  44. pos.push_back(x);
  45. }
  46. }
  47. if (matches > 1) {
  48. // O != E for both ports, and O != O
  49. if ((p1.foe[ORG] != p1.foe[EQU]) && (p2.foe[ORG] != p2.foe[EQU]) &&
  50. (p1.foe[ORG] != p2.foe[ORG])) {
  51. return trade_type_result{1, inv2};
  52. }
  53. // at least 2 matches. but are they trade pairs?
  54. // I can tell by comparing the last two positions in the same port.
  55. if (p1.foe[pos[matches - 1]] == p1.foe[pos[matches - 2]]) {
  56. // they are NOT.
  57. return trade_type_result{3, inv2};
  58. }
  59. return trade_type_result{2, inv2};
  60. }
  61. if (matches == 1) {
  62. if (inv2.foe[FUEL]) return trade_type_result{4, inv2};
  63. return trade_type_result{3, inv2};
  64. }
  65. return trade_type_result{0, inv2};
  66. }
  67. int trade_type(port_type port1, port_type port2) {
  68. trade_type_result r = trade_type_info(port1, port2);
  69. return r.type;
  70. }
  71. /*
  72. // adding this breaks test-galaxy's port = {2, 2, {1,2,3}, {1,2,3}} code.
  73. port::port() {
  74. sector = 0;
  75. type = 0;
  76. for (int x = 0; x < 3; x++) {
  77. amount[x] = 0;
  78. percent[x] = 0;
  79. }
  80. }
  81. */
  82. sector_warps::sector_warps() {
  83. sector = 0;
  84. // for (int x = 0; x < MAX_WARPS; ++x) warps[x] = 0;
  85. }
  86. void sector_warps::add(sector_type new_sector) {
  87. warps.insert(new_sector);
  88. /*
  89. for (int x = 0; x < MAX_WARPS; ++x) {
  90. if (warps[x] == new_sector) return;
  91. if (warps[x] == 0) {
  92. warps[x] = new_sector;
  93. return;
  94. }
  95. }
  96. std::string message = str(boost::format("More then MAX %1% sectors for %2%") %
  97. MAX_WARPS % (int)sector);
  98. throw std::out_of_range(message);
  99. */
  100. }
  101. std::ostream &operator<<(std::ostream &os, const sector_warps &warps) {
  102. os << "Sector: " << warps.sector << " ";
  103. bool comma = false;
  104. for (auto const &warp : warps.warps) {
  105. if (comma)
  106. os << ",";
  107. else
  108. comma = true;
  109. os << warp;
  110. }
  111. /*
  112. for (int x = 0; x < MAX_WARPS; ++x) {
  113. if (warps.warps[x] != 0) {
  114. if (x != 0) os << ",";
  115. os << warps.warps[x];
  116. }
  117. }
  118. */
  119. return os;
  120. }
  121. #define GTEST_COUT std::cerr << "[ ] [ INFO ]"
  122. // #define GTEST_DEBUG
  123. // TODO: fix this. I want some trace output, but I don't want
  124. // my logs flooded ...
  125. struct port parse_portcim(const std::string line) {
  126. struct port p;
  127. p.sector = std::stoi(line);
  128. // 20 - 1708 97% - 710 56% 287 15%
  129. static std::regex portrx(
  130. "[ ]*([0-9]+) (.)[ ]+([0-9]+)[ ]+([0-9]+%) (.)[ "
  131. "]+([0-9]+)[ ]+([0-9]+%) (.)[ ]+([0-9]+)[ ]+([0-9]+%)[ ]*",
  132. std::regex_constants::ECMAScript);
  133. // does it not understand {3} ??
  134. // NO, it does not, from regex101.com:
  135. // A repeated capturing group will only capture the last iteration. Put a
  136. // capturing group around the repeated group to capture all iterations or use
  137. // a non-capturing group instead if you're not interested in the data
  138. //
  139. // static std::regex portrx("[ ]*([0-9]+)( (.)[ ]+([0-9]+)[ ]+([0-9]+%)){3}[
  140. // ]*",
  141. // std::regex_constants::ECMAScript);
  142. // sector + amount pct + amount pct + amount pct
  143. // 1 2 3 4 5 6 7 8 9 10
  144. #ifdef GTEST_DEBUG
  145. GTEST_COUT << "Sector: " << p.sector << std::endl;
  146. GTEST_COUT << "Line: [" << line << "]" << std::endl;
  147. #endif
  148. buysell port_buysell;
  149. std::smatch matches;
  150. if (std::regex_match(line, matches, portrx)) {
  151. #ifdef GTEST_DEBUG
  152. for (size_t x = 1; x < matches.size(); ++x) {
  153. GTEST_COUT << x << " : " << matches[x] << std::endl;
  154. }
  155. #endif
  156. if (matches.size() != 11) {
  157. #ifdef GTEST_DEBUG
  158. GTEST_COUT << "Now you have 101 problems." << std::endl;
  159. #endif
  160. p.sector = 0;
  161. p.type = 0;
  162. return p;
  163. }
  164. // GTEST_COUT << "matches: " << matches.size() << std::endl;
  165. p.sector = stoi(matches[1]);
  166. // GTEST_COUT << "sector: " << matches[1] << std::endl;
  167. // for (int x = 1; x < 11; ++x) {
  168. // GTEST_COUT << x << " : " << matches[x] << std::endl;
  169. // }
  170. for (int x = 0; x < 3; ++x) {
  171. int pos = x * 3;
  172. port_buysell.foe[x] = matches[pos + 2] == "-";
  173. p.amount[x] = stoi(matches[pos + 3]);
  174. p.percent[x] = stoi(matches[pos + 4]);
  175. }
  176. p.type = type_from_buysell(port_buysell);
  177. #ifdef GTEST_DEBUG
  178. GTEST_COUT << "port is type " << (int)p.type << std::endl;
  179. #endif
  180. return p;
  181. } else {
  182. #ifdef GTEST_DEBUG
  183. GTEST_COUT << "regex_match failed." << std::endl;
  184. #endif
  185. p.type = 0;
  186. p.sector = 0;
  187. return p;
  188. }
  189. }
  190. Galaxy::Galaxy() { burnt_percent = 40; }
  191. Galaxy::~Galaxy() { BUGZ_LOG(fatal) << "Galaxy::~Galaxy()"; }
  192. void Galaxy::reset(void) {
  193. meta = YAML::Node();
  194. config = YAML::Node();
  195. ports.clear();
  196. warps.clear();
  197. }
  198. void Galaxy::add_warp(sector_warps sw) {
  199. auto pos = warps.find(sw.sector);
  200. if (pos == warps.end()) {
  201. // not found
  202. // sw.sort();
  203. warps[sw.sector] = sw;
  204. // BUGZ_LOG(info) << "add_warp NEW " << sw.sector;
  205. } else {
  206. // found!
  207. if (pos->second.warps == sw.warps) {
  208. // BUGZ_LOG(trace) << "add_warp: Yup, I already know about " << sw.sector;
  209. } else {
  210. BUGZ_LOG(info) << "add_warp: Warps don't match! Updating...";
  211. BUGZ_LOG(warning) << "Have: " << pos->second;
  212. BUGZ_LOG(warning) << "Got : " << sw;
  213. warps[sw.sector] = sw;
  214. }
  215. }
  216. }
  217. void Galaxy::add_port(sector_type sector, int port_type) {
  218. auto pos = ports.find(sector);
  219. if (pos == ports.end()) {
  220. // no such port.
  221. port p;
  222. p.sector = sector;
  223. p.type = port_type;
  224. for (int x = 0; x < 3; x++) {
  225. p.amount[x] = 0;
  226. p.percent[x] = 0;
  227. }
  228. // BUGZ_LOG(trace) << "add_port: " << sector << ", " << port_type << " : "
  229. // << p;
  230. ports[sector] = p;
  231. } else {
  232. // port was found, so:
  233. if (pos->second.type == port_type) {
  234. // BUGZ_LOG(trace) << "add_port: Yup, port " << sector << " is class " <<
  235. // port_type;
  236. } else {
  237. BUGZ_LOG(fatal) << "add_port: " << sector << " shows " << pos->second.type
  238. << " >> set to " << port_type;
  239. pos->second.type = port_type;
  240. }
  241. }
  242. }
  243. void Galaxy::add_port(port p) {
  244. auto pos = ports.find(p.sector);
  245. if (pos == ports.end()) {
  246. // BUGZ_LOG(trace) << "add_port: NEW " << p;
  247. ports[p.sector] = p;
  248. } else {
  249. if (pos->second.type != p.type) {
  250. if ((pos->second.type == 9) && (p.type == 8)) {
  251. BUGZ_LOG(trace) << "add_port: StarDock " << p.sector;
  252. p.type = 9;
  253. ports[p.sector] = p;
  254. } else {
  255. BUGZ_LOG(fatal) << "add_port: " << pos->second << " NEW : " << p;
  256. ports[p.sector] = p;
  257. }
  258. } else {
  259. if (pos->second.amount != p.amount) {
  260. // BUGZ_LOG(info) << "add_port: UPDATE " << p.sector;
  261. pos->second = p;
  262. } else {
  263. // BUGZ_LOG(info) << "add_port: Yup " << p.sector;
  264. }
  265. }
  266. }
  267. }
  268. void Galaxy::load(void) {
  269. std::string filename =
  270. str(boost::format("galaxy-%1%-%2%.yaml") % game % username);
  271. // reset ?
  272. meta = YAML::Node();
  273. config = YAML::Node();
  274. ports.clear();
  275. warps.clear();
  276. if (file_exists(filename)) {
  277. YAML::Node data = YAML::LoadFile(filename);
  278. if (config["meta"]) meta = config["meta"];
  279. meta["load_from"] = filename;
  280. std::chrono::_V2::system_clock::time_point now =
  281. std::chrono::system_clock::now();
  282. meta["load_time"] = std::chrono::system_clock::to_time_t(now); // time_t
  283. if (data["config"]) {
  284. config = data["config"];
  285. } else {
  286. BUGZ_LOG(fatal) << "YAML Missing config section.";
  287. }
  288. if (data["ports"]) {
  289. const YAML::Node ports = data["ports"];
  290. for (auto const &port_iter : ports) {
  291. port p;
  292. p.sector = port_iter.first.as<int>();
  293. p.type = port_iter.second["class"].as<int>();
  294. int x;
  295. if (p.type == 0) {
  296. // nothing to load for type = 0, set defaults.
  297. for (x = 0; x < 3; ++x) {
  298. p.amount[x] = 0;
  299. p.percent[x] = 0;
  300. }
  301. } else {
  302. x = 0;
  303. for (auto const &amount : port_iter.second["amount"]) {
  304. p.amount[x] = amount.as<int>();
  305. ++x;
  306. }
  307. x = 0;
  308. for (auto const &pct : port_iter.second["pct"]) {
  309. p.percent[x] = pct.as<int>();
  310. ++x;
  311. }
  312. }
  313. add_port(p);
  314. }
  315. } else {
  316. BUGZ_LOG(fatal) << "YAML Missing ports section.";
  317. }
  318. if (data["warps"]) {
  319. const YAML::Node &warps = data["warps"];
  320. // if (warps.IsMap()) {
  321. for (auto const warp_iter : warps) {
  322. sector_warps sw;
  323. sw.sector = warp_iter.first.as<int>();
  324. for (auto const sector_iter : warp_iter.second) {
  325. sw.add(sector_iter.as<int>());
  326. }
  327. // BUGZ_LOG(trace) << "YAML warp: " << sw;
  328. add_warp(sw);
  329. }
  330. // }
  331. } else {
  332. BUGZ_LOG(fatal) << "YAML Missing warps section.";
  333. }
  334. BUGZ_LOG(fatal) << "YAML: config keys: " << config.size();
  335. BUGZ_LOG(fatal) << "YAML: warp keys: " << warps.size();
  336. BUGZ_LOG(fatal) << "YAML: port keys: " << ports.size();
  337. } else {
  338. BUGZ_LOG(fatal) << "Missing YAML: " << filename;
  339. }
  340. }
  341. void Galaxy::save(void) {
  342. std::string filename =
  343. str(boost::format("galaxy-%1%-%2%.yaml") % game % username);
  344. YAML::Node data;
  345. // add some information to meta before saving.
  346. meta["save_to"] = filename;
  347. std::chrono::_V2::system_clock::time_point now =
  348. std::chrono::system_clock::now();
  349. meta["save_time"] = std::chrono::system_clock::to_time_t(now); // time_t
  350. data["meta"] = meta;
  351. // data["meta"].SetStyle(YAML::EmitterStyle::Flow);
  352. BUGZ_LOG(fatal) << "YAML config: " << config.size();
  353. data["config"] = config;
  354. data["config"].SetStyle(YAML::EmitterStyle::Flow);
  355. /*
  356. for (auto const &config_iter : config) {
  357. data["config"][config_iter.first] = config_iter.second;
  358. }
  359. */
  360. BUGZ_LOG(fatal) << "YAML warps: " << warps.size();
  361. for (auto const &warp : warps) {
  362. for (auto const &sector : warp.second.warps) {
  363. data["warps"][warp.first].push_back(sector);
  364. }
  365. data["warps"][warp.first].SetStyle(YAML::EmitterStyle::Flow);
  366. /*
  367. for (int x = 0; x < MAX_WARPS; ++x) {
  368. if (warp.second.warps[x] == 0) break;
  369. data["warps"][warp.first].push_back(warp.second.warps[x]);
  370. }
  371. */
  372. }
  373. BUGZ_LOG(fatal) << "YAML ports: " << ports.size();
  374. /*
  375. When saving to yaml, my sector_type is like char. So, it wants
  376. to save the values as a character. Cast to int.
  377. */
  378. for (auto const &port : ports) {
  379. data["ports"][port.second.sector]["class"] = (int)port.second.type;
  380. if (port.second.type == 0) {
  381. data["ports"][port.second.sector].SetStyle(YAML::EmitterStyle::Flow);
  382. } else {
  383. // nothing to save for type = 0
  384. for (int x = 0; x < 3; x++) {
  385. data["ports"][port.second.sector]["amount"].push_back(
  386. (int)port.second.amount[x]);
  387. data["ports"][port.second.sector]["pct"].push_back(
  388. (int)port.second.percent[x]);
  389. }
  390. data["ports"][port.second.sector]["amount"].SetStyle(
  391. YAML::EmitterStyle::Flow);
  392. data["ports"][port.second.sector]["pct"].SetStyle(
  393. YAML::EmitterStyle::Flow);
  394. data["ports"][port.second.sector].SetStyle(YAML::EmitterStyle::Flow);
  395. }
  396. }
  397. std::ofstream fout(filename);
  398. fout << data << std::endl;
  399. BUGZ_LOG(fatal) << "YAML: " << filename;
  400. }
  401. std::vector<port_pair_type> Galaxy::find_trades(sector_type sector,
  402. bool highest) {
  403. std::vector<port_pair_type> pptv;
  404. // Does this sector have a port?
  405. auto port = ports.find(sector);
  406. if (port == ports.end()) return pptv;
  407. auto port_warps = warps.find(sector);
  408. // warps not found for port sector?
  409. if (port_warps == warps.end()) return pptv;
  410. for (auto const &s : port_warps->second.warps) {
  411. // only count the ports > our sector number -- so we don't have dups
  412. if (highest && (s < sector)) continue;
  413. // verify we have a way back
  414. {
  415. auto warpback = warps.find(s);
  416. // can we find that warp?
  417. if (warpback == warps.end()) continue;
  418. // does it link back to the port's sector?
  419. if (warpback->second.warps.find(sector) == warpback->second.warps.end())
  420. continue;
  421. }
  422. // Does this sector have a port?
  423. auto possible_port = ports.find(s);
  424. if (possible_port == ports.end()) continue;
  425. // calculate trade type
  426. // int t = trade_type(port->second.type, possible_port->second.type);
  427. trade_type_result ttr =
  428. trade_type_info(port->second.type, possible_port->second.type);
  429. if ((ttr.type == 0) || (ttr.type == 4)) continue;
  430. bool burnt = false;
  431. for (int x = 0; x < 3; ++x) {
  432. if (ttr.trades.foe[x]) {
  433. // if port isn't unknown, check to see if it's burnt out.
  434. if (!possible_port->second.unknown())
  435. if (possible_port->second.percent[x] < burnt_percent) burnt = true;
  436. if (!port->second.unknown())
  437. if (port->second.percent[x] < burnt_percent) burnt = true;
  438. }
  439. }
  440. if (burnt) continue;
  441. pptv.push_back(port_pair_type{ttr.type, sector, s});
  442. BUGZ_LOG(trace) << "sector: " << sector << " and " << s
  443. << " tt:" << ttr.type;
  444. }
  445. return pptv;
  446. }
  447. bool compare_port_pair(const port_pair_type &ppt1, const port_pair_type &ppt2) {
  448. if (ppt1.type == ppt2.type) {
  449. return ppt1.s1 < ppt2.s1;
  450. } else {
  451. return (ppt1.type < ppt2.type);
  452. }
  453. }
  454. void Galaxy::sort_port_pair_type(std::vector<port_pair_type> &pptv) {
  455. sort(pptv.begin(), pptv.end(), compare_port_pair);
  456. }
  457. std::vector<port_pair_type> Galaxy::find_best_trades(void) {
  458. std::vector<port_pair_type> pptv;
  459. burnt_percent = config["burnt_percent"].as<int>();
  460. if (burnt_percent > 90) {
  461. burnt_percent = 90;
  462. config["burnt_percent"] = 90;
  463. }
  464. if (burnt_percent < 10) {
  465. burnt_percent = 10;
  466. config["burnt_percent"] = 10;
  467. }
  468. for (auto const &pi : ports) {
  469. if (pi.second.type == 0) continue;
  470. sector_type sector = pi.second.sector;
  471. auto port_warps = warps.find(sector);
  472. if (port_warps == warps.end()) continue;
  473. std::vector<port_pair_type> ppt_sector = find_trades(sector, true);
  474. if (ppt_sector.empty()) continue;
  475. pptv.insert(pptv.end(), ppt_sector.begin(), ppt_sector.end());
  476. }
  477. return pptv;
  478. }