प्रमुख मजकुराकडे जा

कॉलडेटा ऑप्टिमायझेशनसाठी लहान ABI

स्तर 2
मध्यम
Ori Pomerantz
१ एप्रिल, २०२२
13 मिनिट वाचन

प्रस्तावना

या लेखात, आपण optimistic rollups बद्दल शिकाल, त्यावरील व्यवहारांची किंमत, आणि ती वेगळी किंमत संरचना Ethereum मेननेटपेक्षा आपल्याला वेगवेगळ्या गोष्टींसाठी ऑप्टिमाइझ करण्याची आवश्यकता कशी निर्माण करते. हे ऑप्टिमायझेशन कसे लागू करायचे हे देखील आपण शिकाल.

संपूर्ण प्रकटीकरण

मी एक पूर्णवेळ Optimism (opens in a new tab) कर्मचारी आहे, त्यामुळे या लेखातील उदाहरणे Optimism वर चालतील. तथापि, येथे स्पष्ट केलेले तंत्र इतर रोलअपसाठी देखील तितकेच चांगले कार्य करेल.

परिभाषा

रोलअपवर चर्चा करताना, 'स्तर 1' (L1) ही संज्ञा मेननेटसाठी वापरली जाते, जे उत्पादन Ethereum नेटवर्क आहे. 'स्तर 2' (L2) ही संज्ञा रोलअप किंवा इतर कोणत्याही प्रणालीसाठी वापरली जाते जी सुरक्षिततेसाठी L1 वर अवलंबून असते परंतु तिची बहुतेक प्रक्रिया ऑफचेन करते.

आपण L2 व्यवहारांची किंमत आणखी कशी कमी करू शकतो?

ऑप्टिमिस्टिक रोलअप्सना प्रत्येक ऐतिहासिक व्यवहाराचा रेकॉर्ड जतन करावा लागतो जेणेकरून कोणीही त्यातून जाऊन सध्याची स्थिती योग्य आहे हे सत्यापित करू शकेल. Ethereum मेननेटमध्ये डेटा मिळवण्याचा सर्वात स्वस्त मार्ग म्हणजे तो कॅलडेटा म्हणून लिहिणे. हे समाधान Optimism (opens in a new tab) आणि Arbitrum (opens in a new tab) या दोघांनी निवडले होते.

L2 व्यवहारांची किंमत

L2 व्यवहारांच्या किंमतीत दोन घटक असतात:

  1. L2 प्रक्रिया, जी सहसा अत्यंत स्वस्त असते
  2. L1 स्टोरेज, जे मेननेट गॅस खर्चाशी जोडलेले आहे

मी हे लिहित असताना, Optimism वर L2 गॅसची किंमत 0.001 Gwei आहे. दुसरीकडे, L1 गॅसची किंमत अंदाजे 40 gwei आहे. आपण सध्याच्या किमती येथे पाहू शकता (opens in a new tab).

कॅलडेटाच्या एका बाइटला एकतर 4 गॅस (जर तो शून्य असेल तर) किंवा 16 गॅस (जर ते इतर कोणतेही मूल्य असेल तर) लागतात. EVM वरील सर्वात महागड्या क्रियांपैकी एक म्हणजे स्टोरेजमध्ये लिहिणे. L2 वर स्टोरेजमध्ये 32-बाइट शब्द लिहिण्याची कमाल किंमत 22100 गॅस आहे. सध्या, हे 22.1 gwei आहे. म्हणून जर आपण कॅलडेटाचा एक शून्य बाइट वाचवू शकलो, तर आपण स्टोरेजमध्ये सुमारे 200 बाइट्स लिहू शकू आणि तरीही फायद्यात राहू.

ABI

बहुतेक व्यवहार बाह्य-मालकीच्या खात्यातून करारावर प्रवेश करतात. बहुतेक कॉन्ट्रॅक्ट्स Solidity मध्ये लिहिलेले आहेत आणि ऍप्लिकेशन बायनरी इंटरफेस (ABI) (opens in a new tab) नुसार त्यांचे डेटा फील्ड इंटरप्रिट करतात.

