Atomic swaps
Or, trading apples for bananas.
The Problem
Andy has some apples. Barb has some bananas. Andy agrees to give Barb two apples in exchange for one banana. In order to make sure both parties hold up their end of the deal they need to create a single transaction that sends assets both directions.
The Inputs
We'll need to find 3 UTXOs to use as inputs. Andy's apples, Barb's bananas and some AOK to pay the network fee (Andy will pay).
Andy uses listunspent
to find a suitable txid
and vout
:
{
"txid": "73cff8d5595d9134841e252aa95a01a408cc1473cc8ef590cdd81e819e62fccc",
"vout": 1,
"address": "Kj6iUJQvVZrd4wxeijebcivC6a4Y53qVgR",
"scriptPubKey": "21027e1fde02d2cbfac3629aeaf669abd156d0c4dfbf52f6a5e4dd6664e81a621045ac",
"amount": 1,
"confirmations": 3088,
"spendable": true,
"solvable": true,
"safe": true
}
Then each party uses listmyassets
with verbose=true
to find asset UTXOs:
Andy:
listmytokens APPLES true
:
{
"APPLES": {
"balance": 1000,
"outpoints": [
{
"txid": "744c61abe6d237939c3567bc44b912ff4c375984229908c964be44f36dec79e3",
"vout": 3,
"amount": 1000
}
]
}
}
Barb:
listmytokens BANANAS true
:
{
"BANANAS": {
"balance": 1000,
"outpoints": [
{
"txid": "8468ef5193b1f5d6a7bd501f5f8ef5aec7c3d86fa87cec5a0b6f6d86fba78a4f",
"vout": 3,
"amount": 1000
}
]
}
}
Extracting the txids and vouts gives us our raw inputs:
'[{"txid":"73cff8d5595d9134841e252aa95a01a408cc1473cc8ef590cdd81e819e62fccc","vout":1}, \
{"txid":"744c61abe6d237939c3567bc44b912ff4c375984229908c964be44f36dec79e3","vout":3}, \
{"txid":"8468ef5193b1f5d6a7bd501f5f8ef5aec7c3d86fa87cec5a0b6f6d86fba78a4f","vout":3}]'
The Outputs
We'll be using 5 new AOK addresses:
Andy's Raven change address:
KZzymtqctNdW4fWhLdWKRYUDGXP3KzG8Y9
Andy's banana receive address:
KiNMHYArytp2D6jQSPHrGd7rq3qAxvpdt5
Barb's banana change address:
KdahQMGWTJgFjziWadqejMU7gYAvpumzkk
Barb's apple receive address:
KtPia5TPzjw1XibYh3de3RKxcpq9gusCBu
Andy's apple change address:
KeW8oMzwyQZbgME9Tm633TDRV8bW347tEM
All asset transfers have to be balanced. Since we have 1000 APPLES coming in, we have to have 1000 going out. So Andy will send 2 to Barb's receive address and the rest (998) to his change address. The AOK will pay 0.001 for the network fee as normal.
'{"KZzymtqctNdW4fWhLdWKRYUDGXP3KzG8Y9":4.8827125, \
"KiNMHYArytp2D6jQSPHrGd7rq3qAxvpdt5":{"transfer":{"BANANAS":1}}, \
"KdahQMGWTJgFjziWadqejMU7gYAvpumzkk":{"transfer":{"BANANAS":999}}, \
"KtPia5TPzjw1XibYh3de3RKxcpq9gusCBu":{"transfer":{"APPLES":2}}, \
"KeW8oMzwyQZbgME9Tm633TDRV8bW347tEM":{"transfer":{"APPLES":998}}}'
Creating The Transaction
Andy can now use createrawtransaction
, passing in the inputs and outputs and receiving the transaction hex:
createrawtransaction '[{"txid":"631cf1566165803f0b89fbfb169d8f0c89129ec3f8536a48e4c4f0f3c4081cff","vout":0},{"txid":"744c61abe6d237939c3567bc44b912ff4c375984229908c964be44f36dec79e3","vout":3},{"txid":"8468ef5193b1f5d6a7bd501f5f8ef5aec7c3d86fa87cec5a0b6f6d86fba78a4f","vout":3}]' '{"KZzymtqctNdW4fWhLdWKRYUDGXP3KzG8Y9":4.8827125,"KiNMHYArytp2D6jQSPHrGd7rq3qAxvpdt5":{"transfer":{"BANANAS":1}},"KdahQMGWTJgFjziWadqejMU7gYAvpumzkk":{"transfer":{"BANANAS":999}},"KtPia5TPzjw1XibYh3de3RKxcpq9gusCBu":{"transfer":{"APPLES":2}},"KeW8oMzwyQZbgME9Tm633TDRV8bW347tEM":{"transfer":{"APPLES":998}}}'
0200000003ff1c08c4f3f0c4e4486a53f8c39e12890c8f9d16fbfb890b3f80656156f11c630000000000ffffffffe379ec6df344be64c90899228459374cff12b944bc67359c9337d2e6ab614c740300000000ffffffff4f8aa7fb866d6f0b5aec7ca86fd8c3c7aef58e5f1f50bda7d6f5b19351ef68840300000000ffffffff05926d1a1d000000001976a914a1d62b4f8a6710ad43d56f656c27966513ae8fbf88ac00000000000000003076a91483b7a59ce25d8159fdc2ab70320512d2bc07a1c488acc01472766e740742414e414e415300e1f505000000007500000000000000003076a91430f446c2ac37dc072516603e44e86aea7af2bf6788acc01472766e740742414e414e415300078142170000007500000000000000002f76a914cf084a93cf87d3030b74a1939d9c8f02f6152c8c88acc01372766e74064150504c455300c2eb0b000000007500000000000000002f76a914d18967e005ff7208cc6a9c483b028c7c72ca2d6e88acc01372766e74064150504c455300268b3c170000007500000000
Signing The Transaction
Ok, the structure of the transaction is set. Now each party needs to sign it in order to unlock the inputs. Here are the steps:
- Andy signs it using
signrawtransaction
. This will alter the hex, using his wallet to insert the signatures.
Note: Since Andy doesn't have the keys for all of the inputs he will see an error:
"Unable to sign input, invalid stack size (possibly missing key)" This is normal and expected.
- Andy sends the signed hex to Barb.
- Barb uses
signrawtransaction
to sign the rest of the inputs. Again capture the hex output.
Submit The Transaction
Almost there. Barb now passes the fully signed hex to sendrawtransaction
! It will be communicated to the network and put into the next block.