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

ネームスペースのモザイクへのリンクと解除⚓︎

ネームスペースモザイク にリンクさせることができます。これにより、トランザクションにおいて長い16進数のモザイクIDの代わりに、人間が読み取り可能なエイリアス(別名)を使用できるようになります。

このチュートリアルでは、ネームスペースをモザイク識別子にリンクする方法と、不要になった際にリンクを解除する方法を説明します。

前提条件⚓︎

開始する前に、以下を確認してください。

さらに、トランザクションがどのようにアナウンスされ承認されるかを理解するために、転送トランザクション のチュートリアルを復習しておいてください。

ネームスペースとモザイクの両方の所有権が必要

ネームスペースとモザイクの両方を所有しているアカウントのみが、それらをリンクさせることができます。

完全なコード⚓︎

import json
import os
import time
import urllib.request

from symbolchain.CryptoTypes import PrivateKey
from symbolchain.facade.SymbolFacade import SymbolFacade
from symbolchain.sc import Amount
from symbolchain.symbol.IdGenerator import (
    generate_mosaic_alias_id,
    generate_namespace_path
)
from symbolchain.symbol.Network import NetworkTimestamp

NODE_URL = os.getenv('NODE_URL', 'https://reference.symboltest.net:3001')
print(f'Using node {NODE_URL}')


# Helper function to announce a transaction
def announce_transaction(payload, label):
    print(f'Announcing {label} to /transactions')
    request = urllib.request.Request(
        f'{NODE_URL}/transactions',
        data=payload.encode(),
        headers={'Content-Type': 'application/json'},
        method='PUT'
    )
    with urllib.request.urlopen(request) as announce_response:
        print(f'  Response: {announce_response.read().decode()}')


# Helper function to wait for transaction confirmation
def wait_for_confirmation(tx_hash, label):
    print(f'Waiting for {label} confirmation...')
    for attempt in range(60):
        time.sleep(1)
        try:
            url = f'{NODE_URL}/transactionStatus/{tx_hash}'
            with urllib.request.urlopen(url) as confirm_response:
                status = json.loads(confirm_response.read().decode())
                print(f"  Transaction status: {status['group']}")
                if status['group'] == 'confirmed':
                    print(f'{label} confirmed in {attempt} seconds')
                    return
                if status['group'] == 'failed':
                    raise RuntimeError(f"{label} failed: {status['code']}")
        except urllib.error.HTTPError:
            print('  Transaction status: unknown')
    raise TimeoutError(f'{label} not confirmed after 60 seconds')


SIGNER_PRIVATE_KEY = os.getenv(
    'SIGNER_PRIVATE_KEY',
    '0000000000000000000000000000000000000000000000000000000000000000')
signer_key_pair = SymbolFacade.KeyPair(PrivateKey(SIGNER_PRIVATE_KEY))

facade = SymbolFacade('testnet')
signer_address = facade.network.public_key_to_address(
    signer_key_pair.public_key)
print(f'Signer address: {signer_address}')

namespace_name = os.getenv('NAMESPACE_NAME', 'my_namespace')
print(f'Namespace name: {namespace_name}')

namespace_id = generate_namespace_path(namespace_name)[-1]
print(f'Namespace ID: {namespace_id} ({hex(namespace_id)})')

# Target mosaic ID to link the namespace to
mosaic_id = int(os.getenv('MOSAIC_ID', '0x45C8C3733983AAC2'), 16)
print(f'Mosaic ID: {mosaic_id} ({hex(mosaic_id)})')

