package commands import ( "context" "fmt" "os" "os/signal" "strconv" "syscall" "time" "github.com/photoprism/photoprism/internal/config" "github.com/photoprism/photoprism/internal/server" "github.com/photoprism/photoprism/internal/service" "github.com/photoprism/photoprism/internal/workers" "github.com/photoprism/photoprism/pkg/fs" "github.com/sevlyar/go-daemon" "github.com/urfave/cli" ) // StartCommand is used to register the start cli command var StartCommand = cli.Command{ Name: "start", Aliases: []string{"up"}, Usage: "Starts web server", Flags: startFlags, Action: startAction, } var startFlags = []cli.Flag{ cli.BoolFlag{ Name: "detach-server, d", Usage: "detach from the console (daemon mode)", EnvVar: "PHOTOPRISM_DETACH_SERVER", }, cli.BoolFlag{ Name: "config, c", Usage: "show config", }, } // startAction start the web server and initializes the daemon func startAction(ctx *cli.Context) error { conf := config.NewConfig(ctx) service.SetConfig(conf) if err := conf.CreateDirectories(); err != nil { return err } if ctx.IsSet("config") { fmt.Printf("NAME VALUE\n") fmt.Printf("detach-server %t\n", conf.DetachServer()) fmt.Printf("tidb-host %s\n", conf.TidbServerHost()) fmt.Printf("tidb-port %d\n", conf.TidbServerPort()) fmt.Printf("tidb-password %s\n", conf.TidbServerPassword()) fmt.Printf("tidb-path %s\n", conf.TidbServerPath()) fmt.Printf("http-host %s\n", conf.HttpServerHost()) fmt.Printf("http-port %d\n", conf.HttpServerPort()) fmt.Printf("http-mode %s\n", conf.HttpServerMode()) return nil } // pass this context down the chain cctx, cancel := context.WithCancel(context.Background()) if conf.HttpServerPort() < 1 || conf.HttpServerPort() > 65535 { log.Fatal("server port must be a number between 1 and 65535") } if err := conf.CreateDirectories(); err != nil { log.Fatal(err) } if err := conf.Init(cctx); err != nil { log.Fatal(err) } // initialize the database conf.InitDb() // check if daemon is running, if not initialize the daemon dctx := new(daemon.Context) dctx.LogFileName = conf.LogFilename() dctx.PidFileName = conf.PIDFilename() dctx.Args = ctx.Args() if !daemon.WasReborn() && conf.DetachServer() { conf.Shutdown() cancel() if pid, ok := childAlreadyRunning(conf.PIDFilename()); ok { log.Infof("daemon already running with process id %v\n", pid) return nil } child, err := dctx.Reborn() if err != nil { log.Fatal(err) } if child != nil { if !fs.Overwrite(conf.PIDFilename(), []byte(strconv.Itoa(child.Pid))) { log.Fatalf("failed writing process id to \"%s\"", conf.PIDFilename()) } log.Infof("daemon started with process id %v\n", child.Pid) return nil } } if conf.ReadOnly() { log.Infof("read-only mode enabled") } // start web server go server.Start(cctx, conf) // start share & sync workers workers.Start(conf) // set up proper shutdown of daemon and web server quit := make(chan os.Signal) signal.Notify(quit, syscall.SIGINT, syscall.SIGTERM) <-quit // stop share & sync workers workers.Stop() log.Info("shutting down...") conf.Shutdown() cancel() err := dctx.Release() if err != nil { log.Error(err) } time.Sleep(3 * time.Second) return nil }