Source code for erc7730.convert.ledger.eip712.convert_erc7730_to_eip712

from typing import assert_never, final, override

from eip712.model.input.contract import InputEIP712Contract
from eip712.model.input.descriptor import InputEIP712DAppDescriptor
from eip712.model.input.message import InputEIP712Mapper, InputEIP712MapperField, InputEIP712Message
from eip712.model.schema import EIP712SchemaField
from eip712.model.types import EIP712Format

from erc7730.common.ledger import ledger_network_id
from erc7730.common.output import ExceptionsToOutput, OutputAdder
from erc7730.convert import ERC7730Converter
from erc7730.model.context import EIP712Schema
from erc7730.model.display import FieldFormat
from erc7730.model.paths import Array, ContainerField, ContainerPath, DataPath
from erc7730.model.paths.path_ops import data_path_concat, to_relative
from erc7730.model.resolved.context import ResolvedDeployment, ResolvedEIP712Context
from erc7730.model.resolved.descriptor import ResolvedERC7730Descriptor
from erc7730.model.resolved.display import (
    ResolvedField,
    ResolvedFieldDescription,
    ResolvedNestedFields,
    ResolvedTokenAmountParameters,
    ResolvedValueConstant,
    ResolvedValuePath,
)


[docs] @final class ERC7730toEIP712Converter(ERC7730Converter[ResolvedERC7730Descriptor, InputEIP712DAppDescriptor]): """ Converts ERC-7730 descriptor to Ledger legacy EIP-712 descriptor. Generates 1 output InputEIP712DAppDescriptor per chain id, as EIP-712 descriptors are chain-specific. """
[docs] @override def convert( self, descriptor: ResolvedERC7730Descriptor, out: OutputAdder ) -> dict[str, InputEIP712DAppDescriptor] | None: with ExceptionsToOutput(out): context = descriptor.context if not isinstance(context, ResolvedEIP712Context): return out.error("context is not EIP712") if (domain := context.eip712.domain) is None or (dapp_name := domain.name) is None: return out.error("EIP712 domain is not defined") if (contract_name := descriptor.metadata.owner) is None: return out.error("metadata.owner is not defined") messages: list[InputEIP712Message] = [] for primary_type, format in descriptor.display.formats.items(): schema = self._get_schema(primary_type, context.eip712.schemas, out) if schema is None: return out.error(f"EIP-712 schema for {primary_type} is missing") label = format.intent if isinstance(format.intent, str) else primary_type output_fields = [] for input_field in format.fields: if (out_field := self.convert_field(input_field, None, out)) is None: return None output_fields.extend(out_field) messages.append( InputEIP712Message(schema=schema, mapper=InputEIP712Mapper(label=label, fields=output_fields)) ) descriptors: dict[str, InputEIP712DAppDescriptor] = {} for deployment in context.eip712.deployments: output_descriptor = self._build_network_descriptor(deployment, dapp_name, contract_name, messages, out) if output_descriptor is not None: descriptors[str(deployment.chainId)] = output_descriptor return descriptors
@classmethod def _build_network_descriptor( cls, deployment: ResolvedDeployment, dapp_name: str, contract_name: str, messages: list[InputEIP712Message], out: OutputAdder, ) -> InputEIP712DAppDescriptor | None: if (network := ledger_network_id(deployment.chainId)) is None: return out.error(f"network id {deployment.chainId} not supported") return InputEIP712DAppDescriptor( blockchainName=network, chainId=deployment.chainId, name=dapp_name, contracts=[ InputEIP712Contract(address=deployment.address.lower(), contractName=contract_name, messages=messages) ], ) @classmethod def _get_schema( cls, primary_type: str, schemas: list[EIP712Schema], out: OutputAdder ) -> dict[str, list[EIP712SchemaField]] | None: for schema in schemas: if schema.primaryType == primary_type: return schema.types return out.error(f"schema for type {primary_type} not found")
[docs] @classmethod def convert_field( cls, field: ResolvedField, prefix: DataPath | None, out: OutputAdder ) -> list[InputEIP712MapperField] | None: match field: case ResolvedFieldDescription(): if (output_field := cls.convert_field_description(field, prefix, out)) is None: return None return [output_field] case ResolvedNestedFields(): output_fields = [] for in_field in field.fields: if (output_field := cls.convert_field(in_field, prefix, out)) is None: return None output_fields.extend(output_field) return output_fields case _: assert_never(field)
[docs] @classmethod def convert_field_description( cls, field: ResolvedFieldDescription, prefix: DataPath | None, out: OutputAdder, ) -> InputEIP712MapperField | None: field_path: DataPath asset_path: DataPath | None = None field_format: EIP712Format | None = None in_array: bool = False match field.value: case ResolvedValueConstant(): return out.error("Constant values are not supported") case ResolvedValuePath(path=path): match path: case DataPath() as field_path: field_path = data_path_concat(prefix, field_path) for element in field_path.elements: match element: case Array(): in_array = True break case _: pass case ContainerPath() as container_path: return out.error(f"Path {container_path} is not supported") case _: assert_never(field.value) case _: assert_never(field.value) match field.format: case None: field_format = None case FieldFormat.ADDRESS_NAME: field_format = EIP712Format.RAW case FieldFormat.RAW: field_format = EIP712Format.RAW case FieldFormat.ENUM: field_format = EIP712Format.RAW case FieldFormat.UNIT: field_format = EIP712Format.RAW case FieldFormat.DURATION: field_format = EIP712Format.RAW case FieldFormat.NFT_NAME: field_format = EIP712Format.RAW case FieldFormat.CALL_DATA: field_format = EIP712Format.RAW case FieldFormat.DATE: field_format = EIP712Format.DATETIME case FieldFormat.AMOUNT: field_format = EIP712Format.AMOUNT case FieldFormat.TOKEN_AMOUNT: if in_array: # EIP-712 does not support token references in arrays, fallback to raw format field_format = EIP712Format.RAW else: field_format = EIP712Format.AMOUNT if field.params is not None and isinstance(field.params, ResolvedTokenAmountParameters): match field.params.token: case None: return out.error("Token path or reference must be set") case ResolvedValueConstant(): return out.error("Constant values are not supported") case ResolvedValuePath(path=path): match path: case None: pass case DataPath() as token_path: asset_path = data_path_concat(prefix, token_path) case ContainerPath() as container_path if container_path.field == ContainerField.TO: # In EIP-712 protocol, format=token with no token path # => refers to verifyingContract asset_path = None case ContainerPath() as container_path: return out.error(f"Path {container_path} is not supported") case _: assert_never(path) case _: assert_never(field.value) case _: assert_never(field.format) return InputEIP712MapperField( path=str(to_relative(field_path)), label=field.label, assetPath=None if asset_path is None else str(to_relative(asset_path)), format=field_format, )