aboutsummaryrefslogtreecommitdiffstatshomepage
path: root/src/storage_sqlite.c
diff options
context:
space:
mode:
authorEgor Tensin <Egor.Tensin@gmail.com>2023-04-11 02:29:57 +0200
committerEgor Tensin <Egor.Tensin@gmail.com>2023-04-11 02:29:57 +0200
commit85357ce9b82d595415f64c9ee3517627d158de2e (patch)
tree56e29008040ae17d9a1756b3df1a7f0455b031e2 /src/storage_sqlite.c
parentadd .editorconfig (diff)
downloadcimple-85357ce9b82d595415f64c9ee3517627d158de2e.tar.gz
cimple-85357ce9b82d595415f64c9ee3517627d158de2e.zip
actually create some tables in SQLite database
Diffstat (limited to 'src/storage_sqlite.c')
-rw-r--r--src/storage_sqlite.c135
1 files changed, 115 insertions, 20 deletions
diff --git a/src/storage_sqlite.c b/src/storage_sqlite.c
index b170e73..3f2c5c5 100644
--- a/src/storage_sqlite.c
+++ b/src/storage_sqlite.c
@@ -7,26 +7,16 @@
#include "storage_sqlite.h"
#include "log.h"
+#include "sqlite.h"
+#include "sqlite_sql.h"
#include "storage.h"
#include <sqlite3.h>
+#include <stdio.h>
#include <stdlib.h>
#include <string.h>
-#define sqlite_errno(var, fn) \
- do { \
- log_err("%s: %s\n", fn, sqlite3_errstr(var)); \
- var = -var; \
- } while (0)
-
-#define sqlite_errno_if(expr, fn) \
- do { \
- int CONCAT(ret, __LINE__) = expr; \
- if (CONCAT(ret, __LINE__)) \
- sqlite_errno(CONCAT(ret, __LINE__), fn); \
- } while (0)
-
int storage_settings_create_sqlite(struct storage_settings *settings, const char *path)
{
settings->sqlite.path = strdup(path);
@@ -48,27 +38,132 @@ struct storage_sqlite {
sqlite3 *db;
};
+static int storage_upgrade_sqlite_to(struct storage_sqlite *storage, size_t version)
+{
+ static const char *const FMT = "%s PRAGMA user_version = %zu;";
+
+ const char *script;
+ char *full_script;
+ size_t nb;
+ int ret = 0;
+
+ script = sql_sqlite_files[version];
+
+ ret = snprintf(NULL, 0, FMT, script, version + 1);
+ nb = (size_t)ret + 1;
+ ret = 0;
+
+ full_script = malloc(nb);
+ if (!full_script) {
+ log_errno("malloc");
+ return -1;
+ }
+ snprintf(full_script, nb, FMT, script, version + 1);
+
+ ret = sqlite_exec_as_transaction(storage->db, full_script);
+ goto free;
+
+free:
+ free(full_script);
+
+ return ret;
+}
+
+static int storage_upgrade_sqlite_from_to(struct storage_sqlite *storage, size_t from, size_t to)
+{
+ int ret = 0;
+
+ for (size_t i = from; i < to; ++i) {
+ log("Upgrading SQLite database from version %zu to version %zu\n", i, i + 1);
+ ret = storage_upgrade_sqlite_to(storage, i);
+ if (ret < 0) {
+ log_err("Failed to upgrade to version %zu\n", i + 1);
+ return ret;
+ }
+ }
+
+ return ret;
+}
+
+static int storage_upgrade_sqlite(struct storage_sqlite *storage)
+{
+ size_t newest_version;
+ unsigned int current_version;
+ int ret = 0;
+
+ ret = sqlite_get_user_version(storage->db, &current_version);
+ if (ret < 0)
+ return ret;
+ log("SQLite database version: %u\n", current_version);
+
+ newest_version = sizeof(sql_sqlite_files) / sizeof(sql_sqlite_files[0]);
+ log("Newest database version: %zu\n", newest_version);
+
+ if (current_version > newest_version) {
+ log_err("Unknown database version: %u\n", current_version);
+ return -1;
+ }
+
+ if (current_version == newest_version) {
+ log("SQLite database already at the newest version\n");
+ return 0;
+ }
+
+ return storage_upgrade_sqlite_from_to(storage, current_version, newest_version);
+}
+
+static int storage_prepare_sqlite(struct storage_sqlite *storage)
+{
+ int ret = 0;
+
+ ret = sqlite_set_foreign_keys(storage->db);
+ if (ret < 0)
+ return ret;
+
+ ret = storage_upgrade_sqlite(storage);
+ if (ret < 0)
+ return ret;
+
+ return ret;
+}
+
int storage_create_sqlite(struct storage *storage, const struct storage_settings *settings)
{
int ret = 0;
+ log("Using SQLite database at %s\n", settings->sqlite.path);
+
storage->sqlite = malloc(sizeof(storage->sqlite));
if (!storage->sqlite) {
log_errno("malloc");
return -1;
}
- ret = sqlite3_open(settings->sqlite.path, &storage->sqlite->db);
- if (ret) {
- sqlite_errno(ret, "sqlite3_open");
- return ret;
- }
+ ret = sqlite_init();
+ if (ret < 0)
+ goto free;
+ ret = sqlite_open_rw(settings->sqlite.path, &storage->sqlite->db);
+ if (ret < 0)
+ goto destroy;
+ ret = storage_prepare_sqlite(storage->sqlite);
+ if (ret < 0)
+ goto close;
- return 0;
+ return ret;
+
+close:
+ sqlite_close(storage->sqlite->db);
+destroy:
+ sqlite_destroy();
+free:
+ free(storage->sqlite);
+
+ return ret;
}
void storage_destroy_sqlite(struct storage *storage)
{
- sqlite_errno_if(sqlite3_close(storage->sqlite->db), "sqlite3_close");
+ sqlite_close(storage->sqlite->db);
+ sqlite_destroy();
free(storage->sqlite);
}