/* Copyright 2016 Alex 'AdUser' Z (ad_user@runbox.com) * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as * published by the Free Software Foundation. */ #include "filter.h" #include /* draft */ #define HOST_REGEX "([0-9]{1,3}\\.[0-9]{1,3}\\.[0-9]{1,3}\\.[0-9]{1,3})" typedef struct f2b_regex_t { struct f2b_regex_t *next; char pattern[PATTERN_MAX]; int matches; regex_t regex; } f2b_regex_t; struct _config { char id[ID_MAX]; char error[256]; bool icase; f2b_regex_t *regexps; f2b_regex_t *statp; }; cfg_t * create(const char *id) { cfg_t *cfg = NULL; if ((cfg = calloc(1, sizeof(cfg_t))) == NULL) return NULL; strlcpy(cfg->id, id, sizeof(cfg->id)); return cfg; } bool config(cfg_t *cfg, const char *key, const char *value) { assert(cfg != NULL); assert(key != NULL); assert(value != NULL); if (strcmp(key, "icase") == 0) { cfg->icase = (strcmp(value, "yes") == 0) ? true : false; return true; } return false; } bool append(cfg_t *cfg, const char *pattern) { f2b_regex_t *regex = NULL; int flags = REG_EXTENDED; size_t bufsize; char *buf = NULL; char *token = NULL; assert(pattern != NULL); if (cfg->icase) flags |= REG_ICASE; if ((token = strstr(pattern, HOST_TOKEN)) == NULL) return false; bufsize = strlen(pattern) + strlen(HOST_REGEX) + 1; if ((buf = alloca(bufsize)) == NULL) return false; memset(buf, 0x0, bufsize); memcpy(buf, pattern, token - pattern); strlcat(buf, HOST_REGEX, bufsize); strlcat(buf, token + strlen(HOST_TOKEN), bufsize); if ((regex = calloc(1, sizeof(f2b_regex_t))) == NULL) return false; if (regcomp(®ex->regex, buf, flags) == 0) { regex->next = cfg->regexps; cfg->regexps = regex; strlcpy(regex->pattern, pattern, sizeof(regex->pattern)); return true; } free(regex); return false; } bool ready(cfg_t *cfg) { assert(cfg != NULL); if (cfg->regexps) return true; return false; } bool stats(cfg_t *cfg, int *matches, char **pattern, bool reset) { assert(cfg != NULL); if (reset) cfg->statp = cfg->regexps; if (cfg->statp) { *matches = cfg->statp->matches; *pattern = cfg->statp->pattern; cfg->statp = cfg->statp->next; return true; } return false; } const char * error(cfg_t *cfg) { assert(cfg != NULL); return cfg->error; } bool match(cfg_t *cfg, const char *line, char *buf, size_t buf_size) { f2b_regex_t *r = NULL; size_t match_len = 0; regmatch_t match[2]; assert(cfg != NULL); assert(line != NULL); assert(buf != NULL); for (r = cfg->regexps; r != NULL; r = r->next) { if (regexec(&r->regex, line, 2, &match[0], 0) != 0) continue; /* matched */ r->matches++; match_len = match[1].rm_eo - match[1].rm_so; assert(buf_size > match_len); memcpy(buf, &line[match[1].rm_so], match_len); buf[match_len] = '\0'; buf[buf_size - 1] = '\0'; return true; } return false; } void flush(cfg_t *cfg) { f2b_regex_t *next = NULL, *r = NULL; assert(cfg != NULL); for (r = cfg->regexps; r != NULL; r = next) { next = r->next; regfree(&r->regex); free(r); } cfg->regexps = NULL; } void destroy(cfg_t *cfg) { assert(cfg != NULL); flush(cfg); free(cfg); }