simple_pattern.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419
  1. // SPDX-License-Identifier: GPL-3.0-or-later
  2. #include "../libnetdata.h"
  3. struct simple_pattern {
  4. const char *match;
  5. uint32_t len;
  6. SIMPLE_PREFIX_MODE mode;
  7. bool negative;
  8. bool case_sensitive;
  9. struct simple_pattern *child;
  10. struct simple_pattern *next;
  11. };
  12. static struct simple_pattern *parse_pattern(char *str, SIMPLE_PREFIX_MODE default_mode, size_t count) {
  13. if(unlikely(count >= 1000))
  14. return NULL;
  15. // fprintf(stderr, "PARSING PATTERN: '%s'\n", str);
  16. SIMPLE_PREFIX_MODE mode;
  17. struct simple_pattern *child = NULL;
  18. char *s = str, *c = str;
  19. // skip asterisks in front
  20. while(*c == '*') c++;
  21. // find the next asterisk
  22. while(*c && *c != '*') c++;
  23. // do we have an asterisk in the middle?
  24. if(*c == '*' && c[1] != '\0') {
  25. // yes, we have
  26. child = parse_pattern(c, default_mode, count + 1);
  27. c[1] = '\0';
  28. }
  29. // check what this one matches
  30. size_t len = strlen(s);
  31. if(len >= 2 && *s == '*' && s[len - 1] == '*') {
  32. s[len - 1] = '\0';
  33. s++;
  34. mode = SIMPLE_PATTERN_SUBSTRING;
  35. }
  36. else if(len >= 1 && *s == '*') {
  37. s++;
  38. mode = SIMPLE_PATTERN_SUFFIX;
  39. }
  40. else if(len >= 1 && s[len - 1] == '*') {
  41. s[len - 1] = '\0';
  42. mode = SIMPLE_PATTERN_PREFIX;
  43. }
  44. else
  45. mode = default_mode;
  46. // allocate the structure
  47. struct simple_pattern *m = callocz(1, sizeof(struct simple_pattern));
  48. if(*s) {
  49. m->match = strdupz(s);
  50. m->len = strlen(m->match);
  51. m->mode = mode;
  52. }
  53. else {
  54. m->mode = SIMPLE_PATTERN_SUBSTRING;
  55. }
  56. m->child = child;
  57. return m;
  58. }
  59. SIMPLE_PATTERN *simple_pattern_create(const char *list, const char *separators, SIMPLE_PREFIX_MODE default_mode, bool case_sensitive) {
  60. struct simple_pattern *root = NULL, *last = NULL;
  61. if(unlikely(!list || !*list)) return root;
  62. char isseparator[256] = {
  63. [' '] = 1 // space
  64. , ['\t'] = 1 // tab
  65. , ['\r'] = 1 // carriage return
  66. , ['\n'] = 1 // new line
  67. , ['\f'] = 1 // form feed
  68. , ['\v'] = 1 // vertical tab
  69. };
  70. if (unlikely(separators && *separators)) {
  71. memset(&isseparator[0], 0, sizeof(isseparator));
  72. while(*separators) isseparator[(unsigned char)*separators++] = 1;
  73. }
  74. char *buf = mallocz(strlen(list) + 1);
  75. const char *s = list;
  76. while(s && *s) {
  77. buf[0] = '\0';
  78. char *c = buf;
  79. bool negative = false;
  80. // skip all spaces
  81. while(isseparator[(unsigned char)*s])
  82. s++;
  83. if(*s == '!') {
  84. negative = true;
  85. s++;
  86. }
  87. // empty string
  88. if(unlikely(!*s))
  89. break;
  90. // find the next space
  91. char escape = 0;
  92. while(*s) {
  93. if(*s == '\\' && !escape) {
  94. escape = 1;
  95. s++;
  96. }
  97. else {
  98. if (isseparator[(unsigned char)*s] && !escape) {
  99. s++;
  100. break;
  101. }
  102. *c++ = *s++;
  103. escape = 0;
  104. }
  105. }
  106. // terminate our string
  107. *c = '\0';
  108. // if we matched the empty string, skip it
  109. if(unlikely(!*buf))
  110. continue;
  111. // fprintf(stderr, "FOUND PATTERN: '%s'\n", buf);
  112. struct simple_pattern *m = parse_pattern(buf, default_mode, 0);
  113. m->negative = negative;
  114. m->case_sensitive = case_sensitive;
  115. // link it at the end
  116. if(unlikely(!root))
  117. root = last = m;
  118. else {
  119. last->next = m;
  120. last = m;
  121. }
  122. }
  123. freez(buf);
  124. return (SIMPLE_PATTERN *)root;
  125. }
  126. static inline char *add_wildcarded(const char *matched, size_t matched_size, char *wildcarded, size_t *wildcarded_size) {
  127. //if(matched_size) {
  128. // char buf[matched_size + 1];
  129. // strncpyz(buf, matched, matched_size);
  130. // fprintf(stderr, "ADD WILDCARDED '%s' of length %zu\n", buf, matched_size);
  131. //}
  132. if(unlikely(wildcarded && *wildcarded_size && matched && *matched && matched_size)) {
  133. size_t wss = *wildcarded_size - 1;
  134. size_t len = (matched_size < wss)?matched_size:wss;
  135. if(likely(len)) {
  136. strncpyz(wildcarded, matched, len);
  137. *wildcarded_size -= len;
  138. return &wildcarded[len];
  139. }
  140. }
  141. return wildcarded;
  142. }
  143. static inline int sp_strcmp(const char *s1, const char *s2, bool case_sensitive) {
  144. if(case_sensitive)
  145. return strcmp(s1, s2);
  146. return strcasecmp(s1, s2);
  147. }
  148. static inline int sp_strncmp(const char *s1, const char *s2, size_t n, bool case_sensitive) {
  149. if(case_sensitive)
  150. return strncmp(s1, s2, n);
  151. return strncasecmp(s1, s2, n);
  152. }
  153. static inline char *sp_strstr(const char *haystack, const char *needle, bool case_sensitive) {
  154. if(case_sensitive)
  155. return strstr(haystack, needle);
  156. return strcasestr(haystack, needle);
  157. }
  158. static inline bool match_pattern(struct simple_pattern *m, const char *str, size_t len, char *wildcarded, size_t *wildcarded_size) {
  159. char *s;
  160. bool loop = true;
  161. while(loop && m->len <= len) {
  162. loop = false;
  163. switch(m->mode) {
  164. default:
  165. case SIMPLE_PATTERN_EXACT:
  166. if(unlikely(sp_strcmp(str, m->match, m->case_sensitive) == 0)) {
  167. if(!m->child) return true;
  168. return false;
  169. }
  170. break;
  171. case SIMPLE_PATTERN_SUBSTRING:
  172. if(!m->len) return true;
  173. if((s = sp_strstr(str, m->match, m->case_sensitive))) {
  174. wildcarded = add_wildcarded(str, s - str, wildcarded, wildcarded_size);
  175. if(!m->child) {
  176. add_wildcarded(&s[m->len], len - (&s[m->len] - str), wildcarded, wildcarded_size);
  177. return true;
  178. }
  179. // instead of recursion
  180. {
  181. len = len - (s - str) - m->len;
  182. str = &s[m->len];
  183. m = m->child;
  184. loop = true;
  185. // return match_pattern(m->child, &s[m->len], len - (s - str) - m->len, wildcarded, wildcarded_size);
  186. }
  187. }
  188. break;
  189. case SIMPLE_PATTERN_PREFIX:
  190. if(unlikely(sp_strncmp(str, m->match, m->len, m->case_sensitive) == 0)) {
  191. if(!m->child) {
  192. add_wildcarded(&str[m->len], len - m->len, wildcarded, wildcarded_size);
  193. return true;
  194. }
  195. // instead of recursion
  196. {
  197. len = len - m->len;
  198. str = &str[m->len];
  199. m = m->child;
  200. loop = true;
  201. // return match_pattern(m->child, &str[m->len], len - m->len, wildcarded, wildcarded_size);
  202. }
  203. }
  204. break;
  205. case SIMPLE_PATTERN_SUFFIX:
  206. if(unlikely(sp_strcmp(&str[len - m->len], m->match, m->case_sensitive) == 0)) {
  207. add_wildcarded(str, len - m->len, wildcarded, wildcarded_size);
  208. if(!m->child) return true;
  209. return false;
  210. }
  211. break;
  212. }
  213. }
  214. return false;
  215. }
  216. static inline SIMPLE_PATTERN_RESULT simple_pattern_matches_extract_with_length(SIMPLE_PATTERN *list, const char *str, size_t len, char *wildcarded, size_t wildcarded_size) {
  217. struct simple_pattern *m, *root = (struct simple_pattern *)list;
  218. for(m = root; m ; m = m->next) {
  219. char *ws = wildcarded;
  220. size_t wss = wildcarded_size;
  221. if(unlikely(ws)) *ws = '\0';
  222. if (match_pattern(m, str, len, ws, &wss)) {
  223. if (m->negative) return SP_MATCHED_NEGATIVE;
  224. return SP_MATCHED_POSITIVE;
  225. }
  226. }
  227. return SP_NOT_MATCHED;
  228. }
  229. SIMPLE_PATTERN_RESULT simple_pattern_matches_buffer_extract(SIMPLE_PATTERN *list, BUFFER *str, char *wildcarded, size_t wildcarded_size) {
  230. if(!list || !str || buffer_strlen(str)) return SP_NOT_MATCHED;
  231. return simple_pattern_matches_extract_with_length(list, buffer_tostring(str), buffer_strlen(str), wildcarded, wildcarded_size);
  232. }
  233. SIMPLE_PATTERN_RESULT simple_pattern_matches_string_extract(SIMPLE_PATTERN *list, STRING *str, char *wildcarded, size_t wildcarded_size) {
  234. if(!list || !str) return SP_NOT_MATCHED;
  235. return simple_pattern_matches_extract_with_length(list, string2str(str), string_strlen(str), wildcarded, wildcarded_size);
  236. }
  237. SIMPLE_PATTERN_RESULT simple_pattern_matches_extract(SIMPLE_PATTERN *list, const char *str, char *wildcarded, size_t wildcarded_size) {
  238. if(!list || !str || !*str) return SP_NOT_MATCHED;
  239. return simple_pattern_matches_extract_with_length(list, str, strlen(str), wildcarded, wildcarded_size);
  240. }
  241. SIMPLE_PATTERN_RESULT simple_pattern_matches_length_extract(SIMPLE_PATTERN *list, const char *str, size_t len, char *wildcarded, size_t wildcarded_size) {
  242. if(!list || !str || !*str || !len) return SP_NOT_MATCHED;
  243. return simple_pattern_matches_extract_with_length(list, str, len, wildcarded, wildcarded_size);
  244. }
  245. static inline void free_pattern(struct simple_pattern *m) {
  246. if(!m) return;
  247. free_pattern(m->child);
  248. free_pattern(m->next);
  249. freez((void *)m->match);
  250. freez(m);
  251. }
  252. void simple_pattern_free(SIMPLE_PATTERN *list) {
  253. if(!list) return;
  254. free_pattern(((struct simple_pattern *)list));
  255. }
  256. /* Debugging patterns
  257. This code should be dead - it is useful for debugging but should not be called by production code.
  258. Feel free to comment it out, but please leave it in the file.
  259. */
  260. extern void simple_pattern_dump(uint64_t debug_type, SIMPLE_PATTERN *p)
  261. {
  262. struct simple_pattern *root = (struct simple_pattern *)p;
  263. if(root==NULL) {
  264. netdata_log_debug(debug_type,"dump_pattern(NULL)");
  265. return;
  266. }
  267. netdata_log_debug(debug_type,"dump_pattern(%p) child=%p next=%p mode=%u match=%s", root, root->child, root->next, root->mode,
  268. root->match);
  269. if(root->child!=NULL)
  270. simple_pattern_dump(debug_type, (SIMPLE_PATTERN*)root->child);
  271. if(root->next!=NULL)
  272. simple_pattern_dump(debug_type, (SIMPLE_PATTERN*)root->next);
  273. }
  274. /* Heuristic: decide if the pattern could match a DNS name.
  275. Although this functionality is used directly by socket.c:connection_allowed() it must be in this file
  276. because of the SIMPLE_PATTERN/simple_pattern structure hiding.
  277. Based on RFC952 / RFC1123. We need to decide if the pattern may match a DNS name, or not. For the negative
  278. cases we need to be sure that it can only match an ipv4 or ipv6 address:
  279. * IPv6 addresses contain ':', which are illegal characters in DNS.
  280. * IPv4 addresses cannot contain alpha- characters.
  281. * DNS TLDs must be alphanumeric to distinguish from IPv4.
  282. Some patterns (e.g. "*a*" ) could match multiple cases (i.e. DNS or IPv6).
  283. Some patterns will be awkward (e.g. "192.168.*") as they look like they are intended to match IPv4-only
  284. but could match DNS (i.e. "192.168.com" is a valid name).
  285. */
  286. static void scan_is_potential_name(struct simple_pattern *p, int *alpha, int *colon, int *wildcards)
  287. {
  288. while (p) {
  289. if (p->match) {
  290. if(p->mode == SIMPLE_PATTERN_EXACT && !strcmp("localhost", p->match)) {
  291. p = p->child;
  292. continue;
  293. }
  294. char const *scan = p->match;
  295. while (*scan != 0) {
  296. if ((*scan >= 'a' && *scan <= 'z') || (*scan >= 'A' && *scan <= 'Z'))
  297. *alpha = 1;
  298. if (*scan == ':')
  299. *colon = 1;
  300. scan++;
  301. }
  302. if (p->mode != SIMPLE_PATTERN_EXACT)
  303. *wildcards = 1;
  304. p = p->child;
  305. }
  306. }
  307. }
  308. extern int simple_pattern_is_potential_name(SIMPLE_PATTERN *p)
  309. {
  310. int alpha=0, colon=0, wildcards=0;
  311. struct simple_pattern *root = (struct simple_pattern*)p;
  312. while (root != NULL) {
  313. if (root->match != NULL) {
  314. scan_is_potential_name(root, &alpha, &colon, &wildcards);
  315. }
  316. if (root->mode != SIMPLE_PATTERN_EXACT)
  317. wildcards = 1;
  318. root = root->next;
  319. }
  320. return (alpha || wildcards) && !colon;
  321. }
  322. char *simple_pattern_trim_around_equal(char *src) {
  323. char *store = mallocz(strlen(src) + 1);
  324. char *dst = store;
  325. while (*src) {
  326. if (*src == '=') {
  327. if (*(dst -1) == ' ')
  328. dst--;
  329. *dst++ = *src++;
  330. if (*src == ' ')
  331. src++;
  332. }
  333. *dst++ = *src++;
  334. }
  335. *dst = 0x00;
  336. return store;
  337. }
  338. char *simple_pattern_iterate(SIMPLE_PATTERN **p)
  339. {
  340. struct simple_pattern *root = (struct simple_pattern *) *p;
  341. struct simple_pattern **Proot = (struct simple_pattern **)p;
  342. (*Proot) = (*Proot)->next;
  343. return (char *) root->match;
  344. }