angular/packages/service-worker/worker/test/prefetch_spec.ts
Joey Perrott 9dbe6fc18b refactor: update license text to point to angular.dev (#57901)
Update license text to point to angular.dev instead of angular.io

PR Close #57901
2024-09-24 15:33:00 +02:00

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);
}