Skip to content

Commit 5ad7589

Browse files
Add service worker support (gpuweb#3419)
* Generate Web Worker script for every test file This will be needed to launch Service Workers, which cannot be generated at runtime and cannot use dynamic import(). * Add service worker support * Update wpt.ts * Run tests * Address kainino0x feedback * Address feedback | part 2 * Address feedback | part 3 * Address feedback | part 4 * Address feedback | part 5 * Address feedback | part 6 * Address feedback | part 7 * Address feedback | part 8 * Apply suggestions from code review * use WorkerTestRunRequest in the postMessage/onmessage interface * Clean up resolvers map * Use express routing for .worker.js * Skip worker tests when run in a worker that can't support them DedicatedWorker can be created from DedicatedWorker, but none of the other nested worker pairs are allowed. * Clean up all service workers on startup and shutdown * Avoid reinitializing service workers for every single case * lint fixes * Catch errors in wrapTestGroupForWorker onMessage * Make sure the service worker has the correct URL --------- Co-authored-by: Kai Ninomiya <[email protected]>
1 parent f9f6c90 commit 5ad7589

27 files changed

+512
-279
lines changed

Gruntfile.js

+5-5
Original file line numberDiff line numberDiff line change
@@ -24,9 +24,9 @@ module.exports = function (grunt) {
2424
cmd: 'node',
2525
args: ['tools/gen_version'],
2626
},
27-
'generate-listings': {
27+
'generate-listings-and-webworkers': {
2828
cmd: 'node',
29-
args: ['tools/gen_listings', 'gen/', ...kAllSuites.map(s => 'src/' + s)],
29+
args: ['tools/gen_listings_and_webworkers', 'gen/', ...kAllSuites.map(s => 'src/' + s)],
3030
},
3131
validate: {
3232
cmd: 'node',
@@ -159,14 +159,14 @@ module.exports = function (grunt) {
159159
// Must run after generate-common and run:build-out.
160160
files: [
161161
{ expand: true, dest: 'out/', cwd: 'gen', src: 'common/internal/version.js' },
162-
{ expand: true, dest: 'out/', cwd: 'gen', src: '*/listing.js' },
162+
{ expand: true, dest: 'out/', cwd: 'gen', src: '*/**/*.js' },
163163
],
164164
},
165165
'gen-to-out-wpt': {
166166
// Must run after generate-common and run:build-out-wpt.
167167
files: [
168168
{ expand: true, dest: 'out-wpt/', cwd: 'gen', src: 'common/internal/version.js' },
169-
{ expand: true, dest: 'out-wpt/', cwd: 'gen', src: 'webgpu/listing.js' },
169+
{ expand: true, dest: 'out-wpt/', cwd: 'gen', src: 'webgpu/**/*.js' },
170170
],
171171
},
172172
'htmlfiles-to-out': {
@@ -243,7 +243,7 @@ module.exports = function (grunt) {
243243

244244
grunt.registerTask('generate-common', 'Generate files into gen/ and src/', [
245245
'run:generate-version',
246-
'run:generate-listings',
246+
'run:generate-listings-and-webworkers',
247247
'run:generate-cache',
248248
]);
249249
grunt.registerTask('build-standalone', 'Build out/ (no checks; run after generate-common)', [

docs/intro/developing.md

+1
Original file line numberDiff line numberDiff line change
@@ -58,6 +58,7 @@ The following url parameters change how the harness runs:
5858
- `debug=1` enables verbose debug logging from tests.
5959
- `worker=dedicated` runs the tests on a dedicated worker instead of the main thread.
6060
- `worker=shared` runs the tests on a shared worker instead of the main thread.
61+
- `worker=service` runs the tests on a service worker instead of the main thread.
6162
- `power_preference=low-power` runs most tests passing `powerPreference: low-power` to `requestAdapter`
6263
- `power_preference=high-performance` runs most tests passing `powerPreference: high-performance` to `requestAdapter`
6364

docs/terms.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -111,7 +111,7 @@ Each Suite has one Listing File (`suite/listing.[tj]s`), containing a list of th
111111
in the suite.
112112

113113
In `src/suite/listing.ts`, this is computed dynamically.
114-
In `out/suite/listing.js`, the listing has been pre-baked (by `tools/gen_listings`).
114+
In `out/suite/listing.js`, the listing has been pre-baked (by `tools/gen_listings_and_webworkers`).
115115

116116
**Type:** Once `import`ed, `ListingFile`
117117

src/common/internal/query/compare.ts

+4-1
Original file line numberDiff line numberDiff line change
@@ -58,7 +58,10 @@ function compareOneLevel(ordering: Ordering, aIsBig: boolean, bIsBig: boolean):
5858
return Ordering.Unordered;
5959
}
6060

61-
function comparePaths(a: readonly string[], b: readonly string[]): Ordering {
61+
/**
62+
* Compare two file paths, or file-local test paths, returning an Ordering between the two.
63+
*/
64+
export function comparePaths(a: readonly string[], b: readonly string[]): Ordering {
6265
const shorter = Math.min(a.length, b.length);
6366

6467
for (let i = 0; i < shorter; ++i) {

src/common/internal/test_suite_listing.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
// A listing of all specs within a single suite. This is the (awaited) type of
22
// `groups` in '{cts,unittests}/listing.ts' and `listing` in the auto-generated
3-
// 'out/{cts,unittests}/listing.js' files (see tools/gen_listings).
3+
// 'out/{cts,unittests}/listing.js' files (see tools/gen_listings_and_webworkers).
44
export type TestSuiteListing = TestSuiteListingEntry[];
55

66
export type TestSuiteListingEntry = TestSuiteListingEntrySpec | TestSuiteListingEntryReadme;

src/common/runtime/helper/options.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,7 @@ export function optionString(
2525
* The possible options for the tests.
2626
*/
2727
export interface CTSOptions {
28-
worker?: 'dedicated' | 'shared' | '';
28+
worker?: 'dedicated' | 'shared' | 'service' | '';
2929
debug: boolean;
3030
compatibility: boolean;
3131
forceFallbackAdapter: boolean;
@@ -68,6 +68,7 @@ export const kCTSOptionsInfo: OptionsInfos<CTSOptions> = {
6868
{ value: '', description: 'no worker' },
6969
{ value: 'dedicated', description: 'dedicated worker' },
7070
{ value: 'shared', description: 'shared worker' },
71+
{ value: 'service', description: 'service worker' },
7172
],
7273
},
7374
debug: { description: 'show more info' },

src/common/runtime/helper/test_worker-worker.ts

+4-23
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,9 @@
11
import { setBaseResourcePath } from '../../framework/resources.js';
2-
import { globalTestConfig } from '../../framework/test_config.js';
32
import { DefaultTestFileLoader } from '../../internal/file_loader.js';
4-
import { Logger } from '../../internal/logging/logger.js';
53
import { parseQuery } from '../../internal/query/parseQuery.js';
6-
import { TestQueryWithExpectation } from '../../internal/query/query.js';
7-
import { setDefaultRequestAdapterOptions } from '../../util/navigator_gpu.js';
84
import { assert } from '../../util/util.js';
95

10-
import { CTSOptions } from './options.js';
6+
import { setupWorkerEnvironment, WorkerTestRunRequest } from './utils_worker.js';
117

128
// Should be WorkerGlobalScope, but importing lib "webworker" conflicts with lib "dom".
139
/* eslint-disable-next-line @typescript-eslint/no-explicit-any */
@@ -18,24 +14,9 @@ const loader = new DefaultTestFileLoader();
1814
setBaseResourcePath('../../../resources');
1915

2016
async function reportTestResults(this: MessagePort | Worker, ev: MessageEvent) {
21-
const query: string = ev.data.query;
22-
const expectations: TestQueryWithExpectation[] = ev.data.expectations;
23-
const ctsOptions: CTSOptions = ev.data.ctsOptions;
17+
const { query, expectations, ctsOptions } = ev.data as WorkerTestRunRequest;
2418

25-
const { debug, unrollConstEvalLoops, powerPreference, compatibility } = ctsOptions;
26-
globalTestConfig.unrollConstEvalLoops = unrollConstEvalLoops;
27-
globalTestConfig.compatibility = compatibility;
28-
29-
Logger.globalDebugMode = debug;
30-
const log = new Logger();
31-
32-
if (powerPreference || compatibility) {
33-
setDefaultRequestAdapterOptions({
34-
...(powerPreference && { powerPreference }),
35-
// MAINTENANCE_TODO: Change this to whatever the option ends up being
36-
...(compatibility && { compatibilityMode: true }),
37-
});
38-
}
19+
const log = setupWorkerEnvironment(ctsOptions);
3920

4021
const testcases = Array.from(await loader.loadCases(parseQuery(query)));
4122
assert(testcases.length === 1, 'worker query resulted in != 1 cases');
@@ -48,7 +29,7 @@ async function reportTestResults(this: MessagePort | Worker, ev: MessageEvent) {
4829
}
4930

5031
self.onmessage = (ev: MessageEvent) => {
51-
void reportTestResults.call(self, ev);
32+
void reportTestResults.call(ev.source || self, ev);
5233
};
5334

5435
self.onconnect = (event: MessageEvent) => {

src/common/runtime/helper/test_worker.ts

+102-50
Original file line numberDiff line numberDiff line change
@@ -2,95 +2,147 @@ import { LogMessageWithStack } from '../../internal/logging/log_message.js';
22
import { TransferredTestCaseResult, LiveTestCaseResult } from '../../internal/logging/result.js';
33
import { TestCaseRecorder } from '../../internal/logging/test_case_recorder.js';
44
import { TestQueryWithExpectation } from '../../internal/query/query.js';
5+
import { timeout } from '../../util/timeout.js';
6+
import { assert } from '../../util/util.js';
57

68
import { CTSOptions, kDefaultCTSOptions } from './options.js';
9+
import { WorkerTestRunRequest } from './utils_worker.js';
710

8-
export class TestDedicatedWorker {
9-
private readonly ctsOptions: CTSOptions;
10-
private readonly worker: Worker;
11-
private readonly resolvers = new Map<string, (result: LiveTestCaseResult) => void>();
11+
/** Query all currently-registered service workers, and unregister them. */
12+
function unregisterAllServiceWorkers() {
13+
void navigator.serviceWorker.getRegistrations().then(registrations => {
14+
for (const registration of registrations) {
15+
void registration.unregister();
16+
}
17+
});
18+
}
1219

13-
constructor(ctsOptions?: CTSOptions) {
14-
this.ctsOptions = { ...(ctsOptions || kDefaultCTSOptions), ...{ worker: 'dedicated' } };
15-
const selfPath = import.meta.url;
16-
const selfPathDir = selfPath.substring(0, selfPath.lastIndexOf('/'));
17-
const workerPath = selfPathDir + '/test_worker-worker.js';
18-
this.worker = new Worker(workerPath, { type: 'module' });
19-
this.worker.onmessage = ev => {
20-
const query: string = ev.data.query;
21-
const result: TransferredTestCaseResult = ev.data.result;
22-
if (result.logs) {
23-
for (const l of result.logs) {
24-
Object.setPrototypeOf(l, LogMessageWithStack.prototype);
25-
}
20+
// NOTE: This code runs on startup for any runtime with worker support. Here, we use that chance to
21+
// delete any leaked service workers, and register to clean up after ourselves at shutdown.
22+
unregisterAllServiceWorkers();
23+
window.addEventListener('beforeunload', () => {
24+
unregisterAllServiceWorkers();
25+
});
26+
27+
class TestBaseWorker {
28+
protected readonly ctsOptions: CTSOptions;
29+
protected readonly resolvers = new Map<string, (result: LiveTestCaseResult) => void>();
30+
31+
constructor(worker: CTSOptions['worker'], ctsOptions?: CTSOptions) {
32+
this.ctsOptions = { ...(ctsOptions || kDefaultCTSOptions), ...{ worker } };
33+
}
34+
35+
onmessage(ev: MessageEvent) {
36+
const query: string = ev.data.query;
37+
const result: TransferredTestCaseResult = ev.data.result;
38+
if (result.logs) {
39+
for (const l of result.logs) {
40+
Object.setPrototypeOf(l, LogMessageWithStack.prototype);
2641
}
27-
this.resolvers.get(query)!(result as LiveTestCaseResult);
42+
}
43+
this.resolvers.get(query)!(result as LiveTestCaseResult);
44+
this.resolvers.delete(query);
2845

29-
// MAINTENANCE_TODO(kainino0x): update the Logger with this result (or don't have a logger and
30-
// update the entire results JSON somehow at some point).
31-
};
46+
// MAINTENANCE_TODO(kainino0x): update the Logger with this result (or don't have a logger and
47+
// update the entire results JSON somehow at some point).
3248
}
3349

34-
async run(
50+
async makeRequestAndRecordResult(
51+
target: MessagePort | Worker | ServiceWorker,
3552
rec: TestCaseRecorder,
3653
query: string,
37-
expectations: TestQueryWithExpectation[] = []
38-
): Promise<void> {
39-
this.worker.postMessage({
54+
expectations: TestQueryWithExpectation[]
55+
) {
56+
const request: WorkerTestRunRequest = {
4057
query,
4158
expectations,
4259
ctsOptions: this.ctsOptions,
43-
});
60+
};
61+
target.postMessage(request);
62+
4463
const workerResult = await new Promise<LiveTestCaseResult>(resolve => {
64+
assert(!this.resolvers.has(query), "can't request same query twice simultaneously");
4565
this.resolvers.set(query, resolve);
4666
});
4767
rec.injectResult(workerResult);
4868
}
4969
}
5070

71+
export class TestDedicatedWorker extends TestBaseWorker {
72+
private readonly worker: Worker;
73+
74+
constructor(ctsOptions?: CTSOptions) {
75+
super('dedicated', ctsOptions);
76+
const selfPath = import.meta.url;
77+
const selfPathDir = selfPath.substring(0, selfPath.lastIndexOf('/'));
78+
const workerPath = selfPathDir + '/test_worker-worker.js';
79+
this.worker = new Worker(workerPath, { type: 'module' });
80+
this.worker.onmessage = ev => this.onmessage(ev);
81+
}
82+
83+
async run(
84+
rec: TestCaseRecorder,
85+
query: string,
86+
expectations: TestQueryWithExpectation[] = []
87+
): Promise<void> {
88+
await this.makeRequestAndRecordResult(this.worker, rec, query, expectations);
89+
}
90+
}
91+
5192
export class TestWorker extends TestDedicatedWorker {}
5293

53-
export class TestSharedWorker {
54-
private readonly ctsOptions: CTSOptions;
94+
export class TestSharedWorker extends TestBaseWorker {
5595
private readonly port: MessagePort;
56-
private readonly resolvers = new Map<string, (result: LiveTestCaseResult) => void>();
5796

5897
constructor(ctsOptions?: CTSOptions) {
59-
this.ctsOptions = { ...(ctsOptions || kDefaultCTSOptions), ...{ worker: 'shared' } };
98+
super('shared', ctsOptions);
6099
const selfPath = import.meta.url;
61100
const selfPathDir = selfPath.substring(0, selfPath.lastIndexOf('/'));
62101
const workerPath = selfPathDir + '/test_worker-worker.js';
63102
const worker = new SharedWorker(workerPath, { type: 'module' });
64103
this.port = worker.port;
65104
this.port.start();
66-
this.port.onmessage = ev => {
67-
const query: string = ev.data.query;
68-
const result: TransferredTestCaseResult = ev.data.result;
69-
if (result.logs) {
70-
for (const l of result.logs) {
71-
Object.setPrototypeOf(l, LogMessageWithStack.prototype);
72-
}
73-
}
74-
this.resolvers.get(query)!(result as LiveTestCaseResult);
105+
this.port.onmessage = ev => this.onmessage(ev);
106+
}
75107

76-
// MAINTENANCE_TODO(kainino0x): update the Logger with this result (or don't have a logger and
77-
// update the entire results JSON somehow at some point).
78-
};
108+
async run(
109+
rec: TestCaseRecorder,
110+
query: string,
111+
expectations: TestQueryWithExpectation[] = []
112+
): Promise<void> {
113+
await this.makeRequestAndRecordResult(this.port, rec, query, expectations);
114+
}
115+
}
116+
117+
export class TestServiceWorker extends TestBaseWorker {
118+
constructor(ctsOptions?: CTSOptions) {
119+
super('service', ctsOptions);
79120
}
80121

81122
async run(
82123
rec: TestCaseRecorder,
83124
query: string,
84125
expectations: TestQueryWithExpectation[] = []
85126
): Promise<void> {
86-
this.port.postMessage({
87-
query,
88-
expectations,
89-
ctsOptions: this.ctsOptions,
90-
});
91-
const workerResult = await new Promise<LiveTestCaseResult>(resolve => {
92-
this.resolvers.set(query, resolve);
127+
const [suite, name] = query.split(':', 2);
128+
const fileName = name.split(',').join('/');
129+
const serviceWorkerURL = new URL(
130+
`/out/${suite}/webworker/${fileName}.worker.js`,
131+
window.location.href
132+
).toString();
133+
134+
// If a registration already exists for this path, it will be ignored.
135+
const registration = await navigator.serviceWorker.register(serviceWorkerURL, {
136+
type: 'module',
93137
});
94-
rec.injectResult(workerResult);
138+
// Make sure the registration we just requested is active. (We don't worry about it being
139+
// outdated from a previous page load, because we wipe all service workers on shutdown/startup.)
140+
while (!registration.active || registration.active.scriptURL !== serviceWorkerURL) {
141+
await new Promise(resolve => timeout(resolve, 0));
142+
}
143+
const serviceWorker = registration.active;
144+
145+
navigator.serviceWorker.onmessage = ev => this.onmessage(ev);
146+
await this.makeRequestAndRecordResult(serviceWorker, rec, query, expectations);
95147
}
96148
}
+34
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
import { globalTestConfig } from '../../framework/test_config.js';
2+
import { Logger } from '../../internal/logging/logger.js';
3+
import { TestQueryWithExpectation } from '../../internal/query/query.js';
4+
import { setDefaultRequestAdapterOptions } from '../../util/navigator_gpu.js';
5+
6+
import { CTSOptions } from './options.js';
7+
8+
export interface WorkerTestRunRequest {
9+
query: string;
10+
expectations: TestQueryWithExpectation[];
11+
ctsOptions: CTSOptions;
12+
}
13+
14+
/**
15+
* Set config environment for workers with ctsOptions and return a Logger.
16+
*/
17+
export function setupWorkerEnvironment(ctsOptions: CTSOptions): Logger {
18+
const { debug, unrollConstEvalLoops, powerPreference, compatibility } = ctsOptions;
19+
globalTestConfig.unrollConstEvalLoops = unrollConstEvalLoops;
20+
globalTestConfig.compatibility = compatibility;
21+
22+
Logger.globalDebugMode = debug;
23+
const log = new Logger();
24+
25+
if (powerPreference || compatibility) {
26+
setDefaultRequestAdapterOptions({
27+
...(powerPreference && { powerPreference }),
28+
// MAINTENANCE_TODO: Change this to whatever the option ends up being
29+
...(compatibility && { compatibilityMode: true }),
30+
});
31+
}
32+
33+
return log;
34+
}

0 commit comments

Comments
 (0)