106 lines
4.3 KiB
TypeScript
106 lines
4.3 KiB
TypeScript
/**
|
|
* @license
|
|
* Copyright Google LLC All Rights Reserved.
|
|
*
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
* found in the LICENSE file at https://angular.io/license
|
|
*/
|
|
|
|
import {PrefetchAssetGroup} from '../src/assets';
|
|
import {CacheDatabase} from '../src/db-cache';
|
|
import {IdleScheduler} from '../src/idle';
|
|
import {MockCache} from '../testing/cache';
|
|
import {MockExtendableEvent} from '../testing/events';
|
|
import {MockRequest} from '../testing/fetch';
|
|
import {MockFileSystemBuilder, MockServerStateBuilder, tmpHashTable, tmpManifestSingleAssetGroup} from '../testing/mock';
|
|
import {SwTestHarnessBuilder} from '../testing/scope';
|
|
import {envIsSupported} from '../testing/utils';
|
|
|
|
(function() {
|
|
// Skip environments that don't support the minimum APIs needed to run the SW tests.
|
|
if (!envIsSupported()) {
|
|
return;
|
|
}
|
|
|
|
const dist = new MockFileSystemBuilder()
|
|
.addFile('/foo.txt', 'this is foo', {Vary: 'Accept'})
|
|
.addFile('/bar.txt', 'this is bar')
|
|
.build();
|
|
|
|
const manifest = tmpManifestSingleAssetGroup(dist);
|
|
|
|
const server = new MockServerStateBuilder().withStaticFiles(dist).withManifest(manifest).build();
|
|
|
|
const scope = new SwTestHarnessBuilder().withServerState(server).build();
|
|
|
|
const db = new CacheDatabase(scope);
|
|
|
|
const testEvent = new MockExtendableEvent('test');
|
|
|
|
|
|
describe('prefetch assets', () => {
|
|
let group: PrefetchAssetGroup;
|
|
let idle: IdleScheduler;
|
|
beforeEach(() => {
|
|
idle = new IdleScheduler(null!, 3000, 30000, {
|
|
log: (v, ctx = '') => console.error(v, ctx),
|
|
});
|
|
group = new PrefetchAssetGroup(
|
|
scope, scope, idle, manifest.assetGroups![0], tmpHashTable(manifest), db, 'test');
|
|
});
|
|
it('initializes without crashing', async () => {
|
|
await group.initializeFully();
|
|
});
|
|
it('fully caches the two files', async () => {
|
|
await group.initializeFully();
|
|
scope.updateServerState();
|
|
const res1 = await group.handleFetch(scope.newRequest('/foo.txt'), testEvent);
|
|
const res2 = await group.handleFetch(scope.newRequest('/bar.txt'), testEvent);
|
|
expect(await res1!.text()).toEqual('this is foo');
|
|
expect(await res2!.text()).toEqual('this is bar');
|
|
});
|
|
it('persists the cache across restarts', async () => {
|
|
await group.initializeFully();
|
|
const freshScope =
|
|
new SwTestHarnessBuilder().withCacheState(scope.caches.original.dehydrate()).build();
|
|
group = new PrefetchAssetGroup(
|
|
freshScope, freshScope, idle, manifest.assetGroups![0], tmpHashTable(manifest),
|
|
new CacheDatabase(freshScope), 'test');
|
|
await group.initializeFully();
|
|
const res1 = await group.handleFetch(scope.newRequest('/foo.txt'), testEvent);
|
|
const res2 = await group.handleFetch(scope.newRequest('/bar.txt'), testEvent);
|
|
expect(await res1!.text()).toEqual('this is foo');
|
|
expect(await res2!.text()).toEqual('this is bar');
|
|
});
|
|
it('caches properly if resources are requested before initialization', async () => {
|
|
const res1 = await group.handleFetch(scope.newRequest('/foo.txt'), testEvent);
|
|
const res2 = await group.handleFetch(scope.newRequest('/bar.txt'), testEvent);
|
|
expect(await res1!.text()).toEqual('this is foo');
|
|
expect(await res2!.text()).toEqual('this is bar');
|
|
scope.updateServerState();
|
|
await group.initializeFully();
|
|
});
|
|
it('throws if the server-side content does not match the manifest hash', async () => {
|
|
const badHashFs = dist.extend().addFile('/foo.txt', 'corrupted file').build();
|
|
const badServer =
|
|
new MockServerStateBuilder().withManifest(manifest).withStaticFiles(badHashFs).build();
|
|
const badScope = new SwTestHarnessBuilder().withServerState(badServer).build();
|
|
group = new PrefetchAssetGroup(
|
|
badScope, badScope, idle, manifest.assetGroups![0], tmpHashTable(manifest),
|
|
new CacheDatabase(badScope), 'test');
|
|
const err = await errorFrom(group.initializeFully());
|
|
expect(err.message).toContain('Hash mismatch');
|
|
});
|
|
it('CacheQueryOptions are passed through', async () => {
|
|
await group.initializeFully();
|
|
const matchSpy = spyOn(MockCache.prototype, 'match').and.callThrough();
|
|
await group.handleFetch(scope.newRequest('/foo.txt'), testEvent);
|
|
expect(matchSpy).toHaveBeenCalledWith(new MockRequest('/foo.txt'), {ignoreVary: true});
|
|
});
|
|
});
|
|
})();
|
|
|
|
function errorFrom(promise: Promise<any>): Promise<any> {
|
|
return promise.catch(err => err);
|
|
}
|