Skip to main content

Client for Ledger Nano Bitcoin application

Project description

Ledger Bitcoin application client

Overview

Client library for Ledger Bitcoin application.

Main repository and documentation: https://github.com/LedgerHQ/app-bitcoin-new

Install

If you just want to communicate through TCP socket (for example with the Speculos emulator), there is no dependency:

$ pip install ledger_bitcoin

otherwise, hidapi must be installed as an extra dependency:

$ pip install ledger_bitcoin[hid]

Getting started

The main method exported by the library is createClient, which queries the hardware wallet for the version of the running app, and then returns the appropriate implementation of the Client class.

See the documentation of the class and the example below for the supported methods.

When running on a legacy version of the app (below version 2.0.0), only the features that were available on the app are supported. Any unsopported method (e.g.: multisig registration or addresses, taproot addresses) will raise a NotImplementedError.

Running with speculos

It is possible to run the app and the library with the speculos emulator.

⚠️ Currently, speculos does not correctly emulate the version of the app, always returning a dummy value; in order to use the library, it is necessary to set the SPECULOS_APPNAME environment variable before starting speculos, for example with:

$ export SPECULOS_APPNAME="Bitcoin Test:2.1.0"

Similarly, to test the library behavior on a legacy version of the app, one can set the version to 1.6.5 (the final version of the 1.X series).

The expected application name is Bitcoin for mainnet, Bitcoin Test for testnet.

Example

The following example showcases all the main methods of the Client's interface.

If you are not using the context manager syntax when creating the client, remember to call the stop() method to release the communication channel.

Testing the sign_psbt method requires producing a valid PSBT (with any external tool that supports either PSBTv0 or PSBTv2), and provide the corresponding wallet policy; it is skipped by default in the following example.

from typing import Optional
from ledger_bitcoin import createClient, Chain, MultisigWallet, MultisigWallet, WalletPolicy, AddressType, TransportClient
from ledger_bitcoin.psbt import PSBT


def main():
    # speculos on default host/port
    # with createClient(TransportClient(), chain=Chain.TEST) as client:

    # Ledger Nano connected via USB
    with createClient(chain=Chain.TEST) as client:
        # ==> Get the master key fingerprint

        fpr = client.get_master_fingerprint().hex()
        print(f"Master key fingerprint: {fpr}")

        # ==> Get and display on screen the first taproot address

        first_taproot_account_pubkey = client.get_extended_pubkey("m/86'/1'/0'")
        first_taproot_account_policy = WalletPolicy(
            "",
            "tr(@0/**)",
            [
                f"[{fpr}/86'/1'/0']{first_taproot_account_pubkey}/**"
            ],
        )
        first_taproot_account_address = client.get_wallet_address(
            first_taproot_account_policy,
            None,
            change=0,
            address_index=0,
            display=True # show address on the wallet's screen
        )

        print(f"First taproot account receive address: {first_taproot_account_address}")

        # ==> Register a multisig wallet named "Cold storage"

        our_pubkey = client.get_extended_pubkey("m/48'/1'/0'/2'")
        other_key_info = "[76223a6e/48'/1'/0'/2']tpubDE7NQymr4AFtewpAsWtnreyq9ghkzQBXpCZjWLFVRAvnbf7vya2eMTvT2fPapNqL8SuVvLQdbUbMfWLVDCZKnsEBqp6UK93QEzL8Ck23AwF/**"

        multisig_policy = MultisigWallet(
            name="Cold storage",
            address_type=AddressType.WIT,
            threshold=2,
            keys_info=[
                other_key_info,                       # some other bitcoiner
                f"[{fpr}/48'/1'/0'/2']{our_pubkey}",  # that's us
            ],
        )

        policy_id, policy_hmac = client.register_wallet(multisig_policy)

        print(f"Policy hmac: {policy_hmac.hex()}. Store it safely (together with the policy).")

        assert policy_id == multisig_policy.id  # should never fail

        # ==> Derive and show an address for "Cold storage"

        multisig_address = client.get_wallet_address(multisig_policy, policy_hmac, change=0, address_index=0, display=True)
        print(f"Multisig wallet address: {multisig_address}")

        # ==> Sign a psbt

        # TODO: set a wallet policy and a valid psbt file in order to test psbt signing
        psbt_filename: Optional[str] = None
        signing_policy: Optional[WalletPolicy] = None
        signing_policy_hmac: Optional[bytes] = None
        if not psbt_filename or not signing_policy:
            print("Nothing to sign :(")
            return

        raw_psbt_base64 = open(psbt_filename, "r").read()
        psbt = PSBT()
        psbt.deserialize(raw_psbt_base64)

        result = client.sign_psbt(psbt, signing_policy, signing_policy_hmac)

        print("Returned signatures:")
        print(result)

if __name__ == "__main__":
    main()

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

ledger_bitcoin-0.1.2.tar.gz (63.4 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

ledger_bitcoin-0.1.2-py3-none-any.whl (75.4 kB view details)

Uploaded Python 3

File details

Details for the file ledger_bitcoin-0.1.2.tar.gz.

File metadata

  • Download URL: ledger_bitcoin-0.1.2.tar.gz
  • Upload date:
  • Size: 63.4 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/4.0.2 CPython/3.10.6

File hashes

Hashes for ledger_bitcoin-0.1.2.tar.gz
Algorithm Hash digest
SHA256 e9cfdfaad07f7c602a9aa21e7f371c56b04fca347d63278afae7199f3e1be111
MD5 ddc34201ab82ca210db8b7fa1b0de25b
BLAKE2b-256 b3a7c5745e8991a51efb64c9bd9899f0e0fceee2b3620faea0772c13f0b43278

See more details on using hashes here.

File details

Details for the file ledger_bitcoin-0.1.2-py3-none-any.whl.

File metadata

  • Download URL: ledger_bitcoin-0.1.2-py3-none-any.whl
  • Upload date:
  • Size: 75.4 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/4.0.2 CPython/3.10.6

File hashes

Hashes for ledger_bitcoin-0.1.2-py3-none-any.whl
Algorithm Hash digest
SHA256 e8edad936c3aa7c50de6743e1f50b9a01e84e23634823dbadd90d90cd55b53dd
MD5 96c7ca49ebf45b6a0e91b4f9f2854b09
BLAKE2b-256 eb07b448f43b119805d67c6a6dc33f98400ae87817104f7b3920d9d841f8e027

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page