client/web: server /index.html on 404 requests

In production, the asset handler is receiving requests for pages like
/details, which results in a 404. Instead, if we know the requested file
does not exist, serve the main index page and let wouter route it
appropriately on the frontend.

Updates tailscale/corp/#14335

Signed-off-by: Will Norris <will@tailscale.com>
This commit is contained in:
Will Norris 2023-11-14 15:00:46 -08:00 committed by Will Norris
parent 74947ce459
commit fb984c2b71
1 changed files with 14 additions and 1 deletions

View File

@ -4,6 +4,7 @@
package web package web
import ( import (
"io/fs"
"log" "log"
"net/http" "net/http"
"net/http/httputil" "net/http/httputil"
@ -22,7 +23,19 @@ func assetsHandler(devMode bool) (_ http.Handler, cleanup func()) {
cleanup := startDevServer() cleanup := startDevServer()
return devServerProxy(), cleanup return devServerProxy(), cleanup
} }
return http.FileServer(http.FS(prebuilt.FS())), nil
fsys := prebuilt.FS()
fileserver := http.FileServer(http.FS(fsys))
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
_, err := fs.Stat(fsys, strings.TrimPrefix(r.URL.Path, "/"))
if os.IsNotExist(err) {
// rewrite request to just fetch /index.html and let
// the frontend router handle it.
r = r.Clone(r.Context())
r.URL.Path = "/"
}
fileserver.ServeHTTP(w, r)
}), nil
} }
// startDevServer starts the JS dev server that does on-demand rebuilding // startDevServer starts the JS dev server that does on-demand rebuilding