I have the following line in my controller:
string lTempPath = Path.Combine(Server.MapPath("~/Temp"), lRandomFileName);
Problem is Server is not virtual and can only be accessed with a getter.
I get a
"The method or operation is not implemented."
How can I mock this server?
The test I have created is as below:
[TestCase]
public void PreviewActionShouldGenerateUrlOfPdf()
{
//Arrange
var server = MockRepository.GenerateMock<HttpServerUtilityBase>();
server.Stub(s => s.MapPath("~Temp")).Return("~/Temp");
var httpContext = MockRepository.GenerateMock<HttpContextBase>();
httpContext.Stub(hc => hc.Server).Return(server);
httpContext.Server.Stub(s => s.MapPath("~/Temp")).Return("~/Temp");
var controller = new StudiesController()
{
ReportingService = MockRepository.GenerateMock<IReportingService>(),
SecurityService = MockRepository.GenerateMock<ISecurityService>()
};
controller.ControllerContext = new ControllerContext(httpContext, new RouteData(), controller);
controller.ReportingService.Stub(rs => rs.GetStudyByGID(new Guid())).Return(new Study());
controller.ReportingService.Stub(rs => rs.ListPractices()).Return(new[] { new Practice(), new Practice() });
controller.SecurityService.Stub(ss => ss.GetUser("")).IgnoreArguments().Return(new User());
controller.ControllerContext.HttpContext = MockRepository.GeneratePartialMock<FakeHttpContext>("http://test.com");
controller.HttpContext.User = new FakePrincipal(new FakeIdentity("test"), new string[0]);
controller.ControllerContext.HttpContext.Stub(x => x.Request).Return(MockRepository.GenerateMock<HttpRequestBase>());
controller.ControllerContext.HttpContext.Request.Stub(x => x.Url).Return(new Uri("http://test.com"));
controller.ReportingService.Stub(
rs =>
rs.GenerateReport(new Study(), new Practice(), new User(), false, ReportGenerationOutputFormat.PDF)).IgnoreArguments().Return(new StudyReportSnapshot());
var content = new ContentResult();
//Act
var result = (ContentResult)controller.Preview(new Guid());
//Assert
Assert.AreEqual(result.Content, content.Content);
}
Assuming you're using some sort of IOC/DI container, you shouldn't depend on
Controller.Server
. Instead you should useHttpServerUtilityBase
.This example assumes Ninject as the IOC container, but any of the popular containers will do:
First, register
HttpServerUtilityBase
with your IOC container like so:This will make sure that at run time your app will use the current request's server property.
Then, add a constructor to your controller that accepts an instance of
HttpServerUtilityBase
:Now, anywhere before where you were calling
Server.MapPath
, just call_server.MapPath
instead.Finally, in your tests, you can mock HttpServerUtilityBase like so (assuming Moq as the mocking framework):
EDIT
Since you mentioned you're not using an DI framework, you can resort to "Poor Man's Dependency Injection". Essentially adding an overloaded constructor:
This will allow production code to use the current web request, but then you can create your own implementation of
HttpServerUtilityBase
to use for testing.You can use a Mocking framework in your Unit Test code such as Rhino.Mocks, Moq or FakeItEasy
An example unit test body would be (this example uses Moq)
Then you can assert the result.
Hope that gives you a few ideas