तथापि, ABI हे L1 साठी डिझाइन केले होते, जिथे कॅलडेटाच्या एका बाइटची किंमत अंदाजे चार अंकगणितीय क्रियांइतकी असते, L2 साठी नाही, जिथे कॅलडेटाच्या एका बाइटची किंमत हजारहून अधिक अंकगणितीय क्रियांइतकी असते. कॅलडेटा खालीलप्रमाणे विभागलेला आहे:

विभागलांबीबाइट्सवाया गेलेले बाइट्सवाया गेलेला गॅसआवश्यक बाइट्सआवश्यक गॅस
फंक्शन सिलेक्टर40-3348116
शून्य124-15124800
गंतव्य पत्ता2016-350020320
रक्कम3236-67176415240
एकूण68160576

स्पष्टीकरण:

  • फंक्शन सिलेक्टर: कॉन्ट्रॅक्टमध्ये 256 पेक्षा कमी फंक्शन्स आहेत, त्यामुळे आपण त्यांना एका बाइटने वेगळे करू शकतो. हे बाइट्स सहसा गैर-शून्य असतात आणि त्यामुळे सोळा गॅस खर्च येतो (opens in a new tab).
  • शून्य: हे बाइट्स नेहमी शून्य असतात कारण वीस-बाइट पत्त्याला ठेवण्यासाठी बत्तीस-बाइट शब्दाची आवश्यकता नसते. शून्य धारण करणार्‍या बाइट्सना चार गॅस खर्च येतो (पिवळे पेपर पहा (opens in a new tab), परिशिष्ट G, पृ. 27, Gtxdatazero चे मूल्य).
  • रक्कम: जर आपण असे गृहीत धरले की या कॉन्ट्रॅक्टमध्ये decimals अठरा आहे (सामान्य मूल्य) आणि आपण हस्तांतरित करत असलेल्या टोकन्सची कमाल रक्कम 1018 असेल, तर आपल्याला 1036 ची कमाल रक्कम मिळते. 25615 > 1036, म्हणून पंधरा बाइट्स पुरेसे आहेत.

L1 वर 160 गॅसचा अपव्यय सहसा नगण्य असतो. एका व्यवहाराला किमान 21,000 गॅस (opens in a new tab) खर्च येतो, त्यामुळे अतिरिक्त 0.8% ने काही फरक पडत नाही. तथापि, L2 वर, गोष्टी वेगळ्या आहेत. व्यवहाराचा जवळजवळ संपूर्ण खर्च तो L1 वर लिहिण्याचा असतो. व्यवहार कॅलडेटा व्यतिरिक्त, 109 बाइट्सचे व्यवहार हेडर (गंतव्य पत्ता, स्वाक्षरी, इ.) आहे. म्हणून एकूण खर्च 109*16+576+160=2480 आहे, आणि आपण त्यापैकी सुमारे 6.5% वाया घालवत आहोत.

जेव्हा आपण गंतव्यस्थानावर नियंत्रण ठेवत नाही तेव्हा खर्च कमी करणे

आपले गंतव्य कॉन्ट्रॅक्टवर नियंत्रण नाही असे गृहीत धरून, आपण तरीही यासारखा (opens in a new tab) उपाय वापरू शकता. चला संबंधित फाइल्स पाहूया.

Token.sol

हे गंतव्य कॉन्ट्रॅक्ट आहे (opens in a new tab). हे एक मानक ERC-20 कॉन्ट्रॅक्ट आहे, ज्यात एक अतिरिक्त वैशिष्ट्य आहे. हे faucet फंक्शन कोणत्याही वापरकर्त्याला वापरण्यासाठी काही टोकन मिळवू देते. हे एका उत्पादन ERC-20 कॉन्ट्रॅक्टला निरुपयोगी बनवेल, परंतु जेव्हा एखादे ERC-20 केवळ चाचणी सुलभ करण्यासाठी अस्तित्वात असते तेव्हा ते जीवन सोपे करते.

