void, free);
void path_hash_func(const char *q, struct siphash *state) {
- size_t n;
+ bool add_slash = false;
assert(q);
assert(state);
* similar checks and also doesn't care for trailing slashes. Note that relative and absolute paths (i.e. those
* which begin in a slash or not) will hash differently though. */
- n = strspn(q, "/");
- if (n > 0) { /* Eat up initial slashes, and add one "/" to the hash for all of them */
- siphash24_compress(q, 1, state);
- q += n;
- }
+ /* if path is absolute, add one "/" to the hash. */
+ if (path_is_absolute(q))
+ siphash24_compress("/", 1, state);
for (;;) {
- /* Determine length of next component */
- n = strcspn(q, "/");
- if (n == 0) /* Reached the end? */
- break;
-
- /* Add this component to the hash and skip over it */
- siphash24_compress(q, n, state);
- q += n;
-
- /* How many slashes follow this component? */
- n = strspn(q, "/");
- if (q[n] == 0) /* Is this a trailing slash? If so, we are at the end, and don't care about the slashes anymore */
- break;
-
- /* We are not add the end yet. Hash exactly one slash for all of the ones we just encountered. */
- siphash24_compress(q, 1, state);
- q += n;
+ const char *e;
+ int r;
+
+ r = path_find_first_component(&q, true, &e);
+ if (r == 0)
+ return;
+
+ if (add_slash)
+ siphash24_compress_byte('/', state);
+
+ if (r < 0) {
+ /* if a component is invalid, then add remaining part as a string. */
+ string_hash_func(q, state);
+ return;
+ }
+
+ /* Add this component to the hash. */
+ siphash24_compress(e, r, state);
+
+ add_slash = true;
}
}
assert_se(hashmap_put(h, "//foo", INT_TO_PTR(3)) == -EEXIST);
assert_se(hashmap_put(h, "//foox/", INT_TO_PTR(4)) >= 0);
assert_se(hashmap_put(h, "/foox////", INT_TO_PTR(5)) == -EEXIST);
+ assert_se(hashmap_put(h, "//././/foox//.//.", INT_TO_PTR(5)) == -EEXIST);
assert_se(hashmap_put(h, "foo//////bar/quux//", INT_TO_PTR(6)) >= 0);
assert_se(hashmap_put(h, "foo/bar//quux/", INT_TO_PTR(8)) == -EEXIST);
+ assert_se(hashmap_put(h, "foo./ba.r//.quux/", INT_TO_PTR(9)) >= 0);
+ assert_se(hashmap_put(h, "foo./ba.r//.//.quux///./", INT_TO_PTR(10)) == -EEXIST);
assert_se(hashmap_get(h, "foo") == INT_TO_PTR(1));
assert_se(hashmap_get(h, "foo/") == INT_TO_PTR(1));
assert_se(hashmap_get(h, "//foo") == INT_TO_PTR(2));
assert_se(hashmap_get(h, "/////foo////") == INT_TO_PTR(2));
assert_se(hashmap_get(h, "/////foox////") == INT_TO_PTR(4));
+ assert_se(hashmap_get(h, "/.///./foox//.//") == INT_TO_PTR(4));
assert_se(hashmap_get(h, "/foox/") == INT_TO_PTR(4));
assert_se(hashmap_get(h, "/foox") == INT_TO_PTR(4));
assert_se(!hashmap_get(h, "foox"));
assert_se(hashmap_get(h, "foo/bar/quux") == INT_TO_PTR(6));
assert_se(hashmap_get(h, "foo////bar////quux/////") == INT_TO_PTR(6));
assert_se(!hashmap_get(h, "/foo////bar////quux/////"));
+ assert_se(hashmap_get(h, "foo././//ba.r////.quux///.//.") == INT_TO_PTR(9));
}
static void test_string_strv_hashmap(void) {
log_info("DEFAULT_USER_PATH=%s", DEFAULT_USER_PATH);
}
-#define test_path_compare(a, b, result) { \
- assert_se(path_compare(a, b) == result); \
- assert_se(path_compare(b, a) == -result); \
- assert_se(path_equal(a, b) == !result); \
- assert_se(path_equal(b, a) == !result); \
- }
-
static void test_path_simplify(const char *in, const char *out, const char *out_dot) {
char *p;
static void test_path(void) {
log_info("/* %s */", __func__);
- test_path_compare("/goo", "/goo", 0);
- test_path_compare("/goo", "/goo", 0);
- test_path_compare("//goo", "/goo", 0);
- test_path_compare("//goo/////", "/goo", 0);
- test_path_compare("goo/////", "goo", 0);
-
- test_path_compare("/goo/boo", "/goo//boo", 0);
- test_path_compare("//goo/boo", "/goo/boo//", 0);
-
- test_path_compare("/", "///", 0);
-
- test_path_compare("/x", "x/", 1);
- test_path_compare("x/", "/", -1);
-
- test_path_compare("/x/./y", "x/y", 1);
- test_path_compare("x/.y", "x/y", -1);
-
- test_path_compare("foo", "/foo", -1);
- test_path_compare("/foo", "/foo/bar", -1);
- test_path_compare("/foo/aaa", "/foo/b", -1);
- test_path_compare("/foo/aaa", "/foo/b/a", -1);
- test_path_compare("/foo/a", "/foo/aaa", -1);
- test_path_compare("/foo/a/b", "/foo/aaa", -1);
-
assert_se(path_is_absolute("/"));
assert_se(!path_is_absolute("./"));
assert_se(!path_equal_filename("/b", "/c"));
}
+static void test_path_compare_one(const char *a, const char *b, int expected) {
+ int r;
+
+ assert_se(path_compare(a, a) == 0);
+ assert_se(path_compare(b, b) == 0);
+
+ r = path_compare(a, b);
+ assert_se((r > 0) == (expected > 0) && (r < 0) == (expected < 0));
+ r = path_compare(b, a);
+ assert_se((r < 0) == (expected > 0) && (r > 0) == (expected < 0));
+
+ assert_se(path_equal(a, a) == 1);
+ assert_se(path_equal(b, b) == 1);
+ assert_se(path_equal(a, b) == (expected == 0));
+ assert_se(path_equal(b, a) == (expected == 0));
+}
+
+static void test_path_compare(void) {
+ test_path_compare_one("/goo", "/goo", 0);
+ test_path_compare_one("/goo", "/goo", 0);
+ test_path_compare_one("//goo", "/goo", 0);
+ test_path_compare_one("//goo/////", "/goo", 0);
+ test_path_compare_one("goo/////", "goo", 0);
+ test_path_compare_one("/goo/boo", "/goo//boo", 0);
+ test_path_compare_one("//goo/boo", "/goo/boo//", 0);
+ test_path_compare_one("//goo/././//./boo//././//", "/goo/boo//.", 0);
+ test_path_compare_one("/.", "//.///", 0);
+ test_path_compare_one("/x", "x/", 1);
+ test_path_compare_one("x/", "/", -1);
+ test_path_compare_one("/x/./y", "x/y", 1);
+ test_path_compare_one("x/.y", "x/y", -1);
+ test_path_compare_one("foo", "/foo", -1);
+ test_path_compare_one("/foo", "/foo/bar", -1);
+ test_path_compare_one("/foo/aaa", "/foo/b", -1);
+ test_path_compare_one("/foo/aaa", "/foo/b/a", -1);
+ test_path_compare_one("/foo/a", "/foo/aaa", -1);
+ test_path_compare_one("/foo/a/b", "/foo/aaa", -1);
+}
+
static void test_path_equal_root(void) {
/* Nail down the details of how path_equal("/", ...) works. */
assert_se(path_equal("/", "/"));
assert_se(path_equal("/", "//"));
- assert_se(!path_equal("/", "/./"));
+ assert_se(path_equal("/", "/./"));
assert_se(!path_equal("/", "/../"));
assert_se(!path_equal("/", "/.../"));
test_path_extract_filename_one("/..", NULL, -EINVAL);
test_path_extract_filename_one("../", NULL, -EINVAL);
test_path_extract_filename_one(".", NULL, -EINVAL);
- test_path_extract_filename_one("/.", NULL, -EINVAL);
+ test_path_extract_filename_one("/.", NULL, -EADDRNOTAVAIL);
test_path_extract_filename_one("./", NULL, -EINVAL);
}
test_path_extract_directory_one("/..", "/", 0);
test_path_extract_directory_one("../", NULL, -EDESTADDRREQ);
test_path_extract_directory_one(".", NULL, -EDESTADDRREQ);
- test_path_extract_directory_one("/.", "/", 0);
+ test_path_extract_directory_one("/.", NULL, -EADDRNOTAVAIL);
test_path_extract_directory_one("./", NULL, -EDESTADDRREQ);
}
test_print_paths();
test_path();
+ test_path_compare();
test_path_equal_root();
test_find_executable_full();
test_find_executable(argv[0]);