mirror of
https://github.com/amir20/dozzle.git
synced 2026-06-23 04:10:12 +00:00
365 lines
10 KiB
Go
365 lines
10 KiB
Go
package mcp
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"io"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/amir20/dozzle/internal/container"
|
|
container_support "github.com/amir20/dozzle/internal/support/container"
|
|
"github.com/amir20/dozzle/internal/utils"
|
|
"github.com/modelcontextprotocol/go-sdk/mcp"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
type mockHostService struct {
|
|
containers []container.Container
|
|
hosts []container.Host
|
|
findErr error
|
|
listErrs []error
|
|
logEvents []*container.LogEvent
|
|
logErr error
|
|
}
|
|
|
|
type stubClientService struct {
|
|
container_support.ClientService
|
|
events []*container.LogEvent
|
|
err error
|
|
}
|
|
|
|
func (s *stubClientService) LogsBetweenDates(ctx context.Context, _ container.Container, _ time.Time, _ time.Time, _ container.StdType) (<-chan *container.LogEvent, error) {
|
|
if s.err != nil {
|
|
return nil, s.err
|
|
}
|
|
ch := make(chan *container.LogEvent)
|
|
go func() {
|
|
defer close(ch)
|
|
for _, e := range s.events {
|
|
select {
|
|
case ch <- e:
|
|
case <-ctx.Done():
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
return ch, nil
|
|
}
|
|
|
|
func (s *stubClientService) RawLogs(context.Context, container.Container, time.Time, time.Time, container.StdType) (io.ReadCloser, error) {
|
|
return nil, fmt.Errorf("not implemented")
|
|
}
|
|
|
|
func (m *mockHostService) FindContainer(host string, id string, labels container.ContainerLabels) (*container_support.ContainerService, error) {
|
|
if m.findErr != nil {
|
|
return nil, m.findErr
|
|
}
|
|
for _, c := range m.containers {
|
|
if c.ID == id && c.Host == host {
|
|
stub := &stubClientService{events: m.logEvents, err: m.logErr}
|
|
return container_support.NewContainerService(stub, c), nil
|
|
}
|
|
}
|
|
return nil, assert.AnError
|
|
}
|
|
|
|
func (m *mockHostService) ListAllContainers(labels container.ContainerLabels) ([]container.Container, []error) {
|
|
return m.containers, m.listErrs
|
|
}
|
|
|
|
func (m *mockHostService) Hosts() []container.Host {
|
|
return m.hosts
|
|
}
|
|
|
|
func TestListContainers(t *testing.T) {
|
|
svc := &mockHostService{
|
|
containers: []container.Container{
|
|
{ID: "abc123", Name: "web", Image: "nginx:latest", State: "running", Host: "local"},
|
|
{ID: "def456", Name: "db", Image: "postgres:15", State: "exited", Host: "local"},
|
|
},
|
|
}
|
|
|
|
s := NewServer(svc, nil, "test")
|
|
|
|
ctx := context.Background()
|
|
ct, st := mcp.NewInMemoryTransports()
|
|
|
|
_, err := s.mcpServer.Connect(ctx, st, nil)
|
|
require.NoError(t, err)
|
|
|
|
client := mcp.NewClient(&mcp.Implementation{Name: "test-client", Version: "v0.0.1"}, nil)
|
|
session, err := client.Connect(ctx, ct, nil)
|
|
require.NoError(t, err)
|
|
defer session.Close()
|
|
|
|
// Test listing all containers
|
|
result, err := session.CallTool(ctx, &mcp.CallToolParams{
|
|
Name: "list_containers",
|
|
Arguments: map[string]any{},
|
|
})
|
|
require.NoError(t, err)
|
|
assert.False(t, result.IsError)
|
|
assert.Len(t, result.Content, 1)
|
|
text := result.Content[0].(*mcp.TextContent).Text
|
|
assert.Contains(t, text, "abc123")
|
|
assert.Contains(t, text, "def456")
|
|
|
|
// Test filtering by state
|
|
result, err = session.CallTool(ctx, &mcp.CallToolParams{
|
|
Name: "list_containers",
|
|
Arguments: map[string]any{"state": "running"},
|
|
})
|
|
require.NoError(t, err)
|
|
assert.False(t, result.IsError)
|
|
text = result.Content[0].(*mcp.TextContent).Text
|
|
assert.Contains(t, text, "abc123")
|
|
assert.NotContains(t, text, "def456")
|
|
}
|
|
|
|
func TestListContainersPartialFailure(t *testing.T) {
|
|
svc := &mockHostService{
|
|
containers: []container.Container{
|
|
{ID: "abc123", Name: "web", Image: "nginx:latest", State: "running", Host: "host1"},
|
|
},
|
|
listErrs: []error{nil, fmt.Errorf("host2 unreachable")},
|
|
}
|
|
|
|
s := NewServer(svc, nil, "test")
|
|
|
|
ctx := context.Background()
|
|
ct, st := mcp.NewInMemoryTransports()
|
|
|
|
_, err := s.mcpServer.Connect(ctx, st, nil)
|
|
require.NoError(t, err)
|
|
|
|
client := mcp.NewClient(&mcp.Implementation{Name: "test-client", Version: "v0.0.1"}, nil)
|
|
session, err := client.Connect(ctx, ct, nil)
|
|
require.NoError(t, err)
|
|
defer session.Close()
|
|
|
|
result, err := session.CallTool(ctx, &mcp.CallToolParams{
|
|
Name: "list_containers",
|
|
Arguments: map[string]any{},
|
|
})
|
|
require.NoError(t, err)
|
|
assert.False(t, result.IsError)
|
|
text := result.Content[0].(*mcp.TextContent).Text
|
|
assert.Contains(t, text, "abc123")
|
|
}
|
|
|
|
func TestListHosts(t *testing.T) {
|
|
svc := &mockHostService{
|
|
hosts: []container.Host{
|
|
{ID: "local", Name: "localhost", NCPU: 4, MemTotal: 8000000000, DockerVersion: "24.0", Type: "local", Available: true},
|
|
},
|
|
}
|
|
|
|
s := NewServer(svc, nil, "test")
|
|
|
|
ctx := context.Background()
|
|
ct, st := mcp.NewInMemoryTransports()
|
|
|
|
_, err := s.mcpServer.Connect(ctx, st, nil)
|
|
require.NoError(t, err)
|
|
|
|
client := mcp.NewClient(&mcp.Implementation{Name: "test-client", Version: "v0.0.1"}, nil)
|
|
session, err := client.Connect(ctx, ct, nil)
|
|
require.NoError(t, err)
|
|
defer session.Close()
|
|
|
|
result, err := session.CallTool(ctx, &mcp.CallToolParams{
|
|
Name: "list_hosts",
|
|
Arguments: map[string]any{},
|
|
})
|
|
require.NoError(t, err)
|
|
assert.False(t, result.IsError)
|
|
text := result.Content[0].(*mcp.TextContent).Text
|
|
assert.Contains(t, text, "localhost")
|
|
assert.Contains(t, text, "24.0")
|
|
}
|
|
|
|
func TestGetContainerStats(t *testing.T) {
|
|
stats := utils.RingBufferFrom(300, []container.ContainerStat{
|
|
{ID: "abc123", CPUPercent: 25.5, MemoryPercent: 50.0, MemoryUsage: 1024000},
|
|
{ID: "abc123", CPUPercent: 30.0, MemoryPercent: 55.0, MemoryUsage: 1100000},
|
|
})
|
|
|
|
svc := &mockHostService{
|
|
containers: []container.Container{
|
|
{ID: "abc123", Name: "web", Host: "local", Stats: stats, MemoryLimit: 2048000, CPULimit: 2.0},
|
|
},
|
|
}
|
|
|
|
s := NewServer(svc, nil, "test")
|
|
|
|
ctx := context.Background()
|
|
ct, st := mcp.NewInMemoryTransports()
|
|
|
|
_, err := s.mcpServer.Connect(ctx, st, nil)
|
|
require.NoError(t, err)
|
|
|
|
client := mcp.NewClient(&mcp.Implementation{Name: "test-client", Version: "v0.0.1"}, nil)
|
|
session, err := client.Connect(ctx, ct, nil)
|
|
require.NoError(t, err)
|
|
defer session.Close()
|
|
|
|
result, err := session.CallTool(ctx, &mcp.CallToolParams{
|
|
Name: "get_container_stats",
|
|
Arguments: map[string]any{"host": "local", "container_id": "abc123"},
|
|
})
|
|
require.NoError(t, err)
|
|
assert.False(t, result.IsError)
|
|
text := result.Content[0].(*mcp.TextContent).Text
|
|
assert.Contains(t, text, "25.5")
|
|
assert.Contains(t, text, "2048000")
|
|
}
|
|
|
|
func TestGetContainerStatsNotFound(t *testing.T) {
|
|
svc := &mockHostService{
|
|
containers: []container.Container{},
|
|
}
|
|
|
|
s := NewServer(svc, nil, "test")
|
|
|
|
ctx := context.Background()
|
|
ct, st := mcp.NewInMemoryTransports()
|
|
|
|
_, err := s.mcpServer.Connect(ctx, st, nil)
|
|
require.NoError(t, err)
|
|
|
|
client := mcp.NewClient(&mcp.Implementation{Name: "test-client", Version: "v0.0.1"}, nil)
|
|
session, err := client.Connect(ctx, ct, nil)
|
|
require.NoError(t, err)
|
|
defer session.Close()
|
|
|
|
result, err := session.CallTool(ctx, &mcp.CallToolParams{
|
|
Name: "get_container_stats",
|
|
Arguments: map[string]any{"host": "local", "container_id": "nonexistent"},
|
|
})
|
|
require.NoError(t, err)
|
|
assert.True(t, result.IsError)
|
|
}
|
|
|
|
func TestGetContainerLogsRequiredParams(t *testing.T) {
|
|
svc := &mockHostService{}
|
|
|
|
s := NewServer(svc, nil, "test")
|
|
|
|
ctx := context.Background()
|
|
ct, st := mcp.NewInMemoryTransports()
|
|
|
|
_, err := s.mcpServer.Connect(ctx, st, nil)
|
|
require.NoError(t, err)
|
|
|
|
client := mcp.NewClient(&mcp.Implementation{Name: "test-client", Version: "v0.0.1"}, nil)
|
|
session, err := client.Connect(ctx, ct, nil)
|
|
require.NoError(t, err)
|
|
defer session.Close()
|
|
|
|
// Missing required params
|
|
result, err := session.CallTool(ctx, &mcp.CallToolParams{
|
|
Name: "get_container_logs",
|
|
Arguments: map[string]any{},
|
|
})
|
|
require.NoError(t, err)
|
|
assert.True(t, result.IsError)
|
|
}
|
|
|
|
func TestNewServerRegistersTools(t *testing.T) {
|
|
svc := &mockHostService{}
|
|
s := NewServer(svc, nil, "v1.0.0")
|
|
|
|
ctx := context.Background()
|
|
ct, st := mcp.NewInMemoryTransports()
|
|
|
|
_, err := s.mcpServer.Connect(ctx, st, nil)
|
|
require.NoError(t, err)
|
|
|
|
client := mcp.NewClient(&mcp.Implementation{Name: "test-client", Version: "v0.0.1"}, nil)
|
|
session, err := client.Connect(ctx, ct, nil)
|
|
require.NoError(t, err)
|
|
defer session.Close()
|
|
|
|
tools, err := session.ListTools(ctx, nil)
|
|
require.NoError(t, err)
|
|
|
|
toolNames := make([]string, len(tools.Tools))
|
|
for i, tool := range tools.Tools {
|
|
toolNames[i] = tool.Name
|
|
}
|
|
|
|
assert.Contains(t, toolNames, "list_containers")
|
|
assert.Contains(t, toolNames, "get_container_logs")
|
|
assert.Contains(t, toolNames, "list_hosts")
|
|
assert.Contains(t, toolNames, "get_container_stats")
|
|
assert.Len(t, tools.Tools, 4)
|
|
}
|
|
|
|
func TestGetContainerLogs(t *testing.T) {
|
|
now := time.Now()
|
|
svc := &mockHostService{
|
|
containers: []container.Container{
|
|
{ID: "abc123", Name: "web", Host: "local"},
|
|
},
|
|
logEvents: []*container.LogEvent{
|
|
{Timestamp: now.UnixMilli(), Level: "info", Stream: "stdout", Type: container.LogTypeSingle, RawMessage: "hello"},
|
|
{Timestamp: now.UnixMilli(), Level: "error", Stream: "stderr", Type: container.LogTypeSingle, RawMessage: "boom"},
|
|
},
|
|
}
|
|
|
|
s := NewServer(svc, nil, "test")
|
|
|
|
ctx := context.Background()
|
|
ct, st := mcp.NewInMemoryTransports()
|
|
|
|
_, err := s.mcpServer.Connect(ctx, st, nil)
|
|
require.NoError(t, err)
|
|
|
|
client := mcp.NewClient(&mcp.Implementation{Name: "test-client", Version: "v0.0.1"}, nil)
|
|
session, err := client.Connect(ctx, ct, nil)
|
|
require.NoError(t, err)
|
|
defer session.Close()
|
|
|
|
result, err := session.CallTool(ctx, &mcp.CallToolParams{
|
|
Name: "get_container_logs",
|
|
Arguments: map[string]any{"host": "local", "container_id": "abc123"},
|
|
})
|
|
require.NoError(t, err)
|
|
assert.False(t, result.IsError)
|
|
text := result.Content[0].(*mcp.TextContent).Text
|
|
assert.Contains(t, text, "hello")
|
|
assert.Contains(t, text, "boom")
|
|
}
|
|
|
|
func TestGetContainerLogsInvalidStream(t *testing.T) {
|
|
svc := &mockHostService{
|
|
containers: []container.Container{
|
|
{ID: "abc123", Name: "web", Host: "local"},
|
|
},
|
|
}
|
|
|
|
s := NewServer(svc, nil, "test")
|
|
|
|
ctx := context.Background()
|
|
ct, st := mcp.NewInMemoryTransports()
|
|
|
|
_, err := s.mcpServer.Connect(ctx, st, nil)
|
|
require.NoError(t, err)
|
|
|
|
client := mcp.NewClient(&mcp.Implementation{Name: "test-client", Version: "v0.0.1"}, nil)
|
|
session, err := client.Connect(ctx, ct, nil)
|
|
require.NoError(t, err)
|
|
defer session.Close()
|
|
|
|
result, err := session.CallTool(ctx, &mcp.CallToolParams{
|
|
Name: "get_container_logs",
|
|
Arguments: map[string]any{"host": "local", "container_id": "abc123", "stream": "bogus"},
|
|
})
|
|
require.NoError(t, err)
|
|
assert.True(t, result.IsError)
|
|
text := result.Content[0].(*mcp.TextContent).Text
|
|
assert.Contains(t, text, "invalid stream")
|
|
}
|