Add transaction executor helper

This commit is contained in:
nolash 2021-01-09 22:05:24 +01:00
parent 6ba0ac6820
commit f7c1f05a1f
Signed by: lash
GPG Key ID: 21D2E7BB88C2A746
8 changed files with 185 additions and 20 deletions

View File

@ -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

View File

@ -1,2 +1,6 @@
class UnknownAccountError(Exception):
pass
class TransactionRevertError(Exception):
pass

View File

@ -0,0 +1 @@
from .tx import TxExecutor

View 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

View 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

View File

@ -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
View 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()

View File

@ -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, '')