Integrate standalone tests

This commit is contained in:
David Gonzalez Martin 2024-02-27 13:29:42 -06:00
parent f857dead54
commit cb0906d6ec

View File

@ -11,6 +11,7 @@ const TestError = error{
}; };
fn collectDirectoryDirEntries(allocator: Allocator, path: []const u8) ![]const []const u8{ fn collectDirectoryDirEntries(allocator: Allocator, path: []const u8) ![]const []const u8{
std.debug.print("Opening from cwd: {s}\n", .{try std.fs.cwd().realpathAlloc(allocator, ".")});
var dir = try std.fs.cwd().openDir(path, .{ var dir = try std.fs.cwd().openDir(path, .{
.iterate = true, .iterate = true,
}); });
@ -29,6 +30,92 @@ fn collectDirectoryDirEntries(allocator: Allocator, path: []const u8) ![]const [
return dir_entries.items; return dir_entries.items;
} }
fn runStandalone(allocator: Allocator, args: struct {
directory_path: []const u8,
group_name: []const u8,
is_test: bool,
}) !void {
const test_names = try collectDirectoryDirEntries(allocator, args.directory_path);
const total_compilation_count = test_names.len;
var ran_compilation_count: usize = 0;
var failed_compilation_count: usize = 0;
var ran_test_count: usize = 0;
var failed_test_count: usize = 0;
const total_test_count = test_names.len;
std.debug.print("[{s} START]\n", .{args.group_name});
for (test_names) |test_name| {
std.debug.print("{s}... ", .{test_name});
const source_file_path = try std.mem.concat(allocator, u8, &.{args.directory_path, "/", test_name, "/main.nat"});
const compile_run = try std.ChildProcess.run(.{
.allocator = allocator,
// TODO: delete -main_source_file?
.argv = &.{"zig-out/bin/nat", if (args.is_test) "test" else "exe", "-main_source_file", source_file_path},
});
ran_compilation_count += 1;
const compilation_result: TestError!bool = switch (compile_run.term) {
.Exited => |exit_code| if (exit_code == 0) true else error.abnormal_exit_code,
.Signal => error.signaled,
.Stopped => error.stopped,
.Unknown => error.unknown,
};
const compilation_success = compilation_result catch b: {
failed_compilation_count += 1;
break :b false;
};
std.debug.print("[COMPILATION {s}] ", .{if (compilation_success) "\x1b[32mOK\x1b[0m" else "\x1b[31mFAILED\x1b[0m"});
if (compile_run.stdout.len > 0) {
std.debug.print("STDOUT:\n\n{s}\n\n", .{compile_run.stdout});
}
if (compile_run.stderr.len > 0) {
std.debug.print("STDERR:\n\n{s}\n\n", .{compile_run.stderr});
}
if (compilation_success) {
const test_path = try std.mem.concat(allocator, u8, &.{"nat/", test_name});
const test_run = try std.ChildProcess.run(.{
.allocator = allocator,
// TODO: delete -main_source_file?
.argv = &.{ test_path },
});
ran_test_count += 1;
const test_result: TestError!bool = switch (test_run.term) {
.Exited => |exit_code| if (exit_code == 0) true else error.abnormal_exit_code,
.Signal => error.signaled,
.Stopped => error.stopped,
.Unknown => error.unknown,
};
const test_success = test_result catch b: {
failed_test_count += 1;
break :b false;
};
std.debug.print("[TEST {s}]\n", .{if (test_success) "\x1b[32mOK\x1b[0m" else "\x1b[31mFAILED\x1b[0m"});
if (test_run.stdout.len > 0) {
std.debug.print("STDOUT:\n\n{s}\n\n", .{test_run.stdout});
}
if (test_run.stderr.len > 0) {
std.debug.print("STDERR:\n\n{s}\n\n", .{test_run.stderr});
}
} else {
std.debug.print("\n", .{});
}
}
std.debug.print("\n{s} COMPILATIONS: {}. FAILED: {}\n", .{args.group_name, total_compilation_count, failed_compilation_count});
std.debug.print("{s} TESTS: {}. RAN: {}. FAILED: {}\n", .{args.group_name, total_test_count, ran_test_count, failed_test_count});
if (failed_compilation_count > 0 or failed_test_count > 0) {
return error.fail;
}
}
fn runStandaloneTests(allocator: Allocator) !void { fn runStandaloneTests(allocator: Allocator) !void {
const standalone_test_dir_path = "test/standalone"; const standalone_test_dir_path = "test/standalone";
const standalone_test_names = try collectDirectoryDirEntries(allocator, standalone_test_dir_path); const standalone_test_names = try collectDirectoryDirEntries(allocator, standalone_test_dir_path);
@ -112,6 +199,7 @@ fn runStandaloneTests(allocator: Allocator) !void {
fn runBuildTests(allocator: Allocator) !void { fn runBuildTests(allocator: Allocator) !void {
std.debug.print("\n[BUILD TESTS]\n\n", .{}); std.debug.print("\n[BUILD TESTS]\n\n", .{});
const previous_cwd = try std.fs.cwd().realpathAlloc(allocator, ".");
const test_dir_path = "test/build"; const test_dir_path = "test/build";
const test_names = try collectDirectoryDirEntries(allocator, test_dir_path); const test_names = try collectDirectoryDirEntries(allocator, test_dir_path);
const test_dir_realpath = try std.fs.cwd().realpathAlloc(allocator, test_dir_path); const test_dir_realpath = try std.fs.cwd().realpathAlloc(allocator, test_dir_path);
@ -194,6 +282,8 @@ fn runBuildTests(allocator: Allocator) !void {
std.debug.print("\nTOTAL COMPILATIONS: {}. FAILED: {}\n", .{total_compilation_count, failed_compilation_count}); std.debug.print("\nTOTAL COMPILATIONS: {}. FAILED: {}\n", .{total_compilation_count, failed_compilation_count});
std.debug.print("TOTAL TESTS: {}. RAN: {}. FAILED: {}\n", .{total_test_count, ran_test_count, failed_test_count}); std.debug.print("TOTAL TESTS: {}. RAN: {}. FAILED: {}\n", .{total_test_count, ran_test_count, failed_test_count});
try std.os.chdir(previous_cwd);
if (failed_compilation_count > 0 or failed_test_count > 0) { if (failed_compilation_count > 0 or failed_test_count > 0) {
return error.fail; return error.fail;
} }
@ -203,6 +293,15 @@ pub fn main() !void {
std.debug.print("\n",.{}); std.debug.print("\n",.{});
var arena = std.heap.ArenaAllocator.init(std.heap.page_allocator); var arena = std.heap.ArenaAllocator.init(std.heap.page_allocator);
const allocator = arena.allocator(); const allocator = arena.allocator();
try runStandaloneTests(allocator); try runStandalone(allocator, .{
.directory_path = "test/standalone",
.group_name = "STANDALONE",
.is_test = false,
});
try runBuildTests(allocator); try runBuildTests(allocator);
try runStandalone(allocator, .{
.directory_path = "test/tests",
.group_name = "TESTS",
.is_test = true,
});
} }