seed.py 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053
  1. #!/usr/bin/env python
  2. #
  3. # mmgen = Multi-Mode GENerator, command-line Bitcoin cold storage solution
  4. # Copyright (C)2013-2017 Philemon <mmgen-py@yandex.com>
  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. seed.py: Seed-related classes and methods for the MMGen suite
  20. """
  21. import os
  22. from binascii import hexlify,unhexlify
  23. from mmgen.common import *
  24. from mmgen.obj import *
  25. from mmgen.filename import *
  26. from mmgen.crypto import *
  27. pnm = g.proj_name
  28. def check_usr_seed_len(seed_len):
  29. if opt.seed_len != seed_len and 'seed_len' in opt.set_by_user:
  30. m = 'ERROR: requested seed length (%s) ' + \
  31. "doesn't match seed length of source (%s)"
  32. die(1, m % (opt.seed_len,seed_len))
  33. class Seed(MMGenObject):
  34. def __init__(self,seed_bin=None):
  35. if not seed_bin:
  36. # Truncate random data for smaller seed lengths
  37. seed_bin = sha256(get_random(1033)).digest()[:opt.seed_len/8]
  38. elif len(seed_bin)*8 not in g.seed_lens:
  39. die(3,'%s: invalid seed length' % len(seed_bin))
  40. self.data = seed_bin
  41. self.hexdata = hexlify(seed_bin)
  42. self.sid = SeedID(seed=self)
  43. self.length = len(seed_bin) * 8
  44. def get_data(self):
  45. return self.data
  46. class SeedSource(MMGenObject):
  47. desc = g.proj_name + ' seed source'
  48. file_mode = 'text'
  49. stdin_ok = False
  50. ask_tty = True
  51. no_tty = False
  52. op = None
  53. _msg = {}
  54. class SeedSourceData(MMGenObject): pass
  55. def __new__(cls,fn=None,ss=None,seed=None,ignore_in_fmt=False,passchg=False):
  56. def die_on_opt_mismatch(opt,sstype):
  57. opt_sstype = cls.fmt_code_to_type(opt)
  58. compare_or_die(
  59. opt_sstype.__name__, 'input format requested on command line',
  60. sstype.__name__, 'input file format'
  61. )
  62. if ss:
  63. sstype = ss.__class__ if passchg else cls.fmt_code_to_type(opt.out_fmt)
  64. me = super(cls,cls).__new__(sstype or Wallet) # default: Wallet
  65. me.seed = ss.seed
  66. me.ss_in = ss
  67. me.op = ('conv','pwchg_new')[bool(passchg)]
  68. elif fn or opt.hidden_incog_input_params:
  69. if fn:
  70. f = Filename(fn)
  71. else:
  72. # permit comma in filename
  73. fn = ','.join(opt.hidden_incog_input_params.split(',')[:-1])
  74. f = Filename(fn,ftype=IncogWalletHidden)
  75. if opt.in_fmt and not ignore_in_fmt:
  76. die_on_opt_mismatch(opt.in_fmt,f.ftype)
  77. me = super(cls,cls).__new__(f.ftype)
  78. me.infile = f
  79. me.op = ('old','pwchg_old')[bool(passchg)]
  80. elif opt.in_fmt: # Input format
  81. sstype = cls.fmt_code_to_type(opt.in_fmt)
  82. me = super(cls,cls).__new__(sstype)
  83. me.op = ('old','pwchg_old')[bool(passchg)]
  84. else: # Called with no inputs - initialize with random seed
  85. sstype = cls.fmt_code_to_type(opt.out_fmt)
  86. me = super(cls,cls).__new__(sstype or Wallet) # default: Wallet
  87. me.seed = Seed(seed_bin=seed or None)
  88. me.op = 'new'
  89. # die(1,me.seed.sid.hl()) # DEBUG
  90. return me
  91. def __init__(self,fn=None,ss=None,seed=None,ignore_in_fmt=False,passchg=False):
  92. self.ssdata = self.SeedSourceData()
  93. self.msg = {}
  94. for c in reversed(self.__class__.__mro__):
  95. if hasattr(c,'_msg'):
  96. self.msg.update(c._msg)
  97. if hasattr(self,'seed'):
  98. self._encrypt()
  99. return
  100. elif hasattr(self,'infile') or not g.stdin_tty:
  101. self._deformat_once()
  102. self._decrypt_retry()
  103. else:
  104. if not self.stdin_ok:
  105. die(1,'Reading from standard input not supported for %s format'
  106. % self.desc)
  107. self._deformat_retry()
  108. self._decrypt_retry()
  109. m = ('',', seed length %s' % self.seed.length)[self.seed.length!=256]
  110. qmsg('Valid %s for Seed ID %s%s' % (self.desc,self.seed.sid.hl(),m))
  111. def _get_data(self):
  112. if hasattr(self,'infile'):
  113. self.fmt_data = get_data_from_file(self.infile.name,self.desc,
  114. binary=self.file_mode=='binary')
  115. else:
  116. self.fmt_data = self._get_data_from_user(self.desc)
  117. def _get_data_from_user(self,desc):
  118. return get_data_from_user(desc)
  119. def _deformat_once(self):
  120. self._get_data()
  121. if not self._deformat():
  122. die(2,'Invalid format for input data')
  123. def _deformat_retry(self):
  124. while True:
  125. self._get_data()
  126. if self._deformat(): break
  127. msg('Trying again...')
  128. def _decrypt_retry(self):
  129. while True:
  130. if self._decrypt(): break
  131. if opt.passwd_file:
  132. die(2,'Passphrase from password file, so exiting')
  133. msg('Trying again...')
  134. @classmethod
  135. def get_subclasses_str(cls): # returns name of calling class too
  136. return cls.__name__ + ' ' + ''.join([c.get_subclasses_str() for c in cls.__subclasses__()])
  137. @classmethod
  138. def get_subclasses_easy(cls,acc=[]):
  139. return [globals()[c] for c in cls.get_subclasses_str().split()]
  140. @classmethod
  141. def get_subclasses(cls): # returns calling class too
  142. def GetSubclassesTree(cls,acc):
  143. acc += [cls]
  144. for c in cls.__subclasses__(): GetSubclassesTree(c,acc)
  145. acc = []
  146. GetSubclassesTree(cls,acc)
  147. return acc
  148. @classmethod
  149. def get_extensions(cls):
  150. return [s.ext for s in cls.get_subclasses() if hasattr(s,'ext')]
  151. @classmethod
  152. def fmt_code_to_type(cls,fmt_code):
  153. if not fmt_code: return None
  154. for c in cls.get_subclasses():
  155. if hasattr(c,'fmt_codes') and fmt_code in c.fmt_codes:
  156. return c
  157. return None
  158. @classmethod
  159. def ext_to_type(cls,ext):
  160. if not ext: return None
  161. for c in cls.get_subclasses():
  162. if hasattr(c,'ext') and ext == c.ext:
  163. return c
  164. return None
  165. @classmethod
  166. def format_fmt_codes(cls):
  167. d = [(c.__name__,('.'+c.ext if c.ext else c.ext),','.join(c.fmt_codes))
  168. for c in cls.get_subclasses()
  169. if hasattr(c,'fmt_codes')]
  170. w = max([len(a) for a,b,c in d])
  171. ret = ['{:<{w}} {:<9} {}'.format(a,b,c,w=w) for a,b,c in [
  172. ('Format','FileExt','Valid codes'),
  173. ('------','-------','-----------')
  174. ] + sorted(d)]
  175. return '\n'.join(ret) + '\n'
  176. def get_fmt_data(self):
  177. self._format()
  178. return self.fmt_data
  179. def write_to_file(self,outdir='',desc=''):
  180. self._format()
  181. kwargs = {
  182. 'desc': desc or self.desc,
  183. 'ask_tty': self.ask_tty,
  184. 'no_tty': self.no_tty,
  185. 'binary': self.file_mode == 'binary'
  186. }
  187. # write_data_to_file(): outfile with absolute path overrides opt.outdir
  188. if outdir:
  189. of = os.path.abspath(os.path.join(outdir,self._filename()))
  190. write_data_to_file(of if outdir else self._filename(),self.fmt_data,**kwargs)
  191. class SeedSourceUnenc(SeedSource):
  192. def _decrypt_retry(self): pass
  193. def _encrypt(self): pass
  194. def _filename(self):
  195. return '%s[%s].%s' % (self.seed.sid,self.seed.length,self.ext)
  196. class SeedSourceEnc(SeedSource):
  197. _msg = {
  198. 'choose_passphrase': """
  199. You must choose a passphrase to encrypt your new %s with.
  200. A key will be generated from your passphrase using a hash preset of '%s'.
  201. Please note that no strength checking of passphrases is performed. For
  202. an empty passphrase, just hit ENTER twice.
  203. """.strip()
  204. }
  205. def _get_hash_preset_from_user(self,hp,desc_suf=''):
  206. # hp=a,
  207. n = ('','old ')[self.op=='pwchg_old']
  208. m,n = (('to accept the default',n),('to reuse the old','new '))[
  209. int(self.op=='pwchg_new')]
  210. fs = "Enter {}hash preset for {}{}{},\n or hit ENTER {} value ('{}'): "
  211. p = fs.format(
  212. n,
  213. ('','new ')[self.op=='new'],
  214. self.desc,
  215. ('',' '+desc_suf)[bool(desc_suf)],
  216. m,
  217. hp
  218. )
  219. while True:
  220. ret = my_raw_input(p)
  221. if ret:
  222. if ret in g.hash_presets.keys():
  223. self.ssdata.hash_preset = ret
  224. return ret
  225. else:
  226. msg('Invalid input. Valid choices are %s' %
  227. ', '.join(sorted(g.hash_presets.keys())))
  228. else:
  229. self.ssdata.hash_preset = hp
  230. return hp
  231. def _get_hash_preset(self,desc_suf=''):
  232. if hasattr(self,'ss_in') and hasattr(self.ss_in.ssdata,'hash_preset'):
  233. old_hp = self.ss_in.ssdata.hash_preset
  234. if opt.keep_hash_preset:
  235. qmsg("Reusing hash preset '%s' at user request" % old_hp)
  236. self.ssdata.hash_preset = old_hp
  237. elif 'hash_preset' in opt.set_by_user:
  238. hp = self.ssdata.hash_preset = opt.hash_preset
  239. qmsg("Using hash preset '%s' requested on command line"
  240. % opt.hash_preset)
  241. else: # Prompt, using old value as default
  242. hp = self._get_hash_preset_from_user(old_hp,desc_suf)
  243. if (not opt.keep_hash_preset) and self.op == 'pwchg_new':
  244. m = ("changed to '%s'" % hp,'unchanged')[hp==old_hp]
  245. qmsg('Hash preset %s' % m)
  246. elif 'hash_preset' in opt.set_by_user:
  247. self.ssdata.hash_preset = opt.hash_preset
  248. qmsg("Using hash preset '%s' requested on command line"%opt.hash_preset)
  249. else:
  250. self._get_hash_preset_from_user(opt.hash_preset,desc_suf)
  251. def _get_new_passphrase(self):
  252. desc = '{}passphrase for {}{}'.format(
  253. ('','new ')[self.op=='pwchg_new'],
  254. ('','new ')[self.op in ('new','conv')],
  255. self.desc
  256. )
  257. if opt.passwd_file:
  258. w = pwfile_reuse_warning()
  259. pw = ' '.join(get_words_from_file(opt.passwd_file,desc,silent=w))
  260. elif opt.echo_passphrase:
  261. pw = ' '.join(get_words_from_user('Enter %s: ' % desc))
  262. else:
  263. for i in range(g.passwd_max_tries):
  264. pw = ' '.join(get_words_from_user('Enter %s: ' % desc))
  265. pw2 = ' '.join(get_words_from_user('Repeat passphrase: '))
  266. dmsg('Passphrases: [%s] [%s]' % (pw,pw2))
  267. if pw == pw2:
  268. vmsg('Passphrases match'); break
  269. else: msg('Passphrases do not match. Try again.')
  270. else:
  271. die(2,'User failed to duplicate passphrase in %s attempts' %
  272. g.passwd_max_tries)
  273. if pw == '': qmsg('WARNING: Empty passphrase')
  274. self.ssdata.passwd = pw
  275. return pw
  276. def _get_passphrase(self,desc_suf=''):
  277. desc ='{}passphrase for {}{}'.format(
  278. ('','old ')[self.op=='pwchg_old'],
  279. self.desc,
  280. ('',' '+desc_suf)[bool(desc_suf)]
  281. )
  282. if opt.passwd_file:
  283. w = pwfile_reuse_warning()
  284. ret = ' '.join(get_words_from_file(opt.passwd_file,desc,silent=w))
  285. else:
  286. ret = ' '.join(get_words_from_user('Enter %s: ' % desc))
  287. self.ssdata.passwd = ret
  288. def _get_first_pw_and_hp_and_encrypt_seed(self):
  289. d = self.ssdata
  290. self._get_hash_preset()
  291. if hasattr(self,'ss_in') and hasattr(self.ss_in.ssdata,'passwd'):
  292. old_pw = self.ss_in.ssdata.passwd
  293. if opt.keep_passphrase:
  294. d.passwd = old_pw
  295. qmsg('Reusing passphrase at user request')
  296. else:
  297. pw = self._get_new_passphrase()
  298. if self.op == 'pwchg_new':
  299. m = ('changed','unchanged')[pw==old_pw]
  300. qmsg('Passphrase %s' % m)
  301. else:
  302. qmsg(self.msg['choose_passphrase'] % (self.desc,d.hash_preset))
  303. self._get_new_passphrase()
  304. d.salt = sha256(get_random(128)).digest()[:g.salt_len]
  305. key = make_key(d.passwd, d.salt, d.hash_preset)
  306. d.key_id = make_chksum_8(key)
  307. d.enc_seed = encrypt_seed(self.seed.data,key)
  308. class Mnemonic (SeedSourceUnenc):
  309. stdin_ok = True
  310. fmt_codes = 'mmwords','words','mnemonic','mnem','mn','m'
  311. desc = 'mnemonic data'
  312. ext = 'mmwords'
  313. mn_lens = [i / 32 * 3 for i in g.seed_lens]
  314. wl_id = 'electrum' # or 'tirosh'
  315. def _get_data_from_user(self,desc):
  316. if not g.stdin_tty:
  317. return get_data_from_user(desc)
  318. from mmgen.term import get_char_raw,get_char
  319. def choose_mn_len():
  320. prompt = 'Choose a mnemonic length: 1) 12 words, 2) 18 words, 3) 24 words: '
  321. urange = [str(i+1) for i in range(len(self.mn_lens))]
  322. while True:
  323. r = get_char('\r'+prompt)
  324. if r in urange: break
  325. msg_r('\r' + ' '*len(prompt) + '\r')
  326. return self.mn_lens[int(r)-1]
  327. while True:
  328. mn_len = choose_mn_len()
  329. prompt = 'Mnemonic length of {} words chosen. OK?'.format(mn_len)
  330. if keypress_confirm(prompt,default_yes=True,no_nl=True): break
  331. m = 'Enter your {}-word mnemonic, hitting RETURN or SPACE after each word:'
  332. msg(m.format(mn_len))
  333. def get_word():
  334. s = ''
  335. while True:
  336. ch = get_char_raw('')
  337. if ch in '\b\x7f':
  338. if s: s = s[:-1]
  339. elif ch in '\n ':
  340. if s: break
  341. else: s += ch
  342. return s
  343. wl = baseconv.digits[self.wl_id]
  344. def in_list(w):
  345. from bisect import bisect_left
  346. idx = bisect_left(wl,w)
  347. return(True,False)[idx == len(wl) or w != wl[idx]]
  348. words,i,p = [],0,('Enter word #{}: ','Incorrect entry. Repeat word #{}: ')
  349. while len(words) < mn_len:
  350. msg_r('{r}{s}{r}'.format(r='\r',s=' '*40))
  351. if i == 1: time.sleep(0.1)
  352. msg_r(p[i].format(len(words)+1))
  353. s = get_word()
  354. if in_list(s):
  355. words.append(s); i = 0
  356. else:
  357. i = 1
  358. msg('')
  359. qmsg('Mnemonic successfully entered')
  360. return ' '.join(words)
  361. @staticmethod
  362. def _mn2hex_pad(mn): return len(mn) * 8 / 3
  363. @staticmethod
  364. def _hex2mn_pad(hexnum): return len(hexnum) * 3 / 8
  365. def _format(self):
  366. hexseed = self.seed.hexdata
  367. mn = baseconv.fromhex(hexseed,self.wl_id,self._hex2mn_pad(hexseed))
  368. ret = baseconv.tohex(mn,self.wl_id,self._mn2hex_pad(mn))
  369. # Internal error, so just die on fail
  370. compare_or_die(ret,'recomputed seed',hexseed,'original',e='Internal error')
  371. self.ssdata.mnemonic = mn
  372. self.fmt_data = ' '.join(mn) + '\n'
  373. def _deformat(self):
  374. mn = self.fmt_data.split()
  375. if len(mn) not in self.mn_lens:
  376. msg('Invalid mnemonic (%i words). Allowed numbers of words: %s' %
  377. (len(mn),', '.join([str(i) for i in self.mn_lens])))
  378. return False
  379. for n,w in enumerate(mn,1):
  380. if w not in baseconv.digits[self.wl_id]:
  381. msg('Invalid mnemonic: word #%s is not in the wordlist' % n)
  382. return False
  383. hexseed = baseconv.tohex(mn,self.wl_id,self._mn2hex_pad(mn))
  384. ret = baseconv.fromhex(hexseed,self.wl_id,self._hex2mn_pad(hexseed))
  385. if len(hexseed) * 4 not in g.seed_lens:
  386. msg('Invalid mnemonic (produces too large a number)')
  387. return False
  388. # Internal error, so just die
  389. compare_or_die(' '.join(ret),'recomputed mnemonic',' '.join(mn),'original',e='Internal error')
  390. self.seed = Seed(unhexlify(hexseed))
  391. self.ssdata.mnemonic = mn
  392. check_usr_seed_len(self.seed.length)
  393. return True
  394. class SeedFile (SeedSourceUnenc):
  395. stdin_ok = True
  396. fmt_codes = 'mmseed','seed','s'
  397. desc = 'seed data'
  398. ext = 'mmseed'
  399. def _format(self):
  400. b58seed = baseconv.b58encode(self.seed.data,pad=True)
  401. self.ssdata.chksum = make_chksum_6(b58seed)
  402. self.ssdata.b58seed = b58seed
  403. self.fmt_data = '%s %s\n' % (
  404. self.ssdata.chksum,
  405. split_into_cols(4,b58seed)
  406. )
  407. def _deformat(self):
  408. desc = self.desc
  409. ld = self.fmt_data.split()
  410. if not (7 <= len(ld) <= 12): # 6 <= padded b58 data (ld[1:]) <= 11
  411. msg('Invalid data length (%s) in %s' % (len(ld),desc))
  412. return False
  413. a,b = ld[0],''.join(ld[1:])
  414. if not is_chksum_6(a):
  415. msg("'%s': invalid checksum format in %s" % (a, desc))
  416. return False
  417. if not is_b58_str(b):
  418. msg("'%s': not a base 58 string, in %s" % (b, desc))
  419. return False
  420. vmsg_r('Validating %s checksum...' % desc)
  421. if not compare_chksums(a,'file',make_chksum_6(b),'computed',verbose=True):
  422. return False
  423. ret = baseconv.b58decode(b,pad=True)
  424. if ret == False:
  425. msg('Invalid base-58 encoded seed: %s' % val)
  426. return False
  427. self.seed = Seed(ret)
  428. self.ssdata.chksum = a
  429. self.ssdata.b58seed = b
  430. check_usr_seed_len(self.seed.length)
  431. return True
  432. class HexSeedFile (SeedSourceUnenc):
  433. stdin_ok = True
  434. fmt_codes = 'seedhex','hexseed','hex','mmhex'
  435. desc = 'hexadecimal seed data'
  436. ext = 'mmhex'
  437. def _format(self):
  438. h = self.seed.hexdata
  439. self.ssdata.chksum = make_chksum_6(h)
  440. self.ssdata.hexseed = h
  441. self.fmt_data = '%s %s\n' % (self.ssdata.chksum, split_into_cols(4,h))
  442. def _deformat(self):
  443. desc = self.desc
  444. d = self.fmt_data.split()
  445. try:
  446. d[1]
  447. chk,hstr = d[0],''.join(d[1:])
  448. except:
  449. msg("'%s': invalid %s" % (self.fmt_data.strip(),desc))
  450. return False
  451. if not len(hstr)*4 in g.seed_lens:
  452. msg('Invalid data length (%s) in %s' % (len(hstr),desc))
  453. return False
  454. if not is_chksum_6(chk):
  455. msg("'%s': invalid checksum format in %s" % (chk, desc))
  456. return False
  457. if not is_hex_str(hstr):
  458. msg("'%s': not a hexadecimal string, in %s" % (hstr, desc))
  459. return False
  460. vmsg_r('Validating %s checksum...' % desc)
  461. if not compare_chksums(chk,'file',make_chksum_6(hstr),'computed',verbose=True):
  462. return False
  463. self.seed = Seed(unhexlify(hstr))
  464. self.ssdata.chksum = chk
  465. self.ssdata.hexseed = hstr
  466. check_usr_seed_len(self.seed.length)
  467. return True
  468. class Wallet (SeedSourceEnc):
  469. fmt_codes = 'wallet','w'
  470. desc = g.proj_name + ' wallet'
  471. ext = 'mmdat'
  472. def _get_label_from_user(self,old_lbl=''):
  473. d = ("to reuse the label '%s'" % old_lbl.hl()) if old_lbl else 'for no label'
  474. p = 'Enter a wallet label, or hit ENTER %s: ' % d
  475. while True:
  476. msg_r(p)
  477. ret = my_raw_input('')
  478. if ret:
  479. self.ssdata.label = MMGenWalletLabel(ret,on_fail='return')
  480. if self.ssdata.label:
  481. break
  482. else:
  483. msg('Invalid label. Trying again...')
  484. else:
  485. self.ssdata.label = old_lbl or MMGenWalletLabel('No Label')
  486. break
  487. return self.ssdata.label
  488. # nearly identical to _get_hash_preset() - factor?
  489. def _get_label(self):
  490. if hasattr(self,'ss_in') and hasattr(self.ss_in.ssdata,'label'):
  491. old_lbl = self.ss_in.ssdata.label
  492. if opt.keep_label:
  493. qmsg("Reusing label '%s' at user request" % old_lbl.hl())
  494. self.ssdata.label = old_lbl
  495. elif opt.label:
  496. qmsg("Using label '%s' requested on command line" % opt.label.hl())
  497. lbl = self.ssdata.label = opt.label
  498. else: # Prompt, using old value as default
  499. lbl = self._get_label_from_user(old_lbl)
  500. if (not opt.keep_label) and self.op == 'pwchg_new':
  501. m = ("changed to '%s'" % lbl,'unchanged')[lbl==old_lbl]
  502. qmsg('Label %s' % m)
  503. elif opt.label:
  504. qmsg("Using label '%s' requested on command line" % opt.label.hl())
  505. self.ssdata.label = opt.label
  506. else:
  507. self._get_label_from_user()
  508. def _encrypt(self):
  509. self._get_first_pw_and_hp_and_encrypt_seed()
  510. self._get_label()
  511. d = self.ssdata
  512. d.pw_status = ('NE','E')[len(d.passwd)==0]
  513. d.timestamp = make_timestamp()
  514. def _format(self):
  515. d = self.ssdata
  516. s = self.seed
  517. slt_fmt = baseconv.b58encode(d.salt,pad=True)
  518. es_fmt = baseconv.b58encode(d.enc_seed,pad=True)
  519. lines = (
  520. d.label,
  521. '{} {} {} {} {}'.format(s.sid.lower(), d.key_id.lower(),
  522. s.length, d.pw_status, d.timestamp),
  523. '{}: {} {} {}'.format(d.hash_preset,*get_hash_params(d.hash_preset)),
  524. '{} {}'.format(make_chksum_6(slt_fmt),split_into_cols(4,slt_fmt)),
  525. '{} {}'.format(make_chksum_6(es_fmt), split_into_cols(4,es_fmt))
  526. )
  527. chksum = make_chksum_6(' '.join(lines))
  528. self.fmt_data = '%s\n' % '\n'.join((chksum,)+lines)
  529. def _deformat(self):
  530. def check_master_chksum(lines,desc):
  531. if len(lines) != 6:
  532. msg('Invalid number of lines (%s) in %s data' %
  533. (len(lines),desc))
  534. return False
  535. if not is_chksum_6(lines[0]):
  536. msg('Incorrect master checksum (%s) in %s data' %
  537. (lines[0],desc))
  538. return False
  539. chk = make_chksum_6(' '.join(lines[1:]))
  540. if not compare_chksums(lines[0],'master',chk,'computed',
  541. hdr='For wallet master checksum',verbose=True):
  542. return False
  543. return True
  544. lines = self.fmt_data.splitlines()
  545. if not check_master_chksum(lines,self.desc): return False
  546. d = self.ssdata
  547. d.label = MMGenWalletLabel(lines[1])
  548. d1,d2,d3,d4,d5 = lines[2].split()
  549. d.seed_id = d1.upper()
  550. d.key_id = d2.upper()
  551. check_usr_seed_len(int(d3))
  552. d.pw_status,d.timestamp = d4,d5
  553. hpdata = lines[3].split()
  554. d.hash_preset = hp = hpdata[0][:-1] # a string!
  555. qmsg("Hash preset of wallet: '%s'" % hp)
  556. if 'hash_preset' in opt.set_by_user:
  557. uhp = opt.hash_preset
  558. if uhp != hp:
  559. qmsg("Warning: ignoring user-requested hash preset '%s'" % uhp)
  560. hash_params = [int(i) for i in hpdata[1:]]
  561. if hash_params != get_hash_params(d.hash_preset):
  562. msg("Hash parameters '%s' don't match hash preset '%s'" %
  563. (' '.join(hash_params), d.hash_preset))
  564. return False
  565. lmin,foo,lmax = [v for k,v in baseconv.b58pad_lens] # 22,33,44
  566. for i,key in (4,'salt'),(5,'enc_seed'):
  567. l = lines[i].split(' ')
  568. chk = l.pop(0)
  569. b58_val = ''.join(l)
  570. if len(b58_val) < lmin or len(b58_val) > lmax:
  571. msg('Invalid format for %s in %s: %s' % (key,self.desc,l))
  572. return False
  573. if not compare_chksums(chk,key,
  574. make_chksum_6(b58_val),'computed checksum',verbose=True):
  575. return False
  576. val = baseconv.b58decode(b58_val,pad=True)
  577. if val == False:
  578. msg('Invalid base 58 number: %s' % b58_val)
  579. return False
  580. setattr(d,key,val)
  581. return True
  582. def _decrypt(self):
  583. d = self.ssdata
  584. # Needed for multiple transactions with {}-txsign
  585. suf = ('',os.path.basename(self.infile.name))[bool(opt.quiet)]
  586. self._get_passphrase(desc_suf=suf)
  587. key = make_key(d.passwd, d.salt, d.hash_preset)
  588. ret = decrypt_seed(d.enc_seed, key, d.seed_id, d.key_id)
  589. if ret:
  590. self.seed = Seed(ret)
  591. return True
  592. else:
  593. return False
  594. def _filename(self):
  595. return '{}-{}[{},{}].{}'.format(
  596. self.seed.sid,
  597. self.ssdata.key_id,
  598. self.seed.length,
  599. self.ssdata.hash_preset,
  600. self.ext
  601. )
  602. class Brainwallet (SeedSourceEnc):
  603. stdin_ok = True
  604. fmt_codes = 'mmbrain','brainwallet','brain','bw','b'
  605. desc = 'brainwallet'
  606. ext = 'mmbrain'
  607. # brainwallet warning message? TODO
  608. def get_bw_params(self):
  609. # already checked
  610. a = opt.brain_params.split(',')
  611. return int(a[0]),a[1]
  612. def _deformat(self):
  613. self.brainpasswd = ' '.join(self.fmt_data.split())
  614. return True
  615. def _decrypt(self):
  616. d = self.ssdata
  617. # Don't set opt.seed_len! In txsign, BW seed len might differ from other seed srcs
  618. if opt.brain_params:
  619. seed_len,d.hash_preset = self.get_bw_params()
  620. else:
  621. if 'seed_len' not in opt.set_by_user:
  622. m1 = 'Using default seed length of %s bits'
  623. m2 = 'If this is not what you want, use the --seed-len option'
  624. qmsg((m1+'\n'+m2) % yellow(str(opt.seed_len)))
  625. self._get_hash_preset()
  626. seed_len = opt.seed_len
  627. qmsg_r('Hashing brainwallet data. Please wait...')
  628. # Use buflen arg of scrypt.hash() to get seed of desired length
  629. seed = scrypt_hash_passphrase(self.brainpasswd, '',
  630. d.hash_preset, buflen=seed_len/8)
  631. qmsg('Done')
  632. self.seed = Seed(seed)
  633. msg('Seed ID: %s' % self.seed.sid)
  634. qmsg('Check this value against your records')
  635. return True
  636. class IncogWallet (SeedSourceEnc):
  637. file_mode = 'binary'
  638. fmt_codes = 'mmincog','incog','icg','i'
  639. desc = 'incognito data'
  640. ext = 'mmincog'
  641. no_tty = True
  642. _msg = {
  643. 'check_incog_id': """
  644. Check the generated Incog ID above against your records. If it doesn't
  645. match, then your incognito data is incorrect or corrupted.
  646. """,
  647. 'record_incog_id': """
  648. Make a record of the Incog ID but keep it secret. You will use it to
  649. identify your incog wallet data in the future.
  650. """,
  651. 'incorrect_incog_passphrase_try_again': """
  652. Incorrect passphrase, hash preset, or maybe old-format incog wallet.
  653. Try again? (Y)es, (n)o, (m)ore information:
  654. """.strip(),
  655. 'confirm_seed_id': """
  656. If the Seed ID above is correct but you're seeing this message, then you need
  657. to exit and re-run the program with the '--old-incog-fmt' option.
  658. """.strip(),
  659. 'dec_chk': " %s hash preset"
  660. }
  661. def _make_iv_chksum(self,s): return sha256(s).hexdigest()[:8].upper()
  662. def _get_incog_data_len(self,seed_len):
  663. e = (g.hincog_chk_len,0)[bool(opt.old_incog_fmt)]
  664. return g.aesctr_iv_len + g.salt_len + e + seed_len/8
  665. def _incog_data_size_chk(self):
  666. # valid sizes: 56, 64, 72
  667. dlen = len(self.fmt_data)
  668. valid_dlen = self._get_incog_data_len(opt.seed_len)
  669. if dlen == valid_dlen:
  670. return True
  671. else:
  672. if opt.old_incog_fmt:
  673. msg('WARNING: old-style incognito format requested. ' +
  674. 'Are you sure this is correct?')
  675. msg(('Invalid incognito data size (%s bytes) for this ' +
  676. 'seed length (%s bits)') % (dlen,opt.seed_len))
  677. msg('Valid data size for this seed length: %s bytes' % valid_dlen)
  678. for sl in g.seed_lens:
  679. if dlen == self._get_incog_data_len(sl):
  680. die(1,'Valid seed length for this data size: %s bits' % sl)
  681. msg(('This data size (%s bytes) is invalid for all available ' +
  682. 'seed lengths') % dlen)
  683. return False
  684. def _encrypt (self):
  685. self._get_first_pw_and_hp_and_encrypt_seed()
  686. if opt.old_incog_fmt:
  687. die(1,'Writing old-format incog wallets is unsupported')
  688. d = self.ssdata
  689. # IV is used BOTH to initialize counter and to salt password!
  690. d.iv = get_random(g.aesctr_iv_len)
  691. d.iv_id = self._make_iv_chksum(d.iv)
  692. msg('New Incog Wallet ID: %s' % d.iv_id)
  693. qmsg('Make a record of this value')
  694. vmsg(self.msg['record_incog_id'])
  695. d.salt = get_random(g.salt_len)
  696. key = make_key(d.passwd, d.salt, d.hash_preset, 'incog wallet key')
  697. chk = sha256(self.seed.data).digest()[:8]
  698. d.enc_seed = encrypt_data(chk + self.seed.data, key, 1, 'seed')
  699. d.wrapper_key = make_key(d.passwd, d.iv, d.hash_preset, 'incog wrapper key')
  700. d.key_id = make_chksum_8(d.wrapper_key)
  701. vmsg('Key ID: %s' % d.key_id)
  702. d.target_data_len = self._get_incog_data_len(self.seed.length)
  703. def _format(self):
  704. d = self.ssdata
  705. # print len(d.iv), len(d.salt), len(d.enc_seed), len(d.wrapper_key)
  706. self.fmt_data = d.iv + encrypt_data(
  707. d.salt + d.enc_seed,
  708. d.wrapper_key,
  709. int(hexlify(d.iv),16),
  710. self.desc)
  711. # print len(self.fmt_data)
  712. def _filename(self):
  713. s = self.seed
  714. d = self.ssdata
  715. return '{}-{}-{}[{},{}].{}'.format(
  716. s.sid,
  717. d.key_id,
  718. d.iv_id,
  719. s.length,
  720. d.hash_preset,
  721. self.ext)
  722. def _deformat(self):
  723. if not self._incog_data_size_chk(): return False
  724. d = self.ssdata
  725. d.iv = self.fmt_data[0:g.aesctr_iv_len]
  726. d.incog_id = self._make_iv_chksum(d.iv)
  727. d.enc_incog_data = self.fmt_data[g.aesctr_iv_len:]
  728. msg('Incog Wallet ID: %s' % d.incog_id)
  729. qmsg('Check this value against your records')
  730. vmsg(self.msg['check_incog_id'])
  731. return True
  732. def _verify_seed_newfmt(self,data):
  733. chk,seed = data[:8],data[8:]
  734. if sha256(seed).digest()[:8] == chk:
  735. qmsg('Passphrase%s are correct' % (self.msg['dec_chk'] % 'and'))
  736. return seed
  737. else:
  738. msg('Incorrect passphrase%s' % (self.msg['dec_chk'] % 'or'))
  739. return False
  740. def _verify_seed_oldfmt(self,seed):
  741. m = 'Seed ID: %s. Is the Seed ID correct?' % make_chksum_8(seed)
  742. if keypress_confirm(m, True):
  743. return seed
  744. else:
  745. return False
  746. def _decrypt(self):
  747. d = self.ssdata
  748. self._get_hash_preset(desc_suf=d.incog_id)
  749. self._get_passphrase(desc_suf=d.incog_id)
  750. # IV is used BOTH to initialize counter and to salt password!
  751. key = make_key(d.passwd, d.iv, d.hash_preset, 'wrapper key')
  752. dd = decrypt_data(d.enc_incog_data, key,
  753. int(hexlify(d.iv),16), 'incog data')
  754. d.salt = dd[0:g.salt_len]
  755. d.enc_seed = dd[g.salt_len:]
  756. key = make_key(d.passwd, d.salt, d.hash_preset, 'main key')
  757. qmsg('Key ID: %s' % make_chksum_8(key))
  758. verify_seed = getattr(self,'_verify_seed_'+
  759. ('newfmt','oldfmt')[bool(opt.old_incog_fmt)])
  760. seed = verify_seed(decrypt_seed(d.enc_seed, key, '', ''))
  761. if seed:
  762. self.seed = Seed(seed)
  763. msg('Seed ID: %s' % self.seed.sid)
  764. return True
  765. else:
  766. return False
  767. class IncogWalletHex (IncogWallet):
  768. file_mode = 'text'
  769. desc = 'hex incognito data'
  770. fmt_codes = 'mmincox','incox','incog_hex','xincog','ix','xi'
  771. ext = 'mmincox'
  772. no_tty = False
  773. def _deformat(self):
  774. ret = decode_pretty_hexdump(self.fmt_data)
  775. if ret:
  776. self.fmt_data = ret
  777. return IncogWallet._deformat(self)
  778. else:
  779. return False
  780. def _format(self):
  781. IncogWallet._format(self)
  782. self.fmt_data = pretty_hexdump(self.fmt_data)
  783. class IncogWalletHidden (IncogWallet):
  784. desc = 'hidden incognito data'
  785. fmt_codes = 'incog_hidden','hincog','ih','hi'
  786. ext = None
  787. _msg = {
  788. 'choose_file_size': """
  789. You must choose a size for your new hidden incog data. The minimum size is
  790. {} bytes, which puts the incog data right at the end of the file. Since you
  791. probably want to hide your data somewhere in the middle of the file where it's
  792. harder to find, you're advised to choose a much larger file size than this.
  793. """.strip(),
  794. 'check_incog_id': """
  795. Check generated Incog ID above against your records. If it doesn't
  796. match, then your incognito data is incorrect or corrupted, or you
  797. may have specified an incorrect offset.
  798. """,
  799. 'record_incog_id': """
  800. Make a record of the Incog ID but keep it secret. You will used it to
  801. identify the incog wallet data in the future and to locate the offset
  802. where the data is hidden in the event you forget it.
  803. """,
  804. 'dec_chk': ', hash preset, offset %s seed length'
  805. }
  806. def _get_hincog_params(self,wtype):
  807. a = getattr(opt,'hidden_incog_'+ wtype +'_params').split(',')
  808. return ','.join(a[:-1]),int(a[-1]) # permit comma in filename
  809. def _check_valid_offset(self,fn,action):
  810. d = self.ssdata
  811. m = ('Input','Destination')[action=='write']
  812. if fn.size < d.hincog_offset + d.target_data_len:
  813. die(1,
  814. "%s file '%s' has length %s, too short to %s %s bytes of data at offset %s"
  815. % (m,fn.name,fn.size,action,d.target_data_len,d.hincog_offset))
  816. def _get_data(self):
  817. d = self.ssdata
  818. d.hincog_offset = self._get_hincog_params('input')[1]
  819. qmsg("Getting hidden incog data from file '%s'" % self.infile.name)
  820. # Already sanity-checked:
  821. d.target_data_len = self._get_incog_data_len(opt.seed_len)
  822. self._check_valid_offset(self.infile,'read')
  823. flgs = os.O_RDONLY|os.O_BINARY if g.platform == 'win' else os.O_RDONLY
  824. fh = os.open(self.infile.name,flgs)
  825. os.lseek(fh,int(d.hincog_offset),os.SEEK_SET)
  826. self.fmt_data = os.read(fh,d.target_data_len)
  827. os.close(fh)
  828. qmsg("Data read from file '%s' at offset %s" %
  829. (self.infile.name,d.hincog_offset), 'Data read from file')
  830. # overrides method in SeedSource
  831. def write_to_file(self):
  832. d = self.ssdata
  833. self._format()
  834. compare_or_die(d.target_data_len, 'target data length',
  835. len(self.fmt_data),'length of formatted ' + self.desc)
  836. k = ('output','input')[self.op=='pwchg_new']
  837. fn,d.hincog_offset = self._get_hincog_params(k)
  838. if opt.outdir and not os.path.dirname(fn):
  839. fn = os.path.join(opt.outdir,fn)
  840. check_offset = True
  841. try:
  842. os.stat(fn)
  843. except:
  844. if keypress_confirm("Requested file '%s' does not exist. Create?"
  845. % fn, default_yes=True):
  846. min_fsize = d.target_data_len + d.hincog_offset
  847. msg(self.msg['choose_file_size'].format(min_fsize))
  848. while True:
  849. fsize = parse_nbytes(my_raw_input('Enter file size: '))
  850. if fsize >= min_fsize: break
  851. msg('File size must be an integer no less than %s' %
  852. min_fsize)
  853. from mmgen.tool import Rand2file # threaded routine
  854. Rand2file(fn,str(fsize))
  855. check_offset = False
  856. else:
  857. die(1,'Exiting at user request')
  858. f = Filename(fn,ftype=type(self),write=True)
  859. dmsg('%s data len %s, offset %s' % (
  860. capfirst(self.desc),d.target_data_len,d.hincog_offset))
  861. if check_offset:
  862. self._check_valid_offset(f,'write')
  863. if not opt.quiet: confirm_or_exit('',"alter file '%s'" % f.name)
  864. flgs = os.O_RDWR|os.O_BINARY if g.platform == 'win' else os.O_RDWR
  865. fh = os.open(f.name,flgs)
  866. os.lseek(fh, int(d.hincog_offset), os.SEEK_SET)
  867. os.write(fh, self.fmt_data)
  868. os.close(fh)
  869. msg("%s written to file '%s' at offset %s" % (
  870. capfirst(self.desc),f.name,d.hincog_offset))