384 lines
11 KiB
C
384 lines
11 KiB
C
#include "api.h"
|
|
|
|
#define PCRE2_CODE_UNIT_WIDTH 8
|
|
|
|
#include <string.h>
|
|
#include <pcre2.h>
|
|
#include <stdbool.h>
|
|
|
|
typedef struct RegexState {
|
|
pcre2_code* re;
|
|
pcre2_match_data* match_data;
|
|
const char* subject;
|
|
size_t subject_len;
|
|
size_t offset;
|
|
bool regex_compiled;
|
|
bool found;
|
|
} RegexState;
|
|
|
|
static pcre2_code* regex_get_pattern(lua_State *L, bool* should_free) {
|
|
pcre2_code* re = NULL;
|
|
*should_free = false;
|
|
|
|
if (lua_type(L, 1) == LUA_TTABLE) {
|
|
lua_rawgeti(L, 1, 1);
|
|
re = (pcre2_code*)lua_touserdata(L, -1);
|
|
lua_settop(L, -2);
|
|
} else {
|
|
int errornumber;
|
|
PCRE2_SIZE erroroffset;
|
|
size_t pattern_len = 0;
|
|
const char* pattern = luaL_checklstring(L, 1, &pattern_len);
|
|
|
|
re = pcre2_compile(
|
|
(PCRE2_SPTR)pattern,
|
|
pattern_len, PCRE2_UTF,
|
|
&errornumber, &erroroffset, NULL
|
|
);
|
|
|
|
if (re == NULL) {
|
|
PCRE2_UCHAR errmsg[256];
|
|
pcre2_get_error_message(errornumber, errmsg, sizeof(errmsg));
|
|
luaL_error(
|
|
L, "regex pattern error at offset %d: %s",
|
|
(int)erroroffset, errmsg
|
|
);
|
|
return NULL;
|
|
}
|
|
|
|
if (pcre2_config(PCRE2_CONFIG_JIT, NULL) == 1) {
|
|
pcre2_jit_compile(re, PCRE2_JIT_COMPLETE);
|
|
}
|
|
|
|
*should_free = true;
|
|
}
|
|
|
|
return re;
|
|
}
|
|
|
|
static int regex_gmatch_iterator(lua_State *L) {
|
|
RegexState *state = (RegexState*)lua_touserdata(L, lua_upvalueindex(3));
|
|
|
|
if (state->found) {
|
|
int rc = pcre2_match(
|
|
state->re,
|
|
(PCRE2_SPTR)state->subject, state->subject_len,
|
|
state->offset, 0, state->match_data, NULL
|
|
);
|
|
|
|
if (rc < 0) {
|
|
if (rc != PCRE2_ERROR_NOMATCH) {
|
|
PCRE2_UCHAR buffer[120];
|
|
pcre2_get_error_message(rc, buffer, sizeof(buffer));
|
|
luaL_error(L, "regex matching error %d: %s", rc, buffer);
|
|
}
|
|
goto clean;
|
|
} else {
|
|
size_t ovector_count = pcre2_get_ovector_count(state->match_data);
|
|
if (ovector_count > 0) {
|
|
PCRE2_SIZE* ovector = pcre2_get_ovector_pointer(state->match_data);
|
|
if (ovector[0] > ovector[1]) {
|
|
/* We must guard against patterns such as /(?=.\K)/ that use \K in an
|
|
assertion to set the start of a match later than its end. In the editor,
|
|
we just detect this case and give up. */
|
|
luaL_error(L, "regex matching error: \\K was used in an assertion to "
|
|
" set the match start after its end");
|
|
goto clean;
|
|
}
|
|
|
|
int index = 0;
|
|
if (ovector_count > 1) index = 2;
|
|
|
|
int total = 0;
|
|
int total_results = ovector_count * 2;
|
|
size_t last_offset = 0;
|
|
for (int i = index; i < total_results; i+=2) {
|
|
lua_pushlstring(L, state->subject+ovector[i], ovector[i+1] - ovector[i]);
|
|
last_offset = ovector[i+1];
|
|
total++;
|
|
}
|
|
|
|
if (last_offset - 1 < state->subject_len)
|
|
state->offset = last_offset;
|
|
else
|
|
state->found = false;
|
|
|
|
return total;
|
|
} else {
|
|
state->found = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
clean:
|
|
if (state->regex_compiled) pcre2_code_free(state->re);
|
|
pcre2_match_data_free(state->match_data);
|
|
|
|
return 0; /* not found */
|
|
}
|
|
|
|
static size_t regex_offset_relative(lua_Integer pos, size_t len) {
|
|
if (pos > 0)
|
|
return (size_t)pos;
|
|
else if (pos == 0)
|
|
return 1;
|
|
else if (pos < -(lua_Integer)len) /* inverted comparison */
|
|
return 1; /* clip to 1 */
|
|
else return len + (size_t)pos + 1;
|
|
}
|
|
|
|
static int f_pcre_gc(lua_State* L) {
|
|
lua_rawgeti(L, -1, 1);
|
|
pcre2_code* re = (pcre2_code*)lua_touserdata(L, -1);
|
|
if (re)
|
|
pcre2_code_free(re);
|
|
return 0;
|
|
}
|
|
|
|
static int f_pcre_compile(lua_State *L) {
|
|
size_t len;
|
|
PCRE2_SIZE errorOffset;
|
|
int errorNumber;
|
|
int pattern = PCRE2_UTF;
|
|
const char* str = luaL_checklstring(L, 1, &len);
|
|
if (lua_gettop(L) > 1) {
|
|
const char* options = luaL_checkstring(L, 2);
|
|
if (strstr(options,"i"))
|
|
pattern |= PCRE2_CASELESS;
|
|
if (strstr(options,"m"))
|
|
pattern |= PCRE2_MULTILINE;
|
|
if (strstr(options,"s"))
|
|
pattern |= PCRE2_DOTALL;
|
|
}
|
|
pcre2_code* re = pcre2_compile(
|
|
(PCRE2_SPTR)str,
|
|
len,
|
|
pattern,
|
|
&errorNumber,
|
|
&errorOffset,
|
|
NULL
|
|
);
|
|
if (re) {
|
|
lua_newtable(L);
|
|
lua_pushlightuserdata(L, re);
|
|
lua_rawseti(L, -2, 1);
|
|
luaL_setmetatable(L, "regex");
|
|
return 1;
|
|
}
|
|
PCRE2_UCHAR buffer[256];
|
|
pcre2_get_error_message(errorNumber, buffer, sizeof(buffer));
|
|
lua_pushnil(L);
|
|
char message[1024];
|
|
len = snprintf(message, sizeof(message), "regex compilation failed at offset %d: %s", (int)errorOffset, buffer);
|
|
lua_pushlstring(L, message, len);
|
|
return 2;
|
|
}
|
|
|
|
// Takes string, compiled regex, returns list of indices of matched groups
|
|
// (including the whole match), if a match was found.
|
|
static int f_pcre_match(lua_State *L) {
|
|
size_t len, offset = 1, opts = 0;
|
|
bool regex_compiled = false;
|
|
pcre2_code* re = regex_get_pattern(L, ®ex_compiled);
|
|
if (!re) return 0 ;
|
|
const char* str = luaL_checklstring(L, 2, &len);
|
|
if (lua_gettop(L) > 2)
|
|
offset = regex_offset_relative(luaL_checknumber(L, 3), len);
|
|
offset -= 1;
|
|
len -= offset;
|
|
if (lua_gettop(L) > 3)
|
|
opts = luaL_checknumber(L, 4);
|
|
lua_rawgeti(L, 1, 1);
|
|
pcre2_match_data* md = pcre2_match_data_create_from_pattern(re, NULL);
|
|
int rc = pcre2_match(re, (PCRE2_SPTR)&str[offset], len, 0, opts, md, NULL);
|
|
if (rc < 0) {
|
|
if (regex_compiled) pcre2_code_free(re);
|
|
pcre2_match_data_free(md);
|
|
if (rc != PCRE2_ERROR_NOMATCH) {
|
|
PCRE2_UCHAR buffer[120];
|
|
pcre2_get_error_message(rc, buffer, sizeof(buffer));
|
|
luaL_error(L, "regex matching error %d: %s", rc, buffer);
|
|
}
|
|
return 0;
|
|
}
|
|
PCRE2_SIZE* ovector = pcre2_get_ovector_pointer(md);
|
|
if (ovector[0] > ovector[1]) {
|
|
/* We must guard against patterns such as /(?=.\K)/ that use \K in an
|
|
assertion to set the start of a match later than its end. In the editor,
|
|
we just detect this case and give up. */
|
|
luaL_error(L, "regex matching error: \\K was used in an assertion to "
|
|
" set the match start after its end");
|
|
if (regex_compiled) pcre2_code_free(re);
|
|
pcre2_match_data_free(md);
|
|
return 0;
|
|
}
|
|
for (int i = 0; i < rc*2; i++)
|
|
lua_pushinteger(L, ovector[i]+offset+1);
|
|
if (regex_compiled) pcre2_code_free(re);
|
|
pcre2_match_data_free(md);
|
|
return rc*2;
|
|
}
|
|
|
|
static int f_pcre_gmatch(lua_State *L) {
|
|
/* pattern param */
|
|
bool regex_compiled = false;
|
|
pcre2_code* re = regex_get_pattern(L, ®ex_compiled);
|
|
if (!re) return 0;
|
|
size_t subject_len = 0;
|
|
|
|
/* subject param */
|
|
const char* subject = luaL_checklstring(L, 2, &subject_len);
|
|
|
|
/* offset param */
|
|
size_t offset = regex_offset_relative(
|
|
luaL_optnumber(L, 3, 1), subject_len
|
|
) - 1;
|
|
|
|
/* keep strings on closure to avoid being collected */
|
|
lua_settop(L, 2);
|
|
|
|
RegexState *state;
|
|
state = (RegexState*)lua_newuserdata(L, sizeof(RegexState));
|
|
|
|
state->re = re;
|
|
state->match_data = pcre2_match_data_create_from_pattern(re, NULL);
|
|
state->subject = subject;
|
|
state->subject_len = subject_len;
|
|
state->offset = offset;
|
|
state->found = true;
|
|
state->regex_compiled = regex_compiled;
|
|
|
|
lua_pushcclosure(L, regex_gmatch_iterator, 3);
|
|
return 1;
|
|
}
|
|
|
|
static int f_pcre_gsub(lua_State *L) {
|
|
size_t subject_len = 0, replacement_len = 0;
|
|
|
|
bool regex_compiled = false;
|
|
pcre2_code* re = regex_get_pattern(L, ®ex_compiled);
|
|
if (!re) return 0 ;
|
|
|
|
char* subject = (char*) luaL_checklstring(L, 2, &subject_len);
|
|
const char* replacement = luaL_checklstring(L, 3, &replacement_len);
|
|
int limit = luaL_optinteger(L, 4, 0);
|
|
if (limit < 0 ) limit = 0;
|
|
|
|
pcre2_match_data* match_data = pcre2_match_data_create_from_pattern(re, NULL);
|
|
|
|
size_t buffer_size = 1024;
|
|
char *output = (char *)malloc(buffer_size);
|
|
|
|
int options = PCRE2_SUBSTITUTE_OVERFLOW_LENGTH | PCRE2_SUBSTITUTE_EXTENDED;
|
|
if (limit == 0) options |= PCRE2_SUBSTITUTE_GLOBAL;
|
|
|
|
int results_count = 0;
|
|
int limit_count = 0;
|
|
bool done = false;
|
|
size_t offset = 0;
|
|
PCRE2_SIZE outlen = buffer_size;
|
|
while (!done) {
|
|
results_count = pcre2_substitute(
|
|
re,
|
|
(PCRE2_SPTR)subject, subject_len,
|
|
offset, options,
|
|
match_data, NULL,
|
|
(PCRE2_SPTR)replacement, replacement_len,
|
|
(PCRE2_UCHAR*)output, &outlen
|
|
);
|
|
|
|
if (results_count != PCRE2_ERROR_NOMEMORY || buffer_size >= outlen) {
|
|
/* PCRE2_SUBSTITUTE_GLOBAL code path (fastest) */
|
|
if(limit == 0) {
|
|
done = true;
|
|
/* non PCRE2_SUBSTITUTE_GLOBAL with limit code path (slower) */
|
|
} else {
|
|
size_t ovector_count = pcre2_get_ovector_count(match_data);
|
|
if (results_count > 0 && ovector_count > 0) {
|
|
limit_count++;
|
|
PCRE2_SIZE* ovector = pcre2_get_ovector_pointer(match_data);
|
|
if (outlen > subject_len) {
|
|
offset = ovector[1] + (outlen - subject_len);
|
|
} else {
|
|
offset = ovector[1] - (subject_len - outlen);
|
|
}
|
|
if (limit_count > 1) free(subject);
|
|
if (limit_count == limit || offset-1 == outlen) {
|
|
done = true;
|
|
results_count = limit_count;
|
|
} else {
|
|
subject = output;
|
|
subject_len = outlen;
|
|
output = (char *)malloc(buffer_size);
|
|
outlen = buffer_size;
|
|
}
|
|
} else {
|
|
if (limit_count > 1) {
|
|
free(subject);
|
|
}
|
|
done = true;
|
|
results_count = limit_count;
|
|
}
|
|
}
|
|
} else {
|
|
buffer_size = outlen;
|
|
output = (char *)realloc(output, buffer_size);
|
|
}
|
|
}
|
|
|
|
int return_count = 0;
|
|
|
|
if (results_count > 0) {
|
|
lua_pushlstring(L, (const char*) output, outlen);
|
|
lua_pushinteger(L, results_count);
|
|
return_count = 2;
|
|
} else if (results_count == 0) {
|
|
lua_pushlstring(L, subject, subject_len);
|
|
lua_pushinteger(L, 0);
|
|
return_count = 2;
|
|
}
|
|
|
|
free(output);
|
|
pcre2_match_data_free(match_data);
|
|
if (regex_compiled)
|
|
pcre2_code_free(re);
|
|
|
|
if (results_count < 0) {
|
|
PCRE2_UCHAR errmsg[256];
|
|
pcre2_get_error_message(results_count, errmsg, sizeof(errmsg));
|
|
return luaL_error(L, "regex substitute error: %s", errmsg);
|
|
}
|
|
|
|
return return_count;
|
|
}
|
|
|
|
static const luaL_Reg lib[] = {
|
|
{ "compile", f_pcre_compile },
|
|
{ "cmatch", f_pcre_match },
|
|
{ "gmatch", f_pcre_gmatch },
|
|
{ "gsub", f_pcre_gsub },
|
|
{ "__gc", f_pcre_gc },
|
|
{ NULL, NULL }
|
|
};
|
|
|
|
int luaopen_regex(lua_State *L) {
|
|
luaL_newlib(L, lib);
|
|
lua_pushliteral(L, "regex");
|
|
lua_setfield(L, -2, "__name");
|
|
lua_pushvalue(L, -1);
|
|
lua_setfield(L, LUA_REGISTRYINDEX, "regex");
|
|
lua_pushinteger(L, PCRE2_ANCHORED);
|
|
lua_setfield(L, -2, "ANCHORED");
|
|
lua_pushinteger(L, PCRE2_ANCHORED) ;
|
|
lua_setfield(L, -2, "ENDANCHORED");
|
|
lua_pushinteger(L, PCRE2_NOTBOL);
|
|
lua_setfield(L, -2, "NOTBOL");
|
|
lua_pushinteger(L, PCRE2_NOTEOL);
|
|
lua_setfield(L, -2, "NOTEOL");
|
|
lua_pushinteger(L, PCRE2_NOTEMPTY);
|
|
lua_setfield(L, -2, "NOTEMPTY");
|
|
lua_pushinteger(L, PCRE2_NOTEMPTY_ATSTART);
|
|
lua_setfield(L, -2, "NOTEMPTY_ATSTART");
|
|
return 1;
|
|
}
|