common.py 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260
  1. #!/usr/bin/env python3
  2. #
  3. # mmgen = Multi-Mode GENerator, command-line Bitcoin cold storage solution
  4. # Copyright (C)2013-2023 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.include.common: Shared routines and data for the MMGen test suites
  20. """
  21. import os
  22. from subprocess import run,PIPE
  23. from mmgen.common import *
  24. from mmgen.fileutil import write_data_to_file,get_data_from_file
  25. def strip_ansi_escapes(s):
  26. import re
  27. return re.sub('\x1b' + r'\[[;0-9]+?m','',s)
  28. ascii_uc = ''.join(map(chr,list(range(65,91)))) # 26 chars
  29. ascii_lc = ''.join(map(chr,list(range(97,123)))) # 26 chars
  30. lat_accent = ''.join(map(chr,list(range(192,383)))) # 191 chars, L,S
  31. ru_uc = ''.join(map(chr,list(range(1040,1072)))) # 32 chars
  32. gr_uc = ''.join(map(chr,list(range(913,930)) + list(range(931,940)))) # 26 chars (930 is ctrl char)
  33. gr_uc_w_ctrl = ''.join(map(chr,list(range(913,940)))) # 27 chars, L,C
  34. lat_cyr_gr = lat_accent[:130:5] + ru_uc + gr_uc # 84 chars
  35. ascii_cyr_gr = ascii_uc + ru_uc + gr_uc # 84 chars
  36. utf8_text = '[α-$ample UTF-8 text-ω]' * 10 # 230 chars, L,N,P,S,Z
  37. utf8_combining = '[α-$ámple UTF-8 téxt-ω]' * 10 # L,N,P,S,Z,M
  38. utf8_ctrl = '[α-$ample\nUTF-8\ntext-ω]' * 10 # L,N,P,S,Z,C
  39. text_jp = '必要なのは、信用ではなく暗号化された証明に基づく電子取引システムであり、これにより希望する二者が信用できる第三者機関を介さずに直接取引できるよう' # 72 chars ('W'ide)
  40. text_zh = '所以,我們非常需要這樣一種電子支付系統,它基於密碼學原理而不基於信用,使得任何達成一致的雙方,能夠直接進行支付,從而不需要協力廠商仲介的參與。。' # 72 chars ('F'ull + 'W'ide)
  41. sample_text = 'The Times 03/Jan/2009 Chancellor on brink of second bailout for banks'
  42. sample_mn = {
  43. 'mmgen': { # 'able': 0, 'youth': 1625, 'after' == 'afternoon'[:5]
  44. 'mn': 'able cast forgive master funny gaze after afternoon million paint moral youth',
  45. 'hex': '0005685ab4e94cbe3b228cf92112bc5f',
  46. },
  47. 'bip39': { # len('sun') < uniq_ss_len
  48. 'mn': 'vessel ladder alter error federal sibling chat ability sun glass valve picture',
  49. 'hex': 'f30f8c1da665478f49b001d94c5fc452',
  50. },
  51. 'xmrseed': {
  52. 'mn': 'viewpoint donuts ardent template unveil agile meant unafraid urgent athlete rustled mime azure jaded hawk baby jagged haystack baby jagged haystack ramped oncoming point template',
  53. 'hex': 'e8164dda6d42bd1e261a3406b2038dcbddadbeefdeadbeefdeadbeefdeadbe0f',
  54. },
  55. }
  56. ref_kafile_pass = 'kafile password'
  57. ref_kafile_hash_preset = '1'
  58. def getrand(n):
  59. if g.test_suite_deterministic:
  60. from mmgen.test import fake_urandom
  61. return fake_urandom(n)
  62. else:
  63. return os.urandom(n)
  64. def getrandnum(n):
  65. return int(getrand(n).hex(),16)
  66. def getrandhex(n):
  67. return getrand(n).hex()
  68. def getrandnum_range(nbytes,rn_max):
  69. while True:
  70. rn = int(getrand(nbytes).hex(),16)
  71. if rn < rn_max:
  72. return rn
  73. def getrandstr(num_chars,no_space=False):
  74. n,m = (94,33) if no_space else (95,32)
  75. return ''.join( chr(i % n + m) for i in list(getrand(num_chars)) )
  76. def get_data_dir():
  77. return os.path.join('test','data_dir' + ('','-α')[bool(os.getenv('MMGEN_DEBUG_UTF8'))])
  78. # Windows uses non-UTF8 encodings in filesystem, so use raw bytes here
  79. def cleandir(d,do_msg=False):
  80. d_enc = d.encode()
  81. try: files = os.listdir(d_enc)
  82. except: return
  83. from shutil import rmtree
  84. if do_msg:
  85. gmsg(f'Cleaning directory {d!r}')
  86. for f in files:
  87. try:
  88. os.unlink(os.path.join(d_enc,f))
  89. except:
  90. rmtree(os.path.join(d_enc,f),ignore_errors=True)
  91. def mk_tmpdir(d):
  92. try:
  93. os.makedirs( d, mode=0o755, exist_ok=True )
  94. except OSError as e:
  95. if e.errno != 17:
  96. raise
  97. else:
  98. vmsg(f'Created directory {d!r}')
  99. def get_tmpfile(cfg,fn):
  100. return os.path.join(cfg['tmpdir'],fn)
  101. def write_to_file(fn,data,binary=False):
  102. write_data_to_file(
  103. fn,
  104. data,
  105. quiet = True,
  106. binary = binary,
  107. ignore_opt_outdir = True )
  108. def write_to_tmpfile(cfg,fn,data,binary=False):
  109. write_to_file( os.path.join(cfg['tmpdir'],fn), data=data, binary=binary )
  110. def read_from_file(fn,binary=False):
  111. return get_data_from_file(fn,quiet=True,binary=binary)
  112. def read_from_tmpfile(cfg,fn,binary=False):
  113. return read_from_file(os.path.join(cfg['tmpdir'],fn),binary=binary)
  114. def joinpath(*args,**kwargs):
  115. return os.path.join(*args,**kwargs)
  116. def ok():
  117. if opt.profile:
  118. return
  119. if opt.verbose or opt.exact_output:
  120. gmsg('OK')
  121. else:
  122. msg(' OK')
  123. def cmp_or_die(s,t,desc=None):
  124. if s != t:
  125. die( 'TestSuiteFatalException',
  126. (f'For {desc}:\n' if desc else '') +
  127. f'ERROR: recoded data:\n{t!r}\ndiffers from original data:\n{s!r}'
  128. )
  129. def init_coverage():
  130. coverdir = os.path.join('test','trace')
  131. acc_file = os.path.join('test','trace.acc')
  132. try: os.mkdir(coverdir,0o755)
  133. except: pass
  134. return coverdir,acc_file
  135. def silence():
  136. if not (opt.verbose or opt.exact_output):
  137. g.stdout = g.stderr = open(os.devnull,'w')
  138. def end_silence():
  139. if not (opt.verbose or opt.exact_output):
  140. g.stdout.close()
  141. g.stdout = sys.stdout
  142. g.stderr = sys.stderr
  143. def omsg(s):
  144. sys.stderr.write(s + '\n')
  145. def omsg_r(s):
  146. sys.stderr.write(s)
  147. sys.stderr.flush()
  148. def imsg(s):
  149. if opt.verbose or opt.exact_output:
  150. omsg(s)
  151. def imsg_r(s):
  152. if opt.verbose or opt.exact_output:
  153. omsg_r(s)
  154. def iqmsg(s):
  155. if not opt.quiet:
  156. omsg(s)
  157. def iqmsg_r(s):
  158. if not opt.quiet:
  159. omsg_r(s)
  160. def oqmsg(s):
  161. if not (opt.verbose or opt.exact_output):
  162. omsg(s)
  163. def oqmsg_r(s):
  164. if not (opt.verbose or opt.exact_output):
  165. omsg_r(s)
  166. def end_msg(t):
  167. omsg(green(
  168. 'All requested tests finished OK' +
  169. ('' if g.test_suite_deterministic else f', elapsed time: {t//60:02d}:{t%60:02d}')
  170. ))
  171. def start_test_daemons(*network_ids,remove_datadir=False):
  172. if not opt.no_daemon_autostart:
  173. return test_daemons_ops(*network_ids,op='start',remove_datadir=remove_datadir)
  174. def stop_test_daemons(*network_ids,force=False,remove_datadir=False):
  175. if force or not opt.no_daemon_stop:
  176. return test_daemons_ops(*network_ids,op='stop',remove_datadir=remove_datadir)
  177. def restart_test_daemons(*network_ids,remove_datadir=False):
  178. if not stop_test_daemons(*network_ids):
  179. return False
  180. return start_test_daemons(*network_ids,remove_datadir=remove_datadir)
  181. def test_daemons_ops(*network_ids,op,remove_datadir=False):
  182. if not opt.no_daemon_autostart:
  183. from mmgen.daemon import CoinDaemon
  184. silent = not (opt.verbose or opt.exact_output)
  185. ret = False
  186. for network_id in network_ids:
  187. d = CoinDaemon(network_id,test_suite=True)
  188. if remove_datadir:
  189. d.stop(silent=True)
  190. d.remove_datadir()
  191. ret = d.cmd(op,silent=silent)
  192. return ret
  193. tested_solc_ver = '0.8.7'
  194. def check_solc_ver():
  195. cmd = 'python3 scripts/create-token.py --check-solc-version'
  196. try:
  197. cp = run(cmd.split(),check=False,stdout=PIPE)
  198. except Exception as e:
  199. die(4,f'Unable to execute {cmd!r}: {e}')
  200. res = cp.stdout.decode().strip()
  201. if cp.returncode == 0:
  202. omsg(
  203. orange(f'Found supported solc version {res}') if res == tested_solc_ver else
  204. yellow(f'WARNING: solc version ({res}) does not match tested version ({tested_solc_ver})')
  205. )
  206. return True
  207. else:
  208. omsg(yellow('Warning: Solidity compiler (solc) could not be executed or has unsupported version'))
  209. omsg(res)
  210. return False
  211. def get_ethkey():
  212. cmdnames = ('ethkey','openethereum-ethkey')
  213. for cmdname in cmdnames:
  214. try: run([cmdname,'--help'],stdout=PIPE)
  215. except: pass
  216. else:
  217. return cmdname
  218. else:
  219. die(1,f'ethkey executable not found (tried {cmdnames})')