ビルドしたASP.NET MVCアプリケーションに単体テストを追加しようとしています。ユニットテストでは、次のコードを使用します。
[TestMethod]
public void IndexAction_Should_Return_View() {
var controller = new MembershipController();
controller.SetFakeControllerContext("TestUser");
...
}
次のヘルパーを使用して、コントローラーコンテキストをモックします。
public static class FakeControllerContext {
public static HttpContextBase FakeHttpContext(string username) {
var context = new Mock<HttpContextBase>();
context.SetupGet(ctx => ctx.Request.IsAuthenticated).Returns(!string.IsNullOrEmpty(username));
if (!string.IsNullOrEmpty(username))
context.SetupGet(ctx => ctx.User.Identity).Returns(FakeIdentity.CreateIdentity(username));
return context.Object;
}
public static void SetFakeControllerContext(this Controller controller, string username = null) {
var httpContext = FakeHttpContext(username);
var context = new ControllerContext(new RequestContext(httpContext, new RouteData()), controller);
controller.ControllerContext = context;
}
}
このテストクラスは、次の基本クラスを継承します。
[TestInitialize]
public void Init() {
...
}
このメソッド内で、次のコードを実行しようとするライブラリ(私は制御できません)を呼び出します。
HttpContext.Current.User.Identity.IsAuthenticated
これでおそらく問題を見ることができます。コントローラーに対して偽のHttpContextを設定しましたが、このベースInitメソッドには設定していません。ユニットテスト/モックは私にとって非常に新しいので、これが正しいことを確認したいと思います。 HttpContextをモックアウトして、コントローラーとInitメソッドで呼び出されるライブラリ全体で共有するための正しい方法は何ですか?.
HttpContext.Current
は System.Web.HttpContext
のインスタンスを返しますが、これは System.Web.HttpContextBase
を拡張しません。 HttpContextBase
は、後で模擬するのが難しいHttpContext
に対処するために追加されました。 2つのクラスは基本的に無関係です( HttpContextWrapper
はそれらの間のアダプターとして使用されます)。
幸い、HttpContext
自体は、IPrincipal
(ユーザー)とIIdentity
を置き換えるのに十分な偽物です。
次のコードは、コンソールアプリケーションでも期待どおりに実行されます。
HttpContext.Current = new HttpContext(
new HttpRequest("", "http://tempuri.org", ""),
new HttpResponse(new StringWriter())
);
// User is logged in
HttpContext.Current.User = new GenericPrincipal(
new GenericIdentity("username"),
new string[0]
);
// User is logged out
HttpContext.Current.User = new GenericPrincipal(
new GenericIdentity(String.Empty),
new string[0]
);
Test Initの下でもジョブを実行します。
[TestInitialize]
public void TestInit()
{
HttpContext.Current = new HttpContext(new HttpRequest(null, "http://tempuri.org", null), new HttpResponse(null));
YourControllerToBeTestedController = GetYourToBeTestedController();
}
これは古いテーマですが、単体テスト用のMVCアプリケーションのモックは非常に定期的に行っています。
Visual Studio 2013にアップグレードした後、Moq 4を使用してMVC 3アプリケーションをモックするエクスペリエンスを追加したいだけです。ユニットテストはデバッグモードで動作せず、HttpContextが変数を覗こうとすると「式を評価できません」と表示されました。
Visual Studio 2013には、一部のオブジェクトの評価に問題があることがわかりました。模擬Webアプリケーションのデバッグを再び機能させるには、ツール=>オプション=>デバッグ=>一般設定で「管理互換モードを使用」をチェックする必要がありました。
私は一般的に次のようなことをします:
public static class FakeHttpContext
{
public static void SetFakeContext(this Controller controller)
{
var httpContext = MakeFakeContext();
ControllerContext context =
new ControllerContext(
new RequestContext(httpContext,
new RouteData()), controller);
controller.ControllerContext = context;
}
private static HttpContextBase MakeFakeContext()
{
var context = new Mock<HttpContextBase>();
var request = new Mock<HttpRequestBase>();
var response = new Mock<HttpResponseBase>();
var session = new Mock<HttpSessionStateBase>();
var server = new Mock<HttpServerUtilityBase>();
var user = new Mock<IPrincipal>();
var identity = new Mock<IIdentity>();
context.Setup(c=> c.Request).Returns(request.Object);
context.Setup(c=> c.Response).Returns(response.Object);
context.Setup(c=> c.Session).Returns(session.Object);
context.Setup(c=> c.Server).Returns(server.Object);
context.Setup(c=> c.User).Returns(user.Object);
user.Setup(c=> c.Identity).Returns(identity.Object);
identity.Setup(i => i.IsAuthenticated).Returns(true);
identity.Setup(i => i.Name).Returns("admin");
return context.Object;
}
}
そして、このようなコンテキストを開始する
FakeHttpContext.SetFakeContext(moController);
コントローラーのメソッドを直接呼び出す
long lReportStatusID = -1;
var result = moController.CancelReport(lReportStatusID);
アプリケーションのサードパーティが内部的にリダイレクトする場合、以下の方法でHttpContextをモックすることをお勧めします。
HttpWorkerRequest initWorkerRequest = new SimpleWorkerRequest("","","","",new StringWriter(CultureInfo.InvariantCulture));
System.Web.HttpContext.Current = new HttpContext(initWorkerRequest);
System.Web.HttpContext.Current.Request.Browser = new HttpBrowserCapabilities();
System.Web.HttpContext.Current.Request.Browser.Capabilities = new Dictionary<string, string> { { "requiresPostRedirectionHandling", "false" } };