mirror of
https://github.com/mudler/LocalAI.git
synced 2026-06-25 09:09:07 -04:00
* fix(http): harden BaseURL proxy scheme/host detection Split comma-separated X-Forwarded-Proto and honor the RFC 7239 Forwarded header so generated links use https behind common reverse-proxy setups. Refs #10482 Assisted-by: Claude:claude-opus-4-8 Signed-off-by: Ettore Di Giacinto <mudler@localai.io> * feat(http): honor explicit external base URL in BaseURL When _external_base_url is set in the request context it dictates the origin (scheme+host+port); the proxy path prefix is still appended. Refs #10482 Assisted-by: Claude:claude-opus-4-8 Signed-off-by: Ettore Di Giacinto <mudler@localai.io> * feat(config): generalize LOCALAI_BASE_URL to ExternalBaseURL LOCALAI_BASE_URL now sets a single instance-wide external base URL used for OAuth callbacks and all self-referential links. A Pre middleware stamps it into the request context for middleware.BaseURL. Refs #10482 Assisted-by: Claude:claude-opus-4-8 Signed-off-by: Ettore Di Giacinto <mudler@localai.io> * docs: document LOCALAI_BASE_URL and reverse-proxy headers Refs #10482 Assisted-by: Claude:claude-opus-4-8 Signed-off-by: Ettore Di Giacinto <mudler@localai.io> * test(http): cover parseForwarded edge cases; clarify base-url flag group Adds direct unit coverage for quoted/malformed/multi-element Forwarded headers and regroups the external base URL flag away from auth-only. Refs #10482 Assisted-by: Claude:claude-opus-4-8 Signed-off-by: Ettore Di Giacinto <mudler@localai.io> --------- Signed-off-by: Ettore Di Giacinto <mudler@localai.io> Co-authored-by: Ettore Di Giacinto <mudler@localai.io>
273 lines
8.6 KiB
Go
273 lines
8.6 KiB
Go
package middleware
|
|
|
|
import (
|
|
"net/http/httptest"
|
|
|
|
"github.com/labstack/echo/v4"
|
|
. "github.com/onsi/ginkgo/v2"
|
|
. "github.com/onsi/gomega"
|
|
)
|
|
|
|
var _ = Describe("BaseURL", func() {
|
|
Context("without prefix", func() {
|
|
It("should return base URL without prefix", func() {
|
|
app := echo.New()
|
|
actualURL := ""
|
|
|
|
// Register route - use the actual request path so routing works
|
|
routePath := "/hello/world"
|
|
app.GET(routePath, func(c echo.Context) error {
|
|
actualURL = BaseURL(c)
|
|
return nil
|
|
})
|
|
|
|
req := httptest.NewRequest("GET", "/hello/world", nil)
|
|
rec := httptest.NewRecorder()
|
|
app.ServeHTTP(rec, req)
|
|
|
|
Expect(rec.Code).To(Equal(200), "response status code")
|
|
Expect(actualURL).To(Equal("http://example.com/"), "base URL")
|
|
})
|
|
})
|
|
|
|
Context("with prefix", func() {
|
|
It("should return base URL with prefix", func() {
|
|
app := echo.New()
|
|
actualURL := ""
|
|
|
|
// Register route with the stripped path (after middleware removes prefix)
|
|
routePath := "/hello/world"
|
|
app.GET(routePath, func(c echo.Context) error {
|
|
// Simulate what StripPathPrefix middleware does - store original path
|
|
c.Set("_original_path", "/myprefix/hello/world")
|
|
// Modify the request path to simulate prefix stripping
|
|
c.Request().URL.Path = "/hello/world"
|
|
actualURL = BaseURL(c)
|
|
return nil
|
|
})
|
|
|
|
// Make request with stripped path (middleware would have already processed it)
|
|
req := httptest.NewRequest("GET", "/hello/world", nil)
|
|
rec := httptest.NewRecorder()
|
|
app.ServeHTTP(rec, req)
|
|
|
|
Expect(rec.Code).To(Equal(200), "response status code")
|
|
Expect(actualURL).To(Equal("http://example.com/myprefix/"), "base URL")
|
|
})
|
|
})
|
|
|
|
// Caddy's handle_path (and similar reverse-proxy directives) strips the
|
|
// matched prefix before forwarding upstream, so LocalAI receives the
|
|
// already-stripped path together with X-Forwarded-Prefix. In that case
|
|
// StripPathPrefix never stores _original_path, but BaseURL must still
|
|
// honor the header so that <base href> and asset URLs include the prefix.
|
|
Context("with X-Forwarded-Prefix header but pre-stripped path", func() {
|
|
It("should return base URL with prefix from header", func() {
|
|
app := echo.New()
|
|
actualURL := ""
|
|
|
|
routePath := "/app"
|
|
app.GET(routePath, func(c echo.Context) error {
|
|
actualURL = BaseURL(c)
|
|
return nil
|
|
})
|
|
|
|
req := httptest.NewRequest("GET", "/app", nil)
|
|
req.Header.Set("X-Forwarded-Prefix", "/localai")
|
|
rec := httptest.NewRecorder()
|
|
app.ServeHTTP(rec, req)
|
|
|
|
Expect(rec.Code).To(Equal(200), "response status code")
|
|
Expect(actualURL).To(Equal("http://example.com/localai/"), "base URL")
|
|
})
|
|
|
|
It("should normalize a prefix that already ends with a slash", func() {
|
|
app := echo.New()
|
|
actualURL := ""
|
|
|
|
routePath := "/app"
|
|
app.GET(routePath, func(c echo.Context) error {
|
|
actualURL = BaseURL(c)
|
|
return nil
|
|
})
|
|
|
|
req := httptest.NewRequest("GET", "/app", nil)
|
|
req.Header.Set("X-Forwarded-Prefix", "/localai/")
|
|
rec := httptest.NewRecorder()
|
|
app.ServeHTTP(rec, req)
|
|
|
|
Expect(rec.Code).To(Equal(200), "response status code")
|
|
Expect(actualURL).To(Equal("http://example.com/localai/"), "base URL")
|
|
})
|
|
})
|
|
|
|
// X-Forwarded-Prefix is attacker controllable on misconfigured proxy
|
|
// chains, and the value flows into the SPA HTML response (<base href>
|
|
// and asset URLs). BasePathPrefix must gate the header through
|
|
// SafeForwardedPrefix so values that turn the prefix into an open
|
|
// redirect or a protocol-relative URL are ignored and the base falls
|
|
// back to "/".
|
|
Context("with unsafe X-Forwarded-Prefix header", func() {
|
|
DescribeTable("falls back to / when the header is unsafe",
|
|
func(header string) {
|
|
app := echo.New()
|
|
actualURL := ""
|
|
|
|
app.GET("/app", func(c echo.Context) error {
|
|
actualURL = BaseURL(c)
|
|
return nil
|
|
})
|
|
|
|
req := httptest.NewRequest("GET", "/app", nil)
|
|
req.Header.Set("X-Forwarded-Prefix", header)
|
|
rec := httptest.NewRecorder()
|
|
app.ServeHTTP(rec, req)
|
|
|
|
Expect(rec.Code).To(Equal(200), "response status code")
|
|
Expect(actualURL).To(Equal("http://example.com/"), "base URL")
|
|
},
|
|
Entry("protocol-relative URL", "//evil.com"),
|
|
Entry("protocol-relative URL with path", "//evil.com/assets"),
|
|
Entry("backslash path", `/foo\bar`),
|
|
Entry("embedded NUL", "/foo\x00bar"),
|
|
Entry("CR injection", "/foo\rbar"),
|
|
Entry("LF injection", "/foo\nbar"),
|
|
Entry("missing leading slash", "evil"),
|
|
)
|
|
})
|
|
|
|
Context("scheme detection hardening", func() {
|
|
It("treats comma-separated X-Forwarded-Proto as https when first token is https", func() {
|
|
app := echo.New()
|
|
actualURL := ""
|
|
app.GET("/x", func(c echo.Context) error {
|
|
actualURL = BaseURL(c)
|
|
return nil
|
|
})
|
|
req := httptest.NewRequest("GET", "/x", nil)
|
|
req.Header.Set("X-Forwarded-Proto", "https,http")
|
|
rec := httptest.NewRecorder()
|
|
app.ServeHTTP(rec, req)
|
|
Expect(actualURL).To(Equal("https://example.com/"))
|
|
})
|
|
|
|
It("derives https from the RFC 7239 Forwarded proto directive", func() {
|
|
app := echo.New()
|
|
actualURL := ""
|
|
app.GET("/x", func(c echo.Context) error {
|
|
actualURL = BaseURL(c)
|
|
return nil
|
|
})
|
|
req := httptest.NewRequest("GET", "/x", nil)
|
|
req.Header.Set("Forwarded", "for=192.0.2.1;proto=https;host=proxy.example")
|
|
rec := httptest.NewRecorder()
|
|
app.ServeHTTP(rec, req)
|
|
Expect(actualURL).To(Equal("https://proxy.example/"))
|
|
})
|
|
|
|
It("prefers X-Forwarded-Host over the Forwarded host directive", func() {
|
|
app := echo.New()
|
|
actualURL := ""
|
|
app.GET("/x", func(c echo.Context) error {
|
|
actualURL = BaseURL(c)
|
|
return nil
|
|
})
|
|
req := httptest.NewRequest("GET", "/x", nil)
|
|
req.Header.Set("X-Forwarded-Host", "xfh.example")
|
|
req.Header.Set("Forwarded", "host=fwd.example;proto=https")
|
|
rec := httptest.NewRecorder()
|
|
app.ServeHTTP(rec, req)
|
|
Expect(actualURL).To(Equal("https://xfh.example/"))
|
|
})
|
|
})
|
|
|
|
Context("explicit external base URL override", func() {
|
|
It("uses the configured origin over conflicting forwarded headers", func() {
|
|
app := echo.New()
|
|
actualURL := ""
|
|
app.GET("/x", func(c echo.Context) error {
|
|
c.Set("_external_base_url", "https://192.168.0.13:34567")
|
|
actualURL = BaseURL(c)
|
|
return nil
|
|
})
|
|
req := httptest.NewRequest("GET", "/x", nil)
|
|
req.Header.Set("X-Forwarded-Proto", "http")
|
|
req.Header.Set("X-Forwarded-Host", "internal:8080")
|
|
rec := httptest.NewRecorder()
|
|
app.ServeHTTP(rec, req)
|
|
Expect(actualURL).To(Equal("https://192.168.0.13:34567/"))
|
|
})
|
|
|
|
It("combines the configured origin with a detected path prefix", func() {
|
|
app := echo.New()
|
|
actualURL := ""
|
|
app.GET("/hello", func(c echo.Context) error {
|
|
c.Set("_original_path", "/localai/hello")
|
|
c.Set("_external_base_url", "https://ext.example")
|
|
actualURL = BaseURL(c)
|
|
return nil
|
|
})
|
|
req := httptest.NewRequest("GET", "/hello", nil)
|
|
rec := httptest.NewRecorder()
|
|
app.ServeHTTP(rec, req)
|
|
Expect(actualURL).To(Equal("https://ext.example/localai/"))
|
|
})
|
|
|
|
It("ignores an empty override", func() {
|
|
app := echo.New()
|
|
actualURL := ""
|
|
app.GET("/x", func(c echo.Context) error {
|
|
c.Set("_external_base_url", "")
|
|
actualURL = BaseURL(c)
|
|
return nil
|
|
})
|
|
req := httptest.NewRequest("GET", "/x", nil)
|
|
rec := httptest.NewRecorder()
|
|
app.ServeHTTP(rec, req)
|
|
Expect(actualURL).To(Equal("http://example.com/"))
|
|
})
|
|
})
|
|
|
|
Context("parseForwarded helper", func() {
|
|
It("parses unquoted proto and host", func() {
|
|
proto, host := parseForwarded("for=192.0.2.1;proto=https;host=h.example")
|
|
Expect(proto).To(Equal("https"))
|
|
Expect(host).To(Equal("h.example"))
|
|
})
|
|
|
|
It("strips quotes around values", func() {
|
|
proto, host := parseForwarded(`proto="https";host="h.example"`)
|
|
Expect(proto).To(Equal("https"))
|
|
Expect(host).To(Equal("h.example"))
|
|
})
|
|
|
|
It("uses only the first element of a multi-element header", func() {
|
|
proto, host := parseForwarded("proto=https;host=first.example, proto=http;host=second.example")
|
|
Expect(proto).To(Equal("https"))
|
|
Expect(host).To(Equal("first.example"))
|
|
})
|
|
|
|
It("returns empty strings for an empty header", func() {
|
|
proto, host := parseForwarded("")
|
|
Expect(proto).To(BeEmpty())
|
|
Expect(host).To(BeEmpty())
|
|
})
|
|
|
|
It("skips directives without a value", func() {
|
|
proto, host := parseForwarded("proto;host=h.example")
|
|
Expect(proto).To(BeEmpty())
|
|
Expect(host).To(Equal("h.example"))
|
|
})
|
|
})
|
|
|
|
Context("firstToken helper", func() {
|
|
It("returns the whole trimmed string when there is no comma", func() {
|
|
Expect(firstToken(" https ")).To(Equal("https"))
|
|
})
|
|
|
|
It("returns the first trimmed token when there is a comma", func() {
|
|
Expect(firstToken("https , http")).To(Equal("https"))
|
|
})
|
|
})
|
|
})
|