Skip to content

Latest commit

 

History

History
557 lines (521 loc) · 18.6 KB

e2ee-multisig.md

File metadata and controls

557 lines (521 loc) · 18.6 KB

E2EE for multi-party multisig in Bitcoin

This document describes how peer-to-peer communication and data exchange for multi-party multisig in Bitcoin can be facilitated using Matrix.

Matrix is an open standard and communication protocol for real-time communication. Matrix supports end-to-end-encryption (E2EE) for one-on-one and group chats, thanks to its Olm and Megolm protocols.

Matrix clients that follow the below schema can construct Bitcoin transactions across different Matrix federations.

The schema is currently used in all Nunchuk applications.

If you have any feedback or questions, please email [email protected] or join our Slack.

API

See https://github.com/nunchuk-io/libnunchuk/blob/main/include/nunchukmatrix.h.

Data structures

Workflows

There are 2 main flows:

  • Wallet creation (or recovery)

    • Initialize wallet creation/recovery session
    • Join wallet
    • Leave/cancel wallet
    • Finalize wallet
  • Transaction creation

    • Initialize transaction
    • Cancel or Sign transaction (repeated until enough signatures have been collected)
    • Broadcast transaction

Matrix notes

  • If the Matrix room is E2EE-enabled, all Bitcoin wallet and transaction events will also be end-to-end-encrypted.
  • m.relates_to is used to establish the relationship graph of events for each workflow.
  • Matrix limits the event size to be <= 64kB (65536 bytes). This works for the majority of PSBTs, but not all.
  • For extremely large PSBTs (such as Bitcoin transactions with lots of inputs and/or outputs), the PSBTs need to be converted to media files and uploaded to the Matrix room.

Schema

1. Wallet creation or recovery

1.1 Init wallet

  • Room ID: !DPZjhLvVqrnlOtrqAy:matrix.org
  • Event ID: $5NCjFf5HI6yVlgBhBHo2_qAqTZWs6MXKJh3QUPc9AsA
  • Sender: @olvaus:nunchuk.io
{
  "room_id": "!DPZjhLvVqrnlOtrqAy:matrix.org",
  "type": "io.nunchuk.wallet",
  "content": {
    "msgtype": "io.nunchuk.wallet.init",
    "v": 1,
    "body": {
        "name": "Wallet Name",
        "description": "Wallet description (optional)",
        "m": 2,
        "n": 3,
        "address_type": "NATIVE_SEGWIT", // LEGACY, NESTED_SEGWIT or NATIVE_SEGWIT (default NATIVE_SEGWIT)
        // address_type could be replaced with script_format (P2WSH, P2WSH-P2SH, P2SH)
        "is_escrow": false,
        "member": [], // Optional key whitelisting (if not empty, join event key must belong to this list)
        "chain": "MAIN", // MAIN, TESTNET or REGTEST (default MAIN)
        "bsms": "BSMS 1.0 ..."
    }
  }
}

1.2 Join wallet

  • Room ID: !DPZjhLvVqrnlOtrqAy:matrix.org
  • Event ID: $maTWq8emJUHXMIWzOpowzgfKp4jBE-i_edNhOwhWKVM
  • Sender: @olvaus:nunchuk.io (@liz:nunchuk.io, @niro:nunchuk.io)
{
  "room_id": "!DPZjhLvVqrnlOtrqAy:matrix.org",
  "type": "io.nunchuk.wallet",
  "content": {
    "msgtype": "io.nunchuk.wallet.join",
    "v": 1,
    "body": {
        "key": "[1cf0bf7e/48'/0'/0'/2']xpub6FL8FhxNNUVnG64YurPd16AfGyvFLhh7S2uSsDqR3Qfcm6o9jtcMYwh6DvmcBF9qozxNQmTCVvWtxLpKTnhVLN3Pgnu2D3pAoXYFgVyd8Yz", // KEY format follow https://github.com/bitcoin/bips/blob/master/bip-0129.mediawiki#signer-1
        // If init_event.content.body.member != [], member must include key
        "type": "HARDWARE", // NFC, HARDWARE, AIRGAP, SOFTWARE or FOREIGN_SOFTWARE
        "io.nunchuk.relates_to": {
            "init_event": {
              "room_id": "!DPZjhLvVqrnlOtrqAy:matrix.org",
              "type": "io.nunchuk.wallet",
              "sender": "@olvaus:nunchuk.io",
              "ts": 1631596260590,
              "event_id": "$5NCjFf5HI6yVlgBhBHo2_qAqTZWs6MXKJh3QUPc9AsA",
              "content": {
                "msgtype": "io.nunchuk.wallet.init",
                "v": 1,
                "body": {
                    "name": "Wallet Name",
                    "description": "Wallet description (optional)",
                    "m": 2,
                    "n": 3,
                    "address_type": "NATIVE_SEGWIT",
                    "is_escrow": false,
                    "member": [],
                    "chain": "MAIN",
                    "bsms": "BSMS 1.0 ..."
                }
              }
            }
        }
    }
  }
}