1 /**
2 * @dev कॉलरला खेळण्यासाठी 1000 टोकन देते
3 */
4 function faucet() external {
5 _mint(msg.sender, 1000);
6 } // फंक्शन फॉसेट

CalldataInterpreter.sol

हे ते कॉन्ट्रॅक्ट आहे ज्याला व्यवहारांनी लहान कॅलडेटासह कॉल करणे अपेक्षित आहे (opens in a new tab). चला ओळीनुसार पाहूया.

1//SPDX-License-Identifier: Unlicense
2pragma solidity ^0.8.0;
3
4
5import { OrisUselessToken } from "./Token.sol";

त्याला कसे कॉल करायचे हे जाणून घेण्यासाठी आम्हाला टोकन फंक्शनची आवश्यकता आहे.

1contract CalldataInterpreter {
2
3 OrisUselessToken public immutable token;

ज्या टोकनसाठी आम्ही प्रॉक्सी आहोत त्याचा पत्ता.

1
2 /**
3 * @dev टोकनचा पत्ता निर्दिष्ट करा
4 * @param tokenAddr_ ERC-20 कॉन्ट्रॅक्टचा पत्ता
5 */
6 constructor(
7 address tokenAddr_
8 ) {
9 token = OrisUselessToken(tokenAddr_);
10 } // कन्स्ट्रक्टर
सर्व दाखवा

आम्हाला निर्दिष्ट करण्याची आवश्यकता असलेला टोकन पत्ता हा एकमेव पॅरामीटर आहे.

1 function calldataVal(uint startByte, uint length)
2 private pure returns (uint) {

कॅलडेटा मधून मूल्य वाचा.

1 uint _retVal;
2
3 require(length < 0x21,
4 "calldataVal लांबीची मर्यादा 32 बाइट्स आहे");
5
6 require(length + startByte <= msg.data.length,
7 "calldataVal calldatasize च्या पलीकडे वाचण्याचा प्रयत्न करत आहे");

आपण मेमरीमध्ये एकच 32-बाइट (256-बिट) शब्द लोड करणार आहोत आणि जे बाइट्स आपल्याला हव्या असलेल्या फील्डचा भाग नाहीत ते काढून टाकणार आहोत. हे अल्गोरिदम 32 बाइटपेक्षा जास्त लांबीच्या मूल्यांसाठी कार्य करत नाही, आणि अर्थातच आपण कॅलडेटाच्या शेवटाच्या पुढे वाचू शकत नाही. L1 वर गॅस वाचवण्यासाठी या चाचण्या वगळणे आवश्यक असू शकते, परंतु L2 वर गॅस अत्यंत स्वस्त आहे, ज्यामुळे आपण विचार करू शकणाऱ्या कोणत्याही सॅनिटी तपासण्या शक्य होतात.

1 assembly {
2 _retVal := calldataload(startByte)
3 }

आपण fallback() (खाली पहा) च्या कॉलमधून डेटा कॉपी करू शकलो असतो, परंतु EVM ची असेंब्ली भाषा Yul (opens in a new tab) वापरणे सोपे आहे.

येथे आपण स्टॅकमध्ये startByte ते startByte+31 बाइट्स वाचण्यासाठी CALLDATALOAD ऑपकोड (opens in a new tab) वापरतो. सर्वसाधारणपणे, Yul मधील ऑपकोडचे सिंटॅक्स <opcode name>(<first stack value, if any>,<second stack value, if any>...) असे आहे.

1
2 _retVal = _retVal >> (256-length*8);

केवळ सर्वात लक्षणीय length बाइट्स फील्डचा भाग आहेत, म्हणून आपण इतर मूल्ये काढून टाकण्यासाठी राइट-शिफ्ट (opens in a new tab) करतो. याचा अतिरिक्त फायदा असा आहे की ते मूल्य फील्डच्या उजवीकडे हलवते, त्यामुळे ते मूल्य स्वतःच आहे, 256something पट मूल्य नाही.

1
2 return _retVal;
3 }
4
5
6 fallback() external {

जेव्हा Solidity कॉन्ट्रॅक्टला केलेला कॉल कोणत्याही फंक्शन सिग्नेचरशी जुळत नाही, तेव्हा ते (एक आहे असे गृहीत धरून) fallback() फंक्शन (opens in a new tab) कॉल करते. CalldataInterpreter च्या बाबतीत, कोणताही कॉल येथे येतो कारण इतर कोणतेही external किंवा public फंक्शन्स नाहीत.

1 uint _func;
2
3 _func = calldataVal(0, 1);

कॅलडेटाचा पहिला बाइट वाचा, जो आपल्याला फंक्शन सांगतो. येथे एखादे फंक्शन उपलब्ध नसण्याची दोन कारणे आहेत:

  1. pure किंवा view असणारी फंक्शन्स स्थिती बदलत नाहीत आणि त्यांना गॅस लागत नाही (जेव्हा ऑफचेन कॉल केले जाते). त्यांचा गॅस खर्च कमी करण्याचा प्रयत्न करण्यात काही अर्थ नाही.
  2. msg.sender (opens in a new tab) वर अवलंबून असलेली फंक्शन्स. msg.sender चे मूल्य CalldataInterpreter चा पत्ता असेल, कॉलरचा नाही.

दुर्दैवाने, ERC-20 स्पेसिफिकेशन्स पाहिल्यास (opens in a new tab), फक्त एकच फंक्शन उरते, transfer. यामुळे आपल्याकडे फक्त दोन फंक्शन्स उरतात: transfer (कारण आपण transferFrom कॉल करू शकतो) आणि faucet (कारण आपण ज्याने आपल्याला कॉल केला आहे त्याला टोकन परत हस्तांतरित करू शकतो).

1
2 // कॅलडेटामधील माहिती वापरून टोकनच्या स्थिती बदलणाऱ्या पद्धतींना कॉल करा
3 //
4
5 // फॉसेट
6 if (_func == 1) {

faucet() ला कॉल, ज्यामध्ये पॅरामीटर्स नाहीत.

1 token.faucet();
2 token.transfer(msg.sender,
3 token.balanceOf(address(this)));
4 }

आपण token.faucet() कॉल केल्यावर आपल्याला टोकन्स मिळतात. तथापि, प्रॉक्सी कॉन्ट्रॅक्ट म्हणून, आम्हाला टोकनची गरज नाही. आपल्याला कॉल करणार्‍या EOA (बाह्य मालकीच्या खात्याला) किंवा कॉन्ट्रॅक्टला गरज आहे. म्हणून आम्ही आमचे सर्व टोकन ज्याने आम्हाला कॉल केला आहे त्याला हस्तांतरित करतो.

1 // हस्तांतरण (आपल्याकडे त्यासाठी भत्ता आहे असे गृहीत धरा)
2 if (_func == 2) {

टोकन हस्तांतरित करण्यासाठी दोन पॅरामीटर्स आवश्यक आहेत: गंतव्य पत्ता आणि रक्कम.

1 token.transferFrom(
2 msg.sender,

आम्ही फक्त कॉल करणाऱ्यांना त्यांच्या मालकीचे टोकन हस्तांतरित करण्याची परवानगी देतो

1 address(uint160(calldataVal(1, 20))),

गंतव्य पत्ता बाइट #1 पासून सुरू होतो (बाइट #0 हे फंक्शन आहे). एक पत्ता म्हणून, तो 20-बाइट लांब आहे.

1 calldataVal(21, 2)

या विशिष्ट कॉन्ट्रॅक्टसाठी आपण असे गृहीत धरतो की कोणीही हस्तांतरित करू इच्छित असलेल्या टोकनची कमाल संख्या दोन बाइटमध्ये (65536 पेक्षा कमी) बसते.

1 );
2 }

एकंदरीत, हस्तांतरणासाठी 35 बाइट्सचा कॅलडेटा लागतो:

विभागलांबीबाइट्स
फंक्शन सिलेक्टर10
गंतव्य पत्ता321-32
रक्कम233-34
1 } // फॉलबॅक
2
3} // कॉन्ट्रॅक्ट CalldataInterpreter