try:
    # Fetch current network time
    time_path = '/node/time'
    print(f'Fetching current network time from {time_path}')
    with urllib.request.urlopen(f'{NODE_URL}{time_path}') as response:
        response_json = json.loads(response.read().decode())
        receive_timestamp = (
            response_json['communicationTimestamps']['receiveTimestamp'])
        timestamp = NetworkTimestamp(int(receive_timestamp))
        print(f'  Network time: {timestamp.timestamp} ms since nemesis')

    # Fetch recommended fees
    fee_path = '/network/fees/transaction'
    print(f'Fetching recommended fees from {fee_path}')
    with urllib.request.urlopen(f'{NODE_URL}{fee_path}') as response:
        response_json = json.loads(response.read().decode())
        median_multiplier = response_json['medianFeeMultiplier']
        minimum_multiplier = response_json['minFeeMultiplier']
        fee_multiplier = max(median_multiplier, minimum_multiplier)
        print(f'  Fee multiplier: {fee_multiplier}')

    # Build the alias transaction
    transaction = facade.transaction_factory.create({
        'type': 'mosaic_alias_transaction_v1',
        'signer_public_key': signer_key_pair.public_key,
        'deadline': timestamp.add_hours(2).timestamp,
        'namespace_id': namespace_id,
        'mosaic_id': mosaic_id,
        'alias_action': 'link'
    })
    transaction.fee = Amount(fee_multiplier * transaction.size)

    # Sign transaction and generate final payload
    json_payload = facade.transaction_factory.attach_signature(
        transaction,
        facade.sign_transaction(signer_key_pair, transaction))
    print('Mosaic alias transaction:')
    print(json.dumps(transaction.to_json(), indent=2))

    transaction_hash = facade.hash_transaction(transaction)
    print(f'Transaction hash: {transaction_hash}')

    # Announce and confirm transaction
    announce_transaction(json_payload, 'mosaic alias transaction')
    wait_for_confirmation(transaction_hash, 'mosaic alias transaction')

    # Retrieve the namespace to verify the alias
    namespace_path = f'/namespaces/{namespace_id:x}'
    print(f'Fetching namespace information from {namespace_path}')
    with urllib.request.urlopen(f'{NODE_URL}{namespace_path}') as response:
        response_json = json.loads(response.read().decode())
        namespace_info = response_json['namespace']
        print('Alias information:')
        alias_type = namespace_info['alias']['type']
        print(f'  Alias type: {alias_type}')
        if alias_type == 1:  # MOSAIC type
            aliased_mosaic_id = namespace_info['alias']['mosaicId']
            print(f'  Linked mosaic ID: {aliased_mosaic_id}')

    # Send a transfer using the alias instead of a raw mosaic ID
    print(f'Using alias in transfer: {namespace_name}')

    # Convert namespace to mosaic alias ID
    mosaic_alias_id = generate_mosaic_alias_id(namespace_name)
    print(f'Mosaic ID (alias):'
        f' {mosaic_alias_id} ({hex(mosaic_alias_id)})')

    test_transaction = facade.transaction_factory.create({
        'type': 'transfer_transaction_v1',
        'signer_public_key': signer_key_pair.public_key,
        'deadline': timestamp.add_hours(2).timestamp,
        'recipient_address':
            facade.network.public_key_to_address(
                signer_key_pair.public_key),
        'mosaics': [{
            'mosaic_id': mosaic_alias_id,
            'amount': 1
        }]
    })
    test_transaction.fee = Amount(fee_multiplier * test_transaction.size)
    test_json_payload = facade.transaction_factory.attach_signature(
        test_transaction,
        facade.sign_transaction(signer_key_pair, test_transaction))
    print('Test transaction:')
    print(json.dumps(test_transaction.to_json(), indent=2))
    test_transaction_hash = facade.hash_transaction(test_transaction)
    print(f'Transaction hash: {test_transaction_hash}')
    announce_transaction(test_json_payload, 'test transaction')
    wait_for_confirmation(test_transaction_hash, 'test transaction')

except Exception as e:
    print(e)

Download source

import { PrivateKey } from 'symbol-sdk';
import {
    NetworkTimestamp,
    SymbolFacade,
    generateMosaicAliasId,
    generateNamespacePath,
    models
} from 'symbol-sdk/symbol';

const NODE_URL = process.env.NODE_URL ||
    'https://reference.symboltest.net:3001';
console.log('Using node', NODE_URL);

// Helper function to announce a transaction
async function announceTransaction(payload, label) {
    console.log(`Announcing ${label} to /transactions`);
    const response = await fetch(`${NODE_URL}/transactions`, {
        method: 'PUT',
        headers: { 'Content-Type': 'application/json' },
        body: payload
    });
    console.log('  Response:', await response.text());
}

