package service import ( "fmt" "html/template" "io" "net/http" "net/url" assetfs "github.com/elazarl/go-bindata-assetfs" "github.com/fleetdm/fleet/v4/server/bindata" "github.com/fleetdm/fleet/v4/server/fleet" "github.com/go-kit/log" ) func newBinaryFileSystem(root string) *assetfs.AssetFS { return &assetfs.AssetFS{ Asset: bindata.Asset, AssetDir: bindata.AssetDir, AssetInfo: bindata.AssetInfo, Prefix: root, } } func ServeFrontend(urlPrefix string, sandbox bool, logger log.Logger) http.Handler { herr := func(w http.ResponseWriter, err string) { logger.Log("err", err) http.Error(w, err, http.StatusInternalServerError) } return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { writeBrowserSecurityHeaders(w) // The following check is to prevent a misconfigured osquery from submitting // data to the root endpoint (the osquery remote API uses POST for all its endpoints). // See https://github.com/fleetdm/fleet/issues/16182. if r.Method == "POST" { http.Error(w, http.StatusText(http.StatusMethodNotAllowed), http.StatusMethodNotAllowed) return } fs := newBinaryFileSystem("/frontend") file, err := fs.Open("templates/react.tmpl") if err != nil { herr(w, "load react template: "+err.Error()) return } data, err := io.ReadAll(file) if err != nil { herr(w, "read bindata file: "+err.Error()) return } t, err := template.New("react").Parse(string(data)) if err != nil { herr(w, "create react template: "+err.Error()) return } serverType := "on-premise" if sandbox { serverType = "sandbox" } if err := t.Execute(w, struct { URLPrefix string ServerType string }{ URLPrefix: urlPrefix, ServerType: serverType, }); err != nil { herr(w, "execute react template: "+err.Error()) return } }) } func ServeEndUserEnrollOTA(svc fleet.Service, urlPrefix string, logger log.Logger) http.Handler { herr := func(w http.ResponseWriter, err string) { logger.Log("err", err) http.Error(w, err, http.StatusInternalServerError) } return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { writeBrowserSecurityHeaders(w) setupRequired, err := svc.SetupRequired(r.Context()) if err != nil { herr(w, "setup required err: "+err.Error()) return } if setupRequired { herr(w, "fleet instance not setup") return } fs := newBinaryFileSystem("/frontend") file, err := fs.Open("templates/enroll-ota.html") if err != nil { herr(w, "load enroll ota template: "+err.Error()) return } data, err := io.ReadAll(file) if err != nil { herr(w, "read bindata file: "+err.Error()) return } t, err := template.New("enroll-ota").Parse(string(data)) if err != nil { herr(w, "create react template: "+err.Error()) return } enrollURL, err := generateEnrollOTAURL(urlPrefix, r.URL.Query().Get("enroll_secret")) if err != nil { herr(w, "generate enroll ota url: "+err.Error()) return } if err := t.Execute(w, struct { EnrollURL string URLPrefix string }{ URLPrefix: urlPrefix, EnrollURL: enrollURL, }); err != nil { herr(w, "execute react template: "+err.Error()) return } }) } func generateEnrollOTAURL(fleetURL string, enrollSecret string) (string, error) { path, err := url.JoinPath(fleetURL, "/api/v1/fleet/enrollment_profiles/ota") if err != nil { return "", fmt.Errorf("creating path for end user ota enrollment url: %w", err) } enrollURL, err := url.Parse(path) if err != nil { return "", fmt.Errorf("parsing end user ota enrollment url: %w", err) } q := enrollURL.Query() q.Set("enroll_secret", enrollSecret) enrollURL.RawQuery = q.Encode() return enrollURL.String(), nil } func ServeStaticAssets(path string) http.Handler { return http.StripPrefix(path, http.FileServer(newBinaryFileSystem("/assets"))) }