Visão geral do Microsoft IntelliTest

O IntelliTest permite localizar bugs no início e reduz os custos de manutenção de teste. Usando uma abordagem de teste transparente e automatizada, o IntelliTest pode gerar um conjunto de testes candidato para seu código .NET. A geração de conjunto de testes pode ser ainda mais orientada por propriedades de correção especificadas por você. O IntelliTest até mesmo evoluirá o conjunto de testes automaticamente à medida que o código em teste evolui.

Observação

O IntelliTest está disponível apenas na edição Enterprise. Há suporte para o código no C# direcionado ao .NET Framework. Para obter suporte ao .NET 6 com o IntelliTest, instale a versão prévia do Visual Studio Enterprise e confira o comunicado .

Testes de caracterização O IntelliTest permite que você determine o comportamento do código em termos de um pacote de testes de unidade tradicionais. Esse conjunto de testes pode ser usado como um pacote de regressão, formando a base para lidar com a complexidade associados à refatoração do código herdado ou desconhecido.

Geração de entrada de teste guiada O IntelliTest usa uma abordagem de resolução de restrição e análise de código aberta para gerar automaticamente valores de entrada de teste precisos, normalmente sem necessidade de intervenção do usuário. Para tipos de objeto complexos, ele gera automaticamente as fábricas. Você pode guiar a geração de entrada de teste estendendo e configurando as fábricas para atender aos seus requisitos. As propriedades de correção especificadas como asserções no código são usadas automaticamente para orientar ainda mais a geração de entrada de teste.

Integração de ISE O IntelliTest está totalmente integrado ao Visual Studio IDE. Todas as informações coletadas durante a geração do conjunto de testes (como as entradas geradas automaticamente, a saída do código, os casos de teste gerados e seu status de aprovação ou falha) aparecem no Visual Studio IDE. Você pode iterar facilmente entre corrigir seu código e executar novamente o IntelliTest sem sair do Visual Studio IDE. Os testes podem ser salvos na solução como um Projeto de Teste de Unidade e serão automaticamente detectados posteriormente pelo Gerenciador de Testes do Visual Studio.

Complementar práticas de teste existentes Use o IntelliTest para complementar qualquer prática de teste existente que você já esteja seguindo.

Se desejar testar:

O Olá, Mundo do IntelliTest

O IntelliTest localiza entradas relevantes para o programa testado, o que significa que você pode usá-lo para gerar a famosa cadeia de caracteres Olá, Mundo. Isso pressupõe que você criou um projeto de teste com base em MSTest do C# e adicionou uma referência ao Microsoft.Pex.Framework. Se você estiver usando uma estrutura de teste diferente, crie uma biblioteca de classes do C# e consulte a documentação da estrutura de teste sobre como configurar o projeto.

O exemplo a seguir cria duas restrições no parâmetro chamado value de forma que o IntelliTest gera a cadeia de caracteres necessária:

using System;
using Microsoft.Pex.Framework;
using Microsoft.VisualStudio.TestTools.UnitTesting;

[TestClass]
public partial class HelloWorldTest {
    [PexMethod]
    public void HelloWorld([PexAssumeNotNull]string value) {
        if (value.StartsWith("Hello")
            && value.EndsWith("World!")
            && value.Contains(" "))
            throw new Exception("found it!");
    }
}

Depois de compilado e executado, o IntelliTest gera um conjunto de testes, como o conjunto a seguir:

  1. ""
  2. "\0\0\0\0\0"
  3. "Olá"
  4. "\0\0\0\0\0\0"
  5. "Olá\0"
  6. "Olá\0\0"
  7. "Olá\0Mundo!"
  8. "Olá, Mundo!"

Observação

Para problemas de build, tente substituir as referências Microsoft.VisualStudio.TestPlatform.TestFramework e Microsoft.VisualStudio.TestPlatform.TestFramework.Extensions por uma referência ao Microsoft.VisualStudio.QualityTools.UnitTestFramework.

Leia Gerar testes de unidade com IntelliTest para entender onde os testes gerados são salvos. O código de teste gerado deve incluir um teste como o código a seguir:

