galaxy.cpp 14 KB

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