/*- * 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 "config.h" #include "rspamd.h" #include "util.h" #include "lua/lua_common.h" #include "unix-std.h" #ifdef HAVE_GLOB_H #include #endif static const char *lua_src_name = "lua/tests.lua"; extern char *lua_test; extern char *lua_test_case; extern char *argv0_dirname; extern struct rspamd_main *rspamd_main; static int traceback(lua_State *L) { if (!lua_isstring(L, 1)) { return 1; } lua_getglobal(L, "debug"); if (!lua_istable(L, -1)) { lua_pop(L, 1); return 1; } lua_getfield(L, -1, "traceback"); if (!lua_isfunction(L, -1)) { lua_pop(L, 2); return 1; } lua_pushvalue(L, 1); lua_pushinteger(L, 2); lua_call(L, 2, 1); return 1; } _Noreturn void rspamd_lua_test_func(void) { lua_State *L = (lua_State *) rspamd_main->cfg->lua_state; char *lua_src, *rp, rp_buf[PATH_MAX], path_buf[PATH_MAX], *tmp, *dir, *pattern; const char *old_path; glob_t globbuf; int i, len; rspamd_lua_set_env(L, NULL, NULL, NULL); rspamd_lua_set_globals(rspamd_main->cfg, L); rspamd_lua_start_gc(rspamd_main->cfg); if (lua_test_case) { lua_pushstring(L, lua_test_case); lua_setglobal(L, "test_pattern"); } rspamd_printf("Starting lua tests\n"); lua_src = g_build_filename(argv0_dirname, lua_src_name, NULL); if ((rp = realpath(lua_src, rp_buf)) == NULL) { msg_err("cannot find path %s: %s", lua_src, strerror(errno)); g_assert(0); } g_free(lua_src); tmp = g_strdup(rp); dir = dirname(tmp); /* Set lua path */ lua_getglobal(L, "package"); lua_getfield(L, -1, "path"); old_path = luaL_checkstring(L, -1); rspamd_snprintf(path_buf, sizeof(path_buf), "%s;%s/?.lua;%s/unit/?.lua", old_path, dir, dir); lua_pop(L, 1); lua_pushstring(L, path_buf); lua_setfield(L, -2, "path"); lua_pop(L, 1); lua_newtable(L); globbuf.gl_offs = 0; len = strlen(dir) + sizeof("/unit/") + sizeof("*.lua"); pattern = g_malloc(len); rspamd_snprintf(pattern, len, "%s/unit/%s", dir, "*.lua"); int lua_test_len = 0; int inserted_file = 1; int path_start; if (lua_test) { lua_test_len = strlen(lua_test); } if (glob(pattern, GLOB_DOOFFS, NULL, &globbuf) == 0) { for (i = 0; i < (int) globbuf.gl_pathc; i++) { if (lua_test) { path_start = strlen(globbuf.gl_pathv[i]) - lua_test_len; if (path_start < 0 || strncmp(globbuf.gl_pathv[i] + path_start, lua_test, lua_test_len) != 0) { continue; } } lua_pushinteger(L, inserted_file); lua_pushstring(L, globbuf.gl_pathv[i]); lua_settable(L, -3); inserted_file++; } globfree(&globbuf); g_free(pattern); } else { msg_err("pattern %s doesn't match: %s", pattern, strerror(errno)); g_assert(0); } lua_setglobal(L, "tests_list"); rspamd_lua_set_path(L, NULL, NULL); lua_pushcfunction(L, traceback); luaL_loadfile(L, rp); if (lua_pcall(L, 0, 0, lua_gettop(L) - 1) != 0) { msg_err("run test failed: %s", lua_tostring(L, -1)); g_assert(0); } exit(EXIT_SUCCESS); }