summaryrefslogtreecommitdiffstats
path: root/modules/httpcache
diff options
context:
space:
mode:
authorsilverwind <me@silverwind.io>2020-11-17 23:44:52 +0100
committerGitHub <noreply@github.com>2020-11-17 17:44:52 -0500
commit0615b668dcbdeb8819662f2532cd5843f427dbcc (patch)
tree240a16a14d8ab503c83b75b25617c8275bc450d0 /modules/httpcache
parent9ec5e6c40b0c0e4a8009acbdd5dfea8c0e60cfcd (diff)
downloadgitea-0615b668dcbdeb8819662f2532cd5843f427dbcc.tar.gz
gitea-0615b668dcbdeb8819662f2532cd5843f427dbcc.zip
HTTP cache rework and enable caching for storage assets (#13569)
This enabled HTTP time-based cache for storage assets, primarily avatars. I have not observed If-Modified-Since from browsers during tests but I guess it's good to support regardless. It introduces a new generic httpcache module that can handle both time-based and etag-based caching. Additionally, manifest.json and robots.txt are now also cachable.
Diffstat (limited to 'modules/httpcache')
-rw-r--r--modules/httpcache/httpcache.go59
1 files changed, 59 insertions, 0 deletions
diff --git a/modules/httpcache/httpcache.go b/modules/httpcache/httpcache.go
new file mode 100644
index 0000000000..c4134f8e17
--- /dev/null
+++ b/modules/httpcache/httpcache.go
@@ -0,0 +1,59 @@
+// Copyright 2020 The Gitea Authors. All rights reserved.
+// Use of this source code is governed by a MIT-style
+// license that can be found in the LICENSE file.
+
+package httpcache
+
+import (
+ "encoding/base64"
+ "fmt"
+ "net/http"
+ "os"
+ "strconv"
+ "time"
+
+ "code.gitea.io/gitea/modules/setting"
+)
+
+// GetCacheControl returns a suitable "Cache-Control" header value
+func GetCacheControl() string {
+ if setting.RunMode == "dev" {
+ return "no-store"
+ }
+ return "private, max-age=" + strconv.FormatInt(int64(setting.StaticCacheTime.Seconds()), 10)
+}
+
+// generateETag generates an ETag based on size, filename and file modification time
+func generateETag(fi os.FileInfo) string {
+ etag := fmt.Sprint(fi.Size()) + fi.Name() + fi.ModTime().UTC().Format(http.TimeFormat)
+ return base64.StdEncoding.EncodeToString([]byte(etag))
+}
+
+// HandleTimeCache handles time-based caching for a HTTP request
+func HandleTimeCache(req *http.Request, w http.ResponseWriter, fi os.FileInfo) (handled bool) {
+ ifModifiedSince := req.Header.Get("If-Modified-Since")
+ if ifModifiedSince != "" {
+ t, err := time.Parse(http.TimeFormat, ifModifiedSince)
+ if err == nil && fi.ModTime().Unix() <= t.Unix() {
+ w.WriteHeader(http.StatusNotModified)
+ return true
+ }
+ }
+
+ w.Header().Set("Cache-Control", GetCacheControl())
+ w.Header().Set("Last-Modified", fi.ModTime().Format(http.TimeFormat))
+ return false
+}
+
+// HandleEtagCache handles ETag-based caching for a HTTP request
+func HandleEtagCache(req *http.Request, w http.ResponseWriter, fi os.FileInfo) (handled bool) {
+ etag := generateETag(fi)
+ if req.Header.Get("If-None-Match") == etag {
+ w.WriteHeader(http.StatusNotModified)
+ return true
+ }
+
+ w.Header().Set("Cache-Control", GetCacheControl())
+ w.Header().Set("ETag", etag)
+ return false
+}