[TestMethod]
[PexGeneratedBy(typeof(global::HelloWorldTest))]
[PexRaisedException(typeof(Exception))]
public void HelloWorldThrowsException167()
{
    this.HelloWorld("Hello World!");
}

É fácil assim!

Recursos adicionais:

Atributos importantes

using Microsoft.Pex.Framework;

[..., PexClass(typeof(Foo))]
public partial class FooTest {
    [PexMethod]
    public void Bar([PexAssumeNotNull]Foo target, int i) {
        target.Bar(i);
    }
}
[assembly: PexAssemblyUnderTest("MyAssembly")] // also instruments "MyAssembly"
[assembly: PexInstrumentAssembly("Lib")]

Classes auxiliares estáticas importantes

  • PexAssume avalia suposições (filtragem de entrada)
  • PexAssert avalia declarações
  • PexChoose gera novas opções (entradas adicionais)
  • PexObserve registra valores dinâmicos para os testes gerados
[PexMethod]
void StaticHelpers(Foo target) {
    PexAssume.IsNotNull(target);

    int i = PexChoose.Value<int>("i");
    string result = target.Bar(i);

    PexObserve.ValueForViewing<string>("result", result);
    PexAssert.IsNotNull(result);
}

Limitações

Esta seção descreve as limitações do IntelliTest:

Não determinismo

O IntelliTest pressupõe que o programa analisado é determinístico. Se não for, o IntelliTest rodará até chegar a um limite de exploração.

O IntelliTest considera um programa como não determinístico se ele depende de entradas que o IntelliTest não pode controlar.

O IntelliTest controla entradas fornecidas para testes de unidade parametrizados e obtidas de PexChoose. Nesse sentido, os resultados de chamadas para código não gerenciado ou não instrumentado também são considerados como "entradas" para o programa instrumentado, mas o IntelliTest não pode controlá-los. Se o fluxo de controle do programa depende de valores específicos provenientes dessas fontes externas, o IntelliTest não poderá "direcionar" o programa para as áreas descobertas anteriormente.

Além disso, o programa é considerado não determinístico se os valores das fontes externas mudam ao executar o programa novamente. Em tais casos o IntelliTest perde o controle sobre a execução do programa e a pesquisa se torna ineficiente.

Às vezes, não é óbvio quando isso acontece. Considere os seguintes exemplos:

  • O resultado do método GetHashCode() é fornecido pelo código não gerenciado e não é previsível.
  • A classe System.Random usa a hora atual do sistema para entregar valores verdadeiramente aleatórios.
  • A classe System.DateTime fornece a hora atual, que não está sob o controle do IntelliTest.

Simultaneidade

O IntelliTest não trata de programas multi-threaded.

Código nativo

O IntelliTest não entende o código nativo, como instruções x86 chamadas por meio de P/Invoke. Ele não sabe como converter essas chamadas em restrições que podem ser passadas para o solver de restrição. Mesmo para o código .NET, ele pode analisar somente o código que instrumenta. O IntelliTest não pode instrumentar certas partes do mscorlib, incluindo a biblioteca de reflexão. O DynamicMethod não pode ser instrumentado.

A solução alternativa sugerida é ter um modo de teste em que esses métodos estão localizados em tipos em um assembly dinâmico. No entanto, mesmo se alguns métodos não forem instrumentados, o IntelliTest tentará abranger o máximo de código instrumentado possível.

Plataforma

O IntelliTest tem suporte apenas no .NET Framework X86 de 32 bits.

Idioma

Em princípio, o IntelliTest pode analisar programas .NET arbitrários, gravados em qualquer linguagem .NET. No entanto, no Visual Studio ele dá suporte apenas a C#.

Raciocínio simbólico

O IntelliTest usa um solver de restrições automático para determinar quais valores são relevantes para o teste e o programa em teste. No entanto, as habilidades do solver de restrição são, e sempre serão, limitadas.

Rastreamentos de pilha incorretos

Como o IntelliTest captura e "relança" exceções em cada método instrumentado, os números de linha em rastreamentos de pilha não estarão corretos. Essa é uma limitação pelo design da instrução "relançar".