message.go 17 KB

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