Source code for erc7730.model.paths.path_schemas

from dataclasses import dataclass
from typing import assert_never

from eip712.model.schema import EIP712SchemaField

from erc7730.model.abi import Component, Function, InputOutput
from erc7730.model.context import EIP712Schema
from erc7730.model.paths import (
    ROOT_DATA_PATH,
    Array,
    ArrayElement,
    ArraySlice,
    ContainerPath,
    DataPath,
    DataPathElement,
    Field,
)
from erc7730.model.paths.path_ops import data_path_append
from erc7730.model.resolved.display import (
    ResolvedAddressNameParameters,
    ResolvedCallDataParameters,
    ResolvedDateParameters,
    ResolvedEnumParameters,
    ResolvedField,
    ResolvedFieldDescription,
    ResolvedFormat,
    ResolvedNestedFields,
    ResolvedNftNameParameters,
    ResolvedTokenAmountParameters,
    ResolvedUnitParameters,
    ResolvedValue,
    ResolvedValueConstant,
    ResolvedValuePath,
)
from erc7730.model.resolved.path import ResolvedPath


[docs] @dataclass(kw_only=True, frozen=True) class FormatPaths: data_paths: set[DataPath] # References to values in the serialized data container_paths: set[ContainerPath] # References to values in the container
[docs] def compute_eip712_schema_paths(schema: EIP712Schema) -> set[DataPath]: """ Compute the sets of valid schema paths for an EIP-712 schema. :param schema: EIP-712 schema :return: valid schema paths """ if (primary_type := schema.types.get(schema.primaryType)) is None: raise ValueError(f"Invalid schema: primaryType {schema.primaryType} not in types") paths: set[DataPath] = set() def append_paths(path: DataPath, current_type: list[EIP712SchemaField]) -> None: for field in current_type: if len(field.name) == 0: continue # skip unnamed parameters sub_path = data_path_append(path, Field(identifier=field.name)) field_base_type = field.type.rstrip("[]") if field_base_type in {"bytes"}: paths.add(data_path_append(sub_path, Array())) if field_base_type != field.type: sub_path = data_path_append(sub_path, Array()) paths.add(sub_path) if (target_type := schema.types.get(field_base_type)) is not None: append_paths(sub_path, target_type) else: paths.add(sub_path) append_paths(ROOT_DATA_PATH, primary_type) return paths
[docs] def compute_abi_schema_paths(abi: Function) -> set[DataPath]: """ Compute the sets of valid schema paths for an ABI function. :param abi: Solidity ABI function :return: valid schema paths """ paths: set[DataPath] = set() def append_paths(path: DataPath, params: list[InputOutput] | list[Component] | None) -> None: if not params: return None for param in params: if len(param.name) == 0: continue # skip unnamed parameters sub_path = data_path_append(path, Field(identifier=param.name)) param_base_type = param.type.rstrip("[]") if param_base_type in {"bytes"}: paths.add(data_path_append(sub_path, Array())) if param_base_type != param.type: sub_path = data_path_append(sub_path, Array()) paths.add(sub_path) if param.components: append_paths(sub_path, param.components) # type: ignore else: paths.add(sub_path) append_paths(ROOT_DATA_PATH, abi.inputs) return paths
[docs] def compute_format_schema_paths(format: ResolvedFormat) -> FormatPaths: """ Compute the sets of schema paths referred in an ERC7730 Format section. :param format: resolved $.display.format section :return: schema paths used by field formats """ data_paths: set[DataPath] = set() # references to values in the serialized data container_paths: set[ContainerPath] = set() # references to values in the container if format.fields is not None: def add_path(path: ResolvedPath | None) -> None: match path: case None: pass case ContainerPath(): container_paths.add(path) case DataPath(): data_paths.add(data_path_to_schema_path(path)) case _: assert_never(path) def add_value(value: ResolvedValue | None) -> None: match value: case None: pass case ResolvedValueConstant(): pass case ResolvedValuePath(path=path): add_path(path) case _: assert_never(value) def append_paths(field: ResolvedField) -> None: add_value(field.value) match field: case ResolvedFieldDescription(): match field.params: case None: pass case ResolvedAddressNameParameters(): pass case ResolvedCallDataParameters(callee=callee): add_value(callee) case ResolvedTokenAmountParameters(token=token): add_value(token) case ResolvedNftNameParameters(collection=collection): add_value(collection) case ResolvedDateParameters(): pass case ResolvedUnitParameters(): pass case ResolvedEnumParameters(): pass case _: assert_never(field.params) case ResolvedNestedFields(): for nested_field in field.fields: append_paths(nested_field) case _: assert_never(field) for field in format.fields: append_paths(field) return FormatPaths(data_paths=data_paths, container_paths=container_paths)
[docs] def data_path_to_schema_path(path: DataPath) -> DataPath: """ Convert a data path to a schema path. Example: #.foo.[].[-2].[1:5].bar -> #.foo.[].[].[].bar :param path: data path :return: schema path """ def to_schema(element: DataPathElement) -> DataPathElement: match element: case Field() as f: return f case Array() | ArrayElement() | ArraySlice(): return Array() case _: assert_never(element) return path.model_copy(update={"elements": [to_schema(e) for e in path.elements]})