adding custom test runner
This commit is contained in:
@@ -56,7 +56,10 @@ pub fn build(b: *std.Build) void {
|
|||||||
.{ .name = "nzsl", .module = nzsl.module("nzigsl") },
|
.{ .name = "nzsl", .module = nzsl.module("nzigsl") },
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
const lib_tests = b.addTest(.{ .root_module = test_mod });
|
const lib_tests = b.addTest(.{
|
||||||
|
.root_module = test_mod,
|
||||||
|
.test_runner = .{ .path = b.path("test/test_runner.zig"), .mode = .simple },
|
||||||
|
});
|
||||||
const run_tests = b.addRunArtifact(lib_tests);
|
const run_tests = b.addRunArtifact(lib_tests);
|
||||||
const test_step = b.step("test", "Run Zig unit tests");
|
const test_step = b.step("test", "Run Zig unit tests");
|
||||||
test_step.dependOn(&run_tests.step);
|
test_step.dependOn(&run_tests.step);
|
||||||
|
|||||||
289
test/test_runner.zig
git.filemode.normal_file
289
test/test_runner.zig
git.filemode.normal_file
@@ -0,0 +1,289 @@
|
|||||||
|
// See https://gist.github.com/karlseguin/c6bea5b35e4e8d26af6f81c22cb5d76b/1f317ebc9cd09bc50fd5591d09c34255e15d1d85
|
||||||
|
|
||||||
|
const std = @import("std");
|
||||||
|
const builtin = @import("builtin");
|
||||||
|
|
||||||
|
const Allocator = std.mem.Allocator;
|
||||||
|
|
||||||
|
const BORDER = "=" ** 80;
|
||||||
|
|
||||||
|
// use in custom panic handler
|
||||||
|
var current_test: ?[]const u8 = null;
|
||||||
|
|
||||||
|
pub fn main() !void {
|
||||||
|
var mem: [8192]u8 = undefined;
|
||||||
|
var fba = std.heap.FixedBufferAllocator.init(&mem);
|
||||||
|
|
||||||
|
const allocator = fba.allocator();
|
||||||
|
|
||||||
|
const env = Env.init(allocator);
|
||||||
|
defer env.deinit(allocator);
|
||||||
|
|
||||||
|
var slowest = SlowTracker.init(allocator, 5);
|
||||||
|
defer slowest.deinit();
|
||||||
|
|
||||||
|
var pass: usize = 0;
|
||||||
|
var fail: usize = 0;
|
||||||
|
var skip: usize = 0;
|
||||||
|
var leak: usize = 0;
|
||||||
|
|
||||||
|
Printer.fmt("\r\x1b[0K", .{}); // beginning of line and clear to end of line
|
||||||
|
|
||||||
|
for (builtin.test_functions) |t| {
|
||||||
|
if (isSetup(t)) {
|
||||||
|
t.func() catch |err| {
|
||||||
|
Printer.status(.fail, "\nsetup \"{s}\" failed: {}\n", .{ t.name, err });
|
||||||
|
return err;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (builtin.test_functions) |t| {
|
||||||
|
if (isSetup(t) or isTeardown(t)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
var status = Status.pass;
|
||||||
|
slowest.startTiming();
|
||||||
|
|
||||||
|
const is_unnamed_test = isUnnamed(t);
|
||||||
|
if (env.filter) |f| {
|
||||||
|
if (!is_unnamed_test and std.mem.indexOf(u8, t.name, f) == null) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const friendly_name = blk: {
|
||||||
|
const name = t.name;
|
||||||
|
var it = std.mem.splitScalar(u8, name, '.');
|
||||||
|
while (it.next()) |value| {
|
||||||
|
if (std.mem.eql(u8, value, "test")) {
|
||||||
|
const rest = it.rest();
|
||||||
|
break :blk if (rest.len > 0) rest else name;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break :blk name;
|
||||||
|
};
|
||||||
|
|
||||||
|
current_test = friendly_name;
|
||||||
|
std.testing.allocator_instance = .{};
|
||||||
|
const result = t.func();
|
||||||
|
current_test = null;
|
||||||
|
|
||||||
|
const ns_taken = slowest.endTiming(friendly_name);
|
||||||
|
|
||||||
|
if (std.testing.allocator_instance.deinit() == .leak) {
|
||||||
|
leak += 1;
|
||||||
|
Printer.status(.fail, "\n{s}\n\"{s}\" - Memory Leak\n{s}\n", .{ BORDER, friendly_name, BORDER });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (result) |_| {
|
||||||
|
pass += 1;
|
||||||
|
} else |err| switch (err) {
|
||||||
|
error.SkipZigTest => {
|
||||||
|
skip += 1;
|
||||||
|
status = .skip;
|
||||||
|
},
|
||||||
|
else => {
|
||||||
|
status = .fail;
|
||||||
|
fail += 1;
|
||||||
|
Printer.status(.fail, "\n{s}\n\"{s}\" - {s}\n{s}\n", .{ BORDER, friendly_name, @errorName(err), BORDER });
|
||||||
|
if (@errorReturnTrace()) |trace| {
|
||||||
|
std.debug.dumpStackTrace(trace.*);
|
||||||
|
}
|
||||||
|
if (env.fail_first) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
if (env.verbose) {
|
||||||
|
const ms = @as(f64, @floatFromInt(ns_taken)) / 1_000_000.0;
|
||||||
|
Printer.status(status, "\x1b[35m[{d: >10.2} ms]\x1b[0m {s: <30}", .{ ms, friendly_name });
|
||||||
|
} else {
|
||||||
|
Printer.status(status, ".", .{});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (builtin.test_functions) |t| {
|
||||||
|
if (isTeardown(t)) {
|
||||||
|
t.func() catch |err| {
|
||||||
|
Printer.status(.fail, "\nteardown \"{s}\" failed: {}\n", .{ t.name, err });
|
||||||
|
return err;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const total_tests = pass + fail;
|
||||||
|
const status = if (fail == 0) Status.pass else Status.fail;
|
||||||
|
Printer.status(status, "\n{d} of {d} test{s} passed\n", .{ pass, total_tests, if (total_tests != 1) "s" else "" });
|
||||||
|
if (skip > 0) {
|
||||||
|
Printer.status(.skip, "{d} test{s} skipped\n", .{ skip, if (skip != 1) "s" else "" });
|
||||||
|
}
|
||||||
|
if (leak > 0) {
|
||||||
|
Printer.status(.fail, "{d} test{s} leaked\n", .{ leak, if (leak != 1) "s" else "" });
|
||||||
|
}
|
||||||
|
Printer.fmt("\n", .{});
|
||||||
|
try slowest.display();
|
||||||
|
Printer.fmt("\n", .{});
|
||||||
|
std.posix.exit(if (fail == 0) 0 else 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
const Printer = struct {
|
||||||
|
fn fmt(comptime format: []const u8, args: anytype) void {
|
||||||
|
std.debug.print(format, args);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn status(s: Status, comptime format: []const u8, args: anytype) void {
|
||||||
|
switch (s) {
|
||||||
|
.pass => std.debug.print(format ++ "\x1b[32m✓\x1b[0m\n", args),
|
||||||
|
.fail => std.debug.print(format ++ "\x1b[31m✗\x1b[0m\n", args),
|
||||||
|
.skip => std.debug.print(format ++ "\x1b[33m \x1b[0m\n", args),
|
||||||
|
else => unreachable,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const Status = enum {
|
||||||
|
pass,
|
||||||
|
fail,
|
||||||
|
skip,
|
||||||
|
text,
|
||||||
|
};
|
||||||
|
|
||||||
|
const SlowTracker = struct {
|
||||||
|
const SlowestQueue = std.PriorityDequeue(TestInfo, void, compareTiming);
|
||||||
|
max: usize,
|
||||||
|
slowest: SlowestQueue,
|
||||||
|
timer: std.time.Timer,
|
||||||
|
|
||||||
|
fn init(allocator: Allocator, count: u32) SlowTracker {
|
||||||
|
const timer = std.time.Timer.start() catch @panic("failed to start timer");
|
||||||
|
var slowest = SlowestQueue.init(allocator, {});
|
||||||
|
slowest.ensureTotalCapacity(count) catch @panic("OOM");
|
||||||
|
return .{
|
||||||
|
.max = count,
|
||||||
|
.timer = timer,
|
||||||
|
.slowest = slowest,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const TestInfo = struct {
|
||||||
|
ns: u64,
|
||||||
|
name: []const u8,
|
||||||
|
};
|
||||||
|
|
||||||
|
fn deinit(self: SlowTracker) void {
|
||||||
|
self.slowest.deinit();
|
||||||
|
}
|
||||||
|
|
||||||
|
fn startTiming(self: *SlowTracker) void {
|
||||||
|
self.timer.reset();
|
||||||
|
}
|
||||||
|
|
||||||
|
fn endTiming(self: *SlowTracker, test_name: []const u8) u64 {
|
||||||
|
var timer = self.timer;
|
||||||
|
const ns = timer.lap();
|
||||||
|
|
||||||
|
var slowest = &self.slowest;
|
||||||
|
|
||||||
|
if (slowest.count() < self.max) {
|
||||||
|
// Capacity is fixed to the # of slow tests we want to track
|
||||||
|
// If we've tracked fewer tests than this capacity, than always add
|
||||||
|
slowest.add(TestInfo{ .ns = ns, .name = test_name }) catch @panic("failed to track test timing");
|
||||||
|
return ns;
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
// Optimization to avoid shifting the dequeue for the common case
|
||||||
|
// where the test isn't one of our slowest.
|
||||||
|
const fastest_of_the_slow = slowest.peekMin() orelse unreachable;
|
||||||
|
if (fastest_of_the_slow.ns > ns) {
|
||||||
|
// the test was faster than our fastest slow test, don't add
|
||||||
|
return ns;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// the previous fastest of our slow tests, has been pushed off.
|
||||||
|
_ = slowest.removeMin();
|
||||||
|
slowest.add(TestInfo{ .ns = ns, .name = test_name }) catch @panic("failed to track test timing");
|
||||||
|
return ns;
|
||||||
|
}
|
||||||
|
|
||||||
|
fn display(self: *SlowTracker) !void {
|
||||||
|
var slowest = self.slowest;
|
||||||
|
const count = slowest.count();
|
||||||
|
Printer.fmt("Slowest {d} test{s}: \n", .{ count, if (count != 1) "s" else "" });
|
||||||
|
while (slowest.removeMinOrNull()) |info| {
|
||||||
|
const ms = @as(f64, @floatFromInt(info.ns)) / 1_000_000.0;
|
||||||
|
Printer.fmt(" {d:.2}ms\t{s}\n", .{ ms, info.name });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn compareTiming(context: void, a: TestInfo, b: TestInfo) std.math.Order {
|
||||||
|
_ = context;
|
||||||
|
return std.math.order(a.ns, b.ns);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const Env = struct {
|
||||||
|
verbose: bool,
|
||||||
|
fail_first: bool,
|
||||||
|
filter: ?[]const u8,
|
||||||
|
|
||||||
|
fn init(allocator: Allocator) Env {
|
||||||
|
return .{
|
||||||
|
.verbose = readEnvBool(allocator, "TEST_VERBOSE", true),
|
||||||
|
.fail_first = readEnvBool(allocator, "TEST_FAIL_FIRST", false),
|
||||||
|
.filter = readEnv(allocator, "TEST_FILTER"),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
fn deinit(self: Env, allocator: Allocator) void {
|
||||||
|
if (self.filter) |f| {
|
||||||
|
allocator.free(f);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn readEnv(allocator: Allocator, key: []const u8) ?[]const u8 {
|
||||||
|
const v = std.process.getEnvVarOwned(allocator, key) catch |err| {
|
||||||
|
if (err == error.EnvironmentVariableNotFound) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
std.log.warn("failed to get env var {s} due to err {}", .{ key, err });
|
||||||
|
return null;
|
||||||
|
};
|
||||||
|
return v;
|
||||||
|
}
|
||||||
|
|
||||||
|
fn readEnvBool(allocator: Allocator, key: []const u8, deflt: bool) bool {
|
||||||
|
const value = readEnv(allocator, key) orelse return deflt;
|
||||||
|
defer allocator.free(value);
|
||||||
|
return std.ascii.eqlIgnoreCase(value, "true");
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
pub const panic = std.debug.FullPanic(struct {
|
||||||
|
pub fn panicFn(msg: []const u8, first_trace_addr: ?usize) noreturn {
|
||||||
|
if (current_test) |ct| {
|
||||||
|
std.debug.print("\x1b[31m{s}\npanic running \"{s}\"\n{s}\x1b[0m\n", .{ BORDER, ct, BORDER });
|
||||||
|
}
|
||||||
|
std.debug.defaultPanic(msg, first_trace_addr);
|
||||||
|
}
|
||||||
|
}.panicFn);
|
||||||
|
|
||||||
|
fn isUnnamed(t: std.builtin.TestFn) bool {
|
||||||
|
const marker = ".test_";
|
||||||
|
const test_name = t.name;
|
||||||
|
const index = std.mem.indexOf(u8, test_name, marker) orelse return false;
|
||||||
|
_ = std.fmt.parseInt(u32, test_name[index + marker.len ..], 10) catch return false;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
fn isSetup(t: std.builtin.TestFn) bool {
|
||||||
|
return std.mem.endsWith(u8, t.name, "tests:beforeAll");
|
||||||
|
}
|
||||||
|
|
||||||
|
fn isTeardown(t: std.builtin.TestFn) bool {
|
||||||
|
return std.mem.endsWith(u8, t.name, "tests:afterAll");
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user