|
@@ -461,13 +461,169 @@ retry:
|
|
return plays;
|
|
return plays;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+struct month_user {
|
|
|
|
+ time_t date;
|
|
|
|
+ std::string username;
|
|
|
|
+ // friend bool operator<(const month_user &l, const month_user &r);
|
|
|
|
+};
|
|
|
|
+
|
|
|
|
+bool operator<(const month_user &l, const month_user &r) {
|
|
|
|
+ if (l.date < r.date)
|
|
|
|
+ return true;
|
|
|
|
+ if (l.date > r.date)
|
|
|
|
+ return false;
|
|
|
|
+
|
|
|
|
+ // Otherwise a are equal
|
|
|
|
+ if (l.username < r.username)
|
|
|
|
+ return true;
|
|
|
|
+ if (l.username > r.username)
|
|
|
|
+ return false;
|
|
|
|
+
|
|
|
|
+ // Otherwise both are equal
|
|
|
|
+ return false;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+struct month_stats {
|
|
|
|
+ int days;
|
|
|
|
+ int hands_won;
|
|
|
|
+ int score;
|
|
|
|
+};
|
|
|
|
+
|
|
/**
|
|
/**
|
|
* @brief This will expire out old scores
|
|
* @brief This will expire out old scores
|
|
*
|
|
*
|
|
* @todo implement, but don't throw away high scores.
|
|
* @todo implement, but don't throw away high scores.
|
|
*
|
|
*
|
|
|
|
+ * @param month_first_t
|
|
|
|
+ *
|
|
*/
|
|
*/
|
|
-void DBData::expireScores(void) {}
|
|
|
|
|
|
+void DBData::expireScores(time_t month_first_t) {
|
|
|
|
+ // step 1: aquire lock
|
|
|
|
+ std::ofstream lockfile;
|
|
|
|
+ lockfile.open("db.lock", std::ofstream::out | std::ofstream::app);
|
|
|
|
+ long pos = lockfile.tellp();
|
|
|
|
+ if (pos == 0) {
|
|
|
|
+ lockfile << "OK.";
|
|
|
|
+ lockfile.flush();
|
|
|
|
+ lockfile.close();
|
|
|
|
+ } else {
|
|
|
|
+ if (get_logger()) {
|
|
|
|
+ get_logger() << "db.lock file exists. Skipping maint." << std::endl;
|
|
|
|
+ }
|
|
|
|
+ lockfile.close();
|
|
|
|
+ return;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ std::map<month_user, month_stats> monthly;
|
|
|
|
+
|
|
|
|
+ // Ok, do maint things here
|
|
|
|
+ SQLite::Statement stmt(
|
|
|
|
+ db, "SELECT `date`,username,SUM(score),SUM(won) FROM scores "
|
|
|
|
+ "WHERE `date` < ? GROUP BY `date`,username ORDER BY "
|
|
|
|
+ "`date`,SUM(score) DESC;");
|
|
|
|
+
|
|
|
|
+ try {
|
|
|
|
+ stmt.bind(1, (long)month_first_t);
|
|
|
|
+
|
|
|
|
+ while (stmt.executeStep()) {
|
|
|
|
+ // get time_t, conver to time_point, find first of month, convert back to
|
|
|
|
+ // time_t
|
|
|
|
+ std::chrono::_V2::system_clock::time_point date;
|
|
|
|
+
|
|
|
|
+ date = std::chrono::system_clock::from_time_t((long)stmt.getColumn(0));
|
|
|
|
+ firstOfMonthDate(date);
|
|
|
|
+ time_t date_t = std::chrono::system_clock::to_time_t(date);
|
|
|
|
+
|
|
|
|
+ month_user mu;
|
|
|
|
+
|
|
|
|
+ mu.date = date_t;
|
|
|
|
+ mu.username = (const char *)stmt.getColumn(1);
|
|
|
|
+
|
|
|
|
+ auto map_iter = monthly.find(mu);
|
|
|
|
+ if (map_iter == monthly.end()) {
|
|
|
|
+ // not found
|
|
|
|
+ month_stats ms;
|
|
|
|
+ ms.days = 1;
|
|
|
|
+ ms.score = stmt.getColumn(2);
|
|
|
|
+ ms.hands_won = stmt.getColumn(3);
|
|
|
|
+ monthly[mu] = ms;
|
|
|
|
+ } else {
|
|
|
|
+ // was found
|
|
|
|
+ month_stats &ms = monthly[mu];
|
|
|
|
+ ms.days++;
|
|
|
|
+ ms.score += (int)stmt.getColumn(2);
|
|
|
|
+ ms.hands_won += (int)stmt.getColumn(3);
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ } catch (std::exception &e) {
|
|
|
|
+ if (get_logger) {
|
|
|
|
+ get_logger() << "expireScores() SELECT failed" << std::endl;
|
|
|
|
+ get_logger() << "SQLite exception: " << e.what() << std::endl;
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ // Ok! I have the data that I need!
|
|
|
|
+ if (monthly.empty()) {
|
|
|
|
+ std::remove("db.lock");
|
|
|
|
+ return;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ try {
|
|
|
|
+
|
|
|
|
+ SQLite::Transaction transaction(db);
|
|
|
|
+
|
|
|
|
+ // If we fail for any reason within here -- we should be safe.
|
|
|
|
+ SQLite::Statement stmt_delete(db, "DELETE FROM scores WHERE `date` < ?;");
|
|
|
|
+
|
|
|
|
+ stmt_delete.bind(1, (long)month_first_t);
|
|
|
|
+ stmt_delete.exec();
|
|
|
|
+
|
|
|
|
+ /*
|
|
|
|
+ if (get_logger) {
|
|
|
|
+ get_logger() << "Ok, deleted records < " << month_first_t <<
|
|
|
|
+ std::endl;
|
|
|
|
+ }
|
|
|
|
+ */
|
|
|
|
+
|
|
|
|
+ SQLite::Statement stmt_insert(db,
|
|
|
|
+ "INSERT INTO monthly(month, username, days, "
|
|
|
|
+ "hands_won, score) VALUES(?,?,?,?,?);");
|
|
|
|
+ for (auto key : monthly) {
|
|
|
|
+ /*
|
|
|
|
+ if (get_logger) {
|
|
|
|
+ get_logger() << key.first.date << ":" << key.first.username << " "
|
|
|
|
+ << key.second.score << std::endl;
|
|
|
|
+ }
|
|
|
|
+ */
|
|
|
|
+ stmt_insert.bind(1, (long)key.first.date);
|
|
|
|
+ stmt_insert.bind(2, key.first.username.c_str());
|
|
|
|
+ stmt_insert.bind(3, key.second.days);
|
|
|
|
+ stmt_insert.bind(4, key.second.hands_won);
|
|
|
|
+ stmt_insert.bind(5, key.second.score);
|
|
|
|
+ stmt_insert.exec();
|
|
|
|
+ stmt_insert.reset();
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ transaction.commit();
|
|
|
|
+
|
|
|
|
+ } catch (std::exception &e) {
|
|
|
|
+ if (get_logger) {
|
|
|
|
+ get_logger() << "expireScores() DELETE/INSERTs failed" << std::endl;
|
|
|
|
+ get_logger() << "SQLite exception: " << e.what() << std::endl;
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ if (get_logger) {
|
|
|
|
+ for (auto key : monthly) {
|
|
|
|
+ get_logger() << key.first.date << ":" << key.first.username << " "
|
|
|
|
+ << key.second.days << ", " << key.second.hands_won << ", "
|
|
|
|
+ << key.second.score << ::std::endl;
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ // clean up
|
|
|
|
+
|
|
|
|
+ std::remove("db.lock");
|
|
|
|
+}
|
|
|
|
|
|
/**
|
|
/**
|
|
* @brief Format date to string.
|
|
* @brief Format date to string.
|