تخطي إلى المحتوى الرئيسي

أنشئ وكيل تداول ذكاء اصطناعي خاص بك على إيثيريوم

الذكاء الاصطناعي
التداول
وكيل
python
متوسط
أوري بوميرانتس
13 فبراير 2026
22 دقيقة للقراءة

في هذا البرنامج التعليمي، ستتعلم كيفية بناء وكيل تداول ذكاء اصطناعي بسيط. يعمل هذا الوكيل باستخدام الخطوات التالية:

  1. قراءة الأسعار الحالية والسابقة لرمز مميز، بالإضافة إلى معلومات أخرى قد تكون ذات صلة
  2. بناء استعلام باستخدام هذه المعلومات، إلى جانب معلومات أساسية لشرح كيف يمكن أن تكون ذات صلة
  3. إرسال الاستعلام وتلقي سعر متوقع
  4. التداول بناءً على التوصية
  5. الانتظار والتكرار

يوضح هذا الوكيل كيفية قراءة المعلومات، وترجمتها إلى استعلام ينتج إجابة قابلة للاستخدام، واستخدام تلك الإجابة. كل هذه خطوات مطلوبة لأي وكيل ذكاء اصطناعي. تم تنفيذ هذا الوكيل بلغة 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))

  1. إذا لم يكن لديك بالفعل، فقم بتنزيل وتثبيت Python (opens in a new tab).

  2. استنسخ مستودع GitHub.

    git clone https://github.com/qbzzt/260215-ai-agent.git -b 01-getting-started
    cd 260215-ai-agent
    
  3. قم بتثبيت uv (opens in a new tab). قد يكون الأمر مختلفًا في نظامك.

    pipx install uv
    
  4. قم بتنزيل المكتبات.

    uv sync
    
  5. قم بتنشيط البيئة الافتراضية.

    source .venv/bin/activate
    
  6. للتحقق من أن 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).

إليك شرح سطر بسطر.

استيراد المكتبات التي نحتاجها. سيتم شرحها أدناه عند استخدامها.

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) لجعل الكود قابلاً لإعادة الاستخدام.

