This allows using CHASE_MKDIR_0755 without CHASE_NONEXISTENT or
CHASE_PARENT, so that it will create the final component of the path
too should it be missing.
This is really useful as a mkdir_p() replacement that returns an fd to
the final component, and knows how to operate relative to a root fs.
Kinda reverts
4ea0bcb9229fe12e0c428659d76934351b821872 (which only
refused the flags combination which didn't work, instead of making it
work, which is what this commit does.)
This also corrects behaviour if CHASE_MKDIR_0755 is used in one more
way: we'll now always open the dir as O_PATH. This is generally the
better idea, but matters in particular once with allow using
CHASE_MKDIR_0755 to create the final component: we should uniformly
return an O_PATH dir that must be converted to a proper fd first before
using it.
assert(!FLAGS_SET(flags, CHASE_PREFIX_ROOT));
assert(!FLAGS_SET(flags, CHASE_STEP|CHASE_EXTRACT_FILENAME));
assert(!FLAGS_SET(flags, CHASE_TRAIL_SLASH|CHASE_EXTRACT_FILENAME));
- assert(!FLAGS_SET(flags, CHASE_MKDIR_0755) || (flags & (CHASE_NONEXISTENT | CHASE_PARENT)) != 0);
assert(dir_fd >= 0 || dir_fd == AT_FDCWD);
/* Either the file may be missing, or we return an fd to the final object, but both make no sense */
if (r != -ENOENT)
return r;
- if (!isempty(todo) && !path_is_safe(todo))
+ if (!isempty(todo) && !path_is_safe(todo)) /* Refuse parent/mkdir handling if suffix contains ".." or something weird */
return r;
- if (FLAGS_SET(flags, CHASE_MKDIR_0755) && !isempty(todo)) {
+ if (FLAGS_SET(flags, CHASE_MKDIR_0755) && (!isempty(todo) || !(flags & (CHASE_PARENT|CHASE_NONEXISTENT)))) {
child = xopenat_full(fd,
first,
- O_DIRECTORY|O_CREAT|O_EXCL|O_NOFOLLOW|O_CLOEXEC,
+ O_DIRECTORY|O_CREAT|O_EXCL|O_NOFOLLOW|O_PATH|O_CLOEXEC,
/* xopen_flags = */ 0,
0755);
if (child < 0)
* also points to the result path even if this flag is set.
* When this specified, chase() will succeed with 1 even if the
* file points to the last path component does not exist. */
- CHASE_MKDIR_0755 = 1 << 11, /* Create any missing parent directories in the given path. This
- * needs to be set with CHASE_NONEXISTENT and/or CHASE_PARENT.
- * Note, chase_and_open() or friends always add CHASE_PARENT flag
- * when internally call chase(), hence CHASE_MKDIR_0755 can be
- * safely set without CHASE_NONEXISTENT and CHASE_PARENT. */
+ CHASE_MKDIR_0755 = 1 << 11, /* Create any missing directories in the given path. */
CHASE_EXTRACT_FILENAME = 1 << 12, /* Only return the last component of the resolved path */
} ChaseFlags;
#include "id128-util.h"
#include "mkdir.h"
#include "path-util.h"
+#include "random-util.h"
#include "rm-rf.h"
#include "string-util.h"
#include "tests.h"
assert_se(path_equal(fdpath, expected2));
}
+TEST(use_chase_as_mkdir_p) {
+ _cleanup_free_ char *p = NULL;
+ ASSERT_OK_ERRNO(asprintf(&p, "/tmp/chasemkdir%" PRIu64 "/a/b/c", random_u64()));
+
+ _cleanup_close_ int fd = -EBADF;
+ ASSERT_OK(chase(p, NULL, CHASE_PREFIX_ROOT|CHASE_MKDIR_0755, NULL, &fd));
+
+ ASSERT_OK_EQ(inode_same_at(AT_FDCWD, p, fd, NULL, AT_EMPTY_PATH), 1);
+
+ _cleanup_close_ int fd2 = -EBADF;
+ ASSERT_OK(chase(p, p, CHASE_PREFIX_ROOT|CHASE_MKDIR_0755, NULL, &fd2));
+
+ _cleanup_free_ char *pp = ASSERT_PTR(path_join(p, p));
+
+ ASSERT_OK_EQ(inode_same_at(AT_FDCWD, pp, fd2, NULL, AT_EMPTY_PATH), 1);
+
+ _cleanup_free_ char *f = NULL;
+ ASSERT_OK(path_extract_directory(p, &f));
+
+ _cleanup_free_ char *ff = NULL;
+ ASSERT_OK(path_extract_directory(f, &ff));
+
+ _cleanup_free_ char *fff = NULL;
+ ASSERT_OK(path_extract_directory(ff, &fff));
+
+ ASSERT_OK(rm_rf(fff, REMOVE_PHYSICAL));
+}
+
static int intro(void) {
arg_test_dir = saved_argv[1];
return EXIT_SUCCESS;