コンテンツにスキップ
初級

新しいブロックのリスニング⚓︎

block WSfinalizedBlock WS WebSocket チャネルは、新しい ブロック が生成されたとき、または ファイナライズ されたときにリアルタイムの通知を送信します。 /chain/info GET エンドポイントをポーリングする場合と比較して、WebSocket は API 呼び出しを繰り返すオーバーヘッドなしに、更新が発生した瞬間にプッシュします。

このチュートリアルでは、両方のチャネルをサブスクライブし、到着した各更新を表示する方法を説明します。

メモ

ポーリングベースのアプローチについては、チェーンとファイナライズの最新高の照会 チュートリアルを参照してください。

前提条件⚓︎

websockets ライブラリをインストールします。

pip install websockets

このチュートリアルでは、Node.js 22 以降で利用可能なネイティブの WebSocket API を使用します。 追加のパッケージは必要ありません。

完全なコード⚓︎

このチュートリアルの完全なコード一覧を以下に示します。 詳細な手順ごとの説明は次のセクションで行います。

import asyncio
import json
import os

from websockets import connect

NODE_URL = os.getenv('NODE_URL', 'https://reference.symboltest.net:3001')
WS_URL = NODE_URL.replace('http', 'ws', 1) + '/ws'
print(f'Using node {NODE_URL}')


async def main():
    async with connect(WS_URL) as websocket:
        # Connect to websocket endpoint
        response = json.loads(await websocket.recv())
        uid = response['uid']
        print(f'Connected to {WS_URL} with uid {uid}')

        # Subscribe to block channel
        await websocket.send(json.dumps(
            {'uid': uid, 'subscribe': 'block'}))
        print('Subscribed to block channel')

        # Subscribe to finalizedBlock channel
        await websocket.send(json.dumps(
            {'uid': uid, 'subscribe': 'finalizedBlock'}))
        print('Subscribed to finalizedBlock channel')

        # Handle incoming messages
        try:
            async for raw_message in websocket:
                message = json.loads(raw_message)
                topic = message['topic']

                if topic == 'block':
                    block = message['data']['block']
                    block_meta = message['data']['meta']
                    print(
                        f'New block: height={int(block["height"]):,}'
                        f' hash={block_meta["hash"][:16]}...'
                    )

                if topic == 'finalizedBlock':
                    finalized = message['data']
                    print(
                        f'Finalized: height={int(finalized["height"]):,}'
                        f' hash={finalized["hash"][:16]}...'
                    )

        # Unsubscribe on exit
        finally:
            await websocket.send(json.dumps(
                {'uid': uid, 'unsubscribe': 'block'}))
            await websocket.send(json.dumps(
                {'uid': uid, 'unsubscribe': 'finalizedBlock'}))
            print('Unsubscribed from all channels')


try:
    asyncio.run(main())
except KeyboardInterrupt:
    pass
except Exception as error:
    print(error)

Download source

const NODE_URL = process.env.NODE_URL ||
    'https://reference.symboltest.net:3001';
const WS_URL = NODE_URL.replace('http', 'ws') + '/ws';
console.log(`Using node ${NODE_URL}`);

try {
    const websocket = new WebSocket(WS_URL);

    // Connect to websocket endpoint
    const uid = await new Promise((resolve) => {
        websocket.addEventListener('message', (event) => {
            const message = JSON.parse(event.data);
            resolve(message.uid);
        }, { once: true });
    });
    console.log(`Connected to ${WS_URL} with uid ${uid}`);

    // Subscribe to block channel
    websocket.send(JSON.stringify({ uid, subscribe: 'block' }));
    console.log('Subscribed to block channel');

    // Subscribe to finalizedBlock channel
    websocket.send(JSON.stringify({ uid, subscribe: 'finalizedBlock' }));
    console.log('Subscribed to finalizedBlock channel');

    // Handle incoming messages
    websocket.addEventListener('message', (event) => {
        const message = JSON.parse(event.data);
        const topic = message.topic;

        if (topic === 'block') {
            const block = message.data.block;
            const blockMeta = message.data.meta;
            console.log(
                'New block:'
                + ` height=${parseInt(block.height, 10).toLocaleString()}`
                + ` hash=${blockMeta.hash.substring(0, 16)}...`
            );
        }

        if (topic === 'finalizedBlock') {
            const finalized = message.data;
            console.log(
                'Finalized:'
                + ' height='
                + `${parseInt(finalized.height, 10).toLocaleString()}`
                + ` hash=${finalized.hash.substring(0, 16)}...`
            );
        }
    });

    // Unsubscribe on exit
    process.on('SIGINT', () => {
        websocket.send(JSON.stringify({ uid, unsubscribe: 'block' }));
        websocket.send(
            JSON.stringify({ uid, unsubscribe: 'finalizedBlock' }));
        console.log('Unsubscribed from all channels');
        websocket.close();
        process.exit(0);
    });
} catch (error) {
    console.error(error);
}