// Helper function to wait for transaction confirmation
async function waitForConfirmation(transactionHash, label) {
    console.log(`Waiting for ${label} confirmation...`);
    for (let attempt = 0; 60 > attempt; attempt++) {
        await new Promise(resolve => { setTimeout(resolve, 1000); });
        try {
            const response = await fetch(
                `${NODE_URL}/transactionStatus/${transactionHash}`);
            const status = await response.json();
            console.log('  Transaction status:', status.group);
            if ('confirmed' === status.group) {
                console.log(`${label} confirmed in`, attempt, 'seconds');
                return;
            }
            if ('failed' === status.group)
                throw new Error(`${label} failed: ${status.code}`);
        } catch (e) {
            if (e.message.includes('failed'))
                throw e;
            console.log('  Transaction status: unknown');
        }
    }
    throw new Error(`${label} not confirmed after 60 seconds`);
}


const SIGNER_PRIVATE_KEY = process.env.SIGNER_PRIVATE_KEY ||
    '0000000000000000000000000000000000000000000000000000000000000000';
const signerKeyPair = new SymbolFacade.KeyPair(
    new PrivateKey(SIGNER_PRIVATE_KEY));

const facade = new SymbolFacade('testnet');
const signerAddress = facade.network.publicKeyToAddress(
    signerKeyPair.publicKey);
console.log('Signer address:', signerAddress.toString());

const namespaceName = process.env.NAMESPACE_NAME || 'my_namespace';
console.log('Namespace name:', namespaceName);

const nsPath = generateNamespacePath(namespaceName);
const namespaceId = nsPath[nsPath.length - 1];
console.log(
    'Namespace ID:',
    `${namespaceId} (0x${namespaceId.toString(16)})`);

// Target mosaic ID to link the namespace to
const mosaicId = BigInt(process.env.MOSAIC_ID || '0x45C8C3733983AAC2');
console.log('Mosaic ID:', `${mosaicId} (0x${mosaicId.toString(16)})`);

try {
    // Fetch current network time
    const timePath = '/node/time';
    console.log('Fetching current network time from', timePath);
    const timeResponse = await fetch(`${NODE_URL}${timePath}`);
    const timeJSON = await timeResponse.json();
    const receiveTimestamp =
        timeJSON.communicationTimestamps.receiveTimestamp;
    const timestamp = new NetworkTimestamp(receiveTimestamp);
    console.log('  Network time:', timestamp.timestamp,
        'ms since nemesis');

    // Fetch recommended fees
    const feePath = '/network/fees/transaction';
    console.log('Fetching recommended fees from', feePath);
    const feeResponse = await fetch(`${NODE_URL}${feePath}`);
    const feeJSON = await feeResponse.json();
    const medianMultiplier = feeJSON.medianFeeMultiplier;
    const minimumMultiplier = feeJSON.minFeeMultiplier;
    const feeMultiplier = Math.max(medianMultiplier, minimumMultiplier);
    console.log('  Fee multiplier:', feeMultiplier);

    // Build the alias transaction
    const transaction = facade.transactionFactory.create({
        type: 'mosaic_alias_transaction_v1',
        signerPublicKey: signerKeyPair.publicKey.toString(),
        deadline: timestamp.addHours(2).timestamp,
        namespaceId,
        mosaicId,
        aliasAction: 'link'
    });
    transaction.fee = new models.Amount(feeMultiplier * transaction.size);

    // Sign transaction and generate final payload
    const jsonPayload = facade.transactionFactory.static.attachSignature(
        transaction,
        facade.signTransaction(signerKeyPair, transaction));
    console.log('Built transaction:');
    console.dir(transaction.toJson(), { colors: true });

    const transactionHash = facade.hashTransaction(transaction).toString();
    console.log('Transaction hash:', transactionHash);

    // Announce and confirm transaction
    await announceTransaction(jsonPayload, 'mosaic alias transaction');
    await waitForConfirmation(transactionHash, 'mosaic alias transaction');

    // Retrieve the namespace to verify the alias
    const namespacePath = `/namespaces/${namespaceId.toString(16)}`;
    console.log('Fetching namespace information from', namespacePath);
    const namespaceResponse = await fetch(`${NODE_URL}${namespacePath}`);
    const namespaceJSON = await namespaceResponse.json();
    const namespaceInfo = namespaceJSON.namespace;
    console.log('Alias information:');
    console.log('  Alias type:', namespaceInfo.alias.type);
    if (1 === namespaceInfo.alias.type) { // MOSAIC type
        console.log('  Linked mosaic ID:', namespaceInfo.alias.mosaicId);
    }

    // Send a transfer using the alias instead of a raw mosaic ID
    console.log('Using alias in transfer:', namespaceName);

    // Convert namespace to mosaic alias ID
    const mosaicAliasId = generateMosaicAliasId(namespaceName);
    console.log('  Mosaic ID (alias):',
        `${mosaicAliasId} (0x${mosaicAliasId.toString(16)})`);

    const test_transaction = facade.transactionFactory.create({
        type: 'transfer_transaction_v1',
        signerPublicKey: signerKeyPair.publicKey.toString(),
        deadline: timestamp.addHours(2).timestamp,
        recipientAddress: facade.network.publicKeyToAddress(
            signerKeyPair.publicKey).toString(),
        mosaics: [{
            mosaicId: mosaicAliasId,
            amount: 1n
        }]
    });
    test_transaction.fee = new models.Amount(
        feeMultiplier * test_transaction.size);
    const testJsonPayload =
        facade.transactionFactory.static.attachSignature(
            test_transaction,
            facade.signTransaction(signerKeyPair, test_transaction));
    console.log('Test transaction:');
    console.dir(test_transaction.toJson(), { colors: true });
    const testTransactionHash =
        facade.hashTransaction(test_transaction).toString();
    console.log('Transaction hash:', testTransactionHash);
    await announceTransaction(testJsonPayload, 'test transaction');
    await waitForConfirmation(testTransactionHash, 'test transaction');

} catch (e) {
    console.error(e.message);
}

