Testes de integração no ASP.NET Core
Por Jos van der Til, Martin Costello e Javier Calvarro Nelson.
Os testes de integração garantem que os componentes de um aplicativo funcionem corretamente em um nível que inclua a infraestrutura de suporte do aplicativo, como o banco de dados, o sistema de arquivos e a rede. ASP.NET Core dá suporte a testes de integração usando uma estrutura de teste de unidade com um host Web de teste e um servidor de teste na memória.
Este artigo pressupõe uma compreensão básica sobre testes de unidade. Se não estiver familiarizado com os conceitos de teste, consulte o artigo Teste de Unidade no .NET Core e no .NET Standard e seu conteúdo vinculado.
Exibir ou baixar código de exemplo (como baixar)
O aplicativo de exemplo é um Razor aplicativo Pages e pressupõe uma compreensão básica do Razor Pages. Se você não estiver familiarizado com o Razor Pages, consulte os seguintes artigos:
Para testar SPAs, recomendamos uma ferramenta como o Playwright para .NET, que pode automatizar um navegador.
Introdução aos testes de integração
Os testes de integração avaliam os componentes de um aplicativo em um nível mais amplo do que os testes de unidade. Os testes de unidade são usados para testar componentes de software isolados, como métodos de classe individuais. Os testes de integração confirmam que dois ou mais componentes de aplicativo trabalham juntos para produzir um resultado esperado, possivelmente incluindo todos os componentes necessários para processar totalmente uma solicitação.
Esses testes mais amplos são usados para testar a infraestrutura do aplicativo e toda a estrutura, geralmente incluindo os seguintes componentes:
- Banco de dados
- Sistema de arquivos
- Dispositivos de rede
- Pipeline de solicitação-resposta
Os testes de unidade usam componentes fabricados, conhecidos como fakes ou objetos fictícios, no lugar de componentes de infraestrutura.
Ao contrário dos testes de unidade, os testes de integração:
- Usam os componentes reais que o aplicativo usa em produção.
- Exigem mais código e processamento de dados.
- Demoraram mais para serem executados.
Portanto, limite o uso de testes de integração aos cenários de infraestrutura mais importantes. Se um comportamento puder ser testado usando um teste de unidade ou um teste de integração, escolha o teste de unidade.
Em discussões sobre testes de integração, o projeto testado é frequentemente chamado de System Under Test ou "SUT" para abreviar. "SUT" é usado ao longo deste artigo para se referir ao aplicativo ASP.NET Core que está sendo testado.
Não escreva testes de integração para cada permutação de dados e acesso a arquivos com bancos de dados e sistemas de arquivos. Independentemente de quantos lugares em um aplicativo interagem com bancos de dados e sistemas de arquivos, um conjunto focado de testes de integração de leitura, gravação, atualização e exclusão geralmente é capaz de testar adequadamente os componentes do banco de dados e do sistema de arquivos. Use testes de unidade para testes de rotina da lógica do método que interage com esses componentes. Em testes de unidade, o uso de infraestruturas falsas ou fictícias resulta em uma execução de teste mais rápida.
Testes de integração de ASP.NET Core
Os testes de integração no ASP.NET Core exigem o seguinte:
- Um projeto de teste é usado para conter e executar os testes. O projeto de teste tem uma referência ao SUT.
- O projeto de teste cria um host Web de teste para o SUT e usa um cliente do servidor de teste para lidar com solicitações e respostas com o SUT.
- Um executor de teste é usado para executar os testes e relatar os resultados.
Os testes de integração seguem uma sequência de eventos que incluem as etapas de teste Organizar, Atuar e Afirmar:
- O host da Web do SUT está configurado.
- Um cliente do servidor de teste é criado para enviar solicitações ao aplicativo.
- A etapa de teste Organizar é executada: o aplicativo de teste prepara uma solicitação.
- A etapa de teste Atuar é executada: o cliente envia a solicitação e recebe a resposta.
- A etapa de teste Afirmar é executada: a resposta real é validada como uma aprovada ou reprovada com base em uma resposta esperada.
- O processo continua até que todos os testes sejam executados.
- Os resultados do teste são reportados.
Normalmente, o host da Web de teste é configurado de forma diferente do host da Web normal do aplicativo para as execuções de teste. Por exemplo, um banco de dados diferente ou configurações de aplicativo diferentes podem ser usadas para os testes.
Os componentes de infraestrutura, como o host da Web de teste e o servidor de teste na memória (TestServer), são fornecidos ou gerenciados pelo pacote Microsoft.AspNetCore.Mvc.Testing. O uso desse pacote simplifica a criação e a execução do teste.
O pacote Microsoft.AspNetCore.Mvc.Testing
manipula as seguintes tarefas:
- Copia o arquivo de dependências (
.deps
) do SUT para o diretóriobin
do projeto de teste. - Define a raiz de conteúdo para a raiz do projeto do SUT para que arquivos estáticos e páginas/exibições sejam encontradas quando os testes forem executados.
- Fornece a classe WebApplicationFactory para simplificar a inicialização do aplicativo testado com
TestServer
.
A documentação de testes de unidade descreve como configurar um projeto de teste e um executor de teste, juntamente com instruções detalhadas sobre como executar testes e recomendações sobre como nomear testes e classes de teste.
Separe os testes de unidade dos testes de integração em projetos diferentes. Separando os testes:
- Ajuda a garantir que os componentes de teste de infraestrutura não sejam incluídos acidentalmente nos testes de unidade.
- Permite o controle sobre quais conjuntos de testes são executados.
Praticamente não há diferença entre a configuração de testes de aplicativos do Razor Pages e aplicativos MVC. A única diferença é como os testes são nomeados. Em um aplicativo Razor Pages, os testes de pontos de extremidade de página geralmente recebem o mesmo nome da classe de modelo de página (por exemplo, IndexPageTests
para testar a integração de componentes para a página Índice). Em um aplicativo MVC, os testes geralmente são organizados por classes de controlador e recebem o mesmo nome dos controladores que testam (por exemplo, HomeControllerTests
para testar a integração de componentes para o controlador Home).
Pré-requisitos do aplicativo de teste
O projeto de teste deve:
- Referência ao pacote
Microsoft.AspNetCore.Mvc.Testing
. - Especifique o SDK da Web no arquivo de projeto (
<Project Sdk="Microsoft.NET.Sdk.Web">
).
Esses pré-requisitos podem ser vistos no aplicativo de exemplo. Inspecionar o tests/RazorPagesProject.Tests/RazorPagesProject.Tests.csproj
arquivo. O aplicativo de exemplo usa a estrutura de teste xUnit e a biblioteca do analisador AngleSharp, portanto, o aplicativo de exemplo também faz referência a:
Em aplicativos que usam xunit.runner.visualstudio
na versão 2.4.2 ou posterior, o projeto de teste deve referenciar o pacote Microsoft.NET.Test.Sdk
.
O Entity Framework Core também é usado nos testes. Consulte o arquivo de projeto no GitHub.
Ambiente do SUT
Se o ambiente do SUT não estiver definido, o ambiente usará como padrão o Desenvolvimento.
Testes básicos com o WebApplicationFactory padrão
Exponha a classe definida Program
implicitamente ao projeto de teste realizando um dos seguintes procedimentos:
Expor tipos internos do aplicativo Web ao projeto de teste. Isso pode ser feito no arquivo do projeto SUT (
.csproj
):<ItemGroup> <InternalsVisibleTo Include="MyTestProject" /> </ItemGroup>
Torne a
Program
classe pública usando uma declaração de classe parcial:var builder = WebApplication.CreateBuilder(args); // ... Configure services, routes, etc. app.Run(); + public partial class Program { }
O aplicativo de exemplo usa a abordagem de classe parcial
Program
.
WebApplicationFactory<TEntryPoint> é usado para criar um TestServer para os testes de integração. TEntryPoint
é a classe de ponto de entrada do SUT, geralmente Program.cs
.
As classes de teste implementam uma interface de acessório de classe (IClassFixture
) para indicar que a classe contém testes e fornece instâncias de objeto compartilhadas entre os testes na classe .
A classe de teste a seguir, BasicTests
, usa o WebApplicationFactory
para iniciar o SUT e fornecer um HttpClient para um método de teste, Get_EndpointsReturnSuccessAndCorrectContentType
. O método verifica se o código de status da resposta foi bem-sucedido (200-299) e o cabeçalho Content-Type
é text/html; charset=utf-8
em várias páginas do aplicativo.
CreateClient() cria uma instância do HttpClient
que segue automaticamente redirecionamentos e manipula cookies.
public class BasicTests
: IClassFixture<WebApplicationFactory<Program>>
{
private readonly WebApplicationFactory<Program> _factory;
public BasicTests(WebApplicationFactory<Program> factory)
{
_factory = factory;
}
[Theory]
[InlineData("/")]
[InlineData("/Index")]
[InlineData("/About")]
[InlineData("/Privacy")]
[InlineData("/Contact")]
public async Task Get_EndpointsReturnSuccessAndCorrectContentType(string url)
{
// Arrange
var client = _factory.CreateClient();
// Act
var response = await client.GetAsync(url);
// Assert
response.EnsureSuccessStatusCode(); // Status Code 200-299
Assert.Equal("text/html; charset=utf-8",
response.Content.Headers.ContentType.ToString());
}
}
Por padrão, os cookies não essenciais não são preservados entre solicitações quando a política de consentimento do Regulamento Geral sobre a Proteção de Dados está habilitada. Para preservar cookies não essenciais, como aqueles usados pelo provedor TempData, marque-os como essenciais em seus testes. Para obter instruções sobre como marcar um cookie como essencial, consulte Cookies essenciais.
Comparação entre AngleSharp e Application Parts
a respeito de verificações de proteção contra falsificações
Este artigo usa o analisador AngleSharp para manipular as verificações de proteção contra falsificações carregando páginas e analisando o HTML. Para testar os pontos de extremidade do controlador e as visualizações do Razor Pages em um nível inferior, sem se preocupar com a forma como eles são renderizados no navegador, considere o uso de Application Parts
. A abordagem Partes do Aplicativo injeta um controlador ou Razor page no aplicativo que pode ser usado para fazer solicitações JSON para obter os valores necessários. Para obter mais informações, consulte o blog Recursos de Teste de Integração do ASP.NET Core Protegido contra Falsificações Usando Partes do Aplicativo e o repositório GitHub associado por Martin Costello.
Personalizar WebApplicationFactory
A configuração de host da Web pode ser criada independentemente das classes de teste herdando de WebApplicationFactory<TEntryPoint> para criar uma ou mais fábricas personalizadas:
Herdar do
WebApplicationFactory
e substituir ConfigureWebHost. O IWebHostBuilder permite a configuração da coleção de serviços comIWebHostBuilder.ConfigureServices
public class CustomWebApplicationFactory<TProgram> : WebApplicationFactory<TProgram> where TProgram : class { protected override void ConfigureWebHost(IWebHostBuilder builder) { builder.ConfigureServices(services => { var dbContextDescriptor = services.SingleOrDefault( d => d.ServiceType == typeof(DbContextOptions<ApplicationDbContext>)); services.Remove(dbContextDescriptor); var dbConnectionDescriptor = services.SingleOrDefault( d => d.ServiceType == typeof(DbConnection)); services.Remove(dbConnectionDescriptor); // Create open SqliteConnection so EF won't automatically close it. services.AddSingleton<DbConnection>(container => { var connection = new SqliteConnection("DataSource=:memory:"); connection.Open(); return connection; }); services.AddDbContext<ApplicationDbContext>((container, options) => { var connection = container.GetRequiredService<DbConnection>(); options.UseSqlite(connection); }); }); builder.UseEnvironment("Development"); } }
A propagação de banco de dados no aplicativo de exemplo é executada pelo método
InitializeDbForTests
. O método é descrito na seção Exemplo de testes de integração: testar a organização do aplicativo .O contexto de banco de dados do SUT é registrado em
Program.cs
. O retorno de chamadabuilder.ConfigureServices
do aplicativo de teste é executado depois que o códigoProgram.cs
do aplicativo é executado. Para usar um banco de dados diferente para os testes além do banco de dados do aplicativo, o contexto do banco de dados do aplicativo deve ser substituído embuilder.ConfigureServices
.O aplicativo de exemplo localiza o descritor de serviço para o contexto do banco de dados e usa o descritor para remover o registro do serviço. Em seguida, a fábrica adiciona um novo
ApplicationDbContext
que usa um banco de dados na memória para os testes.Para se conectar a um banco de dados diferente, altere o
DbConnection
. Para usar um banco de dados de teste do SQL Server:- Faça referência ao pacote NuGet
Microsoft.EntityFrameworkCore.SqlServer
do arquivo de projeto. - Chame
UseInMemoryDatabase
.
- Faça referência ao pacote NuGet
Use o
CustomWebApplicationFactory
personalizado em classes de teste. O exemplo a seguir usa a fábrica na classeIndexPageTests
:public class IndexPageTests : IClassFixture<CustomWebApplicationFactory<Program>> { private readonly HttpClient _client; private readonly CustomWebApplicationFactory<Program> _factory; public IndexPageTests( CustomWebApplicationFactory<Program> factory) { _factory = factory; _client = factory.CreateClient(new WebApplicationFactoryClientOptions { AllowAutoRedirect = false }); }
O cliente do aplicativo de exemplo está configurado para evitar que
HttpClient
siga redirecionamentos. Conforme explicado posteriormente na seção Autenticação fictícia, isso permite que os testes verifiquem o resultado da primeira resposta do aplicativo. A primeira resposta é um redirecionamento em muitos desses testes com um cabeçalhoLocation
.Um teste típico usa
HttpClient
e métodos auxiliares para processar a solicitação e a resposta:[Fact] public async Task Post_DeleteAllMessagesHandler_ReturnsRedirectToRoot() { // Arrange var defaultPage = await _client.GetAsync("/"); var content = await HtmlHelpers.GetDocumentAsync(defaultPage); //Act var response = await _client.SendAsync( (IHtmlFormElement)content.QuerySelector("form[id='messages']"), (IHtmlButtonElement)content.QuerySelector("button[id='deleteAllBtn']")); // Assert Assert.Equal(HttpStatusCode.OK, defaultPage.StatusCode); Assert.Equal(HttpStatusCode.Redirect, response.StatusCode); Assert.Equal("/", response.Headers.Location.OriginalString); }
Qualquer solicitação POST para o SUT deve atender a verificação contra falsificações feita automaticamente pelo sistema de proteção de dados contra falsificações do aplicativo. Para organizar a solicitação POST de um teste, o aplicativo de teste deve:
- Fazer uma solicitação para a página.
- Analisar o cookie de validação de solicitação e antifalsificação da resposta.
- Faça a solicitação POST com o token de validação de solicitação e antifalsificação cookie em vigor.
Os métodos de extensão auxiliar SendAsync
(Helpers/HttpClientExtensions.cs
) e o método auxiliar GetDocumentAsync
(Helpers/HtmlHelpers.cs
) no aplicativo de exemplo usam o analisador AngleSharp para lidar com a verificação antifalsificação usando os seguintes métodos:
GetDocumentAsync
: recebe o HttpResponseMessage e retorna umIHtmlDocument
.GetDocumentAsync
usa uma fábrica que prepara uma resposta virtual com base noHttpResponseMessage
original. Para obter mais informações, consulte a documentação do AngleSharp.SendAsync
métodos de extensão para a composiçãoHttpClient
de um HttpRequestMessage e fazer a chamada de SendAsync(HttpRequestMessage) e para enviar solicitações para o SUT. As sobrecargas paraSendAsync
aceitam o formulário HTML (IHtmlFormElement
) e os seguintes itens:- Botão Enviar do formulário (
IHtmlElement
) - Coleção de valores do formulário (
IEnumerable<KeyValuePair<string, string>>
) - Botão Enviar (
IHtmlElement
) e valores do formulário (IEnumerable<KeyValuePair<string, string>>
)
- Botão Enviar do formulário (
O AngleSharp é uma biblioteca de análise de terceiros usada para fins de demonstração neste artigo e no aplicativo de exemplo. O AngleSharp não tem suporte nem é necessário para testes de integração de aplicativos ASP.NET Core. Outros analisadores podem ser usados, como o HAP (Pacote de Agilidade Html). Outra abordagem é escrever código para lidar diretamente com o token de verificação de solicitação do sistema antifalsificação e com o cookie antifalsificação diretamente. Consulte AngleSharp vs Application Parts
para verificações antifalsificação neste artigo para obter mais informações.
O provedor de banco de dados na memória do EF-Core pode ser usado para testes limitados e básicos; no entanto, o provedor SQLite é a opção recomendada para testes na memória.
Consulte Estender a inicialização com filtros de inicialização que mostra como configurar o middleware usando IStartupFilter, o que é útil quando um teste requer um serviço ou middleware personalizado.
Personalizar o cliente com WithWebHostBuilder
Quando uma configuração adicional é necessária em um método de teste, WithWebHostBuilder cria um novo WebApplicationFactory
com um IWebHostBuilder que é personalizado ainda mais pela configuração.
O código de exemplo chama WithWebHostBuilder
para substituir serviços configurados com stubs de teste. Para obter mais informações e exemplos de uso, consulte Injetar serviços fictícios neste artigo.
O método de teste Post_DeleteMessageHandler_ReturnsRedirectToRoot
do aplicativo de exemplo demonstra o uso de WithWebHostBuilder
. Esse teste executa uma exclusão de registro no banco de dados disparando um envio de formulário no SUT.
Como outro teste na classe IndexPageTests
executa uma operação que exclui todos os registros no banco de dados e pode ser executada antes do método Post_DeleteMessageHandler_ReturnsRedirectToRoot
, o banco de dados é propagado novamente neste método de teste para garantir que o registro esteja presente para o SUT excluir. A seleção do primeiro botão excluir do formulário messages
do SUT é simulada na solicitação para o SUT:
[Fact]
public async Task Post_DeleteMessageHandler_ReturnsRedirectToRoot()
{
// Arrange
using (var scope = _factory.Services.CreateScope())
{
var scopedServices = scope.ServiceProvider;
var db = scopedServices.GetRequiredService<ApplicationDbContext>();
Utilities.ReinitializeDbForTests(db);
}
var defaultPage = await _client.GetAsync("/");
var content = await HtmlHelpers.GetDocumentAsync(defaultPage);
//Act
var response = await _client.SendAsync(
(IHtmlFormElement)content.QuerySelector("form[id='messages']"),
(IHtmlButtonElement)content.QuerySelector("form[id='messages']")
.QuerySelector("div[class='panel-body']")
.QuerySelector("button"));
// Assert
Assert.Equal(HttpStatusCode.OK, defaultPage.StatusCode);
Assert.Equal(HttpStatusCode.Redirect, response.StatusCode);
Assert.Equal("/", response.Headers.Location.OriginalString);
}
Opções do cliente
Consulte a página WebApplicationFactoryClientOptions para ver as definições padrão e as opções disponíveis ao criar instâncias HttpClient
.
Crie a classe WebApplicationFactoryClientOptions
e passe-a para o método CreateClient():
public class IndexPageTests :
IClassFixture<CustomWebApplicationFactory<Program>>
{
private readonly HttpClient _client;
private readonly CustomWebApplicationFactory<Program>
_factory;
public IndexPageTests(
CustomWebApplicationFactory<Program> factory)
{
_factory = factory;
_client = factory.CreateClient(new WebApplicationFactoryClientOptions
{
AllowAutoRedirect = false
});
}
NOTA: Para evitar avisos de redirecionamento HTTPS em logs ao usar o middleware de redirecionamento HTTPS, defina BaseAddress = new Uri("https://localhost")
Injetar serviços fictícios
Os serviços podem ser substituídos em um teste com uma chamada para ConfigureTestServices no construtor de host. Para definir o escopo dos serviços substituídos para o próprio teste, o método WithWebHostBuilder é usado para recuperar um construtor de host. Isso pode ser visto nos seguintes testes:
- Get_QuoteService_ProvidesQuoteInPage
- Get_GithubProfilePageCanGetAGithubUser
- Get_SecurePageIsReturnedForAnAuthenticatedUser
O SUT de exemplo inclui um serviço com escopo que retorna uma citação. A citação é integrada em um campo oculto na página de Índice quando a página de Índice é solicitada.
Services/IQuoteService.cs
:
public interface IQuoteService
{
Task<string> GenerateQuote();
}
Services/QuoteService.cs
:
// Quote ©1975 BBC: The Doctor (Tom Baker); Dr. Who: Planet of Evil
// https://www.bbc.co.uk/programmes/p00pyrx6
public class QuoteService : IQuoteService
{
public Task<string> GenerateQuote()
{
return Task.FromResult<string>(
"Come on, Sarah. We've an appointment in London, " +
"and we're already 30,000 years late.");
}
}
Program.cs
:
services.AddScoped<IQuoteService, QuoteService>();
Pages/Index.cshtml.cs
:
public class IndexModel : PageModel
{
private readonly ApplicationDbContext _db;
private readonly IQuoteService _quoteService;
public IndexModel(ApplicationDbContext db, IQuoteService quoteService)
{
_db = db;
_quoteService = quoteService;
}
[BindProperty]
public Message Message { get; set; }
public IList<Message> Messages { get; private set; }
[TempData]
public string MessageAnalysisResult { get; set; }
public string Quote { get; private set; }
public async Task OnGetAsync()
{
Messages = await _db.GetMessagesAsync();
Quote = await _quoteService.GenerateQuote();
}
Pages/Index.cs
:
<input id="quote" type="hidden" value="@Model.Quote">
A marcação a seguir é gerada quando o aplicativo SUT é executado:
<input id="quote" type="hidden" value="Come on, Sarah. We've an appointment in
London, and we're already 30,000 years late.">
Para testar a injeção de serviço e de citação em um teste de integração, um serviço fictício é injetado no SUT pelo teste. O serviço fictício substitui o QuoteService
do aplicativo por um serviço fornecido pelo aplicativo de teste, chamado TestQuoteService
:
IntegrationTests.IndexPageTests.cs
:
// Quote ©1975 BBC: The Doctor (Tom Baker); Pyramids of Mars
// https://www.bbc.co.uk/programmes/p00pys55
public class TestQuoteService : IQuoteService
{
public Task<string> GenerateQuote()
{
return Task.FromResult(
"Something's interfering with time, Mr. Scarman, " +
"and time is my business.");
}
}
ConfigureTestServices
é chamado e o serviço com escopo é registrado:
[Fact]
public async Task Get_QuoteService_ProvidesQuoteInPage()
{
// Arrange
var client = _factory.WithWebHostBuilder(builder =>
{
builder.ConfigureTestServices(services =>
{
services.AddScoped<IQuoteService, TestQuoteService>();
});
})
.CreateClient();
//Act
var defaultPage = await client.GetAsync("/");
var content = await HtmlHelpers.GetDocumentAsync(defaultPage);
var quoteElement = content.QuerySelector("#quote");
// Assert
Assert.Equal("Something's interfering with time, Mr. Scarman, " +
"and time is my business.", quoteElement.Attributes["value"].Value);
}
A marcação produzida durante a execução do teste reflete o texto entre aspas fornecido por TestQuoteService
, portanto, a asserção passa:
<input id="quote" type="hidden" value="Something's interfering with time,
Mr. Scarman, and time is my business.">
Autenticação fictícia
Os testes na classe AuthTests
verificam que um ponto de extremidade seguro:
- Redireciona o usuário não autenticado para a página de entrada do aplicativo.
- Retorna conteúdo para o usuário autenticado.
No SUT, a página /SecurePage
usa uma convenção AuthorizePage para aplicar um AuthorizeFilter à página. Para obter mais informações, consulte Convenções de autorização do Razor Pages.
services.AddRazorPages(options =>
{
options.Conventions.AuthorizePage("/SecurePage");
});
No teste Get_SecurePageRedirectsAnUnauthenticatedUser
, um WebApplicationFactoryClientOptions é definido para revogar a permissão de redirecionamentos definindo AllowAutoRedirect como false
:
[Fact]
public async Task Get_SecurePageRedirectsAnUnauthenticatedUser()
{
// Arrange
var client = _factory.CreateClient(
new WebApplicationFactoryClientOptions
{
AllowAutoRedirect = false
});
// Act
var response = await client.GetAsync("/SecurePage");
// Assert
Assert.Equal(HttpStatusCode.Redirect, response.StatusCode);
Assert.StartsWith("http://localhost/Identity/Account/Login",
response.Headers.Location.OriginalString);
}
Ao não permitir que o cliente siga o redirecionamento, as seguintes verificações podem ser feitas:
- O código de status retornado pelo SUT pode ser verificado em relação ao resultado esperado HttpStatusCode.Redirect, não o código de status final após o redirecionamento para a página de entrada, que seria HttpStatusCode.OK.
- O valor do cabeçalho
Location
nos cabeçalhos de resposta é verificado para confirmar que ele começa comhttp://localhost/Identity/Account/Login
, não a resposta final da página de entrada, em que o cabeçalhoLocation
não estaria presente.
O aplicativo de teste pode simular um AuthenticationHandler<TOptions> no ConfigureTestServices para testar aspectos de autenticação e autorização. Um cenário mínimo retorna um AuthenticateResult.Success:
public class TestAuthHandler : AuthenticationHandler<AuthenticationSchemeOptions>
{
public TestAuthHandler(IOptionsMonitor<AuthenticationSchemeOptions> options,
ILoggerFactory logger, UrlEncoder encoder)
: base(options, logger, encoder)
{
}
protected override Task<AuthenticateResult> HandleAuthenticateAsync()
{
var claims = new[] { new Claim(ClaimTypes.Name, "Test user") };
var identity = new ClaimsIdentity(claims, "Test");
var principal = new ClaimsPrincipal(identity);
var ticket = new AuthenticationTicket(principal, "TestScheme");
var result = AuthenticateResult.Success(ticket);
return Task.FromResult(result);
}
}
O TestAuthHandler
é chamado para autenticar um usuário quando o esquema de autenticação é definido como TestScheme
onde AddAuthentication
está registrado para ConfigureTestServices
. É importante que o esquema TestScheme
corresponda ao esquema que seu aplicativo espera. Caso contrário, a autenticação não funcionará.
[Fact]
public async Task Get_SecurePageIsReturnedForAnAuthenticatedUser()
{
// Arrange
var client = _factory.WithWebHostBuilder(builder =>
{
builder.ConfigureTestServices(services =>
{
services.AddAuthentication(defaultScheme: "TestScheme")
.AddScheme<AuthenticationSchemeOptions, TestAuthHandler>(
"TestScheme", options => { });
});
})
.CreateClient(new WebApplicationFactoryClientOptions
{
AllowAutoRedirect = false,
});
client.DefaultRequestHeaders.Authorization =
new AuthenticationHeaderValue(scheme: "TestScheme");
//Act
var response = await client.GetAsync("/SecurePage");
// Assert
Assert.Equal(HttpStatusCode.OK, response.StatusCode);
}
Para obter mais informações sobre WebApplicationFactoryClientOptions
, confira a seção Opções do cliente.
Testes básicos para middlewares de autenticação
Consulte este repositório do GitHub para obter testes básicos de middlewares de autenticação. Ele contém um servidor de teste específico para o cenário de teste.
Definir o ambiente
Defina o ambiente na fábrica de aplicativos personalizada:
public class CustomWebApplicationFactory<TProgram>
: WebApplicationFactory<TProgram> where TProgram : class
{
protected override void ConfigureWebHost(IWebHostBuilder builder)
{
builder.ConfigureServices(services =>
{
var dbContextDescriptor = services.SingleOrDefault(
d => d.ServiceType ==
typeof(DbContextOptions<ApplicationDbContext>));
services.Remove(dbContextDescriptor);
var dbConnectionDescriptor = services.SingleOrDefault(
d => d.ServiceType ==
typeof(DbConnection));
services.Remove(dbConnectionDescriptor);
// Create open SqliteConnection so EF won't automatically close it.
services.AddSingleton<DbConnection>(container =>
{
var connection = new SqliteConnection("DataSource=:memory:");
connection.Open();
return connection;
});
services.AddDbContext<ApplicationDbContext>((container, options) =>
{
var connection = container.GetRequiredService<DbConnection>();
options.UseSqlite(connection);
});
});
builder.UseEnvironment("Development");
}
}
Como a infraestrutura de teste infere o caminho raiz de conteúdo do aplicativo
O construtor WebApplicationFactory
infere o caminho raiz de conteúdo do aplicativo pesquisando um WebApplicationFactoryContentRootAttribute no assembly que contém os testes de integração com uma chave igual ao assembly TEntryPoint
System.Reflection.Assembly.FullName
. Caso não seja encontrado um atributo com a chave correta, WebApplicationFactory
volta a procurar um arquivo de solução (.sln) e acrescente o nome do assembly TEntryPoint
ao diretório da solução. O diretório raiz do aplicativo (o caminho raiz do conteúdo) é usado para descobrir exibições e arquivos de conteúdo.
Desabilitar a cópia de sombra
A cópia de sombra faz com que os testes sejam executados em um diretório diferente do diretório de saída. Se seus testes dependerem do carregamento de arquivos relativos a Assembly.Location
e você encontrar problemas, talvez seja necessário desabilitar a cópia de sombra.
Para desabilitar a cópia de sombra ao usar xUnit, crie um arquivo xunit.runner.json
no diretório do projeto de teste, com a configuração correta:
{
"shadowCopy": false
}
Descarte de objetos
Depois que os testes da implementação de IClassFixture
são executados, TestServer e HttpClient são descartados quando xUnit descarta o WebApplicationFactory
. Se os objetos instanciados pelo desenvolvedor exigirem descarte, descarte-os na IClassFixture
implementação. Para saber mais, confira Implementação de um método Dispose.
Exemplo de testes de integração
O aplicativo de exemplo é composto por dois aplicativos:
Aplicativo | Diretório do projeto | Descrição |
---|---|---|
Aplicativo de mensagens (o SUT) | src/RazorPagesProject |
Permite que o usuário adicione, exclua uma ou todas as mensagens e as analise. |
Aplicativo de teste | tests/RazorPagesProject.Tests |
Usado para fazer o teste de integração do SUT. |
Os testes podem ser executados usando os recursos de teste nativos do IDE, como o Visual Studio. Se estiver usando o Visual Studio Code ou a linha de comando, execute o seguinte comando em um prompt de comando na pasta tests/RazorPagesProject.Tests
:
dotnet test
Organização do aplicativo de mensagens (SUT)
O SUT é um sistema de mensagens do Razor Pages com as seguintes características:
- A página de Índice do aplicativo (
Pages/Index.cshtml
ePages/Index.cshtml.cs
) fornece uma interface do usuário e métodos de modelo de página para controlar a adição, exclusão e análise de mensagens (número médio de palavras por mensagem). - Uma mensagem é descrita pela classe
Message
(Data/Message.cs
) com duas propriedades:Id
(chave) eText
(mensagem). A propriedadeText
é necessária e limitada a 200 caracteres. - As mensagens são armazenadas usando o banco de dados na memória do Entity Framework†.
- O aplicativo contém uma DAL (camada de acesso a dados) em sua classe de contexto de banco de dados,
AppDbContext
(Data/AppDbContext.cs
). - Se o banco de dados estiver vazio na inicialização do aplicativo, o repositório de mensagens será inicializado com três mensagens.
- O aplicativo inclui um
/SecurePage
que só pode ser acessado por um usuário autenticado.
†O tópico do EF, Teste com InMemory, explica como usar um banco de dados na memória para testes com MSTest. Este tópico usa a estrutura de teste xUnit. Os conceitos de teste e as implementações de teste em diferentes estruturas de teste são semelhantes, mas não idênticos.
Embora o aplicativo não use o padrão do repositório e não seja um exemplo eficaz do padrão UoW (Unidade de Trabalho), o Razor Pages dá suporte a esses padrões de desenvolvimento. Para obter mais informações, confira Criando a camada de persistência da infraestrutura e Lógica do controlador de teste (o exemplo implementa o padrão do repositório).
Organização do aplicativo de teste
O aplicativo de teste é um aplicativo de console dentro do diretório tests/RazorPagesProject.Tests
.
Diretório do aplicativo de teste | Descrição |
---|---|
AuthTests |
Contém métodos de teste para:
|
BasicTests |
Contém um método de teste para roteamento e tipo de conteúdo. |
IntegrationTests |
Contém os testes de integração para a página de Índice usando a classe personalizada WebApplicationFactory . |
Helpers/Utilities |
|
A estrutura de teste é xUnit. Os testes de integração são realizados usando o Microsoft.AspNetCore.TestHost, que inclui o TestServer. Como o pacote Microsoft.AspNetCore.Mvc.Testing
é usado para configurar o host de teste e o servidor de teste, os pacotes TestHost
e TestServer
não exigem referências diretas do pacote no arquivo de projeto do aplicativo de teste ou na configuração do desenvolvedor no aplicativo de teste.
Os testes de integração geralmente exigem um pequeno conjunto de dados no banco de dados antes da execução do teste. Por exemplo, um teste de exclusão exige a exclusão de um registro do banco de dados; portanto, o banco de dados deve ter pelo menos um registro para que a solicitação de exclusão tenha êxito.
O aplicativo de exemplo propaga o banco de dados com três mensagens em Utilities.cs
que os testes podem usar quando são executados:
public static void InitializeDbForTests(ApplicationDbContext db)
{
db.Messages.AddRange(GetSeedingMessages());
db.SaveChanges();
}
public static void ReinitializeDbForTests(ApplicationDbContext db)
{
db.Messages.RemoveRange(db.Messages);
InitializeDbForTests(db);
}
public static List<Message> GetSeedingMessages()
{
return new List<Message>()
{
new Message(){ Text = "TEST RECORD: You're standing on my scarf." },
new Message(){ Text = "TEST RECORD: Would you like a jelly baby?" },
new Message(){ Text = "TEST RECORD: To the rational mind, " +
"nothing is inexplicable; only unexplained." }
};
}
O contexto de banco de dados do SUT é registrado em Program.cs
. O retorno de chamada builder.ConfigureServices
do aplicativo de teste é executado depois que o código Program.cs
do aplicativo é executado. Para usar um banco de dados diferente para os testes, o contexto do banco de dados do aplicativo deve ser substituído em builder.ConfigureServices
. Para obter mais informações, consulte a seção Personalizar WebApplicationFactory .
Recursos adicionais
Este artigo pressupõe uma compreensão básica sobre testes de unidade. Se não estiver familiarizado com os conceitos de teste, consulte o artigo Teste de Unidade no .NET Core e no .NET Standard e seu conteúdo vinculado.
Exibir ou baixar código de exemplo (como baixar)
O aplicativo de exemplo é um Razor aplicativo Pages e pressupõe uma compreensão básica do Razor Pages. Se não estiver familiarizado com o Razor Pages, consulte os seguintes tópicos:
Observação
Para testar SPAs, recomendamos uma ferramenta como o Playwright para .NET, que pode automatizar um navegador.
Introdução aos testes de integração
Os testes de integração avaliam os componentes de um aplicativo em um nível mais amplo do que os testes de unidade. Os testes de unidade são usados para testar componentes de software isolados, como métodos de classe individuais. Os testes de integração confirmam que dois ou mais componentes de aplicativo trabalham juntos para produzir um resultado esperado, possivelmente incluindo todos os componentes necessários para processar totalmente uma solicitação.
Esses testes mais amplos são usados para testar a infraestrutura do aplicativo e toda a estrutura, geralmente incluindo os seguintes componentes:
- Banco de dados
- Sistema de arquivos
- Dispositivos de rede
- Pipeline de solicitação-resposta
Os testes de unidade usam componentes fabricados, conhecidos como fakes ou objetos fictícios, no lugar de componentes de infraestrutura.
Ao contrário dos testes de unidade, os testes de integração:
- Usam os componentes reais que o aplicativo usa em produção.
- Exigem mais código e processamento de dados.
- Demoraram mais para serem executados.
Portanto, limite o uso de testes de integração aos cenários de infraestrutura mais importantes. Se um comportamento puder ser testado usando um teste de unidade ou um teste de integração, escolha o teste de unidade.
Em discussões sobre testes de integração, o projeto testado é frequentemente chamado de System Under Test ou "SUT" para abreviar. "SUT" é usado ao longo deste artigo para se referir ao aplicativo ASP.NET Core que está sendo testado.
Não escreva testes de integração para cada permutação de dados e acesso a arquivos com bancos de dados e sistemas de arquivos. Independentemente de quantos lugares em um aplicativo interagem com bancos de dados e sistemas de arquivos, um conjunto focado de testes de integração de leitura, gravação, atualização e exclusão geralmente é capaz de testar adequadamente os componentes do banco de dados e do sistema de arquivos. Use testes de unidade para testes de rotina da lógica do método que interage com esses componentes. Em testes de unidade, o uso de infraestruturas falsas ou fictícias resulta em uma execução de teste mais rápida.
Testes de integração de ASP.NET Core
Os testes de integração no ASP.NET Core exigem o seguinte:
- Um projeto de teste é usado para conter e executar os testes. O projeto de teste tem uma referência ao SUT.
- O projeto de teste cria um host Web de teste para o SUT e usa um cliente do servidor de teste para lidar com solicitações e respostas com o SUT.
- Um executor de teste é usado para executar os testes e relatar os resultados.
Os testes de integração seguem uma sequência de eventos que incluem as etapas de teste Organizar, Atuar e Afirmar:
- O host da Web do SUT está configurado.
- Um cliente do servidor de teste é criado para enviar solicitações ao aplicativo.
- A etapa de teste Organizar é executada: o aplicativo de teste prepara uma solicitação.
- A etapa de teste Atuar é executada: o cliente envia a solicitação e recebe a resposta.
- A etapa de teste Afirmar é executada: a resposta real é validada como uma aprovada ou reprovada com base em uma resposta esperada.
- O processo continua até que todos os testes sejam executados.
- Os resultados do teste são reportados.
Normalmente, o host da Web de teste é configurado de forma diferente do host da Web normal do aplicativo para as execuções de teste. Por exemplo, um banco de dados diferente ou configurações de aplicativo diferentes podem ser usadas para os testes.
Os componentes de infraestrutura, como o host da Web de teste e o servidor de teste na memória (TestServer), são fornecidos ou gerenciados pelo pacote Microsoft.AspNetCore.Mvc.Testing. O uso desse pacote simplifica a criação e a execução do teste.
O pacote Microsoft.AspNetCore.Mvc.Testing
manipula as seguintes tarefas:
- Copia o arquivo de dependências (
.deps
) do SUT para o diretóriobin
do projeto de teste. - Define a raiz de conteúdo para a raiz do projeto do SUT para que arquivos estáticos e páginas/exibições sejam encontradas quando os testes forem executados.
- Fornece a classe WebApplicationFactory para simplificar a inicialização do aplicativo testado com
TestServer
.
A documentação de testes de unidade descreve como configurar um projeto de teste e um executor de teste, juntamente com instruções detalhadas sobre como executar testes e recomendações sobre como nomear testes e classes de teste.
Separe os testes de unidade dos testes de integração em projetos diferentes. Separando os testes:
- Ajuda a garantir que os componentes de teste de infraestrutura não sejam incluídos acidentalmente nos testes de unidade.
- Permite o controle sobre quais conjuntos de testes são executados.
Praticamente não há diferença entre a configuração de testes de aplicativos do Razor Pages e aplicativos MVC. A única diferença é como os testes são nomeados. Em um aplicativo Razor Pages, os testes de pontos de extremidade de página geralmente recebem o mesmo nome da classe de modelo de página (por exemplo, IndexPageTests
para testar a integração de componentes para a página Índice). Em um aplicativo MVC, os testes geralmente são organizados por classes de controlador e recebem o mesmo nome dos controladores que testam (por exemplo, HomeControllerTests
para testar a integração de componentes para o controlador Home).
Pré-requisitos do aplicativo de teste
O projeto de teste deve:
- Referência ao pacote
Microsoft.AspNetCore.Mvc.Testing
. - Especifique o SDK da Web no arquivo de projeto (
<Project Sdk="Microsoft.NET.Sdk.Web">
).
Esses pré-requisitos podem ser vistos no aplicativo de exemplo. Inspecionar o tests/RazorPagesProject.Tests/RazorPagesProject.Tests.csproj
arquivo. O aplicativo de exemplo usa a estrutura de teste xUnit e a biblioteca do analisador AngleSharp, portanto, o aplicativo de exemplo também faz referência a:
Em aplicativos que usam xunit.runner.visualstudio
na versão 2.4.2 ou posterior, o projeto de teste deve referenciar o pacote Microsoft.NET.Test.Sdk
.
O Entity Framework Core também é usado nos testes. O aplicativo faz referência:
Microsoft.AspNetCore.Diagnostics.EntityFrameworkCore
Microsoft.AspNetCore.Identity.EntityFrameworkCore
Microsoft.EntityFrameworkCore
Microsoft.EntityFrameworkCore.InMemory
Microsoft.EntityFrameworkCore.Tools
Ambiente do SUT
Se o ambiente do SUT não estiver definido, o ambiente usará como padrão o Desenvolvimento.
Testes básicos com o WebApplicationFactory padrão
WebApplicationFactory<TEntryPoint> é usado para criar um TestServer para os testes de integração. TEntryPoint
é a classe de ponto de entrada do SUT, geralmente a classe Startup
.
As classes de teste implementam uma interface de acessório de classe (IClassFixture
) para indicar que a classe contém testes e fornece instâncias de objeto compartilhadas entre os testes na classe .
A classe de teste a seguir, BasicTests
, usa o WebApplicationFactory
para iniciar o SUT e fornecer um HttpClient para um método de teste, Get_EndpointsReturnSuccessAndCorrectContentType
. O método verifica se a resposta do código de status é bem-sucedida (códigos de status no intervalo de 200 a 299) e o cabeçalho Content-Type
é text/html; charset=utf-8
para várias páginas de aplicativos.
CreateClient() cria uma instância do HttpClient
que segue automaticamente redirecionamentos e manipula cookies.
public class BasicTests
: IClassFixture<WebApplicationFactory<RazorPagesProject.Startup>>
{
private readonly WebApplicationFactory<RazorPagesProject.Startup> _factory;
public BasicTests(WebApplicationFactory<RazorPagesProject.Startup> factory)
{
_factory = factory;
}
[Theory]
[InlineData("/")]
[InlineData("/Index")]
[InlineData("/About")]
[InlineData("/Privacy")]
[InlineData("/Contact")]
public async Task Get_EndpointsReturnSuccessAndCorrectContentType(string url)
{
// Arrange
var client = _factory.CreateClient();
// Act
var response = await client.GetAsync(url);
// Assert
response.EnsureSuccessStatusCode(); // Status Code 200-299
Assert.Equal("text/html; charset=utf-8",
response.Content.Headers.ContentType.ToString());
}
}
Por padrão, os cookies não essenciais não são preservados entre solicitações quando a Política de consentimento do GDPR está habilitada. Para preservar cookies não essenciais, como aqueles usados pelo provedor TempData, marque-os como essenciais em seus testes. Para obter instruções sobre como marcar um cookie como essencial, consulte Cookies essenciais.
Personalizar WebApplicationFactory
A configuração de host da Web pode ser criada independentemente das classes de teste herdando de WebApplicationFactory
para criar uma ou mais fábricas personalizadas:
Herdar do
WebApplicationFactory
e substituir ConfigureWebHost. O IWebHostBuilder permite a configuração da coleção de serviços com ConfigureServices:public class CustomWebApplicationFactory<TStartup> : WebApplicationFactory<TStartup> where TStartup: class { protected override void ConfigureWebHost(IWebHostBuilder builder) { builder.ConfigureServices(services => { var descriptor = services.SingleOrDefault( d => d.ServiceType == typeof(DbContextOptions<ApplicationDbContext>)); services.Remove(descriptor); services.AddDbContext<ApplicationDbContext>(options => { options.UseInMemoryDatabase("InMemoryDbForTesting"); }); var sp = services.BuildServiceProvider(); using (var scope = sp.CreateScope()) { var scopedServices = scope.ServiceProvider; var db = scopedServices.GetRequiredService<ApplicationDbContext>(); var logger = scopedServices .GetRequiredService<ILogger<CustomWebApplicationFactory<TStartup>>>(); db.Database.EnsureCreated(); try { Utilities.InitializeDbForTests(db); } catch (Exception ex) { logger.LogError(ex, "An error occurred seeding the " + "database with test messages. Error: {Message}", ex.Message); } } }); } }
A propagação de banco de dados no aplicativo de exemplo é executada pelo método
InitializeDbForTests
. O método é descrito na seção Exemplo de testes de integração: testar a organização do aplicativo .O contexto de banco de dados do SUT é registrado em seu método
Startup.ConfigureServices
. O retorno de chamadabuilder.ConfigureServices
do aplicativo de teste é executado depois que o códigoStartup.ConfigureServices
do aplicativo é executado. A ordem de execução é uma alteração interruptiva para o Host Genérico com o lançamento do ASP.NET Core 3.0. Para usar um banco de dados diferente para os testes além do banco de dados do aplicativo, o contexto do banco de dados do aplicativo deve ser substituído embuilder.ConfigureServices
.Nos SUTs que ainda usam o Host Da Web, o retorno de chamada
builder.ConfigureServices
do aplicativo de teste é executado antes do códigoStartup.ConfigureServices
do SUT. O retorno de chamadabuilder.ConfigureTestServices
do aplicativo de teste é executado após.O aplicativo de exemplo localiza o descritor de serviço para o contexto do banco de dados e usa o descritor para remover o registro do serviço. Em seguida, a fábrica adiciona um novo
ApplicationDbContext
que usa um banco de dados na memória para os testes.Para se conectar a um banco de dados diferente do banco de dados na memória, altere a chamada
UseInMemoryDatabase
para conectar o contexto a um banco de dados diferente. Para usar um banco de dados de teste do SQL Server:- Faça referência ao pacote NuGet
Microsoft.EntityFrameworkCore.SqlServer
do arquivo de projeto. - Faça a chamada de
UseSqlServer
com uma cadeia de caracteres de conexão ao banco de dados.
services.AddDbContext<ApplicationDbContext>((options, context) => { context.UseSqlServer( Configuration.GetConnectionString("TestingDbConnectionString")); });
- Faça referência ao pacote NuGet
Use o
CustomWebApplicationFactory
personalizado em classes de teste. O exemplo a seguir usa a fábrica na classeIndexPageTests
:public class IndexPageTests : IClassFixture<CustomWebApplicationFactory<RazorPagesProject.Startup>> { private readonly HttpClient _client; private readonly CustomWebApplicationFactory<RazorPagesProject.Startup> _factory; public IndexPageTests( CustomWebApplicationFactory<RazorPagesProject.Startup> factory) { _factory = factory; _client = factory.CreateClient(new WebApplicationFactoryClientOptions { AllowAutoRedirect = false }); }
O cliente do aplicativo de exemplo está configurado para evitar que
HttpClient
siga redirecionamentos. Conforme explicado posteriormente na seção Autenticação fictícia, isso permite que os testes verifiquem o resultado da primeira resposta do aplicativo. A primeira resposta é um redirecionamento em muitos desses testes com um cabeçalhoLocation
.Um teste típico usa
HttpClient
e métodos auxiliares para processar a solicitação e a resposta:[Fact] public async Task Post_DeleteAllMessagesHandler_ReturnsRedirectToRoot() { // Arrange var defaultPage = await _client.GetAsync("/"); var content = await HtmlHelpers.GetDocumentAsync(defaultPage); //Act var response = await _client.SendAsync( (IHtmlFormElement)content.QuerySelector("form[id='messages']"), (IHtmlButtonElement)content.QuerySelector("button[id='deleteAllBtn']")); // Assert Assert.Equal(HttpStatusCode.OK, defaultPage.StatusCode); Assert.Equal(HttpStatusCode.Redirect, response.StatusCode); Assert.Equal("/", response.Headers.Location.OriginalString); }
Qualquer solicitação POST para o SUT deve atender a verificação contra falsificações feita automaticamente pelo sistema de proteção de dados contra falsificações do aplicativo. Para organizar a solicitação POST de um teste, o aplicativo de teste deve:
- Fazer uma solicitação para a página.
- Analisar o cookie de validação de solicitação e antifalsificação da resposta.
- Faça a solicitação POST com o token de validação de solicitação e antifalsificação cookie em vigor.
Os métodos de extensão auxiliar SendAsync
(Helpers/HttpClientExtensions.cs
) e o método auxiliar GetDocumentAsync
(Helpers/HtmlHelpers.cs
) no aplicativo de exemplo usam o analisador AngleSharp para lidar com a verificação antifalsificação usando os seguintes métodos:
GetDocumentAsync
: recebe o HttpResponseMessage e retorna umIHtmlDocument
.GetDocumentAsync
usa uma fábrica que prepara uma resposta virtual com base noHttpResponseMessage
original. Para obter mais informações, consulte a documentação do AngleSharp.SendAsync
métodos de extensão para a composiçãoHttpClient
de um HttpRequestMessage e fazer a chamada de SendAsync(HttpRequestMessage) e para enviar solicitações para o SUT. As sobrecargas paraSendAsync
aceitam o formulário HTML (IHtmlFormElement
) e os seguintes itens:- Botão Enviar do formulário (
IHtmlElement
) - Coleção de valores do formulário (
IEnumerable<KeyValuePair<string, string>>
) - Botão Enviar (
IHtmlElement
) e valores do formulário (IEnumerable<KeyValuePair<string, string>>
)
- Botão Enviar do formulário (
Observação
O AngleSharp é uma biblioteca de análise de terceiros usada para fins de demonstração neste artigo e no aplicativo de exemplo. O AngleSharp não tem suporte nem é necessário para testes de integração de aplicativos ASP.NET Core. Outros analisadores podem ser usados, como o HAP (Pacote de Agilidade Html). Outra abordagem é escrever código para lidar diretamente com o token de verificação de solicitação do sistema antifalsificação e com o cookie antifalsificação diretamente.
Observação
O provedor de banco de dados na memória do EF-Core pode ser usado para testes limitados e básicos; no entanto, o provedor SQLite é a opção recomendada para testes na memória.
Personalizar o cliente com WithWebHostBuilder
Quando uma configuração adicional é necessária em um método de teste, WithWebHostBuilder cria um novo WebApplicationFactory
com um IWebHostBuilder que é personalizado ainda mais pela configuração.
O método de teste Post_DeleteMessageHandler_ReturnsRedirectToRoot
do aplicativo de exemplo demonstra o uso de WithWebHostBuilder
. Esse teste executa uma exclusão de registro no banco de dados disparando um envio de formulário no SUT.
Como outro teste na classe IndexPageTests
executa uma operação que exclui todos os registros no banco de dados e pode ser executada antes do método Post_DeleteMessageHandler_ReturnsRedirectToRoot
, o banco de dados é propagado novamente neste método de teste para garantir que o registro esteja presente para o SUT excluir. A seleção do primeiro botão excluir do formulário messages
do SUT é simulada na solicitação para o SUT:
[Fact]
public async Task Post_DeleteMessageHandler_ReturnsRedirectToRoot()
{
// Arrange
var client = _factory.WithWebHostBuilder(builder =>
{
builder.ConfigureServices(services =>
{
var serviceProvider = services.BuildServiceProvider();
using (var scope = serviceProvider.CreateScope())
{
var scopedServices = scope.ServiceProvider;
var db = scopedServices
.GetRequiredService<ApplicationDbContext>();
var logger = scopedServices
.GetRequiredService<ILogger<IndexPageTests>>();
try
{
Utilities.ReinitializeDbForTests(db);
}
catch (Exception ex)
{
logger.LogError(ex, "An error occurred seeding " +
"the database with test messages. Error: {Message}",
ex.Message);
}
}
});
})
.CreateClient(new WebApplicationFactoryClientOptions
{
AllowAutoRedirect = false
});
var defaultPage = await client.GetAsync("/");
var content = await HtmlHelpers.GetDocumentAsync(defaultPage);
//Act
var response = await client.SendAsync(
(IHtmlFormElement)content.QuerySelector("form[id='messages']"),
(IHtmlButtonElement)content.QuerySelector("form[id='messages']")
.QuerySelector("div[class='panel-body']")
.QuerySelector("button"));
// Assert
Assert.Equal(HttpStatusCode.OK, defaultPage.StatusCode);
Assert.Equal(HttpStatusCode.Redirect, response.StatusCode);
Assert.Equal("/", response.Headers.Location.OriginalString);
}
Opções do cliente
A tabela a seguir mostra o WebApplicationFactoryClientOptions padrão disponível ao criar instâncias de HttpClient
.
Opção | Descrição | Padrão |
---|---|---|
AllowAutoRedirect | Obtém ou define se as instâncias de HttpClient devem ou não seguir automaticamente as respostas de redirecionamento. |
true |
BaseAddress | Obtém ou define o endereço básico das instâncias de HttpClient . |
http://localhost |
HandleCookies | Obtém ou define se as instâncias de HttpClient devem manipular cookies. |
true |
MaxAutomaticRedirections | Obtém ou define o número máximo de respostas de redirecionamento que as instâncias de HttpClient devem seguir. |
7 |
Crie a classe WebApplicationFactoryClientOptions
e passe-a para o método CreateClient() (os valores padrão são mostrados no exemplo de código):
// Default client option values are shown
var clientOptions = new WebApplicationFactoryClientOptions();
clientOptions.AllowAutoRedirect = true;
clientOptions.BaseAddress = new Uri("http://localhost");
clientOptions.HandleCookies = true;
clientOptions.MaxAutomaticRedirections = 7;
_client = _factory.CreateClient(clientOptions);
Injetar serviços fictícios
Os serviços podem ser substituídos em um teste com uma chamada para ConfigureTestServices no construtor de host. Para injetar serviços fictícios, o SUT deve ter uma classe Startup
com um método Startup.ConfigureServices
.
O SUT de exemplo inclui um serviço com escopo que retorna uma citação. A citação é integrada em um campo oculto na página de Índice quando a página de Índice é solicitada.
Services/IQuoteService.cs
:
public interface IQuoteService
{
Task<string> GenerateQuote();
}
Services/QuoteService.cs
:
// Quote ©1975 BBC: The Doctor (Tom Baker); Dr. Who: Planet of Evil
// https://www.bbc.co.uk/programmes/p00pyrx6
public class QuoteService : IQuoteService
{
public Task<string> GenerateQuote()
{
return Task.FromResult<string>(
"Come on, Sarah. We've an appointment in London, " +
"and we're already 30,000 years late.");
}
}
Startup.cs
:
services.AddScoped<IQuoteService, QuoteService>();
Pages/Index.cshtml.cs
:
public class IndexModel : PageModel
{
private readonly ApplicationDbContext _db;
private readonly IQuoteService _quoteService;
public IndexModel(ApplicationDbContext db, IQuoteService quoteService)
{
_db = db;
_quoteService = quoteService;
}
[BindProperty]
public Message Message { get; set; }
public IList<Message> Messages { get; private set; }
[TempData]
public string MessageAnalysisResult { get; set; }
public string Quote { get; private set; }
public async Task OnGetAsync()
{
Messages = await _db.GetMessagesAsync();
Quote = await _quoteService.GenerateQuote();
}
Pages/Index.cs
:
<input id="quote" type="hidden" value="@Model.Quote">
A marcação a seguir é gerada quando o aplicativo SUT é executado:
<input id="quote" type="hidden" value="Come on, Sarah. We've an appointment in
London, and we're already 30,000 years late.">
Para testar a injeção de serviço e de citação em um teste de integração, um serviço fictício é injetado no SUT pelo teste. O serviço fictício substitui o QuoteService
do aplicativo por um serviço fornecido pelo aplicativo de teste, chamado TestQuoteService
:
IntegrationTests.IndexPageTests.cs
:
// Quote ©1975 BBC: The Doctor (Tom Baker); Pyramids of Mars
// https://www.bbc.co.uk/programmes/p00pys55
public class TestQuoteService : IQuoteService
{
public Task<string> GenerateQuote()
{
return Task.FromResult<string>(
"Something's interfering with time, Mr. Scarman, " +
"and time is my business.");
}
}
ConfigureTestServices
é chamado e o serviço com escopo é registrado:
[Fact]
public async Task Get_QuoteService_ProvidesQuoteInPage()
{
// Arrange
var client = _factory.WithWebHostBuilder(builder =>
{
builder.ConfigureTestServices(services =>
{
services.AddScoped<IQuoteService, TestQuoteService>();
});
})
.CreateClient();
//Act
var defaultPage = await client.GetAsync("/");
var content = await HtmlHelpers.GetDocumentAsync(defaultPage);
var quoteElement = content.QuerySelector("#quote");
// Assert
Assert.Equal("Something's interfering with time, Mr. Scarman, " +
"and time is my business.", quoteElement.Attributes["value"].Value);
}
A marcação produzida durante a execução do teste reflete o texto entre aspas fornecido por TestQuoteService
, portanto, a asserção passa:
<input id="quote" type="hidden" value="Something's interfering with time,
Mr. Scarman, and time is my business.">
Autenticação fictícia
Os testes na classe AuthTests
verificam que um ponto de extremidade seguro:
- Redireciona o usuário não autenticado para a página de login do aplicativo.
- Retorna conteúdo para o usuário autenticado.
No SUT, a página /SecurePage
usa uma convenção AuthorizePage para aplicar um AuthorizeFilter à página. Para obter mais informações, consulte Convenções de autorização do Razor Pages.
services.AddRazorPages(options =>
{
options.Conventions.AuthorizePage("/SecurePage");
});
No teste Get_SecurePageRedirectsAnUnauthenticatedUser
, um WebApplicationFactoryClientOptions é definido para revogar a permissão de redirecionamentos definindo AllowAutoRedirect como false
:
[Fact]
public async Task Get_SecurePageRedirectsAnUnauthenticatedUser()
{
// Arrange
var client = _factory.CreateClient(
new WebApplicationFactoryClientOptions
{
AllowAutoRedirect = false
});
// Act
var response = await client.GetAsync("/SecurePage");
// Assert
Assert.Equal(HttpStatusCode.Redirect, response.StatusCode);
Assert.StartsWith("http://localhost/Identity/Account/Login",
response.Headers.Location.OriginalString);
}
Ao não permitir que o cliente siga o redirecionamento, as seguintes verificações podem ser feitas:
- O código de status retornado pelo SUT pode ser verificado em relação ao resultado esperado HttpStatusCode.Redirect, não ao código de status final após o redirecionamento para a página de login, que seria HttpStatusCode.OK.
- O valor do cabeçalho
Location
nos cabeçalhos de resposta é verificado para confirmar que ele começa comhttp://localhost/Identity/Account/Login
, não a resposta final da página de login, em que o cabeçalhoLocation
não estaria presente.
O aplicativo de teste pode simular um AuthenticationHandler<TOptions> no ConfigureTestServices para testar aspectos de autenticação e autorização. Um cenário mínimo retorna um AuthenticateResult.Success:
public class TestAuthHandler : AuthenticationHandler<AuthenticationSchemeOptions>
{
public TestAuthHandler(IOptionsMonitor<AuthenticationSchemeOptions> options,
ILoggerFactory logger, UrlEncoder encoder, ISystemClock clock)
: base(options, logger, encoder, clock)
{
}
protected override Task<AuthenticateResult> HandleAuthenticateAsync()
{
var claims = new[] { new Claim(ClaimTypes.Name, "Test user") };
var identity = new ClaimsIdentity(claims, "Test");
var principal = new ClaimsPrincipal(identity);
var ticket = new AuthenticationTicket(principal, "Test");
var result = AuthenticateResult.Success(ticket);
return Task.FromResult(result);
}
}
O TestAuthHandler
é chamado para autenticar um usuário quando o esquema de autenticação é definido como Test
onde AddAuthentication
está registrado para ConfigureTestServices
. É importante que o esquema Test
corresponda ao esquema que seu aplicativo espera. Caso contrário, a autenticação não funcionará.
[Fact]
public async Task Get_SecurePageIsReturnedForAnAuthenticatedUser()
{
// Arrange
var client = _factory.WithWebHostBuilder(builder =>
{
builder.ConfigureTestServices(services =>
{
services.AddAuthentication("Test")
.AddScheme<AuthenticationSchemeOptions, TestAuthHandler>(
"Test", options => {});
});
})
.CreateClient(new WebApplicationFactoryClientOptions
{
AllowAutoRedirect = false,
});
client.DefaultRequestHeaders.Authorization =
new AuthenticationHeaderValue("Test");
//Act
var response = await client.GetAsync("/SecurePage");
// Assert
Assert.Equal(HttpStatusCode.OK, response.StatusCode);
}
Para obter mais informações sobre WebApplicationFactoryClientOptions
, confira a seção Opções do cliente.
Definir o ambiente
Por padrão, o ambiente de host e aplicativo do SUT é configurado para usar o ambiente de Desenvolvimento. Para substituir o ambiente do SUT ao usar IHostBuilder
:
- Defina a variável de ambiente
ASPNETCORE_ENVIRONMENT
(por exemplo,Staging
,Production
ou outro valor personalizado, comoTesting
). - Substitua
CreateHostBuilder
no aplicativo de teste para ler variáveis de ambiente prefixadas comASPNETCORE
.
protected override IHostBuilder CreateHostBuilder() =>
base.CreateHostBuilder()
.ConfigureHostConfiguration(
config => config.AddEnvironmentVariables("ASPNETCORE"));
Se o SUT usar o Host da Web (IWebHostBuilder
), substitua CreateWebHostBuilder
:
protected override IWebHostBuilder CreateWebHostBuilder() =>
base.CreateWebHostBuilder().UseEnvironment("Testing");
Como a infraestrutura de teste infere o caminho raiz de conteúdo do aplicativo
O construtor WebApplicationFactory
infere o caminho raiz de conteúdo do aplicativo pesquisando um WebApplicationFactoryContentRootAttribute no assembly que contém os testes de integração com uma chave igual ao assembly TEntryPoint
System.Reflection.Assembly.FullName
. Caso não seja encontrado um atributo com a chave correta, WebApplicationFactory
volta a procurar um arquivo de solução (.sln) e acrescente o nome do assembly TEntryPoint
ao diretório da solução. O diretório raiz do aplicativo (o caminho raiz do conteúdo) é usado para descobrir exibições e arquivos de conteúdo.
Desabilitar a cópia de sombra
A cópia de sombra faz com que os testes sejam executados em um diretório diferente do diretório de saída. Se seus testes dependerem do carregamento de arquivos relativos a Assembly.Location
e você encontrar problemas, talvez seja necessário desabilitar a cópia de sombra.
Para desabilitar a cópia de sombra ao usar xUnit, crie um arquivo xunit.runner.json
no diretório do projeto de teste, com a configuração correta:
{
"shadowCopy": false
}
Descarte de objetos
Depois que os testes da implementação de IClassFixture
são executados, TestServer e HttpClient são descartados quando xUnit descarta o WebApplicationFactory
. Se os objetos instanciados pelo desenvolvedor exigirem descarte, descarte-os na IClassFixture
implementação. Para saber mais, confira Implementação de um método Dispose.
Exemplo de testes de integração
O aplicativo de exemplo é composto por dois aplicativos:
Aplicativo | Diretório do projeto | Descrição |
---|---|---|
Aplicativo de mensagens (o SUT) | src/RazorPagesProject |
Permite que o usuário adicione, exclua uma ou todas as mensagens e as analise. |
Aplicativo de teste | tests/RazorPagesProject.Tests |
Usado para fazer o teste de integração do SUT. |
Os testes podem ser executados usando os recursos de teste nativos do IDE, como o Visual Studio. Se estiver usando o Visual Studio Code ou a linha de comando, execute o seguinte comando em um prompt de comando na pasta tests/RazorPagesProject.Tests
:
dotnet test
Organização do aplicativo de mensagens (SUT)
O SUT é um sistema de mensagens do Razor Pages com as seguintes características:
- A página de Índice do aplicativo (
Pages/Index.cshtml
ePages/Index.cshtml.cs
) fornece uma interface do usuário e métodos de modelo de página para controlar a adição, exclusão e análise de mensagens (número médio de palavras por mensagem). - Uma mensagem é descrita pela classe
Message
(Data/Message.cs
) com duas propriedades:Id
(chave) eText
(mensagem). A propriedadeText
é necessária e limitada a 200 caracteres. - As mensagens são armazenadas usando o banco de dados na memória do Entity Framework†.
- O aplicativo contém uma DAL (camada de acesso a dados) em sua classe de contexto de banco de dados,
AppDbContext
(Data/AppDbContext.cs
). - Se o banco de dados estiver vazio na inicialização do aplicativo, o repositório de mensagens será inicializado com três mensagens.
- O aplicativo inclui um
/SecurePage
que só pode ser acessado por um usuário autenticado.
†O tópico do EF, Teste com InMemory, explica como usar um banco de dados na memória para testes com MSTest. Este tópico usa a estrutura de teste xUnit. Os conceitos de teste e as implementações de teste em diferentes estruturas de teste são semelhantes, mas não idênticos.
Embora o aplicativo não use o padrão do repositório e não seja um exemplo eficaz do padrão UoW (Unidade de Trabalho), o Razor Pages dá suporte a esses padrões de desenvolvimento. Para obter mais informações, confira Criando a camada de persistência da infraestrutura e Lógica do controlador de teste (o exemplo implementa o padrão do repositório).
Organização do aplicativo de teste
O aplicativo de teste é um aplicativo de console dentro do diretório tests/RazorPagesProject.Tests
.
Diretório do aplicativo de teste | Descrição |
---|---|
AuthTests |
Contém métodos de teste para:
|
BasicTests |
Contém um método de teste para roteamento e tipo de conteúdo. |
IntegrationTests |
Contém os testes de integração para a página de Índice usando a classe personalizada WebApplicationFactory . |
Helpers/Utilities |
|
A estrutura de teste é xUnit. Os testes de integração são realizados usando o Microsoft.AspNetCore.TestHost, que inclui o TestServer. Como o pacote Microsoft.AspNetCore.Mvc.Testing
é usado para configurar o host de teste e o servidor de teste, os pacotes TestHost
e TestServer
não exigem referências diretas do pacote no arquivo de projeto do aplicativo de teste ou na configuração do desenvolvedor no aplicativo de teste.
Os testes de integração geralmente exigem um pequeno conjunto de dados no banco de dados antes da execução do teste. Por exemplo, um teste de exclusão exige a exclusão de um registro do banco de dados; portanto, o banco de dados deve ter pelo menos um registro para que a solicitação de exclusão tenha êxito.
O aplicativo de exemplo propaga o banco de dados com três mensagens em Utilities.cs
que os testes podem usar quando são executados:
public static void InitializeDbForTests(ApplicationDbContext db)
{
db.Messages.AddRange(GetSeedingMessages());
db.SaveChanges();
}
public static void ReinitializeDbForTests(ApplicationDbContext db)
{
db.Messages.RemoveRange(db.Messages);
InitializeDbForTests(db);
}
public static List<Message> GetSeedingMessages()
{
return new List<Message>()
{
new Message(){ Text = "TEST RECORD: You're standing on my scarf." },
new Message(){ Text = "TEST RECORD: Would you like a jelly baby?" },
new Message(){ Text = "TEST RECORD: To the rational mind, " +
"nothing is inexplicable; only unexplained." }
};
}
O contexto de banco de dados do SUT é registrado em seu método Startup.ConfigureServices
. O retorno de chamada builder.ConfigureServices
do aplicativo de teste é executado depois que o código Startup.ConfigureServices
do aplicativo é executado. Para usar um banco de dados diferente para os testes, o contexto do banco de dados do aplicativo deve ser substituído em builder.ConfigureServices
. Para obter mais informações, consulte a seção Personalizar WebApplicationFactory .
Nos SUTs que ainda usam o Host Da Web, o retorno de chamada builder.ConfigureServices
do aplicativo de teste é executado antes do código Startup.ConfigureServices
do SUT. O retorno de chamada builder.ConfigureTestServices
do aplicativo de teste é executado após.
Recursos adicionais
Este artigo pressupõe uma compreensão básica sobre testes de unidade. Se não estiver familiarizado com os conceitos de teste, consulte o artigo Teste de Unidade no .NET Core e no .NET Standard e seu conteúdo vinculado.
Exibir ou baixar código de exemplo (como baixar)
O aplicativo de exemplo é um Razor aplicativo Pages e pressupõe uma compreensão básica do Razor Pages. Se você não estiver familiarizado com o Razor Pages, consulte os seguintes artigos:
Para testar SPAs, recomendamos uma ferramenta como o Playwright para .NET, que pode automatizar um navegador.
Introdução aos testes de integração
Os testes de integração avaliam os componentes de um aplicativo em um nível mais amplo do que os testes de unidade. Os testes de unidade são usados para testar componentes de software isolados, como métodos de classe individuais. Os testes de integração confirmam que dois ou mais componentes de aplicativo trabalham juntos para produzir um resultado esperado, possivelmente incluindo todos os componentes necessários para processar totalmente uma solicitação.
Esses testes mais amplos são usados para testar a infraestrutura do aplicativo e toda a estrutura, geralmente incluindo os seguintes componentes:
- Banco de dados
- Sistema de arquivos
- Dispositivos de rede
- Pipeline de solicitação-resposta
Os testes de unidade usam componentes fabricados, conhecidos como fakes ou objetos fictícios, no lugar de componentes de infraestrutura.
Ao contrário dos testes de unidade, os testes de integração:
- Usam os componentes reais que o aplicativo usa em produção.
- Exigem mais código e processamento de dados.
- Demoraram mais para serem executados.
Portanto, limite o uso de testes de integração aos cenários de infraestrutura mais importantes. Se um comportamento puder ser testado usando um teste de unidade ou um teste de integração, escolha o teste de unidade.
Em discussões sobre testes de integração, o projeto testado é frequentemente chamado de System Under Test ou "SUT" para abreviar. "SUT" é usado ao longo deste artigo para se referir ao aplicativo ASP.NET Core que está sendo testado.
Não escreva testes de integração para cada permutação de dados e acesso a arquivos com bancos de dados e sistemas de arquivos. Independentemente de quantos lugares em um aplicativo interagem com bancos de dados e sistemas de arquivos, um conjunto focado de testes de integração de leitura, gravação, atualização e exclusão geralmente é capaz de testar adequadamente os componentes do banco de dados e do sistema de arquivos. Use testes de unidade para testes de rotina da lógica do método que interage com esses componentes. Em testes de unidade, o uso de infraestruturas falsas ou fictícias resulta em uma execução de teste mais rápida.
Testes de integração de ASP.NET Core
Os testes de integração no ASP.NET Core exigem o seguinte:
- Um projeto de teste é usado para conter e executar os testes. O projeto de teste tem uma referência ao SUT.
- O projeto de teste cria um host Web de teste para o SUT e usa um cliente do servidor de teste para lidar com solicitações e respostas com o SUT.
- Um executor de teste é usado para executar os testes e relatar os resultados.
Os testes de integração seguem uma sequência de eventos que incluem as etapas de teste Organizar, Atuar e Afirmar:
- O host da Web do SUT está configurado.
- Um cliente do servidor de teste é criado para enviar solicitações ao aplicativo.
- A etapa de teste Organizar é executada: o aplicativo de teste prepara uma solicitação.
- A etapa de teste Atuar é executada: o cliente envia a solicitação e recebe a resposta.
- A etapa de teste Afirmar é executada: a resposta real é validada como uma aprovada ou reprovada com base em uma resposta esperada.
- O processo continua até que todos os testes sejam executados.
- Os resultados do teste são reportados.
Normalmente, o host da Web de teste é configurado de forma diferente do host da Web normal do aplicativo para as execuções de teste. Por exemplo, um banco de dados diferente ou configurações de aplicativo diferentes podem ser usadas para os testes.
Os componentes de infraestrutura, como o host da Web de teste e o servidor de teste na memória (TestServer), são fornecidos ou gerenciados pelo pacote Microsoft.AspNetCore.Mvc.Testing. O uso desse pacote simplifica a criação e a execução do teste.
O pacote Microsoft.AspNetCore.Mvc.Testing
manipula as seguintes tarefas:
- Copia o arquivo de dependências (
.deps
) do SUT para o diretóriobin
do projeto de teste. - Define a raiz de conteúdo para a raiz do projeto do SUT para que arquivos estáticos e páginas/exibições sejam encontradas quando os testes forem executados.
- Fornece a classe WebApplicationFactory para simplificar a inicialização do aplicativo testado com
TestServer
.
A documentação de testes de unidade descreve como configurar um projeto de teste e um executor de teste, juntamente com instruções detalhadas sobre como executar testes e recomendações sobre como nomear testes e classes de teste.
Separe os testes de unidade dos testes de integração em projetos diferentes. Separando os testes:
- Ajuda a garantir que os componentes de teste de infraestrutura não sejam incluídos acidentalmente nos testes de unidade.
- Permite o controle sobre quais conjuntos de testes são executados.
Praticamente não há diferença entre a configuração de testes de aplicativos do Razor Pages e aplicativos MVC. A única diferença é como os testes são nomeados. Em um aplicativo Razor Pages, os testes de pontos de extremidade de página geralmente recebem o mesmo nome da classe de modelo de página (por exemplo, IndexPageTests
para testar a integração de componentes para a página Índice). Em um aplicativo MVC, os testes geralmente são organizados por classes de controlador e recebem o mesmo nome dos controladores que testam (por exemplo, HomeControllerTests
para testar a integração de componentes para o controlador Home).
Pré-requisitos do aplicativo de teste
O projeto de teste deve:
- Referência ao pacote
Microsoft.AspNetCore.Mvc.Testing
. - Especifique o SDK da Web no arquivo de projeto (
<Project Sdk="Microsoft.NET.Sdk.Web">
).
Esses pré-requisitos podem ser vistos no aplicativo de exemplo. Inspecionar o tests/RazorPagesProject.Tests/RazorPagesProject.Tests.csproj
arquivo. O aplicativo de exemplo usa a estrutura de teste xUnit e a biblioteca do analisador AngleSharp, portanto, o aplicativo de exemplo também faz referência a:
Em aplicativos que usam xunit.runner.visualstudio
na versão 2.4.2 ou posterior, o projeto de teste deve referenciar o pacote Microsoft.NET.Test.Sdk
.
O Entity Framework Core também é usado nos testes. Consulte o arquivo de projeto no GitHub.
Ambiente do SUT
Se o ambiente do SUT não estiver definido, o ambiente usará como padrão o Desenvolvimento.
Testes básicos com o WebApplicationFactory padrão
Exponha a classe definida Program
implicitamente ao projeto de teste realizando um dos seguintes procedimentos:
Expor tipos internos do aplicativo Web ao projeto de teste. Isso pode ser feito no arquivo do projeto SUT (
.csproj
):<ItemGroup> <InternalsVisibleTo Include="MyTestProject" /> </ItemGroup>
Torne a
Program
classe pública usando uma declaração de classe parcial:var builder = WebApplication.CreateBuilder(args); // ... Configure services, routes, etc. app.Run(); + public partial class Program { }
O aplicativo de exemplo usa a abordagem de classe parcial
Program
.
WebApplicationFactory<TEntryPoint> é usado para criar um TestServer para os testes de integração. TEntryPoint
é a classe de ponto de entrada do SUT, geralmente Program.cs
.
As classes de teste implementam uma interface de acessório de classe (IClassFixture
) para indicar que a classe contém testes e fornece instâncias de objeto compartilhadas entre os testes na classe .
A classe de teste a seguir, BasicTests
, usa o WebApplicationFactory
para iniciar o SUT e fornecer um HttpClient para um método de teste, Get_EndpointsReturnSuccessAndCorrectContentType
. O método verifica se o código de status da resposta foi bem-sucedido (200-299) e o cabeçalho Content-Type
é text/html; charset=utf-8
em várias páginas do aplicativo.
CreateClient() cria uma instância do HttpClient
que segue automaticamente redirecionamentos e manipula cookies.
public class BasicTests
: IClassFixture<WebApplicationFactory<Program>>
{
private readonly WebApplicationFactory<Program> _factory;
public BasicTests(WebApplicationFactory<Program> factory)
{
_factory = factory;
}
[Theory]
[InlineData("/")]
[InlineData("/Index")]
[InlineData("/About")]
[InlineData("/Privacy")]
[InlineData("/Contact")]
public async Task Get_EndpointsReturnSuccessAndCorrectContentType(string url)
{
// Arrange
var client = _factory.CreateClient();
// Act
var response = await client.GetAsync(url);
// Assert
response.EnsureSuccessStatusCode(); // Status Code 200-299
Assert.Equal("text/html; charset=utf-8",
response.Content.Headers.ContentType.ToString());
}
}
Por padrão, os cookies não essenciais não são preservados entre solicitações quando a política de consentimento do Regulamento Geral sobre a Proteção de Dados está habilitada. Para preservar cookies não essenciais, como aqueles usados pelo provedor TempData, marque-os como essenciais em seus testes. Para obter instruções sobre como marcar um cookie como essencial, consulte Cookies essenciais.
Comparação entre AngleSharp e Application Parts
a respeito de verificações de proteção contra falsificações
Este artigo usa o analisador AngleSharp para manipular as verificações de proteção contra falsificações carregando páginas e analisando o HTML. Para testar os pontos de extremidade do controlador e as visualizações do Razor Pages em um nível inferior, sem se preocupar com a forma como eles são renderizados no navegador, considere o uso de Application Parts
. A abordagem Partes do Aplicativo injeta um controlador ou Razor page no aplicativo que pode ser usado para fazer solicitações JSON para obter os valores necessários. Para obter mais informações, consulte o blog Recursos de Teste de Integração do ASP.NET Core Protegido contra Falsificações Usando Partes do Aplicativo e o repositório GitHub associado por Martin Costello.
Personalizar WebApplicationFactory
A configuração de host da Web pode ser criada independentemente das classes de teste herdando de WebApplicationFactory<TEntryPoint> para criar uma ou mais fábricas personalizadas:
Herdar do
WebApplicationFactory
e substituir ConfigureWebHost. O IWebHostBuilder permite a configuração da coleção de serviços comIWebHostBuilder.ConfigureServices
public class CustomWebApplicationFactory<TProgram> : WebApplicationFactory<TProgram> where TProgram : class { protected override void ConfigureWebHost(IWebHostBuilder builder) { builder.ConfigureServices(services => { var dbContextDescriptor = services.SingleOrDefault( d => d.ServiceType == typeof(DbContextOptions<ApplicationDbContext>)); services.Remove(dbContextDescriptor); var dbConnectionDescriptor = services.SingleOrDefault( d => d.ServiceType == typeof(DbConnection)); services.Remove(dbConnectionDescriptor); // Create open SqliteConnection so EF won't automatically close it. services.AddSingleton<DbConnection>(container => { var connection = new SqliteConnection("DataSource=:memory:"); connection.Open(); return connection; }); services.AddDbContext<ApplicationDbContext>((container, options) => { var connection = container.GetRequiredService<DbConnection>(); options.UseSqlite(connection); }); }); builder.UseEnvironment("Development"); } }
A propagação de banco de dados no aplicativo de exemplo é executada pelo método
InitializeDbForTests
. O método é descrito na seção Exemplo de testes de integração: testar a organização do aplicativo .O contexto de banco de dados do SUT é registrado em
Program.cs
. O retorno de chamadabuilder.ConfigureServices
do aplicativo de teste é executado depois que o códigoProgram.cs
do aplicativo é executado. Para usar um banco de dados diferente para os testes além do banco de dados do aplicativo, o contexto do banco de dados do aplicativo deve ser substituído embuilder.ConfigureServices
.O aplicativo de exemplo localiza o descritor de serviço para o contexto do banco de dados e usa o descritor para remover o registro do serviço. Em seguida, a fábrica adiciona um novo
ApplicationDbContext
que usa um banco de dados na memória para os testes.Para se conectar a um banco de dados diferente, altere o
DbConnection
. Para usar um banco de dados de teste do SQL Server:
- Faça referência ao pacote NuGet
Microsoft.EntityFrameworkCore.SqlServer
do arquivo de projeto. - Chame
UseInMemoryDatabase
.
Use o
CustomWebApplicationFactory
personalizado em classes de teste. O exemplo a seguir usa a fábrica na classeIndexPageTests
:public class IndexPageTests : IClassFixture<CustomWebApplicationFactory<Program>> { private readonly HttpClient _client; private readonly CustomWebApplicationFactory<Program> _factory; public IndexPageTests( CustomWebApplicationFactory<Program> factory) { _factory = factory; _client = factory.CreateClient(new WebApplicationFactoryClientOptions { AllowAutoRedirect = false }); }
O cliente do aplicativo de exemplo está configurado para evitar que
HttpClient
siga redirecionamentos. Conforme explicado posteriormente na seção Autenticação fictícia, isso permite que os testes verifiquem o resultado da primeira resposta do aplicativo. A primeira resposta é um redirecionamento em muitos desses testes com um cabeçalhoLocation
.Um teste típico usa
HttpClient
e métodos auxiliares para processar a solicitação e a resposta:[Fact] public async Task Post_DeleteAllMessagesHandler_ReturnsRedirectToRoot() { // Arrange var defaultPage = await _client.GetAsync("/"); var content = await HtmlHelpers.GetDocumentAsync(defaultPage); //Act var response = await _client.SendAsync( (IHtmlFormElement)content.QuerySelector("form[id='messages']"), (IHtmlButtonElement)content.QuerySelector("button[id='deleteAllBtn']")); // Assert Assert.Equal(HttpStatusCode.OK, defaultPage.StatusCode); Assert.Equal(HttpStatusCode.Redirect, response.StatusCode); Assert.Equal("/", response.Headers.Location.OriginalString); }
Qualquer solicitação POST para o SUT deve atender a verificação contra falsificações feita automaticamente pelo sistema de proteção de dados contra falsificações do aplicativo. Para organizar a solicitação POST de um teste, o aplicativo de teste deve:
- Fazer uma solicitação para a página.
- Analisar o cookie de validação de solicitação e antifalsificação da resposta.
- Faça a solicitação POST com o token de validação de solicitação e antifalsificação cookie em vigor.
Os métodos de extensão auxiliar SendAsync
(Helpers/HttpClientExtensions.cs
) e o método auxiliar GetDocumentAsync
(Helpers/HtmlHelpers.cs
) no aplicativo de exemplo usam o analisador AngleSharp para lidar com a verificação antifalsificação usando os seguintes métodos:
GetDocumentAsync
: recebe o HttpResponseMessage e retorna umIHtmlDocument
.GetDocumentAsync
usa uma fábrica que prepara uma resposta virtual com base noHttpResponseMessage
original. Para obter mais informações, consulte a documentação do AngleSharp.SendAsync
métodos de extensão para a composiçãoHttpClient
de um HttpRequestMessage e fazer a chamada de SendAsync(HttpRequestMessage) e para enviar solicitações para o SUT. As sobrecargas paraSendAsync
aceitam o formulário HTML (IHtmlFormElement
) e os seguintes itens:- Botão Enviar do formulário (
IHtmlElement
) - Coleção de valores do formulário (
IEnumerable<KeyValuePair<string, string>>
) - Botão Enviar (
IHtmlElement
) e valores do formulário (IEnumerable<KeyValuePair<string, string>>
)
- Botão Enviar do formulário (
O AngleSharp é uma biblioteca de análise de terceiros usada para fins de demonstração neste artigo e no aplicativo de exemplo. O AngleSharp não tem suporte nem é necessário para testes de integração de aplicativos ASP.NET Core. Outros analisadores podem ser usados, como o HAP (Pacote de Agilidade Html). Outra abordagem é escrever código para lidar diretamente com o token de verificação de solicitação do sistema antifalsificação e com o cookie antifalsificação diretamente. Consulte AngleSharp vs Application Parts
para verificações antifalsificação neste artigo para obter mais informações.
O provedor de banco de dados na memória do EF-Core pode ser usado para testes limitados e básicos; no entanto, o provedor SQLite é a opção recomendada para testes na memória.
Consulte Estender a inicialização com filtros de inicialização que mostra como configurar o middleware usando IStartupFilter, o que é útil quando um teste requer um serviço ou middleware personalizado.
Personalizar o cliente com WithWebHostBuilder
Quando uma configuração adicional é necessária em um método de teste, WithWebHostBuilder cria um novo WebApplicationFactory
com um IWebHostBuilder que é personalizado ainda mais pela configuração.
O código de exemplo chama WithWebHostBuilder
para substituir serviços configurados com stubs de teste. Para obter mais informações e exemplos de uso, consulte Injetar serviços fictícios neste artigo.
O método de teste Post_DeleteMessageHandler_ReturnsRedirectToRoot
do aplicativo de exemplo demonstra o uso de WithWebHostBuilder
. Esse teste executa uma exclusão de registro no banco de dados disparando um envio de formulário no SUT.
Como outro teste na classe IndexPageTests
executa uma operação que exclui todos os registros no banco de dados e pode ser executada antes do método Post_DeleteMessageHandler_ReturnsRedirectToRoot
, o banco de dados é propagado novamente neste método de teste para garantir que o registro esteja presente para o SUT excluir. A seleção do primeiro botão excluir do formulário messages
do SUT é simulada na solicitação para o SUT:
[Fact]
public async Task Post_DeleteMessageHandler_ReturnsRedirectToRoot()
{
// Arrange
using (var scope = _factory.Services.CreateScope())
{
var scopedServices = scope.ServiceProvider;
var db = scopedServices.GetRequiredService<ApplicationDbContext>();
Utilities.ReinitializeDbForTests(db);
}
var defaultPage = await _client.GetAsync("/");
var content = await HtmlHelpers.GetDocumentAsync(defaultPage);
//Act
var response = await _client.SendAsync(
(IHtmlFormElement)content.QuerySelector("form[id='messages']"),
(IHtmlButtonElement)content.QuerySelector("form[id='messages']")
.QuerySelector("div[class='panel-body']")
.QuerySelector("button"));
// Assert
Assert.Equal(HttpStatusCode.OK, defaultPage.StatusCode);
Assert.Equal(HttpStatusCode.Redirect, response.StatusCode);
Assert.Equal("/", response.Headers.Location.OriginalString);
}
Opções do cliente
Consulte a página WebApplicationFactoryClientOptions para ver as definições padrão e as opções disponíveis ao criar instâncias HttpClient
.
Crie a classe WebApplicationFactoryClientOptions
e passe-a para o método CreateClient():
public class IndexPageTests :
IClassFixture<CustomWebApplicationFactory<Program>>
{
private readonly HttpClient _client;
private readonly CustomWebApplicationFactory<Program>
_factory;
public IndexPageTests(
CustomWebApplicationFactory<Program> factory)
{
_factory = factory;
_client = factory.CreateClient(new WebApplicationFactoryClientOptions
{
AllowAutoRedirect = false
});
}
NOTA: Para evitar avisos de redirecionamento HTTPS em logs ao usar o middleware de redirecionamento HTTPS, defina BaseAddress = new Uri("https://localhost")
Injetar serviços fictícios
Os serviços podem ser substituídos em um teste com uma chamada para ConfigureTestServices no construtor de host. Para definir o escopo dos serviços substituídos para o próprio teste, o método WithWebHostBuilder é usado para recuperar um construtor de host. Isso pode ser visto nos seguintes testes:
- Get_QuoteService_ProvidesQuoteInPage
- Get_GithubProfilePageCanGetAGithubUser
- Get_SecurePageIsReturnedForAnAuthenticatedUser
O SUT de exemplo inclui um serviço com escopo que retorna uma citação. A citação é integrada em um campo oculto na página de Índice quando a página de Índice é solicitada.
Services/IQuoteService.cs
:
public interface IQuoteService
{
Task<string> GenerateQuote();
}
Services/QuoteService.cs
:
// Quote ©1975 BBC: The Doctor (Tom Baker); Dr. Who: Planet of Evil
// https://www.bbc.co.uk/programmes/p00pyrx6
public class QuoteService : IQuoteService
{
public Task<string> GenerateQuote()
{
return Task.FromResult<string>(
"Come on, Sarah. We've an appointment in London, " +
"and we're already 30,000 years late.");
}
}
Program.cs
:
services.AddScoped<IQuoteService, QuoteService>();
Pages/Index.cshtml.cs
:
public class IndexModel : PageModel
{
private readonly ApplicationDbContext _db;
private readonly IQuoteService _quoteService;
public IndexModel(ApplicationDbContext db, IQuoteService quoteService)
{
_db = db;
_quoteService = quoteService;
}
[BindProperty]
public Message Message { get; set; }
public IList<Message> Messages { get; private set; }
[TempData]
public string MessageAnalysisResult { get; set; }
public string Quote { get; private set; }
public async Task OnGetAsync()
{
Messages = await _db.GetMessagesAsync();
Quote = await _quoteService.GenerateQuote();
}
Pages/Index.cs
:
<input id="quote" type="hidden" value="@Model.Quote">
A marcação a seguir é gerada quando o aplicativo SUT é executado:
<input id="quote" type="hidden" value="Come on, Sarah. We've an appointment in
London, and we're already 30,000 years late.">
Para testar a injeção de serviço e de citação em um teste de integração, um serviço fictício é injetado no SUT pelo teste. O serviço fictício substitui o QuoteService
do aplicativo por um serviço fornecido pelo aplicativo de teste, chamado TestQuoteService
:
IntegrationTests.IndexPageTests.cs
:
// Quote ©1975 BBC: The Doctor (Tom Baker); Pyramids of Mars
// https://www.bbc.co.uk/programmes/p00pys55
public class TestQuoteService : IQuoteService
{
public Task<string> GenerateQuote()
{
return Task.FromResult(
"Something's interfering with time, Mr. Scarman, " +
"and time is my business.");
}
}
ConfigureTestServices
é chamado e o serviço com escopo é registrado:
[Fact]
public async Task Get_QuoteService_ProvidesQuoteInPage()
{
// Arrange
var client = _factory.WithWebHostBuilder(builder =>
{
builder.ConfigureTestServices(services =>
{
services.AddScoped<IQuoteService, TestQuoteService>();
});
})
.CreateClient();
//Act
var defaultPage = await client.GetAsync("/");
var content = await HtmlHelpers.GetDocumentAsync(defaultPage);
var quoteElement = content.QuerySelector("#quote");
// Assert
Assert.Equal("Something's interfering with time, Mr. Scarman, " +
"and time is my business.", quoteElement.Attributes["value"].Value);
}
A marcação produzida durante a execução do teste reflete o texto entre aspas fornecido por TestQuoteService
, portanto, a asserção passa:
<input id="quote" type="hidden" value="Something's interfering with time,
Mr. Scarman, and time is my business.">
Autenticação fictícia
Os testes na classe AuthTests
verificam que um ponto de extremidade seguro:
- Redireciona o usuário não autenticado para a página de entrada do aplicativo.
- Retorna conteúdo para o usuário autenticado.
No SUT, a página /SecurePage
usa uma convenção AuthorizePage para aplicar um AuthorizeFilter à página. Para obter mais informações, consulte Convenções de autorização do Razor Pages.
services.AddRazorPages(options =>
{
options.Conventions.AuthorizePage("/SecurePage");
});
No teste Get_SecurePageRedirectsAnUnauthenticatedUser
, um WebApplicationFactoryClientOptions é definido para revogar a permissão de redirecionamentos definindo AllowAutoRedirect como false
:
[Fact]
public async Task Get_SecurePageRedirectsAnUnauthenticatedUser()
{
// Arrange
var client = _factory.CreateClient(
new WebApplicationFactoryClientOptions
{
AllowAutoRedirect = false
});
// Act
var response = await client.GetAsync("/SecurePage");
// Assert
Assert.Equal(HttpStatusCode.Redirect, response.StatusCode);
Assert.StartsWith("http://localhost/Identity/Account/Login",
response.Headers.Location.OriginalString);
}
Ao não permitir que o cliente siga o redirecionamento, as seguintes verificações podem ser feitas:
- O código de status retornado pelo SUT pode ser verificado em relação ao resultado esperado HttpStatusCode.Redirect, não o código de status final após o redirecionamento para a página de entrada, que seria HttpStatusCode.OK.
- O valor do cabeçalho
Location
nos cabeçalhos de resposta é verificado para confirmar que ele começa comhttp://localhost/Identity/Account/Login
, não a resposta final da página de entrada, em que o cabeçalhoLocation
não estaria presente.
O aplicativo de teste pode simular um AuthenticationHandler<TOptions> no ConfigureTestServices para testar aspectos de autenticação e autorização. Um cenário mínimo retorna um AuthenticateResult.Success:
public class TestAuthHandler : AuthenticationHandler<AuthenticationSchemeOptions>
{
public TestAuthHandler(IOptionsMonitor<AuthenticationSchemeOptions> options,
ILoggerFactory logger, UrlEncoder encoder, ISystemClock clock)
: base(options, logger, encoder, clock)
{
}
protected override Task<AuthenticateResult> HandleAuthenticateAsync()
{
var claims = new[] { new Claim(ClaimTypes.Name, "Test user") };
var identity = new ClaimsIdentity(claims, "Test");
var principal = new ClaimsPrincipal(identity);
var ticket = new AuthenticationTicket(principal, "TestScheme");
var result = AuthenticateResult.Success(ticket);
return Task.FromResult(result);
}
}
O TestAuthHandler
é chamado para autenticar um usuário quando o esquema de autenticação é definido como TestScheme
onde AddAuthentication
está registrado para ConfigureTestServices
. É importante que o esquema TestScheme
corresponda ao esquema que seu aplicativo espera. Caso contrário, a autenticação não funcionará.
[Fact]
public async Task Get_SecurePageIsReturnedForAnAuthenticatedUser()
{
// Arrange
var client = _factory.WithWebHostBuilder(builder =>
{
builder.ConfigureTestServices(services =>
{
services.AddAuthentication(defaultScheme: "TestScheme")
.AddScheme<AuthenticationSchemeOptions, TestAuthHandler>(
"TestScheme", options => { });
});
})
.CreateClient(new WebApplicationFactoryClientOptions
{
AllowAutoRedirect = false,
});
client.DefaultRequestHeaders.Authorization =
new AuthenticationHeaderValue(scheme: "TestScheme");
//Act
var response = await client.GetAsync("/SecurePage");
// Assert
Assert.Equal(HttpStatusCode.OK, response.StatusCode);
}
Para obter mais informações sobre WebApplicationFactoryClientOptions
, confira a seção Opções do cliente.
Testes básicos para middlewares de autenticação
Consulte este repositório do GitHub para obter testes básicos de middlewares de autenticação. Ele contém um servidor de teste específico para o cenário de teste.
Definir o ambiente
Defina o ambiente na fábrica de aplicativos personalizada:
public class CustomWebApplicationFactory<TProgram>
: WebApplicationFactory<TProgram> where TProgram : class
{
protected override void ConfigureWebHost(IWebHostBuilder builder)
{
builder.ConfigureServices(services =>
{
var dbContextDescriptor = services.SingleOrDefault(
d => d.ServiceType ==
typeof(DbContextOptions<ApplicationDbContext>));
services.Remove(dbContextDescriptor);
var dbConnectionDescriptor = services.SingleOrDefault(
d => d.ServiceType ==
typeof(DbConnection));
services.Remove(dbConnectionDescriptor);
// Create open SqliteConnection so EF won't automatically close it.
services.AddSingleton<DbConnection>(container =>
{
var connection = new SqliteConnection("DataSource=:memory:");
connection.Open();
return connection;
});
services.AddDbContext<ApplicationDbContext>((container, options) =>
{
var connection = container.GetRequiredService<DbConnection>();
options.UseSqlite(connection);
});
});
builder.UseEnvironment("Development");
}
}
Como a infraestrutura de teste infere o caminho raiz de conteúdo do aplicativo
O construtor WebApplicationFactory
infere o caminho raiz de conteúdo do aplicativo pesquisando um WebApplicationFactoryContentRootAttribute no assembly que contém os testes de integração com uma chave igual ao assembly TEntryPoint
System.Reflection.Assembly.FullName
. Caso não seja encontrado um atributo com a chave correta, WebApplicationFactory
volta a procurar um arquivo de solução (.sln) e acrescente o nome do assembly TEntryPoint
ao diretório da solução. O diretório raiz do aplicativo (o caminho raiz do conteúdo) é usado para descobrir exibições e arquivos de conteúdo.
Desabilitar a cópia de sombra
A cópia de sombra faz com que os testes sejam executados em um diretório diferente do diretório de saída. Se seus testes dependerem do carregamento de arquivos relativos a Assembly.Location
e você encontrar problemas, talvez seja necessário desabilitar a cópia de sombra.
Para desabilitar a cópia de sombra ao usar xUnit, crie um arquivo xunit.runner.json
no diretório do projeto de teste, com a configuração correta:
{
"shadowCopy": false
}
Descarte de objetos
Depois que os testes da implementação de IClassFixture
são executados, TestServer e HttpClient são descartados quando xUnit descarta o WebApplicationFactory
. Se os objetos instanciados pelo desenvolvedor exigirem descarte, descarte-os na IClassFixture
implementação. Para saber mais, confira Implementação de um método Dispose.
Exemplo de testes de integração
O aplicativo de exemplo é composto por dois aplicativos:
Aplicativo | Diretório do projeto | Descrição |
---|---|---|
Aplicativo de mensagens (o SUT) | src/RazorPagesProject |
Permite que o usuário adicione, exclua uma ou todas as mensagens e as analise. |
Aplicativo de teste | tests/RazorPagesProject.Tests |
Usado para fazer o teste de integração do SUT. |
Os testes podem ser executados usando os recursos de teste nativos do IDE, como o Visual Studio. Se estiver usando o Visual Studio Code ou a linha de comando, execute o seguinte comando em um prompt de comando na pasta tests/RazorPagesProject.Tests
:
dotnet test
Organização do aplicativo de mensagens (SUT)
O SUT é um sistema de mensagens do Razor Pages com as seguintes características:
- A página de Índice do aplicativo (
Pages/Index.cshtml
ePages/Index.cshtml.cs
) fornece uma interface do usuário e métodos de modelo de página para controlar a adição, exclusão e análise de mensagens (número médio de palavras por mensagem). - Uma mensagem é descrita pela classe
Message
(Data/Message.cs
) com duas propriedades:Id
(chave) eText
(mensagem). A propriedadeText
é necessária e limitada a 200 caracteres. - As mensagens são armazenadas usando o banco de dados na memória do Entity Framework†.
- O aplicativo contém uma DAL (camada de acesso a dados) em sua classe de contexto de banco de dados,
AppDbContext
(Data/AppDbContext.cs
). - Se o banco de dados estiver vazio na inicialização do aplicativo, o repositório de mensagens será inicializado com três mensagens.
- O aplicativo inclui um
/SecurePage
que só pode ser acessado por um usuário autenticado.
†O tópico do EF, Teste com InMemory, explica como usar um banco de dados na memória para testes com MSTest. Este tópico usa a estrutura de teste xUnit. Os conceitos de teste e as implementações de teste em diferentes estruturas de teste são semelhantes, mas não idênticos.
Embora o aplicativo não use o padrão do repositório e não seja um exemplo eficaz do padrão UoW (Unidade de Trabalho), o Razor Pages dá suporte a esses padrões de desenvolvimento. Para obter mais informações, confira Criando a camada de persistência da infraestrutura e Lógica do controlador de teste (o exemplo implementa o padrão do repositório).
Organização do aplicativo de teste
O aplicativo de teste é um aplicativo de console dentro do diretório tests/RazorPagesProject.Tests
.
Diretório do aplicativo de teste | Descrição |
---|---|
AuthTests |
Contém métodos de teste para:
|
BasicTests |
Contém um método de teste para roteamento e tipo de conteúdo. |
IntegrationTests |
Contém os testes de integração para a página de Índice usando a classe personalizada WebApplicationFactory . |
Helpers/Utilities |
|
A estrutura de teste é xUnit. Os testes de integração são realizados usando o Microsoft.AspNetCore.TestHost, que inclui o TestServer. Como o pacote Microsoft.AspNetCore.Mvc.Testing
é usado para configurar o host de teste e o servidor de teste, os pacotes TestHost
e TestServer
não exigem referências diretas do pacote no arquivo de projeto do aplicativo de teste ou na configuração do desenvolvedor no aplicativo de teste.
Os testes de integração geralmente exigem um pequeno conjunto de dados no banco de dados antes da execução do teste. Por exemplo, um teste de exclusão exige a exclusão de um registro do banco de dados; portanto, o banco de dados deve ter pelo menos um registro para que a solicitação de exclusão tenha êxito.
O aplicativo de exemplo propaga o banco de dados com três mensagens em Utilities.cs
que os testes podem usar quando são executados:
public static void InitializeDbForTests(ApplicationDbContext db)
{
db.Messages.AddRange(GetSeedingMessages());
db.SaveChanges();
}
public static void ReinitializeDbForTests(ApplicationDbContext db)
{
db.Messages.RemoveRange(db.Messages);
InitializeDbForTests(db);
}
public static List<Message> GetSeedingMessages()
{
return new List<Message>()
{
new Message(){ Text = "TEST RECORD: You're standing on my scarf." },
new Message(){ Text = "TEST RECORD: Would you like a jelly baby?" },
new Message(){ Text = "TEST RECORD: To the rational mind, " +
"nothing is inexplicable; only unexplained." }
};
}
O contexto de banco de dados do SUT é registrado em Program.cs
. O retorno de chamada builder.ConfigureServices
do aplicativo de teste é executado depois que o código Program.cs
do aplicativo é executado. Para usar um banco de dados diferente para os testes, o contexto do banco de dados do aplicativo deve ser substituído em builder.ConfigureServices
. Para obter mais informações, consulte a seção Personalizar WebApplicationFactory .