aboutsummaryrefslogtreecommitdiffstats
path: root/src/libstat
diff options
context:
space:
mode:
authorVsevolod Stakhov <vsevolod@highsecure.ru>2015-06-19 17:00:24 +0100
committerVsevolod Stakhov <vsevolod@highsecure.ru>2015-06-19 17:00:24 +0100
commit29c56c31b68a3c55a0b77689b7a3ee141549e485 (patch)
treeb503d9f73f912a438b1e4a415a9f221d4dad3542 /src/libstat
parentd7d83b570be42d732083f1b5839760b78979b7cf (diff)
parent03926b5291fda6282f463fd962ccd74e16ef791d (diff)
downloadrspamd-29c56c31b68a3c55a0b77689b7a3ee141549e485.tar.gz
rspamd-29c56c31b68a3c55a0b77689b7a3ee141549e485.zip
Merge branch 'sqlite-stat'
Diffstat (limited to 'src/libstat')
-rw-r--r--src/libstat/CMakeLists.txt3
-rw-r--r--src/libstat/backends/backends.h68
-rw-r--r--src/libstat/backends/mmaped_file.c47
-rw-r--r--src/libstat/backends/sqlite3_backend.c591
-rw-r--r--src/libstat/stat_api.h5
-rw-r--r--src/libstat/stat_config.c43
6 files changed, 683 insertions, 74 deletions
diff --git a/src/libstat/CMakeLists.txt b/src/libstat/CMakeLists.txt
index f31cf4cf4..2fa16cf98 100644
--- a/src/libstat/CMakeLists.txt
+++ b/src/libstat/CMakeLists.txt
@@ -7,7 +7,8 @@ SET(TOKENIZERSSRC ${CMAKE_CURRENT_SOURCE_DIR}/tokenizers/tokenizers.c
SET(CLASSIFIERSSRC ${CMAKE_CURRENT_SOURCE_DIR}/classifiers/bayes.c)
-SET(BACKENDSSRC ${CMAKE_CURRENT_SOURCE_DIR}/backends/mmaped_file.c)
+SET(BACKENDSSRC ${CMAKE_CURRENT_SOURCE_DIR}/backends/mmaped_file.c
+ ${CMAKE_CURRENT_SOURCE_DIR}/backends/sqlite3_backend.c)
IF(ENABLE_HIREDIS MATCHES "ON")
SET(BACKENDSSRC ${BACKENDSSRC}
${CMAKE_CURRENT_SOURCE_DIR}/backends/redis.c)
diff --git a/src/libstat/backends/backends.h b/src/libstat/backends/backends.h
index b6cc49f9e..18f4fce26 100644
--- a/src/libstat/backends/backends.h
+++ b/src/libstat/backends/backends.h
@@ -54,49 +54,37 @@ struct rspamd_stat_backend {
gulong (*inc_learns)(struct rspamd_statfile_runtime *runtime, gpointer ctx);
gulong (*dec_learns)(struct rspamd_statfile_runtime *runtime, gpointer ctx);
ucl_object_t* (*get_stat)(struct rspamd_statfile_runtime *runtime, gpointer ctx);
+ void (*close)(gpointer ctx);
gpointer ctx;
};
-gpointer rspamd_mmaped_file_init (struct rspamd_stat_ctx *ctx, struct rspamd_config *cfg);
-gpointer rspamd_mmaped_file_runtime (struct rspamd_task *task,
- struct rspamd_statfile_config *stcf,
- gboolean learn, gpointer ctx);
-gboolean rspamd_mmaped_file_process_token (struct token_node_s *tok,
- struct rspamd_token_result *res,
- gpointer ctx);
-gboolean rspamd_mmaped_file_learn_token (struct token_node_s *tok,
- struct rspamd_token_result *res,
- gpointer ctx);
-void rspamd_mmaped_file_finalize_learn (struct rspamd_statfile_runtime *runtime,
- gpointer ctx);
-gulong rspamd_mmaped_file_total_learns (struct rspamd_statfile_runtime *runtime,
- gpointer ctx);
-gulong rspamd_mmaped_file_inc_learns (struct rspamd_statfile_runtime *runtime,
- gpointer ctx);
-gulong rspamd_mmaped_file_dec_learns (struct rspamd_statfile_runtime *runtime,
- gpointer ctx);
-ucl_object_t * rspamd_mmaped_file_get_stat (struct rspamd_statfile_runtime *runtime,
- gpointer ctx);
+#define RSPAMD_STAT_BACKEND_DEF(name) \
+ gpointer rspamd_##name##_init (struct rspamd_stat_ctx *ctx, struct rspamd_config *cfg); \
+ gpointer rspamd_##name##_runtime (struct rspamd_task *task, \
+ struct rspamd_statfile_config *stcf, \
+ gboolean learn, gpointer ctx); \
+ gboolean rspamd_##name##_process_token (struct token_node_s *tok, \
+ struct rspamd_token_result *res, \
+ gpointer ctx); \
+ gboolean rspamd_##name##_learn_token (struct token_node_s *tok, \
+ struct rspamd_token_result *res, \
+ gpointer ctx); \
+ void rspamd_##name##_finalize_learn (struct rspamd_statfile_runtime *runtime, \
+ gpointer ctx); \
+ gulong rspamd_##name##_total_learns (struct rspamd_statfile_runtime *runtime, \
+ gpointer ctx); \
+ gulong rspamd_##name##_inc_learns (struct rspamd_statfile_runtime *runtime, \
+ gpointer ctx); \
+ gulong rspamd_##name##_dec_learns (struct rspamd_statfile_runtime *runtime, \
+ gpointer ctx); \
+ gulong rspamd_##name##_learns (struct rspamd_statfile_runtime *runtime, \
+ gpointer ctx); \
+ ucl_object_t * rspamd_##name##_get_stat (struct rspamd_statfile_runtime *runtime, \
+ gpointer ctx); \
+ void rspamd_##name##_close (gpointer ctx)
-gpointer rspamd_redis_init (struct rspamd_stat_ctx *ctx, struct rspamd_config *cfg);
-gpointer rspamd_redis_runtime (struct rspamd_task *task,
- struct rspamd_statfile_config *stcf,
- gboolean learn, gpointer ctx);
-gboolean rspamd_redis_process_token (struct token_node_s *tok,
- struct rspamd_token_result *res,
- gpointer ctx);
-gboolean rspamd_redis_learn_token (struct token_node_s *tok,
- struct rspamd_token_result *res,
- gpointer ctx);
-void rspamd_redis_finalize_learn (struct rspamd_statfile_runtime *runtime,
- gpointer ctx);
-gulong rspamd_redis_total_learns (struct rspamd_statfile_runtime *runtime,
- gpointer ctx);
-gulong rspamd_redis_inc_learns (struct rspamd_statfile_runtime *runtime,
- gpointer ctx);
-gulong rspamd_redis_learns (struct rspamd_statfile_runtime *runtime,
- gpointer ctx);
-ucl_object_t * rspamd_redis_get_stat (struct rspamd_statfile_runtime *runtime,
- gpointer ctx);
+RSPAMD_STAT_BACKEND_DEF(mmaped_file);
+RSPAMD_STAT_BACKEND_DEF(redis);
+RSPAMD_STAT_BACKEND_DEF(sqlite3);
#endif /* BACKENDS_H_ */
diff --git a/src/libstat/backends/mmaped_file.c b/src/libstat/backends/mmaped_file.c
index 6ee282b78..20bce6212 100644
--- a/src/libstat/backends/mmaped_file.c
+++ b/src/libstat/backends/mmaped_file.c
@@ -638,7 +638,7 @@ rspamd_mmaped_file_open (rspamd_mmaped_file_ctx * pool,
}
gint
-rspamd_mmaped_file_close (rspamd_mmaped_file_ctx * pool,
+rspamd_mmaped_file_close_file (rspamd_mmaped_file_ctx * pool,
rspamd_mmaped_file_t * file)
{
rspamd_mmaped_file_t *pos;
@@ -657,8 +657,6 @@ rspamd_mmaped_file_close (rspamd_mmaped_file_ctx * pool,
close (file->fd);
}
- g_hash_table_remove (pool->files, file->cf);
-
g_slice_free1 (sizeof (*file), file);
return 0;
@@ -797,23 +795,6 @@ rspamd_mmaped_file_create (rspamd_mmaped_file_ctx * pool, const gchar *filename,
return 0;
}
-void
-rspamd_mmaped_file_destroy (rspamd_mmaped_file_ctx * pool)
-{
- GHashTableIter it;
- gpointer k, v;
- rspamd_mmaped_file_t *f;
-
- g_hash_table_iter_init (&it, pool->files);
- while (g_hash_table_iter_next (&it, &k, &v)) {
- f = (rspamd_mmaped_file_t *)v;
- rspamd_mmaped_file_close (pool, f);
- }
-
- g_hash_table_destroy (pool->files);
- rspamd_mempool_delete (pool->pool);
-}
-
gpointer
rspamd_mmaped_file_init (struct rspamd_stat_ctx *ctx, struct rspamd_config *cfg)
{
@@ -826,6 +807,7 @@ rspamd_mmaped_file_init (struct rspamd_stat_ctx *ctx, struct rspamd_config *cfg)
gsize size;
new = rspamd_mempool_alloc0 (cfg->cfg_pool, sizeof (rspamd_mmaped_file_ctx));
+ new->pool = rspamd_mempool_new (rspamd_mempool_suggest_size ());
new->lock = rspamd_mempool_get_mutex (new->pool);
new->mlock_ok = cfg->mlock_statfile_pool;
new->files = g_hash_table_new (g_direct_hash, g_direct_equal);
@@ -844,7 +826,7 @@ rspamd_mmaped_file_init (struct rspamd_stat_ctx *ctx, struct rspamd_config *cfg)
* By default, all statfiles are treated as mmaped files
*/
if (stf->backend == NULL ||
- strcmp (stf->backend, MMAPED_BACKEND_TYPE)) {
+ strcmp (stf->backend, MMAPED_BACKEND_TYPE) == 0) {
/*
* Check configuration sanity
*/
@@ -883,6 +865,29 @@ rspamd_mmaped_file_init (struct rspamd_stat_ctx *ctx, struct rspamd_config *cfg)
return (gpointer)new;
}
+void
+rspamd_mmaped_file_close (gpointer p)
+{
+ rspamd_mmaped_file_ctx *ctx = (rspamd_mmaped_file_ctx *)p;
+ GHashTableIter it;
+ gpointer k, v;
+ rspamd_mmaped_file_t *mf;
+
+ g_assert (ctx != NULL);
+
+ rspamd_mempool_lock_mutex (ctx->lock);
+ g_hash_table_iter_init (&it, ctx->files);
+
+ while (g_hash_table_iter_next (&it, &k, &v)) {
+ mf = v;
+ rspamd_mmaped_file_close_file (ctx, mf);
+ }
+
+ g_hash_table_unref (ctx->files);
+ rspamd_mempool_unlock_mutex (ctx->lock);
+ /* XXX: we don't delete pool here to avoid deadlocks */
+}
+
gpointer
rspamd_mmaped_file_runtime (struct rspamd_task *task,
struct rspamd_statfile_config *stcf,
diff --git a/src/libstat/backends/sqlite3_backend.c b/src/libstat/backends/sqlite3_backend.c
new file mode 100644
index 000000000..21b5918ac
--- /dev/null
+++ b/src/libstat/backends/sqlite3_backend.c
@@ -0,0 +1,591 @@
+/*
+ * Copyright (c) 2015, Vsevolod Stakhov
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY AUTHOR ''AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL AUTHOR BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
+ * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#include "config.h"
+#include "stat_internal.h"
+#include "main.h"
+#include "sqlite3.h"
+
+#define SQLITE3_BACKEND_TYPE "sqlite3"
+#define SQLITE3_SCHEMA_VERSION "1"
+#define SQLITE3_DEFAULT "default"
+
+struct rspamd_sqlite3_prstmt;
+
+struct rspamd_stat_sqlite3_db {
+ sqlite3 *sqlite;
+ struct rspamd_sqlite3_prstmt *prstmt;
+ gboolean in_transaction;
+};
+
+struct rspamd_stat_sqlite3_ctx {
+ GHashTable *files;
+};
+
+static const char *create_tables_sql =
+ "BEGIN;"
+ "CREATE TABLE users("
+ "id INTEGER PRIMARY KEY,"
+ "name TEXT,"
+ "learns INTEGER"
+ ");"
+ "CREATE TABLE languages("
+ "id INTEGER PRIMARY KEY,"
+ "name TEXT,"
+ "learns INTEGER"
+ ");"
+ "CREATE TABLE tokens("
+ "token INTEGER PRIMARY KEY,"
+ "user INTEGER REFERENCES users(id) ON DELETE CASCADE,"
+ "language INTEGER REFERENCES languages(id) ON DELETE CASCADE,"
+ "value INTEGER"
+ ");"
+ "CREATE UNIQUE INDEX IF NOT EXISTS un ON users(name);"
+ "CREATE UNIQUE INDEX IF NOT EXISTS ln ON languages(name);"
+ "PRAGMA user_version=" SQLITE3_SCHEMA_VERSION ";"
+ "INSERT INTO users(id, name, learns) VALUES(0, '" SQLITE3_DEFAULT "',0);"
+ "INSERT INTO languages(id, name, learns) VALUES(0, '" SQLITE3_DEFAULT "',0);"
+ "COMMIT;";
+
+enum rspamd_stat_sqlite3_stmt_idx {
+ RSPAMD_STAT_BACKEND_TRANSACTION_START = 0,
+ RSPAMD_STAT_BACKEND_TRANSACTION_COMMIT,
+ RSPAMD_STAT_BACKEND_TRANSACTION_ROLLBACK,
+ RSPAMD_STAT_BACKEND_GET_TOKEN,
+ RSPAMD_STAT_BACKEND_SET_TOKEN,
+ RSPAMD_STAT_BACKEND_INC_LEARNS,
+ RSPAMD_STAT_BACKEND_DEC_LEARNS,
+ RSPAMD_STAT_BACKEND_GET_LEARNS,
+ RSPAMD_STAT_BACKEND_MAX
+};
+
+static struct rspamd_sqlite3_prstmt {
+ enum rspamd_stat_sqlite3_stmt_idx idx;
+ const gchar *sql;
+ const gchar *args;
+ sqlite3_stmt *stmt;
+ gint result;
+ const gchar *ret;
+} prepared_stmts[RSPAMD_STAT_BACKEND_MAX] =
+{
+ {
+ .idx = RSPAMD_STAT_BACKEND_TRANSACTION_START,
+ .sql = "BEGIN TRANSACTION;",
+ .args = "",
+ .stmt = NULL,
+ .result = SQLITE_DONE,
+ .ret = ""
+ },
+ {
+ .idx = RSPAMD_STAT_BACKEND_TRANSACTION_COMMIT,
+ .sql = "COMMIT;",
+ .args = "",
+ .stmt = NULL,
+ .result = SQLITE_DONE,
+ .ret = ""
+ },
+ {
+ .idx = RSPAMD_STAT_BACKEND_TRANSACTION_ROLLBACK,
+ .sql = "ROLLBACK;",
+ .args = "",
+ .stmt = NULL,
+ .result = SQLITE_DONE,
+ .ret = ""
+ },
+ {
+ .idx = RSPAMD_STAT_BACKEND_GET_TOKEN,
+ .sql = "SELECT value FROM tokens "
+ "LEFT JOIN languages ON tokens.language=languages.id "
+ "LEFT JOIN users ON tokens.user=users.id "
+ "WHERE token=?1 AND users.name=?2 AND languages.name=?3;",
+ .stmt = NULL,
+ .args = "ITT",
+ .result = SQLITE_ROW,
+ .ret = "I"
+ },
+ {
+ .idx = RSPAMD_STAT_BACKEND_SET_TOKEN,
+ .sql = "INSERT OR REPLACE INTO tokens(token, user, language, value)"
+ "VALUES (?1, ?2, ?3, ?4);",
+ .stmt = NULL,
+ .args = "IIII",
+ .result = SQLITE_DONE,
+ .ret = ""
+ },
+ {
+ .idx = RSPAMD_STAT_BACKEND_INC_LEARNS,
+ .sql = "UPDATE languages SET learns=learns + 1 WHERE name=?1;"
+ "UPDATE users SET learns=learns + 1 WHERE name=?2;",
+ .stmt = NULL,
+ .args = "TT",
+ .result = SQLITE_DONE,
+ .ret = ""
+ },
+ {
+ .idx = RSPAMD_STAT_BACKEND_DEC_LEARNS,
+ .sql = "UPDATE languages SET learns=learns - 1 WHERE name=?1;"
+ "UPDATE users SET learns=learns - 1 WHERE name=?2;",
+ .stmt = NULL,
+ .args = "TT",
+ .result = SQLITE_DONE,
+ .ret = ""
+ },
+ {
+ .idx = RSPAMD_STAT_BACKEND_GET_LEARNS,
+ .sql = "SELECT sum(learns) FROM languages;",
+ .stmt = NULL,
+ .args = "",
+ .result = SQLITE_DONE,
+ .ret = "I"
+ }
+};
+
+static GQuark
+rspamd_sqlite3_quark (void)
+{
+ return g_quark_from_static_string ("sqlite3-stat-backend");
+}
+
+static gboolean
+rspamd_sqlite3_init_prstmt (struct rspamd_stat_sqlite3_db *db, GError **err)
+{
+ int i;
+
+ for (i = 0; i < RSPAMD_STAT_BACKEND_MAX; i ++) {
+ if (db->prstmt[i].stmt != NULL) {
+ /* Skip already prepared statements */
+ continue;
+ }
+ if (sqlite3_prepare_v2 (db->sqlite, db->prstmt[i].sql, -1,
+ &db->prstmt[i].stmt, NULL) != SQLITE_OK) {
+ g_set_error (err, rspamd_sqlite3_quark (),
+ -1, "Cannot initialize prepared sql `%s`: %s",
+ db->prstmt[i].sql, sqlite3_errmsg (db->sqlite));
+
+ return FALSE;
+ }
+ }
+
+ return TRUE;
+}
+
+static int
+rspamd_sqlite3_run_prstmt (struct rspamd_stat_sqlite3_db *db, int idx, ...)
+{
+ int retcode;
+ va_list ap;
+ sqlite3_stmt *stmt;
+ int i;
+ const char *argtypes;
+
+ if (idx < 0 || idx >= RSPAMD_STAT_BACKEND_MAX) {
+
+ return -1;
+ }
+
+ stmt = db->prstmt[idx].stmt;
+ if (stmt == NULL) {
+ if ((retcode = sqlite3_prepare_v2 (db->sqlite, db->prstmt[idx].sql, -1,
+ &db->prstmt[idx].stmt, NULL)) != SQLITE_OK) {
+ msg_err ("Cannot initialize prepared sql `%s`: %s",
+ db->prstmt[idx].sql, sqlite3_errmsg (db->sqlite));
+
+ return retcode;
+ }
+ stmt = db->prstmt[idx].stmt;
+ }
+
+ msg_debug ("executing `%s`", db->prstmt[idx].sql);
+ argtypes = db->prstmt[idx].args;
+ sqlite3_reset (stmt);
+ va_start (ap, idx);
+
+ for (i = 0; argtypes[i] != '\0'; i++) {
+ switch (argtypes[i]) {
+ case 'T':
+ sqlite3_bind_text (stmt, i + 1, va_arg (ap, const char*), -1,
+ SQLITE_STATIC);
+ break;
+ case 'I':
+ sqlite3_bind_int64 (stmt, i + 1, va_arg (ap, gint64));
+ break;
+ case 'S':
+ sqlite3_bind_int (stmt, i + 1, va_arg (ap, gint));
+ break;
+ }
+ }
+
+ va_end (ap);
+ retcode = sqlite3_step (stmt);
+
+ if (retcode == db->prstmt[idx].result) {
+ argtypes = db->prstmt[idx].ret;
+
+ for (i = 0; argtypes != NULL && argtypes[i] != '\0'; i ++) {
+ switch (argtypes[i]) {
+ case 'T':
+ *va_arg (ap, char**) = g_strdup (sqlite3_column_text (stmt, i));
+ break;
+ case 'I':
+ *va_arg (ap, gint64*) = sqlite3_column_int64 (stmt, i);
+ break;
+ case 'S':
+ *va_arg (ap, int*) = sqlite3_column_int (stmt, i);
+ break;
+ }
+ }
+
+ return SQLITE_OK;
+ }
+ else if (retcode != SQLITE_DONE) {
+ msg_debug ("failed to execute query %s: %d, %s", db->prstmt[idx].sql,
+ retcode, sqlite3_errmsg (db->sqlite));
+ }
+
+ return retcode;
+}
+
+static void
+rspamd_sqlite3_close_prstmt (struct rspamd_stat_sqlite3_db *db)
+{
+ int i;
+
+ for (i = 0; i < RSPAMD_STAT_BACKEND_MAX; i++) {
+ if (db->prstmt[i].stmt != NULL) {
+ sqlite3_finalize (db->prstmt[i].stmt);
+ db->prstmt[i].stmt = NULL;
+ }
+ }
+
+ return;
+}
+
+static struct rspamd_stat_sqlite3_db *
+rspamd_sqlite3_opendb (const gchar *path, const ucl_object_t *opts,
+ gboolean create, GError **err)
+{
+ struct rspamd_stat_sqlite3_db *bk;
+ sqlite3 *sqlite;
+ gint rc, flags;
+
+ flags = SQLITE_OPEN_READWRITE;
+
+ if (create) {
+ flags |= SQLITE_OPEN_CREATE;
+ }
+
+ if ((rc = sqlite3_open_v2 (path, &sqlite,
+ flags, NULL)) != SQLITE_OK) {
+ g_set_error (err, rspamd_sqlite3_quark (),
+ rc, "cannot open sqlite db %s: %d",
+ path, rc);
+
+ return NULL;
+ }
+
+ if (create) {
+ if (sqlite3_exec (sqlite, create_tables_sql, NULL, NULL, NULL) != SQLITE_OK) {
+ g_set_error (err, rspamd_sqlite3_quark (),
+ -1, "cannot execute create sql `%s`: %s",
+ create_tables_sql, sqlite3_errmsg (sqlite));
+ sqlite3_close (sqlite);
+
+ return NULL;
+ }
+ }
+
+ bk = g_slice_alloc0 (sizeof (*bk));
+ bk->sqlite = sqlite;
+ bk->prstmt = g_slice_alloc0 (sizeof (prepared_stmts));
+ memcpy (bk->prstmt, prepared_stmts, sizeof (prepared_stmts));
+
+ if (!rspamd_sqlite3_init_prstmt (bk, err)) {
+ return NULL;
+ }
+
+ return bk;
+}
+
+gpointer
+rspamd_sqlite3_init (struct rspamd_stat_ctx *ctx,
+ struct rspamd_config *cfg)
+{
+ struct rspamd_stat_sqlite3_ctx *new;
+ struct rspamd_classifier_config *clf;
+ struct rspamd_statfile_config *stf;
+ GList *cur, *curst;
+ const ucl_object_t *filenameo;
+ const gchar *filename;
+ struct rspamd_stat_sqlite3_db *bk;
+ GError *err = NULL;
+
+ new = rspamd_mempool_alloc0 (cfg->cfg_pool, sizeof (*new));
+ new->files = g_hash_table_new (g_direct_hash, g_direct_equal);
+
+ /* Iterate over all classifiers and load matching statfiles */
+ cur = cfg->classifiers;
+
+ while (cur) {
+ clf = cur->data;
+
+ curst = clf->statfiles;
+ while (curst) {
+ stf = curst->data;
+
+ if (stf->backend && strcmp (stf->backend, SQLITE3_BACKEND_TYPE) == 0) {
+ /*
+ * Check configuration sanity
+ */
+ filenameo = ucl_object_find_key (stf->opts, "filename");
+ if (filenameo == NULL || ucl_object_type (filenameo) != UCL_STRING) {
+ filenameo = ucl_object_find_key (stf->opts, "path");
+ if (filenameo == NULL || ucl_object_type (filenameo) != UCL_STRING) {
+ msg_err ("statfile %s has no filename defined", stf->symbol);
+ curst = curst->next;
+ continue;
+ }
+ }
+
+ filename = ucl_object_tostring (filenameo);
+
+ if ((bk = rspamd_sqlite3_opendb (filename, stf->opts, FALSE,
+ &err)) == NULL) {
+ msg_err ("cannot open sqlite3 db: %e", err);
+ g_error_free (err);
+ err = NULL;
+
+ bk = rspamd_sqlite3_opendb (filename, stf->opts, TRUE,
+ &err);
+ }
+
+ if (bk != NULL) {
+ g_hash_table_insert (new->files, stf, bk);
+ }
+ else {
+ msg_err ("cannot create sqlite3 db: %e", err);
+ g_error_free (err);
+ err = NULL;
+ }
+
+ ctx->statfiles ++;
+ }
+
+ curst = curst->next;
+ }
+
+ cur = g_list_next (cur);
+ }
+
+ return (gpointer)new;
+}
+
+void
+rspamd_sqlite3_close (gpointer p)
+{
+ struct rspamd_stat_sqlite3_ctx *ctx = p;
+ struct rspamd_stat_sqlite3_db *bk;
+ GHashTableIter it;
+ gpointer k, v;
+
+ g_hash_table_iter_init (&it, ctx->files);
+
+ while (g_hash_table_iter_next (&it, &k, &v)) {
+ bk = v;
+
+ if (bk->sqlite) {
+ if (bk->in_transaction) {
+ rspamd_sqlite3_run_prstmt (bk, RSPAMD_STAT_BACKEND_TRANSACTION_COMMIT);
+ }
+
+ sqlite3_close (bk->sqlite);
+ rspamd_sqlite3_close_prstmt (bk);
+ g_slice_free1 (sizeof (*bk), bk);
+ }
+ }
+
+ g_hash_table_destroy (ctx->files);
+}
+
+gpointer
+rspamd_sqlite3_runtime (struct rspamd_task *task,
+ struct rspamd_statfile_config *stcf, gboolean learn, gpointer p)
+{
+ struct rspamd_stat_sqlite3_ctx *ctx = p;
+ return g_hash_table_lookup (ctx->files, stcf);
+}
+
+gboolean
+rspamd_sqlite3_process_token (struct token_node_s *tok,
+ struct rspamd_token_result *res, gpointer p)
+{
+ struct rspamd_stat_sqlite3_db *bk;
+ gint64 iv = 0, idx;
+
+ g_assert (res != NULL);
+ g_assert (p != NULL);
+ g_assert (res->st_runtime != NULL);
+ g_assert (tok != NULL);
+ g_assert (tok->datalen >= sizeof (guint32) * 2);
+
+ bk = res->st_runtime->backend_runtime;
+
+ if (bk == NULL) {
+ /* Statfile is does not exist, so all values are zero */
+ res->value = 0.0;
+ return FALSE;
+ }
+
+ memcpy (&idx, tok->data, sizeof (idx));
+
+ /* TODO: language and user support */
+ if (rspamd_sqlite3_run_prstmt (bk, RSPAMD_STAT_BACKEND_GET_TOKEN,
+ idx, SQLITE3_DEFAULT, SQLITE3_DEFAULT, &iv) == SQLITE_OK) {
+ res->value = iv;
+ }
+ else {
+ res->value = 0.0;
+ return FALSE;
+ }
+
+
+ return TRUE;
+}
+
+gboolean
+rspamd_sqlite3_learn_token (struct token_node_s *tok,
+ struct rspamd_token_result *res, gpointer p)
+{
+ struct rspamd_stat_sqlite3_db *bk;
+ gint64 iv = 0, idx;
+
+ g_assert (res != NULL);
+ g_assert (p != NULL);
+ g_assert (res->st_runtime != NULL);
+ g_assert (tok != NULL);
+ g_assert (tok->datalen >= sizeof (guint32) * 2);
+
+ bk = res->st_runtime->backend_runtime;
+
+ if (bk == NULL) {
+ /* Statfile is does not exist, so all values are zero */
+ return FALSE;
+ }
+
+ if (!bk->in_transaction) {
+ rspamd_sqlite3_run_prstmt (bk, RSPAMD_STAT_BACKEND_TRANSACTION_START);
+ bk->in_transaction = TRUE;
+ }
+
+ iv = res->value;
+ memcpy (&idx, tok->data, sizeof (idx));
+
+ if (rspamd_sqlite3_run_prstmt (bk, RSPAMD_STAT_BACKEND_SET_TOKEN,
+ idx, 0, 0, iv) == SQLITE_OK) {
+ return FALSE;
+ }
+
+ return TRUE;
+}
+
+void
+rspamd_sqlite3_finalize_learn (struct rspamd_statfile_runtime *runtime,
+ gpointer ctx)
+{
+ struct rspamd_stat_sqlite3_db *bk;
+
+ g_assert (runtime->backend_runtime != NULL);
+ bk = runtime->backend_runtime;
+
+ if (bk->in_transaction) {
+ rspamd_sqlite3_run_prstmt (bk, RSPAMD_STAT_BACKEND_TRANSACTION_COMMIT);
+ bk->in_transaction = FALSE;
+ }
+
+ return;
+}
+
+gulong
+rspamd_sqlite3_total_learns (struct rspamd_statfile_runtime *runtime,
+ gpointer ctx)
+{
+ struct rspamd_stat_sqlite3_db *bk;
+ guint64 res;
+
+ g_assert (runtime->backend_runtime != NULL);
+ bk = runtime->backend_runtime;
+
+ rspamd_sqlite3_run_prstmt (bk, RSPAMD_STAT_BACKEND_GET_LEARNS, &res);
+
+ return res;
+}
+
+gulong
+rspamd_sqlite3_inc_learns (struct rspamd_statfile_runtime *runtime,
+ gpointer ctx)
+{
+ struct rspamd_stat_sqlite3_db *bk;
+
+ g_assert (runtime->backend_runtime != NULL);
+ bk = runtime->backend_runtime;
+ rspamd_sqlite3_run_prstmt (bk, RSPAMD_STAT_BACKEND_INC_LEARNS,
+ SQLITE3_DEFAULT, SQLITE3_DEFAULT);
+
+ return 0;
+}
+
+gulong
+rspamd_sqlite3_dec_learns (struct rspamd_statfile_runtime *runtime,
+ gpointer ctx)
+{
+ struct rspamd_stat_sqlite3_db *bk;
+
+ g_assert (runtime->backend_runtime != NULL);
+ bk = runtime->backend_runtime;
+ rspamd_sqlite3_run_prstmt (bk, RSPAMD_STAT_BACKEND_DEC_LEARNS,
+ SQLITE3_DEFAULT, SQLITE3_DEFAULT);
+
+ return 0;
+}
+
+gulong
+rspamd_sqlite3_learns (struct rspamd_statfile_runtime *runtime,
+ gpointer ctx)
+{
+ struct rspamd_stat_sqlite3_db *bk;
+ guint64 res;
+
+ g_assert (runtime->backend_runtime != NULL);
+ bk = runtime->backend_runtime;
+
+ rspamd_sqlite3_run_prstmt (bk, RSPAMD_STAT_BACKEND_GET_LEARNS, &res);
+
+ return res;
+}
+
+ucl_object_t *
+rspamd_sqlite3_get_stat (struct rspamd_statfile_runtime *runtime,
+ gpointer ctx)
+{
+ return NULL;
+}
diff --git a/src/libstat/stat_api.h b/src/libstat/stat_api.h
index 48c7c53c6..94734adef 100644
--- a/src/libstat/stat_api.h
+++ b/src/libstat/stat_api.h
@@ -51,6 +51,11 @@ typedef enum rspamd_stat_result_e {
void rspamd_stat_init (struct rspamd_config *cfg);
/**
+ * Finalize statistics
+ */
+void rspamd_stat_close (void);
+
+/**
* Classify the task specified and insert symbols if needed
* @param task
* @return TRUE if task has been classified
diff --git a/src/libstat/stat_config.c b/src/libstat/stat_config.c
index 8c935ec4e..c75b02fe0 100644
--- a/src/libstat/stat_config.c
+++ b/src/libstat/stat_config.c
@@ -55,19 +55,23 @@ static struct rspamd_stat_tokenizer stat_tokenizers[] = {
},
};
-static struct rspamd_stat_backend stat_backends[] = {
- {
- .name = RSPAMD_DEFAULT_BACKEND,
- .init = rspamd_mmaped_file_init,
- .runtime = rspamd_mmaped_file_runtime,
- .process_token = rspamd_mmaped_file_process_token,
- .learn_token = rspamd_mmaped_file_learn_token,
- .finalize_learn = rspamd_mmaped_file_finalize_learn,
- .total_learns = rspamd_mmaped_file_total_learns,
- .inc_learns = rspamd_mmaped_file_inc_learns,
- .dec_learns = rspamd_mmaped_file_dec_learns,
- .get_stat = rspamd_mmaped_file_get_stat
+#define RSPAMD_STAT_BACKEND_ELT(nam, eltn) { \
+ .name = #nam, \
+ .init = rspamd_##eltn##_init, \
+ .runtime = rspamd_##eltn##_runtime, \
+ .process_token = rspamd_##eltn##_process_token, \
+ .learn_token = rspamd_##eltn##_learn_token, \
+ .finalize_learn = rspamd_##eltn##_finalize_learn, \
+ .total_learns = rspamd_##eltn##_total_learns, \
+ .inc_learns = rspamd_##eltn##_inc_learns, \
+ .dec_learns = rspamd_##eltn##_dec_learns, \
+ .get_stat = rspamd_##eltn##_get_stat, \
+ .close = rspamd_##eltn##_close \
}
+
+static struct rspamd_stat_backend stat_backends[] = {
+ RSPAMD_STAT_BACKEND_ELT(mmap, mmaped_file),
+ RSPAMD_STAT_BACKEND_ELT(sqlite3, sqlite3)
};
static struct rspamd_stat_cache stat_caches[] = {
@@ -110,6 +114,21 @@ rspamd_stat_init (struct rspamd_config *cfg)
}
}
+void
+rspamd_stat_close (void)
+{
+ guint i;
+
+ g_assert (stat_ctx != NULL);
+
+ for (i = 0; i < stat_ctx->backends_count; i ++) {
+ if (stat_ctx->backends[i].close != NULL) {
+ stat_ctx->backends[i].close (stat_ctx->backends[i].ctx);
+ msg_debug ("closed backend %s", stat_ctx->backends[i].name);
+ }
+ }
+}
+
struct rspamd_stat_ctx *
rspamd_stat_get_ctx (void)
{