package cli import ( "bytes" "context" "crypto/rand" "crypto/subtle" "encoding/hex" "fmt" "html" "io" "io/ioutil" "net/http" "net/url" "os" "strings" "contrib.go.opencensus.io/exporter/prometheus" "github.com/pkg/errors" prom "github.com/prometheus/client_golang/prometheus" htpasswd "github.com/tg123/go-htpasswd" "github.com/kopia/kopia/internal/clock" "github.com/kopia/kopia/internal/server" "github.com/kopia/kopia/repo" ) var ( serverStartCommand = serverCommands.Command("start", "Start Kopia server").Default() serverStartHTMLPath = serverStartCommand.Flag("html", "Server the provided HTML at the root URL").ExistingDir() serverStartUI = serverStartCommand.Flag("ui", "Start the server with HTML UI").Default("true").Bool() serverStartRefreshInterval = serverStartCommand.Flag("refresh-interval", "Frequency for refreshing repository status").Default("10s").Duration() serverStartRandomPassword = serverStartCommand.Flag("random-password", "Generate random password and print to stderr").Hidden().Bool() serverStartAutoShutdown = serverStartCommand.Flag("auto-shutdown", "Auto shutdown the server if API requests not received within given time").Hidden().Duration() serverStartHtpasswdFile = serverStartCommand.Flag("htpasswd-file", "Path to htpasswd file that contains allowed user@hostname entries").Hidden().ExistingFile() ) func init() { setupConnectOptions(serverStartCommand) serverStartCommand.Action(optionalRepositoryAction(runServer)) } func runServer(ctx context.Context, rep repo.Repository) error { srv, err := server.New(ctx, server.Options{ ConfigFile: repositoryConfigFileName(), ConnectOptions: connectOptions(), RefreshInterval: *serverStartRefreshInterval, }) if err != nil { return errors.Wrap(err, "unable to initialize server") } maybeAutoUpgradeRepository(ctx, rep) if err = srv.SetRepository(ctx, rep); err != nil { return errors.Wrap(err, "error connecting to repository") } mux := http.NewServeMux() mux.Handle("/api/", srv.APIHandlers()) if *serverStartHTMLPath != "" { fileServer := serveIndexFileForKnownUIRoutes(http.Dir(*serverStartHTMLPath)) mux.Handle("/", fileServer) } else if *serverStartUI { mux.Handle("/", serveIndexFileForKnownUIRoutes(server.AssetFile())) } httpServer := &http.Server{Addr: stripProtocol(*serverAddress)} srv.OnShutdown = httpServer.Shutdown onCtrlC(func() { log(ctx).Infof("Shutting down...") if err = httpServer.Shutdown(ctx); err != nil { log(ctx).Warningf("unable to shut down: %v", err) } }) mux, err = requireCredentials(mux) if err != nil { return errors.Wrap(err, "unable to setup credentials") } // init prometheus after adding interceptors that require credentials, so that this // handler can be called without auth if err = initPrometheus(mux); err != nil { return errors.Wrap(err, "error initializing Prometheus") } var handler http.Handler = mux if as := *serverStartAutoShutdown; as > 0 { log(ctx).Infof("starting a watchdog to stop the server if there's no activity for %v", as) handler = startServerWatchdog(handler, as, func() { if serr := httpServer.Shutdown(ctx); err != nil { log(ctx).Warningf("unable to stop the server: %v", serr) } }) } httpServer.Handler = handler err = startServerWithOptionalTLS(ctx, httpServer) if !errors.Is(err, http.ErrServerClosed) { return err } return srv.SetRepository(ctx, nil) } func initPrometheus(mux *http.ServeMux) error { reg := prom.NewRegistry() if err := reg.Register(prom.NewProcessCollector(prom.ProcessCollectorOpts{})); err != nil { return errors.Wrap(err, "error registering process collector") } if err := reg.Register(prom.NewGoCollector()); err != nil { return errors.Wrap(err, "error registering go collector") } pe, err := prometheus.NewExporter(prometheus.Options{ Registry: reg, }) if err != nil { return errors.Wrap(err, "unable to initialize prometheus exporter") } mux.Handle("/metrics", pe) return nil } func stripProtocol(addr string) string { return strings.TrimPrefix(strings.TrimPrefix(addr, "https://"), "http://") } func isKnownUIRoute(path string) bool { return strings.HasPrefix(path, "/snapshots") || strings.HasPrefix(path, "/policies") || strings.HasPrefix(path, "/repo") } func patchIndexBytes(b []byte) []byte { if prefix := os.Getenv("KOPIA_UI_TITLE_PREFIX"); prefix != "" { b = bytes.ReplaceAll(b, []byte("