/*- * Copyright 2016 Vsevolod Stakhov * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "lua_common.h" #include "cfg_file.h" /* Classifier methods */ LUA_FUNCTION_DEF (classifier, register_pre_callback); LUA_FUNCTION_DEF (classifier, register_post_callback); LUA_FUNCTION_DEF (classifier, get_statfiles); LUA_FUNCTION_DEF (classifier, get_statfile_by_label); static const struct luaL_reg classifierlib_m[] = { LUA_INTERFACE_DEF (classifier, register_pre_callback), LUA_INTERFACE_DEF (classifier, register_post_callback), LUA_INTERFACE_DEF (classifier, get_statfiles), LUA_INTERFACE_DEF (classifier, get_statfile_by_label), {"__tostring", rspamd_lua_class_tostring}, {NULL, NULL} }; LUA_FUNCTION_DEF (statfile, get_symbol); LUA_FUNCTION_DEF (statfile, get_label); LUA_FUNCTION_DEF (statfile, is_spam); LUA_FUNCTION_DEF (statfile, get_param); static const struct luaL_reg statfilelib_m[] = { LUA_INTERFACE_DEF (statfile, get_symbol), LUA_INTERFACE_DEF (statfile, get_label), LUA_INTERFACE_DEF (statfile, is_spam), LUA_INTERFACE_DEF (statfile, get_param), {"__tostring", rspamd_lua_class_tostring}, {NULL, NULL} }; struct classifier_callback_data { lua_State *L; const gchar *name; }; static struct rspamd_statfile_config * lua_check_statfile (lua_State * L); /* Classifier implementation */ static struct rspamd_classifier_config * lua_check_classifier (lua_State * L) { void *ud = luaL_checkudata (L, 1, "rspamd{classifier}"); luaL_argcheck (L, ud != NULL, 1, "'classifier' expected"); return ud ? *((struct rspamd_classifier_config **)ud) : NULL; } static GList * call_classifier_pre_callback (struct rspamd_classifier_config *ccf, struct rspamd_task *task, lua_State *L, gboolean is_learn, gboolean is_spam) { struct rspamd_classifier_config **pccf; struct rspamd_task **ptask; struct rspamd_statfile_config **pst; GList *res = NULL; pccf = lua_newuserdata (L, sizeof (struct rspamd_classifier_config *)); rspamd_lua_setclass (L, "rspamd{classifier}", -1); *pccf = ccf; ptask = lua_newuserdata (L, sizeof (struct rspamd_task *)); rspamd_lua_setclass (L, "rspamd{task}", -1); *ptask = task; lua_pushboolean (L, is_learn); lua_pushboolean (L, is_spam); if (lua_pcall (L, 4, 1, 0) != 0) { msg_warn_task ("error running pre classifier callback %s", lua_tostring (L, -1)); lua_pop (L, 1); } else { if (lua_istable (L, -1)) { lua_pushnil (L); while (lua_next (L, -2)) { pst = luaL_checkudata (L, -1, "rspamd{statfile}"); if (pst) { res = g_list_prepend (res, *pst); } lua_pop (L, 1); } } lua_pop (L, 1); } return res; } /* Return list of statfiles that should be checked for this message */ GList * rspamd_lua_call_cls_pre_callbacks (struct rspamd_classifier_config *ccf, struct rspamd_task *task, gboolean is_learn, gboolean is_spam, lua_State *L) { GList *res = NULL, *cur; struct classifier_callback_data *cd; /* Go throught all callbacks and call them, appending results to list */ cur = g_list_first (ccf->pre_callbacks); while (cur) { cd = cur->data; lua_getglobal (L, cd->name); res = g_list_concat (res, call_classifier_pre_callback (ccf, task, L, is_learn, is_spam)); cur = g_list_next (cur); } if (res == NULL) { /* Check function from global table 'classifiers' */ lua_getglobal (L, "classifiers"); if (lua_istable (L, -1)) { lua_pushstring (L, ccf->name); lua_gettable (L, -2); /* Function is now on top */ if (lua_isfunction (L, -1)) { res = call_classifier_pre_callback (ccf, task, L, is_learn, is_spam); } lua_pop (L, 1); } lua_pop (L, 1); } return res; } /* Return result mark for statfile */ double rspamd_lua_call_cls_post_callbacks (struct rspamd_classifier_config *ccf, struct rspamd_task *task, double in, lua_State *L) { struct classifier_callback_data *cd; struct rspamd_classifier_config **pccf; struct rspamd_task **ptask; double out = in; GList *cur; /* Go throught all callbacks and call them, appending results to list */ cur = g_list_first (ccf->pre_callbacks); while (cur) { cd = cur->data; lua_getglobal (L, cd->name); pccf = lua_newuserdata (L, sizeof (struct rspamd_classifier_config *)); rspamd_lua_setclass (L, "rspamd{classifier}", -1); *pccf = ccf; ptask = lua_newuserdata (L, sizeof (struct rspamd_task *)); rspamd_lua_setclass (L, "rspamd{task}", -1); *ptask = task; lua_pushnumber (L, out); if (lua_pcall (L, 3, 1, 0) != 0) { msg_warn_task ("error running function %s: %s", cd->name, lua_tostring (L, -1)); lua_pop (L, 1); } else { if (lua_isnumber (L, 1)) { out = lua_tonumber (L, 1); } lua_pop (L, 1); } cur = g_list_next (cur); } return out; } static gint lua_classifier_register_pre_callback (lua_State *L) { struct rspamd_classifier_config *ccf = lua_check_classifier (L); struct classifier_callback_data *cd; const gchar *name; if (ccf) { name = luaL_checkstring (L, 2); if (name) { cd = g_malloc (sizeof (struct classifier_callback_data)); cd->name = g_strdup (name); cd->L = L; ccf->pre_callbacks = g_list_prepend (ccf->pre_callbacks, cd); } } return 0; } static gint lua_classifier_register_post_callback (lua_State *L) { struct rspamd_classifier_config *ccf = lua_check_classifier (L); struct classifier_callback_data *cd; const gchar *name; if (ccf) { name = luaL_checkstring (L, 2); if (name) { cd = g_malloc (sizeof (struct classifier_callback_data)); cd->name = g_strdup (name); cd->L = L; ccf->pre_callbacks = g_list_prepend (ccf->pre_callbacks, cd); } } return 0; } /* Return table of statfiles indexed by name */ static gint lua_classifier_get_statfiles (lua_State *L) { struct rspamd_classifier_config *ccf = lua_check_classifier (L); GList *cur; struct rspamd_statfile_config *st, **pst; gint i; if (ccf) { lua_newtable (L); cur = g_list_first (ccf->statfiles); i = 1; while (cur) { st = cur->data; pst = lua_newuserdata (L, sizeof (struct rspamd_statfile_config *)); rspamd_lua_setclass (L, "rspamd{statfile}", -1); *pst = st; lua_rawseti (L, -2, i++); cur = g_list_next (cur); } } else { lua_pushnil (L); } return 1; } /* Get statfile with specified label */ static gint lua_classifier_get_statfile_by_label (lua_State *L) { struct rspamd_classifier_config *ccf = lua_check_classifier (L); struct rspamd_statfile_config *st, **pst; const gchar *label; GList *cur; gint i; label = luaL_checkstring (L, 2); if (ccf && label) { cur = g_hash_table_lookup (ccf->labels, label); if (cur) { lua_newtable (L); i = 1; while (cur) { st = cur->data; pst = lua_newuserdata (L, sizeof (struct rspamd_statfile_config *)); rspamd_lua_setclass (L, "rspamd{statfile}", -1); *pst = st; lua_rawseti (L, -2, i++); cur = g_list_next (cur); } return 1; } } lua_pushnil (L); return 1; } /* Statfile functions */ static gint lua_statfile_get_symbol (lua_State *L) { struct rspamd_statfile_config *st = lua_check_statfile (L); if (st != NULL) { lua_pushstring (L, st->symbol); } else { lua_pushnil (L); } return 1; } static gint lua_statfile_get_label (lua_State *L) { struct rspamd_statfile_config *st = lua_check_statfile (L); if (st != NULL && st->label != NULL) { lua_pushstring (L, st->label); } else { lua_pushnil (L); } return 1; } static gint lua_statfile_is_spam (lua_State *L) { struct rspamd_statfile_config *st = lua_check_statfile (L); if (st != NULL) { lua_pushboolean (L, st->is_spam); } else { lua_pushnil (L); } return 1; } static gint lua_statfile_get_param (lua_State *L) { struct rspamd_statfile_config *st = lua_check_statfile (L); const gchar *param; const ucl_object_t *value; param = luaL_checkstring (L, 2); if (st != NULL && param != NULL) { value = ucl_object_lookup (st->opts, param); if (value != NULL) { lua_pushstring (L, ucl_object_tostring_forced (value)); return 1; } } lua_pushnil (L); return 1; } static struct rspamd_statfile_config * lua_check_statfile (lua_State * L) { void *ud = luaL_checkudata (L, 1, "rspamd{statfile}"); luaL_argcheck (L, ud != NULL, 1, "'statfile' expected"); return ud ? *((struct rspamd_statfile_config **)ud) : NULL; } /* Open functions */ void luaopen_classifier (lua_State * L) { rspamd_lua_new_class (L, "rspamd{classifier}", classifierlib_m); lua_pop (L, 1); /* remove metatable from stack */ } void luaopen_statfile (lua_State * L) { rspamd_lua_new_class (L, "rspamd{statfile}", statfilelib_m); lua_pop (L, 1); /* remove metatable from stack */ }