我刚刚开始使用Moq进行单元测试/模拟,并遇到了问题..
我有一个名为“CustomerService”的服务层,其中包含以下代码:
public interface ICustomerService
{
Customer GetCustomerById(int id);
}
public class CustomerService : ICustomerService
{
private IRepository<Customer> customerRepository;
public CustomerService(IRepository<Customer> rep)
{
customerRepository = rep;
}
public Customer GetCustomerById(int id)
{
var customer = customerRepository.Get(x => x.CustomerId == id);
if (customer == null)
return null;
return customer;
}
}
我的存储库类是通用的,并且遵循:
public interface IRepository<T> : IDisposable where T : class
{
T Get(Expression<Func<T, bool>> predicate);
}
public class Repository<T> : IRepository<T> where T : class
{
private ObjectContext context;
private IObjectSet<T> objectSet;
public Repository()
: this(new demonEntities())
{
}
public Repository(ObjectContext ctx)
{
context = ctx;
objectSet = context.CreateObjectSet<T>();
}
public T Get(Expression<Func<T, bool>> predicate)
{
T entity = objectSet.Where<T>(predicate).FirstOrDefault();
if (entity == null)
return null;
return objectSet.Where<T>(predicate).FirstOrDefault();
}
public void Dispose()
{
Dispose(true);
GC.SuppressFinalize(this);
}
protected virtual void Dispose(bool disposing)
{
if (disposing)
{
if (context != null)
{
context.Dispose();
context = null;
}
}
}
}
现在是我的问题..如何进行单元测试以检查我的GetCustomerById是否返回null?
已经尝试过:
[TestMethod]
public void GetCustomerTest()
{
const int customerId = 5;
var mock = new Mock<IRepository<Customer>>();
mock.Setup(x => x.Get(z => z.CustomerId == customerId))
.Returns(new Customer());
var repository = mock.Object;
var service = new CustomerService(repository);
var result = service.GetCustomerById(customerId);
Assert.IsNotNull(result);
}
没有运气......
答案 0 :(得分:5)
您需要将Repository<T>.Get
方法设为虚拟,以便Moq可以覆盖它并返回您设置的值:
public virtual T Get(Expression<Func<T, bool>> predicate)
并在测试中,更改
mock.Setup(x => x.Get(z => z.CustomerId == customerId))
.Returns(new Customer());
到
mock.Setup(x => x.Get(It.IsAny<Expression<Func<Customer, bool>>>()))
.Returns(new Customer());
表示为传入的任何Customer
返回新的Expression<Func<Customer, bool>>
。理想情况下,您会测试一个特定的表达式,但根据此SO question的接受答案,Moq无法执行此操作。< / p>
如果您想测试您的服务层没有对存储库返回的Customer
执行任何意外操作,而不是测试是否返回任何 Customer
,您可以设置模拟Customer
(确保将CustomerId
属性设为虚拟)并声明服务层返回的Customer
具有预期的属性。
[TestMethod]
public void GetCustomerTest()
{
const int customerId = 5;
var mockCustomer = new Mock<Customer>();
mockCustomer.SetupGet(x => x.CustomerId)
.Returns(customerId);
var mock = new Mock<IRepository<Customer>>();
mock.Setup(x => x.Get(It.IsAny<Expression<Func<Customer, bool>>>()))
.Returns(mockCustomer.Object);
var repository = mock.Object;
var service = new CustomerService(repository);
var result = service.GetCustomerById(customerId);
Assert.AreEqual(customerId, result.CustomerId);
}
HTH
答案 1 :(得分:0)
你必须针对界面创建模拟。
然后,您需要将类中的成员设置为mock而不是实现者
然后你需要在你为方法创建的模拟上调用.Setup。记住使用方法链来使它可以验证。
在测试中运行该方法,然后调用mock.Verify()
如果您需要,明天会添加代码。我可以使用大量的代码示例。
答案 2 :(得分:0)
你不能这样做的原因是因为一个lambda x => x.CustomerId == id
不等于另一个lambda x => x.CustomerId == id
所以Moq不能匹配它们。
但是,如果你有一个关于你的共同操作的课程:
public class CustomerQueries {
public static Predicate<Customer> ById(int id) = x => x.CustomerId == id;
}
并在代码和测试中重用这个lambda,然后你的Moq应该没有任何问题。您也可以在类的已知实例上使用方法,只要它是完全相同的委托,而不仅仅是类似的一个或另一个副本。
PS:您是否考虑过使用Predicate<T>
代替Expression<Func<T, bool>>
?它更容易阅读和理解目的。