[C#] .Net 핵심 단위 테스트-Mock IOptions <T>

나는 여기에 분명한 것이 빠져있는 것처럼 느낍니다. .Net Core IOptions 패턴 (?)을 사용하여 옵션을 주입 해야하는 클래스가 있습니다. 해당 클래스를 단위 테스트 할 때 클래스의 기능을 검증하는 옵션의 다양한 버전을 조롱하고 싶습니다. 누구든지 Startup 클래스 외부에서 IOptions를 올바르게 조롱 / 인스턴스화 / 채우기하는 방법을 알고 있습니까?

다음은 작업중 인 클래스 샘플입니다.

설정 / 옵션 모델

using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;

namespace OptionsSample.Models
{
    public class SampleOptions
    {
        public string FirstSetting { get; set; }
        public int SecondSetting { get; set; }
    }
}

설정을 사용하는 테스트 할 클래스 :

using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using OptionsSample.Models
using System.Net.Http;
using Microsoft.Extensions.Options;
using System.IO;
using Microsoft.AspNetCore.Http;
using System.Xml.Linq;
using Newtonsoft.Json;
using System.Dynamic;
using Microsoft.Extensions.Logging;

namespace OptionsSample.Repositories
{
    public class SampleRepo : ISampleRepo
    {
        private SampleOptions _options;
        private ILogger<AzureStorageQueuePassthru> _logger;

        public SampleRepo(IOptions<SampleOptions> options)
        {
            _options = options.Value;
        }

        public async Task Get()
        {
        }
    }
}

다른 클래스와 다른 어셈블리의 단위 테스트 :

using OptionsSample.Repositories;
using OptionsSample.Models;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using Xunit;
using Microsoft.Extensions.Logging;
using Microsoft.AspNetCore.Http;
using Microsoft.Extensions.Options;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Configuration;

namespace OptionsSample.Repositories.Tests
{
    public class SampleRepoTests
    {
        private IOptions<SampleOptions> _options;
        private SampleRepo _sampleRepo;


        public SampleRepoTests()
        {
            //Not sure how to populate IOptions<SampleOptions> here
            _options = options;

            _sampleRepo = new SampleRepo(_options);
        }
    }
}



답변

IOptions<SampleOptions>객체 를 수동으로 생성하고 채워야 합니다. Microsoft.Extensions.Options.Options도우미 클래스 를 통해 그렇게 할 수 있습니다 . 예를 들면 다음과 같습니다.

IOptions<SampleOptions> someOptions = Options.Create<SampleOptions>(new SampleOptions());

다음과 같이 조금 단순화 할 수 있습니다.

var someOptions = Options.Create(new SampleOptions());

분명히 이것은 그다지 유용하지 않습니다. 실제로 SampleOptions 객체를 생성하고 채워서 Create 메서드에 전달해야합니다.


답변

주석에 @TSeng으로 표시된 Mocking Framework를 사용하려는 경우 project.json 파일에 다음 종속성을 추가해야합니다.

   "Moq": "4.6.38-alpha",

일단 종속성이 복원되면, SampleOptions 클래스의 인스턴스를 작성하고 언급 한대로 Value에 지정하는 것처럼 MOQ 프레임 워크를 사용하는 것이 간단합니다.

코드 개요는 다음과 같습니다.

SampleOptions app = new SampleOptions(){Title="New Website Title Mocked"}; // Sample property
// Make sure you include using Moq;
var mock = new Mock<IOptions<SampleOptions>>();
// We need to set the Value of IOptions to be the SampleOptions Class
mock.Setup(ap => ap.Value).Returns(app);

모의 설정이 완료되면 다음과 같이 모의 객체를 생성자에게 전달할 수 있습니다.

SampleRepo sr = new SampleRepo(mock.Object);   

HTH.

참고로 Github / patvin80 에서이 두 가지 접근 방식을 간략하게 설명하는 git 저장소가 있습니다.


답변

MOQ를 전혀 사용하지 않아도됩니다. 테스트 .json 구성 파일에서 사용하십시오. 많은 테스트 클래스 파일을위한 하나의 파일. ConfigurationBuilder이 경우 에 사용 하는 것이 좋습니다.

appsetting.json의 예

{
    "someService" {
        "someProp": "someValue
    }
}

설정 매핑 클래스의 예 :

public class SomeServiceConfiguration
{
     public string SomeProp { get; set; }
}

테스트에 필요한 서비스 예 :

public class SomeService
{
    public SomeService(IOptions<SomeServiceConfiguration> config)
    {
        _config = config ?? throw new ArgumentNullException(nameof(_config));
    }
}

NUnit 테스트 클래스 :

[TestFixture]
public class SomeServiceTests
{

    private IOptions<SomeServiceConfiguration> _config;
    private SomeService _service;

    [OneTimeSetUp]
    public void GlobalPrepare()
    {
         var configuration = new ConfigurationBuilder()
            .SetBasePath(Directory.GetCurrentDirectory())
            .AddJsonFile("appsettings.json", false)
            .Build();

        _config = Options.Create(configuration.GetSection("someService").Get<SomeServiceConfiguration>());
    }

    [SetUp]
    public void PerTestPrepare()
    {
        _service = new SomeService(_config);
    }
}


답변

다음과 같은 클래스 Person가 제공 PersonSettings됩니다.

public class PersonSettings
{
    public string Name;
}

public class Person
{
    PersonSettings _settings;

    public Person(IOptions<PersonSettings> settings)
    {
        _settings = settings.Value;
    }

    public string Name => _settings.Name;
}

IOptions<PersonSettings>조롱하고 Person다음과 같이 테스트 할 수 있습니다.

[TestFixture]
public class Test
{
    ServiceProvider _provider;

    [OneTimeSetUp]
    public void Setup()
    {
        var services = new ServiceCollection();
        // mock PersonSettings
        services.AddTransient<IOptions<PersonSettings>>(
            provider => Options.Create<PersonSettings>(new PersonSettings
            {
                Name = "Matt"
            }));
        _provider = services.BuildServiceProvider();
    }

    [Test]
    public void TestName()
    {
        IOptions<PersonSettings> options = _provider.GetService<IOptions<PersonSettings>>();
        Assert.IsNotNull(options, "options could not be created");

        Person person = new Person(options);
        Assert.IsTrue(person.Name == "Matt", "person is not Matt");
    }
}

주입 IOptions<PersonSettings>Person대신의 ctor에 명시 적으로 전달,이 코드를 사용 :

[TestFixture]
public class Test
{
    ServiceProvider _provider;

    [OneTimeSetUp]
    public void Setup()
    {
        var services = new ServiceCollection();
        services.AddTransient<IOptions<PersonSettings>>(
            provider => Options.Create<PersonSettings>(new PersonSettings
            {
                Name = "Matt"
            }));
        services.AddTransient<Person>();
        _provider = services.BuildServiceProvider();
    }

    [Test]
    public void TestName()
    {
        Person person = _provider.GetService<Person>();
        Assert.IsNotNull(person, "person could not be created");

        Assert.IsTrue(person.Name == "Matt", "person is not Matt");
    }
}


답변

테스트중인 리포지토리의 모의 인스턴스를 실제로 만들기 전에 Options.Create ()를 통해 옵션을 만들고 AutoMocker.Use (options)를 사용하는 것보다 언제든지 옵션을 만들 수 있습니다. AutoMocker.CreateInstance <> ()를 사용하면 매개 변수를 수동으로 전달하지 않고도 인스턴스를 쉽게 만들 수 있습니다.

달성하고자하는 동작을 재현 할 수 있도록 SampleRepo를 약간 변경했습니다.

public class SampleRepoTests
{
    private readonly AutoMocker _mocker = new AutoMocker();
    private readonly ISampleRepo _sampleRepo;

    private readonly IOptions<SampleOptions> _options = Options.Create(new SampleOptions()
        {FirstSetting = "firstSetting"});

    public SampleRepoTests()
    {
        _mocker.Use(_options);
        _sampleRepo = _mocker.CreateInstance<SampleRepo>();
    }

    [Fact]
    public void Test_Options_Injected()
    {
        var firstSetting = _sampleRepo.GetFirstSetting();
        Assert.True(firstSetting == "firstSetting");
    }
}

public class SampleRepo : ISampleRepo
{
    private SampleOptions _options;

    public SampleRepo(IOptions<SampleOptions> options)
    {
        _options = options.Value;
    }

    public string GetFirstSetting()
    {
        return _options.FirstSetting;
    }
}

public interface ISampleRepo
{
    string GetFirstSetting();
}

public class SampleOptions
{
    public string FirstSetting { get; set; }
}


답변

Mock을 필요로하지 않고 대신 OptionsWrapper를 사용하는 또 다른 쉬운 방법이 있습니다.

var myAppSettingsOptions = new MyAppSettingsOptions();
appSettingsOptions.MyObjects = new MyObject[]{new MyObject(){MyProp1 = "one", MyProp2 = "two", }};
var optionsWrapper = new OptionsWrapper<MyAppSettingsOptions>(myAppSettingsOptions );
var myClassToTest = new MyClassToTest(optionsWrapper);


답변

내 시스템 및 통합 테스트의 경우 테스트 프로젝트 내에 구성 파일의 복사 / 링크를 선호합니다. 그런 다음 ConfigurationBuilder를 사용하여 옵션을 가져옵니다.

using System.Linq;
using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.DependencyInjection;

namespace SomeProject.Test
{
public static class TestEnvironment
{
    private static object configLock = new object();

    public static ServiceProvider ServiceProvider { get; private set; }
    public static T GetOption<T>()
    {
        lock (configLock)
        {
            if (ServiceProvider != null) return (T)ServiceProvider.GetServices(typeof(T)).First();

            var builder = new ConfigurationBuilder()
                .AddJsonFile("config/appsettings.json", optional: false, reloadOnChange: true)
                .AddEnvironmentVariables();
            var configuration = builder.Build();
            var services = new ServiceCollection();
            services.AddOptions();

            services.Configure<ProductOptions>(configuration.GetSection("Products"));
            services.Configure<MonitoringOptions>(configuration.GetSection("Monitoring"));
            services.Configure<WcfServiceOptions>(configuration.GetSection("Services"));
            ServiceProvider = services.BuildServiceProvider();
            return (T)ServiceProvider.GetServices(typeof(T)).First();
        }
    }
}
}

이 방법으로 TestProject 내 어디에서나 구성을 사용할 수 있습니다. 단위 테스트의 경우 patvin80과 같은 MOQ를 사용하는 것이 좋습니다.