|
| 1 | +package oauth |
| 2 | + |
| 3 | +import ( |
| 4 | + "context" |
| 5 | + "errors" |
| 6 | + "testing" |
| 7 | + |
| 8 | + "github.com/stretchr/testify/assert" |
| 9 | + "github.com/stretchr/testify/require" |
| 10 | + |
| 11 | + oauthhelpers "github.com/docker/mcp-gateway-oauth-helpers" |
| 12 | + "github.com/docker/mcp-gateway/pkg/desktop" |
| 13 | +) |
| 14 | + |
| 15 | +// mockDCRClient implements dcrRegistrationClient for testing. |
| 16 | +type mockDCRClient struct { |
| 17 | + clients map[string]*desktop.DCRClient |
| 18 | + registered map[string]desktop.RegisterDCRRequest |
| 19 | +} |
| 20 | + |
| 21 | +func newMockDCRClient() *mockDCRClient { |
| 22 | + return &mockDCRClient{ |
| 23 | + clients: make(map[string]*desktop.DCRClient), |
| 24 | + registered: make(map[string]desktop.RegisterDCRRequest), |
| 25 | + } |
| 26 | +} |
| 27 | + |
| 28 | +func (m *mockDCRClient) GetDCRClient(_ context.Context, app string) (*desktop.DCRClient, error) { |
| 29 | + c, ok := m.clients[app] |
| 30 | + if !ok { |
| 31 | + return nil, errors.New("not found") |
| 32 | + } |
| 33 | + return c, nil |
| 34 | +} |
| 35 | + |
| 36 | +func (m *mockDCRClient) RegisterDCRClientPending(_ context.Context, app string, req desktop.RegisterDCRRequest) error { |
| 37 | + m.registered[app] = req |
| 38 | + return nil |
| 39 | +} |
| 40 | + |
| 41 | +// mockProber implements oauthProber for testing. |
| 42 | +type mockProber struct { |
| 43 | + discovery *oauthhelpers.Discovery |
| 44 | + err error |
| 45 | +} |
| 46 | + |
| 47 | +func (m *mockProber) DiscoverOAuthRequirements(_ context.Context, _ string) (*oauthhelpers.Discovery, error) { |
| 48 | + return m.discovery, m.err |
| 49 | +} |
| 50 | + |
| 51 | +func TestRegisterProviderForDynamicDiscovery_SkipsAlreadyRegistered(t *testing.T) { |
| 52 | + client := newMockDCRClient() |
| 53 | + client.clients["my-server"] = &desktop.DCRClient{State: "unregistered"} |
| 54 | + |
| 55 | + prober := &mockProber{} // should not be called |
| 56 | + |
| 57 | + err := registerProviderForDynamicDiscovery(t.Context(), "my-server", "https://example.com/mcp", client, prober) |
| 58 | + require.NoError(t, err) |
| 59 | + assert.Empty(t, client.registered, "should not register when already exists") |
| 60 | +} |
| 61 | + |
| 62 | +func TestRegisterProviderForDynamicDiscovery_RegistersWhenOAuthRequired(t *testing.T) { |
| 63 | + client := newMockDCRClient() |
| 64 | + prober := &mockProber{ |
| 65 | + discovery: &oauthhelpers.Discovery{RequiresOAuth: true}, |
| 66 | + } |
| 67 | + |
| 68 | + err := registerProviderForDynamicDiscovery(t.Context(), "ai-kubit-mcp-server", "https://mcp.kubit.ai/mcp", client, prober) |
| 69 | + require.NoError(t, err) |
| 70 | + |
| 71 | + req, ok := client.registered["ai-kubit-mcp-server"] |
| 72 | + require.True(t, ok, "should have registered DCR client") |
| 73 | + assert.Equal(t, "ai-kubit-mcp-server", req.ProviderName, "provider name should match server name") |
| 74 | +} |
| 75 | + |
| 76 | +func TestRegisterProviderForDynamicDiscovery_SkipsWhenNoOAuthRequired(t *testing.T) { |
| 77 | + client := newMockDCRClient() |
| 78 | + prober := &mockProber{ |
| 79 | + discovery: &oauthhelpers.Discovery{RequiresOAuth: false}, |
| 80 | + } |
| 81 | + |
| 82 | + err := registerProviderForDynamicDiscovery(t.Context(), "my-server", "https://example.com/mcp", client, prober) |
| 83 | + require.NoError(t, err) |
| 84 | + assert.Empty(t, client.registered, "should not register when OAuth not required") |
| 85 | +} |
| 86 | + |
| 87 | +func TestRegisterProviderForDynamicDiscovery_SkipsOnProbeError(t *testing.T) { |
| 88 | + client := newMockDCRClient() |
| 89 | + prober := &mockProber{ |
| 90 | + err: errors.New("connection refused"), |
| 91 | + } |
| 92 | + |
| 93 | + err := registerProviderForDynamicDiscovery(t.Context(), "my-server", "https://unreachable.example.com/mcp", client, prober) |
| 94 | + require.NoError(t, err) |
| 95 | + assert.Empty(t, client.registered, "should not register when probe fails") |
| 96 | +} |
0 commit comments