Skip to main content
Glama

Azure MCP Server

Official
MIT License
1,161
  • Linux
  • Apple
MetricsDefinitionsCommandTests.cs23.4 kB
// Copyright (c) Microsoft Corporation. // Licensed under the MIT License. using System.CommandLine; using System.CommandLine.Parsing; using AzureMcp.Core.Models.Command; using AzureMcp.Core.Options; using AzureMcp.Monitor.Commands.Metrics; using AzureMcp.Monitor.Models; using AzureMcp.Monitor.Services; using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.Logging; using NSubstitute; using Xunit; namespace AzureMcp.Monitor.UnitTests.Metrics; public class MetricsDefinitionsCommandTests { private readonly IServiceProvider _serviceProvider; private readonly IMonitorMetricsService _service; private readonly ILogger<MetricsDefinitionsCommand> _logger; private readonly MetricsDefinitionsCommand _command; public MetricsDefinitionsCommandTests() { _service = Substitute.For<IMonitorMetricsService>(); _logger = Substitute.For<ILogger<MetricsDefinitionsCommand>>(); var collection = new ServiceCollection(); collection.AddSingleton(_service); _serviceProvider = collection.BuildServiceProvider(); _command = new(_logger); } #region Constructor and Command Setup Tests [Fact] public void Constructor_InitializesCommandCorrectly() { var command = _command.GetCommand(); Assert.Equal("definitions", command.Name); Assert.NotNull(command.Description); Assert.NotEmpty(command.Description); Assert.Contains("List available metric definitions", command.Description); } [Fact] public void Name_ReturnsDefinitions() { Assert.Equal("definitions", _command.Name); } [Fact] public void Title_ReturnsCorrectTitle() { Assert.Equal("List Azure Monitor Metric Definitions", _command.Title); } [Fact] public void GetCommand_RegistersAllRequiredOptions() { var command = _command.GetCommand(); // Check that all required options are present var optionNames = command.Options.Select(o => o.Name).ToList(); Assert.Contains("subscription", optionNames); Assert.Contains("resource-type", optionNames); Assert.Contains("resource", optionNames); Assert.Contains("metric-namespace", optionNames); Assert.Contains("search-string", optionNames); Assert.Contains("limit", optionNames); Assert.Contains("tenant", optionNames); // Note: resource-group may not be registered as a separate option if resource-id parsing is used } #endregion #region Validation Tests [Theory] [InlineData("--resource test --subscription sub1", true)] [InlineData("--subscription sub1", false)] [InlineData("--resource test", false)] [InlineData("", false)] public async Task ExecuteAsync_ValidatesInputCorrectly(string args, bool shouldSucceed) { // Arrange if (shouldSucceed) { _service.ListMetricDefinitionsAsync( Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<RetryPolicyOptions?>()) .Returns(new List<MetricDefinition> { new() { Name = "CPU", Description = "CPU Percentage", Category = "Performance", Unit = "Percent", SupportedAggregationTypes = new List<string> { "Average", "Maximum", "Minimum" }, IsDimensionRequired = true, Dimensions = new List<string> { "Instance" } } }); } var context = new CommandContext(_serviceProvider); var parseResult = _command.GetCommand().Parse(args); // Act var response = await _command.ExecuteAsync(context, parseResult); // Assert Assert.Equal(shouldSucceed ? 200 : 400, response.Status); if (shouldSucceed) { Assert.NotNull(response.Results); } else { Assert.Contains("required", response.Message.ToLower()); } } #endregion #region Service Interaction Tests [Fact] public async Task ExecuteAsync_CallsServiceWithCorrectParameters() { // Arrange var expectedResults = new List<MetricDefinition> { new() { Name = "CPU Percentage", Description = "Average CPU usage", Category = "Performance", Unit = "Percent", SupportedAggregationTypes = new List<string> { "Average" }, IsDimensionRequired = false, Dimensions = new List<string>() } }; _service.ListMetricDefinitionsAsync( "sub1", null, // resource-group may be null if not provided or not parsed from resource-id "Microsoft.Storage/storageAccounts", "test", "Microsoft.Storage/storageAccounts", null, "tenant1", Arg.Any<RetryPolicyOptions?>()) .Returns(expectedResults); var context = new CommandContext(_serviceProvider); var parseResult = _command.GetCommand().Parse( "--resource test --subscription sub1 --resource-type Microsoft.Storage/storageAccounts --metric-namespace Microsoft.Storage/storageAccounts --tenant tenant1"); // Act var response = await _command.ExecuteAsync(context, parseResult); // Assert Assert.Equal(200, response.Status); Assert.NotNull(response.Results); Assert.Equal("All 1 metric definitions returned.", response.Message); await _service.Received(1).ListMetricDefinitionsAsync( "sub1", null, "Microsoft.Storage/storageAccounts", "test", "Microsoft.Storage/storageAccounts", null, "tenant1", Arg.Any<RetryPolicyOptions?>()); } [Fact] public async Task ExecuteAsync_WithSearchString_CallsServiceWithSearchParameter() { // Arrange _service.ListMetricDefinitionsAsync( Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string>(), Arg.Any<string?>(), "cpu", Arg.Any<string?>(), Arg.Any<RetryPolicyOptions?>()) .Returns(new List<MetricDefinition> { new() { Name = "CPU Percentage", Description = "Average CPU usage", Category = "Performance", Unit = "Percent" } }); var context = new CommandContext(_serviceProvider); var parseResult = _command.GetCommand().Parse("--resource test --subscription sub1 --search-string cpu"); // Act var response = await _command.ExecuteAsync(context, parseResult); // Assert Assert.Equal(200, response.Status); Assert.NotNull(response.Results); // Verify the service was called with the search string await _service.Received(1).ListMetricDefinitionsAsync( Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string>(), Arg.Any<string?>(), "cpu", Arg.Any<string?>(), Arg.Any<RetryPolicyOptions?>()); } [Fact] public async Task ExecuteAsync_WithAllOptionalParameters_CallsServiceCorrectly() { // Arrange var expectedResults = new List<MetricDefinition> { new() { Name = "Memory Usage", Description = "Memory usage metrics", Category = "Memory", Unit = "Bytes" } }; _service.ListMetricDefinitionsAsync( "sub1", null, // resource-group may be null if not provided or not parsed from resource-id "Microsoft.Storage/storageAccounts", "test", "Microsoft.Storage/storageAccounts", "memory", "tenant1", Arg.Any<RetryPolicyOptions?>()) .Returns(expectedResults); var context = new CommandContext(_serviceProvider); var parseResult = _command.GetCommand().Parse( "--resource test --subscription sub1 --resource-type Microsoft.Storage/storageAccounts --metric-namespace Microsoft.Storage/storageAccounts --search-string memory --tenant tenant1 --limit 20"); // Act var response = await _command.ExecuteAsync(context, parseResult); // Assert Assert.Equal(200, response.Status); Assert.NotNull(response.Results); Assert.Equal("All 1 metric definitions returned.", response.Message); await _service.Received(1).ListMetricDefinitionsAsync( "sub1", null, "Microsoft.Storage/storageAccounts", "test", "Microsoft.Storage/storageAccounts", "memory", "tenant1", Arg.Any<RetryPolicyOptions?>()); } #endregion #region Error Handling Tests [Fact] public async Task ExecuteAsync_HandlesServiceErrors() { // Arrange _service.ListMetricDefinitionsAsync( Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<RetryPolicyOptions?>()) .Returns(Task.FromException<List<MetricDefinition>>(new Exception("Test error"))); var context = new CommandContext(_serviceProvider); var parseResult = _command.GetCommand().Parse("--resource test --subscription sub1"); // Act var response = await _command.ExecuteAsync(context, parseResult); // Assert Assert.Equal(500, response.Status); Assert.Contains("Test error", response.Message); Assert.Contains("troubleshooting", response.Message); } [Fact] public async Task ExecuteAsync_HandlesServiceException_LogsError() { // Arrange var exception = new Exception("Service unavailable"); _service.ListMetricDefinitionsAsync( Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<RetryPolicyOptions?>()) .Returns(Task.FromException<List<MetricDefinition>>(exception)); var context = new CommandContext(_serviceProvider); var parseResult = _command.GetCommand().Parse("--resource test --subscription sub1 --resource-group rg1"); // Act var response = await _command.ExecuteAsync(context, parseResult); // Assert Assert.Equal(500, response.Status); _logger.Received(1).Log( LogLevel.Error, Arg.Any<EventId>(), Arg.Is<object>(o => o.ToString()!.Contains("Error listing metric definitions")), exception, Arg.Any<Func<object, Exception?, string>>()); } #endregion #region Result Processing Tests [Fact] public async Task ExecuteAsync_WithResults_ReturnsCorrectStructure() { // Arrange var metricDefinitions = new List<MetricDefinition> { new() { Name = "CPU Percentage", Description = "Average CPU usage", Category = "Performance", Unit = "Percent", SupportedAggregationTypes = new List<string> { "Average", "Maximum" }, IsDimensionRequired = false, Dimensions = new List<string>() }, new() { Name = "Memory Usage", Description = "Memory usage in bytes", Category = "Memory", Unit = "Bytes", SupportedAggregationTypes = new List<string> { "Average", "Maximum", "Total" }, IsDimensionRequired = true, Dimensions = new List<string> { "Instance", "Role" } } }; _service.ListMetricDefinitionsAsync( Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<RetryPolicyOptions?>()) .Returns(metricDefinitions); var context = new CommandContext(_serviceProvider); var parseResult = _command.GetCommand().Parse("--resource test --subscription sub1"); // Act var response = await _command.ExecuteAsync(context, parseResult); // Assert Assert.Equal(200, response.Status); Assert.NotNull(response.Results); Assert.Equal("All 2 metric definitions returned.", response.Message); } [Fact] public async Task ExecuteAsync_WithNoResults_ReturnsNullResults() { // Arrange _service.ListMetricDefinitionsAsync( Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<RetryPolicyOptions?>()) .Returns(new List<MetricDefinition>()); var context = new CommandContext(_serviceProvider); var parseResult = _command.GetCommand().Parse("--resource test --subscription sub1"); // Act var response = await _command.ExecuteAsync(context, parseResult); // Assert Assert.Equal(200, response.Status); Assert.Null(response.Results); } [Fact] public async Task ExecuteAsync_WithNullResults_ReturnsNullResults() { // Arrange _service.ListMetricDefinitionsAsync( Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<RetryPolicyOptions?>()) .Returns(Task.FromResult<List<MetricDefinition>>(null!)); var context = new CommandContext(_serviceProvider); var parseResult = _command.GetCommand().Parse("--resource test --subscription sub1"); // Act var response = await _command.ExecuteAsync(context, parseResult); // Assert Assert.Equal(200, response.Status); Assert.Null(response.Results); } #endregion #region Limit Processing Tests [Fact] public async Task ExecuteAsync_WithDefaultLimit_TruncatesResultsTo10() { // Arrange var metricDefinitions = GenerateMetricDefinitions(15); // More than default limit of 10 _service.ListMetricDefinitionsAsync( Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<RetryPolicyOptions?>()) .Returns(metricDefinitions); var context = new CommandContext(_serviceProvider); var parseResult = _command.GetCommand().Parse("--resource test --subscription sub1"); // Act var response = await _command.ExecuteAsync(context, parseResult); // Assert Assert.Equal(200, response.Status); Assert.NotNull(response.Results); // Verify that results were truncated - message should indicate truncation Assert.Contains("Results truncated to 10 of 15", response.Message); Assert.Contains("metric definitions", response.Message); // Verify service receives all data but command applies limit internally await _service.Received(1).ListMetricDefinitionsAsync(Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<RetryPolicyOptions?>()); } [Fact] public async Task ExecuteAsync_WithCustomLimit_TruncatesResultsCorrectly() { // Arrange var metricDefinitions = GenerateMetricDefinitions(20); _service.ListMetricDefinitionsAsync( Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<RetryPolicyOptions?>()) .Returns(metricDefinitions); var context = new CommandContext(_serviceProvider); var parseResult = _command.GetCommand().Parse("--resource test --subscription sub1 --limit 5"); // Act var response = await _command.ExecuteAsync(context, parseResult); // Assert Assert.Equal(200, response.Status); Assert.NotNull(response.Results); // Verify that results were truncated to the custom limit Assert.Contains("Results truncated to 5 of 20", response.Message); Assert.Contains("metric definitions", response.Message); // Verify service is called correctly await _service.Received(1).ListMetricDefinitionsAsync(Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<RetryPolicyOptions?>()); } [Fact] public async Task ExecuteAsync_WithResultsExceedingLimit_ShowsTruncationMessage() { // Arrange - Create more results than the limit var metricDefinitions = GenerateMetricDefinitions(25); _service.ListMetricDefinitionsAsync( Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<RetryPolicyOptions?>()) .Returns(metricDefinitions); var context = new CommandContext(_serviceProvider); var parseResult = _command.GetCommand().Parse("--resource test --subscription sub1 --limit 8"); // Act var response = await _command.ExecuteAsync(context, parseResult); // Assert Assert.Equal(200, response.Status); Assert.NotNull(response.Results); // Verify that the message indicates truncation with correct counts Assert.Contains("Results truncated to 8 of 25", response.Message); Assert.Contains("Use --search-string to filter results", response.Message); // Verify the service was called await _service.Received(1).ListMetricDefinitionsAsync(Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<RetryPolicyOptions?>()); } [Fact] public async Task ExecuteAsync_WithResultsUnderLimit_ShowsAllResultsMessage() { // Arrange - Create fewer results than the limit var metricDefinitions = GenerateMetricDefinitions(3); _service.ListMetricDefinitionsAsync( Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<RetryPolicyOptions?>()) .Returns(metricDefinitions); var context = new CommandContext(_serviceProvider); var parseResult = _command.GetCommand().Parse("--resource test --subscription sub1 --limit 10"); // Act var response = await _command.ExecuteAsync(context, parseResult); // Assert Assert.Equal(200, response.Status); Assert.NotNull(response.Results); // Verify that all results are returned without truncation Assert.Equal("All 3 metric definitions returned.", response.Message); // Verify the service was called await _service.Received(1).ListMetricDefinitionsAsync(Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<string?>(), Arg.Any<RetryPolicyOptions?>()); } #endregion #region Option Binding Tests [Fact] public async Task ExecuteAsync_BindsOptionsCorrectly() { // Arrange var expectedResults = new List<MetricDefinition> { new() { Name = "Performance Counter", Description = "VM performance metrics", Category = "Performance", Unit = "Count" } }; _service.ListMetricDefinitionsAsync( "test-subscription", null, // resource-group may be null if not provided or not parsed from resource-id "Microsoft.Compute/virtualMachines", "test-vm", "Microsoft.Compute/virtualMachines", "performance", "test-tenant", Arg.Any<RetryPolicyOptions?>()) .Returns(expectedResults); var context = new CommandContext(_serviceProvider); var parseResult = _command.GetCommand().Parse( "--subscription test-subscription --resource-type Microsoft.Compute/virtualMachines --resource test-vm --metric-namespace Microsoft.Compute/virtualMachines --search-string performance --tenant test-tenant --limit 25"); // Act var response = await _command.ExecuteAsync(context, parseResult); // Assert Assert.Equal(200, response.Status); Assert.NotNull(response.Results); Assert.Equal("All 1 metric definitions returned.", response.Message); await _service.Received(1).ListMetricDefinitionsAsync( "test-subscription", null, "Microsoft.Compute/virtualMachines", "test-vm", "Microsoft.Compute/virtualMachines", "performance", "test-tenant", Arg.Any<RetryPolicyOptions?>()); } #endregion #region Helper Methods private static List<MetricDefinition> GenerateMetricDefinitions(int count) { var definitions = new List<MetricDefinition>(); for (int i = 0; i < count; i++) { definitions.Add(new MetricDefinition { Name = $"Metric{i}", Description = $"Description for metric {i}", Category = "Performance", Unit = "Count", SupportedAggregationTypes = new List<string> { "Average" }, IsDimensionRequired = false, Dimensions = new List<string>() }); } return definitions; } #endregion }

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