From 59011c1f8870b56ed11379f2320388a01f71494c Mon Sep 17 00:00:00 2001 From: Juliya Smith Date: Thu, 19 Dec 2024 14:16:13 -0600 Subject: [PATCH] fix: couldnt decode ABIs when missing leading zeroes --- src/ape/utils/abi.py | 35 ++++++++++++++++++++++++++++-- tests/functional/utils/test_abi.py | 34 +++++++++++++++++++++++++++-- 2 files changed, 65 insertions(+), 4 deletions(-) diff --git a/src/ape/utils/abi.py b/src/ape/utils/abi.py index 7a3dd15c89..6bd71c8395 100644 --- a/src/ape/utils/abi.py +++ b/src/ape/utils/abi.py @@ -3,9 +3,14 @@ from dataclasses import make_dataclass from typing import Any, Optional, Union -from eth_abi import decode, grammar +from eth_abi import grammar +from eth_abi.abi import decode +from eth_abi.decoding import UnsignedIntegerDecoder +from eth_abi.encoding import UnsignedIntegerEncoder from eth_abi.exceptions import DecodingError, InsufficientDataBytes +from eth_abi.registry import BaseEquals, registry from eth_pydantic_types import HexBytes +from eth_pydantic_types.validators import validate_bytes_size from eth_utils import decode_hex from ethpm_types.abi import ABIType, ConstructorABI, EventABI, EventABIType, MethodABI @@ -15,6 +20,30 @@ NATSPEC_KEY_PATTERN = re.compile(r"(@\w+)") +class ApeUnsignedIntegerDecoder(UnsignedIntegerDecoder): + def read_data_from_stream(self, stream): + """ + Override to pad the value instead of raise an error. + """ + data_byte_size: int = self.data_byte_size # type: ignore + data = stream.read(data_byte_size) + + if len(data) != data_byte_size: + # Pad the value (instead of raising InsufficientBytesError). + data = validate_bytes_size(data, 32) + + return data + + +registry.unregister("uint") +registry.register( + BaseEquals("uint"), + UnsignedIntegerEncoder, + ApeUnsignedIntegerDecoder, + label="uint", +) + + def is_array(abi_type: Union[str, ABIType]) -> bool: """ Returns ``True`` if the given type is a probably an array. @@ -418,7 +447,9 @@ def __init__(self, abi: EventABI): def event_name(self): return self.abi.name - def decode(self, topics: list[str], data: str, use_hex_on_fail: bool = False) -> dict: + def decode( + self, topics: list[str], data: Union[str, bytes], use_hex_on_fail: bool = False + ) -> dict: decoded = {} for abi, topic_value in zip(self.topic_abi_types, topics[1:]): # reference types as indexed arguments are written as a hash diff --git a/tests/functional/utils/test_abi.py b/tests/functional/utils/test_abi.py index d4aabfc537..64fbe49522 100644 --- a/tests/functional/utils/test_abi.py +++ b/tests/functional/utils/test_abi.py @@ -53,7 +53,7 @@ def topics(): return ["0xc52ec0ad7872dae440d886040390c13677df7bf3cca136d8d81e5e5e7dd62ff1"] -@pytest.fixture +@pytest.fixture(scope="module") def log_data_missing_trailing_zeroes(): return HexBytes( "0x000000000000000000000000000000000000000000000000000000000000001e" @@ -65,7 +65,9 @@ def log_data_missing_trailing_zeroes(): ) -def test_decoding_with_strict(collection, topics, log_data_missing_trailing_zeroes, ape_caplog): +def test_decode_data_missing_trailing_zeroes( + collection, topics, log_data_missing_trailing_zeroes, ape_caplog +): """ This test is for a time where Alchemy gave us log data when it was missing trailing zeroes. When using strict=False, it was able to properly decode. In this case, in Ape, we warn @@ -84,6 +86,34 @@ def test_decoding_with_strict(collection, topics, log_data_missing_trailing_zero assert actual == expected +def test_decode_topics_missing_leading_zeroes(vyper_contract_type): + # The second value here was the problem before... It has no leading zeroes + # and eth-abi is very strict about that. + topics = [ + "0xa84473122c11e32cd505595f246a28418b8ecd6cf819f4e3915363fad1b8f968", + "0x0141", + "0x9f3d45ac20ccf04b45028b8080bb191eab93e29f7898ed43acf480dd80bba94d", + ] + + # NOTE: data isn't really part of the test but still has to be included. + data = ( + b"\x9c\xe2\xce\xf5\x9b\xf2\xdeu\x83f\xf8s\xdb\x7f&\xef\xab\x9bw\xf7\xcf" + b"\xe9\xc8I\xb6\xb5@\x04g\xa9)\x86\x00\x00\x00\x00\x00\x00\x00\x00\x00" + b"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00" + b"\x00\x00\x00\x00\x00{\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00" + b"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00" + b"\x00\x00`\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00" + b"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x07" + b"Dynamic\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00" + b"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00" + ) + abi = vyper_contract_type.events["NumberChange"] + collection = LogInputABICollection(abi) + + actual = collection.decode(topics, data) + assert actual["newNum"] == 321 # NOTE: Was a bug where this causes issues. + + class TestStruct: @pytest.fixture def struct(self):