2

フィルタコンテキストのHttpContextのHttpApplicationStateにオブジェクトを設定しているアクションフィルタがあります。単体テストでこの機能を使用したいのですが、何らかの理由で、HttpApplicationStateの派生元である基になるNameObjectCollectionBaseにオブジェクトが設定されていません。

MVCアプリケーションを実行すると期待どおりに機能するため、機能が機能することはわかっています。

オブジェクトをアプリケーション状態に設定するためにテストを設定するにはどうすればよいですか?私はMoqを使用していますが、これまでのコードの一部を次に示します。それは失敗します

Asset.IsNotNull(context.HttpContext.Application["config"]);

これがコードです。

public override void OnActionExecuted(ActionExecutedContext filterContext)
{
    ControllerBase controller = filterContext.Controller;
    if (!(controller is ApplicationController))
        return;

    ApplicationController applicationController = (ApplicationController) controller;

    IDictionary<string, string> config;

    // Loads the view configuration values.
    if (filterContext.HttpContext.Application["config"] == null)
    {
        config = applicationController.ApplicationService.GetConfiguration();
        filterContext.HttpContext.Application["config"] = config;
    }
    else
    {
        config = (IDictionary<string, string>) filterContext.HttpContext.Application["config"];
    }

    applicationController.ViewBag.BlogTitle = AddConfigurationValueToViewBag("BlogTitle", config);

}

これがこれまでのテストです。

[TestMethod]
public void ApplicationAttribute_OnActionExecuted_SetsConfigurationDctionaryInAppicationCache()
{
    // Arrange
    Mock<HttpContextBase> httpContext = new Mock<HttpContextBase>();

    var mockApplicationState = new Mock<HttpApplicationStateBase>();
    httpContext.Setup(h => h.Application).Returns(mockApplicationState.Object);

    ApplicationController applicationController = new BlogController(null, null, MocksAndStubs.CreateMockApplicationService());

    Mock<ActionExecutedContext> actionExecutedContext = new Mock<ActionExecutedContext>();
    actionExecutedContext.SetupGet(c => c.HttpContext).Returns(httpContext.Object);
    actionExecutedContext.SetupGet(c => c.Controller).Returns(applicationController);

    // Act
    ApplicationAttribute applicationAttribute = new ApplicationAttribute();
    ActionExecutedContext context = actionExecutedContext.Object;
    applicationAttribute.OnActionExecuted(context);

    // Assert
    Assert.IsNotNull(context.HttpContext.Application["config"]);
}
4

2 に答える 2

2

モックオブジェクトが多かれ少なかれ実際のオブジェクトのように動作することを期待しているようです。それはそのようには機能しません。モックはあなたが指示したことだけを行い、それ以上でもそれ以下でもありません。を呼び出すときに特定の何かを返すようにモックに指示しない場合。。。

context.HttpContext.Application["config"]

。。。その後、それは単にしません。特定の何かを返すようにモックを設定した場合、テストの目的が無効になります。

追加の理解や状況(テスト状態の「何」と「理由」)がなければ、アプリケーション状態のセットが呼び出されるようにしようとしているように見えます。オブジェクト自体の結果の状態をテストするのではなく、セットが発生mockApplicationState.Verify()したことをテストするためのアサーションとしてを実行することをお勧めし ます。

編集: Verify()を使用すると、条件の有無にかかわらず、メソッド(またはプロパティメソッド)が呼び出されたことを表明/確認できます。これで始められるはずです:

http://code.google.com/p/moq/wiki/QuickStart#Verification

したがって、検証は次のようになります(完全にテストされていません!):

mockApplicationState.Verify(x => x["config"] == [expected value], Times.Once());

これは基本的に、mockApplicationState ["config"]が期待値に1回未満または複数回設定された場合、テストに失敗することを意味します。

HttpApplicationStateBaseは封印されていないと思います。その場合、上記は例外をスローする可能性があります。

于 2011-08-29T17:10:06.383 に答える
0

MOQの代わりに、このシナリオでは、System.Web.Abstractionsの基本クラスから派生したスタブを生成することがよくあります。MVC / WebApiコントローラーにはHttpContext(HttpContextBase)の抽象化が含まれているため、MVCアプリケーションでこの手法をよく使用します。

このようにして、ユニット/統合テストでHttpContext要件をスタブ化できます。ここにサンプルがあります...

public class MockHttpApplicationState : HttpApplicationStateBase
{
    private IDictionary<string, object> _appState = new Dictionary<string, object>();

    public override void Add(string name, object value)
    {
        _appState.Add(name, value);
    }

    public override object Get(string name)
    {
        return _appState[name];
    }

    public override object this[string name]
    {
        get
        {
            return _appState[name];
        }

        set
        {
            _appState[name] = value;
        }
    }
}

public class MockHttpContext : HttpContextBase
{
    private IDictionary<string, object> _appKeys;

    public MockHttpContext()
    {

    }

    /// <summary>
    /// Accepts a dictionary of app keys to supply to the HttpApplicationState instance
    /// </summary>
    /// <param name="applicationState"></param>
    public MockHttpContext(IDictionary<string,object> applicationState)
    {
        _appKeys = applicationState;
    }

    public override Cache Cache
    {
        get
        {                
            return HttpRuntime.Cache;
        }
    }

    public override HttpApplicationStateBase Application
    {
        get
        {
            var mockAppState = new MockHttpApplicationState();

            foreach (string key in _appKeys.Keys)
            {
                mockAppState.Add(key, _appKeys[key]);
            }

            return mockAppState;
        }
    }

    public override HttpRequestBase Request
    {
        get
        {
            return new HttpRequestWrapper(new HttpRequest(null,"http://localhost",null));
        }
    }
}

次に、私のテストでコントローラーとHttpコンテキストを確立できます。

private readonly OnlineShop.MVC.Controllers.HomeController _controller = 
        new MVC.Controllers.HomeController(null,new UnitOfWork());

    [OneTimeSetUp]
    public void Init()
    {
        var appKeys = new Dictionary<string, object>();

        appKeys.Add("localhost", 1);

        var httpContext = new MockHttpContext(appKeys);

        _controller.ControllerContext = new ControllerContext()
        {
            Controller = _controller,
            RequestContext = new RequestContext(httpContext, new RouteData())    
        };                        
    }

    [Test]
    public void Index_Returns_HomeView()
    {            
        var view = _controller.Index() as ViewResult;
        var viewModel = view.Model as MVC.ViewModels.Home;

        Assert.IsInstanceOf<OnlineShop.MVC.ViewModels.Home>(viewModel);
        Assert.IsTrue(viewModel.FeaturedProducts.Count > 0);
    }

そして、私のコントローラーは、キャッシュとアプリケーションの状態を提供するアンビエントHttpContextBaseインスタンスを認識しています。

  public ActionResult Index()
    {                        
        string cacheKey = string.Format("FeaturedProducts-{0}",WebsiteId);
        IList<Product> productList = this.HttpContext.Cache[cacheKey] as IList<Product>;


        //My app keeps a list of website contexts in the Application. This test returns 1 based on the unit / int tests or a real world db value when hosted on IIS etc..
        int websiteId = (int)HttpContext.Application[this.Request.Url.Host];
于 2016-09-06T13:46:42.410 に答える