Factor code to prepare SQLite statements into a separate function.
[cumulus.git] / localdb.cc
index 27fec5a..951f35c 100644 (file)
 
 using std::string;
 
-void LocalDb::Open(const char *path, const char *snapshot_name)
+/* Helper function to prepare a statement for execution in the current
+ * database. */
+sqlite3_stmt *LocalDb::Prepare(const char *sql)
 {
+    sqlite3_stmt *stmt;
     int rc;
+    const char *tail;
+
+    rc = sqlite3_prepare(db, sql, strlen(sql), &stmt, &tail);
+    if (rc != SQLITE_OK) {
+        throw IOException(string("Error preparing statement: ") + sql);
+    }
 
-    snapshot = snapshot_name;
+    return stmt;
+}
+
+void LocalDb::Open(const char *path, const char *snapshot_name)
+{
+    int rc;
 
     rc = sqlite3_open(path, &db);
     if (rc) {
@@ -41,6 +55,26 @@ void LocalDb::Open(const char *path, const char *snapshot_name)
         sqlite3_close(db);
         throw IOException("Error starting transaction");
     }
+
+    /* Insert this snapshot into the database, and determine the integer key
+     * which will be used to identify it. */
+    sqlite3_stmt *stmt = Prepare("insert into snapshots(name, timestamp) "
+                                 "values (?, julianday('now'))");
+    sqlite3_bind_text(stmt, 1, snapshot_name, strlen(snapshot_name),
+                      SQLITE_TRANSIENT);
+
+    rc = sqlite3_step(stmt);
+    if (rc != SQLITE_DONE) {
+        sqlite3_close(db);
+        throw IOException("Database execution error!");
+    }
+
+    snapshotid = sqlite3_last_insert_rowid(db);
+    sqlite3_finalize(stmt);
+    if (snapshotid == 0) {
+        sqlite3_close(db);
+        throw IOException("Find snapshot id");
+    }
 }
 
 void LocalDb::Close()
@@ -53,23 +87,73 @@ void LocalDb::Close()
     sqlite3_close(db);
 }
 
