Skip to content

Commit

Permalink
Port pcre dependency to pcre2
Browse files Browse the repository at this point in the history
Fixes rsyslog#366
Upstream stopped the support for the old 'pcre' package. See
https://www.pcre.org/
It only supports the new 'pcre2' version, we should replace
the current pcre with the newer pcre2.
  • Loading branch information
Cropi committed Jun 6, 2023
1 parent 1e18f60 commit 305a017
Show file tree
Hide file tree
Showing 3 changed files with 43 additions and 23 deletions.
6 changes: 3 additions & 3 deletions configure.ac
Original file line number Diff line number Diff line change
Expand Up @@ -80,7 +80,8 @@ AC_ARG_ENABLE(regexp,
)
AM_CONDITIONAL(ENABLE_REGEXP, test x$enable_regexp = xyes)
if test "$enable_regexp" = "yes"; then
PKG_CHECK_MODULES(PCRE, libpcre)
# PKG_CHECK_MODULES(PCRE, libpcre)
PKG_CHECK_MODULES(PCRE, [libpcre2-8 >= 10.00])
AC_DEFINE(FEATURE_REGEXP, 1, [Regular expressions support enabled.])
FEATURE_REGEXP=1
else
Expand Down Expand Up @@ -194,5 +195,4 @@ echo "Testbench enabled: $enable_testbench"
echo "Valgrind enabled: $enable_valgrind"
echo "Debug mode enabled: $enable_debug"
echo "Tools enabled: $enable_tools"
echo "Docs enabled: $enable_docs"

echo "Docs enabled: $enable_docs"
8 changes: 4 additions & 4 deletions src/parser.c
Original file line number Diff line number Diff line change
Expand Up @@ -42,10 +42,10 @@
#include "samp.h"
#include "helpers.h"

#ifdef FEATURE_REGEXP
#include <pcre.h>
#include <errno.h>
#endif
// #ifdef FEATURE_REGEXP
// #include <pcre.h>
// #include <errno.h>
// #endif


/* how should output values be formatted? */
Expand Down
52 changes: 36 additions & 16 deletions src/v1_parser.c
Original file line number Diff line number Diff line change
Expand Up @@ -39,7 +39,8 @@
#include "v1_samp.h"

#ifdef FEATURE_REGEXP
#include <pcre.h>
#define PCRE2_CODE_UNIT_WIDTH 8
#include <pcre2.h>
#include <errno.h>
#endif

