summaryrefslogtreecommitdiffstats
path: root/src/lua/lua_text.c
diff options
context:
space:
mode:
authorVsevolod Stakhov <vsevolod@highsecure.ru>2020-07-24 14:04:11 +0100
committerVsevolod Stakhov <vsevolod@highsecure.ru>2020-07-24 20:33:33 +0100
commit53984d130b23f2c692ca521fda8c66c98bc021d2 (patch)
tree282fdc0f13d1b67d6b87a3056342f933c0ded863 /src/lua/lua_text.c
parent1a634c3d5103c9deb6f5fe1c236b5719995f8d11 (diff)
downloadrspamd-53984d130b23f2c692ca521fda8c66c98bc021d2.tar.gz
rspamd-53984d130b23f2c692ca521fda8c66c98bc021d2.zip
[Project] Lua_text: Add method memchr
Diffstat (limited to 'src/lua/lua_text.c')
-rw-r--r--src/lua/lua_text.c65
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;