test.js

हे JavaScript युनिट टेस्ट (opens in a new tab) आपल्याला हे तंत्र कसे वापरावे (आणि ते योग्यरित्या कार्य करते की नाही हे कसे सत्यापित करावे) हे दाखवते. मी असे गृहीत धरणार आहे की तुम्हाला chai (opens in a new tab) आणि ethers (opens in a new tab) समजतात आणि केवळ कॉन्ट्रॅक्टवर विशेषतः लागू होणारे भाग स्पष्ट करणार आहे.

1const { expect } = require("chai");
2
3describe("CalldataInterpreter", function () {
4 it("Should let us use tokens", async function () {
5 const Token = await ethers.getContractFactory("OrisUselessToken")
6 const token = await Token.deploy()
7 await token.deployed()
8 console.log("Token addr:", token.address)
9
10 const Cdi = await ethers.getContractFactory("CalldataInterpreter")
11 const cdi = await Cdi.deploy(token.address)
12 await cdi.deployed()
13 console.log("CalldataInterpreter addr:", cdi.address)
14
15 const signer = await ethers.getSigner()
सर्व दाखवा

आपण दोन्ही कॉन्ट्रॅक्ट्स तैनात करून सुरुवात करतो.

1 // खेळण्यासाठी टोकन मिळवा
2 const faucetTx = {

आम्ही व्यवहार तयार करण्यासाठी सामान्यतः वापरत असलेली उच्च-स्तरीय फंक्शन्स (जसे की token.faucet()) वापरू शकत नाही, कारण आम्ही ABI चे पालन करत नाही. त्याऐवजी, आपल्याला स्वतः व्यवहार तयार करून तो पाठवावा लागेल.

1 to: cdi.address,
2 data: "0x01"

व्यवहारासाठी आपल्याला दोन पॅरामीटर्स द्यावे लागतील:

  1. to, गंतव्य पत्ता. हा कॅलडेटा इंटरप्रिटर कॉन्ट्रॅक्ट आहे.
  2. data, पाठवण्यासाठी कॅलडेटा. फॉसेट कॉलच्या बाबतीत, डेटा एकच बाइट आहे, 0x01.
1
2 }
3 await (await signer.sendTransaction(faucetTx)).wait()

आम्ही साईनरच्या sendTransaction पद्धतीला (opens in a new tab) कॉल करतो कारण आम्ही आधीच गंतव्य (faucetTx.to) निर्दिष्ट केले आहे आणि आम्हाला व्यवहार स्वाक्षरी केलेला हवा आहे.

1// फॉसेट टोकन योग्यरित्या पुरवतो का ते तपासा
2expect(await token.balanceOf(signer.address)).to.equal(1000)

येथे आपण शिल्लक तपासतो. view फंक्शन्सवर गॅस वाचवण्याची गरज नाही, म्हणून आपण त्यांना सामान्यपणे चालवतो.

1// CDI ला भत्ता द्या (मंजुरी प्रॉक्सी केली जाऊ शकत नाही)
2const approveTX = await token.approve(cdi.address, 10000)
3await approveTX.wait()
4expect(await token.allowance(signer.address, cdi.address)).to.equal(10000)

हस्तांतरण करण्यासाठी कॅलडेटा इंटरप्रिटरला भत्ता द्या.

1// टोकन हस्तांतरित करा
2const destAddr = "0xf5a6ead936fb47f342bb63e676479bddf26ebe1d"
3const transferTx = {
4 to: cdi.address,
5 data: "0x02" + destAddr.slice(2, 42) + "0100",
6}

एक हस्तांतरण व्यवहार तयार करा. पहिला बाइट "0x02" आहे, त्यानंतर गंतव्य पत्ता आणि शेवटी रक्कम (0x0100, जे दशांश मध्ये 256 आहे).

1 await (await signer.sendTransaction(transferTx)).wait()
2
3 // तपासा की आपल्याकडे 256 टोकन कमी आहेत
4 expect (await token.balanceOf(signer.address)).to.equal(1000-256)
5
6 // आणि ते आपल्या गंतव्यस्थानाला मिळाले आहेत
7 expect (await token.balanceOf(destAddr)).to.equal(256)
8 }) // ते
9}) // वर्णन करा
सर्व दाखवा

