7
6

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 1 year has passed since last update.

[ERC5453] 1トランザクション内で複数関数呼び出しや署名関連拡張機能の仕組みを理解しよう!

Last updated at Posted at 2023-10-18

はじめに

初めまして。
CryptoGamesというブロックチェーンゲーム企業でエンジニアをしている cardene(かるでね) です!
スマートコントラクトを書いたり、フロントエンド・バックエンド・インフラと幅広く触れています。

代表的なゲームはクリプトスペルズというブロックチェーンゲームです。

今回は、関数の動作を拡張する規格であるERC5750の機能に依存し、1トランザクション内で複数関数呼び出しなどの仕組みを提案している規格であるERC5453についてまとめていきます!

以下にまとめられているものを翻訳・要約・補足しながらまとめていきます。

5453は現在(2023年10月17日)では「Last Call」段階です。

他にも様々なERCについてまとめています。

概要

この提案は、Ethereumの改善提案(EIP)で、ERC5750を基盤としています。
その主な目的は、同じトランザクション内で複数の関数呼び出しを承認できる一般的なプロトコルを設けることです。
これは、従来のアプローチとは異なります。
以前のアプローチ(例えば、ERC20用のERC2612ERC721用のERC4494)では、通常、単一の動作(ERC20の場合はtransferERC721の場合はsafeTransferFrom)しか許可せず、単一の承認者が関与するため、2つのトランザクションが必要でした(最初にpermit(...)トランザクション、次に転送のようなトランザクション)。

ERC5750については以下を参考にしてください。

ERC20については以下を参考にしてください。

ERC721については以下を参考にしてください。

しかし、このEIPは、異なる動作を許可し、同じトランザクション内で任意の数の承認者からの複数の承認を一括して受け入れる方法を提供します。
これにより、マルチシグ(複数の署名が必要な契約)や閾値署名(特定の条件を満たす必要がある署名)など、多くのシナリオで役立ちます。

このプロトコルは、特定のコントラクト内の特定の関数呼び出しに対して、同じトランザクション内で複数の承認を受け入れることができます。
これにより、以前は2つのトランザクションが必要だった手続きが、1つのトランザクションで完了できるようになり、効率が向上します。
特に、コントラクトが特定のロールを持つ複数の承認者から、異なる操作を同じトランザクションで許可できるため、柔軟性とセキュリティが向上します。

動機

このプロトコルは、次の機能をサポートします。

同時承認と関数呼び出しのサポート

ユーザーはコントラクト内の特定の操作を承認しながら、同じトランザクション内でその操作を実行できます。
これにより、複数の操作を1つのトランザクションで実行できます。

別のユーザーからの追加の承認サポート

別のユーザーからの追加の承認を受け入れることができます。
つまり、複数のユーザーが同じトランザクションに対して承認を行えます。

別のユーザーによる支払いのサポート

別のユーザーがトランザクションのガス料金を支払うことができます。
これにより、トランザクションのコストを複数のユーザーで分担できます。

マルチシグのサポート

複数の承認者が必要な場合、マルチシグの機能を提供します。
複数のユーザーが協力してコントラクトの操作を承認できます。

共同行動のサポート

複数のユーザーが連携してトランザクションを実行できます。
複数の承認者が協力して、特定の操作を実行できます。

累積投票のサポート

承認を累積し、複数の承認を合算できます。
これにより、複数の承認が必要な場合に、それらをまとめて処理できます。

オフライン署名のサポート

オフラインで署名を生成し、後でトランザクションに追加できます。
これにより、セキュリティと柔軟性が向上し、インターネット接続が必要ない環境でもトランザクションを実行できます。

このプロトコルは、コントラクトの操作をより柔軟に、セキュアに、効率的に管理するための多くの機能を提供します。

仕様

インターフェース

ここで参照されるインターフェースと構造は以下になります。

pragma solidity ^0.8.9;

struct ValidityBound {
    bytes32 functionParamStructHash;
    uint256 validSince;
    uint256 validBy;
    uint256 nonce;
}

