@@ -43,13 +43,13 @@ class PrecacheStrategy extends Strategy {
43
43
}
44
44
45
45
return response ;
46
- }
46
+ } ,
47
47
} ;
48
48
49
49
static readonly copyRedirectedCacheableResponsesPlugin : WorkboxPlugin = {
50
50
async cacheWillUpdate ( { response} ) {
51
51
return response . redirected ? await copyResponse ( response ) : response ;
52
- }
52
+ } ,
53
53
} ;
54
54
55
55
/**
@@ -73,7 +73,8 @@ class PrecacheStrategy extends Strategy {
73
73
options . cacheName = cacheNames . getPrecacheName ( options . cacheName ) ;
74
74
super ( options ) ;
75
75
76
- this . _fallbackToNetwork = options . fallbackToNetwork === false ? false : true ;
76
+ this . _fallbackToNetwork =
77
+ options . fallbackToNetwork === false ? false : true ;
77
78
78
79
// Redirected responses cannot be used to satisfy a navigation request, so
79
80
// any redirected response must be "copied" rather than cloned, so the new
@@ -91,31 +92,68 @@ class PrecacheStrategy extends Strategy {
91
92
*/
92
93
async _handle ( request : Request , handler : StrategyHandler ) : Promise < Response > {
93
94
const response = await handler . cacheMatch ( request ) ;
94
- if ( ! response ) {
95
- // If this is an `install` event then populate the cache. If this is a
96
- // `fetch` event (or any other event) then respond with the cached
97
- // response.
98
- if ( handler . event && handler . event . type === 'install' ) {
99
- return await this . _handleInstall ( request , handler ) ;
100
- }
101
- return await this . _handleFetch ( request , handler ) ;
95
+ if ( response ) {
96
+ return response ;
102
97
}
103
98
104
- return response ;
99
+ // If this is an `install` event for an entry that isn't already cached,
100
+ // then populate the cache.
101
+ if ( handler . event && handler . event . type === 'install' ) {
102
+ return await this . _handleInstall ( request , handler ) ;
103
+ }
104
+
105
+ // Getting here means something went wrong. An entry that should have been
106
+ // precached wasn't found in the cache.
107
+ return await this . _handleFetch ( request , handler ) ;
105
108
}
106
109
107
- async _handleFetch ( request : Request , handler : StrategyHandler ) : Promise < Response > {
110
+ async _handleFetch (
111
+ request : Request ,
112
+ handler : StrategyHandler ,
113
+ ) : Promise < Response > {
108
114
let response ;
115
+ const params = ( handler . params || { } ) as {
116
+ cacheKey ?: string ;
117
+ integrity ?: string ;
118
+ } ;
109
119
110
- // Fall back to the network if we don't have a cached response
111
- // (perhaps due to manual cache cleanup).
120
+ // Fall back to the network if we're configured to do so.
112
121
if ( this . _fallbackToNetwork ) {
113
122
if ( process . env . NODE_ENV !== 'production' ) {
114
- logger . warn ( `The precached response for ` +
123
+ logger . warn (
124
+ `The precached response for ` +
115
125
`${ getFriendlyURL ( request . url ) } in ${ this . cacheName } was not ` +
116
- `found. Falling back to the network instead.` ) ;
126
+ `found. Falling back to the network.` ,
127
+ ) ;
128
+ }
129
+
130
+ const integrityInManifest = params . integrity ;
131
+ const integrityInRequest = request . integrity ;
132
+ const noIntegrityConflict =
133
+ ! integrityInRequest || integrityInRequest === integrityInManifest ;
134
+ response = await handler . fetch (
135
+ new Request ( request , {
136
+ integrity : integrityInRequest || integrityInManifest ,
137
+ } ) ,
138
+ ) ;
139
+
140
+ // It's only "safe" to repair the cache if we're using SRI to guarantee
141
+ // that the response matches the precache manifest's expectations,
142
+ // and there's either a) no integrity property in the incoming request
143
+ // or b) there is an integrity, and it matches the precache manifest.
144
+ // See https://github.com/GoogleChrome/workbox/issues/2858
145
+ if ( integrityInManifest && noIntegrityConflict ) {
146
+ this . _useDefaultCacheabilityPluginIfNeeded ( ) ;
147
+ const wasCached = await handler . cachePut ( request , response . clone ( ) ) ;
148
+ if ( process . env . NODE_ENV !== 'production' ) {
149
+ if ( wasCached ) {
150
+ logger . log (
151
+ `A response for ${ getFriendlyURL ( request . url ) } ` +
152
+ `was used to "repair" the precache.` ,
153
+ ) ;
154
+ }
155
+ }
117
156
}
118
- response = await handler . fetch ( request ) ;
119
157
} else {
120
158
// This shouldn't normally happen, but there are edge cases:
121
159
// https://github.com/GoogleChrome/workbox/issues/1441
@@ -126,18 +164,19 @@ class PrecacheStrategy extends Strategy {
126
164
}
127
165
128
166
if ( process . env . NODE_ENV !== 'production' ) {
129
- // Params in handlers is type any, can't change right now.
130
- // eslint-disable-next-line
131
- const cacheKey = handler . params && handler . params . cacheKey ||
132
- await handler . getCacheKey ( request , 'read' ) ;
167
+ const cacheKey =
168
+ params . cacheKey || ( await handler . getCacheKey ( request , 'read' ) ) ;
133
169
134
170
// Workbox is going to handle the route.
135
171
// print the routing details to the console.
136
- logger . groupCollapsed ( `Precaching is responding to: ` +
137
- getFriendlyURL ( request . url ) ) ;
138
- // cacheKey is type any, can't change right now.
139
- // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access
140
- logger . log ( `Serving the precached url: ${ getFriendlyURL ( cacheKey . url ) } ` ) ;
172
+ logger . groupCollapsed (
173
+ `Precaching is responding to: ` + getFriendlyURL ( request . url ) ,
174
+ ) ;
175
+ logger . log (
176
+ `Serving the precached url: ${ getFriendlyURL (
177
+ cacheKey instanceof Request ? cacheKey . url : cacheKey ,
178
+ ) } `,
179
+ ) ;
141
180
142
181
logger . groupCollapsed ( `View request details here.` ) ;
143
182
logger . log ( request ) ;
@@ -149,10 +188,14 @@ class PrecacheStrategy extends Strategy {
149
188
150
189
logger . groupEnd ( ) ;
151
190
}
191
+
152
192
return response ;
153
193
}
154
194
155
- async _handleInstall ( request : Request , handler : StrategyHandler ) : Promise < Response > {
195
+ async _handleInstall (
196
+ request : Request ,
197
+ handler : StrategyHandler ,
198
+ ) : Promise < Response > {
156
199
this . _useDefaultCacheabilityPluginIfNeeded ( ) ;
157
200
158
201
const response = await handler . fetch ( request ) ;
0 commit comments