/* SPDX-License-Identifier: LGPL-2.1-or-later */ #include "alloc-util.h" #include "errno-util.h" #include "fd-util.h" #include "fileio.h" #include "format-util.h" #include "fs-util.h" #include "log.h" #include "path-util.h" #include "process-util.h" #include "string-util.h" #include "tests.h" #include "tmpfile-util.h" static void test_tempfn_random_one(const char *p, const char *extra, const char *expect, int ret) { _cleanup_free_ char *s = NULL; int r; r = tempfn_random(p, extra, &s); log_info("%s+%s → %s vs. %s (%i/%s vs. %i/%s)", p, strna(extra), strna(s), strna(expect), r, STRERROR(r), ret, STRERROR(ret)); assert_se(!s == !expect); if (s) { const char *suffix; assert_se(suffix = startswith(s, expect)); assert_se(in_charset(suffix, HEXDIGITS)); assert_se(strlen(suffix) == 16); } assert_se(ret == r); } TEST(tempfn_random) { _cleanup_free_ char *dir = NULL, *p = NULL, *q = NULL; test_tempfn_random_one("", NULL, NULL, -EINVAL); test_tempfn_random_one(".", NULL, NULL, -EADDRNOTAVAIL); test_tempfn_random_one("..", NULL, NULL, -EINVAL); test_tempfn_random_one("/", NULL, NULL, -EADDRNOTAVAIL); test_tempfn_random_one("foo", "hoge/aaa", NULL, -EINVAL); test_tempfn_random_one("foo", NULL, ".#foo", 0); test_tempfn_random_one("foo", "bar", ".#barfoo", 0); test_tempfn_random_one("/tmp/foo", NULL, "/tmp/.#foo", 0); test_tempfn_random_one("/tmp/foo", "bar", "/tmp/.#barfoo", 0); test_tempfn_random_one("./foo", NULL, ".#foo", 0); test_tempfn_random_one("./foo", "bar", ".#barfoo", 0); test_tempfn_random_one("../foo", NULL, "../.#foo", 0); test_tempfn_random_one("../foo", "bar", "../.#barfoo", 0); test_tempfn_random_one("foo/", NULL, ".#foo", 0); test_tempfn_random_one("foo/", "bar", ".#barfoo", 0); test_tempfn_random_one("/tmp/foo/", NULL, "/tmp/.#foo", 0); test_tempfn_random_one("/tmp/foo/", "bar", "/tmp/.#barfoo", 0); test_tempfn_random_one("./foo/", NULL, ".#foo", 0); test_tempfn_random_one("./foo/", "bar", ".#barfoo", 0); test_tempfn_random_one("../foo/", NULL, "../.#foo", 0); test_tempfn_random_one("../foo/", "bar", "../.#barfoo", 0); assert_se(dir = new(char, PATH_MAX - 20)); memset(dir, 'x', PATH_MAX - 21); dir[PATH_MAX - 21] = '\0'; for (size_t i = 0; i < PATH_MAX - 21; i += NAME_MAX + 1) dir[i] = '/'; assert_se(p = path_join(dir, "a")); assert_se(q = path_join(dir, ".#a")); test_tempfn_random_one(p, NULL, q, 0); test_tempfn_random_one(p, "b", NULL, -EINVAL); p = mfree(p); q = mfree(q); assert_se(p = new(char, NAME_MAX + 1)); memset(p, 'x', NAME_MAX); p[NAME_MAX] = '\0'; assert_se(q = new(char, NAME_MAX + 1)); memset(stpcpy(q, ".#"), 'x', NAME_MAX - STRLEN(".#") - 16); q[NAME_MAX - 16] = '\0'; test_tempfn_random_one(p, NULL, q, 0); memset(stpcpy(q, ".#hoge"), 'x', NAME_MAX - STRLEN(".#hoge") - 16); q[NAME_MAX - 16] = '\0'; test_tempfn_random_one(p, "hoge", q, 0); } static void test_tempfn_xxxxxx_one(const char *p, const char *extra, const char *expect, int ret) { _cleanup_free_ char *s = NULL; int r; r = tempfn_xxxxxx(p, extra, &s); log_info("%s+%s → %s vs. %s (%i/%s vs. %i/%s)", p, strna(extra), strna(s), strna(expect), r, STRERROR(r), ret, STRERROR(ret)); assert_se(!s == !expect); if (s) { const char *suffix; assert_se(suffix = startswith(s, expect)); assert_se(streq(suffix, "XXXXXX")); } assert_se(ret == r); } TEST(tempfn_xxxxxx) { _cleanup_free_ char *dir = NULL, *p = NULL, *q = NULL; test_tempfn_xxxxxx_one("", NULL, NULL, -EINVAL); test_tempfn_xxxxxx_one(".", NULL, NULL, -EADDRNOTAVAIL); test_tempfn_xxxxxx_one("..", NULL, NULL, -EINVAL); test_tempfn_xxxxxx_one("/", NULL, NULL, -EADDRNOTAVAIL); test_tempfn_xxxxxx_one("foo", "hoge/aaa", NULL, -EINVAL); test_tempfn_xxxxxx_one("foo", NULL, ".#foo", 0); test_tempfn_xxxxxx_one("foo", "bar", ".#barfoo", 0); test_tempfn_xxxxxx_one("/tmp/foo", NULL, "/tmp/.#foo", 0); test_tempfn_xxxxxx_one("/tmp/foo", "bar", "/tmp/.#barfoo", 0); test_tempfn_xxxxxx_one("./foo", NULL, ".#foo", 0); test_tempfn_xxxxxx_one("./foo", "bar", ".#barfoo", 0); test_tempfn_xxxxxx_one("../foo", NULL, "../.#foo", 0); test_tempfn_xxxxxx_one("../foo", "bar", "../.#barfoo", 0); test_tempfn_xxxxxx_one("foo/", NULL, ".#foo", 0); test_tempfn_xxxxxx_one("foo/", "bar", ".#barfoo", 0); test_tempfn_xxxxxx_one("/tmp/foo/", NULL, "/tmp/.#foo", 0); test_tempfn_xxxxxx_one("/tmp/foo/", "bar", "/tmp/.#barfoo", 0); test_tempfn_xxxxxx_one("./foo/", NULL, ".#foo", 0); test_tempfn_xxxxxx_one("./foo/", "bar", ".#barfoo", 0); test_tempfn_xxxxxx_one("../foo/", NULL, "../.#foo", 0); test_tempfn_xxxxxx_one("../foo/", "bar", "../.#barfoo", 0); assert_se(dir = new(char, PATH_MAX - 10)); memset(dir, 'x', PATH_MAX - 11); dir[PATH_MAX - 11] = '\0'; for (size_t i = 0; i < PATH_MAX - 11; i += NAME_MAX + 1) dir[i] = '/'; assert_se(p = path_join(dir, "a")); assert_se(q = path_join(dir, ".#a")); test_tempfn_xxxxxx_one(p, NULL, q, 0); test_tempfn_xxxxxx_one(p, "b", NULL, -EINVAL); p = mfree(p); q = mfree(q); assert_se(p = new(char, NAME_MAX + 1)); memset(p, 'x', NAME_MAX); p[NAME_MAX] = '\0'; assert_se(q = new(char, NAME_MAX + 1)); memset(stpcpy(q, ".#"), 'x', NAME_MAX - STRLEN(".#") - 6); q[NAME_MAX - 6] = '\0'; test_tempfn_xxxxxx_one(p, NULL, q, 0); memset(stpcpy(q, ".#hoge"), 'x', NAME_MAX - STRLEN(".#hoge") - 6); q[NAME_MAX - 6] = '\0'; test_tempfn_xxxxxx_one(p, "hoge", q, 0); } static void test_tempfn_random_child_one(const char *p, const char *extra, const char *expect, int ret) { _cleanup_free_ char *s = NULL; int r; r = tempfn_random_child(p, extra, &s); log_info_errno(r, "%s+%s → %s vs. %s (%i/%s vs. %i/%s)", p, strna(extra), strna(s), strna(expect), r, STRERROR(r), ret, STRERROR(ret)); assert_se(!s == !expect); if (s) { const char *suffix; assert_se(suffix = startswith(s, expect)); assert_se(in_charset(suffix, HEXDIGITS)); assert_se(strlen(suffix) == 16); } assert_se(ret == r); } TEST(tempfn_random_child) { _cleanup_free_ char *dir = NULL, *p = NULL, *q = NULL; test_tempfn_random_child_one("", NULL, ".#", 0); test_tempfn_random_child_one(".", NULL, ".#", 0); test_tempfn_random_child_one("..", NULL, "../.#", 0); test_tempfn_random_child_one("/", NULL, "/.#", 0); test_tempfn_random_child_one("foo", "hoge/aaa", NULL, -EINVAL); test_tempfn_random_child_one("foo", NULL, "foo/.#", 0); test_tempfn_random_child_one("foo", "bar", "foo/.#bar", 0); test_tempfn_random_child_one("/tmp/foo", NULL, "/tmp/foo/.#", 0); test_tempfn_random_child_one("/tmp/foo", "bar", "/tmp/foo/.#bar", 0); test_tempfn_random_child_one("./foo", NULL, "foo/.#", 0); test_tempfn_random_child_one("./foo", "bar", "foo/.#bar", 0); test_tempfn_random_child_one("../foo", NULL, "../foo/.#", 0); test_tempfn_random_child_one("../foo", "bar", "../foo/.#bar", 0); test_tempfn_random_child_one("foo/", NULL, "foo/.#", 0); test_tempfn_random_child_one("foo/", "bar", "foo/.#bar", 0); test_tempfn_random_child_one("/tmp/foo/", NULL, "/tmp/foo/.#", 0); test_tempfn_random_child_one("/tmp/foo/", "bar", "/tmp/foo/.#bar", 0); test_tempfn_random_child_one("./foo/", NULL, "foo/.#", 0); test_tempfn_random_child_one("./foo/", "bar", "foo/.#bar", 0); test_tempfn_random_child_one("../foo/", NULL, "../foo/.#", 0); test_tempfn_random_child_one("../foo/", "bar", "../foo/.#bar", 0); assert_se(dir = new(char, PATH_MAX - 21)); memset(dir, 'x', PATH_MAX - 22); dir[PATH_MAX - 22] = '\0'; for (size_t i = 0; i < PATH_MAX - 22; i += NAME_MAX + 1) dir[i] = '/'; assert_se(p = path_join(dir, "a")); assert_se(q = path_join(p, ".#")); test_tempfn_random_child_one(p, NULL, q, 0); test_tempfn_random_child_one(p, "b", NULL, -EINVAL); p = mfree(p); q = mfree(q); assert_se(p = new(char, NAME_MAX + 1)); memset(p, 'x', NAME_MAX); p[NAME_MAX] = '\0'; assert_se(q = path_join(p, ".#")); test_tempfn_random_child_one(p, NULL, q, 0); assert_se(strextend(&q, "hoge")); test_tempfn_random_child_one(p, "hoge", q, 0); } TEST(link_tmpfile) { _cleanup_free_ char *cmd = NULL, *cmd2 = NULL, *ans = NULL, *ans2 = NULL, *d = NULL, *tmp = NULL, *line = NULL; _cleanup_close_ int fd = -EBADF, fd2 = -EBADF; const char *p = saved_argv[1] ?: "/tmp"; char *pattern; pattern = strjoina(p, "/systemd-test-XXXXXX"); fd = open_tmpfile_unlinkable(p, O_RDWR|O_CLOEXEC); assert_se(fd >= 0); assert_se(asprintf(&cmd, "ls -l /proc/"PID_FMT"/fd/%d", getpid_cached(), fd) > 0); (void) system(cmd); assert_se(readlink_malloc(cmd + 6, &ans) >= 0); log_debug("link1: %s", ans); assert_se(endswith(ans, " (deleted)")); fd2 = mkostemp_safe(pattern); assert_se(fd2 >= 0); assert_se(unlink(pattern) == 0); assert_se(asprintf(&cmd2, "ls -l /proc/"PID_FMT"/fd/%d", getpid_cached(), fd2) > 0); (void) system(cmd2); assert_se(readlink_malloc(cmd2 + 6, &ans2) >= 0); log_debug("link2: %s", ans2); assert_se(endswith(ans2, " (deleted)")); pattern = strjoina(p, "/tmpfiles-test"); assert_se(tempfn_random(pattern, NULL, &d) >= 0); fd = safe_close(fd); fd = open_tmpfile_linkable(d, O_RDWR|O_CLOEXEC, &tmp); assert_se(fd >= 0); assert_se(write(fd, "foobar\n", 7) == 7); assert_se(touch(d) >= 0); assert_se(link_tmpfile(fd, tmp, d, /* replace= */ false) == -EEXIST); assert_se(unlink(d) >= 0); assert_se(link_tmpfile(fd, tmp, d, /* replace= */ false) >= 0); assert_se(read_one_line_file(d, &line) >= 0); assert_se(streq(line, "foobar")); fd = safe_close(fd); tmp = mfree(tmp); fd = open_tmpfile_linkable(d, O_RDWR|O_CLOEXEC, &tmp); assert_se(fd >= 0); assert_se(write(fd, "waumiau\n", 8) == 8); assert_se(link_tmpfile(fd, tmp, d, /* replace= */ false) == -EEXIST); assert_se(link_tmpfile(fd, tmp, d, /* replace= */ true) >= 0); line = mfree(line); assert_se(read_one_line_file(d, &line) >= 0); assert_se(streq(line, "waumiau")); assert_se(unlink(d) >= 0); } DEFINE_TEST_MAIN(LOG_DEBUG);