test-oldscripts.py 51 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535
  1. #!/usr/bin/python
  2. # Chdir to repo root.
  3. # Since script is not in repo root, fix sys.path so that modules are
  4. # imported from repo, not system.
  5. import sys,os
  6. pn = os.path.dirname(sys.argv[0])
  7. os.chdir(os.path.join(pn,os.pardir))
  8. sys.path.__setitem__(0,os.path.abspath(os.curdir))
  9. import mmgen.globalvars as g
  10. import mmgen.opt as opt
  11. from mmgen.util import mmsg,mdie,Msg,die,capfirst
  12. from mmgen.test import *
  13. hincog_fn = "rand_data"
  14. hincog_bytes = 1024*1024
  15. hincog_offset = 98765
  16. hincog_seedlen = 256
  17. incog_id_fn = "incog_id"
  18. non_mmgen_fn = "btckey"
  19. ref_dir = os.path.join("test","ref")
  20. ref_wallet_brainpass = "abc"
  21. ref_wallet_hash_preset = "1"
  22. ref_wallet_incog_offset = 123
  23. ref_bw_hash_preset = "1"
  24. ref_bw_file = "brainwallet"
  25. ref_bw_file_spc = "brainwallet-spaced"
  26. ref_kafile_pass = "kafile password"
  27. ref_kafile_hash_preset = "1"
  28. ref_enc_fn = "sample-text.mmenc"
  29. tool_enc_passwd = "Scrypt it, don't hash it!"
  30. sample_text = \
  31. "The Times 03/Jan/2009 Chancellor on brink of second bailout for banks\n"
  32. cfgs = {
  33. '6': {
  34. 'name': "reference wallet check (128-bit)",
  35. 'seed_len': 128,
  36. 'seed_id': "FE3C6545",
  37. 'ref_bw_seed_id': "33F10310",
  38. 'addrfile_chk': "B230 7526 638F 38CB 8FDC 8B76",
  39. 'keyaddrfile_chk': "CF83 32FB 8A8B 08E2 0F00 D601",
  40. 'wpasswd': "reference password",
  41. 'ref_wallet': "FE3C6545-D782B529[128,1].mmdat",
  42. 'ic_wallet': "FE3C6545-E29303EA-5E229E30[128,1].mmincog",
  43. 'ic_wallet_hex': "FE3C6545-BC4BE3F2-32586837[128,1].mmincox",
  44. 'hic_wallet': "FE3C6545-161E495F-BEB7548E[128:1].incog-offset123",
  45. 'hic_wallet_old': "FE3C6545-161E495F-9860A85B[128:1].incog-old.offset123",
  46. 'tmpdir': os.path.join("test","tmp6"),
  47. 'kapasswd': "",
  48. 'addr_idx_list': "1010,500-501,31-33,1,33,500,1011", # 8 addresses
  49. 'dep_generators': {
  50. 'mmdat': "refwalletgen1",
  51. 'addrs': "refaddrgen1",
  52. 'akeys.mmenc': "refkeyaddrgen1"
  53. },
  54. },
  55. '7': {
  56. 'name': "reference wallet check (192-bit)",
  57. 'seed_len': 192,
  58. 'seed_id': "1378FC64",
  59. 'ref_bw_seed_id': "CE918388",
  60. 'addrfile_chk': "8C17 A5FA 0470 6E89 3A87 8182",
  61. 'keyaddrfile_chk': "9648 5132 B98E 3AD9 6FC3 C5AD",
  62. 'wpasswd': "reference password",
  63. 'ref_wallet': "1378FC64-6F0F9BB4[192,1].mmdat",
  64. 'ic_wallet': "1378FC64-2907DE97-F980D21F[192,1].mmincog",
  65. 'ic_wallet_hex': "1378FC64-4DCB5174-872806A7[192,1].mmincox",
  66. 'hic_wallet': "1378FC64-B55E9958-77256FC1[192:1].incog.offset123",
  67. 'hic_wallet_old': "1378FC64-B55E9958-D85FF20C[192:1].incog-old.offset123",
  68. 'tmpdir': os.path.join("test","tmp7"),
  69. 'kapasswd': "",
  70. 'addr_idx_list': "1010,500-501,31-33,1,33,500,1011", # 8 addresses
  71. 'dep_generators': {
  72. 'mmdat': "refwalletgen2",
  73. 'addrs': "refaddrgen2",
  74. 'akeys.mmenc': "refkeyaddrgen2"
  75. },
  76. },
  77. '8': {
  78. 'name': "reference wallet check (256-bit)",
  79. 'seed_len': 256,
  80. 'seed_id': "98831F3A",
  81. 'ref_bw_seed_id': "B48CD7FC",
  82. 'addrfile_chk': "6FEF 6FB9 7B13 5D91 854A 0BD3",
  83. 'keyaddrfile_chk': "9F2D D781 1812 8BAD C396 9DEB",
  84. 'wpasswd': "reference password",
  85. 'ref_wallet': "98831F3A-27F2BF93[256,1].mmdat",
  86. 'ref_addrfile': "98831F3A[1,31-33,500-501,1010-1011].addrs",
  87. 'ref_keyaddrfile': "98831F3A[1,31-33,500-501,1010-1011].akeys.mmenc",
  88. 'ref_addrfile_chksum': "6FEF 6FB9 7B13 5D91 854A 0BD3",
  89. 'ref_keyaddrfile_chksum': "9F2D D781 1812 8BAD C396 9DEB",
  90. # 'ref_fake_unspent_data':"98831F3A_unspent.json",
  91. 'ref_tx_file': "tx_FFB367[1.234].raw",
  92. 'ic_wallet': "98831F3A-5482381C-18460FB1[256,1].mmincog",
  93. 'ic_wallet_hex': "98831F3A-1630A9F2-870376A9[256,1].mmincox",
  94. 'hic_wallet': "98831F3A-F59B07A0-559CEF19[256:1].incog.offset123",
  95. 'hic_wallet_old': "98831F3A-F59B07A0-848535F3[256:1].incog-old.offset123",
  96. 'tmpdir': os.path.join("test","tmp8"),
  97. 'kapasswd': "",
  98. 'addr_idx_list': "1010,500-501,31-33,1,33,500,1011", # 8 addresses
  99. 'dep_generators': {
  100. 'mmdat': "refwalletgen3",
  101. 'addrs': "refaddrgen3",
  102. 'akeys.mmenc': "refkeyaddrgen3"
  103. },
  104. },
  105. '1': {
  106. 'tmpdir': os.path.join("test","tmp1"),
  107. 'wpasswd': "Dorian",
  108. 'kapasswd': "Grok the blockchain",
  109. 'addr_idx_list': "12,99,5-10,5,12", # 8 addresses
  110. 'dep_generators': {
  111. 'mmdat': "walletgen",
  112. 'addrs': "addrgen",
  113. 'raw': "txcreate",
  114. 'sig': "txsign",
  115. 'mmwords': "export_mnemonic",
  116. 'mmseed': "export_seed",
  117. 'mmincog': "export_incog",
  118. 'mmincox': "export_incog_hex",
  119. hincog_fn: "export_incog_hidden",
  120. incog_id_fn: "export_incog_hidden",
  121. 'akeys.mmenc': "keyaddrgen"
  122. },
  123. },
  124. '2': {
  125. 'tmpdir': os.path.join("test","tmp2"),
  126. 'wpasswd': "Hodling away",
  127. 'addr_idx_list': "37,45,3-6,22-23", # 8 addresses
  128. 'seed_len': 128,
  129. 'dep_generators': {
  130. 'mmdat': "walletgen2",
  131. 'addrs': "addrgen2",
  132. 'raw': "txcreate2",
  133. 'sig': "txsign2",
  134. 'mmwords': "export_mnemonic2",
  135. },
  136. },
  137. '3': {
  138. 'tmpdir': os.path.join("test","tmp3"),
  139. 'wpasswd': "Major miner",
  140. 'addr_idx_list': "73,54,1022-1023,2-5", # 8 addresses
  141. 'dep_generators': {
  142. 'mmdat': "walletgen3",
  143. 'addrs': "addrgen3",
  144. 'raw': "txcreate3",
  145. 'sig': "txsign3"
  146. },
  147. },
  148. '4': {
  149. 'tmpdir': os.path.join("test","tmp4"),
  150. 'wpasswd': "Hashrate rising",
  151. 'addr_idx_list': "63,1004,542-544,7-9", # 8 addresses
  152. 'seed_len': 192,
  153. 'dep_generators': {
  154. 'mmdat': "walletgen4",
  155. 'mmbrain': "walletgen4",
  156. 'addrs': "addrgen4",
  157. 'raw': "txcreate4",
  158. 'sig': "txsign4",
  159. },
  160. 'bw_filename': "brainwallet.mmbrain",
  161. 'bw_params': "192,1",
  162. },
  163. '5': {
  164. 'tmpdir': os.path.join("test","tmp5"),
  165. 'wpasswd': "My changed password",
  166. 'dep_generators': {
  167. 'mmdat': "passchg",
  168. },
  169. },
  170. '9': {
  171. 'tmpdir': os.path.join("test","tmp9"),
  172. 'tool_enc_infn': "tool_encrypt.in",
  173. # 'tool_enc_ref_infn': "tool_encrypt_ref.in",
  174. 'dep_generators': {
  175. 'tool_encrypt.in': "tool_encrypt",
  176. 'tool_encrypt.in.mmenc': "tool_encrypt",
  177. # 'tool_encrypt_ref.in': "tool_encrypt_ref",
  178. # 'tool_encrypt_ref.in.mmenc': "tool_encrypt_ref",
  179. },
  180. },
  181. }
  182. from copy import deepcopy
  183. for a,b in ('6','11'),('7','12'),('8','13'):
  184. cfgs[b] = deepcopy(cfgs[a])
  185. cfgs[b]['tmpdir'] = os.path.join("test","tmp"+b)
  186. from collections import OrderedDict
  187. cmd_data = OrderedDict([
  188. # test description depends
  189. ['walletgen', (1,'wallet generation', [[[],1]])],
  190. # ['walletchk', (1,'wallet check', [[["mmdat"],1]])],
  191. ['passchg', (5,'password, label and hash preset change',[[["mmdat"],1]])],
  192. ['walletchk_newpass',(5,'wallet check with new pw, label and hash preset',[[["mmdat"],5]])],
  193. ['addrgen', (1,'address generation', [[["mmdat"],1]])],
  194. ['addrimport', (1,'address import', [[["addrs"],1]])],
  195. ['txcreate', (1,'transaction creation', [[["addrs"],1]])],
  196. ['txsign', (1,'transaction signing', [[["mmdat","raw"],1]])],
  197. ['txsend', (1,'transaction sending', [[["sig"],1]])],
  198. ['export_seed', (1,'seed export to mmseed format', [[["mmdat"],1]])],
  199. ['export_mnemonic', (1,'seed export to mmwords format', [[["mmdat"],1]])],
  200. ['export_incog', (1,'seed export to mmincog format', [[["mmdat"],1]])],
  201. ['export_incog_hex',(1,'seed export to mmincog hex format', [[["mmdat"],1]])],
  202. ['export_incog_hidden',(1,'seed export to hidden mmincog format', [[["mmdat"],1]])],
  203. ['addrgen_seed', (1,'address generation from mmseed file', [[["mmseed","addrs"],1]])],
  204. ['addrgen_mnemonic',(1,'address generation from mmwords file',[[["mmwords","addrs"],1]])],
  205. ['addrgen_incog', (1,'address generation from mmincog file',[[["mmincog","addrs"],1]])],
  206. ['addrgen_incog_hex',(1,'address generation from mmincog hex file',[[["mmincox","addrs"],1]])],
  207. ['addrgen_incog_hidden',(1,'address generation from hidden mmincog file', [[[hincog_fn,"addrs"],1]])],
  208. ['keyaddrgen', (1,'key-address file generation', [[["mmdat"],1]])],
  209. ['txsign_keyaddr',(1,'transaction signing with key-address file', [[["akeys.mmenc","raw"],1]])],
  210. ['walletgen2',(2,'wallet generation (2), 128-bit seed', [])],
  211. ['addrgen2', (2,'address generation (2)', [[["mmdat"],2]])],
  212. ['txcreate2', (2,'transaction creation (2)', [[["addrs"],2]])],
  213. ['txsign2', (2,'transaction signing, two transactions',[[["mmdat","raw"],1],[["mmdat","raw"],2]])],
  214. ['export_mnemonic2', (2,'seed export to mmwords format (2)',[[["mmdat"],2]])],
  215. ['walletgen3',(3,'wallet generation (3)', [])],
  216. ['addrgen3', (3,'address generation (3)', [[["mmdat"],3]])],
  217. ['txcreate3', (3,'tx creation with inputs and outputs from two wallets', [[["addrs"],1],[["addrs"],3]])],
  218. ['txsign3', (3,'tx signing with inputs and outputs from two wallets',[[["mmdat"],1],[["mmdat","raw"],3]])],
  219. ['walletgen4',(4,'wallet generation (4) (brainwallet)', [])],
  220. ['addrgen4', (4,'address generation (4)', [[["mmdat"],4]])],
  221. ['txcreate4', (4,'tx creation with inputs and outputs from four seed sources, plus non-MMGen inputs and outputs', [[["addrs"],1],[["addrs"],2],[["addrs"],3],[["addrs"],4]])],
  222. ['txsign4', (4,'tx signing with inputs and outputs from incog file, mnemonic file, wallet and brainwallet, plus non-MMGen inputs and outputs', [[["mmincog"],1],[["mmwords"],2],[["mmdat"],3],[["mmbrain","raw"],4]])],
  223. ['tool_encrypt', (9,"'mmgen-tool encrypt' (random data)", [])],
  224. ['tool_decrypt', (9,"'mmgen-tool decrypt' (random data)", [[[cfgs['9']['tool_enc_infn'],cfgs['9']['tool_enc_infn']+".mmenc"],9]])],
  225. # ['tool_encrypt_ref', (9,"'mmgen-tool encrypt' (reference text)", [])],
  226. ['tool_find_incog_data', (9,"'mmgen-tool find_incog_data'", [[[hincog_fn],1],[[incog_id_fn],1]])],
  227. ])
  228. # saved reference data
  229. cmd_data_ref = (
  230. # reading
  231. ('ref_wallet_chk', ([],'saved reference wallet')),
  232. ('ref_seed_chk', ([],'saved seed file')),
  233. ('ref_mn_chk', ([],'saved mnemonic file')),
  234. ('ref_hincog_chk', ([],'saved hidden incog reference wallet')),
  235. ('ref_brain_chk', ([],'saved brainwallet')),
  236. # generating new reference ('abc' brainwallet) files:
  237. ('refwalletgen', ([],'gen new refwallet')),
  238. ('refaddrgen', (["mmdat"],'new refwallet addr chksum')),
  239. ('refkeyaddrgen', (["mmdat"],'new refwallet key-addr chksum'))
  240. )
  241. # misc. saved reference data
  242. cmd_data_ref_other = (
  243. ('ref_addrfile_chk', 'saved reference address file'),
  244. ('ref_keyaddrfile_chk','saved reference key-address file'),
  245. # Create the fake inputs:
  246. # ('txcreate8', 'transaction creation (8)'),
  247. ('ref_tx_chk', 'saved reference tx file'),
  248. ('ref_brain_chk_spc3', 'saved brainwallet (non-standard spacing)'),
  249. ('ref_tool_decrypt', 'decryption of saved MMGen-encrypted file'),
  250. )
  251. # mmgen-walletconv:
  252. cmd_data_conv_in = ( # reading
  253. ('ref_wallet_conv', 'conversion of saved reference wallet'),
  254. ('ref_mn_conv', 'conversion of saved mnemonic'),
  255. ('ref_seed_conv', 'conversion of saved seed file'),
  256. ('ref_brain_conv', 'conversion of ref brainwallet'),
  257. ('ref_incog_conv', 'conversion of saved incog wallet'),
  258. ('ref_incox_conv', 'conversion of saved hex incog wallet'),
  259. ('ref_hincog_conv', 'conversion of saved hidden incog wallet'),
  260. ('ref_hincog_conv_old','conversion of saved hidden incog wallet (old format)')
  261. )
  262. cmd_data_conv_out = ( # writing
  263. ('ref_wallet_conv_out', 'ref seed conversion to wallet'),
  264. ('ref_mn_conv_out', 'ref seed conversion to mnemonic'),
  265. ('ref_seed_conv_out', 'ref seed conversion to seed'),
  266. ('ref_incog_conv_out', 'ref seed conversion to incog data'),
  267. ('ref_incox_conv_out', 'ref seed conversion to hex incog data'),
  268. ('ref_hincog_conv_out', 'ref seed conversion to hidden incog data')
  269. )
  270. cmd_groups = OrderedDict([
  271. ('main', cmd_data.keys()),
  272. ('ref', [c[0]+str(i) for c in cmd_data_ref for i in (1,2,3)]),
  273. ('ref_other', [c[0] for c in cmd_data_ref_other]),
  274. ('conv_in', [c[0]+str(i) for c in cmd_data_conv_in for i in (1,2,3)]),
  275. ('conv_out', [c[0]+str(i) for c in cmd_data_conv_out for i in (1,2,3)]),
  276. ])
  277. for a,b in cmd_data_ref:
  278. for i,j in (1,128),(2,192),(3,256):
  279. cmd_data[a+str(i)] = (5+i,"%s (%s-bit)" % (b[1],j),[[b[0],5+i]])
  280. for a,b in cmd_data_ref_other:
  281. cmd_data[a] = (8,b,[[[],8]])
  282. for a,b in cmd_data_conv_in:
  283. for i,j in (1,128),(2,192),(3,256):
  284. cmd_data[a+str(i)] = (10+i,"%s (%s-bit)" % (b,j),[[[],10+i]])
  285. for a,b in cmd_data_conv_out:
  286. for i,j in (1,128),(2,192),(3,256):
  287. cmd_data[a+str(i)] = (10+i,"%s (%s-bit)" % (b,j),[[[],10+i]])
  288. utils = {
  289. 'check_deps': 'check dependencies for specified command',
  290. 'clean': 'clean specified tmp dir(s) 1,2,3,4,5 or 6 (no arg = all dirs)',
  291. }
  292. addrs_per_wallet = 8
  293. # total of two outputs must be < 10 BTC
  294. for k in cfgs.keys():
  295. cfgs[k]['amts'] = [0,0]
  296. for idx,mod in (0,6),(1,4):
  297. cfgs[k]['amts'][idx] = "%s.%s" % ((getrandnum(2) % mod), str(getrandnum(4))[:5])
  298. meta_cmds = OrderedDict([
  299. ['ref1', ("refwalletgen1","refaddrgen1","refkeyaddrgen1")],
  300. ['ref2', ("refwalletgen2","refaddrgen2","refkeyaddrgen2")],
  301. ['ref3', ("refwalletgen3","refaddrgen3","refkeyaddrgen3")],
  302. ['gen', ("walletgen","addrgen")],
  303. ['pass', ("passchg","walletchk_newpass")],
  304. ['tx', ("addrimport","txcreate","txsign","txsend")],
  305. ['export', [k for k in cmd_data if k[:7] == "export_" and cmd_data[k][0] == 1]],
  306. ['gen_sp', [k for k in cmd_data if k[:8] == "addrgen_" and cmd_data[k][0] == 1]],
  307. ['online', ("keyaddrgen","txsign_keyaddr")],
  308. ['2', [k for k in cmd_data if cmd_data[k][0] == 2]],
  309. ['3', [k for k in cmd_data if cmd_data[k][0] == 3]],
  310. ['4', [k for k in cmd_data if cmd_data[k][0] == 4]],
  311. ['tool', ("tool_encrypt","tool_decrypt","tool_find_incog_data")],
  312. ['saved_ref1', [c[0]+"1" for c in cmd_data_ref]],
  313. ['saved_ref2', [c[0]+"2" for c in cmd_data_ref]],
  314. ['saved_ref3', [c[0]+"3" for c in cmd_data_ref]],
  315. ['saved_ref_other', [c[0] for c in cmd_data_ref_other]],
  316. ['saved_ref_conv_in1', [c[0]+"1" for c in cmd_data_conv_in]],
  317. ['saved_ref_conv_in2', [c[0]+"2" for c in cmd_data_conv_in]],
  318. ['saved_ref_conv_in3', [c[0]+"3" for c in cmd_data_conv_in]],
  319. ['saved_ref_conv_out1', [c[0]+"1" for c in cmd_data_conv_out]],
  320. ['saved_ref_conv_out2', [c[0]+"2" for c in cmd_data_conv_out]],
  321. ['saved_ref_conv_out3', [c[0]+"3" for c in cmd_data_conv_out]],
  322. ])
  323. opts_data = {
  324. 'desc': "Test suite for the MMGen suite",
  325. 'usage':"[options] [command(s) or metacommand(s)]",
  326. 'options': """
  327. -h, --help Print this help message
  328. -b, --buf-keypress Use buffered keypresses as with real human input
  329. -d, --debug-scripts Turn on debugging output in executed scripts
  330. -D, --direct-exec Bypass pexpect and execute a command directly (for
  331. debugging only)
  332. -e, --exact-output Show the exact output of the MMGen script(s) being run
  333. -l, --list-cmds List and describe the tests and commands in the test suite
  334. -n, --names Display command names instead of descriptions
  335. -p, --pause Pause between tests, resuming on keypress
  336. -q, --quiet Produce minimal output. Suppress dependency info
  337. -s, --system Test scripts and modules installed on system rather than
  338. those in the repo root
  339. -v, --verbose Produce more verbose output
  340. """,
  341. 'notes': """
  342. If no command is given, the whole suite of tests is run.
  343. """
  344. }
  345. cmd_args = opt.opts.init(opts_data)
  346. if opt.system: sys.path.pop(0)
  347. # temporary
  348. os.environ["MMGEN_USE_OLD_SCRIPTS"] = "1"
  349. if opt.debug_scripts: os.environ["MMGEN_DEBUG"] = "1"
  350. if opt.buf_keypress:
  351. send_delay = 0.3
  352. else:
  353. send_delay = 0
  354. os.environ["MMGEN_DISABLE_HOLD_PROTECT"] = "1"
  355. if opt.exact_output:
  356. def msg(s): pass
  357. vmsg = vmsg_r = msg_r = msg
  358. else:
  359. def msg(s): sys.stderr.write(s+"\n")
  360. def vmsg(s):
  361. if opt.verbose: sys.stderr.write(s+"\n")
  362. def msg_r(s): sys.stderr.write(s)
  363. def vmsg_r(s):
  364. if opt.verbose: sys.stderr.write(s)
  365. stderr_save = sys.stderr
  366. def silence():
  367. if not (opt.verbose or opt.exact_output):
  368. sys.stderr = open("/dev/null","a")
  369. def end_silence():
  370. if not (opt.verbose or opt.exact_output):
  371. sys.stderr = stderr_save
  372. def errmsg(s): stderr_save.write(s+"\n")
  373. def errmsg_r(s): stderr_save.write(s)
  374. if opt.list_cmds:
  375. fs = " {:<{w}} - {}"
  376. Msg("AVAILABLE COMMANDS:")
  377. w = max([len(i) for i in cmd_data])
  378. for cmd in cmd_data:
  379. Msg(fs.format(cmd,cmd_data[cmd][1],w=w))
  380. w = max([len(i) for i in meta_cmds])
  381. Msg("\nAVAILABLE METACOMMANDS:")
  382. for cmd in meta_cmds:
  383. Msg(fs.format(cmd," ".join(meta_cmds[cmd]),w=w))
  384. w = max([len(i) for i in cmd_groups.keys()])
  385. Msg("\nAVAILABLE COMMAND GROUPS:")
  386. for g in cmd_groups.keys():
  387. Msg(fs.format(g," ".join(cmd_groups[g]),w=w))
  388. Msg("\nAVAILABLE UTILITIES:")
  389. w = max([len(i) for i in utils])
  390. for cmd in sorted(utils):
  391. Msg(fs.format(cmd,utils[cmd],w=w))
  392. sys.exit()
  393. import pexpect,time,re
  394. from mmgen.util import get_data_from_file,write_to_file,get_lines_from_file
  395. def my_send(p,t,delay=send_delay,s=False):
  396. if delay: time.sleep(delay)
  397. ret = p.send(t) # returns num bytes written
  398. if delay: time.sleep(delay)
  399. if opt.verbose:
  400. ls = "" if opt.debug or not s else " "
  401. es = "" if s else " "
  402. msg("%sSEND %s%s" % (ls,es,yellow("'%s'"%t.replace('\n',r'\n'))))
  403. return ret
  404. def my_expect(p,s,t='',delay=send_delay,regex=False,nonl=False):
  405. quo = "'" if type(s) == str else ""
  406. if opt.verbose: msg_r("EXPECT %s" % yellow(quo+str(s)+quo))
  407. else: msg_r("+")
  408. try:
  409. if s == '': ret = 0
  410. else:
  411. f = p.expect if regex else p.expect_exact
  412. ret = f(s,timeout=3)
  413. except pexpect.TIMEOUT:
  414. errmsg(red("\nERROR. Expect %s%s%s timed out. Exiting" % (quo,s,quo)))
  415. sys.exit(1)
  416. if opt.debug or (opt.verbose and type(s) != str): msg_r(" ==> %s " % ret)
  417. if ret == -1:
  418. errmsg("Error. Expect returned %s" % ret)
  419. sys.exit(1)
  420. else:
  421. if t == '':
  422. if not nonl: vmsg("")
  423. else:
  424. my_send(p,t,delay,s)
  425. return ret
  426. def get_file_with_ext(ext,mydir,delete=True):
  427. flist = [os.path.join(mydir,f) for f in os.listdir(mydir)
  428. if f == ext or f[-(len(ext)+1):] == "."+ext]
  429. if not flist: return False
  430. if len(flist) > 1:
  431. if delete:
  432. if not opt.quiet:
  433. msg("Multiple *.%s files in '%s' - deleting" % (ext,mydir))
  434. for f in flist: os.unlink(f)
  435. return False
  436. else:
  437. return flist[0]
  438. def get_addrfile_checksum(display=False):
  439. addrfile = get_file_with_ext("addrs",cfg['tmpdir'])
  440. silence()
  441. from mmgen.addr import AddrInfo
  442. chk = AddrInfo(addrfile).checksum
  443. if opt.verbose and display: msg("Checksum: %s" % cyan(chk))
  444. end_silence()
  445. return chk
  446. def verify_checksum_or_exit(checksum,chk):
  447. if checksum != chk:
  448. errmsg(red("Checksum error: %s" % chk))
  449. sys.exit(1)
  450. vmsg(green("Checksums match: %s") % (cyan(chk)))
  451. class MMGenExpect(object):
  452. def __init__(self,name,mmgen_cmd,cmd_args=[],extra_desc=""):
  453. if not opt.system:
  454. mmgen_cmd = os.path.join(os.curdir,mmgen_cmd)
  455. desc = (cmd_data[name][1],name)[int(bool(opt.names))]
  456. if extra_desc: desc += " " + extra_desc
  457. if opt.verbose or opt.exact_output:
  458. sys.stderr.write(
  459. green("Testing: %s\nExecuting " % desc) +
  460. cyan("'%s %s'\n" % (mmgen_cmd," ".join(cmd_args)))
  461. )
  462. else:
  463. msg_r("Testing %s: " % desc)
  464. if opt.direct_exec:
  465. os.system(" ".join([mmgen_cmd] + cmd_args))
  466. sys.exit()
  467. else:
  468. self.p = pexpect.spawn(mmgen_cmd,cmd_args)
  469. if opt.exact_output: self.p.logfile = sys.stdout
  470. def license(self):
  471. p = "'w' for conditions and warranty info, or 'c' to continue: "
  472. my_expect(self.p,p,'c')
  473. def label(self,label="Test Label"):
  474. p = "Enter a wallet label, or hit ENTER for no label: "
  475. my_expect(self.p,p,label+"\n")
  476. def usr_rand_out(self,saved=False):
  477. m = "%suser-supplied entropy" % ("saved " if saved else "")
  478. my_expect(self.p,"Generating encryption key from OS random data plus " + m)
  479. def usr_rand(self,num_chars):
  480. rand_chars = list(getrandstr(num_chars,no_space=True))
  481. my_expect(self.p,'symbols left: ','x')
  482. try:
  483. vmsg_r("SEND ")
  484. while self.p.expect('left: ',0.1) == 0:
  485. ch = rand_chars.pop(0)
  486. msg_r(yellow(ch)+" " if opt.verbose else "+")
  487. self.p.send(ch)
  488. except:
  489. vmsg("EOT")
  490. my_expect(self.p,"ENTER to continue: ",'\n')
  491. def passphrase_new(self,desc,passphrase):
  492. my_expect(self.p,("Enter passphrase for %s: " % desc), passphrase+"\n")
  493. my_expect(self.p,"Repeat passphrase: ", passphrase+"\n")
  494. def passphrase(self,desc,passphrase,pwtype=""):
  495. if pwtype: pwtype += " "
  496. my_expect(self.p,("Enter %spassphrase for %s.*?: " % (pwtype,desc)),
  497. passphrase+"\n",regex=True)
  498. def hash_preset(self,desc,preset=''):
  499. my_expect(self.p,("Enter hash preset for %s," % desc))
  500. my_expect(self.p,("or hit ENTER .*?:"), str(preset)+"\n",regex=True)
  501. def written_to_file(self,desc,overwrite_unlikely=False,query="Overwrite? ",oo=False):
  502. s1 = "%s written to file " % desc
  503. s2 = query + "Type uppercase 'YES' to confirm: "
  504. ret = my_expect(self.p,s1 if overwrite_unlikely else [s1,s2])
  505. if ret == 1:
  506. my_send(self.p,"YES\n")
  507. if oo:
  508. outfile = self.expect_getend("Overwriting file '").rstrip("'")
  509. return outfile
  510. else:
  511. ret = my_expect(self.p,s1)
  512. outfile = self.p.readline().strip().strip("'")
  513. vmsg("%s file: %s" % (desc,cyan(outfile.replace("'",""))))
  514. return outfile
  515. def no_overwrite(self):
  516. self.expect("Overwrite? Type uppercase 'YES' to confirm: ","\n")
  517. self.expect("Exiting at user request")
  518. def tx_view(self):
  519. my_expect(self.p,r"View .*?transaction.*? \(y\)es, \(N\)o, pager \(v\)iew.*?: ","\n",regex=True)
  520. def expect_getend(self,s,regex=False):
  521. ret = self.expect(s,regex=regex,nonl=True)
  522. end = self.readline().strip()
  523. vmsg(" ==> %s" % cyan(end))
  524. return end
  525. def interactive(self):
  526. return self.p.interact()
  527. def logfile(self,arg):
  528. self.p.logfile = arg
  529. def expect(self,*args,**kwargs):
  530. return my_expect(self.p,*args,**kwargs)
  531. def send(self,*args,**kwargs):
  532. return my_send(self.p,*args,**kwargs)
  533. def readline(self):
  534. return self.p.readline()
  535. def close(self):
  536. return self.p.close()
  537. def readlines(self):
  538. return [l.rstrip()+"\n" for l in self.p.readlines()]
  539. def read(self,n=None):
  540. return self.p.read(n)
  541. from mmgen.rpc.data import TransactionInfo
  542. from decimal import Decimal
  543. from mmgen.bitcoin import verify_addr
  544. def add_fake_unspent_entry(out,address,comment):
  545. out.append(TransactionInfo(
  546. account = unicode(comment),
  547. vout = int(getrandnum(4) % 8),
  548. txid = unicode(hexlify(os.urandom(32))),
  549. amount = Decimal("%s.%s" % (10+(getrandnum(4) % 40), getrandnum(4) % 100000000)),
  550. address = address,
  551. spendable = False,
  552. scriptPubKey = ("76a914"+verify_addr(address,return_hex=True)+"88ac"),
  553. confirmations = getrandnum(4) % 500
  554. ))
  555. def create_fake_unspent_data(adata,unspent_data_file,tx_data,non_mmgen_input=''):
  556. out = []
  557. for s in tx_data.keys():
  558. sid = tx_data[s]['sid']
  559. a = adata.addrinfo(sid)
  560. for idx,btcaddr in a.addrpairs():
  561. add_fake_unspent_entry(out,btcaddr,"%s:%s Test Wallet" % (sid,idx))
  562. if non_mmgen_input:
  563. from mmgen.bitcoin import privnum2addr,hextowif
  564. privnum = getrandnum(32)
  565. btcaddr = privnum2addr(privnum,compressed=True)
  566. of = os.path.join(cfgs[non_mmgen_input]['tmpdir'],non_mmgen_fn)
  567. write_to_file(of, hextowif("{:064x}".format(privnum),
  568. compressed=True)+"\n","compressed bitcoin key")
  569. add_fake_unspent_entry(out,btcaddr,"Non-MMGen address")
  570. # msg("\n".join([repr(o) for o in out])); sys.exit()
  571. write_to_file(unspent_data_file,repr(out),"Unspent outputs",verbose=True)
  572. def add_comments_to_addr_file(addrfile,tfile):
  573. silence()
  574. msg(green("Adding comments to address file '%s'" % addrfile))
  575. from mmgen.addr import AddrInfo
  576. a = AddrInfo(addrfile)
  577. for i in a.idxs(): a.set_comment(idx,"Test address %s" % idx)
  578. write_to_file(tfile,a.fmt_data(),{})
  579. end_silence()
  580. def make_brainwallet_file(fn):
  581. # Print random words with random whitespace in between
  582. from mmgen.mn_tirosh import words
  583. wl = words.split("\n")
  584. nwords,ws_list,max_spaces = 10," \n",5
  585. def rand_ws_seq():
  586. nchars = getrandnum(1) % max_spaces + 1
  587. return "".join([ws_list[getrandnum(1)%len(ws_list)] for i in range(nchars)])
  588. rand_pairs = [wl[getrandnum(4) % len(wl)] + rand_ws_seq() for i in range(nwords)]
  589. d = "".join(rand_pairs).rstrip() + "\n"
  590. if opt.verbose: msg_r("Brainwallet password:\n%s" % cyan(d))
  591. write_to_file(fn,d,"brainwallet password")
  592. def do_between():
  593. if opt.pause:
  594. from mmgen.util import keypress_confirm
  595. if keypress_confirm(green("Continue?"),default_yes=True):
  596. if opt.verbose or opt.exact_output: sys.stderr.write("\n")
  597. else:
  598. errmsg("Exiting at user request")
  599. sys.exit()
  600. elif opt.verbose or opt.exact_output:
  601. sys.stderr.write("\n")
  602. rebuild_list = OrderedDict()
  603. def check_needs_rerun(ts,cmd,build=False,root=True,force_delete=False,dpy=False):
  604. rerun = True if root else False # force_delete is not passed to recursive call
  605. fns = []
  606. if force_delete or not root:
  607. # does cmd produce a needed dependency(ies)?
  608. ret = ts.get_num_exts_for_cmd(cmd,dpy)
  609. if ret:
  610. for ext in ret[1]:
  611. fn = get_file_with_ext(ext,cfgs[ret[0]]['tmpdir'],delete=build)
  612. if fn:
  613. if force_delete: os.unlink(fn)
  614. else: fns.append(fn)
  615. else: rerun = True
  616. fdeps = ts.generate_file_deps(cmd)
  617. cdeps = ts.generate_cmd_deps(fdeps)
  618. for fn in fns:
  619. my_age = os.stat(fn).st_mtime
  620. for num,ext in fdeps:
  621. f = get_file_with_ext(ext,cfgs[num]['tmpdir'],delete=build)
  622. if f and os.stat(f).st_mtime > my_age: rerun = True
  623. for cdep in cdeps:
  624. if check_needs_rerun(ts,cdep,build=build,root=False,dpy=cmd): rerun = True
  625. if build:
  626. if rerun:
  627. for fn in fns:
  628. if not root: os.unlink(fn)
  629. ts.do_cmd(cmd)
  630. if not root: do_between()
  631. else:
  632. # If prog produces multiple files:
  633. if cmd not in rebuild_list or rerun == True:
  634. rebuild_list[cmd] = (rerun,fns[0] if fns else "") # FIX
  635. return rerun
  636. def refcheck(desc,chk,refchk):
  637. vmsg("Comparing %s '%s' to stored reference" % (desc,chk))
  638. if chk == refchk:
  639. ok()
  640. else:
  641. if not opt.verbose: errmsg("")
  642. errmsg(red("""
  643. Fatal error - %s '%s' does not match reference value '%s'. Aborting test
  644. """.strip() % (desc,chk,refchk)))
  645. sys.exit(3)
  646. def check_deps(cmds):
  647. if len(cmds) != 1:
  648. msg("Usage: %s check_deps <command>" % g.prog_name)
  649. sys.exit(1)
  650. cmd = cmds[0]
  651. if cmd not in cmd_data:
  652. msg("'%s': unrecognized command" % cmd)
  653. sys.exit(1)
  654. if not opt.quiet:
  655. msg("Checking dependencies for '%s'" % (cmd))
  656. check_needs_rerun(ts,cmd,build=False)
  657. w = max(len(i) for i in rebuild_list) + 1
  658. for cmd in rebuild_list:
  659. c = rebuild_list[cmd]
  660. m = "Rebuild" if (c[0] and c[1]) else "Build" if c[0] else "OK"
  661. msg("cmd {:<{w}} {}".format(cmd+":", m, w=w))
  662. # mmsg(cmd,c)
  663. def clean(dirs=[]):
  664. ts = MMGenTestSuite()
  665. dirlist = ts.list_tmp_dirs()
  666. if not dirs: dirs = dirlist.keys()
  667. for d in sorted(dirs):
  668. if d in dirlist:
  669. cleandir(dirlist[d])
  670. else:
  671. msg("%s: invalid directory number" % d)
  672. sys.exit(1)
  673. class MMGenTestSuite(object):
  674. def __init__(self):
  675. pass
  676. def list_tmp_dirs(self):
  677. d = {}
  678. for k in cfgs: d[k] = cfgs[k]['tmpdir']
  679. return d
  680. def get_num_exts_for_cmd(self,cmd,dpy=False): # dpy ignored here
  681. num = str(cmd_data[cmd][0])
  682. dgl = cfgs[num]['dep_generators']
  683. # mmsg(num,cmd,dgl)
  684. if cmd in dgl.values():
  685. exts = [k for k in dgl if dgl[k] == cmd]
  686. return (num,exts)
  687. else:
  688. return None
  689. def do_cmd(self,cmd):
  690. d = [(str(num),ext) for exts,num in cmd_data[cmd][2] for ext in exts]
  691. al = [get_file_with_ext(ext,cfgs[num]['tmpdir']) for num,ext in d]
  692. global cfg
  693. cfg = cfgs[str(cmd_data[cmd][0])]
  694. self.__class__.__dict__[cmd](*([self,cmd] + al))
  695. def generate_file_deps(self,cmd):
  696. return [(str(n),e) for exts,n in cmd_data[cmd][2] for e in exts]
  697. def generate_cmd_deps(self,fdeps):
  698. return [cfgs[str(n)]['dep_generators'][ext] for n,ext in fdeps]
  699. def walletgen(self,name,brain=False,seed_len=None):
  700. args = ["-d",cfg['tmpdir'],"-p1","-r10"]
  701. if seed_len: args += ["-l",str(seed_len)]
  702. # if 'seed_len' in cfg: args += ["-l",cfg['seed_len']]
  703. if brain:
  704. bwf = os.path.join(cfg['tmpdir'],cfg['bw_filename'])
  705. args += ["-b",cfg['bw_params'],bwf]
  706. make_brainwallet_file(bwf)
  707. t = MMGenExpect(name,"mmgen-walletgen", args)
  708. t.license()
  709. if brain:
  710. t.expect(
  711. "A brainwallet will be secure only if you really know what you're doing")
  712. t.expect("Type uppercase 'YES' to confirm: ","YES\n")
  713. t.usr_rand(10)
  714. # t.usr_rand_out()
  715. t.passphrase_new("new MMGen wallet",cfg['wpasswd'])
  716. t.written_to_file("Wallet")
  717. # if not brain:
  718. # t.usr_rand_out(saved=True)
  719. # t.label()
  720. # t.written_to_file("MMGen wallet")
  721. ok()
  722. def refwalletgen(self,name):
  723. label = "test.py ref. wallet (pw '%s', seed len %s)" \
  724. % (ref_wallet_brainpass,cfg['seed_len'])
  725. bw_arg = "-b%s,%s" % (cfg['seed_len'], ref_wallet_hash_preset)
  726. args = ["-d",cfg['tmpdir'],"-p1","-r10",bw_arg,"-L",label]
  727. d = " (%s-bit seed)" % cfg['seed_len']
  728. t = MMGenExpect(name,"mmgen-walletgen", args)
  729. t.license()
  730. t.expect("Type uppercase 'YES' to confirm: ","YES\n")
  731. t.expect("passphrase: ",ref_wallet_brainpass+"\n")
  732. t.usr_rand(10)
  733. t.passphrase_new("new MMGen wallet",cfg['wpasswd'])
  734. seed_id = t.written_to_file("Wallet").split("-")[0].split("/")[-1]
  735. refcheck("seed ID",seed_id,cfg['seed_id'])
  736. refwalletgen1 = refwalletgen2 = refwalletgen3 = refwalletgen
  737. def passchg(self,name,walletfile):
  738. t = MMGenExpect(name,"mmgen-passchg",
  739. ["-d",cfg['tmpdir'],"-p","2","-L","New Label","-r","16",walletfile])
  740. # t.license()
  741. t.passphrase("MMGen wallet",cfgs['1']['wpasswd'],pwtype="old")
  742. t.expect_getend("Label changed: ")
  743. t.expect_getend("Hash preset changed: ")
  744. t.passphrase("MMGen wallet",cfg['wpasswd'],pwtype="new")
  745. t.expect("Repeat passphrase: ",cfg['wpasswd']+"\n")
  746. t.usr_rand(16)
  747. t.expect_getend("Key ID changed: ")
  748. t.written_to_file("Wallet")
  749. ok()
  750. def walletchk_beg(self,name,args):
  751. t = MMGenExpect(name,"mmgen-walletchk", args)
  752. t.expect("Getting MMGen wallet data from file '%s'" % args[-1])
  753. t.passphrase("MMGen wallet",cfg['wpasswd'])
  754. t.expect("Passphrase is OK")
  755. t.expect("Wallet is OK")
  756. return t
  757. def walletchk(self,name,walletfile):
  758. self.walletchk_beg(name,[walletfile])
  759. ok()
  760. walletchk_newpass = walletchk
  761. def addrgen(self,name,walletfile,check_ref=False):
  762. t = MMGenExpect(name,"mmgen-addrgen",["-d",cfg['tmpdir'],walletfile,cfg['addr_idx_list']])
  763. t.license()
  764. t.passphrase("MMGen wallet",cfg['wpasswd'])
  765. t.expect("Passphrase is OK")
  766. t.expect("[0-9]+ addresses generated",regex=True)
  767. chk = t.expect_getend(r"Checksum for address data .*?: ",regex=True)
  768. if check_ref:
  769. refcheck("address data checksum",chk,cfg['addrfile_chk'])
  770. return
  771. t.written_to_file("Addresses")
  772. ok()
  773. def refaddrgen(self,name,walletfile):
  774. d = " (%s-bit seed)" % cfg['seed_len']
  775. self.addrgen(name,walletfile,check_ref=True)
  776. refaddrgen1 = refaddrgen2 = refaddrgen3 = refaddrgen
  777. def addrimport(self,name,addrfile):
  778. outfile = os.path.join(cfg['tmpdir'],"addrfile_w_comments")
  779. add_comments_to_addr_file(addrfile,outfile)
  780. t = MMGenExpect(name,"mmgen-addrimport",[outfile])
  781. t.expect_getend(r"Checksum for address data .*\[.*\]: ",regex=True)
  782. t.expect_getend("Validating addresses...OK. ")
  783. t.expect("Type uppercase 'YES' to confirm: ","\n")
  784. vmsg("This is a simulation, so no addresses were actually imported into the tracking\nwallet")
  785. ok()
  786. def txcreate(self,name,addrfile):
  787. self.txcreate_common(name,sources=['1'])
  788. def txcreate_common(self,name,sources=['1'],non_mmgen_input=''):
  789. if opt.verbose or opt.exact_output:
  790. sys.stderr.write(green("Generating fake transaction info\n"))
  791. silence()
  792. from mmgen.addr import AddrInfo,AddrInfoList
  793. tx_data,ail = {},AddrInfoList()
  794. from mmgen.util import parse_addr_idxs
  795. for s in sources:
  796. afile = get_file_with_ext("addrs",cfgs[s]["tmpdir"])
  797. ai = AddrInfo(afile)
  798. ail.add(ai)
  799. aix = parse_addr_idxs(cfgs[s]['addr_idx_list'])
  800. if len(aix) != addrs_per_wallet:
  801. errmsg(red("Address index list length != %s: %s" %
  802. (addrs_per_wallet,repr(aix))))
  803. sys.exit()
  804. tx_data[s] = {
  805. 'addrfile': afile,
  806. 'chk': ai.checksum,
  807. 'sid': ai.seed_id,
  808. 'addr_idxs': aix[-2:],
  809. }
  810. unspent_data_file = os.path.join(cfg['tmpdir'],"unspent.json")
  811. create_fake_unspent_data(ail,unspent_data_file,tx_data,non_mmgen_input)
  812. # make the command line
  813. from mmgen.bitcoin import privnum2addr
  814. btcaddr = privnum2addr(getrandnum(32),compressed=True)
  815. cmd_args = ["-d",cfg['tmpdir']]
  816. for num in tx_data.keys():
  817. s = tx_data[num]
  818. cmd_args += [
  819. "%s:%s,%s" % (s['sid'],s['addr_idxs'][0],cfgs[num]['amts'][0]),
  820. ]
  821. # + one BTC address
  822. # + one change address and one BTC address
  823. if num is tx_data.keys()[-1]:
  824. cmd_args += ["%s:%s" % (s['sid'],s['addr_idxs'][1])]
  825. cmd_args += ["%s,%s" % (btcaddr,cfgs[num]['amts'][1])]
  826. for num in tx_data: cmd_args += [tx_data[num]['addrfile']]
  827. os.environ["MMGEN_BOGUS_WALLET_DATA"] = unspent_data_file
  828. end_silence()
  829. if opt.verbose or opt.exact_output: sys.stderr.write("\n")
  830. t = MMGenExpect(name,"mmgen-txcreate",cmd_args)
  831. t.license()
  832. for num in tx_data.keys():
  833. t.expect_getend("Getting address data from file ")
  834. chk=t.expect_getend(r"Checksum for address data .*?: ",regex=True)
  835. verify_checksum_or_exit(tx_data[num]['chk'],chk)
  836. # not in tracking wallet warning, (1 + num sources) times
  837. if t.expect(["Continue anyway? (y/N): ",
  838. "Unable to connect to bitcoind"]) == 0:
  839. t.send("y")
  840. else:
  841. errmsg(red("Error: unable to connect to bitcoind. Exiting"))
  842. sys.exit(1)
  843. for num in tx_data.keys():
  844. t.expect("Continue anyway? (y/N): ","y")
  845. t.expect(r"'q' = quit sorting, .*?: ","M", regex=True)
  846. t.expect(r"'q' = quit sorting, .*?: ","q", regex=True)
  847. outputs_list = [addrs_per_wallet*i + 1 for i in range(len(tx_data))]
  848. if non_mmgen_input: outputs_list.append(len(tx_data)*addrs_per_wallet + 1)
  849. t.expect("Enter a range or space-separated list of outputs to spend: ",
  850. " ".join([str(i) for i in outputs_list])+"\n")
  851. if non_mmgen_input: t.expect("Accept? (y/N): ","y")
  852. t.expect("OK? (Y/n): ","y")
  853. t.expect("Add a comment to transaction? (y/N): ","\n")
  854. t.tx_view()
  855. t.expect("Save transaction? (y/N): ","y")
  856. t.written_to_file("Transaction")
  857. ok()
  858. def txsign_end(self,t,tnum=None):
  859. t.expect("Signing transaction")
  860. t.expect("Edit transaction comment? (y/N): ","\n")
  861. t.expect("Save signed transaction? (y/N): ","y")
  862. add = " #" + tnum if tnum else ""
  863. t.written_to_file("Signed transaction" + add)
  864. def txsign(self,name,txfile,walletfile,save=True):
  865. t = MMGenExpect(name,"mmgen-txsign",
  866. ["-d",cfg['tmpdir'],txfile,walletfile])
  867. t.license()
  868. t.tx_view()
  869. t.passphrase("MMGen wallet",cfg['wpasswd'])
  870. if save:
  871. self.txsign_end(t)
  872. else:
  873. t.expect("Edit transaction comment? (y/N): ","\n")
  874. t.expect("Save signed transaction? (y/N): ","\n")
  875. t.expect("Signed transaction not saved")
  876. ok()
  877. def txsend(self,name,sigfile):
  878. t = MMGenExpect(name,"mmgen-txsend", ["-d",cfg['tmpdir'],sigfile])
  879. t.license()
  880. t.tx_view()
  881. t.expect("Edit transaction comment? (y/N): ","\n")
  882. t.expect("broadcast this transaction to the network?")
  883. t.expect("'YES, I REALLY WANT TO DO THIS' to confirm: ","\n")
  884. t.expect("Exiting at user request")
  885. vmsg("This is a simulation; no transaction was sent")
  886. ok()
  887. def export_seed(self,name,walletfile):
  888. t = self.walletchk_beg(name,["-s","-d",cfg['tmpdir'],walletfile])
  889. f = t.written_to_file("Seed data")
  890. silence()
  891. msg("Seed data: %s" % cyan(get_data_from_file(f,"seed data")))
  892. end_silence()
  893. ok()
  894. def export_mnemonic(self,name,walletfile):
  895. t = self.walletchk_beg(name,["-m","-d",cfg['tmpdir'],walletfile])
  896. f = t.written_to_file("Mnemonic data")
  897. silence()
  898. msg_r("Mnemonic data: %s" % cyan(get_data_from_file(f,"mnemonic data")))
  899. end_silence()
  900. ok()
  901. def export_incog(self,name,walletfile,args=["-g"]):
  902. t = MMGenExpect(name,"mmgen-walletchk",args+["-d",cfg['tmpdir'],"-r","10",walletfile])
  903. t.passphrase("MMGen wallet",cfg['wpasswd'])
  904. t.usr_rand(10)
  905. incog_id = t.expect_getend("Incog ID: ")
  906. write_to_tmpfile(cfg,incog_id_fn,incog_id+"\n")
  907. if args[0] == "-G": return t
  908. t.written_to_file("Incognito wallet data",overwrite_unlikely=True)
  909. ok()
  910. def export_incog_hex(self,name,walletfile):
  911. self.export_incog(name,walletfile,args=["-X"])
  912. # TODO: make outdir and hidden incog compatible (ignore --outdir and warn user?)
  913. def export_incog_hidden(self,name,walletfile):
  914. rf,rd = os.path.join(cfg['tmpdir'],hincog_fn),os.urandom(hincog_bytes)
  915. vmsg(green("Writing %s bytes of data to file '%s'" % (hincog_bytes,rf)))
  916. write_to_file(rf,rd,verbose=opt.verbose)
  917. t = self.export_incog(name,walletfile,args=["-G","%s,%s"%(rf,hincog_offset)])
  918. t.written_to_file("Data",query="")
  919. ok()
  920. def addrgen_seed(self,name,walletfile,foo,desc="seed data",arg="-s"):
  921. t = MMGenExpect(name,"mmgen-addrgen",
  922. [arg,"-d",cfg['tmpdir'],walletfile,cfg['addr_idx_list']])
  923. t.license()
  924. t.expect_getend("Valid %s for seed ID " % desc)
  925. vmsg("Comparing generated checksum with checksum from previous address file")
  926. chk = t.expect_getend(r"Checksum for address data .*?: ",regex=True)
  927. verify_checksum_or_exit(get_addrfile_checksum(),chk)
  928. t.no_overwrite()
  929. ok()
  930. def addrgen_mnemonic(self,name,walletfile,foo):
  931. self.addrgen_seed(name,walletfile,foo,desc="mnemonic",arg="-m")
  932. def addrgen_incog(self,name,walletfile,foo,args=["-g"]):
  933. t = MMGenExpect(name,"mmgen-addrgen",args+["-d",
  934. cfg['tmpdir'],walletfile,cfg['addr_idx_list']])
  935. t.license()
  936. t.expect_getend("Incog ID: ")
  937. t.passphrase("incognito wallet data \w{8}", cfg['wpasswd'])
  938. t.hash_preset("incog wallet",'1')
  939. vmsg("Comparing generated checksum with checksum from address file")
  940. chk = t.expect_getend(r"Checksum for address data .*?: ",regex=True)
  941. verify_checksum_or_exit(get_addrfile_checksum(),chk)
  942. t.no_overwrite()
  943. ok()
  944. def addrgen_incog_hex(self,name,walletfile,foo):
  945. self.addrgen_incog(name,walletfile,foo,args=["-X"])
  946. def addrgen_incog_hidden(self,name,walletfile,foo):
  947. rf = os.path.join(cfg['tmpdir'],hincog_fn)
  948. self.addrgen_incog(name,walletfile,foo,
  949. args=["-G","%s,%s,%s"%(rf,hincog_offset,hincog_seedlen)])
  950. def keyaddrgen(self,name,walletfile,check_ref=False):
  951. t = MMGenExpect(name,"mmgen-keygen",
  952. ["-d",cfg['tmpdir'],walletfile,cfg['addr_idx_list']])
  953. t.license()
  954. t.expect("Type uppercase 'YES' to confirm: ","YES\n")
  955. t.passphrase("MMGen wallet",cfg['wpasswd'])
  956. chk = t.expect_getend(r"Checksum for key-address data .*?: ",regex=True)
  957. if check_ref:
  958. refcheck("key-address data checksum",chk,cfg['keyaddrfile_chk'])
  959. return
  960. t.expect("Encrypt key list? (y/N): ","y")
  961. t.hash_preset("new key list",'1')
  962. t.passphrase_new("new key list",cfg['kapasswd'])
  963. t.written_to_file("Keys")
  964. ok()
  965. def refkeyaddrgen(self,name,walletfile):
  966. self.keyaddrgen(name,walletfile,check_ref=True)
  967. refkeyaddrgen1 = refkeyaddrgen2 = refkeyaddrgen3 = refkeyaddrgen
  968. def txsign_keyaddr(self,name,keyaddr_file,txfile):
  969. t = MMGenExpect(name,"mmgen-txsign", ["-d",cfg['tmpdir'],"-M",keyaddr_file,txfile])
  970. t.license()
  971. t.hash_preset("key-address file",'1')
  972. t.passphrase("key-address file",cfg['kapasswd'])
  973. t.expect("Check key-to-address validity? (y/N): ","y")
  974. t.tx_view()
  975. self.txsign_end(t)
  976. ok()
  977. def walletgen2(self,name):
  978. self.walletgen(name,seed_len=128)
  979. def addrgen2(self,name,walletfile):
  980. self.addrgen(name,walletfile)
  981. def txcreate2(self,name,addrfile):
  982. self.txcreate_common(name,sources=['2'])
  983. def txsign2(self,name,txf1,wf1,txf2,wf2):
  984. t = MMGenExpect(name,"mmgen-txsign", ["-d",cfg['tmpdir'],txf1,wf1,txf2,wf2])
  985. t.license()
  986. for cnum in ('1','2'):
  987. t.tx_view()
  988. t.passphrase("MMGen wallet",cfgs[cnum]['wpasswd'])
  989. self.txsign_end(t,cnum)
  990. ok()
  991. def export_mnemonic2(self,name,walletfile):
  992. self.export_mnemonic(name,walletfile)
  993. def walletgen3(self,name):
  994. self.walletgen(name)
  995. def addrgen3(self,name,walletfile):
  996. self.addrgen(name,walletfile)
  997. def txcreate3(self,name,addrfile1,addrfile2):
  998. self.txcreate_common(name,sources=['1','3'])
  999. def txsign3(self,name,wf1,wf2,txf2):
  1000. t = MMGenExpect(name,"mmgen-txsign", ["-d",cfg['tmpdir'],wf1,wf2,txf2])
  1001. t.license()
  1002. t.tx_view()
  1003. for cnum in ('1','3'):
  1004. t.expect_getend("Getting MMGen wallet data from file ")
  1005. t.passphrase("MMGen wallet",cfgs[cnum]['wpasswd'])
  1006. self.txsign_end(t)
  1007. ok()
  1008. def walletgen4(self,name):
  1009. self.walletgen(name,brain=True)
  1010. def addrgen4(self,name,walletfile):
  1011. self.addrgen(name,walletfile)
  1012. def txcreate4(self,name,f1,f2,f3,f4):
  1013. self.txcreate_common(name,sources=['1','2','3','4'],non_mmgen_input='4')
  1014. def txsign4(self,name,f1,f2,f3,f4,f5):
  1015. non_mm_fn = os.path.join(cfg['tmpdir'],non_mmgen_fn)
  1016. t = MMGenExpect(name,"mmgen-txsign",
  1017. ["-d",cfg['tmpdir'],"-b",cfg['bw_params'],"-k",non_mm_fn,f1,f2,f3,f4,f5])
  1018. t.license()
  1019. t.tx_view()
  1020. for cnum,desc,app in ('1',"incognito","incognito"),('3',"MMGen","MMGen"):
  1021. t.expect_getend("Getting %s wallet data from file " % desc)
  1022. t.passphrase("%s wallet"%app,cfgs[cnum]['wpasswd'])
  1023. if cnum == '1':
  1024. t.hash_preset("incog wallet",'1')
  1025. self.txsign_end(t)
  1026. ok()
  1027. def tool_encrypt(self,name,infile=""):
  1028. if infile:
  1029. infn = infile
  1030. else:
  1031. d = os.urandom(1033)
  1032. tmp_fn = cfg['tool_enc_infn']
  1033. write_to_tmpfile(cfg,tmp_fn,d)
  1034. infn = get_tmpfile_fn(cfg,tmp_fn)
  1035. t = MMGenExpect(name,"mmgen-tool",["-d",cfg['tmpdir'],"encrypt",infn])
  1036. t.hash_preset("user data",'1')
  1037. t.passphrase_new("user data",tool_enc_passwd)
  1038. t.written_to_file("Encrypted data")
  1039. ok()
  1040. # Generate the reference mmenc file
  1041. # def tool_encrypt_ref(self,name):
  1042. # infn = get_tmpfile_fn(cfg,cfg['tool_enc_ref_infn'])
  1043. # write_to_file(infn,cfg['tool_enc_reftext'],silent=True)
  1044. # self.tool_encrypt(name,infn)
  1045. def tool_decrypt(self,name,f1,f2):
  1046. of = name + ".out"
  1047. t = MMGenExpect(name,"mmgen-tool",
  1048. ["-d",cfg['tmpdir'],"decrypt",f2,"outfile="+of,"hash_preset=1"])
  1049. t.passphrase("user data",tool_enc_passwd)
  1050. t.written_to_file("Decrypted data")
  1051. d1 = read_from_file(f1)
  1052. d2 = read_from_file(get_tmpfile_fn(cfg,of))
  1053. cmp_or_die(d1,d2)
  1054. def tool_find_incog_data(self,name,f1,f2):
  1055. i_id = read_from_file(f2).rstrip()
  1056. vmsg("Incog ID: %s" % cyan(i_id))
  1057. t = MMGenExpect(name,"mmgen-tool",
  1058. ["-d",cfg['tmpdir'],"find_incog_data",f1,i_id])
  1059. o = t.expect_getend("Incog data for ID \w{8} found at offset ",regex=True)
  1060. os.unlink(f1)
  1061. cmp_or_die(hincog_offset,int(o))
  1062. def walletconv_out(self,name,desc,out_fmt="w",uopts=[],uopts_chk=[],pw=False):
  1063. opts = ["-d",cfg['tmpdir'],"-r10","-p1","-o",out_fmt] + uopts
  1064. infile = os.path.join(ref_dir,cfg['seed_id']+".mmwords")
  1065. d = "(convert)"
  1066. t = MMGenExpect(name,"mmgen-walletconv",opts+[infile],extra_desc=d)
  1067. t.license()
  1068. if pw:
  1069. t.passphrase_new("new "+desc,cfg['wpasswd'])
  1070. t.usr_rand(10)
  1071. if " ".join(desc.split()[-2:]) == "incognito data":
  1072. for i in (1,2,3):
  1073. t.expect("Generating encryption key from OS random data ")
  1074. if desc == "hidden incognito data":
  1075. ret = t.expect(["Create? (Y/n): ","'YES' to confirm: "],"YES\n")
  1076. if ret == 0:
  1077. t.expect("Enter file size: ","1234\n")
  1078. if out_fmt == "w": t.label()
  1079. wf = t.written_to_file(capfirst(desc),oo=True)
  1080. ok()
  1081. d = "(check)"
  1082. if desc == "hidden incognito data":
  1083. self.keygen_chksum_chk_hincog(name,cfg['seed_id'],uopts_chk)
  1084. # elif pw:
  1085. # self.walletchk_chksum_chk(name,wf,cfg['seed_id'],uopts=uopts_chk)
  1086. else:
  1087. self.keygen_chksum_chk(name,wf,cfg['seed_id'],pw=pw)
  1088. def walletconv_in(self,name,infile,desc,uopts=[],pw=False,oo=False):
  1089. opts = ["-d",cfg['tmpdir'],"-o","words","-r10"]
  1090. if_arg = [infile] if infile else []
  1091. d = "(convert)"
  1092. t = MMGenExpect(name,"mmgen-walletconv",opts+uopts+if_arg,extra_desc=d)
  1093. t.license()
  1094. if desc == "brainwallet":
  1095. t.expect("Enter brainwallet: ",ref_wallet_brainpass+"\n")
  1096. if pw:
  1097. t.passphrase(desc,cfg['wpasswd'])
  1098. if name[:19] == "ref_hincog_conv_old":
  1099. t.expect("Is the seed ID correct? (Y/n): ","\n")
  1100. else:
  1101. t.expect(["Passphrase is OK"," are correct"])
  1102. # Output
  1103. wf = t.written_to_file("Mnemonic data",oo=oo)
  1104. t.close()
  1105. ok()
  1106. # back check of result
  1107. d = "(check)"
  1108. self.keygen_chksum_chk(name,wf,cfg['seed_id'])
  1109. # Saved reference file tests
  1110. def ref_wallet_conv(self,name):
  1111. wf = os.path.join(ref_dir,cfg['ref_wallet'])
  1112. self.walletconv_in(name,wf,"MMGen wallet",pw=True,oo=True)
  1113. def ref_mn_conv(self,name,ext="mmwords",desc="Mnemonic data"):
  1114. wf = os.path.join(ref_dir,cfg['seed_id']+"."+ext)
  1115. self.walletconv_in(name,wf,desc,oo=True)
  1116. def ref_seed_conv(self,name):
  1117. self.ref_mn_conv(name,ext="mmseed",desc="Seed data")
  1118. def ref_brain_conv(self,name):
  1119. uopts = ["-i","b","-p","1","-l",str(cfg['seed_len'])]
  1120. self.walletconv_in(name,None,"brainwallet",uopts,oo=True)
  1121. def ref_incog_conv(self,name,wfk="ic_wallet",in_fmt="i",desc="incognito data"):
  1122. uopts = ["-i",in_fmt,"-p","1","-l",str(cfg['seed_len'])]
  1123. wf = os.path.join(ref_dir,cfg[wfk])
  1124. self.walletconv_in(name,wf,desc,uopts,oo=True,pw=True)
  1125. def ref_incox_conv(self,name):
  1126. self.ref_incog_conv(name,in_fmt="xi",wfk="ic_wallet_hex",desc="hex incognito data")
  1127. def ref_hincog_conv(self,name,wfk='hic_wallet',add_uopts=[]):
  1128. ic_f = os.path.join(ref_dir,cfg[wfk])
  1129. uopts = ["-i","hi","-p","1","-l",str(cfg['seed_len'])] + add_uopts
  1130. hi_opt = ["-H","%s,%s" % (ic_f,ref_wallet_incog_offset)]
  1131. self.walletconv_in(name,None,"hidden incognito data",uopts+hi_opt,oo=True,pw=True)
  1132. def ref_hincog_conv_old(self,name):
  1133. self.ref_hincog_conv(name,wfk='hic_wallet_old',add_uopts=["-O"])
  1134. def ref_wallet_conv_out(self,name):
  1135. self.walletconv_out(name,"MMGen wallet","w",pw=True)
  1136. def ref_mn_conv_out(self,name):
  1137. self.walletconv_out(name,"mnemonic data","mn")
  1138. def ref_seed_conv_out(self,name):
  1139. self.walletconv_out(name,"seed data","seed")
  1140. def ref_incog_conv_out(self,name):
  1141. self.walletconv_out(name,"incognito data",out_fmt="i",pw=True)
  1142. def ref_incox_conv_out(self,name):
  1143. self.walletconv_out(name,"hex incognito data",out_fmt="xi",pw=True)
  1144. def ref_hincog_conv_out(self,name,extra_uopts=[]):
  1145. ic_f = os.path.join(cfg['tmpdir'],"rand.data")
  1146. hi_parms = "%s,%s" % (ic_f,ref_wallet_incog_offset)
  1147. hi_parms_legacy = "%s,%s,%s"%(ic_f,ref_wallet_incog_offset,cfg['seed_len'])
  1148. self.walletconv_out(name,
  1149. "hidden incognito data", "hi",
  1150. uopts=["-J",hi_parms] + extra_uopts,
  1151. uopts_chk=["-G",hi_parms_legacy],
  1152. pw=True
  1153. )
  1154. ref_wallet_conv1 = ref_wallet_conv2 = ref_wallet_conv3 = ref_wallet_conv
  1155. ref_mn_conv1 = ref_mn_conv2 = ref_mn_conv3 = ref_mn_conv
  1156. ref_seed_conv1 = ref_seed_conv2 = ref_seed_conv3 = ref_seed_conv
  1157. ref_brain_conv1 = ref_brain_conv2 = ref_brain_conv3 = ref_brain_conv
  1158. ref_incog_conv1 = ref_incog_conv2 = ref_incog_conv3 = ref_incog_conv
  1159. ref_incox_conv1 = ref_incox_conv2 = ref_incox_conv3 = ref_incox_conv
  1160. ref_hincog_conv1 = ref_hincog_conv2 = ref_hincog_conv3 = ref_hincog_conv
  1161. ref_hincog_conv_old1 = ref_hincog_conv_old2 = ref_hincog_conv_old3 = ref_hincog_conv_old
  1162. ref_wallet_conv_out1 = ref_wallet_conv_out2 = ref_wallet_conv_out3 = ref_wallet_conv_out
  1163. ref_mn_conv_out1 = ref_mn_conv_out2 = ref_mn_conv_out3 = ref_mn_conv_out
  1164. ref_seed_conv_out1 = ref_seed_conv_out2 = ref_seed_conv_out3 = ref_seed_conv_out
  1165. ref_incog_conv_out1 = ref_incog_conv_out2 = ref_incog_conv_out3 = ref_incog_conv_out
  1166. ref_incox_conv_out1 = ref_incox_conv_out2 = ref_incox_conv_out3 = ref_incox_conv_out
  1167. ref_hincog_conv_out1 = ref_hincog_conv_out2 = ref_hincog_conv_out3 = ref_hincog_conv_out
  1168. def ref_wallet_chk(self,name):
  1169. wf = os.path.join(ref_dir,cfg['ref_wallet'])
  1170. self.walletchk(name,wf)
  1171. ref_wallet_chk1 = ref_wallet_chk2 = ref_wallet_chk3 = ref_wallet_chk
  1172. def ref_seed_chk(self,name,ext=g.seed_ext):
  1173. wf = os.path.join(ref_dir,"%s.%s" % (cfg['seed_id'],ext))
  1174. desc = "seed data" if ext == g.seed_ext else "mnemonic"
  1175. self.keygen_chksum_chk(name,wf,cfg['seed_id'])
  1176. ref_seed_chk1 = ref_seed_chk2 = ref_seed_chk3 = ref_seed_chk
  1177. def ref_mn_chk(self,name): self.ref_seed_chk(name,ext=g.mn_ext)
  1178. ref_mn_chk1 = ref_mn_chk2 = ref_mn_chk3 = ref_mn_chk
  1179. def ref_brain_chk(self,name,bw_file=ref_bw_file):
  1180. wf = os.path.join(ref_dir,bw_file)
  1181. arg = "-b%s,%s" % (cfg['seed_len'],ref_bw_hash_preset)
  1182. self.keygen_chksum_chk(name,wf,cfg['ref_bw_seed_id'],[arg])
  1183. def keygen_chksum_chk_hincog(self,name,seed_id,hincog_parm):
  1184. t = MMGenExpect(name,"mmgen-keygen", ["-p1","-q","-S","-A"]+hincog_parm+["1"],extra_desc="(check)")
  1185. t.passphrase("",cfg['wpasswd'])
  1186. t.expect("Encrypt key list? (y/N): ","\n")
  1187. t.expect("any printable ASCII symbol.\r\n")
  1188. chk = t.readline()[:8]
  1189. vmsg("Seed ID: %s" % cyan(chk))
  1190. cmp_or_die(seed_id,chk)
  1191. def keygen_chksum_chk(self,name,wf,seed_id,args=[],pw=False):
  1192. hp_arg = ["-p1"] if pw else []
  1193. t = MMGenExpect(name,"mmgen-keygen", ["-q","-S","-A"]+args+hp_arg+[wf,"1"],extra_desc="(check)")
  1194. if pw:
  1195. t.passphrase("",cfg['wpasswd'])
  1196. t.expect("Encrypt key list? (y/N): ","\n")
  1197. t.expect("any printable ASCII symbol.\r\n")
  1198. chk = t.readline()[:8]
  1199. vmsg("Seed ID: %s" % cyan(chk))
  1200. cmp_or_die(seed_id,chk)
  1201. # Use this for encrypted wallets instead of keygen_chksum_chk()
  1202. def walletchk_chksum_chk(self,name,wf,seed_id,uopts=[]):
  1203. t = MMGenExpect(name,"mmgen-walletchk",["-v", wf]+uopts,
  1204. extra_desc="(check)")
  1205. t.passphrase("",cfg['wpasswd'])
  1206. chk = t.expect_getend("Seed ID checksum OK (")[:8]
  1207. t.close()
  1208. cmp_or_die(seed_id,chk)
  1209. ref_brain_chk1 = ref_brain_chk2 = ref_brain_chk3 = ref_brain_chk
  1210. def ref_brain_chk_spc3(self,name):
  1211. self.ref_brain_chk(name,bw_file=ref_bw_file_spc)
  1212. def ref_hincog_chk(self,name):
  1213. for wtype,desc,earg in ('hic_wallet','',[]), \
  1214. ('hic_wallet_old','(old format)',["-o"]):
  1215. ic_arg = "%s,%s,%s" % (
  1216. os.path.join(ref_dir,cfg[wtype]),
  1217. ref_wallet_incog_offset,cfg['seed_len']
  1218. )
  1219. t = MMGenExpect(name,"mmgen-keygen",
  1220. ["-q","-A"]+earg+["-G"]+[ic_arg]+['1'],extra_desc=desc)
  1221. t.passphrase("incognito wallet",cfg['wpasswd'])
  1222. t.hash_preset("incog wallet","1")
  1223. if wtype == 'hic_wallet_old':
  1224. t.expect("Is the seed ID correct? (Y/n): ","\n")
  1225. chk = t.expect_getend("Valid incog data for seed ID ")
  1226. t.close()
  1227. cmp_or_die(cfg['seed_id'],chk)
  1228. ref_hincog_chk1 = ref_hincog_chk2 = ref_hincog_chk3 = ref_hincog_chk
  1229. def ref_addrfile_chk(self,name,ftype="addr"):
  1230. wf = os.path.join(ref_dir,cfg['ref_'+ftype+'file'])
  1231. t = MMGenExpect(name,"mmgen-tool",[ftype+"file_chksum",wf])
  1232. if ftype == "keyaddr":
  1233. w = "key-address file"
  1234. t.hash_preset(w,ref_kafile_hash_preset)
  1235. t.passphrase(w,ref_kafile_pass)
  1236. t.expect("Check key-to-address validity? (y/N): ","y")
  1237. o = t.expect_getend("Checksum for .*address data .*: ",regex=True)
  1238. cmp_or_die(cfg['ref_'+ftype+'file_chksum'],o)
  1239. def ref_keyaddrfile_chk(self,name):
  1240. self.ref_addrfile_chk(name,ftype="keyaddr")
  1241. # def txcreate8(self,name,addrfile):
  1242. # self.txcreate_common(name,sources=['8'])
  1243. def ref_tx_chk(self,name):
  1244. tf = os.path.join(ref_dir,cfg['ref_tx_file'])
  1245. wf = os.path.join(ref_dir,cfg['ref_wallet'])
  1246. self.txsign(name,tf,wf,save=False)
  1247. def ref_tool_decrypt(self,name):
  1248. f = os.path.join(ref_dir,ref_enc_fn)
  1249. t = MMGenExpect(name,"mmgen-tool",
  1250. ["-q","decrypt",f,"outfile=-","hash_preset=1"])
  1251. t.passphrase("user data",tool_enc_passwd)
  1252. t.readline()
  1253. import re
  1254. o = re.sub('\r\n','\n',t.read())
  1255. cmp_or_die(sample_text,o)
  1256. # main()
  1257. if opt.pause:
  1258. import termios,atexit
  1259. fd = sys.stdin.fileno()
  1260. old = termios.tcgetattr(fd)
  1261. def at_exit():
  1262. termios.tcsetattr(fd, termios.TCSADRAIN, old)
  1263. atexit.register(at_exit)
  1264. start_time = int(time.time())
  1265. ts = MMGenTestSuite()
  1266. for cfg in sorted(cfgs): mk_tmpdir(cfgs[cfg])
  1267. try:
  1268. if cmd_args:
  1269. for arg in cmd_args:
  1270. if arg in utils:
  1271. globals()[arg](cmd_args[cmd_args.index(arg)+1:])
  1272. sys.exit()
  1273. elif arg in meta_cmds:
  1274. for cmd in meta_cmds[arg]:
  1275. check_needs_rerun(ts,cmd,build=True)
  1276. elif arg in cmd_groups.keys():
  1277. for cmd in cmd_groups[arg]:
  1278. check_needs_rerun(ts,cmd,build=True)
  1279. elif arg in cmd_data:
  1280. check_needs_rerun(ts,arg,build=True)
  1281. else:
  1282. die(1,"%s: unrecognized command" % arg)
  1283. else:
  1284. clean()
  1285. for cmd in cmd_data:
  1286. ts.do_cmd(cmd)
  1287. if cmd is not cmd_data.keys()[-1]: do_between()
  1288. except:
  1289. sys.stderr = stderr_save
  1290. raise
  1291. t = int(time.time()) - start_time
  1292. sys.stderr.write(green(
  1293. "All requested tests finished OK, elapsed time: %02i:%02i\n"
  1294. % (t/60,t%60)))