इकाई परीक्षण एक वर्ग है जो कई स्रोतों से डेटा का अनुरोध करता है

Aug 17 2020

प्रसंग

मैं एक ऐसी परियोजना पर काम कर रहा हूं, जो .NET के लिए विभिन्न AWS SDKs का उपयोग करके AWS से डेटा खींचती है। यह विशिष्ट उदाहरण AWSSDK.IdentityManagementएसडीके से संबंधित है

लक्ष्य जानकारी से क्वेरी करना IAmazonIdentityManagementServiceऔर उसे उस मॉडल पर मैप करना है जो मेरे द्वारा काम कर रहे व्यवसाय डोमेन के लिए सहायक है

मुझे IamServiceकक्षा के लिए इकाई परीक्षण लिखने का काम सौंपा गया है ।

मुसीबत

यूनिट टेस्ट सेटअप के बहुत GetIamSummaryAsyncखराब होने के कारण, मैं मदद नहीं कर सकता, लेकिन मुझे लगता है कि मैं यूनिट टेस्टिंग ( ) का तरीका खराब तरीके से बनाया जाना चाहिए।

मैंने "कई ऑब्जेक्ट्स को एकल ऑब्जेक्ट्स के लिए डेटा स्रोत मैप करने के लिए डिज़ाइन पैटर्न" जैसी चीज़ों के लिए गुगली की है, लेकिन मेरे द्वारा दी गई एकमात्र सलाह एडेप्टर या प्रॉक्सी पैटर्न का उपयोग करना है। मुझे यकीन नहीं है कि उन्हें इस परिदृश्य पर कैसे लागू किया जाए

सवाल

  • वहाँ एक बेहतर तरीका है कि मैं IamServiceपरीक्षण करने के लिए अपनी कक्षा का निर्माण आसान (अधिक रसीला) कर सकता हूं ?
  • यदि इस प्रकार के परिदृश्य के लिए एडाप्टर या प्रॉक्सी पैटर्न उपयुक्त हैं, तो उन्हें कैसे लागू किया जाएगा?
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;
    }
}

जहाँ वर्ग IamSummaryModelको इस प्रकार परिभाषित किया गया है:

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

मुझे जो समस्या आ रही है, वह यह है कि मेरी यूनिट टेस्ट असेंबली सेक्शन में बड़े पैमाने पर कोड में बदल जाती है। मुझे प्रत्येक AWS SDK क्लाइंट मेथड के लिए मेरे द्वारा की जाने वाली हर कॉल का मजाक उड़ाना होगा।

उदाहरण इकाई परीक्षण

[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);
}

जवाब

2 Flater Aug 17 2020 at 17:55

इस पद्धति में कुछ भी गलत नहीं है। यह बहुत सारी जानकारी में खींच रहा है, लेकिन कभी-कभी ऐसा कुछ करना पड़ता है (जैसे रिपोर्टिंग के लिए, या एक बड़ा डेटा ट्रांसफर तैयार करना)।
यह अपरिहार्य है कि जब आप अपने डेटा स्रोत का मज़ाक उड़ाते हैं, तो आपके पास जितने अधिक स्रोत होंगे, उतने अधिक आपको मज़ाक करना पड़ेगा। यह आसानी से बचा नहीं है। हालाँकि, आप अपने दृष्टिकोण का पुनर्मूल्यांकन कर सकते हैं जो आपको यहाँ ले गया है।

1. क्या इस डेटा को संयोजित करने की आवश्यकता है?

अपने आप से पूछने वाला पहला सवाल यह है कि क्या इस डेटा का संयोजन आवश्यक है। यदि यह नहीं है, और आप इस डेटा को अलग रख सकते हैं, तो यह आपके कोडबेस को सरल और मॉक (और इस प्रकार परीक्षण) को आसान बनाए रखने का एक शानदार तरीका है।
यदि इस डेटा को किसी बिंदु पर संयोजित करने की आवश्यकता है, तो अपनी कक्षा को फिर से भरने से डेटा-संयोजन तर्क को दूसरे स्तर पर स्थानांतरित कर दिया जाता है, जहां एक ही इकाई-परीक्षण प्रश्न अब पॉप अप करता है: उस परत में कैसे मॉक करें ? तर्क को स्थानांतरित करना इसे ठीक नहीं करता है।

2. क्या मुझे इसकी जांच करने की आवश्यकता है?

दूसरी बात, आपको यह सवाल करना चाहिए कि क्या यूनिट टेस्टिंग का यहां वारंट है। हालांकि हर कोई सहमत नहीं है (व्यक्तिगत रूप से, मैं बाड़ पर हूं), IamServiceइकाई परीक्षण नहीं होने के लिए एक उचित तर्क दिया जाना चाहिए क्योंकि यह एक डोमेन तर्क वर्ग नहीं है, बल्कि इसके बजाय यह एक बाहरी संसाधन का आवरण / मैपर है ।

