faster-eth-abi 5.2.22__cp313-cp313-macosx_11_0_arm64.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- faster_eth_abi/__init__.py +12 -0
- faster_eth_abi/_codec.cpython-313-darwin.so +0 -0
- faster_eth_abi/_codec.py +83 -0
- faster_eth_abi/_decoding.cpython-313-darwin.so +0 -0
- faster_eth_abi/_decoding.py +299 -0
- faster_eth_abi/_encoding.cpython-313-darwin.so +0 -0
- faster_eth_abi/_encoding.py +354 -0
- faster_eth_abi/_grammar.cpython-313-darwin.so +0 -0
- faster_eth_abi/_grammar.py +375 -0
- faster_eth_abi/abi.cpython-313-darwin.so +0 -0
- faster_eth_abi/abi.py +17 -0
- faster_eth_abi/base.py +45 -0
- faster_eth_abi/codec.py +2809 -0
- faster_eth_abi/constants.cpython-313-darwin.so +0 -0
- faster_eth_abi/constants.py +7 -0
- faster_eth_abi/decoding.py +584 -0
- faster_eth_abi/encoding.py +746 -0
- faster_eth_abi/exceptions.py +127 -0
- faster_eth_abi/from_type_str.cpython-313-darwin.so +0 -0
- faster_eth_abi/from_type_str.py +141 -0
- faster_eth_abi/grammar.py +172 -0
- faster_eth_abi/io.py +107 -0
- faster_eth_abi/packed.cpython-313-darwin.so +0 -0
- faster_eth_abi/packed.py +19 -0
- faster_eth_abi/py.typed +0 -0
- faster_eth_abi/registry.py +758 -0
- faster_eth_abi/tools/__init__.cpython-313-darwin.so +0 -0
- faster_eth_abi/tools/__init__.py +3 -0
- faster_eth_abi/tools/_strategies.cpython-313-darwin.so +0 -0
- faster_eth_abi/tools/_strategies.py +243 -0
- faster_eth_abi/typing.py +4627 -0
- faster_eth_abi/utils/__init__.cpython-313-darwin.so +0 -0
- faster_eth_abi/utils/__init__.py +0 -0
- faster_eth_abi/utils/numeric.cpython-313-darwin.so +0 -0
- faster_eth_abi/utils/numeric.py +117 -0
- faster_eth_abi/utils/padding.cpython-313-darwin.so +0 -0
- faster_eth_abi/utils/padding.py +22 -0
- faster_eth_abi/utils/string.cpython-313-darwin.so +0 -0
- faster_eth_abi/utils/string.py +19 -0
- faster_eth_abi/utils/validation.cpython-313-darwin.so +0 -0
- faster_eth_abi/utils/validation.py +18 -0
- faster_eth_abi-5.2.22.dist-info/METADATA +136 -0
- faster_eth_abi-5.2.22.dist-info/RECORD +46 -0
- faster_eth_abi-5.2.22.dist-info/WHEEL +6 -0
- faster_eth_abi-5.2.22.dist-info/top_level.txt +2 -0
- faster_eth_abi__mypyc.cpython-313-darwin.so +0 -0
|
@@ -0,0 +1,746 @@
|
|
|
1
|
+
"""Classes for ABI encoding logic.
|
|
2
|
+
|
|
3
|
+
Implements classes and functions for serializing Python values into binary data
|
|
4
|
+
according to ABI type specifications.
|
|
5
|
+
"""
|
|
6
|
+
import abc
|
|
7
|
+
import codecs
|
|
8
|
+
import decimal
|
|
9
|
+
from functools import (
|
|
10
|
+
cached_property,
|
|
11
|
+
lru_cache,
|
|
12
|
+
)
|
|
13
|
+
from types import (
|
|
14
|
+
MethodType,
|
|
15
|
+
)
|
|
16
|
+
from typing import (
|
|
17
|
+
Any,
|
|
18
|
+
Callable,
|
|
19
|
+
ClassVar,
|
|
20
|
+
Final,
|
|
21
|
+
NoReturn,
|
|
22
|
+
Optional,
|
|
23
|
+
Sequence,
|
|
24
|
+
Tuple,
|
|
25
|
+
Type,
|
|
26
|
+
final,
|
|
27
|
+
)
|
|
28
|
+
|
|
29
|
+
from faster_eth_utils import (
|
|
30
|
+
is_address,
|
|
31
|
+
is_boolean,
|
|
32
|
+
is_bytes,
|
|
33
|
+
is_integer,
|
|
34
|
+
is_list_like,
|
|
35
|
+
is_number,
|
|
36
|
+
is_text,
|
|
37
|
+
to_canonical_address,
|
|
38
|
+
)
|
|
39
|
+
from typing_extensions import (
|
|
40
|
+
Self,
|
|
41
|
+
)
|
|
42
|
+
|
|
43
|
+
from faster_eth_abi._encoding import (
|
|
44
|
+
encode_elements,
|
|
45
|
+
encode_elements_dynamic,
|
|
46
|
+
encode_fixed,
|
|
47
|
+
encode_signed,
|
|
48
|
+
encode_tuple,
|
|
49
|
+
encode_tuple_all_dynamic,
|
|
50
|
+
encode_tuple_no_dynamic,
|
|
51
|
+
encode_tuple_no_dynamic_funcs,
|
|
52
|
+
int_to_big_endian,
|
|
53
|
+
validate_tuple,
|
|
54
|
+
)
|
|
55
|
+
from faster_eth_abi.base import (
|
|
56
|
+
BaseCoder,
|
|
57
|
+
)
|
|
58
|
+
from faster_eth_abi.exceptions import (
|
|
59
|
+
EncodingTypeError,
|
|
60
|
+
IllegalValue,
|
|
61
|
+
ValueOutOfBounds,
|
|
62
|
+
)
|
|
63
|
+
from faster_eth_abi.from_type_str import (
|
|
64
|
+
parse_tuple_type_str,
|
|
65
|
+
parse_type_str,
|
|
66
|
+
)
|
|
67
|
+
from faster_eth_abi.utils.numeric import (
|
|
68
|
+
TEN,
|
|
69
|
+
abi_decimal_context,
|
|
70
|
+
ceil32,
|
|
71
|
+
compute_signed_fixed_bounds,
|
|
72
|
+
compute_signed_integer_bounds,
|
|
73
|
+
compute_unsigned_fixed_bounds,
|
|
74
|
+
compute_unsigned_integer_bounds,
|
|
75
|
+
)
|
|
76
|
+
from faster_eth_abi.utils.padding import (
|
|
77
|
+
zpad_right,
|
|
78
|
+
)
|
|
79
|
+
from faster_eth_abi.utils.string import (
|
|
80
|
+
abbr,
|
|
81
|
+
)
|
|
82
|
+
|
|
83
|
+
|
|
84
|
+
class BaseEncoder(BaseCoder, metaclass=abc.ABCMeta):
|
|
85
|
+
"""
|
|
86
|
+
Base class for all encoder classes. Subclass this if you want to define a
|
|
87
|
+
custom encoder class. Subclasses must also implement
|
|
88
|
+
:any:`BaseCoder.from_type_str`.
|
|
89
|
+
"""
|
|
90
|
+
|
|
91
|
+
@abc.abstractmethod
|
|
92
|
+
def encode(self, value: Any) -> bytes: # pragma: no cover
|
|
93
|
+
"""
|
|
94
|
+
Encodes the given value as a sequence of bytes. Should raise
|
|
95
|
+
:any:`exceptions.EncodingError` if ``value`` cannot be encoded.
|
|
96
|
+
"""
|
|
97
|
+
|
|
98
|
+
@abc.abstractmethod
|
|
99
|
+
def validate_value(self, value: Any) -> None:
|
|
100
|
+
"""
|
|
101
|
+
Checks whether or not the given value can be encoded by this encoder.
|
|
102
|
+
If the given value cannot be encoded, must raise
|
|
103
|
+
:any:`exceptions.EncodingError`.
|
|
104
|
+
"""
|
|
105
|
+
|
|
106
|
+
@classmethod
|
|
107
|
+
def invalidate_value(
|
|
108
|
+
cls,
|
|
109
|
+
value: Any,
|
|
110
|
+
exc: Type[Exception] = EncodingTypeError,
|
|
111
|
+
msg: Optional[str] = None,
|
|
112
|
+
) -> NoReturn:
|
|
113
|
+
"""
|
|
114
|
+
Throws a standard exception for when a value is not encodable by an
|
|
115
|
+
encoder.
|
|
116
|
+
"""
|
|
117
|
+
raise exc(
|
|
118
|
+
f"Value `{abbr(value)}` of type {type(value)} cannot be encoded by "
|
|
119
|
+
f"{cls.__name__}{'' if msg is None else (': ' + msg)}"
|
|
120
|
+
)
|
|
121
|
+
|
|
122
|
+
def __call__(self, value: Any) -> bytes:
|
|
123
|
+
return self.encode(value)
|
|
124
|
+
|
|
125
|
+
|
|
126
|
+
class TupleEncoder(BaseEncoder):
|
|
127
|
+
encoders: Tuple[BaseEncoder, ...] = ()
|
|
128
|
+
|
|
129
|
+
def __init__(self, encoders: Tuple[BaseEncoder, ...], **kwargs: Any) -> None:
|
|
130
|
+
super().__init__(encoders=encoders, **kwargs)
|
|
131
|
+
|
|
132
|
+
self._is_dynamic: Final = tuple(
|
|
133
|
+
getattr(e, "is_dynamic", False) for e in self.encoders
|
|
134
|
+
)
|
|
135
|
+
self.is_dynamic = any(self._is_dynamic)
|
|
136
|
+
|
|
137
|
+
validators = []
|
|
138
|
+
for encoder in self.encoders:
|
|
139
|
+
try:
|
|
140
|
+
validator = encoder.validate_value
|
|
141
|
+
except AttributeError:
|
|
142
|
+
validators.append(encoder)
|
|
143
|
+
else:
|
|
144
|
+
validators.append(validator)
|
|
145
|
+
|
|
146
|
+
self.validators: Final[Callable[[Any], None]] = tuple(validators)
|
|
147
|
+
|
|
148
|
+
if type(self).encode is TupleEncoder.encode:
|
|
149
|
+
encode_func = (
|
|
150
|
+
encode_tuple_all_dynamic
|
|
151
|
+
if all(self._is_dynamic)
|
|
152
|
+
else encode_tuple_no_dynamic_funcs.get(
|
|
153
|
+
len(self.encoders),
|
|
154
|
+
encode_tuple_no_dynamic,
|
|
155
|
+
)
|
|
156
|
+
if not self.is_dynamic
|
|
157
|
+
else encode_tuple
|
|
158
|
+
)
|
|
159
|
+
|
|
160
|
+
self.encode = MethodType(encode_func, self)
|
|
161
|
+
|
|
162
|
+
def validate(self) -> None:
|
|
163
|
+
super().validate()
|
|
164
|
+
|
|
165
|
+
if self.encoders is None:
|
|
166
|
+
raise ValueError("`encoders` may not be none")
|
|
167
|
+
|
|
168
|
+
@final
|
|
169
|
+
def validate_value(self, value: Sequence[Any]) -> None:
|
|
170
|
+
validate_tuple(self, value)
|
|
171
|
+
|
|
172
|
+
def encode(self, values: Sequence[Any]) -> bytes:
|
|
173
|
+
return encode_tuple(self, values)
|
|
174
|
+
|
|
175
|
+
def __call__(self, values: Sequence[Any]) -> bytes:
|
|
176
|
+
return self.encode(values)
|
|
177
|
+
|
|
178
|
+
@parse_tuple_type_str
|
|
179
|
+
def from_type_str(cls, abi_type, registry):
|
|
180
|
+
encoders = tuple(
|
|
181
|
+
registry.get_encoder(c.to_type_str()) for c in abi_type.components
|
|
182
|
+
)
|
|
183
|
+
|
|
184
|
+
return cls(encoders=encoders)
|
|
185
|
+
|
|
186
|
+
|
|
187
|
+
class FixedSizeEncoder(BaseEncoder):
|
|
188
|
+
value_bit_size = None
|
|
189
|
+
data_byte_size = None
|
|
190
|
+
encode_fn = None
|
|
191
|
+
type_check_fn = None
|
|
192
|
+
is_big_endian = None
|
|
193
|
+
|
|
194
|
+
def validate(self) -> None:
|
|
195
|
+
super().validate()
|
|
196
|
+
|
|
197
|
+
value_bit_size = self.value_bit_size
|
|
198
|
+
if value_bit_size is None:
|
|
199
|
+
raise ValueError("`value_bit_size` may not be none")
|
|
200
|
+
data_byte_size = self.data_byte_size
|
|
201
|
+
if data_byte_size is None:
|
|
202
|
+
raise ValueError("`data_byte_size` may not be none")
|
|
203
|
+
if self.encode_fn is None:
|
|
204
|
+
raise ValueError("`encode_fn` may not be none")
|
|
205
|
+
if self.is_big_endian is None:
|
|
206
|
+
raise ValueError("`is_big_endian` may not be none")
|
|
207
|
+
|
|
208
|
+
if value_bit_size % 8 != 0:
|
|
209
|
+
raise ValueError(
|
|
210
|
+
f"Invalid value bit size: {value_bit_size}. Must be a multiple of 8"
|
|
211
|
+
)
|
|
212
|
+
|
|
213
|
+
if value_bit_size > data_byte_size * 8:
|
|
214
|
+
raise ValueError("Value byte size exceeds data size")
|
|
215
|
+
|
|
216
|
+
def validate_value(self, value: Any) -> None:
|
|
217
|
+
raise NotImplementedError("Must be implemented by subclasses")
|
|
218
|
+
|
|
219
|
+
def encode(self, value: Any) -> bytes:
|
|
220
|
+
self.validate_value(value)
|
|
221
|
+
encode_fn = self.encode_fn
|
|
222
|
+
if encode_fn is None:
|
|
223
|
+
raise AssertionError("`encode_fn` is None")
|
|
224
|
+
return encode_fixed(value, encode_fn, self.is_big_endian, self.data_byte_size)
|
|
225
|
+
|
|
226
|
+
__call__ = encode
|
|
227
|
+
|
|
228
|
+
|
|
229
|
+
class Fixed32ByteSizeEncoder(FixedSizeEncoder):
|
|
230
|
+
data_byte_size = 32
|
|
231
|
+
|
|
232
|
+
|
|
233
|
+
class BooleanEncoder(Fixed32ByteSizeEncoder):
|
|
234
|
+
value_bit_size = 8
|
|
235
|
+
is_big_endian = True
|
|
236
|
+
|
|
237
|
+
@classmethod
|
|
238
|
+
def validate_value(cls, value: Any) -> None:
|
|
239
|
+
if not is_boolean(value):
|
|
240
|
+
cls.invalidate_value(value)
|
|
241
|
+
|
|
242
|
+
@classmethod
|
|
243
|
+
def encode_fn(cls, value: bool) -> bytes:
|
|
244
|
+
if value is True:
|
|
245
|
+
return b"\x01"
|
|
246
|
+
elif value is False:
|
|
247
|
+
return b"\x00"
|
|
248
|
+
else:
|
|
249
|
+
raise ValueError("Invariant")
|
|
250
|
+
|
|
251
|
+
@parse_type_str("bool")
|
|
252
|
+
def from_type_str(cls, abi_type, registry):
|
|
253
|
+
return cls()
|
|
254
|
+
|
|
255
|
+
|
|
256
|
+
class PackedBooleanEncoder(BooleanEncoder):
|
|
257
|
+
data_byte_size = 1
|
|
258
|
+
|
|
259
|
+
|
|
260
|
+
class NumberEncoder(Fixed32ByteSizeEncoder):
|
|
261
|
+
is_big_endian = True
|
|
262
|
+
bounds_fn = None
|
|
263
|
+
illegal_value_fn = None
|
|
264
|
+
type_check_fn = None
|
|
265
|
+
|
|
266
|
+
def validate(self) -> None:
|
|
267
|
+
super().validate()
|
|
268
|
+
|
|
269
|
+
if self.bounds_fn is None:
|
|
270
|
+
raise ValueError("`bounds_fn` cannot be null")
|
|
271
|
+
if self.type_check_fn is None:
|
|
272
|
+
raise ValueError("`type_check_fn` cannot be null")
|
|
273
|
+
|
|
274
|
+
def validate_value(self, value: Any) -> None:
|
|
275
|
+
type_check_fn = self.type_check_fn
|
|
276
|
+
if type_check_fn is None:
|
|
277
|
+
raise AssertionError("`type_check_fn` is None")
|
|
278
|
+
if not type_check_fn(value):
|
|
279
|
+
self.invalidate_value(value)
|
|
280
|
+
|
|
281
|
+
illegal_value_fn = self.illegal_value_fn
|
|
282
|
+
illegal_value = illegal_value_fn is not None and illegal_value_fn(value)
|
|
283
|
+
if illegal_value:
|
|
284
|
+
self.invalidate_value(value, exc=IllegalValue)
|
|
285
|
+
|
|
286
|
+
lower_bound, upper_bound = self.bounds_fn(self.value_bit_size)
|
|
287
|
+
if value < lower_bound or value > upper_bound:
|
|
288
|
+
self.invalidate_value(
|
|
289
|
+
value,
|
|
290
|
+
exc=ValueOutOfBounds,
|
|
291
|
+
msg=f"Cannot be encoded in {self.value_bit_size} bits. Must be bounded "
|
|
292
|
+
f"between [{lower_bound}, {upper_bound}].",
|
|
293
|
+
)
|
|
294
|
+
|
|
295
|
+
|
|
296
|
+
class UnsignedIntegerEncoder(NumberEncoder):
|
|
297
|
+
encode_fn = staticmethod(int_to_big_endian)
|
|
298
|
+
bounds_fn = staticmethod(compute_unsigned_integer_bounds)
|
|
299
|
+
type_check_fn = staticmethod(is_integer)
|
|
300
|
+
|
|
301
|
+
@parse_type_str("uint")
|
|
302
|
+
def from_type_str(cls, abi_type, registry):
|
|
303
|
+
return cls(value_bit_size=abi_type.sub)
|
|
304
|
+
|
|
305
|
+
|
|
306
|
+
class UnsignedIntegerEncoderCached(UnsignedIntegerEncoder):
|
|
307
|
+
encode: Final[Callable[[int], bytes]]
|
|
308
|
+
maxsize: Final[Optional[int]]
|
|
309
|
+
|
|
310
|
+
def __init__(self, maxsize: Optional[int] = None, **kwargs: Any) -> None:
|
|
311
|
+
super().__init__(**kwargs)
|
|
312
|
+
self.maxsize = maxsize
|
|
313
|
+
self.encode = lru_cache(maxsize=maxsize)(self.encode)
|
|
314
|
+
|
|
315
|
+
|
|
316
|
+
encode_uint_256 = UnsignedIntegerEncoder(value_bit_size=256, data_byte_size=32)
|
|
317
|
+
|
|
318
|
+
|
|
319
|
+
class PackedUnsignedIntegerEncoder(UnsignedIntegerEncoder):
|
|
320
|
+
@parse_type_str("uint")
|
|
321
|
+
def from_type_str(cls, abi_type, registry):
|
|
322
|
+
return cls(
|
|
323
|
+
value_bit_size=abi_type.sub,
|
|
324
|
+
data_byte_size=abi_type.sub // 8,
|
|
325
|
+
)
|
|
326
|
+
|
|
327
|
+
|
|
328
|
+
class PackedUnsignedIntegerEncoderCached(PackedUnsignedIntegerEncoder):
|
|
329
|
+
encode: Final[Callable[[int], bytes]]
|
|
330
|
+
maxsize: Final[Optional[int]]
|
|
331
|
+
|
|
332
|
+
def __init__(self, maxsize: Optional[int] = None, **kwargs: Any) -> None:
|
|
333
|
+
super().__init__(**kwargs)
|
|
334
|
+
self.maxsize = maxsize
|
|
335
|
+
self.encode = lru_cache(maxsize=maxsize)(self.encode)
|
|
336
|
+
|
|
337
|
+
|
|
338
|
+
class SignedIntegerEncoder(NumberEncoder):
|
|
339
|
+
bounds_fn = staticmethod(compute_signed_integer_bounds)
|
|
340
|
+
type_check_fn = staticmethod(is_integer)
|
|
341
|
+
|
|
342
|
+
def encode_fn(self, value: int) -> bytes:
|
|
343
|
+
return int_to_big_endian(value % (2**self.value_bit_size))
|
|
344
|
+
|
|
345
|
+
def encode(self, value: int) -> bytes:
|
|
346
|
+
self.validate_value(value)
|
|
347
|
+
return encode_signed(value, self.encode_fn, self.data_byte_size)
|
|
348
|
+
|
|
349
|
+
__call__ = encode
|
|
350
|
+
|
|
351
|
+
@parse_type_str("int")
|
|
352
|
+
def from_type_str(cls, abi_type, registry):
|
|
353
|
+
return cls(value_bit_size=abi_type.sub)
|
|
354
|
+
|
|
355
|
+
|
|
356
|
+
class SignedIntegerEncoderCached(SignedIntegerEncoder):
|
|
357
|
+
encode: Final[Callable[[int], bytes]]
|
|
358
|
+
maxsize: Final[Optional[int]]
|
|
359
|
+
|
|
360
|
+
def __init__(self, maxsize: Optional[int] = None, **kwargs: Any) -> None:
|
|
361
|
+
super().__init__(**kwargs)
|
|
362
|
+
self.maxsize = maxsize
|
|
363
|
+
self.encode = lru_cache(maxsize=maxsize)(self.encode)
|
|
364
|
+
|
|
365
|
+
|
|
366
|
+
class PackedSignedIntegerEncoder(SignedIntegerEncoder):
|
|
367
|
+
@parse_type_str("int")
|
|
368
|
+
def from_type_str(cls, abi_type, registry):
|
|
369
|
+
return cls(
|
|
370
|
+
value_bit_size=abi_type.sub,
|
|
371
|
+
data_byte_size=abi_type.sub // 8,
|
|
372
|
+
)
|
|
373
|
+
|
|
374
|
+
|
|
375
|
+
class PackedSignedIntegerEncoderCached(PackedSignedIntegerEncoder):
|
|
376
|
+
encode: Final[Callable[[int], bytes]]
|
|
377
|
+
maxsize: Final[Optional[int]]
|
|
378
|
+
|
|
379
|
+
def __init__(self, maxsize: Optional[int] = None, **kwargs: Any) -> None:
|
|
380
|
+
super().__init__(**kwargs)
|
|
381
|
+
self.maxsize = maxsize
|
|
382
|
+
self.encode = lru_cache(maxsize=maxsize)(self.encode)
|
|
383
|
+
|
|
384
|
+
|
|
385
|
+
class BaseFixedEncoder(NumberEncoder):
|
|
386
|
+
frac_places = None
|
|
387
|
+
|
|
388
|
+
@staticmethod
|
|
389
|
+
def type_check_fn(value):
|
|
390
|
+
return is_number(value) and not isinstance(value, float)
|
|
391
|
+
|
|
392
|
+
@staticmethod
|
|
393
|
+
def illegal_value_fn(value):
|
|
394
|
+
if isinstance(value, decimal.Decimal):
|
|
395
|
+
return value.is_nan() or value.is_infinite()
|
|
396
|
+
|
|
397
|
+
return False
|
|
398
|
+
|
|
399
|
+
@cached_property
|
|
400
|
+
def denominator(self) -> decimal.Decimal:
|
|
401
|
+
return TEN**self.frac_places
|
|
402
|
+
|
|
403
|
+
@cached_property
|
|
404
|
+
def precision(self) -> int:
|
|
405
|
+
return TEN**-self.frac_places
|
|
406
|
+
|
|
407
|
+
def validate_value(self, value):
|
|
408
|
+
super().validate_value(value)
|
|
409
|
+
|
|
410
|
+
with decimal.localcontext(abi_decimal_context):
|
|
411
|
+
residue = value % self.precision
|
|
412
|
+
|
|
413
|
+
if residue > 0:
|
|
414
|
+
self.invalidate_value(
|
|
415
|
+
value,
|
|
416
|
+
exc=IllegalValue,
|
|
417
|
+
msg=f"residue {residue!r} outside allowed fractional precision of "
|
|
418
|
+
f"{self.frac_places}",
|
|
419
|
+
)
|
|
420
|
+
|
|
421
|
+
def validate(self) -> None:
|
|
422
|
+
super().validate()
|
|
423
|
+
|
|
424
|
+
frac_places = self.frac_places
|
|
425
|
+
if frac_places is None:
|
|
426
|
+
raise ValueError("must specify `frac_places`")
|
|
427
|
+
|
|
428
|
+
if frac_places <= 0 or frac_places > 80:
|
|
429
|
+
raise ValueError("`frac_places` must be in range (0, 80]")
|
|
430
|
+
|
|
431
|
+
|
|
432
|
+
class UnsignedFixedEncoder(BaseFixedEncoder):
|
|
433
|
+
def bounds_fn(self, value_bit_size):
|
|
434
|
+
return compute_unsigned_fixed_bounds(self.value_bit_size, self.frac_places)
|
|
435
|
+
|
|
436
|
+
def encode_fn(self, value: decimal.Decimal) -> bytes:
|
|
437
|
+
with decimal.localcontext(abi_decimal_context):
|
|
438
|
+
scaled_value = value * self.denominator
|
|
439
|
+
integer_value = int(scaled_value)
|
|
440
|
+
|
|
441
|
+
return int_to_big_endian(integer_value)
|
|
442
|
+
|
|
443
|
+
@parse_type_str("ufixed")
|
|
444
|
+
def from_type_str(cls, abi_type, registry):
|
|
445
|
+
value_bit_size, frac_places = abi_type.sub
|
|
446
|
+
|
|
447
|
+
return cls(
|
|
448
|
+
value_bit_size=value_bit_size,
|
|
449
|
+
frac_places=frac_places,
|
|
450
|
+
)
|
|
451
|
+
|
|
452
|
+
|
|
453
|
+
class PackedUnsignedFixedEncoder(UnsignedFixedEncoder):
|
|
454
|
+
@parse_type_str("ufixed")
|
|
455
|
+
def from_type_str(cls, abi_type, registry):
|
|
456
|
+
value_bit_size, frac_places = abi_type.sub
|
|
457
|
+
|
|
458
|
+
return cls(
|
|
459
|
+
value_bit_size=value_bit_size,
|
|
460
|
+
data_byte_size=value_bit_size // 8,
|
|
461
|
+
frac_places=frac_places,
|
|
462
|
+
)
|
|
463
|
+
|
|
464
|
+
|
|
465
|
+
class SignedFixedEncoder(BaseFixedEncoder):
|
|
466
|
+
def bounds_fn(self, value_bit_size):
|
|
467
|
+
return compute_signed_fixed_bounds(self.value_bit_size, self.frac_places)
|
|
468
|
+
|
|
469
|
+
def encode_fn(self, value: decimal.Decimal) -> bytes:
|
|
470
|
+
with decimal.localcontext(abi_decimal_context):
|
|
471
|
+
scaled_value = value * self.denominator
|
|
472
|
+
integer_value = int(scaled_value)
|
|
473
|
+
|
|
474
|
+
unsigned_integer_value = integer_value % (2**self.value_bit_size)
|
|
475
|
+
|
|
476
|
+
return int_to_big_endian(unsigned_integer_value)
|
|
477
|
+
|
|
478
|
+
def encode(self, value: decimal.Decimal) -> bytes:
|
|
479
|
+
self.validate_value(value)
|
|
480
|
+
return encode_signed(value, self.encode_fn, self.data_byte_size)
|
|
481
|
+
|
|
482
|
+
__call__ = encode
|
|
483
|
+
|
|
484
|
+
@parse_type_str("fixed")
|
|
485
|
+
def from_type_str(cls, abi_type, registry):
|
|
486
|
+
value_bit_size, frac_places = abi_type.sub
|
|
487
|
+
|
|
488
|
+
return cls(
|
|
489
|
+
value_bit_size=value_bit_size,
|
|
490
|
+
frac_places=frac_places,
|
|
491
|
+
)
|
|
492
|
+
|
|
493
|
+
|
|
494
|
+
class PackedSignedFixedEncoder(SignedFixedEncoder):
|
|
495
|
+
@parse_type_str("fixed")
|
|
496
|
+
def from_type_str(cls, abi_type, registry):
|
|
497
|
+
value_bit_size, frac_places = abi_type.sub
|
|
498
|
+
|
|
499
|
+
return cls(
|
|
500
|
+
value_bit_size=value_bit_size,
|
|
501
|
+
data_byte_size=value_bit_size // 8,
|
|
502
|
+
frac_places=frac_places,
|
|
503
|
+
)
|
|
504
|
+
|
|
505
|
+
|
|
506
|
+
class AddressEncoder(Fixed32ByteSizeEncoder):
|
|
507
|
+
value_bit_size = 20 * 8
|
|
508
|
+
encode_fn = staticmethod(to_canonical_address)
|
|
509
|
+
is_big_endian = True
|
|
510
|
+
|
|
511
|
+
@classmethod
|
|
512
|
+
def validate_value(cls, value: Any) -> None:
|
|
513
|
+
if not is_address(value):
|
|
514
|
+
cls.invalidate_value(value)
|
|
515
|
+
|
|
516
|
+
def validate(self) -> None:
|
|
517
|
+
super().validate()
|
|
518
|
+
|
|
519
|
+
if self.value_bit_size != 20 * 8:
|
|
520
|
+
raise ValueError("Addresses must be 160 bits in length")
|
|
521
|
+
|
|
522
|
+
@parse_type_str("address")
|
|
523
|
+
def from_type_str(cls, abi_type, registry):
|
|
524
|
+
return cls()
|
|
525
|
+
|
|
526
|
+
|
|
527
|
+
class PackedAddressEncoder(AddressEncoder):
|
|
528
|
+
data_byte_size = 20
|
|
529
|
+
|
|
530
|
+
|
|
531
|
+
class BytesEncoder(Fixed32ByteSizeEncoder):
|
|
532
|
+
is_big_endian = False
|
|
533
|
+
|
|
534
|
+
def validate_value(self, value: Any) -> None:
|
|
535
|
+
if not is_bytes(value):
|
|
536
|
+
self.invalidate_value(value)
|
|
537
|
+
|
|
538
|
+
byte_size = self.value_bit_size // 8
|
|
539
|
+
if len(value) > byte_size:
|
|
540
|
+
self.invalidate_value(
|
|
541
|
+
value,
|
|
542
|
+
exc=ValueOutOfBounds,
|
|
543
|
+
msg=f"exceeds total byte size for bytes{byte_size} encoding",
|
|
544
|
+
)
|
|
545
|
+
|
|
546
|
+
@staticmethod
|
|
547
|
+
def encode_fn(value: bytes) -> bytes:
|
|
548
|
+
return value
|
|
549
|
+
|
|
550
|
+
@parse_type_str("bytes")
|
|
551
|
+
def from_type_str(cls, abi_type, registry):
|
|
552
|
+
return cls(value_bit_size=abi_type.sub * 8)
|
|
553
|
+
|
|
554
|
+
|
|
555
|
+
class PackedBytesEncoder(BytesEncoder):
|
|
556
|
+
@parse_type_str("bytes")
|
|
557
|
+
def from_type_str(cls, abi_type, registry):
|
|
558
|
+
return cls(
|
|
559
|
+
value_bit_size=abi_type.sub * 8,
|
|
560
|
+
data_byte_size=abi_type.sub,
|
|
561
|
+
)
|
|
562
|
+
|
|
563
|
+
|
|
564
|
+
class ByteStringEncoder(BaseEncoder):
|
|
565
|
+
is_dynamic = True
|
|
566
|
+
|
|
567
|
+
@classmethod
|
|
568
|
+
def validate_value(cls, value: Any) -> None:
|
|
569
|
+
if not is_bytes(value):
|
|
570
|
+
cls.invalidate_value(value)
|
|
571
|
+
|
|
572
|
+
@classmethod
|
|
573
|
+
def encode(cls, value: bytes) -> bytes:
|
|
574
|
+
cls.validate_value(value)
|
|
575
|
+
value_length = len(value)
|
|
576
|
+
|
|
577
|
+
encoded_size = encode_uint_256(value_length)
|
|
578
|
+
padded_value = zpad_right(value, ceil32(value_length))
|
|
579
|
+
|
|
580
|
+
return encoded_size + padded_value
|
|
581
|
+
|
|
582
|
+
__call__: ClassVar[Callable[[Type[Self], bytes], bytes]] = encode
|
|
583
|
+
|
|
584
|
+
@parse_type_str("bytes")
|
|
585
|
+
def from_type_str(cls, abi_type, registry):
|
|
586
|
+
return cls() # type: ignore [misc]
|
|
587
|
+
|
|
588
|
+
|
|
589
|
+
class PackedByteStringEncoder(ByteStringEncoder):
|
|
590
|
+
is_dynamic = False
|
|
591
|
+
|
|
592
|
+
@classmethod
|
|
593
|
+
def encode(cls, value: bytes) -> bytes:
|
|
594
|
+
cls.validate_value(value)
|
|
595
|
+
return value
|
|
596
|
+
|
|
597
|
+
__call__ = encode
|
|
598
|
+
|
|
599
|
+
|
|
600
|
+
class TextStringEncoder(BaseEncoder):
|
|
601
|
+
is_dynamic = True
|
|
602
|
+
|
|
603
|
+
@classmethod
|
|
604
|
+
def validate_value(cls, value: Any) -> None:
|
|
605
|
+
if not is_text(value):
|
|
606
|
+
cls.invalidate_value(value)
|
|
607
|
+
|
|
608
|
+
@classmethod
|
|
609
|
+
def encode(cls, value: str) -> bytes:
|
|
610
|
+
cls.validate_value(value)
|
|
611
|
+
|
|
612
|
+
value_as_bytes = codecs.encode(value, "utf8")
|
|
613
|
+
value_length = len(value_as_bytes)
|
|
614
|
+
|
|
615
|
+
encoded_size = encode_uint_256(value_length)
|
|
616
|
+
padded_value = zpad_right(value_as_bytes, ceil32(value_length))
|
|
617
|
+
|
|
618
|
+
return encoded_size + padded_value
|
|
619
|
+
|
|
620
|
+
__call__: ClassVar[Callable[[Type[Self], str], bytes]] = encode
|
|
621
|
+
|
|
622
|
+
@parse_type_str("string")
|
|
623
|
+
def from_type_str(cls, abi_type, registry):
|
|
624
|
+
return cls() # type: ignore [misc]
|
|
625
|
+
|
|
626
|
+
|
|
627
|
+
class PackedTextStringEncoder(TextStringEncoder):
|
|
628
|
+
is_dynamic = False
|
|
629
|
+
|
|
630
|
+
@classmethod
|
|
631
|
+
def encode(cls, value: str) -> bytes:
|
|
632
|
+
cls.validate_value(value)
|
|
633
|
+
return codecs.encode(value, "utf8")
|
|
634
|
+
|
|
635
|
+
__call__ = encode
|
|
636
|
+
|
|
637
|
+
|
|
638
|
+
class BaseArrayEncoder(BaseEncoder):
|
|
639
|
+
item_encoder: BaseEncoder = None
|
|
640
|
+
|
|
641
|
+
def validate(self) -> None:
|
|
642
|
+
super().validate()
|
|
643
|
+
|
|
644
|
+
if self.item_encoder is None:
|
|
645
|
+
raise ValueError("`item_encoder` may not be none")
|
|
646
|
+
|
|
647
|
+
def validate_value(self, value: Any) -> None:
|
|
648
|
+
if not is_list_like(value):
|
|
649
|
+
self.invalidate_value(
|
|
650
|
+
value,
|
|
651
|
+
msg="must be list-like such as array or tuple",
|
|
652
|
+
)
|
|
653
|
+
|
|
654
|
+
item_encoder = self.item_encoder
|
|
655
|
+
for item in value:
|
|
656
|
+
item_encoder.validate_value(item)
|
|
657
|
+
|
|
658
|
+
def encode_elements(self, value: Sequence[Any]) -> bytes:
|
|
659
|
+
self.validate_value(value)
|
|
660
|
+
return encode_elements(self.item_encoder, value)
|
|
661
|
+
|
|
662
|
+
@parse_type_str(with_arrlist=True)
|
|
663
|
+
def from_type_str(cls, abi_type, registry):
|
|
664
|
+
item_encoder = registry.get_encoder(abi_type.item_type.to_type_str())
|
|
665
|
+
|
|
666
|
+
array_spec = abi_type.arrlist[-1]
|
|
667
|
+
if len(array_spec) == 1:
|
|
668
|
+
# If array dimension is fixed
|
|
669
|
+
return SizedArrayEncoder(
|
|
670
|
+
array_size=array_spec[0],
|
|
671
|
+
item_encoder=item_encoder,
|
|
672
|
+
)
|
|
673
|
+
else:
|
|
674
|
+
# If array dimension is dynamic
|
|
675
|
+
return DynamicArrayEncoder(item_encoder=item_encoder)
|
|
676
|
+
|
|
677
|
+
|
|
678
|
+
class PackedArrayEncoder(BaseArrayEncoder):
|
|
679
|
+
array_size = None
|
|
680
|
+
|
|
681
|
+
def validate_value(self, value: Any) -> None:
|
|
682
|
+
super().validate_value(value)
|
|
683
|
+
|
|
684
|
+
array_size = self.array_size
|
|
685
|
+
if array_size is not None and len(value) != array_size:
|
|
686
|
+
self.invalidate_value(
|
|
687
|
+
value,
|
|
688
|
+
exc=ValueOutOfBounds,
|
|
689
|
+
msg=f"value has {len(value)} items when {array_size} were expected",
|
|
690
|
+
)
|
|
691
|
+
|
|
692
|
+
def encode(self, value: Sequence[Any]) -> bytes:
|
|
693
|
+
return encode_elements(self.item_encoder, value)
|
|
694
|
+
|
|
695
|
+
__call__ = encode
|
|
696
|
+
|
|
697
|
+
@parse_type_str(with_arrlist=True)
|
|
698
|
+
def from_type_str(cls, abi_type, registry):
|
|
699
|
+
item_encoder = registry.get_encoder(abi_type.item_type.to_type_str())
|
|
700
|
+
|
|
701
|
+
array_spec = abi_type.arrlist[-1]
|
|
702
|
+
if len(array_spec) == 1:
|
|
703
|
+
return cls(
|
|
704
|
+
array_size=array_spec[0],
|
|
705
|
+
item_encoder=item_encoder,
|
|
706
|
+
)
|
|
707
|
+
else:
|
|
708
|
+
return cls(item_encoder=item_encoder)
|
|
709
|
+
|
|
710
|
+
|
|
711
|
+
class SizedArrayEncoder(BaseArrayEncoder):
|
|
712
|
+
array_size = None
|
|
713
|
+
|
|
714
|
+
def __init__(self, **kwargs: Any) -> None:
|
|
715
|
+
super().__init__(**kwargs)
|
|
716
|
+
|
|
717
|
+
self.is_dynamic = self.item_encoder.is_dynamic
|
|
718
|
+
|
|
719
|
+
def validate(self) -> None:
|
|
720
|
+
super().validate()
|
|
721
|
+
|
|
722
|
+
if self.array_size is None:
|
|
723
|
+
raise ValueError("`array_size` may not be none")
|
|
724
|
+
|
|
725
|
+
def validate_value(self, value: Any) -> None:
|
|
726
|
+
super().validate_value(value)
|
|
727
|
+
|
|
728
|
+
if len(value) != self.array_size:
|
|
729
|
+
self.invalidate_value(
|
|
730
|
+
value,
|
|
731
|
+
exc=ValueOutOfBounds,
|
|
732
|
+
msg=f"value has {len(value)} items when {self.array_size} were "
|
|
733
|
+
"expected",
|
|
734
|
+
)
|
|
735
|
+
|
|
736
|
+
def encode(self, value: Sequence[Any]) -> bytes:
|
|
737
|
+
return encode_elements(self.item_encoder, value)
|
|
738
|
+
|
|
739
|
+
__call__ = encode
|
|
740
|
+
|
|
741
|
+
|
|
742
|
+
class DynamicArrayEncoder(BaseArrayEncoder):
|
|
743
|
+
is_dynamic = True
|
|
744
|
+
|
|
745
|
+
def encode(self, value: Sequence[Any]) -> bytes:
|
|
746
|
+
return encode_elements_dynamic(self.item_encoder, value)
|