ఇతీరియముపై మీ స్వంత AI ట్రేడింగ్ ఏజెంట్ను తయారు చేయండి
ఈ ట్యుటోరియల్లో మీరు ఒక సాధారణ AI ట్రేడింగ్ ఏజెంట్ను ఎలా నిర్మించాలో నేర్చుకుంటారు. ఈ ఏజెంట్ ఈ దశలను ఉపయోగించి పనిచేస్తుంది:
- టోకెన్ యొక్క ప్రస్తుత మరియు గత ధరలను, అలాగే ఇతర సంభావ్యంగా సంబంధిత సమాచారాన్ని చదవండి
- ఈ సమాచారంతో, అది ఎలా సంబంధితంగా ఉండవచ్చో వివరించడానికి నేపథ్య సమాచారంతో పాటు ఒక క్వెరీని నిర్మించండి
- క్వెరీని సమర్పించి, అంచనా వేయబడిన ధరను తిరిగి పొందండి
- సిఫార్సు ఆధారంగా ట్రేడ్ చేయండి
- వేచి ఉండి, పునరావృతం చేయండి
ఈ ఏజెంట్ సమాచారాన్ని ఎలా చదవాలో, దానిని ఉపయోగపడే జవాబును ఇచ్చే క్వెరీగా అనువదించాలో, మరియు ఆ జవాబును ఎలా ఉపయోగించాలో ప్రదర్శిస్తుంది. ఇవన్నీ AI ఏజెంట్కు అవసరమైన దశలు. ఈ ఏజెంట్ Pythonలో అమలు చేయబడింది, ఎందుకంటే ఇది AIలో ఉపయోగించే అత్యంత సాధారణ భాష.
ఇది ఎందుకు చేయాలి?
స్వయంచాలక ట్రేడింగ్ ఏజెంట్లు అభివృద్ధి చేసేవారిని ట్రేడింగ్ వ్యూహాన్ని ఎంచుకోవడానికి మరియు అమలు చేయడానికి అనుమతిస్తాయి. AI agents మరింత సంక్లిష్టమైన మరియు డైనమిక్ ట్రేడింగ్ వ్యూహాలకు అనుమతిస్తాయి, అభివృద్ధి చేసేవాడు ఉపయోగించాలని కూడా పరిగణించని సమాచారం మరియు అల్గారిథమ్లను ఉపయోగించే అవకాశం ఉంది.
ఉపకరణాలు
ఈ ట్యుటోరియల్ కోట్స్ మరియు ట్రేడింగ్ కోసం 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 ఇతీరియము API. ఇది ఉపయోగించడానికి చాలా సులభం.
బ్లాక్ చైనుపై ట్రేడింగ్
అనేక వికేంద్రీకృత ఎక్స్ఛేంజ్లు (DEX) ఉన్నాయి, ఇవి ఇతీరియముపై టోకెన్లను ట్రేడ్ చేయడానికి మిమ్మల్ని అనుమతిస్తాయి. అయినప్పటికీ, ఆర్బిట్రేజ్ కారణంగా అవి ఒకే రకమైన ఎక్స్ఛేంజ్ రేట్లను కలిగి ఉంటాయి.
Uniswap (opens in a new tab) అనేది విస్తృతంగా ఉపయోగించే DEX, దీనిని మనం కోట్స్ (టోకెన్ సాపేక్ష విలువలను చూడటానికి) మరియు ట్రేడ్ల కోసం ఉపయోగించవచ్చు.
OpenAI
ఒక పెద్ద భాషా నమూనా కోసం, నేను OpenAI (opens in a new tab)తో ప్రారంభించాలని ఎంచుకున్నాను. ఈ ట్యుటోరియల్లోని అప్లికేషన్ను అమలు చేయడానికి మీరు API యాక్సెస్ కోసం చెల్లించాలి. $5 కనీస చెల్లింపు కంటే ఎక్కువ సరిపోతుంది.
అభివృద్ధి, దశలవారీగా
అభివృద్ధిని సులభతరం చేయడానికి, మేము దశలవారీగా ముందుకు సాగుతాము. ప్రతి దశ GitHubలో ఒక బ్రాంచ్.
ప్రారంభించడం
UNIX లేదా Linux కింద ప్రారంభించడానికి దశలు ఉన్నాయి (WSL (opens in a new tab)తో సహా)
-
మీ దగ్గర ఇప్పటికే లేకపోతే, Python (opens in a new tab)ను డౌన్లోడ్ చేసి ఇన్స్టాల్ చేయండి.
-
GitHub రిపోజిటరీని క్లోన్ చేయండి.
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 -
వర్చువల్ ఎన్విరాన్మెంట్ను యాక్టివేట్ చేయండి.
1source .venv/bin/activate -
Python మరియు Web3 సరిగ్గా పనిచేస్తున్నాయో లేదో ధృవీకరించడానికి,
python3ను అమలు చేసి, దానికి ఈ ప్రోగ్రామ్ను అందించండి. మీరు దానిని>>>ప్రాంప్ట్ వద్ద నమోదు చేయవచ్చు; ఫైల్ను సృష్టించాల్సిన అవసరం లేదు.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 ఆబ్జెక్ట్ల జాబితాను అందుకోవాలి, ప్రతి దానిలో టైమ్స్టాంప్, ధర, మరియు ఆస్తి (ప్రస్తుతం ఎల్లప్పుడూ 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అన్నీ చూపించుమాకు అవసరమైన లైబ్రరీలను దిగుమతి చేయండి. వాటిని ఉపయోగించినప్పుడు కింద వివరించబడ్డాయి.
1print = functools.partial(print, flush=True)Python యొక్క printను అవుట్పుట్ను తక్షణమే ఫ్లష్ చేసే వెర్షన్తో భర్తీ చేస్తుంది. ఇది దీర్ఘకాలం నడిచే స్క్రిప్ట్లో ఉపయోగపడుతుంది, ఎందుకంటే మేము స్టేటస్ అప్డేట్లు లేదా డీబగ్గింగ్ అవుట్పుట్ కోసం వేచి ఉండాలనుకోము.
1MAINNET_URL = "https://eth.drpc.org"మెయిన్నెట్కు చేరుకోవడానికి ఒక URL. మీరు నోడ్ యాజ్ ఎ సర్వీస్ నుండి ఒకటి పొందవచ్చు లేదా 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ఒక ఇతీరియము మెయిన్నెట్ బ్లాక్ సాధారణంగా ప్రతి పన్నెండు సెకన్లకు జరుగుతుంది, కాబట్టి ఇవి ఒక కాల వ్యవధిలో జరగాలని మేము ఆశించే బ్లాక్ల సంఖ్య. ఇది ఖచ్చితమైన సంఖ్య కాదని గమనించండి. బ్లాక్ ప్రపోజర్ డౌన్ అయినప్పుడు, ఆ బ్లాక్ దాటవేయబడుతుంది మరియు తదుపరి బ్లాక్ సమయం 24 సెకన్లు. మేము టైమ్స్టాంప్ కోసం ఖచ్చితమైన బ్లాక్ను పొందాలనుకుంటే, మేము బైనరీ శోధన (opens in a new tab)ను ఉపయోగిస్తాము. అయినప్పటికీ, ఇది మన ప్రయోజనాలకు తగినంత దగ్గరగా ఉంది. భవిష్యత్తును అంచనా వేయడం ఖచ్చితమైన శాస్త్రం కాదు.
1CYCLE_BLOCKS = DAY_BLOCKSచక్రం యొక్క పరిమాణం. మేము ప్రతి చక్రానికి ఒకసారి కోట్లను సమీక్షించి, తదుపరి చక్రం చివరిలో విలువను అంచనా వేయడానికి ప్రయత్నిస్తాము.
1# The address of the pool we're reading2WETHUSDC_ADDRESS = Web3.to_checksum_address("0x88e6A0c2dDD26FEEb64F039a2c41296FcB3f5640")కోట్ విలువలు Uniswap 3 USDC/WETH పూల్ నుండి చిరునామా 0x88e6A0c2dDD26FEEb64F039a2c41296FcB3f5640 (opens in a new tab) వద్ద తీసుకోబడ్డాయి. ఈ చిరునామా ఇప్పటికే చెక్సమ్ రూపంలో ఉంది, కానీ కోడ్ను పునర్వినియోగపరచడానికి Web3.to_checksum_address (opens in a new tab)ను ఉపయోగించడం మంచిది.
1POOL_ABI = [2 { "name": "slot0", ... },3 { "name": "token0", ... },4 { "name": "token1", ... },5]67ERC20_ABI = [8 { "name": "symbol", ... },9 { "name": "decimals", ... }10]అన్నీ చూపించుఇవి మేము సంప్రదించాల్సిన రెండు కాంట్రాక్టుల కోసం ABIలు (opens in a new tab). కోడ్ను సంక్షిప్తంగా ఉంచడానికి, మేము కాల్ చేయాల్సిన ఫంక్షన్లను మాత్రమే చేర్చుతాము.
1w3 = Web3(Web3.HTTPProvider(MAINNET_URL))Web3 (opens in a new tab) లైబ్రరీని ప్రారంభించి, ఒక ఇతీరియము నోడ్కు కనెక్ట్ చేయండి.
1@dataclass(frozen=True)2class ERC20Token:3 address: str4 symbol: str5 decimals: int6 contract: Contractఇది Pythonలో డేటా క్లాస్ను సృష్టించడానికి ఒక మార్గం. Contract (opens in a new tab) డేటా రకం కాంట్రాక్ట్కు కనెక్ట్ చేయడానికి ఉపయోగించబడుతుంది. (frozen=True)ను గమనించండి. Pythonలో బూలియన్లు (opens in a new tab) True లేదా Falseగా నిర్వచించబడ్డాయి, క్యాపిటలైజ్ చేయబడ్డాయి. ఈ డేటా క్లాస్ ఫ్రోజెన్ అంటే ఫీల్డ్లను మార్చలేము.
ఇండెంటేషన్ను గమనించండి. C-ఉత్పన్న భాషలకు (opens in a new tab) విరుద్ధంగా, Python బ్లాక్లను సూచించడానికి ఇండెంటేషన్ను ఉపయోగిస్తుంది. Python ఇంటర్ప్రిటర్ తదుపరి నిర్వచనం ఈ డేటా క్లాస్లో భాగం కాదని తెలుసుకుంటుంది, ఎందుకంటే ఇది డేటా క్లాస్ ఫీల్డ్ల వలె అదే ఇండెంటేషన్లో ప్రారంభం కాదు.
1@dataclass(frozen=True)2class PoolInfo:3 address: str4 token0: ERC20Token5 token1: ERC20Token6 contract: Contract7 asset: str8 decimal_factor: Decimal = 1Decimal (opens in a new tab) రకం దశాంశ భిన్నాలను ఖచ్చితంగా నిర్వహించడానికి ఉపయోగించబడుతుంది.
1 def get_price(self, block: int) -> Decimal:ఇది Pythonలో ఫంక్షన్ను నిర్వచించే మార్గం. నిర్వచనం PoolInfoలో ఇంకా భాగమని చూపించడానికి ఇండెంట్ చేయబడింది.
డేటా క్లాస్లో భాగమైన ఫంక్షన్లో మొదటి పరామితి ఎల్లప్పుడూ self, ఇక్కడ పిలిచిన డేటా క్లాస్ ఉదాహరణ. ఇక్కడ మరొక పరామితి, బ్లాక్ నంబర్ ఉంది.
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])EVM నుండి Web3పై ఫంక్షన్ను కాల్ చేయడానికి సింటాక్స్ ఇది: <contract object>.functions.<function name>().call(<parameters>). పరామితులు EVM ఫంక్షన్ యొక్క పరామితులు (ఏవైనా ఉంటే; ఇక్కడ లేవు) లేదా బ్లాక్ చైను ప్రవర్తనను సవరించడానికి పేరు పెట్టబడిన పరామితులు (opens in a new tab) కావచ్చు. ఇక్కడ మేము ఒకటి, block_identifierను, మనం అమలు చేయాలనుకుంటున్న బ్లాక్ నంబర్ను పేర్కొనడానికి ఉపయోగిస్తాము.
ఫలితం ఈ నిర్మాణం, శ్రేణి రూపంలో (opens in a new tab). మొదటి విలువ రెండు టోకెన్ల మధ్య మార్పిడి రేటు యొక్క ఫంక్షన్.
1 raw_price = (sqrt_price_x96 / Decimal(2**96)) ** 2ఆన్చైన్ గణనలను తగ్గించడానికి, Uniswap v3 వాస్తవ మార్పిడి కారకాన్ని నిల్వ చేయదు కానీ దాని వర్గమూలాన్ని నిల్వ చేస్తుంది. EVM ఫ్లోటింగ్ పాయింట్ గణితం లేదా భిన్నాలకు మద్దతు ఇవ్వనందున, వాస్తవ విలువకు బదులుగా, ప్రతిస్పందన
1 # (token1 per token0)2 return 1/(raw_price * self.decimal_factor)మనం పొందే ముడి ధర ప్రతి token1కి మనం పొందే token0 సంఖ్య. మా పూల్లో token0 USDC (యుఎస్ డాలర్తో సమానమైన విలువ కలిగిన స్టేబుల్కాయిన్) మరియు token1 WETH (opens in a new tab). మనం నిజంగా కోరుకునే విలువ WETHకు డాలర్ల సంఖ్య, దాని విలోమం కాదు.
దశాంశ కారకం రెండు టోకెన్ల కోసం దశాంశ కారకాల (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()56 return ERC20Token(7 address=address,8 symbol=symbol,9 decimals=decimals,10 contract=token11 )అన్నీ చూపించుఈ ఫంక్షన్ ఒక చిరునామాను తీసుకుని, ఆ చిరునామాలోని టోకెన్ కాంట్రాక్ట్ గురించిన సమాచారాన్ని తిరిగి ఇస్తుంది. కొత్త Web3 Contract (opens in a new tab)ని సృష్టించడానికి, మేము w3.eth.contractకు చిరునామా మరియు ABIని అందిస్తాము.
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)78 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>" సింటాక్స్ ఒక ఫార్మాట్ చేయబడిన స్ట్రింగ్ (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 స్టేట్మెంట్ (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[<type>]ను ఉపయోగించి ఒక నిర్దిష్ట రకాన్ని మాత్రమే కలిగి ఉండే జాబితాను (opens in a new tab) నిర్వచిస్తారు.
1 quotes = []2 for block in range(start_block, end_block + 1, step):Pythonలో for లూప్ (opens in a new tab) సాధారణంగా జాబితాపై పునరావృతమవుతుంది. కోట్లను కనుగొనడానికి బ్లాక్ నంబర్ల జాబితా range (opens in a new tab) నుండి వస్తుంది.
1 quote = get_quote(pool, block)2 quotes.append(quote)3 return quotesప్రతి బ్లాక్ నంబర్ కోసం, Quote ఆబ్జెక్ట్ను పొంది, దానిని quotes జాబితాకు చేర్చండి. ఆ జాబితాను తిరిగి ఇవ్వండి.
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)89pprint(quotes)అన్నీ చూపించుఇది స్క్రిప్ట్ యొక్క ప్రధాన కోడ్. పూల్ సమాచారాన్ని చదివి, పన్నెండు కోట్లను పొంది, వాటిని pprint (opens in a new tab) చేయండి.
ఒక ప్రాంప్ట్ సృష్టిస్తోంది
తరువాత, మనం ఈ కోట్ల జాబితాను LLM కోసం ఒక ప్రాంప్ట్గా మార్చాలి మరియు ఊహించిన భవిష్యత్ విలువను పొందాలి.
1git checkout 03-create-prompt2uv run agent.pyఇప్పుడు అవుట్పుట్ LLM కోసం ఒక ప్రాంప్ట్గా ఉంటుంది, ఇలాంటిది:
1ఈ కోట్లను బట్టి:2ఆస్తి: WETH/USDC3 2026-01-20T16:34 3016.214 .5 .6 .7 2026-02-01T17:49 2299.1089ఆస్తి: WBTC/WETH10 2026-01-20T16:34 29.8411 .12 .13 .14 2026-02-01T17:50 33.461516172026-02-02T17:56 సమయంలో WETH/USDC కోసం విలువ ఏమని మీరు ఆశిస్తారు?1819మీ సమాధానాన్ని రెండు దశాంశ స్థానాలకు గుండ్రంగా, ఇతర టెక్స్ట్ లేకుండా ఒకే సంఖ్యగా అందించండి.అన్నీ చూపించుఇక్కడ రెండు ఆస్తులు, WETH/USDC మరియు WBTC/WETH కోసం కోట్లు ఉన్నాయని గమనించండి. మరొక ఆస్తి నుండి కోట్లను జోడించడం అంచనా యొక్క ఖచ్చితత్వాన్ని మెరుగుపరచవచ్చు.
ఒక ప్రాంప్ట్ ఎలా కనిపిస్తుంది
ఈ ప్రాంప్ట్లో మూడు విభాగాలు ఉన్నాయి, ఇవి LLM ప్రాంప్ట్లలో చాలా సాధారణం.
-
సమాచారం. LLMలకు వాటి శిక్షణ నుండి చాలా సమాచారం ఉంటుంది, కానీ అవి సాధారణంగా తాజా సమాచారాన్ని కలిగి ఉండవు. ఇక్కడ తాజా కోట్లను తిరిగి పొందడానికి ఇది కారణం. ఒక ప్రాంప్ట్కు సమాచారాన్ని జోడించడాన్ని రిట్రీవల్ ఆగ్మెంటెడ్ జనరేషన్ (RAG) (opens in a new tab) అంటారు.
-
అసలు ప్రశ్న. ఇది మనం తెలుసుకోవాలనుకుంటున్నది.
-
అవుట్పుట్ ఫార్మాటింగ్ సూచనలు. సాధారణంగా, ఒక LLM దానిని ఎలా చేరుకుందో వివరణతో ఒక అంచనాను ఇస్తుంది. ఇది మానవులకు మంచిది, కానీ కంప్యూటర్ ప్రోగ్రామ్కు కేవలం చివరి ఫలితం మాత్రమే అవసరం.
కోడ్ వివరణ
ఇక్కడ కొత్త కోడ్ ఉంది.
1from datetime import datetime, timezone, timedeltaమేము ఏ సమయం కోసం అంచనా వేయాలనుకుంటున్నామో ఆ సమయాన్ని LLMకి అందించాలి. భవిష్యత్తులో "n నిమిషాలు/గంటలు/రోజులు" సమయం పొందడానికి, మేము timedelta క్లాస్ (opens in a new tab)ను ఉపయోగిస్తాము.
1# The addresses of the pools we're reading2WETHUSDC_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 = False78 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 # (token1 per token0)12 if self.reverse:13 return 1/(raw_price * self.decimal_factor)14 else:15 return raw_price * self.decimal_factorఅన్నీ చూపించుWETH/USDC పూల్లో, token1 (WETH) ఒకటి కొనడానికి ఎన్ని token0 (USDC) అవసరమో తెలుసుకోవాలనుకుంటున్నాము. WETH/WBTC పూల్లో, token0 (WBTC, ఇది చుట్టబడిన Bitcoin) ఒకటి కొనడానికి ఎన్ని token1 (WETH) అవసరమో తెలుసుకోవాలనుకుంటున్నాము. పూల్ నిష్పత్తిని తిప్పికొట్టాల్సిన అవసరం ఉందో లేదో మనం ట్రాక్ చేయాలి.
1def read_pool(address: str, reverse: bool = False) -> PoolInfo:2 .3 .4 .56 return PoolInfo(7 .8 .9 .1011 asset= f"{token1.symbol}/{token0.symbol}" if reverse else f"{token0.symbol}/{token1.symbol}",12 reverse=reverse13 )అన్నీ చూపించుపూల్ను తిప్పికొట్టాల్సిన అవసరం ఉందో లేదో తెలుసుకోవడానికి, మేము దానిని read_poolకి ఇన్పుట్గా పొందుతాము. అలాగే, ఆస్తి చిహ్నాన్ని సరిగ్గా సెటప్ చేయాలి.
<a> if <b> else <c> సింటాక్స్ C-ఉత్పన్న భాషలో <b> ? <a> : <c> అయిన టర్నరీ షరతులతో కూడిన ఆపరేటర్ ternary conditional operator (opens in a new tab) యొక్క Python సమానమైనది.
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 ఆబ్జెక్ట్ల జాబితాను ఫార్మాట్ చేసే స్ట్రింగ్ను నిర్మిస్తుంది, అవన్నీ ఒకే ఆస్తికి వర్తిస్తాయని ఊహిస్తుంది.
1def make_prompt(quotes: list[list[Quote]], expected_time: str, asset: str) -> str:2 return f"""Pythonలో బహుళ-పంక్తి స్ట్రింగ్ లిటరల్స్ (opens in a new tab) """ .... అని వ్రాయబడ్డాయి. """.
1ఈ కోట్లను బట్టి:2{3 functools.reduce(lambda acc, q: acc + '\n' + q,4 map(lambda q: format_quotes(q), quotes))5}ఇక్కడ, మేము format_quotesతో ప్రతి కోట్ జాబితాకు ఒక స్ట్రింగ్ను రూపొందించడానికి మ్యాప్రిడ్యూస్ (opens in a new tab) నమూనాను ఉపయోగిస్తాము, ఆపై వాటిని ప్రాంప్ట్లో ఉపయోగించడానికి ఒకే స్ట్రింగ్గా తగ్గిస్తాము.
1{asset} కోసం {expected_time} సమయంలో విలువ ఏమని మీరు ఆశిస్తారు?23మీ సమాధానాన్ని రెండు దశాంశ స్థానాలకు గుండ్రంగా, ఇతర టెక్స్ట్ లేకుండా ఒకే సంఖ్యగా అందించండి.4 """ప్రాంప్ట్ యొక్క మిగిలిన భాగం ఊహించిన విధంగానే ఉంటుంది.
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)89wethwbtc_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]23print(make_prompt(wethusdc_quotes + wethwbtc_quotes, future_time, wethusdc_pool.asset))మనం ఏ భవిష్యత్ సమయ బిందువు కోసం అంచనా వేయాలనుకుంటున్నామో నిర్ణయించి, ప్రాంప్ట్ను సృష్టించండి.
LLMతో ఇంటర్ఫేస్ చేయడం
తరువాత, మేము వాస్తవ LLMని ప్రాంప్ట్ చేసి, ఊహించిన భవిష్యత్ విలువను పొందుతాము. నేను ఈ ప్రోగ్రామ్ను OpenAI ఉపయోగించి వ్రాశాను, కాబట్టి మీరు వేరే ప్రొవైడర్ను ఉపయోగించాలనుకుంటే, మీరు దానిని సర్దుబాటు చేయాలి.
-
ఖాతాకు నిధులు సమకూర్చండి (opens in a new tab)—వ్రాత సమయంలో కనీస మొత్తం $5
-
కమాండ్ లైన్లో, మీ ప్రోగ్రామ్ దానిని ఉపయోగించడానికి API కీని ఎగుమతి చేయండి
1export OPENAI_API_KEY=sk-<కీ యొక్క మిగిలిన భాగం ఇక్కడ వస్తుంది> -
ఏజెంట్ను చెక్అవుట్ చేసి అమలు చేయండి
1git checkout 04-interface-llm2uv run agent.py
ఇక్కడ కొత్త కోడ్ ఉంది.
1from openai import OpenAI23open_ai = OpenAI() # క్లయింట్ OPENAI_API_KEY ఎన్విరాన్మెంట్ వేరియబుల్ను చదువుతుందిOpenAI APIని దిగుమతి చేసి, ప్రారంభించండి.
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].price34print ("Current price:", wethusdc_quotes[-1].price)5print(f"In {future_time}, expected price: {expected_price} USD")67if (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ఊహించిన ఫలితం ఇలాంటిది:
12026-01-05T19:50 కోసం అంచనా: అంచనా 3138.93 USD, వాస్తవం 3218.92 USD, లోపం 79.99 USD22026-01-06T19:56 కోసం అంచనా: అంచనా 3243.39 USD, వాస్తవం 3221.08 USD, లోపం 22.31 USD32026-01-07T20:02 కోసం అంచనా: అంచనా 3223.24 USD, వాస్తవం 3146.89 USD, లోపం 76.35 USD42026-01-08T20:11 కోసం అంచనా: అంచనా 3150.47 USD, వాస్తవం 3092.04 USD, లోపం 58.43 USD5.6.7.82026-01-31T22:33 కోసం అంచనా: అంచనా 2637.73 USD, వాస్తవం 2417.77 USD, లోపం 219.96 USD92026-02-01T22:41 కోసం అంచనా: అంచనా 2381.70 USD, వాస్తవం 2318.84 USD, లోపం 62.86 USD102026-02-02T22:49 కోసం అంచనా: అంచనా 2234.91 USD, వాస్తవం 2349.28 USD, లోపం 114.37 USD1129 అంచనాలలో సగటు అంచనా లోపం: 83.87103448275862068965517241 USD12సిఫార్సు başına సగటు మార్పు: 4.787931034482758620689655172 USD13మార్పుల ప్రామాణిక విచలనం: 104.42 USD14లాభదాయక రోజులు: 51.72%15నష్టపోయే రోజులు: 48.28%అన్నీ చూపించుటెస్టర్ యొక్క చాలా భాగం ఏజెంట్తో సమానంగా ఉంటుంది, కానీ ఇక్కడ కొత్త లేదా సవరించిన భాగాలు ఉన్నాయి.
1CYCLES_FOR_TEST = 40 # బ్యాక్టెస్ట్ కోసం, మేము ఎన్ని చక్రాలను పరీక్షిస్తాము23# చాలా కోట్లను పొందండి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)1112wethwbtc_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# అంచనాలను సృష్టించి, వాటిని వాస్తవ చరిత్రతో సరిచూడండి23total_error = Decimal(0)4changes = []మనం ఆసక్తి చూపే రెండు రకాల లోపాలు ఉన్నాయి. మొదటిది, total_error, కేవలం ప్రిడిక్టర్ చేసిన లోపాల మొత్తం.
రెండవది, 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]ఏజెంట్ ఉపయోగించే నమూనాల సంఖ్యతో సమానమైన నమూనాలను పొందడానికి స్లైస్లను (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")23length_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)ను ఉపయోగించండి. ఫలితం ఫిల్టర్ ఆబ్జెక్ట్, దాని పొడవును పొందడానికి మనం జాబితాగా మార్చాలి.
లావాదేవీలను సమర్పించడం
ఇప్పుడు మనం నిజంగా లావాదేవీలను సమర్పించాలి. అయినప్పటికీ, వ్యవస్థ నిరూపించబడకముందే నేను ఈ సమయంలో నిజమైన డబ్బు ఖర్చు చేయాలనుకోవడం లేదు. బదులుగా, మేము మెయిన్నెట్ యొక్క స్థానిక ఫోర్క్ను సృష్టిస్తాము మరియు ఆ నెట్వర్క్లో "ట్రేడ్" చేస్తాము.
స్థానిక ఫోర్క్ను సృష్టించడానికి మరియు ట్రేడింగ్ను ప్రారంభించడానికి ఇక్కడ దశలు ఉన్నాయి.
-
Foundry (opens in a new tab) ఇన్స్టాల్ చేయండి
-
anvil(opens in a new tab)ని ప్రారంభించండి1anvil --fork-url https://eth.drpc.org --block-time 12anvilఫౌండ్రీ కోసం డిఫాల్ట్ URL, http://localhost:8545లో (opens in a new tab) వింటోంది, కాబట్టి మనం బ్లాక్ చైనును మార్చడానికి ఉపయోగించేcastకమాండ్ (opens in a new tab) కోసం URLను పేర్కొనవలసిన అవసరం లేదు. -
anvilలో నడుస్తున్నప్పుడు, ETH ఉన్న పది టెస్ట్ ఖాతాలు ఉన్నాయి—మొదటి దాని కోసం ఎన్విరాన్మెంట్ వేరియబుల్స్ సెట్ చేయండి1PRIVATE_KEY=0xac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff802ADDRESS=`cast wallet address $PRIVATE_KEY` -
ఇవి మనం ఉపయోగించాల్సిన కాంట్రాక్టులు.
SwapRouter(opens in a new tab) అనేది మనం నిజంగా ట్రేడ్ చేయడానికి ఉపయోగించే Uniswap v3 కాంట్రాక్ట్. మనం నేరుగా పూల్ ద్వారా ట్రేడ్ చేయవచ్చు, కానీ ఇది చాలా సులభం.రెండు దిగువ వేరియబుల్స్ WETH మరియు USDC మధ్య మార్పిడి చేయడానికి అవసరమైన Uniswap v3 మార్గాలు.
1WETH_ADDRESS=0xC02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc22USDC_ADDRESS=0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB483POOL_ADDRESS=0x88e6A0c2dDD26FEEb64F039a2c41296FcB3f56404SWAP_ROUTER=0xE592427A0AEce92De3Edee1F18E0157C058615645WETH_TO_USDC=0xC02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc20001F4A0b86991c6218b36c1d19D4a2e9Eb0cE3606eB486USDC_TO_WETH=0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB480001F4C02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2 -
ప్రతి టెస్ట్ ఖాతాకు 10,000 ETH ఉన్నాయి. ట్రేడింగ్ కోసం 1000 WETH పొందడానికి 1000 ETH చుట్టడానికి WETH కాంట్రాక్ట్ను ఉపయోగించండి.
1cast send $WETH_ADDRESS "deposit()" --value 1000ether --private-key $PRIVATE_KEY -
USDC కోసం 500 WETH ట్రేడ్ చేయడానికి
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మన టోకెన్లలో కొన్నింటిని ఖర్చు చేయడానికి అనుమతించే అనుమతిని సృష్టిస్తుంది. కాంట్రాక్టులు ఈవెంట్లను పర్యవేక్షించలేవు, కాబట్టి మనం నేరుగా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.py2ప్రస్తుత ధర: 1843.1632026-02-06T23:07లో, ఊహించిన ధర: 1724.41 USD4ట్రేడ్కు ముందు ఖాతా బ్యాలెన్స్లు:5USDC బ్యాలెన్స్: 927301.5782726WETH బ్యాలెన్స్: 5007అమ్ము, ధర 118.75 USD తగ్గుతుందని నేను ఆశిస్తున్నాను8అనుమతి లావాదేవీ పంపబడింది: 74e367ddbb407c1aaf567d87aa5863049991b1d2aa092b6b85195d925e2bd41f9అనుమతి లావాదేవీ మైన్ చేయబడింది.10అమ్మకం లావాదేవీ పంపబడింది: fad1bcf938585c9e90364b26ac7a80eea9efd34c37e5db81e58d7655bcae28bf11అమ్మకం లావాదేవీ మైన్ చేయబడింది.12ట్రేడ్ తర్వాత ఖాతా బ్యాలెన్స్లు:13USDC బ్యాలెన్స్: 929143.79711614WETH బ్యాలెన్స్: 499అన్నీ చూపించునిజంగా దానిని ఉపయోగించడానికి, మీకు కొన్ని చిన్న మార్పులు అవసరం.
- లైన్ 14లో,
MAINNET_URLను నిజమైన యాక్సెస్ పాయింట్కు మార్చండి, ఉదాహరణకుhttps://eth.drpc.org - లైన్ 28లో,
PRIVATE_KEYను మీ స్వంత ప్రైవేట్ కీకి మార్చండి - మీరు చాలా ధనవంతులు అయితే మరియు నిరూపించబడని ఏజెంట్ కోసం ప్రతి రోజు 1 ETH కొనుగోలు లేదా అమ్మకం చేయగలిగితే తప్ప, మీరు 29ని మార్చి
WETH_TRADE_AMOUNTను తగ్గించాలనుకోవచ్చు
కోడ్ వివరణ
ఇక్కడ కొత్త కోడ్ ఉంది.
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 ఫంక్షన్ అవసరం. మనం ముందు మరియు తరువాత బ్యాలెన్స్లను కూడా చూపించాలనుకుంటున్నాము, కాబట్టి మనకు balanceOf కూడా అవసరం.
1SWAP_ROUTER_ABI = [2 { "name": "exactInput", ...},3]SwapRouter ABIలో మనకు కేవలం exactInput అవసరం. సంబంధిత ఫంక్షన్, exactOutput ఉంది, దానిని మనం సరిగ్గా ఒక WETH కొనడానికి ఉపయోగించవచ్చు, కానీ సులభతరం చేయడానికి మనం రెండు సందర్భాలలోనూ కేవలం 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 }లావాదేవీ పరామితులు. ఇక్కడ మనకు ఒక ఫంక్షన్ అవసరం ఎందుకంటే నాన్స్ (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) లావాదేవీ మైన్ చేయబడే వరకు వేచి ఉంటుంది. అవసరమైతే అది రసీదును తిరిగి ఇస్తుంది.
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కి విరుద్ధంగా, కొనుగోలు పరామితులు మారవచ్చు. ఇన్పుట్ మొత్తం 1 WETH ఖర్చు, quoteలో అందుబాటులో ఉంటుంది.
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.")101112def 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()45 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("ట్రేడ్కు ముందు ఖాతా బ్యాలెన్స్లు:")2balances()34if (expected_price > current_price):5 print(f"కొను, ధర {expected_price - current_price} USD పెరుగుతుందని నేను ఆశిస్తున్నాను")6 buy(wethusdc_quotes[-1])7else:8 print(f"అమ్ము, ధర {current_price - expected_price} USD తగ్గుతుందని నేను ఆశిస్తున్నాను")9 sell()1011print("ట్రేడ్ తర్వాత ఖాతా బ్యాలెన్స్లు:")12balances()అన్నీ చూపించుఈ ఏజెంట్ ప్రస్తుతం ఒకసారి మాత్రమే పనిచేస్తుంది. అయినప్పటికీ, మీరు దానిని crontab (opens in a new tab) నుండి అమలు చేయడం ద్వారా లేదా 368-400 పంక్తులను లూప్లో చుట్టి, తదుపరి చక్రం కోసం వేచి ఉండటానికి time.sleep (opens in a new tab)ను ఉపయోగించడం ద్వారా నిరంతరం పనిచేసేలా మార్చవచ్చు.
సాధ్యమయ్యే మెరుగుదలలు
ఇది పూర్తి ఉత్పత్తి వెర్షన్ కాదు; ఇది కేవలం ప్రాథమికాలను బోధించడానికి ఒక ఉదాహరణ. మెరుగుదలల కోసం ఇక్కడ కొన్ని ఆలోచనలు ఉన్నాయి.
స్మార్టర్ ట్రేడింగ్
ఏమి చేయాలో నిర్ణయించేటప్పుడు ఏజెంట్ విస్మరించే రెండు ముఖ్యమైన వాస్తవాలు ఉన్నాయి.
- అంచనా వేసిన మార్పు యొక్క పరిమాణం. ధర తగ్గుతుందని ఊహించినప్పుడు, తగ్గుదల పరిమాణంతో సంబంధం లేకుండా ఏజెంట్ స్థిరమైన మొత్తంలో
WETHను అమ్ముతుంది. వాస్తవానికి, చిన్న మార్పులను విస్మరించి, మనం ఎంత ధర తగ్గుతుందని ఊహిస్తున్నామో దాని ఆధారంగా అమ్మడం మంచిది. - ప్రస్తుత పోర్ట్ఫోలియో. మీ పోర్ట్ఫోలియోలో 10% WETHలో ఉంటే మరియు ధర పెరుగుతుందని మీరు భావిస్తే, బహుశా మరింత కొనడం అర్ధవంతం కావచ్చు. కానీ మీ పోర్ట్ఫోలియోలో 90% WETHలో ఉంటే, మీరు తగినంతగా బహిర్గతం కావచ్చు మరియు మరింత కొనాల్సిన అవసరం లేదు. మీరు ధర తగ్గుతుందని ఊహించినప్పుడు దీనికి విరుద్ధంగా ఉంటుంది.
మీరు మీ ట్రేడింగ్ వ్యూహాన్ని రహస్యంగా ఉంచాలనుకుంటే ఏమిటి?
AI విక్రేతలు మీరు వారి LLMలకు పంపే ప్రశ్నలను చూడగలరు, ఇది మీరు మీ ఏజెంట్తో అభివృద్ధి చేసిన మేధావి ట్రేడింగ్ వ్యవస్థను బహిర్గతం చేయగలదు. చాలా మంది ఉపయోగించే ట్రేడింగ్ సిస్టమ్ పనికిరానిది, ఎందుకంటే మీరు కొనాలనుకున్నప్పుడు చాలా మంది కొనడానికి ప్రయత్నిస్తారు (మరియు ధర పెరుగుతుంది) మరియు మీరు అమ్మాలనుకున్నప్పుడు అమ్మడానికి ప్రయత్నిస్తారు (మరియు ధర తగ్గుతుంది).
ఈ సమస్యను నివారించడానికి మీరు స్థానికంగా LLMని అమలు చేయవచ్చు, ఉదాహరణకు, LM-Studio (opens in a new tab)ని ఉపయోగించి.
AI బాట్ నుండి AI ఏజెంట్కు
ఇది AI బాట్, AI ఏజెంట్ కాదు అని మీరు మంచి వాదన చేయవచ్చు. ఇది ముందే నిర్వచించబడిన సమాచారంపై ఆధారపడే సాపేక్షంగా సరళమైన వ్యూహాన్ని అమలు చేస్తుంది. మనం స్వీయ-మెరుగుదలను ప్రారంభించవచ్చు, ఉదాహరణకు, Uniswap v3 పూల్ల జాబితా మరియు వాటి తాజా విలువలను అందించి, ఏ కలయిక ఉత్తమ అంచనా విలువను కలిగి ఉందో అడగడం ద్వారా.
స్లిపేజ్ రక్షణ
ప్రస్తుతం స్లిపేజ్ రక్షణ (opens in a new tab) లేదు. ప్రస్తుత కోట్ $2000 అయితే, మరియు ఊహించిన ధర $2100 అయితే, ఏజెంట్ కొనుగోలు చేస్తుంది. అయినప్పటికీ, ఏజెంట్ కొనుగోలు చేయడానికి ముందు ఖర్చు $2200కి పెరిగితే, ఇకపై కొనడం అర్ధవంతం కాదు.
స్లిపేజ్ రక్షణను అమలు చేయడానికి, agent.py (opens in a new tab) యొక్క 325 మరియు 334 పంక్తులలో amountOutMinimum విలువను పేర్కొనండి.
ముగింపు
ఆశాజనకంగా, ఇప్పుడు మీకు AI ఏజెంట్లతో ప్రారంభించడానికి తగినంత తెలుసు. ఇది విషయంపై సమగ్రమైన అవలోకనం కాదు; దాని కోసం పూర్తి పుస్తకాలు అంకితం చేయబడ్డాయి, కానీ ఇది మిమ్మల్ని ప్రారంభించడానికి సరిపోతుంది. శుభం కలుగు గాక!
నా మరిన్ని పనుల కోసం ఇక్కడ చూడండి (opens in a new tab).
పేజీ చివరి అప్డేట్: 10 ఫిబ్రవరి, 2026