@@ -32,6 +32,7 @@ import (
32
32
"time"
33
33
34
34
"github.com/creachadair/mds/cache"
35
+ "github.com/creachadair/mds/mapset"
35
36
"github.com/creachadair/scheddle"
36
37
"github.com/creachadair/taskgroup"
37
38
"github.com/tailscale/go-cache-plugin/lib/s3util"
@@ -234,7 +235,8 @@ func (s *Server) ServeHTTP(w http.ResponseWriter, r *http.Request) {
234
235
if canCache {
235
236
proxy .ModifyResponse = func (rsp * http.Response ) error {
236
237
maxAge , isVolatile := s .canMemoryCache (rsp )
237
- if ! isVolatile && ! s .canCacheResponse (rsp ) {
238
+ canCacheResponse := s .canCacheResponse (rsp )
239
+ if ! canCacheResponse && ! isVolatile {
238
240
// A response we cannot cache at all.
239
241
setXCacheInfo (rsp .Header , "fetch, uncached" , "" )
240
242
s .rspNotCached .Add (1 )
@@ -249,7 +251,8 @@ func (s *Server) ServeHTTP(w http.ResponseWriter, r *http.Request) {
249
251
Reader : io .TeeReader (rsp .Body , & buf ),
250
252
Closer : rsp .Body ,
251
253
}
252
- if isVolatile {
254
+ if ! canCacheResponse && isVolatile {
255
+ // A volatile response we can cache temporarily.
253
256
setXCacheInfo (rsp .Header , "fetch, cached, volatile" , hash )
254
257
updateCache = func () {
255
258
body := buf .Bytes ()
@@ -323,16 +326,44 @@ func hostMatchesTarget(host string, targets []string) bool {
323
326
324
327
// canCacheRequest reports whether r is a request whose response can be cached.
325
328
func (s * Server ) canCacheRequest (r * http.Request ) bool {
326
- return r .Method == "GET" && ! slices . Contains ( splitCacheControl ( r .Header ), "no-store" )
329
+ return r .Method == "GET" && ! parseCacheControl ( r .Header . Get ( "Cache-Control" )). Keys . Has ( "no-store" )
327
330
}
328
331
329
332
// canCacheResponse reports whether r is a response whose body can be cached.
330
333
func (s * Server ) canCacheResponse (rsp * http.Response ) bool {
331
334
if rsp .StatusCode != http .StatusOK {
332
335
return false
333
336
}
334
- cc := splitCacheControl (rsp .Header )
335
- return ! slices .Contains (cc , "no-store" ) && slices .Contains (cc , "immutable" )
337
+ cc := parseCacheControl (rsp .Header .Get ("Cache-Control" ))
338
+ if cc .Keys .Has ("no-store" ) {
339
+ return false
340
+ } else if cc .Keys .Has ("immutable" ) {
341
+ return true
342
+ }
343
+
344
+ // We treat a response that is not immutable but requires validation as
345
+ // cacheable if its max-age is so long it doesn't matter.
346
+ const goodLongTime = 60 * 24 * time .Hour
347
+ return cc .Keys .Has ("must-revalidate" ) && cc .MaxAge > goodLongTime
348
+ }
349
+
350
+ type cacheControl struct {
351
+ Keys mapset.Set [string ]
352
+ MaxAge time.Duration
353
+ }
354
+
355
+ func parseCacheControl (s string ) (out cacheControl ) {
356
+ for _ , v := range strings .Split (s , "," ) {
357
+ key , val , ok := strings .Cut (strings .TrimSpace (v ), "=" )
358
+ if ok && key == "max-age" {
359
+ sec , err := strconv .Atoi (val )
360
+ if err == nil {
361
+ out .MaxAge = time .Duration (sec ) * time .Second
362
+ }
363
+ }
364
+ out .Keys .Add (key )
365
+ }
366
+ return
336
367
}
337
368
338
369
// canMemoryCache reports whether r is a volatile response whose body can be
@@ -342,21 +373,19 @@ func (s *Server) canMemoryCache(rsp *http.Response) (time.Duration, bool) {
342
373
if rsp .StatusCode != http .StatusOK {
343
374
return 0 , false
344
375
}
345
- var maxAge time.Duration
346
- for _ , v := range splitCacheControl (rsp .Header ) {
347
- if v == "no-store" || v == "immutable" {
348
- return 0 , false // don't cache immutable things in memory
349
- }
350
- sfx , ok := strings .CutPrefix (v , "max-age=" )
351
- if ! ok {
352
- continue
353
- }
354
- sec , err := strconv .Atoi (sfx )
355
- if err == nil {
356
- maxAge = time .Duration (min (sec , 3600 )) * time .Second
357
- }
376
+ cc := parseCacheControl (rsp .Header .Get ("Cache-Control" ))
377
+ if cc .Keys .Has ("no-store" ) || cc .Keys .Has ("no-cache" ) {
378
+ // While no-cache doesn't mean we can't cache it, it requires
379
+ // re-validation before reusing the response, so treat that as if it were
380
+ // no-store.
381
+ return 0 , false
358
382
}
359
- return maxAge , maxAge > 0
383
+
384
+ // We'll cache things in memory if they aren't expected to last too long.
385
+ if cc .MaxAge > 0 && cc .MaxAge < time .Hour {
386
+ return cc .MaxAge , true
387
+ }
388
+ return 0 , false
360
389
}
361
390
362
391
// hashRequest generates the storage digest for the specified request URL.
@@ -375,12 +404,3 @@ func writeCachedResponse(w http.ResponseWriter, hdr http.Header, body []byte) {
375
404
}
376
405
w .Write (body )
377
406
}
378
-
379
- // splitCacheControl returns the tokens of the cache control header from h.
380
- func splitCacheControl (h http.Header ) []string {
381
- fs := strings .Split (h .Get ("Cache-Control" ), "," )
382
- for i , v := range fs {
383
- fs [i ] = strings .TrimSpace (v )
384
- }
385
- return fs
386
- }
0 commit comments