কলডাটা অপ্টিমাইজেশনের জন্য শর্ট ABI
ভূমিকা
এই আর্টিকেলে, আপনি অপ্টিমেস্টিক রোলআপ, সেগুলোতে লেনদেন এর খরচ এবং কীভাবে সেই ভিন্ন খরচের কাঠামোর কারণে ইথিরিয়াম মেইননেট এর চেয়ে ভিন্ন কিছুর জন্য আমাদের অপ্টিমাইজ করতে হয় সে সম্পর্কে জানবেন। আপনি কীভাবে এই অপ্টিমাইজেশন বাস্তবায়ন করতে হয় তাও শিখবেন।
পূর্ণ প্রকাশ
আমি Optimism (opens in a new tab)-এর একজন পূর্ণকালীন কর্মী, তাই এই আর্টিকেলের উদাহরণগুলো Optimism-এ চলবে। তবে, এখানে ব্যাখ্যা করা কৌশলটি অন্যান্য রোলআপ এর জন্যও একইভাবে কাজ করা উচিত।
পরিভাষা
রোলআপ নিয়ে আলোচনা করার সময়, 'layer 1' (L1) শব্দটি মেইননেট, অর্থাৎ প্রোডাকশন ইথিরিয়াম নেটওয়ার্কের জন্য ব্যবহৃত হয়। 'লেয়ার ২' (L2) শব্দটি রোলআপ বা অন্য যেকোনো সিস্টেমের জন্য ব্যবহৃত হয় যা নিরাপত্তার জন্য L1-এর ওপর নির্ভর করে কিন্তু এর বেশিরভাগ প্রসেসিং অফচেইন করে।
আমরা কীভাবে L2 লেনদেন এর খরচ আরও কমাতে পারি?
অপ্টিমেস্টিক রোলআপ-কে প্রতিটি ঐতিহাসিক লেনদেন এর রেকর্ড সংরক্ষণ করতে হয় যাতে যেকেউ সেগুলো যাচাই করে দেখতে পারে যে বর্তমান স্টেট সঠিক কিনা। ইথিরিয়াম মেইননেট-এ ডাটা পাঠানোর সবচেয়ে সস্তা উপায় হলো এটিকে কলডাটা হিসেবে লেখা। এই সমাধানটি Optimism (opens in a new tab) এবং Arbitrum (opens 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 |
| শূন্য (Zeroes) | 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)।
- শূন্য (Zeroes): এই বাইটগুলো সবসময় শূন্য হয় কারণ একটি বিশ-বাইট এডড্রেস ধারণ করার জন্য বত্রিশ-বাইট শব্দের প্রয়োজন হয় না। শূন্য ধারণকারী বাইটগুলোর খরচ চার গ্যাস (ইয়েলো পেপার দেখুন (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 function faucet() external {4 _mint(msg.sender, 1000);5 } // 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 টোকেন অ্যাড্রেস নির্দিষ্ট করুন4 * @param tokenAddr_ ERC-20 কন্ট্রাক্ট অ্যাড্রেস */5 constructor(6 address tokenAddr_7 ) {8 token = OrisUselessToken(tokenAddr_);9 } // কনস্ট্রাক্টরসব দেখানটোকেন এডড্রেস হলো একমাত্র প্যারামিটার যা আমাদের নির্দিষ্ট করতে হবে।
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-এর অ্যাসেম্বলি ভাষা 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>...)।
12 _retVal = _retVal >> (256-length*8);শুধুমাত্র সবচেয়ে গুরুত্বপূর্ণ length বাইটগুলো ফিল্ডের অংশ, তাই আমরা অন্যান্য মানগুলো থেকে মুক্তি পেতে রাইট-শিফট (opens in a new tab) করি। এর একটি অতিরিক্ত সুবিধা হলো এটি মানটিকে ফিল্ডের ডানদিকে সরিয়ে দেয়, তাই এটি মান গুণ 256something হওয়ার পরিবর্তে মানটি নিজেই হয়।
12 return _retVal;3 }456 fallback() external {যখন কোনো Solidity কন্ট্রাক্টে কল করা কোনো ফাংশন সিগনেচারের সাথে মেলে না, তখন এটি fallback() ফাংশন (opens in a new tab) কল করে (ধরে নিচ্ছি যে একটি আছে)। CalldataInterpreter-এর ক্ষেত্রে, যেকোনো কল এখানে আসে কারণ অন্য কোনো external বা public ফাংশন নেই।
1 uint _func;23 _func = calldataVal(0, 1);কলডাটার প্রথম বাইটটি পড়ুন, যা আমাদের ফাংশনটি বলে দেয়। এখানে একটি ফাংশন উপলব্ধ না হওয়ার দুটি কারণ রয়েছে:
- যে ফাংশনগুলো
pureবাviewসেগুলো স্টেট পরিবর্তন করে না এবং গ্যাস খরচ করে না (যখন অফচেইন কল করা হয়)। তাদের গ্যাস খরচ কমানোর চেষ্টা করার কোনো মানে হয় না। - যে ফাংশনগুলো
msg.sender(opens in a new tab)-এর ওপর নির্ভর করে।msg.sender-এর মান কলারের পরিবর্তেCalldataInterpreter-এর এডড্রেস হতে যাচ্ছে।
দুর্ভাগ্যবশত, ERC-20 স্পেসিফিকেশনগুলো দেখলে (opens in a new tab), এটি শুধুমাত্র একটি ফাংশন, transfer বাকি রাখে। এটি আমাদের কাছে শুধুমাত্র দুটি ফাংশন রাখে: transfer (কারণ আমরা transferFrom কল করতে পারি) এবং faucet (কারণ আমরা যে আমাদের কল করেছে তাকে টোকেনগুলো ফেরত পাঠাতে পারি)।
12 // টোকেনের স্টেট পরিবর্তনকারী মেথডগুলো কল করুন ব্যবহার করে3 // কলডেটা থেকে প্রাপ্ত তথ্য45 // ফসেট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 বাইট কলডাটা লাগে:
| সেকশন | দৈর্ঘ্য | বাইট |
|---|---|---|
| ফাংশন সিলেক্টর | 1 | 0 |
| গন্তব্য এডড্রেস | 32 | 1-32 |
| পরিমাণ | 2 | 33-34 |
1 } // ফলব্যাক23} // contract CalldataInterpretertest.js
এই জাভাস্ক্রিপ্ট ইউনিট টেস্টটি (opens in a new tab) আমাদের দেখায় কীভাবে এই মেকানিজমটি ব্যবহার করতে হয় (এবং এটি সঠিকভাবে কাজ করে কিনা তা কীভাবে যাচাই করতে হয়)। আমি ধরে নিচ্ছি আপনি chai (opens in a new tab) এবং ethers (opens 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 // খেলার জন্য টোকেন পান2 const faucetTx = {লেনদেন তৈরি করতে আমরা সাধারণত যে হাই-লেভেল ফাংশনগুলো ব্যবহার করি (যেমন token.faucet()) তা ব্যবহার করতে পারি না, কারণ আমরা ABI অনুসরণ করি না। এর পরিবর্তে, আমাদের নিজেদেরই লেনদেন তৈরি করতে হবে এবং তারপর এটি পাঠাতে হবে।
1 to: cdi.address,2 data: "0x01"লেনদেন এর জন্য আমাদের দুটি প্যারামিটার প্রদান করতে হবে:
to, গন্তব্য এডড্রেস। এটি হলো কলডাটা ইন্টারপ্রেটার কন্ট্রাক্ট।data, পাঠানোর জন্য কলডাটা। একটি faucet কলের ক্ষেত্রে, ডাটা হলো একটি একক বাইট,0x01।
12 }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()23 // চেক করুন যে আমাদের কাছে 256টি টোকেন কম আছে4 expect (await token.balanceOf(signer.address)).to.equal(1000-256)56 // এবং আমাদের গন্তব্য সেগুলো পেয়েছে কিনা7 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 // একমাত্র অ্যাড্রেস যা CalldataInterpreter অ্যাড্রেস নির্দিষ্ট করার অনুমতিপ্রাপ্ত2 address owner;34 // CalldataInterpreter অ্যাড্রেস5 address proxy = address(0);ERC-20 কন্ট্রাক্টকে অনুমোদিত প্রক্সির পরিচয় জানতে হবে। তবে, আমরা কনস্ট্রাক্টরে এই ভেরিয়েবলটি সেট করতে পারি না, কারণ আমরা এখনও মানটি জানি না। এই কন্ট্রাক্টটি প্রথমে ইনস্ট্যানশিয়েট করা হয় কারণ প্রক্সি তার কনস্ট্রাক্টরে টোকেনের এডড্রেস আশা করে।
1 /* *2 * @dev ERC20 কনস্ট্রাক্টর কল করে। */3 constructor(4 ) ERC20("Oris useless token-2", "OUT-2") {5 owner = msg.sender;6 }ক্রিয়েটরের এডড্রেস (যাকে owner বলা হয়) এখানে সংরক্ষণ করা হয় কারণ শুধুমাত্র এই এডড্রেসটিকেই প্রক্সি সেট করার অনুমতি দেওয়া হয়।
1 /* *2 * @dev প্রক্সির জন্য অ্যাড্রেস সেট করুন (CalldataInterpreter)।3 * শুধুমাত্র ওনার দ্বারা একবার কল করা যেতে পারে */4 function setProxy(address _proxy) external {5 require(msg.sender == owner, "Can only be called by owner");6 require(proxy == address(0), "Proxy is already set");78 proxy = _proxy;9 } // function setProxyসব দেখানপ্রক্সির প্রিভিলেজড অ্যাক্সেস রয়েছে, কারণ এটি সিকিউরিটি চেক বাইপাস করতে পারে। আমরা প্রক্সিকে বিশ্বাস করতে পারি তা নিশ্চিত করতে আমরা শুধুমাত্র owner-কে এই ফাংশনটি কল করতে দিই, এবং তাও শুধুমাত্র একবার। একবার proxy-এর একটি বাস্তব মান (শূন্য নয়) হয়ে গেলে, সেই মানটি পরিবর্তন করা যায় না, তাই যদি মালিক প্রতারক হওয়ার সিদ্ধান্ত নেয়, বা এর নেমোনিক প্রকাশ হয়ে যায়, তবুও আমরা নিরাপদ থাকি।
1 /* *2 * @dev কিছু ফাংশন শুধুমাত্র প্রক্সি দ্বারা কল করা যেতে পারে। */3 modifier onlyProxy {এটি একটি modifier ফাংশন (opens in a new tab), এটি অন্যান্য ফাংশনগুলোর কাজ করার পদ্ধতি পরিবর্তন করে।
1 require(msg.sender == proxy);প্রথমে, যাচাই করুন যে আমাদের প্রক্সি কল করেছে এবং অন্য কেউ নয়। যদি না হয়, তবে revert করুন।
1 _;2 }যদি তাই হয়, তবে আমরা যে ফাংশনটি মডিফাই করি সেটি চালান।
1 /* যে ফাংশনগুলো প্রক্সিকে অ্যাকাউন্টগুলোর জন্য প্রকৃতপক্ষে প্রক্সি করার অনুমতি দেয় */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 প্যারামিটার গ্রহণ করে এবং transfer-এর জন্য কোনো অ্যালাউন্সের প্রয়োজন নেই।
1 // ট্রান্সফার (অ্যালাউন্সের কোনো প্রয়োজন নেই)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// অ্যালাউন্স যাচাই করার জন্য দুজন সাইনার প্রয়োজন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// approval এবং 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// approve / transferFrom কম্বো সঠিকভাবে করা হয়েছে কিনা তা চেক করুন17expect(await token.balanceOf(destAddr2)).to.equal(255)সব দেখানদুটি নতুন ফাংশন টেস্ট করুন। মনে রাখবেন যে transferFromTx-এর জন্য দুটি এডড্রেস প্যারামিটার প্রয়োজন: অ্যালাউন্স প্রদানকারী এবং গ্রহণকারী।
উপসংহার
Optimism (opens in a new tab) এবং Arbitrum (opens in a new tab) উভয়েই L1-এ লেখা কলডাটার আকার কমানোর এবং এর ফলে লেনদেন এর খরচ কমানোর উপায় খুঁজছে। তবে, জেনেরিক সমাধান খোঁজা ইনফ্রাস্ট্রাকচার প্রোভাইডার হিসেবে, আমাদের ক্ষমতা সীমিত। ডিএ্যাপ ডেভেলপার হিসেবে, আপনার কাছে অ্যাপ্লিকেশন-নির্দিষ্ট জ্ঞান রয়েছে, যা আপনাকে একটি জেনেরিক সমাধানের চেয়ে অনেক ভালোভাবে আপনার কলডাটা অপ্টিমাইজ করতে দেয়। আশা করি, এই আর্টিকেলটি আপনাকে আপনার প্রয়োজনের জন্য আদর্শ সমাধান খুঁজে পেতে সাহায্য করবে।
আমার আরও কাজের জন্য এখানে দেখুন (opens in a new tab)।
পেজ সর্বশেষ আপডেট: ৩ মার্চ, ২০২৬