client/web: clean up assets handling

A #cleanup that moves all frontend asset handling into assets.go
(formerly dev.go), and stores a single assetsHandler field back
to web.Server that manages when to serve the dev vite proxy versus
static files itself.

Signed-off-by: Sonia Appasamy <sonia@tailscale.com>
This commit is contained in:
Sonia Appasamy 2023-09-08 15:30:07 -04:00 committed by Sonia Appasamy
parent 4a38d8d372
commit 1eadb2b608
3 changed files with 49 additions and 47 deletions

View File

@ -4,6 +4,8 @@
package web package web
import ( import (
"embed"
"io/fs"
"log" "log"
"net/http" "net/http"
"net/http/httputil" "net/http/httputil"
@ -12,11 +14,42 @@ import (
"os/exec" "os/exec"
"path/filepath" "path/filepath"
"strings" "strings"
"tailscale.com/util/must"
) )
// This contains all files needed to build the frontend assets.
// Because we assign this to the blank identifier, it does not actually embed the files.
// However, this does cause `go mod vendor` to include the files when vendoring the package.
// External packages that use the web client can `go mod vendor`, run `yarn build` to
// build the assets, then those asset bundles will be embedded.
//
//go:embed yarn.lock index.html *.js *.json src/*
var _ embed.FS
//go:embed build/*
var embeddedFS embed.FS
// staticfiles serves static files from the build directory.
var staticfiles http.Handler
func init() {
buildFiles := must.Get(fs.Sub(embeddedFS, "build"))
staticfiles = http.FileServer(http.FS(buildFiles))
}
func assetsHandler(devMode bool) (_ http.Handler, cleanup func()) {
if devMode {
// When in dev mode, proxy asset requests to the Vite dev server.
cleanup := startDevServer()
return devServerProxy(), cleanup
}
return staticfiles, nil
}
// startDevServer starts the JS dev server that does on-demand rebuilding // startDevServer starts the JS dev server that does on-demand rebuilding
// and serving of web client JS and CSS resources. // and serving of web client JS and CSS resources.
func (s *Server) startDevServer() (cleanup func()) { func startDevServer() (cleanup func()) {
root := gitRootDir() root := gitRootDir()
webClientPath := filepath.Join(root, "client", "web") webClientPath := filepath.Join(root, "client", "web")
@ -45,10 +78,8 @@ func (s *Server) startDevServer() (cleanup func()) {
} }
} }
func (s *Server) addProxyToDevServer() { // devServerProxy returns a reverse proxy to the vite dev server.
if !s.devMode { func devServerProxy() *httputil.ReverseProxy {
return // only using Vite proxy in dev mode
}
// We use Vite to develop on the web client. // We use Vite to develop on the web client.
// Vite starts up its own local server for development, // Vite starts up its own local server for development,
// which we proxy requests to from Server.ServeHTTP. // which we proxy requests to from Server.ServeHTTP.
@ -62,8 +93,9 @@ func (s *Server) addProxyToDevServer() {
w.Write([]byte("\n\nError: " + err.Error())) w.Write([]byte("\n\nError: " + err.Error()))
} }
viteTarget, _ := url.Parse("http://127.0.0.1:4000") viteTarget, _ := url.Parse("http://127.0.0.1:4000")
s.devProxy = httputil.NewSingleHostReverseProxy(viteTarget) devProxy := httputil.NewSingleHostReverseProxy(viteTarget)
s.devProxy.ErrorHandler = handleErr devProxy.ErrorHandler = handleErr
return devProxy
} }
func gitRootDir() string { func gitRootDir() string {

View File

@ -6,7 +6,7 @@
<meta name="viewport" content="width=device-width, initial-scale=1" /> <meta name="viewport" content="width=device-width, initial-scale=1" />
<link rel="shortcut icon" href="" /> <link rel="shortcut icon" href="" />
<script type="module" crossorigin src="./assets/index-f8beba53.js"></script> <script type="module" crossorigin src="./assets/index-4d1f45ea.js"></script>
<link rel="stylesheet" href="./assets/index-8612dca6.css"> <link rel="stylesheet" href="./assets/index-8612dca6.css">
</head> </head>
<body> <body>

View File

@ -7,14 +7,11 @@ package web
import ( import (
"context" "context"
"crypto/rand" "crypto/rand"
"embed"
"encoding/json" "encoding/json"
"fmt" "fmt"
"io" "io"
"io/fs"
"log" "log"
"net/http" "net/http"
"net/http/httputil"
"net/netip" "net/netip"
"os" "os"
"path/filepath" "path/filepath"
@ -31,35 +28,20 @@ import (
"tailscale.com/net/netutil" "tailscale.com/net/netutil"
"tailscale.com/tailcfg" "tailscale.com/tailcfg"
"tailscale.com/util/httpm" "tailscale.com/util/httpm"
"tailscale.com/util/must"
"tailscale.com/version/distro" "tailscale.com/version/distro"
) )
// This contains all files needed to build the frontend assets.
// Because we assign this to the blank identifier, it does not actually embed the files.
// However, this does cause `go mod vendor` to include the files when vendoring the package.
// External packages that use the web client can `go mod vendor`, run `yarn build` to
// build the assets, then those asset bundles will be embedded.
//
//go:embed yarn.lock index.html *.js *.json src/*
var _ embed.FS
//go:embed build/*
var embeddedFS embed.FS
// staticfiles serves static files from the build directory.
var staticfiles http.Handler
// Server is the backend server for a Tailscale web client. // Server is the backend server for a Tailscale web client.
type Server struct { type Server struct {
lc *tailscale.LocalClient lc *tailscale.LocalClient
devMode bool devMode bool
devProxy *httputil.ReverseProxy // only filled when devMode is on
cgiMode bool cgiMode bool
pathPrefix string pathPrefix string
apiHandler http.Handler // csrf-protected api handler
assetsHandler http.Handler // serves frontend assets
apiHandler http.Handler // serves api endpoints; csrf-protected
} }
// ServerOpts contains options for constructing a new Server. // ServerOpts contains options for constructing a new Server.
@ -89,11 +71,7 @@ func NewServer(ctx context.Context, opts ServerOpts) (s *Server, cleanup func())
cgiMode: opts.CGIMode, cgiMode: opts.CGIMode,
pathPrefix: opts.PathPrefix, pathPrefix: opts.PathPrefix,
} }
cleanup = func() {} s.assetsHandler, cleanup = assetsHandler(opts.DevMode)
if s.devMode {
cleanup = s.startDevServer()
s.addProxyToDevServer()
}
// Create handler for "/api" requests with CSRF protection. // Create handler for "/api" requests with CSRF protection.
// We don't require secure cookies, since the web client is regularly used // We don't require secure cookies, since the web client is regularly used
@ -107,11 +85,6 @@ func NewServer(ctx context.Context, opts ServerOpts) (s *Server, cleanup func())
return s, cleanup return s, cleanup
} }
func init() {
buildFiles := must.Get(fs.Sub(embeddedFS, "build"))
staticfiles = http.FileServer(http.FS(buildFiles))
}
// ServeHTTP processes all requests for the Tailscale web client. // ServeHTTP processes all requests for the Tailscale web client.
func (s *Server) ServeHTTP(w http.ResponseWriter, r *http.Request) { func (s *Server) ServeHTTP(w http.ResponseWriter, r *http.Request) {
handler := s.serve handler := s.serve
@ -151,14 +124,11 @@ func (s *Server) serve(w http.ResponseWriter, r *http.Request) {
// Pass API requests through to the API handler. // Pass API requests through to the API handler.
s.apiHandler.ServeHTTP(w, r) s.apiHandler.ServeHTTP(w, r)
return return
case s.devMode:
// When in dev mode, proxy non-api requests to the Vite dev server.
s.devProxy.ServeHTTP(w, r)
return
default: default:
// Otherwise, serve static files from the embedded filesystem. if !s.devMode {
s.lc.IncrementCounter(context.Background(), "web_client_page_load", 1) s.lc.IncrementCounter(context.Background(), "web_client_page_load", 1)
staticfiles.ServeHTTP(w, r) }
s.assetsHandler.ServeHTTP(w, r)
return return
} }
} }