2017-02-22 50 views
0

Документация Microsoft (https://docs.microsoft.com/en-us/aspnet/core/testing/integration-testing) объясняет, как реализовать интеграционный тест с использованием класса TestServer. Это легко, если мы используем WEB API, потому что мы получаем сериализованную модель в качестве ответа от действия.Тест интеграции ядра ASP.NET для действия контроллера

Но если я хочу проверить действие Controller, возвращающее HTML-представление, содержащее некоторые данные, как я могу оценить, что содержимое страницы является тем, что я ожидаю (избегая сканирования содержимого HTML-страницы)?

ответ

0

Одним из вариантов является использование автоматизированного тестирования пользовательского интерфейса, используя что-то вроде Селена

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», просто сделать:

  var greetingViewModel = GetModel<GreetingViewModel>(resonseContent); 

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

Для того, чтобы добавить этот JSON сериализованную модель представления на страницу, я осуществил следующий фильтр:

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 
    } 
} 

Тогда , в методе 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()); }); 
     } 

 Смежные вопросы

  • Нет связанных вопросов^_^