struct SingleEndorsementData {
    address endorserAddress; // 32
    bytes sig; // dynamic = 65
}

struct GeneralExtensionDataStruct {
    bytes32 erc5453MagicWord;
    uint256 erc5453Type;
    uint256 nonce;
    uint256 validSince;
    uint256 validBy;
    bytes endorsementPayload;
}

interface IERC5453EndorsementCore {
    function eip5453Nonce(address endorser) external view returns (uint256);
    function isEligibleEndorser(address endorser) external view returns (bool);
}

interface IERC5453EndorsementDigest {
    function computeValidityDigest(
        bytes32 _functionParamStructHash,
        uint256 _validSince,
        uint256 _validBy,
        uint256 _nonce
    ) external view returns (bytes32);

    function computeFunctionParamHash(
        string memory _functionName,
        bytes memory _functionParamPacked
    ) external view returns (bytes32);
}

interface IERC5453EndorsementDataTypeA {
    function computeExtensionDataTypeA(
        uint256 nonce,
        uint256 validSince,
        uint256 validBy,
        address endorserAddress,
        bytes calldata sig
    ) external view returns (bytes memory);
}


interface IERC5453EndorsementDataTypeB {
    function computeExtensionDataTypeB(
        uint256 nonce,
        uint256 validSince,
        uint256 validBy,
        address[] calldata endorserAddress,
        bytes[] calldata sigs
    ) external view returns (bytes memory);
}

ValidityBound

struct ValidityBound {
    bytes32 functionParamStructHash;
    uint256 validSince;
    uint256 validBy;
    uint256 nonce;
}

概要
スマートコントラクトの呼び出しの有効性に関する情報を格納する構造体。

パラメータ

  • functionParamStructHash
    • 関数呼び出しのパラメータ構造のハッシュ値。
  • validSince
    • 有効期間の開始を示すタイムスタンプ(Unix時間)。
  • validBy
    • 有効期間の終了を示すタイムスタンプ(Unix時間)。
  • nonce
    • ユニークな識別子として使用される数値。

SingleEndorsementData

struct SingleEndorsementData {
    address endorserAddress;
    bytes sig;
}

概要
単一の承認データを格納する構造体。

パラメータ

  • endorserAddress
    • 承認者のアドレス。
  • sig
    • 署名データを格納する可変長のバイト列(通常は65バイト)。

GeneralExtensionDataStruct

struct GeneralExtensionDataStruct {
    bytes32 erc5453MagicWord;
    uint256 erc5453Type;
    uint256 nonce;
    uint256 validSince;
    uint256 validBy;
    bytes endorsementPayload;
}

概要
一般的な拡張データを格納する構造体。
ERC5453に関連する情報を含みます。

パラメータ

  • erc5453MagicWord
    • ERC5453プロトコルの識別子として使用される特定のハッシュ値。
  • erc5453Type
    • ERC5453のタイプを示す数値。
  • nonce
    • ユニークな識別子として使用される数値。
  • validSince
    • 有効期間の開始を示すタイムスタンプ(Unix時間)。
  • validBy
    • 有効期間の終了を示すタイムスタンプ(Unix時間)。
  • endorsementPayload
    • 拡張データのペイロードを格納する可変長のバイト列。

IERC5453EndorsementCore

eip5453Nonce

function eip5453Nonce(address endorser) external view returns (uint256);

概要
指定された承認者のユニークな識別子(nonce)を取得する関数。

詳細
指定された endorserアドレスに関連付けられたユニークな識別子(nonce)を返します。
ユニークなnonceは、承認者が特定の操作を実行するたびにインクリメントされる一意の値です。

引数

  • endorser
    • ユーザーまたはアカウントのアドレス。

戻り値

  • nonce
    • 承認者のユニークな識別子。

isEligibleEndorser

function isEligibleEndorser(address endorser) external view returns (bool);

概要
指定されたアドレスが有効な承認者であるか確認する関数。

