2020-09-30 16:53:55 +02:00

103 lines
3.0 KiB
C++

#include <regexp.h>
BOOL test_match(int m, LPSTR target, int pattern[]) /* m = length of target */
{
int *match;
int i = -1; /* Will be advanced to 0 */
int j = 0; /* i = index to pattern, j = index to target */
BOOL fResult = FALSE;
match = new int[INTERNET_MAX_URL_LENGTH];
if (match == NULL)
goto Cleanup;
advance:
++i;
if (j > m)
goto Cleanup;
switch (pattern[i]) {
case PAT_START: if (j != 0) goto Cleanup; match[i] = 0; goto advance;
case PAT_END: if (target[j] == 0) {fResult = TRUE; goto Cleanup;} else goto retreat;
case PAT_STAR: match[i] = j = m; goto advance;
case PAT_QUES: if (j < m) goto match_one; else goto retreat;
case PAT_AUGDOT: if (target[j] == '.') goto match_one;
else if (target[j] == 0) goto match_zero;
else goto retreat;
case PAT_AUGQUES: if (target[j] && target[j] != '.')
goto match_one; else goto match_zero;
case PAT_AUGSTAR: if (target[j] && target[j] != '.')
goto match_one; else goto retreat;
default: if (target[j] == pattern[i])
goto match_one; else goto retreat;
}
match_one: match[i] = ++j; goto advance;
match_zero: match[i] = j; goto advance;
retreat:
--i;
switch (pattern[i]) {
case PAT_START: goto Cleanup;
case PAT_END: goto Cleanup; /* Cannot happen */
case PAT_STAR: if (match[i] == match[i-1]) goto retreat;
j = --match[i]; goto advance;
case PAT_QUES: goto retreat;
case PAT_AUGDOT: goto retreat;
case PAT_AUGQUES: if (match[i] == match[i-1]) goto retreat;
j = --match[i]; goto advance;
case PAT_AUGSTAR: goto retreat;
default: goto retreat;
}
Cleanup:
if (match)
delete [] match;
return fResult;
}
BOOL parse_pattern(LPSTR s, int pattern[])
{
int i = 1;
pattern[0] = PAT_START; /* Can be hard-coded into pattern[] */
for (;;) {
switch (*s) {
case '*': pattern[i] = PAT_STAR; break;
case '?': pattern[i] = PAT_QUES; break;
case '^':
switch (*++s) {
case '.': pattern[i] = PAT_AUGDOT; break;
case '?': pattern[i] = PAT_AUGQUES; break;
case '*': pattern[i] = PAT_AUGSTAR; break;
default: return FALSE;
}
break;
case 0: pattern[i] = PAT_END; return TRUE;
default: pattern[i] = *s; break;
}
if (++i >= INTERNET_MAX_URL_LENGTH) return FALSE;
++s;
}
}
BOOL match( LPSTR target, LPSTR regexp)
{
int *pattern;
BOOL result;
pattern = new int[INTERNET_MAX_URL_LENGTH];
if (!target || (pattern==NULL))
return FALSE;
if (!parse_pattern(regexp,pattern))
return FALSE;
if (lstrlen(target) >= INTERNET_MAX_URL_LENGTH)
return FALSE;
result = test_match(lstrlen(target),target,pattern);
delete [] pattern;
return result;
}