server.go 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899
  1. package cmd
  2. import (
  3. "context"
  4. "errors"
  5. "fmt"
  6. "log/slog"
  7. "os"
  8. "os/signal"
  9. "path/filepath"
  10. "time"
  11. "github.com/charmbracelet/crush/internal/config"
  12. crushlog "github.com/charmbracelet/crush/internal/log"
  13. "github.com/charmbracelet/crush/internal/server"
  14. "github.com/charmbracelet/x/term"
  15. "github.com/spf13/cobra"
  16. )
  17. var serverHost string
  18. func init() {
  19. serverCmd.Flags().StringVarP(&serverHost, "host", "H", server.DefaultHost(), "Server host (TCP or Unix socket)")
  20. rootCmd.AddCommand(serverCmd)
  21. }
  22. var serverCmd = &cobra.Command{
  23. Use: "server",
  24. Short: "Start the Crush server",
  25. RunE: func(cmd *cobra.Command, _ []string) error {
  26. dataDir, err := cmd.Flags().GetString("data-dir")
  27. if err != nil {
  28. return fmt.Errorf("failed to get data directory: %v", err)
  29. }
  30. debug, err := cmd.Flags().GetBool("debug")
  31. if err != nil {
  32. return fmt.Errorf("failed to get debug flag: %v", err)
  33. }
  34. cfg, err := config.Load(config.GlobalWorkspaceDir(), dataDir, debug)
  35. if err != nil {
  36. return fmt.Errorf("failed to load configuration: %v", err)
  37. }
  38. logFile := filepath.Join(config.GlobalCacheDir(), "server-"+safeNameRegexp.ReplaceAllString(serverHost, "_"), "crush.log")
  39. if term.IsTerminal(os.Stderr.Fd()) {
  40. crushlog.Setup(logFile, debug, os.Stderr)
  41. } else {
  42. crushlog.Setup(logFile, debug)
  43. }
  44. hostURL, err := server.ParseHostURL(serverHost)
  45. if err != nil {
  46. return fmt.Errorf("invalid server host: %v", err)
  47. }
  48. srv := server.NewServer(cfg, hostURL.Scheme, hostURL.Host)
  49. srv.SetLogger(slog.Default())
  50. slog.Info("Starting Crush server...", "addr", serverHost)
  51. errch := make(chan error, 1)
  52. sigch := make(chan os.Signal, 1)
  53. sigs := []os.Signal{os.Interrupt}
  54. sigs = append(sigs, addSignals(sigs)...)
  55. signal.Notify(sigch, sigs...)
  56. go func() {
  57. errch <- srv.ListenAndServe()
  58. }()
  59. select {
  60. case <-sigch:
  61. slog.Info("Received interrupt signal...")
  62. case err = <-errch:
  63. if err != nil && !errors.Is(err, server.ErrServerClosed) {
  64. _ = srv.Close()
  65. slog.Error("Server error", "error", err)
  66. return fmt.Errorf("server error: %v", err)
  67. }
  68. }
  69. if errors.Is(err, server.ErrServerClosed) {
  70. return nil
  71. }
  72. ctx, cancel := context.WithTimeout(cmd.Context(), 5*time.Second)
  73. defer cancel()
  74. slog.Info("Shutting down...")
  75. if err := srv.Shutdown(ctx); err != nil {
  76. slog.Error("Failed to shutdown server", "error", err)
  77. return fmt.Errorf("failed to shutdown server: %v", err)
  78. }
  79. return nil
  80. },
  81. }