Skip to content

Commit

Permalink
Handle starknet event keys/data correctly
Browse files Browse the repository at this point in the history
  • Loading branch information
m-kus committed Dec 17, 2024
1 parent add96b7 commit 1cc6e1d
Show file tree
Hide file tree
Showing 4 changed files with 58 additions and 42 deletions.
63 changes: 31 additions & 32 deletions pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -48,38 +48,37 @@ classifiers = [
"Typing :: Typed",
]

dependencies = [
"aiohttp~=3.10",
"aiolimiter~=1.1",
"anyio~=4.4",
"appdirs~=1.4",
"APScheduler~=3.10",
"async-lru~=2.0",
"asyncpg~=0.29",
"click~=8.1",
"datamodel-code-generator~=0.26",
"eth-abi~=5.0",
"lru-dict~=1.3",
"orjson~=3.10",
"prometheus-client~=0.20",
"pycryptodome~=3.20",
"pydantic~=2.10.3",
"pyhumps~=3.8",
"pysignalr~=1.0",
"python-dotenv~=1.0",
"python-json-logger~=2.0",
"ruamel.yaml~=0.18.6",
"sentry-sdk~=2.16",
"sqlparse~=0.5",
"starknet-py==0.24.0",
"strict-rfc3339~=0.7",
"survey~=5.4",
"tabulate~=0.9",
# NOTE: Heavily patched; don't update without testing.
"tortoise-orm==0.21.7",
"uvloop~=0.20",
"web3~=7.2",
]
[tool.poetry.dependencies]
aiohttp = "~3.10"
aiolimiter = "~1.1"
anyio = "~4.4"
appdirs = "~1.4"
APScheduler = "~3.10"
async-lru = "~2.0"
asyncpg = "~0.29"
click = "~8.1"
datamodel-code-generator = "~0.26"
eth-abi = "~5.0"
lru-dict = "~1.3"
orjson = "~3.10"
prometheus-client = "~0.20"
pycryptodome = "~3.20"
pydantic = "~2.9"
pyhumps = "~3.8"
pysignalr = "~1.0"
python-dotenv = "~1.0"
python-json-logger = "~2.0"
ruamel.yaml = "~0.18.6"
sentry-sdk = "~2.16"
sqlparse = "~0.5"
starknet-py = { git = "https://github.com/software-mansion/starknet.py", rev = "a8d73538d409d9ef7c756921e43d10925f2838bc" }
strict-rfc3339 = "~0.7"
survey = "~5.4"
tabulate = "~0.9"
# NOTE: Heavily patched; don't update without testing.
tortoise-orm = "0.21.7"
uvloop = "~0.20"
web3 = "~7.2"

[project.optional-dependencies]
migrations = [
Expand Down
2 changes: 1 addition & 1 deletion requirements.txt
Original file line number Diff line number Diff line change
Expand Up @@ -83,7 +83,7 @@ sentry-sdk==2.19.2
six==1.16.0
sniffio==1.3.1
sqlparse==0.5.3
starknet-py==0.24.0
git+https://github.com/software-mansion/starknet.py@a8d73538d409d9ef7c756921e43d10925f2838bc#egg=starknet-py
strict-rfc3339==0.7
survey==5.4.2
sympy==1.11.1
Expand Down
20 changes: 17 additions & 3 deletions src/dipdup/abi/cairo.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
from __future__ import annotations

from collections import OrderedDict
from functools import cache
from typing import TYPE_CHECKING
from typing import Any
Expand All @@ -16,7 +17,7 @@
from starknet_py.abi.v2 import Abi # type: ignore[import-untyped]
from starknet_py.cairo.data_types import CairoType # type: ignore[import-untyped]
from starknet_py.cairo.data_types import EventType
from starknet_py.serialization import PayloadSerializer # type: ignore[import-untyped]
from starknet_py.serialization import CairoDataSerializer # type: ignore[import-untyped]

from dipdup.package import DipDupPackage

Expand All @@ -25,7 +26,7 @@ class CairoEventAbi(TypedDict):
name: str
event_identifier: str
members: dict[str, CairoType]
serializer: PayloadSerializer
sorted_serializers: OrderedDict[str, CairoDataSerializer]


class CairoAbi(TypedDict):
Expand Down Expand Up @@ -93,12 +94,25 @@ def convert_abi(package: DipDupPackage) -> dict[str, CairoAbi]:
for name, event_type in parsed_abi.events.items():
if name in converted_abi['events']:
raise NotImplementedError('Multiple events with the same name are not supported')

serializers = serializer_for_event(event_type).serializers

# Event payload is returned from RPC in two arrays: keys (including event selector) and data.
# Since any event field can be marked as key, the original ordering might be broken.
#
# We need to reorder deserializers so that the keys remain in the beginning and
# the rest of the fields are moved towards the end (preserving their inner ordering).
#
# That way we can apply the deserializers to the concatenation of keys (without first element) + data.
sorted_members = event_type.keys + [name for name in serializers if name not in event_type.keys]
sorted_serializers = OrderedDict((name, serializers[name]) for name in sorted_members)

converted_abi['events'].append(
CairoEventAbi(
name=name,
event_identifier=sn_keccak(name),
members=event_type.types,
serializer=serializer_for_event(event_type),
sorted_serializers=sorted_serializers,
)
)
abi_by_typename[contract_typename] = converted_abi
Expand Down
15 changes: 9 additions & 6 deletions src/dipdup/indexes/starknet_events/matcher.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import logging
from collections import deque
from collections import deque, OrderedDict
from collections.abc import Iterable
from typing import Any

Expand Down Expand Up @@ -69,15 +69,18 @@ def prepare_event_handler_args(
name=snake_to_pascal(handler_config.name) + 'Payload',
)

serializer = package._cairo_abis.get_event_abi(
event_abi = package._cairo_abis.get_event_abi(
typename=typename,
name=handler_config.name,
)['serializer']
data = [int(s, 16) for s in matched_event.data]
)

# Skipping first key which is the event selector
# Note that some fields might be encoded with more than one felt (complex types)
raw_data = [int(x, 16) for x in matched_event.keys[1:] + matched_event.data]

# holding context for error building
with DeserializationContext.create(data) as context:
data_dict = deserialize_to_dict(serializer.serializers, context)
with DeserializationContext.create(raw_data) as context:
data_dict = deserialize_to_dict(event_abi['sorted_serializers'], context)

typed_payload = parse_object(type_=type_, data=data_dict)
return StarknetEvent(
Expand Down

0 comments on commit 1cc6e1d

Please sign in to comment.