aboutsummaryrefslogtreecommitdiffstats
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/libutil/rrd.c106
-rw-r--r--src/libutil/rrd.h64
2 files changed, 93 insertions, 77 deletions
diff --git a/src/libutil/rrd.c b/src/libutil/rrd.c
index 02df74c64..a48420481 100644
--- a/src/libutil/rrd.c
+++ b/src/libutil/rrd.c
@@ -21,9 +21,29 @@
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
+#include <blake2.h>
#include "config.h"
#include "rrd.h"
#include "util.h"
+#include "logger.h"
+
+#define msg_err_rrd(...) rspamd_default_log_function (G_LOG_LEVEL_CRITICAL, \
+ "rrd", file->id, \
+ G_STRFUNC, \
+ __VA_ARGS__)
+#define msg_warn_rrd(...) rspamd_default_log_function (G_LOG_LEVEL_WARNING, \
+ "rrd", file->id, \
+ G_STRFUNC, \
+ __VA_ARGS__)
+#define msg_info_rrd(...) rspamd_default_log_function (G_LOG_LEVEL_INFO, \
+ "rrd", file->id, \
+ G_STRFUNC, \
+ __VA_ARGS__)
+#define msg_debug_rrd(...) rspamd_default_log_function (G_LOG_LEVEL_DEBUG, \
+ "rrd", file->id, \
+ G_STRFUNC, \
+ __VA_ARGS__)
+
static GQuark
rrd_error_quark (void)
@@ -303,6 +323,29 @@ rspamd_rrd_adjust_pointers (struct rspamd_rrd_file *file, gboolean completed)
}
}
+static void
+rspamd_rrd_calculate_checksum (struct rspamd_rrd_file *file)
+{
+ guchar sigbuf[BLAKE2B_OUTBYTES];
+ struct rrd_ds_def *ds;
+ guint i;
+ blake2b_state st;
+
+ if (file->finalized) {
+ blake2b_init (&st, BLAKE2B_OUTBYTES);
+ blake2b_update (&st, file->filename, strlen (file->filename));
+
+ for (i = 0; i < file->stat_head->ds_cnt; i ++) {
+ ds = &file->ds_def[i];
+ blake2b_update (&st, ds->ds_nam, sizeof (ds->ds_nam));
+ }
+
+ blake2b_final (&st, sigbuf, BLAKE2B_OUTBYTES);
+
+ file->id = rspamd_encode_base32 (sigbuf, sizeof (sigbuf));
+ }
+}
+
/**
* Open completed or incompleted rrd file
* @param filename
@@ -313,7 +356,7 @@ rspamd_rrd_adjust_pointers (struct rspamd_rrd_file *file, gboolean completed)
static struct rspamd_rrd_file *
rspamd_rrd_open_common (const gchar *filename, gboolean completed, GError **err)
{
- struct rspamd_rrd_file *new;
+ struct rspamd_rrd_file *file;
gint fd;
struct stat st;
@@ -321,9 +364,9 @@ rspamd_rrd_open_common (const gchar *filename, gboolean completed, GError **err)
return NULL;
}
- new = g_slice_alloc0 (sizeof (struct rspamd_rrd_file));
+ file = g_slice_alloc0 (sizeof (struct rspamd_rrd_file));
- if (new == NULL) {
+ if (file == NULL) {
g_set_error (err, rrd_error_quark (), ENOMEM, "not enough memory");
return NULL;
}
@@ -343,28 +386,29 @@ rspamd_rrd_open_common (const gchar *filename, gboolean completed, GError **err)
return FALSE;
}
/* Mmap file */
- new->size = st.st_size;
- if ((new->map =
+ file->size = st.st_size;
+ if ((file->map =
mmap (NULL, st.st_size, PROT_READ | PROT_WRITE, MAP_SHARED, fd,
0)) == MAP_FAILED) {
close (fd);
g_set_error (err,
rrd_error_quark (), ENOMEM, "mmap failed: %s", strerror (errno));
- g_slice_free1 (sizeof (struct rspamd_rrd_file), new);
+ g_slice_free1 (sizeof (struct rspamd_rrd_file), file);
return NULL;
}
close (fd);
/* Adjust pointers */
- rspamd_rrd_adjust_pointers (new, completed);
+ rspamd_rrd_adjust_pointers (file, completed);
/* Mark it as finalized */
- new->finalized = completed;
+ file->finalized = completed;
- new->filename = g_strdup (filename);
+ file->filename = g_strdup (filename);
+ rspamd_rrd_calculate_checksum (file);
- return new;
+ return file;
}
/**
@@ -376,7 +420,13 @@ rspamd_rrd_open_common (const gchar *filename, gboolean completed, GError **err)
struct rspamd_rrd_file *
rspamd_rrd_open (const gchar *filename, GError **err)
{
- return rspamd_rrd_open_common (filename, TRUE, err);
+ struct rspamd_rrd_file *file;
+
+ if ((file = rspamd_rrd_open_common (filename, TRUE, err))) {
+ msg_info_rrd ("rrd file opened: %s", filename);
+ }
+
+ return file;
}
/**
@@ -655,6 +705,8 @@ rspamd_rrd_finalize (struct rspamd_rrd_file *file, GError **err)
rspamd_rrd_adjust_pointers (file, TRUE);
file->finalized = TRUE;
+ rspamd_rrd_calculate_checksum (file);
+ msg_info_rrd ("rrd file created: %s", file->filename);
return TRUE;
}
@@ -683,6 +735,8 @@ rspamd_rrd_update_pdp_prep (struct rspamd_rrd_file *file,
rspamd_strlcpy (file->pdp_prep[i].last_ds, "U",
sizeof (file->pdp_prep[i].last_ds));
pdp_new[i] = NAN;
+ msg_debug_rrd ("adding unknown point interval %.3f is less than heartbeat %.3f",
+ interval, file->ds_def[i].par[RRD_DS_mrhb_cnt].lv);
}
else {
switch (type) {
@@ -690,17 +744,21 @@ rspamd_rrd_update_pdp_prep (struct rspamd_rrd_file *file,
case RRD_DST_DERIVE:
if (file->pdp_prep[i].last_ds[0] == 'U') {
pdp_new[i] = NAN;
+ msg_debug_rrd ("last point is NaN for point %ud", i);
}
else {
pdp_new[i] = vals[i] - strtod (file->pdp_prep[i].last_ds,
NULL);
+ msg_debug_rrd ("new PDP %ud, %.3f", i, pdp_new[i]);
}
break;
case RRD_DST_GAUGE:
pdp_new[i] = vals[i] * interval;
+ msg_debug_rrd ("new PDP %ud, %.3f", i, pdp_new[i]);
break;
case RRD_DST_ABSOLUTE:
pdp_new[i] = vals[i];
+ msg_debug_rrd ("new PDP %ud, %.3f", i, pdp_new[i]);
break;
default:
return FALSE;
@@ -773,6 +831,10 @@ rspamd_rrd_update_pdp_step (struct rspamd_rrd_file *file,
scratch[PDP_unkn_sec_cnt].lv = 0;
scratch[PDP_val].dv = pdp_new[i] / interval * post_int;
}
+
+ msg_debug_rrd ("new temp PDP %ud, %.3f -> %.3f, scratch: %3f",
+ i, pdp_new[i], pdp_temp[i],
+ scratch[PDP_val].dv);
}
}
@@ -811,6 +873,7 @@ rspamd_rrd_update_cdp (struct rspamd_rrd_file *file,
if (rra->pdp_cnt > 1) {
/* Do we have any CDP to update for this rra ? */
if (rra_steps[rra_index] > 0) {
+
if (isnan (pdp_temp[i])) {
/* New pdp is nan */
/* Increment unknown points count */
@@ -861,6 +924,7 @@ rspamd_rrd_update_cdp (struct rspamd_rrd_file *file,
break;
}
}
+
/* Init carry of this CDP */
pdp_in_cdp = (pdp_steps - pdp_offset) / rra->pdp_cnt;
if (pdp_in_cdp == 0 || isnan (pdp_temp[i])) {
@@ -889,6 +953,10 @@ rspamd_rrd_update_cdp (struct rspamd_rrd_file *file,
scratch[CDP_val].dv = pdp_temp[i];
}
}
+
+ msg_debug_rrd ("update cdp %d with value %.3f, "
+ "stored value: %.3f, carry: %.3f",
+ i, last_cdp, cur_cdp, scratch[CDP_val].dv);
}
/* In this case we just need to update cdp_prep for this RRA */
else {
@@ -923,6 +991,10 @@ rspamd_rrd_update_cdp (struct rspamd_rrd_file *file,
break;
}
}
+
+ msg_debug_rrd ("aggregate cdp %d with pdp %.3f, "
+ "stored value: %.3f",
+ i, pdp_temp[i], scratch[CDP_val].dv);
}
}
else {
@@ -1008,6 +1080,8 @@ rspamd_rrd_add_record (struct rspamd_rrd_file *file,
interval = ticks - ((gdouble)file->live_head->last_up +
file->live_head->last_up_usec / 1000000.);
+ msg_debug_rrd ("update rrd record after %.3f seconds", interval);
+
/* Update PDP preparation values */
pdp_new = g_malloc0 (sizeof (gdouble) * file->stat_head->ds_cnt);
pdp_temp = g_malloc0 (sizeof (gdouble) * file->stat_head->ds_cnt);
@@ -1097,6 +1171,10 @@ rspamd_rrd_add_record (struct rspamd_rrd_file *file,
/* This rra have not passed enough pdp steps */
rra_steps[i] = 0;
}
+
+ msg_debug_rrd ("cdp: %d, rra steps: %d(%d), pdp steps: %d",
+ i, rra_steps[i], pdp_offset, pdp_steps);
+
/* Update this specific CDP */
rspamd_rrd_update_cdp (file,
pdp_steps,
@@ -1135,9 +1213,9 @@ rspamd_rrd_close (struct rspamd_rrd_file * file)
}
munmap (file->map, file->size);
- if (file->filename != NULL) {
- g_free (file->filename);
- }
+ g_free (file->filename);
+ g_free (file->id);
+
g_slice_free1 (sizeof (struct rspamd_rrd_file), file);
return 0;
diff --git a/src/libutil/rrd.h b/src/libutil/rrd.h
index 43f112750..e6231bab8 100644
--- a/src/libutil/rrd.h
+++ b/src/libutil/rrd.h
@@ -93,25 +93,6 @@ enum rrd_cf_type {
RRD_CF_MINIMUM,
RRD_CF_MAXIMUM,
RRD_CF_LAST,
- RRD_CF_HWPREDICT,
- /* An array of predictions using the seasonal
- * Holt-Winters algorithm. Requires an RRA of type
- * CF_SEASONAL for this data source. */
- RRD_CF_SEASONAL,
- /* An array of seasonal effects. Requires an RRA of
- * type CF_HWPREDICT for this data source. */
- RRD_CF_DEVPREDICT,
- /* An array of deviation predictions based upon
- * smoothed seasonal deviations. Requires an RRA of
- * type CF_DEVSEASONAL for this data source. */
- RRD_CF_DEVSEASONAL,
- /* An array of smoothed seasonal deviations. Requires
- * an RRA of type CF_HWPREDICT for this data source.
- * */
- RRD_CF_FAILURES,
- /* HWPREDICT that follows a moving baseline */
- RRD_CF_MHWPREDICT
- /* new entries must come last !!! */
};
@@ -121,50 +102,6 @@ enum rrd_rra_param {
RRA_cdp_xff_val = 0, /* what part of the consolidated
* datapoint must be known, to produce a
* valid entry in the rra */
- /* CF_HWPREDICT: */
- RRA_hw_alpha = 1,
- /* exponential smoothing parameter for the intercept in
- * the Holt-Winters prediction algorithm. */
- RRA_hw_beta = 2,
- /* exponential smoothing parameter for the slope in
- * the Holt-Winters prediction algorithm. */
-
- RRA_dependent_rra_idx = 3,
- /* For CF_HWPREDICT: index of the RRA with the seasonal
- * effects of the Holt-Winters algorithm (of type
- * CF_SEASONAL).
- * For CF_DEVPREDICT: index of the RRA with the seasonal
- * deviation predictions (of type CF_DEVSEASONAL).
- * For CF_SEASONAL: index of the RRA with the Holt-Winters
- * intercept and slope coefficient (of type CF_HWPREDICT).
- * For CF_DEVSEASONAL: index of the RRA with the
- * Holt-Winters prediction (of type CF_HWPREDICT).
- * For CF_FAILURES: index of the CF_DEVSEASONAL array.
- * */
-
- /* CF_SEASONAL and CF_DEVSEASONAL: */
- RRA_seasonal_gamma = 1,
- /* exponential smoothing parameter for seasonal effects. */
-
- RRA_seasonal_smoothing_window = 2,
- /* fraction of the season to include in the running average
- * smoother */
-
- /* RRA_dependent_rra_idx = 3, */
-
- RRA_seasonal_smooth_idx = 4,
- /* an integer between 0 and row_count - 1 which
- * is index in the seasonal cycle for applying
- * the period smoother. */
-
- /* CF_FAILURES: */
- RRA_delta_pos = 1, /* confidence bound scaling parameters */
- RRA_delta_neg = 2,
- /* RRA_dependent_rra_idx = 3, */
- RRA_window_len = 4,
- RRA_failure_threshold = 5
- /* For CF_FAILURES, number of violations within the last
- * window required to mark a failure. */
};
@@ -275,6 +212,7 @@ struct rspamd_rrd_file {
guint8 * map; /* mmapped area */
gsize size; /* its size */
gboolean finalized;
+ gchar *id;
};