From 54a39b076073c3600d8f2f8bb655fc574937903e Mon Sep 17 00:00:00 2001 From: ssturges Date: Tue, 25 Jan 2022 13:39:09 -0500 Subject: [Enhancement] Add rspamadm configdump option --symbol-detail which provides a consolidated output of every setting for a symbol across all configuration locations. --- src/libserver/rspamd_symcache.c | 21 +++++++ src/libserver/rspamd_symcache.h | 5 ++ src/rspamadm/configdump.c | 120 +++++++++++++++++++++++++++++++++++++++- 3 files changed, 144 insertions(+), 2 deletions(-) diff --git a/src/libserver/rspamd_symcache.c b/src/libserver/rspamd_symcache.c index 4335ada68..21804d543 100644 --- a/src/libserver/rspamd_symcache.c +++ b/src/libserver/rspamd_symcache.c @@ -3384,6 +3384,27 @@ rspamd_symcache_set_symbol_flags (struct rspamd_symcache *cache, return FALSE; } +void +rspamd_symcache_get_symbol_details(struct rspamd_symcache *cache, + const gchar *symbol, + const ucl_object_t *this_sym_ucl) +{ + struct rspamd_symcache_item *item; + + g_assert (cache != NULL); + g_assert (symbol != NULL); + + item = rspamd_symcache_find_filter (cache, symbol, false); + + if (item) { + ucl_object_insert_key (this_sym_ucl, + ucl_object_fromstring(item->type_descr), + "type", strlen("type"), false); + + // any other data? + } +} + guint rspamd_symcache_get_symbol_flags (struct rspamd_symcache *cache, const gchar *symbol) diff --git a/src/libserver/rspamd_symcache.h b/src/libserver/rspamd_symcache.h index 67c753e39..05fe80cef 100644 --- a/src/libserver/rspamd_symcache.h +++ b/src/libserver/rspamd_symcache.h @@ -301,6 +301,11 @@ gboolean rspamd_symcache_set_symbol_flags (struct rspamd_symcache *cache, guint rspamd_symcache_get_symbol_flags (struct rspamd_symcache *cache, const gchar *symbol); +void rspamd_symcache_get_symbol_details(struct rspamd_symcache *cache, + const gchar *symbol, + const ucl_object_t *this_sym_ucl); + + /** * Process settings for task * @param task diff --git a/src/rspamadm/configdump.c b/src/rspamadm/configdump.c index 36ea7eca7..577d05581 100644 --- a/src/rspamadm/configdump.c +++ b/src/rspamadm/configdump.c @@ -28,6 +28,7 @@ static gboolean show_help = FALSE; static gboolean show_comments = FALSE; static gboolean modules_state = FALSE; static gboolean symbol_groups_only = FALSE; +static gboolean symbol_full_details = FALSE; static gboolean skip_template = FALSE; static gchar *config = NULL; extern struct rspamd_main *rspamd_main; @@ -52,7 +53,7 @@ static GOptionEntry entries[] = { {"compact", 'C', 0, G_OPTION_ARG_NONE, &compact, "Compacted json output", NULL}, {"config", 'c', 0, G_OPTION_ARG_STRING, &config, - "Config file to test", NULL}, + "Config file to test", NULL}, {"show-help", 'h', 0, G_OPTION_ARG_NONE, &show_help, "Show help as comments for each option", NULL }, {"show-comments", 's', 0, G_OPTION_ARG_NONE, &show_comments, @@ -61,6 +62,8 @@ static GOptionEntry entries[] = { "Show modules state only", NULL}, {"groups", 'g', 0, G_OPTION_ARG_NONE, &symbol_groups_only, "Show symbols groups only", NULL}, + {"symbol-details", 'd', 0, G_OPTION_ARG_NONE, &symbol_full_details, + "Show full symbol details only", NULL}, {"skip-template", 'T', 0, G_OPTION_ARG_NONE, &skip_template, "Do not apply Jinja templates", NULL}, {NULL, 0, 0, G_OPTION_ARG_NONE, NULL, NULL, NULL} @@ -237,7 +240,7 @@ rspamadm_configdump (gint argc, gchar **argv, const struct rspamadm_command *cmd GOptionContext *context; GError *error = NULL; const gchar *confdir; - const ucl_object_t *obj, *cur, *doc_obj; + const ucl_object_t *obj = NULL, *cur, *doc_obj; struct rspamd_config *cfg = rspamd_main->cfg; gboolean ret = TRUE; worker_t **pworker; @@ -305,6 +308,119 @@ rspamadm_configdump (gint argc, gchar **argv, const struct rspamadm_command *cmd exit (EXIT_SUCCESS); } + if (symbol_full_details) { + /* + * Create object from symbols groups and output it using the + * specified format + */ + ucl_object_t *out = ucl_object_typed_new (UCL_OBJECT); + GHashTableIter it; + gpointer sk, sv; + + g_hash_table_iter_init (&it, cfg->symbols); + ucl_object_t *sym_ucl = ucl_object_typed_new (UCL_OBJECT); + const ucl_object_t *all_symbols_ucl = ucl_object_lookup(cfg->rcl_obj, "symbols"); + + while (g_hash_table_iter_next (&it, &sk, &sv)) { + const gchar *sym_name = (const gchar *)sk; + struct rspamd_symbol *s = (struct rspamd_symbol *)sv; + ucl_object_t *this_sym_ucl = ucl_object_typed_new (UCL_OBJECT); + + ucl_object_insert_key (this_sym_ucl, + ucl_object_fromdouble (s->score), + "score", strlen ("score"), + false); + ucl_object_insert_key (this_sym_ucl, + ucl_object_fromstring (s->description), + "description", strlen ("description"), false); + + rspamd_symcache_get_symbol_details(cfg->cache, sym_name, this_sym_ucl); + + ucl_object_insert_key (this_sym_ucl, + ucl_object_frombool (!!(s->flags & RSPAMD_SYMBOL_FLAG_DISABLED)), + "disabled", strlen ("disabled"), + false); + + if (s->nshots == 1) { + ucl_object_insert_key (this_sym_ucl, + ucl_object_frombool (true), + "one_shot", strlen ("one_shot"), + false); + } + else { + ucl_object_insert_key (this_sym_ucl, + ucl_object_frombool (false), + "one_shot", strlen ("one_shot"), + false); + } + + if (s->gr != NULL) { + struct rspamd_symbols_group *gr = s->gr; + const char *gr_name = gr->name; + if (strcmp(gr_name, "ungrouped") != 0) { + ucl_object_insert_key (this_sym_ucl, + ucl_object_fromstring (gr_name), + "group", strlen("group"), + false); + } + + if (s->groups) { + ucl_object_t *add_groups = ucl_object_typed_new (UCL_ARRAY); + guint j; + struct rspamd_symbols_group *add_gr; + bool one_extra = false; + + PTR_ARRAY_FOREACH (s->groups, j, add_gr) { + if (add_gr->name && strcmp (add_gr->name, gr_name) != 0) { + ucl_array_append (add_groups, + ucl_object_fromstring (add_gr->name)); + one_extra = true; + } + } + + if (one_extra == true) { + ucl_object_insert_key (this_sym_ucl, + add_groups, + "groups", strlen ("groups"), + false); + } + } + } + + const ucl_object_t *loaded_symbol_ucl = ucl_object_lookup(all_symbols_ucl, sym_name); + if (loaded_symbol_ucl) { + ucl_object_iter_t it = NULL; + while ((cur = ucl_iterate_object (loaded_symbol_ucl, &it, true)) != NULL) + { + const char *key = ucl_object_key(cur); + /* If this key isn't something we have direct in the symbol item, grab the key/value */ + if ((strcmp(key, "score") != 0 ) && + (strcmp(key, "description") != 0) && + (strcmp(key, "disabled") != 0) && + (strcmp(key, "condition") != 0) && + (strcmp(key, "score") != 0) && + (strcmp(key, "one_shot") != 0) && + (strcmp(key, "any_shot") != 0) && + (strcmp(key, "nshots") != 0) && + (strcmp(key, "one_param") != 0) && + (strcmp(key, "priority") != 0)) + { + ucl_object_insert_key(this_sym_ucl, (ucl_object_t *)cur, key, strlen(key), false); + } + } + } + + ucl_object_insert_key (sym_ucl, this_sym_ucl, sym_name, + strlen (sym_name), true); + + } + ucl_object_insert_key (out, sym_ucl, "symbols", + strlen ("symbols"), true); + + rspamadm_dump_section_obj (cfg, out, NULL); + exit (EXIT_SUCCESS); + } + if (symbol_groups_only) { /* * Create object from symbols groups and output it using the -- cgit v1.2.3