1.3 Leave wallet

  • Room ID: !DPZjhLvVqrnlOtrqAy:matrix.org
  • Event ID: $maTWq8emJUHXMIWzOpowzgfKp4jBE-i_edNhOwhWKVM
  • Sender: @olvaus:nunchuk.io (@liz:nunchuk.io, @niro:nunchuk.io)
{
  "room_id": "!DPZjhLvVqrnlOtrqAy:matrix.org",
  "type": "io.nunchuk.wallet",
  "content": {
    "msgtype": "io.nunchuk.wallet.leave",
    "v": 1,
    "body": {
        "reason": "",
        "io.nunchuk.relates_to": {
            "init_event": {
              "room_id": "!DPZjhLvVqrnlOtrqAy:matrix.org",
              "type": "io.nunchuk.wallet",
              "sender": "@olvaus:nunchuk.io",
              "ts": 1631596260590,
              "event_id": "$5NCjFf5HI6yVlgBhBHo2_qAqTZWs6MXKJh3QUPc9AsA",
              "content": {
                "msgtype": "io.nunchuk.wallet.init",
                "v": 1,
                "body": {
                    "name": "Wallet Name",
                    "description": "Wallet description (optional)",
                    "m": 2,
                    "n": 3,
                    "address_type": "NATIVE_SEGWIT",
                    "is_escrow": false,
                    "member": [],
                    "chain": "MAIN",
                    "bsms": "BSMS 1.0 ..."
                }
              }
            },
            "join_event_id": "$maTWq8emJUHXMIWzOpowzgfKp4jBE-i_edNhOwhWKVM"
        }
    }
  }
}

1.4 Wallet ready

  • Room ID: !DPZjhLvVqrnlOtrqAy:matrix.org
  • Event ID: $maTWq8emJUHXMIWzOpowzgfKp4jBE-i_edNhOwhWKVM
  • Sender: @olvaus:nunchuk.io
{
  "room_id": "!DPZjhLvVqrnlOtrqAy:matrix.org",
  "type": "io.nunchuk.wallet",
  "content": {
    "msgtype": "io.nunchuk.wallet.ready",
    "v": 1,
    "body": {
        "io.nunchuk.relates_to": {
            "init_event": {
              "room_id": "!DPZjhLvVqrnlOtrqAy:matrix.org",
              "type": "io.nunchuk.wallet",
              "sender": "@olvaus:nunchuk.io",
              "ts": 1631596260590,
              "event_id": "$5NCjFf5HI6yVlgBhBHo2_qAqTZWs6MXKJh3QUPc9AsA",
              "content": {
                "msgtype": "io.nunchuk.wallet.init",
                "v": 1,
                "body": {
                    "name": "Wallet Name",
                    "description": "Wallet description (optional)",
                    "m": 2,
                    "n": 3,
                    "address_type": "NATIVE_SEGWIT",
                    "is_escrow": false,
                    "member": [],
                    "chain": "MAIN",
                    "bsms": "BSMS 1.0 ..."
                }
              }
            },
            "join_ids": [
                "$maTWq8emJUHXMIWzOpowzgfKp4jBE-i_edNhOwhWKVM",
                "liz:nunchuk.io_join_event_id",
                "niro:nunchuk.io_join_event_id"
            ]
        }
    }
  }
}

