aboutsummaryrefslogtreecommitdiffstats
path: root/src/libserver/re_cache.c
diff options
context:
space:
mode:
authorVsevolod Stakhov <vsevolod@highsecure.ru>2018-09-15 14:35:33 +0100
committerVsevolod Stakhov <vsevolod@highsecure.ru>2018-09-15 14:35:33 +0100
commit8faa65961ae7dae4e0782f1cccf05630d4f8750a (patch)
tree9b4b5228356d8ececa590c68fd659fd8ba5fa3eb /src/libserver/re_cache.c
parent4e3cace514317d2535a7b811ecb9b4d962502d48 (diff)
downloadrspamd-8faa65961ae7dae4e0782f1cccf05630d4f8750a.tar.gz
rspamd-8faa65961ae7dae4e0782f1cccf05630d4f8750a.zip
[Project] Implement selectors registration for regular expressions
Diffstat (limited to 'src/libserver/re_cache.c')
-rw-r--r--src/libserver/re_cache.c50
1 files changed, 50 insertions, 0 deletions
diff --git a/src/libserver/re_cache.c b/src/libserver/re_cache.c
index d8f7f3d0d..0a70c85b6 100644
--- a/src/libserver/re_cache.c
+++ b/src/libserver/re_cache.c
@@ -23,6 +23,9 @@
#include "libutil/util.h"
#include "libutil/regexp.h"
#include "lua/lua_common.h"
+
+#include "khash.h"
+
#ifdef WITH_HYPERSCAN
#include "hs.h"
#include "unix-std.h"
@@ -70,6 +73,7 @@ static const guchar rspamd_hs_magic[] = {'r', 's', 'h', 's', 'r', 'e', '1', '1'}
rspamd_hs_magic_vector[] = {'r', 's', 'h', 's', 'r', 'v', '1', '1'};
#endif
+
struct rspamd_re_class {
guint64 id;
enum rspamd_re_type type;
@@ -97,13 +101,18 @@ struct rspamd_re_cache_elt {
enum rspamd_re_cache_elt_match_type match_type;
};
+KHASH_INIT (lua_selectors_hash, gchar *, int, 1, kh_str_hash_func, kh_str_hash_equal);
+
struct rspamd_re_cache {
GHashTable *re_classes;
+
GPtrArray *re;
+ khash_t (lua_selectors_hash) *selectors;
ref_entry_t ref;
guint nre;
guint max_re_data;
gchar hash[rspamd_cryptobox_HASHBYTES + 1];
+ lua_State *L;
#ifdef WITH_HYPERSCAN
gboolean hyperscan_loaded;
gboolean disable_hyperscan;
@@ -149,6 +158,8 @@ rspamd_re_cache_destroy (struct rspamd_re_cache *cache)
GHashTableIter it;
gpointer k, v;
struct rspamd_re_class *re_class;
+ gchar *skey;
+ gint sref;
g_assert (cache != NULL);
g_hash_table_iter_init (&it, cache->re_classes);
@@ -176,6 +187,15 @@ rspamd_re_cache_destroy (struct rspamd_re_cache *cache)
g_free (re_class);
}
+ if (cache->L) {
+ kh_foreach (cache->selectors, skey, sref, {
+ luaL_unref (cache->L, LUA_REGISTRYINDEX, sref);
+ g_free (skey);
+ });
+ }
+
+ kh_destroy (lua_selectors_hash, cache->selectors);
+
g_hash_table_unref (cache->re_classes);
g_ptr_array_free (cache->re, TRUE);
g_free (cache);
@@ -199,6 +219,7 @@ rspamd_re_cache_new (void)
cache->re_classes = g_hash_table_new (g_int64_hash, g_int64_equal);
cache->nre = 0;
cache->re = g_ptr_array_new_full (256, rspamd_re_cache_elt_dtor);
+ cache->selectors = kh_init (lua_selectors_hash);
#ifdef WITH_HYPERSCAN
cache->hyperscan_loaded = FALSE;
#endif
@@ -413,6 +434,8 @@ rspamd_re_cache_init (struct rspamd_re_cache *cache, struct rspamd_config *cfg)
}
}
+ cache->L = cfg->lua_state;
+
#ifdef WITH_HYPERSCAN
const gchar *platform = "generic";
rspamd_fstring_t *features = rspamd_fstring_new ();
@@ -1979,3 +2002,30 @@ rspamd_re_cache_load_hyperscan (struct rspamd_re_cache *cache,
return TRUE;
#endif
}
+
+void rspamd_re_cache_add_selector (struct rspamd_re_cache *cache,
+ const gchar *sname,
+ gint ref)
+{
+ khiter_t k;
+
+ k = kh_get (lua_selectors_hash, cache->selectors, (gchar *)sname);
+
+ if (k == kh_end (cache->selectors)) {
+ gchar *cpy = g_strdup (sname);
+ gint res;
+
+ k = kh_put (lua_selectors_hash, cache->selectors, cpy, &res);
+
+ kh_value (cache->selectors, k) = ref;
+ }
+ else {
+ msg_warn_re_cache ("replacing selector with name %s", sname);
+
+ if (cache->L) {
+ luaL_unref (cache->L, LUA_REGISTRYINDEX, kh_value (cache->selectors, k));
+ }
+
+ kh_value (cache->selectors, k) = ref;
+ }
+}