ASP.Net MVCフレームワークを使用したC#のコントローラーがあります
public class HomeController:Controller{
public ActionResult Index()
{
if (Request.IsAjaxRequest())
{
//do some ajaxy stuff
}
return View("Index");
}
}
モックに関するいくつかのヒントを得て、次のコードとRhinoMocksでコードをテストしたいと考えていました
var mocks = new MockRepository();
var mockedhttpContext = mocks.DynamicMock<HttpContextBase>();
var mockedHttpRequest = mocks.DynamicMock<HttpRequestBase>();
SetupResult.For(mockedhttpContext.Request).Return(mockedHttpRequest);
var controller = new HomeController();
controller.ControllerContext = new ControllerContext(mockedhttpContext, new RouteData(), controller);
var result = controller.Index() as ViewResult;
Assert.AreEqual("About", result.ViewName);
しかし、私はこのエラーを取得し続けます:
例外System.ArgumentNullException:System.ArgumentNullException:値をnullにすることはできません。パラメーター名:System.Web.Mvc.AjaxRequestExtensions.IsAjaxRequest(HttpRequestBase request)のリクエスト
コントローラーのRequest
オブジェクトにはセッターがないためです。以下の回答の推奨コードを使用して、このテストを適切に機能させようとしました。
これは、RhinoMocksの代わりにMoqを使用し、Moqを使用して、同じテストに以下を使用します。
var request = new Mock<HttpRequestBase>();
// Not working - IsAjaxRequest() is static extension method and cannot be mocked
// request.Setup(x => x.IsAjaxRequest()).Returns(true /* or false */);
// use this
request.SetupGet(x => x.Headers["X-Requested-With"]).Returns("XMLHttpRequest");
var context = new Mock<HttpContextBase>();
context.SetupGet(x => x.Request).Returns(request.Object);
var controller = new HomeController(Repository, LoginInfoProvider);
controller.ControllerContext = new ControllerContext(context.Object, new RouteData(), controller);
var result = controller.Index() as ViewResult;
Assert.AreEqual("About", result.ViewName);
ただし、次のエラーが発生します。
例外System.ArgumentException:System.ArgumentException:オーバーライドできないメンバーの無効なセットアップ:x => x.Headers ["X-Requested-With"] at Moq.Mock.ThrowIfCantOverride(Expression setup、MethodInfo methodInfo)
繰り返しますが、リクエストヘッダーを設定できないようです。 RhinoMocksまたはMoqでこの値を設定するにはどうすればよいですか?
Moq を使用:
var request = new Mock<HttpRequestBase>();
// Not working - IsAjaxRequest() is static extension method and cannot be mocked
// request.Setup(x => x.IsAjaxRequest()).Returns(true /* or false */);
// use this
request.SetupGet(x => x.Headers).Returns(
new System.Net.WebHeaderCollection {
{"X-Requested-With", "XMLHttpRequest"}
});
var context = new Mock<HttpContextBase>();
context.SetupGet(x => x.Request).Returns(request.Object);
var controller = new YourController();
controller.ControllerContext = new ControllerContext(context.Object, new RouteData(), controller);
更新:
Request.IsAjaxRequest()
の代わりにRequest.Headers["X-Requested-With"]
またはRequest["X-Requested-With"]
をモックします。
NSubstituteを使用している人のために、私は上記の答えを修正し、このようなことをすることができました...(詳細はコントローラのアクションメソッド名です)
var fakeRequest = Substitute.For<HttpRequestBase>();
var fakeContext = Substitute.For<HttpContextBase>();
fakeRequest.Headers.Returns(new WebHeaderCollection { {"X-Requested-With", "XMLHttpRequest"}});
fakeContext.Request.Returns(fakeRequest);
controller.ControllerContext = new ControllerContext(fakeContext, new RouteData(), controller);
var model = new EntityTypeMaintenanceModel();
var result = controller.Details(model) as PartialViewResult;
Assert.IsNotNull(result);
Assert.AreEqual("EntityType", result.ViewName);
RhinoMocksを使用した実用的なソリューションを次に示します。 http://thegrayzone.co.uk/blog/2010/03/mocking-request-isajaxrequest/ で見つけたMoqソリューションに基づいています。
public static void MakeAjaxRequest(this Controller controller)
{
MockRepository mocks = new MockRepository();
// Create mocks
var mockedhttpContext = mocks.DynamicMock<HttpContextBase>();
var mockedHttpRequest = mocks.DynamicMock<HttpRequestBase>();
// Set headers to pretend it's an Ajax request
SetupResult.For(mockedHttpRequest.Headers)
.Return(new WebHeaderCollection() {
{"X-Requested-With", "XMLHttpRequest"}
});
// Tell the mocked context to return the mocked request
SetupResult.For(mockedhttpContext.Request).Return(mockedHttpRequest);
mocks.ReplayAll();
// Set controllerContext
controller.ControllerContext = new ControllerContext(mockedhttpContext, new RouteData(), controller);
}
Is AjaxRequestは拡張メソッドです。したがって、Rhinoを使用して次の方法で実行できます。
protected HttpContextBase BuildHttpContextStub(bool isAjaxRequest)
{
var httpRequestBase = MockRepository.GenerateStub<HttpRequestBase>();
if (isAjaxRequest)
{
httpRequestBase.Stub(r => r["X-Requested-With"]).Return("XMLHttpRequest");
}
var httpContextBase = MockRepository.GenerateStub<HttpContextBase>();
httpContextBase.Stub(c => c.Request).Return(httpRequestBase);
return httpContextBase;
}
// Build controller
....
controller.ControllerContext = new ControllerContext(BuildHttpContextStub(true), new RouteData(), controller);
これを探しているように見えますが、
var requestMock = new Mock<HttpRequestBase>();
requestMock.SetupGet(rq => rq["Age"]).Returns("2001");
コントローラーでの使用:
public ActionResult Index()
{
var age = Request["Age"]; //This will return 2001
}
次のように、HttpContextBaseをモックしてControllerContextプロパティに配置する必要があります。
controller.ControllerContext =
new ControllerContext(mockedHttpContext, new RouteData(), controller);
ユニットテスト中にIsAjaxRequest()
がfalseを返すようにするには、以下に示すように、テストメソッドでリクエストヘッダーとコレクション値の両方を設定する必要があります。
_request.SetupGet(x => x.Headers).Returns(new System.Net.WebHeaderCollection { { "X-Requested-With", "NotAjaxRequest" } });
_request.SetupGet(x=>x["X-Requested-With"]).Returns("NotAjaxRequest");
両方を設定する理由は、以下に示すIsAjaxRequest()の実装に隠されています。
public static bool IsAjaxRequest(this HttpRequestBase request)<br/>
{
if (request == null)
{
throw new ArgumentNullException("request");
}
return ((request["X-Requested-With"] == "XMLHttpRequest") || ((request.Headers != null) && (request.Headers["X-Requested-With"] == "XMLHttpRequest")));
}
リクエストコレクションとヘッダーの両方を使用するため、ヘッダーとリクエストコレクションの両方のセットアップを作成する必要があります。
これにより、ajaxリクエストでない場合にfalseを返すリクエストが作成されます。 trueを返すには、次のことができます。
_httpContext.SetupGet(x => x.Request["X-Requested-With"]).Returns("XMLHttpRequest");
次のように、Web API中にHttpRequestMessageオブジェクトをリクエストに追加する他の方法を見つけました
[Test]
public void TestMethod()
{
var controllerContext = new HttpControllerContext();
var request = new HttpRequestMessage();
request.Headers.Add("TestHeader", "TestHeader");
controllerContext.Request = request;
_controller.ControllerContext = controllerContext;
var result = _controller.YourAPIMethod();
//Your assertion
}