ts_ethdev.py 38 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004
  1. #!/usr/bin/env python3
  2. #
  3. # mmgen = Multi-Mode GENerator, command-line Bitcoin cold storage solution
  4. # Copyright (C)2013-2021 The MMGen Project <mmgen@tuta.io>
  5. #
  6. # This program is free software: you can redistribute it and/or modify
  7. # it under the terms of the GNU General Public License as published by
  8. # the Free Software Foundation, either version 3 of the License, or
  9. # (at your option) any later version.
  10. #
  11. # This program is distributed in the hope that it will be useful,
  12. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. # GNU General Public License for more details.
  15. #
  16. # You should have received a copy of the GNU General Public License
  17. # along with this program. If not, see <http://www.gnu.org/licenses/>.
  18. """
  19. ts_ethdev.py: Ethdev tests for the test.py test suite
  20. """
  21. import sys,os,re,shutil
  22. from decimal import Decimal
  23. from subprocess import run,PIPE,DEVNULL
  24. from mmgen.globalvars import g
  25. from mmgen.opts import opt
  26. from mmgen.util import die
  27. from mmgen.exception import *
  28. from ..include.common import *
  29. from .common import *
  30. del_addrs = ('4','1')
  31. dfl_sid = '98831F3A'
  32. # The OpenEthereum dev address with lots of coins. Create with "ethkey -b info ''":
  33. dfl_addr = '00a329c0648769a73afac7f9381e08fb43dbea72'
  34. dfl_addr_chk = '00a329c0648769A73afAc7F9381E08FB43dBEA72'
  35. dfl_privkey = '4d5db4107d237df6a3d58ee5f70ae63d73d7658d4026f2eefd2f204c81682cb7'
  36. burn_addr = 'deadbeef'*5
  37. amt1 = '999999.12345689012345678'
  38. amt2 = '888.111122223333444455'
  39. openethereum_key_fn = 'openethereum.devkey'
  40. # Token sends require varying amounts of gas, depending on compiler version
  41. def get_solc_ver():
  42. try: cp = run(['solc','--version'],stdout=PIPE)
  43. except: return None
  44. if cp.returncode:
  45. return None
  46. line = cp.stdout.decode().splitlines()[1]
  47. m = re.search(r'Version:\s*(\d+)\.(\d+)\.(\d+)',line)
  48. return '.'.join(m.groups()) if m else None
  49. solc_ver = get_solc_ver()
  50. if solc_ver == '0.5.1':
  51. vbal1 = '1.2288337'
  52. vbal1a = 'TODO'
  53. vbal2 = '99.997085083'
  54. vbal3 = '1.23142165'
  55. vbal4 = '127.0287837'
  56. else: # 0.5.3 or precompiled 0.5.3
  57. vbal1 = '1.2288487'
  58. vbal1a = '1.22627465'
  59. vbal2 = '99.997092733'
  60. vbal3 = '1.23142915'
  61. vbal4 = '127.0287987'
  62. bals = {
  63. '1': [ ('98831F3A:E:1','123.456')],
  64. '2': [ ('98831F3A:E:1','123.456'),('98831F3A:E:11','1.234')],
  65. '3': [ ('98831F3A:E:1','123.456'),('98831F3A:E:11','1.234'),('98831F3A:E:21','2.345')],
  66. '4': [ ('98831F3A:E:1','100'),
  67. ('98831F3A:E:2','23.45495'),
  68. ('98831F3A:E:11','1.234'),
  69. ('98831F3A:E:21','2.345')],
  70. '5': [ ('98831F3A:E:1','100'),
  71. ('98831F3A:E:2','23.45495'),
  72. ('98831F3A:E:11','1.234'),
  73. ('98831F3A:E:21','2.345'),
  74. (burn_addr + '\s+Non-MMGen',amt1)],
  75. '8': [ ('98831F3A:E:1','0'),
  76. ('98831F3A:E:2','23.45495'),
  77. ('98831F3A:E:11',vbal1,'a1'),
  78. ('98831F3A:E:12','99.99895'),
  79. ('98831F3A:E:21','2.345'),
  80. (burn_addr + '\s+Non-MMGen',amt1)],
  81. '9': [ ('98831F3A:E:1','0'),
  82. ('98831F3A:E:2','23.45495'),
  83. ('98831F3A:E:11',vbal1,'a1'),
  84. ('98831F3A:E:12',vbal2),
  85. ('98831F3A:E:21','2.345'),
  86. (burn_addr + '\s+Non-MMGen',amt1)],
  87. '10': [ ('98831F3A:E:1','0'),
  88. ('98831F3A:E:2','23.0218'),
  89. ('98831F3A:E:3','0.4321'),
  90. ('98831F3A:E:11',vbal1,'a1'),
  91. ('98831F3A:E:12',vbal2),
  92. ('98831F3A:E:21','2.345'),
  93. (burn_addr + '\s+Non-MMGen',amt1)]
  94. }
  95. token_bals = {
  96. '1': [ ('98831F3A:E:11','1000','1.234')],
  97. '2': [ ('98831F3A:E:11','998.76544',vbal3,'a1'),
  98. ('98831F3A:E:12','1.23456','0')],
  99. '3': [ ('98831F3A:E:11','110.654317776666555545',vbal1,'a1'),
  100. ('98831F3A:E:12','1.23456','0')],
  101. '4': [ ('98831F3A:E:11','110.654317776666555545',vbal1,'a1'),
  102. ('98831F3A:E:12','1.23456','0'),
  103. (burn_addr + '\s+Non-MMGen',amt2,amt1)],
  104. '5': [ ('98831F3A:E:11','110.654317776666555545',vbal1,'a1'),
  105. ('98831F3A:E:12','1.23456','99.99895'),
  106. (burn_addr + '\s+Non-MMGen',amt2,amt1)],
  107. '6': [ ('98831F3A:E:11','110.654317776666555545',vbal1,'a1'),
  108. ('98831F3A:E:12','0',vbal2),
  109. ('98831F3A:E:13','1.23456','0'),
  110. (burn_addr + '\s+Non-MMGen',amt2,amt1)],
  111. '7': [ ('98831F3A:E:11','67.444317776666555545',vbal1a,'a2'),
  112. ('98831F3A:E:12','43.21',vbal2),
  113. ('98831F3A:E:13','1.23456','0'),
  114. (burn_addr + '\s+Non-MMGen',amt2,amt1)]
  115. }
  116. token_bals_getbalance = {
  117. '1': (vbal4,'999999.12345689012345678'),
  118. '2': ('111.888877776666555545','888.111122223333444455')
  119. }
  120. from .ts_base import *
  121. from .ts_shared import *
  122. coin = g.coin
  123. class TestSuiteEthdev(TestSuiteBase,TestSuiteShared):
  124. 'Ethereum transacting, token deployment and tracking wallet operations'
  125. networks = ('eth','etc')
  126. passthru_opts = ('coin','daemon_id','http_timeout')
  127. extra_spawn_args = ['--regtest=1']
  128. tmpdir_nums = [22]
  129. solc_vers = ('0.5.1','0.5.3') # 0.5.1: Raspbian Stretch, 0.5.3: Ubuntu Bionic
  130. color = True
  131. cmd_group = (
  132. ('setup', 'dev mode tests for coin {} (start daemon)'.format(coin)),
  133. ('daemon_version', 'mmgen-tool daemon_version'),
  134. ('wallet_upgrade1', 'upgrading the tracking wallet (v1 -> v2)'),
  135. ('wallet_upgrade2', 'upgrading the tracking wallet (v2 -> v3)'),
  136. ('addrgen', 'generating addresses'),
  137. ('addrimport', 'importing addresses'),
  138. ('addrimport_dev_addr', "importing OpenEthereum dev address 'Ox00a329c..'"),
  139. ('txcreate1', 'creating a transaction (spend from dev address to address :1)'),
  140. ('txview1_raw', 'viewing the raw transaction'),
  141. ('txsign1', 'signing the transaction'),
  142. ('txview1_sig', 'viewing the signed transaction'),
  143. ('tx_status0_bad', 'getting the transaction status'),
  144. ('txsign1_ni', 'signing the transaction (non-interactive)'),
  145. ('txsend1', 'sending the transaction'),
  146. ('bal1', 'the {} balance'.format(coin)),
  147. ('txcreate2', 'creating a transaction (spend from dev address to address :11)'),
  148. ('txsign2', 'signing the transaction'),
  149. ('txsend2', 'sending the transaction'),
  150. ('bal2', 'the {} balance'.format(coin)),
  151. ('txcreate3', 'creating a transaction (spend from dev address to address :21)'),
  152. ('txsign3', 'signing the transaction'),
  153. ('txsend3', 'sending the transaction'),
  154. ('bal3', 'the {} balance'.format(coin)),
  155. ('tx_status1', 'getting the transaction status'),
  156. ('txcreate4', 'creating a transaction (spend from MMGen address, low TX fee)'),
  157. ('txbump', 'bumping the transaction fee'),
  158. ('txsign4', 'signing the transaction'),
  159. ('txsend4', 'sending the transaction'),
  160. ('tx_status1a', 'getting the transaction status'),
  161. ('bal4', 'the {} balance'.format(coin)),
  162. ('txcreate5', 'creating a transaction (fund burn address)'),
  163. ('txsign5', 'signing the transaction'),
  164. ('txsend5', 'sending the transaction'),
  165. ('addrimport_burn_addr',"importing burn address"),
  166. ('bal5', 'the {} balance'.format(coin)),
  167. ('add_label1', 'adding a UTF-8 label (zh)'),
  168. ('chk_label1', 'the label'),
  169. ('add_label2', 'adding a UTF-8 label (lat+cyr+gr)'),
  170. ('chk_label2', 'the label'),
  171. ('remove_label', 'removing the label'),
  172. ('token_compile1', 'compiling ERC20 token #1'),
  173. ('token_deploy1a', 'deploying ERC20 token #1 (SafeMath)'),
  174. ('token_deploy1b', 'deploying ERC20 token #1 (Owned)'),
  175. ('token_deploy1c', 'deploying ERC20 token #1 (Token)'),
  176. ('tx_status2', 'getting the transaction status'),
  177. ('bal6', 'the {} balance'.format(coin)),
  178. ('token_compile2', 'compiling ERC20 token #2'),
  179. ('token_deploy2a', 'deploying ERC20 token #2 (SafeMath)'),
  180. ('token_deploy2b', 'deploying ERC20 token #2 (Owned)'),
  181. ('token_deploy2c', 'deploying ERC20 token #2 (Token)'),
  182. ('contract_deploy', 'deploying contract (create,sign,send)'),
  183. ('token_fund_users', 'transferring token funds from dev to user'),
  184. ('token_user_bals', 'show balances after transfer'),
  185. ('token_addrgen', 'generating token addresses'),
  186. ('token_addrimport_badaddr1','importing token addresses (no token address)'),
  187. ('token_addrimport_badaddr2','importing token addresses (bad token address)'),
  188. ('token_addrimport_addr1','importing token addresses using token address (MM1)'),
  189. ('token_addrimport_addr2','importing token addresses using token address (MM2)'),
  190. ('token_addrimport_batch','importing token addresses (dummy batch mode) (MM1)'),
  191. ('token_addrimport_sym', 'importing token addresses using token symbol (MM2)'),
  192. ('bal7', 'the {} balance'.format(coin)),
  193. ('token_bal1', 'the {} balance and token balance'.format(coin)),
  194. ('token_txcreate1', 'creating a token transaction'),
  195. ('token_txview1_raw', 'viewing the raw transaction'),
  196. ('token_txsign1', 'signing the transaction'),
  197. ('token_txsend1', 'sending the transaction'),
  198. ('token_txview1_sig', 'viewing the signed transaction'),
  199. ('tx_status3', 'getting the transaction status'),
  200. ('token_bal2', 'the {} balance and token balance'.format(coin)),
  201. ('token_txcreate2', 'creating a token transaction (to burn address)'),
  202. ('token_txbump', 'bumping the transaction fee'),
  203. ('token_txsign2', 'signing the transaction'),
  204. ('token_txsend2', 'sending the transaction'),
  205. ('token_bal3', 'the {} balance and token balance'.format(coin)),
  206. ('del_dev_addr', "deleting the dev address"),
  207. ('bal1_getbalance', 'the {} balance (getbalance)'.format(coin)),
  208. ('addrimport_token_burn_addr',"importing the token burn address"),
  209. ('token_bal4', 'the {} balance and token balance'.format(coin)),
  210. ('token_bal_getbalance','the token balance (getbalance)'),
  211. ('txcreate_noamt', 'creating a transaction (full amount send)'),
  212. ('txsign_noamt', 'signing the transaction'),
  213. ('txsend_noamt', 'sending the transaction'),
  214. ('bal8', 'the {} balance'.format(coin)),
  215. ('token_bal5', 'the token balance'),
  216. ('token_txcreate_noamt', 'creating a token transaction (full amount send)'),
  217. ('token_txsign_noamt', 'signing the transaction'),
  218. ('token_txsend_noamt', 'sending the transaction'),
  219. ('bal9', 'the {} balance'.format(coin)),
  220. ('token_bal6', 'the token balance'),
  221. ('listaddresses1', 'listaddresses'),
  222. ('listaddresses2', 'listaddresses minconf=999999999 (ignored)'),
  223. ('listaddresses3', 'listaddresses sort=age (ignored)'),
  224. ('listaddresses4', 'listaddresses showempty=1 sort=age (ignored)'),
  225. ('token_listaddresses1','listaddresses --token=mm1'),
  226. ('token_listaddresses2','listaddresses --token=mm1 showempty=1'),
  227. ('twview_cached_balances','twview (cached balances)'),
  228. ('token_twview_cached_balances','token twview (cached balances)'),
  229. ('txcreate_cached_balances','txcreate (cached balances)'),
  230. ('token_txcreate_cached_balances','token txcreate (cached balances)'),
  231. ('txdo_cached_balances', 'txdo (cached balances)'),
  232. ('txcreate_refresh_balances','refreshing balances'),
  233. ('bal10', 'the {} balance'.format(coin)),
  234. ('token_txdo_cached_balances', 'token txdo (cached balances)'),
  235. ('token_txcreate_refresh_balances','refreshing token balances'),
  236. ('token_bal7', 'the token balance'),
  237. ('twview1','twview'),
  238. ('twview2','twview wide=1'),
  239. ('twview3','twview wide=1 sort=age (ignored)'),
  240. ('twview4','twview wide=1 minconf=999999999 (ignored)'),
  241. ('twview5','twview wide=1 minconf=0 (ignored)'),
  242. ('token_twview1','twview --token=mm1'),
  243. ('token_twview2','twview --token=mm1 wide=1'),
  244. ('token_twview3','twview --token=mm1 wide=1 sort=age (ignored)'),
  245. ('edit_label1','adding label to addr #{} in {} tracking wallet (zh)'.format(del_addrs[0],coin)),
  246. ('edit_label2','adding label to addr #{} in {} tracking wallet (lat+cyr+gr)'.format(del_addrs[1],coin)),
  247. ('edit_label3','removing label from addr #{} in {} tracking wallet'.format(del_addrs[0],coin)),
  248. ('token_edit_label1','adding label to addr #{} in {} token tracking wallet'.format(del_addrs[0],coin)),
  249. ('remove_addr1','removing addr #{} from {} tracking wallet'.format(del_addrs[0],coin)),
  250. ('remove_addr2','removing addr #{} from {} tracking wallet'.format(del_addrs[1],coin)),
  251. ('token_remove_addr1','removing addr #{} from {} token tracking wallet'.format(del_addrs[0],coin)),
  252. ('token_remove_addr2','removing addr #{} from {} token tracking wallet'.format(del_addrs[1],coin)),
  253. ('stop', 'stopping daemon'),
  254. )
  255. def __init__(self,trunner,cfgs,spawn):
  256. TestSuiteBase.__init__(self,trunner,cfgs,spawn)
  257. from mmgen.protocol import init_proto
  258. self.proto = init_proto(g.coin,network='regtest')
  259. from mmgen.daemon import CoinDaemon
  260. self.rpc_port = CoinDaemon(proto=self.proto,test_suite=True).rpc_port
  261. @property
  262. def eth_args(self):
  263. return ['--outdir={}'.format(self.tmpdir),'--coin='+self.proto.coin,'--rpc-port={}'.format(self.rpc_port),'--quiet']
  264. async def setup(self):
  265. self.spawn('',msg_only=True)
  266. if solc_ver in self.solc_vers:
  267. imsg('Found solc version {}'.format(solc_ver))
  268. else:
  269. imsg('Solc compiler {}. Using precompiled contract data'.format(
  270. 'version {} not supported by test suite'.format(solc_ver)
  271. if solc_ver else 'not found' ))
  272. srcdir = os.path.join(self.tr.repo_root,'test','ref','ethereum','bin')
  273. from shutil import copytree
  274. for d in ('mm1','mm2'):
  275. copytree(os.path.join(srcdir,d),os.path.join(self.tmpdir,d))
  276. if not opt.no_daemon_autostart:
  277. if not start_test_daemons(self.proto.coin+'_rt',remove_datadir=True):
  278. return False
  279. from mmgen.rpc import rpc_init
  280. rpc = await rpc_init(self.proto)
  281. imsg('Daemon: {} v{}'.format(rpc.daemon.coind_name,rpc.daemon_version_str))
  282. return 'ok'
  283. def wallet_upgrade(self,src_file):
  284. if self.proto.coin == 'ETC':
  285. msg('skipping test {!r} for ETC'.format(self.test_name))
  286. return 'skip'
  287. src_dir = joinpath(ref_dir,'ethereum')
  288. dest_dir = joinpath(self.tr.data_dir,'altcoins',self.proto.coin.lower())
  289. w_from = joinpath(src_dir,src_file)
  290. w_to = joinpath(dest_dir,'tracking-wallet.json')
  291. os.makedirs(dest_dir,mode=0o750,exist_ok=True)
  292. dest = shutil.copy2(w_from,w_to)
  293. assert dest == w_to, dest
  294. t = self.spawn('mmgen-tool', self.eth_args + ['twview'])
  295. t.read()
  296. os.unlink(w_to)
  297. return t
  298. def daemon_version(self):
  299. t = self.spawn('mmgen-tool', self.eth_args + ['daemon_version'])
  300. t.expect('version')
  301. return t
  302. def wallet_upgrade1(self): return self.wallet_upgrade('tracking-wallet-v1.json')
  303. def wallet_upgrade2(self): return self.wallet_upgrade('tracking-wallet-v2.json')
  304. def addrgen(self,addrs='1-3,11-13,21-23'):
  305. t = self.spawn('mmgen-addrgen', self.eth_args + [dfl_words_file,addrs])
  306. t.written_to_file('Addresses')
  307. t.read()
  308. return t
  309. def addrimport(self,ext='21-23]{}.regtest.addrs',expect='9/9',add_args=[],bad_input=False):
  310. ext = ext.format('-α' if g.debug_utf8 else '')
  311. fn = self.get_file_with_ext(ext,no_dot=True,delete=False)
  312. t = self.spawn('mmgen-addrimport', self.eth_args[1:-1] + add_args + [fn])
  313. if bad_input:
  314. t.read()
  315. return t
  316. t.expect('Importing')
  317. t.expect(expect)
  318. t.read()
  319. return t
  320. def addrimport_one_addr(self,addr=None,extra_args=[]):
  321. t = self.spawn('mmgen-addrimport', self.eth_args[1:] + extra_args + ['--address='+addr])
  322. t.expect('OK')
  323. return t
  324. def addrimport_dev_addr(self):
  325. return self.addrimport_one_addr(addr=dfl_addr)
  326. def addrimport_burn_addr(self):
  327. return self.addrimport_one_addr(addr=burn_addr)
  328. def txcreate(self,args=[],menu=[],acct='1',non_mmgen_inputs=0,caller='txcreate',
  329. interactive_fee = '50G',
  330. eth_fee_res = None,
  331. fee_res_data = ('0.00105','50'),
  332. fee_desc = 'gas price',
  333. no_read = False,
  334. tweaks = [] ):
  335. fee_res = r'\D{}\D.*{c} .*\D{}\D.*gas price in Gwei'.format(*fee_res_data,c=self.proto.coin)
  336. t = self.spawn('mmgen-'+caller, self.eth_args + ['-B'] + args)
  337. t.expect(r'add \[l\]abel, .*?:.','p', regex=True)
  338. t.written_to_file('Account balances listing')
  339. t = self.txcreate_ui_common( t, menu=menu, caller=caller,
  340. input_sels_prompt = 'to spend from',
  341. inputs = acct,
  342. file_desc = 'transaction',
  343. bad_input_sels = True,
  344. non_mmgen_inputs = non_mmgen_inputs,
  345. interactive_fee = interactive_fee,
  346. fee_res = fee_res,
  347. fee_desc = fee_desc,
  348. eth_fee_res = eth_fee_res,
  349. add_comment = tx_label_jp,
  350. tweaks = tweaks )
  351. if not no_read:
  352. t.read()
  353. return t
  354. def txsign(self,ni=False,ext='{}.regtest.rawtx',add_args=[]):
  355. ext = ext.format('-α' if g.debug_utf8 else '')
  356. keyfile = joinpath(self.tmpdir,openethereum_key_fn)
  357. write_to_file(keyfile,dfl_privkey+'\n')
  358. txfile = self.get_file_with_ext(ext,no_dot=True)
  359. t = self.spawn( 'mmgen-txsign',
  360. ['--outdir={}'.format(self.tmpdir),'--coin='+self.proto.coin,'--quiet']
  361. + ['--rpc-host=bad_host'] # ETH signing must work without RPC
  362. + add_args
  363. + ([],['--yes'])[ni]
  364. + ['-k', keyfile, txfile, dfl_words_file] )
  365. return self.txsign_ui_common(t,ni=ni,has_label=True)
  366. def txsend(self,ni=False,bogus_send=False,ext='{}.regtest.sigtx',add_args=[]):
  367. ext = ext.format('-α' if g.debug_utf8 else '')
  368. txfile = self.get_file_with_ext(ext,no_dot=True)
  369. if not bogus_send: os.environ['MMGEN_BOGUS_SEND'] = ''
  370. t = self.spawn('mmgen-txsend', self.eth_args + add_args + [txfile])
  371. if not bogus_send: os.environ['MMGEN_BOGUS_SEND'] = '1'
  372. txid = self.txsend_ui_common(t,quiet=not g.debug,bogus_send=bogus_send,has_label=True)
  373. return t
  374. def txview(self,ext_fs):
  375. ext = ext_fs.format('-α' if g.debug_utf8 else '')
  376. txfile = self.get_file_with_ext(ext,no_dot=True)
  377. t = self.spawn( 'mmgen-tool',['--verbose','txview',txfile] )
  378. t.read()
  379. return t
  380. def txcreate1(self):
  381. # valid_keypresses = EthereumTwUnspentOutputs.key_mappings.keys()
  382. menu = ['a','d','r','M','X','e','m','m'] # include one invalid keypress, 'X'
  383. args = ['98831F3A:E:1,123.456']
  384. return self.txcreate(args=args,menu=menu,acct='1',non_mmgen_inputs=1,tweaks=['confirm_non_mmgen'])
  385. def txview1_raw(self):
  386. return self.txview(ext_fs='{}.regtest.rawtx')
  387. def txsign1(self): return self.txsign(add_args=['--use-internal-keccak-module'])
  388. def tx_status0_bad(self):
  389. return self.tx_status(ext='{}.regtest.sigtx',expect_str='neither in mempool nor blockchain',exit_val=1)
  390. def txsign1_ni(self): return self.txsign(ni=True)
  391. def txsend1(self): return self.txsend()
  392. def txview1_sig(self): # do after send so that TxID is displayed
  393. return self.txview(ext_fs='{}.regtest.sigtx')
  394. def bal1(self): return self.bal(n='1')
  395. def txcreate2(self):
  396. args = ['98831F3A:E:11,1.234']
  397. return self.txcreate(args=args,acct='10',non_mmgen_inputs=1,tweaks=['confirm_non_mmgen'])
  398. def txsign2(self): return self.txsign(ni=True,ext='1.234,50000]{}.regtest.rawtx')
  399. def txsend2(self): return self.txsend(ext='1.234,50000]{}.regtest.sigtx')
  400. def bal2(self): return self.bal(n='2')
  401. def txcreate3(self):
  402. args = ['98831F3A:E:21,2.345']
  403. return self.txcreate(args=args,acct='10',non_mmgen_inputs=1,tweaks=['confirm_non_mmgen'])
  404. def txsign3(self): return self.txsign(ni=True,ext='2.345,50000]{}.regtest.rawtx')
  405. def txsend3(self): return self.txsend(ext='2.345,50000]{}.regtest.sigtx')
  406. def bal3(self): return self.bal(n='3')
  407. def tx_status(self,ext,expect_str,expect_str2='',add_args=[],exit_val=0):
  408. ext = ext.format('-α' if g.debug_utf8 else '')
  409. txfile = self.get_file_with_ext(ext,no_dot=True)
  410. t = self.spawn('mmgen-txsend', self.eth_args + add_args + ['--status',txfile])
  411. t.expect(expect_str)
  412. if expect_str2:
  413. t.expect(expect_str2)
  414. t.read()
  415. t.req_exit_val = exit_val
  416. return t
  417. def tx_status1(self):
  418. return self.tx_status(ext='2.345,50000]{}.regtest.sigtx',expect_str='has 1 confirmation')
  419. def tx_status1a(self):
  420. return self.tx_status(ext='2.345,50000]{}.regtest.sigtx',expect_str='has 2 confirmations')
  421. def txcreate4(self):
  422. return self.txcreate( args = ['98831F3A:E:2,23.45495'],
  423. acct = '1',
  424. non_mmgen_inputs = 0,
  425. interactive_fee = '40G',
  426. fee_res_data = ('0.00084','40'),
  427. eth_fee_res = True )
  428. def txbump(self,ext=',40000]{}.regtest.rawtx',fee='50G',add_args=[]):
  429. ext = ext.format('-α' if g.debug_utf8 else '')
  430. txfile = self.get_file_with_ext(ext,no_dot=True)
  431. t = self.spawn('mmgen-txbump', self.eth_args + add_args + ['--yes',txfile])
  432. t.expect('or gas price: ',fee+'\n')
  433. t.read()
  434. return t
  435. def txsign4(self): return self.txsign(ni=True,ext='.45495,50000]{}.regtest.rawtx')
  436. def txsend4(self): return self.txsend(ext='.45495,50000]{}.regtest.sigtx')
  437. def bal4(self): return self.bal(n='4')
  438. def txcreate5(self):
  439. args = [burn_addr + ','+amt1]
  440. return self.txcreate(args=args,acct='10',non_mmgen_inputs=1,tweaks=['confirm_non_mmgen'])
  441. def txsign5(self): return self.txsign(ni=True,ext=amt1+',50000]{}.regtest.rawtx')
  442. def txsend5(self): return self.txsend(ext=amt1+',50000]{}.regtest.sigtx')
  443. def bal5(self): return self.bal(n='5')
  444. #bal_corr = Decimal('0.0000032') # gas use for token sends varies between ETH and ETC!
  445. bal_corr = Decimal('0.0000000') # update: OpenEthereum team seems to have corrected this
  446. def bal(self,n):
  447. t = self.spawn('mmgen-tool', self.eth_args + ['twview','wide=1'])
  448. text = strip_ansi_escapes(t.read())
  449. for b in bals[n]:
  450. addr,amt,adj = b if len(b) == 3 else b + (False,)
  451. if adj and self.proto.coin == 'ETC':
  452. amt = str(Decimal(amt) + Decimal(adj[1]) * self.bal_corr)
  453. pat = r'\D{}\D.*\D{}\D'.format(addr,amt.replace('.',r'\.'))
  454. assert re.search(pat,text), pat
  455. ss = f'Total {self.proto.coin}:'
  456. assert re.search(ss,text),ss
  457. return t
  458. def token_bal(self,n=None):
  459. t = self.spawn('mmgen-tool', self.eth_args + ['--token=mm1','twview','wide=1'])
  460. text = strip_ansi_escapes(t.read())
  461. for b in token_bals[n]:
  462. addr,_amt1,_amt2,adj = b if len(b) == 4 else b + (False,)
  463. if adj and self.proto.coin == 'ETC':
  464. _amt2 = str(Decimal(_amt2) + Decimal(adj[1]) * self.bal_corr)
  465. pat = r'{}\b.*\D{}\D.*\b{}\D'.format(addr,_amt1,_amt2)
  466. assert re.search(pat,text), pat
  467. ss = 'Total MM1:'
  468. assert re.search(ss,text),ss
  469. return t
  470. def bal_getbalance(self,idx,etc_adj=False,extra_args=[]):
  471. bal1 = token_bals_getbalance[idx][0]
  472. bal2 = token_bals_getbalance[idx][1]
  473. bal1 = Decimal(bal1)
  474. if etc_adj and self.proto.coin == 'ETC':
  475. bal1 += self.bal_corr
  476. t = self.spawn('mmgen-tool', self.eth_args + extra_args + ['getbalance'])
  477. t.expect(r'\n[0-9A-F]{8}: .*\D'+str(bal1),regex=True)
  478. t.expect(r'\nNon-MMGen: .*\D'+bal2,regex=True)
  479. total = strip_ansi_escapes(t.expect_getend(r'\nTOTAL:\s+',regex=True)).split()[0]
  480. t.read()
  481. assert Decimal(bal1) + Decimal(bal2) == Decimal(total)
  482. return t
  483. def add_label(self,lbl,addr='98831F3A:E:3'):
  484. t = self.spawn('mmgen-tool', self.eth_args + ['add_label',addr,lbl])
  485. t.expect('Added label.*in tracking wallet',regex=True)
  486. return t
  487. def chk_label(self,lbl_pat,addr='98831F3A:E:3'):
  488. t = self.spawn('mmgen-tool', self.eth_args + ['listaddresses','all_labels=1'])
  489. t.expect(r'{}\b.*\S{{30}}\b.*{}\b'.format(addr,lbl_pat),regex=True)
  490. return t
  491. def add_label1(self): return self.add_label(lbl=tw_label_zh)
  492. def chk_label1(self): return self.chk_label(lbl_pat=tw_label_zh)
  493. def add_label2(self): return self.add_label(lbl=tw_label_lat_cyr_gr)
  494. def chk_label2(self): return self.chk_label(lbl_pat=tw_label_lat_cyr_gr)
  495. def remove_label(self,addr='98831F3A:E:3'):
  496. t = self.spawn('mmgen-tool', self.eth_args + ['remove_label',addr])
  497. t.expect('Removed label.*in tracking wallet',regex=True)
  498. return t
  499. def token_compile(self,token_data={}):
  500. odir = joinpath(self.tmpdir,token_data['symbol'].lower())
  501. if not solc_ver:
  502. imsg('Using precompiled contract data in {}'.format(odir))
  503. return 'skip' if os.path.exists(odir) else False
  504. self.spawn('',msg_only=True)
  505. cmd_args = ['--{}={}'.format(k,v) for k,v in list(token_data.items())]
  506. imsg("Compiling solidity token contract '{}' with 'solc'".format(token_data['symbol']))
  507. try: os.mkdir(odir)
  508. except: pass
  509. cmd = [
  510. 'scripts/traceback_run.py',
  511. 'scripts/create-token.py',
  512. '--coin=' + self.proto.coin,
  513. '--outdir=' + odir
  514. ] + cmd_args + [dfl_addr_chk]
  515. imsg("Executing: {}".format(' '.join(cmd)))
  516. cp = run(cmd,stdout=DEVNULL,stderr=PIPE)
  517. if cp.returncode != 0:
  518. rdie(2,'solc failed with the following output: {}'.format(cp.stderr))
  519. imsg("ERC20 token '{}' compiled".format(token_data['symbol']))
  520. return 'ok'
  521. def token_compile1(self):
  522. token_data = { 'name':'MMGen Token 1', 'symbol':'MM1', 'supply':10**26, 'decimals':18 }
  523. return self.token_compile(token_data)
  524. def token_compile2(self):
  525. token_data = { 'name':'MMGen Token 2', 'symbol':'MM2', 'supply':10**18, 'decimals':10 }
  526. return self.token_compile(token_data)
  527. async def get_exec_status(self,txid):
  528. from mmgen.tx import MMGenTX
  529. tx = MMGenTX.New(proto=self.proto)
  530. from mmgen.rpc import rpc_init
  531. tx.rpc = await rpc_init(self.proto)
  532. return await tx.get_exec_status(txid,True)
  533. async def token_deploy(self,num,key,gas,mmgen_cmd='txdo',tx_fee='8G'):
  534. keyfile = joinpath(self.tmpdir,openethereum_key_fn)
  535. fn = joinpath(self.tmpdir,'mm'+str(num),key+'.bin')
  536. os.environ['MMGEN_BOGUS_SEND'] = ''
  537. args = ['-B',
  538. '--tx-fee='+tx_fee,
  539. '--tx-gas={}'.format(gas),
  540. '--contract-data='+fn,
  541. '--inputs='+dfl_addr,
  542. '--yes' ]
  543. if mmgen_cmd == 'txdo': args += ['-k',keyfile]
  544. t = self.spawn( 'mmgen-'+mmgen_cmd, self.eth_args + args)
  545. if mmgen_cmd == 'txcreate':
  546. t.written_to_file('transaction')
  547. ext = '[0,8000]{}.regtest.rawtx'.format('-α' if g.debug_utf8 else '')
  548. txfile = self.get_file_with_ext(ext,no_dot=True)
  549. t = self.spawn('mmgen-txsign', self.eth_args + ['--yes','-k',keyfile,txfile],no_msg=True)
  550. self.txsign_ui_common(t,ni=True)
  551. txfile = txfile.replace('.rawtx','.sigtx')
  552. t = self.spawn('mmgen-txsend', self.eth_args + [txfile],no_msg=True)
  553. os.environ['MMGEN_BOGUS_SEND'] = '1'
  554. txid = self.txsend_ui_common(t,caller=mmgen_cmd,
  555. quiet = mmgen_cmd == 'txdo' or not g.debug,
  556. bogus_send=False)
  557. addr = strip_ansi_escapes(t.expect_getend('Contract address: '))
  558. assert (await self.get_exec_status(txid)) != 0, f'Contract {num}:{key} failed to execute. Aborting'
  559. if key == 'Token':
  560. self.write_to_tmpfile( f'token_addr{num}', addr+'\n' )
  561. imsg(f'\nToken MM{num} deployed!')
  562. return t
  563. async def token_deploy1a(self): return await self.token_deploy(num=1,key='SafeMath',gas=200000)
  564. async def token_deploy1b(self): return await self.token_deploy(num=1,key='Owned',gas=250000)
  565. async def token_deploy1c(self): return await self.token_deploy(num=1,key='Token',gas=1100000,tx_fee='7G')
  566. def tx_status2(self):
  567. return self.tx_status(ext=self.proto.coin+'[0,7000]{}.regtest.sigtx',expect_str='successfully executed')
  568. def bal6(self): return self.bal5()
  569. async def token_deploy2a(self): return await self.token_deploy(num=2,key='SafeMath',gas=200000)
  570. async def token_deploy2b(self): return await self.token_deploy(num=2,key='Owned',gas=250000)
  571. async def token_deploy2c(self): return await self.token_deploy(num=2,key='Token',gas=1100000)
  572. async def contract_deploy(self): # test create,sign,send
  573. return await self.token_deploy(num=2,key='SafeMath',gas=1100000,mmgen_cmd='txcreate')
  574. async def token_transfer_ops(self,op,amt=1000):
  575. self.spawn('',msg_only=True)
  576. sid = dfl_sid
  577. from mmgen.tool import MMGenToolCmdWallet
  578. usr_mmaddrs = ['{}:E:{}'.format(sid,i) for i in (11,21)]
  579. usr_addrs = [MMGenToolCmdWallet(proto=self.proto).gen_addr(addr,dfl_words_file) for addr in usr_mmaddrs]
  580. from mmgen.altcoins.eth.contract import TokenResolve
  581. from mmgen.altcoins.eth.tx import EthereumMMGenTX as etx
  582. async def do_transfer(rpc):
  583. for i in range(2):
  584. tk = await TokenResolve(
  585. self.proto,
  586. rpc,
  587. self.read_from_tmpfile(f'token_addr{i+1}').strip() )
  588. imsg_r( '\n' + await tk.info() )
  589. imsg('dev token balance (pre-send): {}'.format(await tk.get_balance(dfl_addr)))
  590. imsg('Sending {} {} to address {} ({})'.format(amt,self.proto.dcoin,usr_addrs[i],usr_mmaddrs[i]))
  591. from mmgen.obj import ETHAmt
  592. txid = await tk.transfer(
  593. dfl_addr,
  594. usr_addrs[i],
  595. amt,
  596. dfl_privkey,
  597. start_gas = ETHAmt(60000,'wei'),
  598. gasPrice = ETHAmt(8,'Gwei') )
  599. assert (await self.get_exec_status(txid)) != 0,'Transfer of token funds failed. Aborting'
  600. async def show_bals(rpc):
  601. for i in range(2):
  602. tk = await TokenResolve(
  603. self.proto,
  604. rpc,
  605. self.read_from_tmpfile(f'token_addr{i+1}').strip() )
  606. imsg('Token: {}'.format(await tk.get_symbol()))
  607. imsg('dev token balance: {}'.format(await tk.get_balance(dfl_addr)))
  608. imsg('usr token balance: {} ({} {})'.format(
  609. await tk.get_balance(usr_addrs[i]),usr_mmaddrs[i],usr_addrs[i]))
  610. from mmgen.rpc import rpc_init
  611. rpc = await rpc_init(self.proto)
  612. silence()
  613. if op == 'show_bals':
  614. await show_bals(rpc)
  615. elif op == 'do_transfer':
  616. await do_transfer(rpc)
  617. end_silence()
  618. return 'ok'
  619. def token_fund_users(self):
  620. return self.token_transfer_ops(op='do_transfer')
  621. def token_user_bals(self):
  622. return self.token_transfer_ops(op='show_bals')
  623. def token_addrgen(self):
  624. self.addrgen(addrs='11-13')
  625. ok_msg()
  626. return self.addrgen(addrs='21-23')
  627. def token_addrimport_badaddr1(self):
  628. t = self.addrimport(ext='[11-13]{}.regtest.addrs',add_args=['--token=abc'],bad_input=True)
  629. t.req_exit_val = 2
  630. return t
  631. def token_addrimport_badaddr2(self):
  632. t = self.addrimport(ext='[11-13]{}.regtest.addrs',add_args=['--token='+'00deadbeef'*4],bad_input=True)
  633. t.req_exit_val = 2
  634. return t
  635. def token_addrimport(self,addr_file,addr_range,expect,extra_args=[]):
  636. token_addr = self.read_from_tmpfile(addr_file).strip()
  637. return self.addrimport(
  638. ext = f'[{addr_range}]{{}}.regtest.addrs',
  639. expect = expect,
  640. add_args = ['--token-addr='+token_addr]+extra_args )
  641. def token_addrimport_addr1(self):
  642. return self.token_addrimport('token_addr1','11-13',expect='3/3')
  643. def token_addrimport_addr2(self):
  644. return self.token_addrimport('token_addr2','21-23',expect='3/3')
  645. def token_addrimport_batch(self):
  646. return self.token_addrimport('token_addr1','11-13',expect='OK: 3',extra_args=['--batch'])
  647. def token_addrimport_sym(self):
  648. return self.addrimport(
  649. ext = '[21-23]{}.regtest.addrs',
  650. expect = '3/3',
  651. add_args = ['--token=MM2'] )
  652. def bal7(self): return self.bal5()
  653. def token_bal1(self): return self.token_bal(n='1')
  654. def token_txcreate(self,args=[],token='',inputs='1',fee='50G'):
  655. t = self.spawn('mmgen-txcreate', self.eth_args + ['--token='+token,'-B','--tx-fee='+fee] + args)
  656. t = self.txcreate_ui_common(
  657. t,
  658. menu = [],
  659. inputs = inputs,
  660. input_sels_prompt = 'to spend from',
  661. add_comment = tx_label_lat_cyr_gr )
  662. t.read()
  663. return t
  664. def token_txsign(self,ext='',token=''):
  665. return self.txsign(ni=True,ext=ext,add_args=['--token='+token])
  666. def token_txsend(self,ext='',token=''):
  667. return self.txsend(ext=ext,add_args=['--token=mm1'])
  668. def token_txcreate1(self):
  669. return self.token_txcreate(args=['98831F3A:E:12,1.23456'],token='mm1')
  670. def token_txview1_raw(self):
  671. return self.txview(ext_fs='1.23456,50000]{}.regtest.rawtx')
  672. def token_txsign1(self):
  673. return self.token_txsign(ext='1.23456,50000]{}.regtest.rawtx',token='mm1')
  674. def token_txsend1(self):
  675. return self.token_txsend(ext='1.23456,50000]{}.regtest.sigtx',token='mm1')
  676. def token_txview1_sig(self):
  677. return self.txview(ext_fs='1.23456,50000]{}.regtest.sigtx')
  678. def tx_status3(self):
  679. return self.tx_status(
  680. ext='1.23456,50000]{}.regtest.sigtx',
  681. add_args=['--token=mm1'],
  682. expect_str='successfully executed',
  683. expect_str2='has 1 confirmation')
  684. def token_bal2(self):
  685. return self.token_bal(n='2')
  686. def twview(self,args=[],expect_str='',tool_args=[],exit_val=0):
  687. t = self.spawn('mmgen-tool', self.eth_args + args + ['twview'] + tool_args)
  688. if expect_str:
  689. t.expect(expect_str,regex=True)
  690. t.read()
  691. t.req_exit_val = exit_val
  692. return t
  693. def token_txcreate2(self):
  694. return self.token_txcreate(args=[burn_addr+','+amt2],token='mm1')
  695. def token_txbump(self):
  696. return self.txbump(ext=amt2+',50000]{}.regtest.rawtx',fee='56G',add_args=['--token=mm1'])
  697. def token_txsign2(self):
  698. return self.token_txsign(ext=amt2+',50000]{}.regtest.rawtx',token='mm1')
  699. def token_txsend2(self):
  700. return self.token_txsend(ext=amt2+',50000]{}.regtest.sigtx',token='mm1')
  701. def token_bal3(self):
  702. return self.token_bal(n='3')
  703. def del_dev_addr(self):
  704. t = self.spawn('mmgen-tool', self.eth_args + ['remove_address',dfl_addr])
  705. t.read() # TODO
  706. return t
  707. def bal1_getbalance(self):
  708. return self.bal_getbalance('1',etc_adj=True)
  709. def addrimport_token_burn_addr(self):
  710. return self.addrimport_one_addr(addr=burn_addr,extra_args=['--token=mm1'])
  711. def token_bal4(self):
  712. return self.token_bal(n='4')
  713. def token_bal_getbalance(self):
  714. return self.bal_getbalance('2',extra_args=['--token=mm1'])
  715. def txcreate_noamt(self):
  716. return self.txcreate(args=['98831F3A:E:12'],eth_fee_res=True)
  717. def txsign_noamt(self):
  718. return self.txsign(ext='99.99895,50000]{}.regtest.rawtx')
  719. def txsend_noamt(self):
  720. return self.txsend(ext='99.99895,50000]{}.regtest.sigtx')
  721. def bal8(self): return self.bal(n='8')
  722. def token_bal5(self): return self.token_bal(n='5')
  723. def token_txcreate_noamt(self):
  724. return self.token_txcreate(args=['98831F3A:E:13'],token='mm1',inputs='2',fee='51G')
  725. def token_txsign_noamt(self):
  726. return self.token_txsign(ext='1.23456,51000]{}.regtest.rawtx',token='mm1')
  727. def token_txsend_noamt(self):
  728. return self.token_txsend(ext='1.23456,51000]{}.regtest.sigtx',token='mm1')
  729. def bal9(self): return self.bal(n='9')
  730. def token_bal6(self): return self.token_bal(n='6')
  731. def listaddresses(self,args=[],tool_args=['all_labels=1'],exit_val=0):
  732. t = self.spawn('mmgen-tool', self.eth_args + args + ['listaddresses'] + tool_args)
  733. t.read()
  734. t.req_exit_val = exit_val
  735. return t
  736. def listaddresses1(self):
  737. return self.listaddresses()
  738. def listaddresses2(self):
  739. return self.listaddresses(tool_args=['minconf=999999999'])
  740. def listaddresses3(self):
  741. return self.listaddresses(tool_args=['sort=age'])
  742. def listaddresses4(self):
  743. return self.listaddresses(tool_args=['sort=age','showempty=1'])
  744. def token_listaddresses1(self):
  745. return self.listaddresses(args=['--token=mm1'])
  746. def token_listaddresses2(self):
  747. return self.listaddresses(args=['--token=mm1'],tool_args=['showempty=1'])
  748. def twview_cached_balances(self):
  749. return self.twview(args=['--cached-balances'])
  750. def token_twview_cached_balances(self):
  751. return self.twview(args=['--token=mm1','--cached-balances'])
  752. def txcreate_cached_balances(self):
  753. args = ['--tx-fee=20G','--cached-balances','98831F3A:E:3,0.1276']
  754. return self.txcreate(args=args,acct='2')
  755. def token_txcreate_cached_balances(self):
  756. args=['--cached-balances','--tx-fee=12G','98831F3A:E:12,1.2789']
  757. return self.token_txcreate(args=args,token='mm1')
  758. def txdo_cached_balances(self,
  759. acct = '2',
  760. fee_res_data = ('0.00105','50'),
  761. add_args = ['98831F3A:E:3,0.4321']):
  762. args = ['--tx-fee=20G','--cached-balances'] + add_args + [dfl_words_file]
  763. os.environ['MMGEN_BOGUS_SEND'] = ''
  764. t = self.txcreate(args=args,acct=acct,caller='txdo',fee_res_data=fee_res_data,no_read=True)
  765. os.environ['MMGEN_BOGUS_SEND'] = '1'
  766. self._do_confirm_send(t,quiet=not g.debug,sure=False)
  767. t.read()
  768. return t
  769. def txcreate_refresh_balances(self,
  770. bals=['2','3'],
  771. args=['-B','--cached-balances','-i'],
  772. total= '1000126.14829832312345678',
  773. adj_total=True,
  774. total_coin=None ):
  775. if total_coin is None:
  776. total_coin = self.proto.coin
  777. if self.proto.coin == 'ETC' and adj_total:
  778. total = str(Decimal(total) + self.bal_corr)
  779. t = self.spawn('mmgen-txcreate', self.eth_args + args)
  780. for n in bals:
  781. t.expect('[R]efresh balance:\b','R')
  782. t.expect(' main menu): ',n)
  783. t.expect('Is this what you want? (y/N): ','y')
  784. t.expect('[R]efresh balance:\b','q')
  785. t.expect('Total unspent:.*\D{}\D.*{}'.format(total,total_coin),regex=True)
  786. t.read()
  787. return t
  788. def bal10(self): return self.bal(n='10')
  789. def token_txdo_cached_balances(self):
  790. return self.txdo_cached_balances(
  791. acct='1',
  792. fee_res_data=('0.0026','50'),
  793. add_args=['--token=mm1','98831F3A:E:12,43.21'])
  794. def token_txcreate_refresh_balances(self):
  795. return self.txcreate_refresh_balances(
  796. bals=['1','2'],
  797. args=['--token=mm1','-B','--cached-balances','-i'],
  798. total='1000',adj_total=False,total_coin='MM1')
  799. def token_bal7(self): return self.token_bal(n='7')
  800. def twview1(self):
  801. return self.twview()
  802. def twview2(self):
  803. return self.twview(tool_args=['wide=1'])
  804. def twview3(self):
  805. return self.twview(tool_args=['wide=1','sort=age'])
  806. def twview4(self):
  807. return self.twview(tool_args=['wide=1','minconf=999999999'])
  808. def twview5(self):
  809. return self.twview(tool_args=['wide=1','minconf=0'])
  810. def token_twview1(self):
  811. return self.twview(args=['--token=mm1'])
  812. def token_twview2(self):
  813. return self.twview(args=['--token=mm1'],tool_args=['wide=1'])
  814. def token_twview3(self):
  815. return self.twview(args=['--token=mm1'],tool_args=['wide=1','sort=age'])
  816. def edit_label(self,out_num,args=[],action='l',label_text=None):
  817. t = self.spawn('mmgen-txcreate', self.eth_args + args + ['-B','-i'])
  818. p1,p2 = ('efresh balance:\b','return to main menu): ')
  819. p3,r3 = (p2,label_text+'\n') if label_text is not None else ('(y/N): ','y')
  820. p4,r4 = (('(y/N): ',),('y',)) if label_text == '' else ((),())
  821. for p,r in zip((p1,p1,p2,p3)+p4,('M',action,out_num+'\n',r3)+r4):
  822. t.expect(p,r)
  823. m = ( 'Account #{} removed' if action == 'D' else
  824. 'Label added to account #{}' if label_text else
  825. 'Label removed from account #{}' )
  826. t.expect(m.format(out_num))
  827. for p,r in zip((p1,p1),('M','q')):
  828. t.expect(p,r)
  829. t.expect('Total unspent:')
  830. t.read()
  831. return t
  832. def edit_label1(self):
  833. return self.edit_label(out_num=del_addrs[0],label_text=tw_label_zh)
  834. def edit_label2(self):
  835. return self.edit_label(out_num=del_addrs[1],label_text=tw_label_lat_cyr_gr)
  836. def edit_label3(self):
  837. return self.edit_label(out_num=del_addrs[0],label_text='')
  838. def token_edit_label1(self):
  839. return self.edit_label(out_num='1',label_text='Token label #1',args=['--token=mm1'])
  840. def remove_addr1(self):
  841. return self.edit_label(out_num=del_addrs[0],action='D')
  842. def remove_addr2(self):
  843. return self.edit_label(out_num=del_addrs[1],action='D')
  844. def token_remove_addr1(self):
  845. return self.edit_label(out_num=del_addrs[0],args=['--token=mm1'],action='D')
  846. def token_remove_addr2(self):
  847. return self.edit_label(out_num=del_addrs[1],args=['--token=mm1'],action='D')
  848. def stop(self):
  849. self.spawn('',msg_only=True)
  850. if not opt.no_daemon_stop:
  851. if not stop_test_daemons(self.proto.coin+'_rt'):
  852. return False
  853. return 'ok'