bru*_*ert 7 asp.net-mvc unit-testing repository-pattern asp.net-web-api
我刚刚开始深入研究单元测试,并且刚刚开始掌握存储库模式和IoC.但是,我不认为我完全理解它,因为它的一部分看起来有点傻.让我解释.
我的控制器:
public class UserProfileController : ApiController
{
private IUserProfileRepository repository;
// Optional constructor, passes repository, allows dependency injection
public UserProfileController(IUserProfileRepository userProfileRepository)
{
this.repository = userProfileRepository;
}
// GET api/UserProfile
// Returns a list of all users
public IEnumerable<UserProfile> Get()
{
// Only Admins can see a list of users
if (Roles.IsUserInRole("Admin"))
{
return repository.Get();
}
else
{
throw new HttpResponseException(
new HttpResponseMessage(HttpStatusCode.Forbidden)
{
ReasonPhrase = "Administrator access required"
});
}
}
// Other methods, etc.
Run Code Online (Sandbox Code Playgroud)
(请注意,我有一个依赖项,Roles.IsUserInRole("Admin"),我无法弄清楚如何抽象,这会导致一些问题).
我的典型repo界面:
public interface IUserProfileRepository : IDisposable
{
IEnumerable<UserProfile> Get();
// Other methods, etc.
}
Run Code Online (Sandbox Code Playgroud)
回购:
public class UserProfileRepository : IUserProfileRepository, IDisposable
{
private OfootContext context;
public UserProfileRepository(OfootContext context)
{
this.context = context;
}
public IEnumerable<UserProfile> Get()
{
return context.UserProfiles.AsEnumerable();
}
// ... More code
Run Code Online (Sandbox Code Playgroud)
所以一切似乎都很好,我从业务逻辑中抽象出了业务访问层,现在我可以创建一个虚假的存储库来运行单元测试.
假回购:
public class FakeUserProfileRepository : IUserProfileRepository, IDisposable
{
private List<UserProfile> context;
public FakeUserProfileRepository(List<UserProfile> context)
{
this.context = context;
}
public IEnumerable<UserProfile> Get()
{
return context.AsEnumerable();
}
Run Code Online (Sandbox Code Playgroud)
和测试:
[TestMethod]
public void GetUsers()
{
// Arrange
var items = new List<UserProfile>()
{
new UserProfile
{
UserId = 1,
Username = "Bob",
},
new UserProfile
{
UserId = 2,
Username = "Bob2",
}
};
FakeUserProfileRepository repo = new FakeUserProfileRepository(
items);
UserProfileController controller = new UserProfileController(
repo);
// Act
IEnumerable<UserProfile> result = controller.Get();
// Assert
Assert.IsNotNull(result);
}
Run Code Online (Sandbox Code Playgroud)
现在我们在同一页面上(并且随意指出任何'代码味道'),这是我的想法:
如果我找到一种方法来抽象出用户帐号垃圾,我仍然只是在代码中乱码并创建更多代码(甚至可能是两倍多?因为我需要创建假货)我可以在其中替换Context.
我的问题是:我错过了什么?它是一个整体概念还是具体的东西?
hos*_*ude 11
你走在正确的轨道上.让事情起作用总是很痛苦,但是你会发现它会在未来发展.
我推荐像Moq这样的框架,而不是创建"虚假"对象.它允许您设置测试时所需的行为,而不是重新实现整个接口.例如,在您的测试中,您可以简单地写:
Mock<IUserProfileRepository> mockUserRepo = new Mock<IUserProfileRepository>();
var items = new List<UserProfile>()
{
new UserProfile
{
UserId = 1,
Username = "Bob",
},
new UserProfile
{
UserId = 2,
Username = "Bob2",
}
};
mockUserRepo.Setup(m => m.Get().Returns(items.AsEnumerable());
UserProfileController controller = new UserProfileController(
mockUserRepo.Object);
// Act
IEnumerable<UserProfile> result = controller.Get();
//Now you can keep varying the mock response by changing the Setup(), so now
//check for null response handling, 0 items, exceptions etc...
Run Code Online (Sandbox Code Playgroud)
所有这些努力的最终结果是您已完全将测试隔离到Controller,没有数据库依赖关系,您可以轻松改变输入而无需编写类,而是使用模拟设置.
如果您遵循这种简单的架构模式,您将获得令人敬畏的可测试性和明确的关注点分离.随着系统中的事情变得越来越复杂,您可以利用像Unity这样的DI容器.
在身份验证部分,我建议创建属性,您可以使用ASP.Net MVC使用:[Authorization(Roles ="Admin")]作为示例来装饰您的方法.这创建了另一个有用的交叉模式,使Auth资源与控制器中的业务逻辑分离.
| 归档时间: |
|
| 查看次数: |
5409 次 |
| 最近记录: |