123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248 |
- #!/usr/bin/env python3
- """
- test.unit_tests_d.ut_baseconv: Base conversion unit test for the MMGen suite
- """
- from mmgen.util import msg,msg_r
- from ..include.common import cfg,qmsg,qmsg_r,vmsg,vmsg_r
- class unit_test:
- vectors = {
- 'b58': (
- (('00',None),'1'),
- (('00',1),'1'),
- (('00',2),'11'),
- (('01',None),'2'),
- (('01',1),'2'),
- (('01',2),'12'),
- (('0f',None),'G'),
- (('0f',1),'G'),
- (('0f',2),'1G'),
- (('deadbeef',None),'6h8cQN'),
- (('deadbeef',20),'111111111111116h8cQN'),
- (('00000000',None),'1'),
- (('00000000',20),'11111111111111111111'),
- (('ffffffff',None),'7YXq9G'),
- (('ffffffff',20),'111111111111117YXq9G'),
- (('ff'*16,'seed'),'YcVfxkQb6JRzqk5kF2tNLv'),
- (('ff'*24,'seed'),'QLbz7JHiBTspS962RLKV8GndWFwiEaqKL'),
- (('ff'*32,'seed'),'JEKNVnkbo3jma5nREBBJCDoXFVeKkD56V3xKrvRmWxFG'),
- (('00'*16,'seed'),'1111111111111111111111'),
- (('00'*24,'seed'),'111111111111111111111111111111111'),
- (('00'*32,'seed'),'11111111111111111111111111111111111111111111'),
- ),
- # MMGen-flavored base32 using simple base conversion
- 'b32': (
- (('00',None),'A'),
- (('00',1),'A'),
- (('00',2),'AA'),
- (('01',None),'B'),
- (('01',1),'B'),
- (('01',2),'AB'),
- (('0f',None),'P'),
- (('0f',1),'P'),
- (('0f',2),'AP'),
- (('deadbeef',None),'DPK3PXP'),
- (('deadbeef',20),'AAAAAAAAAAAAADPK3PXP'),
- (('00000000',None),'A'),
- (('00000000',20),'AAAAAAAAAAAAAAAAAAAA'),
- (('ffffffff',None),'D777777'),
- (('ffffffff',20),'AAAAAAAAAAAAAD777777'),
- ),
- 'b16': (
- (('00',None),'0'),
- (('00',1),'0'),
- (('00',2),'00'),
- (('01',None),'1'),
- (('01',1),'1'),
- (('01',2),'01'),
- (('0f',None),'f'),
- (('0f',1),'f'),
- (('0f',2),'0f'),
- (('deadbeef',None),'deadbeef'),
- (('deadbeef',20),'000000000000deadbeef'),
- (('00000000',None),'0'),
- (('00000000',20),'00000000000000000000'),
- (('ffffffff',None),'ffffffff'),
- (('ffffffff',20),'000000000000ffffffff'),
- ),
- 'b10': (
- (('00',None),'0'),
- (('00',1),'0'),
- (('00',2),'00'),
- (('01',None),'1'),
- (('01',1),'1'),
- (('01',2),'01'),
- (('0f',None),'15'),
- (('0f',1),'15'),
- (('0f',2),'15'),
- (('deadbeef',None),'3735928559'),
- (('deadbeef',20),'00000000003735928559'),
- (('00000000',None),'0'),
- (('00000000',20),'00000000000000000000'),
- (('ffffffff',None),'4294967295'),
- (('ffffffff',20),'00000000004294967295'),
- ),
- 'b8': (
- (('00',None),'0'),
- (('00',1),'0'),
- (('00',2),'00'),
- (('01',None),'1'),
- (('01',1),'1'),
- (('01',2),'01'),
- (('0f',None),'17'),
- (('0f',1),'17'),
- (('0f',2),'17'),
- (('deadbeef',None),'33653337357'),
- (('deadbeef',20),'00000000033653337357'),
- (('00000000',None),'0'),
- (('00000000',20),'00000000000000000000'),
- (('ffffffff',None),'37777777777'),
- (('ffffffff',20),'00000000037777777777'),
- ),
- 'b6d': (
- (('00',None),'1'),
- (('00',1),'1'),
- (('00',2),'11'),
- (('01',None),'2'),
- (('01',1),'2'),
- (('01',2),'12'),
- (('0f',None),'34'),
- (('0f',1),'34'),
- (('0f',2),'34'),
- (('010f',None),'2242'),
- (('010f',20),'11111111111111112242'),
- (('deadbeef',None),'2525524636426'),
- (('deadbeef',20),'11111112525524636426'),
- (('00000000',None),'1'),
- (('00000000',20),'11111111111111111111'),
- (('ffffffff',None),'2661215126614'),
- (('ffffffff',20),'11111112661215126614'),
- (('ff'*16,'seed'),'34164464641266661652465154654653354436664555521414'),
- (('ff'*24,'seed'),'246111411433323364222465566552324652566121541623426135163525451613554313654'),
- (('ff'*32,'seed'),'2132521653312613134145131423465414636252114131225324246311141642456513416322412146151432142242565134'),
- (('00'*16,'seed'),'1'*50),
- (('00'*24,'seed'),'1'*75),
- (('00'*32,'seed'),'1'*100),
- ),
- 'mmgen': (
- (('deadbeefdeadbeefdeadbeefdeadbeef','seed'),
- 'table cast forgive master funny gaze sadness ripple million paint moral match' ),
- (('deadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeef','seed'),
- 'swirl maybe anymore mix scale stray fog use approach page crime rhyme ' +
- 'class former strange window snap soon'),
- (('deadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeef','seed'),
- 'swell type milk figure cheese phone fill black test bloom heard comfort ' +
- 'image terrible radio lesson own reply battle goal goodbye need laugh stream'),
- (('ffffffffffffffffffffffffffffffff','seed'),
- 'yellow yeah show bowl season spider cling defeat poison law shelter reflect'),
- (('ffffffffffffffffffffffffffffffffffffffffffffffff','seed'),
- 'yeah youth quit fail perhaps drum out person young click skin ' +
- 'weird inside silently perfectly together anyone memory'),
- (('ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff','seed'),
- 'wrote affection object cell opinion here laughter stare honest north cost begin ' +
- 'murder something yourself effort acid dot doubt game broke tell guilt innocent'),
- (('00000000000000000000000000000001','seed'),
- 'able ' * 11 + 'about'),
- (('000000000000000000000000000000000000000000000001','seed'),
- 'able ' * 17 + 'about'),
- (('0000000000000000000000000000000000000000000000000000000000000001','seed'),
- 'able ' * 23 + 'about'),
- ),
- }
- def run_test(self,name,ut):
- msg_r('Testing base conversion routines...')
- from mmgen.baseconv import baseconv
- perr = "length of {!r} less than pad length ({})"
- rerr = "return value ({!r}) does not match reference value ({!r})"
- qmsg_r('\nChecking hex-to-base conversion:')
- for base,data in self.vectors.items():
- fs = " {h:%s} {p:<6} {r}" % max(len(d[0][0]) for d in data)
- if not cfg.verbose:
- qmsg_r(f' {base}')
- vmsg(f'\nBase: {base}')
- vmsg(fs.format(h='Input',p='Pad',r='Output'))
- for (hexstr,pad),ret_chk in data:
- ret = baseconv(base).fromhex(hexstr,pad=pad,tostr=True)
- if pad != 'seed':
- assert len(ret) >= (pad or 0), perr.format(ret,pad or 0)
- assert ret == ret_chk, rerr.format(ret,ret_chk)
- vmsg(fs.format(h=hexstr,r=ret,p=str(pad)))
- # msg("(('{h}',{p}),'{r}'),".format(h=hexstr,r=ret,c=ret_chk,p=pad))
- # msg('')
- # return True
- qmsg_r('\nChecking base-to-hex conversion:')
- for base,data in self.vectors.items():
- fs = " {h:%s} {p:<6} {r}" % max(len(d[1]) for d in data)
- if not cfg.verbose:
- qmsg_r(f' {base}')
- vmsg(f'\nBase: {base}')
- vmsg(fs.format(h='Input',p='Pad',r='Output'))
- for (hexstr,pad),ret_chk in data:
- if type(pad) is int:
- pad = len(hexstr)
- ret = baseconv(base).tohex( ret_chk.split() if base == 'mmgen' else ret_chk, pad=pad )
- if pad is None:
- assert int(ret,16) == int(hexstr,16), rerr.format(int(ret,16),int(hexstr,16))
- else:
- assert ret == hexstr, rerr.format(ret,hexstr)
- vmsg(fs.format(h=ret_chk,r=ret,p=str(pad)))
- # msg("(('{h}',{p}),'{r}'),".format(h=hexstr,r=ret_chk,c=ret_chk,p=pad))
- qmsg_r('\nChecking wordlist checksums:')
- vmsg('')
- for wl_id in baseconv.constants['wl_chksum']:
- vmsg_r(f' {wl_id+":":9}')
- baseconv(wl_id).check_wordlist(cfg)
- qmsg('')
- qmsg('Checking error handling:')
- bad_b58 = 'I'*22
- bad_b58len = 'a'*23
- fr58 = baseconv('b58').fromhex
- to58 = baseconv('b58').tohex
- to32 = baseconv('b32').tohex
- to8 = baseconv('b8').tohex
- hse = 'HexadecimalStringError'
- bce = 'BaseConversionError'
- bpe = 'BaseConversionPadError'
- sle = 'SeedLengthError'
- bad_data = (
- ('hexstr', hse, ': not a hexadecimal str', lambda:fr58('x')),
- ('hexstr (seed)', hse, 'seed data not a hexadec', lambda:fr58('x',pad='seed')),
- ('hexstr (empty)', bce, 'empty data not allowed', lambda:fr58('')),
- ('b58 data', bce, ': not in base58', lambda:to58('IfFzZ')),
- ('b58 data (seed)', bce, 'seed data not in base58', lambda:to58(bad_b58,pad='seed')),
- ('b58 len (seed)', bce, 'invalid length for', lambda:to58(bad_b58len,pad='seed')),
- ('b58 data (empty)', bce, 'empty base58 data', lambda:to58('')),
- ('b8 data (empty)' , bce, 'empty base8 string data', lambda:to8('')),
- ('b32 data', bce, 'not in MMGen base32', lambda:to32('1az')),
- ('pad arg (in)', bpe, "illegal value for 'pad'", lambda:fr58('ff',pad='foo')),
- ('pad arg (in)', bpe, "illegal value for 'pad'", lambda:fr58('ff',pad=False)),
- ('pad arg (in)', bpe, "illegal value for 'pad'", lambda:fr58('ff',pad=True)),
- ('seedlen (in)', sle, 'invalid byte length', lambda:fr58('ff',pad='seed')),
- ('pad arg (out)', bpe, "illegal value for 'pad'", lambda:to58('Z',pad='foo')),
- ('pad arg (out)', bpe, "illegal value for 'pad'", lambda:to58('Z',pad=False)),
- ('pad arg (out)', bpe, "illegal value for 'pad'", lambda:to58('Z',pad=True)),
- ('seedlen (out)', bce, 'invalid length for seed', lambda:to58('Z',pad='seed')),
- )
- ut.process_bad_data(bad_data)
- msg('OK')
- return True
|