Isolando código em teste com falsificação da Microsoft
O Microsoft Fakes ajuda você a isolar o código que está testando substituindo outras partes do aplicativo por stubs ou shims.Esses são pequenos trechos de código sob o controle de seus testes.Ao isolar seu código para teste, você sabe que se o teste falhar, a causa está lá e não em outro lugar.Stubs e shims também permitem que você teste seu código mesmo que outras partes do aplicativo ainda não estejam funcionando.
O Fakes vem em duas versões:
Um stub substitui uma classe por um pequeno substituto que implementa a mesma interface.Para usar stubs, você precisa criar seu aplicativo de modo que cada componente dependa apenas das interfaces, e não de outros componentes.(Por “componente”, queremos dizer uma classe ou um grupo de classes que são criadas e atualizadas em conjunto e normalmente estão contidas em um assembly.)
Um shim modifica o código compilado de seu aplicativo no tempo de execução para que, em vez de fazer uma chamada de método específica, ele executa o código shim que seu teste fornece.Os shims podem ser usados para substituir chamadas para assemblies que você não pode modificar, como assemblies do .NET.
Requisitos
- Visual Studio Ultimate ou Premium
Escolhendo entre os tipos de stub e shim
Normalmente, você consideraria um projeto do Visual Studio para ser um componente, pois desenvolve e atualiza as classes ao mesmo tempo.Você poderia considerar o uso de stubs e shims para chamadas feitas pelo projeto para outros projetos em sua solução ou para outros assemblies que o projeto referencia.
Como guia geral, use stubs para chamadas em sua solução do Visual Studio, e shims para chamadas a outros assemblies referenciados.Isso é porque em sua própria solução é uma boa prática desacoplar os componentes definindo interfaces da maneira exigida pelo stub.Mas assemblies externos, como System.dll, geralmente não são fornecidos com definições de interface separadas, portanto, você deve usar shims.
Outras considerações são:
Desempenho. Os shims têm execução mais lenta porque reescrevem seu código no tempo de execução.Os stubs não têm essa sobrecarga de desempenho e são tão rápidos quanto os métodos virtuais.
Métodos estáticos, tipos lacrados. É possível usar stubs somente para implementar interfaces.Portanto, os tipos de stub não podem ser usados para métodos estáticos, métodos não virtuais, métodos virtuais lacrados, métodos em tipos lacrados e assim por diante.
Tipos internos. Os stubs e shims podem ser usados com tipos internos que são acessíveis com o atributo de assembly InternalsVisibleToAttribute.
Métodos particulares. Os shims poderão substituir chamadas para métodos particulares se todos os tipos na assinatura do método estiverem visíveis.Os stubs só podem substituir métodos visíveis.
Interfaces e métodos abstratos. Os stubs oferecem implementações de interfaces e métodos abstratos que podem ser usados em testes.Os shims não podem instrumentar interfaces e métodos abstratos, porque não têm corpos de método.
Em geral, recomendamos que você use tipos de stub para isolar das dependências dentro de sua base de código.É possível fazer isso ocultando os componentes atrás das interfaces.Os tipos de shims podem ser usados para isolar de componentes de terceiros que não oferecem uma API testável.
Introdução aos stubs
(Para obter uma descrição mais detalhada, consulte Usando stubs para isolar partes de seu aplicativo para testes de unidade.)
Injetar interfaces
Para usar stubs, você precisa escrever o código que deseja testar de forma que ele não mencione explicitamente as classes em outro componente de seu aplicativo.Por “componente”, queremos dizer uma classe ou um grupo de classes que são criadas e atualizadas em conjunto e normalmente estão contidas em um projeto do Visual Studio.As variáveis e os parâmetros devem ser declarados usando interfaces e instâncias de outros componentes que devem ser passados ou criados usando uma fábrica.Por exemplo, se StockFeed é uma classe em outro componente do aplicativo, ele é considerado incorreto:
return (new StockFeed()).GetSharePrice("COOO"); // Bad
Em vez disso, defina uma interface que possa ser implementada pelo outro componente, e que também possa ser implementada por um stub para fins de teste:
public int GetContosoPrice(IStockFeed feed) { return feed.GetSharePrice("COOO"); }
Public Function GetContosoPrice(feed As IStockFeed) As Integer Return feed.GetSharePrice("COOO") End Function
Adicionar assembly do Fakes
No Gerenciador de Soluções, expanda a lista de referências do projeto de teste.Se você estiver trabalhando no Visual Basic, escolha Mostrar Todos os Arquivos para ver a lista de referências.
Selecione a referência ao assembly em que a interface (por exemplo, IStockFeed) é definida.No menu de atalho dessa referência, escolha Adicionar Assembly do Fakes.
Recompile a solução.
Em seus testes, construa instâncias do stub e forneça o código para seus métodos:
[TestClass] class TestStockAnalyzer { [TestMethod] public void TestContosoStockPrice() { // Arrange: // Create the fake stockFeed: IStockFeed stockFeed = new StockAnalysis.Fakes.StubIStockFeed() // Generated by Fakes. { // Define each method: // Name is original name + parameter types: GetSharePriceString = (company) => { return 1234; } }; // In the completed application, stockFeed would be a real one: var componentUnderTest = new StockAnalyzer(stockFeed); // Act: int actualValue = componentUnderTest.GetContosoPrice(); // Assert: Assert.AreEqual(1234, actualValue); } ... }
<TestClass()> _ Class TestStockAnalyzer <TestMethod()> _ Public Sub TestContosoStockPrice() ' Arrange: ' Create the fake stockFeed: Dim stockFeed As New StockAnalysis.Fakes.StubIStockFeed With stockFeed .GetSharePriceString = Function(company) Return 1234 End Function End With ' In the completed application, stockFeed would be a real one: Dim componentUnderTest As New StockAnalyzer(stockFeed) ' Act: Dim actualValue As Integer = componentUnderTest.GetContosoPrice ' Assert: Assert.AreEqual(1234, actualValue) End Sub End Class
A parte especial da mágica aqui é a classe StubIStockFeed.Para cada interface no assembly referenciado, o mecanismo Microsoft Fakes gera uma classe de stub.O nome da classe stub é derivado do nome da interface, com "Fakes.Stub" como prefixo e os nomes dos tipos de parâmetro anexados.
Os stubs também são gerados para getters e setters de propriedades, para eventos e métodos genéricos.Para obter mais informações, consulte Usando stubs para isolar partes de seu aplicativo para testes de unidade.
Introdução aos shims
(Para obter uma descrição mais detalhada, consulte Usando shims para isolar seu aplicativo de outros assemblies para testes de unidade.)
Suponhamos que seu componente contenha chamadas para DateTime.Now:
// Code under test:
public int GetTheCurrentYear()
{
return DateTime.Now.Year;
}
Durante os testes, você gostaria de corrigir a propriedade Now, porque a versão real retorna inconvenientemente um valor diferente em cada chamada.
Para usar correções, você não precisa alterar o código do aplicativo ou escrevê-lo de uma maneira específica.
Adicionar assembly do Fakes
No Gerenciador de Soluções, abra as referências de seu projeto de teste de unidade e selecione a referência ao assembly que contém o método que você deseja forjar.Nesse exemplo, a classe DateTime está em System.dll.Para ver as referências em projeto do Visual Basic, escolha Mostrar Todos os Arquivos.
Escolha Adicionar Assembly do Fakes.
Inserir um shim em um ShimsContext
[TestClass] public class TestClass1 { [TestMethod] public void TestCurrentYear() { int fixedYear = 2000; // Shims can be used only in a ShimsContext: using (ShimsContext.Create()) { // Arrange: // Shim DateTime.Now to return a fixed date: System.Fakes.ShimDateTime.NowGet = () => { return new DateTime(fixedYear, 1, 1); }; // Instantiate the component under test: var componentUnderTest = new MyComponent(); // Act: int year = componentUnderTest.GetTheCurrentYear(); // Assert: // This will always be true if the component is working: Assert.AreEqual(fixedYear, year); } } }
<TestClass()> _ Public Class TestClass1 <TestMethod()> _ Public Sub TestCurrentYear() Using s = Microsoft.QualityTools.Testing.Fakes.ShimsContext.Create() Dim fixedYear As Integer = 2000 ' Arrange: ' Detour DateTime.Now to return a fixed date: System.Fakes.ShimDateTime.NowGet = _ Function() As DateTime Return New DateTime(fixedYear, 1, 1) End Function ' Instantiate the component under test: Dim componentUnderTest = New MyComponent() ' Act: Dim year As Integer = componentUnderTest.GetTheCurrentYear ' Assert: ' This will always be true if the component is working: Assert.AreEqual(fixedYear, year) End Using End Sub End Class
Os nomes das classes de shims são compostos pelo prefixo Fakes.Shim adicionado ao nome do tipo original.Os nomes dos parâmetros são acrescentados ao nome do método.(Você não precisa adicionar qualquer referência de projeto para System.Fakes)
O exemplo anterior usa um shim para um método estático.Para usar um shim para um método de instância, digite AllInstances entre o nome do tipo e o nome do método:
System.IO.Fakes.ShimFile.AllInstances.ReadToEnd = ...
Você não precisa inserir qualquer referência a System.IO.Fakes: ele é criado pelo processo de geração de correção.
Você também pode criar shims para instâncias específicas, para construtores e propriedades.Para obter mais informações, consulte Usando shims para isolar seu aplicativo de outros assemblies para testes de unidade.
Nesta seção
Usando stubs para isolar partes de seu aplicativo para testes de unidade
Usando shims para isolar seu aplicativo de outros assemblies para testes de unidade
Geração de código, compilação e convenções de nomenclatura no Microsoft Fakes