Add transaction executor helper
This commit is contained in:
parent
6ba0ac6820
commit
f7c1f05a1f
@ -1,6 +1,7 @@
|
||||
* 0.4.13-unreleased
|
||||
- Implement DictKeystore
|
||||
- Remove unused insert_key method in keystore interface
|
||||
- Add transaction executor helper
|
||||
* 0.4.12
|
||||
- Enforce hex strings in signer backend for sign message
|
||||
* 0.4.11
|
||||
|
@ -1,2 +1,6 @@
|
||||
class UnknownAccountError(Exception):
|
||||
pass
|
||||
|
||||
|
||||
class TransactionRevertError(Exception):
|
||||
pass
|
||||
|
1
crypto_dev_signer/helper/__init__.py
Normal file
1
crypto_dev_signer/helper/__init__.py
Normal file
@ -0,0 +1 @@
|
||||
from .tx import TxExecutor
|
71
crypto_dev_signer/helper/tx.py
Normal file
71
crypto_dev_signer/helper/tx.py
Normal file
@ -0,0 +1,71 @@
|
||||
# standard imports
|
||||
import logging
|
||||
|
||||
# third-party imports
|
||||
from crypto_dev_signer.eth.transaction import EIP155Transaction
|
||||
|
||||
# local imports
|
||||
from crypto_dev_signer.error import TransactionRevertError
|
||||
|
||||
logg = logging.getLogger()
|
||||
|
||||
|
||||
class TxExecutor:
|
||||
|
||||
def __init__(self, sender, signer, dispatcher, reporter, nonce, chain_id, fee_helper, fee_price_helper, block=False):
|
||||
self.sender = sender
|
||||
self.nonce = nonce
|
||||
self.signer = signer
|
||||
self.dispatcher = dispatcher
|
||||
self.reporter = reporter
|
||||
self.block = bool(block)
|
||||
self.chain_id = chain_id
|
||||
self.tx_hashes = []
|
||||
self.fee_price_helper = fee_price_helper
|
||||
self.fee_helper = fee_helper
|
||||
|
||||
|
||||
def sign_and_send(self, builder, force_wait=False):
|
||||
fee_units = self.fee_helper(self.sender, None, None)
|
||||
|
||||
tx_tpl = {
|
||||
'from': self.sender,
|
||||
'chainId': self.chain_id,
|
||||
'fee': fee_units,
|
||||
'feePrice': self.fee_price_helper(),
|
||||
'nonce': self.nonce,
|
||||
}
|
||||
tx = None
|
||||
for b in builder:
|
||||
tx = b(tx_tpl, tx)
|
||||
|
||||
logg.debug('from {} nonce {} tx {}'.format(self.sender, self.nonce, tx))
|
||||
|
||||
chain_tx = EIP155Transaction(tx, self.nonce, self.chain_id)
|
||||
signature = self.signer.signTransaction(chain_tx)
|
||||
chain_tx_serialized = chain_tx.rlp_serialize()
|
||||
tx_hash = self.dispatcher('0x' + chain_tx_serialized.hex())
|
||||
self.tx_hashes.append(tx_hash)
|
||||
self.nonce += 1
|
||||
rcpt = None
|
||||
if self.block or force_wait:
|
||||
rcpt = self.wait_for(tx_hash)
|
||||
logg.info('tx {} fee used: {}'.format(tx_hash.hex(), rcpt['feeUsed']))
|
||||
return (tx_hash.hex(), rcpt)
|
||||
|
||||
|
||||
def wait_for(self, tx_hash=None):
|
||||
if tx_hash == None:
|
||||
tx_hash = self.tx_hashes[len(self.tx_hashes)-1]
|
||||
i = 1
|
||||
while True:
|
||||
try:
|
||||
#return self.w3.eth.getTransactionReceipt(tx_hash)
|
||||
return self.reporter(tx_hash)
|
||||
except web3.exceptions.TransactionNotFound:
|
||||
logg.debug('poll #{} for {}'.format(i, tx_hash.hex()))
|
||||
i += 1
|
||||
time.sleep(1)
|
||||
if rcpt['status'] == 0:
|
||||
raise TransactionRevertError(tx_hash)
|
||||
return rcpt
|
33
crypto_dev_signer/keystore/dict.py
Normal file
33
crypto_dev_signer/keystore/dict.py
Normal file
@ -0,0 +1,33 @@
|
||||
# standard imports
|
||||
import logging
|
||||
|
||||
# local imports
|
||||
from . import keyapi
|
||||
from .interface import Keystore
|
||||
from crypto_dev_signer.error import UnknownAccountError
|
||||
from crypto_dev_signer.common import strip_hex_prefix
|
||||
|
||||
logg = logging.getLogger()
|
||||
|
||||
|
||||
class DictKeystore(Keystore):
|
||||
|
||||
def __init__(self):
|
||||
self.keys = {}
|
||||
|
||||
|
||||
def get(self, address, password=None):
|
||||
if password != None:
|
||||
logg.debug('password ignored as dictkeystore doesnt do encryption')
|
||||
try:
|
||||
return self.keys[address]
|
||||
except KeyError:
|
||||
raise UnknownAccountError(address)
|
||||
|
||||
|
||||
def import_key(self, pk, password=None):
|
||||
pubk = keyapi.private_key_to_public_key(pk)
|
||||
address_hex = pubk.to_checksum_address()
|
||||
address_hex_clean = strip_hex_prefix(address_hex)
|
||||
self.keys[address_hex_clean] = pk.to_bytes()
|
||||
return address_hex
|
2
setup.py
2
setup.py
@ -24,7 +24,7 @@ f.close()
|
||||
|
||||
setup(
|
||||
name="crypto-dev-signer",
|
||||
version="0.4.13a3",
|
||||
version="0.4.13a4",
|
||||
description="A signer and keystore daemon and library for cryptocurrency software development",
|
||||
author="Louis Holbrook",
|
||||
author_email="dev@holbrook.no",
|
||||
|
73
test/test_helper.py
Normal file
73
test/test_helper.py
Normal file
@ -0,0 +1,73 @@
|
||||
# standard imports
|
||||
import unittest
|
||||
import logging
|
||||
import os
|
||||
|
||||
# local imports
|
||||
from crypto_dev_signer.keystore import DictKeystore
|
||||
from crypto_dev_signer.eth.signer import ReferenceSigner
|
||||
from crypto_dev_signer.helper import TxExecutor
|
||||
|
||||
logging.basicConfig(level=logging.DEBUG)
|
||||
logg = logging.getLogger()
|
||||
|
||||
script_dir = os.path.realpath(os.path.dirname(__file__))
|
||||
|
||||
|
||||
class MockEthTxBackend:
|
||||
|
||||
def dispatcher(self, tx):
|
||||
logg.debug('sender {}'.format(tx))
|
||||
return os.urandom(32)
|
||||
|
||||
def reporter(self, tx):
|
||||
logg.debug('reporter {}'.format(tx))
|
||||
|
||||
def fee_price_helper(self):
|
||||
return 21
|
||||
|
||||
def fee_helper(self, sender, code, inputs):
|
||||
logg.debug('fee helper code {} inputs {}'.format(code, inputs))
|
||||
return 2
|
||||
|
||||
def builder(self, a, b):
|
||||
b = {
|
||||
'from': a['from'],
|
||||
'to': '0x' + os.urandom(20).hex(),
|
||||
'data': '',
|
||||
'gasPrice': a['feePrice'],
|
||||
'gas': a['fee'],
|
||||
}
|
||||
return b
|
||||
|
||||
def builder_two(self, a, b):
|
||||
b['value'] = 1024
|
||||
return b
|
||||
|
||||
|
||||
class TestHelper(unittest.TestCase):
|
||||
|
||||
def setUp(self):
|
||||
logg.debug('setup')
|
||||
self.db = DictKeystore()
|
||||
|
||||
keystore_filepath = os.path.join(script_dir, 'testdata', 'UTC--2021-01-08T18-37-01.187235289Z--00a329c0648769a73afac7f9381e08fb43dbea72')
|
||||
|
||||
self.address_hex = self.db.import_keystore_file(keystore_filepath, '')
|
||||
self.signer = ReferenceSigner(self.db)
|
||||
|
||||
|
||||
def tearDown(self):
|
||||
pass
|
||||
|
||||
|
||||
def test_helper(self):
|
||||
backend = MockEthTxBackend()
|
||||
executor = TxExecutor(self.address_hex, self.signer, backend.dispatcher, backend.reporter, 666, 13, backend.fee_helper, backend.fee_price_helper)
|
||||
|
||||
tx_ish = {'from': self.address_hex}
|
||||
executor.sign_and_send([backend.builder, backend.builder_two])
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
unittest.main()
|
@ -6,11 +6,6 @@ import logging
|
||||
import base64
|
||||
import os
|
||||
|
||||
# third-party imports
|
||||
import psycopg2
|
||||
from psycopg2 import sql
|
||||
from cryptography.fernet import Fernet, InvalidToken
|
||||
|
||||
# local imports
|
||||
from crypto_dev_signer.keystore import DictKeystore
|
||||
from crypto_dev_signer.error import UnknownAccountError
|
||||
@ -22,29 +17,16 @@ logg = logging.getLogger()
|
||||
script_dir = os.path.realpath(os.path.dirname(__file__))
|
||||
|
||||
|
||||
class TestDatabase(unittest.TestCase):
|
||||
class TestDict(unittest.TestCase):
|
||||
|
||||
conn = None
|
||||
cur = None
|
||||
symkey = None
|
||||
address_hex = None
|
||||
db = None
|
||||
|
||||
def setUp(self):
|
||||
logg.debug('setup')
|
||||
# arbitrary value
|
||||
#symkey_hex = 'E92431CAEE69313A7BE9E443C4ABEED9BF8157E9A13553B4D5D6E7D51B5021D9'
|
||||
#self.symkey = bytes.fromhex(symkey_hex)
|
||||
|
||||
#kw = {
|
||||
# 'symmetric_key': self.symkey,
|
||||
# }
|
||||
self.db = DictKeystore()
|
||||
|
||||
keystore_filepath = os.path.join(script_dir, 'testdata', 'UTC--2021-01-08T18-37-01.187235289Z--00a329c0648769a73afac7f9381e08fb43dbea72')
|
||||
#f = open(
|
||||
#s = f.read()
|
||||
#f.close()
|
||||
|
||||
self.address_hex = self.db.import_keystore_file(keystore_filepath, '')
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user