app.go 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890
  1. package app
  2. import (
  3. "context"
  4. "fmt"
  5. "os"
  6. "path/filepath"
  7. "slices"
  8. "strings"
  9. "log/slog"
  10. tea "github.com/charmbracelet/bubbletea/v2"
  11. "github.com/sst/opencode-sdk-go"
  12. "github.com/sst/opencode/internal/clipboard"
  13. "github.com/sst/opencode/internal/commands"
  14. "github.com/sst/opencode/internal/components/toast"
  15. "github.com/sst/opencode/internal/id"
  16. "github.com/sst/opencode/internal/styles"
  17. "github.com/sst/opencode/internal/theme"
  18. "github.com/sst/opencode/internal/util"
  19. )
  20. type Message struct {
  21. Info opencode.MessageUnion
  22. Parts []opencode.PartUnion
  23. }
  24. type App struct {
  25. Info opencode.App
  26. Agents []opencode.Agent
  27. Providers []opencode.Provider
  28. Version string
  29. StatePath string
  30. Config *opencode.Config
  31. Client *opencode.Client
  32. State *State
  33. AgentIndex int
  34. Provider *opencode.Provider
  35. Model *opencode.Model
  36. Session *opencode.Session
  37. Messages []Message
  38. Permissions []opencode.Permission
  39. CurrentPermission opencode.Permission
  40. Commands commands.CommandRegistry
  41. InitialModel *string
  42. InitialPrompt *string
  43. InitialAgent *string
  44. InitialSession *string
  45. compactCancel context.CancelFunc
  46. IsLeaderSequence bool
  47. IsBashMode bool
  48. ScrollSpeed int
  49. }
  50. func (a *App) Agent() *opencode.Agent {
  51. return &a.Agents[a.AgentIndex]
  52. }
  53. type SessionCreatedMsg = struct {
  54. Session *opencode.Session
  55. }
  56. type SessionSelectedMsg = *opencode.Session
  57. type MessageRevertedMsg struct {
  58. Session opencode.Session
  59. Message Message
  60. }
  61. type SessionUnrevertedMsg struct {
  62. Session opencode.Session
  63. }
  64. type SessionLoadedMsg struct{}
  65. type ModelSelectedMsg struct {
  66. Provider opencode.Provider
  67. Model opencode.Model
  68. }
  69. type AgentSelectedMsg struct {
  70. AgentName string
  71. }
  72. type SessionClearedMsg struct{}
  73. type CompactSessionMsg struct{}
  74. type SendPrompt = Prompt
  75. type SendShell = struct {
  76. Command string
  77. }
  78. type SetEditorContentMsg struct {
  79. Text string
  80. }
  81. type FileRenderedMsg struct {
  82. FilePath string
  83. }
  84. type PermissionRespondedToMsg struct {
  85. Response opencode.SessionPermissionRespondParamsResponse
  86. }
  87. func New(
  88. ctx context.Context,
  89. version string,
  90. appInfo opencode.App,
  91. agents []opencode.Agent,
  92. httpClient *opencode.Client,
  93. initialModel *string,
  94. initialPrompt *string,
  95. initialAgent *string,
  96. initialSession *string,
  97. ) (*App, error) {
  98. util.RootPath = appInfo.Path.Root
  99. util.CwdPath = appInfo.Path.Cwd
  100. configInfo, err := httpClient.Config.Get(ctx)
  101. if err != nil {
  102. return nil, err
  103. }
  104. if configInfo.Keybinds.Leader == "" {
  105. configInfo.Keybinds.Leader = "ctrl+x"
  106. }
  107. appStatePath := filepath.Join(appInfo.Path.State, "tui")
  108. appState, err := LoadState(appStatePath)
  109. if err != nil {
  110. appState = NewState()
  111. SaveState(appStatePath, appState)
  112. }
  113. if appState.AgentModel == nil {
  114. appState.AgentModel = make(map[string]AgentModel)
  115. }
  116. if configInfo.Theme != "" {
  117. appState.Theme = configInfo.Theme
  118. }
  119. themeEnv := os.Getenv("OPENCODE_THEME")
  120. if themeEnv != "" {
  121. appState.Theme = themeEnv
  122. }
  123. agentIndex := slices.IndexFunc(agents, func(a opencode.Agent) bool {
  124. return a.Mode != "subagent"
  125. })
  126. var agent *opencode.Agent
  127. modeName := "build"
  128. if appState.Agent != "" {
  129. modeName = appState.Agent
  130. }
  131. if initialAgent != nil && *initialAgent != "" {
  132. modeName = *initialAgent
  133. }
  134. for i, m := range agents {
  135. if m.Name == modeName {
  136. agentIndex = i
  137. break
  138. }
  139. }
  140. agent = &agents[agentIndex]
  141. if agent.Model.ModelID != "" {
  142. appState.AgentModel[agent.Name] = AgentModel{
  143. ProviderID: agent.Model.ProviderID,
  144. ModelID: agent.Model.ModelID,
  145. }
  146. }
  147. if err := theme.LoadThemesFromDirectories(
  148. appInfo.Path.Config,
  149. appInfo.Path.Root,
  150. appInfo.Path.Cwd,
  151. ); err != nil {
  152. slog.Warn("Failed to load themes from directories", "error", err)
  153. }
  154. if appState.Theme != "" {
  155. if appState.Theme == "system" && styles.Terminal != nil {
  156. theme.UpdateSystemTheme(
  157. styles.Terminal.Background,
  158. styles.Terminal.BackgroundIsDark,
  159. )
  160. }
  161. theme.SetTheme(appState.Theme)
  162. }
  163. slog.Debug("Loaded config", "config", configInfo)
  164. app := &App{
  165. Info: appInfo,
  166. Agents: agents,
  167. Version: version,
  168. StatePath: appStatePath,
  169. Config: configInfo,
  170. State: appState,
  171. Client: httpClient,
  172. AgentIndex: agentIndex,
  173. Session: &opencode.Session{},
  174. Messages: []Message{},
  175. Commands: commands.LoadFromConfig(configInfo),
  176. InitialModel: initialModel,
  177. InitialPrompt: initialPrompt,
  178. InitialAgent: initialAgent,
  179. InitialSession: initialSession,
  180. ScrollSpeed: int(configInfo.Tui.ScrollSpeed),
  181. }
  182. return app, nil
  183. }
  184. func (a *App) Keybind(commandName commands.CommandName) string {
  185. command := a.Commands[commandName]
  186. kb := command.Keybindings[0]
  187. key := kb.Key
  188. if kb.RequiresLeader {
  189. key = a.Config.Keybinds.Leader + " " + kb.Key
  190. }
  191. return key
  192. }
  193. func (a *App) Key(commandName commands.CommandName) string {
  194. t := theme.CurrentTheme()
  195. base := styles.NewStyle().Background(t.Background()).Foreground(t.Text()).Bold(true).Render
  196. muted := styles.NewStyle().
  197. Background(t.Background()).
  198. Foreground(t.TextMuted()).
  199. Faint(true).
  200. Render
  201. command := a.Commands[commandName]
  202. key := a.Keybind(commandName)
  203. return base(key) + muted(" "+command.Description)
  204. }
  205. func SetClipboard(text string) tea.Cmd {
  206. var cmds []tea.Cmd
  207. cmds = append(cmds, func() tea.Msg {
  208. clipboard.Write(clipboard.FmtText, []byte(text))
  209. return nil
  210. })
  211. // try to set the clipboard using OSC52 for terminals that support it
  212. cmds = append(cmds, tea.SetClipboard(text))
  213. return tea.Sequence(cmds...)
  214. }
  215. func (a *App) cycleMode(forward bool) (*App, tea.Cmd) {
  216. if forward {
  217. a.AgentIndex++
  218. if a.AgentIndex >= len(a.Agents) {
  219. a.AgentIndex = 0
  220. }
  221. } else {
  222. a.AgentIndex--
  223. if a.AgentIndex < 0 {
  224. a.AgentIndex = len(a.Agents) - 1
  225. }
  226. }
  227. if a.Agent().Mode == "subagent" {
  228. return a.cycleMode(forward)
  229. }
  230. modelID := a.Agent().Model.ModelID
  231. providerID := a.Agent().Model.ProviderID
  232. if modelID == "" {
  233. if model, ok := a.State.AgentModel[a.Agent().Name]; ok {
  234. modelID = model.ModelID
  235. providerID = model.ProviderID
  236. }
  237. }
  238. if modelID != "" {
  239. for _, provider := range a.Providers {
  240. if provider.ID == providerID {
  241. a.Provider = &provider
  242. for _, model := range provider.Models {
  243. if model.ID == modelID {
  244. a.Model = &model
  245. break
  246. }
  247. }
  248. break
  249. }
  250. }
  251. }
  252. a.State.Agent = a.Agent().Name
  253. a.State.UpdateAgentUsage(a.Agent().Name)
  254. return a, a.SaveState()
  255. }
  256. func (a *App) SwitchAgent() (*App, tea.Cmd) {
  257. return a.cycleMode(true)
  258. }
  259. func (a *App) SwitchAgentReverse() (*App, tea.Cmd) {
  260. return a.cycleMode(false)
  261. }
  262. func (a *App) cycleRecentModel(forward bool) (*App, tea.Cmd) {
  263. recentModels := a.State.RecentlyUsedModels
  264. if len(recentModels) > 5 {
  265. recentModels = recentModels[:5]
  266. }
  267. if len(recentModels) < 2 {
  268. return a, toast.NewInfoToast("Need at least 2 recent models to cycle")
  269. }
  270. nextIndex := 0
  271. prevIndex := 0
  272. for i, recentModel := range recentModels {
  273. if a.Provider != nil && a.Model != nil && recentModel.ProviderID == a.Provider.ID &&
  274. recentModel.ModelID == a.Model.ID {
  275. nextIndex = (i + 1) % len(recentModels)
  276. prevIndex = (i - 1 + len(recentModels)) % len(recentModels)
  277. break
  278. }
  279. }
  280. targetIndex := nextIndex
  281. if !forward {
  282. targetIndex = prevIndex
  283. }
  284. for range recentModels {
  285. currentRecentModel := recentModels[targetIndex%len(recentModels)]
  286. provider, model := findModelByProviderAndModelID(
  287. a.Providers,
  288. currentRecentModel.ProviderID,
  289. currentRecentModel.ModelID,
  290. )
  291. if provider != nil && model != nil {
  292. a.Provider, a.Model = provider, model
  293. a.State.AgentModel[a.Agent().Name] = AgentModel{
  294. ProviderID: provider.ID,
  295. ModelID: model.ID,
  296. }
  297. return a, tea.Sequence(
  298. a.SaveState(),
  299. toast.NewSuccessToast(
  300. fmt.Sprintf("Switched to %s (%s)", model.Name, provider.Name),
  301. ),
  302. )
  303. }
  304. recentModels = append(
  305. recentModels[:targetIndex%len(recentModels)],
  306. recentModels[targetIndex%len(recentModels)+1:]...)
  307. if len(recentModels) < 2 {
  308. a.State.RecentlyUsedModels = recentModels
  309. return a, tea.Sequence(
  310. a.SaveState(),
  311. toast.NewInfoToast("Not enough valid recent models to cycle"),
  312. )
  313. }
  314. }
  315. a.State.RecentlyUsedModels = recentModels
  316. return a, toast.NewErrorToast("Recent model not found")
  317. }
  318. func (a *App) CycleRecentModel() (*App, tea.Cmd) {
  319. return a.cycleRecentModel(true)
  320. }
  321. func (a *App) CycleRecentModelReverse() (*App, tea.Cmd) {
  322. return a.cycleRecentModel(false)
  323. }
  324. func (a *App) SwitchToAgent(agentName string) (*App, tea.Cmd) {
  325. // Find the agent index by name
  326. for i, agent := range a.Agents {
  327. if agent.Name == agentName {
  328. a.AgentIndex = i
  329. break
  330. }
  331. }
  332. // Set up model for the new agent
  333. modelID := a.Agent().Model.ModelID
  334. providerID := a.Agent().Model.ProviderID
  335. if modelID == "" {
  336. if model, ok := a.State.AgentModel[a.Agent().Name]; ok {
  337. modelID = model.ModelID
  338. providerID = model.ProviderID
  339. }
  340. }
  341. if modelID != "" {
  342. for _, provider := range a.Providers {
  343. if provider.ID == providerID {
  344. a.Provider = &provider
  345. for _, model := range provider.Models {
  346. if model.ID == modelID {
  347. a.Model = &model
  348. break
  349. }
  350. }
  351. break
  352. }
  353. }
  354. }
  355. a.State.Agent = a.Agent().Name
  356. a.State.UpdateAgentUsage(agentName)
  357. return a, a.SaveState()
  358. }
  359. // findModelByFullID finds a model by its full ID in the format "provider/model"
  360. func findModelByFullID(
  361. providers []opencode.Provider,
  362. fullModelID string,
  363. ) (*opencode.Provider, *opencode.Model) {
  364. modelParts := strings.SplitN(fullModelID, "/", 2)
  365. if len(modelParts) < 2 {
  366. return nil, nil
  367. }
  368. providerID := modelParts[0]
  369. modelID := modelParts[1]
  370. return findModelByProviderAndModelID(providers, providerID, modelID)
  371. }
  372. // findModelByProviderAndModelID finds a model by provider ID and model ID
  373. func findModelByProviderAndModelID(
  374. providers []opencode.Provider,
  375. providerID, modelID string,
  376. ) (*opencode.Provider, *opencode.Model) {
  377. for _, provider := range providers {
  378. if provider.ID != providerID {
  379. continue
  380. }
  381. for _, model := range provider.Models {
  382. if model.ID == modelID {
  383. return &provider, &model
  384. }
  385. }
  386. // Provider found but model not found
  387. return nil, nil
  388. }
  389. // Provider not found
  390. return nil, nil
  391. }
  392. // findProviderByID finds a provider by its ID
  393. func findProviderByID(providers []opencode.Provider, providerID string) *opencode.Provider {
  394. for _, provider := range providers {
  395. if provider.ID == providerID {
  396. return &provider
  397. }
  398. }
  399. return nil
  400. }
  401. func (a *App) InitializeProvider() tea.Cmd {
  402. providersResponse, err := a.Client.App.Providers(context.Background())
  403. if err != nil {
  404. slog.Error("Failed to list providers", "error", err)
  405. // TODO: notify user
  406. return nil
  407. }
  408. providers := providersResponse.Providers
  409. if len(providers) == 0 {
  410. slog.Error("No providers configured")
  411. return nil
  412. }
  413. a.Providers = providers
  414. // retains backwards compatibility with old state format
  415. if model, ok := a.State.AgentModel[a.State.Agent]; ok {
  416. a.State.Provider = model.ProviderID
  417. a.State.Model = model.ModelID
  418. }
  419. var selectedProvider *opencode.Provider
  420. var selectedModel *opencode.Model
  421. // Priority 1: Command line --model flag (InitialModel)
  422. if a.InitialModel != nil && *a.InitialModel != "" {
  423. if provider, model := findModelByFullID(providers, *a.InitialModel); provider != nil &&
  424. model != nil {
  425. selectedProvider = provider
  426. selectedModel = model
  427. slog.Debug(
  428. "Selected model from command line",
  429. "provider",
  430. provider.ID,
  431. "model",
  432. model.ID,
  433. )
  434. } else {
  435. slog.Debug("Command line model not found", "model", *a.InitialModel)
  436. }
  437. }
  438. // Priority 2: Config file model setting
  439. if selectedProvider == nil && a.Config.Model != "" {
  440. if provider, model := findModelByFullID(providers, a.Config.Model); provider != nil &&
  441. model != nil {
  442. selectedProvider = provider
  443. selectedModel = model
  444. slog.Debug("Selected model from config", "provider", provider.ID, "model", model.ID)
  445. } else {
  446. slog.Debug("Config model not found", "model", a.Config.Model)
  447. }
  448. }
  449. // Priority 3: Current agent's preferred model
  450. if selectedProvider == nil && a.Agent().Model.ModelID != "" {
  451. if provider, model := findModelByProviderAndModelID(providers, a.Agent().Model.ProviderID, a.Agent().Model.ModelID); provider != nil &&
  452. model != nil {
  453. selectedProvider = provider
  454. selectedModel = model
  455. slog.Debug(
  456. "Selected model from current agent",
  457. "provider",
  458. provider.ID,
  459. "model",
  460. model.ID,
  461. "agent",
  462. a.Agent().Name,
  463. )
  464. } else {
  465. slog.Debug("Agent model not found", "provider", a.Agent().Model.ProviderID, "model", a.Agent().Model.ModelID, "agent", a.Agent().Name)
  466. }
  467. }
  468. // Priority 4: Recent model usage (most recently used model)
  469. if selectedProvider == nil && len(a.State.RecentlyUsedModels) > 0 {
  470. recentUsage := a.State.RecentlyUsedModels[0] // Most recent is first
  471. if provider, model := findModelByProviderAndModelID(providers, recentUsage.ProviderID, recentUsage.ModelID); provider != nil &&
  472. model != nil {
  473. selectedProvider = provider
  474. selectedModel = model
  475. slog.Debug(
  476. "Selected model from recent usage",
  477. "provider",
  478. provider.ID,
  479. "model",
  480. model.ID,
  481. )
  482. } else {
  483. slog.Debug("Recent model not found", "provider", recentUsage.ProviderID, "model", recentUsage.ModelID)
  484. }
  485. }
  486. // Priority 5: State-based model (backwards compatibility)
  487. if selectedProvider == nil && a.State.Provider != "" && a.State.Model != "" {
  488. if provider, model := findModelByProviderAndModelID(providers, a.State.Provider, a.State.Model); provider != nil &&
  489. model != nil {
  490. selectedProvider = provider
  491. selectedModel = model
  492. slog.Debug("Selected model from state", "provider", provider.ID, "model", model.ID)
  493. } else {
  494. slog.Debug("State model not found", "provider", a.State.Provider, "model", a.State.Model)
  495. }
  496. }
  497. // Priority 6: Internal priority fallback (Anthropic preferred, then first available)
  498. if selectedProvider == nil {
  499. // Try Anthropic first as internal priority
  500. if provider := findProviderByID(providers, "anthropic"); provider != nil {
  501. if model := getDefaultModel(providersResponse, *provider); model != nil {
  502. selectedProvider = provider
  503. selectedModel = model
  504. slog.Debug(
  505. "Selected model from internal priority (Anthropic)",
  506. "provider",
  507. provider.ID,
  508. "model",
  509. model.ID,
  510. )
  511. }
  512. }
  513. // If Anthropic not available, use first available provider
  514. if selectedProvider == nil && len(providers) > 0 {
  515. provider := &providers[0]
  516. if model := getDefaultModel(providersResponse, *provider); model != nil {
  517. selectedProvider = provider
  518. selectedModel = model
  519. slog.Debug(
  520. "Selected model from fallback (first available)",
  521. "provider",
  522. provider.ID,
  523. "model",
  524. model.ID,
  525. )
  526. }
  527. }
  528. }
  529. // Final safety check
  530. if selectedProvider == nil || selectedModel == nil {
  531. slog.Error("Failed to select any model")
  532. return nil
  533. }
  534. var cmds []tea.Cmd
  535. cmds = append(cmds, util.CmdHandler(ModelSelectedMsg{
  536. Provider: *selectedProvider,
  537. Model: *selectedModel,
  538. }))
  539. // Load initial session if provided
  540. if a.InitialSession != nil && *a.InitialSession != "" {
  541. cmds = append(cmds, func() tea.Msg {
  542. // Find the session by ID
  543. sessions, err := a.ListSessions(context.Background())
  544. if err != nil {
  545. slog.Error("Failed to list sessions for initial session", "error", err)
  546. return toast.NewErrorToast("Failed to load initial session")()
  547. }
  548. for _, session := range sessions {
  549. if session.ID == *a.InitialSession {
  550. return SessionSelectedMsg(&session)
  551. }
  552. }
  553. slog.Warn("Initial session not found", "sessionID", *a.InitialSession)
  554. return toast.NewErrorToast("Session not found: " + *a.InitialSession)()
  555. })
  556. }
  557. if a.InitialPrompt != nil && *a.InitialPrompt != "" {
  558. cmds = append(cmds, util.CmdHandler(SendPrompt{Text: *a.InitialPrompt}))
  559. }
  560. return tea.Sequence(cmds...)
  561. }
  562. func getDefaultModel(
  563. response *opencode.AppProvidersResponse,
  564. provider opencode.Provider,
  565. ) *opencode.Model {
  566. if match, ok := response.Default[provider.ID]; ok {
  567. model := provider.Models[match]
  568. return &model
  569. } else {
  570. for _, model := range provider.Models {
  571. return &model
  572. }
  573. }
  574. return nil
  575. }
  576. func (a *App) IsBusy() bool {
  577. if len(a.Messages) == 0 {
  578. return false
  579. }
  580. lastMessage := a.Messages[len(a.Messages)-1]
  581. if casted, ok := lastMessage.Info.(opencode.AssistantMessage); ok {
  582. return casted.Time.Completed == 0
  583. }
  584. return false
  585. }
  586. func (a *App) SaveState() tea.Cmd {
  587. return func() tea.Msg {
  588. err := SaveState(a.StatePath, a.State)
  589. if err != nil {
  590. slog.Error("Failed to save state", "error", err)
  591. }
  592. return nil
  593. }
  594. }
  595. func (a *App) InitializeProject(ctx context.Context) tea.Cmd {
  596. cmds := []tea.Cmd{}
  597. session, err := a.CreateSession(ctx)
  598. if err != nil {
  599. // status.Error(err.Error())
  600. return nil
  601. }
  602. a.Session = session
  603. cmds = append(cmds, util.CmdHandler(SessionCreatedMsg{Session: session}))
  604. go func() {
  605. _, err := a.Client.Session.Init(ctx, a.Session.ID, opencode.SessionInitParams{
  606. MessageID: opencode.F(id.Ascending(id.Message)),
  607. ProviderID: opencode.F(a.Provider.ID),
  608. ModelID: opencode.F(a.Model.ID),
  609. })
  610. if err != nil {
  611. slog.Error("Failed to initialize project", "error", err)
  612. // status.Error(err.Error())
  613. }
  614. }()
  615. return tea.Batch(cmds...)
  616. }
  617. func (a *App) CompactSession(ctx context.Context) tea.Cmd {
  618. if a.compactCancel != nil {
  619. a.compactCancel()
  620. }
  621. compactCtx, cancel := context.WithCancel(ctx)
  622. a.compactCancel = cancel
  623. go func() {
  624. defer func() {
  625. a.compactCancel = nil
  626. }()
  627. _, err := a.Client.Session.Summarize(
  628. compactCtx,
  629. a.Session.ID,
  630. opencode.SessionSummarizeParams{
  631. ProviderID: opencode.F(a.Provider.ID),
  632. ModelID: opencode.F(a.Model.ID),
  633. },
  634. )
  635. if err != nil {
  636. if compactCtx.Err() != context.Canceled {
  637. slog.Error("Failed to compact session", "error", err)
  638. }
  639. }
  640. }()
  641. return nil
  642. }
  643. func (a *App) MarkProjectInitialized(ctx context.Context) error {
  644. _, err := a.Client.App.Init(ctx)
  645. if err != nil {
  646. slog.Error("Failed to mark project as initialized", "error", err)
  647. return err
  648. }
  649. return nil
  650. }
  651. func (a *App) CreateSession(ctx context.Context) (*opencode.Session, error) {
  652. session, err := a.Client.Session.New(ctx, opencode.SessionNewParams{})
  653. if err != nil {
  654. return nil, err
  655. }
  656. return session, nil
  657. }
  658. func (a *App) SendPrompt(ctx context.Context, prompt Prompt) (*App, tea.Cmd) {
  659. var cmds []tea.Cmd
  660. if a.Session.ID == "" {
  661. session, err := a.CreateSession(ctx)
  662. if err != nil {
  663. return a, toast.NewErrorToast(err.Error())
  664. }
  665. a.Session = session
  666. cmds = append(cmds, util.CmdHandler(SessionCreatedMsg{Session: session}))
  667. }
  668. messageID := id.Ascending(id.Message)
  669. message := prompt.ToMessage(messageID, a.Session.ID)
  670. a.Messages = append(a.Messages, message)
  671. cmds = append(cmds, func() tea.Msg {
  672. _, err := a.Client.Session.Chat(ctx, a.Session.ID, opencode.SessionChatParams{
  673. ProviderID: opencode.F(a.Provider.ID),
  674. ModelID: opencode.F(a.Model.ID),
  675. Agent: opencode.F(a.Agent().Name),
  676. MessageID: opencode.F(messageID),
  677. Parts: opencode.F(message.ToSessionChatParams()),
  678. })
  679. if err != nil {
  680. errormsg := fmt.Sprintf("failed to send message: %v", err)
  681. slog.Error(errormsg)
  682. return toast.NewErrorToast(errormsg)()
  683. }
  684. return nil
  685. })
  686. // The actual response will come through SSE
  687. // For now, just return success
  688. return a, tea.Batch(cmds...)
  689. }
  690. func (a *App) SendShell(ctx context.Context, command string) (*App, tea.Cmd) {
  691. var cmds []tea.Cmd
  692. if a.Session.ID == "" {
  693. session, err := a.CreateSession(ctx)
  694. if err != nil {
  695. return a, toast.NewErrorToast(err.Error())
  696. }
  697. a.Session = session
  698. cmds = append(cmds, util.CmdHandler(SessionCreatedMsg{Session: session}))
  699. }
  700. cmds = append(cmds, func() tea.Msg {
  701. _, err := a.Client.Session.Shell(
  702. context.Background(),
  703. a.Session.ID,
  704. opencode.SessionShellParams{
  705. Agent: opencode.F(a.Agent().Name),
  706. Command: opencode.F(command),
  707. },
  708. )
  709. if err != nil {
  710. slog.Error("Failed to submit shell command", "error", err)
  711. return toast.NewErrorToast("Failed to submit shell command")()
  712. }
  713. return nil
  714. })
  715. // The actual response will come through SSE
  716. // For now, just return success
  717. return a, tea.Batch(cmds...)
  718. }
  719. func (a *App) Cancel(ctx context.Context, sessionID string) error {
  720. // Cancel any running compact operation
  721. if a.compactCancel != nil {
  722. a.compactCancel()
  723. a.compactCancel = nil
  724. }
  725. _, err := a.Client.Session.Abort(ctx, sessionID)
  726. if err != nil {
  727. slog.Error("Failed to cancel session", "error", err)
  728. return err
  729. }
  730. return nil
  731. }
  732. func (a *App) ListSessions(ctx context.Context) ([]opencode.Session, error) {
  733. response, err := a.Client.Session.List(ctx)
  734. if err != nil {
  735. return nil, err
  736. }
  737. if response == nil {
  738. return []opencode.Session{}, nil
  739. }
  740. sessions := *response
  741. return sessions, nil
  742. }
  743. func (a *App) DeleteSession(ctx context.Context, sessionID string) error {
  744. _, err := a.Client.Session.Delete(ctx, sessionID)
  745. if err != nil {
  746. slog.Error("Failed to delete session", "error", err)
  747. return err
  748. }
  749. return nil
  750. }
  751. func (a *App) UpdateSession(ctx context.Context, sessionID string, title string) error {
  752. _, err := a.Client.Session.Update(ctx, sessionID, opencode.SessionUpdateParams{
  753. Title: opencode.F(title),
  754. })
  755. if err != nil {
  756. slog.Error("Failed to update session", "error", err)
  757. return err
  758. }
  759. return nil
  760. }
  761. func (a *App) ListMessages(ctx context.Context, sessionId string) ([]Message, error) {
  762. response, err := a.Client.Session.Messages(ctx, sessionId)
  763. if err != nil {
  764. return nil, err
  765. }
  766. if response == nil {
  767. return []Message{}, nil
  768. }
  769. messages := []Message{}
  770. for _, message := range *response {
  771. msg := Message{
  772. Info: message.Info.AsUnion(),
  773. Parts: []opencode.PartUnion{},
  774. }
  775. for _, part := range message.Parts {
  776. msg.Parts = append(msg.Parts, part.AsUnion())
  777. }
  778. messages = append(messages, msg)
  779. }
  780. return messages, nil
  781. }
  782. func (a *App) ListProviders(ctx context.Context) ([]opencode.Provider, error) {
  783. response, err := a.Client.App.Providers(ctx)
  784. if err != nil {
  785. return nil, err
  786. }
  787. if response == nil {
  788. return []opencode.Provider{}, nil
  789. }
  790. providers := *response
  791. return providers.Providers, nil
  792. }
  793. // func (a *App) loadCustomKeybinds() {
  794. //
  795. // }