mirror of
https://github.com/0glabs/0g-storage-node.git
synced 2025-01-24 22:15:17 +00:00
616 lines
14 KiB
Python
616 lines
14 KiB
Python
|
import coincurve
|
||
|
import random
|
||
|
import rlp
|
||
|
import sha3 as _sha3
|
||
|
|
||
|
from eth_utils import decode_hex, int_to_big_endian, big_endian_to_int
|
||
|
from eth_utils import encode_hex as encode_hex_0x
|
||
|
from py_ecc.secp256k1 import privtopub, ecdsa_raw_sign, ecdsa_raw_recover
|
||
|
from rlp.sedes import big_endian_int, BigEndianInt, Binary
|
||
|
from rlp.utils import ALL_BYTES
|
||
|
|
||
|
|
||
|
def sha3_256(x):
|
||
|
return _sha3.keccak_256(x).digest()
|
||
|
|
||
|
|
||
|
class Memoize:
|
||
|
def __init__(self, fn):
|
||
|
self.fn = fn
|
||
|
self.memo = {}
|
||
|
|
||
|
def __call__(self, *args):
|
||
|
if args not in self.memo:
|
||
|
self.memo[args] = self.fn(*args)
|
||
|
return self.memo[args]
|
||
|
|
||
|
|
||
|
TT256 = 2**256
|
||
|
TT256M1 = 2**256 - 1
|
||
|
TT255 = 2**255
|
||
|
SECP256K1P = 2**256 - 4294968273
|
||
|
|
||
|
|
||
|
def is_numeric(x):
|
||
|
return isinstance(x, int)
|
||
|
|
||
|
|
||
|
def is_string(x):
|
||
|
return isinstance(x, bytes)
|
||
|
|
||
|
|
||
|
def to_string(value):
|
||
|
if isinstance(value, bytes):
|
||
|
return value
|
||
|
if isinstance(value, str):
|
||
|
return bytes(value, "utf-8")
|
||
|
if isinstance(value, int):
|
||
|
return bytes(str(value), "utf-8")
|
||
|
|
||
|
|
||
|
def int_to_bytes(value):
|
||
|
if isinstance(value, bytes):
|
||
|
return value
|
||
|
return int_to_big_endian(value)
|
||
|
|
||
|
|
||
|
def to_string_for_regexp(value):
|
||
|
return str(to_string(value), "utf-8")
|
||
|
|
||
|
|
||
|
unicode = str
|
||
|
|
||
|
|
||
|
def bytearray_to_bytestr(value):
|
||
|
return bytes(value)
|
||
|
|
||
|
|
||
|
def encode_int32(v):
|
||
|
return v.to_bytes(32, byteorder="big")
|
||
|
|
||
|
|
||
|
def bytes_to_int(value):
|
||
|
return int.from_bytes(value, byteorder="big")
|
||
|
|
||
|
|
||
|
def str_to_bytes(value):
|
||
|
if isinstance(value, bytearray):
|
||
|
value = bytes(value)
|
||
|
if isinstance(value, bytes):
|
||
|
return value
|
||
|
return bytes(value, "utf-8")
|
||
|
|
||
|
|
||
|
def ascii_chr(n):
|
||
|
return ALL_BYTES[n]
|
||
|
|
||
|
|
||
|
def encode_hex(n):
|
||
|
if isinstance(n, str):
|
||
|
return encode_hex(n.encode("ascii"))
|
||
|
return encode_hex_0x(n)[2:]
|
||
|
|
||
|
|
||
|
def ecrecover_to_pub(rawhash, v, r, s):
|
||
|
if coincurve and hasattr(coincurve, "PublicKey"):
|
||
|
try:
|
||
|
pk = coincurve.PublicKey.from_signature_and_message(
|
||
|
zpad(bytearray_to_bytestr(int_to_32bytearray(r)), 32)
|
||
|
+ zpad(bytearray_to_bytestr(int_to_32bytearray(s)), 32)
|
||
|
+ ascii_chr(v - 27),
|
||
|
rawhash,
|
||
|
hasher=None,
|
||
|
)
|
||
|
pub = pk.format(compressed=False)[1:]
|
||
|
x, y = pk.point()
|
||
|
except BaseException:
|
||
|
x, y = 0, 0
|
||
|
pub = b"\x00" * 64
|
||
|
else:
|
||
|
result = ecdsa_raw_recover(rawhash, (v, r, s))
|
||
|
if result:
|
||
|
x, y = result
|
||
|
pub = encode_int32(x) + encode_int32(y)
|
||
|
else:
|
||
|
raise ValueError("Invalid VRS")
|
||
|
assert len(pub) == 64
|
||
|
return pub, x, y
|
||
|
|
||
|
|
||
|
def ecsign(rawhash, key):
|
||
|
if coincurve and hasattr(coincurve, "PrivateKey"):
|
||
|
pk = coincurve.PrivateKey(key)
|
||
|
signature = pk.sign_recoverable(rawhash, hasher=None)
|
||
|
v = safe_ord(signature[64]) + 27
|
||
|
r = big_endian_to_int(signature[0:32])
|
||
|
s = big_endian_to_int(signature[32:64])
|
||
|
else:
|
||
|
v, r, s = ecdsa_raw_sign(rawhash, key)
|
||
|
return v, r, s
|
||
|
|
||
|
|
||
|
def ec_random_keys():
|
||
|
priv_key = random.randint(0, 2**256).to_bytes(32, "big")
|
||
|
pub_key = privtopub(priv_key)
|
||
|
return priv_key, pub_key
|
||
|
|
||
|
|
||
|
def convert_to_nodeid(signature, challenge):
|
||
|
r = big_endian_to_int(signature[:32])
|
||
|
s = big_endian_to_int(signature[32:64])
|
||
|
v = big_endian_to_int(signature[64:]) + 27
|
||
|
signed = int_to_bytes(challenge)
|
||
|
h_signed = sha3_256(signed)
|
||
|
return ecrecover_to_pub(h_signed, v, r, s)
|
||
|
|
||
|
|
||
|
def get_nodeid(node):
|
||
|
challenge = random.randint(0, 2**32 - 1)
|
||
|
signature = node.getnodeid(list(int_to_bytes(challenge)))
|
||
|
return convert_to_nodeid(signature, challenge)
|
||
|
|
||
|
|
||
|
def mk_contract_address(sender, nonce):
|
||
|
return sha3(rlp.encode([normalize_address(sender), nonce]))[12:]
|
||
|
|
||
|
|
||
|
def mk_metropolis_contract_address(sender, initcode):
|
||
|
return sha3(normalize_address(sender) + initcode)[12:]
|
||
|
|
||
|
|
||
|
def safe_ord(value):
|
||
|
if isinstance(value, int):
|
||
|
return value
|
||
|
else:
|
||
|
return ord(value)
|
||
|
|
||
|
|
||
|
# decorator
|
||
|
|
||
|
|
||
|
def debug(label):
|
||
|
def deb(f):
|
||
|
def inner(*args, **kwargs):
|
||
|
i = random.randrange(1000000)
|
||
|
print(label, i, "start", args)
|
||
|
x = f(*args, **kwargs)
|
||
|
print(label, i, "end", x)
|
||
|
return x
|
||
|
|
||
|
return inner
|
||
|
|
||
|
return deb
|
||
|
|
||
|
|
||
|
def flatten(li):
|
||
|
o = []
|
||
|
for l in li:
|
||
|
o.extend(l)
|
||
|
return o
|
||
|
|
||
|
|
||
|
def bytearray_to_int(arr):
|
||
|
o = 0
|
||
|
for a in arr:
|
||
|
o = (o << 8) + a
|
||
|
return o
|
||
|
|
||
|
|
||
|
def int_to_32bytearray(i):
|
||
|
o = [0] * 32
|
||
|
for x in range(32):
|
||
|
o[31 - x] = i & 0xFF
|
||
|
i >>= 8
|
||
|
return o
|
||
|
|
||
|
|
||
|
# sha3_count = [0]
|
||
|
|
||
|
|
||
|
def sha3(seed):
|
||
|
return sha3_256(to_string(seed))
|
||
|
|
||
|
|
||
|
assert (
|
||
|
encode_hex(sha3(b""))
|
||
|
== "c5d2460186f7233c927e7db2dcc703c0e500b653ca82273b7bfad8045d85a470"
|
||
|
)
|
||
|
assert (
|
||
|
encode_hex(sha3(b"\x00" * 256))
|
||
|
== "d397b3b043d87fcd6fad1291ff0bfd16401c274896d8c63a923727f077b8e0b5"
|
||
|
)
|
||
|
|
||
|
|
||
|
@Memoize
|
||
|
def priv_to_addr(k):
|
||
|
k = normalize_key(k)
|
||
|
x, y = privtopub(k)
|
||
|
addr = bytearray(sha3(encode_int32(x) + encode_int32(y))[12:])
|
||
|
addr[0] &= 0x0F
|
||
|
addr[0] |= 0x10
|
||
|
return bytes(addr)
|
||
|
|
||
|
|
||
|
def priv_to_pub(k):
|
||
|
k = normalize_key(k)
|
||
|
x, y = privtopub(k)
|
||
|
return bytes(encode_int32(x) + encode_int32(y))
|
||
|
|
||
|
|
||
|
def pub_to_addr(k):
|
||
|
x = big_endian_to_int(decode_hex(k[2:34]))
|
||
|
y = big_endian_to_int(decode_hex(k[34:66]))
|
||
|
addr = sha3(encode_int32(x) + encode_int32(y))[12:]
|
||
|
addr[0] &= 0x0F
|
||
|
addr[0] |= 0x10
|
||
|
return bytes(addr)
|
||
|
|
||
|
|
||
|
def checksum_encode(addr): # Takes a 20-byte binary address as input
|
||
|
addr = normalize_address(addr)
|
||
|
o = ""
|
||
|
v = big_endian_to_int(sha3(encode_hex(addr)))
|
||
|
for i, c in enumerate(encode_hex(addr)):
|
||
|
if c in "0123456789":
|
||
|
o += c
|
||
|
else:
|
||
|
o += c.upper() if (v & (2 ** (255 - 4 * i))) else c.lower()
|
||
|
return "0x" + o
|
||
|
|
||
|
|
||
|
def check_checksum(addr):
|
||
|
return checksum_encode(normalize_address(addr)) == addr
|
||
|
|
||
|
|
||
|
def normalize_address(x, allow_blank=False):
|
||
|
if is_numeric(x):
|
||
|
return int_to_addr(x)
|
||
|
if allow_blank and x in {"", b""}:
|
||
|
return b""
|
||
|
if len(x) in (42, 50) and x[:2] in {"0x", b"0x"}:
|
||
|
x = x[2:]
|
||
|
if len(x) in (40, 48):
|
||
|
x = decode_hex(x)
|
||
|
if len(x) == 24:
|
||
|
assert len(x) == 24 and sha3(x[:20])[:4] == x[-4:]
|
||
|
x = x[:20]
|
||
|
if len(x) != 20:
|
||
|
raise Exception("Invalid address format: %r" % x)
|
||
|
return x
|
||
|
|
||
|
|
||
|
def normalize_key(key):
|
||
|
if is_numeric(key):
|
||
|
o = encode_int32(key)
|
||
|
elif len(key) == 32:
|
||
|
o = key
|
||
|
elif len(key) == 64:
|
||
|
o = decode_hex(key)
|
||
|
elif len(key) == 66 and key[:2] == "0x":
|
||
|
o = decode_hex(key[2:])
|
||
|
else:
|
||
|
raise Exception("Invalid key format: %r" % key)
|
||
|
if o == b"\x00" * 32:
|
||
|
raise Exception("Zero privkey invalid")
|
||
|
return o
|
||
|
|
||
|
|
||
|
def zpad(x, l):
|
||
|
"""Left zero pad value `x` at least to length `l`.
|
||
|
|
||
|
>>> zpad('', 1)
|
||
|
'\x00'
|
||
|
>>> zpad('\xca\xfe', 4)
|
||
|
'\x00\x00\xca\xfe'
|
||
|
>>> zpad('\xff', 1)
|
||
|
'\xff'
|
||
|
>>> zpad('\xca\xfe', 2)
|
||
|
'\xca\xfe'
|
||
|
"""
|
||
|
return b"\x00" * max(0, l - len(x)) + x
|
||
|
|
||
|
|
||
|
def rzpad(value, total_length):
|
||
|
"""Right zero pad value `x` at least to length `l`.
|
||
|
|
||
|
>>> zpad('', 1)
|
||
|
'\x00'
|
||
|
>>> zpad('\xca\xfe', 4)
|
||
|
'\xca\xfe\x00\x00'
|
||
|
>>> zpad('\xff', 1)
|
||
|
'\xff'
|
||
|
>>> zpad('\xca\xfe', 2)
|
||
|
'\xca\xfe'
|
||
|
"""
|
||
|
return value + b"\x00" * max(0, total_length - len(value))
|
||
|
|
||
|
|
||
|
def int_to_addr(x):
|
||
|
o = [b""] * 20
|
||
|
for i in range(20):
|
||
|
o[19 - i] = ascii_chr(x & 0xFF)
|
||
|
x >>= 8
|
||
|
return b"".join(o)
|
||
|
|
||
|
|
||
|
def coerce_addr_to_bin(x):
|
||
|
if is_numeric(x):
|
||
|
return encode_hex(zpad(big_endian_int.serialize(x), 20))
|
||
|
elif len(x) == 40 or len(x) == 0:
|
||
|
return decode_hex(x)
|
||
|
else:
|
||
|
return zpad(x, 20)[-20:]
|
||
|
|
||
|
|
||
|
def coerce_addr_to_hex(x):
|
||
|
if is_numeric(x):
|
||
|
return encode_hex(zpad(big_endian_int.serialize(x), 20))
|
||
|
elif len(x) == 40 or len(x) == 0:
|
||
|
return x
|
||
|
else:
|
||
|
return encode_hex(zpad(x, 20)[-20:])
|
||
|
|
||
|
|
||
|
def coerce_to_int(x):
|
||
|
if is_numeric(x):
|
||
|
return x
|
||
|
elif len(x) == 40:
|
||
|
return big_endian_to_int(decode_hex(x))
|
||
|
else:
|
||
|
return big_endian_to_int(x)
|
||
|
|
||
|
|
||
|
def coerce_to_bytes(x):
|
||
|
if is_numeric(x):
|
||
|
return big_endian_int.serialize(x)
|
||
|
elif len(x) == 40:
|
||
|
return decode_hex(x)
|
||
|
else:
|
||
|
return x
|
||
|
|
||
|
|
||
|
def parse_int_or_hex(s):
|
||
|
if is_numeric(s):
|
||
|
return s
|
||
|
elif s[:2] in (b"0x", "0x"):
|
||
|
s = to_string(s)
|
||
|
tail = (b"0" if len(s) % 2 else b"") + s[2:]
|
||
|
return big_endian_to_int(decode_hex(tail))
|
||
|
else:
|
||
|
return int(s)
|
||
|
|
||
|
|
||
|
def ceil32(x):
|
||
|
return x if x % 32 == 0 else x + 32 - (x % 32)
|
||
|
|
||
|
|
||
|
def to_signed(i):
|
||
|
return i if i < TT255 else i - TT256
|
||
|
|
||
|
|
||
|
def sha3rlp(x):
|
||
|
return sha3(rlp.encode(x))
|
||
|
|
||
|
|
||
|
# Format encoders/decoders for bin, addr, int
|
||
|
|
||
|
|
||
|
def decode_bin(v):
|
||
|
"""decodes a bytearray from serialization"""
|
||
|
if not is_string(v):
|
||
|
raise Exception("Value must be binary, not RLP array")
|
||
|
return v
|
||
|
|
||
|
|
||
|
def decode_addr(v):
|
||
|
"""decodes an address from serialization"""
|
||
|
if len(v) not in [0, 20]:
|
||
|
raise Exception("Serialized addresses must be empty or 20 bytes long!")
|
||
|
return encode_hex(v)
|
||
|
|
||
|
|
||
|
def decode_int(v):
|
||
|
"""decodes and integer from serialization"""
|
||
|
if len(v) > 0 and (v[0] == b"\x00" or v[0] == 0):
|
||
|
raise Exception("No leading zero bytes allowed for integers")
|
||
|
return big_endian_to_int(v)
|
||
|
|
||
|
|
||
|
def decode_int256(v):
|
||
|
return big_endian_to_int(v)
|
||
|
|
||
|
|
||
|
def encode_bin(v):
|
||
|
"""encodes a bytearray into serialization"""
|
||
|
return v
|
||
|
|
||
|
|
||
|
def encode_root(v):
|
||
|
"""encodes a trie root into serialization"""
|
||
|
return v
|
||
|
|
||
|
|
||
|
def encode_int(v):
|
||
|
"""encodes an integer into serialization"""
|
||
|
if not is_numeric(v) or v < 0 or v >= TT256:
|
||
|
raise Exception("Integer invalid or out of range: %r" % v)
|
||
|
return int_to_big_endian(v)
|
||
|
|
||
|
|
||
|
def encode_int256(v):
|
||
|
return zpad(int_to_big_endian(v), 256)
|
||
|
|
||
|
|
||
|
def scan_bin(v):
|
||
|
if v[:2] in ("0x", b"0x"):
|
||
|
return decode_hex(v[2:])
|
||
|
else:
|
||
|
return decode_hex(v)
|
||
|
|
||
|
|
||
|
def scan_int(v):
|
||
|
if v[:2] in ("0x", b"0x"):
|
||
|
return big_endian_to_int(decode_hex(v[2:]))
|
||
|
else:
|
||
|
return int(v)
|
||
|
|
||
|
|
||
|
# Decoding from RLP serialization
|
||
|
decoders = {
|
||
|
"bin": decode_bin,
|
||
|
"addr": decode_addr,
|
||
|
"int": decode_int,
|
||
|
"int256b": decode_int256,
|
||
|
}
|
||
|
|
||
|
# Encoding to RLP serialization
|
||
|
encoders = {
|
||
|
"bin": encode_bin,
|
||
|
"int": encode_int,
|
||
|
"trie_root": encode_root,
|
||
|
"int256b": encode_int256,
|
||
|
}
|
||
|
|
||
|
# Encoding to printable format
|
||
|
printers = {
|
||
|
"bin": lambda v: "0x" + encode_hex(v),
|
||
|
"addr": lambda v: v,
|
||
|
"int": lambda v: to_string(v),
|
||
|
"trie_root": lambda v: encode_hex(v),
|
||
|
"int256b": lambda x: encode_hex(zpad(encode_int256(x), 256)),
|
||
|
}
|
||
|
|
||
|
# Decoding from printable format
|
||
|
scanners = {
|
||
|
"bin": scan_bin,
|
||
|
"addr": lambda x: x[2:] if x[:2] in (b"0x", "0x") else x,
|
||
|
"int": scan_int,
|
||
|
"trie_root": lambda x: scan_bin,
|
||
|
"int256b": lambda x: big_endian_to_int(decode_hex(x)),
|
||
|
}
|
||
|
|
||
|
|
||
|
def int_to_hex(x):
|
||
|
o = encode_hex(encode_int(x))
|
||
|
return "0x" + (o[1:] if (len(o) > 0 and o[0] == b"0") else o)
|
||
|
|
||
|
|
||
|
def remove_0x_head(s):
|
||
|
return s[2:] if s[:2] in (b"0x", "0x") else s
|
||
|
|
||
|
|
||
|
def parse_as_bin(s):
|
||
|
return decode_hex(s[2:] if s[:2] == "0x" else s)
|
||
|
|
||
|
|
||
|
def parse_as_int(s):
|
||
|
return s if is_numeric(s) else int("0" + s[2:], 16) if s[:2] == "0x" else int(s)
|
||
|
|
||
|
|
||
|
def print_func_call(ignore_first_arg=False, max_call_number=100):
|
||
|
"""utility function to facilitate debug, it will print input args before
|
||
|
function call, and print return value after function call
|
||
|
|
||
|
usage:
|
||
|
|
||
|
@print_func_call
|
||
|
def some_func_to_be_debu():
|
||
|
pass
|
||
|
|
||
|
:param ignore_first_arg: whether print the first arg or not.
|
||
|
useful when ignore the `self` parameter of an object method call
|
||
|
"""
|
||
|
from functools import wraps
|
||
|
|
||
|
def display(x):
|
||
|
x = to_string(x)
|
||
|
try:
|
||
|
x.decode("ascii")
|
||
|
except BaseException:
|
||
|
return "NON_PRINTABLE"
|
||
|
return x
|
||
|
|
||
|
local = {"call_number": 0}
|
||
|
|
||
|
def inner(f):
|
||
|
@wraps(f)
|
||
|
def wrapper(*args, **kwargs):
|
||
|
local["call_number"] += 1
|
||
|
tmp_args = args[1:] if ignore_first_arg and len(args) else args
|
||
|
this_call_number = local["call_number"]
|
||
|
print(
|
||
|
(
|
||
|
"{0}#{1} args: {2}, {3}".format(
|
||
|
f.__name__,
|
||
|
this_call_number,
|
||
|
", ".join([display(x) for x in tmp_args]),
|
||
|
", ".join(
|
||
|
display(key) + "=" + to_string(value)
|
||
|
for key, value in kwargs.items()
|
||
|
),
|
||
|
)
|
||
|
)
|
||
|
)
|
||
|
res = f(*args, **kwargs)
|
||
|
print(
|
||
|
(
|
||
|
"{0}#{1} return: {2}".format(
|
||
|
f.__name__, this_call_number, display(res)
|
||
|
)
|
||
|
)
|
||
|
)
|
||
|
|
||
|
if local["call_number"] > 100:
|
||
|
raise Exception("Touch max call number!")
|
||
|
return res
|
||
|
|
||
|
return wrapper
|
||
|
|
||
|
return inner
|
||
|
|
||
|
|
||
|
def dump_state(trie):
|
||
|
res = ""
|
||
|
for k, v in list(trie.to_dict().items()):
|
||
|
res += "%r:%r\n" % (encode_hex(k), encode_hex(v))
|
||
|
return res
|
||
|
|
||
|
|
||
|
class Denoms:
|
||
|
def __init__(self):
|
||
|
self.wei = 1
|
||
|
self.babbage = 10**3
|
||
|
self.ada = 10**3
|
||
|
self.kwei = 10**3
|
||
|
self.lovelace = 10**6
|
||
|
self.mwei = 10**6
|
||
|
self.shannon = 10**9
|
||
|
self.gwei = 10**9
|
||
|
self.szabo = 10**12
|
||
|
self.finney = 10**15
|
||
|
self.mether = 10**15
|
||
|
self.ether = 10**18
|
||
|
self.turing = 2**256 - 1
|
||
|
|
||
|
|
||
|
denoms = Denoms()
|
||
|
|
||
|
address = Binary.fixed_length(20, allow_empty=True)
|
||
|
int20 = BigEndianInt(20)
|
||
|
int32 = BigEndianInt(32)
|
||
|
int256 = BigEndianInt(256)
|
||
|
hash32 = Binary.fixed_length(32)
|
||
|
hash20 = Binary.fixed_length(20)
|
||
|
trie_root = Binary.fixed_length(32, allow_empty=True)
|
||
|
|
||
|
|
||
|
class bcolors:
|
||
|
HEADER = "\033[95m"
|
||
|
OKBLUE = "\033[94m"
|
||
|
OKGREEN = "\033[92m"
|
||
|
WARNING = "\033[91m"
|
||
|
FAIL = "\033[91m"
|
||
|
ENDC = "\033[0m"
|
||
|
BOLD = "\033[1m"
|
||
|
UNDERLINE = "\033[4m"
|