我有具有非接口依赖一个构造函数:AutoMockContainer与非接口依赖关系automocking类的支持
public MainWindowViewModel(IWorkItemProvider workItemProvider, WeekNavigatorViewModel weekNavigator)
我使用Moq.Contrib automockcontainer。如果我尝试automock MainWindowViewModel类,由于WeekNavigatorViewModel依赖关系,我得到一个错误。
是否有任何支持嘲讽非接口类型的automocking容器?
正如Mark所示:是的你可以! :-)我用Mark在他的回答中提供的东西替换了Moq.Contrib AutoMockContainer,唯一的区别是自动生成的mock被注册为单例,但是您可以将其设置为可配置。下面是最终的解决方案:
/// <summary>
/// Auto-mocking factory that can create an instance of the
/// class under test and automatically inject mocks for all its dependencies.
/// </summary>
/// <remarks>
/// Mocks interface and class dependencies
/// </remarks>
public class AutoMockContainer
{
readonly IContainer _container;
public AutoMockContainer(MockFactory factory)
{
var builder = new ContainerBuilder();
builder.RegisterSource(new AnyConcreteTypeNotAlreadyRegisteredSource());
builder.RegisterSource(new MoqRegistrationSource(factory));
_container = builder.Build();
}
/// <summary>
/// Gets or creates a mock for the given type, with
/// the default behavior specified by the factory.
/// </summary>
public Mock<T> GetMock<T>() where T : class
{
return (_container.Resolve<T>() as IMocked<T>).Mock;
}
/// <summary>
/// Creates an instance of a class under test,
/// injecting all necessary dependencies as mocks.
/// </summary>
/// <typeparam name="T">Requested object type.</typeparam>
public T Create<T>() where T : class
{
return _container.Resolve<T>();
}
public T Resolve<T>()
{
return _container.Resolve<T>();
}
/// <summary>
/// Registers and resolves the given service on the container.
/// </summary>
/// <typeparam name="TService">Service</typeparam>
/// <typeparam name="TImplementation">The implementation of the service.</typeparam>
public void Register<TService, TImplementation>()
{
var builder = new ContainerBuilder();
builder.RegisterType<TImplementation>().As<TService>().SingleInstance();
builder.Update(_container);
}
/// <summary>
/// Registers the given service instance on the container.
/// </summary>
/// <typeparam name="TService">Service type.</typeparam>
/// <param name="instance">Service instance.</param>
public void Register<TService>(TService instance)
{
var builder = new ContainerBuilder();
if (instance.GetType().IsClass)
builder.RegisterInstance(instance as object).As<TService>();
else
builder.Register(c => instance).As<TService>();
builder.Update(_container);
}
class MoqRegistrationSource : IRegistrationSource
{
private readonly MockFactory _factory;
private readonly MethodInfo _createMethod;
public MoqRegistrationSource(MockFactory factory)
{
_factory = factory;
_createMethod = factory.GetType().GetMethod("Create", new Type[] { });
}
public IEnumerable<IComponentRegistration> RegistrationsFor(Service service, Func<Service, IEnumerable<IComponentRegistration>> registrationAccessor)
{
var swt = service as IServiceWithType;
if (swt == null)
{
yield break;
}
if (!swt.ServiceType.IsInterface)
yield break;
var existingReg = registrationAccessor(service);
if (existingReg.Any())
{
yield break;
}
var reg = RegistrationBuilder.ForDelegate((c, p) =>
{
var createMethod = _createMethod.MakeGenericMethod(swt.ServiceType);
return ((Mock)createMethod.Invoke(_factory, null)).Object;
}).As(swt.ServiceType).SingleInstance().CreateRegistration();
yield return reg;
}
public bool IsAdapterForIndividualComponents
{
get { return false; }
}
}
}
如果MyClass依赖于具体类型而不是接口,这是否工作? – Marius 2010-03-17 13:25:11
是的,Autofac的'AnyConcreteTypeNotAlreadyRegisteredSource'负责解析具体的类型。 – 2010-03-17 13:32:09
我认为,模拟注册应该明确定义为'SingleInstance'在这个语句'As(swt.ServiceType).CreateRegistration()'。我在NSubstitute上遇到了一个问题,如果它不是'SingleInstance',它可能会有问题。例如:'Resolver.Resolve()。Get(“1”)。Returns(new Basket(1,50));'在这里'ICacheManager'由自动替换的容器创建并且是暂时的'返回'永远不会按预期设置,因为它是使用'var sut = Resolver.Resolve ();'的新实例IOrderService使用一个新的替代'ICacheManager'依赖项。 –
2017-02-26 19:10:52