Download source

コード解説⚓︎

アカウントの設定⚓︎

SIGNER_PRIVATE_KEY = os.getenv(
    'SIGNER_PRIVATE_KEY',
    '0000000000000000000000000000000000000000000000000000000000000000')
signer_key_pair = SymbolFacade.KeyPair(PrivateKey(SIGNER_PRIVATE_KEY))

facade = SymbolFacade('testnet')
signer_address = facade.network.public_key_to_address(
    signer_key_pair.public_key)
print(f'Signer address: {signer_address}')
const SIGNER_PRIVATE_KEY = process.env.SIGNER_PRIVATE_KEY ||
    '0000000000000000000000000000000000000000000000000000000000000000';
const signerKeyPair = new SymbolFacade.KeyPair(
    new PrivateKey(SIGNER_PRIVATE_KEY));

const facade = new SymbolFacade('testnet');
const signerAddress = facade.network.publicKeyToAddress(
    signerKeyPair.publicKey);
console.log('Signer address:', signerAddress.toString());

このスニペットは、署名者の秘密鍵を SIGNER_PRIVATE_KEY 環境変数から読み取ります。設定されていない場合はデフォルトのテストキーが使用されます。 署名者のアドレスは公開鍵から派生します。 このアカウントは、リンクされるネームスペースとモザイクの両方を所有している必要があります。

ネームスペースとターゲットモザイクの定義⚓︎

namespace_name = os.getenv('NAMESPACE_NAME', 'my_namespace')
print(f'Namespace name: {namespace_name}')

namespace_id = generate_namespace_path(namespace_name)[-1]
print(f'Namespace ID: {namespace_id} ({hex(namespace_id)})')

# Target mosaic ID to link the namespace to
mosaic_id = int(os.getenv('MOSAIC_ID', '0x45C8C3733983AAC2'), 16)
print(f'Mosaic ID: {mosaic_id} ({hex(mosaic_id)})')
const namespaceName = process.env.NAMESPACE_NAME || 'my_namespace';
console.log('Namespace name:', namespaceName);

const nsPath = generateNamespacePath(namespaceName);
const namespaceId = nsPath[nsPath.length - 1];
console.log(
    'Namespace ID:',
    `${namespaceId} (0x${namespaceId.toString(16)})`);

// Target mosaic ID to link the namespace to
const mosaicId = BigInt(process.env.MOSAIC_ID || '0x45C8C3733983AAC2');
console.log('Mosaic ID:', `${mosaicId} (0x${mosaicId.toString(16)})`);

