app.go 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  1. package app
  2. import (
  3. "context"
  4. "database/sql"
  5. "maps"
  6. "sync"
  7. "time"
  8. "log/slog"
  9. "github.com/opencode-ai/opencode/internal/config"
  10. "github.com/opencode-ai/opencode/internal/history"
  11. "github.com/opencode-ai/opencode/internal/llm/agent"
  12. "github.com/opencode-ai/opencode/internal/logging"
  13. "github.com/opencode-ai/opencode/internal/lsp"
  14. "github.com/opencode-ai/opencode/internal/message"
  15. "github.com/opencode-ai/opencode/internal/permission"
  16. "github.com/opencode-ai/opencode/internal/session"
  17. "github.com/opencode-ai/opencode/internal/status"
  18. "github.com/opencode-ai/opencode/internal/tui/theme"
  19. )
  20. type App struct {
  21. Logs logging.Service
  22. Sessions session.Service
  23. Messages message.Service
  24. History history.Service
  25. Permissions permission.Service
  26. Status status.Service
  27. CoderAgent agent.Service
  28. LSPClients map[string]*lsp.Client
  29. clientsMutex sync.RWMutex
  30. watcherCancelFuncs []context.CancelFunc
  31. cancelFuncsMutex sync.Mutex
  32. watcherWG sync.WaitGroup
  33. }
  34. func New(ctx context.Context, conn *sql.DB) (*App, error) {
  35. err := logging.InitService(conn)
  36. if err != nil {
  37. slog.Error("Failed to initialize logging service", "error", err)
  38. return nil, err
  39. }
  40. err = session.InitService(conn)
  41. if err != nil {
  42. slog.Error("Failed to initialize session service", "error", err)
  43. return nil, err
  44. }
  45. err = message.InitService(conn)
  46. if err != nil {
  47. slog.Error("Failed to initialize message service", "error", err)
  48. return nil, err
  49. }
  50. err = history.InitService(conn)
  51. if err != nil {
  52. slog.Error("Failed to initialize history service", "error", err)
  53. return nil, err
  54. }
  55. err = permission.InitService()
  56. if err != nil {
  57. slog.Error("Failed to initialize permission service", "error", err)
  58. return nil, err
  59. }
  60. err = status.InitService()
  61. if err != nil {
  62. slog.Error("Failed to initialize status service", "error", err)
  63. return nil, err
  64. }
  65. app := &App{
  66. Logs: logging.GetService(),
  67. Sessions: session.GetService(),
  68. Messages: message.GetService(),
  69. History: history.GetService(),
  70. Permissions: permission.GetService(),
  71. Status: status.GetService(),
  72. LSPClients: make(map[string]*lsp.Client),
  73. }
  74. // Initialize theme based on configuration
  75. app.initTheme()
  76. // Initialize LSP clients in the background
  77. go app.initLSPClients(ctx)
  78. app.CoderAgent, err = agent.NewAgent(
  79. config.AgentCoder,
  80. app.Sessions,
  81. app.Messages,
  82. agent.CoderAgentTools(
  83. app.Permissions,
  84. app.Sessions,
  85. app.Messages,
  86. app.History,
  87. app.LSPClients,
  88. ),
  89. )
  90. if err != nil {
  91. slog.Error("Failed to create coder agent", err)
  92. return nil, err
  93. }
  94. return app, nil
  95. }
  96. // initTheme sets the application theme based on the configuration
  97. func (app *App) initTheme() {
  98. cfg := config.Get()
  99. if cfg == nil || cfg.TUI.Theme == "" {
  100. return // Use default theme
  101. }
  102. // Try to set the theme from config
  103. err := theme.SetTheme(cfg.TUI.Theme)
  104. if err != nil {
  105. slog.Warn("Failed to set theme from config, using default theme", "theme", cfg.TUI.Theme, "error", err)
  106. } else {
  107. slog.Debug("Set theme from config", "theme", cfg.TUI.Theme)
  108. }
  109. }
  110. // Shutdown performs a clean shutdown of the application
  111. func (app *App) Shutdown() {
  112. // Cancel all watcher goroutines
  113. app.cancelFuncsMutex.Lock()
  114. for _, cancel := range app.watcherCancelFuncs {
  115. cancel()
  116. }
  117. app.cancelFuncsMutex.Unlock()
  118. app.watcherWG.Wait()
  119. // Perform additional cleanup for LSP clients
  120. app.clientsMutex.RLock()
  121. clients := make(map[string]*lsp.Client, len(app.LSPClients))
  122. maps.Copy(clients, app.LSPClients)
  123. app.clientsMutex.RUnlock()
  124. for name, client := range clients {
  125. shutdownCtx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
  126. if err := client.Shutdown(shutdownCtx); err != nil {
  127. slog.Error("Failed to shutdown LSP client", "name", name, "error", err)
  128. }
  129. cancel()
  130. }
  131. }