aboutsummaryrefslogtreecommitdiffstats
path: root/src/libserver/roll_history.c
diff options
context:
space:
mode:
authorVsevolod Stakhov <vsevolod@highsecure.ru>2014-04-21 16:25:51 +0100
committerVsevolod Stakhov <vsevolod@highsecure.ru>2014-04-21 16:25:51 +0100
commit61555065f3d1c8badcc9573691232f1b6e42988c (patch)
tree563d5b7cb8c468530f7e79c4da0a75267b1184e1 /src/libserver/roll_history.c
parentad5bf825b7f33bc10311673991f0cc888e69c0b1 (diff)
downloadrspamd-61555065f3d1c8badcc9573691232f1b6e42988c.tar.gz
rspamd-61555065f3d1c8badcc9573691232f1b6e42988c.zip
Rework project structure, remove trash files.
Diffstat (limited to 'src/libserver/roll_history.c')
-rw-r--r--src/libserver/roll_history.c212
1 files changed, 212 insertions, 0 deletions
diff --git a/src/libserver/roll_history.c b/src/libserver/roll_history.c
new file mode 100644
index 000000000..504f8ae3b
--- /dev/null
+++ b/src/libserver/roll_history.c
@@ -0,0 +1,212 @@
+/* Copyright (c) 2010-2012, 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 ''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 "main.h"
+#include "roll_history.h"
+
+
+/**
+ * Returns new roll history
+ * @param pool pool for shared memory
+ * @return new structure
+ */
+struct roll_history*
+rspamd_roll_history_new (rspamd_mempool_t *pool)
+{
+ struct roll_history *new;
+
+ if (pool == NULL) {
+ return NULL;
+ }
+
+ new = rspamd_mempool_alloc0_shared (pool, sizeof (struct roll_history));
+ new->pool = pool;
+ new->mtx = rspamd_mempool_get_mutex (pool);
+
+ return new;
+}
+
+struct history_metric_callback_data {
+ gchar *pos;
+ gint remain;
+};
+
+static void
+roll_history_symbols_callback (gpointer key, gpointer value, void *user_data)
+{
+ struct history_metric_callback_data *cb = user_data;
+ struct symbol *s = value;
+ guint wr;
+
+ if (cb->remain > 0) {
+ wr = rspamd_snprintf (cb->pos, cb->remain, "%s, ", s->name);
+ cb->pos += wr;
+ cb->remain -= wr;
+ }
+}
+
+/**
+ * Update roll history with data from task
+ * @param history roll history object
+ * @param task task object
+ */
+void
+rspamd_roll_history_update (struct roll_history *history, struct rspamd_task *task)
+{
+ gint row_num;
+ struct roll_history_row *row;
+ struct metric_result *metric_res;
+ struct history_metric_callback_data cbdata;
+
+ if (history->need_lock) {
+ /* Some process is getting history, so wait on a mutex */
+ rspamd_mempool_lock_mutex (history->mtx);
+ history->need_lock = FALSE;
+ rspamd_mempool_unlock_mutex (history->mtx);
+ }
+
+ /* First of all obtain check and obtain row number */
+ g_atomic_int_compare_and_exchange (&history->cur_row, HISTORY_MAX_ROWS, 0);
+#if ((GLIB_MAJOR_VERSION == 2) && (GLIB_MINOR_VERSION > 30))
+ row_num = g_atomic_int_add (&history->cur_row, 1);
+#else
+ row_num = g_atomic_int_exchange_and_add (&history->cur_row, 1);
+#endif
+
+ if (row_num < HISTORY_MAX_ROWS) {
+ row = &history->rows[row_num];
+ row->completed = FALSE;
+ }
+ else {
+ /* Race condition */
+ history->cur_row = 0;
+ return;
+ }
+
+ /* Add information from task to roll history */
+ memcpy (&row->from_addr, &task->from_addr, sizeof (row->from_addr));
+ memcpy (&row->tv, &task->tv, sizeof (row->tv));
+
+ /* Strings */
+ rspamd_strlcpy (row->message_id, task->message_id, sizeof (row->message_id));
+ if (task->user) {
+ rspamd_strlcpy (row->user, task->user, sizeof (row->message_id));
+ }
+ else {
+ row->user[0] = '\0';
+ }
+
+ /* Get default metric */
+ metric_res = g_hash_table_lookup (task->results, DEFAULT_METRIC);
+ if (metric_res == NULL) {
+ row->symbols[0] = '\0';
+ row->action = METRIC_ACTION_NOACTION;
+ }
+ else {
+ row->score = metric_res->score;
+ row->required_score = metric_res->metric->actions[METRIC_ACTION_REJECT].score;
+ row->action = check_metric_action (metric_res->score,
+ metric_res->metric->actions[METRIC_ACTION_REJECT].score, metric_res->metric);
+ cbdata.pos = row->symbols;
+ cbdata.remain = sizeof (row->symbols);
+ g_hash_table_foreach (metric_res->symbols, roll_history_symbols_callback, &cbdata);
+ if (cbdata.remain > 0) {
+ /* Remove last whitespace and comma */
+ *cbdata.pos-- = '\0';
+ *cbdata.pos-- = '\0';
+ *cbdata.pos = '\0';
+ }
+ }
+
+ row->scan_time = task->scan_milliseconds;
+ row->len = (task->msg == NULL ? 0 : task->msg->len);
+ row->completed = TRUE;
+}
+
+/**
+ * Load previously saved history from file
+ * @param history roll history object
+ * @param filename filename to load from
+ * @return TRUE if history has been loaded
+ */
+gboolean
+rspamd_roll_history_load (struct roll_history *history, const gchar *filename)
+{
+ gint fd;
+ struct stat st;
+
+ if (stat (filename, &st) == -1) {
+ msg_info ("cannot load history from %s: %s", filename, strerror (errno));
+ return FALSE;
+ }
+
+ if (st.st_size != sizeof (history->rows)) {
+ msg_info ("cannot load history from %s: size mismatch", filename);
+ return FALSE;
+ }
+
+ if ((fd = open (filename, O_RDONLY)) == -1) {
+ msg_info ("cannot load history from %s: %s", filename, strerror (errno));
+ return FALSE;
+ }
+
+ if (read (fd, history->rows, sizeof (history->rows)) == -1) {
+ close (fd);
+ msg_info ("cannot read history from %s: %s", filename, strerror (errno));
+ return FALSE;
+ }
+
+ close (fd);
+
+ return TRUE;
+}
+
+/**
+ * Save history to file
+ * @param history roll history object
+ * @param filename filename to load from
+ * @return TRUE if history has been saved
+ */
+gboolean
+rspamd_roll_history_save (struct roll_history *history, const gchar *filename)
+{
+ gint fd;
+
+ if ((fd = open (filename, O_WRONLY | O_CREAT | O_TRUNC, 00600)) == -1) {
+ msg_info ("cannot save history to %s: %s", filename, strerror (errno));
+ return FALSE;
+ }
+
+ if (write (fd, history->rows, sizeof (history->rows)) == -1) {
+ close (fd);
+ msg_info ("cannot write history to %s: %s", filename, strerror (errno));
+ return FALSE;
+ }
+
+ close (fd);
+
+ return TRUE;
+}