মূল কন্টেন্টে যান

কলডাটা অপ্টিমাইজেশনের জন্য শর্ট ABI

লেয়ার ২
ইন্টারমিডিয়েট
ওরি পোমেরান্টজ
১ এপ্রিল, ২০২২
14 মিনিট পড়া

ভূমিকা

এই আর্টিকেলে, আপনি অপ্টিমেস্টিক রোলআপ, সেগুলোতে লেনদেন এর খরচ এবং কীভাবে সেই ভিন্ন খরচের কাঠামোর কারণে ইথিরিয়াম মেইননেট এর চেয়ে ভিন্ন কিছুর জন্য আমাদের অপ্টিমাইজ করতে হয় সে সম্পর্কে জানবেন। আপনি কীভাবে এই অপ্টিমাইজেশন বাস্তবায়ন করতে হয় তাও শিখবেন।

পূর্ণ প্রকাশ

আমি 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 লেনদেন এর খরচ দুটি উপাদানের সমন্বয়ে গঠিত:

  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
শূন্য (Zeroes)124-15124800
গন্তব্য এডড্রেস2016-350020320
পরিমাণ3236-67176415240
মোট68160576

ব্যাখ্যা:

  • ফাংশন সিলেক্টর: কন্ট্রাক্টে 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 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 টোকেন অ্যাড্রেস নির্দিষ্ট করুন
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;
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-এর অ্যাসেম্বলি ভাষা 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} // contract CalldataInterpreter

test.js

এই জাভাস্ক্রিপ্ট ইউনিট টেস্টটি (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, পাঠানোর জন্য কলডাটা। একটি faucet কলের ক্ষেত্রে, ডাটা হলো একটি একক বাইট, 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 }) // it
9}) // describe
সব দেখান

গন্তব্য কন্ট্রাক্ট আপনার নিয়ন্ত্রণে থাকলে খরচ কমানো

যদি গন্তব্য কন্ট্রাক্টের ওপর আপনার নিয়ন্ত্রণ থাকে তবে আপনি এমন ফাংশন তৈরি করতে পারেন যা 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 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");
7
8 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 /* যে ফাংশনগুলো প্রক্সিকে অ্যাকাউন্টগুলোর জন্য প্রকৃতপক্ষে প্রক্সি করার অনুমতি দেয় */
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 // 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// অ্যালাউন্স যাচাই করার জন্য দুজন সাইনার প্রয়োজন
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 এবং 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-এ লেখা কলডাটার আকার কমানোর এবং এর ফলে লেনদেন এর খরচ কমানোর উপায় খুঁজছে। তবে, জেনেরিক সমাধান খোঁজা ইনফ্রাস্ট্রাকচার প্রোভাইডার হিসেবে, আমাদের ক্ষমতা সীমিত। ডিএ্যাপ ডেভেলপার হিসেবে, আপনার কাছে অ্যাপ্লিকেশন-নির্দিষ্ট জ্ঞান রয়েছে, যা আপনাকে একটি জেনেরিক সমাধানের চেয়ে অনেক ভালোভাবে আপনার কলডাটা অপ্টিমাইজ করতে দেয়। আশা করি, এই আর্টিকেলটি আপনাকে আপনার প্রয়োজনের জন্য আদর্শ সমাধান খুঁজে পেতে সাহায্য করবে।

আমার আরও কাজের জন্য এখানে দেখুন (opens in a new tab)

পেজ সর্বশেষ আপডেট: ৩ মার্চ, ২০২৬

এই টিউটোরিয়ালটি কি সহায়ক ছিল?