ct_base.py 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  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.cmdtest_py_d.ct_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
  24. from ..include.common import cfg,write_to_file,read_from_file
  25. from .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. extra_spawn_args = []
  31. networks = ()
  32. segwit_opts_ok = False
  33. color = False
  34. need_daemon = False
  35. win_skip = False
  36. mac_skip = False
  37. tmpdir_nums = []
  38. test_name = None
  39. def __init__(self,trunner,cfgs,spawn):
  40. if hasattr(self,'name'): # init will be called multiple times for classes with multiple inheritance
  41. return
  42. self.name = type(self).__name__
  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.altcoin_pfx = '' if self.proto.base_coin == 'BTC' else '-'+self.proto.base_coin
  51. self.tn_ext = ('','.testnet')[self.proto.testnet]
  52. d = {'bch':'btc','btc':'btc','ltc':'ltc'}
  53. self.fork = d[self.proto.coin.lower()] if self.proto.coin.lower() in d else None
  54. if len(self.tmpdir_nums) == 1:
  55. self.tmpdir_num = self.tmpdir_nums[0]
  56. if self.tr:
  57. self.spawn_env = dict(self.tr.spawn_env)
  58. self.spawn_env['MMGEN_TEST_SUITE_ENABLE_COLOR'] = '1' if self.color else ''
  59. else:
  60. self.spawn_env = {} # placeholder
  61. @property
  62. def tmpdir(self):
  63. return os.path.join('test','tmp','{}{}'.format(self.tmpdir_num,'-α' if cfg.debug_utf8 else ''))
  64. def get_file_with_ext(self,ext,**kwargs):
  65. return get_file_with_ext(self.tmpdir,ext,**kwargs)
  66. def read_from_tmpfile(self,fn,binary=False):
  67. return read_from_file(os.path.join(self.tmpdir,fn),binary=binary)
  68. def write_to_tmpfile(self,fn,data,binary=False):
  69. return write_to_file(os.path.join(self.tmpdir,fn),data,binary=binary)
  70. def delete_tmpfile(self,fn):
  71. fn = os.path.join(self.tmpdir,fn)
  72. try:
  73. return os.unlink(fn)
  74. except:
  75. msg(f'{fn}: file does not exist or could not be deleted')
  76. def skip_for_platform(self, name, extra_msg=None):
  77. if sys.platform == name:
  78. msg(gray('Skipping test {!r} for {} platform{}'.format(
  79. self.test_name,
  80. name,
  81. f' ({extra_msg})' if extra_msg else "")))
  82. return True
  83. else:
  84. return False
  85. def skip_for_mac(self, extra_msg=None):
  86. return self.skip_for_platform('darwin', extra_msg)
  87. def skip_for_win(self, extra_msg=None):
  88. return self.skip_for_platform('win32', extra_msg)
  89. def spawn_chk(self,*args,**kwargs):
  90. """
  91. Drop-in replacement for spawn() + t.read() for tests that spawn more than one process.
  92. Ensures that test script execution stops when a spawned process fails.
  93. """
  94. t = self.spawn(*args,**kwargs)
  95. t.read()
  96. t.ok()
  97. t.skip_ok = True
  98. return t