mirror of
git://holbrook.no/erc20-demurrage-token
synced 2024-11-16 14:16:46 +01:00
Revert to mine for every tx, add limit test
This commit is contained in:
parent
7bdd18664e
commit
4e11f750e8
@ -1 +1,2 @@
|
|||||||
from .sim import DemurrageTokenSimulation
|
from .sim import DemurrageTokenSimulation
|
||||||
|
from .error import TxLimitException
|
||||||
|
2
python/erc20_demurrage_token/sim/error.py
Normal file
2
python/erc20_demurrage_token/sim/error.py
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
class TxLimitException(RuntimeError):
|
||||||
|
pass
|
@ -17,6 +17,7 @@ from chainlib.eth.address import to_checksum_address
|
|||||||
from chainlib.eth.block import (
|
from chainlib.eth.block import (
|
||||||
block_latest,
|
block_latest,
|
||||||
block_by_number,
|
block_by_number,
|
||||||
|
block_by_hash,
|
||||||
)
|
)
|
||||||
from crypto_dev_signer.keystore.dict import DictKeystore
|
from crypto_dev_signer.keystore.dict import DictKeystore
|
||||||
from crypto_dev_signer.eth.signer import ReferenceSigner as EIP155Signer
|
from crypto_dev_signer.eth.signer import ReferenceSigner as EIP155Signer
|
||||||
@ -27,6 +28,7 @@ from hexathon import (
|
|||||||
|
|
||||||
# local imports
|
# local imports
|
||||||
from erc20_demurrage_token import DemurrageToken
|
from erc20_demurrage_token import DemurrageToken
|
||||||
|
from erc20_demurrage_token.sim.error import TxLimitException
|
||||||
|
|
||||||
logg = logging.getLogger(__name__)
|
logg = logging.getLogger(__name__)
|
||||||
|
|
||||||
@ -42,12 +44,10 @@ class DemurrageTokenSimulation:
|
|||||||
self.eth_backend = self.eth_helper.backend
|
self.eth_backend = self.eth_helper.backend
|
||||||
self.gas_oracle = OverrideGasOracle(limit=100000, price=1)
|
self.gas_oracle = OverrideGasOracle(limit=100000, price=1)
|
||||||
self.rpc = TestRPCConnection(None, self.eth_helper, self.signer)
|
self.rpc = TestRPCConnection(None, self.eth_helper, self.signer)
|
||||||
self.period = 1
|
|
||||||
for a in self.keystore.list():
|
for a in self.keystore.list():
|
||||||
self.accounts.append(add_0x(to_checksum_address(a)))
|
self.accounts.append(add_0x(to_checksum_address(a)))
|
||||||
settings.sink_address = self.accounts[0]
|
settings.sink_address = self.accounts[0]
|
||||||
|
|
||||||
|
|
||||||
self.actors = []
|
self.actors = []
|
||||||
for i in range(actors):
|
for i in range(actors):
|
||||||
idx = i % 10
|
idx = i % 10
|
||||||
@ -85,10 +85,17 @@ class DemurrageTokenSimulation:
|
|||||||
raise RuntimeError('contract deployment failed')
|
raise RuntimeError('contract deployment failed')
|
||||||
self.address = r['contract_address']
|
self.address = r['contract_address']
|
||||||
|
|
||||||
|
o = c.decimals(self.address, sender_address=self.accounts[0])
|
||||||
|
r = self.rpc.do(o)
|
||||||
|
self.decimals = c.parse_decimals(r)
|
||||||
|
|
||||||
self.period_seconds = settings.period_minutes * 60
|
self.period_seconds = settings.period_minutes * 60
|
||||||
|
|
||||||
self.last_block += 1
|
self.last_block += 1
|
||||||
self.last_timestamp += 1
|
self.last_timestamp += 1
|
||||||
|
self.period = 1
|
||||||
|
self.period_txs = []
|
||||||
|
self.period_tx_limit = self.period_seconds - 1
|
||||||
|
|
||||||
logg.info('intialized at block {} timestamp {} period {} demurrage level {} sink address {} (first address in keystore)'.format(
|
logg.info('intialized at block {} timestamp {} period {} demurrage level {} sink address {} (first address in keystore)'.format(
|
||||||
self.last_block,
|
self.last_block,
|
||||||
@ -102,15 +109,35 @@ class DemurrageTokenSimulation:
|
|||||||
self.eth_helper.disable_auto_mine_transactions()
|
self.eth_helper.disable_auto_mine_transactions()
|
||||||
|
|
||||||
self.caller_contract = DemurrageToken(self.chain_spec)
|
self.caller_contract = DemurrageToken(self.chain_spec)
|
||||||
|
self.caller_address = self.accounts[0]
|
||||||
|
|
||||||
|
|
||||||
|
def __check_limit(self):
|
||||||
|
if self.period_tx_limit == len(self.period_txs):
|
||||||
|
raise TxLimitException('reached period tx limit {}'.format(self.period_tx_limit))
|
||||||
|
|
||||||
|
|
||||||
|
def __check_tx(self, tx_hash):
|
||||||
|
o = receipt(tx_hash)
|
||||||
|
rcpt = self.rpc.do(o)
|
||||||
|
if rcpt['status'] == 0:
|
||||||
|
raise RuntimeError('tx {} (block {} index {}) failed'.format(tx_hash, self.last_block, rcpt['transaction_index']))
|
||||||
|
logg.debug('tx {} block {} index {} verified'.format(tx_hash, self.last_block, rcpt['transaction_index']))
|
||||||
|
|
||||||
|
|
||||||
|
def from_units(self, v):
|
||||||
|
return v * (10 ** self.decimals)
|
||||||
|
|
||||||
|
|
||||||
def mint(self, recipient, value):
|
def mint(self, recipient, value):
|
||||||
|
self.__check_limit()
|
||||||
nonce_oracle = RPCNonceOracle(self.accounts[0], conn=self.rpc)
|
nonce_oracle = RPCNonceOracle(self.accounts[0], conn=self.rpc)
|
||||||
c = DemurrageToken(self.chain_spec, signer=self.signer, nonce_oracle=nonce_oracle, gas_oracle=self.gas_oracle)
|
c = DemurrageToken(self.chain_spec, signer=self.signer, nonce_oracle=nonce_oracle, gas_oracle=self.gas_oracle)
|
||||||
(tx_hash, o) = c.mint_to(self.address, self.accounts[0], recipient, value)
|
(tx_hash, o) = c.mint_to(self.address, self.accounts[0], recipient, value)
|
||||||
self.rpc.do(o)
|
self.rpc.do(o)
|
||||||
logg.info('tx hash {}'.format(tx_hash))
|
self.next_block()
|
||||||
|
self.__check_tx(tx_hash)
|
||||||
|
self.period_txs.append(tx_hash)
|
||||||
return tx_hash
|
return tx_hash
|
||||||
|
|
||||||
|
|
||||||
@ -119,51 +146,61 @@ class DemurrageTokenSimulation:
|
|||||||
c = DemurrageToken(self.chain_spec, signer=self.signer, nonce_oracle=nonce_oracle, gas_oracle=self.gas_oracle)
|
c = DemurrageToken(self.chain_spec, signer=self.signer, nonce_oracle=nonce_oracle, gas_oracle=self.gas_oracle)
|
||||||
(tx_hash, o) = c.transfer(self.address, sender, recipient, value)
|
(tx_hash, o) = c.transfer(self.address, sender, recipient, value)
|
||||||
self.rpc.do(o)
|
self.rpc.do(o)
|
||||||
|
self.next_block()
|
||||||
|
self.__check_tx(tx_hash)
|
||||||
|
self.period_txs.append(tx_hash)
|
||||||
return tx_hash
|
return tx_hash
|
||||||
|
|
||||||
|
|
||||||
def balance(self, holder):
|
def balance(self, holder, base=False):
|
||||||
#o = self.caller_contract.balance_of(self.address, holder, sender_address=self.accounts[0])
|
o = None
|
||||||
c = DemurrageToken(self.chain_spec)
|
if base:
|
||||||
o = c.balance_of(self.address, holder, sender_address=self.accounts[0])
|
o = self.caller_contract.base_balance_of(self.address, holder, sender_address=self.caller_address)
|
||||||
|
else:
|
||||||
|
o = self.caller_contract.balance_of(self.address, holder, sender_address=self.caller_address)
|
||||||
r = self.rpc.do(o)
|
r = self.rpc.do(o)
|
||||||
return self.caller_contract.parse_balance_of(r)
|
return self.caller_contract.parse_balance_of(r)
|
||||||
|
|
||||||
|
|
||||||
|
def next_block(self):
|
||||||
|
hsh = self.eth_helper.mine_block()
|
||||||
|
o = block_by_hash(hsh)
|
||||||
|
r = self.rpc.do(o)
|
||||||
|
logg.info('now at block {} timestamp {}'.format(r['number'], r['timestamp']))
|
||||||
|
|
||||||
|
|
||||||
def next(self):
|
def next(self):
|
||||||
target_timestamp = self.period * self.period_seconds
|
target_timestamp = self.start_timestamp + (self.period * self.period_seconds) - 1
|
||||||
|
logg.debug('warping to {}, {} from start'.format(target_timestamp, target_timestamp - self.start_timestamp))
|
||||||
self.last_timestamp = target_timestamp
|
self.last_timestamp = target_timestamp
|
||||||
|
|
||||||
self.eth_helper.mine_block()
|
self.eth_helper.time_travel(self.last_timestamp)
|
||||||
|
self.next_block()
|
||||||
|
|
||||||
self.last_timestamp += 1
|
|
||||||
o = block_by_number(self.last_block)
|
o = block_by_number(self.last_block)
|
||||||
r = self.rpc.do(o)
|
r = self.rpc.do(o)
|
||||||
self.last_block = r['number']
|
self.last_block = r['number']
|
||||||
block_base = self.last_block
|
block_base = self.last_block
|
||||||
|
|
||||||
for tx_hash in r['transactions']:
|
# for tx_hash in r['transactions']:
|
||||||
o = receipt(tx_hash)
|
# o = receipt(tx_hash)
|
||||||
rcpt = self.rpc.do(o)
|
# rcpt = self.rpc.do(o)
|
||||||
if rcpt['status'] == 0:
|
# if rcpt['status'] == 0:
|
||||||
raise RuntimeError('tx {} (block {} index {}) failed'.format(tx_hash, self.last_block, rcpt['transaction_index']))
|
# raise RuntimeError('tx {} (block {} index {}) failed'.format(tx_hash, self.last_block, rcpt['transaction_index']))
|
||||||
logg.info('tx {} (block {} index {}) verified'.format(tx_hash, self.last_block, rcpt['transaction_index']))
|
# logg.info('tx {} (block {} index {}) verified'.format(tx_hash, self.last_block, rcpt['transaction_index']))
|
||||||
|
|
||||||
#self.eth_helper.time_travel(self.start_timestamp + self.last_timestamp)
|
|
||||||
|
|
||||||
nonce_oracle = RPCNonceOracle(self.accounts[0], conn=self.rpc)
|
nonce_oracle = RPCNonceOracle(self.accounts[0], conn=self.rpc)
|
||||||
c = DemurrageToken(self.chain_spec, signer=self.signer, nonce_oracle=nonce_oracle, gas_oracle=self.gas_oracle)
|
c = DemurrageToken(self.chain_spec, signer=self.signer, nonce_oracle=nonce_oracle, gas_oracle=self.gas_oracle)
|
||||||
(tx_hash, o) = c.apply_demurrage(self.address, self.accounts[0])
|
(tx_hash, o) = c.apply_demurrage(self.address, self.accounts[0])
|
||||||
self.rpc.do(o)
|
self.rpc.do(o)
|
||||||
#self.eth_helper.time_travel(self.start_timestamp + self.last_timestamp + 9)
|
self.next_block()
|
||||||
self.eth_helper.mine_block()
|
|
||||||
self.last_block += 1
|
|
||||||
|
|
||||||
(tx_hash, o) = c.change_period(self.address, self.accounts[0])
|
(tx_hash, o) = c.change_period(self.address, self.accounts[0])
|
||||||
self.rpc.do(o)
|
self.rpc.do(o)
|
||||||
#self.eth_helper.time_travel(self.start_timestamp + self.last_timestamp + 10)
|
self.next_block()
|
||||||
self.eth_helper.mine_block()
|
|
||||||
self.last_block += 1
|
o = block_latest()
|
||||||
|
self.last_block = self.rpc.do(o)
|
||||||
|
|
||||||
o = block_by_number(self.last_block)
|
o = block_by_number(self.last_block)
|
||||||
r = self.rpc.do(o)
|
r = self.rpc.do(o)
|
||||||
@ -173,19 +210,9 @@ class DemurrageTokenSimulation:
|
|||||||
if rcpt['status'] == 0:
|
if rcpt['status'] == 0:
|
||||||
raise RuntimeError('demurrage step failed on block {}'.format(self.last_block))
|
raise RuntimeError('demurrage step failed on block {}'.format(self.last_block))
|
||||||
|
|
||||||
# while True:
|
self.last_timestamp = r['timestamp']
|
||||||
# self.eth_helper.mine_block()
|
logg.info('next concludes at block {} timestamp {}, {} after start'.format(self.last_block, self.last_timestamp, self.last_timestamp - self.start_timestamp))
|
||||||
# o = block_latest()
|
|
||||||
# r = self.rpc.do(o)
|
|
||||||
# self.last_block = r
|
|
||||||
#
|
|
||||||
# o = block_by_number(r, include_tx=False)
|
|
||||||
# r = self.rpc.do(o)
|
|
||||||
# self.last_block =r['number']
|
|
||||||
# if r['timestamp'] == target_timestamp:
|
|
||||||
# break
|
|
||||||
|
|
||||||
self.last_timestamp += self.last_block - block_base
|
|
||||||
self.period += 1
|
self.period += 1
|
||||||
|
self.period_txs = []
|
||||||
|
|
||||||
return (self.last_block, self.last_timestamp)
|
return (self.last_block, self.last_timestamp)
|
||||||
|
@ -223,6 +223,21 @@ class DemurrageToken(ERC20):
|
|||||||
return o
|
return o
|
||||||
|
|
||||||
|
|
||||||
|
def base_balance_of(self, contract_address, address, sender_address=ZERO_ADDRESS):
|
||||||
|
o = jsonrpc_template()
|
||||||
|
o['method'] = 'eth_call'
|
||||||
|
enc = ABIContractEncoder()
|
||||||
|
enc.method('baseBalanceOf')
|
||||||
|
enc.typ(ABIContractType.ADDRESS)
|
||||||
|
enc.address(address)
|
||||||
|
data = add_0x(enc.get())
|
||||||
|
tx = self.template(sender_address, contract_address)
|
||||||
|
tx = self.set_code(tx, data)
|
||||||
|
o['params'].append(self.normalize(tx))
|
||||||
|
o['params'].append('latest')
|
||||||
|
return o
|
||||||
|
|
||||||
|
|
||||||
def apply_demurrage(self, contract_address, sender_address):
|
def apply_demurrage(self, contract_address, sender_address):
|
||||||
return self.transact_noarg('applyDemurrage', contract_address, sender_address)
|
return self.transact_noarg('applyDemurrage', contract_address, sender_address)
|
||||||
|
|
||||||
|
@ -17,14 +17,13 @@ class TestSim(unittest.TestCase):
|
|||||||
|
|
||||||
def setUp(self):
|
def setUp(self):
|
||||||
self.chain_spec = ChainSpec('evm', 'foochain', 42)
|
self.chain_spec = ChainSpec('evm', 'foochain', 42)
|
||||||
#self.cap = 1000000000
|
|
||||||
self.cap = 0
|
self.cap = 0
|
||||||
settings = DemurrageTokenSettings()
|
settings = DemurrageTokenSettings()
|
||||||
settings.name = 'Simulated Demurrage Token'
|
settings.name = 'Simulated Demurrage Token'
|
||||||
settings.symbol = 'SIM'
|
settings.symbol = 'SIM'
|
||||||
settings.decimals = 6
|
settings.decimals = 6
|
||||||
settings.demurrage_level = 50
|
settings.demurrage_level = 5010590837337300000000000000000000 # equals approx 2% per month
|
||||||
settings.period_minutes = 10800
|
settings.period_minutes = 10800 # 1 week in minutes
|
||||||
self.sim = DemurrageTokenSimulation(self.chain_spec, settings, redistribute=True, cap=self.cap, actors=10)
|
self.sim = DemurrageTokenSimulation(self.chain_spec, settings, redistribute=True, cap=self.cap, actors=10)
|
||||||
|
|
||||||
|
|
||||||
@ -32,7 +31,7 @@ class TestSim(unittest.TestCase):
|
|||||||
self.sim.mint(self.sim.actors[0], 1024)
|
self.sim.mint(self.sim.actors[0], 1024)
|
||||||
self.sim.next()
|
self.sim.next()
|
||||||
balance = self.sim.balance(self.sim.actors[0])
|
balance = self.sim.balance(self.sim.actors[0])
|
||||||
self.assertEqual(balance, 1024)
|
self.assertEqual(balance, 1023)
|
||||||
|
|
||||||
|
|
||||||
def test_transfer(self):
|
def test_transfer(self):
|
||||||
@ -40,21 +39,38 @@ class TestSim(unittest.TestCase):
|
|||||||
self.sim.transfer(self.sim.actors[0], self.sim.actors[1], 500)
|
self.sim.transfer(self.sim.actors[0], self.sim.actors[1], 500)
|
||||||
self.sim.next()
|
self.sim.next()
|
||||||
balance = self.sim.balance(self.sim.actors[0])
|
balance = self.sim.balance(self.sim.actors[0])
|
||||||
self.assertEqual(balance, 524)
|
self.assertEqual(balance, 523)
|
||||||
|
|
||||||
balance = self.sim.balance(self.sim.actors[1])
|
balance = self.sim.balance(self.sim.actors[1])
|
||||||
self.assertEqual(balance, 500)
|
self.assertEqual(balance, 499)
|
||||||
|
|
||||||
|
|
||||||
def test_more_periods(self):
|
def test_more_periods(self):
|
||||||
self.sim.mint(self.sim.actors[0], 1024)
|
self.sim.mint(self.sim.actors[0], 1024)
|
||||||
|
self.sim.mint(self.sim.actors[1], 1024)
|
||||||
self.sim.next()
|
self.sim.next()
|
||||||
|
|
||||||
self.sim.mint(self.sim.actors[0], 1024)
|
self.sim.mint(self.sim.actors[0], 1024)
|
||||||
self.sim.next()
|
self.sim.next()
|
||||||
|
|
||||||
balance = self.sim.balance(self.sim.actors[0])
|
balance = self.sim.balance(self.sim.actors[0])
|
||||||
self.assertEqual(balance, 2048)
|
self.assertEqual(balance, 2047)
|
||||||
|
|
||||||
|
|
||||||
|
def test_demurrage(self):
|
||||||
|
self.sim.mint(self.sim.actors[0], self.sim.from_units(100))
|
||||||
|
self.sim.mint(self.sim.actors[1], self.sim.from_units(100))
|
||||||
|
self.sim.transfer(self.sim.actors[0], self.sim.actors[2], self.sim.from_units(10))
|
||||||
|
self.sim.next()
|
||||||
|
|
||||||
|
balance = self.sim.balance(self.sim.actors[0])
|
||||||
|
self.assertEqual(balance, 89995500)
|
||||||
|
|
||||||
|
balance = self.sim.balance(self.sim.actors[1])
|
||||||
|
self.assertEqual(balance, 99995000)
|
||||||
|
|
||||||
|
balance = self.sim.balance(self.sim.actors[1], base=True)
|
||||||
|
self.assertEqual(balance, 100000000)
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
|
39
python/tests/sim/tests_sim_limit.py
Normal file
39
python/tests/sim/tests_sim_limit.py
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
# standard imports
|
||||||
|
import unittest
|
||||||
|
import logging
|
||||||
|
|
||||||
|
# external imports
|
||||||
|
from chainlib.chain import ChainSpec
|
||||||
|
|
||||||
|
# local imports
|
||||||
|
from erc20_demurrage_token import DemurrageTokenSettings
|
||||||
|
from erc20_demurrage_token.sim import (
|
||||||
|
DemurrageTokenSimulation,
|
||||||
|
TxLimitException,
|
||||||
|
)
|
||||||
|
|
||||||
|
logging.basicConfig(level=logging.INFO)
|
||||||
|
logg = logging.getLogger()
|
||||||
|
|
||||||
|
class TestLimit(unittest.TestCase):
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
self.chain_spec = ChainSpec('evm', 'foochain', 42)
|
||||||
|
self.cap = 0
|
||||||
|
settings = DemurrageTokenSettings()
|
||||||
|
settings.name = 'Simulated Demurrage Token'
|
||||||
|
settings.symbol = 'SIM'
|
||||||
|
settings.decimals = 6
|
||||||
|
settings.demurrage_level = 1
|
||||||
|
settings.period_minutes = 1
|
||||||
|
self.sim = DemurrageTokenSimulation(self.chain_spec, settings, redistribute=True, cap=self.cap, actors=1)
|
||||||
|
|
||||||
|
|
||||||
|
def test_limit(self):
|
||||||
|
with self.assertRaises(TxLimitException):
|
||||||
|
for i in range(60):
|
||||||
|
self.sim.mint(self.sim.actors[0], i)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
unittest.main()
|
Loading…
Reference in New Issue
Block a user