diff options
author | Vsevolod Stakhov <vsevolod@highsecure.ru> | 2016-09-17 13:22:20 +0100 |
---|---|---|
committer | Vsevolod Stakhov <vsevolod@highsecure.ru> | 2016-09-17 13:22:20 +0100 |
commit | 3f0fdfd8561d0e71b13c194d4f3000f6727b3ed9 (patch) | |
tree | 6f2e26f6bf098e1613475e7d3af20851f966ea4a /src/lua | |
parent | ca4f536c6741f8fde57d243c242f4ca10e108905 (diff) | |
download | rspamd-3f0fdfd8561d0e71b13c194d4f3000f6727b3ed9.tar.gz rspamd-3f0fdfd8561d0e71b13c194d4f3000f6727b3ed9.zip |
[Feature] Rework lua tcp module
Diffstat (limited to 'src/lua')
-rw-r--r-- | src/lua/lua_tcp.c | 140 |
1 files changed, 121 insertions, 19 deletions
diff --git a/src/lua/lua_tcp.c b/src/lua/lua_tcp.c index 4c202787f..38d14adfc 100644 --- a/src/lua/lua_tcp.c +++ b/src/lua/lua_tcp.c @@ -17,6 +17,7 @@ #include "buffer.h" #include "dns.h" #include "utlist.h" +#include "ref.h" #include "unix-std.h" static void lua_tcp_handler (int fd, short what, gpointer ud); @@ -49,8 +50,22 @@ end LUA_FUNCTION_DEF (tcp, request); +/*** + * @method tcp:close() + * + * Closes TCP connection + */ +LUA_FUNCTION_DEF (tcp, close); +LUA_FUNCTION_DEF (tcp, gc); + static const struct luaL_reg tcp_libf[] = { LUA_INTERFACE_DEF (tcp, request), + {NULL, NULL} +}; + +static const struct luaL_reg tcp_libm[] = { + LUA_INTERFACE_DEF (tcp, close), + {"__gc", lua_tcp_gc}, {"__tostring", rspamd_lua_class_tostring}, {NULL, NULL} }; @@ -67,14 +82,17 @@ struct lua_tcp_cbdata { gchar *stop_pattern; struct rspamd_async_watcher *w; struct event ev; + ref_entry_t ref; gint fd; gint cbref; + gint connect_cb; guint iovlen; guint pos; guint total; + guint16 port; gboolean partial; gboolean do_shutdown; - guint16 port; + gboolean connected; }; static const int default_tcp_timeout = 5000; @@ -110,6 +128,14 @@ lua_tcp_fin (gpointer arg) g_slice_free1 (sizeof (struct lua_tcp_cbdata), cbd); } +static struct lua_tcp_cbdata * +lua_check_tcp (lua_State *L, gint pos) +{ + void *ud = rspamd_lua_check_udata (L, pos, "rspamd{tcp}"); + luaL_argcheck (L, ud != NULL, pos, "'tcp' expected"); + return ud ? *((struct lua_tcp_cbdata **)ud) : NULL; +} + static void lua_tcp_maybe_free (struct lua_tcp_cbdata *cbd) { @@ -126,13 +152,24 @@ static void lua_tcp_push_error (struct lua_tcp_cbdata *cbd, const char *err, ...) { va_list ap; + struct lua_tcp_cbdata **pcbd; - va_start (ap, err); lua_rawgeti (cbd->L, LUA_REGISTRYINDEX, cbd->cbref); + + /* Error message */ + va_start (ap, err); lua_pushvfstring (cbd->L, err, ap); va_end (ap); - if (lua_pcall (cbd->L, 1, 0, 0) != 0) { + /* Body */ + lua_pushnil (cbd->L); + /* Connection */ + pcbd = lua_newuserdata (cbd->L, sizeof (*pcbd)); + *pcbd = cbd; + REF_RETAIN (cbd); + rspamd_lua_setclass (cbd->L, "rspamd{tcp}", -1); + + if (lua_pcall (cbd->L, 3, 0, 0) != 0) { msg_info ("callback call failed: %s", lua_tostring (cbd->L, -1)); lua_pop (cbd->L, 1); } @@ -142,6 +179,7 @@ static void lua_tcp_push_data (struct lua_tcp_cbdata *cbd, const gchar *str, gsize len) { struct rspamd_lua_text *t; + struct lua_tcp_cbdata **pcbd; lua_rawgeti (cbd->L, LUA_REGISTRYINDEX, cbd->cbref); /* Error */ @@ -152,8 +190,13 @@ lua_tcp_push_data (struct lua_tcp_cbdata *cbd, const gchar *str, gsize len) t->start = str; t->len = len; t->own = FALSE; + /* Connection */ + pcbd = lua_newuserdata (cbd->L, sizeof (*pcbd)); + *pcbd = cbd; + rspamd_lua_setclass (cbd->L, "rspamd{tcp}", -1); + REF_RETAIN (cbd); - if (lua_pcall (cbd->L, 2, 0, 0) != 0) { + if (lua_pcall (cbd->L, 3, 0, 0) != 0) { msg_info ("callback call failed: %s", lua_tostring (cbd->L, -1)); lua_pop (cbd->L, 1); } @@ -208,7 +251,8 @@ lua_tcp_write_helper (struct lua_tcp_cbdata *cbd) if (r == -1) { lua_tcp_push_error (cbd, "IO write error while trying to write %d " "bytes: %s", (gint)remain, strerror (errno)); - lua_tcp_maybe_free (cbd); + REF_RELEASE (cbd); + return; } else { @@ -257,6 +301,8 @@ lua_tcp_handler (int fd, short what, gpointer ud) gssize r; guint slen; + REF_RETAIN (cbd); + if (what == EV_READ) { g_assert (cbd->partial || cbd->in != NULL); @@ -285,7 +331,7 @@ lua_tcp_handler (int fd, short what, gpointer ud) } } - lua_tcp_maybe_free (cbd); + REF_RELEASE (cbd); } else { if (cbd->partial) { @@ -301,7 +347,7 @@ lua_tcp_handler (int fd, short what, gpointer ud) if (memcmp (cbd->stop_pattern, cbd->in->str + (cbd->in->len - slen), slen) == 0) { lua_tcp_push_data (cbd, cbd->in->str, cbd->in->len); - lua_tcp_maybe_free (cbd); + REF_RELEASE (cbd); } } } @@ -309,18 +355,39 @@ lua_tcp_handler (int fd, short what, gpointer ud) } } else if (what == EV_WRITE) { + if (!cbd->connected) { + cbd->connected = TRUE; + + if (cbd->connect_cb != -1) { + struct lua_tcp_cbdata **pcbd; + + lua_rawgeti (cbd->L, LUA_REGISTRYINDEX, cbd->connect_cb); + pcbd = lua_newuserdata (cbd->L, sizeof (*pcbd)); + *pcbd = cbd; + REF_RETAIN (cbd); + rspamd_lua_setclass (cbd->L, "rspamd{tcp}", -1); + + if (lua_pcall (cbd->L, 1, 0, 0) != 0) { + msg_info ("callback call failed: %s", lua_tostring (cbd->L, -1)); + lua_pop (cbd->L, 1); + } + } + } + lua_tcp_write_helper (cbd); } #ifdef EV_CLOSED else if (what == EV_CLOSED) { lua_tcp_push_error (cbd, "Remote peer has closed the connection"); - lua_tcp_maybe_free (cbd); + REF_RELEASE (cbd); } #endif else { lua_tcp_push_error (cbd, "IO timeout"); - lua_tcp_maybe_free (cbd); + REF_RELEASE (cbd); } + + REF_RELEASE (cbd); } static gboolean @@ -354,7 +421,7 @@ lua_tcp_dns_handler (struct rdns_reply *reply, gpointer ud) rn = rdns_request_get_name (reply->request, NULL); lua_tcp_push_error (cbd, "unable to resolve host: %s", rn->name); - lua_tcp_maybe_free (cbd); + REF_RETAIN (cbd); } else { if (reply->entries->type == RDNS_REQUEST_A) { @@ -371,7 +438,7 @@ lua_tcp_dns_handler (struct rdns_reply *reply, gpointer ud) if (!lua_tcp_make_connection (cbd)) { lua_tcp_push_error (cbd, "unable to make connection to the host %s", rspamd_inet_address_to_string (cbd->addr)); - lua_tcp_maybe_free (cbd); + REF_RETAIN (cbd); } } } @@ -443,7 +510,7 @@ lua_tcp_request (lua_State *L) const gchar *host; gchar *stop_pattern = NULL; guint port; - gint cbref, tp; + gint cbref, tp, conn_cbref = -1; struct event_base *ev_base; struct lua_tcp_cbdata *cbd; struct rspamd_dns_resolver *resolver; @@ -529,6 +596,10 @@ lua_tcp_request (lua_State *L) lua_pop (L, 1); } + if (pool == NULL) { + return luaL_error (L, "tcp request has no memory pool associated"); + } + lua_pushstring (L, "timeout"); lua_gettable (L, -2); if (lua_type (L, -1) == LUA_TNUMBER) { @@ -557,11 +628,14 @@ lua_tcp_request (lua_State *L) } lua_pop (L, 1); - if (pool == NULL) { + lua_pushstring (L, "on_connect"); + lua_gettable (L, -2); + + if (lua_type (L, -1) == LUA_TFUNCTION) { + conn_cbref = luaL_ref (L, LUA_REGISTRYINDEX); + } + else { lua_pop (L, 1); - msg_err ("tcp request has no memory pool associated"); - lua_pushboolean (L, FALSE); - return 1; } lua_pushstring (L, "data"); @@ -633,6 +707,8 @@ lua_tcp_request (lua_State *L) cbd->pos = 0; cbd->port = port; cbd->stop_pattern = stop_pattern; + cbd->connect_cb = conn_cbref; + REF_INIT_RETAIN (cbd, lua_tcp_maybe_free); if (session) { cbd->session = session; @@ -648,7 +724,7 @@ lua_tcp_request (lua_State *L) rspamd_inet_address_set_port (cbd->addr, port); /* Host is numeric IP, no need to resolve */ if (!lua_tcp_make_connection (cbd)) { - lua_tcp_maybe_free (cbd); + REF_RELEASE (cbd); lua_pushboolean (L, FALSE); return 1; @@ -659,14 +735,14 @@ lua_tcp_request (lua_State *L) if (!make_dns_request (resolver, session, NULL, lua_tcp_dns_handler, cbd, RDNS_REQUEST_A, host)) { lua_tcp_push_error (cbd, "cannot resolve host: %s", host); - lua_tcp_maybe_free (cbd); + REF_RETAIN (cbd); } } else { if (!make_dns_request_task (task, lua_tcp_dns_handler, cbd, RDNS_REQUEST_A, host)) { lua_tcp_push_error (cbd, "cannot resolve host: %s", host); - lua_tcp_maybe_free (cbd); + REF_RELEASE (cbd); } } } @@ -676,6 +752,30 @@ lua_tcp_request (lua_State *L) } static gint +lua_tcp_close (lua_State *L) +{ + struct lua_tcp_cbdata *cbd = lua_check_tcp (L, 1); + + if (cbd == NULL) { + return luaL_error (L, "invalid arguments"); + } + + REF_RELEASE (cbd); + + return 0; +} + +static gint +lua_tcp_gc (lua_State *L) +{ + struct lua_tcp_cbdata *cbd = lua_check_tcp (L, 1); + + REF_RELEASE (cbd); + + return 0; +} + +static gint lua_load_tcp (lua_State * L) { lua_newtable (L); @@ -688,4 +788,6 @@ void luaopen_tcp (lua_State * L) { rspamd_lua_add_preload (L, "rspamd_tcp", lua_load_tcp); + rspamd_lua_new_class (L, "rspamd{tcp}", tcp_libm); + lua_pop (L, 1); } |