أنشئ وكيل تداول ذكاء اصطناعي خاص بك على إيثيريوم
في هذا البرنامج التعليمي، ستتعلم كيفية بناء وكيل تداول ذكاء اصطناعي بسيط. يعمل هذا الوكيل باستخدام الخطوات التالية:
- قراءة الأسعار الحالية والسابقة لرمز مميز، بالإضافة إلى معلومات أخرى قد تكون ذات صلة
- بناء استعلام باستخدام هذه المعلومات، إلى جانب معلومات أساسية لشرح كيف يمكن أن تكون ذات صلة
- إرسال الاستعلام وتلقي سعر متوقع
- التداول بناءً على التوصية
- الانتظار والتكرار
يوضح هذا الوكيل كيفية قراءة المعلومات، وترجمتها إلى استعلام ينتج إجابة قابلة للاستخدام، واستخدام تلك الإجابة. كل هذه خطوات مطلوبة لأي وكيل ذكاء اصطناعي. تم تنفيذ هذا الوكيل بلغة Python لأنها اللغة الأكثر شيوعًا المستخدمة في الذكاء الاصطناعي.
لماذا نقوم بذلك؟
تسمح وكلاء التداول الآلي للمطورين باختيار وتنفيذ استراتيجية تداول. تسمح وكلاء الذكاء الاصطناعي باستراتيجيات تداول أكثر تعقيدًا وديناميكية، وربما تستخدم معلومات وخوارزميات لم يفكر المطور حتى في استخدامها.
الأدوات
يستخدم هذا البرنامج التعليمي Python (opens in a new tab)، ومكتبة Web3 (opens in a new tab)، ويونيسواب v3 (opens in a new tab) للحصول على الأسعار والتداول.
لماذا Python؟
اللغة الأكثر استخدامًا للذكاء الاصطناعي هي Python (opens in a new tab)، لذلك نستخدمها هنا. لا تقلق إذا كنت لا تعرف Python. اللغة واضحة جدًا، وسأشرح بالضبط ما تفعله.
مكتبة Web3 (opens in a new tab) هي واجهة برمجة تطبيقات (API) إيثيريوم الأكثر شيوعًا في Python. وهي سهلة الاستخدام إلى حد ما.
التداول على سلسلة الكتل
هناك العديد من منصات التداول اللامركزية (DEX) التي تتيح لك تداول الرموز المميزة على إيثيريوم. ومع ذلك، فإنها تميل إلى امتلاك أسعار صرف مماثلة بسبب المراجحة (arbitrage).
يونيسواب (opens in a new tab) هي منصة تداول لامركزية مستخدمة على نطاق واسع يمكننا استخدامها لكل من الأسعار (لرؤية القيم النسبية للرموز المميزة) والتداولات.
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.
git clone https://github.com/qbzzt/260215-ai-agent.git -b 01-getting-started cd 260215-ai-agent -
قم بتثبيت
uv(opens in a new tab). قد يكون الأمر مختلفًا في نظامك.pipx install uv -
قم بتنزيل المكتبات.
uv sync -
قم بتنشيط البيئة الافتراضية.
source .venv/bin/activate -
للتحقق من أن Python و Web3 يعملان بشكل صحيح، قم بتشغيل
python3وزوده بهذا البرنامج. يمكنك إدخاله في موجه>>>؛ ليست هناك حاجة لإنشاء ملف.from web3 import Web3 MAINNET_URL = "https://eth.drpc.org" w3 = Web3(Web3.HTTPProvider(MAINNET_URL)) w3.eth.block_number quit()
القراءة من سلسلة الكتل
الخطوة التالية هي القراءة من سلسلة الكتل. للقيام بذلك، تحتاج إلى التبديل إلى فرع 02-read-quote ثم استخدام uv لتشغيل البرنامج.
git checkout 02-read-quote
uv run agent.py
يجب أن تتلقى قائمة بكائنات Quote، كل منها يحتوي على طابع زمني، وسعر، والأصل (حاليًا دائمًا WETH/USDC).
إليك شرح سطر بسطر.
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
استيراد المكتبات التي نحتاجها. سيتم شرحها أدناه عند استخدامها.
print = functools.partial(print, flush=True)
يستبدل print الخاص بـ Python بإصدار يقوم دائمًا بمسح المخرجات (flushes output) على الفور. هذا مفيد في نص برمجي طويل الأمد لأننا لا نريد انتظار تحديثات الحالة أو مخرجات تصحيح الأخطاء.
MAINNET_URL = "https://eth.drpc.org"
رابط (URL) للوصول إلى الشبكة الرئيسية. يمكنك الحصول على واحد من العقدة كخدمة (Node as a service) أو استخدام أحد الروابط المعلن عنها في 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
تحدث كتلة شبكة إيثيريوم الرئيسية عادةً كل اثنتي عشرة ثانية، لذا فهذا هو عدد الكتل التي نتوقع حدوثها في فترة زمنية معينة. لاحظ أن هذا ليس رقمًا دقيقًا. عندما يكون مقترح الكتلة معطلاً، يتم تخطي تلك الكتلة، ويكون الوقت للكتلة التالية هو 24 ثانية. إذا أردنا الحصول على الكتلة الدقيقة لطابع زمني معين، فسنستخدم البحث الثنائي (binary search) (opens in a new tab). ومع ذلك، هذا قريب بما يكفي لأغراضنا. التنبؤ بالمستقبل ليس علمًا دقيقًا.
CYCLE_BLOCKS = DAY_BLOCKS
حجم الدورة. نقوم بمراجعة الأسعار مرة واحدة في كل دورة ونحاول تقدير القيمة في نهاية الدورة التالية.
# عنوان المجمع الذي نقرأه
WETHUSDC_ADDRESS = Web3.to_checksum_address("0x88e6A0c2dDD26FEEb64F039a2c41296FcB3f5640")
تؤخذ قيم الأسعار من مجمع يونيسواب v3 لـ USDC/WETH في العنوان 0x88e6A0c2dDD26FEEb64F039a2c41296FcB3f5640 (opens in a new tab). هذا العنوان موجود بالفعل في شكل المجموع الاختباري (checksum)، ولكن من الأفضل استخدام Web3.to_checksum_address (opens in a new tab) لجعل الكود قابلاً لإعادة الاستخدام.
POOL_ABI = [
{ "name": "slot0", ... },
{ "name": "token0", ... },
{ "name": "token1", ... },
]
ERC20_ABI = [
{ "name": "symbol", ... },
{ "name": "decimals", ... }
]
هذه هي واجهات التطبيق الثنائية (ABIs) (opens in a new tab) للعقدين اللذين نحتاج إلى الاتصال بهما. للحفاظ على إيجاز الكود، نقوم بتضمين الوظائف التي نحتاج إلى استدعائها فقط.
w3 = Web3(Web3.HTTPProvider(MAINNET_URL))
بدء مكتبة Web3 (opens in a new tab) والاتصال بعقدة إيثيريوم.
@dataclass(frozen=True)
class ERC20Token:
address: str
symbol: str
decimals: int
contract: Contract
هذه إحدى الطرق لإنشاء فئة بيانات (data class) في Python. يُستخدم نوع البيانات Contract (opens in a new tab) للاتصال بالعقد. لاحظ (frozen=True). في Python، يتم تعريف القيم المنطقية (booleans) (opens in a new tab) كـ True أو False، بأحرف كبيرة. فئة البيانات هذه هي frozen، مما يعني أنه لا يمكن تعديل الحقول.
لاحظ المسافة البادئة. على عكس اللغات المشتقة من C (opens in a new tab)، تستخدم Python المسافة البادئة للإشارة إلى الكتل البرمجية. يعرف مترجم Python أن التعريف التالي ليس جزءًا من فئة البيانات هذه لأنه لا يبدأ بنفس المسافة البادئة لحقول فئة البيانات.
@dataclass(frozen=True)
class PoolInfo:
address: str
token0: ERC20Token
token1: ERC20Token
contract: Contract
asset: str
decimal_factor: Decimal = 1
يُستخدم النوع Decimal (opens in a new tab) للتعامل بدقة مع الكسور العشرية.
def get_price(self, block: int) -> Decimal:
هذه هي الطريقة لتعريف دالة في Python. يتم وضع مسافة بادئة للتعريف لإظهار أنه لا يزال جزءًا من PoolInfo.
في الدالة التي تعد جزءًا من فئة بيانات، يكون المعامل الأول دائمًا self، وهو مثيل فئة البيانات الذي تم استدعاؤه هنا. يوجد هنا معامل آخر، وهو رقم الكتلة.
assert block <= w3.eth.block_number, "Block is in the future"
لو كنا نستطيع قراءة المستقبل، لما احتجنا إلى الذكاء الاصطناعي للتداول.
sqrt_price_x96 = Decimal(self.contract.functions.slot0().call(block_identifier=block)[0])
بناء الجملة لاستدعاء دالة على آلة إيثيريوم الافتراضية (EVM) من Web3 هو كالتالي: <contract object>.functions.<function name>().call(<parameters>). يمكن أن تكون المعاملات هي معاملات دالة EVM (إن وجدت؛ هنا لا يوجد) أو معاملات مسماة (named parameters) (opens in a new tab) لتعديل سلوك سلسلة الكتل. هنا نستخدم واحدًا، block_identifier، لتحديد رقم الكتلة الذي نرغب في التشغيل فيه.
النتيجة هي هذا الهيكل (struct)، في شكل مصفوفة (opens in a new tab). القيمة الأولى هي دالة لسعر الصرف بين الرمزين المميزين.
raw_price = (sqrt_price_x96 / Decimal(2**96)) ** 2
لتقليل الحسابات على السلسلة، لا يقوم يونيسواب v3 بتخزين عامل الصرف الفعلي بل جذره التربيعي. نظرًا لأن EVM لا تدعم رياضيات الفاصلة العائمة أو الكسور، فبدلاً من القيمة الفعلية، تكون الاستجابة هي
# (الرمز المميز 1 لكل الرمز المميز 0)
return 1/(raw_price * self.decimal_factor)
السعر الخام الذي نحصل عليه هو عدد token0 الذي نحصل عليه مقابل كل token1. في المجمع الخاص بنا، token0 هو USDC (عملة مستقرة بنفس قيمة الدولار الأمريكي) و token1 هو إيثر مغلف (WETH) (opens in a new tab). القيمة التي نريدها حقًا هي عدد الدولارات لكل WETH، وليس العكس.
العامل العشري هو النسبة بين العوامل العشرية (opens in a new tab) للرمزين المميزين.
@dataclass(frozen=True)
class Quote:
timestamp: str
price: Decimal
asset: str
تمثل فئة البيانات هذه سعرًا: سعر أصل معين في نقطة زمنية معينة. في هذه المرحلة، حقل asset غير ذي صلة لأننا نستخدم مجمعًا واحدًا وبالتالي لدينا أصل واحد. ومع ذلك، سنضيف المزيد من الأصول لاحقًا.
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
)
تأخذ هذه الدالة عنوانًا وتُرجع معلومات حول عقد الرمز المميز في ذلك العنوان. لإنشاء Contract جديد في Web3 (opens in a new tab)، نقدم العنوان و ABI إلى 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)
)
تُرجع هذه الدالة كل ما نحتاجه حول مجمع معين (opens in a new tab). بناء الجملة f"<string>" هو سلسلة نصية منسقة (formatted string) (opens in a new tab).
def get_quote(pool: PoolInfo, block_number: int = None) -> Quote:
الحصول على كائن Quote. القيمة الافتراضية لـ block_number هي None (لا توجد قيمة).
if block_number is None:
block_number = w3.eth.block_number
إذا لم يتم تحديد رقم الكتلة، فاستخدم w3.eth.block_number، وهو أحدث رقم كتلة. هذا هو بناء الجملة لـ عبارة if (opens in a new tab).
قد يبدو كما لو كان من الأفضل تعيين القيمة الافتراضية إلى w3.eth.block_number، ولكن هذا لا يعمل بشكل جيد لأنه سيكون رقم الكتلة في وقت تعريف الدالة. في وكيل يعمل لفترة طويلة، ستكون هذه مشكلة.
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
)
استخدم مكتبة datetime (opens in a new tab) لتنسيقه إلى تنسيق قابل للقراءة للبشر والنماذج اللغوية الكبيرة (LLMs). استخدم Decimal.quantize (opens in a new tab) لتقريب القيمة إلى منزلتين عشريتين.
def get_quotes(pool: PoolInfo, start_block: int, end_block: int, step: int) -> list[Quote]:
في Python، يمكنك تعريف قائمة (opens in a new tab) يمكن أن تحتوي فقط على نوع معين باستخدام list[<type>].
quotes = []
for block in range(start_block, end_block + 1, step):
في Python، عادةً ما تتكرر حلقة for (opens in a new tab) عبر قائمة. تأتي قائمة أرقام الكتل للعثور على الأسعار فيها من range (opens in a new tab).
quote = get_quote(pool, block)
quotes.append(quote)
return quotes
لكل رقم كتلة، احصل على كائن Quote وأضفه إلى قائمة quotes. ثم أرجع تلك القائمة.
pool = read_pool(WETHUSDC_ADDRESS)
quotes = get_quotes(
pool,
w3.eth.block_number - 12*CYCLE_BLOCKS,
w3.eth.block_number,
CYCLE_BLOCKS
)
pprint(quotes)
هذا هو الكود الرئيسي للنص البرمجي. اقرأ معلومات المجمع، واحصل على اثني عشر سعرًا، وقم بـ pprint (opens in a new tab) لها.
إنشاء مطالبة (Prompt)
بعد ذلك، نحتاج إلى تحويل قائمة الأسعار هذه إلى مطالبة لنموذج لغوي كبير (LLM) والحصول على قيمة مستقبلية متوقعة.
git checkout 03-create-prompt
uv run agent.py
سيكون الإخراج الآن عبارة عن مطالبة لنموذج لغوي كبير، مشابهة لـ:
بناءً على هذه الأسعار:
الأصل: WETH/USDC
2026-01-20T16:34 3016.21
.
.
.
2026-02-01T17:49 2299.10
الأصل: WBTC/WETH
2026-01-20T16:34 29.84
.
.
.
2026-02-01T17:50 33.46
ما هي القيمة التي تتوقعها لـ WETH/USDC في الوقت 2026-02-02T17:56؟
قدم إجابتك كرقم واحد مقرب إلى منزلتين عشريتين،
بدون أي نص آخر.
لاحظ أن هناك أسعارًا لأصلين هنا، WETH/USDC و WBTC/WETH. قد تؤدي إضافة أسعار من أصل آخر إلى تحسين دقة التنبؤ.
كيف تبدو المطالبة
تحتوي هذه المطالبة على ثلاثة أقسام، وهي شائعة جدًا في مطالبات النماذج اللغوية الكبيرة.
-
المعلومات. تمتلك النماذج اللغوية الكبيرة الكثير من المعلومات من تدريبها، لكنها عادة لا تمتلك أحدث المعلومات. هذا هو السبب في أننا بحاجة إلى استرداد أحدث الأسعار هنا. تسمى إضافة المعلومات إلى المطالبة التوليد المعزز بالاسترداد (RAG) (opens in a new tab).
-
السؤال الفعلي. هذا ما نريد معرفته.
-
تعليمات تنسيق المخرجات. عادةً، سيعطينا النموذج اللغوي الكبير تقديرًا مع شرح لكيفية وصوله إليه. هذا أفضل للبشر، لكن برنامج الكمبيوتر يحتاج فقط إلى النتيجة النهائية.
شرح الكود
إليك الكود الجديد.
from datetime import datetime, timezone, timedelta
نحتاج إلى تزويد النموذج اللغوي الكبير بالوقت الذي نريد تقديرًا له. للحصول على وقت "n دقائق/ساعات/أيام" في المستقبل، نستخدم فئة timedelta (opens in a new tab).
# عناوين المجمعات التي نقرأها
WETHUSDC_ADDRESS = Web3.to_checksum_address("0x88e6A0c2dDD26FEEb64F039a2c41296FcB3f5640")
WETHWBTC_ADDRESS = Web3.to_checksum_address("0xCBCdF9626bC03E24f779434178A73a0B4bad62eD")
لدينا مجمعان نحتاج إلى قراءتهما.
@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 # (الرمز المميز 1 لكل الرمز المميز 0)
if self.reverse:
return 1/(raw_price * self.decimal_factor)
else:
return raw_price * self.decimal_factor
في مجمع WETH/USDC، نريد أن نعرف كم عدد token0 (USDC) الذي نحتاجه لشراء واحد من token1 (WETH). في مجمع WETH/WBTC، نريد أن نعرف كم عدد token1 (WETH) الذي نحتاجه لشراء واحد token0 (WBTC، وهو بيتكوين مغلف). نحتاج إلى تتبع ما إذا كانت نسبة المجمع بحاجة إلى عكسها.
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
)
لمعرفة ما إذا كان المجمع بحاجة إلى عكسه، نحصل على ذلك كمدخل إلى read_pool. أيضًا، يجب إعداد رمز الأصل بشكل صحيح.
بناء الجملة <a> if <b> else <c> هو المعادل في Python لـ المعامل الشرطي الثلاثي (ternary conditional operator) (opens in a new tab)، والذي سيكون في لغة مشتقة من C هو <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
تقوم هذه الدالة ببناء سلسلة نصية تنسق قائمة من كائنات Quote، بافتراض أنها تنطبق جميعها على نفس الأصل.
def make_prompt(quotes: list[list[Quote]], expected_time: str, asset: str) -> str:
return f"""
في Python، تُكتب السلاسل النصية متعددة الأسطر (opens in a new tab) كـ """ .... """.
Given these quotes:
{
functools.reduce(lambda acc, q: acc + '\n' + q,
map(lambda q: format_quotes(q), quotes))
}
هنا، نستخدم نمط MapReduce (opens in a new tab) لإنشاء سلسلة نصية لكل قائمة أسعار باستخدام format_quotes، ثم تقليلها إلى سلسلة نصية واحدة لاستخدامها في المطالبة.
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.
"""
بقية المطالبة كما هو متوقع.
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
)
مراجعة المجمعين والحصول على أسعار من كليهما.
future_time = (datetime.now(timezone.utc) + timedelta(days=1)).isoformat()[0:16]
print(make_prompt(wethusdc_quotes + wethwbtc_quotes, future_time, wethusdc_pool.asset))
تحديد النقطة الزمنية المستقبلية التي نريد التقدير لها، وإنشاء المطالبة.
التفاعل مع نموذج لغوي كبير (LLM)
بعد ذلك، نقوم بمطالبة نموذج لغوي كبير فعلي ونتلقى قيمة مستقبلية متوقعة. لقد كتبت هذا البرنامج باستخدام OpenAI، لذا إذا كنت ترغب في استخدام مزود مختلف، فستحتاج إلى تعديله.
-
احصل على حساب OpenAI (opens in a new tab)
-
قم بتمويل الحساب (opens in a new tab) — الحد الأدنى للمبلغ في وقت الكتابة هو $5
-
في سطر الأوامر، قم بتصدير مفتاح API حتى يتمكن برنامجك من استخدامه
export OPENAI_API_KEY=sk-<the rest of the key goes here> -
قم بالتبديل (Checkout) وتشغيل الوكيل
git checkout 04-interface-llm uv run agent.py
إليك الكود الجديد.
from openai import OpenAI
open_ai = OpenAI() # يقرأ العميل متغير البيئة OPENAI_API_KEY
استيراد وإنشاء مثيل لواجهة برمجة تطبيقات OpenAI.
response = open_ai.chat.completions.create(
model="gpt-4-turbo",
messages=[
{"role": "user", "content": prompt}
],
temperature=0.0,
max_tokens=16,
)
استدعاء واجهة برمجة تطبيقات OpenAI (open_ai.chat.completions.create) لإنشاء الاستجابة.
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")
إخراج السعر وتقديم توصية بالشراء أو البيع.
اختبار التنبؤات
الآن بعد أن أصبح بإمكاننا إنشاء تنبؤات، يمكننا أيضًا استخدام البيانات التاريخية لتقييم ما إذا كنا ننتج تنبؤات مفيدة.
uv run test-predictor.py
النتيجة المتوقعة مشابهة لـ:
التنبؤ لـ 2026-01-05T19:50: المتوقع 3138.93 USD، الفعلي 3218.92 USD، الخطأ 79.99 USD
التنبؤ لـ 2026-01-06T19:56: المتوقع 3243.39 USD، الفعلي 3221.08 USD، الخطأ 22.31 USD
التنبؤ لـ 2026-01-07T20:02: المتوقع 3223.24 USD، الفعلي 3146.89 USD، الخطأ 76.35 USD
التنبؤ لـ 2026-01-08T20:11: المتوقع 3150.47 USD، الفعلي 3092.04 USD، الخطأ 58.43 USD
.
.
.
التنبؤ لـ 2026-01-31T22:33: المتوقع 2637.73 USD، الفعلي 2417.77 USD، الخطأ 219.96 USD
التنبؤ لـ 2026-02-01T22:41: المتوقع 2381.70 USD، الفعلي 2318.84 USD، الخطأ 62.86 USD
التنبؤ لـ 2026-02-02T22:49: المتوقع 2234.91 USD، الفعلي 2349.28 USD، الخطأ 114.37 USD
متوسط خطأ التنبؤ على مدار 29 تنبؤًا: 83.87103448275862068965517241 USD
متوسط التغيير لكل توصية: 4.787931034482758620689655172 USD
التباين القياسي للتغييرات: 104.42 USD
الأيام المربحة: 51.72%
الأيام الخاسرة: 48.28%
معظم كود الاختبار مطابق للوكيل، ولكن إليك الأجزاء الجديدة أو المعدلة.
CYCLES_FOR_TEST = 40 # للاختبار الرجعي، كم عدد الدورات التي نختبرها
# الحصول على الكثير من عروض الأسعار
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
)
ننظر إلى CYCLES_FOR_TEST (محددة كـ 40 هنا) يومًا للوراء.
# إنشاء توقعات والتحقق منها مقابل التاريخ الحقيقي
total_error = Decimal(0)
changes = []
هناك نوعان من الأخطاء التي نهتم بها. الأول، total_error، هو ببساطة مجموع الأخطاء التي ارتكبها المتنبئ.
لفهم الثاني، changes، نحتاج إلى تذكر الغرض من الوكيل. ليس الغرض هو التنبؤ بنسبة WETH/USDC (سعر ETH). بل إصدار توصيات البيع والشراء. إذا كان السعر حاليًا $2000 وتوقع أن يكون $2010 غدًا، فلا نمانع إذا كانت النتيجة الفعلية هي $2020 وكسبنا أموالاً إضافية. لكننا نمانع إذا توقع $2010، واشترى ETH بناءً على تلك التوصية، وانخفض السعر إلى $1990.
for index in range(0,len(wethusdc_quotes)-CYCLES_BACK):
يمكننا فقط النظر في الحالات التي يتوفر فيها السجل الكامل (القيم المستخدمة للتنبؤ والقيمة في العالم الحقيقي لمقارنتها بها). هذا يعني أن أحدث حالة يجب أن تكون تلك التي بدأت منذ CYCLES_BACK.
wethusdc_slice = wethusdc_quotes[index:index+CYCLES_BACK]
wethwbtc_slice = wethwbtc_quotes[index:index+CYCLES_BACK]
استخدم الشرائح (slices) (opens in a new tab) للحصول على نفس عدد العينات الذي يستخدمه الوكيل. الكود الموجود بين هنا والمقطع التالي هو نفس كود الحصول على التنبؤ الموجود لدينا في الوكيل.
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
احصل على السعر المتوقع، والسعر الفعلي، والسعر في وقت التنبؤ. نحتاج إلى السعر في وقت التنبؤ لتحديد ما إذا كانت التوصية هي الشراء أم البيع.
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")
احسب الخطأ، وأضفه إلى الإجمالي.
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)
بالنسبة لـ changes، نريد التأثير النقدي لشراء أو بيع واحد ETH. لذا أولاً، نحتاج إلى تحديد التوصية، ثم تقييم كيفية تغير السعر الفعلي، وما إذا كانت التوصية قد حققت أرباحًا (تغيير إيجابي) أو تسببت في خسارة (تغيير سلبي).
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")
الإبلاغ عن النتائج.
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%}")
استخدم filter (opens in a new tab) لحساب عدد الأيام المربحة وعدد الأيام الخاسرة. النتيجة هي كائن تصفية (filter object)، والذي نحتاج إلى تحويله إلى قائمة للحصول على الطول.
إرسال المعاملات
الآن نحتاج إلى إرسال المعاملات فعليًا. ومع ذلك، لا أريد إنفاق أموال حقيقية في هذه المرحلة، قبل إثبات كفاءة النظام. بدلاً من ذلك، سنقوم بإنشاء تفرع محلي من الشبكة الرئيسية، و"التداول" على تلك الشبكة.
إليك خطوات إنشاء تفرع محلي وتمكين التداول.
-
قم بتثبيت Foundry (opens in a new tab)
-
ابدأ تشغيل
anvil(opens in a new tab)anvil --fork-url https://eth.drpc.org --block-time 12يستمع
anvilعلى الرابط الافتراضي لـ Foundry، وهو http://localhost:8545، (opens in a new tab) لذلك لا نحتاج إلى تحديد الرابط لـ أمرcast(opens in a new tab) الذي نستخدمه لمعالجة سلسلة الكتل. -
عند التشغيل في
anvil، هناك عشرة حسابات اختبار تحتوي على ETH — قم بتعيين متغيرات البيئة للحساب الأولPRIVATE_KEY=0xac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff80 ADDRESS=`cast wallet address $PRIVATE_KEY` -
هذه هي العقود التي نحتاج إلى استخدامها.
SwapRouter(opens in a new tab) هو عقد يونيسواب v3 الذي نستخدمه للتداول الفعلي. يمكننا التداول مباشرة من خلال المجمع، ولكن هذا أسهل بكثير.المتغيران السفليان هما مسارات يونيسواب v3 المطلوبة للمبادلة بين WETH و USDC.
WETH_ADDRESS=0xC02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2 USDC_ADDRESS=0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48 POOL_ADDRESS=0x88e6A0c2dDD26FEEb64F039a2c41296FcB3f5640 SWAP_ROUTER=0xE592427A0AEce92De3Edee1F18E0157C05861564 WETH_TO_USDC=0xC02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc20001F4A0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48 USDC_TO_WETH=0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB480001F4C02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2 -
يحتوي كل حساب من حسابات الاختبار على 10,000 ETH. استخدم عقد WETH لتغليف 1000 ETH للحصول على 1000 WETH للتداول.
cast send $WETH_ADDRESS "deposit()" --value 1000ether --private-key $PRIVATE_KEY -
استخدم
SwapRouterلمبادلة 500 WETH مقابل 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_KEYيُنشئ استدعاء
approveسماحية تسمح لـSwapRouterبإنفاق بعض الرموز المميزة الخاصة بنا. لا يمكن للعقود مراقبة الأحداث، لذلك إذا قمنا بتحويل الرموز المميزة مباشرة إلى عقدSwapRouter، فلن يعرف أنه تم الدفع له. بدلاً من ذلك، نسمح لعقدSwapRouterبإنفاق مبلغ معين، ثم يقومSwapRouterبذلك. يتم ذلك من خلال دالة يستدعيهاSwapRouter، لذلك يعرف ما إذا كانت ناجحة. -
تحقق من أن لديك ما يكفي من كلا الرمزين المميزين.
cast call $WETH_ADDRESS "balanceOf(address)" $ADDRESS | cast from-wei echo `cast call $USDC_ADDRESS "balanceOf(address)" $ADDRESS | cast to-dec`/10^6 | bc
الآن بعد أن أصبح لدينا WETH و USDC، يمكننا تشغيل الوكيل فعليًا.
git checkout 05-trade
uv run agent.py
سيبدو الإخراج مشابهًا لـ:
(ai-trading-agent) qbzzt@Ori-Cloudnomics:~/260215-ai-agent$ uv run agent.py
السعر الحالي: 1843.16
في 2026-02-06T23:07، السعر المتوقع: 1724.41 USD
أرصدة الحساب قبل التداول:
رصيد USDC: 927301.578272
رصيد WETH: 500
بيع، أتوقع أن ينخفض السعر بمقدار 118.75 USD
تم إرسال معاملة الموافقة: 74e367ddbb407c1aaf567d87aa5863049991b1d2aa092b6b85195d925e2bd41f
تم تعدين معاملة الموافقة.
تم إرسال معاملة البيع: fad1bcf938585c9e90364b26ac7a80eea9efd34c37e5db81e58d7655bcae28bf
تم تعدين معاملة البيع.
أرصدة الحساب بعد التداول:
رصيد USDC: 929143.797116
رصيد WETH: 499
لاستخدامه فعليًا، تحتاج إلى بعض التغييرات الطفيفة.
- في السطر 14، قم بتغيير
MAINNET_URLإلى نقطة وصول حقيقية، مثلhttps://eth.drpc.org - في السطر 28، قم بتغيير
PRIVATE_KEYإلى مفتاحك الخاص - ما لم تكن ثريًا جدًا ويمكنك شراء أو بيع 1 ETH كل يوم لوكيل غير مثبت، فقد ترغب في تغيير السطر 29 لتقليل
WETH_TRADE_AMOUNT
شرح الكود
إليك الكود الجديد.
SWAP_ROUTER_ADDRESS=Web3.to_checksum_address("0xE592427A0AEce92De3Edee1F18E0157C05861564")
WETH_TO_USDC=bytes.fromhex("C02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc20001F4A0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48")
USDC_TO_WETH=bytes.fromhex("A0b86991c6218b36c1d19D4a2e9Eb0cE3606eB480001F4C02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2")
PRIVATE_KEY="0xac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff80"
نفس المتغيرات التي استخدمناها في الخطوة 4.
WETH_TRADE_AMOUNT=1
المبلغ المراد تداوله.
ERC20_ABI = [
{ "name": "symbol", ... },
{ "name": "decimals", ... },
{ "name": "balanceOf", ...},
{ "name": "approve", ...}
]
للتداول الفعلي، نحتاج إلى دالة approve. نريد أيضًا إظهار الأرصدة قبل وبعد، لذلك نحتاج أيضًا إلى balanceOf.
SWAP_ROUTER_ABI = [
{ "name": "exactInput", ...},
]
في ABI الخاص بـ SwapRouter نحتاج فقط إلى exactInput. هناك دالة ذات صلة، exactOutput، يمكننا استخدامها لشراء WETH واحد بالضبط، ولكن للتبسيط نستخدم فقط exactInput في كلتا الحالتين.
account = w3.eth.account.from_key(PRIVATE_KEY)
swap_router = w3.eth.contract(
address=SWAP_ROUTER_ADDRESS,
abi=SWAP_ROUTER_ABI
)
تعريفات Web3 لـ account (opens in a new tab) وعقد SwapRouter.
def txn_params() -> dict:
return {
"from": account.address,
"value": 0,
"gas": 300000,
"nonce": w3.eth.get_transaction_count(account.address),
}
معاملات المعاملة. نحتاج إلى دالة هنا لأن الرقم الفريد (nonce) (opens in a new tab) يجب أن يتغير في كل مرة.
def approve_token(contract: Contract, amount: int):
الموافقة على سماحية الرمز المميز لـ 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)
هذه هي الطريقة التي نرسل بها معاملة في Web3. أولاً نستخدم كائن Contract (opens in a new tab) لبناء المعاملة. ثم نستخدم web3.eth.account.sign_transaction (opens in a new tab) لتوقيع المعاملة، باستخدام PRIVATE_KEY. أخيرًا، نستخدم w3.eth.send_raw_transaction (opens in a new tab) لإرسال المعاملة.
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) حتى يتم تعدين المعاملة. ويُرجع الإيصال إذا لزم الأمر.
SELL_PARAMS = {
"path": WETH_TO_USDC,
"recipient": account.address,
"deadline": 2**256 - 1,
"amountIn": WETH_TRADE_AMOUNT * 10 ** wethusdc_pool.token1.decimals,
"amountOutMinimum": 0,
}
هذه هي المعاملات عند بيع 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,
}
على عكس SELL_PARAMS، يمكن أن تتغير معاملات الشراء. مبلغ الإدخال هو تكلفة 1 WETH، كما هو متاح في 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.")
الدالتان buy() و sell() متطابقتان تقريبًا. أولاً نوافق على سماحية كافية لـ SwapRouter، ثم نستدعيه بالمسار والمبلغ الصحيحين.
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)}")
الإبلاغ عن أرصدة المستخدم بكلتا العملتين.
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()
يعمل هذا الوكيل حاليًا مرة واحدة فقط. ومع ذلك، يمكنك تغييره ليعمل بشكل مستمر إما عن طريق تشغيله من crontab (opens in a new tab) أو عن طريق تغليف الأسطر 368-400 في حلقة واستخدام time.sleep (opens in a new tab) للانتظار حتى يحين وقت الدورة التالية.
تحسينات محتملة
هذه ليست نسخة إنتاج كاملة؛ إنها مجرد مثال لتعليم الأساسيات. إليك بعض الأفكار للتحسينات.
تداول أكثر ذكاءً
هناك حقيقتان مهمتان يتجاهلهما الوكيل عند اتخاذ قرار بشأن ما يجب القيام به.
- حجم التغيير المتوقع. يبيع الوكيل مبلغًا ثابتًا من
WETHإذا كان من المتوقع أن ينخفض السعر، بغض النظر عن حجم الانخفاض. يمكن القول إنه سيكون من الأفضل تجاهل التغييرات الطفيفة والبيع بناءً على مقدار الانخفاض المتوقع في السعر. - المحفظة الحالية. إذا كان 10% من محفظتك في WETH وتعتقد أن السعر سيرتفع، فمن المنطقي على الأرجح شراء المزيد. ولكن إذا كان 90% من محفظتك في WETH، فقد تكون معرضًا للسوق بشكل كافٍ، ولا داعي لشراء المزيد. العكس صحيح إذا كنت تتوقع انخفاض السعر.
ماذا لو كنت تريد الحفاظ على سرية استراتيجية التداول الخاصة بك؟
يمكن لموردي الذكاء الاصطناعي رؤية الاستعلامات التي ترسلها إلى نماذجهم اللغوية الكبيرة، مما قد يكشف عن نظام التداول العبقري الذي طورته مع وكيلك. نظام التداول الذي يستخدمه عدد كبير جدًا من الأشخاص لا قيمة له لأن الكثير من الأشخاص يحاولون الشراء عندما تريد الشراء (ويرتفع السعر) ويحاولون البيع عندما تريد البيع (وينخفض السعر).
يمكنك تشغيل نموذج لغوي كبير محليًا، على سبيل المثال، باستخدام LM-Studio (opens in a new tab)، لتجنب هذه المشكلة.
من روبوت ذكاء اصطناعي إلى وكيل ذكاء اصطناعي
يمكنك تقديم حجة قوية بأن هذا روبوت ذكاء اصطناعي، وليس وكيل ذكاء اصطناعي. فهو ينفذ استراتيجية بسيطة نسبيًا تعتمد على معلومات محددة مسبقًا. يمكننا تمكين التحسين الذاتي، على سبيل المثال، من خلال توفير قائمة بمجمعات يونيسواب v3 وأحدث قيمها والسؤال عن أي مجموعة لها أفضل قيمة تنبؤية.
الحماية من الانزلاق السعري
حاليًا لا توجد حماية من الانزلاق السعري (opens in a new tab). إذا كان السعر الحالي هو $2000، والسعر المتوقع هو $2100، فسيقوم الوكيل بالشراء. ومع ذلك، إذا ارتفعت التكلفة قبل أن يشتري الوكيل إلى $2200، فلن يكون من المنطقي الشراء بعد الآن.
لتنفيذ الحماية من الانزلاق السعري، حدد قيمة amountOutMinimum في السطرين 325 و 334 من agent.py (opens in a new tab).
الخاتمة
نأمل أن تكون الآن تعرف ما يكفي للبدء مع وكلاء الذكاء الاصطناعي. هذه ليست نظرة عامة شاملة على الموضوع؛ هناك كتب كاملة مخصصة لذلك، ولكن هذا يكفي لتبدأ. حظًا موفقًا!