Lock unspents funding lock tx.

This commit is contained in:
tecnovert 2020-11-15 23:31:59 +02:00
parent 18a5322f10
commit eb6bd444c4
No known key found for this signature in database
GPG key ID: 8ED6D8750C4E3F93
3 changed files with 67 additions and 23 deletions

View file

@ -98,6 +98,7 @@ class MessageTypes(IntEnum):
XMR_BID_SPLIT = auto() XMR_BID_SPLIT = auto()
XMR_BID_ACCEPT = auto() XMR_BID_ACCEPT = auto()
XMR_BID_TXN_SIGS_FL = auto() XMR_BID_TXN_SIGS_FL = auto()
XMR_BID_LOCK_REFUND_SPEND_TX_LF = auto()
class SwapTypes(IntEnum): class SwapTypes(IntEnum):
@ -1073,6 +1074,8 @@ class BasicSwap(BaseApp):
session.add(bid.initiate_tx) session.add(bid.initiate_tx)
if bid.participate_tx: if bid.participate_tx:
session.add(bid.participate_tx) session.add(bid.participate_tx)
if bid.xmr_a_lock_tx:
session.add(bid.xmr_a_lock_tx)
if xmr_swap is not None: if xmr_swap is not None:
session.add(xmr_swap) session.add(xmr_swap)
@ -2454,15 +2457,19 @@ class BasicSwap(BaseApp):
q = session.query(EventQueue).filter(EventQueue.trigger_at >= now) q = session.query(EventQueue).filter(EventQueue.trigger_at >= now)
for row in q: for row in q:
if row.event_type == EventTypes.ACCEPT_BID: try:
self.acceptBid(row.linked_id) if row.event_type == EventTypes.ACCEPT_BID:
elif row.event_type == EventTypes.SIGN_XMR_SWAP_LOCK_TX_A: self.acceptBid(row.linked_id)
self.sendXmrBidTxnSigsFtoL(row.linked_id, session) elif row.event_type == EventTypes.SIGN_XMR_SWAP_LOCK_TX_A:
elif row.event_type == EventTypes.SEND_XMR_SWAP_LOCK_TX_A: self.sendXmrBidTxnSigsFtoL(row.linked_id, session)
self.sendXmrBidCoinALockTx(row.linked_id, session) elif row.event_type == EventTypes.SEND_XMR_SWAP_LOCK_TX_A:
else: self.sendXmrBidCoinALockTx(row.linked_id, session)
self.log.warning('Unknown event type: %d', row.event_type) else:
self.log.warning('Unknown event type: %d', row.event_type)
except Exception as ex:
if self.debug:
traceback.print_exc()
self.log.error('checkEvents failed: {}'.format(str(ex)))
session.delete(row) session.delete(row)
session.commit() session.commit()
@ -2802,6 +2809,7 @@ class BasicSwap(BaseApp):
self.saveBidInSession(bid.bid_id, bid, session, xmr_swap) self.saveBidInSession(bid.bid_id, bid, session, xmr_swap)
def receiveXmrBidAccept(self, bid, session): def receiveXmrBidAccept(self, bid, session):
# Follower receiving MSG1F and MSG2F
self.log.debug('Receiving xmr bid accept %s', bid.bid_id.hex()) self.log.debug('Receiving xmr bid accept %s', bid.bid_id.hex())
now = int(time.time()) now = int(time.time())
@ -3017,14 +3025,12 @@ class BasicSwap(BaseApp):
# TODO: set msg_valid based on bid / offer parameters # TODO: set msg_valid based on bid / offer parameters
msg_valid = self.SMSG_SECONDS_IN_HOUR * 48 msg_valid = self.SMSG_SECONDS_IN_HOUR * 48
ro = self.callrpc('smsgsend', [bid.bid_addr, offer.addr_from, payload_hex, False, msg_valid, False, options]) ro = self.callrpc('smsgsend', [bid.bid_addr, offer.addr_from, payload_hex, False, msg_valid, False, options])
xmr_swap.coin_a_lock_tx_sigs_l_id = bytes.fromhex(ro['msgid']) xmr_swap.coin_a_lock_tx_sigs_l_msg_id = bytes.fromhex(ro['msgid'])
self.log.info('Sent XMR_BID_TXN_SIGS_FL %s', xmr_swap.coin_a_lock_tx_sigs_l_id.hex()) self.log.info('Sent XMR_BID_TXN_SIGS_FL %s', xmr_swap.coin_a_lock_tx_sigs_l_msg_id.hex())
bid.setState(BidStates.BID_ACCEPTED) bid.setState(BidStates.BID_ACCEPTED)
session.add(bid) self.saveBidInSession(bid_id, bid, session, xmr_swap)
session.add(xmr_swap)
self.swaps_in_progress[bid_id] = (bid, offer) self.swaps_in_progress[bid_id] = (bid, offer)
except Exception as ex: except Exception as ex:
if self.debug: if self.debug:
@ -3055,11 +3061,40 @@ class BasicSwap(BaseApp):
xmr_swap.al_lock_spend_tx_esig = ci_from.signTxOtVES(kal, xmr_swap.pkasf, xmr_swap.a_lock_spend_tx, 0, xmr_swap.a_lock_tx_script, bid.amount) # self.a_swap_value xmr_swap.al_lock_spend_tx_esig = ci_from.signTxOtVES(kal, xmr_swap.pkasf, xmr_swap.a_lock_spend_tx, 0, xmr_swap.a_lock_tx_script, bid.amount) # self.a_swap_value
msg_buf = XmrBidLockSpendTxMessage(
bid_msg_id=bid_id,
a_lock_spend_tx=xmr_swap.a_lock_spend_tx,
al_lock_spend_tx_esig=xmr_swap.al_lock_spend_tx_esig)
mag_bytes = msg_buf.SerializeToString()
payload_hex = str.format('{:02x}', MessageTypes.XMR_BID_LOCK_REFUND_SPEND_TX_LF) + mag_bytes.hex()
options = {'decodehex': True, 'ttl_is_seconds': True}
# TODO: set msg_valid based on bid / offer parameters
msg_valid = self.SMSG_SECONDS_IN_HOUR * 48
ro = self.callrpc('smsgsend', [offer.addr_from, bid.bid_addr, payload_hex, False, msg_valid, False, options])
xmr_swap.coin_a_lock_refund_spend_tx_msg_id = bytes.fromhex(ro['msgid'])
# TODO: Separate MSG4F and txn sending # TODO: Separate MSG4F and txn sending
# publishalocktx
lock_tx_signed = ci_from.signTxWithWallet(xmr_swap.a_lock_tx)
txid_hex = ci_from.publishTx(lock_tx_signed)
self.log.debug('Submitted lock txn %s to %s chain for bid %s', txid_hex, chainparams[coin_from]['name'], bid_id.hex())
bid.xmr_a_lock_tx = SwapTx(
bid_id=bid_id,
tx_type=TxTypes.XMR_SWAP_A_LOCK,
txid=bytes.fromhex(txid_hex),
)
bid.xmr_a_lock_tx.setState(TxStates.TX_SENT)
bid.setState(BidStates.BID_ACCEPTED)
self.saveBidInSession(bid_id, bid, session, xmr_swap)
self.swaps_in_progress[bid_id] = (bid, offer)
def processXmrBidCoinALockSigs(self, msg): def processXmrBidCoinALockSigs(self, msg):
# Follower processing MSG3L # Leader processing MSG3L
self.log.debug('Processing xmr coin a follower lock sigs msg %s', msg['msgid']) self.log.debug('Processing xmr coin a follower lock sigs msg %s', msg['msgid'])
now = int(time.time()) now = int(time.time())
msg_bytes = bytes.fromhex(msg['hex'][2:-2]) msg_bytes = bytes.fromhex(msg['hex'][2:-2])

