app.go 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  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. // UI state
  36. filepickerOpen bool
  37. }
  38. func New(ctx context.Context, conn *sql.DB) (*App, error) {
  39. err := logging.InitService(conn)
  40. if err != nil {
  41. slog.Error("Failed to initialize logging service", "error", err)
  42. return nil, err
  43. }
  44. err = session.InitService(conn)
  45. if err != nil {
  46. slog.Error("Failed to initialize session service", "error", err)
  47. return nil, err
  48. }
  49. err = message.InitService(conn)
  50. if err != nil {
  51. slog.Error("Failed to initialize message service", "error", err)
  52. return nil, err
  53. }
  54. err = history.InitService(conn)
  55. if err != nil {
  56. slog.Error("Failed to initialize history service", "error", err)
  57. return nil, err
  58. }
  59. err = permission.InitService()
  60. if err != nil {
  61. slog.Error("Failed to initialize permission service", "error", err)
  62. return nil, err
  63. }
  64. err = status.InitService()
  65. if err != nil {
  66. slog.Error("Failed to initialize status service", "error", err)
  67. return nil, err
  68. }
  69. fileutil.Init()
  70. app := &App{
  71. CurrentSession: &session.Session{},
  72. Logs: logging.GetService(),
  73. Sessions: session.GetService(),
  74. Messages: message.GetService(),
  75. History: history.GetService(),
  76. Permissions: permission.GetService(),
  77. Status: status.GetService(),
  78. LSPClients: make(map[string]*lsp.Client),
  79. }
  80. // Initialize theme based on configuration
  81. app.initTheme()
  82. // Initialize LSP clients in the background
  83. go app.initLSPClients(ctx)
  84. app.PrimaryAgent, err = agent.NewAgent(
  85. config.AgentPrimary,
  86. app.Sessions,
  87. app.Messages,
  88. agent.PrimaryAgentTools(
  89. app.Permissions,
  90. app.Sessions,
  91. app.Messages,
  92. app.History,
  93. app.LSPClients,
  94. ),
  95. )
  96. if err != nil {
  97. slog.Error("Failed to create primary agent", "error", err)
  98. return nil, err
  99. }
  100. return app, nil
  101. }
  102. // initTheme sets the application theme based on the configuration
  103. func (app *App) initTheme() {
  104. cfg := config.Get()
  105. if cfg == nil || cfg.TUI.Theme == "" {
  106. return // Use default theme
  107. }
  108. // Try to set the theme from config
  109. err := theme.SetTheme(cfg.TUI.Theme)
  110. if err != nil {
  111. slog.Warn("Failed to set theme from config, using default theme", "theme", cfg.TUI.Theme, "error", err)
  112. } else {
  113. slog.Debug("Set theme from config", "theme", cfg.TUI.Theme)
  114. }
  115. }
  116. // IsFilepickerOpen returns whether the filepicker is currently open
  117. func (app *App) IsFilepickerOpen() bool {
  118. return app.filepickerOpen
  119. }
  120. // SetFilepickerOpen sets the state of the filepicker
  121. func (app *App) SetFilepickerOpen(open bool) {
  122. app.filepickerOpen = open
  123. }
  124. // Shutdown performs a clean shutdown of the application
  125. func (app *App) Shutdown() {
  126. // Cancel all watcher goroutines
  127. app.cancelFuncsMutex.Lock()
  128. for _, cancel := range app.watcherCancelFuncs {
  129. cancel()
  130. }
  131. app.cancelFuncsMutex.Unlock()
  132. app.watcherWG.Wait()
  133. // Perform additional cleanup for LSP clients
  134. app.clientsMutex.RLock()
  135. clients := make(map[string]*lsp.Client, len(app.LSPClients))
  136. maps.Copy(clients, app.LSPClients)
  137. app.clientsMutex.RUnlock()
  138. for name, client := range clients {
  139. shutdownCtx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
  140. if err := client.Shutdown(shutdownCtx); err != nil {
  141. slog.Error("Failed to shutdown LSP client", "name", name, "error", err)
  142. }
  143. cancel()
  144. }
  145. }