db.cpp 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539
  1. #include "db.h"
  2. #include "utils.h"
  3. #include <SQLiteCpp/VariadicBind.h>
  4. #include <chrono>
  5. #include <iomanip>
  6. #include <iostream>
  7. #include <sstream>
  8. #include <thread>
  9. /*
  10. database is locked
  11. This happens when more then one node plays the game.
  12. The database access is slow.
  13. So, make sure you set it up so that you do your writes right
  14. before you collect user input. That way, the user won't see
  15. the lags.
  16. This might be an issue on rPI systems!
  17. Change the strategy so we only update when the game ends.
  18. */
  19. DBData::DBData(void)
  20. : db("space-data.db", SQLite::OPEN_READWRITE | SQLite::OPEN_CREATE) {
  21. locked_retries = 50;
  22. create_tables();
  23. }
  24. DBData::~DBData() {}
  25. void DBData::retry_wait(void) {
  26. std::this_thread::sleep_for(std::chrono::milliseconds(100));
  27. }
  28. #define DBLOCK "database is locked"
  29. /**
  30. * @brief create tables if they don't exist.
  31. */
  32. void DBData::create_tables(void) {
  33. int tries = 0;
  34. retry:
  35. try {
  36. db.exec("CREATE TABLE IF NOT EXISTS settings(username TEXT, setting TEXT, "
  37. "value TEXT, PRIMARY KEY(username, setting));");
  38. db.exec(
  39. "CREATE TABLE IF NOT EXISTS scores ( \"username\" TEXT, \"when\" "
  40. "INTEGER, \"date\" INTEGER, \"hand\" INTEGER, \"won\" INTEGER, "
  41. "\"score\" INTEGER, PRIMARY KEY(\"username\", \"date\", \"hand\"));");
  42. db.exec("CREATE TABLE IF NOT EXISTS monthly ( \"month\" "
  43. "INTEGER, \"username\" TEXT, \"days\" INTEGER, "
  44. "\"score\" INTEGER, PRIMARY KEY(\"month\",\"username\") );");
  45. } catch (std::exception &e) {
  46. if (get_logger) {
  47. get_logger() << "create_tables():" << std::endl;
  48. get_logger() << "SQLite exception: " << e.what() << std::endl;
  49. }
  50. if (strcmp(e.what(), DBLOCK) == 0) {
  51. ++tries;
  52. if (tries < locked_retries) {
  53. retry_wait();
  54. goto retry;
  55. }
  56. if (get_logger)
  57. get_logger() << "giving up! " << tries << " retries." << std::endl;
  58. }
  59. }
  60. if (tries > 0) {
  61. if (get_logger)
  62. get_logger() << "success after " << tries << std::endl;
  63. }
  64. }
  65. /**
  66. * @brief get setting from the settings table
  67. *
  68. * We use user and setting.
  69. * Return isMissing if not found.
  70. *
  71. * @param setting
  72. * @param ifMissing
  73. * @return std::string
  74. */
  75. std::string DBData::getSetting(const std::string &setting,
  76. std::string ifMissing) {
  77. SQLite::Statement stmt_getSet = SQLite::Statement(
  78. db, "SELECT value FROM settings WHERE username=? AND setting=?");
  79. int tries = 0;
  80. retry:
  81. try {
  82. stmt_getSet.bind(1, user);
  83. stmt_getSet.bind(2, setting);
  84. if (stmt_getSet.executeStep()) {
  85. std::string value = stmt_getSet.getColumn(0);
  86. return value;
  87. };
  88. return ifMissing;
  89. } catch (std::exception &e) {
  90. if (get_logger) {
  91. get_logger() << "getSettings( " << setting << "," << ifMissing
  92. << " ): " << user << std::endl;
  93. get_logger() << "SQLite exception: " << e.what() << std::endl;
  94. }
  95. if (strcmp(e.what(), DBLOCK) == 0) {
  96. ++tries;
  97. if (tries < locked_retries) {
  98. retry_wait();
  99. goto retry;
  100. }
  101. if (get_logger)
  102. get_logger() << "giving up! " << tries << " retries." << std::endl;
  103. }
  104. }
  105. if (tries > 0) {
  106. if (get_logger)
  107. get_logger() << "success after " << tries << std::endl;
  108. }
  109. return ifMissing;
  110. }
  111. /**
  112. * @brief save setting in the settings table
  113. *
  114. * We save user setting in the settings table.
  115. * We use SQLite's REPLACE INTO so it does an update if it exists, or an insert
  116. * if it is missing.
  117. * @param setting
  118. * @param value
  119. */
  120. void DBData::setSetting(const std::string &setting, const std::string &value) {
  121. SQLite::Statement stmt_setSet = SQLite::Statement(
  122. db, "REPLACE INTO settings(username, setting, value) VALUES(?,?,?);");
  123. int tries = 0;
  124. retry:
  125. try {
  126. stmt_setSet.bind(1, user);
  127. stmt_setSet.bind(2, setting);
  128. stmt_setSet.bind(3, value);
  129. stmt_setSet.exec();
  130. } catch (std::exception &e) {
  131. if (get_logger) {
  132. get_logger() << "setSettings( " << setting << "," << value
  133. << " ): " << user << std::endl;
  134. get_logger() << "SQLite exception: " << e.what() << std::endl;
  135. }
  136. if (strcmp(e.what(), DBLOCK) == 0) {
  137. ++tries;
  138. if (tries < locked_retries) {
  139. retry_wait();
  140. goto retry;
  141. }
  142. if (get_logger)
  143. get_logger() << "giving up! " << tries << " retries." << std::endl;
  144. }
  145. }
  146. if (tries > 0)
  147. if (get_logger)
  148. get_logger() << "success after " << tries << std::endl;
  149. }
  150. /**
  151. * @brief save the user's score
  152. *
  153. * @param when now()
  154. * @param date what day they played
  155. * @param hand which hand they played
  156. * @param won did they win? 1/0
  157. * @param score
  158. */
  159. void DBData::saveScore(time_t when, time_t date, int hand, int won, int score) {
  160. int tries = 0;
  161. retry:
  162. try {
  163. SQLite::Statement stmt(
  164. db, "INSERT INTO scores( \"username\", \"when\", "
  165. "\"date\", \"hand\", \"won\", \"score\") VALUES(?,?,?,?,?,?);");
  166. stmt.bind(1, user);
  167. stmt.bind(2, when);
  168. stmt.bind(3, date);
  169. stmt.bind(4, hand);
  170. stmt.bind(5, won);
  171. stmt.bind(6, score);
  172. stmt.exec();
  173. } catch (std::exception &e) {
  174. if (get_logger) {
  175. get_logger() << "saveScore( " << when << "," << date << "," << hand << ","
  176. << won << "," << score << " ): " << user << std::endl;
  177. get_logger() << "SQLite exception: " << e.what() << std::endl;
  178. }
  179. if (strcmp(e.what(), DBLOCK) == 0) {
  180. ++tries;
  181. if (tries < locked_retries) {
  182. retry_wait();
  183. goto retry;
  184. }
  185. if (get_logger)
  186. get_logger() << "giving up! " << tries << " retries." << std::endl;
  187. }
  188. }
  189. if (tries > 0)
  190. if (get_logger)
  191. get_logger() << "success after " << tries << std::endl;
  192. }
  193. /**
  194. * @brief Returns number of hands played on given day
  195. *
  196. * returns number of hands they played, or 0
  197. * @param day
  198. * @return int
  199. */
  200. int DBData::handsPlayedOnDay(time_t day) {
  201. int tries = 0;
  202. retry:
  203. try {
  204. SQLite::Statement stmt(
  205. db, "SELECT COUNT(*) FROM scores WHERE \"username\"=? AND \"DATE\"=?;");
  206. stmt.bind(1, user);
  207. stmt.bind(2, day);
  208. int count = 0;
  209. if (stmt.executeStep()) {
  210. count = stmt.getColumn(0);
  211. };
  212. return count;
  213. } catch (std::exception &e) {
  214. if (get_logger) {
  215. get_logger() << "handsPlayedOnDay( " << day << " ): " << user
  216. << std::endl;
  217. get_logger() << "SQLite exception: " << e.what() << std::endl;
  218. }
  219. if (strcmp(e.what(), DBLOCK) == 0) {
  220. ++tries;
  221. if (tries < locked_retries) {
  222. retry_wait();
  223. goto retry;
  224. }
  225. if (get_logger)
  226. get_logger() << "giving up! " << tries << " retries." << std::endl;
  227. }
  228. }
  229. if (tries > 0)
  230. if (get_logger)
  231. get_logger() << "success after " << tries << std::endl;
  232. return 0;
  233. }
  234. /*
  235. * If you're looking at scores, you're not really looking for all the details.
  236. * I think using the group/SUM would be better, and it sorts scores from highest
  237. * to lowest. Let SQL do the work for me.
  238. */
  239. // select date,username,SUM(score),SUM(won) FROM scores group by date,username
  240. // ORDER BY SUM(score) DESC;
  241. std::vector<scores_details> DBData::getScoresOnDay(time_t date) {
  242. std::vector<scores_details> scores;
  243. int tries = 0;
  244. retry:
  245. try {
  246. // \"when\",
  247. SQLite::Statement stmt(db, "SELECT \"username\", \"date\", \"hand\", "
  248. "\"won\", \"score\" FROM SCORES WHERE "
  249. "\"date\"=? ORDER BY \"username\", \"hand\";");
  250. stmt.bind(1, date);
  251. while (stmt.executeStep()) {
  252. scores_details sd;
  253. sd.user = (const char *)stmt.getColumn(0);
  254. sd.date = stmt.getColumn(1);
  255. sd.hand = stmt.getColumn(2);
  256. sd.won = stmt.getColumn(3);
  257. sd.score = stmt.getColumn(4);
  258. scores.push_back(sd);
  259. }
  260. } catch (std::exception &e) {
  261. if (get_logger) {
  262. get_logger() << "getScoresOnDay( " << date << " ): " << std::endl;
  263. get_logger() << "SQLite exception: " << e.what() << std::endl;
  264. }
  265. scores.clear();
  266. if (strcmp(e.what(), DBLOCK) == 0) {
  267. ++tries;
  268. if (tries < locked_retries) {
  269. retry_wait();
  270. goto retry;
  271. }
  272. if (get_logger)
  273. get_logger() << "giving up! " << tries << " retries." << std::endl;
  274. }
  275. }
  276. if (tries > 0)
  277. if (get_logger)
  278. get_logger() << "success after " << tries << std::endl;
  279. return scores;
  280. }
  281. /**
  282. * @brief Gets scores, time_t is day, vector has user and scores sorted highest
  283. * to lowest.
  284. *
  285. * @return std::map<time_t, std::vector<scores_data>>
  286. */
  287. std::map<time_t, std::vector<scores_data>> DBData::getScores(void) {
  288. std::map<time_t, std::vector<scores_data>> scores;
  289. int tries = 0;
  290. retry:
  291. try {
  292. SQLite::Statement stmt(
  293. db, "SELECT `date`,username,SUM(score),SUM(won) FROM scores "
  294. "GROUP BY `date`,username ORDER BY `date`,SUM(score) DESC;");
  295. time_t current = 0;
  296. std::vector<scores_data> vsd;
  297. while (stmt.executeStep()) {
  298. time_t the_date = stmt.getColumn(0);
  299. if (current == 0) {
  300. // ok, we've got the first one!
  301. current = the_date;
  302. } else {
  303. // Ok, are we on another date now?
  304. if (the_date != current) {
  305. scores[current] = std::move(vsd);
  306. vsd.clear();
  307. current = the_date;
  308. }
  309. }
  310. scores_data sd;
  311. sd.user = (const char *)stmt.getColumn(1);
  312. sd.date = the_date;
  313. sd.score = stmt.getColumn(2);
  314. sd.won = stmt.getColumn(3);
  315. vsd.push_back(sd);
  316. }
  317. if (!vsd.empty()) {
  318. scores[current] = std::move(vsd);
  319. }
  320. vsd.clear();
  321. } catch (std::exception &e) {
  322. if (get_logger) {
  323. get_logger() << "getScores(): " << std::endl;
  324. get_logger() << "SQLite exception: " << e.what() << std::endl;
  325. }
  326. scores.clear();
  327. if (strcmp(e.what(), DBLOCK) == 0) {
  328. ++tries;
  329. if (tries < locked_retries) {
  330. retry_wait();
  331. goto retry;
  332. }
  333. if (get_logger)
  334. get_logger() << "giving up! " << locked_retries << " retries."
  335. << std::endl;
  336. }
  337. }
  338. if (tries > 0)
  339. if (get_logger)
  340. get_logger() << "success after " << tries << std::endl;
  341. return scores;
  342. }
  343. /**
  344. * @brief Get hands played per day
  345. *
  346. * Uses the user value.
  347. *
  348. * @return std::map<time_t, int>
  349. */
  350. std::map<time_t, int> DBData::getPlayed(void) {
  351. std::map<time_t, int> hands;
  352. int tries = 0;
  353. retry:
  354. try {
  355. SQLite::Statement stmt(
  356. // select date, count(hand) from scores where username='grinder' group
  357. // by date;
  358. db, "SELECT `date`,COUNT(hand) FROM scores "
  359. "WHERE username=? GROUP BY `date`;");
  360. stmt.bind(1, user);
  361. while (stmt.executeStep()) {
  362. time_t the_date = stmt.getColumn(0);
  363. hands[the_date] = stmt.getColumn(1);
  364. }
  365. } catch (std::exception &e) {
  366. if (get_logger) {
  367. get_logger() << "getPlayed(): " << user << std::endl;
  368. get_logger() << "SQLite exception: " << e.what() << std::endl;
  369. }
  370. hands.clear();
  371. if (strcmp(e.what(), DBLOCK) == 0) {
  372. ++tries;
  373. if (tries < locked_retries) {
  374. retry_wait();
  375. goto retry;
  376. }
  377. if (get_logger)
  378. get_logger() << "giving up! " << tries << " retries." << std::endl;
  379. }
  380. }
  381. if (tries > 0)
  382. if (get_logger)
  383. get_logger() << "success after " << tries << std::endl;
  384. return hands;
  385. }
  386. /**
  387. * @brief When has the user played?
  388. *
  389. * This returns a map of date (time_t), and number of hands played on that date.
  390. *
  391. * @return std::map<time_t, long>
  392. */
  393. std::map<time_t, int> DBData::whenPlayed(void) {
  394. // select "date", count(hand) from scores where username='?' group by
  395. // "date";
  396. std::map<time_t, int> plays;
  397. int tries = 0;
  398. retry:
  399. try {
  400. SQLite::Statement stmt(db, "SELECT `date`, COUNT(hand) FROM scores WHERE "
  401. "username=? GROUP BY `date`;");
  402. stmt.bind(1, user);
  403. while (stmt.executeStep()) {
  404. time_t d = (long)stmt.getColumn(0);
  405. plays[d] = stmt.getColumn(1);
  406. }
  407. } catch (std::exception &e) {
  408. if (get_logger) {
  409. get_logger() << "whenPlayed(): " << std::endl;
  410. get_logger() << "SQLite exception: " << e.what() << std::endl;
  411. }
  412. plays.clear();
  413. if (strcmp(e.what(), DBLOCK) == 0) {
  414. ++tries;
  415. if (tries < locked_retries) {
  416. retry_wait();
  417. goto retry;
  418. }
  419. if (get_logger)
  420. get_logger() << "giving up! " << tries << " retries." << std::endl;
  421. }
  422. }
  423. if (tries > 0)
  424. if (get_logger)
  425. get_logger() << "success after " << tries << std::endl;
  426. return plays;
  427. }
  428. /**
  429. * @brief This will expire out old scores
  430. *
  431. * @todo implement, but don't throw away high scores.
  432. *
  433. */
  434. void DBData::expireScores(void) {}
  435. /**
  436. * @brief Format date to string.
  437. *
  438. * We use default "%0m/%0d/%Y", but can be configured by SysOp via
  439. * config["date_score"] setting. "%Y/%0m/%0d" for non-US?
  440. *
  441. * @param tt
  442. * @return std::string
  443. */
  444. std::string convertDateToDateScoreFormat(time_t tt) {
  445. std::stringstream ss;
  446. if (config["date_score"]) {
  447. std::string custom_format = config["date_score"].as<std::string>();
  448. ss << std::put_time(std::localtime(&tt), custom_format.c_str());
  449. } else {
  450. ss << std::put_time(std::localtime(&tt), "%0m/%0d/%Y");
  451. }
  452. std::string date = ss.str();
  453. return date;
  454. }
  455. void normalizeDate(std::chrono::_V2::system_clock::time_point &date) {
  456. time_t date_t = std::chrono::system_clock::to_time_t(date);
  457. normalizeDate(date_t);
  458. date = std::chrono::system_clock::from_time_t(date_t);
  459. }
  460. /**
  461. * @brief change datetime to have consistent time
  462. *
  463. * This converts the time part to hour:00
  464. *
  465. * @param tt
  466. * @param hour
  467. */
  468. void normalizeDate(time_t &tt, int hour) {
  469. std::tm *local_tm = localtime(&tt);
  470. // adjust date to 2:00:00 AM
  471. tt -= (local_tm->tm_min * 60) + local_tm->tm_sec;
  472. while (local_tm->tm_hour < hour) {
  473. ++local_tm->tm_hour;
  474. tt += 60 * 60;
  475. }
  476. if (local_tm->tm_hour > hour) {
  477. tt -= (60 * 60) * (local_tm->tm_hour - hour);
  478. }
  479. /* // possible DST adjustment. LMTATSM
  480. if (local_tm->tm_isdst) {
  481. // DST in effect
  482. tt -= (60*60);
  483. }
  484. */
  485. }
  486. void firstOfMonthDate(std::chrono::_V2::system_clock::time_point &date) {
  487. using namespace std::literals;
  488. time_t date_t = std::chrono::system_clock::to_time_t(date);
  489. // adjust to first day of the month
  490. std::tm date_tm;
  491. localtime_r(&date_t, &date_tm);
  492. if (date_tm.tm_mday > 1) {
  493. date -= 24h * (date_tm.tm_mday - 1);
  494. }
  495. normalizeDate(date);
  496. }