Skip to main content
Glama

Azure MCP Server

Official
MIT License
1,161
  • Linux
  • Apple
BlobUploadCommandTests.cs7.99 kB
// Copyright (c) Microsoft Corporation. // Licensed under the MIT License. using System.CommandLine.Parsing; using AzureMcp.Core.Models.Command; using AzureMcp.Storage.Commands.Blob; using AzureMcp.Storage.Models; using AzureMcp.Storage.Services; using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.Logging; using NSubstitute; using NSubstitute.ExceptionExtensions; using Xunit; namespace AzureMcp.Storage.UnitTests.Blob; public class BlobUploadCommandTests { private readonly IServiceProvider _serviceProvider; private readonly IStorageService _storageService; private readonly ILogger<BlobUploadCommand> _logger; private readonly BlobUploadCommand _command; private readonly CommandContext _context; private readonly Parser _parser; private readonly string _knownAccount = "account123"; private readonly string _knownContainer = "container123"; private readonly string _knownBlob = "test-blob.txt"; private readonly string _knownLocalFilePath = "C:\\temp\\test-file.txt"; private readonly string _knownSubscription = "sub123"; public BlobUploadCommandTests() { _storageService = Substitute.For<IStorageService>(); _logger = Substitute.For<ILogger<BlobUploadCommand>>(); var collection = new ServiceCollection().AddSingleton(_storageService); _serviceProvider = collection.BuildServiceProvider(); _command = new(_logger); _context = new(_serviceProvider); _parser = new(_command.GetCommand()); } [Fact] public async Task ExecuteAsync_UploadsBlobSuccessfully() { // Arrange var uploadResult = new BlobUploadResult( Blob: _knownBlob, Container: _knownContainer, UploadedFile: Path.GetFileName(_knownLocalFilePath), LastModified: DateTimeOffset.UtcNow, ETag: "\"0x8D123456789ABCD\"", MD5Hash: "abc123def456" ); _storageService.UploadBlob( _knownAccount, _knownContainer, _knownBlob, _knownLocalFilePath, false, _knownSubscription, Arg.Any<string?>(), Arg.Any<Core.Options.RetryPolicyOptions?>()) .Returns(uploadResult); var args = _parser.Parse([ "--account", _knownAccount, "--container", _knownContainer, "--blob", _knownBlob, "--local-file-path", _knownLocalFilePath, "--subscription", _knownSubscription ]); // Act var response = await _command.ExecuteAsync(_context, args); // Assert Assert.NotNull(response); Assert.NotNull(response.Results); Assert.Equal(200, response.Status); } [Fact] public async Task ExecuteAsync_WithOverwriteOption_UploadsBlobWithOverwrite() { // Arrange var uploadResult = new BlobUploadResult( Blob: _knownBlob, Container: _knownContainer, UploadedFile: Path.GetFileName(_knownLocalFilePath), LastModified: DateTimeOffset.UtcNow, ETag: "\"0x8D123456789ABCD\"", MD5Hash: "abc123def456" ); _storageService.UploadBlob( _knownAccount, _knownContainer, _knownBlob, _knownLocalFilePath, true, _knownSubscription, Arg.Any<string?>(), Arg.Any<Core.Options.RetryPolicyOptions?>()) .Returns(uploadResult); var args = _parser.Parse([ "--account", _knownAccount, "--container", _knownContainer, "--blob", _knownBlob, "--local-file-path", _knownLocalFilePath, "--overwrite", "--subscription", _knownSubscription ]); // Act var response = await _command.ExecuteAsync(_context, args); // Assert Assert.NotNull(response); Assert.NotNull(response.Results); Assert.Equal(200, response.Status); } [Fact] public async Task ExecuteAsync_FileNotFoundError_ReturnsError() { // Arrange _storageService.UploadBlob( Arg.Any<string>(), Arg.Any<string>(), Arg.Any<string>(), Arg.Any<string>(), Arg.Any<bool>(), Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<Core.Options.RetryPolicyOptions?>()) .ThrowsAsync(new FileNotFoundException($"Local file not found: {_knownLocalFilePath}")); var args = _parser.Parse([ "--account", _knownAccount, "--container", _knownContainer, "--blob", _knownBlob, "--local-file-path", _knownLocalFilePath, "--subscription", _knownSubscription ]); // Act var response = await _command.ExecuteAsync(_context, args); // Assert Assert.NotEqual(200, response.Status); Assert.Contains("Local file not found", response.Message); } [Fact] public async Task ExecuteAsync_BlobAlreadyExistsError_ReturnsError() { // Arrange _storageService.UploadBlob( Arg.Any<string>(), Arg.Any<string>(), Arg.Any<string>(), Arg.Any<string>(), Arg.Any<bool>(), Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<Core.Options.RetryPolicyOptions?>()) .ThrowsAsync(new InvalidOperationException("Blob 'test-blob.txt' already exists in container 'container123'. Use --overwrite to replace it.")); var args = _parser.Parse([ "--account", _knownAccount, "--container", _knownContainer, "--blob", _knownBlob, "--local-file-path", _knownLocalFilePath, "--subscription", _knownSubscription ]); // Act var response = await _command.ExecuteAsync(_context, args); // Assert Assert.NotEqual(200, response.Status); Assert.Contains("already exists", response.Message); } [Theory] [InlineData("")] [InlineData("--account account123")] [InlineData("--account account123 --container container123")] [InlineData("--account account123 --container container123 --blob test-blob.txt")] [InlineData("--account account123 --container container123 --blob test-blob.txt --local-file-path /path/to/file.txt")] public async Task ExecuteAsync_MissingRequiredOptions_ReturnsValidationError(string argString) { // Arrange var args = string.IsNullOrEmpty(argString) ? _parser.Parse([]) : _parser.Parse(argString.Split(' ', StringSplitOptions.RemoveEmptyEntries)); // Act var response = await _command.ExecuteAsync(_context, args); // Assert Assert.NotEqual(200, response.Status); } [Fact] public async Task ExecuteAsync_AccessDeniedError_ReturnsError() { // Arrange var requestFailedException = new Azure.RequestFailedException(403, "Access denied"); _storageService.UploadBlob( Arg.Any<string>(), Arg.Any<string>(), Arg.Any<string>(), Arg.Any<string>(), Arg.Any<bool>(), Arg.Any<string>(), Arg.Any<string?>(), Arg.Any<Core.Options.RetryPolicyOptions?>()) .ThrowsAsync(requestFailedException); var args = _parser.Parse([ "--account", _knownAccount, "--container", _knownContainer, "--blob", _knownBlob, "--local-file-path", _knownLocalFilePath, "--subscription", _knownSubscription ]); // Act var response = await _command.ExecuteAsync(_context, args); // Assert Assert.NotEqual(200, response.Status); Assert.Contains("Access denied", response.Message); } }

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