Consider a method in a .NET assembly:
public static string GetSecurityContextUserName()
{
//extract the username from request
string sUser = HttpContext.Current.User.Identity.Name;
//everything after the domain
sUser = sUser.Substring(sUser.IndexOf("\\") + 1).ToLower();
return sUser;
}
I’d like to call this method from a unit test using the Moq framework. This assembly is part of a webforms solution. The unit test looks like this, but I am missing the Moq code.
//arrange
string ADAccount = "BUGSBUNNY";
string fullADName = "LOONEYTUNES\BUGSBUNNY";
//act
//need to mock up the HttpContext here somehow -- using Moq.
string foundUserName = MyIdentityBL.GetSecurityContextUserName();
//assert
Assert.AreEqual(foundUserName, ADAccount, true, "Should have been the same User Identity.");
Question:
- How can I use Moq to arrange a fake HttpContext object with some value like ‘MyDomain\MyUser’?
- How do I associate that fake with my call into my static method at
MyIdentityBL.GetSecurityContextUserName()? - Do you have any suggestions on how to improve this code/architecture?
Webforms is notoriously untestable for this exact reason – a lot of code can rely on static classes in the asp.net pipeline.
In order to test this with Moq, you need to refactor your
GetSecurityContextUserName()method to use dependency injection with anHttpContextBaseobject.HttpContextWrapperresides inSystem.Web.Abstractions, which ships with .Net 3.5. It is a wrapper for theHttpContextclass, and extendsHttpContextBase, and you can construct anHttpContextWrapperjust like this:Even better, you can mock an HttpContextBase and set up your expectations on it using Moq. Including the logged in user, etc.
With this in place, you can call
GetSecurityContextUserName(mockContext.Object), and your application is much less coupled to the static WebForms HttpContext. If you’re going to be doing a lot of tests that rely on a mocked context, I highly suggest taking a look at Scott Hanselman’s MvcMockHelpers class, which has a version for use with Moq. It conveniently handles a lot of the setup necessary. And despite the name, you don’t need to be doing it with MVC – I use it successfully with webforms apps when I can refactor them to useHttpContextBase.