app.go 3.9 KB

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