2016-06-07 07:46:45 +00:00
|
|
|
// Copyright (C) 2014 The Syncthing Authors.
|
|
|
|
//
|
|
|
|
// This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
// License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
2017-02-09 07:52:18 +01:00
|
|
|
// You can obtain one at https://mozilla.org/MPL/2.0/.
|
2016-06-07 07:46:45 +00:00
|
|
|
|
2019-03-26 20:53:58 +01:00
|
|
|
package api
|
2016-06-07 07:46:45 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"compress/gzip"
|
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"mime"
|
|
|
|
"net/http"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
2018-05-10 06:53:39 +01:00
|
|
|
"time"
|
2016-06-07 07:46:45 +00:00
|
|
|
|
|
|
|
"github.com/syncthing/syncthing/lib/auto"
|
|
|
|
"github.com/syncthing/syncthing/lib/config"
|
|
|
|
"github.com/syncthing/syncthing/lib/sync"
|
|
|
|
)
|
|
|
|
|
2019-11-08 22:44:37 +01:00
|
|
|
const themePrefix = "theme-assets/"
|
|
|
|
|
2016-06-07 07:46:45 +00:00
|
|
|
type staticsServer struct {
|
|
|
|
assetDir string
|
|
|
|
assets map[string][]byte
|
|
|
|
availableThemes []string
|
|
|
|
|
2019-11-08 21:37:42 +00:00
|
|
|
mut sync.RWMutex
|
|
|
|
theme string
|
|
|
|
lastThemeChange time.Time
|
2016-06-07 07:46:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func newStaticsServer(theme, assetDir string) *staticsServer {
|
|
|
|
s := &staticsServer{
|
2019-11-08 21:37:42 +00:00
|
|
|
assetDir: assetDir,
|
|
|
|
assets: auto.Assets(),
|
|
|
|
mut: sync.NewRWMutex(),
|
|
|
|
theme: theme,
|
|
|
|
lastThemeChange: time.Now().UTC(),
|
2016-06-07 07:46:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
seen := make(map[string]struct{})
|
|
|
|
// Load themes from compiled in assets.
|
|
|
|
for file := range auto.Assets() {
|
|
|
|
theme := strings.Split(file, "/")[0]
|
|
|
|
if _, ok := seen[theme]; !ok {
|
|
|
|
seen[theme] = struct{}{}
|
|
|
|
s.availableThemes = append(s.availableThemes, theme)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if assetDir != "" {
|
|
|
|
// Load any extra themes from the asset override dir.
|
|
|
|
for _, dir := range dirNames(assetDir) {
|
|
|
|
if _, ok := seen[dir]; !ok {
|
|
|
|
seen[dir] = struct{}{}
|
|
|
|
s.availableThemes = append(s.availableThemes, dir)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *staticsServer) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
|
|
|
switch r.URL.Path {
|
|
|
|
case "/themes.json":
|
|
|
|
s.serveThemes(w, r)
|
|
|
|
default:
|
|
|
|
s.serveAsset(w, r)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *staticsServer) serveAsset(w http.ResponseWriter, r *http.Request) {
|
2018-05-10 06:53:39 +01:00
|
|
|
w.Header().Set("Cache-Control", "no-cache, must-revalidate")
|
|
|
|
|
2016-06-07 07:46:45 +00:00
|
|
|
file := r.URL.Path
|
|
|
|
|
|
|
|
if file[0] == '/' {
|
|
|
|
file = file[1:]
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(file) == 0 {
|
|
|
|
file = "index.html"
|
|
|
|
}
|
|
|
|
|
|
|
|
s.mut.RLock()
|
|
|
|
theme := s.theme
|
2019-11-08 21:37:42 +00:00
|
|
|
modificationTime := s.lastThemeChange
|
2016-06-07 07:46:45 +00:00
|
|
|
s.mut.RUnlock()
|
|
|
|
|
2019-11-08 22:44:37 +01:00
|
|
|
// If path starts with special prefix, get theme and file from path
|
|
|
|
if strings.HasPrefix(file, themePrefix) {
|
|
|
|
path := file[len(themePrefix):]
|
|
|
|
i := strings.IndexRune(path, '/')
|
|
|
|
|
|
|
|
if i == -1 {
|
|
|
|
http.NotFound(w, r)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
theme = path[:i]
|
|
|
|
file = path[i+1:]
|
|
|
|
}
|
|
|
|
|
2016-06-07 07:46:45 +00:00
|
|
|
// Check for an override for the current theme.
|
|
|
|
if s.assetDir != "" {
|
|
|
|
p := filepath.Join(s.assetDir, theme, filepath.FromSlash(file))
|
|
|
|
if _, err := os.Stat(p); err == nil {
|
2018-03-28 15:51:24 -04:00
|
|
|
mtype := s.mimeTypeForFile(file)
|
|
|
|
if len(mtype) != 0 {
|
|
|
|
w.Header().Set("Content-Type", mtype)
|
|
|
|
}
|
2016-06-07 07:46:45 +00:00
|
|
|
http.ServeFile(w, r, p)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check for a compiled in asset for the current theme.
|
|
|
|
bs, ok := s.assets[theme+"/"+file]
|
|
|
|
if !ok {
|
|
|
|
// Check for an overridden default asset.
|
|
|
|
if s.assetDir != "" {
|
|
|
|
p := filepath.Join(s.assetDir, config.DefaultTheme, filepath.FromSlash(file))
|
|
|
|
if _, err := os.Stat(p); err == nil {
|
2018-03-28 15:51:24 -04:00
|
|
|
mtype := s.mimeTypeForFile(file)
|
|
|
|
if len(mtype) != 0 {
|
|
|
|
w.Header().Set("Content-Type", mtype)
|
|
|
|
}
|
2016-06-07 07:46:45 +00:00
|
|
|
http.ServeFile(w, r, p)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check for a compiled in default asset.
|
|
|
|
bs, ok = s.assets[config.DefaultTheme+"/"+file]
|
|
|
|
if !ok {
|
|
|
|
http.NotFound(w, r)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-08 21:37:42 +00:00
|
|
|
etag := fmt.Sprintf("%d", modificationTime.Unix())
|
|
|
|
w.Header().Set("Last-Modified", modificationTime.Format(http.TimeFormat))
|
2018-05-10 06:53:39 +01:00
|
|
|
w.Header().Set("Etag", etag)
|
|
|
|
|
2018-06-18 14:14:17 +08:00
|
|
|
if t, err := time.Parse(http.TimeFormat, r.Header.Get("If-Modified-Since")); err == nil {
|
2019-11-08 21:37:42 +00:00
|
|
|
if modificationTime.Equal(t) || modificationTime.Before(t) {
|
2018-06-18 14:14:17 +08:00
|
|
|
w.WriteHeader(http.StatusNotModified)
|
|
|
|
return
|
|
|
|
}
|
2018-05-10 06:53:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if match := r.Header.Get("If-None-Match"); match != "" {
|
|
|
|
if strings.Contains(match, etag) {
|
|
|
|
w.WriteHeader(http.StatusNotModified)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-07 07:46:45 +00:00
|
|
|
mtype := s.mimeTypeForFile(file)
|
|
|
|
if len(mtype) != 0 {
|
|
|
|
w.Header().Set("Content-Type", mtype)
|
|
|
|
}
|
|
|
|
if strings.Contains(r.Header.Get("Accept-Encoding"), "gzip") {
|
|
|
|
w.Header().Set("Content-Encoding", "gzip")
|
|
|
|
} else {
|
|
|
|
// ungzip if browser not send gzip accepted header
|
|
|
|
var gr *gzip.Reader
|
|
|
|
gr, _ = gzip.NewReader(bytes.NewReader(bs))
|
|
|
|
bs, _ = ioutil.ReadAll(gr)
|
|
|
|
gr.Close()
|
|
|
|
}
|
|
|
|
w.Header().Set("Content-Length", fmt.Sprintf("%d", len(bs)))
|
|
|
|
|
2019-02-02 12:16:27 +01:00
|
|
|
w.Write(bs)
|
2016-06-07 07:46:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *staticsServer) serveThemes(w http.ResponseWriter, r *http.Request) {
|
|
|
|
sendJSON(w, map[string][]string{
|
|
|
|
"themes": s.availableThemes,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *staticsServer) mimeTypeForFile(file string) string {
|
|
|
|
// We use a built in table of the common types since the system
|
|
|
|
// TypeByExtension might be unreliable. But if we don't know, we delegate
|
2017-09-10 08:28:12 +00:00
|
|
|
// to the system. All our files are UTF-8.
|
2016-06-07 07:46:45 +00:00
|
|
|
ext := filepath.Ext(file)
|
|
|
|
switch ext {
|
|
|
|
case ".htm", ".html":
|
2017-09-10 08:28:12 +00:00
|
|
|
return "text/html; charset=utf-8"
|
2016-06-07 07:46:45 +00:00
|
|
|
case ".css":
|
2017-09-10 08:28:12 +00:00
|
|
|
return "text/css; charset=utf-8"
|
2016-06-07 07:46:45 +00:00
|
|
|
case ".js":
|
2017-09-10 08:28:12 +00:00
|
|
|
return "application/javascript; charset=utf-8"
|
2016-06-07 07:46:45 +00:00
|
|
|
case ".json":
|
2017-09-10 08:28:12 +00:00
|
|
|
return "application/json; charset=utf-8"
|
2016-06-07 07:46:45 +00:00
|
|
|
case ".png":
|
|
|
|
return "image/png"
|
|
|
|
case ".ttf":
|
|
|
|
return "application/x-font-ttf"
|
|
|
|
case ".woff":
|
|
|
|
return "application/x-font-woff"
|
|
|
|
case ".svg":
|
2017-09-10 08:28:12 +00:00
|
|
|
return "image/svg+xml; charset=utf-8"
|
2016-06-07 07:46:45 +00:00
|
|
|
default:
|
|
|
|
return mime.TypeByExtension(ext)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *staticsServer) setTheme(theme string) {
|
|
|
|
s.mut.Lock()
|
|
|
|
s.theme = theme
|
2019-11-08 21:37:42 +00:00
|
|
|
s.lastThemeChange = time.Now().UTC()
|
2016-06-07 07:46:45 +00:00
|
|
|
s.mut.Unlock()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *staticsServer) String() string {
|
|
|
|
return fmt.Sprintf("staticsServer@%p", s)
|
|
|
|
}
|