diff options
author | Vsevolod Stakhov <vsevolod@highsecure.ru> | 2015-06-19 17:00:24 +0100 |
---|---|---|
committer | Vsevolod Stakhov <vsevolod@highsecure.ru> | 2015-06-19 17:00:24 +0100 |
commit | 29c56c31b68a3c55a0b77689b7a3ee141549e485 (patch) | |
tree | b503d9f73f912a438b1e4a415a9f221d4dad3542 /src/libstat | |
parent | d7d83b570be42d732083f1b5839760b78979b7cf (diff) | |
parent | 03926b5291fda6282f463fd962ccd74e16ef791d (diff) | |
download | rspamd-29c56c31b68a3c55a0b77689b7a3ee141549e485.tar.gz rspamd-29c56c31b68a3c55a0b77689b7a3ee141549e485.zip |
Merge branch 'sqlite-stat'
Diffstat (limited to 'src/libstat')
-rw-r--r-- | src/libstat/CMakeLists.txt | 3 | ||||
-rw-r--r-- | src/libstat/backends/backends.h | 68 | ||||
-rw-r--r-- | src/libstat/backends/mmaped_file.c | 47 | ||||
-rw-r--r-- | src/libstat/backends/sqlite3_backend.c | 591 | ||||
-rw-r--r-- | src/libstat/stat_api.h | 5 | ||||
-rw-r--r-- | src/libstat/stat_config.c | 43 |
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) { |