コードでは以下を定義しています。

  • ネームスペース名: リンクするネームスペース。 NAMESPACE_NAME 環境変数から読み込まれ、設定されていない場合は my_namespace がデフォルトとなります。 この名前は、自身のアカウントがすでに所有しているネームスペースと一致している必要があります。
  • ネームスペース ID: ID は、 を使用してネームスペース名から生成されます。 これは階層内の各レベルの ID 配列を返します。 最終的なネームスペース ID を取得するために、最後の要素が選択されます。 最後の要素を取得する方法は、ルートネームスペースとサブネームスペースの両方で機能します。

    foo のようなルートネームスペースの場合、配列には要素が1つ含まれます。 symbol.xym のようなサブネームスペースの場合、要素は2つ含まれ、最後の要素が symbol 配下の xym の ID となります。

    サブネームスペース ID は一意

    サブネームスペース ID は階層的に派生するため、末尾の名前が同じでも親が異なる2つのサブネームスペースは、異なる ID を生成します。 例えば、 foo.xymbar.xym のパスの最後の要素は異なります。

  • モザイク ID: ネームスペースが指し示す先のモザイクの16進数識別子。 MOSAIC_ID 環境変数から読み込まれます。設定されていない場合は、デフォルトのテスト用モザイクIDが使用されます。

ネットワーク時間と手数料の取得⚓︎

    # Fetch current network time
    time_path = '/node/time'
    print(f'Fetching current network time from {time_path}')
    with urllib.request.urlopen(f'{NODE_URL}{time_path}') as response:
        response_json = json.loads(response.read().decode())
        receive_timestamp = (
            response_json['communicationTimestamps']['receiveTimestamp'])
        timestamp = NetworkTimestamp(int(receive_timestamp))
        print(f'  Network time: {timestamp.timestamp} ms since nemesis')

    # Fetch recommended fees
    fee_path = '/network/fees/transaction'
    print(f'Fetching recommended fees from {fee_path}')
    with urllib.request.urlopen(f'{NODE_URL}{fee_path}') as response:
        response_json = json.loads(response.read().decode())
        median_multiplier = response_json['medianFeeMultiplier']
        minimum_multiplier = response_json['minFeeMultiplier']
        fee_multiplier = max(median_multiplier, minimum_multiplier)
        print(f'  Fee multiplier: {fee_multiplier}')
    // Fetch current network time
    const timePath = '/node/time';
    console.log('Fetching current network time from', timePath);
    const timeResponse = await fetch(`${NODE_URL}${timePath}`);
    const timeJSON = await timeResponse.json();
    const receiveTimestamp =
        timeJSON.communicationTimestamps.receiveTimestamp;
    const timestamp = new NetworkTimestamp(receiveTimestamp);
    console.log('  Network time:', timestamp.timestamp,
        'ms since nemesis');

    // Fetch recommended fees
    const feePath = '/network/fees/transaction';
    console.log('Fetching recommended fees from', feePath);
    const feeResponse = await fetch(`${NODE_URL}${feePath}`);
    const feeJSON = await feeResponse.json();
    const medianMultiplier = feeJSON.medianFeeMultiplier;
    const minimumMultiplier = feeJSON.minFeeMultiplier;
    const feeMultiplier = Math.max(medianMultiplier, minimumMultiplier);
    console.log('  Fee multiplier:', feeMultiplier);

転送トランザクション チュートリアルで説明されているプロセスに従い、ネットワーク時間と推奨手数料をそれぞれ /node/time GET および /network/fees/transaction GET から取得します。

トランザクションの構築⚓︎

    # Build the alias transaction
    transaction = facade.transaction_factory.create({
        'type': 'mosaic_alias_transaction_v1',
        'signer_public_key': signer_key_pair.public_key,
        'deadline': timestamp.add_hours(2).timestamp,
        'namespace_id': namespace_id,
        'mosaic_id': mosaic_id,
        'alias_action': 'link'
    })
    transaction.fee = Amount(fee_multiplier * transaction.size)
    // Build the alias transaction
    const transaction = facade.transactionFactory.create({
        type: 'mosaic_alias_transaction_v1',
        signerPublicKey: signerKeyPair.publicKey.toString(),
        deadline: timestamp.addHours(2).timestamp,
        namespaceId,
        mosaicId,
        aliasAction: 'link'
    });
    transaction.fee = new models.Amount(feeMultiplier * transaction.size);

