ut_tx.py 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162
  1. #!/usr/bin/env python3
  2. """
  3. test.modtest_d.ut_tx: TX unit tests for the MMGen suite
  4. """
  5. import os
  6. from mmgen.tx import CompletedTX, UnsignedTX
  7. from mmgen.tx.file import MMGenTxFile
  8. from mmgen.cfg import Config
  9. from ..include.common import cfg, qmsg, vmsg, gr_uc
  10. async def do_txfile_test(desc, fns, cfg=cfg, check=False):
  11. qmsg(f'\n Testing CompletedTX initializer ({desc})')
  12. for fn in fns:
  13. qmsg(f' parsing: {os.path.basename(fn)}')
  14. fpath = os.path.join('test', 'ref', fn)
  15. tx = await CompletedTX(cfg=cfg, filename=fpath, quiet_open=True)
  16. vmsg('\n' + tx.info.format())
  17. f = MMGenTxFile(tx)
  18. fn_gen = f.make_filename()
  19. if cfg.debug_utf8:
  20. fn_gen = fn_gen.replace('-α', '')
  21. assert fn_gen == os.path.basename(fn), f'{fn_gen} != {fn}'
  22. if check:
  23. text = f.format()
  24. with open(fpath) as fh:
  25. text_chk = fh.read()
  26. assert text == text_chk, f'\nformatted text:\n{text}\n !=\noriginal file:\n{text_chk}'
  27. qmsg(' OK')
  28. return True
  29. class unit_tests:
  30. altcoin_deps = ('txfile_alt', 'txfile_alt_legacy')
  31. async def txfile(self, name, ut, desc='displaying transaction files (BTC)'):
  32. return await do_txfile_test(
  33. 'Bitcoin',
  34. (
  35. 'tx/7A8157[6.65227,34].rawtx',
  36. 'tx/BB3FD2[7.57134314,123].sigtx',
  37. 'tx/0A869F[1.23456,32].regtest.asubtx',
  38. ),
  39. check = True
  40. )
  41. async def txfile_alt(self, name, ut, desc='displaying transaction files (LTC, BCH, ETH)'):
  42. return await do_txfile_test(
  43. 'altcoins',
  44. (
  45. 'tx/C09D73-LTC[981.73747,2000].testnet.rawtx',
  46. 'tx/91060A-BCH[1.23456].regtest.arawtx',
  47. 'tx/D850C6-MM1[43.21,50000].subtx', # token tx
  48. ),
  49. # token resolved by tracking wallet under data_dir:
  50. cfg = Config({'data_dir': 'test/ref/data_dir'}),
  51. check = True
  52. )
  53. async def txfile_legacy(self, name, ut, desc='displaying transaction files (legacy format, BTC)'):
  54. return await do_txfile_test(
  55. 'Bitcoin - legacy file format',
  56. (
  57. '0B8D5A[15.31789,14,tl=1320969600].rawtx',
  58. '542169[5.68152,34].sigtx',
  59. '0C7115[15.86255,14,tl=1320969600].testnet.rawtx',
  60. '25EFA3[2.34].testnet.rawtx',
  61. )
  62. )
  63. async def txfile_alt_legacy(self, name, ut, desc='displaying transaction files (legacy format, LTC, BCH, ETH)'):
  64. return await do_txfile_test(
  65. 'altcoins - legacy file format',
  66. (
  67. '460D4D-BCH[10.19764,tl=1320969600].rawtx',
  68. 'ethereum/5881D2-MM1[1.23456,50000].rawtx',
  69. 'ethereum/6BDB25-MM1[1.23456,50000].testnet.rawtx',
  70. 'ethereum/88FEFD-ETH[23.45495,40000].rawtx',
  71. 'ethereum/B472BD-ETH[23.45495,40000].testnet.rawtx',
  72. 'ethereum/B472BD-ETH[23.45495,40000].testnet.sigtx',
  73. 'litecoin/A5A1E0-LTC[1454.64322,1453,tl=1320969600].testnet.rawtx',
  74. 'litecoin/AF3CDF-LTC[620.76194,1453,tl=1320969600].rawtx',
  75. )
  76. )
  77. def errors(self, name, ut, desc='reading transaction files (error handling)'):
  78. async def bad1():
  79. await CompletedTX(cfg, filename='foo')
  80. def bad2():
  81. UnsignedTX(cfg, filename='foo')
  82. bad_data = (
  83. ('forbidden positional args', 'TypeError', 'positional arguments', bad1),
  84. ('forbidden positional args', 'TypeError', 'positional arguments', bad2),
  85. )
  86. ut.process_bad_data(bad_data)
  87. return True
  88. def op_return_data(self, name, ut, desc='OpReturnData class'):
  89. from mmgen.proto.btc.tx.op_return_data import OpReturnData
  90. vecs = [
  91. 'data:=:ETH.ETH:0x86d526d6624AbC0178cF7296cD538Ecc080A95F1:0/1/0',
  92. 'hexdata:3d3a4554482e4554483a30783836643532366436363234416243303137'
  93. '38634637323936634435333845636330383041393546313a302f312f30',
  94. 'hexdata:00010203040506',
  95. 'data:a\n',
  96. 'data:a\tb',
  97. 'data:' + gr_uc[:24],
  98. ]
  99. assert OpReturnData(cfg._proto, vecs[0]) == OpReturnData(cfg._proto, vecs[1])
  100. for vec in vecs:
  101. d = OpReturnData(cfg._proto, vec)
  102. assert d == OpReturnData(cfg._proto, repr(d)) # repr() must return a valid initializer
  103. assert isinstance(d, bytes)
  104. assert isinstance(str(d), str)
  105. vmsg('-' * 80)
  106. vmsg(vec)
  107. vmsg(repr(d))
  108. vmsg(d.hl())
  109. vmsg(d.hl(add_label=True))
  110. bad_data = [
  111. 'data:',
  112. 'hexdata:',
  113. 'data:' + ('x' * 81),
  114. 'hexdata:' + ('deadbeef' * 20) + 'ee',
  115. 'hex:0abc',
  116. 'da:xyz',
  117. 'hexdata:xyz',
  118. 'hexdata:abcde',
  119. b'data:abc',
  120. ]
  121. def bad(n):
  122. return lambda: OpReturnData(cfg._proto, bad_data[n])
  123. vmsg('-' * 80)
  124. vmsg('Testing error handling:')
  125. ut.process_bad_data((
  126. ('bad1', 'AssertionError', 'not in range', bad(0)),
  127. ('bad2', 'AssertionError', 'not in range', bad(1)),
  128. ('bad3', 'AssertionError', 'not in range', bad(2)),
  129. ('bad4', 'AssertionError', 'not in range', bad(3)),
  130. ('bad5', 'ValueError', 'must start', bad(4)),
  131. ('bad6', 'ValueError', 'must start', bad(5)),
  132. ('bad7', 'AssertionError', 'not in hex', bad(6)),
  133. ('bad8', 'AssertionError', 'even', bad(7)),
  134. ('bad9', 'AssertionError', 'a string', bad(8)),
  135. ), pfx='')
  136. return True