Download source

このスニペットでは、 NODE_URL 環境変数を使用して Symbol API ノード を設定します。 値が指定されない場合は、デフォルト値が使用されます。 WebSocket URL は、HTTP プロトコルを WebSocket プロトコルに置き換え、 /ws を追加することで NODE_URL から派生します。

プログラムは Ctrl+C で中断されるまで実行され、接続を閉じる前にサブスクライブ解除のステップをトリガーします。

コード解説⚓︎

WebSocket への接続⚓︎

    async with connect(WS_URL) as websocket:
        # Connect to websocket endpoint
        response = json.loads(await websocket.recv())
        uid = response['uid']
        print(f'Connected to {WS_URL} with uid {uid}')
    const websocket = new WebSocket(WS_URL);

    // Connect to websocket endpoint
    const uid = await new Promise((resolve) => {
        websocket.addEventListener('message', (event) => {
            const message = JSON.parse(event.data);
            resolve(message.uid);
        }, { once: true });
    });
    console.log(`Connected to ${WS_URL} with uid ${uid}`);

最初のステップは、ノードの /ws エンドポイントへの WebSocket 接続を開くことです。 接続すると、サーバーは以降のすべてのサブスクリプションリクエストに含める必要がある一意の識別子( uid )を含むメッセージを送信します。

接続プロトコルの詳細については、WebSocket リファレンス を参照してください。

チャネルのサブスクライブ⚓︎

        # Subscribe to block channel
        await websocket.send(json.dumps(
            {'uid': uid, 'subscribe': 'block'}))
        print('Subscribed to block channel')

        # Subscribe to finalizedBlock channel
        await websocket.send(json.dumps(
            {'uid': uid, 'subscribe': 'finalizedBlock'}))
        print('Subscribed to finalizedBlock channel')
    // Subscribe to block channel
    websocket.send(JSON.stringify({ uid, subscribe: 'block' }));
    console.log('Subscribed to block channel');

    // Subscribe to finalizedBlock channel
    websocket.send(JSON.stringify({ uid, subscribe: 'finalizedBlock' }));
    console.log('Subscribed to finalizedBlock channel');

コードは2つのチャネルをサブスクライブします。

  • block WS: 新しいブロックが生成されるたびに(約30秒ごとに)通知します。
  • finalizedBlock WS: ファイナライズラウンドが完了するたびに(約10〜20分ごとに)通知します。

各サブスクリプションメッセージには、接続ステップで受信した uid とチャネル名が含まれます。

メッセージの処理⚓︎

        # Handle incoming messages
        try:
            async for raw_message in websocket:
                message = json.loads(raw_message)
                topic = message['topic']

                if topic == 'block':
                    block = message['data']['block']
                    block_meta = message['data']['meta']
                    print(
                        f'New block: height={int(block["height"]):,}'
                        f' hash={block_meta["hash"][:16]}...'
                    )

                if topic == 'finalizedBlock':
                    finalized = message['data']
                    print(
                        f'Finalized: height={int(finalized["height"]):,}'
                        f' hash={finalized["hash"][:16]}...'
                    )
    // Handle incoming messages
    websocket.addEventListener('message', (event) => {
        const message = JSON.parse(event.data);
        const topic = message.topic;

        if (topic === 'block') {
            const block = message.data.block;
            const blockMeta = message.data.meta;
            console.log(
                'New block:'
                + ` height=${parseInt(block.height, 10).toLocaleString()}`
                + ` hash=${blockMeta.hash.substring(0, 16)}...`
            );
        }

        if (topic === 'finalizedBlock') {
            const finalized = message.data;
            console.log(
                'Finalized:'
                + ' height='
                + `${parseInt(finalized.height, 10).toLocaleString()}`
                + ` hash=${finalized.hash.substring(0, 16)}...`
            );
        }
    });

