इकाई परीक्षण एक वर्ग है जो कई स्रोतों से डेटा का अनुरोध करता है
प्रसंग
मैं एक ऐसी परियोजना पर काम कर रहा हूं, जो .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);
}
जवाब
इस पद्धति में कुछ भी गलत नहीं है। यह बहुत सारी जानकारी में खींच रहा है, लेकिन कभी-कभी ऐसा कुछ करना पड़ता है (जैसे रिपोर्टिंग के लिए, या एक बड़ा डेटा ट्रांसफर तैयार करना)।
यह अपरिहार्य है कि जब आप अपने डेटा स्रोत का मज़ाक उड़ाते हैं, तो आपके पास जितने अधिक स्रोत होंगे, उतने अधिक आपको मज़ाक करना पड़ेगा। यह आसानी से बचा नहीं है। हालाँकि, आप अपने दृष्टिकोण का पुनर्मूल्यांकन कर सकते हैं जो आपको यहाँ ले गया है।
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);
}
}
कस्टम मॉक किए गए मानों का उपयोग करते समय आप संभवतः अधिक जटिल तर्क-वितर्क करने जा रहे हैं, लेकिन यह सिर्फ एक मूल उदाहरण है।
नोट: जैसा कि मैंने कैंडिड_ऑरेंज के उत्तर के लिए टिप्पणी में उल्लेख किया है, आपके डोमेन में आपके पुस्तकालयों से इंटरफेस का उपयोग न करने की सलाह दी जाती है (या कम से कम भारी रूप से इसे कम से कम), लेकिन यह आपके प्रश्न के मूल से असंबंधित है इसलिए मैं उस बिंदु पर संदेह कर रहा हूं।
डेटा स्रोतों के बारे में जानने वाला वर्ग परीक्षण नहीं किया जा सकता है। यह केवल एकीकरण परीक्षण किया जा सकता है।
एक वर्ग जो डेटा संरचनाओं से अवगत है, वह इकाई परीक्षण किया जा सकता है। आपको जो आवश्यक है वह डेटा संरचनाओं को प्रदान करने का एक तरीका है जिसमें डेटा स्रोतों के ज्ञान की आवश्यकता नहीं होती है।
ये मेमोरी डेटाबेस में हार्ड कोडेड टेस्ट डेटा से सब कुछ हो सकता है। लेकिन अगर आप अपने वास्तविक डेटा स्रोतों से बात कर रहे हैं, तो आप इकाई परीक्षण कर रहे हैं ।