mirror of
https://github.com/browseros-ai/BrowserOS.git
synced 2026-05-13 23:53:25 +00:00
git-subtree-dir: packages/browseros-agent git-subtree-mainline:8f148d0918git-subtree-split:90bd4be300
136 lines
2.7 KiB
Go
136 lines
2.7 KiB
Go
package output
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"os"
|
|
|
|
"browseros-cli/mcp"
|
|
|
|
"github.com/fatih/color"
|
|
)
|
|
|
|
var (
|
|
errColor = color.New(color.FgRed, color.Bold)
|
|
dimColor = color.New(color.Faint)
|
|
boldColor = color.New(color.Bold)
|
|
)
|
|
|
|
// JSON outputs raw JSON to stdout. If structuredContent exists, use it;
|
|
// otherwise fall back to serializing the full result.
|
|
func JSON(result *mcp.ToolResult) {
|
|
if result.StructuredContent != nil {
|
|
data, _ := json.Marshal(result.StructuredContent)
|
|
fmt.Println(string(data))
|
|
return
|
|
}
|
|
data, _ := json.Marshal(result)
|
|
fmt.Println(string(data))
|
|
}
|
|
|
|
// JSONRaw outputs any value as JSON.
|
|
func JSONRaw(v any) {
|
|
data, _ := json.Marshal(v)
|
|
fmt.Println(string(data))
|
|
}
|
|
|
|
// Text prints a text result to stdout.
|
|
func Text(result *mcp.ToolResult) {
|
|
fmt.Println(result.TextContent())
|
|
}
|
|
|
|
// Confirm prints a short confirmation message.
|
|
func Confirm(msg string) {
|
|
fmt.Println(msg)
|
|
}
|
|
|
|
// Error prints an error to stderr and exits with the given code.
|
|
func Error(msg string, code int) {
|
|
errColor.Fprintf(os.Stderr, "Error: %s\n", msg)
|
|
os.Exit(code)
|
|
}
|
|
|
|
// Errorf formats and prints an error to stderr and exits.
|
|
func Errorf(code int, format string, args ...any) {
|
|
errColor.Fprintf(os.Stderr, "Error: "+format+"\n", args...)
|
|
os.Exit(code)
|
|
}
|
|
|
|
// PageList formats a list of pages for human display.
|
|
func PageList(result *mcp.ToolResult) {
|
|
if result.StructuredContent == nil {
|
|
Text(result)
|
|
return
|
|
}
|
|
|
|
pages, ok := result.StructuredContent["pages"].([]any)
|
|
if !ok {
|
|
Text(result)
|
|
return
|
|
}
|
|
|
|
if len(pages) == 0 {
|
|
fmt.Println("No pages open.")
|
|
return
|
|
}
|
|
|
|
for _, p := range pages {
|
|
page, ok := p.(map[string]any)
|
|
if !ok {
|
|
continue
|
|
}
|
|
pageID := intVal(page["pageId"])
|
|
tabID := intVal(page["tabId"])
|
|
title := strVal(page["title"])
|
|
url := strVal(page["url"])
|
|
active := boolVal(page["isActive"])
|
|
|
|
marker := ""
|
|
if active {
|
|
marker = " " + boldColor.Sprint("[ACTIVE]")
|
|
}
|
|
|
|
fmt.Printf(" %d. %s (tab %d)%s\n", pageID, title, tabID, marker)
|
|
fmt.Printf(" %s\n", dimColor.Sprint(url))
|
|
}
|
|
}
|
|
|
|
// ActivePage formats a single active page for human display.
|
|
func ActivePage(result *mcp.ToolResult) {
|
|
if result.StructuredContent == nil {
|
|
Text(result)
|
|
return
|
|
}
|
|
|
|
sc := result.StructuredContent
|
|
pageID := intVal(sc["pageId"])
|
|
tabID := intVal(sc["tabId"])
|
|
title := strVal(sc["title"])
|
|
url := strVal(sc["url"])
|
|
|
|
fmt.Printf("Active page: %d (tab %d)\n", pageID, tabID)
|
|
fmt.Println(title)
|
|
fmt.Println(dimColor.Sprint(url))
|
|
}
|
|
|
|
func intVal(v any) int {
|
|
if f, ok := v.(float64); ok {
|
|
return int(f)
|
|
}
|
|
return 0
|
|
}
|
|
|
|
func strVal(v any) string {
|
|
if s, ok := v.(string); ok {
|
|
return s
|
|
}
|
|
return ""
|
|
}
|
|
|
|
func boolVal(v any) bool {
|
|
if b, ok := v.(bool); ok {
|
|
return b
|
|
}
|
|
return false
|
|
}
|