2021-05-31 17:34:16 +02:00
|
|
|
# standard imports
|
|
|
|
import logging
|
|
|
|
import io
|
|
|
|
import json
|
|
|
|
|
|
|
|
# external imports
|
|
|
|
import pytest
|
|
|
|
from chainlib.connection import RPCConnection
|
|
|
|
from chainlib.eth.nonce import (
|
|
|
|
nonce,
|
|
|
|
OverrideNonceOracle,
|
|
|
|
RPCNonceOracle,
|
|
|
|
)
|
2021-06-03 15:51:55 +02:00
|
|
|
from chainqueue.sql.tx import create as queue_create
|
2021-05-31 17:34:16 +02:00
|
|
|
from chainlib.eth.tx import (
|
|
|
|
raw,
|
|
|
|
receipt,
|
|
|
|
TxFormat,
|
|
|
|
Tx,
|
|
|
|
)
|
|
|
|
from chainlib.eth.block import block_latest
|
|
|
|
from chainlib.eth.gas import (
|
|
|
|
Gas,
|
|
|
|
OverrideGasOracle,
|
|
|
|
)
|
2021-06-03 15:51:55 +02:00
|
|
|
from chainqueue.sql.state import (
|
2021-05-31 17:34:16 +02:00
|
|
|
set_reserved,
|
|
|
|
set_sent,
|
|
|
|
set_ready,
|
|
|
|
)
|
|
|
|
from chainqueue.db.models.otx import Otx
|
|
|
|
from chainqueue.db.enum import StatusBits
|
|
|
|
from eth_erc20 import ERC20
|
|
|
|
from cic_eth_registry import CICRegistry
|
|
|
|
|
|
|
|
# local imports
|
2021-07-07 09:45:21 +02:00
|
|
|
from cic_eth.api.admin import AdminApi
|
2021-05-31 17:34:16 +02:00
|
|
|
from cic_eth.eth.gas import cache_gas_data
|
|
|
|
from cic_eth.eth.erc20 import cache_transfer_data
|
2021-08-28 06:27:48 +02:00
|
|
|
from cic_eth.queue.query import get_nonce_tx_local
|
2021-05-31 17:34:16 +02:00
|
|
|
|
|
|
|
logg = logging.getLogger()
|
|
|
|
|
|
|
|
|
|
|
|
def test_admin_api_tx(
|
|
|
|
default_chain_spec,
|
|
|
|
init_database,
|
|
|
|
init_celery_tasks,
|
|
|
|
eth_rpc,
|
|
|
|
eth_signer,
|
|
|
|
agent_roles,
|
|
|
|
contract_roles,
|
|
|
|
custodial_roles,
|
|
|
|
celery_session_worker,
|
|
|
|
foo_token,
|
|
|
|
address_declarator,
|
|
|
|
cic_registry,
|
|
|
|
register_tokens,
|
|
|
|
register_lookups,
|
|
|
|
caplog,
|
|
|
|
):
|
|
|
|
|
|
|
|
nonce_oracle = RPCNonceOracle(custodial_roles['FOO_TOKEN_GIFTER'], conn=eth_rpc)
|
|
|
|
gas_oracle = OverrideGasOracle(limit=100000, conn=eth_rpc)
|
|
|
|
|
|
|
|
o = nonce(custodial_roles['FOO_TOKEN_GIFTER'])
|
|
|
|
r = eth_rpc.do(o)
|
|
|
|
gifter_nonce = int(r, 16)
|
|
|
|
|
|
|
|
#c = Gas(default_chain_spec, signer=eth_signer, nonce_oracle=nonce_oracle, gas_oracle=gas_oracle)
|
|
|
|
c = ERC20(default_chain_spec, signer=eth_signer, nonce_oracle=nonce_oracle, gas_oracle=gas_oracle)
|
|
|
|
(tx_hash_hex, tx_signed_raw_hex) = c.transfer(foo_token, custodial_roles['FOO_TOKEN_GIFTER'], agent_roles['ALICE'], 100 * (10 ** 6), tx_format=TxFormat.RLP_SIGNED)
|
|
|
|
queue_create(
|
|
|
|
default_chain_spec,
|
|
|
|
gifter_nonce, # will only work if agent starts at 0
|
|
|
|
agent_roles['ALICE'],
|
|
|
|
tx_hash_hex,
|
|
|
|
tx_signed_raw_hex,
|
|
|
|
session=init_database,
|
|
|
|
)
|
|
|
|
cache_transfer_data(
|
|
|
|
tx_hash_hex,
|
|
|
|
tx_signed_raw_hex,
|
|
|
|
default_chain_spec.asdict(),
|
|
|
|
)
|
|
|
|
|
|
|
|
init_database.commit()
|
|
|
|
|
|
|
|
o = raw(tx_signed_raw_hex)
|
|
|
|
eth_rpc.do(o)
|
|
|
|
|
|
|
|
o = receipt(tx_hash_hex)
|
|
|
|
r = eth_rpc.do(o)
|
|
|
|
assert r['status'] == 1
|
|
|
|
|
|
|
|
set_ready(default_chain_spec, tx_hash_hex, session=init_database)
|
|
|
|
set_reserved(default_chain_spec, tx_hash_hex, session=init_database)
|
|
|
|
set_sent(default_chain_spec, tx_hash_hex, session=init_database)
|
|
|
|
|
|
|
|
# lookup by transaction hash, without registry
|
|
|
|
api = AdminApi(eth_rpc, queue=None, call_address=contract_roles['CONTRACT_DEPLOYER'])
|
|
|
|
tx = api.tx(default_chain_spec, tx_hash=tx_hash_hex)
|
|
|
|
logg.debug('deployed {}'.format(contract_roles['CONTRACT_DEPLOYER']))
|
|
|
|
assert tx['tx_hash'] == tx_hash_hex
|
|
|
|
|
|
|
|
# lookup by RLP transaction, without registry
|
|
|
|
tx = api.tx(default_chain_spec, tx_raw=tx_signed_raw_hex)
|
|
|
|
assert tx['tx_hash'] == tx_hash_hex
|
|
|
|
|
|
|
|
# lookup by transaction hash, with registry
|
|
|
|
registry = CICRegistry(default_chain_spec, eth_rpc)
|
|
|
|
tx = api.tx(default_chain_spec, tx_hash=tx_hash_hex, registry=registry)
|
|
|
|
assert tx['tx_hash'] == tx_hash_hex
|
|
|
|
|
|
|
|
# lookup by transaction hash, using writer
|
|
|
|
buf = io.StringIO()
|
|
|
|
api.tx(default_chain_spec, tx_hash=tx_hash_hex, renderer=json.dumps, w=buf)
|
|
|
|
tx = json.loads(buf.getvalue())
|
|
|
|
assert tx['tx_hash'] == tx_hash_hex
|
|
|
|
|
|
|
|
|
|
|
|
def test_admin_api_account(
|
|
|
|
default_chain_spec,
|
|
|
|
init_database,
|
|
|
|
eth_rpc,
|
|
|
|
eth_signer,
|
|
|
|
agent_roles,
|
|
|
|
contract_roles,
|
|
|
|
celery_session_worker,
|
|
|
|
caplog,
|
|
|
|
):
|
|
|
|
|
|
|
|
nonce_oracle = OverrideNonceOracle(agent_roles['ALICE'], 42)
|
|
|
|
gas_oracle = OverrideGasOracle(limit=21000, conn=eth_rpc)
|
|
|
|
|
|
|
|
tx_hashes_alice = []
|
|
|
|
txs_alice = []
|
|
|
|
|
|
|
|
for i in range(3):
|
|
|
|
c = Gas(default_chain_spec, signer=eth_signer, nonce_oracle=nonce_oracle, gas_oracle=gas_oracle)
|
|
|
|
(tx_hash_hex, tx_signed_raw_hex) = c.create(agent_roles['ALICE'], agent_roles['BOB'], 100 * (10 ** 6), tx_format=TxFormat.RLP_SIGNED)
|
|
|
|
queue_create(
|
|
|
|
default_chain_spec,
|
|
|
|
42+i,
|
|
|
|
agent_roles['ALICE'],
|
|
|
|
tx_hash_hex,
|
|
|
|
tx_signed_raw_hex,
|
|
|
|
session=init_database,
|
|
|
|
)
|
|
|
|
cache_gas_data(
|
|
|
|
tx_hash_hex,
|
|
|
|
tx_signed_raw_hex,
|
|
|
|
default_chain_spec.asdict(),
|
|
|
|
)
|
|
|
|
tx_hashes_alice.append(tx_hash_hex)
|
|
|
|
txs_alice.append(tx_signed_raw_hex)
|
|
|
|
|
|
|
|
init_database.commit()
|
|
|
|
|
|
|
|
nonce_oracle = OverrideNonceOracle(agent_roles['BOB'], 13)
|
|
|
|
tx_hashes_bob = []
|
|
|
|
txs_bob = []
|
|
|
|
|
|
|
|
for i in range(2):
|
|
|
|
c = Gas(default_chain_spec, signer=eth_signer, nonce_oracle=nonce_oracle, gas_oracle=gas_oracle)
|
|
|
|
(tx_hash_hex, tx_signed_raw_hex) = c.create(agent_roles['BOB'], agent_roles['ALICE'], 100 * (10 ** 6), tx_format=TxFormat.RLP_SIGNED)
|
|
|
|
queue_create(
|
|
|
|
default_chain_spec,
|
|
|
|
13+i,
|
|
|
|
agent_roles['BOB'],
|
|
|
|
tx_hash_hex,
|
|
|
|
tx_signed_raw_hex,
|
|
|
|
session=init_database,
|
|
|
|
)
|
|
|
|
cache_gas_data(
|
|
|
|
tx_hash_hex,
|
|
|
|
tx_signed_raw_hex,
|
|
|
|
default_chain_spec.asdict(),
|
|
|
|
)
|
|
|
|
tx_hashes_bob.append(tx_hash_hex)
|
|
|
|
txs_bob.append(tx_signed_raw_hex)
|
|
|
|
|
|
|
|
init_database.commit()
|
|
|
|
|
|
|
|
|
|
|
|
api = AdminApi(eth_rpc, queue=None, call_address=contract_roles['CONTRACT_DEPLOYER'])
|
2021-07-21 19:34:51 +02:00
|
|
|
r = api.account(default_chain_spec, agent_roles['ALICE'], include_sender=True, include_recipient=True)
|
2021-05-31 17:34:16 +02:00
|
|
|
assert len(r) == 5
|
|
|
|
|
|
|
|
api = AdminApi(eth_rpc, queue=None, call_address=contract_roles['CONTRACT_DEPLOYER'])
|
|
|
|
r = api.account(default_chain_spec, agent_roles['ALICE'], include_sender=False)
|
|
|
|
assert len(r) == 2
|
|
|
|
|
|
|
|
api = AdminApi(eth_rpc, queue=None, call_address=contract_roles['CONTRACT_DEPLOYER'])
|
|
|
|
r = api.account(default_chain_spec, agent_roles['ALICE'], include_recipient=False)
|
|
|
|
assert len(r) == 3
|
|
|
|
|
|
|
|
|
|
|
|
def test_admin_api_account_writer(
|
|
|
|
default_chain_spec,
|
|
|
|
init_database,
|
|
|
|
eth_rpc,
|
|
|
|
eth_signer,
|
|
|
|
agent_roles,
|
|
|
|
contract_roles,
|
|
|
|
celery_session_worker,
|
|
|
|
caplog,
|
|
|
|
):
|
|
|
|
|
|
|
|
nonce_oracle = OverrideNonceOracle(agent_roles['ALICE'], 42)
|
|
|
|
gas_oracle = OverrideGasOracle(limit=21000, conn=eth_rpc)
|
|
|
|
|
|
|
|
c = Gas(default_chain_spec, signer=eth_signer, nonce_oracle=nonce_oracle, gas_oracle=gas_oracle)
|
|
|
|
(tx_hash_hex, tx_signed_raw_hex) = c.create(agent_roles['ALICE'], agent_roles['BOB'], 100 * (10 ** 6), tx_format=TxFormat.RLP_SIGNED)
|
|
|
|
queue_create(
|
|
|
|
default_chain_spec,
|
|
|
|
42,
|
|
|
|
agent_roles['ALICE'],
|
|
|
|
tx_hash_hex,
|
|
|
|
tx_signed_raw_hex,
|
|
|
|
session=init_database,
|
|
|
|
)
|
|
|
|
cache_gas_data(
|
|
|
|
tx_hash_hex,
|
|
|
|
tx_signed_raw_hex,
|
|
|
|
default_chain_spec.asdict(),
|
|
|
|
)
|
|
|
|
|
|
|
|
init_database.commit()
|
|
|
|
|
|
|
|
buf = io.StringIO()
|
|
|
|
api = AdminApi(eth_rpc, queue=None, call_address=contract_roles['CONTRACT_DEPLOYER'])
|
|
|
|
api.account(default_chain_spec, agent_roles['ALICE'], renderer=json.dumps, w=buf)
|
|
|
|
|
|
|
|
# TODO: improve eval
|
|
|
|
tx = json.loads(buf.getvalue())
|
|
|
|
assert tx['tx_hash'] == tx_hash_hex
|
|
|
|
|
|
|
|
|
|
|
|
def test_registry(
|
|
|
|
eth_rpc,
|
|
|
|
cic_registry,
|
|
|
|
contract_roles,
|
|
|
|
celery_session_worker,
|
|
|
|
):
|
|
|
|
|
|
|
|
api = AdminApi(eth_rpc, queue=None, call_address=contract_roles['CONTRACT_DEPLOYER'])
|
|
|
|
t = api.registry()
|
|
|
|
r = t.get_leaf()
|
|
|
|
assert r == cic_registry
|
|
|
|
|
|
|
|
|
|
|
|
def test_proxy_do(
|
|
|
|
default_chain_spec,
|
|
|
|
eth_rpc,
|
|
|
|
contract_roles,
|
|
|
|
celery_session_worker,
|
|
|
|
):
|
|
|
|
|
|
|
|
o = block_latest()
|
|
|
|
r = eth_rpc.do(o)
|
|
|
|
|
|
|
|
api = AdminApi(eth_rpc, queue=None, call_address=contract_roles['CONTRACT_DEPLOYER'])
|
|
|
|
t = api.proxy_do(default_chain_spec, o)
|
|
|
|
rr = t.get_leaf()
|
|
|
|
|
|
|
|
assert r == rr
|
|
|
|
|
|
|
|
|
|
|
|
def test_resend_inplace(
|
|
|
|
init_database,
|
|
|
|
default_chain_spec,
|
|
|
|
eth_rpc,
|
|
|
|
eth_signer,
|
|
|
|
agent_roles,
|
|
|
|
contract_roles,
|
|
|
|
celery_session_worker,
|
|
|
|
):
|
|
|
|
|
|
|
|
rpc = RPCConnection.connect(default_chain_spec, 'default')
|
|
|
|
nonce_oracle = OverrideNonceOracle(agent_roles['ALICE'], 42)
|
|
|
|
gas_oracle = OverrideGasOracle(price=1000000000, limit=21000)
|
|
|
|
c = Gas(default_chain_spec, signer=eth_signer, nonce_oracle=nonce_oracle, gas_oracle=gas_oracle)
|
|
|
|
(tx_hash_hex, tx_signed_raw_hex) = c.create(agent_roles['ALICE'], agent_roles['BOB'], 100 * (10 ** 6), tx_format=TxFormat.RLP_SIGNED)
|
|
|
|
|
|
|
|
queue_create(
|
|
|
|
default_chain_spec,
|
|
|
|
42,
|
|
|
|
agent_roles['ALICE'],
|
|
|
|
tx_hash_hex,
|
|
|
|
tx_signed_raw_hex,
|
|
|
|
session=init_database,
|
|
|
|
)
|
|
|
|
cache_gas_data(
|
|
|
|
tx_hash_hex,
|
|
|
|
tx_signed_raw_hex,
|
|
|
|
default_chain_spec.asdict(),
|
|
|
|
)
|
|
|
|
|
|
|
|
set_ready(default_chain_spec, tx_hash_hex, session=init_database)
|
|
|
|
set_reserved(default_chain_spec, tx_hash_hex, session=init_database)
|
|
|
|
set_sent(default_chain_spec, tx_hash_hex, session=init_database)
|
|
|
|
|
|
|
|
init_database.commit()
|
|
|
|
|
|
|
|
api = AdminApi(eth_rpc, queue=None, call_address=contract_roles['CONTRACT_DEPLOYER'])
|
|
|
|
t = api.resend(tx_hash_hex, default_chain_spec, unlock=True)
|
|
|
|
r = t.get_leaf()
|
|
|
|
assert t.successful()
|
|
|
|
|
|
|
|
|
|
|
|
otx = Otx.load(tx_hash_hex, session=init_database)
|
|
|
|
assert otx.status & StatusBits.OBSOLETE == StatusBits.OBSOLETE
|
|
|
|
|
2021-08-28 06:27:48 +02:00
|
|
|
txs = get_nonce_tx_local(default_chain_spec, otx.nonce, agent_roles['ALICE'], session=init_database)
|
2021-05-31 17:34:16 +02:00
|
|
|
assert len(txs) == 2
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.xfail()
|
|
|
|
def test_resend_clone(
|
|
|
|
init_database,
|
|
|
|
default_chain_spec,
|
|
|
|
eth_rpc,
|
|
|
|
eth_signer,
|
|
|
|
agent_roles,
|
|
|
|
contract_roles,
|
|
|
|
celery_session_worker,
|
|
|
|
):
|
|
|
|
|
|
|
|
rpc = RPCConnection.connect(default_chain_spec, 'default')
|
|
|
|
nonce_oracle = OverrideNonceOracle(agent_roles['ALICE'], 42)
|
|
|
|
gas_oracle = OverrideGasOracle(price=1000000000, limit=21000)
|
|
|
|
c = Gas(default_chain_spec, signer=eth_signer, nonce_oracle=nonce_oracle, gas_oracle=gas_oracle)
|
|
|
|
(tx_hash_hex, tx_signed_raw_hex) = c.create(agent_roles['ALICE'], agent_roles['BOB'], 100 * (10 ** 6), tx_format=TxFormat.RLP_SIGNED)
|
|
|
|
|
|
|
|
queue_create(
|
|
|
|
default_chain_spec,
|
|
|
|
42,
|
|
|
|
agent_roles['ALICE'],
|
|
|
|
tx_hash_hex,
|
|
|
|
tx_signed_raw_hex,
|
|
|
|
session=init_database,
|
|
|
|
)
|
|
|
|
cache_gas_data(
|
|
|
|
tx_hash_hex,
|
|
|
|
tx_signed_raw_hex,
|
|
|
|
default_chain_spec.asdict(),
|
|
|
|
)
|
|
|
|
|
|
|
|
set_ready(default_chain_spec, tx_hash_hex, session=init_database)
|
|
|
|
set_reserved(default_chain_spec, tx_hash_hex, session=init_database)
|
|
|
|
set_sent(default_chain_spec, tx_hash_hex, session=init_database)
|
|
|
|
|
|
|
|
init_database.commit()
|
|
|
|
|
|
|
|
api = AdminApi(eth_rpc, queue=None, call_address=contract_roles['CONTRACT_DEPLOYER'])
|
|
|
|
t = api.resend(tx_hash_hex, default_chain_spec, in_place=False)
|
|
|
|
r = t.get_leaf()
|
|
|
|
assert t.successful()
|
|
|
|
|
|
|
|
otx = Otx.load(tx_hash_hex, session=init_database)
|
|
|
|
assert otx.status & StatusBits.IN_NETWORK == StatusBits.IN_NETWORK
|
|
|
|
assert otx.status & StatusBits.OBSOLETE == StatusBits.OBSOLETE
|
|
|
|
|
2021-08-28 06:27:48 +02:00
|
|
|
txs = get_nonce_tx_local(default_chain_spec, otx.nonce, agent_roles['ALICE'], session=init_database)
|
2021-05-31 17:34:16 +02:00
|
|
|
assert len(txs) == 1
|
|
|
|
|
2021-08-28 06:27:48 +02:00
|
|
|
txs = get_nonce_tx_local(default_chain_spec, otx.nonce + 1, agent_roles['ALICE'], session=init_database)
|
2021-05-31 17:34:16 +02:00
|
|
|
assert len(txs) == 1
|
|
|
|
|
|
|
|
otx = Otx.load(txs[0], session=init_database)
|
|
|
|
assert otx.status == 0
|