MetricsDefinitionsCommandTests.cs•23.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
}