ct_ethdev.py 52 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499
  1. #!/usr/bin/env python3
  2. #
  3. # MMGen Wallet, a terminal-based cryptocurrency wallet
  4. # Copyright (C)2013-2024 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. test.cmdtest_d.ct_ethdev: Ethdev tests for the cmdtest.py test suite
  20. """
  21. import sys, os, re, shutil, asyncio, json
  22. from decimal import Decimal
  23. from collections import namedtuple
  24. from subprocess import run, PIPE, DEVNULL
  25. from pathlib import Path
  26. from mmgen.color import yellow, blue, cyan, set_vt100
  27. from mmgen.util import msg, rmsg, die
  28. from ..include.common import (
  29. cfg,
  30. check_solc_ver,
  31. omsg,
  32. imsg,
  33. imsg_r,
  34. joinpath,
  35. read_from_file,
  36. write_to_file,
  37. cmp_or_die,
  38. strip_ansi_escapes,
  39. silence,
  40. end_silence,
  41. gr_uc,
  42. stop_test_daemons
  43. )
  44. from .common import (
  45. ref_dir,
  46. dfl_words_file,
  47. tx_comment_jp,
  48. tx_comment_lat_cyr_gr,
  49. tw_comment_zh,
  50. tw_comment_lat_cyr_gr,
  51. get_file_with_ext,
  52. ok_msg,
  53. Ctrl_U
  54. )
  55. from .ct_base import CmdTestBase
  56. from .ct_shared import CmdTestShared
  57. del_addrs = ('4', '1')
  58. dfl_sid = '98831F3A'
  59. # The OpenEthereum dev address with lots of coins. Create with "ethkey -b info ''":
  60. dfl_devaddr = '00a329c0648769a73afac7f9381e08fb43dbea72'
  61. dfl_devkey = '4d5db4107d237df6a3d58ee5f70ae63d73d7658d4026f2eefd2f204c81682cb7'
  62. burn_addr = 'deadbeef'*5
  63. burn_addr2 = 'beadcafe'*5
  64. amt1 = '999999.12345689012345678'
  65. amt2 = '888.111122223333444455'
  66. parity_devkey_fn = 'parity.devkey'
  67. def set_vbals(daemon_id):
  68. global vbal1, vbal2, vbal3, vbal4, vbal5, vbal6, vbal7, vbal9
  69. if daemon_id == 'geth':
  70. vbal1 = '1.2288334'
  71. vbal2 = '99.996560752'
  72. vbal3 = '1.2314176'
  73. vbal4 = '127.0287834'
  74. vbal5 = '1000126.14775104212345678'
  75. vbal6 = '1000126.14880104212345678'
  76. vbal7 = '1000124.91891764212345678'
  77. vbal9 = '1.2262504'
  78. else:
  79. vbal1 = '1.2288396'
  80. vbal2 = '99.997088092'
  81. vbal3 = '1.23142525'
  82. vbal3 = '1.2314246'
  83. vbal4 = '127.0287896'
  84. vbal5 = '1000126.14828458212345678'
  85. vbal6 = '1000126.14933458212345678'
  86. vbal7 = '1000124.91944498212345678'
  87. vbal9 = '1.226261'
  88. bals = lambda k: {
  89. '1': [ ('98831F3A:E:1', '123.456')],
  90. '2': [ ('98831F3A:E:1', '123.456'), ('98831F3A:E:11', '1.234')],
  91. '3': [ ('98831F3A:E:1', '123.456'), ('98831F3A:E:11', '1.234'), ('98831F3A:E:21', '2.345')],
  92. '4': [ ('98831F3A:E:1', '100'),
  93. ('98831F3A:E:2', '23.45495'),
  94. ('98831F3A:E:11', '1.234'),
  95. ('98831F3A:E:21', '2.345')],
  96. '5': [ ('98831F3A:E:1', '100'),
  97. ('98831F3A:E:2', '23.45495'),
  98. ('98831F3A:E:11', '1.234'),
  99. ('98831F3A:E:21', '2.345'),
  100. (burn_addr + r'\s+non-MMGen', amt1)],
  101. '8': [ ('98831F3A:E:1', '0'),
  102. ('98831F3A:E:2', '23.45495'),
  103. ('98831F3A:E:11', vbal1),
  104. ('98831F3A:E:12', '99.99895'),
  105. ('98831F3A:E:21', '2.345'),
  106. (burn_addr + r'\s+non-MMGen', amt1)],
  107. '9': [ ('98831F3A:E:1', '0'),
  108. ('98831F3A:E:2', '23.45495'),
  109. ('98831F3A:E:11', vbal1),
  110. ('98831F3A:E:12', vbal2),
  111. ('98831F3A:E:21', '2.345'),
  112. (burn_addr + r'\s+non-MMGen', amt1)],
  113. '10': [ ('98831F3A:E:1', '0'),
  114. ('98831F3A:E:2', '23.0218'),
  115. ('98831F3A:E:3', '0.4321'),
  116. ('98831F3A:E:11', vbal1),
  117. ('98831F3A:E:12', vbal2),
  118. ('98831F3A:E:21', '2.345'),
  119. (burn_addr + r'\s+non-MMGen', amt1)]
  120. }[k]
  121. token_bals = lambda k: {
  122. '1': [ ('98831F3A:E:11', '1000', '1.234')],
  123. '2': [ ('98831F3A:E:11', '998.76544', vbal3),
  124. ('98831F3A:E:12', '1.23456', '0')],
  125. '3': [ ('98831F3A:E:11', '110.654317776666555545', vbal1),
  126. ('98831F3A:E:12', '1.23456', '0')],
  127. '4': [ ('98831F3A:E:11', '110.654317776666555545', vbal1),
  128. ('98831F3A:E:12', '1.23456', '0'),
  129. (burn_addr + r'\s+non-MMGen', amt2, amt1)],
  130. '5': [ ('98831F3A:E:11', '110.654317776666555545', vbal1),
  131. ('98831F3A:E:12', '1.23456', '99.99895'),
  132. (burn_addr + r'\s+non-MMGen', amt2, amt1)],
  133. '6': [ ('98831F3A:E:11', '110.654317776666555545', vbal1),
  134. ('98831F3A:E:12', '0', vbal2),
  135. ('98831F3A:E:13', '1.23456', '0'),
  136. (burn_addr + r'\s+non-MMGen', amt2, amt1)],
  137. '7': [ ('98831F3A:E:11', '67.444317776666555545', vbal9),
  138. ('98831F3A:E:12', '43.21', vbal2),
  139. ('98831F3A:E:13', '1.23456', '0'),
  140. (burn_addr + r'\s+non-MMGen', amt2, amt1)]
  141. }[k]
  142. token_bals_getbalance = lambda k: {
  143. '1': (vbal4, '999999.12345689012345678'),
  144. '2': ('111.888877776666555545', '888.111122223333444455')
  145. }[k]
  146. coin = cfg.coin
  147. class CmdTestEthdev(CmdTestBase, CmdTestShared):
  148. 'Ethereum transacting, token deployment and tracking wallet operations'
  149. networks = ('eth', 'etc')
  150. passthru_opts = ('coin', 'daemon_id', 'http_timeout', 'rpc_backend')
  151. tmpdir_nums = [22]
  152. color = True
  153. cmd_group_in = (
  154. ('setup', f'dev mode tests for coin {coin} (start daemon)'),
  155. ('subgroup.misc', []),
  156. ('subgroup.init', []),
  157. ('subgroup.msg', ['init']),
  158. ('subgroup.main', ['init']),
  159. ('subgroup.contract', ['main']),
  160. ('subgroup.token', ['contract']),
  161. ('subgroup.twexport', ['token']),
  162. ('subgroup.cached', ['token']),
  163. ('subgroup.view', ['cached']),
  164. ('subgroup.label', ['cached']),
  165. ('subgroup.remove', ['cached']),
  166. ('stop', 'stopping daemon'),
  167. )
  168. cmd_subgroups = {
  169. 'misc': (
  170. 'miscellaneous commands',
  171. ('daemon_version', 'mmgen-tool daemon_version'),
  172. ),
  173. 'init': (
  174. 'initializing wallets',
  175. ('wallet_upgrade1', 'upgrading the tracking wallet (v1 -> v2)'),
  176. ('wallet_upgrade2', 'upgrading the tracking wallet (v2 -> v3)'),
  177. ('addrgen', 'generating addresses'),
  178. ('addrimport', 'importing addresses'),
  179. ('addrimport_dev_addr', "importing dev faucet address 'Ox00a329c..'"),
  180. ('fund_dev_address', 'funding the default (Parity dev) address'),
  181. ),
  182. 'msg': (
  183. 'message signing',
  184. ('msgsign_chk', "signing a message (low-level, check against 'eth_sign' RPC call)"),
  185. ('msgcreate', 'creating a message file'),
  186. ('msgsign', 'signing the message file'),
  187. ('msgverify', 'verifying the message file'),
  188. ('msgexport', 'exporting the message file data to JSON for third-party verifier'),
  189. ('msgverify_export', 'verifying the exported JSON data'),
  190. ('msgcreate_raw', 'creating a message file (--msghash-type=raw)'),
  191. ('msgsign_raw', 'signing the message file (msghash_type=raw)'),
  192. ('msgverify_raw', 'verifying the message file (msghash_type=raw)'),
  193. ('msgexport_raw', 'exporting the message file data to JSON (msghash_type=raw)'),
  194. ('msgverify_export_raw', 'verifying the exported JSON data (msghash_type=raw)'),
  195. ),
  196. 'main': (
  197. 'creating, signing, sending and bumping Ethereum transactions',
  198. ('txcreate1', 'creating a transaction (spend from dev address to address :1)'),
  199. ('txview1_raw', 'viewing the raw transaction'),
  200. ('txsign1', 'signing the transaction'),
  201. ('txview1_sig', 'viewing the signed transaction'),
  202. ('tx_status0_bad', 'getting the transaction status'),
  203. ('txsign1_ni', 'signing the transaction (non-interactive)'),
  204. ('txsend1', 'sending the transaction'),
  205. ('bal1', f'the {coin} balance'),
  206. ('txcreate2', 'creating a transaction (spend from dev address to address :11)'),
  207. ('txsign2', 'signing the transaction'),
  208. ('txsend2', 'sending the transaction'),
  209. ('bal2', f'the {coin} balance'),
  210. ('txcreate3', 'creating a transaction (spend from dev address to address :21)'),
  211. ('txsign3', 'signing the transaction'),
  212. ('txsend3', 'sending the transaction'),
  213. ('bal3', f'the {coin} balance'),
  214. ('tx_status1', 'getting the transaction status'),
  215. ('txcreate4', 'creating a transaction (spend from MMGen address, low TX fee)'),
  216. ('txbump', 'bumping the transaction fee'),
  217. ('txsign4', 'signing the transaction'),
  218. ('txsend4', 'sending the transaction'),
  219. ('tx_status1a', 'getting the transaction status'),
  220. ('bal4', f'the {coin} balance'),
  221. ('txcreate5', 'creating a transaction (fund burn address)'),
  222. ('txsign5', 'signing the transaction'),
  223. ('txsend5', 'sending the transaction'),
  224. ('addrimport_burn_addr', 'importing burn address'),
  225. ('bal5', f'the {coin} balance'),
  226. ('add_comment1', 'adding a UTF-8 label (zh)'),
  227. ('chk_comment1', 'checking the label'),
  228. ('add_comment2', 'adding a UTF-8 label (lat+cyr+gr)'),
  229. ('chk_comment2', 'checking the label'),
  230. ('remove_comment', 'removing the label'),
  231. ),
  232. 'contract': (
  233. 'creating and deploying ERC20 tokens',
  234. ('token_compile1', 'compiling ERC20 token #1'),
  235. ('token_deploy1a', 'deploying ERC20 token #1 (SafeMath)'),
  236. ('token_deploy1b', 'deploying ERC20 token #1 (Owned)'),
  237. ('token_deploy1c', 'deploying ERC20 token #1 (Token)'),
  238. ('tx_status2', 'getting the transaction status'),
  239. ('bal6', f'the {coin} balance'),
  240. ('token_compile2', 'compiling ERC20 token #2'),
  241. ('token_deploy2a', 'deploying ERC20 token #2 (SafeMath)'),
  242. ('token_deploy2b', 'deploying ERC20 token #2 (Owned)'),
  243. ('token_deploy2c', 'deploying ERC20 token #2 (Token)'),
  244. ('contract_deploy', 'deploying contract (create, sign, send)'),
  245. ),
  246. 'token': (
  247. 'creating, signing, sending and bumping ERC20 token transactions',
  248. ('token_fund_users', 'transferring token funds from dev to user'),
  249. ('token_user_bals', 'show balances after transfer'),
  250. ('token_addrgen', 'generating token addresses'),
  251. ('token_addrimport_badaddr1', 'importing token addresses (no token address)'),
  252. ('token_addrimport_badaddr2', 'importing token addresses (bad token address)'),
  253. ('token_addrimport_addr1', 'importing token addresses using token address (MM1)'),
  254. ('token_addrimport_addr2', 'importing token addresses using token address (MM2)'),
  255. ('token_addrimport_batch', 'importing token addresses (dummy batch mode) (MM1)'),
  256. ('token_addrimport_sym', 'importing token addresses using token symbol (MM2)'),
  257. ('bal7', f'the {coin} balance'),
  258. ('token_bal1', f'the {coin} balance and token balance'),
  259. ('token_txcreate1', 'creating a token transaction'),
  260. ('token_txview1_raw', 'viewing the raw transaction'),
  261. ('token_txsign1', 'signing the transaction'),
  262. ('token_txsend1', 'sending the transaction'),
  263. ('token_txview1_sig', 'viewing the signed transaction'),
  264. ('tx_status3', 'getting the transaction status'),
  265. ('token_bal2', f'the {coin} balance and token balance'),
  266. ('token_txcreate2', 'creating a token transaction (to burn address)'),
  267. ('token_txbump', 'bumping the transaction fee'),
  268. ('token_txsign2', 'signing the transaction'),
  269. ('token_txsend2', 'sending the transaction'),
  270. ('token_bal3', f'the {coin} balance and token balance'),
  271. ('del_dev_addr', 'deleting the dev address'),
  272. ('bal1_getbalance', f'the {coin} balance (getbalance)'),
  273. ('addrimport_token_burn_addr', 'importing the token burn address'),
  274. ('token_bal4', f'the {coin} balance and token balance'),
  275. ('token_bal_getbalance', 'the token balance (getbalance)'),
  276. ('txcreate_noamt', 'creating a transaction (full amount send)'),
  277. ('txsign_noamt', 'signing the transaction'),
  278. ('txsend_noamt', 'sending the transaction'),
  279. ('bal8', f'the {coin} balance'),
  280. ('token_bal5', 'the token balance'),
  281. ('token_txcreate_noamt', 'creating a token transaction (full amount send)'),
  282. ('token_txsign_noamt', 'signing the transaction'),
  283. ('token_txsend_noamt', 'sending the transaction'),
  284. ('bal9', f'the {coin} balance'),
  285. ('token_bal6', 'the token balance'),
  286. ('listaddresses1', 'listaddresses'),
  287. ('listaddresses2', 'listaddresses minconf=999999999 (ignored)'),
  288. ('listaddresses3', 'listaddresses sort=age (ignored)'),
  289. ('listaddresses4', 'listaddresses showempty=1 sort=age (ignored)'),
  290. ('token_listaddresses1', 'listaddresses --token=mm1'),
  291. ('token_listaddresses2', 'listaddresses --token=mm1 showempty=1'),
  292. ),
  293. 'twexport': (
  294. 'exporting and importing tracking wallet to JSON',
  295. ('twexport_noamt', 'exporting the tracking wallet (include_amts=0)'),
  296. ('twmove', 'moving the tracking wallet'),
  297. ('twimport', 'importing the tracking wallet'),
  298. ('twview7', 'twview (cached_balances=1)'),
  299. ('twview8', 'twview'),
  300. ('twexport', 'exporting the tracking wallet'),
  301. ('tw_chktotal', 'checking total value in tracking wallet dump'),
  302. ('twmove', 'moving the tracking wallet'),
  303. ('twimport', 'importing the tracking wallet'),
  304. ('twcompare', 'comparing imported tracking wallet with original'),
  305. ('edit_json_twdump', 'editing the tracking wallet JSON dump'),
  306. ('twmove', 'moving the tracking wallet'),
  307. ('twimport_nochksum', 'importing the edited tracking wallet JSON dump (ignore_checksum=1)'),
  308. ('token_listaddresses3', 'listaddresses --token=mm1 showempty=1'),
  309. ('token_listaddresses4', 'listaddresses --token=mm2 showempty=1'),
  310. ('twview9', 'twview (check balance)'),
  311. ),
  312. 'cached': (
  313. 'creating and sending transactions using cached balances',
  314. ('twview_cached_balances', 'twview (cached balances)'),
  315. ('token_twview_cached_balances', 'token twview (cached balances)'),
  316. ('txcreate_cached_balances', 'txcreate (cached balances)'),
  317. ('token_txcreate_cached_balances', 'token txcreate (cached balances)'),
  318. ('txdo_cached_balances', 'txdo (cached balances)'),
  319. ('txcreate_refresh_balances', 'refreshing balances'),
  320. ('bal10', f'the {coin} balance'),
  321. ('token_txdo_cached_balances', 'token txdo (cached balances)'),
  322. ('token_txcreate_refresh_balances', 'refreshing token balances'),
  323. ('token_bal7', 'the token balance'),
  324. ),
  325. 'view': (
  326. 'viewing addresses and unspent outputs',
  327. ('twview1', 'twview'),
  328. ('twview2', 'twview wide=1'),
  329. ('twview3', 'twview wide=1 sort=age (ignored)'),
  330. ('twview4', 'twview wide=1 minconf=999999999 (ignored)'),
  331. ('twview5', 'twview wide=1 minconf=0 (ignored)'),
  332. ('token_twview1', 'twview --token=mm1'),
  333. ('token_twview2', 'twview --token=mm1 wide=1'),
  334. ('token_twview3', 'twview --token=mm1 wide=1 sort=age (ignored)'),
  335. ),
  336. 'label': (
  337. 'creating, editing and removing labels',
  338. ('edit_comment1', f'adding label to addr #{del_addrs[0]} in {coin} tracking wallet (zh)'),
  339. ('edit_comment2', f'editing label for addr #{del_addrs[0]} in {coin} tracking wallet (zh)'),
  340. ('edit_comment3', f'adding label to addr #{del_addrs[1]} in {coin} tracking wallet (lat+cyr+gr)'),
  341. ('edit_comment4', f'removing label from addr #{del_addrs[0]} in {coin} tracking wallet'),
  342. ('token_edit_comment1', f'adding label to addr #{del_addrs[0]} in {coin} token tracking wallet'),
  343. ),
  344. 'remove': (
  345. 'removing addresses from tracking wallet',
  346. ('remove_addr1', f'removing addr #{del_addrs[0]} from {coin} tracking wallet'),
  347. ('twview6', 'twview (balance reduced after address removal)'),
  348. ('remove_addr2', f'removing addr #{del_addrs[1]} from {coin} tracking wallet'),
  349. ('token_remove_addr1', f'removing addr #{del_addrs[0]} from {coin} token tracking wallet'),
  350. ('token_remove_addr2', f'removing addr #{del_addrs[1]} from {coin} token tracking wallet'),
  351. ),
  352. }
  353. def __init__(self, trunner, cfgs, spawn):
  354. CmdTestBase.__init__(self, trunner, cfgs, spawn)
  355. if trunner is None:
  356. return
  357. self.eth_args = [f'--outdir={self.tmpdir}', '--regtest=1', '--quiet']
  358. self.eth_args_noquiet = [f'--outdir={self.tmpdir}', '--regtest=1']
  359. from mmgen.protocol import init_proto
  360. self.proto = init_proto( cfg, cfg.coin, network='regtest', need_amt=True)
  361. from mmgen.daemon import CoinDaemon
  362. self.daemon = CoinDaemon( cfg, self.proto.coin+'_rt', test_suite=True)
  363. set_vbals(self.daemon.id)
  364. self.using_solc = check_solc_ver()
  365. if not self.using_solc:
  366. omsg(yellow('Using precompiled contract data'))
  367. omsg(blue(f'Coin daemon {self.daemon.id!r} selected'))
  368. self.genesis_fn = joinpath(self.tmpdir, 'genesis.json')
  369. self.keystore_dir = os.path.relpath(joinpath(self.daemon.datadir, 'keystore'))
  370. write_to_file(
  371. joinpath(self.tmpdir, parity_devkey_fn),
  372. dfl_devkey+'\n')
  373. self.message = 'attack at dawn'
  374. self.spawn_env['MMGEN_BOGUS_SEND'] = ''
  375. @property
  376. async def rpc(self):
  377. from mmgen.rpc import rpc_init
  378. return await rpc_init(cfg, self.proto)
  379. async def setup(self):
  380. self.spawn('', msg_only=True)
  381. if not self.using_solc:
  382. srcdir = os.path.join(self.tr.repo_root, 'test', 'ref', 'ethereum', 'bin')
  383. from shutil import copytree
  384. for d in ('mm1', 'mm2'):
  385. copytree(os.path.join(srcdir, d), os.path.join(self.tmpdir, d))
  386. d = self.daemon
  387. if d.id in ('geth', 'erigon'):
  388. self.genesis_setup(d)
  389. set_vt100()
  390. if d.id == 'erigon':
  391. self.write_to_tmpfile('signer_key', self.keystore_data['key']+'\n')
  392. d.usr_coind_args = [
  393. '--miner.sigfile={}'.format(os.path.join(self.tmpdir, 'signer_key')),
  394. '--miner.etherbase={}'.format(self.keystore_data['address'])]
  395. if d.id in ('geth', 'erigon'):
  396. imsg(' {:19} {}'.format('Cmdline:', ' '.join(e for e in d.start_cmd if not 'verbosity' in e)))
  397. if not cfg.no_daemon_autostart:
  398. if not d.id in ('geth', 'erigon'):
  399. d.stop(silent=True)
  400. d.remove_datadir()
  401. d.start( silent = not (cfg.verbose or cfg.exact_output))
  402. rpc = await self.rpc
  403. imsg(f'Daemon: {rpc.daemon.coind_name} v{rpc.daemon_version_str}')
  404. return 'ok'
  405. @property
  406. def keystore_data(self):
  407. if not hasattr(self, '_keystore_data'):
  408. wallet_fn = os.path.join( self.keystore_dir, os.listdir(self.keystore_dir)[0])
  409. from mmgen.proto.eth.misc import decrypt_geth_keystore
  410. key = decrypt_geth_keystore(
  411. cfg = cfg,
  412. wallet_fn = wallet_fn,
  413. passwd = b'')
  414. with open(wallet_fn) as fh:
  415. res = json.loads(fh.read())
  416. res.update( { 'key': key.hex()})
  417. self._keystore_data = res
  418. return self._keystore_data
  419. def genesis_setup(self, d):
  420. def make_key():
  421. pwfile = joinpath(self.tmpdir, 'account_passwd')
  422. write_to_file(pwfile, '')
  423. run(['rm', '-rf', self.keystore_dir])
  424. cmd = f'geth account new --password={pwfile} --lightkdf --keystore {self.keystore_dir}'
  425. cp = run(cmd.split(), stdout=PIPE, stderr=PIPE)
  426. if cp.returncode:
  427. die(1, cp.stderr.decode())
  428. def make_genesis(signer_addr, prealloc_addr):
  429. return {
  430. 'config': {
  431. 'chainId': 1337, # TODO: replace constant with var
  432. 'homesteadBlock': 0,
  433. 'eip150Block': 0,
  434. 'eip155Block': 0,
  435. 'eip158Block': 0,
  436. 'byzantiumBlock': 0,
  437. 'constantinopleBlock': 0,
  438. 'petersburgBlock': 0,
  439. 'istanbulBlock': 0,
  440. 'muirGlacierBlock': 0,
  441. 'berlinBlock': 0,
  442. 'londonBlock': 0,
  443. 'arrowGlacierBlock': 0,
  444. 'grayGlacierBlock': 0,
  445. 'shanghaiTime': 0,
  446. 'terminalTotalDifficulty': 0,
  447. 'terminalTotalDifficultyPassed': True,
  448. 'isDev': True
  449. },
  450. 'nonce': '0x0',
  451. 'timestamp': '0x0',
  452. 'extraData': '0x',
  453. 'gasLimit': '0xaf79e0',
  454. 'difficulty': '0x1',
  455. 'mixHash': '0x0000000000000000000000000000000000000000000000000000000000000000',
  456. 'coinbase': '0x0000000000000000000000000000000000000000',
  457. 'number': '0x0',
  458. 'gasUsed': '0x0',
  459. 'parentHash': '0x0000000000000000000000000000000000000000000000000000000000000000',
  460. 'baseFeePerGas': '0x3b9aca00',
  461. 'excessBlobGas': None,
  462. 'blobGasUsed': None,
  463. 'alloc': {
  464. prealloc_addr: { 'balance': hex(prealloc_amt.toWei())}
  465. }
  466. }
  467. def init_genesis(fn):
  468. cmd = f'{d.exec_fn} init --datadir {d.datadir} {fn}'
  469. cp = run( cmd.split(), stdout=PIPE, stderr=PIPE)
  470. if cp.returncode:
  471. die(1, cp.stderr.decode())
  472. d.stop(quiet=True)
  473. d.remove_datadir()
  474. imsg(cyan('Initializing Genesis Block:'))
  475. prealloc_amt = self.proto.coin_amt('1_000_000_000')
  476. make_key()
  477. signer_addr = self.keystore_data['address']
  478. self.write_to_tmpfile( 'signer_addr', signer_addr + '\n')
  479. imsg(f' Keystore: {self.keystore_dir}')
  480. imsg(f' Signer key: {self.keystore_data["key"]}')
  481. imsg(f' Signer address: {signer_addr}')
  482. imsg(f' Faucet: {dfl_devaddr} ({prealloc_amt} ETH)')
  483. imsg(f' Genesis block data: {self.genesis_fn}')
  484. genesis_data = make_genesis(signer_addr, dfl_devaddr)
  485. write_to_file( self.genesis_fn, json.dumps(genesis_data, indent=' ')+'\n')
  486. init_genesis(self.genesis_fn)
  487. def daemon_version(self):
  488. t = self.spawn('mmgen-tool', self.eth_args + ['daemon_version'])
  489. t.expect('version')
  490. return t
  491. async def _wallet_upgrade(self, src_fn, expect1, expect2=None):
  492. if self.proto.coin == 'ETC':
  493. msg(f'skipping test {self.test_name!r} for ETC')
  494. return 'skip'
  495. from mmgen.tw.ctl import TwCtl
  496. twctl = await TwCtl(cfg, self.proto, no_wallet_init=True)
  497. from_fn = Path(ref_dir) / 'ethereum' / src_fn
  498. bak_fn = twctl.tw_dir / f'upgraded-{src_fn}'
  499. twctl.tw_dir.mkdir(mode=0o750, parents=True, exist_ok=True)
  500. dest = shutil.copy2(from_fn, twctl.tw_path)
  501. assert dest == twctl.tw_path, f'{dest} != {twctl.tw_path}'
  502. t = self.spawn('mmgen-tool', self.eth_args + ['twview'])
  503. t.expect(expect1)
  504. if expect2:
  505. t.expect(expect2)
  506. t.read()
  507. twctl.tw_path.rename(bak_fn)
  508. return t
  509. async def wallet_upgrade1(self):
  510. return await self._wallet_upgrade('tracking-wallet-v1.json', 'accounts field', 'network field')
  511. async def wallet_upgrade2(self):
  512. return await self._wallet_upgrade('tracking-wallet-v2.json', 'token params field', 'network field')
  513. def addrgen(self, addrs='1-3,11-13,21-23'):
  514. t = self.spawn('mmgen-addrgen', self.eth_args + [dfl_words_file, addrs])
  515. t.written_to_file('Addresses')
  516. return t
  517. def addrimport(
  518. self,
  519. ext = '21-23]{}.regtest.addrs',
  520. expect = '9/9',
  521. add_args = [],
  522. bad_input = False,
  523. exit_val = None):
  524. ext = ext.format('-α' if cfg.debug_utf8 else '')
  525. fn = self.get_file_with_ext(ext, no_dot=True, delete=False)
  526. t = self.spawn('mmgen-addrimport', ['--regtest=1'] + add_args + [fn], exit_val=exit_val)
  527. if bad_input:
  528. return t
  529. t.expect('Importing')
  530. t.expect(expect)
  531. return t
  532. def addrimport_one_addr(self, addr=None, extra_args=[]):
  533. t = self.spawn('mmgen-addrimport', ['--regtest=1', '--quiet', f'--address={addr}'] + extra_args)
  534. t.expect('OK')
  535. return t
  536. def addrimport_dev_addr(self):
  537. return self.addrimport_one_addr(addr=dfl_devaddr)
  538. def addrimport_burn_addr(self):
  539. return self.addrimport_one_addr(addr=burn_addr)
  540. def txcreate(
  541. self,
  542. args = [],
  543. menu = [],
  544. acct = '1',
  545. caller = 'txcreate',
  546. interactive_fee = '50G',
  547. fee_info_data = ('0.00105', '50'),
  548. no_read = False,
  549. print_listing = True,
  550. tweaks = []):
  551. fee_info_pat = r'\D{}\D.*{c} .*\D{}\D.*gas price in Gwei'.format(*fee_info_data, c=self.proto.coin)
  552. t = self.spawn(f'mmgen-{caller}', self.eth_args + ['-B'] + args)
  553. if print_listing:
  554. t.expect(r'add \[l\]abel, .*?:.', 'p', regex=True)
  555. t.written_to_file('Account balances listing')
  556. t = self.txcreate_ui_common(
  557. t,
  558. menu = menu,
  559. caller = caller,
  560. input_sels_prompt = 'to spend from',
  561. inputs = acct,
  562. file_desc = 'transaction',
  563. bad_input_sels = True,
  564. interactive_fee = interactive_fee,
  565. fee_info_pat = fee_info_pat,
  566. fee_desc = 'transaction fee or gas price',
  567. add_comment = tx_comment_jp,
  568. tweaks = tweaks)
  569. if not no_read:
  570. t.read()
  571. return t
  572. def txsign(self, ni=False, ext='{}.regtest.rawtx', add_args=[], dev_send=False):
  573. ext = ext.format('-α' if cfg.debug_utf8 else '')
  574. keyfile = joinpath(self.tmpdir, parity_devkey_fn)
  575. txfile = self.get_file_with_ext(ext, no_dot=True)
  576. t = self.spawn(
  577. 'mmgen-txsign',
  578. self.eth_args
  579. + [f'--coin={self.proto.coin}']
  580. + ['--rpc-host=bad_host'] # ETH signing must work without RPC
  581. + add_args
  582. + ([], ['--yes'])[ni]
  583. + ([f'--keys-from-file={keyfile}'] if dev_send else [])
  584. + [txfile, dfl_words_file])
  585. return self.txsign_ui_common(t, ni=ni, has_label=True)
  586. def txsend(self, ext='{}.regtest.sigtx', add_args=[]):
  587. ext = ext.format('-α' if cfg.debug_utf8 else '')
  588. txfile = self.get_file_with_ext(ext, no_dot=True)
  589. t = self.spawn('mmgen-txsend', self.eth_args + add_args + [txfile])
  590. self.txsend_ui_common(
  591. t,
  592. quiet = not cfg.debug,
  593. bogus_send = False,
  594. has_label = True)
  595. return t
  596. def txview(self, ext_fs):
  597. ext = ext_fs.format('-α' if cfg.debug_utf8 else '')
  598. txfile = self.get_file_with_ext(ext, no_dot=True)
  599. return self.spawn('mmgen-tool', ['--verbose', 'txview', txfile])
  600. def fund_dev_address(self):
  601. """
  602. For Erigon, fund the default (Parity) dev address from the Erigon dev address
  603. For the others, send a junk TX to keep block counts equal for all daemons
  604. """
  605. dt = namedtuple('data', ['devkey_fn', 'dest', 'amt'])
  606. d = dt(parity_devkey_fn, burn_addr2, '1')
  607. t = self.txcreate(
  608. args = self.eth_args_noquiet + [
  609. f'--keys-from-file={joinpath(self.tmpdir, d.devkey_fn)}',
  610. f'{d.dest},{d.amt}',
  611. ],
  612. menu = ['a', 'r'],
  613. caller = 'txdo',
  614. acct = '1',
  615. no_read = True)
  616. self._do_confirm_send(t, quiet=not cfg.debug, sure=False)
  617. t.read()
  618. self.get_file_with_ext('sigtx', delete_all=True)
  619. return t
  620. def txcreate1(self):
  621. # include one invalid keypress 'X' -- see EthereumTwUnspentOutputs.key_mappings
  622. menu = ['a', 'd', 'r', 'M', 'X', 'e', 'm', 'm']
  623. args = ['98831F3A:E:1,123.456']
  624. return self.txcreate(args=args, menu=menu, acct='1', tweaks=['confirm_non_mmgen'])
  625. def txview1_raw(self):
  626. return self.txview(ext_fs='{}.regtest.rawtx')
  627. def txsign1(self):
  628. return self.txsign(add_args=['--use-internal-keccak-module'], dev_send=True)
  629. def tx_status0_bad(self):
  630. return self.tx_status(ext='{}.regtest.sigtx', expect_str='neither in mempool nor blockchain', exit_val=1)
  631. def txsign1_ni(self):
  632. return self.txsign(ni=True, dev_send=True)
  633. def txsend1(self):
  634. return self.txsend()
  635. def txview1_sig(self): # do after send so that TxID is displayed
  636. return self.txview(ext_fs='{}.regtest.sigtx')
  637. def bal1(self):
  638. return self.bal(n='1')
  639. def txcreate2(self):
  640. args = ['98831F3A:E:11,1.234']
  641. return self.txcreate(args=args, acct='10', tweaks=['confirm_non_mmgen'])
  642. def txsign2(self):
  643. return self.txsign(ni=True, ext='1.234,50000]{}.regtest.rawtx', dev_send=True)
  644. def txsend2(self):
  645. return self.txsend(ext='1.234,50000]{}.regtest.sigtx')
  646. def bal2(self):
  647. return self.bal(n='2')
  648. def txcreate3(self):
  649. args = ['98831F3A:E:21,2.345']
  650. return self.txcreate(args=args, acct='10', tweaks=['confirm_non_mmgen'])
  651. def txsign3(self):
  652. return self.txsign(ni=True, ext='2.345,50000]{}.regtest.rawtx', dev_send=True)
  653. def txsend3(self):
  654. return self.txsend(ext='2.345,50000]{}.regtest.sigtx')
  655. def bal3(self):
  656. return self.bal(n='3')
  657. def tx_status(self, ext, expect_str, expect_str2='', add_args=[], exit_val=0):
  658. ext = ext.format('-α' if cfg.debug_utf8 else '')
  659. txfile = self.get_file_with_ext(ext, no_dot=True)
  660. t = self.spawn(
  661. 'mmgen-txsend',
  662. self.eth_args + add_args + ['--status', txfile],
  663. exit_val = exit_val)
  664. t.expect(expect_str)
  665. if expect_str2:
  666. t.expect(expect_str2)
  667. return t
  668. def tx_status1(self):
  669. return self.tx_status(ext='2.345,50000]{}.regtest.sigtx', expect_str='has 1 confirmation')
  670. def tx_status1a(self):
  671. return self.tx_status(ext='2.345,50000]{}.regtest.sigtx', expect_str='has 2 confirmations')
  672. async def msgsign_chk(self): # NB: Geth only!
  673. def create_signature_mmgen():
  674. key = self.keystore_data['key']
  675. imsg(f'Key: {key}')
  676. from mmgen.proto.eth.misc import ec_sign_message_with_privkey
  677. return ec_sign_message_with_privkey(cfg, self.message, bytes.fromhex(key), 'eth_sign')
  678. async def create_signature_rpc():
  679. addr = self.read_from_tmpfile('signer_addr').strip()
  680. imsg(f'Address: {addr}')
  681. rpc = await self.rpc
  682. return await rpc.call(
  683. 'eth_sign',
  684. '0x' + addr,
  685. '0x' + self.message.encode().hex())
  686. if not self.daemon.id == 'geth':
  687. return 'skip'
  688. self.spawn('', msg_only=True)
  689. sig = '0x' + create_signature_mmgen()
  690. sig_chk = await create_signature_rpc()
  691. # Compare signatures
  692. imsg(f'Message: {self.message}')
  693. imsg(f'Signature: {sig}')
  694. cmp_or_die(sig, sig_chk, 'message signatures')
  695. imsg('Geth and MMGen signatures match')
  696. return 'ok'
  697. def msgcreate(self, add_args=[]):
  698. t = self.spawn('mmgen-msg', self.eth_args_noquiet + add_args + ['create', self.message, '98831F3A:E:1'])
  699. t.written_to_file('Unsigned message data')
  700. return t
  701. def msgsign(self):
  702. fn = get_file_with_ext(self.tmpdir, 'rawmsg.json')
  703. t = self.spawn('mmgen-msg', self.eth_args_noquiet + ['sign', fn, dfl_words_file])
  704. t.written_to_file('Signed message data')
  705. return t
  706. def msgverify(self, fn=None, msghash_type='eth_sign'):
  707. fn = fn or get_file_with_ext(self.tmpdir, 'sigmsg.json')
  708. t = self.spawn('mmgen-msg', self.eth_args_noquiet + ['verify', fn])
  709. t.expect(msghash_type)
  710. t.expect('1 signature verified')
  711. return t
  712. def msgexport(self):
  713. fn = get_file_with_ext(self.tmpdir, 'sigmsg.json')
  714. t = self.spawn('mmgen-msg', self.eth_args_noquiet + ['export', fn])
  715. t.written_to_file('Signature data')
  716. return t
  717. def msgverify_export(self):
  718. return self.msgverify(
  719. fn = os.path.join(self.tmpdir, 'signatures.json'))
  720. def msgcreate_raw(self):
  721. get_file_with_ext(self.tmpdir, 'rawmsg.json', delete_all=True)
  722. return self.msgcreate(add_args=['--msghash-type=raw'])
  723. def msgsign_raw(self):
  724. get_file_with_ext(self.tmpdir, 'sigmsg.json', delete_all=True)
  725. return self.msgsign()
  726. def msgverify_raw(self):
  727. return self.msgverify(msghash_type='raw')
  728. def msgexport_raw(self):
  729. get_file_with_ext(self.tmpdir, 'signatures.json', no_dot=True, delete_all=True)
  730. return self.msgexport()
  731. def msgverify_export_raw(self):
  732. return self.msgverify(
  733. fn = os.path.join(self.tmpdir, 'signatures.json'),
  734. msghash_type = 'raw')
  735. def txcreate4(self):
  736. return self.txcreate(
  737. args = ['98831F3A:E:2,23.45495'],
  738. acct = '1',
  739. interactive_fee = '40G',
  740. fee_info_data = ('0.00084', '40'))
  741. def txbump(self, ext=',40000]{}.regtest.rawtx', fee='50G', add_args=[]):
  742. ext = ext.format('-α' if cfg.debug_utf8 else '')
  743. txfile = self.get_file_with_ext(ext, no_dot=True)
  744. t = self.spawn('mmgen-txbump', self.eth_args + add_args + ['--yes', txfile])
  745. t.expect('or gas price: ', fee+'\n')
  746. return t
  747. def txsign4(self):
  748. return self.txsign(ni=True, ext='.45495,50000]{}.regtest.rawtx', dev_send=True)
  749. def txsend4(self):
  750. return self.txsend(ext='.45495,50000]{}.regtest.sigtx')
  751. def bal4(self):
  752. return self.bal(n='4')
  753. def txcreate5(self):
  754. args = [burn_addr + ','+amt1]
  755. return self.txcreate(args=args, acct='10', tweaks=['confirm_non_mmgen'])
  756. def txsign5(self):
  757. return self.txsign(ni=True, ext=amt1+',50000]{}.regtest.rawtx', dev_send=True)
  758. def txsend5(self):
  759. return self.txsend(ext=amt1+',50000]{}.regtest.sigtx')
  760. def bal5(self):
  761. return self.bal(n='5')
  762. def bal(self, n):
  763. t = self.spawn('mmgen-tool', self.eth_args + ['twview', 'wide=1'])
  764. text = t.read(strip_color=True)
  765. for addr, amt in bals(n):
  766. pat = r'\D{}\D.*\D{}\D'.format(addr, amt.replace('.', r'\.'))
  767. assert re.search(pat, text), pat
  768. ss = f'Total {self.proto.coin}:'
  769. assert re.search(ss, text), ss
  770. return t
  771. def token_bal(self, n=None):
  772. t = self.spawn('mmgen-tool', self.eth_args + ['--token=mm1', 'twview', 'wide=1'])
  773. text = t.read(strip_color=True)
  774. for addr, _amt1, _amt2 in token_bals(n):
  775. pat = fr'{addr}\b.*\D{_amt1}\D.*\b{_amt2}\D'
  776. assert re.search(pat, text), pat
  777. ss = 'Total MM1:'
  778. assert re.search(ss, text), ss
  779. return t
  780. def bal_getbalance(self, sid, idx, etc_adj=False, extra_args=[]):
  781. bal1 = token_bals_getbalance(idx)[0]
  782. bal2 = token_bals_getbalance(idx)[1]
  783. bal1 = Decimal(bal1)
  784. t = self.spawn('mmgen-tool', self.eth_args + extra_args + ['getbalance'])
  785. t.expect(rf'{sid}:.*'+str(bal1), regex=True)
  786. t.expect(r'Non-MMGen:.*'+bal2, regex=True)
  787. total = strip_ansi_escapes(t.expect_getend(rf'TOTAL {self.proto.coin}')).split()[0]
  788. assert Decimal(bal1) + Decimal(bal2) == Decimal(total)
  789. return t
  790. def add_comment(self, comment, addr='98831F3A:E:3'):
  791. t = self.spawn('mmgen-tool', self.eth_args + ['add_label', addr, comment])
  792. t.expect('Added label.*in tracking wallet', regex=True)
  793. return t
  794. def chk_comment(self, comment_pat, addr='98831F3A:E:3'):
  795. t = self.spawn('mmgen-tool', self.eth_args + ['listaddresses', 'all_labels=1'])
  796. t.expect(fr'{addr}\b.*{comment_pat}', regex=True)
  797. return t
  798. def add_comment1(self):
  799. return self.add_comment(comment=tw_comment_zh)
  800. def chk_comment1(self):
  801. return self.chk_comment(comment_pat=tw_comment_zh[:3])
  802. def add_comment2(self):
  803. return self.add_comment(comment=tw_comment_lat_cyr_gr)
  804. def chk_comment2(self):
  805. return self.chk_comment(comment_pat=tw_comment_lat_cyr_gr[:3])
  806. def remove_comment(self, addr='98831F3A:E:3'):
  807. t = self.spawn('mmgen-tool', self.eth_args + ['remove_label', addr])
  808. t.expect('Removed label.*in tracking wallet', regex=True)
  809. return t
  810. def token_compile(self, token_data={}):
  811. odir = joinpath(self.tmpdir, token_data['symbol'].lower())
  812. if not self.using_solc:
  813. imsg(f'Using precompiled contract data in {odir}')
  814. return 'skip' if os.path.exists(odir) else False
  815. self.spawn('', msg_only=True)
  816. cmd_args = [f'--{k}={v}' for k, v in list(token_data.items())]
  817. imsg("Compiling solidity token contract '{}' with 'solc'".format(token_data['symbol']))
  818. try:
  819. os.mkdir(odir)
  820. except:
  821. pass
  822. cmd = [
  823. 'python3',
  824. 'scripts/create-token.py',
  825. '--coin=' + self.proto.coin,
  826. '--outdir=' + odir
  827. ] + cmd_args + [self.proto.checksummed_addr(dfl_devaddr)]
  828. imsg('Executing: {}'.format(' '.join(cmd)))
  829. cp = run(cmd, stdout=DEVNULL, stderr=PIPE)
  830. if cp.returncode != 0:
  831. rmsg('solc failed with the following output:')
  832. die(2, cp.stderr.decode())
  833. imsg('ERC20 token {!r} compiled'.format(token_data['symbol']))
  834. return 'ok'
  835. def token_compile1(self):
  836. token_data = {'name':'MMGen Token 1', 'symbol':'MM1', 'supply':10**26, 'decimals':18}
  837. return self.token_compile(token_data)
  838. def token_compile2(self):
  839. token_data = {'name':'MMGen Token 2', 'symbol':'MM2', 'supply':10**18, 'decimals':10}
  840. return self.token_compile(token_data)
  841. async def get_tx_receipt(self, txid):
  842. if self.daemon.id == 'geth': # yet another Geth bug
  843. await asyncio.sleep(0.5)
  844. from mmgen.tx import NewTX
  845. tx = await NewTX(cfg=cfg, proto=self.proto)
  846. tx.rpc = await self.rpc
  847. res = await tx.get_receipt(txid)
  848. imsg(f'Gas sent: {res.gas_sent.hl():<9} {(res.gas_sent*res.gas_price).hl2(encl="()")}')
  849. imsg(f'Gas used: {res.gas_used.hl():<9} {(res.gas_used*res.gas_price).hl2(encl="()")}')
  850. imsg(f'Gas price: {res.gas_price.hl()}')
  851. if res.gas_used == res.gas_sent:
  852. omsg(yellow('Warning: all gas was used!'))
  853. return res
  854. async def token_deploy(self, num, key, gas, mmgen_cmd='txdo', tx_fee='8G'):
  855. keyfile = joinpath(self.tmpdir, parity_devkey_fn)
  856. fn = joinpath(self.tmpdir, 'mm'+str(num), key+'.bin')
  857. args = [
  858. '-B',
  859. f'--fee={tx_fee}',
  860. f'--gas={gas}',
  861. f'--contract-data={fn}',
  862. f'--inputs={dfl_devaddr}',
  863. '--yes',
  864. ]
  865. if mmgen_cmd == 'txdo':
  866. args += ['-k', keyfile]
  867. t = self.spawn('mmgen-'+mmgen_cmd, self.eth_args + args)
  868. if mmgen_cmd == 'txcreate':
  869. t.written_to_file('transaction')
  870. ext = '[0,8000]{}.regtest.rawtx'.format('-α' if cfg.debug_utf8 else '')
  871. txfile = self.get_file_with_ext(ext, no_dot=True)
  872. t = self.spawn('mmgen-txsign', self.eth_args + ['--yes', '-k', keyfile, txfile], no_msg=True)
  873. self.txsign_ui_common(t, ni=True)
  874. txfile = txfile.replace('.rawtx', '.sigtx')
  875. t = self.spawn('mmgen-txsend', self.eth_args + [txfile], no_msg=True)
  876. txid = self.txsend_ui_common(t,
  877. caller = mmgen_cmd,
  878. quiet = mmgen_cmd == 'txdo' or not cfg.debug,
  879. bogus_send = False)
  880. addr = strip_ansi_escapes(t.expect_getend('Contract address: '))
  881. if (await self.get_tx_receipt(txid)).status == 0:
  882. die(2, f'Contract {num}:{key} failed to execute. Aborting')
  883. if key == 'Token':
  884. self.write_to_tmpfile(f'token_addr{num}', addr+'\n')
  885. imsg(f'\nToken MM{num} deployed!')
  886. return t
  887. async def token_deploy1a(self):
  888. return await self.token_deploy(num=1, key='SafeMath', gas=500_000)
  889. async def token_deploy1b(self):
  890. return await self.token_deploy(num=1, key='Owned', gas=1_000_000)
  891. async def token_deploy1c(self):
  892. return await self.token_deploy(num=1, key='Token', gas=4_000_000, tx_fee='7G')
  893. def tx_status2(self):
  894. return self.tx_status(
  895. ext = self.proto.coin+'[0,7000]{}.regtest.sigtx',
  896. expect_str = 'successfully executed')
  897. def bal6(self):
  898. return self.bal5()
  899. async def token_deploy2a(self):
  900. return await self.token_deploy(num=2, key='SafeMath', gas=500_000)
  901. async def token_deploy2b(self):
  902. return await self.token_deploy(num=2, key='Owned', gas=1_000_000)
  903. async def token_deploy2c(self):
  904. return await self.token_deploy(num=2, key='Token', gas=4_000_000)
  905. async def contract_deploy(self): # test create, sign, send
  906. return await self.token_deploy(num=2, key='SafeMath', gas=500_000, mmgen_cmd='txcreate')
  907. async def token_transfer_ops(self, op, amt=1000, num_tokens=2):
  908. self.spawn('', msg_only=True)
  909. sid = dfl_sid
  910. from mmgen.tool.wallet import tool_cmd
  911. usr_mmaddrs = [f'{sid}:E:{i}' for i in (11, 21)][:num_tokens]
  912. from mmgen.proto.eth.contract import ResolvedToken
  913. async def do_transfer(rpc):
  914. for i in range(num_tokens):
  915. tk = await ResolvedToken(
  916. cfg,
  917. self.proto,
  918. rpc,
  919. self.read_from_tmpfile(f'token_addr{i+1}').strip())
  920. imsg_r('\n' + await tk.info())
  921. imsg('dev token balance (pre-send): {}'.format(await tk.get_balance(dfl_devaddr)))
  922. imsg(f'Sending {amt} {self.proto.dcoin} to address {usr_addrs[i]} ({usr_mmaddrs[i]})')
  923. txid = await tk.transfer(
  924. dfl_devaddr,
  925. usr_addrs[i],
  926. amt,
  927. dfl_devkey,
  928. start_gas = self.proto.coin_amt(60000, from_unit='wei'),
  929. gasPrice = self.proto.coin_amt(8, from_unit='Gwei'))
  930. if (await self.get_tx_receipt(txid)).status == 0:
  931. die(2, 'Transfer of token funds failed. Aborting')
  932. async def show_bals(rpc):
  933. for i in range(num_tokens):
  934. tk = await ResolvedToken(
  935. cfg,
  936. self.proto,
  937. rpc,
  938. self.read_from_tmpfile(f'token_addr{i+1}').strip())
  939. imsg('Token: {}'.format(await tk.get_symbol()))
  940. imsg(f'dev token balance: {await tk.get_balance(dfl_devaddr)}')
  941. imsg('usr token balance: {} ({} {})'.format(
  942. await tk.get_balance(usr_addrs[i]),
  943. usr_mmaddrs[i],
  944. usr_addrs[i]))
  945. def gen_addr(addr):
  946. return tool_cmd(cfg, cmdname='gen_addr', proto=self.proto).gen_addr(addr, dfl_words_file)
  947. silence()
  948. usr_addrs = list(map(gen_addr, usr_mmaddrs))
  949. if op == 'show_bals':
  950. await show_bals(await self.rpc)
  951. elif op == 'do_transfer':
  952. await do_transfer(await self.rpc)
  953. end_silence()
  954. return 'ok'
  955. def token_fund_users(self):
  956. return self.token_transfer_ops(op='do_transfer')
  957. def token_user_bals(self):
  958. return self.token_transfer_ops(op='show_bals')
  959. def token_addrgen(self, num_tokens=2):
  960. t = self.addrgen(addrs='11-13')
  961. if num_tokens == 1:
  962. return t
  963. ok_msg()
  964. return self.addrgen(addrs='21-23')
  965. def token_addrimport_badaddr1(self):
  966. t = self.addrimport(
  967. ext = '[11-13]{}.regtest.addrs',
  968. add_args = ['--token=abc'],
  969. bad_input = True,
  970. exit_val = 2)
  971. t.expect('could not be resolved')
  972. return t
  973. def token_addrimport_badaddr2(self):
  974. t = self.addrimport(
  975. ext = '[11-13]{}.regtest.addrs',
  976. add_args = ['--token='+'00deadbeef'*4],
  977. bad_input = True,
  978. exit_val = 2)
  979. t.expect('could not be resolved')
  980. return t
  981. def token_addrimport(self, addr_file, addr_range, expect, extra_args=[]):
  982. token_addr = self.read_from_tmpfile(addr_file).strip()
  983. return self.addrimport(
  984. ext = f'[{addr_range}]{{}}.regtest.addrs',
  985. expect = expect,
  986. add_args = ['--token-addr='+token_addr]+extra_args)
  987. def token_addrimport_addr1(self):
  988. return self.token_addrimport('token_addr1', '11-13', expect='3/3')
  989. def token_addrimport_addr2(self):
  990. return self.token_addrimport('token_addr2', '21-23', expect='3/3')
  991. def token_addrimport_batch(self):
  992. return self.token_addrimport('token_addr1', '11-13', expect='3 addresses', extra_args=['--batch'])
  993. def token_addrimport_sym(self):
  994. return self.addrimport(
  995. ext = '[21-23]{}.regtest.addrs',
  996. expect = '3/3',
  997. add_args = ['--token=MM2'])
  998. def bal7(self):
  999. return self.bal5()
  1000. def token_bal1(self):
  1001. return self.token_bal(n='1')
  1002. def token_txcreate(self, args=[], token='', inputs='1', fee='50G', file_desc='Unsigned transaction'):
  1003. return self.txcreate_ui_common(
  1004. self.spawn('mmgen-txcreate', self.eth_args + [f'--token={token}', '-B', f'--fee={fee}'] + args),
  1005. menu = [],
  1006. inputs = inputs,
  1007. input_sels_prompt = 'to spend from',
  1008. add_comment = tx_comment_lat_cyr_gr,
  1009. file_desc = file_desc)
  1010. def token_txsign(self, ext='', token=''):
  1011. return self.txsign(ni=True, ext=ext, add_args=['--token='+token])
  1012. def token_txsend(self, ext='', token=''):
  1013. return self.txsend(ext=ext, add_args=['--token='+token])
  1014. def token_txcreate1(self):
  1015. return self.token_txcreate(args=['98831F3A:E:12,1.23456'], token='mm1')
  1016. def token_txview1_raw(self):
  1017. return self.txview(ext_fs='1.23456,50000]{}.regtest.rawtx')
  1018. def token_txsign1(self):
  1019. return self.token_txsign(ext='1.23456,50000]{}.regtest.rawtx', token='mm1')
  1020. def token_txsend1(self):
  1021. return self.token_txsend(ext='1.23456,50000]{}.regtest.sigtx', token='mm1')
  1022. def token_txview1_sig(self):
  1023. return self.txview(ext_fs='1.23456,50000]{}.regtest.sigtx')
  1024. def tx_status3(self):
  1025. return self.tx_status(
  1026. ext = '1.23456,50000]{}.regtest.sigtx',
  1027. add_args = ['--token=mm1'],
  1028. expect_str = 'successfully executed',
  1029. expect_str2 = 'has 1 confirmation')
  1030. def token_bal2(self):
  1031. return self.token_bal(n='2')
  1032. def twview(self, args=[], expect_str='', tool_args=[]):
  1033. t = self.spawn('mmgen-tool', self.eth_args + args + ['twview'] + tool_args)
  1034. if expect_str:
  1035. t.expect(expect_str, regex=True)
  1036. return t
  1037. def token_txcreate2(self):
  1038. return self.token_txcreate(args=[burn_addr+', '+amt2], token='mm1')
  1039. def token_txbump(self):
  1040. return self.txbump(ext=amt2+',50000]{}.regtest.rawtx', fee='56G', add_args=['--token=mm1'])
  1041. def token_txsign2(self):
  1042. return self.token_txsign(ext=amt2+',50000]{}.regtest.rawtx', token='mm1')
  1043. def token_txsend2(self):
  1044. return self.token_txsend(ext=amt2+',50000]{}.regtest.sigtx', token='mm1')
  1045. def token_bal3(self):
  1046. return self.token_bal(n='3')
  1047. def del_dev_addr(self):
  1048. t = self.spawn('mmgen-tool', self.eth_args + ['remove_address', dfl_devaddr])
  1049. t.expect(f"'{dfl_devaddr}' deleted")
  1050. return t
  1051. def bal1_getbalance(self):
  1052. return self.bal_getbalance(dfl_sid, '1', etc_adj=True)
  1053. def addrimport_token_burn_addr(self):
  1054. return self.addrimport_one_addr(addr=burn_addr, extra_args=['--token=mm1'])
  1055. def token_bal4(self):
  1056. return self.token_bal(n='4')
  1057. def token_bal_getbalance(self):
  1058. return self.bal_getbalance(dfl_sid, '2', extra_args=['--token=mm1'])
  1059. def txcreate_noamt(self):
  1060. return self.txcreate(args=['98831F3A:E:12'])
  1061. def txsign_noamt(self):
  1062. return self.txsign(ext='99.99895,50000]{}.regtest.rawtx')
  1063. def txsend_noamt(self):
  1064. return self.txsend(ext='99.99895,50000]{}.regtest.sigtx')
  1065. def bal8(self):
  1066. return self.bal(n='8')
  1067. def token_bal5(self):
  1068. return self.token_bal(n='5')
  1069. def token_txcreate_noamt(self):
  1070. return self.token_txcreate(args=['98831F3A:E:13'], token='mm1', inputs='2', fee='51G')
  1071. def token_txsign_noamt(self):
  1072. return self.token_txsign(ext='1.23456,51000]{}.regtest.rawtx', token='mm1')
  1073. def token_txsend_noamt(self):
  1074. return self.token_txsend(ext='1.23456,51000]{}.regtest.sigtx', token='mm1')
  1075. def bal9(self):
  1076. return self.bal(n='9')
  1077. def token_bal6(self):
  1078. return self.token_bal(n='6')
  1079. def listaddresses(self, args=[], tool_args=['all_labels=1']):
  1080. return self.spawn('mmgen-tool', self.eth_args + args + ['listaddresses'] + tool_args)
  1081. def listaddresses1(self):
  1082. return self.listaddresses()
  1083. def listaddresses2(self):
  1084. return self.listaddresses(tool_args=['minconf=999999999'])
  1085. def listaddresses3(self):
  1086. return self.listaddresses(tool_args=['sort=amt', 'reverse=1'])
  1087. def listaddresses4(self):
  1088. return self.listaddresses(tool_args=['sort=age', 'showempty=0'])
  1089. def token_listaddresses1(self):
  1090. return self.listaddresses(args=['--token=mm1'])
  1091. def token_listaddresses2(self):
  1092. return self.listaddresses(args=['--token=mm1'], tool_args=['showempty=1'])
  1093. def token_listaddresses3(self):
  1094. return self.listaddresses(args=['--token=mm1'], tool_args=['showempty=0'])
  1095. def token_listaddresses4(self):
  1096. return self.listaddresses(args=['--token=mm2'], tool_args=['sort=age', 'reverse=1'])
  1097. def twview_cached_balances(self):
  1098. return self.twview(args=['--cached-balances'])
  1099. def token_twview_cached_balances(self):
  1100. return self.twview(args=['--token=mm1', '--cached-balances'])
  1101. def txcreate_cached_balances(self):
  1102. args = ['--fee=20G', '--cached-balances', '98831F3A:E:3, 0.1276']
  1103. return self.txcreate(args=args, acct='2')
  1104. def token_txcreate_cached_balances(self):
  1105. args=['--cached-balances', '--fee=12G', '98831F3A:E:12, 1.2789']
  1106. return self.token_txcreate(args=args, token='mm1')
  1107. def txdo_cached_balances(
  1108. self,
  1109. acct = '2',
  1110. fee_info_data = ('0.00105', '50'),
  1111. add_args = ['98831F3A:E:3,0.4321']):
  1112. t = self.txcreate(
  1113. args = ['--fee=20G', '--cached-balances'] + add_args + [dfl_words_file],
  1114. acct = acct,
  1115. caller = 'txdo',
  1116. fee_info_data = fee_info_data,
  1117. no_read = True)
  1118. self._do_confirm_send(t, quiet=not cfg.debug, sure=False)
  1119. return t
  1120. def txcreate_refresh_balances(self):
  1121. return self._txcreate_refresh_balances(
  1122. bals = ['2', '3'],
  1123. args = ['-B', '--cached-balances', '-i'],
  1124. total = vbal5,
  1125. adj_total = True,
  1126. total_coin = None)
  1127. def _txcreate_refresh_balances(self, bals, args, total, adj_total, total_coin):
  1128. if total_coin is None:
  1129. total_coin = self.proto.coin
  1130. t = self.spawn('mmgen-txcreate', self.eth_args + args)
  1131. for n in bals:
  1132. t.expect('[R]efresh balance:\b', 'R')
  1133. t.expect(' main menu): ', n+'\n')
  1134. t.expect('Is this what you want? (y/N): ', 'y')
  1135. t.expect('[R]efresh balance:\b', 'q')
  1136. t.expect(rf'Total unspent:.*\D{total}\D.*{total_coin}', regex=True)
  1137. return t
  1138. def bal10(self):
  1139. return self.bal(n='10')
  1140. def token_txdo_cached_balances(self):
  1141. return self.txdo_cached_balances(
  1142. acct = '1',
  1143. fee_info_data = ('0.0026', '50'),
  1144. add_args = ['--token=mm1', '98831F3A:E:12,43.21'])
  1145. def token_txcreate_refresh_balances(self):
  1146. return self._txcreate_refresh_balances(
  1147. bals = ['1', '2'],
  1148. args = ['--token=mm1', '-B', '--cached-balances', '-i'],
  1149. total = '1000',
  1150. adj_total = False,
  1151. total_coin = 'MM1')
  1152. def token_bal7(self):
  1153. return self.token_bal(n='7')
  1154. def twview1(self):
  1155. return self.twview()
  1156. def twview2(self):
  1157. return self.twview(tool_args=['wide=1'])
  1158. def twview3(self):
  1159. return self.twview(tool_args=['wide=1', 'sort=age'])
  1160. def twview4(self):
  1161. return self.twview(tool_args=['wide=1', 'minconf=999999999'])
  1162. def twview5(self):
  1163. return self.twview(tool_args=['wide=1', 'minconf=0'])
  1164. def twview6(self):
  1165. return self.twview(expect_str=vbal7)
  1166. def twview7(self):
  1167. return self.twview(args=['--cached-balances'])
  1168. def twview8(self):
  1169. return self.twview()
  1170. def twview9(self):
  1171. return self.twview(args=['--cached-balances'], expect_str=vbal6)
  1172. def token_twview1(self):
  1173. return self.twview(args=['--token=mm1'])
  1174. def token_twview2(self):
  1175. return self.twview(args=['--token=mm1'], tool_args=['wide=1'])
  1176. def token_twview3(self):
  1177. return self.twview(args=['--token=mm1'], tool_args=['wide=1', 'sort=age'])
  1178. def edit_comment(
  1179. self,
  1180. out_num,
  1181. args = [],
  1182. action = 'l',
  1183. comment_text = None,
  1184. changed = False,
  1185. pexpect_spawn = None):
  1186. t = self.spawn('mmgen-txcreate', self.eth_args + args + ['-B', '-i'], pexpect_spawn=pexpect_spawn)
  1187. menu_prompt = 'efresh balance:\b'
  1188. t.expect(menu_prompt, 'M')
  1189. t.expect(menu_prompt, action)
  1190. t.expect(r'return to main menu): ', out_num+'\n')
  1191. for p, r in (
  1192. ('Enter label text.*: ', comment_text+'\n') if comment_text is not None else (r'\(y/N\): ', 'y'),
  1193. (r'\(y/N\): ', 'y') if comment_text == Ctrl_U else (None, None),
  1194. ):
  1195. if p:
  1196. t.expect(p, r, regex=True)
  1197. m = (
  1198. 'Label for account #{} edited' if changed else
  1199. 'Account #{} removed' if action == 'D' else
  1200. 'Label added to account #{}' if comment_text and comment_text != Ctrl_U else
  1201. 'Label removed from account #{}')
  1202. t.expect(m.format(out_num))
  1203. t.expect(menu_prompt, 'M')
  1204. t.expect(menu_prompt, 'q')
  1205. t.expect('Total unspent:')
  1206. return t
  1207. def edit_comment1(self):
  1208. return self.edit_comment(out_num=del_addrs[0], comment_text=tw_comment_zh[:3])
  1209. def edit_comment2(self):
  1210. spawn = not sys.platform == 'win32'
  1211. return self.edit_comment(
  1212. out_num = del_addrs[0],
  1213. comment_text = tw_comment_zh[3:],
  1214. changed = True,
  1215. pexpect_spawn = spawn)
  1216. def edit_comment3(self):
  1217. return self.edit_comment(out_num=del_addrs[1], comment_text=tw_comment_lat_cyr_gr)
  1218. def edit_comment4(self):
  1219. if self.skip_for_win('no pexpect_spawn'):
  1220. return 'skip'
  1221. return self.edit_comment(out_num=del_addrs[0], comment_text=Ctrl_U, pexpect_spawn=True)
  1222. def token_edit_comment1(self):
  1223. return self.edit_comment(out_num='1', comment_text='Token label #1', args=['--token=mm1'])
  1224. def remove_addr1(self):
  1225. return self.edit_comment(out_num=del_addrs[0], action='D')
  1226. def remove_addr2(self):
  1227. return self.edit_comment(out_num=del_addrs[1], action='D')
  1228. def token_remove_addr1(self):
  1229. return self.edit_comment(out_num=del_addrs[0], args=['--token=mm1'], action='D')
  1230. def token_remove_addr2(self):
  1231. return self.edit_comment(out_num=del_addrs[1], args=['--token=mm1'], action='D')
  1232. def twexport_noamt(self):
  1233. return self.twexport(add_args=['include_amts=0'])
  1234. def twexport(self, add_args=[]):
  1235. t = self.spawn('mmgen-tool', self.eth_args + ['twexport'] + add_args)
  1236. t.written_to_file('JSON data')
  1237. return t
  1238. async def twmove(self):
  1239. self.spawn('', msg_only=True)
  1240. from mmgen.tw.ctl import TwCtl
  1241. twctl = await TwCtl(cfg, self.proto, no_wallet_init=True)
  1242. imsg('Moving tracking wallet')
  1243. fn_bak = twctl.tw_path.with_suffix('.bak.json')
  1244. fn_bak.unlink(missing_ok=True)
  1245. twctl.tw_path.rename(fn_bak)
  1246. return 'ok'
  1247. def twimport(self, add_args=[], expect_str=None):
  1248. from mmgen.tw.json import TwJSON
  1249. fn = joinpath(self.tmpdir, TwJSON.Base(cfg, self.proto).dump_fn)
  1250. t = self.spawn('mmgen-tool', self.eth_args_noquiet + ['twimport', fn] + add_args)
  1251. t.expect('(y/N): ', 'y')
  1252. if expect_str:
  1253. t.expect(expect_str)
  1254. t.written_to_file('tracking wallet data')
  1255. return t
  1256. def twimport_nochksum(self):
  1257. return self.twimport(add_args=['ignore_checksum=true'], expect_str='ignoring incorrect checksum')
  1258. def tw_chktotal(self):
  1259. self.spawn('', msg_only=True)
  1260. from mmgen.tw.json import TwJSON
  1261. fn = joinpath(self.tmpdir, TwJSON.Base(cfg, self.proto).dump_fn)
  1262. res = json.loads(read_from_file(fn))
  1263. cmp_or_die(res['data']['value'], vbal6, 'value in tracking wallet JSON dump')
  1264. return 'ok'
  1265. async def twcompare(self):
  1266. self.spawn('', msg_only=True)
  1267. from mmgen.tw.ctl import TwCtl
  1268. twctl = await TwCtl(cfg, self.proto, no_wallet_init=True)
  1269. fn = twctl.tw_path
  1270. fn_bak = fn.with_suffix('.bak.json')
  1271. imsg('Comparing imported tracking wallet with original')
  1272. data = [json.dumps(json.loads(f.read_text()), sort_keys=True) for f in (fn, fn_bak)]
  1273. cmp_or_die(*data, 'tracking wallets')
  1274. return 'ok'
  1275. def edit_json_twdump(self):
  1276. self.spawn('', msg_only=True)
  1277. from mmgen.tw.json import TwJSON
  1278. fn = TwJSON.Base(cfg, self.proto).dump_fn
  1279. text = json.loads(self.read_from_tmpfile(fn))
  1280. token_addr = self.read_from_tmpfile('token_addr2').strip()
  1281. text['data']['entries']['tokens'][token_addr][2][3] = f'edited comment [фубар] [{gr_uc}]'
  1282. self.write_to_tmpfile(fn, json.dumps(text, indent=4))
  1283. return 'ok'
  1284. def stop(self):
  1285. self.spawn('', msg_only=True)
  1286. if not cfg.no_daemon_stop:
  1287. if not stop_test_daemons(self.proto.coin+'_rt', remove_datadir=True):
  1288. return False
  1289. set_vt100()
  1290. return 'ok'