| 
									
										
										
										
											2017-09-28 16:18:12 -07:00
										 |  |  | /** | 
					
						
							|  |  |  |  * @license | 
					
						
							| 
									
										
										
										
											2020-05-19 12:08:49 -07:00
										 |  |  |  * Copyright Google LLC All Rights Reserved. | 
					
						
							| 
									
										
										
										
											2017-09-28 16:18:12 -07:00
										 |  |  |  * | 
					
						
							|  |  |  |  * 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 {CacheDatabase} from '../src/db-cache'; | 
					
						
							|  |  |  | import {Driver} from '../src/driver'; | 
					
						
							|  |  |  | import {Manifest} from '../src/manifest'; | 
					
						
							| 
									
										
										
										
											2020-04-29 16:07:34 +02:00
										 |  |  | import {MockCache} from '../testing/cache'; | 
					
						
							| 
									
										
										
										
											2017-09-28 16:18:12 -07:00
										 |  |  | import {MockRequest} from '../testing/fetch'; | 
					
						
							|  |  |  | import {MockFileSystemBuilder, MockServerStateBuilder, tmpHashTableForFs} from '../testing/mock'; | 
					
						
							|  |  |  | import {SwTestHarness, SwTestHarnessBuilder} from '../testing/scope'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-20 23:29:14 +02:00
										 |  |  | (function() { | 
					
						
							| 
									
										
										
										
											2020-04-13 16:40:21 -07:00
										 |  |  | // Skip environments that don't support the minimum APIs needed to run the SW tests.
 | 
					
						
							|  |  |  | if (!SwTestHarness.envIsSupported()) { | 
					
						
							|  |  |  |   return; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const dist = new MockFileSystemBuilder() | 
					
						
							|  |  |  |                  .addFile('/foo.txt', 'this is foo') | 
					
						
							|  |  |  |                  .addFile('/bar.txt', 'this is bar') | 
					
						
							|  |  |  |                  .addFile('/api/test', 'version 1') | 
					
						
							|  |  |  |                  .addFile('/api/a', 'version A') | 
					
						
							|  |  |  |                  .addFile('/api/b', 'version B') | 
					
						
							|  |  |  |                  .addFile('/api/c', 'version C') | 
					
						
							|  |  |  |                  .addFile('/api/d', 'version D') | 
					
						
							|  |  |  |                  .addFile('/api/e', 'version E') | 
					
						
							|  |  |  |                  .addFile('/fresh/data', 'this is fresh data') | 
					
						
							|  |  |  |                  .addFile('/refresh/data', 'this is some data') | 
					
						
							|  |  |  |                  .build(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const distUpdate = new MockFileSystemBuilder() | 
					
						
							|  |  |  |                        .addFile('/foo.txt', 'this is foo v2') | 
					
						
							|  |  |  |                        .addFile('/bar.txt', 'this is bar') | 
					
						
							|  |  |  |                        .addFile('/api/test', 'version 2') | 
					
						
							|  |  |  |                        .addFile('/fresh/data', 'this is fresher data') | 
					
						
							|  |  |  |                        .addFile('/refresh/data', 'this is refreshed data') | 
					
						
							|  |  |  |                        .build(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const manifest: Manifest = { | 
					
						
							|  |  |  |   configVersion: 1, | 
					
						
							|  |  |  |   timestamp: 1234567890123, | 
					
						
							|  |  |  |   index: '/index.html', | 
					
						
							|  |  |  |   assetGroups: [ | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       name: 'assets', | 
					
						
							|  |  |  |       installMode: 'prefetch', | 
					
						
							|  |  |  |       updateMode: 'prefetch', | 
					
						
							|  |  |  |       urls: [ | 
					
						
							|  |  |  |         '/foo.txt', | 
					
						
							|  |  |  |         '/bar.txt', | 
					
						
							|  |  |  |       ], | 
					
						
							|  |  |  |       patterns: [], | 
					
						
							| 
									
										
										
										
											2020-04-29 16:59:15 +02:00
										 |  |  |       cacheQueryOptions: {ignoreVary: true}, | 
					
						
							| 
									
										
										
										
											2020-04-13 16:40:21 -07:00
										 |  |  |     }, | 
					
						
							|  |  |  |   ], | 
					
						
							|  |  |  |   dataGroups: [ | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       name: 'testPerf', | 
					
						
							|  |  |  |       maxSize: 3, | 
					
						
							|  |  |  |       strategy: 'performance', | 
					
						
							|  |  |  |       patterns: ['^/api/.*$'], | 
					
						
							|  |  |  |       timeoutMs: 1000, | 
					
						
							|  |  |  |       maxAge: 5000, | 
					
						
							|  |  |  |       version: 1, | 
					
						
							| 
									
										
										
										
											2020-04-29 16:59:15 +02:00
										 |  |  |       cacheQueryOptions: {ignoreVary: true, ignoreSearch: true}, | 
					
						
							| 
									
										
										
										
											2020-04-13 16:40:21 -07:00
										 |  |  |     }, | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       name: 'testRefresh', | 
					
						
							|  |  |  |       maxSize: 3, | 
					
						
							|  |  |  |       strategy: 'performance', | 
					
						
							|  |  |  |       patterns: ['^/refresh/.*$'], | 
					
						
							|  |  |  |       timeoutMs: 1000, | 
					
						
							|  |  |  |       refreshAheadMs: 1000, | 
					
						
							|  |  |  |       maxAge: 5000, | 
					
						
							|  |  |  |       version: 1, | 
					
						
							| 
									
										
										
										
											2020-04-29 16:59:15 +02:00
										 |  |  |       cacheQueryOptions: {ignoreVary: true}, | 
					
						
							| 
									
										
										
										
											2020-04-13 16:40:21 -07:00
										 |  |  |     }, | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       name: 'testFresh', | 
					
						
							|  |  |  |       maxSize: 3, | 
					
						
							|  |  |  |       strategy: 'freshness', | 
					
						
							|  |  |  |       patterns: ['^/fresh/.*$'], | 
					
						
							|  |  |  |       timeoutMs: 1000, | 
					
						
							|  |  |  |       maxAge: 5000, | 
					
						
							|  |  |  |       version: 1, | 
					
						
							| 
									
										
										
										
											2020-04-29 16:59:15 +02:00
										 |  |  |       cacheQueryOptions: {ignoreVary: true}, | 
					
						
							| 
									
										
										
										
											2020-04-13 16:40:21 -07:00
										 |  |  |     }, | 
					
						
							|  |  |  |   ], | 
					
						
							|  |  |  |   navigationUrls: [], | 
					
						
							| 
									
										
										
										
											2020-08-24 16:04:17 +02:00
										 |  |  |   navigationRequestStrategy: 'performance', | 
					
						
							| 
									
										
										
										
											2020-04-13 16:40:21 -07:00
										 |  |  |   hashTable: tmpHashTableForFs(dist), | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const seqIncreasedManifest: Manifest = { | 
					
						
							|  |  |  |   ...manifest, | 
					
						
							|  |  |  |   dataGroups: [ | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       ...manifest.dataGroups![0], | 
					
						
							|  |  |  |       version: 2, | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     manifest.dataGroups![1], | 
					
						
							|  |  |  |     manifest.dataGroups![2], | 
					
						
							|  |  |  |   ], | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const server = new MockServerStateBuilder().withStaticFiles(dist).withManifest(manifest).build(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const serverUpdate = | 
					
						
							|  |  |  |     new MockServerStateBuilder().withStaticFiles(distUpdate).withManifest(manifest).build(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const serverSeqUpdate = new MockServerStateBuilder() | 
					
						
							|  |  |  |                             .withStaticFiles(distUpdate) | 
					
						
							|  |  |  |                             .withManifest(seqIncreasedManifest) | 
					
						
							|  |  |  |                             .build(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | describe('data cache', () => { | 
					
						
							|  |  |  |   let scope: SwTestHarness; | 
					
						
							|  |  |  |   let driver: Driver; | 
					
						
							|  |  |  |   beforeEach(async () => { | 
					
						
							|  |  |  |     scope = new SwTestHarnessBuilder().withServerState(server).build(); | 
					
						
							|  |  |  |     driver = new Driver(scope, scope, new CacheDatabase(scope, scope)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Initialize.
 | 
					
						
							|  |  |  |     expect(await makeRequest(scope, '/foo.txt')).toEqual('this is foo'); | 
					
						
							|  |  |  |     await driver.initialized; | 
					
						
							|  |  |  |     server.clearRequests(); | 
					
						
							|  |  |  |     serverUpdate.clearRequests(); | 
					
						
							|  |  |  |     serverSeqUpdate.clearRequests(); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  |   afterEach(() => { | 
					
						
							|  |  |  |     server.reset(); | 
					
						
							|  |  |  |     serverUpdate.reset(); | 
					
						
							|  |  |  |     serverSeqUpdate.reset(); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   describe('in performance mode', () => { | 
					
						
							|  |  |  |     it('names the caches correctly', async () => { | 
					
						
							|  |  |  |       expect(await makeRequest(scope, '/api/test')).toEqual('version 1'); | 
					
						
							|  |  |  |       const keys = await scope.caches.keys(); | 
					
						
							|  |  |  |       expect(keys.every(key => key.startsWith('ngsw:/:'))).toEqual(true); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     it('caches a basic request', async () => { | 
					
						
							|  |  |  |       expect(await makeRequest(scope, '/api/test')).toEqual('version 1'); | 
					
						
							|  |  |  |       server.assertSawRequestFor('/api/test'); | 
					
						
							|  |  |  |       scope.advance(1000); | 
					
						
							|  |  |  |       expect(await makeRequest(scope, '/api/test')).toEqual('version 1'); | 
					
						
							|  |  |  |       server.assertNoOtherRequests(); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     it('does not cache opaque responses', async () => { | 
					
						
							|  |  |  |       expect(await makeNoCorsRequest(scope, '/api/test')).toBe(''); | 
					
						
							|  |  |  |       server.assertSawRequestFor('/api/test'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       expect(await makeNoCorsRequest(scope, '/api/test')).toBe(''); | 
					
						
							|  |  |  |       server.assertSawRequestFor('/api/test'); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     it('refreshes after awhile', async () => { | 
					
						
							|  |  |  |       expect(await makeRequest(scope, '/api/test')).toEqual('version 1'); | 
					
						
							|  |  |  |       server.clearRequests(); | 
					
						
							|  |  |  |       scope.advance(10000); | 
					
						
							|  |  |  |       scope.updateServerState(serverUpdate); | 
					
						
							|  |  |  |       expect(await makeRequest(scope, '/api/test')).toEqual('version 2'); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     it('expires the least recently used entry', async () => { | 
					
						
							|  |  |  |       expect(await makeRequest(scope, '/api/a')).toEqual('version A'); | 
					
						
							|  |  |  |       expect(await makeRequest(scope, '/api/b')).toEqual('version B'); | 
					
						
							|  |  |  |       expect(await makeRequest(scope, '/api/c')).toEqual('version C'); | 
					
						
							|  |  |  |       expect(await makeRequest(scope, '/api/d')).toEqual('version D'); | 
					
						
							|  |  |  |       expect(await makeRequest(scope, '/api/e')).toEqual('version E'); | 
					
						
							| 
									
										
										
										
											2019-06-24 15:04:10 +03:00
										 |  |  |       server.clearRequests(); | 
					
						
							| 
									
										
										
										
											2020-04-13 16:40:21 -07:00
										 |  |  |       expect(await makeRequest(scope, '/api/c')).toEqual('version C'); | 
					
						
							|  |  |  |       expect(await makeRequest(scope, '/api/d')).toEqual('version D'); | 
					
						
							|  |  |  |       expect(await makeRequest(scope, '/api/e')).toEqual('version E'); | 
					
						
							|  |  |  |       server.assertNoOtherRequests(); | 
					
						
							|  |  |  |       expect(await makeRequest(scope, '/api/a')).toEqual('version A'); | 
					
						
							|  |  |  |       expect(await makeRequest(scope, '/api/b')).toEqual('version B'); | 
					
						
							|  |  |  |       server.assertSawRequestFor('/api/a'); | 
					
						
							|  |  |  |       server.assertSawRequestFor('/api/b'); | 
					
						
							|  |  |  |       server.assertNoOtherRequests(); | 
					
						
							| 
									
										
										
										
											2019-06-24 15:04:10 +03:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2020-04-13 16:40:21 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     it('does not carry over cache with new version', async () => { | 
					
						
							|  |  |  |       expect(await makeRequest(scope, '/api/test')).toEqual('version 1'); | 
					
						
							|  |  |  |       scope.updateServerState(serverSeqUpdate); | 
					
						
							|  |  |  |       expect(await driver.checkForUpdate()).toEqual(true); | 
					
						
							|  |  |  |       await driver.updateClient(await scope.clients.get('default')); | 
					
						
							|  |  |  |       expect(await makeRequest(scope, '/api/test')).toEqual('version 2'); | 
					
						
							| 
									
										
										
										
											2017-09-28 16:18:12 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2020-04-29 16:07:34 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     it('CacheQueryOptions are passed through', async () => { | 
					
						
							|  |  |  |       await driver.initialized; | 
					
						
							|  |  |  |       const matchSpy = spyOn(MockCache.prototype, 'match').and.callThrough(); | 
					
						
							|  |  |  |       // the first request fetches the resource from the server
 | 
					
						
							|  |  |  |       await makeRequest(scope, '/api/a'); | 
					
						
							|  |  |  |       // the second one will be loaded from the cache
 | 
					
						
							|  |  |  |       await makeRequest(scope, '/api/a'); | 
					
						
							| 
									
										
										
										
											2020-04-29 16:59:15 +02:00
										 |  |  |       expect(matchSpy).toHaveBeenCalledWith( | 
					
						
							|  |  |  |           new MockRequest('/api/a'), {ignoreVary: true, ignoreSearch: true}); | 
					
						
							| 
									
										
										
										
											2020-04-29 16:07:34 +02:00
										 |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     it('still matches if search differs but ignoreSearch is enabled', async () => { | 
					
						
							|  |  |  |       await driver.initialized; | 
					
						
							|  |  |  |       const matchSpy = spyOn(MockCache.prototype, 'match').and.callThrough(); | 
					
						
							|  |  |  |       // the first request fetches the resource from the server
 | 
					
						
							|  |  |  |       await makeRequest(scope, '/api/a?v=1'); | 
					
						
							|  |  |  |       // the second one will be loaded from the cache
 | 
					
						
							|  |  |  |       server.clearRequests(); | 
					
						
							|  |  |  |       await makeRequest(scope, '/api/a?v=2'); | 
					
						
							|  |  |  |       server.assertNoOtherRequests(); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2020-04-13 16:40:21 -07:00
										 |  |  |   }); | 
					
						
							| 
									
										
										
										
											2017-09-28 16:18:12 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-13 16:40:21 -07:00
										 |  |  |   describe('in freshness mode', () => { | 
					
						
							|  |  |  |     it('goes to the server first', async () => { | 
					
						
							|  |  |  |       expect(await makeRequest(scope, '/fresh/data')).toEqual('this is fresh data'); | 
					
						
							|  |  |  |       server.assertSawRequestFor('/fresh/data'); | 
					
						
							|  |  |  |       server.clearRequests(); | 
					
						
							|  |  |  |       expect(await makeRequest(scope, '/fresh/data')).toEqual('this is fresh data'); | 
					
						
							|  |  |  |       server.assertSawRequestFor('/fresh/data'); | 
					
						
							|  |  |  |       server.assertNoOtherRequests(); | 
					
						
							|  |  |  |       scope.updateServerState(serverUpdate); | 
					
						
							|  |  |  |       expect(await makeRequest(scope, '/fresh/data')).toEqual('this is fresher data'); | 
					
						
							|  |  |  |       serverUpdate.assertSawRequestFor('/fresh/data'); | 
					
						
							|  |  |  |       serverUpdate.assertNoOtherRequests(); | 
					
						
							| 
									
										
										
										
											2017-09-28 16:18:12 -07:00
										 |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-13 16:40:21 -07:00
										 |  |  |     it('caches opaque responses', async () => { | 
					
						
							|  |  |  |       expect(await makeNoCorsRequest(scope, '/fresh/data')).toBe(''); | 
					
						
							|  |  |  |       server.assertSawRequestFor('/fresh/data'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       server.online = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       expect(await makeRequest(scope, '/fresh/data')).toBe(''); | 
					
						
							|  |  |  |       server.assertNoOtherRequests(); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     it('falls back on the cache when server times out', async () => { | 
					
						
							|  |  |  |       expect(await makeRequest(scope, '/fresh/data')).toEqual('this is fresh data'); | 
					
						
							|  |  |  |       server.assertSawRequestFor('/fresh/data'); | 
					
						
							|  |  |  |       server.clearRequests(); | 
					
						
							|  |  |  |       scope.updateServerState(serverUpdate); | 
					
						
							|  |  |  |       serverUpdate.pause(); | 
					
						
							|  |  |  |       const [res, done] = makePendingRequest(scope, '/fresh/data'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       await serverUpdate.nextRequest; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // Since the network request doesn't return within the timeout of 1,000ms,
 | 
					
						
							|  |  |  |       // this should return cached data.
 | 
					
						
							|  |  |  |       scope.advance(2000); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       expect(await res).toEqual('this is fresh data'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // Unpausing allows the worker to continue with caching.
 | 
					
						
							|  |  |  |       serverUpdate.unpause(); | 
					
						
							|  |  |  |       await done; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       serverUpdate.pause(); | 
					
						
							|  |  |  |       const [res2, done2] = makePendingRequest(scope, '/fresh/data'); | 
					
						
							|  |  |  |       await serverUpdate.nextRequest; | 
					
						
							|  |  |  |       scope.advance(2000); | 
					
						
							|  |  |  |       expect(await res2).toEqual('this is fresher data'); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     it('refreshes ahead', async () => { | 
					
						
							|  |  |  |       server.assertNoOtherRequests(); | 
					
						
							|  |  |  |       serverUpdate.assertNoOtherRequests(); | 
					
						
							|  |  |  |       expect(await makeRequest(scope, '/refresh/data')).toEqual('this is some data'); | 
					
						
							|  |  |  |       server.assertSawRequestFor('/refresh/data'); | 
					
						
							|  |  |  |       server.clearRequests(); | 
					
						
							|  |  |  |       expect(await makeRequest(scope, '/refresh/data')).toEqual('this is some data'); | 
					
						
							|  |  |  |       server.assertNoOtherRequests(); | 
					
						
							|  |  |  |       scope.updateServerState(serverUpdate); | 
					
						
							|  |  |  |       scope.advance(1500); | 
					
						
							|  |  |  |       expect(await makeRequest(scope, '/refresh/data')).toEqual('this is some data'); | 
					
						
							|  |  |  |       serverUpdate.assertSawRequestFor('/refresh/data'); | 
					
						
							|  |  |  |       expect(await makeRequest(scope, '/refresh/data')).toEqual('this is refreshed data'); | 
					
						
							|  |  |  |       serverUpdate.assertNoOtherRequests(); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     it('caches opaque responses on refresh', async () => { | 
					
						
							|  |  |  |       // Make the initial request and populate the cache.
 | 
					
						
							|  |  |  |       expect(await makeRequest(scope, '/fresh/data')).toBe('this is fresh data'); | 
					
						
							|  |  |  |       server.assertSawRequestFor('/fresh/data'); | 
					
						
							|  |  |  |       server.clearRequests(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // Update the server state and pause the server, so the next request times out.
 | 
					
						
							|  |  |  |       scope.updateServerState(serverUpdate); | 
					
						
							|  |  |  |       serverUpdate.pause(); | 
					
						
							|  |  |  |       const [res, done] = | 
					
						
							|  |  |  |           makePendingRequest(scope, new MockRequest('/fresh/data', {mode: 'no-cors'})); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // The network request times out after 1,000ms and the cached response is returned.
 | 
					
						
							|  |  |  |       await serverUpdate.nextRequest; | 
					
						
							|  |  |  |       scope.advance(2000); | 
					
						
							|  |  |  |       expect(await res).toBe('this is fresh data'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // Unpause the server to allow the network request to complete and be cached.
 | 
					
						
							|  |  |  |       serverUpdate.unpause(); | 
					
						
							|  |  |  |       await done; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // Pause the server to force the cached (opaque) response to be returned.
 | 
					
						
							|  |  |  |       serverUpdate.pause(); | 
					
						
							|  |  |  |       const [res2] = makePendingRequest(scope, '/fresh/data'); | 
					
						
							|  |  |  |       await serverUpdate.nextRequest; | 
					
						
							|  |  |  |       scope.advance(2000); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       expect(await res2).toBe(''); | 
					
						
							| 
									
										
										
										
											2017-09-28 16:18:12 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2020-04-29 16:59:15 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     it('CacheQueryOptions are passed through when falling back to cache', async () => { | 
					
						
							|  |  |  |       const matchSpy = spyOn(MockCache.prototype, 'match').and.callThrough(); | 
					
						
							|  |  |  |       await makeRequest(scope, '/fresh/data'); | 
					
						
							|  |  |  |       server.clearRequests(); | 
					
						
							|  |  |  |       scope.updateServerState(serverUpdate); | 
					
						
							|  |  |  |       serverUpdate.pause(); | 
					
						
							|  |  |  |       const [res, done] = makePendingRequest(scope, '/fresh/data'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       await serverUpdate.nextRequest; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // Since the network request doesn't return within the timeout of 1,000ms,
 | 
					
						
							|  |  |  |       // this should return cached data.
 | 
					
						
							|  |  |  |       scope.advance(2000); | 
					
						
							|  |  |  |       await res; | 
					
						
							|  |  |  |       expect(matchSpy).toHaveBeenCalledWith(new MockRequest('/fresh/data'), {ignoreVary: true}); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // Unpausing allows the worker to continue with caching.
 | 
					
						
							|  |  |  |       serverUpdate.unpause(); | 
					
						
							|  |  |  |       await done; | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2017-09-28 16:18:12 -07:00
										 |  |  |   }); | 
					
						
							| 
									
										
										
										
											2020-04-13 16:40:21 -07:00
										 |  |  | }); | 
					
						
							| 
									
										
										
										
											2017-12-16 14:42:55 -08:00
										 |  |  | })(); | 
					
						
							| 
									
										
										
										
											2017-09-28 16:18:12 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-24 15:04:08 +03:00
										 |  |  | function makeRequest(scope: SwTestHarness, url: string, clientId?: string): Promise<string|null> { | 
					
						
							|  |  |  |   const [resTextPromise, done] = makePendingRequest(scope, url, clientId); | 
					
						
							|  |  |  |   return done.then(() => resTextPromise); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-24 15:04:13 +03:00
										 |  |  | function makeNoCorsRequest( | 
					
						
							| 
									
										
										
										
											2019-11-12 09:08:40 -08:00
										 |  |  |     scope: SwTestHarness, url: string, clientId?: string): Promise<string|null> { | 
					
						
							| 
									
										
										
										
											2019-06-24 15:04:13 +03:00
										 |  |  |   const req = new MockRequest(url, {mode: 'no-cors'}); | 
					
						
							|  |  |  |   const [resTextPromise, done] = makePendingRequest(scope, req, clientId); | 
					
						
							|  |  |  |   return done.then(() => resTextPromise); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-13 16:40:21 -07:00
										 |  |  | function makePendingRequest(scope: SwTestHarness, urlOrReq: string|MockRequest, clientId?: string): | 
					
						
							|  |  |  |     [Promise<string|null>, Promise<void>] { | 
					
						
							| 
									
										
										
										
											2019-06-24 15:04:13 +03:00
										 |  |  |   const req = (typeof urlOrReq === 'string') ? new MockRequest(urlOrReq) : urlOrReq; | 
					
						
							|  |  |  |   const [resPromise, done] = scope.handleFetch(req, clientId || 'default'); | 
					
						
							| 
									
										
										
										
											2019-06-24 15:04:08 +03:00
										 |  |  |   return [ | 
					
						
							|  |  |  |     resPromise.then<string|null>(res => res ? res.text() : null), | 
					
						
							|  |  |  |     done, | 
					
						
							|  |  |  |   ]; | 
					
						
							|  |  |  | } |