1.5 Finalize wallet

  • Room ID: !DPZjhLvVqrnlOtrqAy:matrix.org
  • Event ID: $maTWq8emJUHXMIWzOpowzgfKp4jBE-i_edNhOwhWKVM
  • Sender: @olvaus:nunchuk.io
{
  "room_id": "!DPZjhLvVqrnlOtrqAy:matrix.org",
  "type": "io.nunchuk.wallet",
  "content": {
    "msgtype": "io.nunchuk.wallet.create",
    "v": 1,
    "body": {
        "descriptor": "wsh(sortedmulti(1,[59865f44/48'/0'/0'/2']026d15412460ba0d881c21837bb999233896085a9ed4e5445bd637c10e579768ba,[b7044ca6/48'/0'/0'/2']030baf0497ab406ff50cb48b4013abac8a0338758d2fd54cd934927afa57cc2062))#rzx9dffd",
        // (?) maybe support descriptor_template with path_restriction later
        "first_address": "bc1quqy523xu3l8che3s8vja8n33qtg0uyugr9l5z092s3wa50p8t7rqy6zumf",
        "io.nunchuk.relates_to": {
            "init_event": {
              "room_id": "!DPZjhLvVqrnlOtrqAy:matrix.org",
              "type": "io.nunchuk.wallet",
              "sender": "@olvaus:nunchuk.io",
              "ts": 1631596260590,
              "event_id": "$5NCjFf5HI6yVlgBhBHo2_qAqTZWs6MXKJh3QUPc9AsA",
              "content": {
                "msgtype": "io.nunchuk.wallet.init",
                "v": 1,
                "body": {
                    "name": "Wallet Name",
                    "description": "Wallet description (optional)",
                    "m": 2,
                    "n": 3,
                    "address_type": "NATIVE_SEGWIT",
                    "is_escrow": false,
                    "member": [],
                    "chain": "MAIN",
                    "bsms": "BSMS 1.0 ..."
                }
              }
            },
            "join_ids": [
                "$maTWq8emJUHXMIWzOpowzgfKp4jBE-i_edNhOwhWKVM",
                "liz:nunchuk.io_join_event_id",
                "niro:nunchuk.io_join_event_id"
            ]
        }
    }
  }
}

1.6 Cancel wallet

  • Room ID: !DPZjhLvVqrnlOtrqAy:matrix.org
  • Event ID: $maTWq8emJUHXMIWzOpowzgfKp4jBE-i_edNhOwhWKVM
  • Sender: @olvaus:nunchuk.io
{
  "room_id": "!DPZjhLvVqrnlOtrqAy:matrix.org",
  "type": "io.nunchuk.wallet",
  "content": {
    "msgtype": "io.nunchuk.wallet.cancel",
    "v": 1,
    "body": {
        "reason": "",
        "io.nunchuk.relates_to": {
            "init_event": {
              "room_id": "!DPZjhLvVqrnlOtrqAy:matrix.org",
              "type": "io.nunchuk.wallet",
              "sender": "@olvaus:nunchuk.io",
              "ts": 1631596260590,
              "event_id": "$5NCjFf5HI6yVlgBhBHo2_qAqTZWs6MXKJh3QUPc9AsA",
              "content": {
                "msgtype": "io.nunchuk.wallet.init",
                "v": 1,
                "body": {
                    "name": "Wallet Name",
                    "description": "Wallet description (optional)",
                    "m": 2,
                    "n": 3,
                    "address_type": "NATIVE_SEGWIT",
                    "is_escrow": false,
                    "member": [],
                    "chain": "MAIN",
                    "bsms": "BSMS 1.0 ..."
                }
              }
            }
        }
    }
  }
}

