WIP refactor send, checkgas, refillgas to use chainlib

This commit is contained in:
nolash 2021-03-18 11:57:26 +01:00
parent 9f2773e948
commit 2f07ea1395
Signed by untrusted user who does not match committer: lash
GPG Key ID: 21D2E7BB88C2A746
12 changed files with 141 additions and 290 deletions

View File

@ -23,7 +23,7 @@ from cic_eth.queue.tx import (
from cic_eth.queue.tx import create as queue_create from cic_eth.queue.tx import create as queue_create
from cic_eth.eth.util import unpack_signed_raw_tx from cic_eth.eth.util import unpack_signed_raw_tx
from cic_eth.eth.task import ( from cic_eth.eth.task import (
sign_tx, #sign_tx,
create_check_gas_task, create_check_gas_task,
) )

View File

@ -20,6 +20,7 @@ def upgrade():
op.create_table( op.create_table(
'nonce_task_reservation', 'nonce_task_reservation',
sa.Column('id', sa.Integer, primary_key=True), sa.Column('id', sa.Integer, primary_key=True),
sa.Column('address_hex', sa.String(42), nullable=False),
sa.Column('nonce', sa.Integer, nullable=False), sa.Column('nonce', sa.Integer, nullable=False),
sa.Column('key', sa.String, nullable=False), sa.Column('key', sa.String, nullable=False),
sa.Column('date_created', sa.DateTime, nullable=False), sa.Column('date_created', sa.DateTime, nullable=False),

View File

@ -20,6 +20,7 @@ def upgrade():
op.create_table( op.create_table(
'nonce_task_reservation', 'nonce_task_reservation',
sa.Column('id', sa.Integer, primary_key=True), sa.Column('id', sa.Integer, primary_key=True),
sa.Column('address_hex', sa.String(42), nullable=False),
sa.Column('nonce', sa.Integer, nullable=False), sa.Column('nonce', sa.Integer, nullable=False),
sa.Column('key', sa.String, nullable=False), sa.Column('key', sa.String, nullable=False),
sa.Column('date_created', sa.DateTime, nullable=False), sa.Column('date_created', sa.DateTime, nullable=False),

View File

@ -122,17 +122,19 @@ class NonceReservation(SessionBase):
__tablename__ = 'nonce_task_reservation' __tablename__ = 'nonce_task_reservation'
address_hex = Column(String(42))
nonce = Column(Integer) nonce = Column(Integer)
key = Column(String) key = Column(String)
date_created = Column(DateTime, default=datetime.datetime.utcnow) date_created = Column(DateTime, default=datetime.datetime.utcnow)
@staticmethod @staticmethod
def peek(key, session=None): def peek(address, key, session=None):
session = SessionBase.bind_session(session) session = SessionBase.bind_session(session)
q = session.query(NonceReservation) q = session.query(NonceReservation)
q = q.filter(NonceReservation.key==key) q = q.filter(NonceReservation.key==key)
q = q.filter(NonceReservation.address_hex==address)
o = q.first() o = q.first()
nonce = None nonce = None
@ -147,18 +149,19 @@ class NonceReservation(SessionBase):
@staticmethod @staticmethod
def release(key, session=None): def release(address, key, session=None):
session = SessionBase.bind_session(session) session = SessionBase.bind_session(session)
nonce = NonceReservation.peek(key, session=session) nonce = NonceReservation.peek(key, session=session)
q = session.query(NonceReservation) q = session.query(NonceReservation)
q = q.filter(NonceReservation.address_hex==address)
q = q.filter(NonceReservation.key==key) q = q.filter(NonceReservation.key==key)
o = q.first() o = q.first()
if o == None: if o == None:
raise IntegrityError('nonce for key {}'.format(nonce)) raise IntegrityError('nonce {} for key {} address {}: {}'.format(nonce, key, address))
SessionBase.release_session(session) SessionBase.release_session(session)
session.delete(o) session.delete(o)
@ -173,14 +176,15 @@ class NonceReservation(SessionBase):
def next(address, key, session=None): def next(address, key, session=None):
session = SessionBase.bind_session(session) session = SessionBase.bind_session(session)
if NonceReservation.peek(key, session) != None: if NonceReservation.peek(address, key, session) != None:
raise IntegrityError('nonce for key {}'.format(key)) raise IntegrityError('nonce for key {} address {}'.format(key, address))
nonce = Nonce.next(address, session=session) nonce = Nonce.next(address, session=session)
o = NonceReservation() o = NonceReservation()
o.nonce = nonce o.nonce = nonce
o.key = key o.key = key
o.address_hex = address
session.add(o) session.add(o)
SessionBase.release_session(session) SessionBase.release_session(session)

View File

@ -1,9 +1,8 @@
# standard imports # standard imports
import logging import logging
# third-party imports # external imports
import celery import celery
from cic_registry.chain import ChainSpec
from erc20_single_shot_faucet import Faucet from erc20_single_shot_faucet import Faucet
from chainlib.eth.constant import ZERO_ADDRESS from chainlib.eth.constant import ZERO_ADDRESS
from hexathon import strip_0x from hexathon import strip_0x
@ -14,6 +13,7 @@ from chainlib.eth.sign import (
) )
from chainlib.eth.address import to_checksum_address from chainlib.eth.address import to_checksum_address
from chainlib.eth.tx import TxFormat from chainlib.eth.tx import TxFormat
from chainlib.chain import ChainSpec
from eth_accounts_index import AccountRegistry from eth_accounts_index import AccountRegistry
# local import # local import
@ -196,7 +196,7 @@ def create(self, password, chain_str):
@celery_app.task(bind=True, throws=(RoleMissingError,), base=CriticalSQLAlchemyAndSignerTask) @celery_app.task(bind=True, throws=(RoleMissingError,), base=CriticalSQLAlchemyAndSignerTask)
def register(self, account_address, chain_str, writer_address=None): def register(self, account_address, chain_spec_dict, writer_address=None):
"""Creates a transaction to add the given address to the accounts index. """Creates a transaction to add the given address to the accounts index.
:param account_address: Ethereum address to add :param account_address: Ethereum address to add
@ -209,7 +209,7 @@ def register(self, account_address, chain_str, writer_address=None):
:returns: The account_address input param :returns: The account_address input param
:rtype: str, 0x-hex :rtype: str, 0x-hex
""" """
chain_spec = ChainSpec.from_chain_str(chain_str) chain_spec = ChainSpec.from_dict(chain_spec_dict)
session = self.create_session() session = self.create_session()
#session = SessionBase.create_session() #session = SessionBase.create_session()
@ -223,12 +223,6 @@ def register(self, account_address, chain_str, writer_address=None):
logg.debug('adding account address {} to index; writer {}'.format(account_address, writer_address)) logg.debug('adding account address {} to index; writer {}'.format(account_address, writer_address))
queue = self.request.delivery_info.get('routing_key') queue = self.request.delivery_info.get('routing_key')
# c = RpcClient(chain_spec, holder_address=writer_address)
# registry = safe_registry(c.w3)
# txf = AccountTxFactory(writer_address, c, registry=registry)
# tx_add = txf.add(account_address, chain_spec, self.request.root_id, session=session)
# (tx_hash_hex, tx_signed_raw_hex) = sign_and_register_tx(tx_add, chain_str, queue, 'cic_eth.eth.account.cache_account_data', session=session)
# Retrieve account index address # Retrieve account index address
rpc = RPCConnection.connect(chain_spec, 'default') rpc = RPCConnection.connect(chain_spec, 'default')
reg = CICRegistry(chain_spec, rpc) reg = CICRegistry(chain_spec, rpc)
@ -239,26 +233,27 @@ def register(self, account_address, chain_str, writer_address=None):
rpc_signer = RPCConnection.connect(chain_spec, 'signer') rpc_signer = RPCConnection.connect(chain_spec, 'signer')
nonce_oracle = self.create_nonce_oracle(writer_address, rpc) nonce_oracle = self.create_nonce_oracle(writer_address, rpc)
gas_oracle = self.create_gas_oracle(rpc) gas_oracle = self.create_gas_oracle(rpc)
account_registry = AccountRegistry(signer=rpc_signer, nonce_oracle=nonce_oracle, gas_oracle=gas_oracle) account_registry = AccountRegistry(signer=rpc_signer, nonce_oracle=nonce_oracle, gas_oracle=gas_oracle, chain_id=chain_spec.chain_id())
(tx_hash_hex, tx_signed_raw_hex) = account_registry.add(account_registry_address, writer_address, account_address, tx_format=TxFormat.RLP_SIGNED) (tx_hash_hex, tx_signed_raw_hex) = account_registry.add(account_registry_address, writer_address, account_address, tx_format=TxFormat.RLP_SIGNED)
#cache_task = 'cic_eth.eth.account.cache_account_data' # TODO: if cache task fails, task chain will not return
cache_task = 'cic_eth.eth.account.cache_account_data'
cache_task = None cache_task = None
# add transaction to queue # add transaction to queue
register_tx(tx_hash_hex, tx_signed_raw_hex, chain_str, queue, cache_task=cache_task, session=session) register_tx(tx_hash_hex, tx_signed_raw_hex, chain_spec, queue, cache_task=cache_task, session=session)
session.commit() session.commit()
session.close() session.close()
return tx_hash_hex
#gas_budget = tx_add['gas'] * tx_add['gasPrice'] #gas_budget = tx_add['gas'] * tx_add['gasPrice']
logg.debug('register user tx {}'.format(tx_hash_hex)) gas_budget = account_registry.gas(tx_signed_raw_hex)
s = create_check_gas_and_send_task( logg.debug('register user tx {} {} {}'.format(tx_hash_hex, queue, gas_budget))
s = create_check_gas_task(
[tx_signed_raw_hex], [tx_signed_raw_hex],
chain_str, chain_spec,
writer_address, writer_address,
gas_budget, gas=gas_budget,
tx_hashes_hex=[tx_hash_hex], tx_hashes_hex=[tx_hash_hex],
queue=queue, queue=queue,
) )
@ -305,6 +300,7 @@ def gift(self, account_address, chain_str):
queue=queue, queue=queue,
) )
s.apply_async() s.apply_async()
return [tx_signed_raw_hex] return [tx_signed_raw_hex]

View File

@ -21,46 +21,46 @@ from cic_eth.error import SignerError
celery_app = celery.current_app celery_app = celery.current_app
logg = celery_app.log.get_default_logger() logg = celery_app.log.get_default_logger()
#
@celery_app.task() #@celery_app.task()
def sign_tx(tx, chain_str): #def sign_tx(tx, chain_str):
"""Sign a single transaction against the given chain specification. # """Sign a single transaction against the given chain specification.
#
:param tx: Transaction in standard Ethereum format # :param tx: Transaction in standard Ethereum format
:type tx: dict # :type tx: dict
:param chain_str: Chain spec string representation # :param chain_str: Chain spec string representation
:type chain_str: str # :type chain_str: str
:returns: Transaction hash and raw signed transaction, respectively # :returns: Transaction hash and raw signed transaction, respectively
:rtype: tuple # :rtype: tuple
""" # """
chain_spec = ChainSpec.from_chain_str(chain_str) # chain_spec = ChainSpec.from_chain_str(chain_str)
#c = RpcClient(chain_spec) # #c = RpcClient(chain_spec)
tx_transfer_signed = None # tx_transfer_signed = None
conn = RPCConnection.connect('signer') # conn = RPCConnection.connect('signer')
try: # try:
o = sign_transaction(tx) # o = sign_transaction(tx)
tx_transfer_signed = conn.do(o) # tx_transfer_signed = conn.do(o)
#try: # #try:
# tx_transfer_signed = c.w3.eth.sign_transaction(tx) # # tx_transfer_signed = c.w3.eth.sign_transaction(tx)
except Exception as e: # except Exception as e:
raise SignerError('sign tx {}: {}'.format(tx, e)) # raise SignerError('sign tx {}: {}'.format(tx, e))
logg.debug('tx_transfer_signed {}'.format(tx_transfer_signed)) # logg.debug('tx_transfer_signed {}'.format(tx_transfer_signed))
h = sha3.keccak_256() # h = sha3.keccak_256()
h.update(bytes.fromhex(strip_0x(tx_transfer_signed['raw']))) # h.update(bytes.fromhex(strip_0x(tx_transfer_signed['raw'])))
tx_hash = h.digest() # tx_hash = h.digest()
#tx_hash = c.w3.keccak(hexstr=tx_transfer_signed['raw']) # #tx_hash = c.w3.keccak(hexstr=tx_transfer_signed['raw'])
tx_hash_hex = add_0x(tx_hash.hex()) # tx_hash_hex = add_0x(tx_hash.hex())
return (tx_hash_hex, tx_transfer_signed['raw'],) # return (tx_hash_hex, tx_transfer_signed['raw'],)
#def sign_and_register_tx(tx, chain_str, queue, cache_task=None, session=None): #def sign_and_register_tx(tx, chain_str, queue, cache_task=None, session=None):
def register_tx(tx_hash_hex, tx_signed_raw_hex, chain_str, queue, cache_task=None, session=None): def register_tx(tx_hash_hex, tx_signed_raw_hex, chain_spec, queue, cache_task=None, session=None):
"""Signs the provided transaction, and adds it to the transaction queue cache (with status PENDING). """Signs the provided transaction, and adds it to the transaction queue cache (with status PENDING).
:param tx: Standard ethereum transaction data :param tx: Standard ethereum transaction data
:type tx: dict :type tx: dict
:param chain_str: Chain spec, string representation :param chain_spec: Chain spec of transaction to add to queue
:type chain_str: str :type chain_spec: chainlib.chain.ChainSpec
:param queue: Task queue :param queue: Task queue
:type queue: str :type queue: str
:param cache_task: Cache task to call with signed transaction. If None, no task will be called. :param cache_task: Cache task to call with signed transaction. If None, no task will be called.
@ -73,14 +73,14 @@ def register_tx(tx_hash_hex, tx_signed_raw_hex, chain_str, queue, cache_task=Non
logg.debug('adding queue tx {}'.format(tx_hash_hex)) logg.debug('adding queue tx {}'.format(tx_hash_hex))
tx_signed_raw = bytes.fromhex(strip_0x(tx_signed_raw_hex)) tx_signed_raw = bytes.fromhex(strip_0x(tx_signed_raw_hex))
tx = unpack(tx_signed_raw) tx = unpack(tx_signed_raw, chain_id=chain_spec.chain_id())
queue_create( queue_create(
tx['nonce'], tx['nonce'],
tx['from'], tx['from'],
tx_hash_hex, tx_hash_hex,
tx_signed_raw_hex, tx_signed_raw_hex,
chain_str, chain_spec,
session=session, session=session,
) )
@ -91,7 +91,7 @@ def register_tx(tx_hash_hex, tx_signed_raw_hex, chain_str, queue, cache_task=Non
[ [
tx_hash_hex, tx_hash_hex,
tx_signed_raw_hex, tx_signed_raw_hex,
chain_str, chain_spec.asdict(),
], ],
queue=queue, queue=queue,
) )
@ -101,15 +101,15 @@ def register_tx(tx_hash_hex, tx_signed_raw_hex, chain_str, queue, cache_task=Non
# TODO: rename as we will not be sending task in the chain, this is the responsibility of the dispatcher # TODO: rename as we will not be sending task in the chain, this is the responsibility of the dispatcher
def create_check_gas_task(tx_signed_raws_hex, chain_str, holder_address, gas, tx_hashes_hex=None, queue=None): def create_check_gas_task(tx_signed_raws_hex, chain_spec, holder_address, gas=None, tx_hashes_hex=None, queue=None):
"""Creates a celery task signature for a check_gas task that adds the task to the outgoing queue to be processed by the dispatcher. """Creates a celery task signature for a check_gas task that adds the task to the outgoing queue to be processed by the dispatcher.
If tx_hashes_hex is not spefified, a preceding task chained to check_gas must supply the transaction hashes as its return value. If tx_hashes_hex is not spefified, a preceding task chained to check_gas must supply the transaction hashes as its return value.
:param tx_signed_raws_hex: Raw signed transaction data :param tx_signed_raws_hex: Raw signed transaction data
:type tx_signed_raws_hex: list of str, 0x-hex :type tx_signed_raws_hex: list of str, 0x-hex
:param chain_str: Chain spec, string representation :param chain_spec: Chain spec of address to add check gas for
:type chain_str: str :type chain_spec: chainlib.chain.ChainSpec
:param holder_address: Address sending the transactions :param holder_address: Address sending the transactions
:type holder_address: str, 0x-hex :type holder_address: str, 0x-hex
:param gas: Gas budget hint for transactions :param gas: Gas budget hint for transactions
@ -127,7 +127,7 @@ def create_check_gas_task(tx_signed_raws_hex, chain_str, holder_address, gas, tx
'cic_eth.eth.tx.check_gas', 'cic_eth.eth.tx.check_gas',
[ [
tx_hashes_hex, tx_hashes_hex,
chain_str, chain_spec.asdict(),
tx_signed_raws_hex, tx_signed_raws_hex,
holder_address, holder_address,
gas, gas,
@ -138,7 +138,7 @@ def create_check_gas_task(tx_signed_raws_hex, chain_str, holder_address, gas, tx
s_check_gas = celery.signature( s_check_gas = celery.signature(
'cic_eth.eth.tx.check_gas', 'cic_eth.eth.tx.check_gas',
[ [
chain_str, chain_spec.asdict(),
tx_signed_raws_hex, tx_signed_raws_hex,
holder_address, holder_address,
gas, gas,

View File

@ -4,8 +4,8 @@ import logging
# third-party imports # third-party imports
import celery import celery
import requests import requests
from cic_registry import zero_address from chainlib.eth.constant import ZERO_ADDRESS
from cic_registry.chain import ChainSpec from chainlib.chain import ChainSpec
from chainlib.eth.address import is_checksum_address from chainlib.eth.address import is_checksum_address
from chainlib.eth.gas import balance from chainlib.eth.gas import balance
from chainlib.eth.error import ( from chainlib.eth.error import (
@ -15,7 +15,9 @@ from chainlib.eth.error import (
from chainlib.eth.tx import ( from chainlib.eth.tx import (
transaction, transaction,
receipt, receipt,
raw,
) )
from chainlib.connection import RPCConnection
from chainlib.hash import keccak256_hex_to_hex from chainlib.hash import keccak256_hex_to_hex
from hexathon import add_0x from hexathon import add_0x
@ -43,7 +45,7 @@ from cic_eth.eth.util import unpack_signed_raw_tx
from cic_eth.eth.task import ( from cic_eth.eth.task import (
register_tx, register_tx,
create_check_gas_task, create_check_gas_task,
sign_tx, #sign_tx,
) )
from cic_eth.eth.nonce import NonceOracle from cic_eth.eth.nonce import NonceOracle
from cic_eth.error import ( from cic_eth.error import (
@ -68,7 +70,7 @@ MAX_NONCE_ATTEMPTS = 3
# TODO this function is too long # TODO this function is too long
@celery_app.task(bind=True, throws=(OutOfGasError), base=CriticalSQLAlchemyAndWeb3Task) @celery_app.task(bind=True, throws=(OutOfGasError), base=CriticalSQLAlchemyAndWeb3Task)
def check_gas(self, tx_hashes, chain_str, txs=[], address=None, gas_required=None): def check_gas(self, tx_hashes, chain_spec_dict, txs=[], address=None, gas_required=None):
"""Check the gas level of the sender address of a transaction. """Check the gas level of the sender address of a transaction.
If the account balance is not sufficient for the required gas, gas refill is requested and OutOfGasError raiser. If the account balance is not sufficient for the required gas, gas refill is requested and OutOfGasError raiser.
@ -77,8 +79,8 @@ def check_gas(self, tx_hashes, chain_str, txs=[], address=None, gas_required=Non
:param tx_hashes: Transaction hashes due to be submitted :param tx_hashes: Transaction hashes due to be submitted
:type tx_hashes: list of str, 0x-hex :type tx_hashes: list of str, 0x-hex
:param chain_str: Chain spec string representation :param chain_spec_dict: Chain spec dict representation
:type chain_str: str :type chain_spec_dict: dict
:param txs: Signed raw transaction data, corresponding to tx_hashes :param txs: Signed raw transaction data, corresponding to tx_hashes
:type txs: list of str, 0x-hex :type txs: list of str, 0x-hex
:param address: Sender address :param address: Sender address
@ -99,38 +101,38 @@ def check_gas(self, tx_hashes, chain_str, txs=[], address=None, gas_required=Non
if not is_checksum_address(address): if not is_checksum_address(address):
raise ValueError('invalid address {}'.format(address)) raise ValueError('invalid address {}'.format(address))
chain_spec = ChainSpec.from_chain_str(chain_str) chain_spec = ChainSpec.from_dict(chain_spec_dict)
queue = self.request.delivery_info['routing_key'] queue = self.request.delivery_info.get('routing_key')
#c = RpcClient(chain_spec, holder_address=address) conn = RPCConnection.connect(chain_spec)
#c = RpcClient(chain_spec)
conn = RPCConnection.connect()
# TODO: it should not be necessary to pass address explicitly, if not passed should be derived from the tx # TODO: it should not be necessary to pass address explicitly, if not passed should be derived from the tx
balance = 0 gas_balance = 0
try: try:
#balance = c.w3.eth.getBalance(address)
o = balance(address) o = balance(address)
r = conn.do(o) r = conn.do(o)
except EthException as e: except EthException as e:
raise EthError('balance call for {}: {}'.format(address, e)) raise EthError('gas_balance call for {}: {}'.format(address, e))
logg.debug('address {} has gas {} needs {}'.format(address, balance, gas_required)) logg.debug('address {} has gas {} needs {}'.format(address, gas_balance, gas_required))
session = SessionBase.create_session()
gas_provider = AccountRole.get_address('GAS_GIFTER', session=session)
session.close()
if gas_required > balance: if gas_required > gas_balance:
s_nonce = celery.signature( s_nonce = celery.signature(
'cic_eth.eth.tx.reserve_nonce', 'cic_eth.eth.tx.reserve_nonce',
[ [
address, address,
c.gas_provider(), gas_provider,
], ],
queue=queue, queue=queue,
) )
s_refill_gas = celery.signature( s_refill_gas = celery.signature(
'cic_eth.eth.tx.refill_gas', 'cic_eth.eth.tx.refill_gas',
[ [
chain_str, chain_spec_dict,
], ],
queue=queue, queue=queue,
) )
@ -147,28 +149,28 @@ def check_gas(self, tx_hashes, chain_str, txs=[], address=None, gas_required=Non
) )
wait_tasks.append(s) wait_tasks.append(s)
celery.group(wait_tasks)() celery.group(wait_tasks)()
raise OutOfGasError('need to fill gas, required {}, had {}'.format(gas_required, balance)) raise OutOfGasError('need to fill gas, required {}, had {}'.format(gas_required, gas_balance))
safe_gas = c.safe_threshold_amount() safe_gas = self.safe_gas_threshold_amount
if balance < safe_gas: if gas_balance < safe_gas:
s_nonce = celery.signature( s_nonce = celery.signature(
'cic_eth.eth.tx.reserve_nonce', 'cic_eth.eth.tx.reserve_nonce',
[ [
address, address,
c.gas_provider(), gas_provider,
], ],
queue=queue, queue=queue,
) )
s_refill_gas = celery.signature( s_refill_gas = celery.signature(
'cic_eth.eth.tx.refill_gas', 'cic_eth.eth.tx.refill_gas',
[ [
chain_str, chain_spec_dict,
], ],
queue=queue, queue=queue,
) )
s_nonce.link(s_refill_gas) s_nonce.link(s_refill_gas)
s_nonce.apply_async() s_nonce.apply_async()
logg.debug('requested refill from {} to {}'.format(c.gas_provider(), address)) logg.debug('requested refill from {} to {}'.format(gas_provider, address))
ready_tasks = [] ready_tasks = []
for tx_hash in tx_hashes: for tx_hash in tx_hashes:
s = celery.signature( s = celery.signature(
@ -218,172 +220,9 @@ def hashes_to_txs(self, tx_hashes):
return txs return txs
# TODO: Move this and send to subfolder submodule
class ParityNodeHandler:
def __init__(self, chain_spec, queue):
self.chain_spec = chain_spec
self.chain_str = str(chain_spec)
self.queue = queue
def handle(self, exception, tx_hash_hex, tx_hex):
meth = self.handle_default
if isinstance(exception, (ValueError)):
earg = exception.args[0]
if earg['code'] == -32010:
logg.debug('skipping lock for code {}'.format(earg['code']))
meth = self.handle_invalid_parameters
elif earg['code'] == -32602:
meth = self.handle_invalid_encoding
else:
# TODO: move to status log db comment field
meth = self.handle_invalid
elif isinstance(exception, (requests.exceptions.ConnectionError)):
meth = self.handle_connection
(t, e_fn, message) = meth(tx_hash_hex, tx_hex, str(exception))
return (t, e_fn, '{} {}'.format(message, exception))
def handle_connection(self, tx_hash_hex, tx_hex, debugstr=None):
s_set_sent = celery.signature(
'cic_eth.queue.tx.set_sent_status',
[
tx_hash_hex,
True,
],
queue=self.queue,
)
t = s_set_sent.apply_async()
return (t, TemporaryTxError, 'Sendfail {}'.format(tx_hex_string(tx_hex, self.chain_spec.chain_id())))
def handle_invalid_encoding(self, tx_hash_hex, tx_hex, debugstr=None):
tx_bytes = bytes.fromhex(tx_hex[2:])
tx = unpack_signed_raw_tx(tx_bytes, self.chain_spec.chain_id())
s_lock = celery.signature(
'cic_eth.admin.ctrl.lock_send',
[
tx_hash_hex,
self.chain_str,
tx['from'],
tx_hash_hex,
],
queue=self.queue,
)
s_set_reject = celery.signature(
'cic_eth.queue.tx.set_rejected',
[],
queue=self.queue,
)
nonce_txs = get_nonce_tx(tx['nonce'], tx['from'], self.chain_spec.chain_id())
attempts = len(nonce_txs)
if attempts < MAX_NONCE_ATTEMPTS:
logg.debug('nonce {} address {} retries {} < {}'.format(tx['nonce'], tx['from'], attempts, MAX_NONCE_ATTEMPTS))
s_resend = celery.signature(
'cic_eth.eth.tx.resend_with_higher_gas',
[
self.chain_str,
None,
1.01,
],
queue=self.queue,
)
s_unlock = celery.signature(
'cic_eth.admin.ctrl.unlock_send',
[
self.chain_str,
tx['from'],
],
queue=self.queue,
)
s_resend.link(s_unlock)
s_set_reject.link(s_resend)
s_lock.link(s_set_reject)
t = s_lock.apply_async()
return (t, PermanentTxError, 'Reject invalid encoding {}'.format(tx_hex_string(tx_hex, self.chain_spec.chain_id())))
def handle_invalid_parameters(self, tx_hash_hex, tx_hex, debugstr=None):
s_sync = celery.signature(
'cic_eth.eth.tx.sync_tx',
[
tx_hash_hex,
self.chain_str,
],
queue=self.queue,
)
t = s_sync.apply_async()
return (t, PermanentTxError, 'Reject invalid parameters {}'.format(tx_hex_string(tx_hex, self.chain_spec.chain_id())))
def handle_invalid(self, tx_hash_hex, tx_hex, debugstr=None):
tx_bytes = bytes.fromhex(tx_hex[2:])
tx = unpack_signed_raw_tx(tx_bytes, self.chain_spec.chain_id())
s_lock = celery.signature(
'cic_eth.admin.ctrl.lock_send',
[
tx_hash_hex,
self.chain_str,
tx['from'],
tx_hash_hex,
],
queue=self.queue,
)
s_set_reject = celery.signature(
'cic_eth.queue.tx.set_rejected',
[],
queue=self.queue,
)
s_debug = celery.signature(
'cic_eth.admin.debug.alert',
[
tx_hash_hex,
debugstr,
],
queue=self.queue,
)
s_set_reject.link(s_debug)
s_lock.link(s_set_reject)
t = s_lock.apply_async()
return (t, PermanentTxError, 'Reject invalid {}'.format(tx_hex_string(tx_hex, self.chain_spec.chain_id())))
def handle_default(self, tx_hash_hex, tx_hex, debugstr):
tx_bytes = bytes.fromhex(tx_hex[2:])
tx = unpack_signed_raw_tx(tx_bytes, self.chain_spec.chain_id())
s_lock = celery.signature(
'cic_eth.admin.ctrl.lock_send',
[
tx_hash_hex,
self.chain_str,
tx['from'],
tx_hash_hex,
],
queue=self.queue,
)
s_set_fubar = celery.signature(
'cic_eth.queue.tx.set_fubar',
[],
queue=self.queue,
)
s_debug = celery.signature(
'cic_eth.admin.debug.alert',
[
tx_hash_hex,
debugstr,
],
queue=self.queue,
)
s_set_fubar.link(s_debug)
s_lock.link(s_set_fubar)
t = s_lock.apply_async()
return (t, PermanentTxError, 'Fubar {} {}'.format(tx_hex_string(tx_hex, self.chain_spec.chain_id()), debugstr))
# TODO: A lock should be introduced to ensure that the send status change and the transaction send is atomic. # TODO: A lock should be introduced to ensure that the send status change and the transaction send is atomic.
@celery_app.task(bind=True, base=CriticalWeb3Task) @celery_app.task(bind=True, base=CriticalWeb3Task)
def send(self, txs, chain_str): def send(self, txs, chain_spec_dict):
"""Send transactions to the network. """Send transactions to the network.
If more than one transaction is passed to the task, it will spawn a new send task with the remaining transaction(s) after the first in the list has been processed. If more than one transaction is passed to the task, it will spawn a new send task with the remaining transaction(s) after the first in the list has been processed.
@ -408,7 +247,7 @@ def send(self, txs, chain_str):
if len(txs) == 0: if len(txs) == 0:
raise ValueError('no transaction to send') raise ValueError('no transaction to send')
chain_spec = ChainSpec.from_chain_str(chain_str) chain_spec = ChainSpec.from_dict(chain_spec_dict)
tx_hex = txs[0] tx_hex = txs[0]
logg.debug('send transaction {}'.format(tx_hex)) logg.debug('send transaction {}'.format(tx_hex))
@ -428,17 +267,18 @@ def send(self, txs, chain_str):
queue=queue, queue=queue,
) )
return txs[1:] o = raw(tx_hex)
conn = RPCConnection.connect(chain_spec, 'default')
try: #try:
#r = c.w3.eth.send_raw_transaction(tx_hex) #r = c.w3.eth.send_raw_transaction(tx_hex)
r = c.w3.eth.sendRawTransaction(tx_hex) #r = c.w3.eth.sendRawTransaction(tx_hex)
except requests.exceptions.ConnectionError as e: conn.do(o)
raise(e) #except requests.exceptions.ConnectionError as e:
except Exception as e: # raise(e)
raiser = ParityNodeHandler(chain_spec, queue) # except Exception as e:
(t, e, m) = raiser.handle(e, tx_hash_hex, tx_hex) # raiser = ParityNodeHandler(chain_spec, queue)
raise e(m) # (t, e, m) = raiser.handle(e, tx_hash_hex, tx_hex)
# raise e(m)
s_set_sent.apply_async() s_set_sent.apply_async()
tx_tail = txs[1:] tx_tail = txs[1:]
@ -456,7 +296,7 @@ def send(self, txs, chain_str):
# TODO: if this method fails the nonce will be out of sequence. session needs to be extended to include the queue create, so that nonce is rolled back if the second sql query fails. Better yet, split each state change into separate tasks. # TODO: if this method fails the nonce will be out of sequence. session needs to be extended to include the queue create, so that nonce is rolled back if the second sql query fails. Better yet, split each state change into separate tasks.
# TODO: method is too long, factor out code for clarity # TODO: method is too long, factor out code for clarity
@celery_app.task(bind=True, throws=(NotFoundEthException,), base=CriticalWeb3AndSignerTask) @celery_app.task(bind=True, throws=(NotFoundEthException,), base=CriticalWeb3AndSignerTask)
def refill_gas(self, recipient_address, chain_str): def refill_gas(self, recipient_address, chain_spec_dict):
"""Executes a native token transaction to fund the recipient's gas expenditures. """Executes a native token transaction to fund the recipient's gas expenditures.
:param recipient_address: Recipient in need of gas :param recipient_address: Recipient in need of gas
@ -467,7 +307,7 @@ def refill_gas(self, recipient_address, chain_str):
:returns: Transaction hash. :returns: Transaction hash.
:rtype: str, 0x-hex :rtype: str, 0x-hex
""" """
chain_spec = ChainSpec.from_chain_str(chain_str) chain_spec = ChainSpec.from_dict(chain_spec_dict)
zero_amount = False zero_amount = False
session = SessionBase.create_session() session = SessionBase.create_session()
@ -632,21 +472,22 @@ def resend_with_higher_gas(self, txold_hash_hex, chain_str, gas=None, default_fa
@celery_app.task(bind=True, base=CriticalSQLAlchemyTask) @celery_app.task(bind=True, base=CriticalSQLAlchemyTask)
def reserve_nonce(self, chained_input, signer=None): def reserve_nonce(self, chained_input, signer_address=None):
session = SessionBase.create_session() session = SessionBase.create_session()
address = None address = None
if signer == None: if signer_address == None:
address = chained_input address = chained_input
logg.debug('non-explicit address for reserve nonce, using arg head {}'.format(chained_input)) logg.debug('non-explicit address for reserve nonce, using arg head {}'.format(chained_input))
else: else:
#if web3.Web3.isChecksumAddress(signer): #if web3.Web3.isChecksumAddress(signer_address):
if is_checksum_address(signer): if is_checksum_address(signer_address):
address = signer address = signer_address
logg.debug('explicit address for reserve nonce {}'.format(signer)) logg.debug('explicit address for reserve nonce {}'.format(signer_address))
else: else:
address = AccountRole.get_address(signer, session=session) address = AccountRole.get_address(signer_address, session=session)
logg.debug('role for reserve nonce {} -> {}'.format(signer, address)) logg.debug('role for reserve nonce {} -> {}'.format(signer_address, address))
if not is_checksum_address(address): if not is_checksum_address(address):
raise ValueError('invalid result when resolving address for nonce {}'.format(address)) raise ValueError('invalid result when resolving address for nonce {}'.format(address))
@ -803,8 +644,8 @@ def cache_gas_refill_data(
tx_hash_hex, tx_hash_hex,
tx['from'], tx['from'],
tx['to'], tx['to'],
zero_address, ZERO_ADDRESS,
zero_address, ZERO_ADDRESS,
tx['value'], tx['value'],
tx['value'], tx['value'],
) )

View File

@ -35,7 +35,7 @@ celery_app = celery.current_app
logg = logging.getLogger() logg = logging.getLogger()
def create(nonce, holder_address, tx_hash, signed_tx, chain_str, obsolete_predecessors=True, session=None): def create(nonce, holder_address, tx_hash, signed_tx, chain_spec, obsolete_predecessors=True, session=None):
"""Create a new transaction queue record. """Create a new transaction queue record.
:param nonce: Transaction nonce :param nonce: Transaction nonce
@ -46,13 +46,13 @@ def create(nonce, holder_address, tx_hash, signed_tx, chain_str, obsolete_predec
:type tx_hash: str, 0x-hex :type tx_hash: str, 0x-hex
:param signed_tx: Signed raw transaction :param signed_tx: Signed raw transaction
:type signed_tx: str, 0x-hex :type signed_tx: str, 0x-hex
:param chain_str: Chain spec string representation to create transaction for :param chain_spec: Chain spec to create transaction for
:type chain_str: str :type chain_spec: ChainSpec
:returns: transaction hash :returns: transaction hash
:rtype: str, 0x-hash :rtype: str, 0x-hash
""" """
session = SessionBase.bind_session(session) session = SessionBase.bind_session(session)
lock = Lock.check_aggregate(chain_str, LockEnum.QUEUE, holder_address, session=session) lock = Lock.check_aggregate(str(chain_spec), LockEnum.QUEUE, holder_address, session=session)
if lock > 0: if lock > 0:
SessionBase.release_session(session) SessionBase.release_session(session)
raise LockedError(lock) raise LockedError(lock)

View File

@ -52,6 +52,7 @@ class CriticalWeb3Task(CriticalTask):
autoretry_for = ( autoretry_for = (
requests.exceptions.ConnectionError, requests.exceptions.ConnectionError,
) )
safe_gas_threshold_amount = 2000000000 * 60000 * 3
class CriticalSQLAlchemyAndWeb3Task(CriticalTask): class CriticalSQLAlchemyAndWeb3Task(CriticalTask):
@ -62,6 +63,7 @@ class CriticalSQLAlchemyAndWeb3Task(CriticalTask):
sqlalchemy.exc.ResourceClosedError, sqlalchemy.exc.ResourceClosedError,
EthError, EthError,
) )
safe_gas_threshold_amount = 2000000000 * 60000 * 3
class CriticalSQLAlchemyAndSignerTask(CriticalTask): class CriticalSQLAlchemyAndSignerTask(CriticalTask):
autoretry_for = ( autoretry_for = (

View File

@ -4,7 +4,8 @@ import tempfile
import logging import logging
import shutil import shutil
logg = logging.getLogger(__name__) #logg = logging.getLogger(__name__)
logg = logging.getLogger()
# celery fixtures # celery fixtures
@ -52,7 +53,7 @@ def celery_config():
@pytest.fixture(scope='session') @pytest.fixture(scope='session')
def celery_worker_parameters(): def celery_worker_parameters():
return { return {
# 'queues': ('cic-eth'), # 'queues': ('celery'),
} }
@pytest.fixture(scope='session') @pytest.fixture(scope='session')

View File

@ -9,7 +9,9 @@ from chainlib.eth.address import to_checksum_address
# local imports # local imports
from cic_eth.db.models.role import AccountRole from cic_eth.db.models.role import AccountRole
logg = logging.getLogger(__name__) #logg = logging.getLogger(__name__)
# what the actual fuck, debug is not being shown even though explicitly set
logg = logging.getLogger()
@pytest.fixture(scope='function') @pytest.fixture(scope='function')
@ -22,6 +24,7 @@ def custodial_roles(
r.update(contract_roles) r.update(contract_roles)
r.update({ r.update({
'DEFAULT': eth_accounts[0], 'DEFAULT': eth_accounts[0],
'GAS_GIFTER': eth_accounts[3],
}) })
for k in r.keys(): for k in r.keys():
role = AccountRole.set(k, r[k]) role = AccountRole.set(k, r[k])

View File

@ -67,10 +67,10 @@ def test_register_account(
eth_accounts, eth_accounts,
eth_rpc, eth_rpc,
cic_registry, cic_registry,
celery_session_worker,
eth_empty_accounts, eth_empty_accounts,
custodial_roles, custodial_roles,
call_sender, call_sender,
celery_worker,
): ):
logg.debug('chainspec {}'.format(str(default_chain_spec))) logg.debug('chainspec {}'.format(str(default_chain_spec)))
@ -79,16 +79,17 @@ def test_register_account(
'cic_eth.eth.tx.reserve_nonce', 'cic_eth.eth.tx.reserve_nonce',
[ [
eth_empty_accounts[0], eth_empty_accounts[0],
eth_accounts[0], custodial_roles['ACCOUNT_REGISTRY_WRITER'],
], ],
queue=None, queue=None,
) )
s_register = celery.signature( s_register = celery.signature(
'cic_eth.eth.account.register', 'cic_eth.eth.account.register',
[ [
str(default_chain_spec), default_chain_spec.asdict(),
eth_accounts[0], custodial_roles['ACCOUNT_REGISTRY_WRITER'],
], ],
queue=None,
) )
s_nonce.link(s_register) s_nonce.link(s_register)
t = s_nonce.apply_async() t = s_nonce.apply_async()
@ -101,13 +102,14 @@ def test_register_account(
o = session.query(Otx).first() o = session.query(Otx).first()
tx_signed_hex = o.signed_tx tx_signed_hex = o.signed_tx
session.close() session.close()
s_send = celery.signature( s_send = celery.signature(
'cic_eth.eth.tx.send', 'cic_eth.eth.tx.send',
[ [
[tx_signed_hex], [tx_signed_hex],
str(default_chain_spec), default_chain_spec.asdict(),
], ],
queue=None,
) )
t = s_send.apply_async() t = s_send.apply_async()
address = t.get() address = t.get()