Files
goclaw/internal/mcp/bridge_tool_test.go
T
viettranx 78abdec887 feat(mcp+skills): per-agent registry isolation, skill access filtering, managed skill lifecycle (#57)
Security: fix cross-agent MCP tool leak by cloning tool registry before MCP registration.

MCP: enforce mcp_ prefix on all tool names, add cache invalidation on server/grant changes,
add grant management endpoints, add group:mcp policy support for per-agent allowlisting.

Skills: persist full YAML frontmatter, auto-promote/demote visibility on grant/revoke,
simplify versioning, handle ZIP wrapper directories, expand tilde in skillsDir path.

Fixes: wrap DeleteSkill cascade in transaction, use atomic NOT EXISTS for revoke-demote,
create cancel context before storing server in map.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-04 23:14:31 +07:00

145 lines
3.8 KiB
Go

package mcp
import (
"testing"
mcpgo "github.com/mark3labs/mcp-go/mcp"
)
func TestInputSchemaToMap(t *testing.T) {
schema := mcpgo.ToolInputSchema{
Type: "object",
Properties: map[string]any{
"query": map[string]any{
"type": "string",
"description": "Search query",
},
},
Required: []string{"query"},
}
m := inputSchemaToMap(schema)
if m["type"] != "object" {
t.Errorf("expected type=object, got %v", m["type"])
}
props, ok := m["properties"].(map[string]any)
if !ok || props == nil {
t.Fatal("expected properties map")
}
if _, ok := props["query"]; !ok {
t.Error("expected 'query' in properties")
}
req, ok := m["required"].([]string)
if !ok || len(req) != 1 || req[0] != "query" {
t.Errorf("expected required=[query], got %v", m["required"])
}
}
func TestInputSchemaToMap_EmptyType(t *testing.T) {
schema := mcpgo.ToolInputSchema{}
m := inputSchemaToMap(schema)
if m["type"] != "object" {
t.Errorf("expected default type=object, got %v", m["type"])
}
}
func TestExtractTextContent(t *testing.T) {
result := &mcpgo.CallToolResult{
Content: []mcpgo.Content{
mcpgo.TextContent{Type: "text", Text: "hello"},
mcpgo.TextContent{Type: "text", Text: "world"},
},
}
got := extractTextContent(result)
if got != "hello\nworld" {
t.Errorf("expected 'hello\\nworld', got %q", got)
}
}
func TestExtractTextContent_Nil(t *testing.T) {
if got := extractTextContent(nil); got != "" {
t.Errorf("expected empty for nil, got %q", got)
}
result := &mcpgo.CallToolResult{}
if got := extractTextContent(result); got != "" {
t.Errorf("expected empty for no content, got %q", got)
}
}
func TestBridgeToolNaming(t *testing.T) {
mcpTool := mcpgo.Tool{
Name: "query",
Description: "Run a query",
InputSchema: mcpgo.ToolInputSchema{Type: "object"},
}
// Without prefix → auto-derived from server name
bt := NewBridgeTool("myserver", mcpTool, nil, "", 30, nil)
if bt.Name() != "mcp_myserver__query" {
t.Errorf("expected name=mcp_myserver__query, got %s", bt.Name())
}
if bt.ServerName() != "myserver" {
t.Errorf("expected serverName=myserver, got %s", bt.ServerName())
}
if bt.OriginalName() != "query" {
t.Errorf("expected originalName=query, got %s", bt.OriginalName())
}
// With non-mcp_ prefix → gets mcp_ prepended
bt2 := NewBridgeTool("myserver", mcpTool, nil, "pg", 0, nil)
if bt2.Name() != "mcp_pg__query" {
t.Errorf("expected name=mcp_pg__query, got %s", bt2.Name())
}
if bt2.OriginalName() != "query" {
t.Errorf("expected originalName=query, got %s", bt2.OriginalName())
}
// With mcp_ prefix → unchanged
bt3 := NewBridgeTool("myserver", mcpTool, nil, "mcp_pg", 0, nil)
if bt3.Name() != "mcp_pg__query" {
t.Errorf("expected name=mcp_pg__query, got %s", bt3.Name())
}
// Server name with hyphens → sanitized to underscores
bt4 := NewBridgeTool("my-server", mcpTool, nil, "", 0, nil)
if bt4.Name() != "mcp_my_server__query" {
t.Errorf("expected name=mcp_my_server__query, got %s", bt4.Name())
}
// Default timeout
if bt2.timeoutSec != 60 {
t.Errorf("expected default timeout=60, got %d", bt2.timeoutSec)
}
}
func TestEnsureMCPPrefix(t *testing.T) {
tests := []struct {
name string
prefix string
serverName string
want string
}{
{"empty prefix", "", "vnstock", "mcp_vnstock"},
{"empty prefix hyphenated server", "", "my-server", "mcp_my_server"},
{"non-mcp prefix", "pg", "postgres", "mcp_pg"},
{"already mcp_ prefix", "mcp_pg", "postgres", "mcp_pg"},
{"mcp prefix without underscore", "mcp", "x", "mcp_mcp"},
{"custom prefix with underscores", "vnstock", "vnstock", "mcp_vnstock"},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
got := ensureMCPPrefix(tt.prefix, tt.serverName)
if got != tt.want {
t.Errorf("ensureMCPPrefix(%q, %q) = %q, want %q", tt.prefix, tt.serverName, got, tt.want)
}
})
}
}