Test unitaire d'une classe qui demande des données à plusieurs sources
Le contexte
Je travaille sur un projet qui extrait des données d'AWS à l'aide des différents kits SDK AWS pour .NET. Cet exemple spécifique concerne le AWSSDK.IdentityManagement
SDK
L'objectif est d'interroger des informations IAmazonIdentityManagementService
et de les mapper à un modèle utile pour le domaine d'activité dans lequel je travaille
J'ai été chargé d'écrire des tests unitaires pour la IamService
classe.
Problème
Les configurations de test unitaire étant si verbeuses, je ne peux pas m'empêcher de penser que la méthode I'm Unit Testing ( GetIamSummaryAsync
) doit être mal construite.
J'ai cherché sur Google des choses comme «Modèles de conception pour mapper plusieurs sources de données sur des objets uniques», mais le seul conseil que je vois est d'utiliser les modèles d'adaptateur ou de proxy. Je ne sais pas comment les appliquer à ce scénario
Question
- Y a-t-il une meilleure façon de construire ma
IamService
classe pour la rendre plus facile (plus succincte) à tester? - Si les modèles d'adaptateur ou de proxy sont appropriés pour ce type de scénario, comment seraient-ils appliqués?
public class IamService : IIamService
{
IAmazonIdentityManagementService _iamClient;
public IamService(IAmazonIdentityManagementService iamClient)
{
_iamClient = iamClient;
}
public async Task<IamSummaryModel> GetIamSummaryAsync()
{
var getAccountSummaryResponse = await _iamClient.GetAccountSummaryAsync();
var listCustomerManagedPoliciesResponse = await _iamClient.ListPoliciesAsync();
var listGroupsResponse = await _iamClient.ListGroupsAsync();
var listInstanceProfilesResponse = await _iamClient.ListInstanceProfilesAsync();
var listRolesResponse = await _iamClient.ListRolesAsync();
var listServerCertificatesResponse = await _iamClient.ListServerCertificatesAsync();
var listUsersResponse = await _iamClient.ListUsersAsync();
IamSummaryModel iamSummary = new IamSummaryModel();
iamSummary.CustomerManagedPolicies.Count = listCustomerManagedPoliciesResponse.Policies.Count;
iamSummary.CustomerManagedPolicies.DefaultQuota = getAccountSummaryResponse.SummaryMap["PoliciesQuota"];
iamSummary.Groups.Count = listGroupsResponse.Groups.Count;
iamSummary.Groups.DefaultQuota = getAccountSummaryResponse.SummaryMap["GroupsQuota"];
iamSummary.InstanceProfiles.Count = listInstanceProfilesResponse.InstanceProfiles.Count;
iamSummary.InstanceProfiles.DefaultQuota = getAccountSummaryResponse.SummaryMap["InstanceProfilesQuota"];
iamSummary.Roles.Count = listRolesResponse.Roles.Count;
iamSummary.Roles.DefaultQuota = getAccountSummaryResponse.SummaryMap["RolesQuota"];
iamSummary.ServerCertificates.Count = listServerCertificatesResponse.ServerCertificateMetadataList.Count;
iamSummary.ServerCertificates.DefaultQuota = getAccountSummaryResponse.SummaryMap["ServerCertificatesQuota"];
iamSummary.Users.Count = listUsersResponse.Users.Count;
iamSummary.Users.DefaultQuota = getAccountSummaryResponse.SummaryMap["UsersQuota"];
return iamSummary;
}
}
Où la classe IamSummaryModel
est définie comme:
public sealed class IamSummaryModel
{
public ResourceSummaryModel CustomerManagedPolicies { get; set; } = new ResourceSummaryModel();
public ResourceSummaryModel Groups { get; set; } = new ResourceSummaryModel();
public ResourceSummaryModel InstanceProfiles { get; set; } = new ResourceSummaryModel();
public ResourceSummaryModel Roles { get; set; } = new ResourceSummaryModel();
public ResourceSummaryModel ServerCertificates { get; set; } = new ResourceSummaryModel();
public ResourceSummaryModel Users { get; set; } = new ResourceSummaryModel();
}
public sealed class ResourceSummaryModel
{
public int Count { get; set; }
public int DefaultQuota { get; set; }
}
Le problème auquel je suis confronté est que mes tests unitaires se transforment en une masse de code dans la section Assembler. Je dois me moquer de chaque appel que je fais à chaque méthode client AWS SDK.
Exemple de test unitaire
[Fact]
public async Task GetIamSummaryAsync_CustomerManagerPolicies_MapToModel()
{
// Arrange
var iamClientStub = new Mock<IAmazonIdentityManagementService>();
iamClientStub.Setup(iam => iam.ListPoliciesAsync(It.IsAny<CancellationToken>()))
.Returns(Task.FromResult(
new ListPoliciesResponse()
{
Policies = new List<ManagedPolicy>()
{
new ManagedPolicy(),
new ManagedPolicy()
}
}
));
// Lots of other mocks, one for each dependency
var sut = new IamService(iamClientStub.Object);
// Act
var actual = await sut.GetIamSummaryAsync();
// Assert
Assert.Equal(2, actual.CustomerManagedPolicies.Count);
}
Réponses
Il n'y a rien de mal avec cette méthode. Cela recueille beaucoup d'informations, mais parfois c'est quelque chose qui doit être fait (par exemple pour faire des rapports ou préparer un transfert de données volumineux).
Il est inévitable que lorsque vous vous moquez de votre source de données, plus vous avez de sources, plus vous devez vous moquer. Ce n'est pas facilement évité. Cependant, vous pouvez réévaluer votre approche qui vous a conduit ici.
1. Ces données doivent-elles être combinées?
La première question à se poser est de savoir si la combinaison de ces données est nécessaire. Si ce n'est pas le cas et que vous pouvez conserver ces données séparément, c'est un excellent moyen de garder votre base de code plus simple et plus facile à simuler (et donc à tester).
Si ces données doivent être combinées à un moment donné, la refactorisation de votre classe ne fait que déplacer la logique de combinaison de données vers un autre niveau, où la même question de test unitaire apparaît maintenant: comment se moquer de cette couche? Déplacer la logique ne résout pas le problème.
2. Dois-je effectuer un test unitaire?
Deuxièmement, vous devez vous demander si les tests unitaires sont justifiés ici. Bien que tout le monde ne soit pas d'accord (personnellement, je suis sur la clôture), il y a un argument raisonnable pour IamService
ne pas être testé unitaire car ce n'est pas une classe de logique de domaine, mais plutôt un wrapper / mapper d'une ressource externe .
De même, je ne testerais pas non plus une classe de contexte EntityFramework, à moins qu'elle ne contienne une logique métier personnalisée (par exemple des champs d'audit automatique), car cette logique métier a besoin d'être testée. Le reste de la classe n'est que l'implémentation d'EF, ce qui ne justifie pas de test.
Votre IamService
est actuellement dépourvu de toute logique métier réelle, donc l'argument de ne pas le tester unitaire est assez fort, à mon avis. L'argument selon lequel le mappage de l' IamSummaryModel
objet compte comme logique métier est, eh bien, discutable. Je ne teste pas toujours les mappages triviaux car le code trivial ne doit pas être testé (note: bien que je pense que cela est correct, je suis conscient qu'il est très facile de mal utiliser l'étiquette "trivial" sur du code qui n'est pas vraiment trivial. ATTENTION)
3. Comment minimiser l'effort de moquerie?
Si vous avez atteint ce stade, vous acceptez que la combinaison des données et le test unitaire de votre classe sont nécessaires. Cela conclut logiquement à la nécessité de se moquer de toutes ces sources de données lors du test de cette classe. C'est maintenant devenu un fait inévitable.
Mais cela ne signifie pas que vous ne pouvez pas vous faciliter la vie en réutilisant / simplifiant la logique d'arrangement. Laissez votre classe de test hériter d'une classe de base utilisée comme fixture, ou implémenter une propriété contenant ledit fixture. Pour cette réponse, je choisirai l'itinéraire d'héritage, mais l'un ou l'autre fonctionne.
public class IamServiceTestFixture
{
protected IamService GetService()
{
var mockedAmazonService = GetMockedAmazonService();
return new IamService(mockedAmazonService);
}
private IAmazonIdentityManagementService GetMockedAmazonService()
{
var iamClientStub = new Mock<IAmazonIdentityManagementService>();
// Set up your mocks
return iamClientStub;
}
}
public class IamServiceTests : IamServiceTestFixture
{
[Test]
public void MyTest()
{
// Arrange
var sut = GetService();
// Act
var actual = await sut.GetIamSummaryAsync();
// Assert
Assert.Equal(2, actual.CustomerManagedPolicies.Count);
}
}
Il s'agit d'une mise en œuvre très rapide d'un tel appareil. Ce luminaire peut faire la plupart des démarches pour vous. Si vous avez plus d'un test, ce que je suppose que vous le ferez, cela réduira considérablement la complexité d'avoir à configurer cela pour chaque test individuel.
Lors de la configuration de la maquette, vous pouvez vous fier aux valeurs que vous avez choisies et les rendre accessibles via des propriétés, que vous pouvez ensuite réutiliser ultérieurement pour votre logique d'assertion. Par exemple:
public class IamServiceTestFixture
{
protected ListPoliciesResponse ListPoliciesResponse { get; private set; }
public IamServiceTestFixture()
{
this.ListPoliciesResponse = new ListPoliciesResponse()
{
Policies = new List<ManagedPolicy>()
{
new ManagedPolicy(),
new ManagedPolicy()
}
}
}
protected IamService GetService()
{
var mockedAmazonService = GetMockedAmazonService();
return new IamService(mockedAmazonService);
}
private IAmazonIdentityManagementService GetMockedAmazonService()
{
var iamClientStub = new Mock<IAmazonIdentityManagementService>();
iamClientStub.Setup(iam => iam.ListPoliciesAsync(It.IsAny<CancellationToken>()))
.Returns(Task.FromResult(this.ListPoliciesResponse));
return iamClientStub;
}
}
public class IamServiceTests : IamServiceTestFixture
{
[Test]
public void MyTest()
{
// Arrange
var sut = GetService();
// Act
var actual = await sut.GetIamSummaryAsync();
// Assert
Assert.Equal(
this.ListPoliciesResponse.Policies.Count(),
actual.CustomerManagedPolicies.Count()
);
}
}
Remarquez comment j'ai configuré une réponse fictive spécifique et que je suis ensuite en mesure d'utiliser cette réponse simulée pour la comparer à la réponse réelle reçue de mon unité testée.
Si vous avez besoin d'écrire des tests spécifiques pour des politiques spécifiques, vous pouvez ajouter des paramètres de méthode si nécessaire, par exemple:
public class IamServiceTestFixture
{
protected IamService GetService(IEnumerable<ManagedPolicy> policies)
{
var mockedAmazonService = GetMockedAmazonService(policies);
return new IamService(mockedAmazonService);
}
private IAmazonIdentityManagementService GetMockedAmazonService(IEnumerable<ManagedPolicy> policies)
{
var iamClientStub = new Mock<IAmazonIdentityManagementService>();
iamClientStub.Setup(iam => iam.ListPoliciesAsync(It.IsAny<CancellationToken>()))
.Returns(Task.FromResult(new ListPoliciesResponse()
{
Policies = policies
}));
return iamClientStub;
}
}
public class IamServiceTests : IamServiceTestFixture
{
[Test]
public void MyTest()
{
var customPolicy = new ManagedPolicy();
// Arrange
var sut = GetService(new ManagedPolicy[] { customPolicy });
// Act
var actual = await sut.GetIamSummaryAsync();
// Assert
actual.CustomerManagedPolicies.Should().Contain(customPolicy);
}
}
Vous allez probablement avoir une logique d'assertion plus complexe lorsque vous utilisez des valeurs simulées personnalisées, mais ce n'est qu'un exemple de base.
Remarque: comme je l'ai mentionné dans le commentaire de la réponse de candied_orange, il est conseillé de ne pas utiliser les interfaces de vos bibliothèques dans votre domaine (ou du moins de la minimiser fortement), mais cela n'a aucun rapport avec le cœur de votre question ici, donc je saute ce point.
Une classe qui connaît les sources de données ne peut pas être testée unitaire. Il ne peut être testé que par intégration.
Une classe qui connaît les structures de données peut être testée unitaire. Ce dont vous avez besoin, c'est d'un moyen de fournir les structures de données qui ne nécessitent pas de connaissance des sources de données.
Il peut s'agir de tout, des données de test codées en dur aux bases de données en mémoire. Mais si vous parlez à vos vraies sources de données, vous ne faites pas de tests unitaires .