mirror of
https://github.com/lightpanda-io/browser.git
synced 2025-10-29 23:23:28 +00:00
2
.github/workflows/wpt.yml
vendored
2
.github/workflows/wpt.yml
vendored
@@ -71,7 +71,7 @@ jobs:
|
|||||||
ln -s /usr/local/lib/netsurf/lib vendor/netsurf/lib
|
ln -s /usr/local/lib/netsurf/lib vendor/netsurf/lib
|
||||||
ln -s /usr/local/lib/netsurf/include vendor/netsurf/include
|
ln -s /usr/local/lib/netsurf/include vendor/netsurf/include
|
||||||
|
|
||||||
- run: zig build wpt -Dengine=v8
|
- run: zig build wpt -Dengine=v8 -- --safe --summary
|
||||||
|
|
||||||
# For now WPT tests doesn't pass at all.
|
# For now WPT tests doesn't pass at all.
|
||||||
# We accept then to continue the job on failure.
|
# We accept then to continue the job on failure.
|
||||||
|
|||||||
490
src/main_wpt.zig
490
src/main_wpt.zig
@@ -1,61 +1,32 @@
|
|||||||
const std = @import("std");
|
const std = @import("std");
|
||||||
|
|
||||||
const parser = @import("netsurf.zig");
|
|
||||||
const jsruntime = @import("jsruntime");
|
const jsruntime = @import("jsruntime");
|
||||||
|
|
||||||
const TPL = jsruntime.TPL;
|
const Suite = @import("wpt/testcase.zig").Suite;
|
||||||
const Env = jsruntime.Env;
|
const FileLoader = @import("wpt/fileloader.zig").FileLoader;
|
||||||
const Loop = jsruntime.Loop;
|
const wpt = @import("wpt/run.zig");
|
||||||
|
|
||||||
const DOM = @import("dom.zig");
|
const DOM = @import("dom.zig");
|
||||||
const HTMLElem = @import("html/elements.zig");
|
const HTMLElem = @import("html/elements.zig");
|
||||||
|
|
||||||
const fspath = std.fs.path;
|
|
||||||
|
|
||||||
const wpt_dir = "tests/wpt";
|
const wpt_dir = "tests/wpt";
|
||||||
|
|
||||||
// FileLoader loads files content from the filesystem.
|
const usage =
|
||||||
const FileLoader = struct {
|
\\usage: {s} [options] [test filter]
|
||||||
const FilesMap = std.StringHashMap([]const u8);
|
\\ Run the Web Test Platform.
|
||||||
|
\\
|
||||||
|
\\ -h, --help Print this help message and exit.
|
||||||
|
\\ --json result is formatted in JSON.
|
||||||
|
\\ --safe each test is run in a separate process.
|
||||||
|
\\ --summary print a summary result. Incompatible w/ --json
|
||||||
|
\\
|
||||||
|
;
|
||||||
|
|
||||||
files: FilesMap,
|
// Out list all the ouputs handled by WPT.
|
||||||
path: []const u8,
|
const Out = enum {
|
||||||
alloc: std.mem.Allocator,
|
json,
|
||||||
|
summary,
|
||||||
fn init(alloc: std.mem.Allocator, path: []const u8) FileLoader {
|
text,
|
||||||
const files = FilesMap.init(alloc);
|
|
||||||
|
|
||||||
return FileLoader{
|
|
||||||
.path = path,
|
|
||||||
.alloc = alloc,
|
|
||||||
.files = files,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
fn get(self: *FileLoader, name: []const u8) ![]const u8 {
|
|
||||||
if (!self.files.contains(name)) {
|
|
||||||
try self.load(name);
|
|
||||||
}
|
|
||||||
return self.files.get(name).?;
|
|
||||||
}
|
|
||||||
fn load(self: *FileLoader, name: []const u8) !void {
|
|
||||||
const filename = try fspath.join(self.alloc, &.{ self.path, name });
|
|
||||||
defer self.alloc.free(filename);
|
|
||||||
var file = try std.fs.cwd().openFile(filename, .{});
|
|
||||||
defer file.close();
|
|
||||||
|
|
||||||
const file_size = try file.getEndPos();
|
|
||||||
const content = try file.readToEndAlloc(self.alloc, file_size);
|
|
||||||
const namedup = try self.alloc.dupe(u8, name);
|
|
||||||
try self.files.put(namedup, content);
|
|
||||||
}
|
|
||||||
fn deinit(self: *FileLoader) void {
|
|
||||||
var iter = self.files.iterator();
|
|
||||||
while (iter.next()) |entry| {
|
|
||||||
self.alloc.free(entry.key_ptr.*);
|
|
||||||
self.alloc.free(entry.value_ptr.*);
|
|
||||||
}
|
|
||||||
self.files.deinit();
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
// TODO For now the WPT tests run is specific to WPT.
|
// TODO For now the WPT tests run is specific to WPT.
|
||||||
@@ -68,16 +39,68 @@ pub fn main() !void {
|
|||||||
// generate APIs
|
// generate APIs
|
||||||
const apis = comptime jsruntime.compile(DOM.Interfaces);
|
const apis = comptime jsruntime.compile(DOM.Interfaces);
|
||||||
|
|
||||||
std.debug.print("Running WPT test suite\n", .{});
|
|
||||||
|
|
||||||
var gpa = std.heap.GeneralPurposeAllocator(.{}){};
|
var gpa = std.heap.GeneralPurposeAllocator(.{}){};
|
||||||
defer _ = gpa.deinit();
|
defer _ = gpa.deinit();
|
||||||
const alloc = gpa.allocator();
|
const alloc = gpa.allocator();
|
||||||
|
|
||||||
const args = try std.process.argsAlloc(alloc);
|
var args = try std.process.argsWithAllocator(alloc);
|
||||||
defer std.process.argsFree(alloc, args);
|
defer args.deinit();
|
||||||
|
|
||||||
const filter = args[1..];
|
// get the exec name.
|
||||||
|
const execname = args.next().?;
|
||||||
|
|
||||||
|
var out: Out = .text;
|
||||||
|
var safe = false;
|
||||||
|
|
||||||
|
var filter = std.ArrayList([]const u8).init(alloc);
|
||||||
|
defer filter.deinit();
|
||||||
|
|
||||||
|
while (args.next()) |arg| {
|
||||||
|
if (std.mem.eql(u8, "-h", arg) or std.mem.eql(u8, "--help", arg)) {
|
||||||
|
try std.io.getStdErr().writer().print(usage, .{execname});
|
||||||
|
std.os.exit(0);
|
||||||
|
}
|
||||||
|
if (std.mem.eql(u8, "--json", arg)) {
|
||||||
|
out = .json;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (std.mem.eql(u8, "--safe", arg)) {
|
||||||
|
safe = true;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (std.mem.eql(u8, "--summary", arg)) {
|
||||||
|
out = .summary;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
try filter.append(arg[0..]);
|
||||||
|
}
|
||||||
|
|
||||||
|
// summary is available in safe mode only.
|
||||||
|
if (out == .summary) {
|
||||||
|
safe = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// browse the dir to get the tests dynamically.
|
||||||
|
var list = std.ArrayList([]const u8).init(alloc);
|
||||||
|
try wpt.find(alloc, wpt_dir, &list);
|
||||||
|
defer {
|
||||||
|
for (list.items) |tc| {
|
||||||
|
alloc.free(tc);
|
||||||
|
}
|
||||||
|
list.deinit();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (safe) {
|
||||||
|
return try runSafe(alloc, execname, out, list.items, filter.items);
|
||||||
|
}
|
||||||
|
|
||||||
|
var results = std.ArrayList(Suite).init(alloc);
|
||||||
|
defer {
|
||||||
|
for (results.items) |suite| {
|
||||||
|
suite.deinit();
|
||||||
|
}
|
||||||
|
results.deinit();
|
||||||
|
}
|
||||||
|
|
||||||
// initialize VM JS lib.
|
// initialize VM JS lib.
|
||||||
const vm = jsruntime.VM.init();
|
const vm = jsruntime.VM.init();
|
||||||
@@ -87,35 +110,12 @@ pub fn main() !void {
|
|||||||
var loader = FileLoader.init(alloc, wpt_dir);
|
var loader = FileLoader.init(alloc, wpt_dir);
|
||||||
defer loader.deinit();
|
defer loader.deinit();
|
||||||
|
|
||||||
// browse the dir to get the tests dynamically.
|
|
||||||
var list = std.ArrayList([]const u8).init(alloc);
|
|
||||||
try findWPTTests(alloc, wpt_dir, &list);
|
|
||||||
defer {
|
|
||||||
for (list.items) |tc| {
|
|
||||||
alloc.free(tc);
|
|
||||||
}
|
|
||||||
list.deinit();
|
|
||||||
}
|
|
||||||
|
|
||||||
var run: usize = 0;
|
var run: usize = 0;
|
||||||
var failures: usize = 0;
|
var failures: usize = 0;
|
||||||
for (list.items) |tc| {
|
for (list.items) |tc| {
|
||||||
if (filter.len > 0) {
|
if (!shouldRun(filter.items, tc)) {
|
||||||
var match = false;
|
|
||||||
for (filter) |f| {
|
|
||||||
if (std.mem.startsWith(u8, tc, f)) {
|
|
||||||
match = true;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
if (std.mem.endsWith(u8, tc, f)) {
|
|
||||||
match = true;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (!match) {
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
run += 1;
|
run += 1;
|
||||||
|
|
||||||
@@ -123,178 +123,216 @@ pub fn main() !void {
|
|||||||
var arena = std.heap.ArenaAllocator.init(alloc);
|
var arena = std.heap.ArenaAllocator.init(alloc);
|
||||||
defer arena.deinit();
|
defer arena.deinit();
|
||||||
|
|
||||||
// TODO I don't use testing.expect here b/c I want to execute all the
|
const res = wpt.run(&arena, apis, wpt_dir, tc, &loader) catch |err| {
|
||||||
// tests. And testing.expect stops running test in the first failure.
|
const suite = try Suite.init(alloc, tc, false, @errorName(err), null);
|
||||||
const res = runWPT(&arena, apis, tc, &loader) catch |err| {
|
try results.append(suite);
|
||||||
std.debug.print("FAIL\t{s}\n{any}\n", .{ tc, err });
|
|
||||||
|
if (out == .text) {
|
||||||
|
std.debug.print("FAIL\t{s}\t{}\n", .{ tc, err });
|
||||||
|
}
|
||||||
failures += 1;
|
failures += 1;
|
||||||
continue;
|
continue;
|
||||||
};
|
};
|
||||||
// no need to call res.deinit() thanks to the arena allocator.
|
// no need to call res.deinit() thanks to the arena allocator.
|
||||||
|
|
||||||
if (!res.success) {
|
const suite = try Suite.init(alloc, tc, res.success, res.result, res.stack);
|
||||||
std.debug.print("FAIL\t{s}\n{s}\n", .{ tc, res.stack orelse res.result });
|
try results.append(suite);
|
||||||
failures += 1;
|
|
||||||
continue;
|
if (out == .json) {
|
||||||
}
|
|
||||||
if (!std.mem.eql(u8, res.result, "Pass")) {
|
|
||||||
std.debug.print("FAIL\t{s}\n{s}\n", .{ tc, res.stack orelse res.result });
|
|
||||||
failures += 1;
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
std.debug.print("PASS\t{s}\n", .{tc});
|
if (!suite.pass) {
|
||||||
|
std.debug.print("Fail\t{s}\n{s}\n", .{ suite.name, suite.fmtMessage() });
|
||||||
|
failures += 1;
|
||||||
|
} else {
|
||||||
|
std.debug.print("Pass\t{s}\n", .{suite.name});
|
||||||
}
|
}
|
||||||
|
|
||||||
if (failures > 0) {
|
// display details
|
||||||
std.debug.print("{d}/{d} tests failures\n", .{ failures, run });
|
if (suite.cases) |cases| {
|
||||||
|
for (cases) |case| {
|
||||||
|
std.debug.print("\t{s}\t{s}\t{s}\n", .{ case.fmtStatus(), case.name, case.fmtMessage() });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (out == .json) {
|
||||||
|
var output = std.ArrayList(Test).init(alloc);
|
||||||
|
defer output.deinit();
|
||||||
|
|
||||||
|
for (results.items) |suite| {
|
||||||
|
var cases = std.ArrayList(Case).init(alloc);
|
||||||
|
defer cases.deinit();
|
||||||
|
|
||||||
|
if (suite.cases) |scases| {
|
||||||
|
for (scases) |case| {
|
||||||
|
try cases.append(Case{
|
||||||
|
.pass = case.pass,
|
||||||
|
.name = case.name,
|
||||||
|
.message = case.message,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// no cases, generate a fake one
|
||||||
|
try cases.append(Case{
|
||||||
|
.pass = suite.pass,
|
||||||
|
.name = suite.name,
|
||||||
|
.message = suite.stack orelse suite.message,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
try output.append(Test{
|
||||||
|
.pass = suite.pass,
|
||||||
|
.name = suite.name,
|
||||||
|
.cases = try cases.toOwnedSlice(),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
defer {
|
||||||
|
for (output.items) |suite| {
|
||||||
|
alloc.free(suite.cases);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
try std.json.stringify(output.items, .{ .whitespace = .indent_2 }, std.io.getStdOut().writer());
|
||||||
|
std.os.exit(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (out == .text and failures > 0) {
|
||||||
|
std.debug.print("{d}/{d} tests suites failures\n", .{ failures, run });
|
||||||
std.os.exit(1);
|
std.os.exit(1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// runWPT parses the given HTML file, starts a js env and run the first script
|
// struct used for JSON output.
|
||||||
// tags containing javascript sources.
|
const Case = struct {
|
||||||
// It loads first the js libs files.
|
pass: bool,
|
||||||
fn runWPT(arena: *std.heap.ArenaAllocator, comptime apis: []jsruntime.API, f: []const u8, loader: *FileLoader) !jsruntime.JSResult {
|
name: []const u8,
|
||||||
|
message: ?[]const u8,
|
||||||
|
};
|
||||||
|
const Test = struct {
|
||||||
|
pass: bool,
|
||||||
|
crash: bool = false,
|
||||||
|
name: []const u8,
|
||||||
|
cases: []Case,
|
||||||
|
};
|
||||||
|
|
||||||
|
// shouldRun return true if the test should be run accroding to the given filters.
|
||||||
|
fn shouldRun(filter: [][]const u8, tc: []const u8) bool {
|
||||||
|
if (filter.len == 0) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (filter) |f| {
|
||||||
|
if (std.mem.startsWith(u8, tc, f)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (std.mem.endsWith(u8, tc, f)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// runSafe rune each test cae in a separate child process to detect crashes.
|
||||||
|
fn runSafe(
|
||||||
|
allocator: std.mem.Allocator,
|
||||||
|
execname: []const u8,
|
||||||
|
out: Out,
|
||||||
|
testcases: [][]const u8,
|
||||||
|
filter: [][]const u8,
|
||||||
|
) !void {
|
||||||
|
var arena = std.heap.ArenaAllocator.init(allocator);
|
||||||
|
defer arena.deinit();
|
||||||
|
|
||||||
const alloc = arena.allocator();
|
const alloc = arena.allocator();
|
||||||
|
|
||||||
// document
|
const Result = enum {
|
||||||
const html_doc = try parser.documentHTMLParseFromFileAlloc(alloc, f);
|
pass,
|
||||||
const doc = parser.documentHTMLToDocument(html_doc);
|
fail,
|
||||||
|
crash,
|
||||||
|
};
|
||||||
|
|
||||||
const dirname = fspath.dirname(f[wpt_dir.len..]) orelse unreachable;
|
var argv = try std.ArrayList([]const u8).initCapacity(alloc, 3);
|
||||||
|
defer argv.deinit();
|
||||||
// create JS env
|
argv.appendAssumeCapacity(execname);
|
||||||
var loop = try Loop.init(alloc);
|
if (out == .json) {
|
||||||
defer loop.deinit();
|
argv.appendAssumeCapacity("--json");
|
||||||
var js_env = try Env.init(arena, &loop);
|
|
||||||
defer js_env.deinit();
|
|
||||||
|
|
||||||
// load APIs in JS env
|
|
||||||
var tpls: [apis.len]TPL = undefined;
|
|
||||||
try js_env.load(apis, &tpls);
|
|
||||||
|
|
||||||
// start JS env
|
|
||||||
try js_env.start(alloc, apis);
|
|
||||||
defer js_env.stop();
|
|
||||||
|
|
||||||
// add document object
|
|
||||||
try js_env.addObject(apis, html_doc, "document");
|
|
||||||
|
|
||||||
// alias global as self and window
|
|
||||||
try js_env.attachObject(try js_env.getGlobal(), "self", null);
|
|
||||||
try js_env.attachObject(try js_env.getGlobal(), "window", null);
|
|
||||||
|
|
||||||
// thanks to the arena, we don't need to deinit res.
|
|
||||||
var res: jsruntime.JSResult = undefined;
|
|
||||||
|
|
||||||
const init =
|
|
||||||
\\window.listeners = [];
|
|
||||||
\\window.document = document;
|
|
||||||
\\window.parent = window;
|
|
||||||
\\window.addEventListener = function (type, listener, options) {
|
|
||||||
\\ window.listeners.push({type: type, listener: listener, options: options});
|
|
||||||
\\};
|
|
||||||
\\window.dispatchEvent = function (event) {
|
|
||||||
\\ len = window.listeners.length;
|
|
||||||
\\ for (var i = 0; i < len; i++) {
|
|
||||||
\\ if (window.listeners[i].type == event.target) {
|
|
||||||
\\ window.listeners[i].listener(event);
|
|
||||||
\\ }
|
|
||||||
\\ }
|
|
||||||
\\ return true;
|
|
||||||
\\};
|
|
||||||
\\window.removeEventListener = function () {};
|
|
||||||
\\
|
|
||||||
\\console = [];
|
|
||||||
\\console.log = function () {
|
|
||||||
\\ console.push(...arguments);
|
|
||||||
\\};
|
|
||||||
;
|
|
||||||
res = try evalJS(js_env, alloc, init, "init");
|
|
||||||
if (!res.success) {
|
|
||||||
return res;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// loop hover the scripts.
|
var output = std.ArrayList(Test).init(alloc);
|
||||||
const scripts = try parser.documentGetElementsByTagName(doc, "script");
|
|
||||||
const slen = try parser.nodeListLength(scripts);
|
|
||||||
for (0..slen) |i| {
|
|
||||||
const s = (try parser.nodeListItem(scripts, @intCast(i))).?;
|
|
||||||
|
|
||||||
// If the script contains an src attribute, load it.
|
for (testcases) |tc| {
|
||||||
if (try parser.elementGetAttribute(@as(*parser.Element, @ptrCast(s)), "src")) |src| {
|
if (!shouldRun(filter, tc)) {
|
||||||
var path = src;
|
|
||||||
if (!std.mem.startsWith(u8, src, "/")) {
|
|
||||||
// no need to free path, thanks to the arena.
|
|
||||||
path = try fspath.join(alloc, &.{ "/", dirname, path });
|
|
||||||
}
|
|
||||||
|
|
||||||
res = try evalJS(js_env, alloc, try loader.get(path), src);
|
|
||||||
if (!res.success) {
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// If the script as a source text, execute it.
|
|
||||||
const src = try parser.nodeTextContent(s) orelse continue;
|
|
||||||
res = try evalJS(js_env, alloc, src, "");
|
|
||||||
|
|
||||||
// return the first failure.
|
|
||||||
if (!res.success) {
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Mark tests as ready to run.
|
|
||||||
res = try evalJS(js_env, alloc, "window.dispatchEvent({target: 'load'});", "ready");
|
|
||||||
if (!res.success) {
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
|
|
||||||
// display console logs
|
|
||||||
res = try evalJS(js_env, alloc, "console.join(', ');", "console");
|
|
||||||
if (res.result.len > 0) {
|
|
||||||
std.debug.print("-- CONSOLE LOG\n{s}\n--\n", .{res.result});
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check the final test status.
|
|
||||||
res = try evalJS(js_env, alloc, "report.status;", "teststatus");
|
|
||||||
if (!res.success) {
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
|
|
||||||
// If the test failed, return detailed logs intead of the simple status.
|
|
||||||
if (!std.mem.eql(u8, res.result, "Pass")) {
|
|
||||||
return try evalJS(js_env, alloc, "report.log", "teststatus");
|
|
||||||
}
|
|
||||||
|
|
||||||
// return the final result.
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
|
|
||||||
fn evalJS(env: jsruntime.Env, alloc: std.mem.Allocator, script: []const u8, name: ?[]const u8) !jsruntime.JSResult {
|
|
||||||
var res = jsruntime.JSResult{};
|
|
||||||
try env.run(alloc, script, name, &res, null);
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
|
|
||||||
// browse the path to find the tests list.
|
|
||||||
fn findWPTTests(allocator: std.mem.Allocator, path: []const u8, list: *std.ArrayList([]const u8)) !void {
|
|
||||||
var dir = try std.fs.cwd().openIterableDir(path, .{ .no_follow = true });
|
|
||||||
defer dir.close();
|
|
||||||
|
|
||||||
var walker = try dir.walk(allocator);
|
|
||||||
defer walker.deinit();
|
|
||||||
|
|
||||||
while (try walker.next()) |entry| {
|
|
||||||
if (entry.kind != .file) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
if (!std.mem.endsWith(u8, entry.basename, ".html")) {
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
try list.append(try fspath.join(allocator, &.{ path, entry.path }));
|
argv.appendAssumeCapacity(tc);
|
||||||
|
defer _ = argv.pop();
|
||||||
|
|
||||||
|
// TODO use std.ChildProcess.run after next zig upgrade.
|
||||||
|
var child = std.ChildProcess.init(argv.items, alloc);
|
||||||
|
child.stdin_behavior = .Ignore;
|
||||||
|
child.stdout_behavior = .Pipe;
|
||||||
|
child.stderr_behavior = .Pipe;
|
||||||
|
|
||||||
|
var stdout = std.ArrayList(u8).init(alloc);
|
||||||
|
var stderr = std.ArrayList(u8).init(alloc);
|
||||||
|
|
||||||
|
try child.spawn();
|
||||||
|
try child.collectOutput(&stdout, &stderr, 1024 * 1024);
|
||||||
|
const term = try child.wait();
|
||||||
|
|
||||||
|
var result: Result = undefined;
|
||||||
|
switch (term) {
|
||||||
|
.Exited => |v| {
|
||||||
|
if (v == 0) {
|
||||||
|
result = .pass;
|
||||||
|
} else {
|
||||||
|
result = .fail;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
.Signal => result = .crash,
|
||||||
|
.Stopped => result = .crash,
|
||||||
|
.Unknown => result = .crash,
|
||||||
|
}
|
||||||
|
|
||||||
|
if (out == .summary) {
|
||||||
|
switch (result) {
|
||||||
|
.pass => std.debug.print("Pass", .{}),
|
||||||
|
.fail => std.debug.print("Fail", .{}),
|
||||||
|
.crash => std.debug.print("Crash", .{}),
|
||||||
|
}
|
||||||
|
std.debug.print("\t{s}\n", .{tc});
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (out == .json) {
|
||||||
|
if (result == .crash) {
|
||||||
|
var cases = [_]Case{.{
|
||||||
|
.pass = false,
|
||||||
|
.name = "crash",
|
||||||
|
.message = stderr.items,
|
||||||
|
}};
|
||||||
|
try output.append(Test{
|
||||||
|
.pass = false,
|
||||||
|
.crash = true,
|
||||||
|
.name = tc,
|
||||||
|
.cases = cases[0..1],
|
||||||
|
});
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const jp = try std.json.parseFromSlice([]Test, alloc, stdout.items, .{});
|
||||||
|
try output.appendSlice(jp.value);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
std.debug.print("{s}\n", .{stderr.items});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (out == .json) {
|
||||||
|
try std.json.stringify(output.items, .{ .whitespace = .indent_2 }, std.io.getStdOut().writer());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
46
src/wpt/fileloader.zig
Normal file
46
src/wpt/fileloader.zig
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
const std = @import("std");
|
||||||
|
const fspath = std.fs.path;
|
||||||
|
|
||||||
|
// FileLoader loads files content from the filesystem.
|
||||||
|
pub const FileLoader = struct {
|
||||||
|
const FilesMap = std.StringHashMap([]const u8);
|
||||||
|
|
||||||
|
files: FilesMap,
|
||||||
|
path: []const u8,
|
||||||
|
alloc: std.mem.Allocator,
|
||||||
|
|
||||||
|
pub fn init(alloc: std.mem.Allocator, path: []const u8) FileLoader {
|
||||||
|
const files = FilesMap.init(alloc);
|
||||||
|
|
||||||
|
return FileLoader{
|
||||||
|
.path = path,
|
||||||
|
.alloc = alloc,
|
||||||
|
.files = files,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
pub fn get(self: *FileLoader, name: []const u8) ![]const u8 {
|
||||||
|
if (!self.files.contains(name)) {
|
||||||
|
try self.load(name);
|
||||||
|
}
|
||||||
|
return self.files.get(name).?;
|
||||||
|
}
|
||||||
|
pub fn load(self: *FileLoader, name: []const u8) !void {
|
||||||
|
const filename = try fspath.join(self.alloc, &.{ self.path, name });
|
||||||
|
defer self.alloc.free(filename);
|
||||||
|
var file = try std.fs.cwd().openFile(filename, .{});
|
||||||
|
defer file.close();
|
||||||
|
|
||||||
|
const file_size = try file.getEndPos();
|
||||||
|
const content = try file.readToEndAlloc(self.alloc, file_size);
|
||||||
|
const namedup = try self.alloc.dupe(u8, name);
|
||||||
|
try self.files.put(namedup, content);
|
||||||
|
}
|
||||||
|
pub fn deinit(self: *FileLoader) void {
|
||||||
|
var iter = self.files.iterator();
|
||||||
|
while (iter.next()) |entry| {
|
||||||
|
self.alloc.free(entry.key_ptr.*);
|
||||||
|
self.alloc.free(entry.value_ptr.*);
|
||||||
|
}
|
||||||
|
self.files.deinit();
|
||||||
|
}
|
||||||
|
};
|
||||||
158
src/wpt/run.zig
Normal file
158
src/wpt/run.zig
Normal file
@@ -0,0 +1,158 @@
|
|||||||
|
const std = @import("std");
|
||||||
|
const fspath = std.fs.path;
|
||||||
|
|
||||||
|
const FileLoader = @import("fileloader.zig").FileLoader;
|
||||||
|
|
||||||
|
const parser = @import("../netsurf.zig");
|
||||||
|
|
||||||
|
const jsruntime = @import("jsruntime");
|
||||||
|
const Loop = jsruntime.Loop;
|
||||||
|
const Env = jsruntime.Env;
|
||||||
|
const TPL = jsruntime.TPL;
|
||||||
|
|
||||||
|
// runWPT parses the given HTML file, starts a js env and run the first script
|
||||||
|
// tags containing javascript sources.
|
||||||
|
// It loads first the js libs files.
|
||||||
|
pub fn run(arena: *std.heap.ArenaAllocator, comptime apis: []jsruntime.API, comptime dir: []const u8, f: []const u8, loader: *FileLoader) !jsruntime.JSResult {
|
||||||
|
const alloc = arena.allocator();
|
||||||
|
|
||||||
|
// document
|
||||||
|
const html_doc = try parser.documentHTMLParseFromFileAlloc(alloc, f);
|
||||||
|
const doc = parser.documentHTMLToDocument(html_doc);
|
||||||
|
|
||||||
|
const dirname = fspath.dirname(f[dir.len..]) orelse unreachable;
|
||||||
|
|
||||||
|
// create JS env
|
||||||
|
var loop = try Loop.init(alloc);
|
||||||
|
defer loop.deinit();
|
||||||
|
var js_env = try Env.init(arena, &loop);
|
||||||
|
defer js_env.deinit();
|
||||||
|
|
||||||
|
// load APIs in JS env
|
||||||
|
var tpls: [apis.len]TPL = undefined;
|
||||||
|
try js_env.load(apis, &tpls);
|
||||||
|
|
||||||
|
// start JS env
|
||||||
|
try js_env.start(alloc, apis);
|
||||||
|
defer js_env.stop();
|
||||||
|
|
||||||
|
// add document object
|
||||||
|
try js_env.addObject(apis, html_doc, "document");
|
||||||
|
|
||||||
|
// alias global as self and window
|
||||||
|
try js_env.attachObject(try js_env.getGlobal(), "self", null);
|
||||||
|
try js_env.attachObject(try js_env.getGlobal(), "window", null);
|
||||||
|
|
||||||
|
// thanks to the arena, we don't need to deinit res.
|
||||||
|
var res: jsruntime.JSResult = undefined;
|
||||||
|
|
||||||
|
const init =
|
||||||
|
\\window.listeners = [];
|
||||||
|
\\window.document = document;
|
||||||
|
\\window.parent = window;
|
||||||
|
\\window.addEventListener = function (type, listener, options) {
|
||||||
|
\\ window.listeners.push({type: type, listener: listener, options: options});
|
||||||
|
\\};
|
||||||
|
\\window.dispatchEvent = function (event) {
|
||||||
|
\\ len = window.listeners.length;
|
||||||
|
\\ for (var i = 0; i < len; i++) {
|
||||||
|
\\ if (window.listeners[i].type == event.target) {
|
||||||
|
\\ window.listeners[i].listener(event);
|
||||||
|
\\ }
|
||||||
|
\\ }
|
||||||
|
\\ return true;
|
||||||
|
\\};
|
||||||
|
\\window.removeEventListener = function () {};
|
||||||
|
\\
|
||||||
|
\\console = [];
|
||||||
|
\\console.log = function () {
|
||||||
|
\\ console.push(...arguments);
|
||||||
|
\\};
|
||||||
|
;
|
||||||
|
res = try evalJS(js_env, alloc, init, "init");
|
||||||
|
if (!res.success) {
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
// loop hover the scripts.
|
||||||
|
const scripts = try parser.documentGetElementsByTagName(doc, "script");
|
||||||
|
const slen = try parser.nodeListLength(scripts);
|
||||||
|
for (0..slen) |i| {
|
||||||
|
const s = (try parser.nodeListItem(scripts, @intCast(i))).?;
|
||||||
|
|
||||||
|
// If the script contains an src attribute, load it.
|
||||||
|
if (try parser.elementGetAttribute(@as(*parser.Element, @ptrCast(s)), "src")) |src| {
|
||||||
|
var path = src;
|
||||||
|
if (!std.mem.startsWith(u8, src, "/")) {
|
||||||
|
// no need to free path, thanks to the arena.
|
||||||
|
path = try fspath.join(alloc, &.{ "/", dirname, path });
|
||||||
|
}
|
||||||
|
|
||||||
|
res = try evalJS(js_env, alloc, try loader.get(path), src);
|
||||||
|
if (!res.success) {
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// If the script as a source text, execute it.
|
||||||
|
const src = try parser.nodeTextContent(s) orelse continue;
|
||||||
|
res = try evalJS(js_env, alloc, src, "");
|
||||||
|
|
||||||
|
// return the first failure.
|
||||||
|
if (!res.success) {
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Mark tests as ready to run.
|
||||||
|
res = try evalJS(js_env, alloc, "window.dispatchEvent({target: 'load'});", "ready");
|
||||||
|
if (!res.success) {
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
// display console logs
|
||||||
|
res = try evalJS(js_env, alloc, "console.join(', ');", "console");
|
||||||
|
if (res.result.len > 0) {
|
||||||
|
std.debug.print("-- CONSOLE LOG\n{s}\n--\n", .{res.result});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check the final test status.
|
||||||
|
res = try evalJS(js_env, alloc, "report.status;", "teststatus");
|
||||||
|
if (!res.success) {
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If the test failed, return detailed logs intead of the simple status.
|
||||||
|
if (!std.mem.eql(u8, res.result, "Pass")) {
|
||||||
|
return try evalJS(js_env, alloc, "report.log", "teststatus");
|
||||||
|
}
|
||||||
|
|
||||||
|
// return the final result.
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
fn evalJS(env: jsruntime.Env, alloc: std.mem.Allocator, script: []const u8, name: ?[]const u8) !jsruntime.JSResult {
|
||||||
|
var res = jsruntime.JSResult{};
|
||||||
|
try env.run(alloc, script, name, &res, null);
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
// browse the path to find the tests list.
|
||||||
|
pub fn find(allocator: std.mem.Allocator, comptime path: []const u8, list: *std.ArrayList([]const u8)) !void {
|
||||||
|
var dir = try std.fs.cwd().openIterableDir(path, .{ .no_follow = true });
|
||||||
|
defer dir.close();
|
||||||
|
|
||||||
|
var walker = try dir.walk(allocator);
|
||||||
|
defer walker.deinit();
|
||||||
|
|
||||||
|
while (try walker.next()) |entry| {
|
||||||
|
if (entry.kind != .file) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (!std.mem.endsWith(u8, entry.basename, ".html")) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
try list.append(try fspath.join(allocator, &.{ path, entry.path }));
|
||||||
|
}
|
||||||
|
}
|
||||||
248
src/wpt/testcase.zig
Normal file
248
src/wpt/testcase.zig
Normal file
@@ -0,0 +1,248 @@
|
|||||||
|
const std = @import("std");
|
||||||
|
const testing = std.testing;
|
||||||
|
|
||||||
|
pub const Case = struct {
|
||||||
|
pass: bool,
|
||||||
|
name: []const u8,
|
||||||
|
|
||||||
|
message: ?[]const u8,
|
||||||
|
|
||||||
|
fn init(alloc: std.mem.Allocator, name: []const u8, status: []const u8, message: []const u8) !Case {
|
||||||
|
var case = Case{
|
||||||
|
.pass = std.mem.eql(u8, "Pass", status),
|
||||||
|
.name = try alloc.dupe(u8, name),
|
||||||
|
.message = null,
|
||||||
|
};
|
||||||
|
|
||||||
|
if (message.len > 0) {
|
||||||
|
case.message = try alloc.dupe(u8, message);
|
||||||
|
}
|
||||||
|
|
||||||
|
return case;
|
||||||
|
}
|
||||||
|
|
||||||
|
fn deinit(self: Case, alloc: std.mem.Allocator) void {
|
||||||
|
alloc.free(self.name);
|
||||||
|
|
||||||
|
if (self.message) |msg| {
|
||||||
|
alloc.free(msg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn fmtStatus(self: Case) []const u8 {
|
||||||
|
if (self.pass) {
|
||||||
|
return "Pass";
|
||||||
|
}
|
||||||
|
return "Fail";
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn fmtMessage(self: Case) []const u8 {
|
||||||
|
if (self.message) |v| {
|
||||||
|
return v;
|
||||||
|
}
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
pub const Suite = struct {
|
||||||
|
alloc: std.mem.Allocator,
|
||||||
|
pass: bool,
|
||||||
|
name: []const u8,
|
||||||
|
message: ?[]const u8,
|
||||||
|
stack: ?[]const u8,
|
||||||
|
cases: ?[]Case,
|
||||||
|
|
||||||
|
// caller owns the wpt.Suite.
|
||||||
|
// owner must call deinit().
|
||||||
|
pub fn init(alloc: std.mem.Allocator, name: []const u8, pass: bool, res: []const u8, stack: ?[]const u8) !Suite {
|
||||||
|
var suite = Suite{
|
||||||
|
.alloc = alloc,
|
||||||
|
.pass = false,
|
||||||
|
.name = try alloc.dupe(u8, name),
|
||||||
|
.message = null,
|
||||||
|
.stack = null,
|
||||||
|
.cases = null,
|
||||||
|
};
|
||||||
|
|
||||||
|
// handle JS error.
|
||||||
|
if (!pass) {
|
||||||
|
suite.message = try alloc.dupe(u8, res);
|
||||||
|
if (stack) |st| {
|
||||||
|
suite.stack = try alloc.dupe(u8, st);
|
||||||
|
}
|
||||||
|
|
||||||
|
return suite;
|
||||||
|
}
|
||||||
|
|
||||||
|
// no JS error, let's try to parse the result.
|
||||||
|
suite.pass = true;
|
||||||
|
|
||||||
|
// special case: the result contains only "Pass" message
|
||||||
|
if (std.mem.eql(u8, "Pass", res)) {
|
||||||
|
return suite;
|
||||||
|
}
|
||||||
|
|
||||||
|
var cases = std.ArrayList(Case).init(alloc);
|
||||||
|
defer cases.deinit();
|
||||||
|
|
||||||
|
var lines = std.mem.splitScalar(u8, res, '\n');
|
||||||
|
while (lines.next()) |line| {
|
||||||
|
if (line.len == 0) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
var fields = std.mem.splitScalar(u8, line, '|');
|
||||||
|
var ff: [3][]const u8 = .{ "", "", "" };
|
||||||
|
var i: u8 = 0;
|
||||||
|
while (fields.next()) |field| {
|
||||||
|
if (i >= 3) {
|
||||||
|
suite.pass = false;
|
||||||
|
suite.message = try alloc.dupe(u8, res);
|
||||||
|
return suite;
|
||||||
|
}
|
||||||
|
|
||||||
|
ff[i] = field;
|
||||||
|
i += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// invalid output format
|
||||||
|
if (i != 2 and i != 3) {
|
||||||
|
suite.pass = false;
|
||||||
|
suite.message = try alloc.dupe(u8, res);
|
||||||
|
return suite;
|
||||||
|
}
|
||||||
|
|
||||||
|
const case = try Case.init(alloc, ff[0], ff[1], ff[2]);
|
||||||
|
if (!case.pass) {
|
||||||
|
suite.pass = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
try cases.append(case);
|
||||||
|
}
|
||||||
|
|
||||||
|
suite.cases = try cases.toOwnedSlice();
|
||||||
|
|
||||||
|
return suite;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn deinit(self: Suite) void {
|
||||||
|
self.alloc.free(self.name);
|
||||||
|
|
||||||
|
if (self.stack) |stack| {
|
||||||
|
self.alloc.free(stack);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (self.message) |res| {
|
||||||
|
self.alloc.free(res);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (self.cases) |cases| {
|
||||||
|
for (cases) |case| {
|
||||||
|
case.deinit(self.alloc);
|
||||||
|
}
|
||||||
|
self.alloc.free(cases);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn fmtMessage(self: Suite) []const u8 {
|
||||||
|
if (self.message) |v| {
|
||||||
|
return v;
|
||||||
|
}
|
||||||
|
if (self.stack) |v| {
|
||||||
|
return v;
|
||||||
|
}
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
test "success test case" {
|
||||||
|
const alloc = testing.allocator;
|
||||||
|
|
||||||
|
const Res = struct {
|
||||||
|
pass: bool,
|
||||||
|
result: []const u8,
|
||||||
|
};
|
||||||
|
|
||||||
|
const res = Res{
|
||||||
|
.pass = true,
|
||||||
|
.result =
|
||||||
|
\\Empty string as a name for Document.getElementsByTagName|Pass
|
||||||
|
\\Empty string as a name for Element.getElementsByTagName|Pass
|
||||||
|
\\
|
||||||
|
,
|
||||||
|
};
|
||||||
|
|
||||||
|
const suite = Suite.init(alloc, "foo", res.pass, res.result, null) catch unreachable; // TODO
|
||||||
|
defer suite.deinit();
|
||||||
|
|
||||||
|
try testing.expect(suite.pass == true);
|
||||||
|
try testing.expect(suite.cases != null);
|
||||||
|
try testing.expect(suite.cases.?.len == 2);
|
||||||
|
try testing.expect(suite.cases.?[0].pass == true);
|
||||||
|
try testing.expect(suite.cases.?[1].pass == true);
|
||||||
|
}
|
||||||
|
|
||||||
|
test "failed test case" {
|
||||||
|
const alloc = testing.allocator;
|
||||||
|
|
||||||
|
const Res = struct {
|
||||||
|
pass: bool,
|
||||||
|
result: []const u8,
|
||||||
|
};
|
||||||
|
|
||||||
|
const res = Res{
|
||||||
|
.pass = true,
|
||||||
|
.result =
|
||||||
|
\\Empty string as a name for Document.getElementsByTagName|Pass
|
||||||
|
\\Empty string as a name for Element.getElementsByTagName|Fail|div.getElementsByTagName is not a function
|
||||||
|
\\
|
||||||
|
,
|
||||||
|
};
|
||||||
|
|
||||||
|
const suite = Suite.init(alloc, "foo", res.pass, res.result, null) catch unreachable; // TODO
|
||||||
|
defer suite.deinit();
|
||||||
|
|
||||||
|
try testing.expect(suite.pass == false);
|
||||||
|
try testing.expect(suite.cases != null);
|
||||||
|
try testing.expect(suite.cases.?.len == 2);
|
||||||
|
try testing.expect(suite.cases.?[0].pass == true);
|
||||||
|
try testing.expect(suite.cases.?[1].pass == false);
|
||||||
|
}
|
||||||
|
|
||||||
|
test "invalid result" {
|
||||||
|
const alloc = testing.allocator;
|
||||||
|
|
||||||
|
const Res = struct {
|
||||||
|
pass: bool,
|
||||||
|
result: []const u8,
|
||||||
|
};
|
||||||
|
|
||||||
|
const res = Res{
|
||||||
|
.pass = true,
|
||||||
|
.result =
|
||||||
|
\\this is|an|invalid|result
|
||||||
|
,
|
||||||
|
};
|
||||||
|
|
||||||
|
const suite = Suite.init(alloc, "foo", res.pass, res.result, null) catch unreachable; // TODO
|
||||||
|
defer suite.deinit();
|
||||||
|
|
||||||
|
try testing.expect(suite.pass == false);
|
||||||
|
try testing.expect(suite.message != null);
|
||||||
|
try testing.expect(std.mem.eql(u8, res.result, suite.message.?));
|
||||||
|
try testing.expect(suite.cases == null);
|
||||||
|
|
||||||
|
const res2 = Res{
|
||||||
|
.pass = true,
|
||||||
|
.result =
|
||||||
|
\\this is an invalid result.
|
||||||
|
,
|
||||||
|
};
|
||||||
|
|
||||||
|
const suite2 = Suite.init(alloc, "foo", res2.pass, res2.result, null) catch unreachable; // TODO
|
||||||
|
defer suite2.deinit();
|
||||||
|
|
||||||
|
try testing.expect(suite2.pass == false);
|
||||||
|
try testing.expect(suite2.message != null);
|
||||||
|
try testing.expect(std.mem.eql(u8, res2.result, suite2.message.?));
|
||||||
|
try testing.expect(suite2.cases == null);
|
||||||
|
}
|
||||||
@@ -10,7 +10,7 @@ var report = {
|
|||||||
add_completion_callback(function (tests, status) {
|
add_completion_callback(function (tests, status) {
|
||||||
// report the tests global status.
|
// report the tests global status.
|
||||||
// TODO the status.status is always OK even if a test fail.
|
// TODO the status.status is always OK even if a test fail.
|
||||||
// I ingore the global status for now, but I build one with the tests results.
|
// I ignore the global status for now, but I build one with the tests results.
|
||||||
//report.status = status.status;
|
//report.status = status.status;
|
||||||
|
|
||||||
var status = "Pass";
|
var status = "Pass";
|
||||||
@@ -18,9 +18,9 @@ add_completion_callback(function (tests, status) {
|
|||||||
var log = "";
|
var log = "";
|
||||||
for (var i = 0; i < tests.length; i++) {
|
for (var i = 0; i < tests.length; i++) {
|
||||||
const test = tests[i];
|
const test = tests[i];
|
||||||
log += test.name+": "+test.format_status();
|
log += test.name+"|"+test.format_status();
|
||||||
if (test.message != null) {
|
if (test.message != null) {
|
||||||
log += " " + test.message;
|
log += "|"+test.message.replaceAll("\n"," ");
|
||||||
}
|
}
|
||||||
log += "\n";
|
log += "\n";
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user