util.py 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720
  1. #!/usr/bin/env python3
  2. #
  3. # mmgen = Multi-Mode GENerator, command-line Bitcoin cold storage solution
  4. # Copyright (C)2013-2022 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. util.py: Low-level routines imported by other modules in the MMGen suite
  20. """
  21. import sys,os,time,re
  22. from .color import *
  23. from .globalvars import g
  24. from .opts import opt
  25. import mmgen.color as color_mod
  26. ascii_lowercase = 'abcdefghijklmnopqrstuvwxyz'
  27. hexdigits = '0123456789abcdefABCDEF'
  28. hexdigits_uc = '0123456789ABCDEF'
  29. hexdigits_lc = '0123456789abcdef'
  30. if g.platform == 'win':
  31. def msg_r(s):
  32. try:
  33. g.stderr.write(s)
  34. g.stderr.flush()
  35. except:
  36. os.write(2,s.encode())
  37. def msg(s):
  38. msg_r(s + '\n')
  39. def Msg_r(s):
  40. try:
  41. g.stdout.write(s)
  42. g.stdout.flush()
  43. except:
  44. os.write(1,s.encode())
  45. def Msg(s):
  46. Msg_r(s + '\n')
  47. else:
  48. def msg(s):
  49. g.stderr.write(s + '\n')
  50. def msg_r(s):
  51. g.stderr.write(s)
  52. g.stderr.flush()
  53. def Msg(s):
  54. g.stdout.write(s + '\n')
  55. def Msg_r(s):
  56. g.stdout.write(s)
  57. g.stdout.flush()
  58. def rmsg(s):
  59. msg(red(s))
  60. def ymsg(s):
  61. msg(yellow(s))
  62. def gmsg(s):
  63. msg(green(s))
  64. def gmsg_r(s):
  65. msg_r(green(s))
  66. def bmsg(s):
  67. msg(blue(s))
  68. def pumsg(s):
  69. msg(purple(s))
  70. def qmsg(s):
  71. if not opt.quiet:
  72. msg(s)
  73. def qmsg_r(s):
  74. if not opt.quiet:
  75. msg_r(s)
  76. def vmsg(s,force=False):
  77. if opt.verbose or force:
  78. msg(s)
  79. def vmsg_r(s,force=False):
  80. if opt.verbose or force:
  81. msg_r(s)
  82. def Vmsg(s,force=False):
  83. if opt.verbose or force:
  84. Msg(s)
  85. def Vmsg_r(s,force=False):
  86. if opt.verbose or force:
  87. Msg_r(s)
  88. def dmsg(s):
  89. if opt.debug:
  90. msg(s)
  91. def mmsg(*args):
  92. for d in args:
  93. Msg(repr(d))
  94. def mdie(*args):
  95. mmsg(*args)
  96. sys.exit(0)
  97. def die(ev,s='',stdout=False):
  98. if isinstance(ev,int):
  99. from .exception import MMGenSystemExit,MMGenError
  100. if ev <= 2:
  101. raise MMGenSystemExit(ev,s,stdout)
  102. else:
  103. raise MMGenError(ev,s,stdout)
  104. elif isinstance(ev,str):
  105. import mmgen.exception
  106. raise getattr(mmgen.exception,ev)(s)
  107. else:
  108. raise ValueError(f'{ev}: exit value must be string or int instance')
  109. def die_wait(delay,ev=0,s=''):
  110. assert isinstance(delay,int)
  111. assert isinstance(ev,int)
  112. if s:
  113. msg(s)
  114. time.sleep(delay)
  115. sys.exit(ev)
  116. def die_pause(ev=0,s=''):
  117. assert isinstance(ev,int)
  118. if s:
  119. msg(s)
  120. input('Press ENTER to exit')
  121. sys.exit(ev)
  122. def Die(ev=0,s=''):
  123. die(ev=ev,s=s,stdout=True)
  124. def pp_fmt(d):
  125. import pprint
  126. return pprint.PrettyPrinter(indent=4,compact=False).pformat(d)
  127. def pp_msg(d):
  128. msg(pp_fmt(d))
  129. def fmt(s,indent='',strip_char=None):
  130. "de-indent multiple lines of text, or indent with specified string"
  131. return indent + ('\n'+indent).join([l.strip(strip_char) for l in s.strip().splitlines()]) + '\n'
  132. def fmt_list(iterable,fmt='dfl',indent=''):
  133. "pretty-format a list"
  134. sep,lq,rq = {
  135. 'utf8': ("“, ”", "“", "”"),
  136. 'dfl': ("', '", "'", "'"),
  137. 'bare': (' ', '', '' ),
  138. 'no_quotes': (', ', '', '' ),
  139. 'no_spc': ("','", "'", "'"),
  140. 'min': (",", "'", "'"),
  141. 'col': ('\n'+indent, indent, '' ),
  142. }[fmt]
  143. return lq + sep.join(str(i) for i in iterable) + rq
  144. def list_gen(*data):
  145. """
  146. add element to list if condition is true or absent
  147. """
  148. assert type(data) in (list,tuple), f'{type(data).__name__} not in (list,tuple)'
  149. def gen():
  150. for i in data:
  151. assert type(i) == list, f'{type(i).__name__} != list'
  152. assert len(i) in (1,2), f'{len(i)} not in (1,2)'
  153. if len(i) == 1 or i[1]:
  154. yield i[0]
  155. return list(gen())
  156. def removeprefix(s,pfx): # workaround for pre-Python 3.9
  157. return s[len(pfx):] if s.startswith(pfx) else s
  158. def removesuffix(s,sfx): # workaround for pre-Python 3.9
  159. return s[:len(sfx)] if s.endswith(sfx) else s
  160. def remove_dups(iterable,edesc='element',desc='list',quiet=False,hide=False):
  161. """
  162. Remove duplicate occurrences of iterable elements, preserving first occurrence
  163. If iterable is a generator, return a list, else type(iterable)
  164. """
  165. ret = []
  166. for e in iterable:
  167. if e in ret:
  168. if not quiet:
  169. ymsg(f'Warning: removing duplicate {edesc} {"(hidden)" if hide else e} in {desc}')
  170. else:
  171. ret.append(e)
  172. return ret if type(iterable).__name__ == 'generator' else type(iterable)(ret)
  173. def exit_if_mswin(feature):
  174. if g.platform == 'win':
  175. die(2, capfirst(feature) + ' not supported on the MSWin / MSYS2 platform' )
  176. def get_keccak(cached_ret=[]):
  177. if not cached_ret:
  178. from .opts import opt
  179. # called in opts.init() via CoinProtocol, so must use getattr():
  180. if getattr(opt,'use_internal_keccak_module',False):
  181. qmsg('Using internal keccak module by user request')
  182. from .contrib.keccak import keccak_256
  183. else:
  184. try:
  185. from sha3 import keccak_256
  186. except:
  187. from .contrib.keccak import keccak_256
  188. cached_ret.append(keccak_256)
  189. return cached_ret[0]
  190. # From 'man dd':
  191. # c=1, w=2, b=512, kB=1000, K=1024, MB=1000*1000, M=1024*1024,
  192. # GB=1000*1000*1000, G=1024*1024*1024, and so on for T, P, E, Z, Y.
  193. bytespec_map = (
  194. ('c', 1),
  195. ('w', 2),
  196. ('b', 512),
  197. ('kB', 1000),
  198. ('K', 1024),
  199. ('MB', 1000000),
  200. ('M', 1048576),
  201. ('GB', 1000000000),
  202. ('G', 1073741824),
  203. ('TB', 1000000000000),
  204. ('T', 1099511627776),
  205. ('PB', 1000000000000000),
  206. ('P', 1125899906842624),
  207. ('EB', 1000000000000000000),
  208. ('E', 1152921504606846976),
  209. )
  210. def int2bytespec(n,spec,fmt,print_sym=True):
  211. def spec2int(spec):
  212. for k,v in bytespec_map:
  213. if k == spec:
  214. return v
  215. else:
  216. die('{spec}: unrecognized bytespec')
  217. return '{:{}f}{}'.format( n / spec2int(spec), fmt, spec if print_sym else '' )
  218. def parse_bytespec(nbytes):
  219. m = re.match(r'([0123456789.]+)(.*)',nbytes)
  220. if m:
  221. if m.group(2):
  222. for k,v in bytespec_map:
  223. if k == m.group(2):
  224. from decimal import Decimal
  225. return int(Decimal(m.group(1)) * v)
  226. else:
  227. msg("Valid byte specifiers: '{}'".format("' '".join([i[0] for i in bytespec_map])))
  228. elif '.' in nbytes:
  229. raise ValueError('fractional bytes not allowed')
  230. else:
  231. return int(nbytes)
  232. die(1,f'{nbytes!r}: invalid byte specifier')
  233. def suf(arg,suf_type='s',verb='none'):
  234. suf_types = {
  235. 'none': {
  236. 's': ('s', ''),
  237. 'es': ('es', ''),
  238. 'ies': ('ies','y'),
  239. },
  240. 'is': {
  241. 's': ('s are', ' is'),
  242. 'es': ('es are', ' is'),
  243. 'ies': ('ies are','y is'),
  244. },
  245. 'has': {
  246. 's': ('s have', ' has'),
  247. 'es': ('es have', ' has'),
  248. 'ies': ('ies have','y has'),
  249. },
  250. }
  251. if isinstance(arg,int):
  252. n = arg
  253. elif isinstance(arg,(list,tuple,set,dict)):
  254. n = len(arg)
  255. else:
  256. die(2,f'{arg}: invalid parameter for suf()')
  257. return suf_types[verb][suf_type][n == 1]
  258. def get_extension(fn):
  259. return os.path.splitext(fn)[1][1:]
  260. def remove_extension(fn,ext):
  261. a,b = os.path.splitext(fn)
  262. return a if b[1:] == ext else fn
  263. def make_chksum_N(s,nchars,sep=False):
  264. if isinstance(s,str):
  265. s = s.encode()
  266. if nchars%4 or not (4 <= nchars <= 64):
  267. return False
  268. from hashlib import sha256
  269. s = sha256(sha256(s).digest()).hexdigest().upper()
  270. sep = ('',' ')[bool(sep)]
  271. return sep.join([s[i*4:i*4+4] for i in range(nchars//4)])
  272. def make_chksum_8(s,sep=False):
  273. from .obj import HexStr
  274. from hashlib import sha256
  275. s = HexStr(sha256(sha256(s).digest()).hexdigest()[:8].upper(),case='upper')
  276. return '{} {}'.format(s[:4],s[4:]) if sep else s
  277. def make_chksum_6(s):
  278. from .obj import HexStr
  279. from hashlib import sha256
  280. if isinstance(s,str):
  281. s = s.encode()
  282. return HexStr(sha256(s).hexdigest()[:6])
  283. def is_chksum_6(s):
  284. return len(s) == 6 and set(s) <= set(hexdigits_lc)
  285. def make_iv_chksum(s):
  286. from hashlib import sha256
  287. return sha256(s).hexdigest()[:8].upper()
  288. def split_into_cols(col_wid,s):
  289. return ' '.join([s[col_wid*i:col_wid*(i+1)] for i in range(len(s)//col_wid+1)]).rstrip()
  290. def capfirst(s): # different from str.capitalize() - doesn't downcase any uc in string
  291. return s if len(s) == 0 else s[0].upper() + s[1:]
  292. def decode_timestamp(s):
  293. # tz_save = open('/etc/timezone').read().rstrip()
  294. os.environ['TZ'] = 'UTC'
  295. # os.environ['TZ'] = tz_save
  296. return int(time.mktime( time.strptime(s,'%Y%m%d_%H%M%S') ))
  297. def make_timestamp(secs=None):
  298. return '{:04d}{:02d}{:02d}_{:02d}{:02d}{:02d}'.format(*time.gmtime(
  299. int(secs) if secs != None else time.time() )[:6])
  300. def make_timestr(secs=None):
  301. return '{}-{:02d}-{:02d} {:02d}:{:02d}:{:02d}'.format(*time.gmtime(
  302. int(secs) if secs != None else time.time() )[:6])
  303. def secs_to_dhms(secs):
  304. hrs = secs // 3600
  305. return '{}{:02d}:{:02d}:{:02d} h/m/s'.format(
  306. ('{} day{}, '.format(hrs//24,suf(hrs//24)) if hrs > 24 else ''),
  307. hrs % 24,
  308. (secs // 60) % 60,
  309. secs % 60
  310. )
  311. def secs_to_hms(secs):
  312. return '{:02d}:{:02d}:{:02d}'.format(secs//3600, (secs//60) % 60, secs % 60)
  313. def secs_to_ms(secs):
  314. return '{:02d}:{:02d}'.format(secs//60, secs % 60)
  315. def is_int(s):
  316. try:
  317. int(str(s))
  318. return True
  319. except:
  320. return False
  321. def is_hex_str(s):
  322. return set(s) <= set(hexdigits)
  323. def is_hex_str_lc(s):
  324. return set(s) <= set(hexdigits_lc)
  325. def is_utf8(s):
  326. try: s.decode('utf8')
  327. except: return False
  328. else: return True
  329. def remove_whitespace(s,ws='\t\r\n '):
  330. return s.translate(dict((ord(e),None) for e in ws))
  331. def pretty_format(s,width=80,pfx=''):
  332. out = []
  333. while(s):
  334. if len(s) <= width:
  335. out.append(s)
  336. break
  337. i = s[:width].rfind(' ')
  338. out.append(s[:i])
  339. s = s[i+1:]
  340. return pfx + ('\n'+pfx).join(out)
  341. def block_format(data,gw=2,cols=8,line_nums=None,data_is_hex=False):
  342. assert line_nums in (None,'hex','dec'),"'line_nums' must be one of None, 'hex' or 'dec'"
  343. ln_fs = '{:06x}: ' if line_nums == 'hex' else '{:06}: '
  344. bytes_per_chunk = gw
  345. if data_is_hex:
  346. gw *= 2
  347. nchunks = len(data)//gw + bool(len(data)%gw)
  348. return ''.join(
  349. ('' if (line_nums == None or i % cols) else ln_fs.format(i*bytes_per_chunk))
  350. + data[i*gw:i*gw+gw]
  351. + (' ' if (not cols or (i+1) % cols) else '\n')
  352. for i in range(nchunks)
  353. ).rstrip() + '\n'
  354. def pretty_hexdump(data,gw=2,cols=8,line_nums=None):
  355. return block_format(data.hex(),gw,cols,line_nums,data_is_hex=True)
  356. def decode_pretty_hexdump(data):
  357. pat = re.compile(fr'^[{hexdigits}]+:\s+')
  358. lines = [pat.sub('',line) for line in data.splitlines()]
  359. try:
  360. return bytes.fromhex(''.join((''.join(lines).split())))
  361. except:
  362. msg('Data not in hexdump format')
  363. return False
  364. def strip_comment(line):
  365. return re.sub('#.*','',line).rstrip()
  366. def strip_comments(lines):
  367. pat = re.compile('#.*')
  368. return [m for m in [pat.sub('',l).rstrip() for l in lines] if m != '']
  369. def compare_chksums(chk1,desc1,chk2,desc2,hdr='',die_on_fail=False,verbose=False):
  370. if not chk1 == chk2:
  371. fs = "{} ERROR: {} checksum ({}) doesn't match {} checksum ({})"
  372. m = fs.format((hdr+':\n ' if hdr else 'CHECKSUM'),desc2,chk2,desc1,chk1)
  373. if die_on_fail:
  374. die(3,m)
  375. else:
  376. vmsg(m,force=verbose)
  377. return False
  378. vmsg(f'{capfirst(desc1)} checksum OK ({chk1})')
  379. return True
  380. def compare_or_die(val1, desc1, val2, desc2, e='Error'):
  381. if val1 != val2:
  382. die(3,f"{e}: {desc2} ({val2}) doesn't match {desc1} ({val1})")
  383. dmsg(f'{capfirst(desc2)} OK ({val2})')
  384. return True
  385. def check_wallet_extension(fn):
  386. from .wallet import get_wallet_data
  387. get_wallet_data( ext=get_extension(fn), die_on_fail=True ) # raises exception on failure
  388. def make_full_path(outdir,outfile):
  389. return os.path.normpath(os.path.join(outdir, os.path.basename(outfile)))
  390. def confirm_or_raise(message,action,expect='YES',exit_msg='Exiting at user request'):
  391. if message:
  392. msg(message)
  393. if line_input(
  394. (f'{action} ' if action[0].isupper() else f'Are you sure you want to {action}?\n') +
  395. f'Type uppercase {expect!r} to confirm: '
  396. ).strip() != expect:
  397. die( 'UserNonConfirmation', exit_msg )
  398. def get_words_from_user(prompt):
  399. words = line_input(prompt, echo=opt.echo_passphrase).split()
  400. dmsg('Sanitized input: [{}]'.format(' '.join(words)))
  401. return words
  402. def get_data_from_user(desc='data'): # user input MUST be UTF-8
  403. data = line_input(f'Enter {desc}: ',echo=opt.echo_passphrase)
  404. dmsg(f'User input: [{data}]')
  405. return data
  406. class oneshot_warning:
  407. color = 'nocolor'
  408. def __init__(self,div=None,fmt_args=[],reverse=False):
  409. self.do(type(self),div,fmt_args,reverse)
  410. def do(self,wcls,div,fmt_args,reverse):
  411. def do_warning():
  412. message = getattr(wcls,'message')
  413. color = getattr( color_mod, getattr(wcls,'color') )
  414. msg(color('WARNING: ' + message.format(*fmt_args)))
  415. if not hasattr(wcls,'data'):
  416. setattr(wcls,'data',[])
  417. data = getattr(wcls,'data')
  418. condition = (div in data) if reverse else (not div in data)
  419. if not div in data:
  420. data.append(div)
  421. if condition:
  422. do_warning()
  423. self.warning_shown = True
  424. else:
  425. self.warning_shown = False
  426. class oneshot_warning_group(oneshot_warning):
  427. def __init__(self,wcls,div=None,fmt_args=[],reverse=False):
  428. self.do(getattr(self,wcls),div,fmt_args,reverse)
  429. class pwfile_reuse_warning(oneshot_warning):
  430. message = 'Reusing passphrase from file {!r} at user request'
  431. def __init__(self,fn):
  432. oneshot_warning.__init__(self,div=fn,fmt_args=[fn],reverse=True)
  433. def line_input(prompt,echo=True,insert_txt=''):
  434. """
  435. multi-line prompts OK
  436. one-line prompts must begin at beginning of line
  437. empty prompts forbidden due to interactions with readline
  438. """
  439. assert prompt,'calling line_input() with an empty prompt forbidden'
  440. def init_readline():
  441. try:
  442. import readline
  443. except ImportError:
  444. return False
  445. else:
  446. if insert_txt:
  447. readline.set_startup_hook(lambda: readline.insert_text(insert_txt))
  448. return True
  449. else:
  450. return False
  451. if not sys.stdout.isatty():
  452. msg_r(prompt)
  453. prompt = ''
  454. from .term import kb_hold_protect
  455. kb_hold_protect()
  456. if g.test_suite_popen_spawn:
  457. msg(prompt)
  458. sys.stderr.flush()
  459. reply = os.read(0,4096).decode().rstrip('\n') # strip NL to mimic behavior of input()
  460. elif echo or not sys.stdin.isatty():
  461. clear_buffer = init_readline() if sys.stdin.isatty() else False
  462. reply = input(prompt)
  463. if clear_buffer:
  464. import readline
  465. readline.set_startup_hook(lambda: readline.insert_text(''))
  466. else:
  467. from getpass import getpass
  468. if g.platform == 'win':
  469. # MSWin hack - getpass('foo') doesn't flush stderr
  470. msg_r(prompt.strip()) # getpass('') adds a space
  471. sys.stderr.flush()
  472. reply = getpass('')
  473. else:
  474. reply = getpass(prompt)
  475. kb_hold_protect()
  476. return reply.strip()
  477. def keypress_confirm(prompt,default_yes=False,verbose=False,no_nl=False,complete_prompt=False):
  478. if not complete_prompt:
  479. prompt = '{} {}: '.format( prompt, '(Y/n)' if default_yes else '(y/N)' )
  480. nl = f'\r{" "*len(prompt)}\r' if no_nl else '\n'
  481. if g.accept_defaults:
  482. msg(prompt)
  483. return default_yes
  484. from .term import get_char
  485. while True:
  486. reply = get_char(prompt,immed_chars='yYnN').strip('\n\r')
  487. if not reply:
  488. msg_r(nl)
  489. return True if default_yes else False
  490. elif reply in 'yYnN':
  491. msg_r(nl)
  492. return True if reply in 'yY' else False
  493. else:
  494. msg_r('\nInvalid reply\n' if verbose else '\r')
  495. def do_pager(text):
  496. pagers = ['less','more']
  497. end_msg = '\n(end of text)\n\n'
  498. # --- Non-MSYS Windows code deleted ---
  499. # raw, chop, horiz scroll 8 chars, disable buggy line chopping in MSYS
  500. os.environ['LESS'] = (('--shift 8 -RS'),('-cR -#1'))[g.platform=='win']
  501. if 'PAGER' in os.environ and os.environ['PAGER'] != pagers[0]:
  502. pagers = [os.environ['PAGER']] + pagers
  503. from subprocess import run
  504. from .color import set_vt100
  505. for pager in pagers:
  506. try:
  507. m = text + ('' if pager == 'less' else end_msg)
  508. p = run([pager],input=m.encode(),check=True)
  509. msg_r('\r')
  510. except:
  511. pass
  512. else:
  513. break
  514. else:
  515. Msg(text+end_msg)
  516. set_vt100()
  517. def do_license_msg(immed=False):
  518. if opt.quiet or g.no_license or opt.yes or not g.stdin_tty:
  519. return
  520. import mmgen.contrib.license as gpl
  521. msg(gpl.warning)
  522. from .term import get_char
  523. prompt = "Press 'w' for conditions and warranty info, or 'c' to continue: "
  524. while True:
  525. reply = get_char(prompt, immed_chars=('','wc')[bool(immed)])
  526. if reply == 'w':
  527. do_pager(gpl.conditions)
  528. elif reply == 'c':
  529. msg('')
  530. break
  531. else:
  532. msg_r('\r')
  533. msg('')
  534. def get_subclasses(cls,names=False):
  535. def gen(cls):
  536. for i in cls.__subclasses__():
  537. yield i
  538. for j in gen(i):
  539. yield j
  540. return tuple((c.__name__ for c in gen(cls)) if names else gen(cls))
  541. def base_proto_subclass(cls,proto,subdir,modname,sub_clsname=None):
  542. """
  543. magic module loading and class selection
  544. """
  545. modpath = 'mmgen.base_proto.{}.{}{}'.format(
  546. proto.base_proto.lower(),
  547. subdir + '.' if subdir else '',
  548. modname )
  549. clsname = (
  550. proto.mod_clsname
  551. + ('Token' if proto.tokensym else '')
  552. + cls.__name__ )
  553. import importlib
  554. if sub_clsname:
  555. return getattr(getattr(importlib.import_module(modpath),clsname),sub_clsname)
  556. else:
  557. return getattr(importlib.import_module(modpath),clsname)
  558. # decorator for TrackingWallet
  559. def write_mode(orig_func):
  560. def f(self,*args,**kwargs):
  561. if self.mode != 'w':
  562. die(1,'{} opened in read-only mode: cannot execute method {}()'.format(
  563. type(self).__name__,
  564. locals()['orig_func'].__name__
  565. ))
  566. return orig_func(self,*args,**kwargs)
  567. return f
  568. def run_session(callback,backend=None):
  569. async def do():
  570. if (backend or opt.rpc_backend) == 'aiohttp':
  571. import aiohttp
  572. async with aiohttp.ClientSession(
  573. headers = { 'Content-Type': 'application/json' },
  574. connector = aiohttp.TCPConnector(limit_per_host=g.aiohttp_rpc_queue_len),
  575. ) as g.session:
  576. return await callback
  577. else:
  578. return await callback
  579. import asyncio
  580. return asyncio.run(do())
  581. def wrap_ripemd160(called=[]):
  582. if not called:
  583. try:
  584. import hashlib
  585. hashlib.new('ripemd160')
  586. except ValueError:
  587. def hashlib_new_wrapper(name,*args,**kwargs):
  588. if name == 'ripemd160':
  589. return ripemd160(*args,**kwargs)
  590. else:
  591. return hashlib_new(name,*args,**kwargs)
  592. from .contrib.ripemd160 import ripemd160
  593. hashlib_new = hashlib.new
  594. hashlib.new = hashlib_new_wrapper
  595. called.append(True)