モザイクエイリアストランザクションでは以下を指定します。

  • Type: モザイクエイリアストランザクションにはタイプ MosaicAliasTransactionV1 を使用します。

  • 署名者の公開鍵: ネームスペースとモザイクを所有し、トランザクション手数料を支払うアカウント。

  • ネームスペース ID: リンクされるネームスペースの識別子。

  • モザイク ID: ネームスペースにリンクするモザイクの識別子。

  • エイリアスアクション: link という値はエイリアスを作成します。後にエイリアスを削除するには、代わりに unlink を使用します。

エイリアスのリンク解除

ネームスペースのモザイクへのリンクを解除するには、同じネームスペース ID とモザイク ID を指定し、 alias_action フィールドを unlink に設定した別の MosaicAliasTransactionV1 トランザクションをアナウンスします。

リンク解除プロセスによってネームスペースやモザイク自体が削除されるわけではなく、それらの関連付けのみが削除されます。 リンク解除後、そのネームスペースは別のモザイクやアドレスにリンクさせることができます。

トランザクションの送信⚓︎

    # Sign transaction and generate final payload
    json_payload = facade.transaction_factory.attach_signature(
        transaction,
        facade.sign_transaction(signer_key_pair, transaction))
    print('Mosaic alias transaction:')
    print(json.dumps(transaction.to_json(), indent=2))

    transaction_hash = facade.hash_transaction(transaction)
    print(f'Transaction hash: {transaction_hash}')

    # Announce and confirm transaction
    announce_transaction(json_payload, 'mosaic alias transaction')
    wait_for_confirmation(transaction_hash, 'mosaic alias transaction')
    // Sign transaction and generate final payload
    const jsonPayload = facade.transactionFactory.static.attachSignature(
        transaction,
        facade.signTransaction(signerKeyPair, transaction));
    console.log('Built transaction:');
    console.dir(transaction.toJson(), { colors: true });

    const transactionHash = facade.hashTransaction(transaction).toString();
    console.log('Transaction hash:', transactionHash);

    // Announce and confirm transaction
    await announceTransaction(jsonPayload, 'mosaic alias transaction');
    await waitForConfirmation(transactionHash, 'mosaic alias transaction');

トランザクションは、転送トランザクションの作成 と同じプロセスに従って署名され、アナウンスされます。

エイリアスの検証⚓︎

    # Retrieve the namespace to verify the alias
    namespace_path = f'/namespaces/{namespace_id:x}'
    print(f'Fetching namespace information from {namespace_path}')
    with urllib.request.urlopen(f'{NODE_URL}{namespace_path}') as response:
        response_json = json.loads(response.read().decode())
        namespace_info = response_json['namespace']
        print('Alias information:')
        alias_type = namespace_info['alias']['type']
        print(f'  Alias type: {alias_type}')
        if alias_type == 1:  # MOSAIC type
            aliased_mosaic_id = namespace_info['alias']['mosaicId']
            print(f'  Linked mosaic ID: {aliased_mosaic_id}')
    // Retrieve the namespace to verify the alias
    const namespacePath = `/namespaces/${namespaceId.toString(16)}`;
    console.log('Fetching namespace information from', namespacePath);
    const namespaceResponse = await fetch(`${NODE_URL}${namespacePath}`);
    const namespaceJSON = await namespaceResponse.json();
    const namespaceInfo = namespaceJSON.namespace;
    console.log('Alias information:');
    console.log('  Alias type:', namespaceInfo.alias.type);
    if (1 === namespaceInfo.alias.type) { // MOSAIC type
        console.log('  Linked mosaic ID:', namespaceInfo.alias.mosaicId);
    }

エイリアスが作成されたことを確認するために、コードは /namespaces/{namespaceId} GET エンドポイントを使用してネットワークからネームスペース情報を取得します。

レスポンスにはエイリアスタイプ( mosaic )とリンクされたモザイクIDが含まれ、ネームスペースが指定したモザイクを指していることが確認されます。

