mirror of
https://github.com/angular/angular
synced 2026-05-24 09:28:37 +00:00
131 lines
4.6 KiB
TypeScript
131 lines
4.6 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.dev/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);
|
|
}
|