詳細
指定された endorserアドレスが有効な承認者である場合、trueを返します。
有効な承認者は、特定の操作やトランザクションに署名する権限を持つアドレスです。

引数

  • endorser
    • 承認者のアドレス。

戻り値

  • isEligible
    • 指定されたアドレスが有効な承認者である場合はtrue、それ以外の場合はfalse

IERC5453EndorsementDigest

computeValidityDigest

function computeValidityDigest(
    bytes32 _functionParamStructHash,
    uint256 _validSince,
    uint256 _validBy,
    uint256 _nonce
) external view returns (bytes32);

概要
承認情報の有効性を確認するためのダイジェスト(digest)を計算する関数。

詳細
指定された _functionParamStructHash(関数パラメータのハッシュ)、_validSince(有効期限の開始)、_validBy(有効期限の終了)、_nonce(ユニークな識別子)を使用して、承認情報の有効性を確認するためのダイジェストを計算します。
ダイジェストは、特定の承認情報の一意の識別子であり、署名の検証に使用されます。

引数

  • _functionParamStructHash
    • 関数パラメータのハッシュ値。
  • _validSince
    • 有効期限の開始時刻。
  • _validBy
    • 有効期限の終了時刻。
  • _nonce
    • ユニークな識別子。

戻り値

  • digest
    • 承認情報の有効性を確認するためのダイジェスト。

computeFunctionParamHash

function computeFunctionParamHash(
    string memory _functionName,
    bytes memory _functionParamPacked
) external view returns (bytes32);

概要
関数名とパラメータのパックされたバイト列から関数パラメータのハッシュを計算する関数。

詳細
_functionName_functionParamPacked(パラメータのパックされたバイト列)を使用して、関数パラメータのハッシュを計算します
このハッシュは、関数パラメータの一意の識別子として使用され、承認情報の生成および検証に関与します。

引数

  • _functionName
    • 関数名の文字列。
  • _functionParamPacked
    • パラメータのパックされたバイト列。

戻り値

  • paramHash
    • 関数パラメータのハッシュ。

IERC5453EndorsementDataTypeA

computeExtensionDataTypeA

function computeExtensionDataTypeA(
    uint256 nonce,
    uint256 validSince,
    uint256 validBy,
    address endorserAddress,
    bytes calldata sig
) external view returns (bytes memory);

概要

与えられた引数を使用して拡張データのデータタイプAを計算する関数。

詳細

コントラクト内でデータタイプAを計算するために使用されます。
拡張データタイプAは、ユーザーに関する情報を保持し、署名されたトランザクションの有効性を確認するのに役立ちます。
この関数は、特定の引数を取り、それらを使用してデータタイプAを生成し、その結果を返します。
これにより、トランザクションが有効であるかどうかを検証できます。

引数

  • nonce
    • トランザクションのnonce値。
    • 一意のトランザクション識別子。
  • validSince
    • トランザクションの有効開始時刻。
  • validBy
    • トランザクションの有効終了時刻。
  • endorserAddress
    • 署名者のアドレス。トランザクションを署名したアカウントのアドレス。
  • sig
    • トランザクションの署名データ。トランザクションの正当性を確認するために使用される署名。

戻り値

計算された拡張データタイプAをbytesデータ型で返します。


IERC5453EndorsementDataTypeB

computeExtensionDataTypeB

function computeExtensionDataTypeB(
    uint256 nonce,
    uint256 validSince,
    uint256 validBy,
    address[] calldata endorserAddress,
    bytes[] calldata sigs
) external view returns (bytes memory);

概要

与えられた引数を使用して拡張データのデータタイプBを計算する関数。

詳細

コントラクト内でデータタイプBを計算するために使用されます。
拡張データタイプBは、複数の署名者によって署名されたトランザクションの有効性を確認するのに役立ちます。
この関数は、特定の引数を取り、それらを使用してデータタイプBを生成し、その結果を返します。
これにより、トランザクションが有効であるかどうかを検証できます。