Expand Down Expand Up @@ -1266,7 +1267,7 @@ void* tokenized_parser_data_constructor(ln_fieldList_t *node, ln_ctx ctx) {
* significantly slower than other field-types.
*/
struct regex_parser_data_s {
pcre *re;
pcre2_code *re;
int consume_group;
int return_group;
int max_groups;
Expand All @@ -1276,40 +1277,54 @@ PARSER(Regex)
assert(str != NULL);
assert(offs != NULL);
assert(parsed != NULL);
unsigned int* ovector = NULL;
PCRE2_SIZE *ovector;
pcre2_match_data *match_data = NULL;

struct regex_parser_data_s *pData = (struct regex_parser_data_s*) node->parser_data;
if (pData != NULL) {
ovector = calloc(pData->max_groups, sizeof(unsigned int) * 3);
if (ovector == NULL) FAIL(LN_NOMEM);
match_data = pcre2_match_data_create_from_pattern(pData->re, NULL);
if (match_data == NULL) FAIL(LN_NOMEM);

int result = pcre2_match(
pData->re, /* the compiled pattern */
(PCRE2_SPTR)str, /* the subject string */
(PCRE2_SIZE)strLen, /* the length of the subject */
(PCRE2_SIZE)*offs, /* start at offset 0 in the subject */
0, /* default options */
match_data, /* block for storing the result */
NULL); /* use default match context */

int result = pcre_exec(pData->re, NULL, str, strLen, *offs, 0, (int*) ovector, pData->max_groups * 3);
if (result == 0) result = pData->max_groups;
if (result > pData->consume_group) {
/*please check 'man 3 pcreapi' for cryptic '2 * n' and '2 * n + 1' magic*/
ovector = pcre2_get_ovector_pointer(match_data);
printf("Match succeeded at offset %d\n", (int)ovector[0]);

/* please check 'man 3 pcre2api' for cryptic '2 * n' and '2 * n + 1' magic
* in a nutshell, within the ovector, the first in each pair of values is set to the
* offset of the first code unit of a substring, and the second is set to the
* offset of the first code unit after the end of a substring.
*/
if (ovector[2 * pData->consume_group] == *offs) {
*parsed = ovector[2 * pData->consume_group + 1] - ovector[2 * pData->consume_group];
if (pData->consume_group != pData->return_group) {
char* val = NULL;
if((val = strndup(str + ovector[2 * pData->return_group],
ovector[2 * pData->return_group + 1] -
ovector[2 * pData->return_group])) == NULL) {
free(ovector);
FAIL(LN_NOMEM);
}
*value = json_object_new_string(val);
free(val);
if (*value == NULL) {
free(ovector);
FAIL(LN_NOMEM);
}
}
}
}
free(ovector);
}
r = 0; /* success */
done:
pcre2_match_data_free(match_data);
return r;
}

Expand Down Expand Up @@ -1346,8 +1361,9 @@ void* regex_parser_data_constructor(ln_fieldList_t *node, ln_ctx ctx) {
char* name = NULL;
struct regex_parser_data_s *pData = NULL;
const char *unescaped_exp = NULL;
const char *error = NULL;
int erroffset = 0;
// const char *error = NULL;
PCRE2_SIZE erroffset = 0;
int errcode = 0;


CHKN(name = es_str2cstr(node->name, NULL));
Expand All @@ -1365,7 +1381,8 @@ void* regex_parser_data_constructor(ln_fieldList_t *node, ln_ctx ctx) {
if ((grp_parse_err = regex_parser_configure_consume_and_return_group(args, pData)) != NULL)
FAIL(LN_BADCONFIG);

CHKN(pData->re = pcre_compile(exp, 0, &error, &erroffset, NULL));
// CHKN(pData->re = pcre_compile(exp, 0, &error, &erroffset, NULL));
CHKN(pData->re = pcre2_compile((PCRE2_SPTR)exp, PCRE2_ZERO_TERMINATED, 0, &errcode, &erroffset, NULL));

pData->max_groups = ((pData->consume_group > pData->return_group) ? pData->consume_group :
pData->return_group) + 1;
Expand All @@ -1387,9 +1404,12 @@ void* regex_parser_data_constructor(ln_fieldList_t *node, ln_ctx ctx) {
ln_dbgprintf(ctx, "couldn't allocate memory for regex-string for field: '%s'", name);
else if (grp_parse_err != NULL)
ln_dbgprintf(ctx, "%s for: '%s'", grp_parse_err, name);
else if (pData->re == NULL)
else if (pData->re == NULL) {
PCRE2_UCHAR errbuffer[256];
pcre2_get_error_message(errcode, errbuffer, sizeof(errbuffer));
ln_dbgprintf(ctx, "couldn't compile regex(encountered error '%s' at char '%d' in pattern) "
"for regex-matched field: '%s'", error, erroffset, name);
"for regex-matched field: '%s'", errbuffer, (int)erroffset, name);
}
regex_parser_data_destructor((void**)&pData);
}
if (exp != NULL) free(exp);
Expand All @@ -1401,7 +1421,7 @@ void* regex_parser_data_constructor(ln_fieldList_t *node, ln_ctx ctx) {
void regex_parser_data_destructor(void** dataPtr) {
if ((*dataPtr) != NULL) {
struct regex_parser_data_s *pData = (struct regex_parser_data_s*) *dataPtr;
if (pData->re != NULL) pcre_free(pData->re);
if (pData->re != NULL) pcre2_code_free(pData->re);
free(pData);
*dataPtr = NULL;
}
Expand Down

0 comments on commit 305a017

Please sign in to comment.