जेव्हा तुम्ही गंतव्य कॉन्ट्रॅक्टवर नियंत्रण ठेवता तेव्हा खर्च कमी करणे

जर तुमचे गंतव्य कॉन्ट्रॅक्टवर नियंत्रण असेल तर तुम्ही अशी फंक्शन्स तयार करू शकता जी msg.sender तपासण्यांना बायपास करतात कारण ते कॅलडेटा इंटरप्रिटरवर विश्वास ठेवतात. हे कसे कार्य करते याचे उदाहरण तुम्ही येथे पाहू शकता, control-contract शाखेत (opens in a new tab).

जर कॉन्ट्रॅक्ट केवळ बाह्य व्यवहारांना प्रतिसाद देत असेल, तर आपण फक्त एका कॉन्ट्रॅक्टने काम चालवू शकलो असतो. तथापि, ते कंपोझिबिलिटी खंडित करेल. सामान्य ERC-20 कॉल्सला प्रतिसाद देणारा एक कॉन्ट्रॅक्ट आणि लहान कॉल डेटासह व्यवहारांना प्रतिसाद देणारा दुसरा कॉन्ट्रॅक्ट असणे खूप चांगले आहे.

Token.sol

या उदाहरणात आपण Token.sol मध्ये बदल करू शकतो. हे आपल्याला अनेक फंक्शन्स ठेवण्याची परवानगी देते जे फक्त प्रॉक्सी कॉल करू शकते. हे नवीन भाग आहेत:

