fix(service-worker): improve ServiceWorker cache names (#42622)
This commit improves the cache names generated by the ServiceWorker by making them shorter and non-repetitive. In particular, the following changes are made: - Data-group cache names no longer include the `dynamic` infix, since it does not add any value. Before: `ngsw:<...>:data:dynamic:<...>` After: `ngsw:<...>:data:<...>` - `CacheDatabase` table names no longer include the `ngsw:<path>` prefix twice. Before: `ngsw:<path>:db:ngsw:<path>:<...>` After: `ngsw:<path>:db:<...>` NOTE 1: This change will result in different cache names being generated for the same app-versions with the new SericeWorker script. This means that some of the previously cached data will need to be re-downloaded (because the ServiceWorker will not be able to re-use the old caches), but that should be transparent for the end user. While possible, adding logic to allow the ServiceWorker to retrieve data from the old caches is not worth the extra complecity and maintenance cost. NOTE 2: Generating different cache names for some of the caches means that the ServiceWorker will not be able to clean-up some of the old caches. This will be taken care of in a subsequent commit that will rework the clean-up logic to be more robust (covering changes such as this one and other edgecases). PR Close #42622
This commit is contained in:
parent
7507ed2e54
commit
73b0275dc2
|
@ -77,7 +77,7 @@ export class AppVersion implements UpdateSource {
|
||||||
|
|
||||||
// Process each `AssetGroup` declared in the manifest. Each declared group gets an `AssetGroup`
|
// Process each `AssetGroup` declared in the manifest. Each declared group gets an `AssetGroup`
|
||||||
// instance created for it, of a type that depends on the configuration mode.
|
// instance created for it, of a type that depends on the configuration mode.
|
||||||
const assetCacheNamePrefix = `${adapter.cacheNamePrefix}:${manifestHash}:assets`;
|
const assetCacheNamePrefix = `${manifestHash}:assets`;
|
||||||
this.assetGroups = (manifest.assetGroups || []).map(config => {
|
this.assetGroups = (manifest.assetGroups || []).map(config => {
|
||||||
// Check the caching mode, which determines when resources will be fetched/updated.
|
// Check the caching mode, which determines when resources will be fetched/updated.
|
||||||
switch (config.installMode) {
|
switch (config.installMode) {
|
||||||
|
@ -91,11 +91,11 @@ export class AppVersion implements UpdateSource {
|
||||||
});
|
});
|
||||||
|
|
||||||
// Process each `DataGroup` declared in the manifest.
|
// Process each `DataGroup` declared in the manifest.
|
||||||
this.dataGroups = (manifest.dataGroups || [])
|
this.dataGroups =
|
||||||
|
(manifest.dataGroups || [])
|
||||||
.map(
|
.map(
|
||||||
config => new DataGroup(
|
config => new DataGroup(
|
||||||
scope, adapter, config, database, debugHandler,
|
scope, adapter, config, database, debugHandler, `${config.version}:data`));
|
||||||
`${adapter.cacheNamePrefix}:${config.version}:data`));
|
|
||||||
|
|
||||||
// This keeps backwards compatibility with app versions without navigation urls.
|
// This keeps backwards compatibility with app versions without navigation urls.
|
||||||
// Fix: https://github.com/angular/angular/issues/27209
|
// Fix: https://github.com/angular/angular/issues/27209
|
||||||
|
|
|
@ -67,7 +67,8 @@ export abstract class AssetGroup {
|
||||||
|
|
||||||
// This is the primary cache, which holds all of the cached requests for this group. If a
|
// This is the primary cache, which holds all of the cached requests for this group. If a
|
||||||
// resource isn't in this cache, it hasn't been fetched yet.
|
// resource isn't in this cache, it hasn't been fetched yet.
|
||||||
this.cache = scope.caches.open(`${cacheNamePrefix}:${config.name}:cache`);
|
this.cache =
|
||||||
|
scope.caches.open(`${adapter.cacheNamePrefix}:${cacheNamePrefix}:${config.name}:cache`);
|
||||||
|
|
||||||
// This is the metadata table, which holds specific information for each cached URL, such as
|
// This is the metadata table, which holds specific information for each cached URL, such as
|
||||||
// the timestamp of when it was added to the cache.
|
// the timestamp of when it was added to the cache.
|
||||||
|
@ -103,7 +104,8 @@ export abstract class AssetGroup {
|
||||||
* Clean up all the cached data for this group.
|
* Clean up all the cached data for this group.
|
||||||
*/
|
*/
|
||||||
async cleanup(): Promise<void> {
|
async cleanup(): Promise<void> {
|
||||||
await this.scope.caches.delete(`${this.cacheNamePrefix}:${this.config.name}:cache`);
|
await this.scope.caches.delete(
|
||||||
|
`${this.adapter.cacheNamePrefix}:${this.cacheNamePrefix}:${this.config.name}:cache`);
|
||||||
await this.db.delete(`${this.cacheNamePrefix}:${this.config.name}:meta`);
|
await this.db.delete(`${this.cacheNamePrefix}:${this.config.name}:meta`);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -249,11 +249,10 @@ export class DataGroup {
|
||||||
private config: DataGroupConfig, private db: Database, private debugHandler: DebugHandler,
|
private config: DataGroupConfig, private db: Database, private debugHandler: DebugHandler,
|
||||||
private cacheNamePrefix: string) {
|
private cacheNamePrefix: string) {
|
||||||
this.patterns = config.patterns.map(pattern => new RegExp(pattern));
|
this.patterns = config.patterns.map(pattern => new RegExp(pattern));
|
||||||
this.cache = scope.caches.open(`${cacheNamePrefix}:dynamic:${config.name}:cache`);
|
this.cache =
|
||||||
this.lruTable =
|
scope.caches.open(`${adapter.cacheNamePrefix}:${cacheNamePrefix}:${config.name}:cache`);
|
||||||
this.db.open(`${cacheNamePrefix}:dynamic:${config.name}:lru`, config.cacheQueryOptions);
|
this.lruTable = this.db.open(`${cacheNamePrefix}:${config.name}:lru`, config.cacheQueryOptions);
|
||||||
this.ageTable =
|
this.ageTable = this.db.open(`${cacheNamePrefix}:${config.name}:age`, config.cacheQueryOptions);
|
||||||
this.db.open(`${cacheNamePrefix}:dynamic:${config.name}:age`, config.cacheQueryOptions);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -550,9 +549,10 @@ export class DataGroup {
|
||||||
async cleanup(): Promise<void> {
|
async cleanup(): Promise<void> {
|
||||||
// Remove both the cache and the database entries which track LRU stats.
|
// Remove both the cache and the database entries which track LRU stats.
|
||||||
await Promise.all([
|
await Promise.all([
|
||||||
this.scope.caches.delete(`${this.cacheNamePrefix}:dynamic:${this.config.name}:cache`),
|
this.scope.caches.delete(
|
||||||
this.db.delete(`${this.cacheNamePrefix}:dynamic:${this.config.name}:age`),
|
`${this.adapter.cacheNamePrefix}:${this.cacheNamePrefix}:${this.config.name}:cache`),
|
||||||
this.db.delete(`${this.cacheNamePrefix}:dynamic:${this.config.name}:lru`),
|
this.db.delete(`${this.cacheNamePrefix}:${this.config.name}:age`),
|
||||||
|
this.db.delete(`${this.cacheNamePrefix}:${this.config.name}:lru`),
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1269,12 +1269,12 @@ describe('Driver', () => {
|
||||||
const cacheKeysFor = (baseHref: string, manifestHash: string) =>
|
const cacheKeysFor = (baseHref: string, manifestHash: string) =>
|
||||||
[`ngsw:${baseHref}:db:control`,
|
[`ngsw:${baseHref}:db:control`,
|
||||||
`ngsw:${baseHref}:${manifestHash}:assets:eager:cache`,
|
`ngsw:${baseHref}:${manifestHash}:assets:eager:cache`,
|
||||||
`ngsw:${baseHref}:db:ngsw:${baseHref}:${manifestHash}:assets:eager:meta`,
|
`ngsw:${baseHref}:db:${manifestHash}:assets:eager:meta`,
|
||||||
`ngsw:${baseHref}:${manifestHash}:assets:lazy:cache`,
|
`ngsw:${baseHref}:${manifestHash}:assets:lazy:cache`,
|
||||||
`ngsw:${baseHref}:db:ngsw:${baseHref}:${manifestHash}:assets:lazy:meta`,
|
`ngsw:${baseHref}:db:${manifestHash}:assets:lazy:meta`,
|
||||||
`ngsw:${baseHref}:42:data:dynamic:api:cache`,
|
`ngsw:${baseHref}:42:data:api:cache`,
|
||||||
`ngsw:${baseHref}:db:ngsw:${baseHref}:42:data:dynamic:api:lru`,
|
`ngsw:${baseHref}:db:42:data:api:lru`,
|
||||||
`ngsw:${baseHref}:db:ngsw:${baseHref}:42:data:dynamic:api:age`,
|
`ngsw:${baseHref}:db:42:data:api:age`,
|
||||||
];
|
];
|
||||||
|
|
||||||
const createManifestWithBaseHref = (baseHref: string, distDir: MockFileSystem): Manifest => ({
|
const createManifestWithBaseHref = (baseHref: string, distDir: MockFileSystem): Manifest => ({
|
||||||
|
|
Loading…
Reference in New Issue