fix(service-worker): ensure initialization before handling messages (#32525)

- resolves "Invariant violated (initialize): latest hash null has no known manifest"
- Thanks to @gkalpak and @hsta for helping test and investigate this fix

Fixes #25611

PR Close #32525
This commit is contained in:
Michael Prentice 2019-09-06 23:57:21 -04:00 committed by atscott
parent 083d48e072
commit 72eba7745f
2 changed files with 85 additions and 46 deletions

View File

@ -250,34 +250,23 @@ export class Driver implements Debuggable, UpdateSource {
return; return;
} }
// Initialization is the only event which is sent directly from the SW to itself, event.waitUntil((async() => {
// and thus `event.source` is not a Client. Handle it here, before the check // Initialization is the only event which is sent directly from the SW to itself, and thus
// for Client sources. // `event.source` is not a `Client`. Handle it here, before the check for `Client` sources.
if (data.action === 'INITIALIZE') { if (data.action === 'INITIALIZE') {
// Only initialize if not already initialized (or initializing). return this.ensureInitialized(event);
if (this.initialized === null) {
// Initialize the SW.
this.initialized = this.initialize();
// Wait until initialization is properly scheduled, then trigger idle
// events to allow it to complete (assuming the SW is idle).
event.waitUntil((async() => {
await this.initialized;
await this.idle.trigger();
})());
} }
return; // Only messages from true clients are accepted past this point.
} // This is essentially a typecast.
if (!this.adapter.isClient(event.source)) {
return;
}
// Only messages from true clients are accepted past this point (this is essentially // Handle the message and keep the SW alive until it's handled.
// a typecast). await this.ensureInitialized(event);
if (!this.adapter.isClient(event.source)) { await this.handleMessage(data, event.source);
return; })());
}
// Handle the message and keep the SW alive until it's handled.
event.waitUntil(this.handleMessage(data, event.source));
} }
private onPush(msg: PushEvent): void { private onPush(msg: PushEvent): void {
@ -295,6 +284,32 @@ export class Driver implements Debuggable, UpdateSource {
event.waitUntil(this.handleClick(event.notification, event.action)); event.waitUntil(this.handleClick(event.notification, event.action));
} }
private async ensureInitialized(event: ExtendableEvent): Promise<void> {
// Since the SW may have just been started, it may or may not have been initialized already.
// `this.initialized` will be `null` if initialization has not yet been attempted, or will be a
// `Promise` which will resolve (successfully or unsuccessfully) if it has.
if (this.initialized !== null) {
return this.initialized;
}
// Initialization has not yet been attempted, so attempt it. This should only ever happen once
// per SW instantiation.
try {
this.initialized = this.initialize();
await this.initialized;
} catch (error) {
// If initialization fails, the SW needs to enter a safe state, where it declines to respond
// to network requests.
this.state = DriverReadyState.SAFE_MODE;
this.stateMessage = `Initialization failed due to error: ${errorToString(error)}`;
throw error;
} finally {
// Regardless if initialization succeeded, background tasks still need to happen.
event.waitUntil(this.idle.trigger());
}
}
private async handleMessage(msg: MsgAny&{action: string}, from: Client): Promise<void> { private async handleMessage(msg: MsgAny&{action: string}, from: Client): Promise<void> {
if (isMsgCheckForUpdates(msg)) { if (isMsgCheckForUpdates(msg)) {
const action = (async() => { await this.checkForUpdate(); })(); const action = (async() => { await this.checkForUpdate(); })();
@ -383,28 +398,10 @@ export class Driver implements Debuggable, UpdateSource {
} }
private async handleFetch(event: FetchEvent): Promise<Response> { private async handleFetch(event: FetchEvent): Promise<Response> {
// Since the SW may have just been started, it may or may not have been initialized already.
// this.initialized will be `null` if initialization has not yet been attempted, or will be a
// Promise which will resolve (successfully or unsuccessfully) if it has.
if (this.initialized === null) {
// Initialization has not yet been attempted, so attempt it. This should only ever happen once
// per SW instantiation.
this.initialized = this.initialize();
}
// If initialization fails, the SW needs to enter a safe state, where it declines to respond to
// network requests.
try { try {
// Wait for initialization. // Ensure the SW instance has been initialized.
await this.initialized; await this.ensureInitialized(event);
} catch (e) { } catch {
// Initialization failed. Enter a safe state.
this.state = DriverReadyState.SAFE_MODE;
this.stateMessage = `Initialization failed due to error: ${errorToString(e)}`;
// Even though the driver entered safe mode, background tasks still need to happen.
event.waitUntil(this.idle.trigger());
// Since the SW is already committed to responding to the currently active request, // Since the SW is already committed to responding to the currently active request,
// respond with a network fetch. // respond with a network fetch.
return this.safeFetch(event.request); return this.safeFetch(event.request);

View File

@ -366,6 +366,48 @@ import {SwTestHarness, SwTestHarnessBuilder} from '../testing/scope';
server.assertNoOtherRequests(); server.assertNoOtherRequests();
}); });
it('initializes the service worker on fetch if it has not yet been initialized', async() => {
// Driver is initially uninitialized.
expect(driver.initialized).toBeNull();
expect(driver['latestHash']).toBeNull();
// Making a request initializes the driver (fetches assets).
expect(await makeRequest(scope, '/foo.txt')).toEqual('this is foo');
expect(driver['latestHash']).toEqual(jasmine.any(String));
server.assertSawRequestFor('ngsw.json');
server.assertSawRequestFor('/foo.txt');
server.assertSawRequestFor('/bar.txt');
server.assertSawRequestFor('/redirected.txt');
// Once initialized, cached resources are served without network requests.
expect(await makeRequest(scope, '/foo.txt')).toEqual('this is foo');
expect(await makeRequest(scope, '/bar.txt')).toEqual('this is bar');
server.assertNoOtherRequests();
});
it('initializes the service worker on message if it has not yet been initialized', async() => {
// Driver is initially uninitialized.
expect(driver.initialized).toBeNull();
expect(driver['latestHash']).toBeNull();
// Pushing a message initializes the driver (fetches assets).
await scope.handleMessage({action: 'foo'}, 'someClient');
expect(driver['latestHash']).toEqual(jasmine.any(String));
server.assertSawRequestFor('ngsw.json');
server.assertSawRequestFor('/foo.txt');
server.assertSawRequestFor('/bar.txt');
server.assertSawRequestFor('/redirected.txt');
// Once initialized, pushed messages are handled without re-initializing.
await scope.handleMessage({action: 'bar'}, 'someClient');
server.assertNoOtherRequests();
// Once initialized, cached resources are served without network requests.
expect(await makeRequest(scope, '/foo.txt')).toEqual('this is foo');
expect(await makeRequest(scope, '/bar.txt')).toEqual('this is bar');
server.assertNoOtherRequests();
});
it('handles non-relative URLs', async() => { it('handles non-relative URLs', async() => {
expect(await makeRequest(scope, '/foo.txt')).toEqual('this is foo'); expect(await makeRequest(scope, '/foo.txt')).toEqual('this is foo');
await driver.initialized; await driver.initialized;