galaxy.cpp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450
  1. #include "galaxy.h"
  2. #include <algorithm> // sort
  3. #include <boost/format.hpp>
  4. #include <exception>
  5. #include <fstream>
  6. #include <ostream>
  7. #include <set>
  8. #include <string>
  9. #include "logging.h"
  10. #include "yaml-cpp/yaml.h"
  11. // c++ default exceptions list
  12. // https://en.cppreference.com/w/cpp/error/exception
  13. bool buysell::operator==(const buysell &rhs) const {
  14. return ((foe[0] == rhs.foe[0]) && (foe[1] == rhs.foe[1]) &&
  15. (foe[2] == rhs.foe[2]));
  16. }
  17. std::ostream &operator<<(std::ostream &os, const buysell &bs) {
  18. os << bs.foe[0] << bs.foe[1] << bs.foe[2];
  19. return os;
  20. }
  21. bool buysell_text::operator==(const buysell_text &rhs) const {
  22. return ((txt[0] == rhs.txt[0]) && (txt[1] == rhs.txt[1]) &&
  23. (txt[2] == rhs.txt[2]));
  24. }
  25. std::ostream &operator<<(std::ostream &os, const buysell_text &bst) {
  26. os << '"' << bst.txt[0] << bst.txt[1] << bst.txt[2] << '"';
  27. return os;
  28. }
  29. std::ostream &operator<<(std::ostream &os, const port &p) {
  30. if (p.type == 0) {
  31. os << p.sector << ": " << (int)p.type;
  32. } else {
  33. os << p.sector << ": " << (int)p.type << " " << text_from_type(p.type)
  34. << " " << p.amount[0] << "," << p.amount[1] << "," << p.amount[2];
  35. }
  36. return os;
  37. }
  38. /*
  39. // adding this breaks test-galaxy's port = {2, 2, {1,2,3}, {1,2,3}} code.
  40. port::port() {
  41. sector = 0;
  42. type = 0;
  43. for (int x = 0; x < 3; x++) {
  44. amount[x] = 0;
  45. percent[x] = 0;
  46. }
  47. }
  48. */
  49. sector_warps::sector_warps() {
  50. sector = 0;
  51. // for (int x = 0; x < MAX_WARPS; ++x) warps[x] = 0;
  52. }
  53. void sector_warps::add(sector_type new_sector) {
  54. warps.insert(new_sector);
  55. /*
  56. for (int x = 0; x < MAX_WARPS; ++x) {
  57. if (warps[x] == new_sector) return;
  58. if (warps[x] == 0) {
  59. warps[x] = new_sector;
  60. return;
  61. }
  62. }
  63. std::string message = str(boost::format("More then MAX %1% sectors for %2%") %
  64. MAX_WARPS % (int)sector);
  65. throw std::out_of_range(message);
  66. */
  67. }
  68. std::ostream &operator<<(std::ostream &os, const sector_warps &warps) {
  69. os << "Sector: " << warps.sector << " ";
  70. bool comma = false;
  71. for (auto const &warp : warps.warps) {
  72. if (comma)
  73. os << ",";
  74. else
  75. comma = true;
  76. os << warp;
  77. }
  78. /*
  79. for (int x = 0; x < MAX_WARPS; ++x) {
  80. if (warps.warps[x] != 0) {
  81. if (x != 0) os << ",";
  82. os << warps.warps[x];
  83. }
  84. }
  85. */
  86. return os;
  87. }
  88. #ifdef NOT_SET
  89. bool sector_sort(sector_type st1, sector_type st2) {
  90. /* Sort sectors, put 0's at the end. */
  91. if (st1 == 0) return false;
  92. if (st2 == 0) return false;
  93. return (st1 < st2);
  94. }
  95. void sector_warps::sort(void) {
  96. std::sort(&warps[0], &warps[MAX_WARPS], sector_sort);
  97. }
  98. bool sector_warps::operator==(const sector_warps &rhs) const {
  99. /*
  100. Comapre if the sector_warps are the same.
  101. They do not need to be in the same order.
  102. */
  103. std::set<sector_type> contains;
  104. if (sector == rhs.sector) {
  105. int x;
  106. for (x = 0; x < MAX_WARPS; ++x) {
  107. if (warps[x] == 0) break;
  108. contains.insert(warps[x]);
  109. }
  110. for (x = 0; x < MAX_WARPS; ++x) {
  111. if (warps[0] == 0) break;
  112. auto pos = contains.find(rhs.warps[x]);
  113. if (pos == contains.end()) return false;
  114. contains.erase(pos);
  115. }
  116. return contains.empty();
  117. }
  118. // sector number doesn't match!
  119. return false;
  120. }
  121. #endif
  122. #define GTEST_COUT std::cerr << "[ ] [ INFO ]"
  123. #define GTEST_DEBUG
  124. struct port parse_portcim(const std::string line) {
  125. struct port p;
  126. p.sector = std::stoi(line);
  127. // 20 - 1708 97% - 710 56% 287 15%
  128. static std::regex portrx(
  129. "[ ]*([0-9]+) (.)[ ]+([0-9]+)[ ]+([0-9]+%) (.)[ "
  130. "]+([0-9]+)[ ]+([0-9]+%) (.)[ ]+([0-9]+)[ ]+([0-9]+%)[ ]*",
  131. std::regex_constants::ECMAScript);
  132. // does it not understand {3} ??
  133. // NO, it does not, from regex101.com:
  134. // A repeated capturing group will only capture the last iteration. Put a
  135. // capturing group around the repeated group to capture all iterations or use
  136. // a non-capturing group instead if you're not interested in the data
  137. //
  138. // static std::regex portrx("[ ]*([0-9]+)( (.)[ ]+([0-9]+)[ ]+([0-9]+%)){3}[
  139. // ]*",
  140. // std::regex_constants::ECMAScript);
  141. // sector + amount pct + amount pct + amount pct
  142. // 1 2 3 4 5 6 7 8 9 10
  143. #ifdef GTEST_DEBUG
  144. GTEST_COUT << "Sector: " << p.sector << std::endl;
  145. GTEST_COUT << "Line: [" << line << "]" << std::endl;
  146. #endif
  147. buysell port_buysell;
  148. std::smatch matches;
  149. if (std::regex_match(line, matches, portrx)) {
  150. #ifdef GTEST_DEBUG
  151. for (size_t x = 1; x < matches.size(); ++x) {
  152. GTEST_COUT << x << " : " << matches[x] << std::endl;
  153. }
  154. #endif
  155. if (matches.size() != 11) {
  156. #ifdef GTEST_DEBUG
  157. GTEST_COUT << "Now you have 101 problems." << std::endl;
  158. #endif
  159. p.sector = 0;
  160. p.type = 0;
  161. return p;
  162. }
  163. // GTEST_COUT << "matches: " << matches.size() << std::endl;
  164. p.sector = stoi(matches[1]);
  165. // GTEST_COUT << "sector: " << matches[1] << std::endl;
  166. // for (int x = 1; x < 11; ++x) {
  167. // GTEST_COUT << x << " : " << matches[x] << std::endl;
  168. // }
  169. for (int x = 0; x < 3; ++x) {
  170. int pos = x * 3;
  171. port_buysell.foe[x] = matches[pos + 2] == "-";
  172. p.amount[x] = stoi(matches[pos + 3]);
  173. p.percent[x] = stoi(matches[pos + 4]);
  174. }
  175. p.type = type_from_buysell(port_buysell);
  176. #ifdef GTEST_DEBUG
  177. GTEST_COUT << "port is type " << (int)p.type << std::endl;
  178. #endif
  179. return p;
  180. } else {
  181. #ifdef GTEST_DEBUG
  182. GTEST_COUT << "regex_match failed." << std::endl;
  183. #endif
  184. p.type = 0;
  185. p.sector = 0;
  186. return p;
  187. }
  188. }
  189. Galaxy::Galaxy() {}
  190. Galaxy::~Galaxy() { BUGZ_LOG(fatal) << "Galaxy::~Galaxy()"; }
  191. void Galaxy::add_warp(sector_warps sw) {
  192. auto pos = warps.find(sw.sector);
  193. if (pos == warps.end()) {
  194. // not found
  195. // sw.sort();
  196. warps[sw.sector] = sw;
  197. BUGZ_LOG(info) << "add_warp NEW " << sw.sector;
  198. } else {
  199. // found!
  200. if (pos->second.warps == sw.warps) {
  201. BUGZ_LOG(trace) << "add_warp: Yup, I already know about " << sw.sector;
  202. } else {
  203. BUGZ_LOG(info) << "add_warp: Warps don't match! Updating...";
  204. BUGZ_LOG(warning) << "Have: " << pos->second;
  205. BUGZ_LOG(warning) << "Got : " << sw;
  206. warps[sw.sector] = sw;
  207. }
  208. }
  209. }
  210. void Galaxy::add_port(sector_type sector, int port_class) {
  211. auto pos = ports.find(sector);
  212. if (pos == ports.end()) {
  213. // no such port.
  214. port p;
  215. p.sector = sector;
  216. p.type = port_class;
  217. for( int x= 0; x < 3; x++) {
  218. p.amount[x ] = 0;
  219. p.percent[x] = 0;
  220. }
  221. BUGZ_LOG(info) << "add_port: " << sector << ", " << port_class << " : " << p;
  222. ports[sector] = p;
  223. } else {
  224. // port was found, so:
  225. if ( pos->second.type == port_class) {
  226. BUGZ_LOG(info) << "add_port: Yup, port " << sector << " is class " << port_class;
  227. } else {
  228. BUGZ_LOG(fatal) << "add_port: " << sector << " shows " << pos->second.type << " >> set to " << port_class;
  229. }
  230. }
  231. }
  232. void Galaxy::add_port(port p) {
  233. auto pos = ports.find(p.sector);
  234. if (pos == ports.end()) {
  235. BUGZ_LOG(info) << "add_port: NEW " << p;
  236. ports[p.sector] = p;
  237. } else {
  238. if (pos->second.type != p.type) {
  239. if ( (pos->second.type == 9) && (p.type == 8) ) {
  240. BUGZ_LOG(info) << "add_port: StarDock " << p.sector;
  241. p.type = 9;
  242. ports[p.sector] = p;
  243. } else {
  244. BUGZ_LOG(fatal) << "add_port: " << pos->second << " NEW : " << p;
  245. ports[p.sector] = p;
  246. }
  247. } else {
  248. BUGZ_LOG(info) << "add_port: Yup " << p.sector;
  249. }
  250. }
  251. }
  252. #ifdef NO_NOT_EVER
  253. namespace YAML {
  254. template <>
  255. struct convert<Galaxy> {
  256. static Node encode(const Galaxy &rhs) {
  257. Node node;
  258. for (auto const &config_iter : rhs.config) {
  259. node["config"][config_iter.first] = config_iter.second;
  260. }
  261. for (auto const &warp : rhs.warps) {
  262. for (int x = 0; x < MAX_WARPS; ++x) {
  263. if (warp.second.warps[x] == 0) break;
  264. node["warps"][warp.first].push_back(warp.second.warps[x]);
  265. }
  266. }
  267. /*
  268. node.push_back(rhs.x);
  269. node.push_back(rhs.y);
  270. node.push_back(rhs.z);
  271. return node;
  272. */
  273. return node;
  274. }
  275. static bool decode(const Node &node, Galaxy &rhs) {
  276. if (!node.IsMap()) return false;
  277. if (node["config"]) {
  278. } else {
  279. BUGZ_LOG(fatal) << "YAML missing config section.";
  280. }
  281. if (node["ports"]) {
  282. } else {
  283. BUGZ_LOG(fatal) << "YAML missing ports section.";
  284. }
  285. if (node["warps"]) {
  286. const Node &warps = node["warps"];
  287. if (warps.IsMap()) {
  288. for (auto const warp_iter : warps) {
  289. sector_warps sw;
  290. sw.sector = warp_iter.first.as<int>();
  291. for (auto const sector_iter : warp_iter.second) {
  292. sw.add(sector_iter.as<int>());
  293. }
  294. BUGZ_LOG(fatal) << "YAML warp: " << sw;
  295. rhs.add_warp(sw);
  296. }
  297. }
  298. } else {
  299. BUGZ_LOG(fatal) << "YAML missing warps section.";
  300. }
  301. /*
  302. if(!node.IsSequence() || node.size() != 3) {
  303. return false;
  304. }
  305. rhs.x = node[0].as<double>();
  306. rhs.y = node[1].as<double>();
  307. rhs.z = node[2].as<double>();
  308. */
  309. return true;
  310. }
  311. };
  312. } // namespace YAML
  313. #endif
  314. void Galaxy::load(void) {
  315. std::string filename =
  316. str(boost::format("galaxy-%1%-%2%.json") % game % username);
  317. // reset ?
  318. config.clear();
  319. ports.clear();
  320. warps.clear();
  321. if (file_exists(filename)) {
  322. YAML::Node data = YAML::LoadFile(filename);
  323. if (data["config"]) {
  324. } else {
  325. BUGZ_LOG(fatal) << "YAML Missing config section.";
  326. }
  327. if (data["ports"]) {
  328. const YAML::Node ports = data["ports"];
  329. for (auto const &port_iter : ports) {
  330. port p;
  331. p.sector = port_iter.first.as<int>();
  332. p.type = port_iter.second["class"].as<int>();
  333. int x = 0;
  334. for (auto const &amount : port_iter.second["amount"]) {
  335. p.amount[x] = amount.as<int>();
  336. ++x;
  337. }
  338. x = 0;
  339. for (auto const &pct : port_iter.second["pct"]) {
  340. p.percent[x] = pct.as<int>();
  341. ++x;
  342. }
  343. add_port(p);
  344. }
  345. } else {
  346. BUGZ_LOG(fatal) << "YAML Missing ports section.";
  347. }
  348. if (data["warps"]) {
  349. const YAML::Node &warps = data["warps"];
  350. // if (warps.IsMap()) {
  351. for (auto const warp_iter : warps) {
  352. sector_warps sw;
  353. sw.sector = warp_iter.first.as<int>();
  354. for (auto const sector_iter : warp_iter.second) {
  355. sw.add(sector_iter.as<int>());
  356. }
  357. BUGZ_LOG(fatal) << "YAML warp: " << sw;
  358. add_warp(sw);
  359. }
  360. // }
  361. } else {
  362. BUGZ_LOG(fatal) << "YAML Missing warps section.";
  363. }
  364. BUGZ_LOG(fatal) << "YAML: config keys: " << config.size();
  365. BUGZ_LOG(fatal) << "YAML: warp keys: " << warps.size();
  366. BUGZ_LOG(fatal) << "YAML: port keys: " << ports.size();
  367. } else {
  368. BUGZ_LOG(fatal) << "Missing YAML: " << filename;
  369. }
  370. }
  371. void Galaxy::save(void) {
  372. std::string filename =
  373. str(boost::format("galaxy-%1%-%2%.json") % game % username);
  374. YAML::Node data;
  375. BUGZ_LOG(fatal) << "YAML config: " << config.size();
  376. for (auto const &config_iter : config) {
  377. data["config"][config_iter.first] = config_iter.second;
  378. }
  379. BUGZ_LOG(fatal) << "YAML warps: " << warps.size();
  380. for (auto const &warp : warps) {
  381. for (auto const &sector : warp.second.warps) {
  382. data["warps"][warp.first].push_back(sector);
  383. }
  384. /*
  385. for (int x = 0; x < MAX_WARPS; ++x) {
  386. if (warp.second.warps[x] == 0) break;
  387. data["warps"][warp.first].push_back(warp.second.warps[x]);
  388. }
  389. */
  390. }
  391. BUGZ_LOG(fatal) << "YAML ports: " << ports.size();
  392. /*
  393. When saving to yaml, my sector_type is like char. So, it wants
  394. to save the values as a character. Cast to int.
  395. */
  396. for (auto const &port : ports) {
  397. data["ports"][port.second.sector]["class"] = (int)port.second.type;
  398. for (int x = 0; x < 3; x++) {
  399. data["ports"][port.second.sector]["amount"].push_back(
  400. (int)port.second.amount[x]);
  401. data["ports"][port.second.sector]["pct"].push_back(
  402. (int)port.second.percent[x]);
  403. }
  404. }
  405. std::ofstream fout(filename);
  406. fout << data << std::endl;
  407. BUGZ_LOG(fatal) << "YAML: " << filename;
  408. }