fix(docs-infra): support recovering from unrecoverable SW states (#39651)

Occasionally, the SW would end up in a broken state where some of the
eagerly cached resources of an older version were available in the local
cache, but others (such as lazy-loaded bundles) were not. This would
leave the app in a broken state and a blank screen would be displayed.
See #28114 for a more detailed discussion.

This commit takes advantage of the newly introduced (in v11)
[SwUpdate#unrecoverable][1] API to detect these bad states and recover
by doing a full page reload whenever an [UnrecoverableStateEvent][2] is
emitted.

Partially addresses #28114.

NOTE:
Currently, `SwUpdate.unrecoverable` only works if the app has already
bootstrapped; i.e. if only lazy-loaded bundles have been purged from the
cache.
That should be fine in practice, since the cache entries are removed in
least-recently-used order. Thus the eagerly loaded bundles will be the
last to be removed from the cache (which rarely happens in practice).

[1]: https://v11.angular.io/api/service-worker/SwUpdate#unrecoverable
[2]: https://v11.angular.io/api/service-worker/UnrecoverableStateEvent

PR Close #39651
This commit is contained in:
George Kalpakas 2020-11-12 17:43:51 +02:00 committed by Andrew Kushnir
parent 305d05545a
commit 935cf433ed
5 changed files with 45 additions and 2 deletions

View File

@ -70,6 +70,10 @@ export class LocationService {
window.location.replace(url);
}
reloadPage(): void {
window.location.reload();
}
private stripSlashes(url: string) {
return url.replace(/^\/+/, '').replace(/\/+(\?|#|$)/, '$1');
}

View File

@ -119,6 +119,16 @@ describe('SwUpdatesService', () => {
expect(location.fullPageNavigationNeeded).toHaveBeenCalledTimes(2);
}));
it('should request a page reload when an unrecoverable state has been detected', run(() => {
expect(location.reloadPage).toHaveBeenCalledTimes(0);
swu.$$unrecoverableSubj.next({reason: 'Something bad happened'});
expect(location.reloadPage).toHaveBeenCalledTimes(1);
swu.$$unrecoverableSubj.next({reason: 'Something worse happened'});
expect(location.reloadPage).toHaveBeenCalledTimes(2);
}));
describe('when `SwUpdate` is not enabled', () => {
const runDeactivated = (specFn: VoidFunction) => run(specFn, false);
@ -150,6 +160,13 @@ describe('SwUpdatesService', () => {
expect(location.fullPageNavigationNeeded).not.toHaveBeenCalled();
}));
it('should never request a page reload', runDeactivated(() => {
swu.$$unrecoverableSubj.next({reason: 'Something bad happened'});
swu.$$unrecoverableSubj.next({reason: 'Something worse happened'});
expect(location.reloadPage).not.toHaveBeenCalled();
}));
});
describe('when destroyed', () => {
@ -201,6 +218,17 @@ describe('SwUpdatesService', () => {
swu.$$activatedSubj.next({current: {hash: 'qux'}});
expect(location.fullPageNavigationNeeded).not.toHaveBeenCalled();
}));
it('should stop requesting page reloads when unrecoverable states are detected', run(() => {
swu.$$unrecoverableSubj.next({reason: 'Something bad happened'});
expect(location.reloadPage).toHaveBeenCalledTimes(1);
service.ngOnDestroy();
location.reloadPage.calls.reset();
swu.$$unrecoverableSubj.next({reason: 'Something worse happened'});
expect(location.reloadPage).not.toHaveBeenCalled();
}));
});
});
@ -212,9 +240,11 @@ class MockApplicationRef {
class MockSwUpdate {
$$availableSubj = new Subject<{available: {hash: string}}>();
$$activatedSubj = new Subject<{current: {hash: string}}>();
$$unrecoverableSubj = new Subject<{reason: string}>();
available = this.$$availableSubj.asObservable();
activated = this.$$activatedSubj.asObservable();
unrecoverable = this.$$unrecoverableSubj.asObservable();
activateUpdate = jasmine.createSpy('MockSwUpdate.activateUpdate')
.and.callFake(() => Promise.resolve());

View File

@ -51,6 +51,14 @@ export class SwUpdatesService implements OnDestroy {
takeUntil(this.onDestroy),
)
.subscribe(() => location.fullPageNavigationNeeded());
// Request an immediate page reload once an unrecoverable state has been detected.
this.swu.unrecoverable
.pipe(
tap(evt => this.log(`Unrecoverable state: ${evt.reason}\nReloading...`)),
takeUntil(this.onDestroy),
)
.subscribe(() => location.reloadPage());
}
ngOnDestroy() {

View File

@ -13,6 +13,7 @@ export class MockLocationService {
.callFake((url: string) => this.urlSubject.next(url));
goExternal = jasmine.createSpy('Location.goExternal');
replace = jasmine.createSpy('Location.replace');
reloadPage = jasmine.createSpy('Location.reloadPage');
handleAnchorClick = jasmine.createSpy('Location.handleAnchorClick')
.and.returnValue(false); // prevent click from causing a browser navigation

View File

@ -3,7 +3,7 @@
"master": {
"uncompressed": {
"runtime-es2015": 3033,
"main-es2015": 447565,
"main-es2015": 447766,
"polyfills-es2015": 52343
}
}
@ -12,7 +12,7 @@
"master": {
"uncompressed": {
"runtime-es2015": 3033,
"main-es2015": 447774,
"main-es2015": 447975,
"polyfills-es2015": 52493
}
}