Uniswap-v2 কন্ট্রাক্ট ওয়াক-থ্রু
ভূমিকা
Uniswap v2 (opens in a new tab) যেকোনো দুটি ERC-20 টোকেনের মধ্যে একটি এক্সচেঞ্জ মার্কেট তৈরি করতে পারে। এই আর্টিকেলে আমরা এই প্রটোকলটি ইমপ্লিমেন্ট করা কন্ট্রাক্টগুলোর সোর্স কোড নিয়ে আলোচনা করব এবং দেখব কেন সেগুলো এভাবে লেখা হয়েছে।
Uniswap কী করে?
মূলত, দুই ধরনের ব্যবহারকারী রয়েছে: লিকুইডিটি প্রোভাইডার এবং ট্রেডার।
লিকুইডিটি প্রোভাইডাররা পুলটিতে দুটি টোকেন প্রদান করে যা এক্সচেঞ্জ করা যায় (আমরা সেগুলোকে Token0 এবং Token1 বলব)। এর বিনিময়ে, তারা তৃতীয় একটি টোকেন পায় যা পুলের আংশিক মালিকানাকে উপস্থাপন করে, যাকে লিকুইডিটি টোকেন বলা হয়।
ট্রেডাররা পুলে এক ধরনের টোকেন পাঠায় এবং লিকুইডিটি প্রোভাইডারদের দেওয়া পুল থেকে অন্যটি গ্রহণ করে (উদাহরণস্বরূপ, Token0 পাঠায় এবং Token1 গ্রহণ করে)। এক্সচেঞ্জ রেট নির্ধারিত হয় পুলে থাকা Token0 এবং Token1 এর আপেক্ষিক সংখ্যার ওপর ভিত্তি করে। এছাড়া, পুলটি লিকুইডিটি পুলের জন্য রিওয়ার্ড হিসেবে একটি ছোট শতাংশ কেটে নেয়।
লিকুইডিটি প্রোভাইডাররা যখন তাদের সম্পদ ফেরত পেতে চায়, তখন তারা পুল টোকেনগুলো বার্ন করতে পারে এবং রিওয়ার্ডের অংশসহ তাদের টোকেনগুলো ফেরত পেতে পারে।
আরও বিস্তারিত বর্ণনার জন্য এখানে ক্লিক করুন (opens in a new tab)।
কেন v2? কেন v3 নয়?
Uniswap v3 (opens in a new tab) হলো একটি আপগ্রেড যা v2 এর চেয়ে অনেক বেশি জটিল। প্রথমে v2 শেখা এবং তারপর v3 তে যাওয়া বেশি সহজ।
কোর কন্ট্রাক্ট বনাম পেরিফেরি কন্ট্রাক্ট
Uniswap v2 দুটি উপাদানে বিভক্ত, একটি কোর এবং একটি পেরিফেরি। এই বিভাজন কোর কন্ট্রাক্টগুলোকে, যা সম্পদ ধারণ করে এবং তাই সেগুলোকে সুরক্ষিত হতেই হবে, আরও সহজ এবং অডিট করার জন্য সুবিধাজনক করে তোলে। ট্রেডারদের জন্য প্রয়োজনীয় সমস্ত অতিরিক্ত কার্যকারিতা এরপর পেরিফেরি কন্ট্রাক্টগুলোর মাধ্যমে প্রদান করা যেতে পারে।
ডাটা এবং কন্ট্রোল ফ্লো
ইউনিসোয়াপের (Uniswap) তিনটি প্রধান কাজ করার সময় ডাটা এবং কন্ট্রোলের ফ্লো যেভাবে কাজ করে:
- বিভিন্ন টোকেনের মধ্যে সোয়াপ করা
- মার্কেটে লিকুইডিটি যোগ করা এবং পেয়ার এক্সচেঞ্জ ERC-20 লিকুইডিটি টোকেন দিয়ে পুরস্কৃত হওয়া
- ERC-20 লিকুইডিটি টোকেন বার্ন করা এবং সেই ERC-20 টোকেনগুলো ফেরত পাওয়া যা পেয়ার এক্সচেঞ্জ ট্রেডারদের এক্সচেঞ্জ করার অনুমতি দেয়
সোয়াপ
এটি সবচেয়ে সাধারণ ফ্লো, যা ট্রেডাররা ব্যবহার করে থাকেন:
কলার
- পেরিফেরি একাউন্টকে সোয়াপ করার পরিমাণের একটি এলাউন্স প্রদান করুন।
- পেরিফেরি কন্ট্রাক্টের অনেকগুলো সোয়াপ ফাংশনের মধ্যে একটি কল করুন (কোনটি কল করবেন তা নির্ভর করে ETH জড়িত আছে কি না, ট্রেডার জমা দেওয়ার টোকেনের পরিমাণ বা ফেরত পাওয়ার টোকেনের পরিমাণ নির্দিষ্ট করেছেন কি না, ইত্যাদির ওপর)।
প্রতিটি সোয়াপ ফাংশন একটি
pathগ্রহণ করে, যা হলো অতিক্রম করার জন্য এক্সচেঞ্জগুলোর একটি অ্যারে।
পেরিফেরি কন্ট্রাক্টে (UniswapV2Router02.sol)
- পাথের প্রতিটি এক্সচেঞ্জে ট্রেড করার পরিমাণগুলো চিহ্নিত করুন।
- পাথের ওপর ইটারেট করে। পথের প্রতিটি এক্সচেঞ্জের জন্য এটি ইনপুট টোকেন পাঠায় এবং তারপর এক্সচেঞ্জের
swapফাংশন কল করে। বেশিরভাগ ক্ষেত্রে টোকেনগুলোর গন্তব্য এডড্রেস হলো পাথের পরবর্তী পেয়ার এক্সচেঞ্জ। চূড়ান্ত এক্সচেঞ্জে এটি ট্রেডারের দেওয়া এডড্রেস হয়।
কোর কন্ট্রাক্টে (UniswapV2Pair.sol)
- যাচাই করুন যে কোর কন্ট্রাক্ট প্রতারিত হচ্ছে না এবং সোয়াপের পরে পর্যাপ্ত লিকুইডিটি বজায় রাখতে পারে।
- দেখুন পরিচিত রিজার্ভের পাশাপাশি আমাদের কাছে কতগুলো অতিরিক্ত টোকেন আছে। সেই পরিমাণটি হলো এক্সচেঞ্জ করার জন্য আমাদের প্রাপ্ত ইনপুট টোকেনের সংখ্যা।
- আউটপুট টোকেনগুলো গন্তব্যে পাঠান।
- রিজার্ভের পরিমাণ আপডেট করতে
_updateকল করুন
পেরিফেরি কন্ট্রাক্টে ফিরে আসা (UniswapV2Router02.sol)
- প্রয়োজনীয় যেকোনো ক্লিনআপ সম্পন্ন করুন (উদাহরণস্বরূপ, ট্রেডারকে পাঠানোর জন্য ETH ফেরত পেতে WETH টোকেন বার্ন করুন)
লিকুইডিটি যোগ করা
কলার
- পেরিফেরি একাউন্টকে লিকুইডিটি পুলে যোগ করার পরিমাণের একটি এলাউন্স প্রদান করুন।
- পেরিফেরি কন্ট্রাক্টের যেকোনো একটি
addLiquidityফাংশন কল করুন।
পেরিফেরি কন্ট্রাক্টে (UniswapV2Router02.sol)
- প্রয়োজন হলে একটি নতুন পেয়ার এক্সচেঞ্জ তৈরি করুন
- যদি কোনো বিদ্যমান পেয়ার এক্সচেঞ্জ থাকে, তবে যোগ করার টোকেনের পরিমাণ হিসাব করুন। এটি উভয় টোকেনের জন্য অভিন্ন ভ্যালু হওয়ার কথা, তাই বিদ্যমান টোকেনের সাথে নতুন টোকেনের অনুপাত একই হবে।
- পরিমাণগুলো গ্রহণযোগ্য কি না তা চেক করুন (কলাররা একটি ন্যূনতম পরিমাণ নির্দিষ্ট করতে পারেন যার নিচে তারা লিকুইডিটি যোগ করতে চাইবেন না)
- কোর কন্ট্রাক্ট কল করুন।
কোর কন্ট্রাক্টে (UniswapV2Pair.sol)
- লিকুইডিটি টোকেন মিন্ট করুন এবং সেগুলো কলারের কাছে পাঠান
- রিজার্ভের পরিমাণ আপডেট করতে
_updateকল করুন
লিকুইডিটি সরানো
কলার
- আন্ডারলাইং টোকেনের বিনিময়ে বার্ন করার জন্য পেরিফেরি একাউন্টকে লিকুইডিটি টোকেনের একটি এলাউন্স প্রদান করুন।
- পেরিফেরি কন্ট্রাক্টের যেকোনো একটি
removeLiquidityফাংশন কল করুন।
পেরিফেরি কন্ট্রাক্টে (UniswapV2Router02.sol)
- পেয়ার এক্সচেঞ্জে লিকুইডিটি টোকেনগুলো পাঠান
কোর কন্ট্রাক্টে (UniswapV2Pair.sol)
- বার্ন করা টোকেনের অনুপাতে গন্তব্য এডড্রেসে আন্ডারলাইং টোকেনগুলো পাঠান। উদাহরণস্বরূপ, যদি পুলে 1000 A টোকেন, 500 B টোকেন এবং 90 লিকুইডিটি টোকেন থাকে এবং আমরা বার্ন করার জন্য 9 টোকেন পাই, তবে আমরা 10% লিকুইডিটি টোকেন বার্ন করছি, তাই আমরা ব্যবহারকারীকে 100 A টোকেন এবং 50 B টোকেন ফেরত পাঠাব।
- লিকুইডিটি টোকেনগুলো বার্ন করুন
- রিজার্ভের পরিমাণ আপডেট করতে
_updateকল করুন
মূল কন্ট্রাক্টস
এগুলো হলো সুরক্ষিত কন্ট্রাক্ট যা লিকুইডিটি ধরে রাখে।
UniswapV2Pair.sol
এই কন্ট্রাক্টটি (opens in a new tab) আসল পুলটি বাস্তবায়ন করে যা টোকেন এক্সচেঞ্জ করে। এটি Uniswap-এর মূল কার্যকারিতা।
1pragma solidity =0.5.16;2
3import './interfaces/IUniswapV2Pair.sol';4import './UniswapV2ERC20.sol';5import './libraries/Math.sol';6import './libraries/UQ112x112.sol';7import './interfaces/IERC20.sol';8import './interfaces/IUniswapV2Factory.sol';9import './interfaces/IUniswapV2Callee.sol';এগুলো হলো সেই সব ইন্টারফেস যা সম্পর্কে কন্ট্রাক্টটির জানা প্রয়োজন, কারণ কন্ট্রাক্টটি এগুলো বাস্তবায়ন করে (IUniswapV2Pair এবং UniswapV2ERC20) অথবা এটি এমন কন্ট্রাক্টগুলোকে কল করে যা এগুলো বাস্তবায়ন করে।
1contract UniswapV2Pair is IUniswapV2Pair, UniswapV2ERC20 {এই কন্ট্রাক্টটি UniswapV2ERC20 থেকে ইনহেরিট করে, যা লিকুইডিটি টোকেনগুলোর জন্য ERC-20 ফাংশন প্রদান করে।
1 using SafeMath for uint;ওভারফ্লো এবং আন্ডারফ্লো এড়াতে SafeMath লাইব্রেরি (opens in a new tab) ব্যবহার করা হয়। এটি গুরুত্বপূর্ণ কারণ অন্যথায় আমরা এমন একটি পরিস্থিতিতে পড়তে পারি যেখানে একটি মান -1 হওয়া উচিত, কিন্তু তার বদলে সেটি 2^256-1 হয়ে যায়।
1 using UQ112x112 for uint224;পুল কন্ট্রাক্টের অনেক হিসাবের জন্য ভগ্নাংশের প্রয়োজন হয়। তবে, EVM ভগ্নাংশ সমর্থন করে না।
Uniswap এর যে সমাধানটি বের করেছে তা হলো 224 বিট মান ব্যবহার করা, যার মধ্যে 112 বিট পূর্ণসংখ্যার জন্য এবং 112 বিট ভগ্নাংশের জন্য। তাই 1.0-কে 2^112 হিসেবে, 1.5-কে 2^112 + 2^111 হিসেবে উপস্থাপন করা হয়, ইত্যাদি।
এই লাইব্রেরি সম্পর্কে আরও বিস্তারিত তথ্য ডকুমেন্টের পরের অংশে পাওয়া যাবে।
ভেরিয়েবলস
1 uint public constant MINIMUM_LIQUIDITY = 10**3;শূন্য দিয়ে ভাগের ঘটনা এড়াতে, সবসময় একটি ন্যূনতম সংখ্যক লিকুইডিটি টোকেন থাকে (তবে এগুলো শূন্য একাউন্ট-এর মালিকানাধীন)। সেই সংখ্যাটি হলো MINIMUM_LIQUIDITY, এক হাজার।
1 bytes4 private constant SELECTOR = bytes4(keccak256(bytes('transfer(address,uint256)')));এটি ERC-20 ট্রান্সফার ফাংশনের জন্য ABI সিলেক্টর। এটি দুটি টোকেন একাউন্ট-এ ERC-20 টোকেন ট্রান্সফার করতে ব্যবহৃত হয়।
1 address public factory;এটি হলো ফ্যাক্টরি কন্ট্রাক্ট যা এই পুলটি তৈরি করেছে। প্রতিটি পুল হলো দুটি ERC-20 টোকেনের মধ্যে একটি এক্সচেঞ্জ, ফ্যাক্টরি হলো একটি কেন্দ্রীয় পয়েন্ট যা এই সমস্ত পুলগুলোকে সংযুক্ত করে।
1 address public token0;2 address public token1;এখানে দুটি ধরণের ERC-20 টোকেনের কন্ট্রাক্টগুলোর এডড্রেস রয়েছে যা এই পুলের মাধ্যমে এক্সচেঞ্জ করা যেতে পারে।
1 uint112 private reserve0; // সিঙ্গেল স্টোরেজ স্লট ব্যবহার করে, যা getReserves এর মাধ্যমে অ্যাক্সেসযোগ্য2 uint112 private reserve1; // সিঙ্গেল স্টোরেজ স্লট ব্যবহার করে, যা getReserves এর মাধ্যমে অ্যাক্সেসযোগ্যপ্রতিটি টোকেন টাইপের জন্য পুলের যে রিজার্ভ রয়েছে। আমরা ধরে নিই যে দুটি একই পরিমাণ মান উপস্থাপন করে, এবং তাই প্রতিটি token0-এর মান reserve1/reserve0 সংখ্যক token1-এর সমান।
1 uint32 private blockTimestampLast; // সিঙ্গেল স্টোরেজ স্লট ব্যবহার করে, যা getReserves এর মাধ্যমে অ্যাক্সেসযোগ্যসর্বশেষ যে ব্লক-এ একটি এক্সচেঞ্জ ঘটেছিল তার টাইমস্ট্যাম্প, যা সময়ের সাথে সাথে এক্সচেঞ্জ রেট ট্র্যাক করতে ব্যবহৃত হয়।
ইথিরিয়াম কন্ট্রাক্টগুলোর সবচেয়ে বড় গ্যাস খরচের একটি হলো স্টোরেজ, যা কন্ট্রাক্টের এক কল থেকে পরের কল পর্যন্ত স্থায়ী হয়। প্রতিটি স্টোরেজ সেল 256 বিট দীর্ঘ। তাই তিনটি ভেরিয়েবল, reserve0, reserve1, এবং blockTimestampLast, এমনভাবে বরাদ্দ করা হয় যাতে একটি একক স্টোরেজ মান তাদের তিনটিকে অন্তর্ভুক্ত করতে পারে (112+112+32=256)।
1 uint public price0CumulativeLast;2 uint public price1CumulativeLast;এই ভেরিয়েবলগুলো প্রতিটি টোকেনের জন্য ক্রমবর্ধমান খরচ ধরে রাখে (প্রতিটি অন্যটির সাপেক্ষে)। এগুলো একটি নির্দিষ্ট সময়ের মধ্যে গড় এক্সচেঞ্জ রেট গণনা করতে ব্যবহৃত হতে পারে।
1 uint public kLast; // reserve0 * reserve1, সাম্প্রতিকতম লিকুইডিটি ইভেন্টের ঠিক পরের অবস্থা অনুযায়ীপেয়ার এক্সচেঞ্জ যেভাবে token0 এবং token1-এর মধ্যে এক্সচেঞ্জ রেট নির্ধারণ করে তা হলো ট্রেডের সময় দুটি রিজার্ভের গুণফল স্থির রাখা। kLast হলো এই মান। এটি পরিবর্তিত হয় যখন কোনো লিকুইডিটি প্রোভাইডার টোকেন জমা দেয় বা তুলে নেয়, এবং 0.3% মার্কেট ফি-এর কারণে এটি সামান্য বৃদ্ধি পায়।
এখানে একটি সহজ উদাহরণ দেওয়া হলো। মনে রাখবেন যে সরলতার স্বার্থে টেবিলে দশমিক বিন্দুর পরে মাত্র তিনটি সংখ্যা রয়েছে, এবং আমরা 0.3% ট্রেডিং ফি উপেক্ষা করেছি তাই সংখ্যাগুলো পুরোপুরি নির্ভুল নয়।
| ইভেন্ট | reserve0 | reserve1 | reserve0 * reserve1 | গড় এক্সচেঞ্জ রেট (token1 / token0) |
|---|---|---|---|---|
| প্রাথমিক সেটআপ | 1,000.000 | 1,000.000 | 1,000,000 | |
| ট্রেডার A 50 token0 সোয়াপ করে 47.619 token1 পায় | 1,050.000 | 952.381 | 1,000,000 | 0.952 |
| ট্রেডার B 10 token0 সোয়াপ করে 8.984 token1 পায় | 1,060.000 | 943.396 | 1,000,000 | 0.898 |
| ট্রেডার C 40 token0 সোয়াপ করে 34.305 token1 পায় | 1,100.000 | 909.090 | 1,000,000 | 0.858 |
| ট্রেডার D 100 token1 সোয়াপ করে 109.01 token0 পায় | 990.990 | 1,009.090 | 1,000,000 | 0.917 |
| ট্রেডার E 10 token0 সোয়াপ করে 10.079 token1 পায় | 1,000.990 | 999.010 | 1,000,000 | 1.008 |
ট্রেডাররা যখন আরও বেশি token0 প্রদান করে, তখন token1-এর আপেক্ষিক মান বৃদ্ধি পায়, এবং এর বিপরীতটিও ঘটে, যা সাপ্লাই এবং ডিমান্ডের ওপর ভিত্তি করে হয়।
লক
1 uint private unlocked = 1;নিরাপত্তা দুর্বলতার একটি শ্রেণী রয়েছে যা রিএন্ট্রান্সি অ্যাবিউজ (opens in a new tab)-এর ওপর ভিত্তি করে তৈরি। Uniswap-এর যেকোনো ERC-20 টোকেন ট্রান্সফার করার প্রয়োজন হয়, যার মানে হলো এমন ERC-20 কন্ট্রাক্টগুলোকে কল করা যা তাদের কল করা Uniswap মার্কেটকে অপব্যবহার করার চেষ্টা করতে পারে।
কন্ট্রাক্টের অংশ হিসেবে একটি unlocked ভেরিয়েবল রাখার মাধ্যমে, আমরা ফাংশনগুলো চলাকালীন সময়ে (একই লেনদেন-এর মধ্যে) সেগুলোকে কল করা থেকে আটকাতে পারি।
1 modifier lock() {এই ফাংশনটি হলো একটি মডিফায়ার (opens in a new tab), এমন একটি ফাংশন যা একটি সাধারণ ফাংশনকে আবৃত করে কোনোভাবে এর আচরণ পরিবর্তন করে।
1 require(unlocked == 1, 'UniswapV2: LOCKED');2 unlocked = 0;যদি unlocked এক-এর সমান হয়, তবে এটিকে শূন্যতে সেট করুন। যদি এটি আগে থেকেই শূন্য থাকে তবে কলটি রিভার্ট করুন, এটিকে ব্যর্থ করুন।
1 _;একটি মডিফায়ারে _; হলো মূল ফাংশন কল (সমস্ত প্যারামিটারসহ)। এখানে এর অর্থ হলো ফাংশন কলটি কেবল তখনই ঘটে যদি কল করার সময় unlocked এক থাকে, এবং এটি চলাকালীন সময়ে unlocked-এর মান শূন্য থাকে।
1 unlocked = 1;2 }মূল ফাংশনটি রিটার্ন করার পর, লকটি রিলিজ করুন।
বিবিধ ফাংশন
1 function getReserves() public view returns (uint112 _reserve0, uint112 _reserve1, uint32 _blockTimestampLast) {2 _reserve0 = reserve0;3 _reserve1 = reserve1;4 _blockTimestampLast = blockTimestampLast;5 }এই ফাংশনটি কলারদের এক্সচেঞ্জের বর্তমান স্টেট প্রদান করে। লক্ষ্য করুন যে Solidity ফাংশনগুলো একাধিক মান রিটার্ন করতে পারে (opens in a new tab)।
1 function _safeTransfer(address token, address to, uint value) private {2 (bool success, bytes memory data) = token.call(abi.encodeWithSelector(SELECTOR, to, value));এই ইন্টারনাল ফাংশনটি এক্সচেঞ্জ থেকে অন্য কাউকে একটি নির্দিষ্ট পরিমাণ ERC20 টোকেন ট্রান্সফার করে। SELECTOR নির্দিষ্ট করে যে আমরা যে ফাংশনটি কল করছি তা হলো transfer(address,uint) (ওপরের সংজ্ঞা দেখুন)।
টোকেন ফাংশনের জন্য একটি ইন্টারফেস ইমপোর্ট করা এড়াতে, আমরা ABI ফাংশনগুলোর (opens in a new tab) একটি ব্যবহার করে "ম্যানুয়ালি" কলটি তৈরি করি।
1 require(success && (data.length == 0 || abi.decode(data, (bool))), 'UniswapV2: TRANSFER_FAILED');2 }দুটি উপায়ে একটি ERC-20 ট্রান্সফার কল ব্যর্থতা রিপোর্ট করতে পারে:
- রিভার্ট। যদি কোনো এক্সটার্নাল কন্ট্রাক্টে কল রিভার্ট হয়, তবে বুলিয়ান রিটার্ন মান হয়
false - স্বাভাবিকভাবে শেষ হয় কিন্তু একটি ব্যর্থতা রিপোর্ট করে। সেক্ষেত্রে রিটার্ন মান বাফারের একটি নন-জিরো দৈর্ঘ্য থাকে, এবং যখন একটি বুলিয়ান মান হিসেবে ডিকোড করা হয় তখন এটি
falseহয়
যদি এই শর্তগুলোর কোনোটি ঘটে, তবে রিভার্ট করুন।
ইভেন্টস
1 event Mint(address indexed sender, uint amount0, uint amount1);2 event Burn(address indexed sender, uint amount0, uint amount1, address indexed to);এই দুটি ইভেন্ট এমিট হয় যখন কোনো লিকুইডিটি প্রোভাইডার লিকুইডিটি জমা দেয় (Mint) অথবা তুলে নেয় (Burn)। উভয় ক্ষেত্রেই, জমা দেওয়া বা তুলে নেওয়া token0 এবং token1-এর পরিমাণ ইভেন্টের অংশ, সেইসাথে যে একাউন্ট আমাদের কল করেছে তার পরিচয় (sender)। তুলে নেওয়ার ক্ষেত্রে, ইভেন্টটিতে টোকেন গ্রহণকারী টার্গেটও (to) অন্তর্ভুক্ত থাকে, যা সেন্ডারের মতো একই নাও হতে পারে।
1 event Swap(2 address indexed sender,3 uint amount0In,4 uint amount1In,5 uint amount0Out,6 uint amount1Out,7 address indexed to8 );এই ইভেন্টটি এমিট হয় যখন কোনো ট্রেডার একটি টোকেন অন্যটির জন্য সোয়াপ করে। আবারও, সেন্ডার এবং গন্তব্য একই নাও হতে পারে। প্রতিটি টোকেন এক্সচেঞ্জে পাঠানো হতে পারে, অথবা এটি থেকে গ্রহণ করা হতে পারে।
1 event Sync(uint112 reserve0, uint112 reserve1);অবশেষে, কারণ যাই হোক না কেন, প্রতিবার টোকেন যোগ করা বা তুলে নেওয়ার সময় Sync এমিট হয়, যাতে সর্বশেষ রিজার্ভ তথ্য (এবং সেই কারণে এক্সচেঞ্জ রেট) প্রদান করা যায়।
সেটআপ ফাংশন
নতুন পেয়ার এক্সচেঞ্জ সেট আপ করার সময় এই ফাংশনগুলো একবার কল করার কথা।
1 constructor() public {2 factory = msg.sender;3 }কনস্ট্রাক্টর নিশ্চিত করে যে আমরা সেই ফ্যাক্টরির এডড্রেস ট্র্যাক রাখব যা পেয়ারটি তৈরি করেছে। এই তথ্যটি initialize-এর জন্য এবং ফ্যাক্টরি ফি-এর জন্য (যদি থাকে) প্রয়োজন।
1 // ডিপ্লয়মেন্টের সময় ফ্যাক্টরি দ্বারা একবার কল করা হয়2 function initialize(address _token0, address _token1) external {3 require(msg.sender == factory, 'UniswapV2: FORBIDDEN'); // পর্যাপ্ত চেক4 token0 = _token0;5 token1 = _token1;6 }এই ফাংশনটি ফ্যাক্টরিকে (এবং শুধুমাত্র ফ্যাক্টরিকে) দুটি ERC-20 টোকেন নির্দিষ্ট করার অনুমতি দেয় যা এই পেয়ারটি এক্সচেঞ্জ করবে।
ইন্টারনাল আপডেট ফাংশন
_update
1 // রিজার্ভ আপডেট করে এবং, প্রতি ব্লকের প্রথম কলে, প্রাইস অ্যাকুমুলেটর আপডেট করে2 function _update(uint balance0, uint balance1, uint112 _reserve0, uint112 _reserve1) private {প্রতিবার টোকেন জমা দেওয়া বা তুলে নেওয়ার সময় এই ফাংশনটি কল করা হয়।
1 require(balance0 <= uint112(-1) && balance1 <= uint112(-1), 'UniswapV2: OVERFLOW');যদি balance0 বা balance1 (uint256) uint112(-1) (=2^112-1)-এর চেয়ে বেশি হয় (তাই এটি ওভারফ্লো হয় এবং uint112-এ রূপান্তর করার সময় 0-তে ফিরে আসে) তবে ওভারফ্লো রোধ করতে _update চালিয়ে যেতে অস্বীকার করুন। একটি সাধারণ টোকেন যা 10^18 ইউনিটে বিভক্ত করা যায়, এর মানে হলো প্রতিটি এক্সচেঞ্জ প্রতিটি টোকেনের প্রায় 5.1*10^15-এ সীমাবদ্ধ। এখন পর্যন্ত এটি কোনো সমস্যা হয়নি।
1 uint32 blockTimestamp = uint32(block.timestamp % 2**32);2 uint32 timeElapsed = blockTimestamp - blockTimestampLast; // ওভারফ্লো কাঙ্ক্ষিত3 if (timeElapsed > 0 && _reserve0 != 0 && _reserve1 != 0) {যদি অতিবাহিত সময় শূন্য না হয়, তবে এর মানে হলো আমরা এই ব্লক-এ প্রথম এক্সচেঞ্জ লেনদেন। সেক্ষেত্রে, আমাদের কস্ট অ্যাকুমুলেটরগুলো আপডেট করতে হবে।
1 // * কখনও ওভারফ্লো হয় না, এবং + ওভারফ্লো কাঙ্ক্ষিত2 price0CumulativeLast += uint(UQ112x112.encode(_reserve1).uqdiv(_reserve0)) * timeElapsed;3 price1CumulativeLast += uint(UQ112x112.encode(_reserve0).uqdiv(_reserve1)) * timeElapsed;4 }প্রতিটি কস্ট অ্যাকুমুলেটর সর্বশেষ খরচ (অন্য টোকেনের রিজার্ভ/এই টোকেনের রিজার্ভ) গুণ সেকেন্ডে অতিবাহিত সময়ের সাথে আপডেট করা হয়। একটি গড় মূল্য পেতে, আপনি দুটি ভিন্ন সময়ের ক্রমবর্ধমান মূল্য পড়েন এবং তাদের মধ্যকার সময়ের পার্থক্য দিয়ে ভাগ করেন। উদাহরণস্বরূপ, ইভেন্টগুলোর এই ক্রমটি ধরে নিন:
| ইভেন্ট | reserve0 | reserve1 | টাইমস্ট্যাম্প | প্রান্তিক এক্সচেঞ্জ রেট (reserve1 / reserve0) | price0CumulativeLast |
|---|---|---|---|---|---|
| প্রাথমিক সেটআপ | 1,000.000 | 1,000.000 | 5,000 | 1.000 | 0 |
| ট্রেডার A 50 token0 জমা দেয় এবং 47.619 token1 ফেরত পায় | 1,050.000 | 952.381 | 5,020 | 0.907 | 20 |
| ট্রেডার B 10 token0 জমা দেয় এবং 8.984 token1 ফেরত পায় | 1,060.000 | 943.396 | 5,030 | 0.890 | 20+10*0.907 = 29.07 |
| ট্রেডার C 40 token0 জমা দেয় এবং 34.305 token1 ফেরত পায় | 1,100.000 | 909.090 | 5,100 | 0.826 | 29.07+70*0.890 = 91.37 |
| ট্রেডার D 100 token1 জমা দেয় এবং 109.01 token0 ফেরত পায় | 990.990 | 1,009.090 | 5,110 | 1.018 | 91.37+10*0.826 = 99.63 |
| ট্রেডার E 10 token0 জমা দেয় এবং 10.079 token1 ফেরত পায় | 1,000.990 | 999.010 | 5,150 | 0.998 | 99.63+40*1.1018 = 143.702 |
ধরা যাক আমরা 5,030 এবং 5,150 টাইমস্ট্যাম্পের মধ্যে Token0-এর গড় মূল্য গণনা করতে চাই। price0Cumulative-এর মানের পার্থক্য হলো 143.702-29.07=114.632। এটি দুই মিনিটের (120 সেকেন্ড) গড়। তাই গড় মূল্য হলো 114.632/120 = 0.955।
এই মূল্য গণনার কারণেই আমাদের পুরোনো রিজার্ভের আকারগুলো জানা প্রয়োজন।
1 reserve0 = uint112(balance0);2 reserve1 = uint112(balance1);3 blockTimestampLast = blockTimestamp;4 emit Sync(reserve0, reserve1);5 }অবশেষে, গ্লোবাল ভেরিয়েবলগুলো আপডেট করুন এবং একটি Sync ইভেন্ট এমিট করুন।
_mintFee
1 // যদি ফি চালু থাকে, তবে sqrt(k) এর বৃদ্ধির ১/৬ অংশের সমতুল্য লিকুইডিটি মিন্ট করে2 function _mintFee(uint112 _reserve0, uint112 _reserve1) private returns (bool feeOn) {Uniswap 2.0-তে ট্রেডাররা মার্কেট ব্যবহার করার জন্য 0.30% ফি প্রদান করে। সেই ফি-এর বেশিরভাগ অংশ (ট্রেডের 0.25%) সবসময় লিকুইডিটি প্রোভাইডারদের কাছে যায়। বাকি 0.05% লিকুইডিটি প্রোভাইডারদের কাছে অথবা ফ্যাক্টরি দ্বারা প্রটোকল ফি হিসেবে নির্দিষ্ট করা একটি এডড্রেস-এ যেতে পারে, যা Uniswap-কে তাদের ডেভেলপমেন্ট প্রচেষ্টার জন্য অর্থ প্রদান করে।
হিসাব কমানোর জন্য (এবং সেই কারণে গ্যাস খরচ), এই ফি শুধুমাত্র তখনই গণনা করা হয় যখন পুল থেকে লিকুইডিটি যোগ করা বা সরানো হয়, প্রতিটি লেনদেন-এর সময় নয়।
1 address feeTo = IUniswapV2Factory(factory).feeTo();2 feeOn = feeTo != address(0);ফ্যাক্টরির ফি গন্তব্য পড়ুন। যদি এটি শূন্য হয় তবে কোনো প্রটোকল ফি নেই এবং সেই ফি গণনা করার কোনো প্রয়োজন নেই।
1 uint _kLast = kLast; // গ্যাস সাশ্রয়kLast স্টেট ভেরিয়েবলটি স্টোরেজে অবস্থিত, তাই কন্ট্রাক্টে বিভিন্ন কলের মধ্যে এর একটি মান থাকবে।
স্টোরেজ অ্যাক্সেস করা ভোলাটাইল মেমরি অ্যাক্সেস করার চেয়ে অনেক বেশি ব্যয়বহুল যা কন্ট্রাক্টে ফাংশন কল শেষ হলে রিলিজ হয়ে যায়, তাই আমরা গ্যাস বাঁচাতে একটি ইন্টারনাল ভেরিয়েবল ব্যবহার করি।
1 if (feeOn) {2 if (_kLast != 0) {লিকুইডিটি প্রোভাইডাররা তাদের লিকুইডিটি টোকেনগুলোর মূল্যবৃদ্ধির মাধ্যমেই তাদের অংশ পেয়ে যায়। কিন্তু প্রটোকল ফি-এর জন্য নতুন লিকুইডিটি টোকেন মিন্ট করা এবং feeTo এডড্রেস-এ প্রদান করা প্রয়োজন।
1 uint rootK = Math.sqrt(uint(_reserve0).mul(_reserve1));2 uint rootKLast = Math.sqrt(_kLast);3 if (rootK > rootKLast) {যদি নতুন লিকুইডিটি থাকে যার ওপর প্রটোকল ফি সংগ্রহ করতে হবে। আপনি স্কয়ার রুট ফাংশনটি এই আর্টিকেলের পরের অংশে দেখতে পারেন।
1 uint numerator = totalSupply.mul(rootK.sub(rootKLast));2 uint denominator = rootK.mul(5).add(rootKLast);3 uint liquidity = numerator / denominator;ফি-এর এই জটিল হিসাবটি হোয়াইটপেপার (opens in a new tab)-এর ৫ পৃষ্ঠায় ব্যাখ্যা করা হয়েছে। আমরা জানি যে kLast গণনা করার সময় এবং বর্তমান সময়ের মধ্যে কোনো লিকুইডিটি যোগ করা বা সরানো হয়নি (কারণ আমরা প্রতিবার লিকুইডিটি যোগ করা বা সরানোর সময় এই হিসাবটি চালাই, এটি আসলে পরিবর্তন হওয়ার আগে), তাই reserve0 * reserve1-এ যেকোনো পরিবর্তন লেনদেন ফি থেকে আসতে হবে (সেগুলো ছাড়া আমরা reserve0 * reserve1 স্থির রাখতাম)।
1 if (liquidity > 0) _mint(feeTo, liquidity);2 }3 }অতিরিক্ত লিকুইডিটি টোকেনগুলো আসলে তৈরি করতে এবং সেগুলোকে feeTo-তে অ্যাসাইন করতে UniswapV2ERC20._mint ফাংশনটি ব্যবহার করুন।
1 } else if (_kLast != 0) {2 kLast = 0;3 }4 }যদি কোনো ফি না থাকে তবে kLast-কে শূন্যতে সেট করুন (যদি এটি আগে থেকেই তা না থাকে)। যখন এই কন্ট্রাক্টটি লেখা হয়েছিল তখন একটি গ্যাস রিফান্ড ফিচার (opens in a new tab) ছিল যা কন্ট্রাক্টগুলোকে তাদের অপ্রয়োজনীয় স্টোরেজ শূন্য করে ইথিরিয়াম স্টেট-এর সামগ্রিক আকার কমাতে উৎসাহিত করত।
এই কোডটি সম্ভব হলে সেই রিফান্ড পায়।
এক্সটার্নালি অ্যাক্সেসযোগ্য ফাংশন
মনে রাখবেন যে যদিও যেকোনো লেনদেন বা কন্ট্রাক্ট এই ফাংশনগুলোকে কল করতে পারে, এগুলো পেরিফেরি কন্ট্রাক্ট থেকে কল করার জন্য ডিজাইন করা হয়েছে। আপনি যদি এগুলোকে সরাসরি কল করেন তবে আপনি পেয়ার এক্সচেঞ্জকে প্রতারণা করতে পারবেন না, তবে আপনি কোনো ভুলের মাধ্যমে মান হারাতে পারেন।
mint
1 // এই লো-লেভেল ফাংশনটি এমন একটি কন্ট্রাক্ট থেকে কল করা উচিত যা গুরুত্বপূর্ণ সেফটি চেক সম্পাদন করে2 function mint(address to) external lock returns (uint liquidity) {এই ফাংশনটি কল করা হয় যখন কোনো লিকুইডিটি প্রোভাইডার পুলে লিকুইডিটি যোগ করে। এটি পুরস্কার হিসেবে অতিরিক্ত লিকুইডিটি টোকেন মিন্ট করে। এটিকে একটি পেরিফেরি কন্ট্রাক্ট থেকে কল করা উচিত যা একই লেনদেন-এ লিকুইডিটি যোগ করার পরে এটিকে কল করে (যাতে অন্য কেউ বৈধ মালিকের আগে নতুন লিকুইডিটি দাবি করে এমন কোনো লেনদেন জমা দিতে না পারে)।
1 (uint112 _reserve0, uint112 _reserve1,) = getReserves(); // গ্যাস সাশ্রয়একাধিক মান রিটার্ন করে এমন একটি Solidity ফাংশনের ফলাফল পড়ার উপায় এটি। আমরা সর্বশেষ রিটার্ন করা মান, ব্লক টাইমস্ট্যাম্প, বাতিল করি, কারণ আমাদের এটির প্রয়োজন নেই।
1 uint balance0 = IERC20(token0).balanceOf(address(this));2 uint balance1 = IERC20(token1).balanceOf(address(this));3 uint amount0 = balance0.sub(_reserve0);4 uint amount1 = balance1.sub(_reserve1);বর্তমান ব্যালেন্সগুলো পান এবং দেখুন প্রতিটি টোকেন টাইপের কতটা যোগ করা হয়েছে।
1 bool feeOn = _mintFee(_reserve0, _reserve1);সংগ্রহ করার জন্য প্রটোকল ফি গণনা করুন, যদি থাকে, এবং সেই অনুযায়ী লিকুইডিটি টোকেন মিন্ট করুন। যেহেতু _mintFee-এর প্যারামিটারগুলো হলো পুরোনো রিজার্ভ মান, তাই ফি শুধুমাত্র ফি-এর কারণে পুলের পরিবর্তনের ওপর ভিত্তি করে নির্ভুলভাবে গণনা করা হয়।
1 uint _totalSupply = totalSupply; // গ্যাস সাশ্রয়, অবশ্যই এখানে সংজ্ঞায়িত করতে হবে কারণ totalSupply _mintFee তে আপডেট হতে পারে2 if (_totalSupply == 0) {3 liquidity = Math.sqrt(amount0.mul(amount1)).sub(MINIMUM_LIQUIDITY);4 _mint(address(0), MINIMUM_LIQUIDITY); // প্রথম MINIMUM_LIQUIDITY টোকেনগুলো স্থায়ীভাবে লক করেযদি এটি প্রথম জমা হয়, তবে MINIMUM_LIQUIDITY টোকেন তৈরি করুন এবং সেগুলোকে লক করতে শূন্য এডড্রেস-এ পাঠান। এগুলো কখনোই রিডিম করা যাবে না, যার মানে হলো পুলটি কখনোই পুরোপুরি খালি হবেবিধা হবে না (এটি আমাদের কিছু জায়গায় শূন্য দিয়ে ভাগ করা থেকে বাঁচায়)। MINIMUM_LIQUIDITY-এর মান হলো এক হাজার, যা বিবেচনা করে যে বেশিরভাগ ERC-20 একটি টোকেনের 10^-18 তম ইউনিটে বিভক্ত, যেমন ETH wei-তে বিভক্ত, এটি একটি একক টোকেনের মানের 10^-15। খুব বেশি খরচ নয়।
প্রথম জমার সময় আমরা দুটি টোকেনের আপেক্ষিক মান জানি না, তাই আমরা কেবল পরিমাণগুলো গুণ করি এবং একটি স্কয়ার রুট নিই, এই ধরে নিয়ে যে জমাটি আমাদের উভয় টোকেনে সমান মান প্রদান করে।
আমরা এটিকে বিশ্বাস করতে পারি কারণ আরবিট্রেজ-এ মান হারানো এড়াতে সমান মান প্রদান করা আমানতকারীর স্বার্থে। ধরা যাক দুটি টোকেনের মান অভিন্ন, কিন্তু আমাদের আমানতকারী Token0-এর চেয়ে চারগুণ বেশি Token1 জমা দিয়েছেন। একজন ট্রেডার এই সত্যটি ব্যবহার করতে পারে যে পেয়ার এক্সচেঞ্জ মনে করে Token0 বেশি মূল্যবান, এটি থেকে মান বের করে নিতে।
| ইভেন্ট | reserve0 | reserve1 | reserve0 * reserve1 | পুলের মান (reserve0 + reserve1) |
|---|---|---|---|---|
| প্রাথমিক সেটআপ | 8 | 32 | 256 | 40 |
| ট্রেডার 8টি Token0 টোকেন জমা দেয়, 16টি Token1 ফেরত পায় | 16 | 16 | 256 | 32 |
যেমনটি আপনি দেখতে পাচ্ছেন, ট্রেডার অতিরিক্ত 8টি টোকেন উপার্জন করেছে, যা পুলের মান হ্রাস থেকে আসে, যা এর মালিক আমানতকারীকে ক্ষতিগ্রস্ত করে।
1 } else {2 liquidity = Math.min(amount0.mul(_totalSupply) / _reserve0, amount1.mul(_totalSupply) / _reserve1);পরবর্তী প্রতিটি জমার সাথে আমরা ইতিমধ্যেই দুটি অ্যাসেটের মধ্যে এক্সচেঞ্জ রেট জানি, এবং আমরা আশা করি লিকুইডিটি প্রোভাইডাররা উভয়ের সমান মান প্রদান করবে। যদি তারা তা না করে, তবে আমরা তাদের শাস্তি হিসেবে তাদের দেওয়া কম মানের ওপর ভিত্তি করে লিকুইডিটি টোকেন দিই।
এটি প্রাথমিক জমা হোক বা পরবর্তী কোনো জমা, আমরা যে সংখ্যক লিকুইডিটি টোকেন প্রদান করি তা reserve0*reserve1-এর পরিবর্তনের স্কয়ার রুটের সমান এবং লিকুইডিটি টোকেনের মান পরিবর্তন হয় না (যদি না আমরা এমন কোনো জমা পাই যার উভয় প্রকারের সমান মান নেই, সেক্ষেত্রে "জরিমানা" বিতরণ করা হয়)। এখানে একই মান থাকা দুটি টোকেনের আরেকটি উদাহরণ দেওয়া হলো, যেখানে তিনটি ভালো জমা এবং একটি খারাপ জমা রয়েছে (শুধুমাত্র এক প্রকারের টোকেন জমা, তাই এটি কোনো লিকুইডিটি টোকেন তৈরি করে না)।
| ইভেন্ট | reserve0 | reserve1 | reserve0 * reserve1 | পুলের মান (reserve0 + reserve1) | এই জমার জন্য মিন্ট করা লিকুইডিটি টোকেন | মোট লিকুইডিটি টোকেন | প্রতিটি লিকুইডিটি টোকেনের মান |
|---|---|---|---|---|---|---|---|
| প্রাথমিক সেটআপ | 8.000 | 8.000 | 64 | 16.000 | 8 | 8 | 2.000 |
| প্রতিটি প্রকারের চারটি জমা | 12.000 | 12.000 | 144 | 24.000 | 4 | 12 | 2.000 |
| প্রতিটি প্রকারের দুটি জমা | 14.000 | 14.000 | 196 | 28.000 | 2 | 14 | 2.000 |
| অসম মানের জমা | 18.000 | 14.000 | 252 | 32.000 | 0 | 14 | ~2.286 |
| আরবিট্রেজ-এর পর | ~15.874 | ~15.874 | 252 | ~31.748 | 0 | 14 | ~2.267 |
1 }2 require(liquidity > 0, 'UniswapV2: INSUFFICIENT_LIQUIDITY_MINTED');3 _mint(to, liquidity);অতিরিক্ত লিকুইডিটি টোকেনগুলো আসলে তৈরি করতে এবং সেগুলোকে সঠিক একাউন্ট-এ দিতে UniswapV2ERC20._mint ফাংশনটি ব্যবহার করুন।
1
2 _update(balance0, balance1, _reserve0, _reserve1);3 if (feeOn) kLast = uint(reserve0).mul(reserve1); // reserve0 এবং reserve1 আপ-টু-ডেট আছে4 emit Mint(msg.sender, amount0, amount1);5 }স্টেট ভেরিয়েবলগুলো আপডেট করুন (reserve0, reserve1, এবং প্রয়োজন হলে kLast) এবং উপযুক্ত ইভেন্ট এমিট করুন।
burn
1 // এই লো-লেভেল ফাংশনটি এমন একটি কন্ট্রাক্ট থেকে কল করা উচিত যা গুরুত্বপূর্ণ সেফটি চেক সম্পাদন করে2 function burn(address to) external lock returns (uint amount0, uint amount1) {এই ফাংশনটি কল করা হয় যখন লিকুইডিটি তুলে নেওয়া হয় এবং উপযুক্ত লিকুইডিটি টোকেনগুলো বার্ন করার প্রয়োজন হয়। এটিও একটি পেরিফেরি একাউন্ট থেকে কল করা উচিত।
1 (uint112 _reserve0, uint112 _reserve1,) = getReserves(); // গ্যাস সাশ্রয়2 address _token0 = token0; // গ্যাস সাশ্রয়3 address _token1 = token1; // গ্যাস সাশ্রয়4 uint balance0 = IERC20(_token0).balanceOf(address(this));5 uint balance1 = IERC20(_token1).balanceOf(address(this));6 uint liquidity = balanceOf[address(this)];পেরিফেরি কন্ট্রাক্ট কলের আগে বার্ন করার জন্য লিকুইডিটি এই কন্ট্রাক্টে ট্রান্সফার করেছে। সেভাবে আমরা জানি কতটা লিকুইডিটি বার্ন করতে হবে, এবং আমরা নিশ্চিত করতে পারি যে এটি বার্ন হয়েছে।
1 bool feeOn = _mintFee(_reserve0, _reserve1);2 uint _totalSupply = totalSupply; // গ্যাস সাশ্রয়, অবশ্যই এখানে সংজ্ঞায়িত করতে হবে কারণ totalSupply _mintFee তে আপডেট হতে পারে3 amount0 = liquidity.mul(balance0) / _totalSupply; // ব্যালেন্স ব্যবহার করা আনুপাতিক (pro-rata) বন্টন নিশ্চিত করে4 amount1 = liquidity.mul(balance1) / _totalSupply; // ব্যালেন্স ব্যবহার করা আনুপাতিক (pro-rata) বন্টন নিশ্চিত করে5 require(amount0 > 0 && amount1 > 0, 'UniswapV2: INSUFFICIENT_LIQUIDITY_BURNED');লিকুইডিটি প্রোভাইডার উভয় টোকেনের সমান মান গ্রহণ করে। এইভাবে আমরা এক্সচেঞ্জ রেট পরিবর্তন করি না।
1 _burn(address(this), liquidity);2 _safeTransfer(_token0, to, amount0);3 _safeTransfer(_token1, to, amount1);4 balance0 = IERC20(_token0).balanceOf(address(this));5 balance1 = IERC20(_token1).balanceOf(address(this));6
7 _update(balance0, balance1, _reserve0, _reserve1);8 if (feeOn) kLast = uint(reserve0).mul(reserve1); // reserve0 এবং reserve1 আপ-টু-ডেট আছে9 emit Burn(msg.sender, amount0, amount1, to);10 }11
বাকি burn ফাংশনটি ওপরের mint ফাংশনের মিরর ইমেজ।
swap
1 // এই লো-লেভেল ফাংশনটি এমন একটি কন্ট্রাক্ট থেকে কল করা উচিত যা গুরুত্বপূর্ণ সেফটি চেক সম্পাদন করে2 function swap(uint amount0Out, uint amount1Out, address to, bytes calldata data) external lock {এই ফাংশনটিও একটি পেরিফেরি কন্ট্রাক্ট থেকে কল করার কথা।
1 require(amount0Out > 0 || amount1Out > 0, 'UniswapV2: INSUFFICIENT_OUTPUT_AMOUNT');2 (uint112 _reserve0, uint112 _reserve1,) = getReserves(); // গ্যাস সাশ্রয়3 require(amount0Out < _reserve0 && amount1Out < _reserve1, 'UniswapV2: INSUFFICIENT_LIQUIDITY');4
5 uint balance0;6 uint balance1;7 { // _token{0,1} এর জন্য স্কোপ, স্ট্যাক টু ডিপ (stack too deep) এরর এড়ায়লোকাল ভেরিয়েবলগুলো মেমরিতে সংরক্ষণ করা যেতে পারে অথবা, যদি সেগুলো খুব বেশি না হয়, তবে সরাসরি স্ট্যাক-এ সংরক্ষণ করা যেতে পারে। যদি আমরা সংখ্যাটি সীমিত করতে পারি যাতে আমরা স্ট্যাক ব্যবহার করব তবে আমরা কম গ্যাস ব্যবহার করি। আরও বিস্তারিত তথ্যের জন্য ইয়েলো পেপার, আনুষ্ঠানিক ইথিরিয়াম স্পেসিফিকেশন (opens in a new tab), পৃষ্ঠা ২৬, সমীকরণ ২৯৮ দেখুন।
1 address _token0 = token0;2 address _token1 = token1;3 require(to != _token0 && to != _token1, 'UniswapV2: INVALID_TO');4 if (amount0Out > 0) _safeTransfer(_token0, to, amount0Out); // অপ্টিমিস্টিক্যালি টোকেন ট্রান্সফার করে5 if (amount1Out > 0) _safeTransfer(_token1, to, amount1Out); // অপ্টিমিস্টিক্যালি টোকেন ট্রান্সফার করেএই ট্রান্সফারটি অপ্টিমিস্টিক, কারণ সমস্ত শর্ত পূরণ হয়েছে কিনা তা নিশ্চিত হওয়ার আগেই আমরা ট্রান্সফার করি। ইথিরিয়াম-এ এটি ঠিক আছে কারণ যদি কলের পরে শর্তগুলো পূরণ না হয় তবে আমরা এটি থেকে এবং এটি তৈরি করা যেকোনো পরিবর্তন থেকে রিভার্ট করি।
1 if (data.length > 0) IUniswapV2Callee(to).uniswapV2Call(msg.sender, amount0Out, amount1Out, data);অনুরোধ করা হলে রিসিভারকে সোয়াপ সম্পর্কে জানান।
1 balance0 = IERC20(_token0).balanceOf(address(this));2 balance1 = IERC20(_token1).balanceOf(address(this));3 }বর্তমান ব্যালেন্সগুলো পান। পেরিফেরি কন্ট্রাক্ট সোয়াপ-এর জন্য আমাদের কল করার আগে আমাদের টোকেনগুলো পাঠায়। এটি কন্ট্রাক্টের জন্য চেক করা সহজ করে তোলে যে এটি প্রতারিত হচ্ছে না, এমন একটি চেক যা কোর কন্ট্রাক্টে হতেই হবে (কারণ আমাদের পেরিফেরি কন্ট্রাক্ট ছাড়া অন্য এনটিটিগুলো দ্বারা আমাদের কল করা হতে পারে)।
1 uint amount0In = balance0 > _reserve0 - amount0Out ? balance0 - (_reserve0 - amount0Out) : 0;2 uint amount1In = balance1 > _reserve1 - amount1Out ? balance1 - (_reserve1 - amount1Out) : 0;3 require(amount0In > 0 || amount1In > 0, 'UniswapV2: INSUFFICIENT_INPUT_AMOUNT');4 { // reserve{0,1}Adjusted এর জন্য স্কোপ, স্ট্যাক টু ডিপ (stack too deep) এরর এড়ায়5 uint balance0Adjusted = balance0.mul(1000).sub(amount0In.mul(3));6 uint balance1Adjusted = balance1.mul(1000).sub(amount1In.mul(3));7 require(balance0Adjusted.mul(balance1Adjusted) >= uint(_reserve0).mul(_reserve1).mul(1000**2), 'UniswapV2: K');এটি একটি স্যানিটি চেক যা নিশ্চিত করে যে আমরা সোয়াপ থেকে হারাব না। এমন কোনো পরিস্থিতি নেই যেখানে একটি সোয়াপ reserve0*reserve1 হ্রাস করবে। এখানেই আমরা নিশ্চিত করি যে সোয়াপ-এ 0.3% ফি পাঠানো হচ্ছে; K-এর মান স্যানিটি চেক করার আগে, আমরা উভয় ব্যালেন্সকে 1000 দিয়ে গুণ করি এবং পরিমাণগুলোকে 3 দিয়ে গুণ করে বিয়োগ করি, এর মানে হলো বর্তমান রিজার্ভের K মানের সাথে এর K মান তুলনা করার আগে ব্যালেন্স থেকে 0.3% (3/1000 = 0.003 = 0.3%) কেটে নেওয়া হচ্ছে।
1 }2
3 _update(balance0, balance1, _reserve0, _reserve1);4 emit Swap(msg.sender, amount0In, amount1In, amount0Out, amount1Out, to);5 }reserve0 এবং reserve1 আপডেট করুন, এবং প্রয়োজন হলে প্রাইস অ্যাকুমুলেটর এবং টাইমস্ট্যাম্প আপডেট করুন এবং একটি ইভেন্ট এমিট করুন।
Sync বা Skim
আসল ব্যালেন্সগুলোর পক্ষে পেয়ার এক্সচেঞ্জ-এর মনে করা রিজার্ভের সাথে সিঙ্ক না হওয়া সম্ভব।
কন্ট্রাক্টের সম্মতি ছাড়া টোকেন তুলে নেওয়ার কোনো উপায় নেই, তবে জমা দেওয়া একটি ভিন্ন বিষয়। একটি একাউন্ট mint বা swap কল না করেই এক্সচেঞ্জে টোকেন ট্রান্সফার করতে পারে।
সেক্ষেত্রে দুটি সমাধান রয়েছে:
sync, বর্তমান ব্যালেন্সগুলোতে রিজার্ভ আপডেট করুনskim, অতিরিক্ত পরিমাণ তুলে নিন। মনে রাখবেন যে যেকোনো একাউন্ট-কেskimকল করার অনুমতি দেওয়া হয়েছে কারণ আমরা জানি না কে টোকেনগুলো জমা দিয়েছে। এই তথ্যটি একটি ইভেন্টে এমিট করা হয়, তবে ইভেন্টগুলো ব্লকচেইন থেকে অ্যাক্সেসযোগ্য নয়।
1 // ব্যালেন্সকে রিজার্ভের সাথে মেলাতে বাধ্য করে2 function skim(address to) external lock {3 address _token0 = token0; // গ্যাস সাশ্রয়4 address _token1 = token1; // গ্যাস সাশ্রয়5 _safeTransfer(_token0, to, IERC20(_token0).balanceOf(address(this)).sub(reserve0));6 _safeTransfer(_token1, to, IERC20(_token1).balanceOf(address(this)).sub(reserve1));7 }8
9
10
11 // রিজার্ভকে ব্যালেন্সের সাথে মেলাতে বাধ্য করে12 function sync() external lock {13 _update(IERC20(token0).balanceOf(address(this)), IERC20(token1).balanceOf(address(this)), reserve0, reserve1);14 }15}UniswapV2Factory.sol
এই কন্ট্রাক্টটি (opens in a new tab) পেয়ার এক্সচেঞ্জগুলো তৈরি করে।
1pragma solidity =0.5.16;2
3import './interfaces/IUniswapV2Factory.sol';4import './UniswapV2Pair.sol';5
6contract UniswapV2Factory is IUniswapV2Factory {7 address public feeTo;8 address public feeToSetter;প্রটোকল ফি বাস্তবায়ন করার জন্য এই স্টেট ভেরিয়েবলগুলো প্রয়োজনীয় (হোয়াইটপেপার (opens in a new tab), পৃষ্ঠা ৫ দেখুন)।
feeTo এডড্রেস প্রটোকল ফি-এর জন্য লিকুইডিটি টোকেনগুলো জমা করে, এবং feeToSetter হলো সেই এডড্রেস যাকে feeTo-কে একটি ভিন্ন এডড্রেস-এ পরিবর্তন করার অনুমতি দেওয়া হয়েছে।
1 mapping(address => mapping(address => address)) public getPair;2 address[] public allPairs;এই ভেরিয়েবলগুলো পেয়ারগুলোর, দুটি টোকেন টাইপের মধ্যে এক্সচেঞ্জগুলোর ট্র্যাক রাখে।
প্রথমটি, getPair, হলো একটি ম্যাপিং যা একটি পেয়ার এক্সচেঞ্জ কন্ট্রাক্টকে এটি যে দুটি ERC-20 টোকেন এক্সচেঞ্জ করে তার ওপর ভিত্তি করে শনাক্ত করে। ERC-20 টোকেনগুলো সেই কন্ট্রাক্টগুলোর এডড্রেস দ্বারা শনাক্ত করা হয় যা সেগুলোকে বাস্তবায়ন করে, তাই কী (keys) এবং মান (value) সবই হলো এডড্রেস। পেয়ার এক্সচেঞ্জের এডড্রেস পেতে যা আপনাকে tokenA থেকে tokenB-তে রূপান্তর করতে দেয়, আপনি getPair[<tokenA address>][<tokenB address>] ব্যবহার করেন (অথবা এর উল্টোটা)।
দ্বিতীয় ভেরিয়েবল, allPairs, হলো একটি অ্যারে যা এই ফ্যাক্টরি দ্বারা তৈরি সমস্ত পেয়ার এক্সচেঞ্জের এডড্রেস অন্তর্ভুক্ত করে। ইথিরিয়াম-এ আপনি একটি ম্যাপিংয়ের কন্টেন্টের ওপর ইটারেট করতে পারবেন না, বা সমস্ত কী-এর একটি তালিকা পেতে পারবেন না, তাই এই ফ্যাক্টরি কোন এক্সচেঞ্জগুলো পরিচালনা করে তা জানার একমাত্র উপায় হলো এই ভেরিয়েবলটি।
দ্রষ্টব্য: আপনি একটি ম্যাপিংয়ের সমস্ত কী-এর ওপর ইটারেট করতে না পারার কারণ হলো কন্ট্রাক্ট ডাটা স্টোরেজ ব্যয়বহুল, তাই আমরা এটি যত কম ব্যবহার করি ততই ভালো, এবং আমরা এটি যত কম পরিবর্তন করি ততই ভালো। আপনি ইটারেশন সমর্থন করে এমন ম্যাপিং (opens in a new tab) তৈরি করতে পারেন, তবে সেগুলোর জন্য কী-এর একটি তালিকার জন্য অতিরিক্ত স্টোরেজ প্রয়োজন। বেশিরভাগ অ্যাপ্লিকেশনে আপনার সেটির প্রয়োজন নেই।
1 event PairCreated(address indexed token0, address indexed token1, address pair, uint);এই ইভেন্টটি এমিট হয় যখন একটি নতুন পেয়ার এক্সচেঞ্জ তৈরি করা হয়। এটিতে টোকেনগুলোর এডড্রেস, পেয়ার এক্সচেঞ্জের এডড্রেস এবং ফ্যাক্টরি দ্বারা পরিচালিত মোট এক্সচেঞ্জের সংখ্যা অন্তর্ভুক্ত থাকে।
1 constructor(address _feeToSetter) public {2 feeToSetter = _feeToSetter;3 }কনস্ট্রাক্টর একমাত্র যে কাজটি করে তা হলো feeToSetter নির্দিষ্ট করা। ফ্যাক্টরিগুলো কোনো ফি ছাড়াই শুরু হয়, এবং শুধুমাত্র feeSetter তা পরিবর্তন করতে পারে।
1 function allPairsLength() external view returns (uint) {2 return allPairs.length;3 }এই ফাংশনটি এক্সচেঞ্জ পেয়ারের সংখ্যা রিটার্ন করে।
1 function createPair(address tokenA, address tokenB) external returns (address pair) {এটি ফ্যাক্টরির মূল ফাংশন, দুটি ERC-20 টোকেনের মধ্যে একটি পেয়ার এক্সচেঞ্জ তৈরি করা। মনে রাখবেন যে যেকেউ এই ফাংশনটি কল করতে পারে। একটি নতুন পেয়ার এক্সচেঞ্জ তৈরি করতে আপনার Uniswap থেকে অনুমতির প্রয়োজন নেই।
1 require(tokenA != tokenB, 'UniswapV2: IDENTICAL_ADDRESSES');2 (address token0, address token1) = tokenA < tokenB ? (tokenA, tokenB) : (tokenB, tokenA);আমরা চাই নতুন এক্সচেঞ্জের এডড্রেসটি ডিটারমিনিস্টিক হোক, যাতে এটি আগে থেকেই অফচেইন গণনা করা যায় (এটি লেয়ার ২ লেনদেন-এর জন্য দরকারী হতে পারে)। এটি করার জন্য আমাদের টোকেন এডড্রেসগুলোর একটি সামঞ্জস্যপূর্ণ ক্রম থাকতে হবে, আমরা যে ক্রমেই সেগুলো পেয়ে থাকি না কেন, তাই আমরা সেগুলোকে এখানে সাজাই।
1 require(token0 != address(0), 'UniswapV2: ZERO_ADDRESS');2 require(getPair[token0][token1] == address(0), 'UniswapV2: PAIR_EXISTS'); // একটি চেকই যথেষ্টবড় লিকুইডিটি পুলগুলো ছোটগুলোর চেয়ে ভালো, কারণ সেগুলোর মূল্য বেশি স্থিতিশীল। আমরা প্রতি জোড়া টোকেনের জন্য একটির বেশি লিকুইডিটি পুল রাখতে চাই না। যদি আগে থেকেই একটি এক্সচেঞ্জ থাকে, তবে একই পেয়ারের জন্য আরেকটি তৈরি করার কোনো প্রয়োজন নেই।
1 bytes memory bytecode = type(UniswapV2Pair).creationCode;একটি নতুন কন্ট্রাক্ট তৈরি করতে আমাদের সেই কোডটি প্রয়োজন যা এটি তৈরি করে (কনস্ট্রাক্টর ফাংশন এবং সেই কোড উভয়ই যা আসল কন্ট্রাক্টের EVM বাইটকোড মেমরিতে লেখে)। সাধারণত Solidity-তে আমরা শুধু addr = new <name of contract>(<constructor parameters>) ব্যবহার করি এবং কম্পাইলার আমাদের জন্য সবকিছুর যত্ন নেয়, কিন্তু একটি ডিটারমিনিস্টিক কন্ট্রাক্ট এডড্রেস পেতে আমাদের CREATE2 অপকোড (opens in a new tab) ব্যবহার করতে হবে।
যখন এই কোডটি লেখা হয়েছিল তখন সেই অপকোডটি Solidity দ্বারা সমর্থিত ছিল না, তাই ম্যানুয়ালি কোডটি পাওয়া প্রয়োজনীয় ছিল। এটি আর কোনো সমস্যা নয়, কারণ Solidity এখন CREATE2 সমর্থন করে (opens in a new tab)।
1 bytes32 salt = keccak256(abi.encodePacked(token0, token1));2 assembly {3 pair := create2(0, add(bytecode, 32), mload(bytecode), salt)4 }যখন কোনো অপকোড এখনও Solidity দ্বারা সমর্থিত নয় তখন আমরা ইনলাইন অ্যাসেম্বলি (opens in a new tab) ব্যবহার করে এটিকে কল করতে পারি।
1 IUniswapV2Pair(pair).initialize(token0, token1);নতুন এক্সচেঞ্জকে এটি কোন দুটি টোকেন এক্সচেঞ্জ করে তা জানাতে initialize ফাংশনটি কল করুন।
1 getPair[token0][token1] = pair;2 getPair[token1][token0] = pair; // বিপরীত দিকে ম্যাপিং পপুলেট করে3 allPairs.push(pair);4 emit PairCreated(token0, token1, pair, allPairs.length);5 }স্টেট ভেরিয়েবলগুলোতে নতুন পেয়ারের তথ্য সংরক্ষণ করুন এবং বিশ্বকে নতুন পেয়ার এক্সচেঞ্জ সম্পর্কে জানাতে একটি ইভেন্ট এমিট করুন।
1 function setFeeTo(address _feeTo) external {2 require(msg.sender == feeToSetter, 'UniswapV2: FORBIDDEN');3 feeTo = _feeTo;4 }5
6 function setFeeToSetter(address _feeToSetter) external {7 require(msg.sender == feeToSetter, 'UniswapV2: FORBIDDEN');8 feeToSetter = _feeToSetter;9 }10}এই দুটি ফাংশন feeSetter-কে ফি প্রাপক (যদি থাকে) নিয়ন্ত্রণ করতে এবং feeSetter-কে একটি নতুন এডড্রেস-এ পরিবর্তন করার অনুমতি দেয়।
UniswapV2ERC20.sol
এই কন্ট্রাক্টটি (opens in a new tab) ERC-20 লিকুইডিটি টোকেন বাস্তবায়ন করে। এটি OpenZeppelin ERC-20 কন্ট্রাক্ট-এর মতোই, তাই আমি শুধুমাত্র সেই অংশটি ব্যাখ্যা করব যা ভিন্ন, permit কার্যকারিতা।
ইথিরিয়াম-এ লেনদেন-এর জন্য ইথার (ETH) খরচ হয়, যা আসল টাকার সমতুল্য। যদি আপনার কাছে ERC-20 টোকেন থাকে কিন্তু ETH না থাকে, তবে আপনি লেনদেন পাঠাতে পারবেন না, তাই আপনি সেগুলো দিয়ে কিছুই করতে পারবেন না। এই সমস্যা এড়ানোর একটি সমাধান হলো মেটা-ট্রানজেকশন (opens in a new tab)। টোকেনের মালিক এমন একটি লেনদেন-এ স্বাক্ষর করেন যা অন্য কাউকে অফচেইন টোকেন তুলে নেওয়ার অনুমতি দেয় এবং এটি ইন্টারনেট ব্যবহার করে প্রাপকের কাছে পাঠায়। প্রাপক, যার কাছে ETH আছে, তারপর মালিকের পক্ষে পারমিট জমা দেয়।
1 bytes32 public DOMAIN_SEPARATOR;2 // keccak256("Permit(address owner,address spender,uint256 value,uint256 nonce,uint256 deadline)");3 bytes32 public constant PERMIT_TYPEHASH = 0x6e71edae12b1b97f4d1f60370fef10105fa2faae0126114a169c64845d6126c9;এই হ্যাস হলো লেনদেন টাইপের জন্য আইডেন্টিফায়ার (opens in a new tab)। আমরা এখানে শুধুমাত্র এই প্যারামিটারগুলোর সাথে Permit সমর্থন করি।
1 mapping(address => uint) public nonces;একজন প্রাপকের পক্ষে একটি ডিজিটাল সিগনেচার জাল করা সম্ভব নয়। তবে, একই লেনদেন দুবার পাঠানো খুবই সহজ (এটি এক ধরণের রিপ্লে অ্যাটাক (opens in a new tab))। এটি প্রতিরোধ করতে, আমরা একটি নন্স (opens in a new tab) ব্যবহার করি। যদি একটি নতুন Permit-এর নন্স সর্বশেষ ব্যবহৃত নন্সের চেয়ে এক বেশি না হয়, তবে আমরা ধরে নিই যে এটি অবৈধ।
1 constructor() public {2 uint chainId;3 assembly {4 chainId := chainid5 }এটি চেইন আইডেন্টিফায়ার (opens in a new tab) পুনরুদ্ধার করার কোড। এটি Yul (opens in a new tab) নামক একটি EVM অ্যাসেম্বলি ডায়ালেক্ট ব্যবহার করে। মনে রাখবেন যে Yul-এর বর্তমান সংস্করণে আপনাকে chainid() ব্যবহার করতে হবে, chainid নয়।
1 DOMAIN_SEPARATOR = keccak256(2 abi.encode(3 keccak256('EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)'),4 keccak256(bytes(name)),5 keccak256(bytes('1')),6 chainId,7 address(this)8 )9 );10 }EIP-712-এর জন্য ডোমেইন সেপারেটর (opens in a new tab) গণনা করুন।
1 function permit(address owner, address spender, uint value, uint deadline, uint8 v, bytes32 r, bytes32 s) external {এটি সেই ফাংশন যা পারমিশনগুলো বাস্তবায়ন করে। এটি প্যারামিটার হিসেবে প্রাসঙ্গিক ফিল্ডগুলো এবং সিগনেচারের (opens in a new tab) জন্য তিনটি স্কেলার মান (v, r, এবং s) গ্রহণ করে।
1 require(deadline >= block.timestamp, 'UniswapV2: EXPIRED');ডেডলাইনের পরে লেনদেন গ্রহণ করবেন না।
1 bytes32 digest = keccak256(2 abi.encodePacked(3 '\x19\x01',4 DOMAIN_SEPARATOR,5 keccak256(abi.encode(PERMIT_TYPEHASH, owner, spender, value, nonces[owner]++, deadline))6 )7 );abi.encodePacked(...) হলো সেই মেসেজ যা আমরা পাওয়ার আশা করি। আমরা জানি নন্স কী হওয়া উচিত, তাই আমাদের এটি প্যারামিটার হিসেবে পাওয়ার কোনো প্রয়োজন নেই।
ইথিরিয়াম সিগনেচার এ্যালগরিদম সাইন করার জন্য 256 বিট পাওয়ার আশা করে, তাই আমরা keccak256 হ্যাস ফাংশন ব্যবহার করি।
1 address recoveredAddress = ecrecover(digest, v, r, s);ডাইজেস্ট এবং সিগনেচার থেকে আমরা ecrecover (opens in a new tab) ব্যবহার করে সেই এডড্রেস পেতে পারি যা এটি সাইন করেছে।
1 require(recoveredAddress != address(0) && recoveredAddress == owner, 'UniswapV2: INVALID_SIGNATURE');2 _approve(owner, spender, value);3 }4
যদি সবকিছু ঠিক থাকে, তবে এটিকে একটি ERC-20 অ্যাপ্রুভ (opens in a new tab) হিসেবে বিবেচনা করুন।
পেরিফেরি কন্ট্রাক্টস
পেরিফেরি কন্ট্রাক্টগুলো হলো Uniswap-এর API (এপ্লিকেশন প্রোগ্রাম ইন্টারফেস)। এগুলো অন্যান্য কন্ট্রাক্ট বা ডিসেন্ট্রালাইজড এপ্লিকেশন থেকে এক্সটার্নাল কলের জন্য এভেইলেবল। আপনি চাইলে সরাসরি কোর কন্ট্রাক্টগুলোকে কল করতে পারেন, তবে এটি বেশ জটিল এবং ভুল করলে আপনার ভ্যালু বা ফান্ড হারাতে পারেন। কোর কন্ট্রাক্টগুলোতে শুধুমাত্র এমন টেস্ট থাকে যা নিশ্চিত করে যে তাদের সাথে কোনো প্রতারণা করা হচ্ছে না, অন্য কারো জন্য কোনো স্যানিটি চেক এতে থাকে না। সেগুলো পেরিফেরিতে থাকে যাতে প্রয়োজন অনুযায়ী আপডেট করা যায়।
UniswapV2Router01.sol
এই কন্ট্রাক্টে (opens in a new tab) কিছু সমস্যা রয়েছে এবং এটি আর ব্যবহার করা উচিত নয় (opens in a new tab)। সৌভাগ্যবশত, পেরিফেরি কন্ট্রাক্টগুলো স্টেটলেস এবং এগুলো কোনো অ্যাসেট ধরে রাখে না, তাই এটিকে সহজেই বাতিল করা যায় এবং এর পরিবর্তে মানুষকে UniswapV2Router02 ব্যবহার করার পরামর্শ দেওয়া যায়।
UniswapV2Router02.sol
বেশিরভাগ ক্ষেত্রেই আপনি এই কন্ট্রাক্টের (opens in a new tab) মাধ্যমে Uniswap ব্যবহার করবেন। এটি কীভাবে ব্যবহার করতে হয় তা আপনি এখানে (opens in a new tab) দেখতে পারেন।
1pragma solidity =0.6.6;2
3import '@uniswap/v2-core/contracts/interfaces/IUniswapV2Factory.sol';4import '@uniswap/lib/contracts/libraries/TransferHelper.sol';5
6import './interfaces/IUniswapV2Router02.sol';7import './libraries/UniswapV2Library.sol';8import './libraries/SafeMath.sol';9import './interfaces/IERC20.sol';10import './interfaces/IWETH.sol';এগুলোর বেশিরভাগই আমরা আগে দেখেছি, অথবা বেশ স্পষ্ট। একমাত্র ব্যতিক্রম হলো IWETH.sol। Uniswap v2 যেকোনো জোড়া ERC-20 টোকেনের জন্য এক্সচেঞ্জ করার অনুমতি দেয়, কিন্তু ইথার (ETH) নিজে কোনো ERC-20 টোকেন নয়। এটি স্ট্যান্ডার্ডের আগের এবং ইউনিক মেকানিজমের মাধ্যমে ট্রান্সফার করা হয়। ERC-20 টোকেনের ক্ষেত্রে প্রযোজ্য কন্ট্রাক্টগুলোতে ETH-এর ব্যবহার সক্ষম করতে মানুষ wrapped ether (WETH) (opens in a new tab) কন্ট্রাক্ট তৈরি করেছে। আপনি এই কন্ট্রাক্টে ETH পাঠালে, এটি আপনাকে সমপরিমাণ WETH মিন্ট করে দেয়। অথবা আপনি WETH বার্ন করে, আবার ETH ফেরত পেতে পারেন।
1contract UniswapV2Router02 is IUniswapV2Router02 {2 using SafeMath for uint;3
4 address public immutable override factory;5 address public immutable override WETH;রাউটারকে জানতে হবে কোন ফ্যাক্টরি ব্যবহার করতে হবে, এবং যেসব লেনদেনের জন্য WETH প্রয়োজন সেগুলোর জন্য কোন WETH কন্ট্রাক্ট ব্যবহার করতে হবে। এই ভ্যালুগুলো ইমমিউটেবল (opens in a new tab), যার মানে এগুলো শুধুমাত্র কনস্ট্রাক্টরে সেট করা যায়। এটি ব্যবহারকারীদের এই আত্মবিশ্বাস দেয় যে কেউ এগুলো পরিবর্তন করে কম সৎ কন্ট্রাক্টের দিকে নির্দেশ করতে পারবে না।
1 modifier ensure(uint deadline) {2 require(deadline >= block.timestamp, 'UniswapV2Router: EXPIRED');3 _;4 }এই মডিফায়ারটি নিশ্চিত করে যে সময়-সীমিত লেনদেনগুলো ("যদি পারো তবে Y সময়ের আগে X করো") তাদের সময়সীমা পার হওয়ার পরে যেন না ঘটে।
1 constructor(address _factory, address _WETH) public {2 factory = _factory;3 WETH = _WETH;4 }কনস্ট্রাক্টরটি শুধুমাত্র ইমমিউটেবল স্টেট ভেরিয়েবলগুলো সেট করে।
1 receive() external payable {2 assert(msg.sender == WETH); // শুধুমাত্র WETH কন্ট্রাক্ট থেকে ফলব্যাকের মাধ্যমে ETH গ্রহণ করে3 }যখন আমরা WETH কন্ট্রাক্ট থেকে টোকেন রিডিম করে আবার ETH-এ রূপান্তর করি তখন এই ফাংশনটি কল করা হয়। শুধুমাত্র আমরা যে WETH কন্ট্রাক্টটি ব্যবহার করি তারই এটি করার অনুমতি রয়েছে।
লিকুইডিটি যোগ করা
এই ফাংশনগুলো পেয়ার এক্সচেঞ্জে টোকেন যোগ করে, যা লিকুইডিটি পুল বৃদ্ধি করে।
1
2 // **** লিকুইডিটি যোগ করুন ****3 function _addLiquidity(এই ফাংশনটি পেয়ার এক্সচেঞ্জে জমা করার জন্য A এবং B টোকেনের পরিমাণ গণনা করতে ব্যবহৃত হয়।
1 address tokenA,2 address tokenB,এগুলো হলো ERC-20 টোকেন কন্ট্রাক্টগুলোর এডড্রেস।
1 uint amountADesired,2 uint amountBDesired,এগুলো হলো সেই পরিমাণ যা লিকুইডিটি প্রোভাইডার জমা করতে চান। এগুলো জমা করার জন্য A এবং B-এর সর্বোচ্চ পরিমাণও বটে।
1 uint amountAMin,2 uint amountBMinএগুলো হলো জমা করার জন্য সর্বনিম্ন গ্রহণযোগ্য পরিমাণ। যদি এই পরিমাণ বা তার বেশি দিয়ে লেনদেন সম্পন্ন করা না যায়, তবে এটি রিভার্ট হয়ে যাবে। আপনি যদি এই ফিচারটি না চান, তবে শুধু শূন্য (0) উল্লেখ করুন।
লিকুইডিটি প্রোভাইডাররা সাধারণত একটি সর্বনিম্ন পরিমাণ নির্দিষ্ট করে দেন, কারণ তারা লেনদেনটিকে বর্তমান এক্সচেঞ্জ রেটের কাছাকাছি সীমাবদ্ধ রাখতে চান। যদি এক্সচেঞ্জ রেট খুব বেশি ওঠানামা করে, তবে এর মানে হতে পারে এমন কোনো খবর যা আন্ডারলাইং ভ্যালু পরিবর্তন করে দিয়েছে, এবং তারা ম্যানুয়ালি সিদ্ধান্ত নিতে চান যে কী করতে হবে।
উদাহরণস্বরূপ, এমন একটি পরিস্থিতি কল্পনা করুন যেখানে এক্সচেঞ্জ রেট এক-এর বিপরীতে এক (1:1) এবং লিকুইডিটি প্রোভাইডার এই ভ্যালুগুলো নির্দিষ্ট করেছেন:
| প্যারামিটার | ভ্যালু |
|---|---|
| amountADesired | 1000 |
| amountBDesired | 1000 |
| amountAMin | 900 |
| amountBMin | 800 |
যতক্ষণ এক্সচেঞ্জ রেট 0.9 এবং 1.25 এর মধ্যে থাকে, ততক্ষণ লেনদেনটি সম্পন্ন হয়। যদি এক্সচেঞ্জ রেট এই রেঞ্জের বাইরে চলে যায়, তবে লেনদেনটি বাতিল হয়ে যায়।
এই সতর্কতার কারণ হলো লেনদেনগুলো তাৎক্ষণিক নয়, আপনি সেগুলো সাবমিট করেন এবং শেষ পর্যন্ত একজন ভ্যালিডেটর সেগুলোকে একটি ব্লকে অন্তর্ভুক্ত করে (যদি না আপনার গ্যাস প্রাইস খুব কম হয়, সেক্ষেত্রে এটিকে ওভাররাইট করার জন্য আপনাকে একই নন্স এবং উচ্চতর গ্যাস প্রাইস দিয়ে আরেকটি লেনদেন সাবমিট করতে হবে)। সাবমিশন এবং অন্তর্ভুক্তির মধ্যবর্তী সময়ে কী ঘটে তা আপনি নিয়ন্ত্রণ করতে পারবেন না।
1 ) internal virtual returns (uint amountA, uint amountB) {রিজার্ভগুলোর বর্তমান অনুপাতের সমান অনুপাত বজায় রাখতে লিকুইডিটি প্রোভাইডারের যে পরিমাণ জমা করা উচিত, ফাংশনটি সেই পরিমাণ রিটার্ন করে।
1 // পেয়ারটি এখনও না থাকলে তৈরি করুন2 if (IUniswapV2Factory(factory).getPair(tokenA, tokenB) == address(0)) {3 IUniswapV2Factory(factory).createPair(tokenA, tokenB);4 }যদি এই টোকেন পেয়ারের জন্য এখনও কোনো এক্সচেঞ্জ না থাকে, তবে এটি তৈরি করুন।
1 (uint reserveA, uint reserveB) = UniswapV2Library.getReserves(factory, tokenA, tokenB);পেয়ারের বর্তমান রিজার্ভগুলো পান।
1 if (reserveA == 0 && reserveB == 0) {2 (amountA, amountB) = (amountADesired, amountBDesired);যদি বর্তমান রিজার্ভগুলো খালি থাকে তবে এটি একটি নতুন পেয়ার এক্সচেঞ্জ। জমা করার পরিমাণ ঠিক ততটাই হওয়া উচিত যতটা লিকুইডিটি প্রোভাইডার প্রদান করতে চান।
1 } else {2 uint amountBOptimal = UniswapV2Library.quote(amountADesired, reserveA, reserveB);যদি আমাদের দেখতে হয় যে পরিমাণগুলো কী হবে, তবে আমরা এই ফাংশনটি (opens in a new tab) ব্যবহার করে অপ্টিমাল পরিমাণ পেতে পারি। আমরা বর্তমান রিজার্ভের সমান অনুপাত চাই।
1 if (amountBOptimal <= amountBDesired) {2 require(amountBOptimal >= amountBMin, 'UniswapV2Router: INSUFFICIENT_B_AMOUNT');3 (amountA, amountB) = (amountADesired, amountBOptimal);যদি amountBOptimal লিকুইডিটি প্রোভাইডারের জমা করতে চাওয়া পরিমাণের চেয়ে ছোট হয়, তবে এর মানে হলো টোকেন B বর্তমানে লিকুইডিটি জমাকারীর ধারণার চেয়ে বেশি মূল্যবান, তাই কম পরিমাণ প্রয়োজন।
1 } else {2 uint amountAOptimal = UniswapV2Library.quote(amountBDesired, reserveB, reserveA);3 assert(amountAOptimal <= amountADesired);4 require(amountAOptimal >= amountAMin, 'UniswapV2Router: INSUFFICIENT_A_AMOUNT');5 (amountA, amountB) = (amountAOptimal, amountBDesired);যদি অপ্টিমাল B পরিমাণ কাঙ্ক্ষিত B পরিমাণের চেয়ে বেশি হয়, তবে এর মানে হলো B টোকেনগুলো বর্তমানে লিকুইডিটি জমাকারীর ধারণার চেয়ে কম মূল্যবান, তাই বেশি পরিমাণ প্রয়োজন। তবে, কাঙ্ক্ষিত পরিমাণটি হলো সর্বোচ্চ সীমা, তাই আমরা তা করতে পারি না। এর পরিবর্তে আমরা কাঙ্ক্ষিত B টোকেনের পরিমাণের জন্য A টোকেনের অপ্টিমাল সংখ্যা গণনা করি।
সবকিছু একসাথে মেলালে আমরা এই গ্রাফটি পাই। ধরুন আপনি এক হাজার A টোকেন (নীল রেখা) এবং এক হাজার B টোকেন (লাল রেখা) জমা করার চেষ্টা করছেন। x অক্ষ হলো এক্সচেঞ্জ রেট, A/B। যদি x=1 হয়, তবে তাদের ভ্যালু সমান এবং আপনি প্রতিটির এক হাজার করে জমা করবেন। যদি x=2 হয়, তবে A-এর ভ্যালু B-এর দ্বিগুণ (আপনি প্রতিটি A টোকেনের জন্য দুটি B টোকেন পাবেন) তাই আপনি এক হাজার B টোকেন জমা করবেন, কিন্তু মাত্র 500 A টোকেন। যদি x=0.5 হয়, তবে পরিস্থিতি উল্টো হবে, এক হাজার A টোকেন এবং পাঁচশ B টোকেন।
আপনি সরাসরি কোর কন্ট্রাক্টে লিকুইডিটি জমা করতে পারেন (UniswapV2Pair::mint (opens in a new tab) ব্যবহার করে), কিন্তু কোর কন্ট্রাক্ট শুধুমাত্র চেক করে যে তার নিজের সাথে প্রতারণা করা হচ্ছে না, তাই আপনি লেনদেন সাবমিট করা এবং তা এক্সিকিউট হওয়ার মধ্যবর্তী সময়ে এক্সচেঞ্জ রেট পরিবর্তিত হলে ভ্যালু হারানোর ঝুঁকিতে থাকেন। আপনি যদি পেরিফেরি কন্ট্রাক্ট ব্যবহার করেন, তবে এটি আপনার জমা করার পরিমাণ হিসাব করে এবং তা সাথে সাথে জমা করে দেয়, ফলে এক্সচেঞ্জ রেট পরিবর্তন হয় না এবং আপনি কিছুই হারান না।
1 function addLiquidity(2 address tokenA,3 address tokenB,4 uint amountADesired,5 uint amountBDesired,6 uint amountAMin,7 uint amountBMin,8 address to,9 uint deadlineলিকুইডিটি জমা করার জন্য একটি লেনদেনের মাধ্যমে এই ফাংশনটি কল করা যেতে পারে। বেশিরভাগ প্যারামিটার উপরের _addLiquidity-এর মতোই, তবে দুটি ব্যতিক্রম রয়েছে:
. to হলো সেই এডড্রেস যা পুলে লিকুইডিটি প্রোভাইডারের অংশ দেখানোর জন্য নতুন মিন্ট করা লিকুইডিটি টোকেনগুলো পায়
. deadline হলো লেনদেনের জন্য একটি সময়সীমা
1 ) external virtual override ensure(deadline) returns (uint amountA, uint amountB, uint liquidity) {2 (amountA, amountB) = _addLiquidity(tokenA, tokenB, amountADesired, amountBDesired, amountAMin, amountBMin);3 address pair = UniswapV2Library.pairFor(factory, tokenA, tokenB);আমরা প্রকৃতপক্ষে জমা করার পরিমাণ গণনা করি এবং তারপর লিকুইডিটি পুলের এডড্রেস খুঁজে বের করি। গ্যাস বাঁচানোর জন্য আমরা ফ্যাক্টরিকে জিজ্ঞাসা করে এটি করি না, বরং লাইব্রেরি ফাংশন pairFor ব্যবহার করি (নিচে লাইব্রেরি অংশে দেখুন)
1 TransferHelper.safeTransferFrom(tokenA, msg.sender, pair, amountA);2 TransferHelper.safeTransferFrom(tokenB, msg.sender, pair, amountB);ব্যবহারকারীর কাছ থেকে সঠিক পরিমাণের টোকেন পেয়ার এক্সচেঞ্জে ট্রান্সফার করুন।
1 liquidity = IUniswapV2Pair(pair).mint(to);2 }এর বিনিময়ে পুলের আংশিক মালিকানার জন্য to এডড্রেসকে লিকুইডিটি টোকেন দিন। কোর কন্ট্রাক্টের mint ফাংশন দেখে যে এর কাছে কতগুলো অতিরিক্ত টোকেন আছে (শেষবার লিকুইডিটি পরিবর্তন হওয়ার সময়ের তুলনায়) এবং সেই অনুযায়ী লিকুইডিটি মিন্ট করে।
1 function addLiquidityETH(2 address token,3 uint amountTokenDesired,যখন একজন লিকুইডিটি প্রোভাইডার একটি Token/ETH পেয়ার এক্সচেঞ্জে লিকুইডিটি প্রদান করতে চান, তখন কিছু পার্থক্য থাকে। কন্ট্রাক্টটি লিকুইডিটি প্রোভাইডারের জন্য ETH র্যাপ করার বিষয়টি পরিচালনা করে। ব্যবহারকারী কত ETH জমা করতে চান তা নির্দিষ্ট করার কোনো প্রয়োজন নেই, কারণ ব্যবহারকারী কেবল লেনদেনের সাথে সেগুলো পাঠিয়ে দেন (পরিমাণটি msg.value-তে এভেইলেবল থাকে)।
1 uint amountTokenMin,2 uint amountETHMin,3 address to,4 uint deadline5 ) external virtual override payable ensure(deadline) returns (uint amountToken, uint amountETH, uint liquidity) {6 (amountToken, amountETH) = _addLiquidity(7 token,8 WETH,9 amountTokenDesired,10 msg.value,11 amountTokenMin,12 amountETHMin13 );14 address pair = UniswapV2Library.pairFor(factory, token, WETH);15 TransferHelper.safeTransferFrom(token, msg.sender, pair, amountToken);16 IWETH(WETH).deposit{value: amountETH}();17 assert(IWETH(WETH).transfer(pair, amountETH));ETH জমা করার জন্য কন্ট্রাক্টটি প্রথমে এটিকে WETH-এ র্যাপ করে এবং তারপর WETH-কে পেয়ারে ট্রান্সফার করে। লক্ষ্য করুন যে ট্রান্সফারটি একটি assert-এ র্যাপ করা আছে। এর মানে হলো যদি ট্রান্সফারটি ব্যর্থ হয় তবে এই কন্ট্রাক্ট কলটিও ব্যর্থ হয়, এবং তাই র্যাপিংটি আসলে ঘটে না।
1 liquidity = IUniswapV2Pair(pair).mint(to);2 // ডাস্ট (dust) eth থাকলে তা রিফান্ড করুন3 if (msg.value > amountETH) TransferHelper.safeTransferETH(msg.sender, msg.value - amountETH);4 }ব্যবহারকারী ইতিমধ্যেই আমাদের ETH পাঠিয়েছেন, তাই যদি কোনো অতিরিক্ত অংশ অবশিষ্ট থাকে (কারণ অন্য টোকেনটি ব্যবহারকারীর ধারণার চেয়ে কম মূল্যবান), তবে আমাদের একটি রিফান্ড ইস্যু করতে হবে।
লিকুইডিটি সরানো
এই ফাংশনগুলো লিকুইডিটি সরিয়ে ফেলবে এবং লিকুইডিটি প্রোভাইডারকে অর্থ ফেরত দেবে।
1 // **** লিকুইডিটি সরান ****2 function removeLiquidity(3 address tokenA,4 address tokenB,5 uint liquidity,6 uint amountAMin,7 uint amountBMin,8 address to,9 uint deadline10 ) public virtual override ensure(deadline) returns (uint amountA, uint amountB) {লিকুইডিটি সরানোর সবচেয়ে সহজ ক্ষেত্র। প্রতিটি টোকেনের একটি ন্যূনতম পরিমাণ রয়েছে যা লিকুইডিটি প্রোভাইডার গ্রহণ করতে সম্মত হন, এবং এটি অবশ্যই ডেডলাইনের আগে ঘটতে হবে।
1 address pair = UniswapV2Library.pairFor(factory, tokenA, tokenB);2 IUniswapV2Pair(pair).transferFrom(msg.sender, pair, liquidity); // পেয়ারে লিকুইডিটি পাঠান3 (uint amount0, uint amount1) = IUniswapV2Pair(pair).burn(to);কোর কন্ট্রাক্টের burn ফাংশন ব্যবহারকারীকে টোকেন ফেরত দেওয়ার বিষয়টি পরিচালনা করে।
1 (address token0,) = UniswapV2Library.sortTokens(tokenA, tokenB);যখন একটি ফাংশন একাধিক ভ্যালু রিটার্ন করে, কিন্তু আমরা শুধুমাত্র সেগুলোর কয়েকটিতে আগ্রহী থাকি, তখন আমরা এভাবেই শুধুমাত্র সেই ভ্যালুগুলো পাই। একটি ভ্যালু রিড করে তা কখনোই ব্যবহার না করার চেয়ে গ্যাসের দিক থেকে এটি কিছুটা সস্তা।
1 (amountA, amountB) = tokenA == token0 ? (amount0, amount1) : (amount1, amount0);কোর কন্ট্রাক্ট যেভাবে পরিমাণগুলো রিটার্ন করে (প্রথমে লোয়ার এডড্রেস টোকেন) তা থেকে ব্যবহারকারী যেভাবে আশা করে ( tokenA এবং tokenB অনুযায়ী) সেভাবে পরিমাণগুলোকে রূপান্তর করুন।
1 require(amountA >= amountAMin, 'UniswapV2Router: INSUFFICIENT_A_AMOUNT');2 require(amountB >= amountBMin, 'UniswapV2Router: INSUFFICIENT_B_AMOUNT');3 }প্রথমে ট্রান্সফার করা এবং তারপর এটি বৈধ কিনা তা যাচাই করা ঠিক আছে, কারণ যদি এটি বৈধ না হয় তবে আমরা সমস্ত স্টেট পরিবর্তনগুলো রিভার্ট করে দেব।
1 function removeLiquidityETH(2 address token,3 uint liquidity,4 uint amountTokenMin,5 uint amountETHMin,6 address to,7 uint deadline8 ) public virtual override ensure(deadline) returns (uint amountToken, uint amountETH) {9 (amountToken, amountETH) = removeLiquidity(10 token,11 WETH,12 liquidity,13 amountTokenMin,14 amountETHMin,15 address(this),16 deadline17 );18 TransferHelper.safeTransfer(token, to, amountToken);19 IWETH(WETH).withdraw(amountETH);20 TransferHelper.safeTransferETH(to, amountETH);21 }ETH-এর জন্য লিকুইডিটি সরানো প্রায় একই রকম, শুধু পার্থক্য হলো আমরা WETH টোকেনগুলো গ্রহণ করি এবং তারপর লিকুইডিটি প্রোভাইডারকে ফেরত দেওয়ার জন্য সেগুলোকে ETH-এ রিডিম করি।
1 function removeLiquidityWithPermit(2 address tokenA,3 address tokenB,4 uint liquidity,5 uint amountAMin,6 uint amountBMin,7 address to,8 uint deadline,9 bool approveMax, uint8 v, bytes32 r, bytes32 s10 ) external virtual override returns (uint amountA, uint amountB) {11 address pair = UniswapV2Library.pairFor(factory, tokenA, tokenB);12 uint value = approveMax ? uint(-1) : liquidity;13 IUniswapV2Pair(pair).permit(msg.sender, address(this), value, deadline, v, r, s);14 (amountA, amountB) = removeLiquidity(tokenA, tokenB, liquidity, amountAMin, amountBMin, to, deadline);15 }16
17
18 function removeLiquidityETHWithPermit(19 address token,20 uint liquidity,21 uint amountTokenMin,22 uint amountETHMin,23 address to,24 uint deadline,25 bool approveMax, uint8 v, bytes32 r, bytes32 s26 ) external virtual override returns (uint amountToken, uint amountETH) {27 address pair = UniswapV2Library.pairFor(factory, token, WETH);28 uint value = approveMax ? uint(-1) : liquidity;29 IUniswapV2Pair(pair).permit(msg.sender, address(this), value, deadline, v, r, s);30 (amountToken, amountETH) = removeLiquidityETH(token, liquidity, amountTokenMin, amountETHMin, to, deadline);31 }এই ফাংশনগুলো মেটা-ট্রানজেকশন রিলে করে যাতে ইথার ছাড়া ব্যবহারকারীরা পারমিট মেকানিজম ব্যবহার করে পুল থেকে উইথড্র করতে পারে।
1
2 // **** লিকুইডিটি সরান (ফি-অন-ট্রান্সফার টোকেন সমর্থন করে) ****3 function removeLiquidityETHSupportingFeeOnTransferTokens(4 address token,5 uint liquidity,6 uint amountTokenMin,7 uint amountETHMin,8 address to,9 uint deadline10 ) public virtual override ensure(deadline) returns (uint amountETH) {11 (, amountETH) = removeLiquidity(12 token,13 WETH,14 liquidity,15 amountTokenMin,16 amountETHMin,17 address(this),18 deadline19 );20 TransferHelper.safeTransfer(token, to, IERC20(token).balanceOf(address(this)));21 IWETH(WETH).withdraw(amountETH);22 TransferHelper.safeTransferETH(to, amountETH);23 }24
এই ফাংশনটি এমন টোকেনগুলোর জন্য ব্যবহার করা যেতে পারে যেগুলোর ট্রান্সফার বা স্টোরেজ ফি রয়েছে। যখন কোনো টোকেনের এমন ফি থাকে তখন আমরা কতটা টোকেন ফেরত পাব তা জানার জন্য removeLiquidity ফাংশনের ওপর নির্ভর করতে পারি না, তাই আমাদের প্রথমে উইথড্র করতে হবে এবং তারপর ব্যালেন্স জানতে হবে।
1
2
3 function removeLiquidityETHWithPermitSupportingFeeOnTransferTokens(4 address token,5 uint liquidity,6 uint amountTokenMin,7 uint amountETHMin,8 address to,9 uint deadline,10 bool approveMax, uint8 v, bytes32 r, bytes32 s11 ) external virtual override returns (uint amountETH) {12 address pair = UniswapV2Library.pairFor(factory, token, WETH);13 uint value = approveMax ? uint(-1) : liquidity;14 IUniswapV2Pair(pair).permit(msg.sender, address(this), value, deadline, v, r, s);15 amountETH = removeLiquidityETHSupportingFeeOnTransferTokens(16 token, liquidity, amountTokenMin, amountETHMin, to, deadline17 );18 }চূড়ান্ত ফাংশনটি মেটা-ট্রানজেকশনের সাথে স্টোরেজ ফি যুক্ত করে।
ট্রেড
1 // **** সোয়াপ ****2 // প্রাথমিক পরিমাণটি ইতিমধ্যে প্রথম পেয়ারে পাঠানো হয়েছে তা প্রয়োজন3 function _swap(uint[] memory amounts, address[] memory path, address _to) internal virtual {এই ফাংশনটি ইন্টারনাল প্রসেসিং সম্পন্ন করে যা ট্রেডারদের জন্য উন্মুক্ত ফাংশনগুলোর জন্য প্রয়োজনীয়।
1 for (uint i; i < path.length - 1; i++) {আমি যখন এটি লিখছি তখন 388,160টি ERC-20 টোকেন (opens in a new tab) রয়েছে। যদি প্রতিটি টোকেন পেয়ারের জন্য একটি করে পেয়ার এক্সচেঞ্জ থাকত, তবে এটি 150 বিলিয়নেরও বেশি পেয়ার এক্সচেঞ্জ হতো। এই মুহূর্তে পুরো চেইনে সেই সংখ্যার মাত্র 0.1% একাউন্ট রয়েছে (opens in a new tab)। এর পরিবর্তে, সোয়াপ ফাংশনগুলো একটি পাথের কনসেপ্ট সমর্থন করে। একজন ট্রেডার A-এর বিনিময়ে B, B-এর বিনিময়ে C, এবং C-এর বিনিময়ে D এক্সচেঞ্জ করতে পারেন, তাই সরাসরি A-D পেয়ার এক্সচেঞ্জের কোনো প্রয়োজন নেই।
এই মার্কেটগুলোতে দামগুলো সাধারণত সিঙ্ক্রোনাইজড থাকে, কারণ যখন সেগুলো সিঙ্কের বাইরে থাকে তখন এটি আর্বিট্রেজের সুযোগ তৈরি করে। উদাহরণস্বরূপ, তিনটি টোকেন A, B এবং C কল্পনা করুন। প্রতিটি পেয়ারের জন্য একটি করে মোট তিনটি পেয়ার এক্সচেঞ্জ রয়েছে।
- প্রাথমিক পরিস্থিতি
- একজন ট্রেডার 24.695টি A টোকেন বিক্রি করে 25.305টি B টোকেন পান।
- ট্রেডার 25.305টি C টোকেনের বিনিময়ে 24.695টি B টোকেন বিক্রি করেন, এবং প্রায় 0.61টি B টোকেন লাভ হিসেবে রাখেন।
- তারপর ট্রেডার 25.305টি A টোকেনের বিনিময়ে 24.695টি C টোকেন বিক্রি করেন, এবং প্রায় 0.61টি C টোকেন লাভ হিসেবে রাখেন। ট্রেডারের কাছে 0.61টি অতিরিক্ত A টোকেনও থাকে (ট্রেডারের কাছে শেষে থাকা 25.305টি, বিয়োগ মূল বিনিয়োগ 24.695টি)।
| ধাপ | A-B এক্সচেঞ্জ | B-C এক্সচেঞ্জ | A-C এক্সচেঞ্জ |
|---|---|---|---|
| 1 | A:1000 B:1050 A/B=1.05 | B:1000 C:1050 B/C=1.05 | A:1050 C:1000 C/A=1.05 |
| 2 | A:1024.695 B:1024.695 A/B=1 | B:1000 C:1050 B/C=1.05 | A:1050 C:1000 C/A=1.05 |
| 3 | A:1024.695 B:1024.695 A/B=1 | B:1024.695 C:1024.695 B/C=1 | A:1050 C:1000 C/A=1.05 |
| 4 | A:1024.695 B:1024.695 A/B=1 | B:1024.695 C:1024.695 B/C=1 | A:1024.695 C:1024.695 C/A=1 |
1 (address input, address output) = (path[i], path[i + 1]);2 (address token0,) = UniswapV2Library.sortTokens(input, output);3 uint amountOut = amounts[i + 1];আমরা বর্তমানে যে পেয়ারটি হ্যান্ডেল করছি তা পান, এটিকে সর্ট করুন (পেয়ারের সাথে ব্যবহারের জন্য) এবং প্রত্যাশিত আউটপুট পরিমাণ পান।
1 (uint amount0Out, uint amount1Out) = input == token0 ? (uint(0), amountOut) : (amountOut, uint(0));প্রত্যাশিত আউট পরিমাণগুলো পান, যা পেয়ার এক্সচেঞ্জের প্রত্যাশা অনুযায়ী সর্ট করা থাকে।
1 address to = i < path.length - 2 ? UniswapV2Library.pairFor(factory, output, path[i + 2]) : _to;এটি কি শেষ এক্সচেঞ্জ? যদি তাই হয়, তবে ট্রেডের জন্য প্রাপ্ত টোকেনগুলো গন্তব্যে পাঠান। যদি না হয়, তবে এটিকে পরবর্তী পেয়ার এক্সচেঞ্জে পাঠান।
1
2 IUniswapV2Pair(UniswapV2Library.pairFor(factory, input, output)).swap(3 amount0Out, amount1Out, to, new bytes(0)4 );5 }6 }টোকেনগুলো সোয়াপ করার জন্য প্রকৃতপক্ষে পেয়ার এক্সচেঞ্জকে কল করুন। এক্সচেঞ্জ সম্পর্কে জানানোর জন্য আমাদের কোনো কলব্যাকের প্রয়োজন নেই, তাই আমরা সেই ফিল্ডে কোনো বাইট পাঠাই না।
1 function swapExactTokensForTokens(এই ফাংশনটি ট্রেডাররা সরাসরি একটি টোকেনের বিনিময়ে অন্যটি সোয়াপ করতে ব্যবহার করেন।
1 uint amountIn,2 uint amountOutMin,3 address[] calldata path,এই প্যারামিটারে ERC-20 কন্ট্রাক্টগুলোর এডড্রেস থাকে। উপরে যেমন ব্যাখ্যা করা হয়েছে, এটি একটি অ্যারে কারণ আপনার কাছে থাকা অ্যাসেট থেকে আপনার কাঙ্ক্ষিত অ্যাসেট পেতে আপনাকে বেশ কয়েকটি পেয়ার এক্সচেঞ্জের মধ্য দিয়ে যেতে হতে পারে।
সলিডিটিতে একটি ফাংশন প্যারামিটার memory বা calldata-তে স্টোর করা যেতে পারে। যদি ফাংশনটি কন্ট্রাক্টের একটি এন্ট্রি পয়েন্ট হয়, যা সরাসরি কোনো ব্যবহারকারী (একটি লেনদেন ব্যবহার করে) বা অন্য কোনো কন্ট্রাক্ট থেকে কল করা হয়, তবে প্যারামিটারের ভ্যালু সরাসরি কল ডাটা থেকে নেওয়া যেতে পারে। যদি ফাংশনটি ইন্টারনালি কল করা হয়, যেমন উপরের _swap, তবে প্যারামিটারগুলোকে memory-তে স্টোর করতে হবে। কল করা কন্ট্রাক্টের দৃষ্টিকোণ থেকে calldata হলো রিড অনলি।
uint বা address-এর মতো স্কেলার টাইপগুলোর ক্ষেত্রে কম্পাইলার আমাদের জন্য স্টোরেজ নির্বাচনের বিষয়টি পরিচালনা করে, কিন্তু অ্যারেগুলোর ক্ষেত্রে, যা দীর্ঘ এবং বেশি ব্যয়বহুল, আমরা কোন ধরনের স্টোরেজ ব্যবহার করতে হবে তা নির্দিষ্ট করে দিই।
1 address to,2 uint deadline3 ) external virtual override ensure(deadline) returns (uint[] memory amounts) {রিটার্ন ভ্যালুগুলো সবসময় মেমোরিতে রিটার্ন করা হয়।
1 amounts = UniswapV2Library.getAmountsOut(factory, amountIn, path);2 require(amounts[amounts.length - 1] >= amountOutMin, 'UniswapV2Router: INSUFFICIENT_OUTPUT_AMOUNT');প্রতিটি সোয়াপে ক্রয় করার পরিমাণ গণনা করুন। যদি ফলাফলটি ট্রেডারের গ্রহণ করতে ইচ্ছুক ন্যূনতম পরিমাণের চেয়ে কম হয়, তবে লেনদেনটি রিভার্ট করে দিন।
1 TransferHelper.safeTransferFrom(2 path[0], msg.sender, UniswapV2Library.pairFor(factory, path[0], path[1]), amounts[0]3 );4 _swap(amounts, path, to);5 }সবশেষে, প্রথম পেয়ার এক্সচেঞ্জের জন্য প্রাথমিক ERC-20 টোকেনটি একাউন্টে ট্রান্সফার করুন এবং _swap কল করুন। এই সবকিছু একই লেনদেনে ঘটছে, তাই পেয়ার এক্সচেঞ্জ জানে যে কোনো অপ্রত্যাশিত টোকেন এই ট্রান্সফারেরই অংশ।
1 function swapTokensForExactTokens(2 uint amountOut,3 uint amountInMax,4 address[] calldata path,5 address to,6 uint deadline7 ) external virtual override ensure(deadline) returns (uint[] memory amounts) {8 amounts = UniswapV2Library.getAmountsIn(factory, amountOut, path);9 require(amounts[0] <= amountInMax, 'UniswapV2Router: EXCESSIVE_INPUT_AMOUNT');10 TransferHelper.safeTransferFrom(11 path[0], msg.sender, UniswapV2Library.pairFor(factory, path[0], path[1]), amounts[0]12 );13 _swap(amounts, path, to);14 }আগের ফাংশন, swapTokensForTokens, একজন ট্রেডারকে তিনি ঠিক কতগুলো ইনপুট টোকেন দিতে ইচ্ছুক এবং এর বিনিময়ে তিনি ন্যূনতম কতগুলো আউটপুট টোকেন পেতে ইচ্ছুক তা নির্দিষ্ট করার অনুমতি দেয়। এই ফাংশনটি বিপরীত সোয়াপ করে, এটি একজন ট্রেডারকে তিনি কতগুলো আউটপুট টোকেন চান এবং সেগুলোর জন্য তিনি সর্বোচ্চ কতগুলো ইনপুট টোকেন দিতে ইচ্ছুক তা নির্দিষ্ট করতে দেয়।
উভয় ক্ষেত্রেই, ট্রেডারকে প্রথমে এই পেরিফেরি কন্ট্রাক্টটিকে একটি অ্যালাউন্স দিতে হবে যাতে এটি সেগুলোকে ট্রান্সফার করতে পারে।
1 function swapExactETHForTokens(uint amountOutMin, address[] calldata path, address to, uint deadline)2 external3 virtual4 override5 payable6 ensure(deadline)7 returns (uint[] memory amounts)8 {9 require(path[0] == WETH, 'UniswapV2Router: INVALID_PATH');10 amounts = UniswapV2Library.getAmountsOut(factory, msg.value, path);11 require(amounts[amounts.length - 1] >= amountOutMin, 'UniswapV2Router: INSUFFICIENT_OUTPUT_AMOUNT');12 IWETH(WETH).deposit{value: amounts[0]}();13 assert(IWETH(WETH).transfer(UniswapV2Library.pairFor(factory, path[0], path[1]), amounts[0]));14 _swap(amounts, path, to);15 }16
17
18 function swapTokensForExactETH(uint amountOut, uint amountInMax, address[] calldata path, address to, uint deadline)19 external20 virtual21 override22 ensure(deadline)23 returns (uint[] memory amounts)24 {25 require(path[path.length - 1] == WETH, 'UniswapV2Router: INVALID_PATH');26 amounts = UniswapV2Library.getAmountsIn(factory, amountOut, path);27 require(amounts[0] <= amountInMax, 'UniswapV2Router: EXCESSIVE_INPUT_AMOUNT');28 TransferHelper.safeTransferFrom(29 path[0], msg.sender, UniswapV2Library.pairFor(factory, path[0], path[1]), amounts[0]30 );31 _swap(amounts, path, address(this));32 IWETH(WETH).withdraw(amounts[amounts.length - 1]);33 TransferHelper.safeTransferETH(to, amounts[amounts.length - 1]);34 }35
36
37
38 function swapExactTokensForETH(uint amountIn, uint amountOutMin, address[] calldata path, address to, uint deadline)39 external40 virtual41 override42 ensure(deadline)43 returns (uint[] memory amounts)44 {45 require(path[path.length - 1] == WETH, 'UniswapV2Router: INVALID_PATH');46 amounts = UniswapV2Library.getAmountsOut(factory, amountIn, path);47 require(amounts[amounts.length - 1] >= amountOutMin, 'UniswapV2Router: INSUFFICIENT_OUTPUT_AMOUNT');48 TransferHelper.safeTransferFrom(49 path[0], msg.sender, UniswapV2Library.pairFor(factory, path[0], path[1]), amounts[0]50 );51 _swap(amounts, path, address(this));52 IWETH(WETH).withdraw(amounts[amounts.length - 1]);53 TransferHelper.safeTransferETH(to, amounts[amounts.length - 1]);54 }55
56
57 function swapETHForExactTokens(uint amountOut, address[] calldata path, address to, uint deadline)58 external59 virtual60 override61 payable62 ensure(deadline)63 returns (uint[] memory amounts)64 {65 require(path[0] == WETH, 'UniswapV2Router: INVALID_PATH');66 amounts = UniswapV2Library.getAmountsIn(factory, amountOut, path);67 require(amounts[0] <= msg.value, 'UniswapV2Router: EXCESSIVE_INPUT_AMOUNT');68 IWETH(WETH).deposit{value: amounts[0]}();69 assert(IWETH(WETH).transfer(UniswapV2Library.pairFor(factory, path[0], path[1]), amounts[0]));70 _swap(amounts, path, to);71 // ডাস্ট (dust) eth থাকলে তা রিফান্ড করুন72 if (msg.value > amounts[0]) TransferHelper.safeTransferETH(msg.sender, msg.value - amounts[0]);73 }এই চারটি ভ্যারিয়েন্টেই ETH এবং টোকেনের মধ্যে ট্রেডিং জড়িত। একমাত্র পার্থক্য হলো আমরা হয় ট্রেডারের কাছ থেকে ETH গ্রহণ করি এবং এটি ব্যবহার করে WETH মিন্ট করি, অথবা আমরা পাথের শেষ এক্সচেঞ্জ থেকে WETH গ্রহণ করি এবং এটি বার্ন করে, ট্রেডারকে এর ফলে প্রাপ্ত ETH ফেরত পাঠাই।
1 // **** সোয়াপ (ফি-অন-ট্রান্সফার টোকেন সমর্থন করে) ****2 // প্রাথমিক পরিমাণটি ইতিমধ্যে প্রথম পেয়ারে পাঠানো হয়েছে তা প্রয়োজন3 function _swapSupportingFeeOnTransferTokens(address[] memory path, address _to) internal virtual {এটি হলো ট্রান্সফার বা স্টোরেজ ফি থাকা টোকেনগুলো সোয়াপ করার ইন্টারনাল ফাংশন যা (এই সমস্যাটি (opens in a new tab)) সমাধান করে।
1 for (uint i; i < path.length - 1; i++) {2 (address input, address output) = (path[i], path[i + 1]);3 (address token0,) = UniswapV2Library.sortTokens(input, output);4 IUniswapV2Pair pair = IUniswapV2Pair(UniswapV2Library.pairFor(factory, input, output));5 uint amountInput;6 uint amountOutput;7 { // স্ট্যাক টু ডিপ (stack too deep) এরর এড়াতে স্কোপ8 (uint reserve0, uint reserve1,) = pair.getReserves();9 (uint reserveInput, uint reserveOutput) = input == token0 ? (reserve0, reserve1) : (reserve1, reserve0);10 amountInput = IERC20(input).balanceOf(address(pair)).sub(reserveInput);11 amountOutput = UniswapV2Library.getAmountOut(amountInput, reserveInput, reserveOutput);ট্রান্সফার ফির কারণে আমরা প্রতিটি ট্রান্সফার থেকে কতটা পাব তা জানার জন্য getAmountsOut ফাংশনের ওপর নির্ভর করতে পারি না (যেমনটা আমরা আসল _swap কল করার আগে করি)। এর পরিবর্তে আমাদের প্রথমে ট্রান্সফার করতে হবে এবং তারপর দেখতে হবে আমরা কতগুলো টোকেন ফেরত পেয়েছি।
দ্রষ্টব্য: তাত্ত্বিকভাবে আমরা _swap-এর পরিবর্তে শুধু এই ফাংশনটি ব্যবহার করতে পারতাম, কিন্তু কিছু নির্দিষ্ট ক্ষেত্রে (উদাহরণস্বরূপ, যদি ট্রান্সফারটি শেষে রিভার্ট হয়ে যায় কারণ প্রয়োজনীয় ন্যূনতম পরিমাণ পূরণের জন্য শেষে পর্যাপ্ত পরিমাণ নেই) এতে বেশি গ্যাস খরচ হবে। ট্রান্সফার ফি টোকেনগুলো বেশ বিরল, তাই যদিও আমাদের সেগুলোর ব্যবস্থা করতে হবে, তবে সমস্ত সোয়াপের ক্ষেত্রে ধরে নেওয়ার কোনো প্রয়োজন নেই যে সেগুলো অন্তত একটির মধ্য দিয়ে যাবে।
1 }2 (uint amount0Out, uint amount1Out) = input == token0 ? (uint(0), amountOutput) : (amountOutput, uint(0));3 address to = i < path.length - 2 ? UniswapV2Library.pairFor(factory, output, path[i + 2]) : _to;4 pair.swap(amount0Out, amount1Out, to, new bytes(0));5 }6 }7
8
9 function swapExactTokensForTokensSupportingFeeOnTransferTokens(10 uint amountIn,11 uint amountOutMin,12 address[] calldata path,13 address to,14 uint deadline15 ) external virtual override ensure(deadline) {16 TransferHelper.safeTransferFrom(17 path[0], msg.sender, UniswapV2Library.pairFor(factory, path[0], path[1]), amountIn18 );19 uint balanceBefore = IERC20(path[path.length - 1]).balanceOf(to);20 _swapSupportingFeeOnTransferTokens(path, to);21 require(22 IERC20(path[path.length - 1]).balanceOf(to).sub(balanceBefore) >= amountOutMin,23 'UniswapV2Router: INSUFFICIENT_OUTPUT_AMOUNT'24 );25 }26
27
28 function swapExactETHForTokensSupportingFeeOnTransferTokens(29 uint amountOutMin,30 address[] calldata path,31 address to,32 uint deadline33 )34 external35 virtual36 override37 payable38 ensure(deadline)39 {40 require(path[0] == WETH, 'UniswapV2Router: INVALID_PATH');41 uint amountIn = msg.value;42 IWETH(WETH).deposit{value: amountIn}();43 assert(IWETH(WETH).transfer(UniswapV2Library.pairFor(factory, path[0], path[1]), amountIn));44 uint balanceBefore = IERC20(path[path.length - 1]).balanceOf(to);45 _swapSupportingFeeOnTransferTokens(path, to);46 require(47 IERC20(path[path.length - 1]).balanceOf(to).sub(balanceBefore) >= amountOutMin,48 'UniswapV2Router: INSUFFICIENT_OUTPUT_AMOUNT'49 );50 }51
52
53 function swapExactTokensForETHSupportingFeeOnTransferTokens(54 uint amountIn,55 uint amountOutMin,56 address[] calldata path,57 address to,58 uint deadline59 )60 external61 virtual62 override63 ensure(deadline)64 {65 require(path[path.length - 1] == WETH, 'UniswapV2Router: INVALID_PATH');66 TransferHelper.safeTransferFrom(67 path[0], msg.sender, UniswapV2Library.pairFor(factory, path[0], path[1]), amountIn68 );69 _swapSupportingFeeOnTransferTokens(path, address(this));70 uint amountOut = IERC20(WETH).balanceOf(address(this));71 require(amountOut >= amountOutMin, 'UniswapV2Router: INSUFFICIENT_OUTPUT_AMOUNT');72 IWETH(WETH).withdraw(amountOut);73 TransferHelper.safeTransferETH(to, amountOut);74 }এগুলো সাধারণ টোকেনের জন্য ব্যবহৃত একই ভ্যারিয়েন্ট, তবে এগুলো এর পরিবর্তে _swapSupportingFeeOnTransferTokens কল করে।
1 // **** লাইব্রেরি ফাংশন ****2 function quote(uint amountA, uint reserveA, uint reserveB) public pure virtual override returns (uint amountB) {3 return UniswapV2Library.quote(amountA, reserveA, reserveB);4 }5
6 function getAmountOut(uint amountIn, uint reserveIn, uint reserveOut)7 public8 pure9 virtual10 override11 returns (uint amountOut)12 {13 return UniswapV2Library.getAmountOut(amountIn, reserveIn, reserveOut);14 }15
16 function getAmountIn(uint amountOut, uint reserveIn, uint reserveOut)17 public18 pure19 virtual20 override21 returns (uint amountIn)22 {23 return UniswapV2Library.getAmountIn(amountOut, reserveIn, reserveOut);24 }25
26 function getAmountsOut(uint amountIn, address[] memory path)27 public28 view29 virtual30 override31 returns (uint[] memory amounts)32 {33 return UniswapV2Library.getAmountsOut(factory, amountIn, path);34 }35
36 function getAmountsIn(uint amountOut, address[] memory path)37 public38 view39 virtual40 override41 returns (uint[] memory amounts)42 {43 return UniswapV2Library.getAmountsIn(factory, amountOut, path);44 }45}এই ফাংশনগুলো হলো শুধু প্রক্সি যা UniswapV2Library ফাংশনগুলোকে কল করে।
UniswapV2Migrator.sol
এই কন্ট্রাক্টটি পুরনো v1 থেকে v2-তে এক্সচেঞ্জগুলো মাইগ্রেট করতে ব্যবহৃত হয়েছিল। যেহেতু সেগুলো মাইগ্রেট করা হয়ে গেছে, তাই এটি আর প্রাসঙ্গিক নয়।
লাইব্রেরিগুলো
SafeMath লাইব্রেরি (opens in a new tab) খুব ভালোভাবে ডকুমেন্টেড, তাই এখানে এটি ডকুমেন্ট করার কোনো প্রয়োজন নেই।
Math
এই লাইব্রেরিতে কিছু গাণিতিক ফাংশন রয়েছে যা সাধারণত Solidity কোডে প্রয়োজন হয় না, তাই এগুলো এই ভাষার অংশ নয়।
1pragma solidity =0.5.16;2
3// বিভিন্ন গাণিতিক অপারেশন সম্পাদনের জন্য একটি লাইব্রেরি4
5library Math {6 function min(uint x, uint y) internal pure returns (uint z) {7 z = x < y ? x : y;8 }9
10 // ব্যাবিলনীয় পদ্ধতি (https://wikipedia.org/wiki/Methods_of_computing_square_roots#Babylonian_method)11 function sqrt(uint y) internal pure returns (uint z) {12 if (y > 3) {13 z = y;14 uint x = y / 2 + 1;x-কে এমন একটি অনুমান হিসেবে শুরু করুন যা বর্গমূলের চেয়ে বড় (এ কারণেই আমাদের 1-3 কে বিশেষ ক্ষেত্র হিসেবে বিবেচনা করতে হবে)।
1 while (x < z) {2 z = x;3 x = (y / x + x) / 2;আরও কাছাকাছি একটি অনুমান বের করুন, যা হলো পূর্ববর্তী অনুমান এবং যে সংখ্যার বর্গমূল আমরা বের করার চেষ্টা করছি তাকে পূর্ববর্তী অনুমান দিয়ে ভাগ করার গড়। নতুন অনুমানটি বিদ্যমান অনুমানের চেয়ে ছোট না হওয়া পর্যন্ত পুনরাবৃত্তি করুন। আরও বিস্তারিত জানতে, এখানে দেখুন (opens in a new tab)।
1 }2 } else if (y != 0) {3 z = 1;আমাদের কখনোই শূন্যের বর্গমূলের প্রয়োজন হওয়া উচিত নয়। এক, দুই এবং তিনের বর্গমূল মোটামুটি এক (আমরা পূর্ণসংখ্যা ব্যবহার করি, তাই আমরা ভগ্নাংশকে উপেক্ষা করি)।
1 }2 }3}ফিক্সড পয়েন্ট ফ্র্যাকশন (UQ112x112)
এই লাইব্রেরিটি ভগ্নাংশগুলো পরিচালনা করে, যা সাধারণত ইথিরিয়াম পাটিগণিতের অংশ নয়। এটি x সংখ্যাটিকে x*2^112 হিসেবে এনকোড করে এই কাজটি করে। এটি আমাদের কোনো পরিবর্তন ছাড়াই মূল যোগ এবং বিয়োগ অপকোডগুলো ব্যবহার করতে দেয়।
1pragma solidity =0.5.16;2
3// বাইনারি ফিক্সড পয়েন্ট নম্বরগুলো হ্যান্ডেল করার জন্য একটি লাইব্রেরি (https://wikipedia.org/wiki/Q_(number_format))4
5// রেঞ্জ: [0, 2**112 - 1]6// রেজোলিউশন: 1 / 2**1127
8library UQ112x112 {9 uint224 constant Q112 = 2**112;Q112 হলো একের জন্য এনকোডিং।
1 // একটি uint112 কে UQ112x112 হিসেবে এনকোড করে2 function encode(uint112 y) internal pure returns (uint224 z) {3 z = uint224(y) * Q112; // কখনও ওভারফ্লো হয় না4 }যেহেতু y হলো uint112, এটি সর্বোচ্চ 2^112-1 হতে পারে। সেই সংখ্যাটিকে এখনও UQ112x112 হিসেবে এনকোড করা যেতে পারে।
1 // একটি UQ112x112 কে uint112 দিয়ে ভাগ করে, একটি UQ112x112 রিটার্ন করে2 function uqdiv(uint224 x, uint112 y) internal pure returns (uint224 z) {3 z = x / uint224(y);4 }5}যদি আমরা দুটি UQ112x112 মানকে ভাগ করি, তবে ফলাফলটি আর 2^112 দ্বারা গুণিত হয় না। তাই এর পরিবর্তে আমরা হর হিসেবে একটি পূর্ণসংখ্যা নিই। গুণ করার জন্যও আমাদের একই ধরনের কৌশল ব্যবহার করার প্রয়োজন হতো, কিন্তু আমাদের UQ112x112 মানগুলোর গুণ করার প্রয়োজন নেই।
UniswapV2Library
এই লাইব্রেরিটি শুধুমাত্র পেরিফেরি কন্ট্রাক্টগুলো দ্বারা ব্যবহৃত হয়
1pragma solidity >=0.5.0;2
3import '@uniswap/v2-core/contracts/interfaces/IUniswapV2Pair.sol';4
5import "./SafeMath.sol";6
7library UniswapV2Library {8 using SafeMath for uint;9
10 // সর্ট করা টোকেন অ্যাড্রেস রিটার্ন করে, এই অর্ডারে সর্ট করা পেয়ারগুলো থেকে রিটার্ন ভ্যালু হ্যান্ডেল করতে ব্যবহৃত হয়11 function sortTokens(address tokenA, address tokenB) internal pure returns (address token0, address token1) {12 require(tokenA != tokenB, 'UniswapV2Library: IDENTICAL_ADDRESSES');13 (token0, token1) = tokenA < tokenB ? (tokenA, tokenB) : (tokenB, tokenA);14 require(token0 != address(0), 'UniswapV2Library: ZERO_ADDRESS');15 }এডড্রেস অনুযায়ী দুটি টোকেন সাজান, যাতে আমরা তাদের জন্য পেয়ার এক্সচেঞ্জের এডড্রেস পেতে পারি। এটি প্রয়োজনীয় কারণ অন্যথায় আমাদের দুটি সম্ভাবনা থাকত, একটি A,B প্যারামিটারের জন্য এবং অন্যটি B,A প্যারামিটারের জন্য, যার ফলে একটির পরিবর্তে দুটি এক্সচেঞ্জ তৈরি হতো।
1 // কোনো এক্সটার্নাল কল না করেই একটি পেয়ারের জন্য CREATE2 অ্যাড্রেস ক্যালকুলেট করে2 function pairFor(address factory, address tokenA, address tokenB) internal pure returns (address pair) {3 (address token0, address token1) = sortTokens(tokenA, tokenB);4 pair = address(uint(keccak256(abi.encodePacked(5 hex'ff',6 factory,7 keccak256(abi.encodePacked(token0, token1)),8 hex'96e8ac4277198ff8b6f785478aa9a39f403cb768dd02cbee326c3e7da348845f' // ইনিট কোড হ্যাস9 ))));10 }এই ফাংশনটি দুটি টোকেনের জন্য পেয়ার এক্সচেঞ্জের এডড্রেস গণনা করে। এই কন্ট্রাক্টটি CREATE2 অপকোড (opens in a new tab) ব্যবহার করে তৈরি করা হয়েছে, তাই আমরা যদি এর ব্যবহৃত প্যারামিটারগুলো জানি তবে একই এ্যালগরিদম ব্যবহার করে এডড্রেস গণনা করতে পারি। এটি ফ্যাক্টরিকে জিজ্ঞাসা করার চেয়ে অনেক সস্তা, এবং
1 // একটি পেয়ারের জন্য রিজার্ভ ফেচ করে এবং সর্ট করে2 function getReserves(address factory, address tokenA, address tokenB) internal view returns (uint reserveA, uint reserveB) {3 (address token0,) = sortTokens(tokenA, tokenB);4 (uint reserve0, uint reserve1,) = IUniswapV2Pair(pairFor(factory, tokenA, tokenB)).getReserves();5 (reserveA, reserveB) = tokenA == token0 ? (reserve0, reserve1) : (reserve1, reserve0);6 }এই ফাংশনটি পেয়ার এক্সচেঞ্জে থাকা দুটি টোকেনের রিজার্ভ রিটার্ন করে। মনে রাখবেন যে এটি যেকোনো ক্রমে টোকেনগুলো গ্রহণ করতে পারে এবং অভ্যন্তরীণ ব্যবহারের জন্য সেগুলোকে সাজিয়ে নেয়।
1 // একটি অ্যাসেটের কিছু পরিমাণ এবং পেয়ার রিজার্ভ দেওয়া হলে, অন্য অ্যাসেটের সমতুল্য পরিমাণ রিটার্ন করে2 function quote(uint amountA, uint reserveA, uint reserveB) internal pure returns (uint amountB) {3 require(amountA > 0, 'UniswapV2Library: INSUFFICIENT_AMOUNT');4 require(reserveA > 0 && reserveB > 0, 'UniswapV2Library: INSUFFICIENT_LIQUIDITY');5 amountB = amountA.mul(reserveB) / reserveA;6 }এই ফাংশনটি আপনাকে টোকেন A-এর বিনিময়ে প্রাপ্ত টোকেন B-এর পরিমাণ দেয়, যদি কোনো ফি জড়িত না থাকে। এই গণনাটি বিবেচনা করে যে ট্রান্সফারটি এক্সচেঞ্জ রেট পরিবর্তন করে।
1 // একটি অ্যাসেটের ইনপুট পরিমাণ এবং পেয়ার রিজার্ভ দেওয়া হলে, অন্য অ্যাসেটের সর্বোচ্চ আউটপুট পরিমাণ রিটার্ন করে2 function getAmountOut(uint amountIn, uint reserveIn, uint reserveOut) internal pure returns (uint amountOut) {উপরের quote ফাংশনটি খুব ভালোভাবে কাজ করে যদি পেয়ার এক্সচেঞ্জ ব্যবহার করার জন্য কোনো ফি না থাকে। তবে, যদি 0.3% এক্সচেঞ্জ ফি থাকে তবে আপনি আসলে যে পরিমাণ পাবেন তা কম হবে। এই ফাংশনটি এক্সচেঞ্জ ফির পরের পরিমাণ গণনা করে।
1
2 require(amountIn > 0, 'UniswapV2Library: INSUFFICIENT_INPUT_AMOUNT');3 require(reserveIn > 0 && reserveOut > 0, 'UniswapV2Library: INSUFFICIENT_LIQUIDITY');4 uint amountInWithFee = amountIn.mul(997);5 uint numerator = amountInWithFee.mul(reserveOut);6 uint denominator = reserveIn.mul(1000).add(amountInWithFee);7 amountOut = numerator / denominator;8 }Solidity নেটিভভাবে ভগ্নাংশ পরিচালনা করে না, তাই আমরা কেবল পরিমাণটিকে 0.997 দিয়ে গুণ করতে পারি না। এর পরিবর্তে, আমরা লবকে 997 দিয়ে এবং হরকে 1000 দিয়ে গুণ করি, যা একই ফলাফল অর্জন করে।
1 // একটি অ্যাসেটের আউটপুট পরিমাণ এবং পেয়ার রিজার্ভ দেওয়া হলে, অন্য অ্যাসেটের প্রয়োজনীয় ইনপুট পরিমাণ রিটার্ন করে2 function getAmountIn(uint amountOut, uint reserveIn, uint reserveOut) internal pure returns (uint amountIn) {3 require(amountOut > 0, 'UniswapV2Library: INSUFFICIENT_OUTPUT_AMOUNT');4 require(reserveIn > 0 && reserveOut > 0, 'UniswapV2Library: INSUFFICIENT_LIQUIDITY');5 uint numerator = reserveIn.mul(amountOut).mul(1000);6 uint denominator = reserveOut.sub(amountOut).mul(997);7 amountIn = (numerator / denominator).add(1);8 }এই ফাংশনটি মোটামুটি একই কাজ করে, তবে এটি আউটপুটের পরিমাণ পায় এবং ইনপুট প্রদান করে।
1
2 // যেকোনো সংখ্যক পেয়ারের উপর চেইনড getAmountOut ক্যালকুলেশন সম্পাদন করে3 function getAmountsOut(address factory, uint amountIn, address[] memory path) internal view returns (uint[] memory amounts) {4 require(path.length >= 2, 'UniswapV2Library: INVALID_PATH');5 amounts = new uint[](path.length);6 amounts[0] = amountIn;7 for (uint i; i < path.length - 1; i++) {8 (uint reserveIn, uint reserveOut) = getReserves(factory, path[i], path[i + 1]);9 amounts[i + 1] = getAmountOut(amounts[i], reserveIn, reserveOut);10 }11 }12
13 // যেকোনো সংখ্যক পেয়ারের উপর চেইনড getAmountIn ক্যালকুলেশন সম্পাদন করে14 function getAmountsIn(address factory, uint amountOut, address[] memory path) internal view returns (uint[] memory amounts) {15 require(path.length >= 2, 'UniswapV2Library: INVALID_PATH');16 amounts = new uint[](path.length);17 amounts[amounts.length - 1] = amountOut;18 for (uint i = path.length - 1; i > 0; i--) {19 (uint reserveIn, uint reserveOut) = getReserves(factory, path[i - 1], path[i]);20 amounts[i - 1] = getAmountIn(amounts[i], reserveIn, reserveOut);21 }22 }23}এই দুটি ফাংশন মানগুলো শনাক্ত করার কাজ পরিচালনা করে যখন একাধিক পেয়ার এক্সচেঞ্জের মধ্য দিয়ে যাওয়ার প্রয়োজন হয়।
ট্রান্সফার হেল্পার
এই লাইব্রেরিটি (opens in a new tab) ERC-20 এবং ইথিরিয়াম ট্রান্সফারের চারপাশে সফলতার চেক যোগ করে যাতে একটি রিভার্ট এবং একটি false মান রিটার্নকে একইভাবে বিবেচনা করা যায়।
1// SPDX-License-Identifier: GPL-3.0-or-later2
3pragma solidity >=0.6.0;4
5// ERC20 টোকেনের সাথে ইন্টারঅ্যাক্ট করার এবং ETH পাঠানোর জন্য হেল্পার মেথড যা ধারাবাহিকভাবে true/false রিটার্ন করে না6library TransferHelper {7 function safeApprove(8 address token,9 address to,10 uint256 value11 ) internal {12 // bytes4(keccak256(bytes('approve(address,uint256)')));13 (bool success, bytes memory data) = token.call(abi.encodeWithSelector(0x095ea7b3, to, value));14
আমরা দুটি উপায়ের যেকোনো একটিতে ভিন্ন একটি কন্ট্রাক্ট কল করতে পারি:
- একটি ফাংশন কল তৈরি করতে একটি ইন্টারফেস ডেফিনিশন ব্যবহার করে
- কলটি তৈরি করতে "ম্যানুয়ালি" অ্যাপ্লিকেশন বাইনারি ইন্টারফেস (ABI) (opens in a new tab) ব্যবহার করে। কোডের লেখক এটিই করার সিদ্ধান্ত নিয়েছেন।
1 require(2 success && (data.length == 0 || abi.decode(data, (bool))),3 'TransferHelper::safeApprove: approve failed'4 );5 }ERC-20 স্ট্যান্ডার্ডের আগে তৈরি হওয়া টোকেনগুলোর সাথে ব্যাকওয়ার্ড কম্প্যাটিবিলিটির স্বার্থে, একটি ERC-20 কল রিভার্ট হওয়ার মাধ্যমে (যে ক্ষেত্রে success হলো false) অথবা সফল হয়ে একটি false মান রিটার্ন করার মাধ্যমে (যে ক্ষেত্রে আউটপুট ডাটা থাকে, এবং আপনি যদি এটিকে বুলিয়ান হিসেবে ডিকোড করেন তবে আপনি false পাবেন) ব্যর্থ হতে পারে।
1
2
3 function safeTransfer(4 address token,5 address to,6 uint256 value7 ) internal {8 // bytes4(keccak256(bytes('transfer(address,uint256)')));9 (bool success, bytes memory data) = token.call(abi.encodeWithSelector(0xa9059cbb, to, value));10 require(11 success && (data.length == 0 || abi.decode(data, (bool))),12 'TransferHelper::safeTransfer: transfer failed'13 );14 }এই ফাংশনটি ERC-20-এর ট্রান্সফার কার্যকারিতা (opens in a new tab) বাস্তবায়ন করে, যা একটি একাউন্টকে অন্য একটি একাউন্ট দ্বারা প্রদত্ত অ্যালাউন্স খরচ করার অনুমতি দেয়।
1
2 function safeTransferFrom(3 address token,4 address from,5 address to,6 uint256 value7 ) internal {8 // bytes4(keccak256(bytes('transferFrom(address,address,uint256)')));9 (bool success, bytes memory data) = token.call(abi.encodeWithSelector(0x23b872dd, from, to, value));10 require(11 success && (data.length == 0 || abi.decode(data, (bool))),12 'TransferHelper::transferFrom: transferFrom failed'13 );14 }এই ফাংশনটি ERC-20-এর transferFrom কার্যকারিতা (opens in a new tab) বাস্তবায়ন করে, যা একটি একাউন্টকে অন্য একটি একাউন্ট দ্বারা প্রদত্ত অ্যালাউন্স খরচ করার অনুমতি দেয়।
1
2 function safeTransferETH(address to, uint256 value) internal {3 (bool success, ) = to.call{value: value}(new bytes(0));4 require(success, 'TransferHelper::safeTransferETH: ETH transfer failed');5 }6}এই ফাংশনটি একটি একাউন্টে ইথার ট্রান্সফার করে। ভিন্ন কোনো কন্ট্রাক্টে যেকোনো কল ইথার পাঠানোর চেষ্টা করতে পারে। যেহেতু আমাদের আসলে কোনো ফাংশন কল করার প্রয়োজন নেই, তাই আমরা কলের সাথে কোনো ডাটা পাঠাই না।
উপসংহার
এটি প্রায় 50 পৃষ্ঠার একটি দীর্ঘ প্রবন্ধ। আপনি যদি এতদূর এসে থাকেন, তবে আপনাকে অভিনন্দন! আশা করি এতক্ষণে আপনি একটি বাস্তব-জীবনের অ্যাপ্লিকেশন (ছোট নমুনা প্রোগ্রামের বিপরীতে) লেখার বিবেচ্য বিষয়গুলো বুঝতে পেরেছেন এবং আপনার নিজের ব্যবহারের জন্য কন্ট্রাক্ট লিখতে আরও ভালোভাবে সক্ষম হবেন।
এখন যান এবং দরকারী কিছু লিখুন এবং আমাদের চমকে দিন।
আমার আরও কাজের জন্য এখানে দেখুন (opens in a new tab)।
পেজ সর্বশেষ আপডেট: 3 মার্চ, 2026
