From 8812c5640790df0b15215edbd95065aa81af3bd9 Mon Sep 17 00:00:00 2001 From: Filip Skokan Date: Fri, 3 Feb 2023 15:19:55 +0100 Subject: [PATCH 1/3] test: add WPTRunner support for variants and generating WPT reports PR-URL: https://github.com/nodejs/node/pull/46498 Reviewed-By: Yagiz Nizipli Reviewed-By: Richard Lau Backport-PR-URL: https://github.com/nodejs/node/pull/46767 --- Makefile | 6 ++ test/common/wpt.js | 262 +++++++++++++++++++++++++++++++++++---------- 2 files changed, 209 insertions(+), 59 deletions(-) diff --git a/Makefile b/Makefile index 94013466239e9c..41a8dc13e0a4f4 100644 --- a/Makefile +++ b/Makefile @@ -595,6 +595,12 @@ test-message: test-build test-wpt: all $(PYTHON) tools/test.py $(PARALLEL_ARGS) wpt +.PHONY: test-wpt-report +test-wpt-report: + $(RM) -r out/wpt + mkdir -p out/wpt + WPT_REPORT=1 $(PYTHON) tools/test.py --shell $(NODE) $(PARALLEL_ARGS) wpt + .PHONY: test-simple test-simple: | cctest # Depends on 'all'. $(PYTHON) tools/test.py $(PARALLEL_ARGS) parallel sequential diff --git a/test/common/wpt.js b/test/common/wpt.js index b5e74f4290e197..9edeff22a84554 100644 --- a/test/common/wpt.js +++ b/test/common/wpt.js @@ -6,9 +6,119 @@ const fs = require('fs'); const fsPromises = fs.promises; const path = require('path'); const events = require('events'); +const os = require('os'); const { inspect } = require('util'); const { Worker } = require('worker_threads'); +function getBrowserProperties() { + const { node: version } = process.versions; // e.g. 18.13.0, 20.0.0-nightly202302078e6e215481 + const release = /^\d+\.\d+\.\d+$/.test(version); + const browser = { + browser_channel: release ? 'stable' : 'experimental', + browser_version: version, + }; + + return browser; +} + +/** + * Return one of three expected values + * https://github.com/web-platform-tests/wpt/blob/1c6ff12/tools/wptrunner/wptrunner/tests/test_update.py#L953-L958 + */ +function getOs() { + switch (os.type()) { + case 'Linux': + return 'linux'; + case 'Darwin': + return 'mac'; + case 'Windows_NT': + return 'win'; + default: + throw new Error('Unsupported os.type()'); + } +} + +// https://github.com/web-platform-tests/wpt/blob/b24eedd/resources/testharness.js#L3705 +function sanitizeUnpairedSurrogates(str) { + return str.replace( + /([\ud800-\udbff]+)(?![\udc00-\udfff])|(^|[^\ud800-\udbff])([\udc00-\udfff]+)/g, + function(_, low, prefix, high) { + let output = prefix || ''; // Prefix may be undefined + const string = low || high; // Only one of these alternates can match + for (let i = 0; i < string.length; i++) { + output += codeUnitStr(string[i]); + } + return output; + }); +} + +function codeUnitStr(char) { + return 'U+' + char.charCodeAt(0).toString(16); +} + +class WPTReport { + constructor() { + this.results = []; + this.time_start = Date.now(); + } + + addResult(name, status) { + const result = { + test: name, + status, + subtests: [], + addSubtest(name, status, message) { + const subtest = { + status, + // https://github.com/web-platform-tests/wpt/blob/b24eedd/resources/testharness.js#L3722 + name: sanitizeUnpairedSurrogates(name), + }; + if (message) { + // https://github.com/web-platform-tests/wpt/blob/b24eedd/resources/testharness.js#L4506 + subtest.message = sanitizeUnpairedSurrogates(message); + } + this.subtests.push(subtest); + return subtest; + }, + }; + this.results.push(result); + return result; + } + + write() { + this.time_end = Date.now(); + this.results = this.results.filter((result) => { + return result.status === 'SKIP' || result.subtests.length !== 0; + }).map((result) => { + const url = new URL(result.test, 'http://wpt'); + url.pathname = url.pathname.replace(/\.js$/, '.html'); + result.test = url.href.slice(url.origin.length); + return result; + }); + + if (fs.existsSync('out/wpt/wptreport.json')) { + const prev = JSON.parse(fs.readFileSync('out/wpt/wptreport.json')); + this.results = [...prev.results, ...this.results]; + this.time_start = prev.time_start; + this.time_end = Math.max(this.time_end, prev.time_end); + this.run_info = prev.run_info; + } else { + /** + * Return required and some optional properties + * https://github.com/web-platform-tests/wpt.fyi/blob/60da175/api/README.md?plain=1#L331-L335 + */ + this.run_info = { + product: 'node.js', + ...getBrowserProperties(), + revision: process.env.WPT_REVISION || 'unknown', + os: getOs(), + }; + } + + fs.writeFileSync('out/wpt/wptreport.json', JSON.stringify(this)); + } +} + // https://github.com/web-platform-tests/wpt/blob/HEAD/resources/testharness.js // TODO: get rid of this half-baked harness in favor of the one // pulled from WPT @@ -313,6 +423,10 @@ class WPTRunner { this.unexpectedFailures = []; this.scriptsModifier = null; + + if (process.env.WPT_REPORT != null) { + this.report = new WPTReport(); + } } /** @@ -339,18 +453,27 @@ class WPTRunner { this.scriptsModifier = modifier; } - get fullInitScript() { - if (this.initScript === null && this.dummyGlobalThisScript === null) { + fullInitScript(hasSubsetScript, locationSearchString) { + let { initScript } = this; + if (hasSubsetScript || locationSearchString) { + initScript = `${initScript}\n\n//===\nglobalThis.location ||= {};`; + } + + if (locationSearchString) { + initScript = `${initScript}\n\n//===\nglobalThis.location.search = "${locationSearchString}";`; + } + + if (initScript === null && this.dummyGlobalThisScript === null) { return null; } - if (this.initScript === null) { + if (initScript === null) { return this.dummyGlobalThisScript; } else if (this.dummyGlobalThisScript === null) { - return this.initScript; + return initScript; } - return `${this.dummyGlobalThisScript}\n\n//===\n${this.initScript}`; + return `${this.dummyGlobalThisScript}\n\n//===\n${initScript}`; } /** @@ -398,15 +521,20 @@ class WPTRunner { for (const spec of queue) { const testFileName = spec.filename; const content = spec.getContent(); - const meta = spec.title = this.getMeta(content); + const meta = spec.meta = this.getMeta(content); const absolutePath = spec.getAbsolutePath(); const relativePath = spec.getRelativePath(); const harnessPath = fixtures.path('wpt', 'resources', 'testharness.js'); const scriptsToRun = []; + let hasSubsetScript = false; + // Scripts specified with the `// META: script=` header if (meta.script) { for (const script of meta.script) { + if (script === '/common/subset-tests.js' || script === '/common/subset-tests-by-key.js') { + hasSubsetScript = true; + } const obj = { filename: this.resource.toRealFilePath(relativePath, script), code: this.resource.read(relativePath, script, false), @@ -423,54 +551,65 @@ class WPTRunner { this.scriptsModifier?.(obj); scriptsToRun.push(obj); - const workerPath = path.join(__dirname, 'wpt/worker.js'); - const worker = new Worker(workerPath, { - execArgv: this.flags, - workerData: { - testRelativePath: relativePath, - wptRunner: __filename, - wptPath: this.path, - initScript: this.fullInitScript, - harness: { - code: fs.readFileSync(harnessPath, 'utf8'), - filename: harnessPath, + /** + * Example test with no META variant + * https://github.com/nodejs/node/blob/03854f6/test/fixtures/wpt/WebCryptoAPI/sign_verify/hmac.https.any.js#L1-L4 + * + * Example test with multiple META variants + * https://github.com/nodejs/node/blob/03854f6/test/fixtures/wpt/WebCryptoAPI/generateKey/successes_RSASSA-PKCS1-v1_5.https.any.js#L1-L9 + */ + for (const variant of meta.variant || ['']) { + const workerPath = path.join(__dirname, 'wpt/worker.js'); + const worker = new Worker(workerPath, { + execArgv: this.flags, + workerData: { + testRelativePath: relativePath, + wptRunner: __filename, + wptPath: this.path, + initScript: this.fullInitScript(hasSubsetScript, variant), + harness: { + code: fs.readFileSync(harnessPath, 'utf8'), + filename: harnessPath, + }, + scriptsToRun, }, - scriptsToRun, - }, - }); - this.workers.set(testFileName, worker); - - worker.on('message', (message) => { - switch (message.type) { - case 'result': - return this.resultCallback(testFileName, message.result); - case 'completion': - return this.completionCallback(testFileName, message.status); - default: - throw new Error(`Unexpected message from worker: ${message.type}`); - } - }); + }); + this.workers.set(testFileName, worker); + + let reportResult; + worker.on('message', (message) => { + switch (message.type) { + case 'result': + reportResult ||= this.report?.addResult(`/${relativePath}${variant}`, 'OK'); + return this.resultCallback(testFileName, message.result, reportResult); + case 'completion': + return this.completionCallback(testFileName, message.status); + default: + throw new Error(`Unexpected message from worker: ${message.type}`); + } + }); - worker.on('error', (err) => { - if (!this.inProgress.has(testFileName)) { - // The test is already finished. Ignore errors that occur after it. - // This can happen normally, for example in timers tests. - return; - } - this.fail( - testFileName, - { - status: NODE_UNCAUGHT, - name: 'evaluation in WPTRunner.runJsTests()', - message: err.message, - stack: inspect(err), - }, - kUncaught, - ); - this.inProgress.delete(testFileName); - }); + worker.on('error', (err) => { + if (!this.inProgress.has(testFileName)) { + // The test is already finished. Ignore errors that occur after it. + // This can happen normally, for example in timers tests. + return; + } + this.fail( + testFileName, + { + status: NODE_UNCAUGHT, + name: 'evaluation in WPTRunner.runJsTests()', + message: err.message, + stack: inspect(err), + }, + kUncaught, + ); + this.inProgress.delete(testFileName); + }); - await events.once(worker, 'exit').catch(() => {}); + await events.once(worker, 'exit').catch(() => {}); + } } process.on('exit', () => { @@ -530,6 +669,8 @@ class WPTRunner { } } + this.report?.write(); + const ran = queue.length; const total = ran + skipped; const passed = ran - expectedFailures - failures.length; @@ -554,8 +695,7 @@ class WPTRunner { getTestTitle(filename) { const spec = this.specMap.get(filename); - const title = spec.meta && spec.meta.title; - return title ? `${filename} : ${title}` : filename; + return spec.meta?.title || filename; } // Map WPT test status to strings @@ -581,14 +721,14 @@ class WPTRunner { * @param {string} filename * @param {Test} test The Test object returned by WPT harness */ - resultCallback(filename, test) { + resultCallback(filename, test, reportResult) { const status = this.getTestStatus(test.status); const title = this.getTestTitle(filename); console.log(`---- ${title} ----`); if (status !== kPass) { - this.fail(filename, test, status); + this.fail(filename, test, status, reportResult); } else { - this.succeed(filename, test, status); + this.succeed(filename, test, status, reportResult); } } @@ -636,11 +776,12 @@ class WPTRunner { } } - succeed(filename, test, status) { + succeed(filename, test, status, reportResult) { console.log(`[${status.toUpperCase()}] ${test.name}`); + reportResult?.addSubtest(test.name, 'PASS'); } - fail(filename, test, status) { + fail(filename, test, status, reportResult) { const spec = this.specMap.get(filename); const expected = spec.failedTests.includes(test.name); if (expected) { @@ -656,6 +797,9 @@ class WPTRunner { const command = `${process.execPath} ${process.execArgv}` + ` ${require.main.filename} ${filename}`; console.log(`Command: ${command}\n`); + + reportResult?.addSubtest(test.name, 'FAIL', test.message); + this.addTestResult(filename, { name: test.name, expected, @@ -685,7 +829,7 @@ class WPTRunner { const parts = match.match(/\/\/ META: ([^=]+?)=(.+)/); const key = parts[1]; const value = parts[2]; - if (key === 'script') { + if (key === 'script' || key === 'variant') { if (result[key]) { result[key].push(value); } else { From 5d45158535256bf2ad12f2d3ca01899261b0ccb4 Mon Sep 17 00:00:00 2001 From: Filip Skokan Date: Wed, 22 Feb 2023 17:40:43 +0100 Subject: [PATCH 2/3] test: fix default WPT titles MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit PR-URL: https://github.com/nodejs/node/pull/46778 Reviewed-By: Tobias Nießen Reviewed-By: Yagiz Nizipli Backport-PR-URL: https://github.com/nodejs/node/pull/46767 --- test/common/wpt.js | 3 +++ test/wpt/status/dom/events.json | 4 ++-- 2 files changed, 5 insertions(+), 2 deletions(-) diff --git a/test/common/wpt.js b/test/common/wpt.js index 9edeff22a84554..b5bc85d20618f1 100644 --- a/test/common/wpt.js +++ b/test/common/wpt.js @@ -724,6 +724,9 @@ class WPTRunner { resultCallback(filename, test, reportResult) { const status = this.getTestStatus(test.status); const title = this.getTestTitle(filename); + if (/^Untitled( \d+)?$/.test(test.name)) { + test.name = `${title}${test.name.slice(8)}`; + } console.log(`---- ${title} ----`); if (status !== kPass) { this.fail(filename, test, status, reportResult); diff --git a/test/wpt/status/dom/events.json b/test/wpt/status/dom/events.json index 95fbda98402b14..103403af5da6ca 100644 --- a/test/wpt/status/dom/events.json +++ b/test/wpt/status/dom/events.json @@ -19,8 +19,8 @@ "Event-constructors.any.js": { "fail": { "expected": [ - "Untitled 3", - "Untitled 4" + "Event constructors 3", + "Event constructors 4" ] } }, From e7720c35337cb983de3392c8f7c0512c0b1f95e7 Mon Sep 17 00:00:00 2001 From: Filip Skokan Date: Sat, 25 Feb 2023 19:19:08 +0100 Subject: [PATCH 3/3] test: fix WPT title when no META title is present PR-URL: https://github.com/nodejs/node/pull/46804 Reviewed-By: Yagiz Nizipli Reviewed-By: Richard Lau Backport-PR-URL: https://github.com/nodejs/node/pull/46767 --- test/common/wpt.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/test/common/wpt.js b/test/common/wpt.js index b5bc85d20618f1..4305cc3eee2e89 100644 --- a/test/common/wpt.js +++ b/test/common/wpt.js @@ -695,7 +695,7 @@ class WPTRunner { getTestTitle(filename) { const spec = this.specMap.get(filename); - return spec.meta?.title || filename; + return spec.meta?.title || filename.split('.')[0]; } // Map WPT test status to strings