エイリアスの使用⚓︎

    # Send a transfer using the alias instead of a raw mosaic ID
    print(f'Using alias in transfer: {namespace_name}')

    # Convert namespace to mosaic alias ID
    mosaic_alias_id = generate_mosaic_alias_id(namespace_name)
    print(f'Mosaic ID (alias):'
        f' {mosaic_alias_id} ({hex(mosaic_alias_id)})')

    test_transaction = facade.transaction_factory.create({
        'type': 'transfer_transaction_v1',
        'signer_public_key': signer_key_pair.public_key,
        'deadline': timestamp.add_hours(2).timestamp,
        'recipient_address':
            facade.network.public_key_to_address(
                signer_key_pair.public_key),
        'mosaics': [{
            'mosaic_id': mosaic_alias_id,
            'amount': 1
        }]
    })
    test_transaction.fee = Amount(fee_multiplier * test_transaction.size)
    test_json_payload = facade.transaction_factory.attach_signature(
        test_transaction,
        facade.sign_transaction(signer_key_pair, test_transaction))
    print('Test transaction:')
    print(json.dumps(test_transaction.to_json(), indent=2))
    test_transaction_hash = facade.hash_transaction(test_transaction)
    print(f'Transaction hash: {test_transaction_hash}')
    announce_transaction(test_json_payload, 'test transaction')
    wait_for_confirmation(test_transaction_hash, 'test transaction')
    // Send a transfer using the alias instead of a raw mosaic ID
    console.log('Using alias in transfer:', namespaceName);

    // Convert namespace to mosaic alias ID
    const mosaicAliasId = generateMosaicAliasId(namespaceName);
    console.log('  Mosaic ID (alias):',
        `${mosaicAliasId} (0x${mosaicAliasId.toString(16)})`);

    const test_transaction = facade.transactionFactory.create({
        type: 'transfer_transaction_v1',
        signerPublicKey: signerKeyPair.publicKey.toString(),
        deadline: timestamp.addHours(2).timestamp,
        recipientAddress: facade.network.publicKeyToAddress(
            signerKeyPair.publicKey).toString(),
        mosaics: [{
            mosaicId: mosaicAliasId,
            amount: 1n
        }]
    });
    test_transaction.fee = new models.Amount(
        feeMultiplier * test_transaction.size);
    const testJsonPayload =
        facade.transactionFactory.static.attachSignature(
            test_transaction,
            facade.signTransaction(signerKeyPair, test_transaction));
    console.log('Test transaction:');
    console.dir(test_transaction.toJson(), { colors: true });
    const testTransactionHash =
        facade.hashTransaction(test_transaction).toString();
    console.log('Transaction hash:', testTransactionHash);
    await announceTransaction(testJsonPayload, 'test transaction');
    await waitForConfirmation(testTransactionHash, 'test transaction');

ネームスペースがモザイクにリンクされると、トランザクション内でモザイクIDの代わりにネームスペースを使用できるようになります。 コードでは、16進数のモザイクIDではなく、モザイク配列内でエイリアスを使用した 転送トランザクション の作成を実演しています。

簡単にするため、この例ではモザイクを送信者自身のアドレスに送り返しており、トランザクションのアナウンスや承認の待機は行いません。

ネームスペースをモザイクIDとして使用するには、ネームスペース名を を使用してモザイクエイリアスIDに変換します。 前のセクションで説明したように、ネームスペースパスの最後のコンポーネントがネームスペースIDとして使用されます。

転送トランザクションのアナウンス方法の詳細については、転送トランザクション チュートリアルを参照してください。

モザイク解決レシート

ネットワークがネームスペースエイリアスをモザイクIDとして使用するトランザクションを処理すると、モザイク解決レシート(Mosaic Resolution Receipt) が生成されます。 このレシートには、トランザクションが承認された時点でエイリアスが実際に指し示していたモザイクIDが記録されます。

これは過去の監査可能性にとって重要です。エイリアスはいつでも変更または削除できるため、たとえエイリアスがその後更新されていたとしても、解決されたモザイクIDを常に検証できることがレシートによって保証されます。

解決レシートは /statements/resolutions/mosaic GET エンドポイントを使用して照会できます。 レシートの詳細については、テキストブックの 解決ステートメント セクションを参照してください。

出力⚓︎

以下に示す出力は、プログラムの典型的な実行結果に対応しています。

