common.py 8.5 KB

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