using Xunit;
using OpenHarbor.MCP.Gateway.Infrastructure.Security;
using OpenHarbor.MCP.Gateway.Core.Models;
namespace OpenHarbor.MCP.Gateway.Infrastructure.Tests.Security;
///
/// Unit tests for ApiKeyAuthProvider following TDD approach.
/// Tests API key-based authentication.
///
public class ApiKeyAuthProviderTests
{
[Fact]
public async Task AuthenticateAsync_WithValidApiKey_ReturnsSuccess()
{
// Arrange
var validKeys = new Dictionary
{
["client-1"] = "secret-key-123",
["client-2"] = "secret-key-456"
};
var provider = new ApiKeyAuthProvider(validKeys);
var context = new AuthenticationContext
{
ClientId = "client-1",
Credentials = "secret-key-123"
};
// Act
var result = await provider.AuthenticateAsync(context);
// Assert
Assert.True(result.IsAuthenticated);
Assert.Equal("client-1", result.ClientId);
Assert.Null(result.ErrorMessage);
}
[Fact]
public async Task AuthenticateAsync_WithInvalidApiKey_ReturnsFailure()
{
// Arrange
var validKeys = new Dictionary
{
["client-1"] = "secret-key-123"
};
var provider = new ApiKeyAuthProvider(validKeys);
var context = new AuthenticationContext
{
ClientId = "client-1",
Credentials = "wrong-key"
};
// Act
var result = await provider.AuthenticateAsync(context);
// Assert
Assert.False(result.IsAuthenticated);
Assert.Null(result.ClientId);
Assert.NotNull(result.ErrorMessage);
Assert.Contains("Invalid API key", result.ErrorMessage);
}
[Fact]
public async Task AuthenticateAsync_WithUnknownClient_ReturnsFailure()
{
// Arrange
var validKeys = new Dictionary
{
["client-1"] = "secret-key-123"
};
var provider = new ApiKeyAuthProvider(validKeys);
var context = new AuthenticationContext
{
ClientId = "unknown-client",
Credentials = "any-key"
};
// Act
var result = await provider.AuthenticateAsync(context);
// Assert
Assert.False(result.IsAuthenticated);
Assert.Contains("Unknown client", result.ErrorMessage);
}
[Fact]
public async Task AuthenticateAsync_WithNullCredentials_ReturnsFailure()
{
// Arrange
var provider = new ApiKeyAuthProvider(new Dictionary
{
["client-1"] = "secret-key-123"
});
var context = new AuthenticationContext
{
ClientId = "client-1",
Credentials = null
};
// Act
var result = await provider.AuthenticateAsync(context);
// Assert
Assert.False(result.IsAuthenticated);
Assert.Contains("credentials", result.ErrorMessage, StringComparison.OrdinalIgnoreCase);
}
[Fact]
public async Task AuthenticateAsync_WithEmptyClientId_ReturnsFailure()
{
// Arrange
var provider = new ApiKeyAuthProvider(new Dictionary
{
["client-1"] = "secret-key-123"
});
var context = new AuthenticationContext
{
ClientId = "",
Credentials = "secret-key-123"
};
// Act
var result = await provider.AuthenticateAsync(context);
// Assert
Assert.False(result.IsAuthenticated);
Assert.Contains("Client ID", result.ErrorMessage);
}
[Fact]
public void Constructor_WithNullKeys_ThrowsArgumentNullException()
{
// Act & Assert
Assert.Throws(() => new ApiKeyAuthProvider(null!));
}
[Fact]
public async Task AuthenticateAsync_WithNullContext_ThrowsArgumentNullException()
{
// Arrange
var provider = new ApiKeyAuthProvider(new Dictionary());
// Act & Assert
await Assert.ThrowsAsync(() => provider.AuthenticateAsync(null!));
}
[Fact]
public async Task AuthenticateAsync_CaseSensitiveApiKey_ReturnsFailure()
{
// Arrange
var validKeys = new Dictionary
{
["client-1"] = "SecretKey123"
};
var provider = new ApiKeyAuthProvider(validKeys);
var context = new AuthenticationContext
{
ClientId = "client-1",
Credentials = "secretkey123" // Wrong case
};
// Act
var result = await provider.AuthenticateAsync(context);
// Assert
Assert.False(result.IsAuthenticated);
}
[Fact]
public async Task AuthorizeAsync_WithValidClient_ReturnsSuccess()
{
// Arrange
var validKeys = new Dictionary
{
["client-1"] = "secret-key-123"
};
var provider = new ApiKeyAuthProvider(validKeys);
var context = new AuthorizationContext
{
ClientId = "client-1",
Resource = "search_codex",
Action = "invoke"
};
// Act
var result = await provider.AuthorizeAsync(context);
// Assert
Assert.True(result.IsAuthorized);
}
[Fact]
public async Task AuthorizeAsync_WithNullContext_ThrowsArgumentNullException()
{
// Arrange
var provider = new ApiKeyAuthProvider(new Dictionary());
// Act & Assert
await Assert.ThrowsAsync(() => provider.AuthorizeAsync(null!));
}
}