message.go 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700
  1. package chat
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "slices"
  6. "strings"
  7. "time"
  8. "github.com/charmbracelet/lipgloss/v2"
  9. "github.com/charmbracelet/lipgloss/v2/compat"
  10. "github.com/charmbracelet/x/ansi"
  11. "github.com/sst/opencode-sdk-go"
  12. "github.com/sst/opencode/internal/app"
  13. "github.com/sst/opencode/internal/components/diff"
  14. "github.com/sst/opencode/internal/styles"
  15. "github.com/sst/opencode/internal/theme"
  16. "github.com/sst/opencode/internal/util"
  17. "golang.org/x/text/cases"
  18. "golang.org/x/text/language"
  19. )
  20. type blockRenderer struct {
  21. textColor compat.AdaptiveColor
  22. border bool
  23. borderColor *compat.AdaptiveColor
  24. borderColorRight bool
  25. paddingTop int
  26. paddingBottom int
  27. paddingLeft int
  28. paddingRight int
  29. marginTop int
  30. marginBottom int
  31. }
  32. type renderingOption func(*blockRenderer)
  33. func WithTextColor(color compat.AdaptiveColor) renderingOption {
  34. return func(c *blockRenderer) {
  35. c.textColor = color
  36. }
  37. }
  38. func WithNoBorder() renderingOption {
  39. return func(c *blockRenderer) {
  40. c.border = false
  41. }
  42. }
  43. func WithBorderColor(color compat.AdaptiveColor) renderingOption {
  44. return func(c *blockRenderer) {
  45. c.borderColor = &color
  46. }
  47. }
  48. func WithBorderColorRight(color compat.AdaptiveColor) renderingOption {
  49. return func(c *blockRenderer) {
  50. c.borderColorRight = true
  51. c.borderColor = &color
  52. }
  53. }
  54. func WithMarginTop(padding int) renderingOption {
  55. return func(c *blockRenderer) {
  56. c.marginTop = padding
  57. }
  58. }
  59. func WithMarginBottom(padding int) renderingOption {
  60. return func(c *blockRenderer) {
  61. c.marginBottom = padding
  62. }
  63. }
  64. func WithPadding(padding int) renderingOption {
  65. return func(c *blockRenderer) {
  66. c.paddingTop = padding
  67. c.paddingBottom = padding
  68. c.paddingLeft = padding
  69. c.paddingRight = padding
  70. }
  71. }
  72. func WithPaddingLeft(padding int) renderingOption {
  73. return func(c *blockRenderer) {
  74. c.paddingLeft = padding
  75. }
  76. }
  77. func WithPaddingRight(padding int) renderingOption {
  78. return func(c *blockRenderer) {
  79. c.paddingRight = padding
  80. }
  81. }
  82. func WithPaddingTop(padding int) renderingOption {
  83. return func(c *blockRenderer) {
  84. c.paddingTop = padding
  85. }
  86. }
  87. func WithPaddingBottom(padding int) renderingOption {
  88. return func(c *blockRenderer) {
  89. c.paddingBottom = padding
  90. }
  91. }
  92. func renderContentBlock(
  93. app *app.App,
  94. content string,
  95. width int,
  96. options ...renderingOption,
  97. ) string {
  98. t := theme.CurrentTheme()
  99. renderer := &blockRenderer{
  100. textColor: t.TextMuted(),
  101. border: true,
  102. paddingTop: 1,
  103. paddingBottom: 1,
  104. paddingLeft: 2,
  105. paddingRight: 2,
  106. }
  107. for _, option := range options {
  108. option(renderer)
  109. }
  110. borderColor := t.BackgroundPanel()
  111. if renderer.borderColor != nil {
  112. borderColor = *renderer.borderColor
  113. }
  114. style := styles.NewStyle().
  115. Foreground(renderer.textColor).
  116. Background(t.BackgroundPanel()).
  117. PaddingTop(renderer.paddingTop).
  118. PaddingBottom(renderer.paddingBottom).
  119. PaddingLeft(renderer.paddingLeft).
  120. PaddingRight(renderer.paddingRight).
  121. AlignHorizontal(lipgloss.Left)
  122. if renderer.border {
  123. style = style.
  124. BorderStyle(lipgloss.ThickBorder()).
  125. BorderLeft(true).
  126. BorderRight(true).
  127. BorderLeftForeground(borderColor).
  128. BorderLeftBackground(t.Background()).
  129. BorderRightForeground(t.BackgroundPanel()).
  130. BorderRightBackground(t.Background())
  131. if renderer.borderColorRight {
  132. style = style.
  133. BorderLeftBackground(t.Background()).
  134. BorderLeftForeground(t.BackgroundPanel()).
  135. BorderRightForeground(borderColor).
  136. BorderRightBackground(t.Background())
  137. }
  138. }
  139. content = style.Render(content)
  140. if renderer.marginTop > 0 {
  141. for range renderer.marginTop {
  142. content = "\n" + content
  143. }
  144. }
  145. if renderer.marginBottom > 0 {
  146. for range renderer.marginBottom {
  147. content = content + "\n"
  148. }
  149. }
  150. return content
  151. }
  152. func renderText(
  153. app *app.App,
  154. message opencode.MessageUnion,
  155. text string,
  156. author string,
  157. showToolDetails bool,
  158. width int,
  159. extra string,
  160. toolCalls ...opencode.ToolPart,
  161. ) string {
  162. t := theme.CurrentTheme()
  163. var ts time.Time
  164. backgroundColor := t.BackgroundPanel()
  165. var content string
  166. switch casted := message.(type) {
  167. case opencode.AssistantMessage:
  168. ts = time.UnixMilli(int64(casted.Time.Created))
  169. content = util.ToMarkdown(text, width, backgroundColor)
  170. case opencode.UserMessage:
  171. ts = time.UnixMilli(int64(casted.Time.Created))
  172. base := styles.NewStyle().Foreground(t.Text()).Background(backgroundColor)
  173. words := strings.Fields(text)
  174. for i, word := range words {
  175. if strings.HasPrefix(word, "@") {
  176. words[i] = base.Foreground(t.Secondary()).Render(word + " ")
  177. } else {
  178. words[i] = base.Render(word + " ")
  179. }
  180. }
  181. text = strings.Join(words, "")
  182. text = ansi.WordwrapWc(text, width-6, " -")
  183. content = base.Width(width - 6).Render(text)
  184. }
  185. timestamp := ts.
  186. Local().
  187. Format("02 Jan 2006 03:04 PM")
  188. if time.Now().Format("02 Jan 2006") == timestamp[:11] {
  189. // don't show the date if it's today
  190. timestamp = timestamp[12:]
  191. }
  192. info := fmt.Sprintf("%s (%s)", author, timestamp)
  193. info = styles.NewStyle().Foreground(t.TextMuted()).Render(info)
  194. if !showToolDetails && toolCalls != nil && len(toolCalls) > 0 {
  195. content = content + "\n\n"
  196. for _, toolCall := range toolCalls {
  197. title := renderToolTitle(toolCall, width)
  198. style := styles.NewStyle()
  199. if toolCall.State.Status == opencode.ToolPartStateStatusError {
  200. style = style.Foreground(t.Error())
  201. }
  202. title = style.Render(title)
  203. title = "∟ " + title + "\n"
  204. content = content + title
  205. }
  206. }
  207. sections := []string{content, info}
  208. if extra != "" {
  209. sections = append(sections, "\n"+extra)
  210. }
  211. content = strings.Join(sections, "\n")
  212. switch message.(type) {
  213. case opencode.UserMessage:
  214. return renderContentBlock(
  215. app,
  216. content,
  217. width,
  218. WithTextColor(t.Text()),
  219. WithBorderColorRight(t.Secondary()),
  220. )
  221. case opencode.AssistantMessage:
  222. return renderContentBlock(
  223. app,
  224. content,
  225. width,
  226. WithBorderColor(t.Accent()),
  227. )
  228. }
  229. return ""
  230. }
  231. func renderToolDetails(
  232. app *app.App,
  233. toolCall opencode.ToolPart,
  234. width int,
  235. ) string {
  236. ignoredTools := []string{"todoread"}
  237. if slices.Contains(ignoredTools, toolCall.Tool) {
  238. return ""
  239. }
  240. if toolCall.State.Status == opencode.ToolPartStateStatusPending {
  241. title := renderToolTitle(toolCall, width)
  242. return renderContentBlock(app, title, width)
  243. }
  244. var result *string
  245. if toolCall.State.Output != "" {
  246. result = &toolCall.State.Output
  247. }
  248. toolInputMap := make(map[string]any)
  249. if toolCall.State.Input != nil {
  250. value := toolCall.State.Input
  251. if m, ok := value.(map[string]any); ok {
  252. toolInputMap = m
  253. keys := make([]string, 0, len(toolInputMap))
  254. for key := range toolInputMap {
  255. keys = append(keys, key)
  256. }
  257. slices.Sort(keys)
  258. }
  259. }
  260. body := ""
  261. t := theme.CurrentTheme()
  262. backgroundColor := t.BackgroundPanel()
  263. borderColor := t.BackgroundPanel()
  264. defaultStyle := styles.NewStyle().Background(backgroundColor).Width(width - 6).Render
  265. if toolCall.State.Metadata != nil {
  266. metadata := toolCall.State.Metadata.(map[string]any)
  267. switch toolCall.Tool {
  268. case "read":
  269. var preview any
  270. if metadata != nil {
  271. preview = metadata["preview"]
  272. }
  273. if preview != nil && toolInputMap["filePath"] != nil {
  274. filename := toolInputMap["filePath"].(string)
  275. body = preview.(string)
  276. body = util.RenderFile(filename, body, width, util.WithTruncate(6))
  277. }
  278. case "edit":
  279. if filename, ok := toolInputMap["filePath"].(string); ok {
  280. var diffField any
  281. if metadata != nil {
  282. diffField = metadata["diff"]
  283. }
  284. if diffField != nil {
  285. patch := diffField.(string)
  286. var formattedDiff string
  287. formattedDiff, _ = diff.FormatUnifiedDiff(
  288. filename,
  289. patch,
  290. diff.WithWidth(width-2),
  291. )
  292. body = strings.TrimSpace(formattedDiff)
  293. style := styles.NewStyle().
  294. Background(backgroundColor).
  295. Foreground(t.TextMuted()).
  296. Padding(1, 2).
  297. Width(width - 4)
  298. if diagnostics := renderDiagnostics(metadata, filename, backgroundColor, width-6); diagnostics != "" {
  299. diagnostics = style.Render(diagnostics)
  300. body += "\n" + diagnostics
  301. }
  302. title := renderToolTitle(toolCall, width)
  303. title = style.Render(title)
  304. content := title + "\n" + body
  305. content = renderContentBlock(
  306. app,
  307. content,
  308. width,
  309. WithPadding(0),
  310. WithBorderColor(borderColor),
  311. )
  312. return content
  313. }
  314. }
  315. case "write":
  316. if filename, ok := toolInputMap["filePath"].(string); ok {
  317. if content, ok := toolInputMap["content"].(string); ok {
  318. body = util.RenderFile(filename, content, width)
  319. if diagnostics := renderDiagnostics(metadata, filename, backgroundColor, width-4); diagnostics != "" {
  320. body += "\n\n" + diagnostics
  321. }
  322. }
  323. }
  324. case "bash":
  325. stdout := metadata["stdout"]
  326. if stdout != nil {
  327. command := toolInputMap["command"].(string)
  328. body = fmt.Sprintf("```console\n> %s\n%s```", command, stdout)
  329. body = util.ToMarkdown(body, width, backgroundColor)
  330. }
  331. case "webfetch":
  332. if format, ok := toolInputMap["format"].(string); ok && result != nil {
  333. body = *result
  334. body = util.TruncateHeight(body, 10)
  335. if format == "html" || format == "markdown" {
  336. body = util.ToMarkdown(body, width, backgroundColor)
  337. }
  338. }
  339. case "todowrite":
  340. todos := metadata["todos"]
  341. if todos != nil {
  342. for _, item := range todos.([]any) {
  343. todo := item.(map[string]any)
  344. content := todo["content"].(string)
  345. switch todo["status"] {
  346. case "completed":
  347. body += fmt.Sprintf("- [x] %s\n", content)
  348. case "cancelled":
  349. // strike through cancelled todo
  350. body += fmt.Sprintf("- [~] ~~%s~~\n", content)
  351. case "in_progress":
  352. // highlight in progress todo
  353. body += fmt.Sprintf("- [ ] `%s`\n", content)
  354. default:
  355. body += fmt.Sprintf("- [ ] %s\n", content)
  356. }
  357. }
  358. body = util.ToMarkdown(body, width, backgroundColor)
  359. }
  360. case "task":
  361. summary := metadata["summary"]
  362. if summary != nil {
  363. toolcalls := summary.([]any)
  364. steps := []string{}
  365. for _, item := range toolcalls {
  366. data, _ := json.Marshal(item)
  367. var toolCall opencode.ToolPart
  368. _ = json.Unmarshal(data, &toolCall)
  369. step := renderToolTitle(toolCall, width)
  370. step = "∟ " + step
  371. steps = append(steps, step)
  372. }
  373. body = strings.Join(steps, "\n")
  374. }
  375. body = defaultStyle(body)
  376. default:
  377. if result == nil {
  378. empty := ""
  379. result = &empty
  380. }
  381. body = *result
  382. body = util.TruncateHeight(body, 10)
  383. body = defaultStyle(body)
  384. }
  385. }
  386. error := ""
  387. if toolCall.State.Status == opencode.ToolPartStateStatusError {
  388. error = toolCall.State.Error
  389. }
  390. if error != "" {
  391. body = styles.NewStyle().
  392. Width(width - 6).
  393. Foreground(t.Error()).
  394. Background(backgroundColor).
  395. Render(error)
  396. }
  397. if body == "" && error == "" && result != nil {
  398. body = *result
  399. body = util.TruncateHeight(body, 10)
  400. body = defaultStyle(body)
  401. }
  402. if body == "" {
  403. body = defaultStyle("")
  404. }
  405. title := renderToolTitle(toolCall, width)
  406. content := title + "\n\n" + body
  407. return renderContentBlock(app, content, width, WithBorderColor(borderColor))
  408. }
  409. func renderToolName(name string) string {
  410. switch name {
  411. case "webfetch":
  412. return "Fetch"
  413. default:
  414. normalizedName := name
  415. if after, ok := strings.CutPrefix(name, "opencode_"); ok {
  416. normalizedName = after
  417. }
  418. return cases.Title(language.Und).String(normalizedName)
  419. }
  420. }
  421. func getTodoPhase(metadata map[string]any) string {
  422. todos, ok := metadata["todos"].([]any)
  423. if !ok || len(todos) == 0 {
  424. return "Plan"
  425. }
  426. counts := map[string]int{"pending": 0, "completed": 0}
  427. for _, item := range todos {
  428. if todo, ok := item.(map[string]any); ok {
  429. if status, ok := todo["status"].(string); ok {
  430. counts[status]++
  431. }
  432. }
  433. }
  434. total := len(todos)
  435. switch {
  436. case counts["pending"] == total:
  437. return "Creating plan"
  438. case counts["completed"] == total:
  439. return "Completing plan"
  440. default:
  441. return "Updating plan"
  442. }
  443. }
  444. func getTodoTitle(toolCall opencode.ToolPart) string {
  445. if toolCall.State.Status == opencode.ToolPartStateStatusCompleted {
  446. if metadata, ok := toolCall.State.Metadata.(map[string]any); ok {
  447. return getTodoPhase(metadata)
  448. }
  449. }
  450. return "Plan"
  451. }
  452. func renderToolTitle(
  453. toolCall opencode.ToolPart,
  454. width int,
  455. ) string {
  456. if toolCall.State.Status == opencode.ToolPartStateStatusPending {
  457. title := renderToolAction(toolCall.Tool)
  458. return styles.NewStyle().Width(width - 6).Render(title)
  459. }
  460. toolArgs := ""
  461. toolArgsMap := make(map[string]any)
  462. if toolCall.State.Input != nil {
  463. value := toolCall.State.Input
  464. if m, ok := value.(map[string]any); ok {
  465. toolArgsMap = m
  466. keys := make([]string, 0, len(toolArgsMap))
  467. for key := range toolArgsMap {
  468. keys = append(keys, key)
  469. }
  470. slices.Sort(keys)
  471. firstKey := ""
  472. if len(keys) > 0 {
  473. firstKey = keys[0]
  474. }
  475. toolArgs = renderArgs(&toolArgsMap, firstKey)
  476. }
  477. }
  478. title := renderToolName(toolCall.Tool)
  479. switch toolCall.Tool {
  480. case "read":
  481. toolArgs = renderArgs(&toolArgsMap, "filePath")
  482. title = fmt.Sprintf("%s %s", title, toolArgs)
  483. case "edit", "write":
  484. if filename, ok := toolArgsMap["filePath"].(string); ok {
  485. title = fmt.Sprintf("%s %s", title, util.Relative(filename))
  486. }
  487. case "bash", "task":
  488. if description, ok := toolArgsMap["description"].(string); ok {
  489. title = fmt.Sprintf("%s %s", title, description)
  490. }
  491. case "webfetch":
  492. toolArgs = renderArgs(&toolArgsMap, "url")
  493. title = fmt.Sprintf("%s %s", title, toolArgs)
  494. case "todowrite":
  495. title = getTodoTitle(toolCall)
  496. case "todoread":
  497. return "Plan"
  498. default:
  499. toolName := renderToolName(toolCall.Tool)
  500. title = fmt.Sprintf("%s %s", toolName, toolArgs)
  501. }
  502. return title
  503. }
  504. func renderToolAction(name string) string {
  505. switch name {
  506. case "task":
  507. return "Planning..."
  508. case "bash":
  509. return "Writing command..."
  510. case "edit":
  511. return "Preparing edit..."
  512. case "webfetch":
  513. return "Fetching from the web..."
  514. case "glob":
  515. return "Finding files..."
  516. case "grep":
  517. return "Searching content..."
  518. case "list":
  519. return "Listing directory..."
  520. case "read":
  521. return "Reading file..."
  522. case "write":
  523. return "Preparing write..."
  524. case "todowrite", "todoread":
  525. return "Planning..."
  526. case "patch":
  527. return "Preparing patch..."
  528. }
  529. return "Working..."
  530. }
  531. func renderArgs(args *map[string]any, titleKey string) string {
  532. if args == nil || len(*args) == 0 {
  533. return ""
  534. }
  535. keys := make([]string, 0, len(*args))
  536. for key := range *args {
  537. keys = append(keys, key)
  538. }
  539. slices.Sort(keys)
  540. title := ""
  541. parts := []string{}
  542. for _, key := range keys {
  543. value := (*args)[key]
  544. if value == nil {
  545. continue
  546. }
  547. if key == "filePath" || key == "path" {
  548. value = util.Relative(value.(string))
  549. }
  550. if key == titleKey {
  551. title = fmt.Sprintf("%s", value)
  552. continue
  553. }
  554. parts = append(parts, fmt.Sprintf("%s=%v", key, value))
  555. }
  556. if len(parts) == 0 {
  557. return title
  558. }
  559. return fmt.Sprintf("%s (%s)", title, strings.Join(parts, ", "))
  560. }
  561. // Diagnostic represents an LSP diagnostic
  562. type Diagnostic struct {
  563. Range struct {
  564. Start struct {
  565. Line int `json:"line"`
  566. Character int `json:"character"`
  567. } `json:"start"`
  568. } `json:"range"`
  569. Severity int `json:"severity"`
  570. Message string `json:"message"`
  571. }
  572. // renderDiagnostics formats LSP diagnostics for display in the TUI
  573. func renderDiagnostics(
  574. metadata map[string]any,
  575. filePath string,
  576. backgroundColor compat.AdaptiveColor,
  577. width int,
  578. ) string {
  579. if diagnosticsData, ok := metadata["diagnostics"].(map[string]any); ok {
  580. if fileDiagnostics, ok := diagnosticsData[filePath].([]any); ok {
  581. var errorDiagnostics []string
  582. for _, diagInterface := range fileDiagnostics {
  583. diagMap, ok := diagInterface.(map[string]any)
  584. if !ok {
  585. continue
  586. }
  587. // Parse the diagnostic
  588. var diag Diagnostic
  589. diagBytes, err := json.Marshal(diagMap)
  590. if err != nil {
  591. continue
  592. }
  593. if err := json.Unmarshal(diagBytes, &diag); err != nil {
  594. continue
  595. }
  596. // Only show error diagnostics (severity === 1)
  597. if diag.Severity != 1 {
  598. continue
  599. }
  600. line := diag.Range.Start.Line + 1 // 1-based
  601. column := diag.Range.Start.Character + 1 // 1-based
  602. errorDiagnostics = append(
  603. errorDiagnostics,
  604. fmt.Sprintf("Error [%d:%d] %s", line, column, diag.Message),
  605. )
  606. }
  607. if len(errorDiagnostics) == 0 {
  608. return ""
  609. }
  610. t := theme.CurrentTheme()
  611. var result strings.Builder
  612. for _, diagnostic := range errorDiagnostics {
  613. if result.Len() > 0 {
  614. result.WriteString("\n\n")
  615. }
  616. diagnostic = ansi.WordwrapWc(diagnostic, width, " -")
  617. result.WriteString(
  618. styles.NewStyle().
  619. Background(backgroundColor).
  620. Foreground(t.Error()).
  621. Render(diagnostic),
  622. )
  623. }
  624. return result.String()
  625. }
  626. }
  627. return ""
  628. // diagnosticsData should be a map[string][]Diagnostic
  629. // strDiagnosticsData := diagnosticsData.Raw()
  630. // diagnosticsMap := gjson.Parse(strDiagnosticsData).Value().(map[string]any)
  631. // fileDiagnostics, ok := diagnosticsMap[filePath]
  632. // if !ok {
  633. // return ""
  634. // }
  635. // diagnosticsList, ok := fileDiagnostics.([]any)
  636. // if !ok {
  637. // return ""
  638. // }
  639. }