snac2/xs_regex.h

146 lines
3.7 KiB
C
Raw Normal View History

2024-01-04 08:22:03 +00:00
/* copyright (c) 2022 - 2024 grunfink et al. / MIT license */
2022-09-27 08:03:41 +00:00
#ifndef _XS_REGEX_H
#define _XS_REGEX_H
2024-05-15 11:27:23 +00:00
int xs_regex_match(const char *str, const char *rx);
2023-01-28 16:49:02 +00:00
xs_list *xs_regex_split_n(const char *str, const char *rx, int count);
2022-11-24 07:47:02 +00:00
#define xs_regex_split(str, rx) xs_regex_split_n(str, rx, XS_ALL)
2023-09-17 00:52:44 +00:00
xs_list *xs_regex_select_n(const char *str, const char *rx, int count);
#define xs_regex_select(str, rx) xs_regex_select_n(str, rx, XS_ALL)
xs_list *xs_regex_replace_in(xs_str *str, const char *rx, const char *rep, int count);
#define xs_regex_replace_i(str, rx, rep) xs_regex_replace_in(str, rx, rep, XS_ALL)
#define xs_regex_replace_n(str, rx, rep, count) xs_regex_replace_in(xs_dup(str), rx, rep, count)
#define xs_regex_replace(str, rx, rep) xs_regex_replace_in(xs_dup(str), rx, rep, XS_ALL)
2022-09-27 08:03:41 +00:00
#ifdef XS_IMPLEMENTATION
#include <regex.h>
2023-01-28 16:49:02 +00:00
xs_list *xs_regex_split_n(const char *str, const char *rx, int count)
2024-05-15 11:27:23 +00:00
/* splits str using regex as a separator, at most count times.
Always returns a list:
len == 0: regcomp error
len == 1: full string (no matches)
len == odd: first part [ separator / next part ]...
*/
2022-09-27 08:03:41 +00:00
{
regex_t re;
regmatch_t rm;
int offset = 0;
2024-05-15 11:27:23 +00:00
xs_list *list = xs_list_new();
2022-09-27 15:19:59 +00:00
const char *p;
2022-09-27 08:03:41 +00:00
if (regcomp(&re, rx, REG_EXTENDED))
2024-05-15 11:27:23 +00:00
return list;
2022-09-27 08:03:41 +00:00
while (count > 0 && !regexec(&re, (p = str + offset), 1, &rm, offset > 0 ? REG_NOTBOL : 0)) {
2022-09-27 15:19:59 +00:00
/* add first the leading part of the string */
2023-09-13 16:19:19 +00:00
xs *s1 = xs_str_new_sz(p, rm.rm_so);
list = xs_list_append(list, s1);
2022-09-27 08:03:41 +00:00
2022-09-27 15:19:59 +00:00
/* add now the matched text as the separator */
2023-09-13 16:19:19 +00:00
xs *s2 = xs_str_new_sz(p + rm.rm_so, rm.rm_eo - rm.rm_so);
list = xs_list_append(list, s2);
2022-09-27 08:03:41 +00:00
2022-09-27 15:19:59 +00:00
/* move forward */
2022-09-27 08:03:41 +00:00
offset += rm.rm_eo;
count--;
}
2022-09-27 15:19:59 +00:00
/* add the rest of the string */
list = xs_list_append(list, p);
2022-09-27 08:03:41 +00:00
regfree(&re);
return list;
}
2022-09-27 15:19:59 +00:00
2023-09-17 00:52:44 +00:00
xs_list *xs_regex_select_n(const char *str, const char *rx, int count)
/* selects all matches and return them as a list */
2022-09-27 15:19:59 +00:00
{
2023-01-28 16:49:02 +00:00
xs_list *list = xs_list_new();
2022-09-27 15:19:59 +00:00
xs *split = NULL;
2024-05-23 08:01:37 +00:00
const xs_val *v;
2022-09-27 15:19:59 +00:00
int n = 0;
2024-05-15 11:27:23 +00:00
int c = 0;
2022-09-27 15:19:59 +00:00
/* split */
2022-10-07 11:48:53 +00:00
split = xs_regex_split_n(str, rx, count);
2022-09-27 15:19:59 +00:00
/* now iterate to get only the 'separators' (odd ones) */
2024-05-15 11:27:23 +00:00
while (xs_list_next(split, &v, &c)) {
2022-09-27 15:19:59 +00:00
if (n & 0x1)
list = xs_list_append(list, v);
n++;
}
return list;
}
xs_list *xs_regex_replace_in(xs_str *str, const char *rx, const char *rep, int count)
/* replaces all matches with the rep string. If it contains unescaped &,
they are replaced with the match */
{
xs_str *s = xs_str_new(NULL);
xs *split = xs_regex_split_n(str, rx, count);
2024-05-23 08:01:37 +00:00
const xs_val *v;
int n = 0;
2024-05-15 11:27:23 +00:00
int c = 0;
int pholder = !!strchr(rep, '&');
2024-05-15 11:27:23 +00:00
while (xs_list_next(split, &v, &c)) {
if (n & 0x1) {
if (pholder) {
/* rep has a placeholder; process char by char */
const char *p = rep;
while (*p) {
if (*p == '&')
s = xs_str_cat(s, v);
else {
if (*p == '\\')
p++;
if (!*p)
break;
s = xs_append_m(s, p, 1);
}
p++;
}
}
else
s = xs_str_cat(s, rep);
}
else
s = xs_str_cat(s, v);
n++;
}
xs_free(str);
return s;
}
2024-05-15 11:27:23 +00:00
int xs_regex_match(const char *str, const char *rx)
/* returns if str matches the regex at least once */
{
xs *l = xs_regex_select_n(str, rx, 1);
return xs_list_len(l) == 1;
}
2022-09-27 08:03:41 +00:00
#endif /* XS_IMPLEMENTATION */
#endif /* XS_REGEX_H */