ప్రధాన కంటెంట్‌కి స్కిప్ చేయండి

కాల్‌డేటా ఆప్టిమైజేషన్ కోసం చిన్న ABIలు

లేయర్ 2
మధ్యస్థ
Ori Pomerantz
1 ఏప్రిల్, 2022
12 నిమిషం పఠనం

పరిచయం

ఈ ఆర్టికల్‌లో, మీరు ఆప్టిమిస్టిక్ రోలప్‌లు, వాటిపై లావాదేవీల ఖర్చు, మరియు 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 లావాదేవీల ఖర్చు రెండు భాగాలతో కూడి ఉంటుంది:

  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, 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 with
3 */
4 function faucet() external {
5 _mint(msg.sender, 1000);
6 } // function faucet

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 Specify the token address
4 * @param tokenAddr_ ERC-20 contract address
5 */
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;
2
3 require(length < 0x21,
4 "calldataVal length limit is 32 bytes");
5
6 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>...).

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 కాంట్రాక్ట్‌కు చేసిన కాల్ ఏ ఫంక్షన్ సంతకాలతో సరిపోలకపోతే, అది the fallback() ఫంక్షన్opens in a new tabను కాల్ చేస్తుంది (ఒకటి ఉందని ఊహిస్తూ). CalldataInterpreter విషయంలో, కాల్ అయినా ఇక్కడికి వస్తుంది ఎందుకంటే ఇతర external లేదా public ఫంక్షన్‌లు లేవు.

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

కాల్‌డేటా యొక్క మొదటి బైట్‌ను చదవండి, ఇది మనకు ఫంక్షన్‌ను తెలియజేస్తుంది. ఇక్కడ ఒక ఫంక్షన్ అందుబాటులో లేకపోవడానికి రెండు కారణాలు ఉన్నాయి:

  1. pure లేదా viewగా ఉండే ఫంక్షన్‌లు స్థితిని మార్చవు మరియు గ్యాస్ ఖర్చు చేయవు (ఆఫ్‌చెయిన్‌లో కాల్ చేసినప్పుడు). వాటి గ్యాస్ ధరను తగ్గించడానికి ప్రయత్నించడంలో అర్థం లేదు.
  2. msg.senderopens in a new tabపై ఆధారపడే ఫంక్షన్‌లు. msg.sender యొక్క విలువ కాలర్ కాదు, CalldataInterpreter యొక్క చిరునామా అవుతుంది.

దురదృష్టవశాత్తు, ERC-20 స్పెసిఫికేషన్‌లను చూస్తేopens in a new tab, ఇది కేవలం ఒక ఫంక్షన్, transferను మాత్రమే వదిలివేస్తుంది. ఇది మనకు కేవలం రెండు ఫంక్షన్‌లను మాత్రమే మిగిల్చింది: transfer (ఎందుకంటే మనం transferFromని కాల్ చేయవచ్చు) మరియు faucet (ఎందుకంటే మనల్ని పిలిచిన వారికి టోకెన్‌లను తిరిగి బదిలీ చేయవచ్చు).

1
2 // Call the state changing methods of token using
3 // information from the calldata
4
5 // faucet
6 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 బైట్ల కాల్‌డేటా పడుతుంది:

విభాగంపొడవుబైట్‌లు
ఫంక్షన్ సెలెక్టర్10
గమ్యస్థాన చిరునామా321-32
మొత్తం233-34
1 } // fallback
2
3} // contract CalldataInterpreter

test.js

ఈ జావాస్క్రిప్ట్ యూనిట్ టెస్ట్opens in a new tab ఈ మెకానిజంను ఎలా ఉపయోగించాలో (మరియు అది సరిగ్గా పనిచేస్తుందో లేదో ఎలా ధృవీకరించాలో) చూపిస్తుంది. మీకు chaiopens in a new tab మరియు ethersopens 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 // Get tokens to play with
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// Check the faucet provides the tokens correctly
2expect(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 tokens
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 // Check that we have 256 tokens less
4 expect (await token.balanceOf(signer.address)).to.equal(1000-256)
5
6 // And that our destination got them
7 expect (await token.balanceOf(destAddr)).to.equal(256)
8 }) // it
9}) // 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 address
2 address owner;
3
4 // The CalldataInterpreter address
5 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 owner
4 */
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");
8
9 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 */
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 పారామీటర్‌ను స్వీకరిస్తాయి మరియు ట్రాన్స్‌ఫర్ కోసం అనుమతి అవసరం లేదు.

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 }
9
10 // approve
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// Need two signers to verify allowances
4const signers = await ethers.getSigners()
5const signer = signers[0]
6const poorSigner = signers[1]

approve() మరియు transferFrom() ని తనిఖీ చేయడానికి మనకు రెండవ సైనర్ అవసరం. మేము దానిని poorSigner అని పిలుస్తాము ఎందుకంటే దానికి మా టోకెన్లు ఏవీ రావు (దానికి ETH ఉండాలి, వాస్తవానికి).

1// Transfer tokens
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// approval and 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// Check the approve / transferFrom combo was done correctly
17expect(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

ఈ ట్యుటోరియల్ ఉపయోగపడిందా?