1package messages
2
3import (
4 "encoding/json"
5 "fmt"
6 "strings"
7 "time"
8
9 "github.com/charmbracelet/crush/internal/fsext"
10 "github.com/charmbracelet/crush/internal/highlight"
11 "github.com/charmbracelet/crush/internal/llm/agent"
12 "github.com/charmbracelet/crush/internal/llm/tools"
13 "github.com/charmbracelet/crush/internal/tui/components/core"
14 "github.com/charmbracelet/crush/internal/tui/styles"
15 "github.com/charmbracelet/lipgloss/v2"
16 "github.com/charmbracelet/lipgloss/v2/tree"
17 "github.com/charmbracelet/x/ansi"
18)
19
20// responseContextHeight limits the number of lines displayed in tool output
21const responseContextHeight = 10
22
23// renderer defines the interface for tool-specific rendering implementations
24type renderer interface {
25 // Render returns the complete (already styled) tool‑call view, not
26 // including the outer border.
27 Render(v *toolCallCmp) string
28}
29
30// rendererFactory creates new renderer instances
31type rendererFactory func() renderer
32
33// renderRegistry manages the mapping of tool names to their renderers
34type renderRegistry map[string]rendererFactory
35
36// register adds a new renderer factory to the registry
37func (rr renderRegistry) register(name string, f rendererFactory) { rr[name] = f }
38
39// lookup retrieves a renderer for the given tool name, falling back to generic renderer
40func (rr renderRegistry) lookup(name string) renderer {
41 if f, ok := rr[name]; ok {
42 return f()
43 }
44 return genericRenderer{} // sensible fallback
45}
46
47// registry holds all registered tool renderers
48var registry = renderRegistry{}
49
50// baseRenderer provides common functionality for all tool renderers
51type baseRenderer struct{}
52
53// paramBuilder helps construct parameter lists for tool headers
54type paramBuilder struct {
55 args []string
56}
57
58// newParamBuilder creates a new parameter builder
59func newParamBuilder() *paramBuilder {
60 return ¶mBuilder{args: make([]string, 0)}
61}
62
63// addMain adds the main parameter (first argument)
64func (pb *paramBuilder) addMain(value string) *paramBuilder {
65 if value != "" {
66 pb.args = append(pb.args, value)
67 }
68 return pb
69}
70
71// addKeyValue adds a key-value pair parameter
72func (pb *paramBuilder) addKeyValue(key, value string) *paramBuilder {
73 if value != "" {
74 pb.args = append(pb.args, key, value)
75 }
76 return pb
77}
78
79// addFlag adds a boolean flag parameter
80func (pb *paramBuilder) addFlag(key string, value bool) *paramBuilder {
81 if value {
82 pb.args = append(pb.args, key, "true")
83 }
84 return pb
85}
86
87// build returns the final parameter list
88func (pb *paramBuilder) build() []string {
89 return pb.args
90}
91
92// renderWithParams provides a common rendering pattern for tools with parameters
93func (br baseRenderer) renderWithParams(v *toolCallCmp, toolName string, args []string, contentRenderer func() string) string {
94 width := v.textWidth()
95 if v.isNested {
96 width -= 4 // Adjust for nested tool call indentation
97 }
98 header := br.makeHeader(v, toolName, width, args...)
99 if v.isNested {
100 return v.style().Render(header)
101 }
102 if res, done := earlyState(header, v); done {
103 return res
104 }
105 body := contentRenderer()
106 return joinHeaderBody(header, body)
107}
108
109// unmarshalParams safely unmarshal JSON parameters
110func (br baseRenderer) unmarshalParams(input string, target any) error {
111 return json.Unmarshal([]byte(input), target)
112}
113
114// makeHeader builds the tool call header with status icon and parameters for a nested tool call.
115func (br baseRenderer) makeNestedHeader(v *toolCallCmp, tool string, width int, params ...string) string {
116 t := styles.CurrentTheme()
117 tool = t.S().Base.Foreground(t.FgHalfMuted).Render(tool) + " "
118 return tool + renderParamList(true, width-lipgloss.Width(tool), params...)
119}
120
121// makeHeader builds "<Tool>: param (key=value)" and truncates as needed.
122func (br baseRenderer) makeHeader(v *toolCallCmp, tool string, width int, params ...string) string {
123 if v.isNested {
124 return br.makeNestedHeader(v, tool, width, params...)
125 }
126 t := styles.CurrentTheme()
127 icon := t.S().Base.Foreground(t.GreenDark).Render(styles.ToolPending)
128 if v.result.ToolCallID != "" {
129 if v.result.IsError {
130 icon = t.S().Base.Foreground(t.RedDark).Render(styles.ToolError)
131 } else {
132 icon = t.S().Base.Foreground(t.Green).Render(styles.ToolSuccess)
133 }
134 } else if v.cancelled {
135 icon = t.S().Muted.Render(styles.ToolPending)
136 }
137 tool = t.S().Base.Foreground(t.Blue).Render(tool)
138 prefix := fmt.Sprintf("%s %s ", icon, tool)
139 return prefix + renderParamList(false, width-lipgloss.Width(prefix), params...)
140}
141
142// renderError provides consistent error rendering
143func (br baseRenderer) renderError(v *toolCallCmp, message string) string {
144 t := styles.CurrentTheme()
145 header := br.makeHeader(v, prettifyToolName(v.call.Name), v.textWidth(), "")
146 message = t.S().Error.Render(v.fit(message, v.textWidth()-2)) // -2 for padding
147 return joinHeaderBody(header, message)
148}
149
150// Register tool renderers
151func init() {
152 registry.register(tools.BashToolName, func() renderer { return bashRenderer{} })
153 registry.register(tools.ViewToolName, func() renderer { return viewRenderer{} })
154 registry.register(tools.EditToolName, func() renderer { return editRenderer{} })
155 registry.register(tools.WriteToolName, func() renderer { return writeRenderer{} })
156 registry.register(tools.FetchToolName, func() renderer { return fetchRenderer{} })
157 registry.register(tools.GlobToolName, func() renderer { return globRenderer{} })
158 registry.register(tools.GrepToolName, func() renderer { return grepRenderer{} })
159 registry.register(tools.LSToolName, func() renderer { return lsRenderer{} })
160 registry.register(tools.SourcegraphToolName, func() renderer { return sourcegraphRenderer{} })
161 registry.register(tools.DiagnosticsToolName, func() renderer { return diagnosticsRenderer{} })
162 registry.register(agent.AgentToolName, func() renderer { return agentRenderer{} })
163}
164
165// -----------------------------------------------------------------------------
166// Generic renderer
167// -----------------------------------------------------------------------------
168
169// genericRenderer handles unknown tool types with basic parameter display
170type genericRenderer struct {
171 baseRenderer
172}
173
174// Render displays the tool call with its raw input and plain content output
175func (gr genericRenderer) Render(v *toolCallCmp) string {
176 return gr.renderWithParams(v, prettifyToolName(v.call.Name), []string{v.call.Input}, func() string {
177 return renderPlainContent(v, v.result.Content)
178 })
179}
180
181// -----------------------------------------------------------------------------
182// Bash renderer
183// -----------------------------------------------------------------------------
184
185// bashRenderer handles bash command execution display
186type bashRenderer struct {
187 baseRenderer
188}
189
190// Render displays the bash command with sanitized newlines and plain output
191func (br bashRenderer) Render(v *toolCallCmp) string {
192 var params tools.BashParams
193 if err := br.unmarshalParams(v.call.Input, ¶ms); err != nil {
194 return br.renderError(v, "Invalid bash parameters")
195 }
196
197 cmd := strings.ReplaceAll(params.Command, "\n", " ")
198 args := newParamBuilder().addMain(cmd).build()
199
200 return br.renderWithParams(v, "Bash", args, func() string {
201 return renderPlainContent(v, v.result.Content)
202 })
203}
204
205// -----------------------------------------------------------------------------
206// View renderer
207// -----------------------------------------------------------------------------
208
209// viewRenderer handles file viewing with syntax highlighting and line numbers
210type viewRenderer struct {
211 baseRenderer
212}
213
214// Render displays file content with optional limit and offset parameters
215func (vr viewRenderer) Render(v *toolCallCmp) string {
216 var params tools.ViewParams
217 if err := vr.unmarshalParams(v.call.Input, ¶ms); err != nil {
218 return vr.renderError(v, "Invalid view parameters")
219 }
220
221 file := fsext.PrettyPath(params.FilePath)
222 args := newParamBuilder().
223 addMain(file).
224 addKeyValue("limit", formatNonZero(params.Limit)).
225 addKeyValue("offset", formatNonZero(params.Offset)).
226 build()
227
228 return vr.renderWithParams(v, "View", args, func() string {
229 var meta tools.ViewResponseMetadata
230 if err := vr.unmarshalParams(v.result.Metadata, &meta); err != nil {
231 return renderPlainContent(v, v.result.Content)
232 }
233 return renderCodeContent(v, meta.FilePath, meta.Content, params.Offset)
234 })
235}
236
237// formatNonZero returns string representation of non-zero integers, empty string for zero
238func formatNonZero(value int) string {
239 if value == 0 {
240 return ""
241 }
242 return fmt.Sprintf("%d", value)
243}
244
245// -----------------------------------------------------------------------------
246// Edit renderer
247// -----------------------------------------------------------------------------
248
249// editRenderer handles file editing with diff visualization
250type editRenderer struct {
251 baseRenderer
252}
253
254// Render displays the edited file with a formatted diff of changes
255func (er editRenderer) Render(v *toolCallCmp) string {
256 var params tools.EditParams
257 if err := er.unmarshalParams(v.call.Input, ¶ms); err != nil {
258 return er.renderError(v, "Invalid edit parameters")
259 }
260
261 file := fsext.PrettyPath(params.FilePath)
262 args := newParamBuilder().addMain(file).build()
263
264 return er.renderWithParams(v, "Edit", args, func() string {
265 var meta tools.EditResponseMetadata
266 if err := er.unmarshalParams(v.result.Metadata, &meta); err != nil {
267 return renderPlainContent(v, v.result.Content)
268 }
269
270 formatter := core.DiffFormatter().
271 Before(fsext.PrettyPath(params.FilePath), meta.OldContent).
272 After(fsext.PrettyPath(params.FilePath), meta.NewContent).
273 Width(v.textWidth() - 2) // -2 for padding
274 if v.textWidth() > 120 {
275 formatter = formatter.Split()
276 }
277 return formatter.String()
278 })
279}
280
281// -----------------------------------------------------------------------------
282// Write renderer
283// -----------------------------------------------------------------------------
284
285// writeRenderer handles file writing with syntax-highlighted content preview
286type writeRenderer struct {
287 baseRenderer
288}
289
290// Render displays the file being written with syntax highlighting
291func (wr writeRenderer) Render(v *toolCallCmp) string {
292 var params tools.WriteParams
293 if err := wr.unmarshalParams(v.call.Input, ¶ms); err != nil {
294 return wr.renderError(v, "Invalid write parameters")
295 }
296
297 file := fsext.PrettyPath(params.FilePath)
298 args := newParamBuilder().addMain(file).build()
299
300 return wr.renderWithParams(v, "Write", args, func() string {
301 return renderCodeContent(v, file, params.Content, 0)
302 })
303}
304
305// -----------------------------------------------------------------------------
306// Fetch renderer
307// -----------------------------------------------------------------------------
308
309// fetchRenderer handles URL fetching with format-specific content display
310type fetchRenderer struct {
311 baseRenderer
312}
313
314// Render displays the fetched URL with format and timeout parameters
315func (fr fetchRenderer) Render(v *toolCallCmp) string {
316 var params tools.FetchParams
317 if err := fr.unmarshalParams(v.call.Input, ¶ms); err != nil {
318 return fr.renderError(v, "Invalid fetch parameters")
319 }
320
321 args := newParamBuilder().
322 addMain(params.URL).
323 addKeyValue("format", params.Format).
324 addKeyValue("timeout", formatTimeout(params.Timeout)).
325 build()
326
327 return fr.renderWithParams(v, "Fetch", args, func() string {
328 file := fr.getFileExtension(params.Format)
329 return renderCodeContent(v, file, v.result.Content, 0)
330 })
331}
332
333// getFileExtension returns appropriate file extension for syntax highlighting
334func (fr fetchRenderer) getFileExtension(format string) string {
335 switch format {
336 case "text":
337 return "fetch.txt"
338 case "html":
339 return "fetch.html"
340 default:
341 return "fetch.md"
342 }
343}
344
345// formatTimeout converts timeout seconds to duration string
346func formatTimeout(timeout int) string {
347 if timeout == 0 {
348 return ""
349 }
350 return (time.Duration(timeout) * time.Second).String()
351}
352
353// -----------------------------------------------------------------------------
354// Glob renderer
355// -----------------------------------------------------------------------------
356
357// globRenderer handles file pattern matching with path filtering
358type globRenderer struct {
359 baseRenderer
360}
361
362// Render displays the glob pattern with optional path parameter
363func (gr globRenderer) Render(v *toolCallCmp) string {
364 var params tools.GlobParams
365 if err := gr.unmarshalParams(v.call.Input, ¶ms); err != nil {
366 return gr.renderError(v, "Invalid glob parameters")
367 }
368
369 args := newParamBuilder().
370 addMain(params.Pattern).
371 addKeyValue("path", params.Path).
372 build()
373
374 return gr.renderWithParams(v, "Glob", args, func() string {
375 return renderPlainContent(v, v.result.Content)
376 })
377}
378
379// -----------------------------------------------------------------------------
380// Grep renderer
381// -----------------------------------------------------------------------------
382
383// grepRenderer handles content searching with pattern matching options
384type grepRenderer struct {
385 baseRenderer
386}
387
388// Render displays the search pattern with path, include, and literal text options
389func (gr grepRenderer) Render(v *toolCallCmp) string {
390 var params tools.GrepParams
391 if err := gr.unmarshalParams(v.call.Input, ¶ms); err != nil {
392 return gr.renderError(v, "Invalid grep parameters")
393 }
394
395 args := newParamBuilder().
396 addMain(params.Pattern).
397 addKeyValue("path", params.Path).
398 addKeyValue("include", params.Include).
399 addFlag("literal", params.LiteralText).
400 build()
401
402 return gr.renderWithParams(v, "Grep", args, func() string {
403 return renderPlainContent(v, v.result.Content)
404 })
405}
406
407// -----------------------------------------------------------------------------
408// LS renderer
409// -----------------------------------------------------------------------------
410
411// lsRenderer handles directory listing with default path handling
412type lsRenderer struct {
413 baseRenderer
414}
415
416// Render displays the directory path, defaulting to current directory
417func (lr lsRenderer) Render(v *toolCallCmp) string {
418 var params tools.LSParams
419 if err := lr.unmarshalParams(v.call.Input, ¶ms); err != nil {
420 return lr.renderError(v, "Invalid ls parameters")
421 }
422
423 path := params.Path
424 if path == "" {
425 path = "."
426 }
427 path = fsext.PrettyPath(path)
428
429 args := newParamBuilder().addMain(path).build()
430
431 return lr.renderWithParams(v, "List", args, func() string {
432 return renderPlainContent(v, v.result.Content)
433 })
434}
435
436// -----------------------------------------------------------------------------
437// Sourcegraph renderer
438// -----------------------------------------------------------------------------
439
440// sourcegraphRenderer handles code search with count and context options
441type sourcegraphRenderer struct {
442 baseRenderer
443}
444
445// Render displays the search query with optional count and context window parameters
446func (sr sourcegraphRenderer) Render(v *toolCallCmp) string {
447 var params tools.SourcegraphParams
448 if err := sr.unmarshalParams(v.call.Input, ¶ms); err != nil {
449 return sr.renderError(v, "Invalid sourcegraph parameters")
450 }
451
452 args := newParamBuilder().
453 addMain(params.Query).
454 addKeyValue("count", formatNonZero(params.Count)).
455 addKeyValue("context", formatNonZero(params.ContextWindow)).
456 build()
457
458 return sr.renderWithParams(v, "Sourcegraph", args, func() string {
459 return renderPlainContent(v, v.result.Content)
460 })
461}
462
463// -----------------------------------------------------------------------------
464// Diagnostics renderer
465// -----------------------------------------------------------------------------
466
467// diagnosticsRenderer handles project-wide diagnostic information
468type diagnosticsRenderer struct {
469 baseRenderer
470}
471
472// Render displays project diagnostics with plain content formatting
473func (dr diagnosticsRenderer) Render(v *toolCallCmp) string {
474 args := newParamBuilder().addMain("project").build()
475
476 return dr.renderWithParams(v, "Diagnostics", args, func() string {
477 return renderPlainContent(v, v.result.Content)
478 })
479}
480
481// -----------------------------------------------------------------------------
482// Task renderer
483// -----------------------------------------------------------------------------
484
485// agentRenderer handles project-wide diagnostic information
486type agentRenderer struct {
487 baseRenderer
488}
489
490func RoundedEnumerator(children tree.Children, index int) string {
491 if children.Length()-1 == index {
492 return " ╰──"
493 }
494 return " ├──"
495}
496
497// Render displays agent task parameters and result content
498func (tr agentRenderer) Render(v *toolCallCmp) string {
499 t := styles.CurrentTheme()
500 var params agent.AgentParams
501 if err := tr.unmarshalParams(v.call.Input, ¶ms); err != nil {
502 return tr.renderError(v, "Invalid task parameters")
503 }
504 prompt := params.Prompt
505 prompt = strings.ReplaceAll(prompt, "\n", " ")
506
507 header := tr.makeHeader(v, "Agent", v.textWidth())
508 taskTag := t.S().Base.Padding(0, 1).MarginLeft(1).Background(t.BlueLight).Foreground(t.White).Render("Task")
509 remainingWidth := v.textWidth() - lipgloss.Width(header) - lipgloss.Width(taskTag) - 2 // -2 for padding
510 prompt = t.S().Muted.Width(remainingWidth).Render(prompt)
511 header = lipgloss.JoinVertical(
512 lipgloss.Left,
513 header,
514 "",
515 lipgloss.JoinHorizontal(
516 lipgloss.Left,
517 taskTag,
518 " ",
519 prompt,
520 ),
521 )
522 childTools := tree.Root(header)
523
524 for _, call := range v.nestedToolCalls {
525 childTools.Child(call.View())
526 }
527 parts := []string{
528 childTools.Enumerator(RoundedEnumerator).String(),
529 }
530 if v.result.ToolCallID == "" {
531 v.spinning = true
532 parts = append(parts, v.anim.View().String())
533 } else {
534 v.spinning = false
535 }
536
537 header = lipgloss.JoinVertical(
538 lipgloss.Left,
539 parts...,
540 )
541
542 if v.result.ToolCallID == "" {
543 return header
544 }
545
546 body := renderPlainContent(v, v.result.Content)
547 return joinHeaderBody(header, body)
548}
549
550// renderParamList renders params, params[0] (params[1]=params[2] ....)
551func renderParamList(nested bool, paramsWidth int, params ...string) string {
552 t := styles.CurrentTheme()
553 if len(params) == 0 {
554 return ""
555 }
556 mainParam := params[0]
557 if len(mainParam) > paramsWidth {
558 mainParam = mainParam[:paramsWidth-3] + "..."
559 }
560
561 if len(params) == 1 {
562 if nested {
563 return t.S().Muted.Render(mainParam)
564 }
565 return t.S().Subtle.Render(mainParam)
566 }
567 otherParams := params[1:]
568 // create pairs of key/value
569 // if odd number of params, the last one is a key without value
570 if len(otherParams)%2 != 0 {
571 otherParams = append(otherParams, "")
572 }
573 parts := make([]string, 0, len(otherParams)/2)
574 for i := 0; i < len(otherParams); i += 2 {
575 key := otherParams[i]
576 value := otherParams[i+1]
577 if value == "" {
578 continue
579 }
580 parts = append(parts, fmt.Sprintf("%s=%s", key, value))
581 }
582
583 partsRendered := strings.Join(parts, ", ")
584 remainingWidth := paramsWidth - lipgloss.Width(partsRendered) - 3 // count for " ()"
585 if remainingWidth < 30 {
586 if nested {
587 return t.S().Muted.Render(mainParam)
588 }
589 // No space for the params, just show the main
590 return t.S().Subtle.Render(mainParam)
591 }
592
593 if len(parts) > 0 {
594 mainParam = fmt.Sprintf("%s (%s)", mainParam, strings.Join(parts, ", "))
595 }
596
597 if nested {
598 return t.S().Muted.Render(ansi.Truncate(mainParam, paramsWidth, "..."))
599 }
600 return t.S().Subtle.Render(ansi.Truncate(mainParam, paramsWidth, "..."))
601}
602
603// earlyState returns immediately‑rendered error/cancelled/ongoing states.
604func earlyState(header string, v *toolCallCmp) (string, bool) {
605 t := styles.CurrentTheme()
606 message := ""
607 switch {
608 case v.result.IsError:
609 message = v.renderToolError()
610 case v.cancelled:
611 message = "Cancelled"
612 case v.result.ToolCallID == "":
613 message = "Waiting for tool to start..."
614 default:
615 return "", false
616 }
617
618 message = t.S().Base.PaddingLeft(2).Render(message)
619 return lipgloss.JoinVertical(lipgloss.Left, header, message), true
620}
621
622func joinHeaderBody(header, body string) string {
623 t := styles.CurrentTheme()
624 body = t.S().Base.PaddingLeft(2).Render(body)
625 return lipgloss.JoinVertical(lipgloss.Left, header, "", body, "")
626}
627
628func renderPlainContent(v *toolCallCmp, content string) string {
629 t := styles.CurrentTheme()
630 content = strings.TrimSpace(content)
631 lines := strings.Split(content, "\n")
632
633 width := v.textWidth() - 2 // -2 for left padding
634 var out []string
635 for i, ln := range lines {
636 if i >= responseContextHeight {
637 break
638 }
639 ln = " " + ln // left padding
640 if len(ln) > width {
641 ln = v.fit(ln, width)
642 }
643 out = append(out, t.S().Muted.
644 Width(width).
645 Background(t.BgBaseLighter).
646 Render(ln))
647 }
648
649 if len(lines) > responseContextHeight {
650 out = append(out, t.S().Muted.
651 Background(t.BgBaseLighter).
652 Width(width).
653 Render(fmt.Sprintf("... (%d lines)", len(lines)-responseContextHeight)))
654 }
655 return strings.Join(out, "\n")
656}
657
658func pad(v any, width int) string {
659 s := fmt.Sprintf("%v", v)
660 w := ansi.StringWidth(s)
661 if w >= width {
662 return s
663 }
664 return strings.Repeat(" ", width-w) + s
665}
666
667func renderCodeContent(v *toolCallCmp, path, content string, offset int) string {
668 t := styles.CurrentTheme()
669 truncated := truncateHeight(content, responseContextHeight)
670
671 highlighted, _ := highlight.SyntaxHighlight(truncated, path, t.BgBase)
672 lines := strings.Split(highlighted, "\n")
673
674 if len(strings.Split(content, "\n")) > responseContextHeight {
675 lines = append(lines, t.S().Muted.
676 Background(t.BgBase).
677 Render(fmt.Sprintf(" …(%d lines)", len(strings.Split(content, "\n"))-responseContextHeight)))
678 }
679
680 maxLineNumber := len(lines) + offset
681 padding := lipgloss.Width(fmt.Sprintf("%d", maxLineNumber))
682 for i, ln := range lines {
683 num := t.S().Base.
684 Foreground(t.FgMuted).
685 Background(t.BgBase).
686 PaddingRight(1).
687 PaddingLeft(1).
688 Render(pad(i+1+offset, padding))
689 w := v.textWidth() - 10 - lipgloss.Width(num) // -4 for left padding
690 lines[i] = lipgloss.JoinHorizontal(lipgloss.Left,
691 num,
692 t.S().Base.
693 PaddingLeft(1).
694 Render(v.fit(ln, w-1)))
695 }
696 return lipgloss.JoinVertical(lipgloss.Left, lines...)
697}
698
699func (v *toolCallCmp) renderToolError() string {
700 t := styles.CurrentTheme()
701 err := strings.ReplaceAll(v.result.Content, "\n", " ")
702 err = fmt.Sprintf("Error: %s", err)
703 return t.S().Base.Foreground(t.Error).Render(v.fit(err, v.textWidth()-2))
704}
705
706func truncateHeight(s string, h int) string {
707 lines := strings.Split(s, "\n")
708 if len(lines) > h {
709 return strings.Join(lines[:h], "\n")
710 }
711 return s
712}
713
714func prettifyToolName(name string) string {
715 switch name {
716 case agent.AgentToolName:
717 return "Agent"
718 case tools.BashToolName:
719 return "Bash"
720 case tools.EditToolName:
721 return "Edit"
722 case tools.FetchToolName:
723 return "Fetch"
724 case tools.GlobToolName:
725 return "Glob"
726 case tools.GrepToolName:
727 return "Grep"
728 case tools.LSToolName:
729 return "List"
730 case tools.SourcegraphToolName:
731 return "Sourcegraph"
732 case tools.ViewToolName:
733 return "View"
734 case tools.WriteToolName:
735 return "Write"
736 default:
737 return name
738 }
739}