+int64_t LocalDb::SegmentToId(const string &segment)
+{
+    int rc;
+    sqlite3_stmt *stmt;
+    int64_t result;
+
+    stmt = Prepare("insert or ignore into segments(segment) values (?)");
+    sqlite3_bind_text(stmt, 1, segment.c_str(), segment.size(),
+                      SQLITE_TRANSIENT);
+    rc = sqlite3_step(stmt);
+    if (rc != SQLITE_DONE) {
+        throw IOException("Could not execute INSERT statement!");
+    }
+    sqlite3_finalize(stmt);
+
+    stmt = Prepare("select segmentid from segments where segment = ?");
+    sqlite3_bind_text(stmt, 1, segment.c_str(), segment.size(),
+                      SQLITE_TRANSIENT);
+
+    rc = sqlite3_step(stmt);
+    if (rc == SQLITE_DONE) {
+        throw IOException("No segment found by id");
+    } else if (rc == SQLITE_ROW) {
+        result = sqlite3_column_int64(stmt, 0);
+    } else {
+        throw IOException("Error executing find segment by id query");
+    }
+
+    sqlite3_finalize(stmt);
+
+    return result;
+}
+
+string LocalDb::IdToSegment(int64_t segmentid)
+{
+    int rc;
+    sqlite3_stmt *stmt;
+    string result;
+
+    stmt = Prepare("select segment from segments where segmentid = ?");
+    sqlite3_bind_int64(stmt, 1, segmentid);
+
+    rc = sqlite3_step(stmt);
+    if (rc == SQLITE_DONE) {
+        throw IOException("No segment found by id");
+    } else if (rc == SQLITE_ROW) {
+        result = (const char *)sqlite3_column_text(stmt, 0);
+    } else {
+        throw IOException("Error executing find segment by id query");
+    }
+
+    sqlite3_finalize(stmt);
+
+    return result;
+}
+
 void LocalDb::StoreObject(const ObjectReference& ref,
                           const string &checksum, int64_t size)
 {
     int rc;
     sqlite3_stmt *stmt;
-    static const char s[] =
-        "insert into block_index(segment, object, checksum, size) "
-        "values (?, ?, ?, ?)";
-    const char *tail;
 
-    rc = sqlite3_prepare_v2(db, s, strlen(s), &stmt, &tail);
-    if (rc != SQLITE_OK) {
-        return;
-    }
+    stmt = Prepare("insert into "
+                   "block_index(segmentid, object, checksum, size, timestamp) "
+                   "values (?, ?, ?, ?, julianday('now'))");
 
-    string seg = ref.get_segment();
-    sqlite3_bind_text(stmt, 1, seg.c_str(), seg.size(), SQLITE_TRANSIENT);
+    sqlite3_bind_int64(stmt, 1, SegmentToId(ref.get_segment()));
     string obj = ref.get_sequence();
     sqlite3_bind_text(stmt, 2, obj.c_str(), obj.size(), SQLITE_TRANSIENT);
     sqlite3_bind_text(stmt, 3, checksum.c_str(), checksum.size(),
@@ -88,18 +172,10 @@ ObjectReference LocalDb::FindObject(const string &checksum, int64_t size)
 {
     int rc;
     sqlite3_stmt *stmt;
-    static const char s[] =
-        "select segment, object from block_index "
-        "where checksum = ? and size = ?";
-    const char *tail;
-
     ObjectReference ref;
 
-    rc = sqlite3_prepare_v2(db, s, strlen(s), &stmt, &tail);
-    if (rc != SQLITE_OK) {
-        return ref;
-    }
-
+    stmt = Prepare("select segmentid, object from block_index "
+                   "where checksum = ? and size = ? and expired is null");
     sqlite3_bind_text(stmt, 1, checksum.c_str(), checksum.size(),
                       SQLITE_TRANSIENT);
     sqlite3_bind_int64(stmt, 2, size);
@@ -107,7 +183,7 @@ ObjectReference LocalDb::FindObject(const string &checksum, int64_t size)
     rc = sqlite3_step(stmt);
     if (rc == SQLITE_DONE) {
     } else if (rc == SQLITE_ROW) {
-        ref = ObjectReference((const char *)sqlite3_column_text(stmt, 0),
+        ref = ObjectReference(IdToSegment(sqlite3_column_int64(stmt, 0)),
                               (const char *)sqlite3_column_text(stmt, 1));
     } else {
         fprintf(stderr, "Could not execute SELECT statement!\n");
@@ -118,25 +194,42 @@ ObjectReference LocalDb::FindObject(const string &checksum, int64_t size)
     return ref;
 }
 
-void LocalDb::UseObject(const ObjectReference& ref)
+bool LocalDb::IsOldObject(const string &checksum, int64_t size)
 {
     int rc;
     sqlite3_stmt *stmt;
-    static const char s[] =
-        "insert into snapshot_contents "
-        "select blockid, ? as snapshot from block_index "
-        "where segment = ? and object = ?";
-    const char *tail;
+    bool found = false;
 
-    rc = sqlite3_prepare_v2(db, s, strlen(s), &stmt, &tail);
-    if (rc != SQLITE_OK) {
-        return;
+    stmt = Prepare("select segmentid, object from block_index "
+                   "where checksum = ? and size = ?");
+    sqlite3_bind_text(stmt, 1, checksum.c_str(), checksum.size(),
+                      SQLITE_TRANSIENT);
+    sqlite3_bind_int64(stmt, 2, size);
+
+    rc = sqlite3_step(stmt);
+    if (rc == SQLITE_DONE) {
+        found = false;
+    } else if (rc == SQLITE_ROW) {
+        found = true;
+    } else {
+        fprintf(stderr, "Could not execute SELECT statement!\n");
     }
 
-    sqlite3_bind_text(stmt, 1, snapshot.c_str(), snapshot.size(),
-                      SQLITE_TRANSIENT);
-    string seg = ref.get_segment();
-    sqlite3_bind_text(stmt, 2, seg.c_str(), seg.size(), SQLITE_TRANSIENT);
+    sqlite3_finalize(stmt);
+
+    return found;
+}
+
+void LocalDb::UseObject(const ObjectReference& ref)
+{
+    int rc;
+    sqlite3_stmt *stmt;
+
+    stmt = Prepare("insert or ignore into snapshot_contents "
+                   "select blockid, ? as snapshotid from block_index "
+                   "where segmentid = ? and object = ?");
+    sqlite3_bind_int64(stmt, 1, snapshotid);
+    sqlite3_bind_int64(stmt, 2, SegmentToId(ref.get_segment()));
     string obj = ref.get_sequence();
     sqlite3_bind_text(stmt, 3, obj.c_str(), obj.size(), SQLITE_TRANSIENT);