1 // CalldataInterpreter पत्ता निर्दिष्ट करण्याची परवानगी असलेला एकमेव पत्ता
2 address owner;
3
4 // CalldataInterpreter पत्ता
5 address proxy = address(0);

ERC-20 कॉन्ट्रॅक्टला अधिकृत प्रॉक्सीची ओळख माहित असणे आवश्यक आहे. तथापि, आपण हे व्हेरिएबल कन्स्ट्रक्टरमध्ये सेट करू शकत नाही, कारण आपल्याला अद्याप मूल्य माहित नाही. हा कॉन्ट्रॅक्ट प्रथम इन्स्टंटिएट केला जातो कारण प्रॉक्सीला त्याच्या कन्स्ट्रक्टरमध्ये टोकनच्या पत्त्याची अपेक्षा असते.

1 /**
2 * @dev ERC20 कन्स्ट्रक्टरला कॉल करते.
3 */
4 constructor(
5 ) ERC20("Oris useless token-2", "OUT-2") {
6 owner = msg.sender;
7 }

निर्मात्याचा पत्ता (ज्याला owner म्हणतात) येथे संग्रहित केला जातो कारण प्रॉक्सी सेट करण्याची परवानगी असलेला तो एकमेव पत्ता आहे.

1 /**
2 * @dev प्रॉक्सीसाठी (CalldataInterpreter) पत्ता सेट करा.
3 * मालकाद्वारे फक्त एकदाच कॉल केले जाऊ शकते
4 */
5 function setProxy(address _proxy) external {
6 require(msg.sender == owner, "फक्त मालकाद्वारे कॉल केले जाऊ शकते");
7 require(proxy == address(0), "प्रॉक्सी आधीच सेट आहे");
8
9 proxy = _proxy;
10 } // फंक्शन setProxy
सर्व दाखवा

