以前のバージョンUser
では、コントローラーに直接設定できたため、非常に簡単な単体テストが可能でした。
ControllerBaseのソースコードを見ると、User
がから抽出されていることがわかりますHttpContext
。
public ClaimsPrincipal User => HttpContext?.User;
コントローラーはHttpContext
ビアにアクセスしますControllerContext
public HttpContext HttpContext => ControllerContext.HttpContext;
これら2つは読み取り専用のプロパティであることがわかります。良いニュースは、ControllerContext
プロパティがその値を設定できるので、それがあなたの道になるということです。
したがって、目標はそのオブジェクトに到達することです。In CoreHttpContext
は抽象的であるため、モックするのははるかに簡単です。
のようなコントローラーを想定
public class MyController : Controller {
IMyContext _context;
public MyController(IMyContext context) {
_context = context;
}
public IActionResult Index() {
SettingsViewModel svm = _context.MySettings(User.Identity.Name);
return View(svm);
}
}
Moqを使用すると、テストは次のようになります。
public void Given_User_Index_Should_Return_ViewResult_With_Model() {
var username = "FakeUserName";
var identity = new GenericIdentity(username, "");
var mockPrincipal = new Mock<ClaimsPrincipal>();
mockPrincipal.Setup(x => x.Identity).Returns(identity);
mockPrincipal.Setup(x => x.IsInRole(It.IsAny<string>())).Returns(true);
var mockHttpContext = new Mock<HttpContext>();
mockHttpContext.Setup(m => m.User).Returns(mockPrincipal.Object);
var model = new SettingsViewModel() {
};
var mockContext = new Mock<IMyContext>();
mockContext.Setup(m => m.MySettings(username)).Returns(model);
var controller = new MyController(mockContext.Object) {
ControllerContext = new ControllerContext {
HttpContext = mockHttpContext.Object
}
};
var viewResult = controller.Index() as ViewResult;
Assert.IsNotNull(viewResult);
Assert.IsNotNull(viewResult.Model);
Assert.AreEqual(model, viewResult.Model);
}
new Claim(ClaimTypes.Name, "1")
コントローラーの使用と一致させることでしたuser.Identity.Name
。しかし、そうでなければ、それはまさに私が達成しようとしていたことです...ダンケ・ション!