Skip to main content
Glama

Azure MCP Server

Official
MIT License
1,161
  • Linux
  • Apple
MySqlServiceQueryValidationTests.cs6.84 kB
// Copyright (c) Microsoft Corporation. // Licensed under the MIT License. using AzureMcp.Core.Services.Azure.ResourceGroup; using AzureMcp.Core.Services.Azure.Tenant; using AzureMcp.MySql.Services; using Microsoft.Extensions.Logging; using NSubstitute; using Xunit; using System.Reflection; namespace AzureMcp.MySql.UnitTests.Services; public class MySqlServiceQueryValidationTests { private readonly IResourceGroupService _resourceGroupService; private readonly ITenantService _tenantService; private readonly ILogger<MySqlService> _logger; private readonly MySqlService _mysqlService; public MySqlServiceQueryValidationTests() { _resourceGroupService = Substitute.For<IResourceGroupService>(); _tenantService = Substitute.For<ITenantService>(); _logger = Substitute.For<ILogger<MySqlService>>(); _mysqlService = new MySqlService(_resourceGroupService, _tenantService, _logger); } [Theory] [InlineData("SELECT * FROM users LIMIT 100")] [InlineData("SELECT COUNT(*) FROM products LIMIT 1")] public void ValidateQuerySafety_WithSafeQueries_ShouldNotThrow(string query) { // Arrange var validateMethod = GetValidateQuerySafetyMethod(); // Act & Assert - Should not throw any exception validateMethod.Invoke(null, new object[] { query }); } [Theory] [InlineData("DROP TABLE users")] [InlineData("DELETE FROM users")] [InlineData("INSERT INTO users")] [InlineData("UPDATE users SET")] [InlineData("CREATE TABLE test")] [InlineData("GRANT ALL PRIVILEGES")] [InlineData("LOAD DATA INFILE")] [InlineData("SELECT * INTO OUTFILE")] public void ValidateQuerySafety_WithDangerousQueries_ShouldThrowInvalidOperationException(string query) { // Arrange var validateMethod = GetValidateQuerySafetyMethod(); // Act & Assert var exception = Assert.Throws<TargetInvocationException>(() => validateMethod.Invoke(null, new object[] { query })); Assert.IsType<InvalidOperationException>(exception.InnerException); Assert.True( exception.InnerException!.Message.Contains("dangerous keyword") || exception.InnerException.Message.Contains("dangerous patterns"), $"Expected error message to contain either 'dangerous keyword' or 'dangerous patterns', but got: {exception.InnerException.Message}"); } [Theory] [InlineData("SHOW DATABASES")] [InlineData("DESCRIBE users")] [InlineData("EXPLAIN SELECT * FROM users")] public void ValidateQuerySafety_WithDisallowedStatements_ShouldThrowInvalidOperationException(string query) { // Arrange var validateMethod = GetValidateQuerySafetyMethod(); // Act & Assert var exception = Assert.Throws<TargetInvocationException>(() => validateMethod.Invoke(null, new object[] { query })); Assert.IsType<InvalidOperationException>(exception.InnerException); Assert.Contains("Only SELECT statements are allowed", exception.InnerException!.Message); } [Theory] [InlineData("")] [InlineData(" ")] public void ValidateQuerySafety_WithEmptyQuery_ShouldThrowArgumentException(string query) { // Arrange var validateMethod = GetValidateQuerySafetyMethod(); // Act & Assert var exception = Assert.Throws<TargetInvocationException>(() => validateMethod.Invoke(null, new object[] { query })); Assert.IsType<ArgumentException>(exception.InnerException); Assert.Contains("Query cannot be null or empty", exception.InnerException!.Message); } [Fact] public void ValidateQuerySafety_WithNullQuery_ShouldThrowArgumentException() { // Arrange var validateMethod = GetValidateQuerySafetyMethod(); // Act & Assert var exception = Assert.Throws<TargetInvocationException>(() => validateMethod.Invoke(null, new object[] { null! })); Assert.IsType<ArgumentException>(exception.InnerException); Assert.Contains("Query cannot be null or empty", exception.InnerException!.Message); } [Fact] public void ValidateQuerySafety_WithLongQuery_ShouldThrowInvalidOperationException() { // Arrange var validateMethod = GetValidateQuerySafetyMethod(); var longQuery = "SELECT * FROM users WHERE " + new string('X', 10000); // Act & Assert var exception = Assert.Throws<TargetInvocationException>(() => validateMethod.Invoke(null, new object[] { longQuery })); Assert.IsType<InvalidOperationException>(exception.InnerException); Assert.Contains("Query length exceeds the maximum allowed limit of 10,000 characters", exception.InnerException!.Message); } [Theory] [InlineData("SELECT * FROM users; DROP TABLE users")] [InlineData("SELECT * FROM users; SELECT * FROM products")] public void ValidateQuerySafety_WithMultipleStatements_ShouldThrowInvalidOperationException(string query) { // Arrange var validateMethod = GetValidateQuerySafetyMethod(); // Act & Assert var exception = Assert.Throws<TargetInvocationException>(() => validateMethod.Invoke(null, new object[] { query })); Assert.IsType<InvalidOperationException>(exception.InnerException); Assert.Contains("Query contains dangerous patterns that could indicate SQL injection attempts", exception.InnerException!.Message); } [Theory] [InlineData("SELECT HEX('abc') FROM users")] [InlineData("SELECT UNHEX('616263') FROM users")] [InlineData("SELECT CONV('a',16,2) FROM users")] public void ValidateQuerySafety_WithObfuscationFunctions_ShouldThrowInvalidOperationException(string query) { // Arrange var validateMethod = GetValidateQuerySafetyMethod(); // Act & Assert var exception = Assert.Throws<TargetInvocationException>(() => validateMethod.Invoke(null, new object[] { query })); Assert.IsType<InvalidOperationException>(exception.InnerException); Assert.True( exception.InnerException!.Message.Contains("Character conversion and obfuscation functions") || exception.InnerException.Message.Contains("dangerous keyword"), $"Expected obfuscation or keyword validation error, but got: {exception.InnerException.Message}"); } private static MethodInfo GetValidateQuerySafetyMethod() { var method = typeof(MySqlService).GetMethod("ValidateQuerySafety", BindingFlags.NonPublic | BindingFlags.Static); Assert.NotNull(method); return method; } }

MCP directory API

We provide all the information about MCP servers via our MCP API.

curl -X GET 'https://glama.ai/api/mcp/v1/servers/Azure/azure-mcp'

If you have feedback or need assistance with the MCP directory API, please join our Discord server