aboutsummaryrefslogtreecommitdiff
path: root/lib/std
diff options
context:
space:
mode:
authorAndrew Kelley <andrew@ziglang.org>2024-01-06 16:47:39 -0800
committerGitHub <noreply@github.com>2024-01-06 16:47:39 -0800
commit282ff8d3bd4a0d870a98f145aa87039e0409b745 (patch)
tree08fbc5f4193f95a4669676a872c070d07f42f3cd /lib/std
parent42389cb9c58cfe71db055142eb7b8cc15193ddf9 (diff)
parent51946f5adcc0e2a7c26f17ab7784e5e90aba2ea1 (diff)
downloadzig-282ff8d3bd4a0d870a98f145aa87039e0409b745.tar.gz
zig-282ff8d3bd4a0d870a98f145aa87039e0409b745.zip
Merge pull request #18453 from squeek502/makepath-parent-components
Add more `Dir.makePath` tests and document how `..` is handled in sub_path
Diffstat (limited to 'lib/std')
-rw-r--r--lib/std/fs/Dir.zig8
-rw-r--r--lib/std/fs/test.zig95
2 files changed, 96 insertions, 7 deletions
diff --git a/lib/std/fs/Dir.zig b/lib/std/fs/Dir.zig
index 4a45975ccc..8a1d38996e 100644
--- a/lib/std/fs/Dir.zig
+++ b/lib/std/fs/Dir.zig
@@ -1111,6 +1111,14 @@ pub fn makeDirW(self: Dir, sub_path: [*:0]const u16) !void {
/// Returns success if the path already exists and is a directory.
/// This function is not atomic, and if it returns an error, the file system may
/// have been modified regardless.
+///
+/// Paths containing `..` components are handled differently depending on the platform:
+/// - On Windows, `..` are resolved before the path is passed to NtCreateFile, meaning
+/// a `sub_path` like "first/../second" will resolve to "second" and only a
+/// `./second` directory will be created.
+/// - On other platforms, `..` are not resolved before the path is passed to `mkdirat`,
+/// meaning a `sub_path` like "first/../second" will create both a `./first`
+/// and a `./second` directory.
pub fn makePath(self: Dir, sub_path: []const u8) !void {
var it = try fs.path.componentIterator(sub_path);
var component = it.last() orelse return;
diff --git a/lib/std/fs/test.zig b/lib/std/fs/test.zig
index cd50b8aae8..0cdcd0a631 100644
--- a/lib/std/fs/test.zig
+++ b/lib/std/fs/test.zig
@@ -545,7 +545,7 @@ test "Dir.Iterator but dir is deleted during iteration" {
var iterator = subdir.iterate();
// Create something to iterate over within the subdir
- try tmp.dir.makePath("subdir/b");
+ try tmp.dir.makePath("subdir" ++ fs.path.sep_str ++ "b");
// Then, before iterating, delete the directory that we're iterating.
// This is a contrived reproduction, but this could happen outside of the program, in another thread, etc.
@@ -754,7 +754,7 @@ test "deleteDir" {
try testWithAllSupportedPathTypes(struct {
fn impl(ctx: *TestContext) !void {
const test_dir_path = try ctx.transformPath("test_dir");
- const test_file_path = try ctx.transformPath("test_dir" ++ std.fs.path.sep_str ++ "test_file");
+ const test_file_path = try ctx.transformPath("test_dir" ++ fs.path.sep_str ++ "test_file");
// deleting a non-existent directory
try testing.expectError(error.FileNotFound, ctx.dir.deleteDir(test_dir_path));
@@ -1089,6 +1089,88 @@ test "makePath in a directory that no longer exists" {
try testing.expectError(error.FileNotFound, tmp.dir.makePath("sub-path"));
}
+fn expectDir(dir: Dir, path: []const u8) !void {
+ var d = try dir.openDir(path, .{});
+ d.close();
+}
+
+test "makepath existing directories" {
+ var tmp = tmpDir(.{});
+ defer tmp.cleanup();
+
+ try tmp.dir.makeDir("A");
+ const tmpA = try tmp.dir.openDir("A", .{});
+ try tmpA.makeDir("B");
+
+ const testPath = "A" ++ fs.path.sep_str ++ "B" ++ fs.path.sep_str ++ "C";
+ try tmp.dir.makePath(testPath);
+
+ try expectDir(tmp.dir, testPath);
+}
+
+test "makepath through existing valid symlink" {
+ var tmp = tmpDir(.{});
+ defer tmp.cleanup();
+
+ try tmp.dir.makeDir("realfolder");
+ try tmp.dir.symLink("." ++ fs.path.sep_str ++ "realfolder", "working-symlink", .{});
+
+ try tmp.dir.makePath("working-symlink" ++ fs.path.sep_str ++ "in-realfolder");
+
+ try expectDir(tmp.dir, "realfolder" ++ fs.path.sep_str ++ "in-realfolder");
+}
+
+test "makepath relative walks" {
+ var tmp = tmpDir(.{});
+ defer tmp.cleanup();
+
+ const relPath = try fs.path.join(testing.allocator, &.{
+ "first", "..", "second", "..", "third", "..", "first", "A", "..", "B", "..", "C",
+ });
+ defer testing.allocator.free(relPath);
+
+ try tmp.dir.makePath(relPath);
+
+ // How .. is handled is different on Windows than non-Windows
+ switch (builtin.os.tag) {
+ .windows => {
+ // On Windows, .. is resolved before passing the path to NtCreateFile,
+ // meaning everything except `first/C` drops out.
+ try expectDir(tmp.dir, "first" ++ fs.path.sep_str ++ "C");
+ try testing.expectError(error.FileNotFound, tmp.dir.access("second", .{}));
+ try testing.expectError(error.FileNotFound, tmp.dir.access("third", .{}));
+ },
+ else => {
+ try expectDir(tmp.dir, "first" ++ fs.path.sep_str ++ "A");
+ try expectDir(tmp.dir, "first" ++ fs.path.sep_str ++ "B");
+ try expectDir(tmp.dir, "first" ++ fs.path.sep_str ++ "C");
+ try expectDir(tmp.dir, "second");
+ try expectDir(tmp.dir, "third");
+ },
+ }
+}
+
+test "makepath ignores '.'" {
+ var tmp = tmpDir(.{});
+ defer tmp.cleanup();
+
+ // Path to create, with "." elements:
+ const dotPath = try fs.path.join(testing.allocator, &.{
+ "first", ".", "second", ".", "third",
+ });
+ defer testing.allocator.free(dotPath);
+
+ // Path to expect to find:
+ const expectedPath = try fs.path.join(testing.allocator, &.{
+ "first", "second", "third",
+ });
+ defer testing.allocator.free(expectedPath);
+
+ try tmp.dir.makePath(dotPath);
+
+ try expectDir(tmp.dir, expectedPath);
+}
+
fn testFilenameLimits(iterable_dir: Dir, maxed_filename: []const u8) !void {
// setup, create a dir and a nested file both with maxed filenames, and walk the dir
{
@@ -1496,6 +1578,7 @@ test "open file with exclusive nonblocking lock twice (absolute paths)" {
.lock = .exclusive,
.lock_nonblocking = true,
});
+ defer fs.deleteFileAbsolute(filename) catch {};
const file2 = fs.createFileAbsolute(filename, .{
.lock = .exclusive,
@@ -1503,8 +1586,6 @@ test "open file with exclusive nonblocking lock twice (absolute paths)" {
});
file1.close();
try testing.expectError(error.WouldBlock, file2);
-
- try fs.deleteFileAbsolute(filename);
}
test "walker" {
@@ -1520,9 +1601,9 @@ test "walker" {
.{"dir2"},
.{"dir3"},
.{"dir4"},
- .{"dir3" ++ std.fs.path.sep_str ++ "sub1"},
- .{"dir3" ++ std.fs.path.sep_str ++ "sub2"},
- .{"dir3" ++ std.fs.path.sep_str ++ "sub2" ++ std.fs.path.sep_str ++ "subsub1"},
+ .{"dir3" ++ fs.path.sep_str ++ "sub1"},
+ .{"dir3" ++ fs.path.sep_str ++ "sub2"},
+ .{"dir3" ++ fs.path.sep_str ++ "sub2" ++ fs.path.sep_str ++ "subsub1"},
});
const expected_basenames = std.ComptimeStringMap(void, .{