इसी तरह, मैं तब तक EntityFramework संदर्भ वर्ग का परीक्षण नहीं करूंगा, जब तक कि इसमें कस्टम व्यावसायिक तर्क (जैसे स्वचालित ऑडिटिंग फ़ील्ड) शामिल न हों, क्योंकि परीक्षण के लिए व्यावसायिक तर्क की आवश्यकता होती है। बाकी वर्ग केवल EF का कार्यान्वयन है, जो कि वारंट परीक्षण नहीं करता है।

आपका IamServiceवर्तमान में किसी भी वास्तविक व्यावसायिक तर्क से रहित है, इसलिए इकाई परीक्षण नहीं करने का तर्क मेरी राय में काफी मजबूत है। तर्क है कि IamSummaryModelवस्तु का मानचित्रण व्यावसायिक तर्क के रूप में गिना जाता है, ठीक है, तर्कपूर्ण है। मैं हमेशा तुच्छ मैपिंग का परीक्षण नहीं करता हूं क्योंकि तुच्छ कोड का परीक्षण नहीं किया जाना चाहिए (ध्यान दें: जबकि मेरा मानना ​​है कि यह सही है, मुझे पता है कि कोड पर "तुच्छ" लेबल का दुरुपयोग करना बहुत आसान है जो वास्तव में तुच्छ नहीं है। सॉफ़्टवेयर)।

3. मैं मॉकिंग के प्रयास को कैसे कम करूँ?

यदि आप इस बिंदु पर पहुंच गए हैं, तो आप सहमत हैं कि आपकी कक्षा का परीक्षण और डेटा संयोजन दोनों आवश्यक हैं। उस कक्षा का परीक्षण करते समय इन सभी डेटा स्रोतों का मजाक उड़ाने की आवश्यकता में तार्किक रूप से निष्कर्ष निकाला गया है। यह अब एक अपरिहार्य तथ्य बन गया है।

लेकिन इसका मतलब यह नहीं है कि आप व्यवस्था तर्क का पुन: उपयोग / सरलीकरण करके अपने जीवन को आसान नहीं बना सकते। अपने परीक्षण वर्ग को या तो एक आधार वर्ग से विरासत में मिला, जिसे एक स्थिरता के रूप में उपयोग किया जाता है, या कहा गया है कि एक संपत्ति है। इस उत्तर के लिए, मैं उत्तराधिकार मार्ग चुनूंगा, लेकिन या तो काम करता है।

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

यह इस तरह के एक स्थिरता का एक बहुत ही त्वरित कार्यान्वयन है। यह स्थिरता आपके लिए अधिकांश लेगवर्क कर सकती है। यदि आपके पास एक से अधिक परीक्षण हैं, जिन्हें मैं बहुत मान लूंगा, तो यह प्रत्येक व्यक्तिगत परीक्षण के लिए इसे स्थापित करने की जटिलता में काफी कटौती करेगा।

मॉक सेट करते समय, आप अपने द्वारा चुने गए मूल्यों पर भरोसा कर सकते हैं और गुणों के माध्यम से सुलभ हो सकते हैं, जिसे आप बाद में अपने मुखर तर्क के लिए पुन: उपयोग कर सकते हैं। उदाहरण के लिए:

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()
        );
    }
}

ध्यान दें कि मैंने एक विशिष्ट नकली प्रतिक्रिया कैसे निर्धारित की, और फिर परीक्षण के तहत अपनी इकाई से प्राप्त वास्तविक प्रतिक्रिया से तुलना करने के लिए उस नकली प्रतिक्रिया का उपयोग करने में सक्षम हूं।

यदि आपको विशिष्ट नीतियों के लिए विशिष्ट परीक्षण लिखने की आवश्यकता है, तो आप विधि पैरामीटर जोड़ सकते हैं जहां आवश्यक हो, जैसे:

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

कस्टम मॉक किए गए मानों का उपयोग करते समय आप संभवतः अधिक जटिल तर्क-वितर्क करने जा रहे हैं, लेकिन यह सिर्फ एक मूल उदाहरण है।


नोट: जैसा कि मैंने कैंडिड_ऑरेंज के उत्तर के लिए टिप्पणी में उल्लेख किया है, आपके डोमेन में आपके पुस्तकालयों से इंटरफेस का उपयोग न करने की सलाह दी जाती है (या कम से कम भारी रूप से इसे कम से कम), लेकिन यह आपके प्रश्न के मूल से असंबंधित है इसलिए मैं उस बिंदु पर संदेह कर रहा हूं।

4 candied_orange Aug 17 2020 at 15:03

डेटा स्रोतों के बारे में जानने वाला वर्ग परीक्षण नहीं किया जा सकता है। यह केवल एकीकरण परीक्षण किया जा सकता है।

एक वर्ग जो डेटा संरचनाओं से अवगत है, वह इकाई परीक्षण किया जा सकता है। आपको जो आवश्यक है वह डेटा संरचनाओं को प्रदान करने का एक तरीका है जिसमें डेटा स्रोतों के ज्ञान की आवश्यकता नहीं होती है।

ये मेमोरी डेटाबेस में हार्ड कोडेड टेस्ट डेटा से सब कुछ हो सकता है। लेकिन अगर आप अपने वास्तविक डेटा स्रोतों से बात कर रहे हैं, तो आप इकाई परीक्षण कर रहे हैं ।