1.7 Delete wallet

  • Room ID: !DPZjhLvVqrnlOtrqAy:matrix.org
  • Event ID: $maTWq8emJUHXMIWzOpowzgfKp4jBE-i_edNhOwhWKVM
  • Sender: @olvaus:nunchuk.io
{
  "room_id": "!DPZjhLvVqrnlOtrqAy:matrix.org",
  "type": "io.nunchuk.wallet",
  "content": {
    "msgtype": "io.nunchuk.wallet.delete",
    "v": 1,
    "body": {
        "wallet_id": "5pu4agtf",
        "io.nunchuk.relates_to": {
            "init_event": {
              "room_id": "!DPZjhLvVqrnlOtrqAy:matrix.org",
              "type": "io.nunchuk.wallet",
              "sender": "@olvaus:nunchuk.io",
              "ts": 1631596260590,
              "event_id": "$5NCjFf5HI6yVlgBhBHo2_qAqTZWs6MXKJh3QUPc9AsA",
              "content": {
                "msgtype": "io.nunchuk.wallet.init",
                "v": 1,
                "body": {
                    "name": "Wallet Name",
                    "description": "Wallet description (optional)",
                    "m": 2,
                    "n": 3,
                    "address_type": "NATIVE_SEGWIT",
                    "is_escrow": false,
                    "member": [],
                    "chain": "MAIN",
                    "bsms": "BSMS 1.0 ..."
                }
              }
            }
        }
    }
  }
}

2. Transaction creation

2.1 Init transaction

  • Room ID: !DPZjhLvVqrnlOtrqAy:matrix.org
  • Event ID: $5NCjFf5HI6yVlgBhBHo2_qAqTZWs6MXKJh3QUPc9AsA
  • Sender: @olvaus:nunchuk.io
{
  "room_id": "!DPZjhLvVqrnlOtrqAy:matrix.org",
  "type": "io.nunchuk.transaction",
  "content": {
    "msgtype": "io.nunchuk.transaction.init",
    "v": 1,
    "body": {
        "wallet_id": "",
        "memo": "Transaction memo (optional)",
        "psbt": "unsigned PSBT encoded in base64",
        "fee_rate": 1,
        "subtract_fee_from_amount": false,
        "chain": "MAIN", // MAIN, TESTNET or REGTEST (default is MAIN)
    }
  }
}

2.2 Sign transaction

  • Room ID: !DPZjhLvVqrnlOtrqAy:matrix.org
  • Event ID: $maTWq8emJUHXMIWzOpowzgfKp4jBE-i_edNhOwhWKVM
  • Sender: @olvaus:nunchuk.io (@liz:nunchuk.io, @niro:nunchuk.io)
{
  "room_id": "!DPZjhLvVqrnlOtrqAy:matrix.org",
  "type": "io.nunchuk.transaction",
  "content": {
    "msgtype": "io.nunchuk.transaction.sign",
    "v": 1,
    "body": {
        "psbt": "signed PSBT encoded in base64",
        "master_fingerprint": "bc4e023d",
        "io.nunchuk.relates_to": {
            "init_event": {
              "room_id": "!DPZjhLvVqrnlOtrqAy:matrix.org",
              "type": "io.nunchuk.transaction",
              "sender": "@olvaus:nunchuk.io",
              "ts": 1631596260590,
              "event_id": "$5NCjFf5HI6yVlgBhBHo2_qAqTZWs6MXKJh3QUPc9AsA",
              "content": {
                "msgtype": "io.nunchuk.transaction.init",
                "v": 1,
                "body": {
                    "wallet_id": "",
                    "memo": "Transaction memo (optional)",
                    "psbt": "unsigned PSBT encoded in base64",
                    "fee_rate": 1,
                    "subtract_fee_from_amount": false,
                    "chain": "MAIN"
                }
              }
            }
        }
    }
  }
}

