मुख्य सामग्री पर जाएं

परीक्षण के लिए Solidity स्मार्ट अनुबंधों को मॉक कैसे करें

Solidity
स्मार्ट अनुबंध
परीक्षण
मॉकिंग
मध्यवर्ती
मार्कस वास
2 मई 2020
5 मिनट पढ़ें

मॉक ऑब्जेक्ट्स (Mock objects) (opens in a new tab) ऑब्जेक्ट-ओरिएंटेड प्रोग्रामिंग में एक सामान्य डिज़ाइन पैटर्न हैं। पुराने फ्रांसीसी शब्द 'mocquer' से आने वाले, जिसका अर्थ 'मज़ाक उड़ाना' है, यह 'किसी वास्तविक चीज़ की नकल करने' के रूप में विकसित हुआ, जो वास्तव में हम प्रोग्रामिंग में कर रहे हैं। कृपया केवल तभी अपने स्मार्ट अनुबंधों का मज़ाक उड़ाएं जब आप चाहें, लेकिन जब भी संभव हो उन्हें मॉक (mock) करें। यह आपके जीवन को आसान बनाता है।

मॉक्स के साथ अनुबंधों की यूनिट-टेस्टिंग (Unit-testing)

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

आप हर बार जटिल परीक्षण सेटअप लॉजिक लिख सकते हैं जो अनुबंध को आवश्यक स्थिति में लाता है या आप एक मॉक लिख सकते हैं। इनहेरिटेंस (inheritance) के साथ किसी अनुबंध को मॉक करना आसान है। बस एक दूसरा मॉक अनुबंध बनाएं जो मूल अनुबंध से इनहेरिट करता हो। अब आप अपने मॉक में फ़ंक्शंस को ओवरराइड (override) कर सकते हैं। आइए इसे एक उदाहरण से समझते हैं।

उदाहरण: निजी ERC-20

हम एक उदाहरण ERC-20 अनुबंध का उपयोग करते हैं जिसमें एक प्रारंभिक निजी समय (private time) होता है। मालिक (owner) निजी उपयोगकर्ताओं को प्रबंधित कर सकता है और शुरुआत में केवल उन्हें ही टोकन प्राप्त करने की अनुमति होगी। एक बार एक निश्चित समय बीत जाने के बाद, सभी को टोकन का उपयोग करने की अनुमति होगी। यदि आप उत्सुक हैं, तो हम नए ओपनजेपेलिन (OpenZeppelin) अनुबंध v3 से _beforeTokenTransfer (opens in a new tab) हुक का उपयोग कर रहे हैं।

और अब आइए इसे मॉक करें।

आपको निम्नलिखित त्रुटि संदेशों (error messages) में से एक मिलेगा:

  • PrivateERC20Mock.sol: TypeError: Overriding function is missing "override" specifier.
  • PrivateERC20.sol: TypeError: Trying to override non-virtual function. Did you forget to add "virtual"?.

चूंकि हम नए 0.6 Solidity संस्करण का उपयोग कर रहे हैं, इसलिए हमें उन फ़ंक्शंस के लिए virtual कीवर्ड जोड़ना होगा जिन्हें ओवरराइड किया जा सकता है और ओवरराइड करने वाले फ़ंक्शन के लिए override जोड़ना होगा। तो आइए उन्हें दोनों isPublic फ़ंक्शंस में जोड़ें।

अब अपने यूनिट परीक्षणों में, आप इसके बजाय PrivateERC20Mock का उपयोग कर सकते हैं। जब आप निजी उपयोग के समय के दौरान व्यवहार का परीक्षण करना चाहते हैं, तो setIsPublic(false) का उपयोग करें और इसी तरह सार्वजनिक उपयोग के समय का परीक्षण करने के लिए setIsPublic(true) का उपयोग करें। बेशक हमारे उदाहरण में, हम समय को तदनुसार बदलने के लिए टाइम हेल्पर्स (time helpers) (opens in a new tab) का भी उपयोग कर सकते हैं। लेकिन मॉकिंग का विचार अब स्पष्ट होना चाहिए और आप ऐसे परिदृश्यों की कल्पना कर सकते हैं जहां यह केवल समय को आगे बढ़ाने जितना आसान नहीं है।

कई अनुबंधों को मॉक करना

यदि आपको हर एक मॉक के लिए एक और अनुबंध बनाना पड़े तो यह अव्यवस्थित हो सकता है। यदि यह आपको परेशान करता है, तो आप MockContract (opens in a new tab) लाइब्रेरी पर एक नज़र डाल सकते हैं। यह आपको तुरंत (on-the-fly) अनुबंधों के व्यवहार को ओवरराइड करने और बदलने की अनुमति देता है। हालाँकि, यह केवल किसी अन्य अनुबंध पर कॉल को मॉक करने के लिए काम करता है, इसलिए यह हमारे उदाहरण के लिए काम नहीं करेगा।

मॉकिंग और भी अधिक शक्तिशाली हो सकती है

मॉकिंग की शक्तियां यहीं समाप्त नहीं होती हैं।

  • फ़ंक्शंस जोड़ना: न केवल किसी विशिष्ट फ़ंक्शन को ओवरराइड करना उपयोगी है, बल्कि केवल अतिरिक्त फ़ंक्शंस जोड़ना भी उपयोगी है। टोकन के लिए एक अच्छा उदाहरण केवल एक अतिरिक्त mint फ़ंक्शन होना है ताकि किसी भी उपयोगकर्ता को मुफ्त में नए टोकन प्राप्त करने की अनुमति मिल सके।
  • टेस्टनेट्स (testnets) में उपयोग: जब आप अपने विकेंद्रीकृत एप्लिकेशन (dapp) के साथ टेस्टनेट्स पर अपने अनुबंधों को तैनात करते हैं और उनका परीक्षण करते हैं, तो एक मॉक किए गए संस्करण का उपयोग करने पर विचार करें। जब तक आपको वास्तव में आवश्यकता न हो, फ़ंक्शंस को ओवरराइड करने से बचें। आखिरकार आप वास्तविक लॉजिक का परीक्षण करना चाहते हैं। लेकिन उदाहरण के लिए एक रीसेट फ़ंक्शन जोड़ना उपयोगी हो सकता है जो अनुबंध की स्थिति को शुरुआत में रीसेट कर देता है, किसी नई तैनाती की आवश्यकता नहीं होती है। स्पष्ट रूप से आप इसे मेननेट (Mainnet) अनुबंध में नहीं रखना चाहेंगे।