2017-02-22 5 views
0

Microsoftドキュメント(https://docs.microsoft.com/en-us/aspnet/core/testing/integration-testing)は、TestServerクラスを使用して統合テストを実装する方法を説明しています。アクションからの応答としてシリアライズドモデルを取得するので、WEB APIを使用している場合は簡単です。コントローラのアクションのASP.NETコア統合テスト

しかし、いくつかのデータを含むHTMLビューを返すコントローラアクションをテストしたい場合、ページコンテンツが私の期待するものであると評価できますか(HTMLページのコンテンツをスキャンしないでください)?

答えて

0

1つのオプションは、セレンのようなものを使用して自動化されたUIテストを使用することです

0

あなたがセットアップしている一つのあなたのWebHostBuilderを同様TESTSERVERオブジェクトとして、あなたはあなたのHTMLレンダリングされたページに作成するために使用されるJSONシリアライズされたモデルを追加することができますコンテンツを表示するたとえば:

<!DOCTYPE html> 
<html> 
<head></head> 
<body>  
    <div>Subject</div><div>Kiss Me</div> 
    <div>Message</div><div>Ciao</div> 
</body> 
</html> 

<script type="model/json"> 
    {"Date":"2017-02-22","Subject":"Kiss Me","Result":true,"Message":"Ciao"} 
</script> 

は、その後、あなたのテストでは、単にモデルをデシリアライズするために、一般的なメソッドを作成します。

public const string StartViewModelContainer = "<script type=\"model/json\">"; 
    public const string EndViewModelContainer = "</script>"; 

    protected virtual T GetModel<T>(string responseContent) 
    { 
     var result = default(T); 

     if (!string.IsNullOrWhiteSpace(responseContent)) 
     { 
      var index = responseContent.IndexOf(StartViewModelContainer, 0, StringComparison.InvariantCulture); 
      if (index > 0) 
      { 
       var startingPosition = index + StartViewModelContainer.Length; 

       var endingPosition = responseContent.IndexOf(
        EndViewModelContainer, 
        startingPosition, 
        StringComparison.InvariantCulture); 

       if (endingPosition <= startingPosition) return result; 

       var jSonModel = responseContent.Substring(startingPosition, endingPosition - startingPosition); 
       result = JsonConvert.DeserializeObject<T>(jSonModel); 
      } 
     } 

     return result; 
    } 

その結果を評価するためにそれを使用した後。 、ちょうどんViewModelには、ビューから使用すると仮定すると、「GreetingViewModel」です。そして、

using System; 
using System.Collections.Generic; 
using System.Linq; 
using Microsoft.AspNetCore.Http; 
using Microsoft.AspNetCore.Mvc; 
using Microsoft.AspNetCore.Mvc.Filters; 
using Microsoft.AspNetCore.Mvc.ModelBinding; 
using Newtonsoft.Json; 
using Ticketino.Web.Components.Extensions.Request; 
using Ticketino.Web.OnlineShop.Serializations; 
using Ticketino.Web.OnlineShop.ViewModels.Base; 

namespace Ticketino.Web.OnlineShop.Filters 
{ 
    /// <summary> 
    /// This is a filter used only for integration tests. 
    /// It format the ViewModel as jSon and appends it to the end of HMTL page, so that it can be deserialized from the test in order to check its values. 
    /// </summary> 
    /// <seealso cref="Microsoft.AspNetCore.Mvc.Filters.ResultFilterAttribute" /> 
    [AttributeUsage(AttributeTargets.Method)] 
    public class IntegrationTestFilterAttribute : ResultFilterAttribute 
    { 
     public const string StartViewModelContainer = "<script type=\"model/json\">"; 
     public const string EndViewModelContainer = "</script>"; 

     public override void OnResultExecuting(ResultExecutingContext filterContext) 
     { 
      if (!filterContext.ModelState.IsValid) 
      { 
       var viewResult = filterContext.Result as ViewResult; 
       if (viewResult?.Model is BaseViewModel) 
       { 
        var errors = IntegrationTestFilterAttribute.GetModelErrors(filterContext.ModelState); 
        ((BaseViewModel)viewResult.Model).ValidationErrors = errors; 
       } 
      } 

      base.OnResultExecuting(filterContext); 
     } 

     public override void OnResultExecuted(ResultExecutedContext filterContext) 
     { 
      if (!filterContext.HttpContext.Request.IsAjaxRequest()) 
      { 
       var viewResult = filterContext.Result as ViewResult; 
       if (viewResult?.Model != null) 
       { 
        var jsonViewModel = string.Concat(
         IntegrationTestFilterAttribute.StartViewModelContainer, 
         JsonConvert.SerializeObject(viewResult.Model, Formatting.None, CommonJsonSerializerSettings.Settings()), 
         IntegrationTestFilterAttribute.EndViewModelContainer); 

        filterContext.HttpContext.Response.WriteAsync(jsonViewModel); 
       } 
      } 

      base.OnResultExecuted(filterContext); 
     } 

     #region Private methods 

     private static IDictionary<string, string> GetModelErrors(ModelStateDictionary errDictionary) 
     { 
      var errors = new Dictionary<string, string>(); 

      //get all entries from the ModelStateDictionary that have any errors and add them to our Dictionary 
      errDictionary.Where(k => k.Value.Errors.Count > 0).ForEach(i => 
      { 
       foreach (var errorMessage in i.Value.Errors.Select(e => e.ErrorMessage)) 
       { 
        errors.Add(i.Key, errorMessage); 
       } 
      }); 

      return errors; 
     } 

     #endregion 
    } 
} 

  var greetingViewModel = GetModel<GreetingViewModel>(resonseContent); 

      Assert.AreEqual(greetingViewModel.Message, "Ciao"); 
      Assert.AreEqual(greetingViewModel.Subject, "Kiss Me"); 

あなたのページにこのJSONシリアライズされたビューモデルを追加するために、私は次のフィルタを実装しました、ConfigureServices(IServiceCollection serviceCollection)メソッドで統合テストを実行すると、次のように注入されます。

 // Filter to append json serialized view model to buttom html response page, in order to eveluate from integration test class 
     if (_hostingEnvironment.IsIntegrationTest()) 
     { 
      mvcBuilder.AddMvcOptions(opt => { opt.Filters.Add(new IntegrationTestFilterAttribute()); }); 
     } 
関連する問題