View file

@ -107,6 +107,7 @@ class Bid(Base):
initiate_tx = None initiate_tx = None
participate_tx = None participate_tx = None
xmr_a_lock_tx = None
def getITxState(self): def getITxState(self):
if self.initiate_tx is None: if self.initiate_tx is None:
@ -115,8 +116,7 @@ class Bid(Base):
def setITxState(self, new_state): def setITxState(self, new_state):
if self.initiate_tx is not None: if self.initiate_tx is not None:
self.initiate_tx.state = new_state self.initiate_tx.setState(new_state)
self.initiate_tx.states = (self.initiate_tx.states if self.initiate_tx.states is not None else bytes()) + struct.pack('<iq', new_state, int(time.time()))
def getPTxState(self): def getPTxState(self):
if self.participate_tx is None: if self.participate_tx is None:
@ -125,8 +125,7 @@ class Bid(Base):
def setPTxState(self, new_state): def setPTxState(self, new_state):
if self.participate_tx is not None: if self.participate_tx is not None:
self.participate_tx.state = new_state self.participate_tx.setState(new_state)
self.participate_tx.states = (self.participate_tx.states if self.participate_tx.states is not None else bytes()) + struct.pack('<iq', new_state, int(time.time()))
def setState(self, new_state, state_note=None): def setState(self, new_state, state_note=None):
now = int(time.time()) now = int(time.time())
@ -166,6 +165,10 @@ class SwapTx(Base):
state = sa.Column(sa.Integer) state = sa.Column(sa.Integer)
states = sa.Column(sa.LargeBinary) # Packed states and times states = sa.Column(sa.LargeBinary) # Packed states and times
def setState(self, new_state):
self.state = new_state
self.states = (self.states if self.states is not None else bytes()) + struct.pack('<iq', new_state, int(time.time()))
class PooledAddress(Base): class PooledAddress(Base):
__tablename__ = 'addresspool' __tablename__ = 'addresspool'
@ -228,7 +231,8 @@ class XmrSwap(Base):
bid_accept_msg_id2 = sa.Column(sa.LargeBinary) bid_accept_msg_id2 = sa.Column(sa.LargeBinary)
bid_accept_msg_id3 = sa.Column(sa.LargeBinary) bid_accept_msg_id3 = sa.Column(sa.LargeBinary)
coin_a_lock_tx_sigs_l_id = sa.Column(sa.LargeBinary) # MSG3L F -> L coin_a_lock_tx_sigs_l_msg_id = sa.Column(sa.LargeBinary) # MSG3L F -> L
coin_a_lock_refund_spend_tx_msg_id = sa.Column(sa.LargeBinary) # MSG4F L -> F
contract_count = sa.Column(sa.Integer) contract_count = sa.Column(sa.Integer)

