Макет сервера в контроллере
В моем контроллере есть следующая строка:
string lTempPath = Path.Combine(Server.MapPath("~/Temp"), lRandomFileName);
Проблема заключается в том, что сервер не является виртуальным и доступен только с помощью геттера.
Я получаю
"Метод или операция не реализованы".
Как я могу издеваться над этим сервером?
Тестирование, которое я создал, выглядит следующим образом:
[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);
}
Ответы
Ответ 1
Предполагая, что вы используете какой-либо контейнер IOC/DI, вы не должны зависеть от Controller.Server
. Вместо этого вы должны использовать HttpServerUtilityBase
.
В этом примере предполагается, что Ninject является контейнером IOC, но любой из популярных контейнеров будет делать:
Сначала зарегистрируйте HttpServerUtilityBase
с контейнером IOC следующим образом:
kernel.Bind<HttpServerUtilityBase>().ToMethod(c => new HttpServerUtilityWrapper(HttpContext.Current.Server));
Это гарантирует, что во время выполнения ваше приложение будет использовать текущее свойство сервера запросов.
Затем добавьте конструктор к контроллеру, который принимает экземпляр HttpServerUtilityBase
:
public MyController(HttpServerUtilityBase server)
{
this._server = server;
}
Теперь, где бы вы ни находились, когда вы звонили Server.MapPath
, просто вызовите _server.MapPath
.
Наконец, в ваших тестах вы можете издеваться над HttpServerUtilityBase так (считая Moq как насмешливую структуру):
var server = new Mock<HttpServerUtilityBase>();
server.Setup(s => s.MapPath(It.IsAny<string>())).Returns<string>(s => /* set up how you want MapPath to behave here */);
ИЗМЕНИТЬ
Поскольку вы упомянули, что не используете рамки DI, вы можете прибегнуть к "инъекции зависимых людей". По существу добавление перегруженного конструктора:
public MyController()
: this(new HttpServerUtilityWrapper(HttpContext.Current.Server))
{
}
public MyController(HttpServerUtilityBase server)
{
this._server = server;
}
Это позволит производственному коду использовать текущий веб-запрос, но затем вы можете создать свою собственную реализацию HttpServerUtilityBase
для тестирования.
Ответ 2
Вы можете использовать фреймворк Mocking в вашем коде Unit Test, таком как Rhino.Mocks, Moq или FakeItEasy
Пример тела Unit Test будет (этот пример использует Moq)
var homeController = new HomeController();
//create mock of HttpServerUtilityBase
var server = new Mock<HttpServerUtilityBase>();
//set up mock to return known value on call.
server.Setup(x => x.MapPath("~/Temp")).Returns("c:\\temp\\");
var httpContext = new Mock<HttpContextBase>();
httpContext.Setup(x => x.Server).Returns(server.Object);
homeController.ControllerContext = new ControllerContext(httpContext.Object, new RouteData(), homeController);
YourModelName yourModelName = new YourModelName();
yourModelName.lRandomFileName = "zzzz.so";
var result = homeController.YourActionName(yourModelName);
Затем вы можете утверждать результат.
Надеюсь, что вы получите несколько идей