فهرست منبع

Added maint/expireScores < 1st of month.

Steve Thielemann 3 سال پیش
والد
کامیت
4d16ebc89b
3فایلهای تغییر یافته به همراه172 افزوده شده و 6 حذف شده
  1. 157 1
      db.cpp
  2. 1 1
      db.h
  3. 14 4
      main.cpp

+ 157 - 1
db.cpp

@@ -461,13 +461,169 @@ retry:
   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
  *
  * @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.

+ 1 - 1
db.h

@@ -55,7 +55,7 @@ public:
   std::vector<scores_details> getScoresOnDay(time_t date);
   std::map<time_t, std::vector<scores_data>> getScores(void);
   std::map<time_t, int> getPlayed(void);
-  void expireScores(void);
+  void expireScores(time_t month_first_t);
 
   int handsPlayedOnDay(time_t day);
   std::map<time_t, int> whenPlayed(void);

+ 14 - 4
main.cpp

@@ -451,16 +451,26 @@ int main(int argc, char *argv[]) {
   // retrieve lastcall
   time_t last_call = std::stol(spacedb.getSetting("LastCall", "0"));
 
+  std::chrono::_V2::system_clock::time_point now =
+      std::chrono::system_clock::now();
+
   // store now as lastcall
-  time_t now =
-      std::chrono::system_clock::to_time_t(std::chrono::system_clock::now());
+  time_t now_t = std::chrono::system_clock::to_time_t(now);
+
+  spacedb.setSetting("LastCall", std::to_string(now_t));
 
-  spacedb.setSetting("LastCall", std::to_string(now));
+  // run maint
+  {
+    std::chrono::_V2::system_clock::time_point maint_date = now;
+    firstOfMonthDate(maint_date);
+
+    spacedb.expireScores(std::chrono::system_clock::to_time_t(maint_date));
+  }
 
   // Have they used this door before?
   if (last_call != 0) {
     door << "Welcome Back!" << door::nl;
-    auto nowClock = std::chrono::system_clock::from_time_t(now);
+    auto nowClock = std::chrono::system_clock::from_time_t(now_t);
     auto lastClock = std::chrono::system_clock::from_time_t(last_call);
     auto delta = nowClock - lastClock;