mirror of
https://github.com/seaweedfs/seaweedfs.git
synced 2026-06-17 00:07:18 +03:00
e56a1c4c05
The admin UI served embedded static files uncompressed and without cache headers: embed.FS has zero mod times, so no Last-Modified, no ETag, no 304s -- every page load re-downloaded ~700KB of css/js in full, which gets painful over slow or tunneled links. Gzip the static tree at generation time (go generate ./weed/admin) and embed only the compressed mirror, shrinking the binary ~1.5MB. The handler hands the pre-compressed bytes to gzip-capable clients, decompresses for the rest, and sets Cache-Control, per-variant content-hash ETags and Vary so repeat loads revalidate with a 304. bootstrap.min.css goes 232KB -> 30KB on the wire. A drift test keeps static_gz/ in sync with static/.
117 lines
2.9 KiB
Go
117 lines
2.9 KiB
Go
package admin
|
|
|
|
import (
|
|
"bytes"
|
|
"compress/gzip"
|
|
"crypto/sha256"
|
|
"encoding/hex"
|
|
"io"
|
|
"io/fs"
|
|
"net/http"
|
|
"path"
|
|
"strconv"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
)
|
|
|
|
// Assets only change with the binary; the ETag revalidates across upgrades.
|
|
const staticCacheControl = "public, max-age=3600"
|
|
|
|
type staticAsset struct {
|
|
name string // base name, drives Content-Type in http.ServeContent
|
|
gz []byte
|
|
etag string
|
|
etagGz string
|
|
}
|
|
|
|
// identity returns the uncompressed bytes.
|
|
func (a *staticAsset) identity() ([]byte, error) {
|
|
zr, err := gzip.NewReader(bytes.NewReader(a.gz))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer zr.Close()
|
|
return io.ReadAll(zr)
|
|
}
|
|
|
|
var staticAssets = sync.OnceValue(func() map[string]*staticAsset {
|
|
assets := make(map[string]*staticAsset)
|
|
err := fs.WalkDir(staticGzFS, "static_gz", func(p string, d fs.DirEntry, err error) error {
|
|
if err != nil || d.IsDir() {
|
|
return err
|
|
}
|
|
gz, err := fs.ReadFile(staticGzFS, p)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
key := strings.TrimSuffix(strings.TrimPrefix(p, "static_gz/"), ".gz")
|
|
sum := sha256.Sum256(gz)
|
|
hash := hex.EncodeToString(sum[:8])
|
|
assets[key] = &staticAsset{
|
|
name: path.Base(key),
|
|
gz: gz,
|
|
etag: `"` + hash + `"`,
|
|
etagGz: `"` + hash + `-gz"`,
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
panic("walk embedded static assets: " + err.Error())
|
|
}
|
|
return assets
|
|
})
|
|
|
|
// StaticHandler serves the embedded admin static assets, which are gzipped
|
|
// at generation time. Gzip-capable clients get the compressed bytes as-is;
|
|
// others get them decompressed on the fly.
|
|
func StaticHandler() http.Handler {
|
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
key := strings.TrimPrefix(path.Clean("/"+r.URL.Path), "/")
|
|
asset, ok := staticAssets()[key]
|
|
if !ok {
|
|
http.NotFound(w, r)
|
|
return
|
|
}
|
|
h := w.Header()
|
|
h.Set("Cache-Control", staticCacheControl)
|
|
h.Add("Vary", "Accept-Encoding")
|
|
if acceptsGzip(r) {
|
|
h.Set("Content-Encoding", "gzip")
|
|
h.Set("ETag", asset.etagGz)
|
|
// ServeContent skips Content-Length when Content-Encoding is set
|
|
if r.Header.Get("Range") == "" {
|
|
h.Set("Content-Length", strconv.Itoa(len(asset.gz)))
|
|
}
|
|
http.ServeContent(w, r, asset.name, time.Time{}, bytes.NewReader(asset.gz))
|
|
return
|
|
}
|
|
data, err := asset.identity()
|
|
if err != nil {
|
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
|
return
|
|
}
|
|
h.Set("ETag", asset.etag)
|
|
http.ServeContent(w, r, asset.name, time.Time{}, bytes.NewReader(data))
|
|
})
|
|
}
|
|
|
|
func acceptsGzip(r *http.Request) bool {
|
|
for _, part := range strings.Split(r.Header.Get("Accept-Encoding"), ",") {
|
|
token, attr, hasAttr := strings.Cut(strings.TrimSpace(part), ";")
|
|
if strings.TrimSpace(token) != "gzip" {
|
|
continue
|
|
}
|
|
if !hasAttr {
|
|
return true
|
|
}
|
|
q, ok := strings.CutPrefix(strings.TrimSpace(attr), "q=")
|
|
if !ok {
|
|
return true
|
|
}
|
|
v, err := strconv.ParseFloat(q, 64)
|
|
return err == nil && v > 0
|
|
}
|
|
return false
|
|
}
|