123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130 |
- #!/usr/bin/env python3
- #
- # MMGen Wallet, a terminal-based cryptocurrency wallet
- # Copyright (C)2013-2025 The MMGen Project <mmgen@tuta.io>
- # Licensed under the GNU General Public License, Version 3:
- # https://www.gnu.org/licenses
- # Public project repositories:
- # https://github.com/mmgen/mmgen-wallet
- # https://gitlab.com/mmgen/mmgen-wallet
- """
- proto.eth.tx.unsigned: Ethereum unsigned transaction class
- """
- import json
- from ....tx import unsigned as TxBase
- from ....util import msg, msg_r, die
- from ....obj import CoinTxID, ETHNonce, Int, HexStr
- from ....addr import CoinAddr, TokenAddr
- from ..contract import Token
- from .completed import Completed, TokenCompleted
- class Unsigned(Completed, TxBase.Unsigned):
- desc = 'unsigned transaction'
- def parse_txfile_serialized_data(self):
- d = json.loads(self.serialized)
- o = {
- 'from': CoinAddr(self.proto, d['from']),
- # NB: for token, 'to' is sendto address
- 'to': CoinAddr(self.proto, d['to']) if d['to'] else None,
- 'amt': self.proto.coin_amt(d['amt']),
- 'gasPrice': self.proto.coin_amt(d['gasPrice']),
- 'startGas': self.proto.coin_amt(d['startGas']),
- 'nonce': ETHNonce(d['nonce']),
- 'chainId': None if d['chainId'] == 'None' else Int(d['chainId']),
- 'data': HexStr(d['data'])}
- self.gas = o['startGas']
- self.txobj = o
- return d # 'token_addr', 'decimals' required by Token subclass
- async def do_sign(self, o, wif):
- o_conv = {
- 'to': bytes.fromhex(o['to'] or ''),
- 'startgas': o['startGas'].toWei(),
- 'gasprice': o['gasPrice'].toWei(),
- 'value': o['amt'].toWei() if o['amt'] else 0,
- 'nonce': o['nonce'],
- 'data': bytes.fromhex(o['data'])}
- from ..pyethereum.transactions import Transaction
- etx = Transaction(**o_conv).sign(wif, o['chainId'])
- assert etx.sender.hex() == o['from'], (
- 'Sender address recovered from signature does not match true sender')
- from .. import rlp
- self.serialized = rlp.encode(etx).hex()
- self.coin_txid = CoinTxID(etx.hash.hex())
- if o['data']: # contract-creating transaction
- if o['to']:
- if not self.is_swap:
- raise ValueError('contract-creating transaction cannot have to-address')
- else:
- self.txobj['token_addr'] = TokenAddr(self.proto, etx.creates.hex())
- async def sign(self, tx_num_str, keys): # return TX object or False; don't exit or raise exception
- from ....exception import TransactionChainMismatch
- try:
- self.check_correct_chain()
- except TransactionChainMismatch:
- return False
- o = self.txobj
- def do_mismatch_err(io, j, k, desc):
- m = 'A compromised online installation may have altered your serialized data!'
- fs = '\n{} mismatch!\n{}\n orig: {}\n serialized: {}'
- die(3, fs.format(desc.upper(), m, getattr(io[0], k), o[j]))
- if o['from'] != self.inputs[0].addr:
- do_mismatch_err(self.inputs, 'from', 'addr', 'from-address')
- if self.outputs:
- if o['to'] != self.outputs[0].addr:
- do_mismatch_err(self.outputs, 'to', 'addr', 'to-address')
- if o['amt'] != self.outputs[0].amt:
- do_mismatch_err(self.outputs, 'amt', 'amt', 'amount')
- msg_r(f'Signing transaction{tx_num_str}...')
- try:
- await self.do_sign(o, keys[0].sec.wif)
- msg('OK')
- from ....tx import SignedTX
- return await SignedTX(cfg=self.cfg, data=self.__dict__, automount=self.automount)
- except Exception as e:
- msg(f'{e}: transaction signing failed!')
- return False
- class TokenUnsigned(TokenCompleted, Unsigned):
- desc = 'unsigned transaction'
- def parse_txfile_serialized_data(self):
- d = Unsigned.parse_txfile_serialized_data(self)
- o = self.txobj
- o['token_addr'] = TokenAddr(self.proto, d['token_addr'])
- o['decimals'] = Int(d['decimals'])
- t = Token(self.cfg, self.proto, o['token_addr'], o['decimals'])
- o['data'] = t.create_data(o['to'], o['amt'])
- o['token_to'] = t.transferdata2sendaddr(o['data'])
- async def do_sign(self, o, wif):
- t = Token(self.cfg, self.proto, o['token_addr'], o['decimals'])
- tx_in = t.make_tx_in(
- to_addr = o['to'],
- amt = o['amt'],
- gas = self.gas,
- gasPrice = o['gasPrice'],
- nonce = o['nonce'])
- res = await t.txsign(tx_in, wif, o['from'], chain_id=o['chainId'])
- self.serialized = res.txhex
- self.coin_txid = res.txid
- class AutomountUnsigned(TxBase.AutomountUnsigned, Unsigned):
- pass
- class TokenAutomountUnsigned(TxBase.AutomountUnsigned, TokenUnsigned):
- pass
|