引数

  • nonce
    • トランザクションのnonce値。
    • 一意のトランザクション識別子。
  • validSince
    • トランザクションの有効開始時刻。
  • validBy
    • トランザクションの有効終了時刻。
  • endorserAddress
    • 署名者のアドレスの配列。トランザクションを署名したアカウントのアドレスを含む。
  • sigs
    • トランザクションの署名データの配列。各署名者の署名データを含む。

戻り値

計算された拡張データタイプBをbytesデータ型で返します。


動作仕様

ERC5750の「メソッド動作の一般的な拡張性」では、特定のユーザーからの許可を示すための方法として、最後のメソッドとしてbytes extraDataを持つ準拠メソッドがERC5453に従うことができると指定されています。

このEIPに準拠するメソッドは、必ずERC5750に準拠したメソッドである必要があります。
呼び出し元は、最後のパラメータとして、Section Interfacesで指定されているGeneralExtensonDataStructのソリッドステートエンコードされたレイアウトを持つbytes extraDataを提供する必要があります。
以下の説明は、bytes extraDataをGeneralExtensonDataStructにデコードする場合に基づいています。

GeneralExtensonDataStructにデコードされたextraData内で、呼び出し元はGeneralExtensonDataStruct.erc5453MagicWordの値をkeccak256("ERC5453-ENDORSEMENT")に設定する必要があります。
呼び出し元は、GeneralExtensonDataStruct.erc5453Typeの値をサポートされている値の1つに設定する必要があります。

このEIPはERC5750に従うメソッドに対して、特定のユーザーからの許可を示す方法を提供します。
呼び出し元は、extraData内に特定の情報を設定し、それをデコードして正しい許可情報を提供する必要があります。
これにより、コントラクトは異なるメソッドを拡張し、許可情報を効果的に管理できるようになります。

uint256 constant ERC5453_TYPE_A = 1;
uint256 constant ERC5453_TYPE_B = 2;

GeneralExtensonDataStruct.erc5453Typeの値がERC5453_TYPE_Aの場合、GeneralExtensonDataStruct.endorsementPayloadSingleEndorsementDataのABIエンコードされたバイト列である必要があります。
一方、GeneralExtensonDataStruct.erc5453Typeの値がERC5453_TYPE_Bの場合、GeneralExtensonDataStruct.endorsementPayloadSingleEndorsementDataの可変長配列のABIエンコードされたバイト列である必要があります。

SingleEndorsementDataには、2つの要素が含まれています。

  • endorserAddress
    • このフィールドはアドレスであり、署名を行った承認者のアドレスを表します。
  • sig
    • このフィールドは65バイトの署名で、ECDSA(secp256k1)署名です。
    • この署名はendorserAddressに対応する署名者のプライベートキーを使用して生成されます。

それぞれのsig署名は、validityDigestと呼ばれるハッシュに対して作成されます。
このハッシュはValidityBoundデータ構造体のhashStructに基づいてEIP712hashTypeDataV4を使用して計算されます。

ERC5453では、特定のユーザーからの許可情報を持つ際に、署名者のアドレスと署名を含むSingleEndorsementDataを使用し、それをGeneralExtensonDataStruct内にABIエンコードされた形式で格納します。
これにより、許可情報が正確に構造化され、検証されることが保証されます。

bytes32 validityDigest =
    eip712HashTypedDataV4(
        keccak256(
            abi.encode(
                keccak256(
                    "ValidityBound(bytes32 functionParamStructHash,uint256 validSince,uint256 validBy,uint256 nonce)"
                ),
                functionParamStructHash,
                _validSince,
                _validBy,
                _nonce
            )
        )
    );

functionParamStructHashは次のように計算されます。

bytes32 functionParamStructHash = keccak256(
    abi.encodePacked(
        keccak256(bytes(_functionStructure)),
        _functionParamPacked
    )
);
return functionParamStructHash;

