aboutsummaryrefslogtreecommitdiff
path: root/test
diff options
context:
space:
mode:
authorRyan Liptak <squeek502@hotmail.com>2022-12-18 02:31:00 -0800
committerRyan Liptak <squeek502@hotmail.com>2022-12-18 03:55:28 -0800
commit0cbc59f2274663b9d956170fff3bf7f477b2b3cf (patch)
treec8a63d9bfd0eeecaef1b53831df50e97eddcfead /test
parente9c48e663145910d4934c3d1d4249da20682ba90 (diff)
downloadzig-0cbc59f2274663b9d956170fff3bf7f477b2b3cf.tar.gz
zig-0cbc59f2274663b9d956170fff3bf7f477b2b3cf.zip
standalone tests: Add windows spawn test
Tests a decent amount of edge cases dealing with how PATH and PATHEXT searching is handled.
Diffstat (limited to 'test')
-rw-r--r--test/standalone.zig4
-rw-r--r--test/standalone/windows_spawn/build.zig16
-rw-r--r--test/standalone/windows_spawn/hello.zig6
-rw-r--r--test/standalone/windows_spawn/main.zig161
4 files changed, 187 insertions, 0 deletions
diff --git a/test/standalone.zig b/test/standalone.zig
index 8605c2ad7d..f2d497e6ea 100644
--- a/test/standalone.zig
+++ b/test/standalone.zig
@@ -63,6 +63,10 @@ pub fn addCases(cases: *tests.StandaloneContext) void {
cases.addBuildFile("test/standalone/load_dynamic_library/build.zig", .{});
}
+ if (builtin.os.tag == .windows) {
+ cases.addBuildFile("test/standalone/windows_spawn/build.zig", .{});
+ }
+
cases.addBuildFile("test/standalone/c_compiler/build.zig", .{
.build_modes = true,
.cross_targets = true,
diff --git a/test/standalone/windows_spawn/build.zig b/test/standalone/windows_spawn/build.zig
new file mode 100644
index 0000000000..10a1132d3a
--- /dev/null
+++ b/test/standalone/windows_spawn/build.zig
@@ -0,0 +1,16 @@
+const Builder = @import("std").build.Builder;
+
+pub fn build(b: *Builder) void {
+ const mode = b.standardReleaseOptions();
+
+ const hello = b.addExecutable("hello", "hello.zig");
+ hello.setBuildMode(mode);
+
+ const main = b.addExecutable("main", "main.zig");
+ main.setBuildMode(mode);
+ const run = main.run();
+ run.addArtifactArg(hello);
+
+ const test_step = b.step("test", "Test it");
+ test_step.dependOn(&run.step);
+}
diff --git a/test/standalone/windows_spawn/hello.zig b/test/standalone/windows_spawn/hello.zig
new file mode 100644
index 0000000000..dcf917c430
--- /dev/null
+++ b/test/standalone/windows_spawn/hello.zig
@@ -0,0 +1,6 @@
+const std = @import("std");
+
+pub fn main() !void {
+ const stdout = std.io.getStdOut().writer();
+ try stdout.writeAll("hello from exe\n");
+}
diff --git a/test/standalone/windows_spawn/main.zig b/test/standalone/windows_spawn/main.zig
new file mode 100644
index 0000000000..970ff959f3
--- /dev/null
+++ b/test/standalone/windows_spawn/main.zig
@@ -0,0 +1,161 @@
+const std = @import("std");
+const windows = std.os.windows;
+const utf16Literal = std.unicode.utf8ToUtf16LeStringLiteral;
+
+pub fn main() anyerror!void {
+ var gpa = std.heap.GeneralPurposeAllocator(.{}){};
+ defer if (gpa.deinit()) @panic("found memory leaks");
+ const allocator = gpa.allocator();
+
+ var it = try std.process.argsWithAllocator(allocator);
+ defer it.deinit();
+ _ = it.next() orelse unreachable; // skip binary name
+ const hello_exe_cache_path = it.next() orelse unreachable;
+
+ var tmp = std.testing.tmpDir(.{});
+ defer tmp.cleanup();
+
+ const tmp_absolute_path = try tmp.dir.realpathAlloc(allocator, ".");
+ defer allocator.free(tmp_absolute_path);
+ const tmp_absolute_path_w = try std.unicode.utf8ToUtf16LeWithNull(allocator, tmp_absolute_path);
+ defer allocator.free(tmp_absolute_path_w);
+ const cwd_absolute_path = try std.fs.cwd().realpathAlloc(allocator, ".");
+ defer allocator.free(cwd_absolute_path);
+ const tmp_relative_path = try std.fs.path.relative(allocator, cwd_absolute_path, tmp_absolute_path);
+ defer allocator.free(tmp_relative_path);
+
+ // Clear PATH
+ std.debug.assert(std.os.windows.kernel32.SetEnvironmentVariableW(
+ utf16Literal("PATH"),
+ null,
+ ) == windows.TRUE);
+
+ // Set PATHEXT to something predictable
+ std.debug.assert(std.os.windows.kernel32.SetEnvironmentVariableW(
+ utf16Literal("PATHEXT"),
+ utf16Literal(".COM;.EXE;.BAT;.CMD;.JS"),
+ ) == windows.TRUE);
+
+ // No PATH, so it should fail to find anything not in the cwd
+ try testExecError(error.FileNotFound, allocator, "something_missing");
+
+ std.debug.assert(std.os.windows.kernel32.SetEnvironmentVariableW(
+ utf16Literal("PATH"),
+ tmp_absolute_path_w,
+ ) == windows.TRUE);
+
+ // Move hello.exe into the tmp dir which is now added to the path
+ try std.fs.cwd().copyFile(hello_exe_cache_path, tmp.dir, "hello.exe", .{});
+
+ // with extension should find the .exe (case insensitive)
+ try testExec(allocator, "HeLLo.exe", "hello from exe\n");
+ // without extension should find the .exe (case insensitive)
+ try testExec(allocator, "heLLo", "hello from exe\n");
+
+ // now add a .bat
+ try tmp.dir.writeFile("hello.bat", "@echo hello from bat");
+ // and a .cmd
+ try tmp.dir.writeFile("hello.cmd", "@echo hello from cmd");
+
+ // with extension should find the .bat (case insensitive)
+ try testExec(allocator, "heLLo.bat", "hello from bat\r\n");
+ // with extension should find the .cmd (case insensitive)
+ try testExec(allocator, "heLLo.cmd", "hello from cmd\r\n");
+ // without extension should find the .exe (since its first in PATHEXT)
+ try testExec(allocator, "heLLo", "hello from exe\n");
+
+ // now rename the exe to not have an extension
+ try tmp.dir.rename("hello.exe", "hello");
+
+ // with extension should now fail
+ try testExecError(error.FileNotFound, allocator, "hello.exe");
+ // without extension should succeed (case insensitive)
+ try testExec(allocator, "heLLo", "hello from exe\n");
+
+ try tmp.dir.makeDir("something");
+ try tmp.dir.rename("hello", "something/hello.exe");
+
+ const relative_path_no_ext = try std.fs.path.join(allocator, &.{ tmp_relative_path, "something/hello" });
+ defer allocator.free(relative_path_no_ext);
+
+ // Giving a full relative path to something/hello should work
+ try testExec(allocator, relative_path_no_ext, "hello from exe\n");
+ // But commands with path separators get excluded from PATH searching, so this will fail
+ try testExecError(error.FileNotFound, allocator, "something/hello");
+
+ // Now that .BAT is the first PATHEXT that should be found, this should succeed
+ try testExec(allocator, "heLLo", "hello from bat\r\n");
+
+ // Add a hello.exe that is not a valid executable
+ try tmp.dir.writeFile("hello.exe", "invalid");
+
+ // Trying to execute it with extension will give InvalidExe. This is a special
+ // case for .EXE extensions, where if they ever try to get executed but they are
+ // invalid, that gets treated as a fatal error wherever they are found and InvalidExe
+ // is returned immediately.
+ try testExecError(error.InvalidExe, allocator, "hello.exe");
+ // Same thing applies to the command with no extension--even though there is a
+ // hello.bat that could be executed, it should stop after it tries executing
+ // hello.exe and getting InvalidExe.
+ try testExecError(error.InvalidExe, allocator, "hello");
+
+ // If we now rename hello.exe to have no extension, it will behave differently
+ try tmp.dir.rename("hello.exe", "hello");
+
+ // Now, trying to execute it without an extension should treat InvalidExe as recoverable
+ // and skip over it and find hello.bat and execute that
+ try testExec(allocator, "hello", "hello from bat\r\n");
+
+ // If we rename the invalid exe to something else
+ try tmp.dir.rename("hello", "goodbye");
+ // Then we should now get FileNotFound when trying to execute 'goodbye',
+ // since that is what the original error will be after searching for 'goodbye'
+ // in the cwd. It will try to execute 'goodbye' from the PATH but the InvalidExe error
+ // should be ignored in this case.
+ try testExecError(error.FileNotFound, allocator, "goodbye");
+
+ // Now let's set the tmp dir as the cwd and set the path only include the "something" sub dir
+ try tmp.dir.setAsCwd();
+ const something_subdir_abs_path = try std.mem.concatWithSentinel(allocator, u16, &.{ tmp_absolute_path_w, utf16Literal("\\something") }, 0);
+ defer allocator.free(something_subdir_abs_path);
+
+ std.debug.assert(std.os.windows.kernel32.SetEnvironmentVariableW(
+ utf16Literal("PATH"),
+ something_subdir_abs_path,
+ ) == windows.TRUE);
+
+ // Now trying to execute goodbye should give error.InvalidExe since it's the original
+ // error that we got when trying within the cwd
+ try testExecError(error.InvalidExe, allocator, "goodbye");
+
+ // hello should still find the .bat
+ try testExec(allocator, "hello", "hello from bat\r\n");
+
+ // If we rename something/hello.exe to something/goodbye.exe
+ try tmp.dir.rename("something/hello.exe", "something/goodbye.exe");
+ // And try to execute goodbye, then the one in something should be found
+ // since the one in cwd is an invalid executable
+ try testExec(allocator, "goodbye", "hello from exe\n");
+
+ // If we use an absolute path to execute the invalid goodbye
+ const goodbye_abs_path = try std.mem.join(allocator, "\\", &.{ tmp_absolute_path, "goodbye" });
+ defer allocator.free(goodbye_abs_path);
+ // then the PATH should not be searched and we should get InvalidExe
+ try testExecError(error.InvalidExe, allocator, goodbye_abs_path);
+}
+
+fn testExecError(err: anyerror, allocator: std.mem.Allocator, command: []const u8) !void {
+ return std.testing.expectError(err, testExec(allocator, command, ""));
+}
+
+fn testExec(allocator: std.mem.Allocator, command: []const u8, expected_stdout: []const u8) !void {
+ var result = try std.ChildProcess.exec(.{
+ .allocator = allocator,
+ .argv = &[_][]const u8{command},
+ });
+ defer allocator.free(result.stdout);
+ defer allocator.free(result.stderr);
+
+ try std.testing.expectEqualStrings("", result.stderr);
+ try std.testing.expectEqualStrings(expected_stdout, result.stdout);
+}