هذه هي واجهات التطبيق الثنائية (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 لا تدعم رياضيات الفاصلة العائمة أو الكسور، فبدلاً من القيمة الفعلية، تكون الاستجابة هي price296

         # (الرمز المميز 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 غير ذي صلة لأننا نستخدم مجمعًا واحدًا وبالتالي لدينا أصل واحد. ومع ذلك، سنضيف المزيد من الأصول لاحقًا.

تأخذ هذه الدالة عنوانًا وتُرجع معلومات حول عقد الرمز المميز في ذلك العنوان. لإنشاء Contract جديد في Web3 (opens in a new tab)، نقدم العنوان و ABI إلى w3.eth.contract.

تُرجع هذه الدالة كل ما نحتاجه حول مجمع معين (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. ثم أرجع تلك القائمة.

هذا هو الكود الرئيسي للنص البرمجي. اقرأ معلومات المجمع، واحصل على اثني عشر سعرًا، وقم بـ pprint (opens in a new tab) لها.

إنشاء مطالبة (Prompt)

بعد ذلك، نحتاج إلى تحويل قائمة الأسعار هذه إلى مطالبة لنموذج لغوي كبير (LLM) والحصول على قيمة مستقبلية متوقعة.

git checkout 03-create-prompt
uv run agent.py

سيكون الإخراج الآن عبارة عن مطالبة لنموذج لغوي كبير، مشابهة لـ:

لاحظ أن هناك أسعارًا لأصلين هنا، WETH/USDC و WBTC/WETH. قد تؤدي إضافة أسعار من أصل آخر إلى تحسين دقة التنبؤ.

كيف تبدو المطالبة

تحتوي هذه المطالبة على ثلاثة أقسام، وهي شائعة جدًا في مطالبات النماذج اللغوية الكبيرة.

  1. المعلومات. تمتلك النماذج اللغوية الكبيرة الكثير من المعلومات من تدريبها، لكنها عادة لا تمتلك أحدث المعلومات. هذا هو السبب في أننا بحاجة إلى استرداد أحدث الأسعار هنا. تسمى إضافة المعلومات إلى المطالبة التوليد المعزز بالاسترداد (RAG) (opens in a new tab).

  2. السؤال الفعلي. هذا ما نريد معرفته.

  3. تعليمات تنسيق المخرجات. عادةً، سيعطينا النموذج اللغوي الكبير تقديرًا مع شرح لكيفية وصوله إليه. هذا أفضل للبشر، لكن برنامج الكمبيوتر يحتاج فقط إلى النتيجة النهائية.

شرح الكود

إليك الكود الجديد.

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")

لدينا مجمعان نحتاج إلى قراءتهما.

في مجمع WETH/USDC، نريد أن نعرف كم عدد token0 (USDC) الذي نحتاجه لشراء واحد من token1 (WETH). في مجمع WETH/WBTC، نريد أن نعرف كم عدد token1 (WETH) الذي نحتاجه لشراء واحد token0 (WBTC، وهو بيتكوين مغلف). نحتاج إلى تتبع ما إذا كانت نسبة المجمع بحاجة إلى عكسها.

لمعرفة ما إذا كان المجمع بحاجة إلى عكسه، نحصل على ذلك كمدخل إلى 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.
    """

بقية المطالبة كما هو متوقع.

مراجعة المجمعين والحصول على أسعار من كليهما.

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، لذا إذا كنت ترغب في استخدام مزود مختلف، فستحتاج إلى تعديله.

  1. احصل على حساب OpenAI (opens in a new tab)

  2. قم بتمويل الحساب (opens in a new tab) — الحد الأدنى للمبلغ في وقت الكتابة هو $5

  3. أنشئ مفتاح API (opens in a new tab)

  4. في سطر الأوامر، قم بتصدير مفتاح API حتى يتمكن برنامجك من استخدامه

    export OPENAI_API_KEY=sk-<the rest of the key goes here>
    
  5. قم بالتبديل (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) لإنشاء الاستجابة.

إخراج السعر وتقديم توصية بالشراء أو البيع.

اختبار التنبؤات

الآن بعد أن أصبح بإمكاننا إنشاء تنبؤات، يمكننا أيضًا استخدام البيانات التاريخية لتقييم ما إذا كنا ننتج تنبؤات مفيدة.

uv run test-predictor.py

النتيجة المتوقعة مشابهة لـ:

معظم كود الاختبار مطابق للوكيل، ولكن إليك الأجزاء الجديدة أو المعدلة.

ننظر إلى 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)، والذي نحتاج إلى تحويله إلى قائمة للحصول على الطول.

إرسال المعاملات

الآن نحتاج إلى إرسال المعاملات فعليًا. ومع ذلك، لا أريد إنفاق أموال حقيقية في هذه المرحلة، قبل إثبات كفاءة النظام. بدلاً من ذلك، سنقوم بإنشاء تفرع محلي من الشبكة الرئيسية، و"التداول" على تلك الشبكة.

إليك خطوات إنشاء تفرع محلي وتمكين التداول.

  1. قم بتثبيت Foundry (opens in a new tab)

  2. ابدأ تشغيل 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) الذي نستخدمه لمعالجة سلسلة الكتل.

  3. عند التشغيل في anvil، هناك عشرة حسابات اختبار تحتوي على ETH — قم بتعيين متغيرات البيئة للحساب الأول

    PRIVATE_KEY=0xac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff80
    ADDRESS=`cast wallet address $PRIVATE_KEY`
    
  4. هذه هي العقود التي نحتاج إلى استخدامها. 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
    
  5. يحتوي كل حساب من حسابات الاختبار على 10,000 ETH. استخدم عقد WETH لتغليف 1000 ETH للحصول على 1000 WETH للتداول.

    cast send $WETH_ADDRESS "deposit()" --value 1000ether --private-key $PRIVATE_KEY
    
  6. استخدم 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، لذلك يعرف ما إذا كانت ناجحة.

  7. تحقق من أن لديك ما يكفي من كلا الرمزين المميزين.

    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

سيبدو الإخراج مشابهًا لـ:

لاستخدامه فعليًا، تحتاج إلى بعض التغييرات الطفيفة.

  • في السطر 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.

الدالتان 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)}")

الإبلاغ عن أرصدة المستخدم بكلتا العملتين.

يعمل هذا الوكيل حاليًا مرة واحدة فقط. ومع ذلك، يمكنك تغييره ليعمل بشكل مستمر إما عن طريق تشغيله من 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).

الخاتمة

نأمل أن تكون الآن تعرف ما يكفي للبدء مع وكلاء الذكاء الاصطناعي. هذه ليست نظرة عامة شاملة على الموضوع؛ هناك كتب كاملة مخصصة لذلك، ولكن هذا يكفي لتبدأ. حظًا موفقًا!

انظر هنا للمزيد من أعمالي (opens in a new tab).