thornode.py 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144
  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. swap.proto.thorchain.thornode: THORChain swap protocol network query ops
  12. """
  13. import json
  14. class ThornodeRPCClient:
  15. http_hdrs = {'Content-Type': 'application/json'}
  16. proto = 'https'
  17. host = 'thornode.ninerealms.com'
  18. verify = True
  19. timeout = 5
  20. def __init__(self, tx, *, proto=None, host=None):
  21. self.cfg = tx.cfg
  22. if proto:
  23. self.proto = proto
  24. if host:
  25. self.host = host
  26. import requests
  27. self.session = requests.Session()
  28. self.session.trust_env = False # ignore *_PROXY environment vars
  29. self.session.headers = self.http_hdrs
  30. if self.cfg.proxy:
  31. self.session.proxies.update({
  32. 'http': f'socks5h://{self.cfg.proxy}',
  33. 'https': f'socks5h://{self.cfg.proxy}'
  34. })
  35. def get(self, path, *, timeout=None):
  36. return self.session.get(
  37. url = self.proto + '://' + self.host + path,
  38. timeout = timeout or self.timeout,
  39. verify = self.verify)
  40. class Thornode:
  41. def __init__(self, tx, amt):
  42. self.tx = tx
  43. self.in_amt = amt
  44. self.rpc = ThornodeRPCClient(tx)
  45. def get_quote(self):
  46. self.get_str = '/thorchain/quote/swap?from_asset={a}.{a}&to_asset={b}.{b}&amount={c}'.format(
  47. a = self.tx.send_proto.coin,
  48. b = self.tx.recv_proto.coin,
  49. c = self.in_amt.to_unit('satoshi'))
  50. self.result = self.rpc.get(self.get_str)
  51. self.data = json.loads(self.result.content)
  52. if not 'expiry' in self.data:
  53. from ....util import pp_fmt, die
  54. die(2, pp_fmt(self.data))
  55. def format_quote(self, trade_limit, usr_trade_limit, *, deduct_est_fee=False):
  56. from ....util import make_timestr, ymsg
  57. from ....util2 import format_elapsed_hr
  58. from ....color import blue, green, cyan, pink, orange, redbg, yelbg, grnbg
  59. from . import name
  60. d = self.data
  61. tx = self.tx
  62. in_coin = tx.send_proto.coin
  63. out_coin = tx.recv_proto.coin
  64. in_amt = self.in_amt
  65. out_amt = tx.recv_proto.coin_amt(int(d['expected_amount_out']), from_unit='satoshi')
  66. if trade_limit:
  67. from . import ExpInt4
  68. e = ExpInt4(trade_limit.to_unit('satoshi'))
  69. tl_rounded = tx.recv_proto.coin_amt(e.trunc, from_unit='satoshi')
  70. ratio = usr_trade_limit if type(usr_trade_limit) is float else float(tl_rounded / out_amt)
  71. direction = 'ABOVE' if ratio > 1 else 'below'
  72. mcolor, lblcolor = (
  73. (redbg, redbg) if (ratio < 0.93 or ratio > 0.999) else
  74. (yelbg, yelbg) if ratio < 0.97 else
  75. (green, grnbg))
  76. trade_limit_disp = f"""
  77. {lblcolor('Trade limit:')} {tl_rounded.hl()} {out_coin} """ + mcolor(
  78. f'({abs(1 - ratio) * 100:0.2f}% {direction} expected amount)')
  79. tx_size_adj = len(e.enc) - 1
  80. else:
  81. trade_limit_disp = ''
  82. tx_size_adj = 0
  83. _amount_in_label = 'Amount in:'
  84. if deduct_est_fee:
  85. if d['gas_rate_units'] == 'satsperbyte':
  86. in_amt -= tx.feespec2abs(d['recommended_gas_rate'] + 's', tx.estimate_size() + tx_size_adj)
  87. out_amt *= (in_amt / self.in_amt)
  88. _amount_in_label = 'Amount in (estimated):'
  89. else:
  90. ymsg('Warning: unknown gas unit ‘{}’, cannot estimate fee'.format(d['gas_rate_units']))
  91. min_in_amt = tx.send_proto.coin_amt(int(d['recommended_min_amount_in']), from_unit='satoshi')
  92. gas_unit = {
  93. 'satsperbyte': 'sat/byte',
  94. }.get(d['gas_rate_units'], d['gas_rate_units'])
  95. elapsed_disp = format_elapsed_hr(d['expiry'], future_msg='from now')
  96. fees = d['fees']
  97. fees_t = tx.recv_proto.coin_amt(int(fees['total']), from_unit='satoshi')
  98. fees_pct_disp = str(fees['total_bps'] / 100) + '%'
  99. slip_pct_disp = str(fees['slippage_bps'] / 100) + '%'
  100. hdr = f'SWAP QUOTE (source: {self.rpc.host})'
  101. return f"""
  102. {cyan(hdr)}
  103. Protocol: {blue(name)}
  104. Direction: {orange(f'{in_coin} => {out_coin}')}
  105. Vault address: {cyan(d['inbound_address'])}
  106. Quote expires: {pink(elapsed_disp)} [{make_timestr(d['expiry'])}]
  107. {_amount_in_label:<22} {in_amt.hl()} {in_coin}
  108. Expected amount out: {out_amt.hl()} {out_coin}{trade_limit_disp}
  109. Rate: {(out_amt / in_amt).hl()} {out_coin}/{in_coin}
  110. Reverse rate: {(in_amt / out_amt).hl()} {in_coin}/{out_coin}
  111. Recommended minimum in amount: {min_in_amt.hl()} {in_coin}
  112. Recommended fee: {pink(d['recommended_gas_rate'])} {pink(gas_unit)}
  113. Fees:
  114. Total: {fees_t.hl()} {out_coin} ({pink(fees_pct_disp)})
  115. Slippage: {pink(slip_pct_disp)}
  116. """
  117. @property
  118. def inbound_address(self):
  119. return self.data['inbound_address']
  120. @property
  121. def rel_fee_hint(self):
  122. if self.data['gas_rate_units'] == 'satsperbyte':
  123. return f'{self.data["recommended_gas_rate"]}s'
  124. def __str__(self):
  125. from pprint import pformat
  126. return pformat(self.data)