కాల్డేటా ఆప్టిమైజేషన్ కోసం చిన్న ABIలు
పరిచయం
ఈ ఆర్టికల్లో, మీరు ఆప్టిమిస్టిక్ రోలప్లు, వాటిపై లావాదేవీల ఖర్చు, మరియు Ethereum మెయిన్నెట్లో కంటే భిన్నమైన విషయాల కోసం ఆప్టిమైజ్ చేయడానికి ఆ విభిన్న వ్యయ నిర్మాణం మనల్ని ఎలా కోరుతుందో తెలుసుకుంటారు. ఈ ఆప్టిమైజేషన్ను ఎలా అమలు చేయాలో కూడా మీరు తెలుసుకుంటారు.
పూర్తి బహిర్గతం
నేను పూర్తి సమయం Optimismopens in a new tab ఉద్యోగిని, కాబట్టి ఈ ఆర్టికల్లోని ఉదాహరణలు Optimismలో రన్ అవుతాయి. అయితే, ఇక్కడ వివరించిన టెక్నిక్ ఇతర రోలప్లకు కూడా అలాగే పని చేయాలి.
పదజాలం
రోలప్ల గురించి చర్చిస్తున్నప్పుడు, 'లేయర్ 1' (L1) అనే పదం మెయిన్నెట్, అంటే ఉత్పత్తి Ethereum నెట్వర్క్ కోసం ఉపయోగించబడుతుంది. 'లేయర్ 2' (L2) అనే పదం రోలప్ లేదా భద్రత కోసం L1పై ఆధారపడి ఉండే ఏదైనా ఇతర సిస్టమ్ కోసం ఉపయోగించబడుతుంది, కానీ దాని ప్రాసెసింగ్లో ఎక్కువ భాగాన్ని ఆఫ్చెయిన్లో చేస్తుంది.
L2 లావాదేవీల ఖర్చును మనం ఇంకా ఎలా తగ్గించగలం?
ఆప్టిమిస్టిక్ రోలప్లు ప్రతి చారిత్రక లావాదేవీ రికార్డును భద్రపరచాలి, తద్వారా ఎవరైనా వాటి ద్వారా వెళ్లి ప్రస్తుత స్థితి సరైనదని ధృవీకరించగలరు. Ethereum మెయిన్నెట్లోకి డేటాను పొందడానికి చౌకైన మార్గం దానిని కాల్డేటాగా వ్రాయడం. ఈ పరిష్కారాన్ని Optimismopens in a new tab మరియు Arbitrumopens in a new tab రెండూ ఎంచుకున్నాయి.
L2 లావాదేవీల ఖర్చు
L2 లావాదేవీల ఖర్చు రెండు భాగాలతో కూడి ఉంటుంది:
- L2 ప్రాసెసింగ్, ఇది సాధారణంగా చాలా చౌకగా ఉంటుంది
- 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లో కాదు, ఇక్కడ ఒక బైట్ కాల్డేటా వెయ్యి కంటే ఎక్కువ అంకగణిత కార్యకలాపాలకు ఖర్చవుతుంది. కాల్డేటా ఈ విధంగా విభజించబడింది:
| విభాగం | పొడవు | బైట్లు | వృధా అయిన బైట్లు | వృధా అయిన గ్యాస్ | అవసరమైన బైట్లు | అవసరమైన గ్యాస్ |
|---|---|---|---|---|---|---|
| ఫంక్షన్ సెలెక్టర్ | 4 | 0-3 | 3 | 48 | 1 | 16 |
| సున్నాలు | 12 | 4-15 | 12 | 48 | 0 | 0 |
| గమ్యస్థాన చిరునామా | 20 | 16-35 | 0 | 0 | 20 | 320 |
| మొత్తం | 32 | 36-67 | 17 | 64 | 15 | 240 |
| మొత్తం | 68 | 160 | 576 |
వివరణ:
- ఫంక్షన్ సెలెక్టర్: కాంట్రాక్ట్లో 256 కంటే తక్కువ ఫంక్షన్లు ఉన్నాయి, కాబట్టి మనం వాటిని ఒకే బైట్తో వేరు చేయవచ్చు. ఈ బైట్లు సాధారణంగా సున్నా కానివి మరియు అందువల్ల పదహారు గ్యాస్ ఖర్చుopens in a new tab అవుతుంది.
- సున్నాలు: ఈ బైట్లు ఎల్లప్పుడూ సున్నాగా ఉంటాయి, ఎందుకంటే ఇరవై-బైట్ చిరునామాను నిల్వ చేయడానికి ముప్పై-రెండు-బైట్ పదం అవసరం లేదు.
సున్నాను కలిగి ఉన్న బైట్లకు నాలుగు గ్యాస్ ఖర్చవుతుంది (పసుపు పత్రాన్ని చూడండిopens in a new tab, అనుబంధం G,
p. 27,
Gtxdatazeroకోసం విలువ). - మొత్తం: ఈ కాంట్రాక్ట్లో
డెసిమల్స్పద్దెనిమిది (సాధారణ విలువ) అని మరియు మేము బదిలీ చేసే టోకెన్ల గరిష్ట మొత్తం 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 Gives the caller 1000 tokens to play with3 */4 function faucet() external {5 _mint(msg.sender, 1000);6 } // function faucetCalldataInterpreter.sol
ఇది తక్కువ కాల్డేటాతో లావాదేవీలు కాల్ చేయాల్సిన కాంట్రాక్ట్opens in a new tab. దాన్ని లైన్ వారీగా చూద్దాం.
1//SPDX-License-Identifier: Unlicense2pragma solidity ^0.8.0;345import { OrisUselessToken } from "./Token.sol";దాన్ని ఎలా కాల్ చేయాలో తెలుసుకోవడానికి మనకు టోకెన్ ఫంక్షన్ అవసరం.
1contract CalldataInterpreter {23 OrisUselessToken public immutable token;మనం ప్రాక్సీగా ఉన్న టోకెన్ చిరునామా.
12 /**3 * @dev Specify the token address4 * @param tokenAddr_ ERC-20 contract address5 */6 constructor(7 address tokenAddr_8 ) {9 token = OrisUselessToken(tokenAddr_);10 } // constructorఅన్నీ చూపించుటోకెన్ చిరునామా మాత్రమే మనం పేర్కొనవలసిన పారామీటర్.
1 function calldataVal(uint startByte, uint length)2 private pure returns (uint) {కాల్డేటా నుండి ఒక విలువను చదవండి.
1 uint _retVal;23 require(length < 0x21,4 "calldataVal length limit is 32 bytes");56 require(length + startByte <= msg.data.length,7 "calldataVal trying to read beyond calldatasize");మనం ఒకే 32-బైట్ (256-బిట్) పదాన్ని మెమరీకి లోడ్ చేయబోతున్నాము మరియు మనకు కావలసిన ఫీల్డ్లో భాగం కాని బైట్లను తీసివేయబోతున్నాము. ఈ అల్గోరిథం 32 బైట్ల కంటే ఎక్కువ పొడవు ఉన్న విలువల కోసం పని చేయదు మరియు వాస్తవానికి మనం కాల్డేటా చివరిని దాటి చదవలేము. L1లో గ్యాస్ను ఆదా చేయడానికి ఈ పరీక్షలను దాటవేయడం అవసరం కావచ్చు, కానీ L2లో గ్యాస్ చాలా చౌకగా ఉంటుంది, ఇది మనం ఆలోచించగల ఏవైనా శానిటీ చెక్లను అనుమతిస్తుంది.
1 assembly {2 _retVal := calldataload(startByte)3 }మనం fallback() (క్రింద చూడండి) కు కాల్ నుండి డేటాను కాపీ చేసి ఉండవచ్చు, కానీ EVM యొక్క అసెంబ్లీ భాష అయిన Yulopens 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>...).
12 _retVal = _retVal >> (256-length*8);అత్యంత ముఖ్యమైన length బైట్లు మాత్రమే ఫీల్డ్లో భాగంగా ఉంటాయి, కాబట్టి ఇతర విలువలను వదిలించుకోవడానికి మేము రైట్-షిఫ్ట్opens in a new tab చేస్తాము.
ఇది విలువను ఫీల్డ్ యొక్క కుడి వైపుకు తరలించే అదనపు ప్రయోజనాన్ని కలిగి ఉంది, కాబట్టి ఇది విలువ 256something రెట్లు కాకుండా స్వయంగా విలువ అవుతుంది.
12 return _retVal;3 }456 fallback() external {Solidity కాంట్రాక్ట్కు చేసిన కాల్ ఏ ఫంక్షన్ సంతకాలతో సరిపోలకపోతే, అది the fallback() ఫంక్షన్opens in a new tabను కాల్ చేస్తుంది (ఒకటి ఉందని ఊహిస్తూ).
CalldataInterpreter విషయంలో, ఏ కాల్ అయినా ఇక్కడికి వస్తుంది ఎందుకంటే ఇతర external లేదా public ఫంక్షన్లు లేవు.
1 uint _func;23 _func = calldataVal(0, 1);కాల్డేటా యొక్క మొదటి బైట్ను చదవండి, ఇది మనకు ఫంక్షన్ను తెలియజేస్తుంది. ఇక్కడ ఒక ఫంక్షన్ అందుబాటులో లేకపోవడానికి రెండు కారణాలు ఉన్నాయి:
pureలేదాviewగా ఉండే ఫంక్షన్లు స్థితిని మార్చవు మరియు గ్యాస్ ఖర్చు చేయవు (ఆఫ్చెయిన్లో కాల్ చేసినప్పుడు). వాటి గ్యాస్ ధరను తగ్గించడానికి ప్రయత్నించడంలో అర్థం లేదు.msg.senderopens in a new tabపై ఆధారపడే ఫంక్షన్లు.msg.senderయొక్క విలువ కాలర్ కాదు,CalldataInterpreterయొక్క చిరునామా అవుతుంది.
దురదృష్టవశాత్తు, ERC-20 స్పెసిఫికేషన్లను చూస్తేopens in a new tab, ఇది కేవలం ఒక ఫంక్షన్, transferను మాత్రమే వదిలివేస్తుంది.
ఇది మనకు కేవలం రెండు ఫంక్షన్లను మాత్రమే మిగిల్చింది: transfer (ఎందుకంటే మనం transferFromని కాల్ చేయవచ్చు) మరియు faucet (ఎందుకంటే మనల్ని పిలిచిన వారికి టోకెన్లను తిరిగి బదిలీ చేయవచ్చు).
12 // Call the state changing methods of token using3 // information from the calldata45 // faucet6 if (_func == 1) {faucet()కు కాల్, దీనికి పారామితులు లేవు.
1 token.faucet();2 token.transfer(msg.sender,3 token.balanceOf(address(this)));4 }మనం token.faucet()ని కాల్ చేసిన తర్వాత మనకు టోకెన్లు వస్తాయి. అయితే, ప్రాక్సీ కాంట్రాక్ట్గా, మాకు టోకెన్లు అవసరం లేదు.
మనల్ని పిలిచిన EOA (బాహ్యంగా యాజమాన్యంలోని ఖాతా) లేదా కాంట్రాక్ట్కు అవసరం.
కాబట్టి మనం మన టోకెన్లన్నింటినీ మనల్ని పిలిచిన వారికి బదిలీ చేస్తాము.
1 // transfer (assume we have an allowance for it)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 బైట్ల కాల్డేటా పడుతుంది:
| విభాగం | పొడవు | బైట్లు |
|---|---|---|
| ఫంక్షన్ సెలెక్టర్ | 1 | 0 |
| గమ్యస్థాన చిరునామా | 32 | 1-32 |
| మొత్తం | 2 | 33-34 |
1 } // fallback23} // contract CalldataInterpretertest.js
ఈ జావాస్క్రిప్ట్ యూనిట్ టెస్ట్opens in a new tab ఈ మెకానిజంను ఎలా ఉపయోగించాలో (మరియు అది సరిగ్గా పనిచేస్తుందో లేదో ఎలా ధృవీకరించాలో) చూపిస్తుంది. మీకు chaiopens in a new tab మరియు ethersopens in a new tab గురించి అర్థమైందని నేను భావిస్తున్నాను మరియు కాంట్రాక్ట్కు ప్రత్యేకంగా వర్తించే భాగాలను మాత్రమే వివరిస్తాను.
1const { expect } = require("chai");23describe("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)910 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)1415 const signer = await ethers.getSigner()అన్నీ చూపించురెండు కాంట్రాక్టులను డిప్లోయ్ చేయడం ద్వారా మేము ప్రారంభిస్తాము.
1 // Get tokens to play with2 const faucetTx = {లావాదేవీలను సృష్టించడానికి మేము సాధారణంగా ఉపయోగించే ఉన్నత-స్థాయి ఫంక్షన్లను (ఉదాహరణకు token.faucet()) ఉపయోగించలేము, ఎందుకంటే మేము ABIని అనుసరించడం లేదు.
బదులుగా, మనమే లావాదేవీని నిర్మించి, ఆపై పంపాలి.
1 to: cdi.address,2 data: "0x01"లావాదేవీ కోసం మనం అందించాల్సిన రెండు పారామితులు ఉన్నాయి:
to, గమ్యస్థాన చిరునామా. ఇది కాల్డేటా ఇంటర్ప్రెటర్ కాంట్రాక్ట్.data, పంపాల్సిన కాల్డేటా. ఫాసెట్ కాల్ విషయంలో, డేటా ఒకే బైట్,0x01.
12 }3 await (await signer.sendTransaction(faucetTx)).wait()మేము సైనర్ యొక్క sendTransaction పద్ధతినిopens in a new tab కాల్ చేస్తాము ఎందుకంటే మేము ఇప్పటికే గమ్యాన్ని (faucetTx.to) పేర్కొన్నాము మరియు లావాదేవీపై సంతకం చేయాలి.
1// Check the faucet provides the tokens correctly2expect(await token.balanceOf(signer.address)).to.equal(1000)ఇక్కడ మేము బ్యాలెన్స్ని ధృవీకరిస్తాము.
view ఫంక్షన్లపై గ్యాస్ ఆదా చేయాల్సిన అవసరం లేదు, కాబట్టి మేము వాటిని సాధారణంగా నడుపుతాము.
1// Give the CDI an allowance (approvals cannot be proxied)2const approveTX = await token.approve(cdi.address, 10000)3await approveTX.wait()4expect(await token.allowance(signer.address, cdi.address)).to.equal(10000)బదిలీలు చేయడానికి కాల్డేటా ఇంటర్ప్రెటర్కు అనుమతి ఇవ్వండి.
1// Transfer tokens2const 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()23 // Check that we have 256 tokens less4 expect (await token.balanceOf(signer.address)).to.equal(1000-256)56 // And that our destination got them7 expect (await token.balanceOf(destAddr)).to.equal(256)8 }) // it9}) // describeఅన్నీ చూపించుమీరు గమ్యస్థాన కాంట్రాక్ట్ను నియంత్రించినప్పుడు ఖర్చులను తగ్గించడం
మీరు గమ్యస్థాన కాంట్రాక్ట్పై నియంత్రణ కలిగి ఉంటే, మీరు కాల్డేటా ఇంటర్ప్రెటర్ను విశ్వసించే కారణంగా msg.sender తనిఖీలను దాటవేసే ఫంక్షన్లను సృష్టించవచ్చు.
ఇది ఎలా పనిచేస్తుందో మీరు ఇక్కడ ఒక ఉదాహరణను control-contract బ్రాంచ్లో చూడవచ్చుopens in a new tab.
కాంట్రాక్ట్ బాహ్య లావాదేవీలకు మాత్రమే ప్రతిస్పందిస్తున్నట్లయితే, మనం కేవలం ఒక కాంట్రాక్ట్తో సరిపెట్టుకోవచ్చు. అయితే, అది కంపోజబిలిటీని దెబ్బతీస్తుంది. సాధారణ ERC-20 కాల్స్కు ప్రతిస్పందించే కాంట్రాక్ట్, మరియు చిన్న కాల్ డేటాతో లావాదేవీలకు ప్రతిస్పందించే మరొక కాంట్రాక్ట్ ఉండటం చాలా మంచిది.
Token.sol
ఈ ఉదాహరణలో మనం Token.solని మార్చవచ్చు.
ఇది ప్రాక్సీ మాత్రమే కాల్ చేయగల అనేక ఫంక్షన్లను కలిగి ఉండటానికి మాకు అనుమతిస్తుంది.
కొత్త భాగాలు ఇక్కడ ఉన్నాయి:
1 // The only address allowed to specify the CalldataInterpreter address2 address owner;34 // The CalldataInterpreter address5 address proxy = address(0);ERC-20 కాంట్రాక్ట్కు అధీకృత ప్రాక్సీ యొక్క గుర్తింపు తెలియాలి. అయితే, మనం ఈ వేరియబుల్ని కన్స్ట్రక్టర్లో సెట్ చేయలేము, ఎందుకంటే మనకు ఇంకా విలువ తెలియదు. ఈ కాంట్రాక్ట్ మొదట ఇన్స్టాన్షియేట్ చేయబడింది ఎందుకంటే ప్రాక్సీ దాని కన్స్ట్రక్టర్లో టోకెన్ చిరునామాను ఆశిస్తుంది.
1 /**2 * @dev Calls the ERC20 constructor.3 */4 constructor(5 ) ERC20("Oris useless token-2", "OUT-2") {6 owner = msg.sender;7 }సృష్టికర్త యొక్క చిరునామా (దీనిని owner అని పిలుస్తారు) ఇక్కడ నిల్వ చేయబడుతుంది ఎందుకంటే ప్రాక్సీని సెట్ చేయడానికి అనుమతించబడిన ఏకైక చిరునామా అదే.
1 /**2 * @dev set the address for the proxy (the CalldataInterpreter).3 * Can only be called once by the owner4 */5 function setProxy(address _proxy) external {6 require(msg.sender == owner, "Can only be called by owner");7 require(proxy == address(0), "Proxy is already set");89 proxy = _proxy;10 } // function setProxyఅన్నీ చూపించుప్రాక్సీకి విశేష ప్రాప్యత ఉంది, ఎందుకంటే ఇది భద్రతా తనిఖీలను దాటవేయగలదు.
మేము ప్రాక్సీని విశ్వసించగలమని నిర్ధారించుకోవడానికి మేము owner ని మాత్రమే ఈ ఫంక్షన్ను కాల్ చేయడానికి అనుమతిస్తాము మరియు ఒక్కసారి మాత్రమే.
proxy కి నిజమైన విలువ (సున్నా కాదు) ఒకసారి ఉంటే, ఆ విలువ మారదు, కాబట్టి యజమాని మోసపూరితంగా మారాలని నిర్ణయించుకున్నా, లేదా దాని కోసం స్మృతిచిహ్నం వెల్లడైనా, మేము ఇప్పటికీ సురక్షితంగా ఉన్నాము.
1 /**2 * @dev Some functions may only be called by the proxy.3 */4 modifier onlyProxy {ఇది ఒక modifier ఫంక్షన్opens in a new tab, ఇది ఇతర ఫంక్షన్లు పనిచేసే విధానాన్ని మారుస్తుంది.
1 require(msg.sender == proxy);మొదట, మేము ప్రాక్సీ ద్వారా పిలువబడ్డామని మరియు మరెవరూ కాదని ధృవీకరించండి.
కాకపోతే, revert చేయండి.
1 _;2 }అలా అయితే, మేము సవరించే ఫంక్షన్ను అమలు చేయండి.
1 /* Functions that allow the proxy to actually proxy for accounts */23 function transferProxy(address from, address to, uint256 amount)4 public virtual onlyProxy() returns (bool)5 {6 _transfer(from, to, amount);7 return true;8 }910 function approveProxy(address from, address spender, uint256 amount)11 public virtual onlyProxy() returns (bool)12 {13 _approve(from, spender, amount);14 return true;15 }1617 function transferFromProxy(18 address spender,19 address from,20 address to,21 uint256 amount22 ) public virtual onlyProxy() returns (bool)23 {24 _spendAllowance(from, spender, amount);25 _transfer(from, to, amount);26 return true;27 }అన్నీ చూపించుఇవి మూడు కార్యకలాపాలు, వీటికి సాధారణంగా టోకెన్లను బదిలీ చేసే లేదా అనుమతిని ఆమోదించే సంస్థ నుండి నేరుగా సందేశం రావాలి. ఇక్కడ ఈ కార్యకలాపాల యొక్క ప్రాక్సీ వెర్షన్ ఉంది:
onlyProxy()ద్వారా సవరించబడింది, కాబట్టి మరెవరూ వాటిని నియంత్రించడానికి అనుమతించబడరు.- సాధారణంగా
msg.senderగా ఉండే చిరునామాను అదనపు పారామీటర్గా పొందుతుంది.
CalldataInterpreter.sol
కాల్డేటా ఇంటర్ప్రెటర్ పైన ఉన్న దానితో దాదాపు సమానంగా ఉంటుంది, ప్రాక్సీ చేయబడిన ఫంక్షన్లు msg.sender పారామీటర్ను స్వీకరిస్తాయి మరియు ట్రాన్స్ఫర్ కోసం అనుమతి అవసరం లేదు.
1 // transfer (no need for allowance)2 if (_func == 2) {3 token.transferProxy(4 msg.sender,5 address(uint160(calldataVal(1, 20))),6 calldataVal(21, 2)7 );8 }910 // approve11 if (_func == 3) {12 token.approveProxy(13 msg.sender,14 address(uint160(calldataVal(1, 20))),15 calldataVal(21, 2)16 );17 }1819 // transferFrom20 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)23// Need two signers to verify allowances4const signers = await ethers.getSigners()5const signer = signers[0]6const poorSigner = signers[1]approve() మరియు transferFrom() ని తనిఖీ చేయడానికి మనకు రెండవ సైనర్ అవసరం.
మేము దానిని poorSigner అని పిలుస్తాము ఎందుకంటే దానికి మా టోకెన్లు ఏవీ రావు (దానికి ETH ఉండాలి, వాస్తవానికి).
1// Transfer tokens2const 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// approval and transferFrom2const approveTx = {3 to: cdi.address,4 data: "0x03" + poorSigner.address.slice(2, 42) + "00FF",5}6await (await signer.sendTransaction(approveTx)).wait()78const destAddr2 = "0xE1165C689C0c3e9642cA7606F5287e708d846206"910const 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()1516// Check the approve / transferFrom combo was done correctly17expect(await token.balanceOf(destAddr2)).to.equal(255)అన్నీ చూపించురెండు కొత్త ఫంక్షన్లను పరీక్షించండి.
transferFromTx కి రెండు చిరునామా పారామితులు అవసరమని గమనించండి: అనుమతి ఇచ్చేవాడు మరియు గ్రహీత.
ముగింపు
Optimismopens in a new tab మరియు Arbitrumopens in a new tab రెండూ L1కి వ్రాయబడిన కాల్డేటా పరిమాణాన్ని మరియు అందువల్ల లావాదేవీల ఖర్చును తగ్గించే మార్గాలను అన్వేషిస్తున్నాయి. అయితే, సాధారణ పరిష్కారాల కోసం చూస్తున్న మౌలిక సదుపాయాల ప్రదాతలుగా, మా సామర్థ్యాలు పరిమితంగా ఉన్నాయి. డాప్స్ డెవలపర్గా, మీకు అప్లికేషన్-నిర్దిష్ట పరిజ్ఞానం ఉంది, ఇది ఒక సాధారణ పరిష్కారంలో మేము చేయగలిగిన దానికంటే మెరుగ్గా మీ కాల్డేటాను ఆప్టిమైజ్ చేయడానికి మిమ్మల్ని అనుమతిస్తుంది. ఈ ఆర్టికల్ మీ అవసరాలకు ఆదర్శవంతమైన పరిష్కారాన్ని కనుగొనడంలో మీకు సహాయపడుతుందని ఆశిస్తున్నాము.
నా మరిన్ని పనుల కోసం ఇక్కడ చూడండిopens in a new tab.
పేజీ చివరి అప్డేట్: 22 ఆగస్టు, 2025