runner.py 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559
  1. #!/usr/bin/env python3
  2. #
  3. # MMGen Wallet, a terminal-based cryptocurrency wallet
  4. # Copyright (C)2013-2025 The MMGen Project <mmgen@tuta.io>
  5. # Licensed under the GNU General Public License, Version 3:
  6. # https://www.gnu.org/licenses
  7. # Public project repositories:
  8. # https://github.com/mmgen/mmgen-wallet
  9. # https://gitlab.com/mmgen/mmgen-wallet
  10. """
  11. test.cmdtest_d.include.runner: test runner for the MMGen Wallet cmdtest suite
  12. """
  13. import sys, os, time, asyncio
  14. from mmgen.cfg import gc
  15. from mmgen.color import red, yellow, green, blue, cyan, gray, nocolor
  16. from mmgen.util import msg, Msg, rmsg, bmsg, die, suf, make_timestr
  17. from ...include.common import (
  18. cmdtest_py_log_fn,
  19. iqmsg,
  20. omsg,
  21. omsg_r,
  22. ok,
  23. start_test_daemons,
  24. init_coverage,
  25. clean
  26. )
  27. from .common import get_file_with_ext, confirm_continue
  28. from .cfg import cfgs, cmd_groups_dfl
  29. from .group_mgr import CmdGroupMgr
  30. def format_args(args):
  31. try:
  32. return ' '.join((f"'{a}'" if ' ' in a else a) for a in args).replace('\\', '/') # for MSYS2
  33. except Exception as e:
  34. print(type(e), e)
  35. print('cmdline:', args)
  36. class CmdTestRunner:
  37. 'cmdtest.py test runner'
  38. def __del__(self):
  39. if self.logging:
  40. self.log_fd.close()
  41. def __init__(self, cfg, repo_root, data_dir, trash_dir, trash_dir2):
  42. self.cfg = cfg
  43. self.proto = cfg._proto
  44. self.data_dir = data_dir
  45. self.trash_dir = trash_dir
  46. self.trash_dir2 = trash_dir2
  47. self.cmd_total = 0
  48. self.rebuild_list = {}
  49. self.gm = CmdGroupMgr(cfg)
  50. self.repo_root = repo_root
  51. self.skipped_warnings = []
  52. self.resume_cmd = None
  53. self.deps_only = None
  54. self.logging = self.cfg.log or os.getenv('MMGEN_EXEC_WRAPPER')
  55. self.testing_segwit = cfg.segwit or cfg.segwit_random or cfg.bech32
  56. self.network_id = self.proto.coin.lower() + ('_tn' if self.proto.testnet else '')
  57. self.daemon_started = False
  58. self.quiet = not (cfg.exact_output or cfg.verbose)
  59. global qmsg, qmsg_r
  60. if cfg.exact_output:
  61. qmsg = qmsg_r = lambda s: None
  62. else:
  63. qmsg = cfg._util.qmsg
  64. qmsg_r = cfg._util.qmsg_r
  65. if self.logging:
  66. self.log_fd = open(cmdtest_py_log_fn, 'a')
  67. self.log_fd.write(f'\nLog started: {make_timestr()} UTC\n')
  68. omsg(f'INFO → Logging to file {cmdtest_py_log_fn!r}')
  69. else:
  70. self.log_fd = None
  71. if self.cfg.coverage:
  72. coverdir, accfile = init_coverage()
  73. omsg(f'INFO → Writing coverage files to {coverdir!r}')
  74. self.pre_args = ['python3', '-m', 'trace', '--count', '--coverdir='+coverdir, '--file='+accfile]
  75. else:
  76. self.pre_args = ['python3'] if sys.platform == 'win32' else []
  77. if self.cfg.pexpect_spawn:
  78. omsg('INFO → Using pexpect.spawn() for real terminal emulation')
  79. self.set_spawn_env()
  80. self.start_time = time.time()
  81. def do_between(self):
  82. if self.cfg.pause:
  83. confirm_continue()
  84. elif not (self.quiet or self.cfg.skipping_deps):
  85. sys.stderr.write('\n')
  86. def set_spawn_env(self):
  87. self.spawn_env = dict(os.environ)
  88. self.spawn_env.update({
  89. 'MMGEN_NO_LICENSE': '1',
  90. 'MMGEN_BOGUS_SEND': '1',
  91. 'MMGEN_TEST_SUITE_PEXPECT': '1',
  92. 'EXEC_WRAPPER_DO_RUNTIME_MSG':'1',
  93. # if cmdtest.py itself is running under exec_wrapper, disable writing of traceback file for spawned script
  94. 'EXEC_WRAPPER_TRACEBACK': '' if os.getenv('MMGEN_EXEC_WRAPPER') else '1',
  95. })
  96. if self.cfg.exact_output:
  97. from mmgen.term import get_terminal_size
  98. self.spawn_env['MMGEN_COLUMNS'] = str(get_terminal_size().width)
  99. else:
  100. self.spawn_env['MMGEN_COLUMNS'] = '120'
  101. def spawn_wrapper(
  102. self,
  103. cmd = '',
  104. args = [],
  105. extra_desc = '',
  106. no_output = False,
  107. msg_only = False,
  108. log_only = False,
  109. no_msg = False,
  110. cmd_dir = 'cmds',
  111. no_exec_wrapper = False,
  112. timeout = None,
  113. pexpect_spawn = None,
  114. direct_exec = False,
  115. no_passthru_opts = False,
  116. spawn_env_override = None,
  117. exit_val = None,
  118. silent = False,
  119. env = {}):
  120. self.exit_val = exit_val
  121. desc = self.tg.test_name if self.cfg.names else self.gm.dpy_data[self.tg.test_name][1]
  122. if extra_desc:
  123. desc += ' ' + extra_desc
  124. cmd_path = (
  125. cmd if self.cfg.system # self.cfg.system is broken for main test group with overlay tree
  126. else os.path.relpath(os.path.join(self.repo_root, cmd_dir, cmd)))
  127. passthru_opts = (
  128. self.passthru_opts if not no_passthru_opts else
  129. [] if no_passthru_opts is True else
  130. [o for o in self.passthru_opts
  131. if o[2:].split('=')[0].replace('-','_') not in no_passthru_opts])
  132. args = (
  133. self.pre_args +
  134. ([] if no_exec_wrapper else ['scripts/exec_wrapper.py']) +
  135. [cmd_path] +
  136. passthru_opts +
  137. args)
  138. cmd_disp = format_args(args)
  139. if self.logging:
  140. self.log_fd.write('[{}][{}:{}] {}\n'.format(
  141. (self.proto.coin.lower() if 'coin' in self.tg.passthru_opts else 'NONE'),
  142. self.tg.group_name,
  143. self.tg.test_name,
  144. cmd_disp))
  145. if log_only:
  146. return
  147. for i in args: # die only after writing log entry
  148. if not isinstance(i, str):
  149. die(2, 'Error: missing input files in cmd line?:\nName: {}\nCmdline: {!r}'.format(
  150. self.tg.test_name,
  151. args))
  152. if not no_msg:
  153. t_pfx = '' if self.cfg.no_timings else f'[{time.time() - self.start_time:08.2f}] '
  154. if (not self.quiet) or self.cfg.print_cmdline:
  155. omsg(green(f'{t_pfx}Testing: {desc}'))
  156. if not msg_only:
  157. clr1, clr2 = (nocolor, nocolor) if self.cfg.print_cmdline else (green, cyan)
  158. omsg(
  159. clr1('Executing: ') +
  160. clr2(repr(cmd_disp) if sys.platform == 'win32' else cmd_disp)
  161. )
  162. else:
  163. omsg_r('{a}Testing {b}: {c}'.format(
  164. a = t_pfx,
  165. b = desc,
  166. c = 'OK\n' if direct_exec or self.cfg.direct_exec else ''))
  167. if msg_only:
  168. return
  169. # NB: the `pexpect_spawn` arg enables hold_protect and send_delay while the corresponding cmdline
  170. # option does not. For performance reasons, this is the desired behavior. For full emulation of
  171. # the user experience with hold protect enabled, specify --buf-keypress or --demo.
  172. send_delay = 0.4 if pexpect_spawn is True or self.cfg.buf_keypress else None
  173. pexpect_spawn = pexpect_spawn if pexpect_spawn is not None else bool(self.cfg.pexpect_spawn)
  174. spawn_env = dict(spawn_env_override or self.tg.spawn_env)
  175. spawn_env.update({
  176. 'MMGEN_HOLD_PROTECT_DISABLE': '' if send_delay else '1',
  177. 'MMGEN_TEST_SUITE_POPEN_SPAWN': '' if pexpect_spawn else '1',
  178. 'EXEC_WRAPPER_EXIT_VAL': '' if exit_val is None else str(exit_val),
  179. })
  180. spawn_env.update(env)
  181. from .pexpect import CmdTestPexpect
  182. return CmdTestPexpect(
  183. args = args,
  184. no_output = no_output,
  185. spawn_env = spawn_env,
  186. pexpect_spawn = pexpect_spawn,
  187. timeout = timeout,
  188. send_delay = send_delay,
  189. silent = silent,
  190. direct_exec = direct_exec)
  191. def end_msg(self):
  192. t = int(time.time() - self.start_time)
  193. sys.stderr.write(green(
  194. f'{self.cmd_total} test{suf(self.cmd_total)} performed' +
  195. ('\n' if self.cfg.no_timings else f'. Elapsed time: {t//60:02d}:{t%60:02d}\n')
  196. ))
  197. def init_group(self, gname, sg_name=None, cmd=None, quiet=False, do_clean=True):
  198. from .cfg import cmd_groups_altcoin
  199. if self.cfg.no_altcoin and gname in cmd_groups_altcoin:
  200. omsg(gray(f'INFO → skipping test group {gname!r} (--no-altcoin)'))
  201. return None
  202. ct_cls = self.gm.load_mod(gname)
  203. if sys.platform in ct_cls.platform_skip:
  204. omsg(gray(f'INFO → skipping test {gname!r} for platform {sys.platform!r}'))
  205. return None
  206. for k in ('segwit', 'segwit_random', 'bech32'):
  207. if getattr(self.cfg, k):
  208. segwit_opt = k
  209. break
  210. else:
  211. segwit_opt = None
  212. def gen_msg():
  213. yield ('{g}:{c}' if cmd else 'test group {g!r}').format(g=gname, c=cmd)
  214. if len(ct_cls.networks) != 1:
  215. yield f' for {self.proto.coin} {self.proto.network}'
  216. if segwit_opt:
  217. yield ' (--{})'.format(segwit_opt.replace('_', '-'))
  218. m = ''.join(gen_msg())
  219. if segwit_opt and not ct_cls.segwit_opts_ok:
  220. iqmsg(gray(f'INFO → skipping {m}'))
  221. return None
  222. # 'networks = ()' means all networks allowed
  223. nws = [(e.split('_')[0], 'testnet') if '_' in e else (e, 'mainnet') for e in ct_cls.networks]
  224. if nws:
  225. coin = self.proto.coin.lower()
  226. for a, b in nws:
  227. if a == coin and b == self.proto.network:
  228. break
  229. else:
  230. iqmsg(gray(f'INFO → skipping {m} for {self.proto.coin} {self.proto.network}'))
  231. return None
  232. if do_clean and not self.cfg.skipping_deps:
  233. clean(
  234. cfgs,
  235. tmpdir_ids = ct_cls.tmpdir_nums,
  236. extra_dirs = [self.data_dir, self.trash_dir, self.trash_dir2])
  237. if not quiet:
  238. bmsg('Executing ' + m)
  239. if (not self.daemon_started) and self.gm.get_cls_by_gname(gname).need_daemon:
  240. start_test_daemons(self.network_id, remove_datadir=True)
  241. self.daemon_started = True
  242. if hasattr(self, 'tg'):
  243. del self.tg
  244. self.tg = self.gm.gm_init_group(self.cfg, self, gname, sg_name, self.spawn_wrapper)
  245. self.ct_clsname = type(self.tg).__name__
  246. # pass through opts from cmdline (po.user_opts)
  247. self.passthru_opts = ['--{}{}'.format(
  248. k.replace('_', '-'),
  249. '' if self.cfg._uopts[k] is True else '=' + self.cfg._uopts[k]
  250. ) for k in self.cfg._uopts
  251. if self.cfg._uopts[k] and k in self.tg.base_passthru_opts + self.tg.passthru_opts]
  252. if self.cfg.resuming:
  253. rc = self.cfg.resume or self.cfg.resume_after
  254. offset = 1 if self.cfg.resume_after else 0
  255. self.resume_cmd = self.gm.cmd_list[self.gm.cmd_list.index(rc)+offset]
  256. omsg(f'INFO → Resuming at command {self.resume_cmd!r}')
  257. if self.cfg.step:
  258. self.cfg.exit_after = self.resume_cmd
  259. if self.cfg.exit_after and self.cfg.exit_after not in self.gm.cmd_list:
  260. die(1, f'{self.cfg.exit_after!r}: command not recognized')
  261. return self.tg
  262. def run_tests(self, cmd_args):
  263. gname_save = None
  264. def parse_arg(arg):
  265. if '.' in arg:
  266. a, b = arg.split('.')
  267. return [a] + b.split(':') if ':' in b else [a, b, None]
  268. elif ':' in arg:
  269. a, b = arg.split(':')
  270. return [a, None, b]
  271. else:
  272. return [self.gm.find_cmd_in_groups(arg), None, arg]
  273. if cmd_args:
  274. for arg in cmd_args:
  275. if arg in self.gm.cmd_groups:
  276. if self.init_group(arg):
  277. for cmd in self.gm.cmd_list:
  278. self.check_needs_rerun(cmd, build=True)
  279. self.do_between()
  280. else:
  281. gname, sg_name, cmdname = parse_arg(arg)
  282. if gname:
  283. same_grp = gname == gname_save # same group as previous cmd: don't clean, suppress blue msg
  284. if self.init_group(gname, sg_name, cmdname, quiet=same_grp, do_clean=not same_grp):
  285. if cmdname:
  286. if self.cfg.deps_only:
  287. self.deps_only = cmdname
  288. try:
  289. self.check_needs_rerun(cmdname, build=True)
  290. except Exception as e: # allow calling of functions not in cmd_group
  291. if isinstance(e, KeyError) and e.args[0] == cmdname:
  292. ret = getattr(self.tg, cmdname)()
  293. if type(ret).__name__ == 'coroutine':
  294. ret = asyncio.run(ret)
  295. self.process_retval(cmdname, ret)
  296. else:
  297. raise
  298. self.do_between()
  299. else:
  300. for cmd in self.gm.cmd_list:
  301. self.check_needs_rerun(cmd, build=True)
  302. self.do_between()
  303. gname_save = gname
  304. else:
  305. die(1, f'{arg!r}: command not recognized')
  306. else:
  307. if self.cfg.exclude_groups:
  308. exclude = self.cfg.exclude_groups.split(',')
  309. for e in exclude:
  310. if e not in cmd_groups_dfl:
  311. die(1, f'{e!r}: group not recognized')
  312. for gname in cmd_groups_dfl:
  313. if self.cfg.exclude_groups and gname in exclude:
  314. continue
  315. if self.init_group(gname):
  316. for cmd in self.gm.cmd_list:
  317. self.check_needs_rerun(cmd, build=True)
  318. self.do_between()
  319. self.end_msg()
  320. def check_needs_rerun(
  321. self,
  322. cmd,
  323. build = False,
  324. root = True,
  325. force_delete = False,
  326. dpy = False):
  327. self.tg.test_name = cmd
  328. if self.ct_clsname == 'CmdTestMain' and self.testing_segwit and cmd not in self.tg.segwit_do:
  329. return False
  330. rerun = root # force_delete is not passed to recursive call
  331. fns = []
  332. if force_delete or not root:
  333. # does cmd produce a needed dependency(ies)?
  334. ret = self.get_num_exts_for_cmd(cmd)
  335. if ret:
  336. for ext in ret[1]:
  337. fn = get_file_with_ext(cfgs[ret[0]]['tmpdir'], ext, delete=build)
  338. if fn:
  339. if force_delete:
  340. os.unlink(fn)
  341. else: fns.append(fn)
  342. else: rerun = True
  343. fdeps = self.generate_file_deps(cmd)
  344. cdeps = self.generate_cmd_deps(fdeps)
  345. for fn in fns:
  346. my_age = os.stat(fn).st_mtime
  347. for num, ext in fdeps:
  348. f = get_file_with_ext(cfgs[num]['tmpdir'], ext, delete=build)
  349. if f and os.stat(f).st_mtime > my_age:
  350. rerun = True
  351. for cdep in cdeps:
  352. if self.check_needs_rerun(cdep, build=build, root=False, dpy=cmd):
  353. rerun = True
  354. if build:
  355. if rerun:
  356. for fn in fns:
  357. if not root:
  358. os.unlink(fn)
  359. if not (dpy and self.cfg.skipping_deps):
  360. self.run_test(cmd)
  361. if not root:
  362. self.do_between()
  363. else:
  364. # If prog produces multiple files:
  365. if cmd not in self.rebuild_list or rerun is True:
  366. self.rebuild_list[cmd] = (rerun, fns[0] if fns else '') # FIX
  367. return rerun
  368. def run_test(self, cmd, sub=False):
  369. if self.deps_only and cmd == self.deps_only:
  370. sys.exit(0)
  371. if self.tg.full_data:
  372. d = [(str(num), ext) for exts, num in self.gm.dpy_data[cmd][2] for ext in exts]
  373. # delete files depended on by this cmd
  374. arg_list = [get_file_with_ext(cfgs[num]['tmpdir'], ext) for num, ext in d]
  375. # remove shared_deps from arg list
  376. if hasattr(self.tg, 'shared_deps'):
  377. arg_list = arg_list[:-len(self.tg.shared_deps)]
  378. else:
  379. arg_list = []
  380. if self.resume_cmd:
  381. if cmd != self.resume_cmd:
  382. return
  383. bmsg(f'Resuming at {self.resume_cmd!r}')
  384. self.resume_cmd = None
  385. self.cfg.skipping_deps = False
  386. self.cfg.resuming = False
  387. if self.cfg.profile:
  388. start = time.time()
  389. self.tg.test_name = cmd # NB: Do not remove, this needs to be set twice
  390. if self.tg.full_data:
  391. tmpdir_num = self.gm.dpy_data[cmd][0]
  392. self.tg.tmpdir_num = tmpdir_num
  393. for k in (test_cfg := cfgs[str(tmpdir_num)]):
  394. if k in self.gm.cfg_attrs:
  395. setattr(self.tg, k, test_cfg[k])
  396. ret = getattr(self.tg, cmd)(*arg_list) # run the test
  397. if sub:
  398. return ret
  399. if type(ret).__name__ == 'coroutine':
  400. ret = asyncio.run(ret)
  401. self.process_retval(cmd, ret)
  402. if self.cfg.profile:
  403. omsg('\r\033[50C{:.4f}'.format(time.time() - start))
  404. if cmd == self.cfg.exit_after:
  405. sys.exit(0)
  406. def warn_skipped(self):
  407. if self.skipped_warnings:
  408. print(yellow('The following tests were skipped and may require attention:'))
  409. r = '-' * 72 + '\n'
  410. print(r+('\n'+r).join(self.skipped_warnings))
  411. def process_retval(self, cmd, ret):
  412. if type(ret).__name__ == 'CmdTestPexpect':
  413. ret.ok(exit_val=self.exit_val)
  414. self.cmd_total += 1
  415. elif ret == 'ok':
  416. ok()
  417. self.cmd_total += 1
  418. elif ret in ('skip', 'skip_msg', 'silent'):
  419. if ret == 'silent':
  420. self.cmd_total += 1
  421. elif ret == 'skip_msg':
  422. ok('SKIP')
  423. elif ret == 'error':
  424. die(2, red(f'\nTest {self.tg.test_name!r} failed'))
  425. elif isinstance(ret, tuple) and ret[0] == 'skip_warn':
  426. self.skipped_warnings.append(
  427. 'Test {!r} was skipped:\n {}'.format(cmd, '\n '.join(ret[1].split('\n'))))
  428. else:
  429. die(2, f'{cmd!r} returned {ret}')
  430. def check_deps(self, cmds): # TODO: broken, unused
  431. if len(cmds) != 1:
  432. die(1, f'Usage: {gc.prog_name} check_deps <command>')
  433. cmd = cmds[0]
  434. if cmd not in self.gm.cmd_list:
  435. die(1, f'{cmd!r}: unrecognized command')
  436. if not self.cfg.quiet:
  437. omsg(f'Checking dependencies for {cmd!r}')
  438. self.check_needs_rerun(cmd)
  439. w = max(map(len, self.rebuild_list)) + 1
  440. for cmd in self.rebuild_list:
  441. c = self.rebuild_list[cmd]
  442. m = 'Rebuild' if (c[0] and c[1]) else 'Build' if c[0] else 'OK'
  443. omsg('cmd {:<{w}} {}'.format(cmd+':', m, w=w))
  444. def generate_file_deps(self, cmd):
  445. return [(str(n), e) for exts, n in self.gm.dpy_data[cmd][2] for e in exts]
  446. def generate_cmd_deps(self, fdeps):
  447. return [cfgs[str(n)]['dep_generators'][ext] for n, ext in fdeps]
  448. def get_num_exts_for_cmd(self, cmd):
  449. try:
  450. num = str(self.gm.dpy_data[cmd][0])
  451. except KeyError:
  452. qmsg_r(f'Missing dependency {cmd!r}')
  453. gname = self.gm.find_cmd_in_groups(cmd)
  454. if gname:
  455. kwargs = self.gm.cmd_groups[gname][1]
  456. kwargs.update({'add_dpy':True})
  457. self.gm.create_group(gname, None, **kwargs)
  458. num = str(self.gm.dpy_data[cmd][0])
  459. qmsg(f' found in group {gname!r}')
  460. else:
  461. qmsg(' not found in any command group!')
  462. raise
  463. dgl = cfgs[num]['dep_generators']
  464. if cmd in dgl.values():
  465. exts = [k for k in dgl if dgl[k] == cmd]
  466. return (num, exts)
  467. else:
  468. return None