170 lines
4.9 KiB
Python
170 lines
4.9 KiB
Python
|
#!python3
|
|||
|
|
|||
|
"""Token balance query script
|
|||
|
|
|||
|
.. moduleauthor:: Louis Holbrook <dev@holbrook.no>
|
|||
|
.. pgp:: 0826EDA1702D1E87C6E2875121D2E7BB88C2A746
|
|||
|
|
|||
|
"""
|
|||
|
|
|||
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
|||
|
|
|||
|
# standard imports
|
|||
|
import datetime
|
|||
|
import sys
|
|||
|
import os
|
|||
|
import json
|
|||
|
import argparse
|
|||
|
import logging
|
|||
|
|
|||
|
# third-party imports
|
|||
|
from hexathon import (
|
|||
|
add_0x,
|
|||
|
strip_0x,
|
|||
|
even,
|
|||
|
)
|
|||
|
import sha3
|
|||
|
|
|||
|
# local imports
|
|||
|
from chainlib.eth.address import (
|
|||
|
to_checksum_address,
|
|||
|
is_checksum_address,
|
|||
|
)
|
|||
|
from chainlib.eth.chain import network_id
|
|||
|
from chainlib.eth.block import (
|
|||
|
block_latest,
|
|||
|
block_by_number,
|
|||
|
Block,
|
|||
|
)
|
|||
|
from chainlib.eth.tx import count
|
|||
|
from chainlib.eth.connection import EthHTTPConnection
|
|||
|
from chainlib.eth.gas import (
|
|||
|
OverrideGasOracle,
|
|||
|
balance,
|
|||
|
price,
|
|||
|
)
|
|||
|
from chainlib.jsonrpc import (
|
|||
|
IntSequenceGenerator,
|
|||
|
)
|
|||
|
from chainlib.chain import ChainSpec
|
|||
|
|
|||
|
BLOCK_SAMPLES = 10
|
|||
|
|
|||
|
logging.basicConfig(level=logging.WARNING)
|
|||
|
logg = logging.getLogger()
|
|||
|
|
|||
|
default_eth_provider = os.environ.get('RPC_PROVIDER')
|
|||
|
if default_eth_provider == None:
|
|||
|
default_eth_provider = os.environ.get('ETH_PROVIDER', 'http://localhost:8545')
|
|||
|
|
|||
|
argparser = argparse.ArgumentParser()
|
|||
|
argparser.add_argument('-p', '--provider', dest='p', default=default_eth_provider, type=str, help='Web3 provider url (http only)')
|
|||
|
argparser.add_argument('-i', '--chain-spec', dest='i', type=str, default='evm:ethereum:1', help='Chain specification string')
|
|||
|
argparser.add_argument('-H', '--human', dest='human', action='store_true', help='Use human-friendly formatting')
|
|||
|
argparser.add_argument('-u', '--unsafe', dest='u', action='store_true', help='Auto-convert address to checksum adddress')
|
|||
|
argparser.add_argument('-l', '--long', dest='l', action='store_true', help='Calculate averages through sampling of blocks and txs')
|
|||
|
argparser.add_argument('-v', action='store_true', help='Be verbose')
|
|||
|
argparser.add_argument('--seq', action='store_true', help='Use sequential rpc ids')
|
|||
|
argparser.add_argument('-vv', action='store_true', help='Be more verbose')
|
|||
|
argparser.add_argument('-y', '--key-file', dest='y', type=str, help='Include summary for keyfile')
|
|||
|
argparser.add_argument('address', nargs='?', type=str, help='Include summary for address (conflicts with -y)')
|
|||
|
args = argparser.parse_args()
|
|||
|
|
|||
|
|
|||
|
if args.vv:
|
|||
|
logg.setLevel(logging.DEBUG)
|
|||
|
elif args.v:
|
|||
|
logg.setLevel(logging.INFO)
|
|||
|
|
|||
|
signer = None
|
|||
|
holder_address = None
|
|||
|
if args.address != None:
|
|||
|
if not args.u and not is_checksum_address(args.address):
|
|||
|
raise ValueError('invalid checksum address {}'.format(args.address))
|
|||
|
holder_address = add_0x(args.address)
|
|||
|
elif args.y != None:
|
|||
|
f = open(args.y, 'r')
|
|||
|
o = json.load(f)
|
|||
|
f.close()
|
|||
|
holder_address = add_0x(to_checksum_address(o['address']))
|
|||
|
|
|||
|
rpc_id_generator = None
|
|||
|
if args.seq:
|
|||
|
rpc_id_generator = IntSequenceGenerator()
|
|||
|
|
|||
|
auth = None
|
|||
|
if os.environ.get('RPC_AUTHENTICATION') == 'basic':
|
|||
|
from chainlib.auth import BasicAuth
|
|||
|
auth = BasicAuth(os.environ['RPC_USERNAME'], os.environ['RPC_PASSWORD'])
|
|||
|
conn = EthHTTPConnection(args.p, auth=auth)
|
|||
|
|
|||
|
gas_oracle = OverrideGasOracle(conn)
|
|||
|
|
|||
|
token_symbol = 'eth'
|
|||
|
|
|||
|
chain_spec = ChainSpec.from_chain_str(args.i)
|
|||
|
|
|||
|
human = args.human
|
|||
|
|
|||
|
longmode = args.l
|
|||
|
|
|||
|
def main():
|
|||
|
o = network_id(id_generator=rpc_id_generator)
|
|||
|
r = conn.do(o)
|
|||
|
#if human:
|
|||
|
# n = format(n, ',')
|
|||
|
sys.stdout.write('Network id: {}\n'.format(r))
|
|||
|
|
|||
|
o = block_latest(id_generator=rpc_id_generator)
|
|||
|
r = conn.do(o)
|
|||
|
n = int(r, 16)
|
|||
|
first_block_number = n
|
|||
|
if human:
|
|||
|
n = format(n, ',')
|
|||
|
sys.stdout.write('Block: {}\n'.format(n))
|
|||
|
|
|||
|
o = block_by_number(first_block_number, False, id_generator=rpc_id_generator)
|
|||
|
r = conn.do(o)
|
|||
|
last_block = Block(r)
|
|||
|
last_timestamp = last_block.timestamp
|
|||
|
|
|||
|
if longmode:
|
|||
|
aggr_time = 0.0
|
|||
|
aggr_gas = 0
|
|||
|
for i in range(BLOCK_SAMPLES):
|
|||
|
o = block_by_number(first_block_number-i, False, id_generator=rpc_id_generator)
|
|||
|
r = conn.do(o)
|
|||
|
block = Block(r)
|
|||
|
aggr_time += last_block.timestamp - block.timestamp
|
|||
|
|
|||
|
gas_limit = int(r['gasLimit'], 16)
|
|||
|
aggr_gas += gas_limit
|
|||
|
|
|||
|
last_block = block
|
|||
|
last_timestamp = block.timestamp
|
|||
|
|
|||
|
n = int(aggr_gas / BLOCK_SAMPLES)
|
|||
|
if human:
|
|||
|
n = format(n, ',')
|
|||
|
|
|||
|
sys.stdout.write('Gaslimit: {}\n'.format(n))
|
|||
|
sys.stdout.write('Blocktime: {}\n'.format(aggr_time / BLOCK_SAMPLES))
|
|||
|
|
|||
|
o = price(id_generator=rpc_id_generator)
|
|||
|
r = conn.do(o)
|
|||
|
n = int(r, 16)
|
|||
|
if human:
|
|||
|
n = format(n, ',')
|
|||
|
sys.stdout.write('Gasprice: {}\n'.format(n))
|
|||
|
|
|||
|
if holder_address != None:
|
|||
|
o = count(holder_address)
|
|||
|
r = conn.do(o)
|
|||
|
n = int(r, 16)
|
|||
|
sys.stdout.write('Address: {}\n'.format(holder_address))
|
|||
|
sys.stdout.write('Nonce: {}\n'.format(n))
|
|||
|
|
|||
|
|
|||
|
if __name__ == '__main__':
|
|||
|
main()
|