2.3 Transaction ready

  • Room ID: !DPZjhLvVqrnlOtrqAy:matrix.org
  • Event ID: $maTWq8emJUHXMIWzOpowzgfKp4jBE-i_edNhOwhWKVM
  • Sender: @olvaus:nunchuk.io
{
  "room_id": "!DPZjhLvVqrnlOtrqAy:matrix.org",
  "type": "io.nunchuk.transaction",
  "content": {
    "msgtype": "io.nunchuk.transaction.ready",
    "v": 1,
    "body": {
        "io.nunchuk.relates_to": {
            "init_event": {
              "room_id": "!DPZjhLvVqrnlOtrqAy:matrix.org",
              "type": "io.nunchuk.transaction",
              "sender": "@olvaus:nunchuk.io",
              "ts": 1631596260590,
              "event_id": "$5NCjFf5HI6yVlgBhBHo2_qAqTZWs6MXKJh3QUPc9AsA",
              "content": {
                "msgtype": "io.nunchuk.transaction.init",
                "v": 1,
                "body": {
                    "wallet_id": "",
                    "memo": "Transaction memo (optional)",
                    "psbt": "unsigned PSBT encoded in base64",
                    "fee_rate": 1,
                    "subtract_fee_from_amount": false,
                    "chain": "MAIN"
                }
              }
            },
            "sign_ids": [
                "$maTWq8emJUHXMIWzOpowzgfKp4jBE-i_edNhOwhWKVM",
                "liz:nunchuk.io_sign_event_id",
                "niro:nunchuk.io_sign_event_id"
            ]
        }
    }
  }
}

2.4 Broadcast transaction

  • Room ID: !DPZjhLvVqrnlOtrqAy:matrix.org
  • Event ID: $maTWq8emJUHXMIWzOpowzgfKp4jBE-i_edNhOwhWKVM
  • Sender: @olvaus:nunchuk.io
{
  "room_id": "!DPZjhLvVqrnlOtrqAy:matrix.org",
  "type": "io.nunchuk.transaction",
  "content": {
    "msgtype": "io.nunchuk.transaction.broadcast",
    "v": 1,
    "body": {
        "tx_id": "",
        "io.nunchuk.relates_to": {
            "init_event": {
              "room_id": "!DPZjhLvVqrnlOtrqAy:matrix.org",
              "type": "io.nunchuk.transaction",
              "sender": "@olvaus:nunchuk.io",
              "ts": 1631596260590,
              "event_id": "$5NCjFf5HI6yVlgBhBHo2_qAqTZWs6MXKJh3QUPc9AsA",
              "content": {
                "msgtype": "io.nunchuk.transaction.init",
                "v": 1,
                "body": {
                    "wallet_id": "",
                    "memo": "Transaction memo (optional)",
                    "psbt": "unsigned PSBT encoded in base64",
                    "fee_rate": 1,
                    "subtract_fee_from_amount": false,
                    "chain": "MAIN"
                }
              }
            },
            "sign_ids": [
                "$maTWq8emJUHXMIWzOpowzgfKp4jBE-i_edNhOwhWKVM",
                "liz:nunchuk.io_sign_event_id",
                "niro:nunchuk.io_sign_event_id"
            ]
        }
    }
  }
}

2.5 Cancel transaction

  • Room ID: !DPZjhLvVqrnlOtrqAy:matrix.org
  • Event ID: $maTWq8emJUHXMIWzOpowzgfKp4jBE-i_edNhOwhWKVM
  • Sender: @olvaus:nunchuk.io
{
  "room_id": "!DPZjhLvVqrnlOtrqAy:matrix.org",
  "type": "io.nunchuk.transaction",
  "content": {
    "msgtype": "io.nunchuk.transaction.cancel",
    "v": 1,
    "body": {
        "reason": "",
        "io.nunchuk.relates_to": {
            "init_event": {
              "room_id": "!DPZjhLvVqrnlOtrqAy:matrix.org",
              "type": "io.nunchuk.transaction",
              "sender": "@olvaus:nunchuk.io",
              "ts": 1631596260590,
              "event_id": "$5NCjFf5HI6yVlgBhBHo2_qAqTZWs6MXKJh3QUPc9AsA",
              "content": {
                "msgtype": "io.nunchuk.transaction.init",
                "v": 1,
                "body": {
                    "wallet_id": "",
                    "memo": "Transaction memo (optional)",
                    "psbt": "unsigned PSBT encoded in base64",
                    "fee_rate": 1,
                    "subtract_fee_from_amount": false,
                    "chain": "MAIN"
                }
              }
            }
        }
    }
  }
}