प्रॉक्सीला विशेषाधिकारित प्रवेश आहे, कारण ते सुरक्षा तपासण्या बायपास करू शकते. आपण प्रॉक्सीवर विश्वास ठेवू शकतो याची खात्री करण्यासाठी, आम्ही फक्त owner ला हे फंक्शन कॉल करू देतो, आणि फक्त एकदाच. एकदा proxy चे वास्तविक मूल्य (शून्य नाही) आले की, ते मूल्य बदलू शकत नाही, त्यामुळे मालकाने बदमाश होण्याचा निर्णय घेतला तरी, किंवा त्यासाठीचा मेमोनिक उघड झाला तरी, आपण तरीही सुरक्षित आहोत.

1 /**
2 * @dev काही फंक्शन्स फक्त प्रॉक्सीद्वारे कॉल केली जाऊ शकतात.
3 */
4 modifier onlyProxy {

हे एक modifier फंक्शन (opens in a new tab) आहे, ते इतर फंक्शन्सच्या कार्यपद्धतीत बदल करते.

1 require(msg.sender == proxy);

प्रथम, सत्यापित करा की आपल्याला प्रॉक्सीने आणि इतर कोणीही कॉल केलेला नाही. नसल्यास, revert करा.

1 _;
2 }

तसे असल्यास, आपण सुधारित करत असलेले फंक्शन चालवा.

1 /* फंक्शन्स जे प्रॉक्सीला खात्यांसाठी प्रत्यक्षात प्रॉक्सी करण्याची परवानगी देतात */
2
3 function transferProxy(address from, address to, uint256 amount)
4 public virtual onlyProxy() returns (bool)
5 {
6 _transfer(from, to, amount);
7 return true;
8 }
9
10 function approveProxy(address from, address spender, uint256 amount)
11 public virtual onlyProxy() returns (bool)
12 {
13 _approve(from, spender, amount);
14 return true;
15 }
16
17 function transferFromProxy(
18 address spender,
19 address from,
20 address to,
21 uint256 amount
22 ) public virtual onlyProxy() returns (bool)
23 {
24 _spendAllowance(from, spender, amount);
25 _transfer(from, to, amount);
26 return true;
27 }
सर्व दाखवा

या तीन क्रिया आहेत ज्यांना सामान्यतः टोकन हस्तांतरित करणाऱ्या किंवा भत्ता मंजूर करणाऱ्या घटकाकडून थेट संदेश येण्याची आवश्यकता असते. येथे आमच्याकडे या ऑपरेशन्सची प्रॉक्सी आवृत्ती आहे जी:

  1. onlyProxy() द्वारे सुधारित केले आहे जेणेकरून इतर कोणालाही त्यांना नियंत्रित करण्याची परवानगी नाही.
  2. जो पत्ता सामान्यतः msg.sender असतो तो अतिरिक्त पॅरामीटर म्हणून मिळवतो.

CalldataInterpreter.sol

कॅलडेटा इंटरप्रिटर वरील इंटरप्रिटरसारखाच आहे, फक्त प्रॉक्सी केलेल्या फंक्शन्सना msg.sender पॅरामीटर मिळतो आणि transfer साठी भत्त्याची गरज नसते.

1 // हस्तांतरण (भत्त्याची गरज नाही)
2 if (_func == 2) {
3 token.transferProxy(
4 msg.sender,
5 address(uint160(calldataVal(1, 20))),
6 calldataVal(21, 2)
7 );
8 }
9
10 // मंजूर करा
11 if (_func == 3) {
12 token.approveProxy(
13 msg.sender,
14 address(uint160(calldataVal(1, 20))),
15 calldataVal(21, 2)
16 );
17 }
18
19 // transferFrom
20 if (_func == 4) {
21 token.transferFromProxy(
22 msg.sender,
23 address(uint160(calldataVal( 1, 20))),
24 address(uint160(calldataVal(21, 20))),
25 calldataVal(41, 2)
26 );
27 }
सर्व दाखवा

