2019-01-05 22:44:33 +00:00
|
|
|
package http
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"log"
|
|
|
|
"net/http"
|
|
|
|
"os"
|
2020-05-31 20:24:18 +00:00
|
|
|
"path"
|
2019-01-05 22:44:33 +00:00
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
2019-01-05 23:01:16 +00:00
|
|
|
"text/template"
|
2019-01-05 22:44:33 +00:00
|
|
|
|
2019-01-08 18:40:14 +00:00
|
|
|
rice "github.com/GeertJohan/go.rice"
|
2020-05-31 23:12:36 +00:00
|
|
|
|
2019-01-05 22:44:33 +00:00
|
|
|
"github.com/filebrowser/filebrowser/v2/auth"
|
2019-01-08 10:29:09 +00:00
|
|
|
"github.com/filebrowser/filebrowser/v2/settings"
|
2019-01-05 22:44:33 +00:00
|
|
|
"github.com/filebrowser/filebrowser/v2/storage"
|
|
|
|
"github.com/filebrowser/filebrowser/v2/version"
|
|
|
|
)
|
|
|
|
|
2020-05-31 23:12:36 +00:00
|
|
|
func handleWithStaticData(w http.ResponseWriter, _ *http.Request, d *data, box *rice.Box, file, contentType string) (int, error) {
|
2019-01-05 22:44:33 +00:00
|
|
|
w.Header().Set("Content-Type", contentType)
|
|
|
|
|
2019-02-15 12:58:45 +00:00
|
|
|
auther, err := d.store.Auth.Get(d.settings.AuthMethod)
|
|
|
|
if err != nil {
|
|
|
|
return http.StatusInternalServerError, err
|
|
|
|
}
|
|
|
|
|
2019-01-05 22:44:33 +00:00
|
|
|
data := map[string]interface{}{
|
|
|
|
"Name": d.settings.Branding.Name,
|
|
|
|
"DisableExternal": d.settings.Branding.DisableExternal,
|
2019-01-08 10:29:09 +00:00
|
|
|
"BaseURL": d.server.BaseURL,
|
2019-01-05 22:44:33 +00:00
|
|
|
"Version": version.Version,
|
2020-05-31 20:24:18 +00:00
|
|
|
"StaticURL": path.Join(d.server.BaseURL, "/static"),
|
2019-01-05 22:44:33 +00:00
|
|
|
"Signup": d.settings.Signup,
|
|
|
|
"NoAuth": d.settings.AuthMethod == auth.MethodNoAuth,
|
2019-12-04 01:31:11 +00:00
|
|
|
"AuthMethod": d.settings.AuthMethod,
|
2019-02-15 12:58:45 +00:00
|
|
|
"LoginPage": auther.LoginPage(),
|
2019-01-05 22:44:33 +00:00
|
|
|
"CSS": false,
|
|
|
|
"ReCaptcha": false,
|
2020-01-02 00:48:48 +00:00
|
|
|
"Theme": d.settings.Branding.Theme,
|
2020-07-23 10:38:03 +00:00
|
|
|
"EnableThumbs": d.server.EnableThumbnails,
|
2020-08-18 13:08:58 +00:00
|
|
|
"ResizePreview": d.server.ResizePreview,
|
2020-10-01 14:45:24 +00:00
|
|
|
"EnableExec": d.server.EnableExec,
|
2019-01-05 22:44:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if d.settings.Branding.Files != "" {
|
2020-05-31 23:12:36 +00:00
|
|
|
fPath := filepath.Join(d.settings.Branding.Files, "custom.css")
|
|
|
|
_, err := os.Stat(fPath) //nolint:shadow
|
2019-01-05 22:44:33 +00:00
|
|
|
|
|
|
|
if err != nil && !os.IsNotExist(err) {
|
|
|
|
log.Printf("couldn't load custom styles: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err == nil {
|
|
|
|
data["CSS"] = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if d.settings.AuthMethod == auth.MethodJSONAuth {
|
2020-05-31 23:12:36 +00:00
|
|
|
raw, err := d.store.Auth.Get(d.settings.AuthMethod) //nolint:shadow
|
2019-01-05 22:44:33 +00:00
|
|
|
if err != nil {
|
|
|
|
return http.StatusInternalServerError, err
|
|
|
|
}
|
|
|
|
|
|
|
|
auther := raw.(*auth.JSONAuth)
|
|
|
|
|
|
|
|
if auther.ReCaptcha != nil {
|
|
|
|
data["ReCaptcha"] = auther.ReCaptcha.Key != "" && auther.ReCaptcha.Secret != ""
|
|
|
|
data["ReCaptchaHost"] = auther.ReCaptcha.Host
|
|
|
|
data["ReCaptchaKey"] = auther.ReCaptcha.Key
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
b, err := json.MarshalIndent(data, "", " ")
|
|
|
|
if err != nil {
|
|
|
|
return http.StatusInternalServerError, err
|
|
|
|
}
|
|
|
|
|
|
|
|
data["Json"] = string(b)
|
|
|
|
|
2020-10-02 13:09:03 +00:00
|
|
|
fileContents, err := box.String(file)
|
|
|
|
if err != nil {
|
|
|
|
if err == os.ErrNotExist {
|
|
|
|
return http.StatusNotFound, err
|
|
|
|
}
|
|
|
|
return http.StatusInternalServerError, err
|
|
|
|
}
|
|
|
|
index := template.Must(template.New("index").Delims("[{[", "]}]").Parse(fileContents))
|
2019-01-05 22:44:33 +00:00
|
|
|
err = index.Execute(w, data)
|
|
|
|
if err != nil {
|
|
|
|
return http.StatusInternalServerError, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0, nil
|
|
|
|
}
|
|
|
|
|
2020-05-31 23:12:36 +00:00
|
|
|
func getStaticHandlers(store *storage.Storage, server *settings.Server) (index, static http.Handler) {
|
2019-01-05 22:44:33 +00:00
|
|
|
box := rice.MustFindBox("../frontend/dist")
|
|
|
|
handler := http.FileServer(box.HTTPBox())
|
|
|
|
|
2020-05-31 23:12:36 +00:00
|
|
|
index = handle(func(w http.ResponseWriter, r *http.Request, d *data) (int, error) {
|
2019-01-05 22:44:33 +00:00
|
|
|
if r.Method != http.MethodGet {
|
|
|
|
return http.StatusNotFound, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
w.Header().Set("x-xss-protection", "1; mode=block")
|
|
|
|
return handleWithStaticData(w, r, d, box, "index.html", "text/html; charset=utf-8")
|
2020-05-31 23:12:36 +00:00
|
|
|
}, "", store, server)
|
2019-01-05 22:44:33 +00:00
|
|
|
|
2020-05-31 23:12:36 +00:00
|
|
|
static = handle(func(w http.ResponseWriter, r *http.Request, d *data) (int, error) {
|
2019-01-05 22:44:33 +00:00
|
|
|
if r.Method != http.MethodGet {
|
|
|
|
return http.StatusNotFound, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if d.settings.Branding.Files != "" {
|
|
|
|
if strings.HasPrefix(r.URL.Path, "img/") {
|
2020-05-31 23:12:36 +00:00
|
|
|
fPath := filepath.Join(d.settings.Branding.Files, r.URL.Path)
|
|
|
|
if _, err := os.Stat(fPath); err == nil {
|
|
|
|
http.ServeFile(w, r, fPath)
|
2019-01-05 22:44:33 +00:00
|
|
|
return 0, nil
|
|
|
|
}
|
|
|
|
} else if r.URL.Path == "custom.css" && d.settings.Branding.Files != "" {
|
|
|
|
http.ServeFile(w, r, filepath.Join(d.settings.Branding.Files, "custom.css"))
|
|
|
|
return 0, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !strings.HasSuffix(r.URL.Path, ".js") {
|
|
|
|
handler.ServeHTTP(w, r)
|
|
|
|
return 0, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return handleWithStaticData(w, r, d, box, r.URL.Path, "application/javascript; charset=utf-8")
|
2020-05-31 23:12:36 +00:00
|
|
|
}, "/static/", store, server)
|
2019-01-05 22:44:33 +00:00
|
|
|
|
|
|
|
return index, static
|
|
|
|
}
|