_functionStructureは、関数のmethodName(type1 param1, type2 param2, ...)の形式で計算されます。
これは、関数名とそのパラメータの型を表します。
例として以下のようになります。

transfer(address to, uint256 amount)

この構造は関数のシグネチャを表します。

_functionParamPackedは、関数の各パラメータをABIエンコードし、それらを連結したバイト列です。
例えば、関数に対するパラメータが(address, uint256)である場合、このフィールドはパラメータをABIエンコードしたバイト列になります。

endorserAddressがecrecover(validityDigest, signature)またはEIP1271(endorserAddress).isValidSignature(validityDigest, signature) == ERC1271.MAGICVALUEと一致する場合、単一の承認は有効と見なされます。
つまり、署名が正当であることが確認されます。

準拠メソッドは、同じERC5453_TYPE_BendorsementPayload内で有効な承認の数に対して閾値を設定できます。
つまり、複数の承認が必要な場合、その数を制限することができます。

validSincevalidByは、承認の有効期間を指定します。
これらはブロック番号またはタイムスタンプのどちらかで指定できます。
実装者は、有効期間がブロック番号またはタイムスタンプであるかどうかを明示する方法を見つける必要があります。
これにより、有効期間が明確に設定され、適切に検証できます。

ERC5453では、関数の構造、パラメータ、承認情報の検証に特定の規則が適用されます。
これにより、セキュアで信頼性の高い承認プロセスを実装することができます。

補足

我々はERC5453プロトコルにおいて、2つの異なるタイプ、つまりERC5453_TYPE_A(単一の承認)とERC5453_TYPE_B(複数の承認、コントラクト全体で同じnonce)を選択しました。
これを採用することで、さまざまなユースケースに対応できます。

まず、ERC5453_TYPE_Aでは、単一の承認が行われます。
これは、特定のユーザーからの個別の承認を表します。
このタイプは、ERC2612ERC4494などの既存のプロトコルでサポートされるユースケースに適しています。

次に、ERC5453_TYPE_Bでは、複数の承認が同じnonceを共有し、コントラクト全体で協力して行われます。
これにより、閾値承認など、より複雑な承認パターンを実現できます。
また、新しいERC5453_TYPE_を定義することで、さらに多様な承認タイプをサポートできます。

また、有効期限に関して、validSincevalidByの両方を含めました。
これにより、承認の有効期間を柔軟に設定できます。
ERC5081が採用されない可能性が高いため、プロトコル内にこれらの2つの数値を追加し、スマートコントラクトレベルで有効期限をサポートできるようにしました。

ERC5453プロトコルは、さまざまな承認タイプと有効期限の柔軟性を提供し、異なるユースケースに対応できるように設計されています。

後方互換性

この設計では、将来の変更や拡張性を最大限に活用するために、bytes calldata extraDataを前提としています。
このextraDataは、コントラクトの関数呼び出し時に追加されるデータフィールドであり、将来の変更や機能追加に対応するための柔軟性を提供します。

このデザインの優れた点は、既存のERC規格に準拠している多くのコントラクトと互換性があることです。
たとえば、ERC721ERC1155など、すでに存在する規格に従っているコントラクトは、extraDataを活用して新しい機能を容易に統合できます。

さらに、このデザインに対応していないコントラクト(例:ERC20)も、ラッパーコントラクトやプロキシのアップグレードなどを介して、extraDataのサポートを後から追加することができます。
これにより、多くの異なるコントラクトが将来の変更に対応し、新しい機能を取り入れることができます。

extraDataを使用することで、コントラクトは将来の変更や拡張に対してオープンで柔軟なアプローチを提供し、様々なERC規格に適用できるようになります。

参考実装

特定の署名者による署名の妥当性を確認するためのアルゴリズムに加えて、以下のリファレンス実装も提供されています。

pragma solidity ^0.8.9;

import "@openzeppelin/contracts/utils/cryptography/SignatureChecker.sol";
import "@openzeppelin/contracts/utils/cryptography/EIP712.sol";

import "./IERC5453.sol";

