Fix background images when BaseURL is specified

This commit is contained in:
Deluan 2022-11-29 14:40:44 -05:00
parent d8c5944ef1
commit 03640ca93d
9 changed files with 115 additions and 28 deletions

View file

@ -5,6 +5,7 @@ import (
"errors" "errors"
"fmt" "fmt"
"os" "os"
"strings"
"time" "time"
"github.com/navidrome/navidrome/conf" "github.com/navidrome/navidrome/conf"
@ -91,7 +92,7 @@ func startServer(ctx context.Context) func() error {
if conf.Server.Prometheus.Enabled { if conf.Server.Prometheus.Enabled {
a.MountRouter("Prometheus metrics", conf.Server.Prometheus.MetricsPath, promhttp.Handler()) a.MountRouter("Prometheus metrics", conf.Server.Prometheus.MetricsPath, promhttp.Handler())
} }
if conf.Server.UILoginBackgroundURL == consts.DefaultUILoginBackgroundURL { if strings.HasPrefix(conf.Server.UILoginBackgroundURL, "/") {
a.MountRouter("Background images", consts.DefaultUILoginBackgroundURL, backgrounds.NewHandler()) a.MountRouter("Background images", consts.DefaultUILoginBackgroundURL, backgrounds.NewHandler())
} }
return a.Run(ctx, fmt.Sprintf("%s:%d", conf.Server.Address, conf.Server.Port)) return a.Run(ctx, fmt.Sprintf("%s:%d", conf.Server.Address, conf.Server.Port))

View file

@ -0,0 +1,10 @@
package configtest
import "github.com/navidrome/navidrome/conf"
func SetupConfig() func() {
oldValues := *conf.Server
return func() {
conf.Server = &oldValues
}
}

2
go.mod
View file

@ -47,6 +47,7 @@ require (
golang.org/x/sync v0.1.0 golang.org/x/sync v0.1.0
golang.org/x/text v0.4.0 golang.org/x/text v0.4.0
golang.org/x/tools v0.3.0 golang.org/x/tools v0.3.0
gopkg.in/yaml.v3 v3.0.1
) )
require ( require (
@ -236,7 +237,6 @@ require (
google.golang.org/protobuf v1.28.1 // indirect google.golang.org/protobuf v1.28.1 // indirect
gopkg.in/ini.v1 v1.67.0 // indirect gopkg.in/ini.v1 v1.67.0 // indirect
gopkg.in/yaml.v2 v2.4.0 // indirect gopkg.in/yaml.v2 v2.4.0 // indirect
gopkg.in/yaml.v3 v3.0.1 // indirect
honnef.co/go/tools v0.3.3 // indirect honnef.co/go/tools v0.3.3 // indirect
mvdan.cc/gofumpt v0.4.0 // indirect mvdan.cc/gofumpt v0.4.0 // indirect
mvdan.cc/interfacer v0.0.0-20180901003855-c20040233aed // indirect mvdan.cc/interfacer v0.0.0-20180901003855-c20040233aed // indirect

View file

@ -5,17 +5,16 @@ import (
"crypto/rand" "crypto/rand"
"encoding/base64" "encoding/base64"
"errors" "errors"
"io"
"math/big" "math/big"
"net/http" "net/http"
"net/url" "net/url"
"path" "path"
"strings"
"sync" "sync"
"time" "time"
"github.com/navidrome/navidrome/consts" "github.com/navidrome/navidrome/consts"
"github.com/navidrome/navidrome/log" "github.com/navidrome/navidrome/log"
"gopkg.in/yaml.v3"
) )
type Handler struct { type Handler struct {
@ -31,7 +30,7 @@ func NewHandler() *Handler {
return h return h
} }
const ndImageServiceURL = "https://www.navidrome.org" const ndImageServiceURL = "https://www.navidrome.org/images"
func (h *Handler) ServeHTTP(w http.ResponseWriter, r *http.Request) { func (h *Handler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
image, err := h.getRandomImage(r.Context()) image, err := h.getRandomImage(r.Context())
@ -42,7 +41,7 @@ func (h *Handler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
return return
} }
http.Redirect(w, r, buildPath(ndImageServiceURL, "backgrounds", image+".jpg"), http.StatusFound) http.Redirect(w, r, buildPath(ndImageServiceURL, image), http.StatusFound)
} }
func (h *Handler) getRandomImage(ctx context.Context) (string, error) { func (h *Handler) getRandomImage(ctx context.Context) (string, error) {
@ -73,15 +72,15 @@ func (h *Handler) getImageList(ctx context.Context) ([]string, error) {
Timeout: 5 * time.Second, Timeout: 5 * time.Second,
} }
req, _ := http.NewRequestWithContext(ctx, http.MethodGet, buildPath(ndImageServiceURL, "images"), nil) req, _ := http.NewRequestWithContext(ctx, http.MethodGet, buildPath(ndImageServiceURL, "index.yml"), nil)
resp, err := c.Do(req) resp, err := c.Do(req)
if err != nil { if err != nil {
log.Warn(ctx, "Could not get list from image service", err) log.Warn(ctx, "Could not get list from image service", err)
return nil, err return nil, err
} }
defer resp.Body.Close() defer resp.Body.Close()
body, err := io.ReadAll(resp.Body) dec := yaml.NewDecoder(resp.Body)
h.list = strings.Split(string(body), "\n") err = dec.Decode(&h.list)
log.Debug(ctx, "Loaded images from image service", "total", len(h.list), "elapsed", time.Since(start)) log.Debug(ctx, "Loaded images from image service", "total", len(h.list), "elapsed", time.Since(start))
return h.list, err return h.list, err
} }

View file

@ -6,6 +6,7 @@ import (
"io" "io"
"io/fs" "io/fs"
"net/http" "net/http"
"path"
"strings" "strings"
"github.com/navidrome/navidrome/conf" "github.com/navidrome/navidrome/conf"
@ -53,6 +54,9 @@ func serveIndex(ds model.DataStore, fs fs.FS) http.HandlerFunc {
"devShowArtistPage": conf.Server.DevShowArtistPage, "devShowArtistPage": conf.Server.DevShowArtistPage,
"listenBrainzEnabled": conf.Server.ListenBrainz.Enabled, "listenBrainzEnabled": conf.Server.ListenBrainz.Enabled,
} }
if strings.HasPrefix(conf.Server.UILoginBackgroundURL, "/") {
appConfig["loginBackgroundURL"] = path.Join(conf.Server.BaseURL, conf.Server.UILoginBackgroundURL)
}
auth := handleLoginFromHeaders(ds, r) auth := handleLoginFromHeaders(ds, r)
if auth != nil { if auth != nil {
appConfig["auth"] = auth appConfig["auth"] = auth

View file

@ -10,6 +10,7 @@ import (
"strings" "strings"
"github.com/navidrome/navidrome/conf" "github.com/navidrome/navidrome/conf"
"github.com/navidrome/navidrome/conf/configtest"
"github.com/navidrome/navidrome/consts" "github.com/navidrome/navidrome/consts"
"github.com/navidrome/navidrome/model" "github.com/navidrome/navidrome/model"
"github.com/navidrome/navidrome/tests" "github.com/navidrome/navidrome/tests"
@ -24,7 +25,7 @@ var _ = Describe("serveIndex", func() {
BeforeEach(func() { BeforeEach(func() {
ds = &tests.MockDataStore{MockedUser: mockUser} ds = &tests.MockDataStore{MockedUser: mockUser}
conf.Server.UILoginBackgroundURL = "" DeferCleanup(configtest.SetupConfig())
}) })
It("adds app_config to index.html", func() { It("adds app_config to index.html", func() {
@ -82,17 +83,6 @@ var _ = Describe("serveIndex", func() {
Expect(config).To(HaveKeyWithValue("baseURL", "base_url_test")) Expect(config).To(HaveKeyWithValue("baseURL", "base_url_test"))
}) })
It("sets the uiLoginBackgroundURL", func() {
conf.Server.UILoginBackgroundURL = "my_background_url"
r := httptest.NewRequest("GET", "/index.html", nil)
w := httptest.NewRecorder()
serveIndex(ds, fs)(w, r)
config := extractAppConfig(w.Body.String())
Expect(config).To(HaveKeyWithValue("loginBackgroundURL", "my_background_url"))
})
It("sets the welcomeMessage", func() { It("sets the welcomeMessage", func() {
conf.Server.UIWelcomeMessage = "Hello" conf.Server.UIWelcomeMessage = "Hello"
r := httptest.NewRequest("GET", "/index.html", nil) r := httptest.NewRequest("GET", "/index.html", nil)
@ -298,9 +288,94 @@ var _ = Describe("serveIndex", func() {
config := extractAppConfig(w.Body.String()) config := extractAppConfig(w.Body.String())
Expect(config).To(HaveKeyWithValue("listenBrainzEnabled", true)) Expect(config).To(HaveKeyWithValue("listenBrainzEnabled", true))
}) })
Describe("loginBackgroundURL", func() {
Context("empty BaseURL", func() {
BeforeEach(func() {
conf.Server.BaseURL = "/"
})
When("it is the default URL", func() {
It("points to the default URL", func() {
conf.Server.UILoginBackgroundURL = consts.DefaultUILoginBackgroundURL
r := httptest.NewRequest("GET", "/index.html", nil)
w := httptest.NewRecorder()
serveIndex(ds, fs)(w, r)
config := extractAppConfig(w.Body.String())
Expect(config).To(HaveKeyWithValue("loginBackgroundURL", consts.DefaultUILoginBackgroundURL))
})
})
When("it is the default offline URL", func() {
It("points to the offline URL", func() {
conf.Server.UILoginBackgroundURL = consts.DefaultUILoginBackgroundURLOffline
r := httptest.NewRequest("GET", "/index.html", nil)
w := httptest.NewRecorder()
serveIndex(ds, fs)(w, r)
config := extractAppConfig(w.Body.String())
Expect(config).To(HaveKeyWithValue("loginBackgroundURL", consts.DefaultUILoginBackgroundURLOffline))
})
})
When("it is a custom URL", func() {
It("points to the offline URL", func() {
conf.Server.UILoginBackgroundURL = "https://example.com/images/1.jpg"
r := httptest.NewRequest("GET", "/index.html", nil)
w := httptest.NewRecorder()
serveIndex(ds, fs)(w, r)
config := extractAppConfig(w.Body.String())
Expect(config).To(HaveKeyWithValue("loginBackgroundURL", "https://example.com/images/1.jpg"))
})
})
})
Context("with a BaseURL", func() {
BeforeEach(func() {
conf.Server.BaseURL = "/music"
})
When("it is the default URL", func() {
It("points to the default URL with BaseURL prefix", func() {
conf.Server.UILoginBackgroundURL = consts.DefaultUILoginBackgroundURL
r := httptest.NewRequest("GET", "/index.html", nil)
w := httptest.NewRecorder()
serveIndex(ds, fs)(w, r)
config := extractAppConfig(w.Body.String())
Expect(config).To(HaveKeyWithValue("loginBackgroundURL", "/music"+consts.DefaultUILoginBackgroundURL))
})
})
When("it is the default offline URL", func() {
It("points to the offline URL", func() {
conf.Server.UILoginBackgroundURL = consts.DefaultUILoginBackgroundURLOffline
r := httptest.NewRequest("GET", "/index.html", nil)
w := httptest.NewRecorder()
serveIndex(ds, fs)(w, r)
config := extractAppConfig(w.Body.String())
Expect(config).To(HaveKeyWithValue("loginBackgroundURL", consts.DefaultUILoginBackgroundURLOffline))
})
})
When("it is a custom URL", func() {
It("points to the offline URL", func() {
conf.Server.UILoginBackgroundURL = "https://example.com/images/1.jpg"
r := httptest.NewRequest("GET", "/index.html", nil)
w := httptest.NewRecorder()
serveIndex(ds, fs)(w, r)
config := extractAppConfig(w.Body.String())
Expect(config).To(HaveKeyWithValue("loginBackgroundURL", "https://example.com/images/1.jpg"))
})
})
})
})
}) })
var appConfigRegex = regexp.MustCompile(`(?m)window.__APP_CONFIG__="([^"]*)`) var appConfigRegex = regexp.MustCompile(`(?m)window.__APP_CONFIG__=(.*);</script>`)
func extractAppConfig(body string) map[string]interface{} { func extractAppConfig(body string) map[string]interface{} {
config := make(map[string]interface{}) config := make(map[string]interface{})
@ -308,7 +383,7 @@ func extractAppConfig(body string) map[string]interface{} {
if match == nil { if match == nil {
return config return config
} }
str, err := strconv.Unquote("\"" + match[1] + "\"") str, err := strconv.Unquote(match[1])
if err != nil { if err != nil {
panic(fmt.Sprintf("%s: %s", match[1], err)) panic(fmt.Sprintf("%s: %s", match[1], err))
} }

View file

@ -7,9 +7,8 @@
content="Navidrome Music Server - {{.Version}}" content="Navidrome Music Server - {{.Version}}"
/> />
<title>Navidrome</title> <title>Navidrome</title>
<script> <!-- The line below has to match the exact format of the equivalent line in ui/build/index.html -->
window.__APP_CONFIG__="{{.AppConfig}}" <script>window.__APP_CONFIG__={{ .AppConfig }};</script>
</script>
</head> </head>
<body> <body>
</body> </body>

View file

@ -31,7 +31,7 @@
--> -->
<title>Navidrome</title> <title>Navidrome</title>
<script> <script>
window.__APP_CONFIG__ = "{{.AppConfig}}" window.__APP_CONFIG__ = {{ .AppConfig }}
</script> </script>
</head> </head>
<body> <body>

View file

@ -34,7 +34,6 @@ let config
try { try {
const appConfig = JSON.parse(window.__APP_CONFIG__) const appConfig = JSON.parse(window.__APP_CONFIG__)
config = { config = {
...defaultConfig, ...defaultConfig,
...appConfig, ...appConfig,