コードは、プログラムが中断されるまで受信メッセージをリスニングします。 各メッセージには、チャネルを識別する topic フィールドと、イベントペイロードを含む data オブジェクトが含まれます。

block メッセージの場合、ペイロードは BlockInfoDTO スキーマに従います。 このチュートリアルでは、各ブロックを識別するためにそのうちの2つを使用します。

  • data.block.height: 新しいブロックの高さ。
  • data.meta.hash: 新しいブロックのハッシュ。

finalizedBlock メッセージの場合、ペイロードは FinalizedBlockDTO スキーマに従います。 このチュートリアルでは以下を使用します。

  • data.height: ファイナライズされたブロックの高さ。
  • data.hash: ファイナライズされたブロックのハッシュ。

チェーンの高さは、新しいブロックが生成されるたびに増加します。 ファイナライズは通常、ブロック生成の10〜20分後に行われるため、ファイナライズされた高さはチェーンの先端(最新高)より遅れます。

投票ノード がこのプロセスをどのように推進するかについての詳細は、テキストブックの コンセンサス セクションを参照してください。

終了時のサブスクライブ解除⚓︎

        # Unsubscribe on exit
        finally:
            await websocket.send(json.dumps(
                {'uid': uid, 'unsubscribe': 'block'}))
            await websocket.send(json.dumps(
                {'uid': uid, 'unsubscribe': 'finalizedBlock'}))
            print('Unsubscribed from all channels')
    // Unsubscribe on exit
    process.on('SIGINT', () => {
        websocket.send(JSON.stringify({ uid, unsubscribe: 'block' }));
        websocket.send(
            JSON.stringify({ uid, unsubscribe: 'finalizedBlock' }));
        console.log('Unsubscribed from all channels');
        websocket.close();
        process.exit(0);
    });

プログラムが中断されたとき( Ctrl+C )、コードは接続を閉じる前に両方のチャネルのサブスクライブ解除メッセージを送信します。 これにより、ノードからのクリーンな切断が保証されます。

出力⚓︎

以下の出力は、新しいブロックとファイナライズイベントをリスニングする典型的な実行例を示しています。

Using node https://reference.symboltest.net:3001
Connected to wss://reference.symboltest.net:3001/ws with uid 9AQEv+DFuCuddfrJlNh7ERf8Zlg=
Subscribed to block channel
Subscribed to finalizedBlock channel
New block: height=3,176,948 hash=EDD1ED4C92E29655...
New block: height=3,176,949 hash=18DBF7E75B2CC003...
New block: height=3,176,950 hash=A3409E1951E8FC8C...
Finalized: height=3,176,948 hash=EDD1ED4C92E29655...
New block: height=3,176,951 hash=A7D80AFD75C4C918...
New block: height=3,176,952 hash=027D939F6E9D0DAE...
Unsubscribed from all channels

出力の主なポイント:

  • 接続 (2行目): wss:// URL への WebSocket 接続が確立され、サーバーは一意の uid を返します。
  • サブスクリプション (3-4行目): block チャネルと finalizedBlock チャネルの両方がサブスクライブされます。
  • 新しいブロック (5-7行目、9-10行目): 新しいブロックの通知が約30秒ごとに届きます。
  • ファイナライズ (8行目): ファイナライズラウンドが完了すると、一度に複数のブロックをカバーするファイナライズ通知が届きます。
  • サブスクライブ解除 (11行目): Ctrl+C を押すと、コードは両方のチャネルのサブスクライブを解除します。

結論⚓︎

このチュートリアルでは、以下の方法を説明しました。

ステップ 関連ドキュメント
block チャネルのサブスクライブ block WS
finalized チャネルのサブスクライブ finalizedBlock WS
ブロックメッセージの処理 BlockInfoDTO
ファイナライズメッセージの処理 FinalizedBlockDTO