abstract contract AERC5453Endorsible is EIP712,
    IERC5453EndorsementCore, IERC5453EndorsementDigest, IERC5453EndorsementDataTypeA, IERC5453EndorsementDataTypeB {
    // ...

    function _validate(
        bytes32 msgDigest,
        SingleEndorsementData memory endersement
    ) internal virtual {
        require(
            endersement.sig.length == 65,
            "AERC5453Endorsible: wrong signature length"
        );
        require(
            SignatureChecker.isValidSignatureNow(
                endersement.endorserAddress,
                msgDigest,
                endersement.sig
            ),
            "AERC5453Endorsible: invalid signature"
        );
    }
    // ...

    modifier onlyEndorsed(
        bytes32 _functionParamStructHash,
        bytes calldata _extensionData
    ) {
        require(_isEndorsed(_functionParamStructHash, _extensionData));
        _;
    }

    function computeExtensionDataTypeB(
        uint256 nonce,
        uint256 validSince,
        uint256 validBy,
        address[] calldata endorserAddress,
        bytes[] calldata sigs
    ) external pure override returns (bytes memory) {
        require(endorserAddress.length == sigs.length);
        SingleEndorsementData[]
            memory endorsements = new SingleEndorsementData[](
                endorserAddress.length
            );
        for (uint256 i = 0; i < endorserAddress.length; ++i) {
            endorsements[i] = SingleEndorsementData(
                endorserAddress[i],
                sigs[i]
            );
        }
        return
            abi.encode(
                GeneralExtensionDataStruct(
                    MAGIC_WORLD,
                    ERC5453_TYPE_B,
                    nonce,
                    validSince,
                    validBy,
                    abi.encode(endorsements)
                )
            );
    }
}

_validate

function _validate(
    bytes32 msgDigest,
    SingleEndorsementData memory endersement
) internal virtual {
    require(
        endersement.sig.length == 65,
        "AERC5453Endorsible: wrong signature length"
    );
    require(
        SignatureChecker.isValidSignatureNow(
            endersement.endorserAddress,
            msgDigest,
            endersement.sig
        ),
        "AERC5453Endorsible: invalid signature"
    );
}

概要

与えられたメッセージダイジェストと単一のエンドースメントデータを検証する関数。

詳細

AERC5453Endorsibleコントラクト内で使用され、単一のエンドースメントデータが有効であるかどうかを検証します。
エンドースメントデータには、署名者のアドレスと署名データが含まれています。
検証は、署名の長さが正しいことと、与えられたメッセージダイジェストに対して署名が有効であることを確認します。

引数

  • msgDigest
    • メッセージダイジェスト。
    • 検証対象のメッセージのハッシュ値。
  • endersement
    • 単一のエンドースメントデータを表す構造体。
    • 署名者のアドレスと署名データが含まれています。

onlyEndorsed

modifier onlyEndorsed(
    bytes32 _functionParamStructHash,
    bytes calldata _extensionData
) {
    require(_isEndorsed(_functionParamStructHash, _extensionData));
    _;
}

概要

指定された関数がエンドースメントされている場合にのみ実行を許可する修飾子。

詳細

関数が呼び出される前に_isEndorsed関数を使用してエンドースメントの有効性を検証します。
エンドースメントが有効である場合、関数の実行が許可されます。
エンドースメントが無効である場合、関数の呼び出しは失敗します。

引数

  • _functionParamStructHash
    • 関数呼び出しのパラメータ構造体のハッシュ値。
  • _extensionData
    • 拡張データ。

computeExtensionDataTypeB

function computeExtensionDataTypeB(
    uint256 nonce,
    uint256 validSince,
    uint256 validBy,
    address[] calldata endorserAddress,
    bytes[] calldata sigs
) external pure override returns (bytes memory) {
    require(endorserAddress.length == sigs.length);
    SingleEndorsementData[] memory endorsements = new SingleEndorsementData[](endorserAddress.length);
    for (uint256 i = 0; i < endorserAddress.length; ++i) {
        endorsements[i] = SingleEndorsementData(
            endorserAddress[i],
            sigs[i]
        );
    }
    return abi.encode(
        GeneralExtensionDataStruct(
            MAGIC_WORLD,
            ERC5453_TYPE_B,
            nonce,
            validSince,
            validBy,
            abi.encode(endorsements)
        )
    );
}

