模拟ClaimsPrincipal类上的扩展方法,以便在HttpContext中使用.用户



扩展不能被Moq等框架嘲笑,因为它们是静态方法。

我的ClaimsPrincipal类有一个扩展方法,在我的应用程序中广泛使用,无法更改。扩展方法称为GetUserSerial()

我希望能够以某种方式模拟这个扩展方法,以便在单元测试中使用它,因为扩展方法本身在各种ASP中使用。NET核心控制器,因此我需要能够模拟它的输出,以正确地测试控制器。

我试图通过以下操作来模拟ControllerContext

private Mock<ControllerContext> GetMockControllerContext(Mock<ClaimsPrincipal> mockClaimsPrincipal)
{
if (mockClaimsPrincipal == null)
{
mockClaimsPrincipal = new Mock<ClaimsPrincipal>();
mockClaimsPrincipal.Setup(x => x.GetSerialNo()).Returns("serial");
}
var contextMock = new Mock<HttpContext>();
contextMock.SetupGet(ctx => ctx.User).Returns(mockClaimsPrincipal.Object);
var controllerContextMock = new Mock<ControllerContext>();
controllerContextMock.SetupGet(con => con.HttpContext).Returns(contextMock.Object);
return controllerContextMock;
}
public static string GetSerialNo(this ClaimsPrincipal claimsPrincipal)
{
return claimsPrincipal?.Claims?.FirstOrDefault(c => c.Type == "SerialNo")?.Value;
}

然而,这显然会产生运行时错误,告诉我扩展方法不能被嘲笑。

这里的主要问题是扩展方法在控制器内部使用,我无法更改它的输出

有没有办法绕过这一点,让我可以指定GetSerialNo方法的输出,并在我的ControllerContext中提供它?

不需要mock。创建具有所需索赔的主体实例

string expectedSerial = "serial";
var claims = new List<Claim> {
new Claim("SerialNo", expectedSerial),
};
var principal = new ClaimsPrincipal(new ClaimsIdentity(claims));
//...

这样,当调用扩展时,它将按预期运行。

如果修改现有身份,则只需添加声明

//... principal already exists
string expectedSerial = "serial";
Claim claim = new Claim("SerialNo", expectedSerial);
(principal.Identity as ClaimsIdentity).AddClaim(claim);
//...

最新更新