diff options
author | Vsevolod Stakhov <vsevolod@highsecure.ru> | 2020-07-24 14:04:11 +0100 |
---|---|---|
committer | Vsevolod Stakhov <vsevolod@highsecure.ru> | 2020-07-24 20:33:33 +0100 |
commit | 53984d130b23f2c692ca521fda8c66c98bc021d2 (patch) | |
tree | 282fdc0f13d1b67d6b87a3056342f933c0ded863 | |
parent | 1a634c3d5103c9deb6f5fe1c236b5719995f8d11 (diff) | |
download | rspamd-53984d130b23f2c692ca521fda8c66c98bc021d2.tar.gz rspamd-53984d130b23f2c692ca521fda8c66c98bc021d2.zip |
[Project] Lua_text: Add method memchr
-rw-r--r-- | src/lua/lua_text.c | 65 |
1 files changed, 65 insertions, 0 deletions
diff --git a/src/lua/lua_text.c b/src/lua/lua_text.c index fcaa9b90a..663e4d49b 100644 --- a/src/lua/lua_text.c +++ b/src/lua/lua_text.c @@ -120,6 +120,15 @@ LUA_FUNCTION_DEF (text, split); */ LUA_FUNCTION_DEF (text, at); /*** + * @method rspamd_text:memchr(chr, [reverse]) + * Returns the first or the last position of the character `chr` in the text or + * -1 in case if a character has not been found. Indexes start from `1` + * @param {string/number} chr character or a character code to find + * @param {boolean} reverse last character if `true` + * @return {integer} position of the character or `-1` + */ +LUA_FUNCTION_DEF (text, memchr); +/*** * @method rspamd_text:bytes() * Converts text to an array of bytes * @return {table|integer} bytes in the array (as unsigned char) @@ -214,6 +223,7 @@ static const struct luaL_reg textlib_m[] = { LUA_INTERFACE_DEF (text, lines), LUA_INTERFACE_DEF (text, split), LUA_INTERFACE_DEF (text, at), + LUA_INTERFACE_DEF (text, memchr), LUA_INTERFACE_DEF (text, bytes), LUA_INTERFACE_DEF (text, lower), LUA_INTERFACE_DEF (text, exclude_chars), @@ -924,6 +934,61 @@ lua_text_at (lua_State *L) } static gint +lua_text_memchr (lua_State *L) +{ + LUA_TRACE_POINT; + struct rspamd_lua_text *t = lua_check_text (L, 1); + int c; + bool reverse = false; + + if (lua_isnumber (L, 2)) { + c = lua_tonumber (L, 2); + } + else { + gsize l; + const gchar *str = lua_tolstring (L, 2, &l); + + if (str) { + c = str[0]; + + if (l != 1) { + return luaL_error (L, "need exactly one character to search"); + } + } + else { + return luaL_error (L, "invalid arguments"); + } + } + + if (t) { + void *f; + + if (lua_isboolean (L, 3)) { + reverse = lua_toboolean (L, 3); + } + + if (reverse) { + f = rspamd_memrchr (t->start, c, t->len); + } + else { + f = memchr (t->start, c, t->len); + } + + if (f) { + lua_pushinteger (L, ((const char *)f) - t->start + 1); + } + else { + lua_pushinteger (L, -1); + } + } + else { + return luaL_error (L, "invalid arguments"); + } + + return 1; +} + +static gint lua_text_bytes (lua_State *L) { LUA_TRACE_POINT; |