概要

拡張データのデータタイプBを計算する関数。

詳細

指定された引数を使用してデータタイプBを生成します。
データタイプBには、ノンス、有効期間、署名者のアドレス、および署名データが含まれています。
これらの情報を使用してデータタイプBを生成し、それをバイトデータにエンコードして返します。

引数

  • nonce
    • トランザクションのnonce値。
    • 一意のトランザクション識別子。
  • validSince
    • トランザクションの有効開始時刻。
  • validBy
    • トランザクションの有効終了時刻。
  • endorserAddress
    • 署名者のアドレスの配列。
    • トランザクションを署名したアカウントのアドレスを含む。
  • sigs
    • トランザクションの署名データの配列。各署名者の署名データを含む。

戻り値

この関数は計算されたデータタイプBをbytesデータ型で返します。
データタイプBには、エンドースメントデータが含まれており、他のコントラクトで使用できる形式にエンコードされています。


EndorsableERC721の参考実装

以下は、ERC4494と同様の動作を実現するEndorsableERC721の参考実装です。

pragma solidity ^0.8.9;

contract EndorsableERC721 is ERC721, AERC5453Endorsible {
    //...

    function mint(
        address _to,
        uint256 _tokenId,
        bytes calldata _extraData
    )
        external
        onlyEndorsed(
            _computeFunctionParamHash(
                "function mint(address _to,uint256 _tokenId)",
                abi.encode(_to, _tokenId)
            ),
            _extraData
        )
    {
        _mint(_to, _tokenId);
    }
}

mint

function mint(
    address _to,
    uint256 _tokenId,
    bytes calldata _extraData
)
    external
    onlyEndorsed(
        _computeFunctionParamHash(
            "function mint(address _to,uint256 _tokenId)",
            abi.encode(_to, _tokenId)
        ),
        _extraData
    )
{
    _mint(_to, _tokenId);
}

概要

新しいトークンを発行して_toアドレスに割り当てる関数。

詳細

ERC721コントラクトに新しいトークンを発行し、指定されたアドレス_toに割り当てます。
また、この関数はエンドースメントを要求し、指定された_extraDataを検証します。
エンドースメントが有効である場合、新しいトークンが_toに発行されます。

引数

  • _to
    • トークンを割り当てるアドレス。
  • _tokenId
    • 新しいトークンのID。
  • _extraData
    • エンドースメントの追加データ。

ThresholdMultiSigForwarderの参考実装

ThresholdMultiSigForwarder(閾値マルチシグフォワーダー)の参考実装です。
この実装は、**Gnosis-Safeウォレットのように、複数の承認者が協力してリモートコントラクトの呼び出しを行うためのものです。

通常、このコントラクトを使用する場合、特定のトランザクションを実行するためには、ある条件(閾値)を満たす必要があります。
例えば、3人の承認者が必要な場合、3人の承認者がトランザクションに署名する必要があります。
一定数の承認者が署名した場合、トランザクションは実行されます。

この仕組みにより、重要なトランザクションや取引の実行において、複数の信頼性のある承認が必要な場合にセキュリティを向上させることができます。
つまり、一つの個人やエンティティだけでなく、複数の関係者が同意を示す必要があります。
これにより、トランザクションの不正使用や誤操作を防ぎ、信頼性とセキュリティを高めることができます。

pragma solidity ^0.8.9;

