base.py 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139
  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. #
  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.cmdtest_d.base: Base class for the cmdtest.py test suite
  20. """
  21. import sys, os
  22. from mmgen.util import msg
  23. from mmgen.color import gray, purple, yellow
  24. from ..include.common import write_to_file, read_from_file, imsg
  25. from .include.common import get_file_with_ext
  26. class CmdTestBase:
  27. 'initializer class for the cmdtest.py test suite'
  28. base_passthru_opts = ('data_dir', 'skip_cfg_file')
  29. passthru_opts = ()
  30. networks = ()
  31. segwit_opts_ok = False
  32. color = False
  33. need_daemon = False
  34. platform_skip = ()
  35. tmpdir_nums = []
  36. test_name = None
  37. is_helper = False
  38. def __init__(self, cfg, trunner, cfgs, spawn):
  39. if hasattr(self, 'name'): # init will be called multiple times for classes with multiple inheritance
  40. return
  41. self.name = type(self).__name__
  42. self.cfg = cfg
  43. self.proto = cfg._proto
  44. self.tr = trunner
  45. self.cfgs = cfgs
  46. self.spawn = spawn
  47. self.have_dfl_wallet = False
  48. self.usr_rand_chars = (5, 30)[bool(cfg.usr_random)]
  49. self.usr_rand_arg = f'-r{self.usr_rand_chars}'
  50. self.tn_ext = ('', '.testnet')[self.proto.testnet]
  51. self.coin = self.proto.coin.lower()
  52. self.fork = 'btc' if self.coin == 'bch' and not cfg.cashaddr else self.coin
  53. self.altcoin_pfx = '' if self.fork == 'btc' else f'-{self.proto.coin}'
  54. self.testnet_opt = ['--testnet=1'] if cfg.testnet else []
  55. if len(self.tmpdir_nums) == 1:
  56. self.tmpdir_num = self.tmpdir_nums[0]
  57. if self.tr:
  58. self.spawn_env = dict(self.tr.spawn_env)
  59. self.spawn_env['MMGEN_TEST_SUITE_ENABLE_COLOR'] = '1' if self.color else ''
  60. else:
  61. self.spawn_env = {} # placeholder
  62. def get_altcoin_pfx(self, coin, cashaddr=True):
  63. coin = coin.lower()
  64. fork = 'btc' if coin == 'bch' and not cashaddr else coin
  65. return '' if fork == 'btc' else f'-{coin.upper()}'
  66. @property
  67. def tmpdir(self):
  68. return os.path.join('test', 'tmp', '{}{}'.format(self.tmpdir_num, '-α' if self.cfg.debug_utf8 else ''))
  69. def get_file_with_ext(self, ext, **kwargs):
  70. return get_file_with_ext(self.tmpdir, ext, **kwargs)
  71. def read_from_tmpfile(self, fn, binary=False, subdir=None):
  72. tdir = os.path.join(self.tmpdir, subdir) if subdir else self.tmpdir
  73. return read_from_file(os.path.join(tdir, fn), binary=binary)
  74. def write_to_tmpfile(self, fn, data, binary=False):
  75. return write_to_file(os.path.join(self.tmpdir, fn), data, binary=binary)
  76. def delete_tmpfile(self, fn):
  77. fn = os.path.join(self.tmpdir, fn)
  78. try:
  79. return os.unlink(fn)
  80. except:
  81. msg(f'{fn}: file does not exist or could not be deleted')
  82. def skip_for_platform(self, name, extra_msg=None):
  83. if sys.platform == name:
  84. msg(gray('Skipping test {!r} for {} platform{}'.format(
  85. self.test_name,
  86. name,
  87. f' ({extra_msg})' if extra_msg else "")))
  88. return True
  89. else:
  90. return False
  91. def skip_for_mac(self, extra_msg=None):
  92. return self.skip_for_platform('darwin', extra_msg)
  93. def skip_for_win(self, extra_msg=None):
  94. return self.skip_for_platform('win32', extra_msg)
  95. def spawn_chk(self, *args, **kwargs):
  96. """
  97. Drop-in replacement for spawn() + t.read() for tests that spawn more than one process.
  98. Ensures that test script execution stops when a spawned process fails.
  99. """
  100. t = self.spawn(*args, **kwargs)
  101. t.read()
  102. t.ok()
  103. t.skip_ok = True
  104. return t
  105. def noop(self):
  106. return 'ok'
  107. def _cashaddr_opt(self, val):
  108. return [f'--cashaddr={val}'] if self.proto.coin == 'BCH' else []
  109. def _kill_process_from_pid_file(self, fn, desc):
  110. self.spawn(msg_only=True)
  111. pid = int(self.read_from_tmpfile(fn))
  112. self.delete_tmpfile(fn)
  113. from signal import SIGTERM
  114. imsg(purple(f'Killing {desc} [PID {pid}]'))
  115. try:
  116. os.kill(pid, SIGTERM)
  117. except:
  118. imsg(yellow(f'{pid}: no such process'))
  119. return 'ok'