message.go 17 KB

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