test(service-worker): better align mock event implementations with actual implementations (#42736)

This commit better aligns the mock event implementations used in
ServiceWorker tests (and the associated typings) with the actual
implementations (and the official TypeScript typings). This allows
verifying the ServiceWorker behavior in a slightly more realistic
environment.

This is in preparation of switching from our custom typings to the
official TypeScript typings (`lib.webworker.d.ts`).

PR Close #42736
This commit is contained in:
George Kalpakas 2021-07-08 16:25:15 +03:00 committed by atscott
parent a86c404f14
commit 22a81231f2
4 changed files with 41 additions and 36 deletions

View File

@ -64,29 +64,24 @@ type WindowClientState = 'hidden'|'visible'|'prerender'|'unloaded';
// Fetch API // Fetch API
interface FetchEvent extends ExtendableEvent { interface FetchEvent extends ExtendableEvent {
request: Request; readonly clientId: string;
clientId?: string; readonly preloadResponse: Promise<any>;
resultingClientId?: string; readonly request: Request;
respondWith(response: Promise<Response>|Response): Promise<Response>; readonly resultingClientId: string;
respondWith(r: Response|Promise<Response>): void;
} }
interface InstallEvent extends ExtendableEvent {
activeWorker: ServiceWorker;
}
interface ActivateEvent extends ExtendableEvent {}
// Notification API // Notification API
interface NotificationEvent extends ExtendableEvent { interface NotificationEvent extends ExtendableEvent {
action: string; readonly action: string;
notification: Notification; readonly notification: Notification;
} }
// Push API // Push API
interface PushEvent extends ExtendableEvent { interface PushEvent extends ExtendableEvent {
data: PushMessageData; readonly data: PushMessageData|null;
} }
interface PushMessageData { interface PushMessageData {
@ -99,13 +94,16 @@ interface PushMessageData {
// Sync API // Sync API
interface SyncEvent extends ExtendableEvent { interface SyncEvent extends ExtendableEvent {
lastChance: boolean; readonly lastChance: boolean;
tag: string; readonly tag: string;
} }
interface ExtendableMessageEvent extends ExtendableEvent { interface ExtendableMessageEvent extends ExtendableEvent {
data: any; readonly data: any;
source: Client|Object; readonly lastEventId: string;
readonly origin: string;
readonly ports: ReadonlyArray<MessagePort>;
readonly source: Client|ServiceWorker|MessagePort|null;
} }
// ServiceWorkerGlobalScope // ServiceWorkerGlobalScope

View File

@ -783,7 +783,7 @@ describe('Driver', () => {
const message: any = scope.clients.getMock('default')!.messages[0]; const message: any = scope.clients.getMock('default')!.messages[0];
expect(message.type).toEqual('NOTIFICATION_CLICK'); expect(message.type).toEqual('NOTIFICATION_CLICK');
expect(message.data.action).toBeUndefined(); expect(message.data.action).toBe('');
expect(message.data.notification.title).toEqual('This is a test without action'); expect(message.data.notification.title).toEqual('This is a test without action');
expect(message.data.notification.body).toEqual('Test body without action'); expect(message.data.notification.body).toEqual('Test body without action');
}); });

View File

@ -49,7 +49,7 @@ export class MockEvent implements Event {
} }
export class MockExtendableEvent extends MockEvent implements ExtendableEvent { export class MockExtendableEvent extends MockEvent implements ExtendableEvent {
private queue: Promise<void>[] = []; private queue: Promise<unknown>[] = [];
get ready(): Promise<void> { get ready(): Promise<void> {
return (async () => { return (async () => {
@ -59,7 +59,7 @@ export class MockExtendableEvent extends MockEvent implements ExtendableEvent {
})(); })();
} }
waitUntil(promise: Promise<void>): void { waitUntil(promise: Promise<unknown>): void {
this.queue.push(promise); this.queue.push(promise);
} }
} }
@ -70,7 +70,8 @@ export class MockActivateEvent extends MockExtendableEvent {
} }
} }
export class MockFetchEvent extends MockExtendableEvent { export class MockFetchEvent extends MockExtendableEvent implements FetchEvent {
readonly preloadResponse = Promise.resolve();
response: Promise<Response|undefined> = Promise.resolve(undefined); response: Promise<Response|undefined> = Promise.resolve(undefined);
constructor( constructor(
@ -78,9 +79,8 @@ export class MockFetchEvent extends MockExtendableEvent {
super('fetch'); super('fetch');
} }
respondWith(promise: Promise<Response>): Promise<Response> { respondWith(r: Response|Promise<Response>): void {
this.response = promise; this.response = Promise.resolve(r);
return promise;
} }
} }
@ -90,27 +90,33 @@ export class MockInstallEvent extends MockExtendableEvent {
} }
} }
export class MockMessageEvent extends MockExtendableEvent { export class MockExtendableMessageEvent extends MockExtendableEvent implements
constructor(readonly data: Object, readonly source: MockClient|null) { ExtendableMessageEvent {
readonly lastEventId = '';
readonly origin = '';
readonly ports: ReadonlyArray<MessagePort> = [];
constructor(readonly data: any, readonly source: Client|MessagePort|ServiceWorker|null) {
super('message'); super('message');
} }
} }
export class MockNotificationEvent extends MockExtendableEvent { export class MockNotificationEvent extends MockExtendableEvent implements NotificationEvent {
readonly notification = { readonly notification = {
...this._notification, ...this._notification,
close: () => undefined, close: () => undefined,
}; } as Notification;
constructor(private _notification: any, readonly action?: string) { constructor(private _notification: Partial<Notification>, readonly action = '') {
super('notification'); super('notification');
} }
} }
export class MockPushEvent extends MockExtendableEvent { export class MockPushEvent extends MockExtendableEvent implements PushEvent {
data = { readonly data = {
json: () => this._data, json: () => this._data,
}; text: () => JSON.stringify(this._data),
} as PushMessageData;
constructor(private _data: object) { constructor(private _data: object) {
super('push'); super('push');

View File

@ -13,7 +13,7 @@ import {AssetGroupConfig, Manifest} from '../src/manifest';
import {sha1} from '../src/sha1'; import {sha1} from '../src/sha1';
import {MockCacheStorage} from './cache'; import {MockCacheStorage} from './cache';
import {MockActivateEvent, MockFetchEvent, MockInstallEvent, MockMessageEvent, MockNotificationEvent, MockPushEvent} from './events'; import {MockActivateEvent, MockExtendableMessageEvent, MockFetchEvent, MockInstallEvent, MockNotificationEvent, MockPushEvent} from './events';
import {MockHeaders, MockRequest, MockResponse} from './fetch'; import {MockHeaders, MockRequest, MockResponse} from './fetch';
import {MockServerState, MockServerStateBuilder} from './mock'; import {MockServerState, MockServerStateBuilder} from './mock';
import {normalizeUrl, parseUrl} from './utils'; import {normalizeUrl, parseUrl} from './utils';
@ -244,12 +244,13 @@ export class SwTestHarness extends Adapter<MockCacheStorage> implements ServiceW
if (!this.eventHandlers.has('message')) { if (!this.eventHandlers.has('message')) {
throw new Error('No message handler registered'); throw new Error('No message handler registered');
} }
let event: MockMessageEvent; let event: MockExtendableMessageEvent;
if (clientId === null) { if (clientId === null) {
event = new MockMessageEvent(data, null); event = new MockExtendableMessageEvent(data, null);
} else { } else {
this.clients.add(clientId); this.clients.add(clientId);
event = new MockMessageEvent(data, this.clients.getMock(clientId) || null); event = new MockExtendableMessageEvent(
data, this.clients.getMock(clientId) as unknown as Client || null);
} }
this.eventHandlers.get('message')!.call(this, event); this.eventHandlers.get('message')!.call(this, event);
return event.ready; return event.ready;