tooltest2.py 33 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876
  1. #!/usr/bin/env python3
  2. #
  3. # mmgen = Multi-Mode GENerator, command-line Bitcoin cold storage solution
  4. # Copyright (C)2013-2019 The MMGen Project <mmgen@tuta.io>
  5. #
  6. # This program is free software: you can redistribute it and/or modify
  7. # it under the terms of the GNU General Public License as published by
  8. # the Free Software Foundation, either version 3 of the License, or
  9. # (at your option) any later version.
  10. #
  11. # This program is distributed in the hope that it will be useful,
  12. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. # GNU General Public License for more details.
  15. #
  16. # You should have received a copy of the GNU General Public License
  17. # along with this program. If not, see <http://www.gnu.org/licenses/>.
  18. """
  19. test/tooltest2.py: Simple tests for the 'mmgen-tool' utility
  20. """
  21. # TODO: move all non-interactive 'mmgen-tool' tests in 'test.py' here
  22. # TODO: move all(?) tests in 'tooltest.py' here (or duplicate them?)
  23. import sys,os,time
  24. from subprocess import run,PIPE
  25. from decimal import Decimal
  26. from tests_header import repo_root
  27. from mmgen.common import *
  28. from test.common import *
  29. from mmgen.obj import is_wif,is_coin_addr
  30. from mmgen.seed import is_bip39_mnemonic,is_mmgen_mnemonic
  31. from mmgen.baseconv import *
  32. NL = ('\n','\r\n')[g.platform=='win']
  33. def is_str(s): return type(s) == str
  34. def md5_hash(s):
  35. from hashlib import md5
  36. return md5(s.encode()).hexdigest()
  37. def md5_hash_strip(s):
  38. import re
  39. s = re.sub('\x1b\[[;0-9]+?m','',s) # strip ANSI color sequences
  40. s = s.replace(NL,'\n') # fix DOS newlines
  41. return md5_hash(s.strip())
  42. opts_data = {
  43. 'text': {
  44. 'desc': "Simple test suite for the 'mmgen-tool' utility",
  45. 'usage':'[options] [command]...',
  46. 'options': """
  47. -h, --help Print this help message
  48. -C, --coverage Produce code coverage info using trace module
  49. -d, --die-on-missing Abort if no test data found for given command
  50. --, --longhelp Print help message for long options (common options)
  51. -l, --list-tests List the test groups in this test suite
  52. -L, --list-tested-cmds Output the 'mmgen-tool' commands that are tested by this test suite
  53. -n, --names Print command names instead of descriptions
  54. -q, --quiet Produce quieter output
  55. -s, --system Test scripts and modules installed on system rather than
  56. those in the repo root
  57. -t, --type= Specify coin type
  58. -f, --fork Run commands via tool executable instead of importing tool module
  59. -t, --traceback Run tool inside traceback script
  60. -v, --verbose Produce more verbose output
  61. """,
  62. 'notes': """
  63. If no command is given, the whole suite of tests is run.
  64. """
  65. }
  66. }
  67. sample_text_hexdump = (
  68. '000000: 5468 6520 5469 6d65 7320 3033 2f4a 616e{n}' +
  69. '000010: 2f32 3030 3920 4368 616e 6365 6c6c 6f72{n}' +
  70. '000020: 206f 6e20 6272 696e 6b20 6f66 2073 6563{n}' +
  71. '000030: 6f6e 6420 6261 696c 6f75 7420 666f 7220{n}' +
  72. '000040: 6261 6e6b 73').format(n=NL)
  73. kafile_opts = ['-p1','-Ptest/ref/keyaddrfile_password']
  74. kafile_code = (
  75. "\nopt.hash_preset = '1'" +
  76. "\nopt.set_by_user = ['hash_preset']" +
  77. "\nopt.use_old_ed25519 = None" +
  78. "\nopt.passwd_file = 'test/ref/keyaddrfile_password'" )
  79. from test.unit_tests_d.ut_bip39 import unit_test as bip39
  80. tests = {
  81. 'Mnemonic': {
  82. 'hex2mn': [
  83. ( ['deadbeefdeadbeefdeadbeefdeadbeef','fmt=mmgen'],
  84. 'table cast forgive master funny gaze sadness ripple million paint moral match' ),
  85. ( ['deadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeef'],
  86. ('swirl maybe anymore mix scale stray fog use approach page crime rhyme ' +
  87. 'class former strange window snap soon') ),
  88. ( ['deadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeef'],
  89. ('swell type milk figure cheese phone fill black test bloom heard comfort ' +
  90. 'image terrible radio lesson own reply battle goal goodbye need laugh stream') ),
  91. ( ['ffffffffffffffffffffffffffffffff'],
  92. 'yellow yeah show bowl season spider cling defeat poison law shelter reflect' ),
  93. ( ['ffffffffffffffffffffffffffffffffffffffffffffffff'],
  94. ('yeah youth quit fail perhaps drum out person young click skin ' +
  95. 'weird inside silently perfectly together anyone memory') ),
  96. ( ['ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff'],
  97. ('wrote affection object cell opinion here laughter stare honest north cost begin ' +
  98. 'murder something yourself effort acid dot doubt game broke tell guilt innocent') ),
  99. ( ['0000000000000000000000000000000000000000000000000000000000000001'],
  100. ('able able able able able able able able able able able able ' +
  101. 'able able able able able able able able able able able about') ),
  102. ] + [([a,'fmt=bip39'],b) for a,b in bip39.vectors],
  103. 'mn2hex': [
  104. ( ['table cast forgive master funny gaze sadness ripple million paint moral match','fmt=mmgen'],
  105. 'deadbeefdeadbeefdeadbeefdeadbeef' ),
  106. ( ['swirl maybe anymore mix scale stray fog use approach page crime rhyme ' +
  107. 'class former strange window snap soon'],
  108. 'deadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeef'),
  109. ( ['swell type milk figure cheese phone fill black test bloom heard comfort ' +
  110. 'image terrible radio lesson own reply battle goal goodbye need laugh stream'],
  111. 'deadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeef' ),
  112. ( ['yellow yeah show bowl season spider cling defeat poison law shelter reflect'],
  113. 'ffffffffffffffffffffffffffffffff' ),
  114. ( ['yeah youth quit fail perhaps drum out person young click skin ' +
  115. 'weird inside silently perfectly together anyone memory'],
  116. 'ffffffffffffffffffffffffffffffffffffffffffffffff' ) ,
  117. ( ['wrote affection object cell opinion here laughter stare honest north cost begin ' +
  118. 'murder something yourself effort acid dot doubt game broke tell guilt innocent'],
  119. 'ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff'),
  120. ( ['able able able able able able able able able able able able ' +
  121. 'able able able able able able able able able able able about'],
  122. '0000000000000000000000000000000000000000000000000000000000000001'),
  123. ] + [([b,'fmt=bip39'],a) for a,b in bip39.vectors],
  124. 'mn_rand128': [
  125. ( [], is_mmgen_mnemonic, ['-r0']),
  126. ( ['fmt=mmgen'], is_mmgen_mnemonic, ['-r0']),
  127. ( ['fmt=bip39'], is_bip39_mnemonic, ['-r0']),
  128. ],
  129. 'mn_rand192': [
  130. ( ['fmt=mmgen'], is_mmgen_mnemonic, ['-r0']),
  131. ( ['fmt=bip39'], is_bip39_mnemonic, ['-r0']),
  132. ],
  133. 'mn_rand256': [
  134. ( ['fmt=mmgen'], is_mmgen_mnemonic, ['-r0']),
  135. ( ['fmt=bip39'], is_bip39_mnemonic, ['-r0']),
  136. ],
  137. 'mn_stats': [
  138. ( [], is_str ),
  139. ( ['fmt=mmgen'], is_str ),
  140. ( ['fmt=bip39'], is_str ),
  141. ],
  142. 'mn_printlist': [
  143. ( [], is_str ),
  144. ( ['fmt=mmgen'], is_str ),
  145. ( ['fmt=bip39'], is_str ),
  146. ],
  147. },
  148. 'Util': {
  149. 'hextob32': [
  150. ( ['deadbeef'], 'DPK3PXP' ),
  151. ( ['deadbeefdeadbeef'], 'N5LN657PK3PXP' ),
  152. ( ['ffffffffffffffff'], 'P777777777777' ),
  153. ( ['0000000000000000'], 'A' ),
  154. ( ['0000000000000000','pad=10'], 'AAAAAAAAAA' ),
  155. ( ['ff','pad=10'], 'AAAAAAAAH7' ),
  156. ],
  157. 'b32tohex': [
  158. ( ['DPK3PXP'], 'deadbeef' ),
  159. ( ['N5LN657PK3PXP'], 'deadbeefdeadbeef' ),
  160. ( ['P777777777777'], 'ffffffffffffffff' ),
  161. ( ['A','pad=16'], '0000000000000000' ),
  162. ( ['AAAAAAAAAA','pad=16'], '0000000000000000' ),
  163. ( ['AAAAAAAAH7','pad=2'], 'ff' ),
  164. ],
  165. 'hextob6d': [
  166. ( ['deadbeef'], '25255 24636 426' ),
  167. ( ['deadbeefdeadbeef'], '43263 51255 35545 36422 42642' ),
  168. ( ['ffffffffffffffff'], '46316 33121 21321 15553 55534' ),
  169. ( ['0000000000000000'], '1' ),
  170. ( ['0000000000000000','pad=10'], '11111 11111' ),
  171. ( ['ff','pad=10'], '11111 12214' ),
  172. ( ['ff'*16],
  173. '34164 46464 12666 61652 46515 46546 53354 43666 45555 21414' ),
  174. ( ['ff'*24],
  175. '24611 14114 33323 36422 24655 66552 32465 25661 21541 62342 '
  176. '61351 63525 45161 35543 13654' ),
  177. ( ['ff'*32],
  178. '21325 21653 31261 31341 45131 42346 54146 36252 11413 12253 '
  179. '24246 31114 16424 56513 41632 24121 46151 43214 22425 65134' ),
  180. ],
  181. 'b6dtohex': [
  182. ( ['25255 24636 426'], 'deadbeef' ),
  183. ( ['43263 51255 35545 36422 42642'], 'deadbeefdeadbeef' ),
  184. ( ['46316 33121 21321 15553 55534'], 'ffffffffffffffff' ),
  185. ( ['1','pad=16'], '0000000000000000' ),
  186. ( ['11111 11111','pad=16'], '0000000000000000' ),
  187. ( ['11111 12214','pad=2'], 'ff' ),
  188. ( ['22222 22222'], 'b88733' ),
  189. ( ['66666 66666'], '039aa3ff' ),
  190. ( ['6'*50], {
  191. 'len': 34,
  192. 'value':'0260154fc36cbf42778f23ffffffffffff' # 130 bits
  193. } ),
  194. ( ['6'*75], {
  195. 'len': 50,
  196. 'value':'03a92ef1c3432e71a7679561bb6817d7ffffffffffffffffff' # 194 bits
  197. } ),
  198. ( ['6'*100], {
  199. 'len': 66,
  200. 'value':'05a4653ca673768565b41f775d6947d55cf3813d0fffffffffffffffffffffffff' # 259 bits
  201. } ),
  202. ],
  203. 'hextob58chk': [
  204. ( ['deadbeef'], 'eFGDJPketnz' ),
  205. ( ['deadbeefdeadbeef'], '5CizhNNRPYpBjrbYX' ),
  206. ( ['ffffffffffffffff'], '5qCHTcgbQwprzjWrb' ),
  207. ( ['0000000000000000'], '111111114FCKVB' ),
  208. ( ['00'], '1Wh4bh' ),
  209. ( ['000000000000000000000000000000000000000000'], '1111111111111111111114oLvT2' ),
  210. ],
  211. 'b58chktohex': [
  212. ( ['eFGDJPketnz'], 'deadbeef' ),
  213. ( ['5CizhNNRPYpBjrbYX'], 'deadbeefdeadbeef' ),
  214. ( ['5qCHTcgbQwprzjWrb'], 'ffffffffffffffff' ),
  215. ( ['111111114FCKVB'], '0000000000000000' ),
  216. ( ['3QJmnh'], '' ),
  217. ( ['1111111111111111111114oLvT2'], '000000000000000000000000000000000000000000' ),
  218. ],
  219. 'bytestob58': [
  220. ( [b'\xde\xad\xbe\xef'], '6h8cQN' ),
  221. ( [b'\xde\xad\xbe\xef\xde\xad\xbe\xef'], 'eFGDJURJykA' ),
  222. ( [b'\xff\xff\xff\xff\xff\xff\xff\xff'], 'jpXCZedGfVQ' ),
  223. ( [b'\x00\x00\x00\x00\x00\x00\x00\x00'], '1' ),
  224. ( [b'\x00\x00\x00\x00\x00\x00\x00\x00','pad=10'], '1111111111' ),
  225. ( [b'\xff','pad=10'], '111111115Q' ),
  226. ],
  227. 'b58tobytes': [
  228. ( ['6h8cQN'], b'\xde\xad\xbe\xef' ),
  229. ( ['eFGDJURJykA'], b'\xde\xad\xbe\xef\xde\xad\xbe\xef' ),
  230. ( ['jpXCZedGfVQ'], b'\xff\xff\xff\xff\xff\xff\xff\xff' ),
  231. ( ['1','pad=8'], b'\x00\x00\x00\x00\x00\x00\x00\x00' ),
  232. ( ['1111111111','pad=8'], b'\x00\x00\x00\x00\x00\x00\x00\x00' ),
  233. ( ['111111115Q','pad=1'], b'\xff' ),
  234. ],
  235. 'hextob58': [
  236. ( ['deadbeef'], '6h8cQN' ),
  237. ( ['deadbeefdeadbeef'], 'eFGDJURJykA' ),
  238. ( ['ffffffffffffffff'], 'jpXCZedGfVQ' ),
  239. ( ['0000000000000000'], '1' ),
  240. ( ['0000000000000000','pad=10'], '1111111111' ),
  241. ( ['ff','pad=10'], '111111115Q' ),
  242. ],
  243. 'b58tohex': [
  244. ( ['6h8cQN'], 'deadbeef' ),
  245. ( ['eFGDJURJykA'], 'deadbeefdeadbeef' ),
  246. ( ['jpXCZedGfVQ'], 'ffffffffffffffff' ),
  247. ( ['1','pad=16'], '0000000000000000' ),
  248. ( ['1111111111','pad=16'], '0000000000000000' ),
  249. ( ['111111115Q','pad=2'], 'ff' ),
  250. ],
  251. 'bytespec': [
  252. ( ['1G'], str(1024*1024*1024) ),
  253. ( ['1234G'], str(1234*1024*1024*1024) ),
  254. ( ['1GB'], str(1000*1000*1000) ),
  255. ( ['1234GB'], str(1234*1000*1000*1000) ),
  256. ( ['1.234MB'], str(1234*1000) ),
  257. ( ['1.234567M'], str(int(Decimal('1.234567')*1024*1024)) ),
  258. ( ['1234'], str(1234) ),
  259. ],
  260. 'hash160': [ # TODO: check that hextob58chk(hash160) = pubhex2addr
  261. ( ['deadbeef'], 'f04df4c4b30d2b7ac6e1ed2445aeb12a9cb4d2ec' ),
  262. ( ['000000000000000000000000000000000000000000'], '2db95e704e2d9b0474acf76182f3f985b7064a8a' ),
  263. ( [''], 'b472a266d0bd89c13706a4132ccfb16f7c3b9fcb' ),
  264. ( ['ffffffffffffffff'], 'f86221f5a1fca059a865c0b7d374dfa9d5f3aeb4' ),
  265. ],
  266. 'hash256': [
  267. ( ['deadbeef'], 'e107944e77a688feae4c2d4db5951923812dd0f72026a11168104ee1b248f8a9' ),
  268. ( ['000000000000000000000000000000000000000000'], 'fd5181fcd097a334ab340569e5edcd09f702fef7994abab01f4b66e86b32ebbe' ),
  269. ( [''], '5df6e0e2761359d30a8275058e299fcc0381534545f55cf43e41983f5d4c9456' ),
  270. ( ['ffffffffffffffff'], '57b2d2c3455e0f76c61c5237ff04fc9fc0f3fe691e587ea9c951949e1a5e0fed' ),
  271. ],
  272. 'hexdump': [
  273. ( [sample_text.encode()], sample_text_hexdump ),
  274. ],
  275. 'unhexdump': [
  276. ( [sample_text_hexdump.encode()], sample_text.encode() ),
  277. ],
  278. 'hexlify': [
  279. ( [b'foobar'], '666f6f626172' ),
  280. ],
  281. 'unhexlify': [
  282. ( ['666f6f626172'], 'foobar' ),
  283. ],
  284. 'hexreverse': [
  285. ( ['deadbeefcafe'], 'fecaefbeadde' ),
  286. ],
  287. 'id6': [
  288. ( [sample_text.encode()], 'a6d72b' ),
  289. ],
  290. 'id8': [
  291. ( [sample_text.encode()], '687C09C2' ),
  292. ],
  293. 'str2id6': [
  294. ( ['74ev zjeq Zw2g DspF RKpE 7H'], '70413d' ), # checked
  295. ],
  296. 'randhex': [
  297. ( [], {'boolfunc':is_hex_str,'len':64}, ['-r0'] ),
  298. ( ['nbytes=16'], {'boolfunc':is_hex_str,'len':32}, ['-r0'] ),
  299. ( ['nbytes=6'], {'boolfunc':is_hex_str,'len':12}, ['-r0'] ),
  300. ],
  301. 'randb58': [
  302. ( [], {'boolfunc':is_b58_str}, ['-r0'] ),
  303. ( ['nbytes=16'], {'boolfunc':is_b58_str}, ['-r0'] ),
  304. ( ['nbytes=12','pad=0'], is_b58_str, ['-r0'] ),
  305. ],
  306. },
  307. 'Wallet': {
  308. 'gen_key': [
  309. ( ['98831F3A:11','wallet=test/ref/98831F3A.mmwords'],
  310. '5JKLcdYbhP6QQ4BXc9HtjfqJ79FFRXP2SZTKUyEuyXJo9QSFUkv'
  311. ),
  312. ( ['98831F3A:C:11','wallet=test/ref/98831F3A.mmwords'],
  313. 'L2LwXv94XTU2HjCbJPXCFuaHjrjucGipWPWUi1hkM5EykgektyqR'
  314. ),
  315. ( ['98831F3A:B:11','wallet=test/ref/98831F3A.mmwords'],
  316. 'L2K4Y9MWb5oUfKKZtwdgCm6FLZdUiWJDHjh9BYxpEvtfcXt4iM5g'
  317. ),
  318. ( ['98831F3A:S:11','wallet=test/ref/98831F3A.mmwords'],
  319. 'KwmkkfC9GghnJhnKoRXRn5KwGCgXrCmDw6Uv83NzE4kJS5axCR9A'
  320. ),
  321. ],
  322. 'gen_addr': [
  323. ( ['98831F3A:11','wallet=test/ref/98831F3A.mmwords'],
  324. '12bYUGXS8SRArZneQDN9YEEYAtEa59Rykm'
  325. ),
  326. ( ['98831F3A:L:11','wallet=test/ref/98831F3A.mmwords'],
  327. '12bYUGXS8SRArZneQDN9YEEYAtEa59Rykm'
  328. ),
  329. ( ['98831F3A:C:11','wallet=test/ref/98831F3A.mmwords'],
  330. '1MPsZ7BY9qikqfPxqmrovE8gLDX2rYArZk'
  331. ),
  332. ( ['98831F3A:B:11','wallet=test/ref/98831F3A.mmwords'],
  333. 'bc1qxptlvmwaymaxa7pxkr2u5pn7c0508stcncv7ms'
  334. ),
  335. ( ['98831F3A:S:11','wallet=test/ref/98831F3A.mmwords'],
  336. '3Eevao3DRVXnYym3tdrJDqS3Wc39PQzahn'
  337. ),
  338. ],
  339. 'get_subseed': [
  340. ( ['3s','wallet=test/ref/98831F3A.mmwords'], '4018EB17' ),
  341. ( ['200','wallet=test/ref/98831F3A.mmwords'], '2B05AE73' ),
  342. ],
  343. 'get_subseed_by_seed_id': [
  344. ( ['4018EB17','wallet=test/ref/98831F3A.mmwords'], '3S' ),
  345. ( ['2B05AE73','wallet=test/ref/98831F3A.mmwords'], None ),
  346. ( ['2B05AE73','wallet=test/ref/98831F3A.mmwords','last_idx=200'], '200L' ),
  347. ],
  348. 'list_subseeds': [
  349. ( ['1-5','wallet=test/ref/98831F3A.mmwords'],
  350. (md5_hash_strip,'996c047e8543d5dde6f82efc3214a6a1')
  351. ),
  352. ],
  353. 'list_shares': [
  354. ( ['3','wallet=test/ref/98831F3A.bip39'],
  355. (md5_hash_strip,'84e8bdaebf9c816a8a3bd2ebec5a2e12')
  356. ),
  357. ( ['3','id_str=default','wallet=test/ref/98831F3A.mmwords'],
  358. (md5_hash_strip,'84e8bdaebf9c816a8a3bd2ebec5a2e12')
  359. ),
  360. ( ['3','id_str=foo','wallet=test/ref/98831F3A.bip39'],
  361. (md5_hash_strip,'d2ac20823c4ea26f15234b5ca8df5d6f')
  362. ),
  363. ( ['3','id_str=foo','master_share=0','wallet=test/ref/98831F3A.mmwords'],
  364. (md5_hash_strip,'d2ac20823c4ea26f15234b5ca8df5d6f')
  365. ),
  366. ( ['3','id_str=foo','master_share=5','wallet=test/ref/98831F3A.mmwords'],
  367. (md5_hash_strip,'c4feedce40bb5959011ee4a996710832')
  368. ),
  369. ( ['3','id_str=βαρ','master_share=5','wallet=test/ref/98831F3A.mmwords'],
  370. (md5_hash_strip,'f7d254798fe2e34b94b5f4ff312998db')
  371. ),
  372. ( ['4','id_str=βαρ','master_share=5','wallet=test/ref/98831F3A.bip39'],
  373. (md5_hash_strip,'d3e479f55792181372a9f32a569c04e5')
  374. ),
  375. ],
  376. },
  377. 'Coin': {
  378. 'addr2pubhash': {
  379. 'btc_mainnet': [
  380. ( ['12bYUGXS8SRArZneQDN9YEEYAtEa59Rykm'], '118089d66b4a5853765e94923abdd5de4616c6e5' ),
  381. ( ['3Eevao3DRVXnYym3tdrJDqS3Wc39PQzahn'], '8e34586186551f6320fa3eb2d238a9c61ab8264b' ),
  382. ( ['bc1qxptlvmwaymaxa7pxkr2u5pn7c0508stcncv7ms'], '3057f66ddd26fa6ef826b0d5ca067ec3e8f3c178' ),
  383. ],
  384. },
  385. 'pubhash2addr': {
  386. 'btc_mainnet': [
  387. ( ['118089d66b4a5853765e94923abdd5de4616c6e5'], '12bYUGXS8SRArZneQDN9YEEYAtEa59Rykm',
  388. None, 'opt.type="legacy"' ),
  389. ( ['8e34586186551f6320fa3eb2d238a9c61ab8264b'], '3Eevao3DRVXnYym3tdrJDqS3Wc39PQzahn',
  390. ['--type=segwit'], 'opt.type="segwit"' ),
  391. ( ['3057f66ddd26fa6ef826b0d5ca067ec3e8f3c178'], 'bc1qxptlvmwaymaxa7pxkr2u5pn7c0508stcncv7ms',
  392. ['--type=bech32'], 'opt.type="bech32"' ),
  393. ],
  394. },
  395. 'addr2scriptpubkey': {
  396. 'btc_mainnet': [
  397. ( ['12bYUGXS8SRArZneQDN9YEEYAtEa59Rykm'], '76a914118089d66b4a5853765e94923abdd5de4616c6e588ac' ),
  398. ( ['3Eevao3DRVXnYym3tdrJDqS3Wc39PQzahn'], 'a9148e34586186551f6320fa3eb2d238a9c61ab8264b87' ),
  399. ( ['bc1qxptlvmwaymaxa7pxkr2u5pn7c0508stcncv7ms'], '00143057f66ddd26fa6ef826b0d5ca067ec3e8f3c178' ),
  400. ],
  401. },
  402. 'scriptpubkey2addr': {
  403. 'btc_mainnet': [
  404. ( ['76a914118089d66b4a5853765e94923abdd5de4616c6e588ac'], '12bYUGXS8SRArZneQDN9YEEYAtEa59Rykm' ),
  405. ( ['a9148e34586186551f6320fa3eb2d238a9c61ab8264b87'], '3Eevao3DRVXnYym3tdrJDqS3Wc39PQzahn' ),
  406. ( ['00143057f66ddd26fa6ef826b0d5ca067ec3e8f3c178'], 'bc1qxptlvmwaymaxa7pxkr2u5pn7c0508stcncv7ms' ),
  407. ],
  408. },
  409. 'hex2wif': {
  410. 'btc_mainnet': [
  411. ( ['118089d66b4a5853765e94923abdd5de4616c6e5118089d66b4a5853765e9492'],
  412. '5HwzecKMWD82ppJK3qMKpC7ohXXAwcyAN5VgdJ9PLFaAzpBG4sX',
  413. None, 'opt.type="legacy"' ),
  414. ( ['118089d66b4a5853765e94923abdd5de4616c6e5118089d66b4a5853765e9492'],
  415. 'KwojSzt1VvW343mQfWQi3J537siAt5ktL2qbuCg1ZyKR8BLQ6UJm',
  416. ['--type=compressed'], 'opt.type="compressed"' ),
  417. ( ['118089d66b4a5853765e94923abdd5de4616c6e5118089d66b4a5853765e9492'],
  418. 'KwojSzt1VvW343mQfWQi3J537siAt5ktL2qbuCg1ZyKR8BLQ6UJm',
  419. ['--type=segwit'], 'opt.type="segwit"' ),
  420. ( ['118089d66b4a5853765e94923abdd5de4616c6e5118089d66b4a5853765e9492'],
  421. 'KwojSzt1VvW343mQfWQi3J537siAt5ktL2qbuCg1ZyKR8BLQ6UJm',
  422. ['--type=bech32'], 'opt.type="bech32"' ),
  423. ],
  424. },
  425. 'privhex2addr': {
  426. 'btc_mainnet': [
  427. ( ['118089d66b4a5853765e94923abdd5de4616c6e5118089d66b4a5853765e9492'],
  428. '1C5VPtgq9xQ6AcTgMAR3J6GDrs72HC4pS1',
  429. None, 'opt.type="legacy"' ),
  430. ( ['118089d66b4a5853765e94923abdd5de4616c6e5118089d66b4a5853765e9492'],
  431. '1Kz9fVSUMshzPejpzW9D95kScgA3rY6QxF',
  432. ['--type=compressed'], 'opt.type="compressed"' ),
  433. ( ['118089d66b4a5853765e94923abdd5de4616c6e5118089d66b4a5853765e9492'],
  434. '3AhjTiWHhVJAi1s5CfKMcLzYps12x3gZhg',
  435. ['--type=segwit'], 'opt.type="segwit"' ),
  436. ( ['118089d66b4a5853765e94923abdd5de4616c6e5118089d66b4a5853765e9492'],
  437. 'bc1q6pqnfwwakuuejpm9w52ds342f9d5u36v0qnz7c',
  438. ['--type=bech32'], 'opt.type="bech32"' ),
  439. ],
  440. },
  441. 'privhex2pubhex': {
  442. 'btc_mainnet': [
  443. ( ['118089d66b4a5853765e94923abdd5de4616c6e5118089d66b4a5853765e9492'],
  444. '044281a85c9ce87279e028410b851410d65136304cfbbbeaaa8e2e3931cf4e972757f3254c322eeaa3cb6bf97cc5ecf8d4387b0df2c0b1e6ee18fe3a6977a7d57a',
  445. None, 'opt.type="legacy"' ),
  446. ( ['118089d66b4a5853765e94923abdd5de4616c6e5118089d66b4a5853765e9492'],
  447. '024281a85c9ce87279e028410b851410d65136304cfbbbeaaa8e2e3931cf4e9727',
  448. ['--type=compressed'], 'opt.type="compressed"' ),
  449. ( ['118089d66b4a5853765e94923abdd5de4616c6e5118089d66b4a5853765e9492'],
  450. '024281a85c9ce87279e028410b851410d65136304cfbbbeaaa8e2e3931cf4e9727',
  451. ['--type=segwit'], 'opt.type="segwit"' ),
  452. ( ['118089d66b4a5853765e94923abdd5de4616c6e5118089d66b4a5853765e9492'],
  453. '024281a85c9ce87279e028410b851410d65136304cfbbbeaaa8e2e3931cf4e9727',
  454. ['--type=bech32'], 'opt.type="bech32"' ),
  455. ],
  456. },
  457. 'pubhex2addr': {
  458. 'btc_mainnet': [
  459. ( ['044281a85c9ce87279e028410b851410d65136304cfbbbeaaa8e2e3931cf4e972757f3254c322eeaa3cb6bf97cc5ecf8d4387b0df2c0b1e6ee18fe3a6977a7d57a'],
  460. '1C5VPtgq9xQ6AcTgMAR3J6GDrs72HC4pS1',
  461. None, 'opt.type="legacy"' ),
  462. ( ['024281a85c9ce87279e028410b851410d65136304cfbbbeaaa8e2e3931cf4e9727'],
  463. '1Kz9fVSUMshzPejpzW9D95kScgA3rY6QxF',
  464. ['--type=compressed'], 'opt.type="compressed"' ),
  465. ( ['024281a85c9ce87279e028410b851410d65136304cfbbbeaaa8e2e3931cf4e9727'],
  466. '3AhjTiWHhVJAi1s5CfKMcLzYps12x3gZhg',
  467. ['--type=segwit'], 'opt.type="segwit"' ),
  468. ( ['024281a85c9ce87279e028410b851410d65136304cfbbbeaaa8e2e3931cf4e9727'],
  469. 'bc1q6pqnfwwakuuejpm9w52ds342f9d5u36v0qnz7c',
  470. ['--type=bech32'], 'opt.type="bech32"' ),
  471. ],
  472. },
  473. 'pubhex2redeem_script': {
  474. 'btc_mainnet': [
  475. ( ['024281a85c9ce87279e028410b851410d65136304cfbbbeaaa8e2e3931cf4e9727'],
  476. '0014d04134b9ddb7399907657514d846aa495b4e474c',
  477. ['--type=segwit'], 'opt.type="segwit"' ),
  478. ],
  479. },
  480. 'redeem_script2addr': {
  481. 'btc_mainnet': [
  482. ( ['0014d04134b9ddb7399907657514d846aa495b4e474c'],
  483. '3AhjTiWHhVJAi1s5CfKMcLzYps12x3gZhg',
  484. ['--type=segwit'], 'opt.type="segwit"' ),
  485. ],
  486. },
  487. 'randpair': {
  488. 'btc_mainnet': [ ( [], [is_wif,is_coin_addr], ['-r0'] ) ],
  489. 'btc_testnet': [ ( [], [is_wif,is_coin_addr], ['-r0'] ) ],
  490. },
  491. 'randwif': {
  492. 'btc_mainnet': [ ( [], is_wif, ['-r0'] ) ],
  493. 'btc_testnet': [ ( [], is_wif, ['-r0'] ) ],
  494. },
  495. 'wif2addr': {
  496. 'btc_mainnet': [
  497. ( ['5HwzecKMWD82ppJK3qMKpC7ohXXAwcyAN5VgdJ9PLFaAzpBG4sX'],
  498. '1C5VPtgq9xQ6AcTgMAR3J6GDrs72HC4pS1', ['--type=legacy'], 'opt.type="legacy"' ),
  499. ( ['KwojSzt1VvW343mQfWQi3J537siAt5ktL2qbuCg1ZyKR8BLQ6UJm'],
  500. '1Kz9fVSUMshzPejpzW9D95kScgA3rY6QxF', ['--type=compressed'], 'opt.type="compressed"' ),
  501. ( ['KwojSzt1VvW343mQfWQi3J537siAt5ktL2qbuCg1ZyKR8BLQ6UJm'],
  502. '3AhjTiWHhVJAi1s5CfKMcLzYps12x3gZhg', ['--type=segwit'], 'opt.type="segwit"' ),
  503. ( ['KwojSzt1VvW343mQfWQi3J537siAt5ktL2qbuCg1ZyKR8BLQ6UJm'],
  504. 'bc1q6pqnfwwakuuejpm9w52ds342f9d5u36v0qnz7c', ['--type=bech32'], 'opt.type="bech32"' ),
  505. ],
  506. },
  507. 'wif2hex': {
  508. 'btc_mainnet': [
  509. ( ['5HwzecKMWD82ppJK3qMKpC7ohXXAwcyAN5VgdJ9PLFaAzpBG4sX'],
  510. '118089d66b4a5853765e94923abdd5de4616c6e5118089d66b4a5853765e9492',
  511. None, 'opt.type="legacy"' ),
  512. ( ['KwojSzt1VvW343mQfWQi3J537siAt5ktL2qbuCg1ZyKR8BLQ6UJm'],
  513. '118089d66b4a5853765e94923abdd5de4616c6e5118089d66b4a5853765e9492',
  514. ['--type=compressed'], 'opt.type="compressed"' ),
  515. ( ['KwojSzt1VvW343mQfWQi3J537siAt5ktL2qbuCg1ZyKR8BLQ6UJm'],
  516. '118089d66b4a5853765e94923abdd5de4616c6e5118089d66b4a5853765e9492',
  517. ['--type=segwit'], 'opt.type="segwit"' ),
  518. ( ['KwojSzt1VvW343mQfWQi3J537siAt5ktL2qbuCg1ZyKR8BLQ6UJm'],
  519. '118089d66b4a5853765e94923abdd5de4616c6e5118089d66b4a5853765e9492',
  520. ['--type=bech32'], 'opt.type="bech32"' ),
  521. ],
  522. },
  523. 'wif2redeem_script': {
  524. 'btc_mainnet': [
  525. ( ['KwojSzt1VvW343mQfWQi3J537siAt5ktL2qbuCg1ZyKR8BLQ6UJm'],
  526. '0014d04134b9ddb7399907657514d846aa495b4e474c',
  527. ['--type=segwit'], 'opt.type="segwit"' ),
  528. ],
  529. },
  530. 'wif2segwit_pair': {
  531. 'btc_mainnet': [
  532. ( ['KwojSzt1VvW343mQfWQi3J537siAt5ktL2qbuCg1ZyKR8BLQ6UJm'],
  533. ('0014d04134b9ddb7399907657514d846aa495b4e474c','3AhjTiWHhVJAi1s5CfKMcLzYps12x3gZhg'),
  534. ['--type=segwit'], 'opt.type="segwit"' ),
  535. ],
  536. },
  537. },
  538. # TODO: compressed address files are missing
  539. # 'addrfile_compressed_chk':
  540. # 'btc': ('A33C 4FDE F515 F5BC','6C48 AA57 2056 C8C8'),
  541. # 'ltc': ('3FC0 8F03 C2D6 BD19','4C0A 49B6 2DD1 1BE0'),
  542. 'File': {
  543. 'addrfile_chksum': {
  544. 'btc_mainnet': [
  545. ( ['test/ref/98831F3A[1,31-33,500-501,1010-1011].addrs'],
  546. '6FEF 6FB9 7B13 5D91'),
  547. ( ['test/ref/98831F3A-S[1,31-33,500-501,1010-1011].addrs'],
  548. '06C1 9C87 F25C 4EE6'),
  549. ( ['test/ref/98831F3A-B[1,31-33,500-501,1010-1011].addrs'],
  550. '9D2A D4B6 5117 F02E'),
  551. ],
  552. 'btc_testnet': [
  553. ( ['test/ref/98831F3A[1,31-33,500-501,1010-1011].testnet.addrs'],
  554. '424E 4326 CFFE 5F51'),
  555. ( ['test/ref/98831F3A-S[1,31-33,500-501,1010-1011].testnet.addrs'],
  556. '072C 8B07 2730 CB7A'),
  557. ( ['test/ref/98831F3A-B[1,31-33,500-501,1010-1011].testnet.addrs'],
  558. '0527 9C39 6C1B E39A'),
  559. ],
  560. 'ltc_mainnet': [
  561. ( ['test/ref/litecoin/98831F3A-LTC[1,31-33,500-501,1010-1011].addrs'],
  562. 'AD52 C3FE 8924 AAF0'),
  563. ( ['test/ref/litecoin/98831F3A-LTC-S[1,31-33,500-501,1010-1011].addrs'],
  564. '63DF E42A 0827 21C3'),
  565. ( ['test/ref/litecoin/98831F3A-LTC-B[1,31-33,500-501,1010-1011].addrs'],
  566. 'FF1C 7939 5967 AB82'),
  567. ],
  568. 'ltc_testnet': [
  569. ( ['test/ref/litecoin/98831F3A-LTC[1,31-33,500-501,1010-1011].testnet.addrs'],
  570. '4EBE 2E85 E969 1B30'),
  571. ( ['test/ref/litecoin/98831F3A-LTC-S[1,31-33,500-501,1010-1011].testnet.addrs'],
  572. '5DD1 D186 DBE1 59F2'),
  573. ( ['test/ref/litecoin/98831F3A-LTC-B[1,31-33,500-501,1010-1011].testnet.addrs'],
  574. 'ED3D 8AA4 BED4 0B40'),
  575. ],
  576. 'zec_mainnet': [
  577. ( ['test/ref/zcash/98831F3A-ZEC-C[1,31-33,500-501,1010-1011].addrs'],'903E 7225 DD86 6E01'),
  578. ( ['test/ref/zcash/98831F3A-ZEC-Z[1,31-33,500-501,1010-1011].addrs'], '9C7A 72DC 3D4A B3AF',
  579. ['--type=zcash_z'], 'opt.type = "zcash_z"' ),
  580. ],
  581. 'xmr_mainnet': [
  582. ( ['test/ref/monero/98831F3A-XMR-M[1,31-33,500-501,1010-1011].addrs'],'4369 0253 AC2C 0E38'), ],
  583. 'dash_mainnet': [
  584. ( ['test/ref/dash/98831F3A-DASH-C[1,31-33,500-501,1010-1011].addrs'],'FBC1 6B6A 0988 4403'), ],
  585. 'eth_mainnet': [
  586. ( ['test/ref/ethereum/98831F3A-ETH[1,31-33,500-501,1010-1011].addrs'],'E554 076E 7AF6 66A3'), ],
  587. 'etc_mainnet': [
  588. ( ['test/ref/ethereum_classic/98831F3A-ETC[1,31-33,500-501,1010-1011].addrs'],
  589. 'E97A D796 B495 E8BC'), ],
  590. },
  591. 'keyaddrfile_chksum': {
  592. 'btc_mainnet': [
  593. ( ['test/ref/98831F3A[1,31-33,500-501,1010-1011].akeys.mmenc'],
  594. '9F2D D781 1812 8BAD', kafile_opts, kafile_code ),
  595. ],
  596. 'btc_testnet': [
  597. ( ['test/ref/98831F3A[1,31-33,500-501,1010-1011].testnet.akeys.mmenc'],
  598. '88CC 5120 9A91 22C2', kafile_opts, kafile_code ),
  599. ],
  600. 'ltc_mainnet': [
  601. ( ['test/ref/litecoin/98831F3A-LTC[1,31-33,500-501,1010-1011].akeys.mmenc'],
  602. 'B804 978A 8796 3ED4', kafile_opts, kafile_code ),
  603. ],
  604. 'ltc_testnet': [
  605. ( ['test/ref/litecoin/98831F3A-LTC[1,31-33,500-501,1010-1011].testnet.akeys.mmenc'],
  606. '98B5 AC35 F334 0398', kafile_opts, kafile_code ),
  607. ],
  608. 'zec_mainnet': [
  609. ( ['test/ref/zcash/98831F3A-ZEC-C[1,31-33,500-501,1010-1011].akeys.mmenc'],
  610. 'F05A 5A5C 0C8E 2617', kafile_opts, kafile_code ),
  611. ( ['test/ref/zcash/98831F3A-ZEC-Z[1,31-33,500-501,1010-1011].akeys.mmenc'], '6B87 9B2D 0D8D 8D1E',
  612. kafile_opts + ['--type=zcash_z'], kafile_code + '\nopt.type = "zcash_z"' ),
  613. ],
  614. 'xmr_mainnet': [
  615. ( ['test/ref/monero/98831F3A-XMR-M[1,31-33,500-501,1010-1011].akeys.mmenc'],
  616. 'E0D7 9612 3D67 404A', kafile_opts, kafile_code ), ],
  617. 'dash_mainnet': [
  618. ( ['test/ref/dash/98831F3A-DASH-C[1,31-33,500-501,1010-1011].akeys.mmenc'],
  619. 'E83D 2C63 FEA2 4142', kafile_opts, kafile_code ), ],
  620. 'eth_mainnet': [
  621. ( ['test/ref/ethereum/98831F3A-ETH[1,31-33,500-501,1010-1011].akeys.mmenc'],
  622. 'E400 70D9 0AE3 C7C2', kafile_opts, kafile_code ), ],
  623. 'etc_mainnet': [
  624. ( ['test/ref/ethereum_classic/98831F3A-ETC[1,31-33,500-501,1010-1011].akeys.mmenc'],
  625. 'EF49 967D BD6C FE45', kafile_opts, kafile_code ), ],
  626. },
  627. 'passwdfile_chksum': {
  628. 'btc_mainnet': [
  629. ( ['test/ref/98831F3A-фубар@crypto.org-b58-20[1,4,1100].pws'],
  630. 'DDD9 44B0 CA28 183F', kafile_opts, kafile_code ), ],
  631. },
  632. 'txview': {
  633. 'btc_mainnet': [ ( ['test/ref/0B8D5A[15.31789,14,tl=1320969600].rawtx'], None ), ],
  634. 'btc_testnet': [ ( ['test/ref/0C7115[15.86255,14,tl=1320969600].testnet.rawtx'], None ), ],
  635. 'bch_mainnet': [ ( ['test/ref/460D4D-BCH[10.19764,tl=1320969600].rawtx'], None ), ],
  636. 'bch_testnet': [ ( ['test/ref/359FD5-BCH[6.68868,tl=1320969600].testnet.rawtx'], None ), ],
  637. 'ltc_mainnet': [ ( ['test/ref/litecoin/AF3CDF-LTC[620.76194,1453,tl=1320969600].rawtx'], None ), ],
  638. 'ltc_testnet': [ ( ['test/ref/litecoin/A5A1E0-LTC[1454.64322,1453,tl=1320969600].testnet.rawtx'],
  639. None ), ],
  640. 'eth_mainnet': [ ( ['test/ref/ethereum/88FEFD-ETH[23.45495,40000].rawtx'], None ), ],
  641. 'eth_testnet': [ ( ['test/ref/ethereum/B472BD-ETH[23.45495,40000].testnet.rawtx'], None ), ],
  642. 'mm1_mainnet': [ ( ['test/ref/ethereum/5881D2-MM1[1.23456,50000].rawtx'], None ), ],
  643. 'mm1_testnet': [ ( ['test/ref/ethereum/6BDB25-MM1[1.23456,50000].testnet.rawtx'], None ), ],
  644. 'etc_mainnet': [ ( ['test/ref/ethereum_classic/ED3848-ETC[1.2345,40000].rawtx'], None ), ],
  645. },
  646. },
  647. }
  648. coin_dependent_groups = ('Coin','File') # TODO: do this as attr of each group in tool.py
  649. def run_test(gid,cmd_name):
  650. data = tests[gid][cmd_name]
  651. # behavior is like test.py: run coin-dependent tests only if g.testnet or g.coin != BTC
  652. if gid in coin_dependent_groups:
  653. k = '{}_{}net'.format((g.token.lower() if g.token else g.coin.lower()),('main','test')[g.testnet])
  654. if k in data:
  655. data = data[k]
  656. m2 = ' ({})'.format(k)
  657. else:
  658. qmsg("-- no data for {} ({}) - skipping".format(cmd_name,k))
  659. return
  660. else:
  661. if g.coin != 'BTC' or g.testnet: return
  662. m2 = ''
  663. m = '{} {}{}'.format(purple('Testing'), cmd_name if opt.names else
  664. extract_docstring(getattr(getattr(tool,'MMGenToolCmd'+gid),cmd_name)),m2)
  665. msg_r(green(m)+'\n' if opt.verbose else m)
  666. def fork_cmd(cmd_name,args,out,opts,exec_code):
  667. cmd = list(tool_cmd) + (opts or []) + [cmd_name] + args
  668. vmsg('{} {}'.format(green('Executing'),cyan(' '.join(cmd))))
  669. cp = run(cmd,input=stdin_input or None,stdout=PIPE,stderr=PIPE)
  670. try: cmd_out = cp.stdout.decode()
  671. except: cmd_out = cp.stdout
  672. if cp.stderr:
  673. vmsg(cp.stderr.strip().decode())
  674. if cp.returncode != 0:
  675. import re
  676. m = re.match(b"tool command returned '(None|False)'"+NL.encode(),cp.stderr)
  677. if m:
  678. return { b'None': None, b'False': False }[m.group(1)]
  679. else:
  680. ydie(1,'Spawned program exited with error: {}'.format(cp.stderr))
  681. return cmd_out.strip()
  682. def run_func(cmd_name,args,out,opts,exec_code):
  683. vmsg('{}: {}{}'.format(purple('Running'),
  684. ' '.join([cmd_name]+[repr(e) for e in args]),
  685. ' '+exec_code if exec_code else '' ))
  686. if exec_code: exec(exec_code)
  687. aargs,kwargs = tool._process_args(cmd_name,args)
  688. oq_save = opt.quiet
  689. if not opt.verbose: opt.quiet = True
  690. if stdin_input:
  691. fd0,fd1 = os.pipe()
  692. if os.fork(): # parent
  693. os.close(fd1)
  694. stdin_save = os.dup(0)
  695. os.dup2(fd0,0)
  696. cmd_out = getattr(tc,cmd_name)(*aargs,**kwargs)
  697. os.dup2(stdin_save,0)
  698. os.wait()
  699. opt.quiet = oq_save
  700. return cmd_out
  701. else: # child
  702. os.close(fd0)
  703. os.write(fd1,stdin_input)
  704. vmsg('Input: {!r}'.format(stdin_input))
  705. sys.exit(0)
  706. else:
  707. ret = getattr(tc,cmd_name)(*aargs,**kwargs)
  708. opt.quiet = oq_save
  709. return ret
  710. for d in data:
  711. args,out,opts,exec_code = d + tuple([None] * (4-len(d)))
  712. stdin_input = None
  713. if args and type(args[0]) == bytes:
  714. stdin_input = args[0]
  715. args[0] = '-'
  716. if opt.fork:
  717. cmd_out = fork_cmd(cmd_name,args,out,opts,exec_code)
  718. else:
  719. if stdin_input and g.platform == 'win':
  720. msg('Skipping for MSWin - no os.fork()')
  721. continue
  722. cmd_out = run_func(cmd_name,args,out,opts,exec_code)
  723. try: vmsg('Output:\n{}\n'.format(cmd_out))
  724. except: vmsg('Output:\n{}\n'.format(repr(cmd_out)))
  725. def check_output(out,chk):
  726. if isinstance(chk,str): chk = chk.encode()
  727. if isinstance(out,int): out = str(out).encode()
  728. if isinstance(out,str): out = out.encode()
  729. err_fs = "Output ({!r}) doesn't match expected output ({!r})"
  730. try: outd = out.decode()
  731. except: outd = None
  732. if type(chk).__name__ == 'function':
  733. assert chk(outd),"{}({}) failed!".format(chk.__name__,outd)
  734. elif type(chk) == dict:
  735. for k,v in chk.items():
  736. if k == 'boolfunc':
  737. assert v(outd),"{}({}) failed!".format(v.__name__,outd)
  738. elif k == 'value':
  739. assert outd == v, err_fs.format(outd,v)
  740. else:
  741. outval = getattr(__builtins__,k)(out)
  742. if outval != v:
  743. die(1,"{}({}) returned {}, not {}!".format(k,out,outval,v))
  744. elif chk is not None:
  745. assert out == chk, err_fs.format(out,chk)
  746. if type(out) == tuple and type(out[0]).__name__ == 'function':
  747. func_out = out[0](cmd_out)
  748. assert func_out == out[1],(
  749. "{}({}) == {} failed!\nOutput: {}".format(out[0].__name__,cmd_out,out[1],func_out))
  750. elif isinstance(out,(list,tuple)):
  751. for co,o in zip(cmd_out.split(NL) if opt.fork else cmd_out,out):
  752. check_output(co,o)
  753. else:
  754. check_output(cmd_out,out)
  755. if not opt.verbose: msg_r('.')
  756. if not opt.verbose:
  757. msg('OK')
  758. def extract_docstring(obj):
  759. return obj.__doc__.strip().split('\n')[0]
  760. def do_group(gid):
  761. qmsg(blue("Testing {}".format(
  762. "command group '{}'".format(gid) if opt.names
  763. else extract_docstring(getattr(tool,'MMGenToolCmd'+gid)))))
  764. for cname in [e for e in dir(getattr(tool,'MMGenToolCmd'+gid)) if e[0] != '_']:
  765. if cname not in tests[gid]:
  766. m = 'No test for command {!r} in group {!r}!'.format(cname,gid)
  767. if opt.die_on_missing:
  768. die(1,m+' Aborting')
  769. else:
  770. msg(m)
  771. continue
  772. run_test(gid,cname)
  773. def do_cmd_in_group(cmd):
  774. for gid in tests:
  775. for cname in tests[gid]:
  776. if cname == cmd:
  777. run_test(gid,cname)
  778. return True
  779. return False
  780. def list_tested_cmds():
  781. for gid in tests:
  782. for cname in [e for e in dir(getattr(tool,'MMGenToolCmd'+gid)) if e[0] != '_']:
  783. Msg(cname)
  784. sys.argv = [sys.argv[0]] + ['--skip-cfg-file'] + sys.argv[1:]
  785. cmd_args = opts.init(opts_data,add_opts=['use_old_ed25519'])
  786. import mmgen.tool as tool
  787. if opt.list_tests:
  788. Msg('Available tests:')
  789. for gid in tests:
  790. Msg(' {:6} - {}'.format(gid,extract_docstring(getattr(tool,'MMGenToolCmd'+gid))))
  791. sys.exit(0)
  792. if opt.list_tested_cmds:
  793. list_tested_cmds()
  794. sys.exit(0)
  795. if opt.system:
  796. tool_exec = 'mmgen-tool'
  797. sys.path.pop(0)
  798. else:
  799. os.environ['PYTHONPATH'] = repo_root
  800. tool_exec = os.path.relpath(os.path.join('cmds','mmgen-tool'))
  801. if opt.fork:
  802. tool_cmd = (tool_exec,'--skip-cfg-file')
  803. passthru_args = ['coin','type','testnet','token']
  804. tool_cmd += tuple(['--{}{}'.format(k.replace('_','-'),
  805. '='+getattr(opt,k) if getattr(opt,k) != True else ''
  806. ) for k in passthru_args if getattr(opt,k)])
  807. if opt.traceback:
  808. tool_cmd = (os.path.join('scripts','traceback_run.py'),) + tool_cmd
  809. if opt.coverage:
  810. d,f = init_coverage()
  811. tool_cmd = ('python3','-m','trace','--count','--coverdir='+d,'--file='+f) + tool_cmd
  812. elif g.platform == 'win':
  813. tool_cmd = ('python3',) + tool_cmd
  814. else:
  815. opt.usr_randchars = 0
  816. tc = tool.MMGenToolCmd()
  817. start_time = int(time.time())
  818. try:
  819. if cmd_args:
  820. for cmd in cmd_args:
  821. if cmd in tests:
  822. do_group(cmd)
  823. else:
  824. if not do_cmd_in_group(cmd):
  825. die(1,"'{}': not a recognized test or test group".format(cmd))
  826. else:
  827. for garg in tests:
  828. do_group(garg)
  829. except KeyboardInterrupt:
  830. die(1,green('\nExiting at user request'))
  831. t = int(time.time()) - start_time
  832. gmsg('All requested tests finished OK, elapsed time: {:02}:{:02}'.format(t//60,t%60))