package main import ( "context" "fmt" "log/slog" "net/http" "os" "os/signal" "syscall" "github.com/spf13/cobra" "github.com/spf13/viper" "github.com/usememos/memos/server" "github.com/usememos/memos/server/profile" "github.com/usememos/memos/store" "github.com/usememos/memos/store/db" ) const ( greetingBanner = ` ███╗ ███╗███████╗███╗ ███╗ ██████╗ ███████╗ ████╗ ████║██╔════╝████╗ ████║██╔═══██╗██╔════╝ ██╔████╔██║█████╗ ██╔████╔██║██║ ██║███████╗ ██║╚██╔╝██║██╔══╝ ██║╚██╔╝██║██║ ██║╚════██║ ██║ ╚═╝ ██║███████╗██║ ╚═╝ ██║╚██████╔╝███████║ ╚═╝ ╚═╝╚══════╝╚═╝ ╚═╝ ╚═════╝ ╚══════╝ ` ) var ( mode string addr string port int data string driver string dsn string instanceProfile *profile.Profile rootCmd = &cobra.Command{ Use: "memos", Short: `An open source, lightweight note-taking service. Easily capture and share your great thoughts.`, Run: func(_ *cobra.Command, _ []string) { ctx, cancel := context.WithCancel(context.Background()) dbDriver, err := db.NewDBDriver(instanceProfile) if err != nil { cancel() slog.Error("failed to create db driver", err) return } if err := dbDriver.Migrate(ctx); err != nil { cancel() slog.Error("failed to migrate database", err) return } storeInstance := store.New(dbDriver, instanceProfile) if err := storeInstance.MigrateManually(ctx); err != nil { cancel() slog.Error("failed to migrate manually", err) return } s, err := server.NewServer(ctx, instanceProfile, storeInstance) if err != nil { cancel() slog.Error("failed to create server", err) return } c := make(chan os.Signal, 1) // Trigger graceful shutdown on SIGINT or SIGTERM. // The default signal sent by the `kill` command is SIGTERM, // which is taken as the graceful shutdown signal for many systems, eg., Kubernetes, Gunicorn. signal.Notify(c, os.Interrupt, syscall.SIGTERM) if err := s.Start(ctx); err != nil { if err != http.ErrServerClosed { slog.Error("failed to start server", err) cancel() } } printGreetings() go func() { <-c s.Shutdown(ctx) cancel() }() // Wait for CTRL-C. <-ctx.Done() }, } ) func Execute() error { return rootCmd.Execute() } func init() { cobra.OnInitialize(initConfig) rootCmd.PersistentFlags().StringVarP(&mode, "mode", "m", "demo", `mode of server, can be "prod" or "dev" or "demo"`) rootCmd.PersistentFlags().StringVarP(&addr, "addr", "a", "", "address of server") rootCmd.PersistentFlags().IntVarP(&port, "port", "p", 8081, "port of server") rootCmd.PersistentFlags().StringVarP(&data, "data", "d", "", "data directory") rootCmd.PersistentFlags().StringVarP(&driver, "driver", "", "", "database driver") rootCmd.PersistentFlags().StringVarP(&dsn, "dsn", "", "", "database source name(aka. DSN)") err := viper.BindPFlag("mode", rootCmd.PersistentFlags().Lookup("mode")) if err != nil { panic(err) } err = viper.BindPFlag("addr", rootCmd.PersistentFlags().Lookup("addr")) if err != nil { panic(err) } err = viper.BindPFlag("port", rootCmd.PersistentFlags().Lookup("port")) if err != nil { panic(err) } err = viper.BindPFlag("data", rootCmd.PersistentFlags().Lookup("data")) if err != nil { panic(err) } err = viper.BindPFlag("driver", rootCmd.PersistentFlags().Lookup("driver")) if err != nil { panic(err) } err = viper.BindPFlag("dsn", rootCmd.PersistentFlags().Lookup("dsn")) if err != nil { panic(err) } viper.SetDefault("mode", "demo") viper.SetDefault("driver", "sqlite") viper.SetDefault("addr", "") viper.SetDefault("port", 8081) viper.SetEnvPrefix("memos") } func initConfig() { viper.AutomaticEnv() var err error instanceProfile, err = profile.GetProfile() if err != nil { slog.Error("failed to get profile", err) return } fmt.Printf(`--- Server profile version: %s data: %s dsn: %s addr: %s port: %d mode: %s driver: %s --- `, instanceProfile.Version, instanceProfile.Data, instanceProfile.DSN, instanceProfile.Addr, instanceProfile.Port, instanceProfile.Mode, instanceProfile.Driver) } func printGreetings() { print(greetingBanner) if len(instanceProfile.Addr) == 0 { fmt.Printf("Version %s has been started on port %d\n", instanceProfile.Version, instanceProfile.Port) } else { fmt.Printf("Version %s has been started on address '%s' and port %d\n", instanceProfile.Version, instanceProfile.Addr, instanceProfile.Port) } fmt.Printf(`--- See more in: 👉Website: %s 👉GitHub: %s --- `, "https://usememos.com", "https://github.com/usememos/memos") } func main() { err := Execute() if err != nil { panic(err) } }