contract ThresholdMultiSigForwarder is AERC5453Endorsible {
    //...
    function forward(
        address _dest,
        uint256 _value,
        uint256 _gasLimit,
        bytes calldata _calldata,
        bytes calldata _extraData
    )
        external
        onlyEndorsed(
            _computeFunctionParamHash(
                "function forward(address _dest,uint256 _value,uint256 _gasLimit,bytes calldata _calldata)",
                abi.encode(_dest, _value, _gasLimit, keccak256(_calldata))
            ),
            _extraData
        )
    {
        string memory errorMessage = "Fail to call remote contract";
        (bool success, bytes memory returndata) = _dest.call{value: _value}(
            _calldata
        );
        Address.verifyCallResult(success, returndata, errorMessage);
    }

}

forward

function forward(
    address _dest,
    uint256 _value,
    uint256 _gasLimit,
    bytes calldata _calldata,
    bytes calldata _extraData
)
    external
    onlyEndorsed(
        _computeFunctionParamHash(
            "function forward(address _dest,uint256 _value,uint256 _gasLimit,bytes calldata _calldata)",
            abi.encode(_dest, _value, _gasLimit, keccak256(_calldata))
        ),
        _extraData
    )
{
    string memory errorMessage = "Fail to call remote contract";
    (bool success, bytes memory returndata) = _dest.call{value: _value}(
        _calldata
    );
    Address.verifyCallResult(success, returndata, errorMessage);
}

概要

指定されたアドレスにトークンとデータを転送し、外部コントラクトを呼び出す関数。

詳細

指定されたアドレス_destに対して指定されたトークン_valueとデータ_calldataを転送し、外部コントラクトを呼び出します。
エンドースメントを要求し、指定された_extraDataを検証します。
呼び出し結果がエラーの場合、エラーメッセージがスローされます。

引数

  • _dest
    • 転送先のアドレス。
  • _value
    • トークンの転送量。
  • _gasLimit
    • ガス制限。
  • _calldata
    • 呼び出しデータ。
  • _extraData
    • エンドースメントの追加データ。

セキュリティ考慮事項

リプレイ攻撃

リプレイ攻撃は、暗号認証に対する一種の攻撃です。
具体的には、既存の署名を再利用して、同じメッセージに対する署名の検証を回避する攻撃です。
この攻撃は、このEIPに依存する実装に影響を与える可能性があります。
なぜなら、ここで説明されているすべてのスマートエンドースメントは、誰でも取得できる公開の暗号的署名であるためです。
実装者は、リプレイ攻撃が同じスマートコントラクトの異なるデプロイメントや、類似のスマートコントラクトのデプロイメント、または別のchainIdでの同じコントラクトのバージョンなど、さまざまな攻撃シナリオで発生する可能性を考慮する必要があります。
noncevalidSincevalidByフィールドは攻撃の表面を制限するために使用できますが、すべての攻撃リスクを完全に排除することは難しいかもしれません。

フィッシング攻撃

フィッシング(詐欺)攻撃にも注意が必要です。
フィッシング攻撃では、悪意のある攻撃者が、ユーザーを騙して、正当なように見せかけたスマートエンドースメントに署名させることがあります。
しかし、実際にはそのデータは悪意のあるアプリケーションに一致しています。

リプレイ攻撃は暗号認証を迂回する攻撃であり、フィッシング攻撃はユーザーを欺いて悪意のあるスマートコントラクトに署名させる攻撃です。
実装者はこれらの攻撃に対処するための対策を講じる必要があります。

引用

Zainan Victor Zhou (@xinbenlv), "ERC-5453: Endorsement - Permit for Any Functions [DRAFT]," Ethereum Improvement Proposals, no. 5453, August 2022. [Online serial]. Available: https://eips.ethereum.org/EIPS/eip-5453.

最後に

今回は「関数の動作を拡張する規格であるERC5750の機能に依存し、1トランザクション内で複数関数呼び出しなどの仕組みを提案している規格であるERC5453」についてまとめてきました!
いかがだったでしょうか?

質問などがある方は以下のTwitterのDMなどからお気軽に質問してください!

Twitter @cardene777

他の媒体でも情報発信しているのでぜひ他も見ていってください!

7
6
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
7
6

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?