Crea il tuo agente IA di trading su Ethereum
In questo tutorial imparerai come costruire un semplice agente IA di trading. Questo agente funziona seguendo questi passaggi:
- Leggere i prezzi attuali e passati di un token, oltre ad altre informazioni potenzialmente rilevanti
- Costruire una query con queste informazioni, insieme a informazioni di base per spiegare come potrebbero essere rilevanti
- Inviare la query e ricevere in risposta un prezzo previsto
- Fare trading in base alla raccomandazione
- Attendere e ripetere
Questo agente dimostra come leggere le informazioni, tradurle in una query che produce una risposta utilizzabile e utilizzare tale risposta. Tutti questi sono passaggi richiesti per un agente IA. Questo agente è implementato in Python perché è il linguaggio più comune utilizzato nell'IA.
Perché farlo?
Gli agenti di trading automatizzati consentono agli sviluppatori di selezionare ed eseguire una strategia di trading. Gli agenti IA consentono strategie di trading più complesse e dinamiche, utilizzando potenzialmente informazioni e algoritmi che lo sviluppatore non ha nemmeno preso in considerazione.
Gli strumenti
Questo tutorial utilizza Python (opens in a new tab), la libreria Web3 (opens in a new tab) e Uniswap v3 (opens in a new tab) per le quotazioni e il trading.
Perché Python?
Il linguaggio più utilizzato per l'IA è Python (opens in a new tab), quindi lo useremo qui. Non preoccuparti se non conosci Python. Il linguaggio è molto chiaro e spiegherò esattamente cosa fa.
La libreria Web3 (opens in a new tab) è l'API di Ethereum per Python più comune. È piuttosto facile da usare.
Fare trading sulla blockchain
Ci sono molti exchange distribuiti (DEX) che ti permettono di scambiare token su Ethereum. Tuttavia, tendono ad avere tassi di cambio simili a causa dell'arbitraggio.
Uniswap (opens in a new tab) è un DEX ampiamente utilizzato che possiamo usare sia per le quotazioni (per vedere i valori relativi dei token) sia per le operazioni di trading.
OpenAI
Per un modello linguistico di grandi dimensioni (LLM), ho scelto di iniziare con OpenAI (opens in a new tab). Per eseguire l'applicazione in questo tutorial dovrai pagare per l'accesso all'API. Il pagamento minimo di 5 $ è più che sufficiente.
Sviluppo, passo dopo passo
Per semplificare lo sviluppo, procederemo per fasi. Ogni passaggio è un branch su GitHub.
Per iniziare
Ci sono dei passaggi per iniziare su UNIX o Linux (incluso WSL (opens in a new tab))
-
Se non lo hai già, scarica e installa Python (opens in a new tab).
-
Clona il repository GitHub.
git clone https://github.com/qbzzt/260215-ai-agent.git -b 01-getting-started cd 260215-ai-agent -
Installa
uv(opens in a new tab). Il comando sul tuo sistema potrebbe essere diverso.pipx install uv -
Scarica le librerie.
uv sync -
Attiva l'ambiente virtuale.
source .venv/bin/activate -
Per verificare che Python e Web3 funzionino correttamente, esegui
python3e forniscigli questo programma. Puoi inserirlo al prompt>>>; non è necessario creare un file.from web3 import Web3 MAINNET_URL = "https://eth.drpc.org" w3 = Web3(Web3.HTTPProvider(MAINNET_URL)) w3.eth.block_number quit()
Leggere dalla blockchain
Il passaggio successivo è leggere dalla blockchain. Per farlo, devi passare al branch 02-read-quote e poi usare uv per eseguire il programma.
git checkout 02-read-quote
uv run agent.py
Dovresti ricevere un elenco di oggetti Quote, ciascuno con un timestamp, un prezzo e l'asset (attualmente sempre WETH/USDC).
Ecco una spiegazione riga per riga.
from web3 import Web3
from web3.contract import Contract
from decimal import Decimal, ROUND_HALF_UP
from dataclasses import dataclass
from datetime import datetime, timezone
from pprint import pprint
import time
import functools
import sys
Importa le librerie di cui abbiamo bisogno. Sono spiegate di seguito quando vengono utilizzate.
print = functools.partial(print, flush=True)
Sostituisce print di Python con una versione che svuota sempre l'output immediatamente. Questo è utile in uno script a esecuzione prolungata perché non vogliamo aspettare per gli aggiornamenti di stato o l'output di debug.
MAINNET_URL = "https://eth.drpc.org"
Un URL per accedere alla Mainnet. Puoi ottenerne uno da un Nodo come servizio o usarne uno tra quelli pubblicizzati su Chainlist (opens in a new tab).
BLOCK_TIME_SECONDS = 12
MINUTE_BLOCKS = int(60 / BLOCK_TIME_SECONDS)
HOUR_BLOCKS = MINUTE_BLOCKS * 60
DAY_BLOCKS = HOUR_BLOCKS * 24
Un blocco della Mainnet di Ethereum viene tipicamente generato ogni dodici secondi, quindi questo è il numero di blocchi che ci aspetteremmo in un determinato periodo di tempo. Nota che questa non è una cifra esatta. Quando il proponente del blocco è inattivo, quel blocco viene saltato e il tempo per il blocco successivo è di 24 secondi. Se volessimo ottenere il blocco esatto per un timestamp, useremmo la ricerca binaria (opens in a new tab). Tuttavia, questo è abbastanza preciso per i nostri scopi. Prevedere il futuro non è una scienza esatta.
CYCLE_BLOCKS = DAY_BLOCKS
La dimensione del ciclo. Esaminiamo le quotazioni una volta per ciclo e cerchiamo di stimare il valore alla fine del ciclo successivo.
# L'indirizzo della pool che stiamo leggendo
WETHUSDC_ADDRESS = Web3.to_checksum_address("0x88e6A0c2dDD26FEEb64F039a2c41296FcB3f5640")
I valori delle quotazioni sono presi dal pool USDC/WETH di Uniswap v3 all'indirizzo 0x88e6A0c2dDD26FEEb64F039a2c41296FcB3f5640 (opens in a new tab). Questo indirizzo è già in formato checksum, ma è meglio usare Web3.to_checksum_address (opens in a new tab) per rendere il codice riutilizzabile.
POOL_ABI = [
{ "name": "slot0", ... },
{ "name": "token0", ... },
{ "name": "token1", ... },
]
ERC20_ABI = [
{ "name": "symbol", ... },
{ "name": "decimals", ... }
]
Queste sono le ABI (opens in a new tab) per i due contratti che dobbiamo contattare. Per mantenere il codice conciso, includiamo solo le funzioni che dobbiamo chiamare.
w3 = Web3(Web3.HTTPProvider(MAINNET_URL))
Inizializza la libreria Web3 (opens in a new tab) e connettiti a un nodo Ethereum.
@dataclass(frozen=True)
class ERC20Token:
address: str
symbol: str
decimals: int
contract: Contract
Questo è un modo per creare una data class in Python. Il tipo di dato Contract (opens in a new tab) viene utilizzato per connettersi al contratto. Nota il (frozen=True). In Python i booleani (opens in a new tab) sono definiti come True o False, con l'iniziale maiuscola. Questa data class è frozen, il che significa che i campi non possono essere modificati.
Nota l'indentazione. A differenza dei linguaggi derivati dal C (opens in a new tab), Python usa l'indentazione per denotare i blocchi. L'interprete Python sa che la definizione seguente non fa parte di questa data class perché non inizia con la stessa indentazione dei campi della data class.
@dataclass(frozen=True)
class PoolInfo:
address: str
token0: ERC20Token
token1: ERC20Token
contract: Contract
asset: str
decimal_factor: Decimal = 1
Il tipo Decimal (opens in a new tab) viene utilizzato per gestire accuratamente le frazioni decimali.
def get_price(self, block: int) -> Decimal:
Questo è il modo per definire una funzione in Python. La definizione è indentata per mostrare che fa ancora parte di PoolInfo.
In una funzione che fa parte di una data class, il primo parametro è sempre self, l'istanza della data class che ha effettuato la chiamata. Qui c'è un altro parametro, il numero del blocco.
assert block <= w3.eth.block_number, "Block is in the future"
Se potessimo leggere il futuro, non avremmo bisogno dell'IA per il trading.
sqrt_price_x96 = Decimal(self.contract.functions.slot0().call(block_identifier=block)[0])
La sintassi per chiamare una funzione sull'EVM da Web3 è questa: <contract object>.functions.<function name>().call(<parameters>). I parametri possono essere i parametri della funzione EVM (se presenti; qui non ce ne sono) o parametri nominati (opens in a new tab) per modificare il comportamento della blockchain. Qui ne usiamo uno, block_identifier, per specificare il numero del blocco in cui desideriamo eseguire l'operazione.
Il risultato è questa struct, in formato array (opens in a new tab). Il primo valore è una funzione del tasso di cambio tra i due token.
raw_price = (sqrt_price_x96 / Decimal(2**96)) ** 2
Per ridurre i calcoli onchain, Uniswap v3 non memorizza il fattore di cambio effettivo, ma piuttosto la sua radice quadrata. Poiché l'EVM non supporta la matematica in virgola mobile o le frazioni, invece del valore effettivo, la risposta è
# (token1 per token0)
return 1/(raw_price * self.decimal_factor)
Il prezzo grezzo che otteniamo è il numero di token0 che riceviamo per ogni token1. Nel nostro pool token0 è USDC (stablecoin con lo stesso valore di un dollaro USA) e token1 è WETH (opens in a new tab). Il valore che vogliamo veramente è il numero di dollari per WETH, non l'inverso.
Il fattore decimale è il rapporto tra i fattori decimali (opens in a new tab) per i due token.
@dataclass(frozen=True)
class Quote:
timestamp: str
price: Decimal
asset: str
Questa data class rappresenta una quotazione: il prezzo di un asset specifico in un dato momento. A questo punto, il campo asset è irrilevante perché usiamo un singolo pool e quindi abbiamo un singolo asset. Tuttavia, aggiungeremo altri asset in seguito.
def read_token(address: str) -> ERC20Token:
token = w3.eth.contract(address=address, abi=ERC20_ABI)
symbol = token.functions.symbol().call()
decimals = token.functions.decimals().call()
return ERC20Token(
address=address,
symbol=symbol,
decimals=decimals,
contract=token
)
Questa funzione accetta un indirizzo e restituisce informazioni sul contratto del token a quell'indirizzo. Per creare un nuovo Contract Web3 (opens in a new tab), forniamo l'indirizzo e l'ABI a w3.eth.contract.
def read_pool(address: str) -> PoolInfo:
pool_contract = w3.eth.contract(address=address, abi=POOL_ABI)
token0Address = pool_contract.functions.token0().call()
token1Address = pool_contract.functions.token1().call()
token0 = read_token(token0Address)
token1 = read_token(token1Address)
return PoolInfo(
address=address,
asset=f"{token1.symbol}/{token0.symbol}",
token0=token0,
token1=token1,
contract=pool_contract,
decimal_factor=Decimal(10) ** Decimal(token0.decimals - token1.decimals)
)
Questa funzione restituisce tutto ciò di cui abbiamo bisogno su un pool specifico (opens in a new tab). La sintassi f"<string>" è una stringa formattata (opens in a new tab).
def get_quote(pool: PoolInfo, block_number: int = None) -> Quote:
Ottieni un oggetto Quote. Il valore predefinito per block_number è None (nessun valore).
if block_number is None:
block_number = w3.eth.block_number
Se non è stato specificato un numero di blocco, usa w3.eth.block_number, che è l'ultimo numero di blocco. Questa è la sintassi per un'istruzione if (opens in a new tab).
Potrebbe sembrare che sarebbe stato meglio impostare semplicemente il valore predefinito su w3.eth.block_number, ma non funziona bene perché sarebbe il numero del blocco al momento in cui la funzione viene definita. In un agente a esecuzione prolungata, questo sarebbe un problema.
block = w3.eth.get_block(block_number)
price = pool.get_price(block_number)
return Quote(
timestamp=datetime.fromtimestamp(block.timestamp, timezone.utc).isoformat(),
price=price.quantize(Decimal("0.01")),
asset=pool.asset
)
Usa la libreria datetime (opens in a new tab) per formattarlo in un formato leggibile per gli esseri umani e per i modelli linguistici di grandi dimensioni (LLM). Usa Decimal.quantize (opens in a new tab) per arrotondare il valore a due cifre decimali.
def get_quotes(pool: PoolInfo, start_block: int, end_block: int, step: int) -> list[Quote]:
In Python definisci una lista (opens in a new tab) che può contenere solo un tipo specifico usando list[<type>].
quotes = []
for block in range(start_block, end_block + 1, step):
In Python un ciclo for (opens in a new tab) tipicamente itera su una lista. L'elenco dei numeri di blocco in cui trovare le quotazioni proviene da range (opens in a new tab).
quote = get_quote(pool, block)
quotes.append(quote)
return quotes
Per ogni numero di blocco, ottieni un oggetto Quote e aggiungilo alla lista quotes. Quindi restituisci quella lista.
pool = read_pool(WETHUSDC_ADDRESS)
quotes = get_quotes(
pool,
w3.eth.block_number - 12*CYCLE_BLOCKS,
w3.eth.block_number,
CYCLE_BLOCKS
)
pprint(quotes)
Questo è il codice principale dello script. Leggi le informazioni del pool, ottieni dodici quotazioni e stampale (pprint) (opens in a new tab).
Creare un prompt
Successivamente, dobbiamo convertire questo elenco di quotazioni in un prompt per un LLM e ottenere un valore futuro atteso.
git checkout 03-create-prompt
uv run agent.py
L'output ora sarà un prompt per un LLM, simile a:
Date queste quotazioni:
Asset: WETH/USDC
2026-01-20T16:34 3016.21
.
.
.
2026-02-01T17:49 2299.10
Asset: WBTC/WETH
2026-01-20T16:34 29.84
.
.
.
2026-02-01T17:50 33.46
Quale ti aspetteresti che sia il valore per WETH/USDC al momento 2026-02-02T17:56?
Fornisci la tua risposta come un singolo numero arrotondato a due cifre decimali,
senza alcun altro testo.
Nota che qui ci sono quotazioni per due asset, WETH/USDC e WBTC/WETH. L'aggiunta di quotazioni da un altro asset potrebbe migliorare l'accuratezza della previsione.
Come si presenta un prompt
Questo prompt contiene tre sezioni, che sono piuttosto comuni nei prompt per LLM.
-
Informazioni. Gli LLM hanno molte informazioni derivanti dal loro addestramento, ma di solito non hanno le più recenti. Questo è il motivo per cui dobbiamo recuperare le ultime quotazioni qui. L'aggiunta di informazioni a un prompt è chiamata retrieval augmented generation (RAG) (opens in a new tab).
-
La domanda vera e propria. Questo è ciò che vogliamo sapere.
-
Istruzioni per la formattazione dell'output. Normalmente, un LLM ci darà una stima con una spiegazione di come ci è arrivato. Questo è meglio per gli esseri umani, ma un programma per computer ha solo bisogno del risultato finale.
Spiegazione del codice
Ecco il nuovo codice.
from datetime import datetime, timezone, timedelta
Dobbiamo fornire all'LLM il momento per il quale vogliamo una stima. Per ottenere un tempo "n minuti/ore/giorni" nel futuro, usiamo la classe timedelta (opens in a new tab).
# Gli indirizzi delle pool che stiamo leggendo
WETHUSDC_ADDRESS = Web3.to_checksum_address("0x88e6A0c2dDD26FEEb64F039a2c41296FcB3f5640")
WETHWBTC_ADDRESS = Web3.to_checksum_address("0xCBCdF9626bC03E24f779434178A73a0B4bad62eD")
Abbiamo due pool che dobbiamo leggere.
@dataclass(frozen=True)
class PoolInfo:
.
.
.
reverse: bool = False
def get_price(self, block: int) -> Decimal:
assert block <= w3.eth.block_number, "Block is in the future"
sqrt_price_x96 = Decimal(self.contract.functions.slot0().call(block_identifier=block)[0])
raw_price = (sqrt_price_x96 / Decimal(2**96)) ** 2 # (token1 per token0)
if self.reverse:
return 1/(raw_price * self.decimal_factor)
else:
return raw_price * self.decimal_factor
Nel pool WETH/USDC, vogliamo sapere quanti token0 (USDC) ci servono per comprare un token1 (WETH). Nel pool WETH/WBTC, vogliamo sapere quanti token1 (WETH) ci servono per comprare un token0 (WBTC, che è Bitcoin incapsulato). Dobbiamo tenere traccia se il rapporto del pool deve essere invertito.
def read_pool(address: str, reverse: bool = False) -> PoolInfo:
.
.
.
return PoolInfo(
.
.
.
asset= f"{token1.symbol}/{token0.symbol}" if reverse else f"{token0.symbol}/{token1.symbol}",
reverse=reverse
)
Per sapere se un pool deve essere invertito, dobbiamo ottenerlo come input per read_pool. Inoltre, il simbolo dell'asset deve essere impostato correttamente.
La sintassi <a> if <b> else <c> è l'equivalente in Python dell'operatore condizionale ternario (opens in a new tab), che in un linguaggio derivato dal C sarebbe <b> ? <a> : <c>.
def format_quotes(quotes: list[Quote]) -> str:
result = f"Asset: {quotes[0].asset}\n"
for quote in quotes:
result += f"\t{quote.timestamp[0:16]} {quote.price.quantize(Decimal('0.01'), rounding=ROUND_HALF_UP)}\n"
return result
Questa funzione costruisce una stringa che formatta un elenco di oggetti Quote, supponendo che si applichino tutti allo stesso asset.
def make_prompt(quotes: list[list[Quote]], expected_time: str, asset: str) -> str:
return f"""
In Python le stringhe letterali multilinea (opens in a new tab) sono scritte come """ .... """.
Given these quotes:
{
functools.reduce(lambda acc, q: acc + '\n' + q,
map(lambda q: format_quotes(q), quotes))
}
Qui, usiamo il pattern MapReduce (opens in a new tab) per generare una stringa per ogni elenco di quotazioni con format_quotes, per poi ridurle in una singola stringa da usare nel prompt.
What would you expect the value for {asset} to be at time {expected_time}?
Provide your answer as a single number rounded to two decimal places,
without any other text.
"""
Il resto del prompt è come previsto.
wethusdc_pool = read_pool(WETHUSDC_ADDRESS, True)
wethusdc_quotes = get_quotes(
wethusdc_pool,
w3.eth.block_number - 12*CYCLE_BLOCKS,
w3.eth.block_number,
CYCLE_BLOCKS,
)
wethwbtc_pool = read_pool(WETHWBTC_ADDRESS)
wethwbtc_quotes = get_quotes(
wethwbtc_pool,
w3.eth.block_number - 12*CYCLE_BLOCKS,
w3.eth.block_number,
CYCLE_BLOCKS
)
Esamina i due pool e ottieni le quotazioni da entrambi.
future_time = (datetime.now(timezone.utc) + timedelta(days=1)).isoformat()[0:16]
print(make_prompt(wethusdc_quotes + wethwbtc_quotes, future_time, wethusdc_pool.asset))
Determina il momento futuro per il quale vogliamo la stima e crea il prompt.
Interfacciarsi con un LLM
Successivamente, interroghiamo un vero LLM e riceviamo un valore futuro atteso. Ho scritto questo programma usando OpenAI, quindi se vuoi usare un provider diverso, dovrai adattarlo.
-
Ottieni un account OpenAI (opens in a new tab)
-
Finanzia l'account (opens in a new tab): l'importo minimo al momento della stesura è di 5 $
-
Nella riga di comando, esporta la chiave API in modo che il tuo programma possa usarla
export OPENAI_API_KEY=sk-<the rest of the key goes here> -
Fai il checkout ed esegui l'agente
git checkout 04-interface-llm uv run agent.py
Ecco il nuovo codice.
from openai import OpenAI
open_ai = OpenAI() # Il client legge la variabile d'ambiente OPENAI_API_KEY
Importa e istanzia l'API di OpenAI.
response = open_ai.chat.completions.create(
model="gpt-4-turbo",
messages=[
{"role": "user", "content": prompt}
],
temperature=0.0,
max_tokens=16,
)
Chiama l'API di OpenAI (open_ai.chat.completions.create) per creare la risposta.
expected_price = Decimal(response.choices[0].message.content.strip())
current_price = wethusdc_quotes[-1].price
print ("Current price:", wethusdc_quotes[-1].price)
print(f"In {future_time}, expected price: {expected_price} USD")
if (expected_price > current_price):
print(f"Buy, I expect the price to go up by {expected_price - current_price} USD")
else:
print(f"Sell, I expect the price to go down by {current_price - expected_price} USD")
Stampa il prezzo e fornisci una raccomandazione di acquisto o vendita.
Testare le previsioni
Ora che possiamo generare previsioni, possiamo anche usare i dati storici per valutare se produciamo previsioni utili.
uv run test-predictor.py
Il risultato atteso è simile a:
Previsione per 2026-01-05T19:50: previsto 3138.93 USD, reale 3218.92 USD, errore 79.99 USD
Previsione per 2026-01-06T19:56: previsto 3243.39 USD, reale 3221.08 USD, errore 22.31 USD
Previsione per 2026-01-07T20:02: previsto 3223.24 USD, reale 3146.89 USD, errore 76.35 USD
Previsione per 2026-01-08T20:11: previsto 3150.47 USD, reale 3092.04 USD, errore 58.43 USD
.
.
.
Previsione per 2026-01-31T22:33: previsto 2637.73 USD, reale 2417.77 USD, errore 219.96 USD
Previsione per 2026-02-01T22:41: previsto 2381.70 USD, reale 2318.84 USD, errore 62.86 USD
Previsione per 2026-02-02T22:49: previsto 2234.91 USD, reale 2349.28 USD, errore 114.37 USD
Errore medio di previsione su 29 previsioni: 83.87103448275862068965517241 USD
Variazione media per raccomandazione: 4.787931034482758620689655172 USD
Varianza standard delle variazioni: 104.42 USD
Giorni redditizi: 51.72%
Giorni in perdita: 48.28%
La maggior parte del tester è identica all'agente, ma ecco le parti nuove o modificate.
CYCLES_FOR_TEST = 40 # Per il backtest, su quanti cicli effettuiamo il test
# Ottieni molte quotazioni
wethusdc_pool = read_pool(WETHUSDC_ADDRESS, True)
wethusdc_quotes = get_quotes(
wethusdc_pool,
w3.eth.block_number - CYCLE_BLOCKS*CYCLES_FOR_TEST,
w3.eth.block_number,
CYCLE_BLOCKS,
)
wethwbtc_pool = read_pool(WETHWBTC_ADDRESS)
wethwbtc_quotes = get_quotes(
wethwbtc_pool,
w3.eth.block_number - CYCLE_BLOCKS*CYCLES_FOR_TEST,
w3.eth.block_number,
CYCLE_BLOCKS
)
Guardiamo indietro di CYCLES_FOR_TEST (specificato come 40 qui) giorni.
# Crea previsioni e confrontale con lo storico reale
total_error = Decimal(0)
changes = []
Ci sono due tipi di errori a cui siamo interessati. Il primo, total_error, è semplicemente la somma degli errori commessi dal predittore.
Per capire il secondo, changes, dobbiamo ricordare lo scopo dell'agente. Non è prevedere il rapporto WETH/USDC (prezzo di ETH). È emettere raccomandazioni di vendita e acquisto. Se il prezzo è attualmente di 2000 $ e prevede 2010 $ per domani, non ci dispiace se il risultato effettivo è 2020 $ e guadagniamo soldi extra. Ma ci dispiace se ha previsto 2010 $ e ha comprato ETH in base a quella raccomandazione, e il prezzo scende a 1990 $.
for index in range(0,len(wethusdc_quotes)-CYCLES_BACK):
Possiamo esaminare solo i casi in cui è disponibile la cronologia completa (i valori utilizzati per la previsione e il valore reale con cui confrontarla). Ciò significa che il caso più recente deve essere quello iniziato CYCLES_BACK fa.
wethusdc_slice = wethusdc_quotes[index:index+CYCLES_BACK]
wethwbtc_slice = wethwbtc_quotes[index:index+CYCLES_BACK]
Usa gli slice (opens in a new tab) per ottenere lo stesso numero di campioni del numero utilizzato dall'agente. Il codice tra qui e il segmento successivo è lo stesso codice per ottenere una previsione che abbiamo nell'agente.
predicted_price = Decimal(response.choices[0].message.content.strip())
real_price = wethusdc_quotes[index+CYCLES_BACK].price
prediction_time_price = wethusdc_quotes[index+CYCLES_BACK-1].price
Ottieni il prezzo previsto, il prezzo reale e il prezzo al momento della previsione. Abbiamo bisogno del prezzo al momento della previsione per determinare se la raccomandazione era di comprare o vendere.
error = abs(predicted_price - real_price)
total_error += error
print (f"Prediction for {prediction_time}: predicted {predicted_price} USD, real {real_price} USD, error {error} USD")
Calcola l'errore e aggiungilo al totale.
recomended_action = 'buy' if predicted_price > prediction_time_price else 'sell'
price_increase = real_price - prediction_time_price
changes.append(price_increase if recomended_action == 'buy' else -price_increase)
Per changes, vogliamo l'impatto monetario dell'acquisto o della vendita di un ETH. Quindi, per prima cosa, dobbiamo determinare la raccomandazione, poi valutare come è cambiato il prezzo effettivo e se la raccomandazione ha fatto guadagnare (variazione positiva) o perdere denaro (variazione negativa).
print (f"Mean prediction error over {len(wethusdc_quotes)-CYCLES_BACK} predictions: {total_error / Decimal(len(wethusdc_quotes)-CYCLES_BACK)} USD")
length_changes = Decimal(len(changes))
mean_change = sum(changes, Decimal(0)) / length_changes
print (f"Mean change per recommendation: {mean_change} USD")
var = sum((x - mean_change) ** 2 for x in changes) / length_changes
print (f"Standard variance of changes: {var.sqrt().quantize(Decimal("0.01"))} USD")
Riporta i risultati.
print (f"Profitable days: {len(list(filter(lambda x: x > 0, changes)))/length_changes:.2%}")
print (f"Losing days: {len(list(filter(lambda x: x < 0, changes)))/length_changes:.2%}")
Usa filter (opens in a new tab) per contare il numero di giorni redditizi e il numero di giorni in perdita. Il risultato è un oggetto filtro, che dobbiamo convertire in una lista per ottenerne la lunghezza.
Inviare transazioni
Ora dobbiamo effettivamente inviare le transazioni. Tuttavia, non voglio spendere soldi veri a questo punto, prima che il sistema sia collaudato. Invece, creeremo un fork locale della Mainnet e faremo "trading" su quella rete.
Ecco i passaggi per creare un fork locale e abilitare il trading.
-
Installa Foundry (opens in a new tab)
-
Avvia
anvil(opens in a new tab)anvil --fork-url https://eth.drpc.org --block-time 12anvilè in ascolto sull'URL predefinito per Foundry, http://localhost:8545 (opens in a new tab), quindi non abbiamo bisogno di specificare l'URL per il comandocast(opens in a new tab) che usiamo per manipolare la blockchain. -
Quando si esegue in
anvil, ci sono dieci account di test che hanno ETH: imposta le variabili d'ambiente per il primoPRIVATE_KEY=0xac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff80 ADDRESS=`cast wallet address $PRIVATE_KEY` -
Questi sono i contratti che dobbiamo usare.
SwapRouter(opens in a new tab) è il contratto Uniswap v3 che usiamo per fare effettivamente trading. Potremmo fare trading direttamente tramite il pool, ma questo è molto più semplice.Le due variabili in basso sono i percorsi di Uniswap v3 necessari per effettuare lo swap tra WETH e USDC.
WETH_ADDRESS=0xC02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2 USDC_ADDRESS=0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48 POOL_ADDRESS=0x88e6A0c2dDD26FEEb64F039a2c41296FcB3f5640 SWAP_ROUTER=0xE592427A0AEce92De3Edee1F18E0157C05861564 WETH_TO_USDC=0xC02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc20001F4A0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48 USDC_TO_WETH=0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB480001F4C02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2 -
Ciascuno degli account di test ha 10.000 ETH. Usa il contratto WETH per incapsulare 1000 ETH per ottenere 1000 WETH per il trading.
cast send $WETH_ADDRESS "deposit()" --value 1000ether --private-key $PRIVATE_KEY -
Usa
SwapRouterper scambiare 500 WETH con USDC.cast send $WETH_ADDRESS "approve(address,uint256)" $SWAP_ROUTER 500ether --private-key $PRIVATE_KEY MAXINT=`cast max-int uint256` cast send $SWAP_ROUTER \ "exactInput((bytes,address,uint256,uint256,uint256))" \ "($WETH_TO_USDC,$ADDRESS,$MAXINT,500ether,1000000)" \ --private-key $PRIVATE_KEYLa chiamata
approvecrea un'autorizzazione di spesa che consente aSwapRouterdi spendere alcuni dei nostri token. I contratti non possono monitorare gli eventi, quindi se trasferissimo i token direttamente al contrattoSwapRouter, non saprebbe di essere stato pagato. Invece, permettiamo al contrattoSwapRouterdi spendere un certo importo, e poiSwapRouterlo fa. Questo viene fatto tramite una funzione chiamata daSwapRouter, in modo che sappia se ha avuto successo. -
Verifica di avere abbastanza di entrambi i token.
cast call $WETH_ADDRESS "balanceOf(address)" $ADDRESS | cast from-wei echo `cast call $USDC_ADDRESS "balanceOf(address)" $ADDRESS | cast to-dec`/10^6 | bc
Ora che abbiamo WETH e USDC, possiamo effettivamente eseguire l'agente.
git checkout 05-trade
uv run agent.py
L'output sarà simile a:
(ai-trading-agent) qbzzt@Ori-Cloudnomics:~/260215-ai-agent$ uv run agent.py
Prezzo attuale: 1843.16
Nel 2026-02-06T23:07, prezzo atteso: 1724.41 USD
Saldi dell'account prima del trade:
Saldo USDC: 927301.578272
Saldo WETH: 500
Vendi, mi aspetto che il prezzo scenda di 118.75 USD
Transazione di approvazione inviata: 74e367ddbb407c1aaf567d87aa5863049991b1d2aa092b6b85195d925e2bd41f
Transazione di approvazione minata.
Transazione di vendita inviata: fad1bcf938585c9e90364b26ac7a80eea9efd34c37e5db81e58d7655bcae28bf
Transazione di vendita minata.
Saldi dell'account dopo il trade:
Saldo USDC: 929143.797116
Saldo WETH: 499
Per usarlo effettivamente, hai bisogno di alcune piccole modifiche.
- Alla riga 14, cambia
MAINNET_URLin un vero punto di accesso, comehttps://eth.drpc.org - Alla riga 28, cambia
PRIVATE_KEYcon la tua chiave privata - A meno che tu non sia molto ricco e possa comprare o vendere 1 ETH ogni giorno per un agente non collaudato, potresti voler cambiare la riga 29 per diminuire
WETH_TRADE_AMOUNT
Spiegazione del codice
Ecco il nuovo codice.
SWAP_ROUTER_ADDRESS=Web3.to_checksum_address("0xE592427A0AEce92De3Edee1F18E0157C05861564")
WETH_TO_USDC=bytes.fromhex("C02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc20001F4A0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48")
USDC_TO_WETH=bytes.fromhex("A0b86991c6218b36c1d19D4a2e9Eb0cE3606eB480001F4C02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2")
PRIVATE_KEY="0xac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff80"
Le stesse variabili che abbiamo usato nel passaggio 4.
WETH_TRADE_AMOUNT=1
L'importo da scambiare.
ERC20_ABI = [
{ "name": "symbol", ... },
{ "name": "decimals", ... },
{ "name": "balanceOf", ...},
{ "name": "approve", ...}
]
Per fare effettivamente trading, abbiamo bisogno della funzione approve. Vogliamo anche mostrare i saldi prima e dopo, quindi abbiamo bisogno anche di balanceOf.
SWAP_ROUTER_ABI = [
{ "name": "exactInput", ...},
]
Nell'ABI di SwapRouter abbiamo solo bisogno di exactInput. C'è una funzione correlata, exactOutput, che potremmo usare per comprare esattamente un WETH, ma per semplicità usiamo solo exactInput in entrambi i casi.
account = w3.eth.account.from_key(PRIVATE_KEY)
swap_router = w3.eth.contract(
address=SWAP_ROUTER_ADDRESS,
abi=SWAP_ROUTER_ABI
)
Le definizioni Web3 per il contratto account (opens in a new tab) e il contratto SwapRouter.
def txn_params() -> dict:
return {
"from": account.address,
"value": 0,
"gas": 300000,
"nonce": w3.eth.get_transaction_count(account.address),
}
I parametri della transazione. Abbiamo bisogno di una funzione qui perché il nonce (opens in a new tab) deve cambiare ogni volta.
def approve_token(contract: Contract, amount: int):
Approva un'autorizzazione di spesa del token per SwapRouter.
txn = contract.functions.approve(SWAP_ROUTER_ADDRESS, amount).build_transaction(txn_params())
signed_txn = w3.eth.account.sign_transaction(txn, private_key=PRIVATE_KEY)
tx_hash = w3.eth.send_raw_transaction(signed_txn.raw_transaction)
Ecco come inviamo una transazione in Web3. Per prima cosa usiamo l'oggetto Contract (opens in a new tab) per costruire la transazione. Poi usiamo web3.eth.account.sign_transaction (opens in a new tab) per firmare la transazione, usando PRIVATE_KEY. Infine, usiamo w3.eth.send_raw_transaction (opens in a new tab) per inviare la transazione.
print(f"Approve transaction sent: {tx_hash.hex()}")
w3.eth.wait_for_transaction_receipt(tx_hash)
print("Approve transaction mined.")
w3.eth.wait_for_transaction_receipt (opens in a new tab) attende finché la transazione non viene minata. Restituisce la ricevuta se necessario.
SELL_PARAMS = {
"path": WETH_TO_USDC,
"recipient": account.address,
"deadline": 2**256 - 1,
"amountIn": WETH_TRADE_AMOUNT * 10 ** wethusdc_pool.token1.decimals,
"amountOutMinimum": 0,
}
Questi sono i parametri quando si vende WETH.
def make_buy_params(quote: Quote) -> dict:
return {
"path": USDC_TO_WETH,
"recipient": account.address,
"deadline": 2**256 - 1,
"amountIn": int(quote.price*WETH_TRADE_AMOUNT) * 10**wethusdc_pool.token0.decimals,
"amountOutMinimum": 0,
}
A differenza di SELL_PARAMS, i parametri di acquisto possono cambiare. L'importo di input è il costo di 1 WETH, come disponibile in quote.
def buy(quote: Quote):
buy_params = make_buy_params(quote)
approve_token(wethusdc_pool.token0.contract, buy_params["amountIn"])
txn = swap_router.functions.exactInput(buy_params).build_transaction(txn_params())
signed_txn = w3.eth.account.sign_transaction(txn, private_key=PRIVATE_KEY)
tx_hash = w3.eth.send_raw_transaction(signed_txn.raw_transaction)
print(f"Buy transaction sent: {tx_hash.hex()}")
w3.eth.wait_for_transaction_receipt(tx_hash)
print("Buy transaction mined.")
def sell():
approve_token(wethusdc_pool.token1.contract,
WETH_TRADE_AMOUNT * 10**wethusdc_pool.token1.decimals)
txn = swap_router.functions.exactInput(SELL_PARAMS).build_transaction(txn_params())
signed_txn = w3.eth.account.sign_transaction(txn, private_key=PRIVATE_KEY)
tx_hash = w3.eth.send_raw_transaction(signed_txn.raw_transaction)
print(f"Sell transaction sent: {tx_hash.hex()}")
w3.eth.wait_for_transaction_receipt(tx_hash)
print("Sell transaction mined.")
Le funzioni buy() e sell() sono quasi identiche. Per prima cosa approviamo un'autorizzazione di spesa sufficiente per SwapRouter, e poi lo chiamiamo con il percorso e l'importo corretti.
def balances():
token0_balance = wethusdc_pool.token0.contract.functions.balanceOf(account.address).call()
token1_balance = wethusdc_pool.token1.contract.functions.balanceOf(account.address).call()
print(f"{wethusdc_pool.token0.symbol} Balance: {Decimal(token0_balance) / Decimal(10 ** wethusdc_pool.token0.decimals)}")
print(f"{wethusdc_pool.token1.symbol} Balance: {Decimal(token1_balance) / Decimal(10 ** wethusdc_pool.token1.decimals)}")
Riporta i saldi dell'utente in entrambe le valute.
print("Account balances before trade:")
balances()
if (expected_price > current_price):
print(f"Buy, I expect the price to go up by {expected_price - current_price} USD")
buy(wethusdc_quotes[-1])
else:
print(f"Sell, I expect the price to go down by {current_price - expected_price} USD")
sell()
print("Account balances after trade:")
balances()
Questo agente attualmente funziona solo una volta. Tuttavia, puoi modificarlo per farlo funzionare continuamente eseguendolo da crontab (opens in a new tab) o racchiudendo le righe 368-400 in un ciclo e usando time.sleep (opens in a new tab) per attendere fino al momento del ciclo successivo.
Possibili miglioramenti
Questa non è una versione di produzione completa; è semplicemente un esempio per insegnare le basi. Ecco alcune idee per miglioramenti.
Trading più intelligente
Ci sono due fatti importanti che l'agente ignora quando decide cosa fare.
- L'entità del cambiamento previsto. L'agente vende un importo fisso di
WETHse si prevede che il prezzo diminuirà, indipendentemente dall'entità del calo. Sarebbe probabilmente meglio ignorare i cambiamenti minori e vendere in base a quanto ci aspettiamo che il prezzo scenda. - Il portafoglio attuale. Se il 10% del tuo portafoglio è in WETH e pensi che il prezzo salirà, probabilmente ha senso comprarne di più. Ma se il 90% del tuo portafoglio è in WETH, potresti essere sufficientemente esposto e non c'è bisogno di comprarne di più. È vero il contrario se ti aspetti che il prezzo scenda.
E se volessi mantenere segreta la tua strategia di trading?
I fornitori di IA possono vedere le query che invii ai loro LLM, il che potrebbe esporre il geniale sistema di trading che hai sviluppato con il tuo agente. Un sistema di trading che troppe persone usano è inutile perché troppe persone cercano di comprare quando tu vuoi comprare (e il prezzo sale) e cercano di vendere quando tu vuoi vendere (e il prezzo scende).
Puoi eseguire un LLM localmente, ad esempio usando LM-Studio (opens in a new tab), per evitare questo problema.
Da bot IA ad agente IA
Si può sostenere a ragione che questo sia un bot IA, non un agente IA. Implementa una strategia relativamente semplice che si basa su informazioni predefinite. Possiamo abilitare l'auto-miglioramento, ad esempio, fornendo un elenco di pool di Uniswap v3 e i loro ultimi valori e chiedendo quale combinazione abbia il miglior valore predittivo.
Protezione dallo slittamento
Attualmente non c'è alcuna protezione dallo slittamento (opens in a new tab). Se la quotazione attuale è di 2000 $ e il prezzo atteso è di 2100 $, l'agente comprerà. Tuttavia, se prima che l'agente compri il costo sale a 2200 $, non ha più senso comprare.
Per implementare la protezione dallo slittamento, specifica un valore amountOutMinimum nelle righe 325 e 334 di agent.py (opens in a new tab).
Conclusione
Speriamo che ora tu ne sappia abbastanza per iniziare con gli agenti IA. Questa non è una panoramica completa dell'argomento; ci sono interi libri dedicati a questo, ma è sufficiente per iniziare. Buona fortuna!