ts_main.py 32 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755
  1. #!/usr/bin/env python3
  2. #
  3. # mmgen = Multi-Mode GENerator, command-line Bitcoin cold storage solution
  4. # Copyright (C)2013-2020 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_main.py: Basic operations tests for the test.py test suite
  20. """
  21. from mmgen.globalvars import g
  22. from mmgen.opts import opt
  23. from mmgen.seed import SeedSource,Wallet,MMGenMnemonic,IncogWallet,MMGenSeedFile
  24. from test.common import *
  25. from test.test_py_d.common import *
  26. from test.test_py_d.ts_base import *
  27. from test.test_py_d.ts_shared import *
  28. def make_brainwallet_file(fn):
  29. # Print random words with random whitespace in between
  30. wl = rwords.split()
  31. nwords,ws_list,max_spaces = 10,' \n',5
  32. def rand_ws_seq():
  33. nchars = getrandnum(1) % max_spaces + 1
  34. return ''.join([ws_list[getrandnum_range(1,200) % len(ws_list)] for i in range(nchars)])
  35. rand_pairs = [wl[getrandnum_range(1,200) % len(wl)] + rand_ws_seq() for i in range(nwords)]
  36. d = ''.join(rand_pairs).rstrip() + '\n'
  37. if opt.verbose: msg_r('Brainwallet password:\n{}'.format(cyan(d)))
  38. write_data_to_file(fn,d,'brainwallet password',quiet=True,ignore_opt_outdir=True)
  39. def verify_checksum_or_exit(checksum,chk):
  40. if checksum != chk:
  41. raise TestSuiteFatalException('Checksum error: {}'.format(chk))
  42. vmsg(green('Checksums match: ') + cyan(chk))
  43. addrs_per_wallet = 8
  44. # 100 words chosen randomly from here:
  45. # https://github.com/bitcoin/bips/pull/432/files/6332230d63149a950d05db78964a03bfd344e6b0
  46. rwords = """
  47. алфавит алый амнезия амфора артист баян белый биатлон брат бульвар веревка вернуть весть возраст
  48. восток горло горный десяток дятел ежевика жест жизнь жрать заговор здание зона изделие итог кабина
  49. кавалер каждый канал керосин класс клятва князь кривой крыша крючок кузнец кукла ландшафт мальчик
  50. масса масштаб матрос мрак муравей мычать негодяй носок ночной нрав оборот оружие открытие оттенок
  51. палуба пароход период пехота печать письмо позор полтора понятие поцелуй почему приступ пруд пятно
  52. ранее режим речь роса рынок рябой седой сердце сквозь смех снимок сойти соперник спичка стон
  53. сувенир сугроб суть сцена театр тираж толк удивить улыбка фирма читатель эстония эстрада юность
  54. """
  55. class TestSuiteMain(TestSuiteBase,TestSuiteShared):
  56. 'basic operations with emulated tracking wallet'
  57. tmpdir_nums = [1,2,3,4,5,14,15,16,20,21]
  58. networks = ('btc','btc_tn','ltc','ltc_tn','bch','bch_tn')
  59. passthru_opts = ('daemon_data_dir','rpc_port','coin','testnet')
  60. segwit_opts_ok = True
  61. cmd_group = (
  62. ('walletgen_dfl_wallet', (15,'wallet generation (default wallet)',[[[],15]])),
  63. ('subwalletgen_dfl_wallet', (15,'subwallet generation (default wallet)',[[[pwfile],15]])),
  64. ('export_seed_dfl_wallet',(15,'seed export to mmseed format (default wallet)',[[[pwfile],15]])),
  65. ('addrgen_dfl_wallet',(15,'address generation (default wallet)',[[[pwfile],15]])),
  66. ('txcreate_dfl_wallet',(15,'transaction creation (default wallet)',[[['addrs'],15]])),
  67. ('txsign_dfl_wallet',(15,'transaction signing (default wallet)',[[['rawtx',pwfile],15]])),
  68. ('passchg_dfl_wallet',(16,'password, label and hash preset change (default wallet)',[[[pwfile],15]])),
  69. ('walletchk_newpass_dfl_wallet',(16,'wallet check with new pw, label and hash preset',[[[pwfile],16]])),
  70. ('delete_dfl_wallet',(15,'delete default wallet',[[[pwfile],15]])),
  71. ('walletgen', (1,'wallet generation', [[['del_dw_run'],15]])),
  72. ('subwalletgen', (1,'subwallet generation', [[['mmdat'],1]])),
  73. ('subwalletgen_mnemonic',(1,'subwallet generation (to mnemonic format)',[[['mmdat'],1]])),
  74. # ('walletchk', (1,'wallet check', [[['mmdat'],1]])),
  75. ('passchg', (5,'password, label and hash preset change',[[['mmdat',pwfile],1]])),
  76. ('passchg_keeplabel',(5,'password, label and hash preset change (keep label)',[[['mmdat',pwfile],1]])),
  77. ('passchg_usrlabel',(5,'password, label and hash preset change (interactive label)',[[['mmdat',pwfile],1]])),
  78. ('walletchk_newpass',(5,'wallet check with new pw, label and hash preset',[[['mmdat',pwfile],5]])),
  79. ('addrgen', (1,'address generation', [[['mmdat',pwfile],1]])),
  80. ('txcreate', (1,'transaction creation', [[['addrs'],1]])),
  81. ('txbump', (1,'transaction fee bumping (no send)',[[['rawtx'],1]])),
  82. ('txsign', (1,'transaction signing', [[['mmdat','rawtx',pwfile,'txbump'],1]])),
  83. ('txsend', (1,'transaction sending', [[['sigtx'],1]])),
  84. # txdo must go after txsign
  85. ('txdo', (1,'online transaction', [[['sigtx','mmdat'],1]])),
  86. ('export_seed', (1,'seed export to mmseed format', [[['mmdat'],1]])),
  87. ('export_hex', (1,'seed export to hexadecimal format', [[['mmdat'],1]])),
  88. ('export_mnemonic', (1,'seed export to mmwords format', [[['mmdat'],1]])),
  89. ('export_bip39', (1,'seed export to bip39 format', [[['mmdat'],1]])),
  90. ('export_incog', (1,'seed export to mmincog format', [[['mmdat'],1]])),
  91. ('export_incog_hex',(1,'seed export to mmincog hex format', [[['mmdat'],1]])),
  92. ('export_incog_hidden',(1,'seed export to hidden mmincog format', [[['mmdat'],1]])),
  93. ('addrgen_seed', (1,'address generation from mmseed file', [[['mmseed','addrs'],1]])),
  94. ('addrgen_hex', (1,'address generation from mmhex file', [[['mmhex','addrs'],1]])),
  95. ('addrgen_mnemonic',(1,'address generation from mmwords file',[[['mmwords','addrs'],1]])),
  96. ('addrgen_incog', (1,'address generation from mmincog file',[[['mmincog','addrs'],1]])),
  97. ('addrgen_incog_hex',(1,'address generation from mmincog hex file',[[['mmincox','addrs'],1]])),
  98. ('addrgen_incog_hidden',(1,'address generation from hidden mmincog file', [[[hincog_fn,'addrs'],1]])),
  99. ('keyaddrgen', (1,'key-address file generation', [[['mmdat',pwfile],1]])),
  100. ('txsign_keyaddr',(1,'transaction signing with key-address file', [[['akeys.mmenc','rawtx'],1]])),
  101. ('txcreate_ni', (1,'transaction creation (non-interactive)', [[['addrs'],1]])),
  102. ('walletgen2',(2,'wallet generation (2), 128-bit seed', [[['del_dw_run'],15]])),
  103. ('addrgen2', (2,'address generation (2)', [[['mmdat'],2]])),
  104. ('txcreate2', (2,'transaction creation (2)', [[['addrs'],2]])),
  105. ('txsign2', (2,'transaction signing, two transactions',[[['mmdat','rawtx'],1],[['mmdat','rawtx'],2]])),
  106. ('export_mnemonic2', (2,'seed export to mmwords format (2)',[[['mmdat'],2]])),
  107. ('walletgen3',(3,'wallet generation (3)', [[['del_dw_run'],15]])),
  108. ('addrgen3', (3,'address generation (3)', [[['mmdat'],3]])),
  109. ('txcreate3', (3,'tx creation with inputs and outputs from two wallets', [[['addrs'],1],[['addrs'],3]])),
  110. ('txsign3', (3,'tx signing with inputs and outputs from two wallets',[[['mmdat'],1],[['mmdat','rawtx'],3]])),
  111. ('walletgen14', (14,'wallet generation (14)', [[['del_dw_run'],15]],14)),
  112. ('addrgen14', (14,'address generation (14)', [[['mmdat'],14]])),
  113. ('keyaddrgen14',(14,'key-address file generation (14)', [[['mmdat'],14]],14)),
  114. ('walletgen4',(4,'wallet generation (4) (brainwallet)', [[['del_dw_run'],15]])),
  115. ('addrgen4', (4,'address generation (4)', [[['mmdat'],4]])),
  116. ('txcreate4', (4,'tx creation with inputs and outputs from four seed sources, key-address file and non-MMGen inputs and outputs', [[['addrs'],1],[['addrs'],2],[['addrs'],3],[['addrs'],4],[['addrs','akeys.mmenc'],14]])),
  117. ('txsign4', (4,'tx signing with inputs and outputs from incog file, mnemonic file, wallet, brainwallet, key-address file and non-MMGen inputs and outputs', [[['mmincog'],1],[['mmwords'],2],[['mmdat'],3],[['mmbrain','rawtx'],4],[['akeys.mmenc'],14]])),
  118. ('txdo4', (4,'tx creation,signing and sending with inputs and outputs from four seed sources, key-address file and non-MMGen inputs and outputs', [[['addrs'],1],[['addrs'],2],[['addrs'],3],[['addrs'],4],[['addrs','akeys.mmenc'],14],[['mmincog'],1],[['mmwords'],2],[['mmdat'],3],[['mmbrain','rawtx'],4],[['akeys.mmenc'],14]])), # must go after txsign4
  119. ('txbump4', (4,'tx fee bump + send with inputs and outputs from four seed sources, key-address file and non-MMGen inputs and outputs', [[['akeys.mmenc'],14],[['mmincog'],1],[['mmwords'],2],[['mmdat'],3],[['akeys.mmenc'],14],[['mmbrain','sigtx','mmdat','txdo'],4]])), # must go after txsign4
  120. ('walletgen5',(20,'wallet generation (5)', [[['del_dw_run'],15]],20)),
  121. ('addrgen5', (20,'address generation (5)', [[['mmdat'],20]])),
  122. ('txcreate5', (20,'transaction creation with bad vsize (5)', [[['addrs'],20]])),
  123. ('txsign5', (20,'transaction signing with bad vsize', [[['mmdat','rawtx'],20]])),
  124. ('walletgen6',(21,'wallet generation (6)', [[['del_dw_run'],15]],21)),
  125. ('addrgen6', (21,'address generation (6)', [[['mmdat'],21]])),
  126. ('txcreate6', (21,'transaction creation with corrected vsize (6)', [[['addrs'],21]])),
  127. ('txsign6', (21,'transaction signing with corrected vsize', [[['mmdat','rawtx'],21]])),
  128. )
  129. def __init__(self,trunner,cfgs,spawn):
  130. if g.coin.lower() not in self.networks:
  131. return
  132. rpc_init()
  133. self.lbl_id = ('account','label')['label_api' in g.rpch.caps]
  134. if g.coin in ('BTC','BCH','LTC'):
  135. self.tx_fee = {'btc':'0.0001','bch':'0.001','ltc':'0.01'}[g.coin.lower()]
  136. self.txbump_fee = {'btc':'123s','bch':'567s','ltc':'12345s'}[g.coin.lower()]
  137. return TestSuiteBase.__init__(self,trunner,cfgs,spawn)
  138. def _get_addrfile_checksum(self,display=False):
  139. addrfile = self.get_file_with_ext('addrs')
  140. silence()
  141. from mmgen.addr import AddrList
  142. chk = AddrList(addrfile).chksum
  143. if opt.verbose and display: msg('Checksum: {}'.format(cyan(chk)))
  144. end_silence()
  145. return chk
  146. def walletgen_dfl_wallet(self,seed_len=None):
  147. return self.walletgen(seed_len=seed_len,gen_dfl_wallet=True)
  148. def subwalletgen_dfl_wallet(self,pf):
  149. return self.subwalletgen(wf='default')
  150. def export_seed_dfl_wallet(self,pf,out_fmt='seed'):
  151. return self.export_seed(wf=None,out_fmt=out_fmt,pf=pf)
  152. def addrgen_dfl_wallet(self,pf=None,check_ref=False):
  153. return self.addrgen(wf=None,pf=pf,check_ref=check_ref,dfl_wallet=True)
  154. def txcreate_dfl_wallet(self,addrfile):
  155. return self.txcreate_common(sources=['15'])
  156. def txsign_dfl_wallet(self,txfile,pf='',save=True,has_label=False):
  157. return self.txsign(None,txfile,pf=pf,save=save,has_label=has_label,dfl_wallet=True)
  158. def passchg_dfl_wallet(self,pf):
  159. return self.passchg(wf=None,pf=pf,dfl_wallet=True)
  160. def walletchk_newpass_dfl_wallet(self,pf):
  161. return self.walletchk_newpass(wf=None,wcls=Wallet,pf=pf,dfl_wallet=True)
  162. def delete_dfl_wallet(self,pf):
  163. self.write_to_tmpfile('del_dw_run',b'',binary=True)
  164. if opt.no_dw_delete: return 'skip'
  165. for wf in [f for f in os.listdir(g.data_dir) if f[-6:]=='.mmdat']:
  166. os.unlink(joinpath(g.data_dir,wf))
  167. self.spawn('',msg_only=True)
  168. self.have_dfl_wallet = False
  169. return 'ok'
  170. def walletgen(self,del_dw_run='dummy',seed_len=None,gen_dfl_wallet=False):
  171. self.write_to_tmpfile(pwfile,self.wpasswd+'\n')
  172. args = ['-p1']
  173. if not gen_dfl_wallet: args += ['-d',self.tmpdir]
  174. if seed_len: args += ['-l',str(seed_len)]
  175. t = self.spawn('mmgen-walletgen', args + [self.usr_rand_arg])
  176. t.license()
  177. t.usr_rand(self.usr_rand_chars)
  178. t.expect('Generating')
  179. wcls = Wallet
  180. t.passphrase_new('new '+wcls.desc,self.wpasswd)
  181. t.label()
  182. if not self.have_dfl_wallet and gen_dfl_wallet:
  183. t.expect('move it to the data directory? (Y/n): ','y')
  184. self.have_dfl_wallet = True
  185. t.written_to_file(capfirst(wcls.desc))
  186. return t
  187. def subwalletgen(self,wf):
  188. args = [self.usr_rand_arg,'-p1','-d',self.tr.trash_dir,'-L','Label']
  189. if wf != 'default': args += [wf]
  190. t = self.spawn('mmgen-subwalletgen', args + ['10s'])
  191. t.license()
  192. wcls = Wallet
  193. t.passphrase(wcls.desc,self.cfgs['1']['wpasswd'])
  194. t.expect('Generating subseed 10S')
  195. t.passphrase_new('new '+wcls.desc,'foo')
  196. t.usr_rand(self.usr_rand_chars)
  197. fn = t.written_to_file(capfirst(wcls.desc))
  198. ext = get_extension(fn)
  199. assert ext,'incorrect file extension: {}'.format(ext)
  200. return t
  201. def subwalletgen_mnemonic(self,wf):
  202. icls = SeedSource.ext_to_type(get_extension(wf))
  203. ocls = MMGenMnemonic
  204. args = [self.usr_rand_arg,'-p1','-d',self.tr.trash_dir,'-o',ocls.fmt_codes[0],wf,'3L']
  205. t = self.spawn('mmgen-subwalletgen', args)
  206. t.license()
  207. t.passphrase(icls.desc,self.cfgs['1']['wpasswd'])
  208. t.expect('Generating subseed 3L')
  209. fn = t.written_to_file(capfirst(ocls.desc))
  210. ext = get_extension(fn)
  211. assert ext == ocls.ext,'incorrect file extension: {}'.format(ext)
  212. return t
  213. def passchg(self,wf,pf,label_action='cmdline',dfl_wallet=False):
  214. silence()
  215. self.write_to_tmpfile(pwfile,get_data_from_file(pf))
  216. end_silence()
  217. add_args = {'cmdline': ['-d',self.tmpdir,'-L','Changed label (UTF-8) α'],
  218. 'keep': ['-d',self.tr.trash_dir,'--keep-label'],
  219. 'user': ['-d',self.tr.trash_dir]
  220. }[label_action]
  221. t = self.spawn('mmgen-passchg', add_args + [self.usr_rand_arg, '-p2'] + ([wf] if wf else []))
  222. t.license()
  223. wcls = Wallet
  224. t.passphrase(wcls.desc,self.cfgs['1']['wpasswd'],pwtype='old')
  225. t.expect_getend('Hash preset changed to ')
  226. t.passphrase(wcls.desc,self.wpasswd,pwtype='new') # reuse passphrase?
  227. t.expect('Repeat passphrase: ',self.wpasswd+'\n')
  228. t.usr_rand(self.usr_rand_chars)
  229. if label_action == 'user':
  230. t.expect('Enter a wallet label.*: ','Interactive Label (UTF-8) α\n',regex=True)
  231. t.expect_getend(('Label changed to ','Reusing label ')[label_action=='keep'])
  232. # t.expect_getend('Key ID changed: ')
  233. if dfl_wallet:
  234. t.expect("Type uppercase 'YES' to confirm: ",'YES\n')
  235. t.written_to_file('New wallet')
  236. t.expect('Securely deleting old wallet')
  237. # t.expect('Okay to WIPE 1 regular file ? (Yes/No)','Yes\n')
  238. t.expect('Wallet passphrase has changed')
  239. t.expect_getend('has been changed to ')
  240. else:
  241. t.written_to_file(capfirst(wcls.desc))
  242. return t
  243. def passchg_keeplabel(self,wf,pf):
  244. return self.passchg(wf,pf,label_action='keep')
  245. def passchg_usrlabel(self,wf,pf):
  246. return self.passchg(wf,pf,label_action='user')
  247. def walletchk_newpass(self,wf,pf,wcls=None,dfl_wallet=False):
  248. return self.walletchk(wf,pf,wcls=wcls,dfl_wallet=dfl_wallet)
  249. def _write_fake_data_to_file(self,d):
  250. unspent_data_file = joinpath(self.tmpdir,'unspent.json')
  251. write_data_to_file(unspent_data_file,d,'Unspent outputs',quiet=True,ignore_opt_outdir=True)
  252. os.environ['MMGEN_BOGUS_WALLET_DATA'] = unspent_data_file
  253. bwd_msg = 'MMGEN_BOGUS_WALLET_DATA={}'.format(unspent_data_file)
  254. if opt.print_cmdline: msg(bwd_msg)
  255. if opt.log: self.tr.log_fd.write(bwd_msg + ' ')
  256. if opt.verbose or opt.exact_output:
  257. sys.stderr.write("Fake transaction wallet data written to file {!r}\n".format(unspent_data_file))
  258. def _create_fake_unspent_entry(self,coinaddr,al_id=None,idx=None,lbl=None,non_mmgen=False,segwit=False):
  259. if 'S' not in g.proto.mmtypes: segwit = False
  260. if lbl: lbl = ' ' + lbl
  261. k = coinaddr.addr_fmt
  262. if not segwit and k == 'p2sh': k = 'p2pkh'
  263. s_beg,s_end = { 'p2pkh': ('76a914','88ac'),
  264. 'p2sh': ('a914','87'),
  265. 'bech32': (g.proto.witness_vernum_hex + '14','') }[k]
  266. amt1,amt2 = {'btc':(10,40),'bch':(10,40),'ltc':(1000,4000)}[g.coin.lower()]
  267. ret = {
  268. self.lbl_id: '{}:{}'.format(g.proto.base_coin.lower(),coinaddr) if non_mmgen \
  269. else ('{}:{}{}'.format(al_id,idx,lbl)),
  270. 'vout': int(getrandnum(4) % 8),
  271. 'txid': os.urandom(32).hex(),
  272. 'amount': g.proto.coin_amt('{}.{}'.format(amt1 + getrandnum(4) % amt2, getrandnum(4) % 100000000)),
  273. 'address': coinaddr,
  274. 'spendable': False,
  275. 'scriptPubKey': '{}{}{}'.format(s_beg,coinaddr.hex,s_end),
  276. 'confirmations': getrandnum(3) // 2 # max: 8388608 (7 digits)
  277. }
  278. return ret
  279. def _create_fake_unspent_data(self,adata,tx_data,non_mmgen_input='',non_mmgen_input_compressed=True):
  280. out = []
  281. for d in tx_data.values():
  282. al = adata.addrlist(al_id=d['al_id'])
  283. for n,(idx,coinaddr) in enumerate(al.addrpairs()):
  284. lbl = get_label(do_shuffle=True)
  285. out.append(self._create_fake_unspent_entry(coinaddr,d['al_id'],idx,lbl,segwit=d['segwit']))
  286. if n == 0: # create a duplicate address. This means addrs_per_wallet += 1
  287. out.append(self._create_fake_unspent_entry(coinaddr,d['al_id'],idx,lbl,segwit=d['segwit']))
  288. if non_mmgen_input:
  289. from mmgen.obj import PrivKey
  290. privkey = PrivKey(os.urandom(32),compressed=non_mmgen_input_compressed,pubkey_type='std')
  291. from mmgen.addr import AddrGenerator,KeyGenerator
  292. rand_coinaddr = AddrGenerator('p2pkh').to_addr(KeyGenerator('std').to_pubhex(privkey))
  293. of = joinpath(self.cfgs[non_mmgen_input]['tmpdir'],non_mmgen_fn)
  294. write_data_to_file(of, privkey.wif+'\n','compressed {} key'.format(g.proto.name),
  295. quiet=True,ignore_opt_outdir=True)
  296. out.append(self._create_fake_unspent_entry(rand_coinaddr,non_mmgen=True,segwit=False))
  297. return out
  298. def _create_tx_data(self,sources,addrs_per_wallet=addrs_per_wallet):
  299. from mmgen.addr import AddrData,AddrList
  300. from mmgen.obj import AddrIdxList
  301. tx_data,ad = {},AddrData()
  302. for s in sources:
  303. afile = get_file_with_ext(self.cfgs[s]['tmpdir'],'addrs')
  304. al = AddrList(afile)
  305. ad.add(al)
  306. aix = AddrIdxList(fmt_str=self.cfgs[s]['addr_idx_list'])
  307. if len(aix) != addrs_per_wallet:
  308. raise TestSuiteFatalException(
  309. 'Address index list length != {}: {}'.format(addrs_per_wallet,repr(aix)))
  310. tx_data[s] = {
  311. 'addrfile': afile,
  312. 'chk': al.chksum,
  313. 'al_id': al.al_id,
  314. 'addr_idxs': aix[-2:],
  315. 'segwit': self.cfgs[s]['segwit']
  316. }
  317. return ad,tx_data
  318. def _make_txcreate_cmdline(self,tx_data):
  319. from mmgen.obj import PrivKey
  320. privkey = PrivKey(os.urandom(32),compressed=True,pubkey_type='std')
  321. t = ('p2pkh','segwit')['S' in g.proto.mmtypes]
  322. from mmgen.addr import AddrGenerator,KeyGenerator
  323. rand_coinaddr = AddrGenerator(t).to_addr(KeyGenerator('std').to_pubhex(privkey))
  324. # total of two outputs must be < 10 BTC (<1000 LTC)
  325. mods = {'btc':(6,4),'bch':(6,4),'ltc':(600,400)}[g.coin.lower()]
  326. for k in self.cfgs:
  327. self.cfgs[k]['amts'] = [None,None]
  328. for idx,mod in enumerate(mods):
  329. self.cfgs[k]['amts'][idx] = '{}.{}'.format(getrandnum(4) % mod, str(getrandnum(4))[:5])
  330. cmd_args = ['--outdir='+self.tmpdir]
  331. for num in tx_data:
  332. s = tx_data[num]
  333. cmd_args += [
  334. '{}:{},{}'.format(s['al_id'],s['addr_idxs'][0],self.cfgs[num]['amts'][0]),
  335. ]
  336. # + one change address and one BTC address
  337. if num is list(tx_data.keys())[-1]:
  338. cmd_args += ['{}:{}'.format(s['al_id'],s['addr_idxs'][1])]
  339. cmd_args += ['{},{}'.format(rand_coinaddr,self.cfgs[num]['amts'][1])]
  340. return cmd_args + [tx_data[num]['addrfile'] for num in tx_data]
  341. def txcreate_common(self,
  342. sources=['1'],
  343. non_mmgen_input='',
  344. do_label=False,
  345. txdo_args=[],
  346. add_args=[],
  347. view='n',
  348. addrs_per_wallet=addrs_per_wallet,
  349. non_mmgen_input_compressed=True,
  350. cmdline_inputs=False):
  351. if opt.verbose or opt.exact_output:
  352. sys.stderr.write(green('Generating fake tracking wallet info\n'))
  353. silence()
  354. ad,tx_data = self._create_tx_data(sources,addrs_per_wallet)
  355. dfake = self._create_fake_unspent_data(ad,tx_data,non_mmgen_input,non_mmgen_input_compressed)
  356. self._write_fake_data_to_file(repr(dfake))
  357. cmd_args = self._make_txcreate_cmdline(tx_data)
  358. if cmdline_inputs:
  359. from mmgen.tx import TwLabel
  360. cmd_args = ['--inputs={},{},{},{},{},{}'.format(
  361. TwLabel(dfake[0][self.lbl_id]).mmid,dfake[1]['address'],
  362. TwLabel(dfake[2][self.lbl_id]).mmid,dfake[3]['address'],
  363. TwLabel(dfake[4][self.lbl_id]).mmid,dfake[5]['address']
  364. ),'--outdir='+self.tr.trash_dir] + cmd_args[1:]
  365. end_silence()
  366. if opt.verbose or opt.exact_output: sys.stderr.write('\n')
  367. t = self.spawn(
  368. 'mmgen-'+('txcreate','txdo')[bool(txdo_args)],
  369. ([],['--rbf'])[g.proto.cap('rbf')] +
  370. ['-f',self.tx_fee,'-B'] + add_args + cmd_args + txdo_args)
  371. if t.expect([('Get','Transac')[cmdline_inputs],'Unable to connect to \S+'],regex=True) == 1:
  372. raise TestSuiteException('\n'+t.p.after)
  373. if cmdline_inputs:
  374. t.written_to_file('tion')
  375. return t
  376. t.license()
  377. if txdo_args and add_args: # txdo4
  378. t.do_decrypt_ka_data(hp='1',pw=self.cfgs['14']['kapasswd'])
  379. for num in tx_data:
  380. t.expect_getend('ting address data from file ')
  381. chk=t.expect_getend(r'Checksum for address data .*?: ',regex=True)
  382. verify_checksum_or_exit(tx_data[num]['chk'],chk)
  383. # not in tracking wallet warning, (1 + num sources) times
  384. for num in range(len(tx_data) + 1):
  385. t.expect('Continue anyway? (y/N): ','y')
  386. outputs_list = [(addrs_per_wallet+1)*i + 1 for i in range(len(tx_data))]
  387. if non_mmgen_input: outputs_list.append(len(tx_data)*(addrs_per_wallet+1) + 1)
  388. self.txcreate_ui_common(t,
  389. menu=(['M'],['M','D','m','g'])[self.test_name=='txcreate'],
  390. inputs=' '.join(map(str,outputs_list)),
  391. add_comment=('',tx_label_lat_cyr_gr)[do_label],
  392. non_mmgen_inputs=(0,1)[bool(non_mmgen_input and not txdo_args)],
  393. view=view)
  394. return t
  395. def txcreate(self,addrfile):
  396. return self.txcreate_common(sources=['1'],add_args=['--vsize-adj=1.01'])
  397. def txbump(self,txfile,prepend_args=[],seed_args=[]):
  398. if not g.proto.cap('rbf'):
  399. msg('Skipping RBF'); return 'skip'
  400. args = prepend_args + ['--quiet','--outdir='+self.tmpdir,txfile] + seed_args
  401. t = self.spawn('mmgen-txbump',args)
  402. if seed_args:
  403. t.do_decrypt_ka_data(hp='1',pw=self.cfgs['14']['kapasswd'])
  404. t.expect('deduct the fee from (Hit ENTER for the change output): ','1\n')
  405. # Fee must be > tx_fee + network relay fee (currently 0.00001)
  406. t.expect('OK? (Y/n): ','\n')
  407. t.expect('Enter transaction fee: ',self.txbump_fee+'\n')
  408. t.expect('OK? (Y/n): ','\n')
  409. if seed_args: # sign and send
  410. t.do_comment(False,has_label=True)
  411. for cnum,wcls in (('1',IncogWallet),('3',Wallet),('4',Wallet)):
  412. t.passphrase(wcls.desc,self.cfgs[cnum]['wpasswd'])
  413. self._do_confirm_send(t,quiet=not g.debug,confirm_send=True)
  414. if g.debug:
  415. t.written_to_file('Transaction')
  416. else:
  417. t.do_comment(False)
  418. t.expect('Save transaction? (y/N): ','y')
  419. t.written_to_file('Transaction')
  420. os.unlink(txfile) # our tx file replaces the original
  421. cmd = 'touch ' + joinpath(self.tmpdir,'txbump')
  422. os.system(cmd)
  423. return t
  424. def txsend(self,sigfile,bogus_send=True,extra_opts=[]):
  425. if not bogus_send: os.environ['MMGEN_BOGUS_SEND'] = ''
  426. t = self.spawn('mmgen-txsend', extra_opts + ['-d',self.tmpdir,sigfile])
  427. if not bogus_send: os.environ['MMGEN_BOGUS_SEND'] = '1'
  428. self.txsend_ui_common(t,view='t',add_comment='')
  429. return t
  430. def txdo(self,addrfile,wallet):
  431. t = self.txcreate_common(sources=['1'],txdo_args=[wallet])
  432. self.txsign_ui_common(t,view='n',do_passwd=True)
  433. self.txsend_ui_common(t)
  434. return t
  435. def _walletconv_export(self,wf,uargs=[],out_fmt='w',pf=None):
  436. opts = ['-d',self.tmpdir,'-o',out_fmt] + uargs + \
  437. ([],[wf])[bool(wf)] + ([],['-P',pf])[bool(pf)]
  438. t = self.spawn('mmgen-walletconv',opts)
  439. t.license()
  440. if not pf:
  441. icls = SeedSource.ext_to_type(get_extension(wf))
  442. t.passphrase(icls.desc,self.wpasswd)
  443. ocls = SeedSource.fmt_code_to_type(out_fmt)
  444. out_pw = issubclass(ocls,SeedSourceEnc) and ocls != Brainwallet
  445. if out_pw:
  446. t.passphrase_new('new '+ocls.desc,self.wpasswd)
  447. t.usr_rand(self.usr_rand_chars)
  448. if ocls.__name__.startswith('Incog'):
  449. m = 'Encrypting OS random data with key'
  450. t.expect(m)
  451. t.expect(m)
  452. incog_id = t.expect_getend('New Incog Wallet ID: ')
  453. t.expect(m)
  454. if ocls == IncogWalletHidden:
  455. self.write_to_tmpfile(incog_id_fn,incog_id)
  456. t.hincog_create(hincog_bytes)
  457. elif ocls == Wallet:
  458. t.label()
  459. return t.written_to_file(capfirst(ocls.desc),oo=True),t
  460. def export_seed(self,wf,out_fmt='seed',pf=None):
  461. f,t = self._walletconv_export(wf,out_fmt=out_fmt,pf=pf)
  462. silence()
  463. wcls = SeedSource.fmt_code_to_type(out_fmt)
  464. msg('{}: {}'.format(capfirst(wcls.desc),cyan(get_data_from_file(f,wcls.desc))))
  465. end_silence()
  466. return t
  467. def export_hex(self,wf,out_fmt='mmhex',pf=None):
  468. return self.export_seed(wf,out_fmt=out_fmt,pf=pf)
  469. def export_mnemonic(self,wf):
  470. return self.export_seed(wf,out_fmt='words')
  471. def export_bip39(self,wf):
  472. return self.export_seed(wf,out_fmt='bip39')
  473. def export_incog(self,wf,out_fmt='i',add_args=[]):
  474. uargs = ['-p1',self.usr_rand_arg] + add_args
  475. f,t = self._walletconv_export(wf,out_fmt=out_fmt,uargs=uargs)
  476. return t
  477. def export_incog_hex(self,wf):
  478. return self.export_incog(wf,out_fmt='xi')
  479. # TODO: make outdir and hidden incog compatible (ignore --outdir and warn user?)
  480. def export_incog_hidden(self,wf):
  481. rf = joinpath(self.tmpdir,hincog_fn)
  482. add_args = ['-J','{},{}'.format(rf,hincog_offset)]
  483. return self.export_incog(wf,out_fmt='hi',add_args=add_args)
  484. def addrgen_seed(self,wf,foo,in_fmt='seed'):
  485. wcls = SeedSource.fmt_code_to_type(in_fmt)
  486. stdout = wcls == MMGenSeedFile # capture output to screen once
  487. add_args = ([],['-S'])[bool(stdout)] + self.segwit_arg
  488. t = self.spawn('mmgen-addrgen', add_args +
  489. ['-i'+in_fmt,'-d',self.tmpdir,wf,self.addr_idx_list])
  490. t.license()
  491. t.expect_getend('Valid {} for Seed ID '.format(wcls.desc))
  492. vmsg('Comparing generated checksum with checksum from previous address file')
  493. chk = t.expect_getend(r'Checksum for address data .*?: ',regex=True)
  494. if stdout: t.read()
  495. verify_checksum_or_exit(self._get_addrfile_checksum(),chk)
  496. if in_fmt != 'seed':
  497. t.no_overwrite()
  498. t.req_exit_val = 1
  499. return t
  500. def addrgen_hex(self,wf,foo,in_fmt='mmhex'):
  501. return self.addrgen_seed(wf,foo,in_fmt=in_fmt)
  502. def addrgen_mnemonic(self,wf,foo):
  503. return self.addrgen_seed(wf,foo,in_fmt='words')
  504. def addrgen_incog(self,wf=[],foo='',in_fmt='i',args=[]):
  505. t = self.spawn('mmgen-addrgen', args + self.segwit_arg + ['-i'+in_fmt,'-d',self.tmpdir]+
  506. ([],[wf])[bool(wf)] + [self.addr_idx_list])
  507. t.license()
  508. t.expect_getend('Incog Wallet ID: ')
  509. wcls = SeedSource.fmt_code_to_type(in_fmt)
  510. t.hash_preset(wcls.desc,'1')
  511. t.passphrase('{} \w{{8}}'.format(wcls.desc),self.wpasswd)
  512. vmsg('Comparing generated checksum with checksum from address file')
  513. chk = t.expect_getend(r'Checksum for address data .*?: ',regex=True)
  514. verify_checksum_or_exit(self._get_addrfile_checksum(),chk)
  515. t.no_overwrite()
  516. t.req_exit_val = 1
  517. return t
  518. def addrgen_incog_hex(self,wf,foo):
  519. return self.addrgen_incog(wf,'',in_fmt='xi')
  520. def addrgen_incog_hidden(self,wf,foo):
  521. rf = joinpath(self.tmpdir,hincog_fn)
  522. return self.addrgen_incog([],'',in_fmt='hi',
  523. args=['-H','{},{}'.format(rf,hincog_offset),'-l',str(hincog_seedlen)])
  524. def txsign_keyaddr(self,keyaddr_file,txfile):
  525. t = self.spawn('mmgen-txsign', ['-d',self.tmpdir,'-p1','-M',keyaddr_file,txfile])
  526. t.license()
  527. t.do_decrypt_ka_data(hp='1',pw=self.kapasswd)
  528. t.view_tx('n')
  529. self.txsign_end(t)
  530. return t
  531. def txcreate_ni(self,addrfile):
  532. return self.txcreate_common(sources=['1'],cmdline_inputs=True,add_args=['--yes'])
  533. def walletgen2(self,del_dw_run='dummy'):
  534. return self.walletgen(seed_len=128)
  535. def addrgen2(self,wf):
  536. return self.addrgen(wf,pf='')
  537. def txcreate2(self,addrfile):
  538. return self.txcreate_common(sources=['2'])
  539. def txsign2(self,wf1,txf1,wf2,txf2):
  540. t = self.spawn('mmgen-txsign', ['-d',self.tmpdir,txf1,wf1,txf2,wf2])
  541. t.license()
  542. for cnum,wf in (('1',wf1),('2',wf2)):
  543. wcls = SeedSource.ext_to_type(get_extension(wf))
  544. t.view_tx('n')
  545. t.passphrase(wcls.desc,self.cfgs[cnum]['wpasswd'])
  546. self.txsign_end(t,cnum)
  547. return t
  548. def export_mnemonic2(self,wf):
  549. return self.export_mnemonic(wf)
  550. def walletgen3(self,del_dw_run='dummy'):
  551. return self.walletgen()
  552. def addrgen3(self,wf):
  553. return self.addrgen(wf,pf='')
  554. def txcreate3(self,addrfile1,addrfile2):
  555. return self.txcreate_common(sources=['1','3'])
  556. def txsign3(self,wf1,wf2,txf2):
  557. t = self.spawn('mmgen-txsign', ['-d',self.tmpdir,wf1,wf2,txf2])
  558. t.license()
  559. t.view_tx('n')
  560. for cnum,wf in (('1',wf1),('3',wf2)):
  561. wcls = SeedSource.ext_to_type(get_extension(wf))
  562. t.passphrase(wcls.desc,self.cfgs[cnum]['wpasswd'])
  563. self.txsign_end(t)
  564. return t
  565. walletgen14 = walletgen
  566. addrgen14 = TestSuiteShared.addrgen
  567. keyaddrgen14 = TestSuiteShared.keyaddrgen
  568. def walletgen4(self,del_dw_run='dummy'):
  569. bwf = joinpath(self.tmpdir,self.bw_filename)
  570. make_brainwallet_file(bwf)
  571. seed_len = str(self.seed_len)
  572. args = ['-d',self.tmpdir,'-p1',self.usr_rand_arg,'-l'+seed_len,'-ib']
  573. t = self.spawn('mmgen-walletconv', args + [bwf])
  574. t.license()
  575. wcls = Wallet
  576. t.passphrase_new('new ' +wcls.desc,self.wpasswd)
  577. t.usr_rand(self.usr_rand_chars)
  578. t.label()
  579. t.written_to_file(capfirst(wcls.desc))
  580. return t
  581. def addrgen4(self,wf):
  582. return self.addrgen(wf,pf='')
  583. def txcreate4(self,f1,f2,f3,f4,f5,f6):
  584. return self.txcreate_common(sources=['1','2','3','4','14'],non_mmgen_input='4',do_label=True,view='y')
  585. def txsign4(self,f1,f2,f3,f4,f5,f6):
  586. non_mm_file = joinpath(self.tmpdir,non_mmgen_fn)
  587. a = ['-d',self.tmpdir,'-i','brain','-b'+self.bw_params,'-p1','-k',non_mm_file,'-M',f6,f1,f2,f3,f4,f5]
  588. t = self.spawn('mmgen-txsign',a)
  589. t.license()
  590. t.do_decrypt_ka_data(hp='1',pw=self.cfgs['14']['kapasswd'])
  591. t.view_tx('t')
  592. for cnum,wcls in (('1',IncogWallet),('3',Wallet)):
  593. t.passphrase('{}'.format(wcls.desc),self.cfgs[cnum]['wpasswd'])
  594. self.txsign_end(t,has_label=True)
  595. return t
  596. def txdo4(self,f1,f2,f3,f4,f5,f6,f7,f8,f9,f10,f11,f12):
  597. non_mm_file = joinpath(self.tmpdir,non_mmgen_fn)
  598. add_args = ['-d',self.tmpdir,'-i','brain','-b'+self.bw_params,'-p1','-k',non_mm_file,'-M',f12]
  599. self.get_file_with_ext('sigtx',delete_all=True) # delete tx signed by txsign4
  600. t = self.txcreate_common(sources=['1','2','3','4','14'],
  601. non_mmgen_input='4',do_label=True,txdo_args=[f7,f8,f9,f10],add_args=add_args)
  602. for cnum,wcls in (('1',IncogWallet),('3',Wallet)):
  603. t.passphrase('{}'.format(wcls.desc),self.cfgs[cnum]['wpasswd'])
  604. self.txsign_ui_common(t)
  605. self.txsend_ui_common(t)
  606. cmd = 'touch ' + joinpath(self.tmpdir,'txdo')
  607. os.system(cmd)
  608. return t
  609. def txbump4(self,f1,f2,f3,f4,f5,f6,f7,f8,f9): # f7:txfile,f9:'txdo'
  610. non_mm_file = joinpath(self.tmpdir,non_mmgen_fn)
  611. return self.txbump(f7,prepend_args=['-p1','-k',non_mm_file,'-M',f1],seed_args=[f2,f3,f4,f5,f6,f8])
  612. def walletgen5(self,del_dw_run='dummy'):
  613. return self.walletgen()
  614. def addrgen5(self,wf):
  615. return self.addrgen(wf,pf='')
  616. def txcreate5(self,addrfile):
  617. return self.txcreate_common(sources=['20'],non_mmgen_input='20',non_mmgen_input_compressed=False)
  618. def txsign5(self,wf,txf,bad_vsize=True,add_args=[]):
  619. non_mm_file = joinpath(self.tmpdir,non_mmgen_fn)
  620. t = self.spawn('mmgen-txsign', add_args + ['-d',self.tmpdir,'-k',non_mm_file,txf,wf])
  621. t.license()
  622. t.view_tx('n')
  623. wcls = SeedSource.ext_to_type(get_extension(wf))
  624. t.passphrase(wcls.desc,self.cfgs['20']['wpasswd'])
  625. if bad_vsize:
  626. t.expect('Estimated transaction vsize')
  627. t.expect('1 transaction could not be signed')
  628. exit_val = 2
  629. else:
  630. t.do_comment(False)
  631. t.expect('Save signed transaction? (Y/n): ','y')
  632. exit_val = 0
  633. t.read()
  634. t.req_exit_val = exit_val
  635. return t
  636. def walletgen6(self,del_dw_run='dummy'):
  637. return self.walletgen()
  638. def addrgen6(self,wf):
  639. return self.addrgen(wf,pf='')
  640. def txcreate6(self,addrfile):
  641. return self.txcreate_common(
  642. sources=['21'],non_mmgen_input='21',non_mmgen_input_compressed=False,add_args=['--vsize-adj=1.08'])
  643. def txsign6(self,txf,wf):
  644. return self.txsign5(txf,wf,bad_vsize=False,add_args=['--vsize-adj=1.08'])