1999-10-27 05:42:43 +02:00
|
|
|
/*
|
1999-11-24 14:26:21 +01:00
|
|
|
*
|
|
|
|
* match.c
|
|
|
|
*
|
|
|
|
* Author: Tatu Ylonen <ylo@cs.hut.fi>
|
|
|
|
*
|
|
|
|
* Copyright (c) 1995 Tatu Ylonen <ylo@cs.hut.fi>, Espoo, Finland
|
|
|
|
* All rights reserved
|
|
|
|
*
|
|
|
|
* Created: Thu Jun 22 01:17:50 1995 ylo
|
|
|
|
*
|
|
|
|
* Simple pattern matching, with '*' and '?' as wildcards.
|
|
|
|
*
|
|
|
|
*/
|
1999-10-27 05:42:43 +02:00
|
|
|
|
|
|
|
#include "includes.h"
|
2000-03-26 05:04:51 +02:00
|
|
|
RCSID("$Id: match.c,v 1.4 2000/03/26 03:04:53 damien Exp $");
|
1999-10-27 05:42:43 +02:00
|
|
|
|
|
|
|
#include "ssh.h"
|
|
|
|
|
1999-11-25 01:54:57 +01:00
|
|
|
/*
|
|
|
|
* Returns true if the given string matches the pattern (which may contain ?
|
|
|
|
* and * as wildcards), and zero if it does not match.
|
|
|
|
*/
|
1999-10-27 05:42:43 +02:00
|
|
|
|
1999-11-24 14:26:21 +01:00
|
|
|
int
|
|
|
|
match_pattern(const char *s, const char *pattern)
|
|
|
|
{
|
|
|
|
for (;;) {
|
|
|
|
/* If at end of pattern, accept if also at end of string. */
|
|
|
|
if (!*pattern)
|
|
|
|
return !*s;
|
1999-10-27 05:42:43 +02:00
|
|
|
|
1999-11-24 14:26:21 +01:00
|
|
|
if (*pattern == '*') {
|
|
|
|
/* Skip the asterisk. */
|
|
|
|
pattern++;
|
1999-10-27 05:42:43 +02:00
|
|
|
|
1999-11-24 14:26:21 +01:00
|
|
|
/* If at end of pattern, accept immediately. */
|
|
|
|
if (!*pattern)
|
|
|
|
return 1;
|
1999-10-27 05:42:43 +02:00
|
|
|
|
1999-11-24 14:26:21 +01:00
|
|
|
/* If next character in pattern is known, optimize. */
|
|
|
|
if (*pattern != '?' && *pattern != '*') {
|
1999-11-25 01:54:57 +01:00
|
|
|
/*
|
|
|
|
* Look instances of the next character in
|
|
|
|
* pattern, and try to match starting from
|
|
|
|
* those.
|
|
|
|
*/
|
1999-11-24 14:26:21 +01:00
|
|
|
for (; *s; s++)
|
|
|
|
if (*s == *pattern &&
|
|
|
|
match_pattern(s + 1, pattern + 1))
|
|
|
|
return 1;
|
|
|
|
/* Failed. */
|
|
|
|
return 0;
|
|
|
|
}
|
1999-11-25 01:54:57 +01:00
|
|
|
/*
|
|
|
|
* Move ahead one character at a time and try to
|
|
|
|
* match at each position.
|
|
|
|
*/
|
1999-11-24 14:26:21 +01:00
|
|
|
for (; *s; s++)
|
|
|
|
if (match_pattern(s, pattern))
|
|
|
|
return 1;
|
|
|
|
/* Failed. */
|
|
|
|
return 0;
|
|
|
|
}
|
1999-11-25 01:54:57 +01:00
|
|
|
/*
|
|
|
|
* There must be at least one more character in the string.
|
|
|
|
* If we are at the end, fail.
|
|
|
|
*/
|
1999-11-24 14:26:21 +01:00
|
|
|
if (!*s)
|
|
|
|
return 0;
|
1999-10-27 05:42:43 +02:00
|
|
|
|
1999-11-25 01:54:57 +01:00
|
|
|
/* Check if the next character of the string is acceptable. */
|
1999-11-24 14:26:21 +01:00
|
|
|
if (*pattern != '?' && *pattern != *s)
|
|
|
|
return 0;
|
1999-10-27 05:42:43 +02:00
|
|
|
|
1999-11-25 01:54:57 +01:00
|
|
|
/* Move to the next character, both in string and in pattern. */
|
1999-11-24 14:26:21 +01:00
|
|
|
s++;
|
|
|
|
pattern++;
|
|
|
|
}
|
|
|
|
/* NOTREACHED */
|
1999-10-27 05:42:43 +02:00
|
|
|
}
|
2000-03-26 05:04:51 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Tries to match the host name (which must be in all lowercase) against the
|
|
|
|
* comma-separated sequence of subpatterns (each possibly preceded by ! to
|
|
|
|
* indicate negation). Returns true if there is a positive match; zero
|
|
|
|
* otherwise.
|
|
|
|
*/
|
|
|
|
|
|
|
|
int
|
|
|
|
match_hostname(const char *host, const char *pattern, unsigned int len)
|
|
|
|
{
|
|
|
|
char sub[1024];
|
|
|
|
int negated;
|
|
|
|
int got_positive;
|
|
|
|
unsigned int i, subi;
|
|
|
|
|
|
|
|
got_positive = 0;
|
|
|
|
for (i = 0; i < len;) {
|
|
|
|
/* Check if the subpattern is negated. */
|
|
|
|
if (pattern[i] == '!') {
|
|
|
|
negated = 1;
|
|
|
|
i++;
|
|
|
|
} else
|
|
|
|
negated = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Extract the subpattern up to a comma or end. Convert the
|
|
|
|
* subpattern to lowercase.
|
|
|
|
*/
|
|
|
|
for (subi = 0;
|
|
|
|
i < len && subi < sizeof(sub) - 1 && pattern[i] != ',';
|
|
|
|
subi++, i++)
|
|
|
|
sub[subi] = isupper(pattern[i]) ? tolower(pattern[i]) : pattern[i];
|
|
|
|
/* If subpattern too long, return failure (no match). */
|
|
|
|
if (subi >= sizeof(sub) - 1)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
/* If the subpattern was terminated by a comma, skip the comma. */
|
|
|
|
if (i < len && pattern[i] == ',')
|
|
|
|
i++;
|
|
|
|
|
|
|
|
/* Null-terminate the subpattern. */
|
|
|
|
sub[subi] = '\0';
|
|
|
|
|
|
|
|
/* Try to match the subpattern against the host name. */
|
|
|
|
if (match_pattern(host, sub)) {
|
|
|
|
if (negated)
|
|
|
|
return 0; /* Fail */
|
|
|
|
else
|
|
|
|
got_positive = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return success if got a positive match. If there was a negative
|
|
|
|
* match, we have already returned zero and never get here.
|
|
|
|
*/
|
|
|
|
return got_positive;
|
|
|
|
}
|