mirror of
https://github.com/mudler/LocalAI.git
synced 2026-05-17 21:21:23 -04:00
fix(http): honor X-Forwarded-Prefix when proxy strips the prefix (#9614)
* fix(http): honor X-Forwarded-Prefix when proxy strips the prefix Closes #9145. Two related issues kept the React UI from loading when a reverse proxy rewrites a sub-path with prefix-stripping (e.g. Caddy `handle_path`): 1. `BaseURL` only computed a prefix from the path StripPathPrefix had removed, so when the proxy strips the prefix before forwarding, the request arrives without it and the base URL was returned without a prefix. Extract a `BasePathPrefix` helper and add an `X-Forwarded-Prefix` header fallback so the prefix is recovered. 2. `<base href>` only changes how relative URLs resolve; the build emits path-absolute references like `/assets/...` and `/favicon.svg`, which still resolve against the origin and bypass the proxy prefix. Rewrite those references in the served `index.html` so the browser requests them through the proxy. Adds unit coverage for `BaseURL` with a pre-stripped path and an end-to-end test for the proxy-stripped scenario. Assisted-by: Claude:claude-opus-4-7 * fix(http): gate X-Forwarded-Prefix through SafeForwardedPrefix in BasePathPrefix BasePathPrefix consumed X-Forwarded-Prefix directly, so a value the codebase elsewhere rejects (e.g. "//evil.com") slipped through and was interpolated into the SPA index.html — both into the path-absolute asset URL rewrite in serveIndex (turning "/assets/..." into "//evil.com/assets/...", a protocol-relative URL that loads JS from a foreign origin) and into <base href>. Route the header through the existing SafeForwardedPrefix validator that StripPathPrefix and prefixRedirect already use, and HTML-escape the prefix before injecting it into the asset rewrite as defense in depth against attribute breakout. Tests cover //evil.com, backslashes, control chars, CR/LF and a missing leading slash; the integration test asserts an unsafe prefix can't poison asset URLs. Signed-off-by: Ettore Di Giacinto <mudler@localai.io> Assisted-by: claude-code:claude-opus-4-7-1m [Read] [Edit] [Bash] --------- Signed-off-by: Ettore Di Giacinto <mudler@localai.io> Co-authored-by: Ettore Di Giacinto <mudler@localai.io>
This commit is contained in:
@@ -6,20 +6,55 @@ import (
|
||||
"github.com/labstack/echo/v4"
|
||||
)
|
||||
|
||||
// BasePathPrefix returns the URL path prefix that the request was reached
|
||||
// under (e.g. "/myprefix/"). It always returns a value that starts and ends
|
||||
// with `/`, defaulting to "/" when the app is not behind a path prefix.
|
||||
//
|
||||
// It first looks at the path StripPathPrefix removed (when the proxy forwards
|
||||
// the prefix in the URL), then falls back to the X-Forwarded-Prefix header
|
||||
// (when the proxy strips the prefix before forwarding, e.g. Caddy's
|
||||
// handle_path).
|
||||
//
|
||||
// The header fallback is gated through SafeForwardedPrefix because the value
|
||||
// flows into the SPA HTML response (both <base href> and the path-absolute
|
||||
// asset URL rewrite in serveIndex). X-Forwarded-Prefix is attacker
|
||||
// controllable on misconfigured proxy chains; without that gate a value like
|
||||
// "//evil.com" turns the asset rewrite into a protocol-relative URL that
|
||||
// loads JS from a foreign origin.
|
||||
func BasePathPrefix(c echo.Context) string {
|
||||
path := c.Path()
|
||||
origPath := c.Request().URL.Path
|
||||
|
||||
if storedPath, ok := c.Get("_original_path").(string); ok && storedPath != "" {
|
||||
origPath = storedPath
|
||||
}
|
||||
|
||||
if path != origPath && strings.HasSuffix(origPath, path) && len(path) > 0 {
|
||||
prefixLen := len(origPath) - len(path)
|
||||
if prefixLen > 0 {
|
||||
pathPrefix := origPath[:prefixLen]
|
||||
if !strings.HasSuffix(pathPrefix, "/") {
|
||||
pathPrefix += "/"
|
||||
}
|
||||
return pathPrefix
|
||||
}
|
||||
}
|
||||
|
||||
if validated, ok := SafeForwardedPrefix(c.Request().Header.Get("X-Forwarded-Prefix")); ok {
|
||||
if !strings.HasSuffix(validated, "/") {
|
||||
validated += "/"
|
||||
}
|
||||
return validated
|
||||
}
|
||||
|
||||
return "/"
|
||||
}
|
||||
|
||||
// BaseURL returns the base URL for the given HTTP request context.
|
||||
// It takes into account that the app may be exposed by a reverse-proxy under a different protocol, host and path.
|
||||
// The returned URL is guaranteed to end with `/`.
|
||||
// The method should be used in conjunction with the StripPathPrefix middleware.
|
||||
func BaseURL(c echo.Context) string {
|
||||
path := c.Path()
|
||||
origPath := c.Request().URL.Path
|
||||
|
||||
// Check if StripPathPrefix middleware stored the original path
|
||||
if storedPath, ok := c.Get("_original_path").(string); ok && storedPath != "" {
|
||||
origPath = storedPath
|
||||
}
|
||||
|
||||
// Check X-Forwarded-Proto for scheme
|
||||
scheme := "http"
|
||||
if c.Request().Header.Get("X-Forwarded-Proto") == "https" {
|
||||
scheme = "https"
|
||||
@@ -27,22 +62,10 @@ func BaseURL(c echo.Context) string {
|
||||
scheme = "https"
|
||||
}
|
||||
|
||||
// Check X-Forwarded-Host for host
|
||||
host := c.Request().Host
|
||||
if forwardedHost := c.Request().Header.Get("X-Forwarded-Host"); forwardedHost != "" {
|
||||
host = forwardedHost
|
||||
}
|
||||
|
||||
if path != origPath && strings.HasSuffix(origPath, path) && len(path) > 0 {
|
||||
prefixLen := len(origPath) - len(path)
|
||||
if prefixLen > 0 && prefixLen <= len(origPath) {
|
||||
pathPrefix := origPath[:prefixLen]
|
||||
if !strings.HasSuffix(pathPrefix, "/") {
|
||||
pathPrefix += "/"
|
||||
}
|
||||
return scheme + "://" + host + pathPrefix
|
||||
}
|
||||
}
|
||||
|
||||
return scheme + "://" + host + "/"
|
||||
return scheme + "://" + host + BasePathPrefix(c)
|
||||
}
|
||||
|
||||
@@ -55,4 +55,84 @@ var _ = Describe("BaseURL", func() {
|
||||
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"),
|
||||
)
|
||||
})
|
||||
})
|
||||
|
||||
Reference in New Issue
Block a user