Test.js

मागील टेस्टिंग कोड आणि या कोडमध्ये काही बदल आहेत.

1const Cdi = await ethers.getContractFactory("CalldataInterpreter")
2const cdi = await Cdi.deploy(token.address)
3await cdi.deployed()
4await token.setProxy(cdi.address)

आम्हाला ERC-20 कॉन्ट्रॅक्टला कोणत्या प्रॉक्सीवर विश्वास ठेवावा हे सांगावे लागेल

1console.log("CalldataInterpreter addr:", cdi.address)
2
3// भत्ते तपासण्यासाठी दोन स्वाक्षरीकर्त्यांची आवश्यकता आहे
4const signers = await ethers.getSigners()
5const signer = signers[0]
6const poorSigner = signers[1]

approve() आणि transferFrom() तपासण्यासाठी आम्हाला दुसऱ्या स्वाक्षरीकर्त्याची आवश्यकता आहे. आम्ही त्याला poorSigner म्हणतो कारण त्याला आमचे कोणतेही टोकन मिळत नाहीत (अर्थात, त्याच्याकडे ETH असणे आवश्यक आहे).

1// टोकन हस्तांतरित करा
2const destAddr = "0xf5a6ead936fb47f342bb63e676479bddf26ebe1d"
3const transferTx = {
4 to: cdi.address,
5 data: "0x02" + destAddr.slice(2, 42) + "0100",
6}
7await (await signer.sendTransaction(transferTx)).wait()

कारण ERC-20 कॉन्ट्रॅक्ट प्रॉक्सी (cdi) वर विश्वास ठेवतो, आम्हाला हस्तांतरण रिले करण्यासाठी भत्त्याची गरज नाही.

1// मंजूरी आणि transferFrom
2const approveTx = {
3 to: cdi.address,
4 data: "0x03" + poorSigner.address.slice(2, 42) + "00FF",
5}
6await (await signer.sendTransaction(approveTx)).wait()
7
8const destAddr2 = "0xE1165C689C0c3e9642cA7606F5287e708d846206"
9
10const transferFromTx = {
11 to: cdi.address,
12 data: "0x04" + signer.address.slice(2, 42) + destAddr2.slice(2, 42) + "00FF",
13}
14await (await poorSigner.sendTransaction(transferFromTx)).wait()
15
16// approve / transferFrom कॉम्बो योग्यरित्या केले होते की नाही हे तपासा
17expect(await token.balanceOf(destAddr2)).to.equal(255)
सर्व दाखवा

दोन नवीन फंक्शन्सची चाचणी घ्या. लक्षात घ्या की transferFromTx ला दोन पत्ता पॅरामीटर्स आवश्यक आहेत: भत्त्याचा दाता आणि स्वीकारकर्ता.

निष्कर्ष

दोन्ही Optimism (opens in a new tab) आणि Arbitrum (opens in a new tab) L1 वर लिहिलेल्या कॅलडेटाचा आकार आणि त्यामुळे व्यवहारांची किंमत कमी करण्याचे मार्ग शोधत आहेत. तथापि, सामान्य उपायांचा शोध घेणारे पायाभूत सुविधा प्रदाते म्हणून, आमच्या क्षमता मर्यादित आहेत. dapp डेव्हलपर म्हणून, आपल्याकडे अनुप्रयोगा-विशिष्ट ज्ञान आहे, जे आपल्याला सामान्य समाधानापेक्षा आपले कॅलडेटा अधिक चांगल्या प्रकारे ऑप्टिमाइझ करू देते. आशा आहे की, हा लेख आपल्याला आपल्या गरजांसाठी आदर्श समाधान शोधण्यात मदत करेल.

माझ्या कामाबद्दल अधिक माहितीसाठी येथे पहा (opens in a new tab).

पृष्ठ अखेरचे अद्यतन: ३ मार्च, २०२६

हे मार्गदर्शन उपयुक्त होते का?