Repository URL to install this package:
|
Version:
3.10.1 ▾
|
/* rxclass.c
* Copyright (c) 2017, Peter Ohler
* All rights reserved.
*/
#include <sys/types.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
#include <stdio.h>
#if !IS_WINDOWS
#include <regex.h>
#endif
#include "rxclass.h"
typedef struct _rxC {
struct _rxC *next;
VALUE rrx;
#if !IS_WINDOWS
regex_t rx;
#endif
VALUE clas;
char src[256];
} *RxC;
void
oj_rxclass_init(RxClass rc) {
*rc->err = '\0';
rc->head = NULL;
rc->tail = NULL;
}
void
oj_rxclass_cleanup(RxClass rc) {
RxC rxc;
while (NULL != (rxc = rc->head)) {
rc->head = rc->head->next;
#if !IS_WINDOWS
if (Qnil == rxc->rrx) {
regfree(&rxc->rx);
}
xfree(rxc);
#endif
}
}
void
oj_rxclass_rappend(RxClass rc, VALUE rx, VALUE clas) {
RxC rxc = ALLOC_N(struct _rxC, 1);
memset(rxc, 0, sizeof(struct _rxC));
rxc->rrx = rx;
rxc->clas = clas;
if (NULL == rc->tail) {
rc->head = rxc;
} else {
rc->tail->next = rxc;
}
rc->tail = rxc;
}
// Attempt to compile the expression. If it fails populate the error code..
int
oj_rxclass_append(RxClass rc, const char *expr, VALUE clas) {
RxC rxc;
#if !IS_WINDOWS
int err;
int flags = 0;
#endif
if (sizeof(rxc->src) <= strlen(expr)) {
snprintf(rc->err, sizeof(rc->err), "expressions must be less than %lu characters", (unsigned long)sizeof(rxc->src));
return EINVAL;
}
rxc = ALLOC_N(struct _rxC, 1);
rxc->next = 0;
rxc->clas = clas;
#if IS_WINDOWS
rxc->rrx = rb_funcall(rb_cRegexp, rb_intern("new"), 1, rb_str_new2(expr));
#else
rxc->rrx = Qnil;
if (0 != (err = regcomp(&rxc->rx, expr, flags))) {
regerror(err, &rxc->rx, rc->err, sizeof(rc->err));
free(rxc);
return err;
}
#endif
if (NULL == rc->tail) {
rc->head = rxc;
} else {
rc->tail->next = rxc;
}
rc->tail = rxc;
return 0;
}
VALUE
oj_rxclass_match(RxClass rc, const char *str, int len) {
RxC rxc;
char buf[4096];
for (rxc = rc->head; NULL != rxc; rxc = rxc->next) {
if (Qnil != rxc->rrx) {
// Must use a valiabel for this to work.
volatile VALUE rstr = rb_str_new(str, len);
//if (Qtrue == rb_funcall(rxc->rrx, rb_intern("match?"), 1, rstr)) {
if (Qnil != rb_funcall(rxc->rrx, rb_intern("match"), 1, rstr)) {
return rxc->clas;
}
} else if (len < (int)sizeof(buf)) {
#if !IS_WINDOWS
// string is not \0 terminated so copy and atempt a match
memcpy(buf, str, len);
buf[len] = '\0';
if (0 == regexec(&rxc->rx, buf, 0, NULL, 0)) { // match
return rxc->clas;
}
#endif
} else {
// TBD allocate a larger buffer and attempt
}
}
return Qnil;
}
void
oj_rxclass_copy(RxClass src, RxClass dest) {
dest->head = NULL;
dest->tail = NULL;
if (NULL != src->head) {
RxC rxc;
for (rxc = src->head; NULL != rxc; rxc = rxc->next) {
if (Qnil != rxc->rrx) {
oj_rxclass_rappend(dest, rxc->rrx, rxc->clas);
} else {
#if !IS_WINDOWS
oj_rxclass_append(dest, rxc->src, rxc->clas);
#endif
}
}
}
}