mirror of
https://github.com/systemd/systemd-stable.git
synced 2025-01-11 05:17:44 +03:00
path-util: introduce path_glob_can_match()
This commit is contained in:
parent
7177ac4572
commit
3b703fe269
@ -1,6 +1,7 @@
|
||||
/* SPDX-License-Identifier: LGPL-2.1-or-later */
|
||||
|
||||
#include <errno.h>
|
||||
#include <fnmatch.h>
|
||||
#include <limits.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
@ -17,6 +18,7 @@
|
||||
#include "extract-word.h"
|
||||
#include "fd-util.h"
|
||||
#include "fs-util.h"
|
||||
#include "glob-util.h"
|
||||
#include "log.h"
|
||||
#include "macro.h"
|
||||
#include "path-util.h"
|
||||
@ -1310,3 +1312,63 @@ bool prefixed_path_strv_contains(char **l, const char *path) {
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
int path_glob_can_match(const char *pattern, const char *prefix, char **ret) {
|
||||
assert(pattern);
|
||||
assert(prefix);
|
||||
|
||||
for (const char *a = pattern, *b = prefix;;) {
|
||||
_cleanup_free_ char *g = NULL, *h = NULL;
|
||||
const char *p, *q;
|
||||
int r, s;
|
||||
|
||||
r = path_find_first_component(&a, /* accept_dot_dot = */ false, &p);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
s = path_find_first_component(&b, /* accept_dot_dot = */ false, &q);
|
||||
if (s < 0)
|
||||
return s;
|
||||
|
||||
if (s == 0) {
|
||||
/* The pattern matches the prefix. */
|
||||
if (ret) {
|
||||
char *t;
|
||||
|
||||
t = path_join(prefix, p);
|
||||
if (!t)
|
||||
return -ENOMEM;
|
||||
|
||||
*ret = t;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
if (r == 0)
|
||||
break;
|
||||
|
||||
if (r == s && strneq(p, q, r))
|
||||
continue; /* common component. Check next. */
|
||||
|
||||
g = strndup(p, r);
|
||||
if (!g)
|
||||
return -ENOMEM;
|
||||
|
||||
if (!string_is_glob(g))
|
||||
break;
|
||||
|
||||
/* We found a glob component. Check if the glob pattern matches the prefix component. */
|
||||
|
||||
h = strndup(q, s);
|
||||
if (!h)
|
||||
return -ENOMEM;
|
||||
|
||||
if (fnmatch(g, h, 0) != 0)
|
||||
break;
|
||||
}
|
||||
|
||||
/* The pattern does not match the prefix. */
|
||||
if (ret)
|
||||
*ret = NULL;
|
||||
return false;
|
||||
}
|
||||
|
@ -196,3 +196,5 @@ static inline const char *empty_to_root(const char *path) {
|
||||
|
||||
bool path_strv_contains(char **l, const char *path);
|
||||
bool prefixed_path_strv_contains(char **l, const char *path);
|
||||
|
||||
int path_glob_can_match(const char *pattern, const char *prefix, char **ret);
|
||||
|
@ -1027,6 +1027,44 @@ TEST(path_startswith_strv) {
|
||||
assert_se(streq_ptr(path_startswith_strv("/foo2/bar", STRV_MAKE("/foo/quux", "", "/zzz")), NULL));
|
||||
}
|
||||
|
||||
static void test_path_glob_can_match_one(const char *pattern, const char *prefix, const char *expected) {
|
||||
_cleanup_free_ char *result = NULL;
|
||||
|
||||
log_debug("%s(%s, %s, %s)", __func__, pattern, prefix, strnull(expected));
|
||||
|
||||
assert_se(path_glob_can_match(pattern, prefix, &result) == !!expected);
|
||||
assert_se(streq_ptr(result, expected));
|
||||
}
|
||||
|
||||
TEST(path_glob_can_match) {
|
||||
test_path_glob_can_match_one("/foo/hoge/aaa", "/foo/hoge/aaa/bbb", NULL);
|
||||
test_path_glob_can_match_one("/foo/hoge/aaa", "/foo/hoge/aaa", "/foo/hoge/aaa");
|
||||
test_path_glob_can_match_one("/foo/hoge/aaa", "/foo/hoge", "/foo/hoge/aaa");
|
||||
test_path_glob_can_match_one("/foo/hoge/aaa", "/foo", "/foo/hoge/aaa");
|
||||
test_path_glob_can_match_one("/foo/hoge/aaa", "/", "/foo/hoge/aaa");
|
||||
|
||||
test_path_glob_can_match_one("/foo/*/aaa", "/foo/hoge/aaa/bbb", NULL);
|
||||
test_path_glob_can_match_one("/foo/*/aaa", "/foo/hoge/aaa", "/foo/hoge/aaa");
|
||||
test_path_glob_can_match_one("/foo/*/aaa", "/foo/hoge", "/foo/hoge/aaa");
|
||||
test_path_glob_can_match_one("/foo/*/aaa", "/foo", "/foo/*/aaa");
|
||||
test_path_glob_can_match_one("/foo/*/aaa", "/", "/foo/*/aaa");
|
||||
|
||||
test_path_glob_can_match_one("/foo/*/*/aaa", "/foo/xxx/yyy/aaa/bbb", NULL);
|
||||
test_path_glob_can_match_one("/foo/*/*/aaa", "/foo/xxx/yyy/aaa", "/foo/xxx/yyy/aaa");
|
||||
test_path_glob_can_match_one("/foo/*/*/aaa", "/foo/xxx/yyy", "/foo/xxx/yyy/aaa");
|
||||
test_path_glob_can_match_one("/foo/*/*/aaa", "/foo/xxx", "/foo/xxx/*/aaa");
|
||||
test_path_glob_can_match_one("/foo/*/*/aaa", "/foo", "/foo/*/*/aaa");
|
||||
test_path_glob_can_match_one("/foo/*/*/aaa", "/", "/foo/*/*/aaa");
|
||||
|
||||
test_path_glob_can_match_one("/foo/*/aaa/*", "/foo/xxx/aaa/bbb/ccc", NULL);
|
||||
test_path_glob_can_match_one("/foo/*/aaa/*", "/foo/xxx/aaa/bbb", "/foo/xxx/aaa/bbb");
|
||||
test_path_glob_can_match_one("/foo/*/aaa/*", "/foo/xxx/ccc", NULL);
|
||||
test_path_glob_can_match_one("/foo/*/aaa/*", "/foo/xxx/aaa", "/foo/xxx/aaa/*");
|
||||
test_path_glob_can_match_one("/foo/*/aaa/*", "/foo/xxx", "/foo/xxx/aaa/*");
|
||||
test_path_glob_can_match_one("/foo/*/aaa/*", "/foo", "/foo/*/aaa/*");
|
||||
test_path_glob_can_match_one("/foo/*/aaa/*", "/", "/foo/*/aaa/*");
|
||||
}
|
||||
|
||||
TEST(print_MAX) {
|
||||
log_info("PATH_MAX=%zu\n"
|
||||
"FILENAME_MAX=%zu\n"
|
||||
|
Loading…
Reference in New Issue
Block a user