View file

@ -681,17 +681,22 @@ class BTCInterface(CoinInterface):
def fundTx(self, tx, feerate): def fundTx(self, tx, feerate):
feerate_str = format_amount(feerate, self.exp()) feerate_str = format_amount(feerate, self.exp())
rv = self.rpc_callback('fundrawtransaction', [tx.hex(), {'feeRate': feerate_str}]) # TODO: unlock unspents if bid cancelled
options = {
'lockUnspents': True,
'feeRate': feerate_str,
}
rv = self.rpc_callback('fundrawtransaction', [tx.hex(), options])
return bytes.fromhex(rv['hex']) return bytes.fromhex(rv['hex'])
def signTxWithWallet(self, tx): def signTxWithWallet(self, tx):
rv = self.rpc_callback('signrawtransactionwithwallet', [ToHex(tx)]) rv = self.rpc_callback('signrawtransactionwithwallet', [tx.hex()])
#return FromHex(tx, rv['hex']) #return FromHex(tx, rv['hex'])
return bytes.fromhex(rv['hex']) return bytes.fromhex(rv['hex'])
def publishTx(self, tx): def publishTx(self, tx):
return self.rpc_callback('sendrawtransaction', [ToHex(tx)]) return self.rpc_callback('sendrawtransaction', [tx.hex()])
def encodeTx(self, tx): def encodeTx(self, tx):
return tx.serialize() return tx.serialize()