Using node https://reference.symboltest.net:3001
Signer address: TCHBDENCLKEBILBPWP3JPB2XNY64OE7PYHHE32I
Namespace name: ns_1778514754.sub_1778514998
Namespace ID: 18182428759753405815 (0xfc54f69f52f2b577)
Mosaic ID: 507540378882865098 (0x70b2549187af7ca)
Fetching current network time from /node/time
  Network time: 111579658666 ms since nemesis
Fetching recommended fees from /network/fees/transaction
  Fee multiplier: 100
Mosaic alias transaction:
{
  "signature": "D969758584AA819BC7135E817368DBCAFEBFFBE39B7FE30807017F418C770B5D19FBD3E08CA2EE67D54DDBD4C1D3323965AFA87509174B737AAA4BA116AF2604",
  "signer_public_key": "3B6A27BCCEB6A42D62A3A8D02A6F0D73653215771DE243A63AC048A18B59DA29",
  "version": 1,
  "network": 152,
  "type": 17230,
  "fee": "14500",
  "deadline": "111586858666",
  "namespace_id": "18182428759753405815",
  "mosaic_id": "507540378882865098",
  "alias_action": 1
}
Transaction hash: 67D18EB6A8F09B8C94AA96D7EB06AE9D2700A6457AAAD2037BAF5DFB9F51E7DF
Announcing mosaic alias transaction to /transactions
  Response: {"message":"packet 9 was pushed to the network via /transactions"}
Waiting for mosaic alias transaction confirmation...
  Transaction status: unconfirmed
  Transaction status: confirmed
mosaic alias transaction confirmed in 7 seconds
Fetching namespace information from /namespaces/fc54f69f52f2b577
Alias information:
  Alias type: 1
  Linked mosaic ID: 070B2549187AF7CA
Using alias in transfer: ns_1778514754.sub_1778514998
Mosaic ID (alias): 18182428759753405815 (0xfc54f69f52f2b577)
Test transaction:
{
  "signature": "B5EBF90EE702EEBF4788401B4508036CD1E46EE43E383AF0932B4F83B77533BDD0B94670F82A0EB4BCAEAE7987D535599D1D4914FB077D2A55EF52ABB6738906",
  "signer_public_key": "3B6A27BCCEB6A42D62A3A8D02A6F0D73653215771DE243A63AC048A18B59DA29",
  "version": 1,
  "network": 152,
  "type": 16724,
  "fee": "17600",
  "deadline": "111586858666",
  "recipient_address": "988E1191A25A88142C2FB3F69787576E3DC713EFC1CE4DE9",
  "mosaics": [
    {
      "mosaic_id": "18182428759753405815",
      "amount": "1"
    }
  ],
  "message": ""
}
Transaction hash: 631FAB32C1021560995B7619A0BCB92EA436CF525C3116C483B0A0B0E288D8B0
Announcing test transaction to /transactions
  Response: {"message":"packet 9 was pushed to the network via /transactions"}
Waiting for test transaction confirmation...
  Transaction status: unconfirmed
  Transaction status: confirmed
test transaction confirmed in 29 seconds

出力の主なポイント:

  • ネームスペースとターゲット (3、5行目): ネームスペース nsmos_1770541301 がターゲットのモザイクIDにリンクされています。

  • トランザクションハッシュ (23行目): トランザクションハッシュを使用して、 Symbol Testnet Explorer でトランザクションを検索できます。

  • エイリアスの検証 (32-33行目): ネームスペース情報により、エイリアスタイプが 1 (モザイク) であることが確認され、リンクされたモザイクIDが表示されています。

  • エイリアスの使用 (36行目): モザイク配列内でエイリアスを使用して転送トランザクションが作成されており、完全なモザイクIDの代わりに使用できることが実証されています。

    異なるモザイク ID

    転送で使用されているモザイクIDが元のモザイクIDと異なるのは、それがモザイクID自体ではなく エンコードされたネームスペース ID であるためです。 ネットワークはトランザクションを処理する際に、エイリアスをリンクされたモザイクに解決します。

結論⚓︎

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

ステップ 関連ドキュメント
ネームスペース ID を生成する
モザイクエイリアストランザクションを構築する
エイリアスを検証する /namespaces/{namespaceId} GET
転送内でエイリアスを使用する
モザイク解決レシートを照会する /statements/resolutions/mosaic GET