Ethereum-இல் உங்கள் சொந்த AI வர்த்தக முகவரை (trading agent) உருவாக்குங்கள்
இந்த வழிகாட்டியில் ஒரு எளிய AI வர்த்தக முகவரை எவ்வாறு உருவாக்குவது என்பதை நீங்கள் கற்றுக் கொள்வீர்கள். இந்த முகவர் பின்வரும் படிகளைப் பயன்படுத்தி செயல்படுகிறது:
- ஒரு டோக்கனின் தற்போதைய மற்றும் கடந்த கால விலைகள், அத்துடன் தொடர்புடைய பிற தகவல்களைப் படித்தல்
- இந்தத் தகவல்களுடன், அது எவ்வாறு தொடர்புடையதாக இருக்கலாம் என்பதை விளக்கும் பின்னணித் தகவல்களையும் சேர்த்து ஒரு வினவலை (query) உருவாக்குதல்
- வினவலைச் சமர்ப்பித்து, கணிக்கப்பட்ட விலையைத் திரும்பப் பெறுதல்
- பரிந்துரையின் அடிப்படையில் வர்த்தகம் செய்தல்
- காத்திருந்து மீண்டும் செய்தல்
இந்த முகவர் தகவல்களை எவ்வாறு படிப்பது, அதைப் பயன்படுத்தக்கூடிய பதிலைத் தரும் வினவலாக எவ்வாறு மாற்றுவது மற்றும் அந்தப் பதிலை எவ்வாறு பயன்படுத்துவது என்பதை விளக்குகிறது. இவை அனைத்தும் ஒரு AI முகவருக்குத் தேவையான படிகளாகும். AI-இல் அதிகம் பயன்படுத்தப்படும் மொழி என்பதால், இந்த முகவர் Python-இல் செயல்படுத்தப்பட்டுள்ளது.
இதை ஏன் செய்ய வேண்டும்?
தானியங்கி வர்த்தக முகவர்கள் டெவலப்பர்கள் ஒரு வர்த்தக உத்தியைத் தேர்ந்தெடுத்து செயல்படுத்த அனுமதிக்கின்றன. AI முகவர்கள் மிகவும் சிக்கலான மற்றும் மாறும் வர்த்தக உத்திகளை அனுமதிக்கின்றன, டெவலப்பர் பயன்படுத்தக் கருதிப் பார்க்காத தகவல்கள் மற்றும் அல்காரிதம்களைக் கூட இவை பயன்படுத்தக்கூடும்.
கருவிகள்
இந்த வழிகாட்டி மேற்கோள்கள் (quotes) மற்றும் வர்த்தகத்திற்கு Python (opens in a new tab), Web3 லைப்ரரி (opens in a new tab) மற்றும் Uniswap v3 (opens in a new tab) ஆகியவற்றைப் பயன்படுத்துகிறது.
ஏன் Python?
AI-க்கு மிகவும் பரவலாகப் பயன்படுத்தப்படும் மொழி Python (opens in a new tab) ஆகும், எனவே அதை நாம் இங்கு பயன்படுத்துகிறோம். உங்களுக்கு Python தெரியாவிட்டாலும் கவலைப்பட வேண்டாம். இந்த மொழி மிகவும் தெளிவானது, மேலும் அது என்ன செய்கிறது என்பதை நான் சரியாக விளக்குவேன்.
Web3 லைப்ரரி (opens in a new tab) என்பது மிகவும் பொதுவான Python Ethereum API ஆகும். இதைப் பயன்படுத்துவது மிகவும் எளிதானது.
பிளாக்செயினில் வர்த்தகம் செய்தல்
Ethereum-இல் டோக்கன்களை வர்த்தகம் செய்ய உங்களை அனுமதிக்கும் பல பரவலாக்கப்பட்ட பரிமாற்றங்கள் (DEX) உள்ளன. இருப்பினும், ஆர்பிட்ரேஜ் (arbitrage) காரணமாக அவை ஒரே மாதிரியான பரிமாற்ற விகிதங்களைக் கொண்டிருக்கின்றன.
Uniswap (opens in a new tab) என்பது பரவலாகப் பயன்படுத்தப்படும் ஒரு DEX ஆகும், இதை நாம் மேற்கோள்களுக்கும் (டோக்கனின் ஒப்பீட்டு மதிப்புகளைப் பார்க்க) மற்றும் வர்த்தகத்திற்கும் பயன்படுத்தலாம்.
OpenAI
ஒரு பெரிய மொழி மாதிரிக்கு (LLM), நான் OpenAI (opens in a new tab)-உடன் தொடங்க முடிவு செய்தேன். இந்த வழிகாட்டியில் உள்ள பயன்பாட்டை இயக்க, நீங்கள் API அணுகலுக்குப் பணம் செலுத்த வேண்டும். குறைந்தபட்ச கட்டணமான $5 போதுமானதை விட அதிகமாகும்.
மேம்பாடு, படிப்படியாக
மேம்பாட்டை எளிதாக்க, நாம் கட்டம் கட்டமாகச் செல்கிறோம். ஒவ்வொரு படியும் GitHub-இல் ஒரு கிளையாக (branch) உள்ளது.
தொடங்குதல்
UNIX அல்லது Linux-இல் (உள்ளடங்கிய WSL (opens in a new tab)) தொடங்குவதற்கான படிகள் உள்ளன
-
உங்களிடம் ஏற்கனவே இல்லையென்றால், Python (opens in a new tab)-ஐப் பதிவிறக்கி நிறுவவும்.
-
GitHub ரெபோசிட்டரியை குளோன் (clone) செய்யவும்.
1git clone https://github.com/qbzzt/260215-ai-agent.git -b 01-getting-started2cd 260215-ai-agent -
uv(opens in a new tab)-ஐ நிறுவவும். உங்கள் கணினியில் உள்ள கட்டளை வேறுபட்டிருக்கலாம்.1pipx install uv -
லைப்ரரிகளைப் பதிவிறக்கவும்.
1uv sync -
விர்ச்சுவல் என்விரான்மென்ட்டை (virtual environment) செயல்படுத்தவும்.
1source .venv/bin/activate -
Python மற்றும் Web3 சரியாக வேலை செய்கிறதா என்பதைச் சரிபார்க்க,
python3-ஐ இயக்கி, இந்த நிரலை அதற்கு வழங்கவும். நீங்கள் அதை>>>ப்ராம்ட்டில் (prompt) உள்ளிடலாம்; ஒரு கோப்பை உருவாக்க வேண்டிய அவசியமில்லை.1from web3 import Web32MAINNET_URL = "https://eth.drpc.org"3w3 = Web3(Web3.HTTPProvider(MAINNET_URL))4w3.eth.block_number5quit()
பிளாக்செயினிலிருந்து படித்தல்
அடுத்த படி பிளாக்செயினிலிருந்து படிப்பதாகும். அதைச் செய்ய, நீங்கள் 02-read-quote கிளைக்கு மாற வேண்டும், பின்னர் நிரலை இயக்க uv-ஐப் பயன்படுத்த வேண்டும்.
1git checkout 02-read-quote2uv run agent.pyநீங்கள் Quote ஆப்ஜெக்ட்களின் பட்டியலைப் பெற வேண்டும், ஒவ்வொன்றும் ஒரு நேரமுத்திரை (timestamp), ஒரு விலை மற்றும் சொத்து (தற்போது எப்போதும் WETH/USDC) ஆகியவற்றைக் கொண்டிருக்கும்.
இதோ வரிக்கு வரி விளக்கம்.
1from web3 import Web32from web3.contract import Contract3from decimal import Decimal, ROUND_HALF_UP4from dataclasses import dataclass5from datetime import datetime, timezone6from pprint import pprint7import time8import functools9import sysநமக்குத் தேவையான லைப்ரரிகளை இறக்குமதி (import) செய்யவும். அவை பயன்படுத்தப்படும்போது கீழே விளக்கப்பட்டுள்ளன.
1print = functools.partial(print, flush=True)Python-இன் print-ஐ எப்போதும் வெளியீட்டை உடனடியாக ஃப்ளஷ் (flush) செய்யும் பதிப்பைக் கொண்டு மாற்றுகிறது. நீண்ட நேரம் இயங்கும் ஸ்கிரிப்ட்டில் இது பயனுள்ளதாக இருக்கும், ஏனெனில் நிலை புதுப்பிப்புகள் அல்லது பிழைத்திருத்த (debugging) வெளியீட்டிற்காக நாம் காத்திருக்க விரும்பவில்லை.
1MAINNET_URL = "https://eth.drpc.org"மெயின்நெட்டிற்குச் (mainnet) செல்வதற்கான URL. நீங்கள் சேவையாக நோடு (Node as a service)-இலிருந்து ஒன்றைப் பெறலாம் அல்லது Chainlist (opens in a new tab)-இல் விளம்பரப்படுத்தப்பட்டவற்றில் ஒன்றைப் பயன்படுத்தலாம்.
1BLOCK_TIME_SECONDS = 122MINUTE_BLOCKS = int(60 / BLOCK_TIME_SECONDS)3HOUR_BLOCKS = MINUTE_BLOCKS * 604DAY_BLOCKS = HOUR_BLOCKS * 24ஒரு Ethereum மெயின்நெட் பிளாக் பொதுவாக ஒவ்வொரு பன்னிரண்டு வினாடிகளுக்கும் நிகழ்கிறது, எனவே ஒரு குறிப்பிட்ட காலப்பகுதியில் நாம் எதிர்பார்க்கும் பிளாக்குகளின் எண்ணிக்கை இதுவாகும். இது ஒரு சரியான எண்ணிக்கை அல்ல என்பதை நினைவில் கொள்க. பிளாக் முன்மொழிபவர் (block proposer) செயலிழக்கும்போது, அந்த பிளாக் தவிர்க்கப்படும், மேலும் அடுத்த பிளாக்கிற்கான நேரம் 24 வினாடிகள் ஆகும். ஒரு நேரமுத்திரைக்குச் சரியான பிளாக்கைப் பெற விரும்பினால், நாம் பைனரி தேடலைப் (binary search) (opens in a new tab) பயன்படுத்துவோம். இருப்பினும், இது நமது நோக்கங்களுக்குப் போதுமான அளவு நெருக்கமாக உள்ளது. எதிர்காலத்தைக் கணிப்பது ஒரு துல்லியமான அறிவியல் அல்ல.
1CYCLE_BLOCKS = DAY_BLOCKSசுழற்சியின் (cycle) அளவு. ஒரு சுழற்சிக்கு ஒருமுறை மேற்கோள்களை மதிப்பாய்வு செய்து, அடுத்த சுழற்சியின் முடிவில் மதிப்பை மதிப்பிட முயற்சிக்கிறோம்.
1# நாம் படிக்கும் தொகுப்பின் முகவரி2WETHUSDC_ADDRESS = Web3.to_checksum_address("0x88e6A0c2dDD26FEEb64F039a2c41296FcB3f5640")மேற்கோள் மதிப்புகள் 0x88e6A0c2dDD26FEEb64F039a2c41296FcB3f5640 (opens in a new tab) என்ற முகவரியில் உள்ள Uniswap 3 USDC/WETH பூலில் (pool) இருந்து எடுக்கப்படுகின்றன. இந்த முகவரி ஏற்கனவே செக்சம் (checksum) வடிவத்தில் உள்ளது, ஆனால் குறியீட்டை மீண்டும் பயன்படுத்தக்கூடியதாக மாற்ற Web3.to_checksum_address (opens in a new tab)-ஐப் பயன்படுத்துவது நல்லது.
1POOL_ABI = [2 { "name": "slot0", ... },3 { "name": "token0", ... },4 { "name": "token1", ... },5]6
7ERC20_ABI = [8 { "name": "symbol", ... },9 { "name": "decimals", ... }10]நாம் தொடர்பு கொள்ள வேண்டிய இரண்டு ஒப்பந்தங்களுக்கான ABI-கள் (opens in a new tab) இவை. குறியீட்டைச் சுருக்கமாக வைத்திருக்க, நாம் அழைக்க வேண்டிய செயல்பாடுகளை (functions) மட்டுமே சேர்த்துள்ளோம்.
1w3 = Web3(Web3.HTTPProvider(MAINNET_URL))Web3 (opens in a new tab) லைப்ரரியைத் துவக்கி, ஒரு Ethereum நோடுடன் இணைக்கவும்.
1@dataclass(frozen=True)2class ERC20Token:3 address: str4 symbol: str5 decimals: int6 contract: ContractPython-இல் டேட்டா கிளாஸை (data class) உருவாக்குவதற்கான ஒரு வழி இது. ஒப்பந்தத்துடன் இணைக்க Contract (opens in a new tab) தரவு வகை பயன்படுத்தப்படுகிறது. (frozen=True) என்பதைக் கவனிக்கவும். Python-இல் பூலியன்கள் (booleans) (opens in a new tab) True அல்லது False எனப் பெரிய எழுத்துக்களில் வரையறுக்கப்படுகின்றன. இந்த டேட்டா கிளாஸ் frozen ஆக உள்ளது, அதாவது புலங்களை (fields) மாற்ற முடியாது.
உள்தள்ளலைக் (indentation) கவனிக்கவும். C-யிலிருந்து பெறப்பட்ட மொழிகளுக்கு (C-derived languages) (opens in a new tab) மாறாக, Python பிளாக்குகளைக் குறிக்க உள்தள்ளலைப் பயன்படுத்துகிறது. பின்வரும் வரையறை இந்த டேட்டா கிளாஸின் ஒரு பகுதியல்ல என்பதை Python இன்டர்ப்ரெட்டர் (interpreter) அறியும், ஏனெனில் இது டேட்டா கிளாஸ் புலங்களின் அதே உள்தள்ளலில் தொடங்கவில்லை.
1@dataclass(frozen=True)2class PoolInfo:3 address: str4 token0: ERC20Token5 token1: ERC20Token6 contract: Contract7 asset: str8 decimal_factor: Decimal = 1தசம பின்னங்களைத் துல்லியமாகக் கையாள Decimal (opens in a new tab) வகை பயன்படுத்தப்படுகிறது.
1 def get_price(self, block: int) -> Decimal:Python-இல் ஒரு செயல்பாட்டை (function) வரையறுக்கும் முறை இதுதான். இது இன்னும் PoolInfo-இன் ஒரு பகுதி என்பதைக் காட்ட வரையறை உள்தள்ளப்பட்டுள்ளது.
ஒரு டேட்டா கிளாஸின் பகுதியாக இருக்கும் ஒரு செயல்பாட்டில், முதல் அளவுரு (parameter) எப்போதும் self ஆக இருக்கும், இது இங்கே அழைக்கப்பட்ட டேட்டா கிளாஸ் நிகழ்வாகும் (instance). இங்கே மற்றொரு அளவுரு உள்ளது, அது பிளாக் எண்.
1 assert block <= w3.eth.block_number, "Block is in the future"நம்மால் எதிர்காலத்தைப் படிக்க முடிந்தால், வர்த்தகத்திற்கு நமக்கு AI தேவைப்படாது.
1 sqrt_price_x96 = Decimal(self.contract.functions.slot0().call(block_identifier=block)[0])Web3-இலிருந்து EVM-இல் ஒரு செயல்பாட்டை அழைப்பதற்கான தொடரியல் (syntax) இதுதான்: <contract object>.functions.<function name>().call(<parameters>). அளவுருக்கள் EVM செயல்பாட்டின் அளவுருக்களாக இருக்கலாம் (ஏதேனும் இருந்தால்; இங்கே எதுவும் இல்லை) அல்லது பிளாக்செயின் நடத்தையை மாற்றுவதற்கான பெயரிடப்பட்ட அளவுருக்களாக (named parameters) (opens in a new tab) இருக்கலாம். இங்கே நாம் இயக்க விரும்பும் பிளாக் எண்ணைக் குறிப்பிட block_identifier என்ற ஒன்றைப் பயன்படுத்துகிறோம்.
இதன் முடிவு இந்த ஸ்ட்ரக்ட் (struct), வரிசை (array) வடிவத்தில் (opens in a new tab) இருக்கும். முதல் மதிப்பு இரண்டு டோக்கன்களுக்கு இடையிலான பரிமாற்ற விகிதத்தின் செயல்பாடாகும்.
1 raw_price = (sqrt_price_x96 / Decimal(2**96)) ** 2ஆன்செயின் (onchain) கணக்கீடுகளைக் குறைக்க, Uniswap v3 உண்மையான பரிமாற்ற காரணியைச் சேமிக்காமல் அதன் வர்க்க மூலத்தை (square root) சேமிக்கிறது. EVM மிதக்கும் புள்ளி (floating point) கணிதம் அல்லது பின்னங்களை ஆதரிக்காததால், உண்மையான மதிப்பிற்குப் பதிலாக, பதில் ஆக இருக்கும்.
1 # (ஒரு token0-க்கு token1)2 return 1/(raw_price * self.decimal_factor)நாம் பெறும் மூல விலை (raw price) என்பது ஒவ்வொரு token1-க்கும் நாம் பெறும் token0-இன் எண்ணிக்கையாகும். நமது பூலில் token0 என்பது USDC (அமெரிக்க டாலரின் அதே மதிப்பைக் கொண்ட ஸ்டேபிள்காயின்) மற்றும் token1 என்பது WETH (opens in a new tab) ஆகும். நாம் உண்மையில் விரும்பும் மதிப்பு ஒரு WETH-க்கு எத்தனை டாலர்கள் என்பதுதான், அதன் தலைகீழ் அல்ல.
தசம காரணி (decimal factor) என்பது இரண்டு டோக்கன்களுக்கான தசம காரணிகளுக்கு (opens in a new tab) இடையிலான விகிதமாகும்.
1@dataclass(frozen=True)2class Quote:3 timestamp: str4 price: Decimal5 asset: strஇந்த டேட்டா கிளாஸ் ஒரு மேற்கோளைக் குறிக்கிறது: ஒரு குறிப்பிட்ட நேரத்தில் ஒரு குறிப்பிட்ட சொத்தின் விலை. இந்த கட்டத்தில், asset புலம் பொருத்தமற்றது, ஏனெனில் நாம் ஒரு பூலை மட்டுமே பயன்படுத்துகிறோம், எனவே ஒரே ஒரு சொத்து மட்டுமே உள்ளது. இருப்பினும், பின்னர் நாம் மேலும் சொத்துகளைச் சேர்ப்போம்.
1def read_token(address: str) -> ERC20Token:2 token = w3.eth.contract(address=address, abi=ERC20_ABI)3 symbol = token.functions.symbol().call()4 decimals = token.functions.decimals().call()5
6 return ERC20Token(7 address=address,8 symbol=symbol,9 decimals=decimals,10 contract=token11 )இந்தச் செயல்பாடு ஒரு முகவரியை எடுத்து, அந்த முகவரியில் உள்ள டோக்கன் ஒப்பந்தம் பற்றிய தகவலை வழங்குகிறது. புதிய Web3 Contract (opens in a new tab)-ஐ உருவாக்க, நாம் முகவரி மற்றும் ABI-ஐ w3.eth.contract-க்கு வழங்குகிறோம்.
1def read_pool(address: str) -> PoolInfo:2 pool_contract = w3.eth.contract(address=address, abi=POOL_ABI)3 token0Address = pool_contract.functions.token0().call()4 token1Address = pool_contract.functions.token1().call()5 token0 = read_token(token0Address)6 token1 = read_token(token1Address)7
8 return PoolInfo(9 address=address,10 asset=f"{token1.symbol}/{token0.symbol}",11 token0=token0,12 token1=token1,13 contract=pool_contract,14 decimal_factor=Decimal(10) ** Decimal(token0.decimals - token1.decimals)15 )இந்தச் செயல்பாடு ஒரு குறிப்பிட்ட பூல் (opens in a new tab) பற்றி நமக்குத் தேவையான அனைத்தையும் வழங்குகிறது. f"<string>" என்ற தொடரியல் ஒரு வடிவமைக்கப்பட்ட சரம் (formatted string) (opens in a new tab) ஆகும்.
1def get_quote(pool: PoolInfo, block_number: int = None) -> Quote:ஒரு Quote ஆப்ஜெக்ட்டைப் பெறவும். block_number-க்கான இயல்புநிலை மதிப்பு None (மதிப்பு இல்லை) ஆகும்.
1 if block_number is None:2 block_number = w3.eth.block_numberபிளாக் எண் குறிப்பிடப்படவில்லை என்றால், w3.eth.block_number-ஐப் பயன்படுத்தவும், இது சமீபத்திய பிளாக் எண்ணாகும். இது ஒரு if அறிக்கைக்கான (statement) (opens in a new tab) தொடரியல் ஆகும்.
இயல்புநிலையை w3.eth.block_number என அமைப்பது சிறப்பாக இருந்திருக்கும் என்று தோன்றலாம், ஆனால் அது சரியாக வேலை செய்யாது, ஏனெனில் அது செயல்பாடு வரையறுக்கப்படும் நேரத்தில் உள்ள பிளாக் எண்ணாக இருக்கும். நீண்ட நேரம் இயங்கும் முகவரில், இது ஒரு சிக்கலாக இருக்கும்.
1 block = w3.eth.get_block(block_number)2 price = pool.get_price(block_number)3 return Quote(4 timestamp=datetime.fromtimestamp(block.timestamp, timezone.utc).isoformat(),5 price=price.quantize(Decimal("0.01")),6 asset=pool.asset7 )மனிதர்கள் மற்றும் பெரிய மொழி மாதிரிகள் (LLM-கள்) படிக்கக்கூடிய வடிவத்திற்கு அதை வடிவமைக்க datetime லைப்ரரியைப் (opens in a new tab) பயன்படுத்தவும். மதிப்பை இரண்டு தசம இடங்களுக்கு முழுமையாக்க Decimal.quantize (opens in a new tab)-ஐப் பயன்படுத்தவும்.
1def get_quotes(pool: PoolInfo, start_block: int, end_block: int, step: int) -> list[Quote]:Python-இல் ஒரு குறிப்பிட்ட வகையை மட்டுமே கொண்டிருக்கக்கூடிய ஒரு பட்டியலை (list) (opens in a new tab) list[<type>] என்பதைப் பயன்படுத்தி நீங்கள் வரையறுக்கலாம்.
1 quotes = []2 for block in range(start_block, end_block + 1, step):Python-இல் ஒரு for லூப் (loop) (opens in a new tab) பொதுவாக ஒரு பட்டியலின் மீது மீண்டும் மீண்டும் செயல்படும் (iterates). மேற்கோள்களைக் கண்டறிய வேண்டிய பிளாக் எண்களின் பட்டியல் range (opens in a new tab)-இலிருந்து வருகிறது.
1 quote = get_quote(pool, block)2 quotes.append(quote)3 return quotesஒவ்வொரு பிளாக் எண்ணுக்கும், ஒரு Quote ஆப்ஜெக்ட்டைப் பெற்று அதை quotes பட்டியலில் சேர்க்கவும் (append). பின்னர் அந்தப் பட்டியலைத் திருப்பி அனுப்பவும்.
1pool = read_pool(WETHUSDC_ADDRESS)2quotes = get_quotes(3 pool,4 w3.eth.block_number - 12*CYCLE_BLOCKS,5 w3.eth.block_number,6 CYCLE_BLOCKS7)8
9pprint(quotes)இது ஸ்கிரிப்ட்டின் முக்கிய குறியீடாகும். பூல் தகவலைப் படித்து, பன்னிரண்டு மேற்கோள்களைப் பெற்று, அவற்றை pprint (opens in a new tab) செய்யவும்.
ஒரு ப்ராம்ட்டை (prompt) உருவாக்குதல்
அடுத்து, இந்த மேற்கோள்களின் பட்டியலை ஒரு LLM-க்கான ப்ராம்ட்டாக மாற்றி, எதிர்பார்க்கப்படும் எதிர்கால மதிப்பைப் பெற வேண்டும்.
1git checkout 03-create-prompt2uv run agent.pyவெளியீடு இப்போது ஒரு LLM-க்கான ப்ராம்ட்டாக இருக்கப் போகிறது, இது போன்றது:
1Given these quotes:2Asset: WETH/USDC3 2026-01-20T16:34 3016.214 .5 .6 .7 2026-02-01T17:49 2299.108
9Asset: WBTC/WETH10 2026-01-20T16:34 29.8411 .12 .13 .14 2026-02-01T17:50 33.4615
16
17What would you expect the value for WETH/USDC to be at time 2026-02-02T17:56?18
19Provide your answer as a single number rounded to two decimal places,20without any other text.இங்கே WETH/USDC மற்றும் WBTC/WETH ஆகிய இரண்டு சொத்துகளுக்கான மேற்கோள்கள் இருப்பதைக் கவனிக்கவும். மற்றொரு சொத்திலிருந்து மேற்கோள்களைச் சேர்ப்பது கணிப்புத் துல்லியத்தை மேம்படுத்தக்கூடும்.
ஒரு ப்ராம்ட் எப்படி இருக்கும்
இந்த ப்ராம்ட் மூன்று பிரிவுகளைக் கொண்டுள்ளது, இவை LLM ப்ராம்ட்களில் மிகவும் பொதுவானவை.
-
தகவல். LLM-கள் அவற்றின் பயிற்சியிலிருந்து நிறைய தகவல்களைக் கொண்டுள்ளன, ஆனால் அவை பொதுவாகச் சமீபத்திய தகவல்களைக் கொண்டிருக்காது. இதனால்தான் நாம் சமீபத்திய மேற்கோள்களை இங்கே மீட்டெடுக்க வேண்டும். ஒரு ப்ராம்ட்டில் தகவலைச் சேர்ப்பது மீட்டெடுப்பு மேம்படுத்தப்பட்ட உருவாக்கம் (retrieval augmented generation - RAG) (opens in a new tab) என்று அழைக்கப்படுகிறது.
-
உண்மையான கேள்வி. இதைத்தான் நாம் தெரிந்து கொள்ள விரும்புகிறோம்.
-
வெளியீட்டு வடிவமைப்பு வழிமுறைகள். பொதுவாக, ஒரு LLM ஒரு மதிப்பீட்டையும் அது எவ்வாறு அந்த முடிவுக்கு வந்தது என்பதற்கான விளக்கத்தையும் நமக்கு வழங்கும். இது மனிதர்களுக்குச் சிறந்தது, ஆனால் ஒரு கணினி நிரலுக்கு இறுதி முடிவு மட்டுமே தேவை.
குறியீடு விளக்கம்
இதோ புதிய குறியீடு.
1from datetime import datetime, timezone, timedeltaநாம் எந்த நேரத்திற்கான மதிப்பீட்டை விரும்புகிறோம் என்பதை LLM-க்கு வழங்க வேண்டும். எதிர்காலத்தில் "n நிமிடங்கள்/மணிநேரங்கள்/நாட்கள்" என்ற நேரத்தைப் பெற, நாம் timedelta கிளாஸைப் (opens in a new tab) பயன்படுத்துகிறோம்.
1# நாம் படிக்கும் தொகுப்புகளின் முகவரிகள்2WETHUSDC_ADDRESS = Web3.to_checksum_address("0x88e6A0c2dDD26FEEb64F039a2c41296FcB3f5640")3WETHWBTC_ADDRESS = Web3.to_checksum_address("0xCBCdF9626bC03E24f779434178A73a0B4bad62eD")நாம் படிக்க வேண்டிய இரண்டு பூல்கள் உள்ளன.
1@dataclass(frozen=True)2class PoolInfo:3 .4 .5 .6 reverse: bool = False7
8 def get_price(self, block: int) -> Decimal:9 assert block <= w3.eth.block_number, "Block is in the future"10 sqrt_price_x96 = Decimal(self.contract.functions.slot0().call(block_identifier=block)[0])11 raw_price = (sqrt_price_x96 / Decimal(2**96)) ** 2 # (ஒரு token0-க்கு token1)12 if self.reverse:13 return 1/(raw_price * self.decimal_factor)14 else:15 return raw_price * self.decimal_factorWETH/USDC பூலில், ஒரு token1 (WETH)-ஐ வாங்க எத்தனை token0 (USDC) தேவை என்பதை நாம் அறிய விரும்புகிறோம். WETH/WBTC பூலில், ஒரு token0 (WBTC, இது ரேப்டு பிட்காயின்) வாங்க எத்தனை token1 (WETH) தேவை என்பதை நாம் அறிய விரும்புகிறோம். பூலின் விகிதத்தைத் தலைகீழாக மாற்ற வேண்டுமா என்பதை நாம் கண்காணிக்க வேண்டும்.
1def read_pool(address: str, reverse: bool = False) -> PoolInfo:2 .3 .4 .5
6 return PoolInfo(7 .8 .9 .10
11 asset= f"{token1.symbol}/{token0.symbol}" if reverse else f"{token0.symbol}/{token1.symbol}",12 reverse=reverse13 )ஒரு பூலைத் தலைகீழாக மாற்ற வேண்டுமா என்பதை அறிய, அதை read_pool-க்கு உள்ளீடாகப் பெற வேண்டும். மேலும், சொத்து குறியீடு (asset symbol) சரியாக அமைக்கப்பட வேண்டும்.
<a> if <b> else <c> என்ற தொடரியல் என்பது மும்மை நிபந்தனை ஆபரேட்டரின் (ternary conditional operator) (opens in a new tab) Python சமமானதாகும், இது C-யிலிருந்து பெறப்பட்ட மொழியில் <b> ? <a> : <c> ஆக இருக்கும்.
1def format_quotes(quotes: list[Quote]) -> str:2 result = f"Asset: {quotes[0].asset}\n"3 for quote in quotes:4 result += f"\t{quote.timestamp[0:16]} {quote.price.quantize(Decimal('0.01'), rounding=ROUND_HALF_UP)}\n"5 return resultஇந்தச் செயல்பாடு Quote ஆப்ஜெக்ட்களின் பட்டியலை வடிவமைக்கும் ஒரு சரத்தை (string) உருவாக்குகிறது, அவை அனைத்தும் ஒரே சொத்துக்குப் பொருந்தும் என்று கருதுகிறது.
1def make_prompt(quotes: list[list[Quote]], expected_time: str, asset: str) -> str:2 return f"""Python-இல் பல வரி சரம் லிட்டரல்கள் (multi-line string literals) (opens in a new tab) """ .... """ என எழுதப்படுகின்றன.
1Given these quotes:2{3 functools.reduce(lambda acc, q: acc + '\n' + q,4 map(lambda q: format_quotes(q), quotes))5}இங்கே, format_quotes-உடன் ஒவ்வொரு மேற்கோள் பட்டியலுக்கும் ஒரு சரத்தை உருவாக்க MapReduce (opens in a new tab) முறையைப் பயன்படுத்துகிறோம், பின்னர் ப்ராம்ட்டில் பயன்படுத்துவதற்காக அவற்றை ஒற்றைச் சரமாகக் குறைக்கிறோம்.
1What would you expect the value for {asset} to be at time {expected_time}?2
3Provide your answer as a single number rounded to two decimal places,4without any other text.5 """ப்ராம்ட்டின் மீதமுள்ள பகுதி எதிர்பார்த்தபடியே உள்ளது.
1wethusdc_pool = read_pool(WETHUSDC_ADDRESS, True)2wethusdc_quotes = get_quotes(3 wethusdc_pool,4 w3.eth.block_number - 12*CYCLE_BLOCKS,5 w3.eth.block_number,6 CYCLE_BLOCKS,7)8
9wethwbtc_pool = read_pool(WETHWBTC_ADDRESS)10wethwbtc_quotes = get_quotes(11 wethwbtc_pool,12 w3.eth.block_number - 12*CYCLE_BLOCKS,13 w3.eth.block_number,14 CYCLE_BLOCKS15)இரண்டு பூல்களையும் மதிப்பாய்வு செய்து இரண்டிலிருந்தும் மேற்கோள்களைப் பெறவும்.
1future_time = (datetime.now(timezone.utc) + timedelta(days=1)).isoformat()[0:16]2
3print(make_prompt(wethusdc_quotes + wethwbtc_quotes, future_time, wethusdc_pool.asset))நாம் மதிப்பீட்டை விரும்பும் எதிர்கால நேரப் புள்ளியைத் தீர்மானித்து, ப்ராம்ட்டை உருவாக்கவும்.
ஒரு LLM-உடன் இடைமுகப்படுத்துதல்
அடுத்து, நாம் ஒரு உண்மையான LLM-ஐ ப்ராம்ட் செய்து, எதிர்பார்க்கப்படும் எதிர்கால மதிப்பைப் பெறுகிறோம். நான் இந்த நிரலை OpenAI-ஐப் பயன்படுத்தி எழுதினேன், எனவே நீங்கள் வேறு வழங்குநரைப் பயன்படுத்த விரும்பினால், நீங்கள் அதைச் சரிசெய்ய வேண்டும்.
-
ஒரு OpenAI கணக்கைப் (opens in a new tab) பெறவும்
-
கணக்கிற்கு நிதியளிக்கவும் (opens in a new tab)—இதை எழுதும் நேரத்தில் குறைந்தபட்ச தொகை $5 ஆகும்
-
கட்டளை வரியில் (command line), API திறவுகோலை ஏற்றுமதி (export) செய்யவும், இதனால் உங்கள் நிரல் அதைப் பயன்படுத்த முடியும்
1export OPENAI_API_KEY=sk-<the rest of the key goes here> -
முகவரை செக்அவுட் (checkout) செய்து இயக்கவும்
1git checkout 04-interface-llm2uv run agent.py
இதோ புதிய குறியீடு.
1from openai import OpenAI2
3open_ai = OpenAI() # கிளையண்ட் OPENAI_API_KEY சூழல் மாறியைப் படிக்கிறதுOpenAI API-ஐ இறக்குமதி செய்து இன்ஸ்டான்ஷியேட் (instantiate) செய்யவும்.
1response = open_ai.chat.completions.create(2 model="gpt-4-turbo",3 messages=[4 {"role": "user", "content": prompt}5 ],6 temperature=0.0,7 max_tokens=16,8)பதிலை உருவாக்க OpenAI API-ஐ (open_ai.chat.completions.create) அழைக்கவும்.
1expected_price = Decimal(response.choices[0].message.content.strip())2current_price = wethusdc_quotes[-1].price3
4print ("Current price:", wethusdc_quotes[-1].price)5print(f"In {future_time}, expected price: {expected_price} USD")6
7if (expected_price > current_price):8 print(f"Buy, I expect the price to go up by {expected_price - current_price} USD")9else:10 print(f"Sell, I expect the price to go down by {current_price - expected_price} USD")விலையை வெளியிட்டு, வாங்குதல் அல்லது விற்றல் பரிந்துரையை வழங்கவும்.
கணிப்புகளைச் சோதித்தல்
இப்போது நாம் கணிப்புகளை உருவாக்க முடியும் என்பதால், நாம் பயனுள்ள கணிப்புகளை உருவாக்குகிறோமா என்பதை மதிப்பிட வரலாற்றுத் தரவையும் பயன்படுத்தலாம்.
1uv run test-predictor.pyஎதிர்பார்க்கப்படும் முடிவு இது போன்றது:
1Prediction for 2026-01-05T19:50: predicted 3138.93 USD, real 3218.92 USD, error 79.99 USD2Prediction for 2026-01-06T19:56: predicted 3243.39 USD, real 3221.08 USD, error 22.31 USD3Prediction for 2026-01-07T20:02: predicted 3223.24 USD, real 3146.89 USD, error 76.35 USD4Prediction for 2026-01-08T20:11: predicted 3150.47 USD, real 3092.04 USD, error 58.43 USD5.6.7.8Prediction for 2026-01-31T22:33: predicted 2637.73 USD, real 2417.77 USD, error 219.96 USD9Prediction for 2026-02-01T22:41: predicted 2381.70 USD, real 2318.84 USD, error 62.86 USD10Prediction for 2026-02-02T22:49: predicted 2234.91 USD, real 2349.28 USD, error 114.37 USD11Mean prediction error over 29 predictions: 83.87103448275862068965517241 USD12Mean change per recommendation: 4.787931034482758620689655172 USD13Standard variance of changes: 104.42 USD14Profitable days: 51.72%15Losing days: 48.28%சோதனையாளரின் (tester) பெரும்பகுதி முகவரைப் போலவே உள்ளது, ஆனால் புதிய அல்லது மாற்றியமைக்கப்பட்ட பகுதிகள் இங்கே உள்ளன.
1CYCLES_FOR_TEST = 40 # பேக்டெஸ்ட்டிற்கு, நாம் எத்தனை சுழற்சிகளைச் சோதிக்கிறோம்2
3# நிறைய விலைப்புள்ளிகளைப் பெறவும்4wethusdc_pool = read_pool(WETHUSDC_ADDRESS, True)5wethusdc_quotes = get_quotes(6 wethusdc_pool,7 w3.eth.block_number - CYCLE_BLOCKS*CYCLES_FOR_TEST,8 w3.eth.block_number,9 CYCLE_BLOCKS,10)11
12wethwbtc_pool = read_pool(WETHWBTC_ADDRESS)13wethwbtc_quotes = get_quotes(14 wethwbtc_pool,15 w3.eth.block_number - CYCLE_BLOCKS*CYCLES_FOR_TEST,16 w3.eth.block_number,17 CYCLE_BLOCKS18)நாம் CYCLES_FOR_TEST (இங்கே 40 எனக் குறிப்பிடப்பட்டுள்ளது) நாட்களுக்குப் பின்னால் பார்க்கிறோம்.
1# கணிப்புகளை உருவாக்கி, உண்மையான வரலாற்றுடன் அவற்றைச் சரிபார்க்கவும்2
3total_error = Decimal(0)4changes = []நாம் ஆர்வமாக உள்ள இரண்டு வகையான பிழைகள் உள்ளன. முதலாவது, total_error, இது முன்னறிவிப்பாளர் (predictor) செய்த பிழைகளின் கூட்டுத்தொகையாகும்.
இரண்டாவதான changes-ஐப் புரிந்து கொள்ள, முகவரின் நோக்கத்தை நாம் நினைவில் கொள்ள வேண்டும். இது WETH/USDC விகிதத்தை (ETH விலை) கணிப்பது அல்ல. இது விற்றல் மற்றும் வாங்குதல் பரிந்துரைகளை வழங்குவதாகும். தற்போது விலை $2000 ஆக இருந்து, நாளை $2010 ஆக இருக்கும் என்று கணித்தால், உண்மையான முடிவு $2020 ஆக இருந்து நாம் கூடுதல் பணம் சம்பாதித்தால் நாம் கவலைப்பட மாட்டோம். ஆனால் அது $2010 எனக் கணித்து, அந்தப் பரிந்துரையின் அடிப்படையில் ETH-ஐ வாங்கி, விலை $1990 ஆகக் குறைந்தால் நாம் கண்டிப்பாக கவலைப்படுவோம்.
1for index in range(0,len(wethusdc_quotes)-CYCLES_BACK):முழுமையான வரலாறு (கணிப்பிற்குப் பயன்படுத்தப்பட்ட மதிப்புகள் மற்றும் அதை ஒப்பிடுவதற்கான நிஜ உலக மதிப்பு) கிடைக்கும் நிகழ்வுகளை மட்டுமே நாம் பார்க்க முடியும். இதன் பொருள், புதிய நிகழ்வு CYCLES_BACK-க்கு முன்பு தொடங்கியதாக இருக்க வேண்டும்.
1 wethusdc_slice = wethusdc_quotes[index:index+CYCLES_BACK]2 wethwbtc_slice = wethwbtc_quotes[index:index+CYCLES_BACK]முகவர் பயன்படுத்தும் அதே எண்ணிக்கையிலான மாதிரிகளைப் பெற ஸ்லைஸ்களைப் (slices) (opens in a new tab) பயன்படுத்தவும். இங்கிருந்து அடுத்த பகுதிக்கு இடையிலான குறியீடு முகவரில் உள்ள அதே கணிப்பைப் பெறும் குறியீடாகும்.
1 predicted_price = Decimal(response.choices[0].message.content.strip())2 real_price = wethusdc_quotes[index+CYCLES_BACK].price3 prediction_time_price = wethusdc_quotes[index+CYCLES_BACK-1].priceகணிக்கப்பட்ட விலை, உண்மையான விலை மற்றும் கணிப்பின் போது இருந்த விலை ஆகியவற்றைப் பெறவும். பரிந்துரை வாங்குவதற்கா அல்லது விற்பதற்கா என்பதைத் தீர்மானிக்க, கணிப்பின் போது இருந்த விலை நமக்குத் தேவை.
1 error = abs(predicted_price - real_price)2 total_error += error3 print (f"Prediction for {prediction_time}: predicted {predicted_price} USD, real {real_price} USD, error {error} USD")பிழையைக் கணக்கிட்டு, அதை மொத்தத்தில் சேர்க்கவும்.
1 recomended_action = 'buy' if predicted_price > prediction_time_price else 'sell'2 price_increase = real_price - prediction_time_price3 changes.append(price_increase if recomended_action == 'buy' else -price_increase)changes-க்கு, ஒரு ETH-ஐ வாங்குவது அல்லது விற்பதன் பணவியல் தாக்கத்தை நாம் விரும்புகிறோம். எனவே முதலில், நாம் பரிந்துரையைத் தீர்மானிக்க வேண்டும், பின்னர் உண்மையான விலை எவ்வாறு மாறியது என்பதையும், பரிந்துரை பணம் சம்பாதித்ததா (நேர்மறையான மாற்றம்) அல்லது பணத்தை இழக்கச் செய்ததா (எதிர்மறையான மாற்றம்) என்பதையும் மதிப்பிட வேண்டும்.
1print (f"Mean prediction error over {len(wethusdc_quotes)-CYCLES_BACK} predictions: {total_error / Decimal(len(wethusdc_quotes)-CYCLES_BACK)} USD")2
3length_changes = Decimal(len(changes))4mean_change = sum(changes, Decimal(0)) / length_changes5print (f"Mean change per recommendation: {mean_change} USD")6var = sum((x - mean_change) ** 2 for x in changes) / length_changes7print (f"Standard variance of changes: {var.sqrt().quantize(Decimal("0.01"))} USD")முடிவுகளைத் தெரிவிக்கவும்.
1print (f"Profitable days: {len(list(filter(lambda x: x > 0, changes)))/length_changes:.2%}")2print (f"Losing days: {len(list(filter(lambda x: x < 0, changes)))/length_changes:.2%}")லாபகரமான நாட்களின் எண்ணிக்கை மற்றும் நஷ்டமான நாட்களின் எண்ணிக்கையைக் கணக்கிட filter (opens in a new tab)-ஐப் பயன்படுத்தவும். இதன் முடிவு ஒரு ஃபில்டர் ஆப்ஜெக்ட் (filter object) ஆகும், இதன் நீளத்தைப் பெற நாம் அதை ஒரு பட்டியலாக மாற்ற வேண்டும்.
பரிவர்த்தனைகளைச் சமர்ப்பித்தல்
இப்போது நாம் உண்மையில் பரிவர்த்தனைகளைச் சமர்ப்பிக்க வேண்டும். இருப்பினும், கணினி நிரூபிக்கப்படுவதற்கு முன்பு, இந்த கட்டத்தில் உண்மையான பணத்தைச் செலவிட நான் விரும்பவில்லை. அதற்குப் பதிலாக, நாம் மெயின்நெட்டின் உள்ளூர் ஃபோர்க்கை (local fork) உருவாக்கி, அந்த நெட்வொர்க்கில் "வர்த்தகம்" செய்வோம்.
உள்ளூர் ஃபோர்க்கை உருவாக்கி வர்த்தகத்தை இயக்குவதற்கான படிகள் இங்கே.
-
Foundry (opens in a new tab)-ஐ நிறுவவும்
-
anvil(opens in a new tab)-ஐத் தொடங்கவும்1anvil --fork-url https://eth.drpc.org --block-time 12anvilFoundry-க்கான இயல்புநிலை URL-ஆன http://localhost:8545-இல் (opens in a new tab) கேட்கிறது, எனவே பிளாக்செயினைக் கையாள நாம் பயன்படுத்தும்castகட்டளைக்கு (opens in a new tab) URL-ஐக் குறிப்பிட வேண்டியதில்லை. -
anvil-இல் இயங்கும்போது, ETH-ஐக் கொண்ட பத்து சோதனைக் கணக்குகள் உள்ளன—முதலாவது கணக்கிற்கு என்விரான்மென்ட் மாறிகளை (environment variables) அமைக்கவும்1PRIVATE_KEY=0xac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff802ADDRESS=`cast wallet address $PRIVATE_KEY` -
நாம் பயன்படுத்த வேண்டிய ஒப்பந்தங்கள் இவை.
SwapRouter(opens in a new tab) என்பது நாம் உண்மையில் வர்த்தகம் செய்யப் பயன்படுத்தும் Uniswap v3 ஒப்பந்தமாகும். நாம் நேரடியாகப் பூல் மூலமாகவும் வர்த்தகம் செய்யலாம், ஆனால் இது மிகவும் எளிதானது.கீழே உள்ள இரண்டு மாறிகள் WETH மற்றும் USDC-க்கு இடையில் மாற்றுவதற்குத் (swap) தேவையான Uniswap v3 பாதைகளாகும்.
1WETH_ADDRESS=0xC02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc22USDC_ADDRESS=0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB483POOL_ADDRESS=0x88e6A0c2dDD26FEEb64F039a2c41296FcB3f56404SWAP_ROUTER=0xE592427A0AEce92De3Edee1F18E0157C058615645WETH_TO_USDC=0xC02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc20001F4A0b86991c6218b36c1d19D4a2e9Eb0cE3606eB486USDC_TO_WETH=0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB480001F4C02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2 -
ஒவ்வொரு சோதனைக் கணக்கிலும் 10,000 ETH உள்ளது. வர்த்தகத்திற்காக 1000 WETH-ஐப் பெற 1000 ETH-ஐ ரேப் (wrap) செய்ய WETH ஒப்பந்தத்தைப் பயன்படுத்தவும்.
1cast send $WETH_ADDRESS "deposit()" --value 1000ether --private-key $PRIVATE_KEY -
500 WETH-ஐ USDC-க்கு வர்த்தகம் செய்ய
SwapRouter-ஐப் பயன்படுத்தவும்.1cast send $WETH_ADDRESS "approve(address,uint256)" $SWAP_ROUTER 500ether --private-key $PRIVATE_KEY2MAXINT=`cast max-int uint256`3cast send $SWAP_ROUTER \4 "exactInput((bytes,address,uint256,uint256,uint256))" \5 "($WETH_TO_USDC,$ADDRESS,$MAXINT,500ether,1000000)" \6 --private-key $PRIVATE_KEYapproveஅழைப்புSwapRouter-ஐ நமது சில டோக்கன்களைச் செலவிட அனுமதிக்கும் ஒரு அலவன்ஸை (allowance) உருவாக்குகிறது. ஒப்பந்தங்களால் நிகழ்வுகளைக் கண்காணிக்க முடியாது, எனவே நாம் டோக்கன்களை நேரடியாகSwapRouterஒப்பந்தத்திற்கு மாற்றினால், அதற்குப் பணம் செலுத்தப்பட்டதா என்பது தெரியாது. அதற்குப் பதிலாக,SwapRouterஒப்பந்தத்தை ஒரு குறிப்பிட்ட தொகையைச் செலவிட நாம் அனுமதிக்கிறோம், பின்னர்SwapRouterஅதைச் செய்கிறது. இதுSwapRouter-ஆல் அழைக்கப்படும் ஒரு செயல்பாட்டின் மூலம் செய்யப்படுகிறது, எனவே அது வெற்றிகரமாக இருந்ததா என்பதை அது அறியும். -
உங்களிடம் இரண்டு டோக்கன்களும் போதுமான அளவு உள்ளதா என்பதைச் சரிபார்க்கவும்.
1cast call $WETH_ADDRESS "balanceOf(address)" $ADDRESS | cast from-wei2echo `cast call $USDC_ADDRESS "balanceOf(address)" $ADDRESS | cast to-dec`/10^6 | bc
இப்போது நம்மிடம் WETH மற்றும் USDC இருப்பதால், நாம் உண்மையில் முகவரை இயக்கலாம்.
1git checkout 05-trade2uv run agent.pyவெளியீடு இது போன்றதாக இருக்கும்:
1(ai-trading-agent) qbzzt@Ori-Cloudnomics:~/260215-ai-agent$ uv run agent.py2Current price: 1843.163In 2026-02-06T23:07, expected price: 1724.41 USD4Account balances before trade:5USDC Balance: 927301.5782726WETH Balance: 5007Sell, I expect the price to go down by 118.75 USD8Approve transaction sent: 74e367ddbb407c1aaf567d87aa5863049991b1d2aa092b6b85195d925e2bd41f9Approve transaction mined.10Sell transaction sent: fad1bcf938585c9e90364b26ac7a80eea9efd34c37e5db81e58d7655bcae28bf11Sell transaction mined.12Account balances after trade:13USDC Balance: 929143.79711614WETH Balance: 499இதை உண்மையில் பயன்படுத்த, உங்களுக்குச் சில சிறிய மாற்றங்கள் தேவை.
- வரி 14-இல்,
MAINNET_URL-ஐhttps://eth.drpc.orgபோன்ற உண்மையான அணுகல் புள்ளிக்கு (access point) மாற்றவும் - வரி 28-இல்,
PRIVATE_KEY-ஐ உங்கள் சொந்த தனிப்பட்ட திறவுகோலுக்கு (private key) மாற்றவும் - நீங்கள் மிகவும் செல்வந்தராக இருந்து, நிரூபிக்கப்படாத ஒரு முகவருக்காக ஒவ்வொரு நாளும் 1 ETH-ஐ வாங்கவோ விற்கவோ முடியாவிட்டால்,
WETH_TRADE_AMOUNT-ஐக் குறைக்க நீங்கள் 29-ஐ மாற்ற விரும்பலாம்
குறியீடு விளக்கம்
இதோ புதிய குறியீடு.
1SWAP_ROUTER_ADDRESS=Web3.to_checksum_address("0xE592427A0AEce92De3Edee1F18E0157C05861564")2WETH_TO_USDC=bytes.fromhex("C02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc20001F4A0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48")3USDC_TO_WETH=bytes.fromhex("A0b86991c6218b36c1d19D4a2e9Eb0cE3606eB480001F4C02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2")4PRIVATE_KEY="0xac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff80"படி 4-இல் நாம் பயன்படுத்திய அதே மாறிகள்.
1WETH_TRADE_AMOUNT=1வர்த்தகம் செய்ய வேண்டிய தொகை.
1ERC20_ABI = [2 { "name": "symbol", ... },3 { "name": "decimals", ... },4 { "name": "balanceOf", ...},5 { "name": "approve", ...}6]உண்மையில் வர்த்தகம் செய்ய, நமக்கு approve செயல்பாடு தேவை. முன்னும் பின்னும் உள்ள நிலுவைகளைக் (balances) காட்டவும் நாம் விரும்புகிறோம், எனவே நமக்கு balanceOf-உம் தேவை.
1SWAP_ROUTER_ABI = [2 { "name": "exactInput", ...},3]SwapRouter ABI-இல் நமக்கு exactInput மட்டுமே தேவை. சரியாக ஒரு WETH-ஐ வாங்க நாம் பயன்படுத்தக்கூடிய exactOutput என்ற தொடர்புடைய செயல்பாடு உள்ளது, ஆனால் எளிமைக்காக நாம் இரண்டு சந்தர்ப்பங்களிலும் exactInput-ஐப் பயன்படுத்துகிறோம்.
1account = w3.eth.account.from_key(PRIVATE_KEY)2swap_router = w3.eth.contract(3 address=SWAP_ROUTER_ADDRESS,4 abi=SWAP_ROUTER_ABI5)account (opens in a new tab) மற்றும் SwapRouter ஒப்பந்தத்திற்கான Web3 வரையறைகள்.
1def txn_params() -> dict:2 return {3 "from": account.address,4 "value": 0,5 "gas": 300000,6 "nonce": w3.eth.get_transaction_count(account.address),7 }பரிவர்த்தனை அளவுருக்கள். நான்ஸ் (nonce) (opens in a new tab) ஒவ்வொரு முறையும் மாற வேண்டும் என்பதால் நமக்கு இங்கே ஒரு செயல்பாடு தேவை.
1def approve_token(contract: Contract, amount: int):SwapRouter-க்கான டோக்கன் அலவன்ஸை அங்கீகரிக்கவும்.
1 txn = contract.functions.approve(SWAP_ROUTER_ADDRESS, amount).build_transaction(txn_params())2 signed_txn = w3.eth.account.sign_transaction(txn, private_key=PRIVATE_KEY)3 tx_hash = w3.eth.send_raw_transaction(signed_txn.raw_transaction)Web3-இல் நாம் ஒரு பரிவர்த்தனையை இப்படித்தான் அனுப்புகிறோம். முதலில் பரிவர்த்தனையை உருவாக்க Contract ஆப்ஜெக்ட்டைப் (opens in a new tab) பயன்படுத்துகிறோம். பின்னர் PRIVATE_KEY-ஐப் பயன்படுத்திப் பரிவர்த்தனையில் கையொப்பமிட web3.eth.account.sign_transaction (opens in a new tab)-ஐப் பயன்படுத்துகிறோம். இறுதியாக, பரிவர்த்தனையை அனுப்ப w3.eth.send_raw_transaction (opens in a new tab)-ஐப் பயன்படுத்துகிறோம்.
1 print(f"Approve transaction sent: {tx_hash.hex()}")2 w3.eth.wait_for_transaction_receipt(tx_hash)3 print("Approve transaction mined.")w3.eth.wait_for_transaction_receipt (opens in a new tab) பரிவர்த்தனை மைன் (mine) செய்யப்படும் வரை காத்திருக்கிறது. தேவைப்பட்டால் அது ரசீதைத் திருப்பித் தரும்.
1SELL_PARAMS = {2 "path": WETH_TO_USDC,3 "recipient": account.address,4 "deadline": 2**256 - 1,5 "amountIn": WETH_TRADE_AMOUNT * 10 ** wethusdc_pool.token1.decimals,6 "amountOutMinimum": 0,7}WETH-ஐ விற்கும்போது இவை அளவுருக்களாகும்.
1def make_buy_params(quote: Quote) -> dict:2 return {3 "path": USDC_TO_WETH,4 "recipient": account.address,5 "deadline": 2**256 - 1,6 "amountIn": int(quote.price*WETH_TRADE_AMOUNT) * 10**wethusdc_pool.token0.decimals,7 "amountOutMinimum": 0,8 }SELL_PARAMS-க்கு மாறாக, வாங்கும் அளவுருக்கள் மாறலாம். உள்ளீட்டுத் தொகை என்பது quote-இல் கிடைக்கும் 1 WETH-இன் விலையாகும்.
1def buy(quote: Quote):2 buy_params = make_buy_params(quote)3 approve_token(wethusdc_pool.token0.contract, buy_params["amountIn"])4 txn = swap_router.functions.exactInput(buy_params).build_transaction(txn_params())5 signed_txn = w3.eth.account.sign_transaction(txn, private_key=PRIVATE_KEY)6 tx_hash = w3.eth.send_raw_transaction(signed_txn.raw_transaction)7 print(f"Buy transaction sent: {tx_hash.hex()}")8 w3.eth.wait_for_transaction_receipt(tx_hash)9 print("Buy transaction mined.")10
11
12def sell():13 approve_token(wethusdc_pool.token1.contract,14 WETH_TRADE_AMOUNT * 10**wethusdc_pool.token1.decimals)15 txn = swap_router.functions.exactInput(SELL_PARAMS).build_transaction(txn_params())16 signed_txn = w3.eth.account.sign_transaction(txn, private_key=PRIVATE_KEY)17 tx_hash = w3.eth.send_raw_transaction(signed_txn.raw_transaction)18 print(f"Sell transaction sent: {tx_hash.hex()}")19 w3.eth.wait_for_transaction_receipt(tx_hash)20 print("Sell transaction mined.")buy() மற்றும் sell() செயல்பாடுகள் கிட்டத்தட்ட ஒரே மாதிரியானவை. முதலில் நாம் SwapRouter-க்கு போதுமான அலவன்ஸை அங்கீகரிக்கிறோம், பின்னர் சரியான பாதை மற்றும் தொகையுடன் அதை அழைக்கிறோம்.
1def balances():2 token0_balance = wethusdc_pool.token0.contract.functions.balanceOf(account.address).call()3 token1_balance = wethusdc_pool.token1.contract.functions.balanceOf(account.address).call()4
5 print(f"{wethusdc_pool.token0.symbol} Balance: {Decimal(token0_balance) / Decimal(10 ** wethusdc_pool.token0.decimals)}")6 print(f"{wethusdc_pool.token1.symbol} Balance: {Decimal(token1_balance) / Decimal(10 ** wethusdc_pool.token1.decimals)}")இரண்டு நாணயங்களிலும் பயனர் நிலுவைகளைத் தெரிவிக்கவும்.
1print("Account balances before trade:")2balances()3
4if (expected_price > current_price):5 print(f"Buy, I expect the price to go up by {expected_price - current_price} USD")6 buy(wethusdc_quotes[-1])7else:8 print(f"Sell, I expect the price to go down by {current_price - expected_price} USD")9 sell()10
11print("Account balances after trade:")12balances()இந்த முகவர் தற்போது ஒரு முறை மட்டுமே வேலை செய்கிறது. இருப்பினும், இதை crontab (opens in a new tab)-இலிருந்து இயக்குவதன் மூலமாகவோ அல்லது 368-400 வரிகளை ஒரு லூப்பில் (loop) ரேப் செய்து, அடுத்த சுழற்சிக்கான நேரம் வரும் வரை காத்திருக்க time.sleep (opens in a new tab)-ஐப் பயன்படுத்துவதன் மூலமாகவோ தொடர்ந்து வேலை செய்யும்படி நீங்கள் மாற்றலாம்.
சாத்தியமான மேம்பாடுகள்
இது ஒரு முழுமையான தயாரிப்புப் பதிப்பு அல்ல; இது அடிப்படைகளைக் கற்பிப்பதற்கான ஒரு எடுத்துக்காட்டு மட்டுமே. மேம்பாடுகளுக்கான சில யோசனைகள் இங்கே உள்ளன.
புத்திசாலித்தனமான வர்த்தகம்
என்ன செய்ய வேண்டும் என்பதைத் தீர்மானிக்கும்போது முகவர் புறக்கணிக்கும் இரண்டு முக்கியமான உண்மைகள் உள்ளன.
- எதிர்பார்க்கப்படும் மாற்றத்தின் அளவு. விலை குறையும் என்று எதிர்பார்க்கப்பட்டால், சரிவின் அளவைப் பொருட்படுத்தாமல் முகவர் ஒரு நிலையான அளவு
WETH-ஐ விற்கிறது. சிறிய மாற்றங்களைப் புறக்கணித்து, விலை எவ்வளவு குறையும் என்று நாம் எதிர்பார்க்கிறோம் என்பதன் அடிப்படையில் விற்பது சிறப்பாக இருக்கும் என்று வாதிடலாம். - தற்போதைய போர்ட்ஃபோலியோ (portfolio). உங்கள் போர்ட்ஃபோலியோவில் 10% WETH-இல் இருந்து, விலை உயரும் என்று நீங்கள் நினைத்தால், மேலும் வாங்குவது அர்த்தமுள்ளதாக இருக்கும். ஆனால் உங்கள் போர்ட்ஃபோலியோவில் 90% WETH-இல் இருந்தால், நீங்கள் போதுமான அளவு வெளிப்பட்டிருக்கலாம், மேலும் வாங்க வேண்டிய அவசியமில்லை. விலை குறையும் என்று நீங்கள் எதிர்பார்த்தால் இதற்கு நேர்மாறானது உண்மையாகும்.
உங்கள் வர்த்தக உத்தியை ரகசியமாக வைத்திருக்க விரும்பினால் என்ன செய்வது?
AI விற்பனையாளர்கள் நீங்கள் அவர்களின் LLM-களுக்கு அனுப்பும் வினவல்களைப் பார்க்க முடியும், இது உங்கள் முகவருடன் நீங்கள் உருவாக்கிய மேதை வர்த்தக அமைப்பை வெளிப்படுத்தக்கூடும். அதிகமான மக்கள் பயன்படுத்தும் ஒரு வர்த்தக அமைப்பு பயனற்றது, ஏனெனில் நீங்கள் வாங்க விரும்பும்போது அதிகமான மக்கள் வாங்க முயற்சிப்பார்கள் (மற்றும் விலை உயரும்) மற்றும் நீங்கள் விற்க விரும்பும்போது விற்க முயற்சிப்பார்கள் (மற்றும் விலை குறையும்).
இந்தச் சிக்கலைத் தவிர்க்க, எடுத்துக்காட்டாக, LM-Studio (opens in a new tab)-ஐப் பயன்படுத்தி நீங்கள் ஒரு LLM-ஐ உள்ளூரில் (locally) இயக்கலாம்.
AI பாட்டிலிருந்து (bot) AI முகவருக்கு
இது ஒரு AI பாட், AI முகவர் அல்ல என்று நீங்கள் ஒரு நல்ல வாதத்தை முன்வைக்கலாம். இது முன்வரையறுக்கப்பட்ட தகவல்களை நம்பியிருக்கும் ஒப்பீட்டளவில் எளிமையான உத்தியைச் செயல்படுத்துகிறது. எடுத்துக்காட்டாக, Uniswap v3 பூல்களின் பட்டியல் மற்றும் அவற்றின் சமீபத்திய மதிப்புகளை வழங்கி, எந்தக் கலவையானது சிறந்த முன்கணிப்பு மதிப்பைக் கொண்டுள்ளது என்று கேட்பதன் மூலம் நாம் சுய-மேம்பாட்டை (self-improvement) இயக்கலாம்.
ஸ்லிப்பேஜ் (Slippage) பாதுகாப்பு
தற்போது ஸ்லிப்பேஜ் பாதுகாப்பு (opens in a new tab) இல்லை. தற்போதைய மேற்கோள் $2000 ஆகவும், எதிர்பார்க்கப்படும் விலை $2100 ஆகவும் இருந்தால், முகவர் வாங்கும். இருப்பினும், முகவர் வாங்குவதற்கு முன்பு விலை $2200 ஆக உயர்ந்தால், இனி வாங்குவதில் அர்த்தமில்லை.
ஸ்லிப்பேஜ் பாதுகாப்பைச் செயல்படுத்த, agent.py (opens in a new tab)-இன் 325 மற்றும் 334 வரிகளில் amountOutMinimum மதிப்பைக் குறிப்பிடவும்.
முடிவுரை
AI முகவர்களுடன் தொடங்குவதற்கு இப்போது உங்களுக்குப் போதுமான அளவு தெரியும் என்று நம்புகிறேன். இது இந்த விஷயத்தின் விரிவான கண்ணோட்டம் அல்ல; அதற்காக அர்ப்பணிக்கப்பட்ட முழுப் புத்தகங்களும் உள்ளன, ஆனால் நீங்கள் தொடங்குவதற்கு இது போதுமானது. நல்வாழ்த்துக்கள்!
எனது மேலும் பல பணிகளுக்கு இங்கே பார்க்கவும் (opens in a new tab).
பக்கம் கடைசியாகப் புதுப்பிக்கப்பட்டது: 3 மார்ச், 2026