2022-10-02 12:42:20 +00:00
|
|
|
//! Default test runner for unit tests.
|
2017-10-31 08:47:55 +00:00
|
|
|
const std = @import("std");
|
|
|
|
const io = std.io;
|
2017-05-01 17:12:38 +00:00
|
|
|
const builtin = @import("builtin");
|
2016-02-04 01:02:01 +00:00
|
|
|
|
2024-01-27 22:30:24 +00:00
|
|
|
pub const std_options = .{
|
|
|
|
.logFn = log,
|
2023-01-03 17:37:11 +00:00
|
|
|
};
|
2020-02-05 21:53:29 +00:00
|
|
|
|
2020-08-11 02:34:27 +00:00
|
|
|
var log_err_count: usize = 0;
|
2023-03-12 07:39:21 +00:00
|
|
|
var cmdline_buffer: [4096]u8 = undefined;
|
|
|
|
var fba = std.heap.FixedBufferAllocator.init(&cmdline_buffer);
|
2020-08-11 02:34:27 +00:00
|
|
|
|
2021-07-24 05:23:03 +00:00
|
|
|
pub fn main() void {
|
2023-11-01 18:45:22 +00:00
|
|
|
if (builtin.zig_backend == .stage2_aarch64) {
|
2023-03-12 07:39:21 +00:00
|
|
|
return mainSimple() catch @panic("test failure");
|
|
|
|
}
|
|
|
|
|
|
|
|
const args = std.process.argsAlloc(fba.allocator()) catch
|
|
|
|
@panic("unable to parse command line args");
|
|
|
|
|
|
|
|
var listen = false;
|
|
|
|
|
|
|
|
for (args[1..]) |arg| {
|
|
|
|
if (std.mem.eql(u8, arg, "--listen=-")) {
|
|
|
|
listen = true;
|
|
|
|
} else {
|
|
|
|
@panic("unrecognized command line argument");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (listen) {
|
2023-03-13 19:21:42 +00:00
|
|
|
return mainServer() catch @panic("internal test runner failure");
|
2023-03-12 07:39:21 +00:00
|
|
|
} else {
|
|
|
|
return mainTerminal();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-13 19:21:42 +00:00
|
|
|
fn mainServer() !void {
|
2023-03-12 07:39:21 +00:00
|
|
|
var server = try std.zig.Server.init(.{
|
|
|
|
.gpa = fba.allocator(),
|
|
|
|
.in = std.io.getStdIn(),
|
|
|
|
.out = std.io.getStdOut(),
|
|
|
|
.zig_version = builtin.zig_version_string,
|
|
|
|
});
|
|
|
|
defer server.deinit();
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
const hdr = try server.receiveMessage();
|
|
|
|
switch (hdr.tag) {
|
|
|
|
.exit => {
|
|
|
|
return std.process.exit(0);
|
|
|
|
},
|
|
|
|
.query_test_metadata => {
|
|
|
|
std.testing.allocator_instance = .{};
|
2023-04-22 11:09:44 +00:00
|
|
|
defer if (std.testing.allocator_instance.deinit() == .leak) {
|
2023-03-12 07:39:21 +00:00
|
|
|
@panic("internal test runner memory leak");
|
|
|
|
};
|
|
|
|
|
|
|
|
var string_bytes: std.ArrayListUnmanaged(u8) = .{};
|
|
|
|
defer string_bytes.deinit(std.testing.allocator);
|
|
|
|
try string_bytes.append(std.testing.allocator, 0); // Reserve 0 for null.
|
|
|
|
|
|
|
|
const test_fns = builtin.test_functions;
|
|
|
|
const names = try std.testing.allocator.alloc(u32, test_fns.len);
|
|
|
|
defer std.testing.allocator.free(names);
|
|
|
|
const expected_panic_msgs = try std.testing.allocator.alloc(u32, test_fns.len);
|
|
|
|
defer std.testing.allocator.free(expected_panic_msgs);
|
|
|
|
|
2024-01-27 23:04:38 +00:00
|
|
|
for (test_fns, names, expected_panic_msgs) |test_fn, *name, *expected_panic_msg| {
|
2023-06-22 17:46:56 +00:00
|
|
|
name.* = @as(u32, @intCast(string_bytes.items.len));
|
2023-03-12 07:39:21 +00:00
|
|
|
try string_bytes.ensureUnusedCapacity(std.testing.allocator, test_fn.name.len + 1);
|
|
|
|
string_bytes.appendSliceAssumeCapacity(test_fn.name);
|
|
|
|
string_bytes.appendAssumeCapacity(0);
|
|
|
|
expected_panic_msg.* = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
try server.serveTestMetadata(.{
|
|
|
|
.names = names,
|
|
|
|
.expected_panic_msgs = expected_panic_msgs,
|
|
|
|
.string_bytes = string_bytes.items,
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
.run_test => {
|
|
|
|
std.testing.allocator_instance = .{};
|
2023-08-25 18:59:10 +00:00
|
|
|
log_err_count = 0;
|
2023-03-12 07:39:21 +00:00
|
|
|
const index = try server.receiveBody_u32();
|
|
|
|
const test_fn = builtin.test_functions[index];
|
|
|
|
var fail = false;
|
|
|
|
var skip = false;
|
|
|
|
var leak = false;
|
|
|
|
test_fn.func() catch |err| switch (err) {
|
|
|
|
error.SkipZigTest => skip = true,
|
|
|
|
else => {
|
|
|
|
fail = true;
|
|
|
|
if (@errorReturnTrace()) |trace| {
|
|
|
|
std.debug.dumpStackTrace(trace.*);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
};
|
2023-04-22 11:09:44 +00:00
|
|
|
leak = std.testing.allocator_instance.deinit() == .leak;
|
2023-03-12 07:39:21 +00:00
|
|
|
try server.serveTestResults(.{
|
|
|
|
.index = index,
|
|
|
|
.flags = .{
|
|
|
|
.fail = fail,
|
|
|
|
.skip = skip,
|
|
|
|
.leak = leak,
|
2024-04-07 19:16:29 +00:00
|
|
|
.log_err_count = std.math.lossyCast(
|
|
|
|
@TypeOf(@as(std.zig.Server.Message.TestResults.Flags, undefined).log_err_count),
|
|
|
|
log_err_count,
|
|
|
|
),
|
2023-03-12 07:39:21 +00:00
|
|
|
},
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
else => {
|
2023-06-15 07:14:16 +00:00
|
|
|
std.debug.print("unsupported message: {x}", .{@intFromEnum(hdr.tag)});
|
2023-03-12 07:39:21 +00:00
|
|
|
std.process.exit(1);
|
|
|
|
},
|
|
|
|
}
|
2021-05-17 22:28:22 +00:00
|
|
|
}
|
2023-03-12 07:39:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn mainTerminal() void {
|
2019-12-12 23:27:17 +00:00
|
|
|
const test_fn_list = builtin.test_functions;
|
2018-07-22 03:43:43 +00:00
|
|
|
var ok_count: usize = 0;
|
|
|
|
var skip_count: usize = 0;
|
2021-05-04 16:36:59 +00:00
|
|
|
var fail_count: usize = 0;
|
2021-12-19 06:44:31 +00:00
|
|
|
var progress = std.Progress{
|
|
|
|
.dont_print_on_dumb = true,
|
|
|
|
};
|
2022-02-09 00:26:55 +00:00
|
|
|
const root_node = progress.start("Test", test_fn_list.len);
|
2022-05-13 03:23:05 +00:00
|
|
|
const have_tty = progress.terminal != null and
|
|
|
|
(progress.supports_ansi_escape_codes or progress.is_windows_terminal);
|
2016-02-04 01:02:01 +00:00
|
|
|
|
2023-07-07 13:22:27 +00:00
|
|
|
var async_frame_buffer: []align(builtin.target.stackAlignment()) u8 = undefined;
|
2020-02-06 22:56:40 +00:00
|
|
|
// TODO this is on the next line (using `undefined` above) because otherwise zig incorrectly
|
|
|
|
// ignores the alignment of the slice.
|
|
|
|
async_frame_buffer = &[_]u8{};
|
|
|
|
|
2020-08-08 06:26:58 +00:00
|
|
|
var leaks: usize = 0;
|
2023-02-18 16:02:57 +00:00
|
|
|
for (test_fn_list, 0..) |test_fn, i| {
|
2022-03-01 19:57:06 +00:00
|
|
|
std.testing.allocator_instance = .{};
|
2020-08-08 06:26:58 +00:00
|
|
|
defer {
|
2023-04-22 11:09:44 +00:00
|
|
|
if (std.testing.allocator_instance.deinit() == .leak) {
|
2020-08-08 06:26:58 +00:00
|
|
|
leaks += 1;
|
|
|
|
}
|
|
|
|
}
|
2020-07-09 04:01:13 +00:00
|
|
|
std.testing.log_level = .warn;
|
2020-01-29 19:18:04 +00:00
|
|
|
|
2020-12-19 04:51:18 +00:00
|
|
|
var test_node = root_node.start(test_fn.name, 0);
|
2019-10-18 01:46:41 +00:00
|
|
|
test_node.activate();
|
2019-10-21 23:01:08 +00:00
|
|
|
progress.refresh();
|
2021-12-19 06:44:31 +00:00
|
|
|
if (!have_tty) {
|
2022-02-27 14:40:59 +00:00
|
|
|
std.debug.print("{d}/{d} {s}... ", .{ i + 1, test_fn_list.len, test_fn.name });
|
2019-12-12 23:27:17 +00:00
|
|
|
}
|
2024-01-27 23:04:38 +00:00
|
|
|
if (test_fn.func()) |_| {
|
2018-07-22 03:43:43 +00:00
|
|
|
ok_count += 1;
|
2019-10-18 01:46:41 +00:00
|
|
|
test_node.end();
|
2021-12-19 06:44:31 +00:00
|
|
|
if (!have_tty) std.debug.print("OK\n", .{});
|
2018-07-22 03:43:43 +00:00
|
|
|
} else |err| switch (err) {
|
|
|
|
error.SkipZigTest => {
|
|
|
|
skip_count += 1;
|
2022-02-13 20:58:41 +00:00
|
|
|
progress.log("SKIP\n", .{});
|
2022-02-13 13:19:33 +00:00
|
|
|
test_node.end();
|
2018-07-22 03:43:43 +00:00
|
|
|
},
|
2019-10-21 22:35:14 +00:00
|
|
|
else => {
|
2021-05-04 16:36:59 +00:00
|
|
|
fail_count += 1;
|
2022-02-13 20:58:41 +00:00
|
|
|
progress.log("FAIL ({s})\n", .{@errorName(err)});
|
2022-04-22 20:10:02 +00:00
|
|
|
if (@errorReturnTrace()) |trace| {
|
2021-05-11 17:50:17 +00:00
|
|
|
std.debug.dumpStackTrace(trace.*);
|
2022-04-22 20:10:02 +00:00
|
|
|
}
|
2022-02-13 13:19:33 +00:00
|
|
|
test_node.end();
|
2019-10-21 22:35:14 +00:00
|
|
|
},
|
2018-07-22 03:43:43 +00:00
|
|
|
}
|
|
|
|
}
|
2019-10-18 01:46:41 +00:00
|
|
|
root_node.end();
|
2019-10-22 02:20:45 +00:00
|
|
|
if (ok_count == test_fn_list.len) {
|
2020-11-26 08:48:12 +00:00
|
|
|
std.debug.print("All {d} tests passed.\n", .{ok_count});
|
2019-10-22 02:20:45 +00:00
|
|
|
} else {
|
2022-02-27 14:40:59 +00:00
|
|
|
std.debug.print("{d} passed; {d} skipped; {d} failed.\n", .{ ok_count, skip_count, fail_count });
|
2016-02-04 01:02:01 +00:00
|
|
|
}
|
2020-08-11 02:34:27 +00:00
|
|
|
if (log_err_count != 0) {
|
2020-11-26 08:48:12 +00:00
|
|
|
std.debug.print("{d} errors were logged.\n", .{log_err_count});
|
2020-08-11 02:34:27 +00:00
|
|
|
}
|
2020-08-08 06:26:58 +00:00
|
|
|
if (leaks != 0) {
|
2020-11-26 08:48:12 +00:00
|
|
|
std.debug.print("{d} tests leaked memory.\n", .{leaks});
|
2020-08-11 02:34:27 +00:00
|
|
|
}
|
2021-05-04 16:36:59 +00:00
|
|
|
if (leaks != 0 or log_err_count != 0 or fail_count != 0) {
|
2020-08-08 06:26:58 +00:00
|
|
|
std.process.exit(1);
|
|
|
|
}
|
2016-02-04 01:02:01 +00:00
|
|
|
}
|
2020-07-09 04:01:13 +00:00
|
|
|
|
|
|
|
pub fn log(
|
|
|
|
comptime message_level: std.log.Level,
|
|
|
|
comptime scope: @Type(.EnumLiteral),
|
|
|
|
comptime format: []const u8,
|
2020-07-11 11:09:04 +00:00
|
|
|
args: anytype,
|
2020-07-09 04:01:13 +00:00
|
|
|
) void {
|
2023-06-15 07:14:16 +00:00
|
|
|
if (@intFromEnum(message_level) <= @intFromEnum(std.log.Level.err)) {
|
2023-08-25 18:59:10 +00:00
|
|
|
log_err_count +|= 1;
|
2020-08-11 02:34:27 +00:00
|
|
|
}
|
2023-06-15 07:14:16 +00:00
|
|
|
if (@intFromEnum(message_level) <= @intFromEnum(std.testing.log_level)) {
|
2022-05-23 22:48:18 +00:00
|
|
|
std.debug.print(
|
|
|
|
"[" ++ @tagName(scope) ++ "] (" ++ @tagName(message_level) ++ "): " ++ format ++ "\n",
|
|
|
|
args,
|
|
|
|
);
|
2020-07-09 04:01:13 +00:00
|
|
|
}
|
|
|
|
}
|
2021-05-17 22:28:22 +00:00
|
|
|
|
2023-03-12 07:39:21 +00:00
|
|
|
/// Simpler main(), exercising fewer language features, so that
|
|
|
|
/// work-in-progress backends can handle it.
|
|
|
|
pub fn mainSimple() anyerror!void {
|
2023-04-24 06:26:37 +00:00
|
|
|
const enable_print = false;
|
2023-04-28 23:38:18 +00:00
|
|
|
const print_all = false;
|
2023-04-24 06:26:37 +00:00
|
|
|
|
|
|
|
var passed: u64 = 0;
|
|
|
|
var skipped: u64 = 0;
|
|
|
|
var failed: u64 = 0;
|
|
|
|
const stderr = if (enable_print) std.io.getStdErr() else {};
|
2021-05-17 22:28:22 +00:00
|
|
|
for (builtin.test_functions) |test_fn| {
|
2023-04-28 23:38:18 +00:00
|
|
|
if (enable_print and print_all) {
|
|
|
|
stderr.writeAll(test_fn.name) catch {};
|
|
|
|
stderr.writeAll("... ") catch {};
|
|
|
|
}
|
2021-09-17 04:03:55 +00:00
|
|
|
test_fn.func() catch |err| {
|
2023-04-28 23:38:18 +00:00
|
|
|
if (enable_print and !print_all) {
|
|
|
|
stderr.writeAll(test_fn.name) catch {};
|
|
|
|
stderr.writeAll("... ") catch {};
|
|
|
|
}
|
2021-09-17 04:03:55 +00:00
|
|
|
if (err != error.SkipZigTest) {
|
2023-04-28 23:38:18 +00:00
|
|
|
if (enable_print) stderr.writeAll("FAIL\n") catch {};
|
2023-04-24 06:26:37 +00:00
|
|
|
failed += 1;
|
|
|
|
if (!enable_print) return err;
|
|
|
|
continue;
|
2021-09-17 04:03:55 +00:00
|
|
|
}
|
2023-04-28 23:38:18 +00:00
|
|
|
if (enable_print) stderr.writeAll("SKIP\n") catch {};
|
2023-04-24 06:26:37 +00:00
|
|
|
skipped += 1;
|
|
|
|
continue;
|
2021-09-17 04:03:55 +00:00
|
|
|
};
|
2023-04-28 23:38:18 +00:00
|
|
|
if (enable_print and print_all) stderr.writeAll("PASS\n") catch {};
|
2023-04-24 06:26:37 +00:00
|
|
|
passed += 1;
|
|
|
|
}
|
|
|
|
if (enable_print) {
|
|
|
|
stderr.writer().print("{} passed, {} skipped, {} failed\n", .{ passed, skipped, failed }) catch {};
|
|
|
|
if (failed != 0) std.process.exit(1);
|
2021-09-17 04:03:55 +00:00
|
|
|
}
|
2022-01-18 05:17:02 +00:00
|
|
|
}
|