btctool v0.7.0 BtcTool View Source

Bitcoin utils related to Elliptic curve cryptography (ECC) algorithms used in bitcoin to create addresses or public keys from private keys, brainwallets, WIFs, etc.

Link to this section Summary

Types

All valid types of bitcoin addresses

Hash which includes WIF, private key and metadata

Public key derived from the private key in binary format

Wallet Import Format string ready to be imported into a wallet. It uses base58check characters. The raw private key can be extracted from this

Functions

Returns Wallet Import Format (WIF) generated from any arbitrary text (passphrase)

Create Wallet Import Format (WIF) private key from raw private key. A raw private key can be presented by a binary of 32 bytes or in 64 hexadecimal characters (0-9a-fA-F)

Generate the address corresponding to the Wallet Import Format (WIP) string where to receive funds

Returns the raw private key from a Wallet Import Format (WIF) string. Including metadata from WIF telling:

  • compressed: If when generating the public key, should use compressed format or uncompressed.
  • network: Where the private key should be used. In mainnet, or testnet

Generate public key from a Wallet Import Format string (WIF)

Link to this section Types

Link to this type address_type() View Source
address_type() :: :p2pkh

All valid types of bitcoin addresses.

Link to this type privkey_result() View Source
privkey_result() :: %{wif: wif_type(), privkey_bin: <<_::256>>, privkey_hex: <<_::512>>, network: :testnet | :mainnet, compressed: boolean()}

Hash which includes WIF, private key and metadata.

WIF string

Wallet Import Format string containing the private key. It also includes metadata with information extracted from the WIF string.

WIF will be a base58check string of 51 characters (408 bits) if user want to use uncompressed public keys in the bitcoin addresses, or 52 characters (416 bits) if wants to use compressed public keys.

Raw private key

Raw private key in binary format (512bits) and hexadecimal format (characters a-z0-9).

Available metadata

Metadata like network or compressed is deducted from the WIP string:

  • network. Which network (:mainnet, or :testnet) is intended to be used the private key.
  • compressed. States if when using private key to generate an address should use the compressed or uncompressed version of the public key. Note: Nowadays is normal to use the compressed version.
Link to this type pubkey_type() View Source
pubkey_type() :: <<_::264>> | <<_::520>>

Public key derived from the private key in binary format.

This key can be compressed or uncompressed. Compressed keys include only the x coordinate. Uncompressed public key includes the x and y coordinates. To differentiate between them, an additional byte is added to the beginning:

  • Uncompressed public key starts with 0x04
  • Compressed public key begins with 0x02 or 0x03 depending on if the y coordinate is ever or odd respectively.
Link to this type wif_type() View Source
wif_type() :: <<_::408>> | <<_::416>>

Wallet Import Format string ready to be imported into a wallet. It uses base58check characters. The raw private key can be extracted from this.

Examples:

  • Uncompressed private key to be imported (51 characters in base58, starts with 5) E.g.: 5HpneLQNKrcznVCQpzodYwAmZ4AoHeyjuRf9iAHAa498rP5kuWb
  • Compressed private key to be imported (52 characters in base58, starts with K or L) E.g.: KwFvTne98E1t3mTNAr8pKx67eUzFJWdSNPqPSfxMEtrueW7PcQzL

Link to this section Functions

Link to this function brainwallet_to_wif(passphrase, options \\ []) View Source
brainwallet_to_wif(binary(), [{atom(), any()}]) ::
  {:ok, privkey_result()} |
  {:error, atom()}

Returns Wallet Import Format (WIF) generated from any arbitrary text (passphrase).

Disclaimer

Using any low entropy text like any text from a book, poem, song or sentence will result in you losing your coins. In other words, if you don’t know what your doing: Don’t use brainwallets. More info at DEF CON 23 - Ryan Castellucci - Cracking CryptoCurrency Brainwallets

Options

It accepts the same options that the function privkey_to_wif/2:

  • compressed - Generate a WIF which signals that a compressed public key should be used if true. Default to true.
  • network - Specifies the network this private key intended to be used on. Can be :mainnet or :testnet. Default is :mainnet.

Examples

iex> brainwallet_to_wif("correct horse battery staple")
{:ok, %{
  compressed: true, network: :mainnet,
  privkey_bin: <<196, 187, 203, 31, 190, 201, 157, 101, 191, 89, 216, 92, 140, 182, 46, 226, 219, 150, 63, 15, 225, 6, 244, 131, 217, 175, 167, 59, 212, 227, 154, 138>>,
  privkey_hex: "C4BBCB1FBEC99D65BF59D85C8CB62EE2DB963F0FE106F483D9AFA73BD4E39A8A",
  wif: "L3p8oAcQTtuokSCRHQ7i4MhjWc9zornvpJLfmg62sYpLRJF9woSu"}}
Link to this function privkey_to_wif(privkey, options \\ []) View Source
privkey_to_wif(<<_::512>> | <<_::256>>, [{atom(), any()}]) ::
  {:ok, privkey_result()} |
  {:error, atom()}

Create Wallet Import Format (WIF) private key from raw private key. A raw private key can be presented by a binary of 32 bytes or in 64 hexadecimal characters (0-9a-fA-F)

It assumes you want the compressed WIF version by default. That way you are signalling that the bitcoin address which should be used when imported into a wallet will be also compressed.

Options

  • compressed - Generate a WIF which signals that a compressed public key should be used if true. Default to true.
  • network - Specifies the network this private key intended to be used on. Can be :mainnet or :testnet. Default is :mainnet.
  • case - Ensures the character case to accept when decoding. Valid values are: :upper, :lower, :mixed. Only useful when the raw private key is passed in hex format. If case is not satisfied will return an error. Default is :mixed

Examples

iex> hexprivkey = "0123456789ABCDEF0123456789ABCDEF0123456789ABCDEF0123456789ABCDEF"
iex> privkey_to_wif(hexprivkey)
{:ok, %{
  wif: "KwFvTne98E1t3mTNAr8pKx67eUzFJWdSNPqPSfxMEtrueW7PcQzL",
  privkey_bin: <<1, 35, 69, 103, 137, 171, 205, 239, 1, 35, 69, 103, 137, 171, 205, 239, 1, 35, 69, 103, 137, 171, 205, 239, 1, 35, 69, 103, 137, 171, 205, 239>>,
  privkey_hex: "0123456789ABCDEF0123456789ABCDEF0123456789ABCDEF0123456789ABCDEF",
  compressed: true,
  network: :mainnet
}}
iex> binprivkey = hexprivkey |> Base.decode16!()
<<1, 35, 69, 103, 137, 171, 205, 239, 1, 35, 69, 103, 137, 171, 205, 239, 1, 35, 69, 103, 137, 171, 205, 239, 1, 35, 69, 103, 137, 171, 205, 239>>
iex> privkey_to_wif(binprivkey)
{:ok, %{
  wif: "KwFvTne98E1t3mTNAr8pKx67eUzFJWdSNPqPSfxMEtrueW7PcQzL",
  privkey_bin: <<1, 35, 69, 103, 137, 171, 205, 239, 1, 35, 69, 103, 137, 171, 205, 239, 1, 35, 69, 103, 137, 171, 205, 239, 1, 35, 69, 103, 137, 171, 205, 239>>,
  privkey_hex: "0123456789ABCDEF0123456789ABCDEF0123456789ABCDEF0123456789ABCDEF",
  compressed: true,
  network: :mainnet
}}
iex> privkey_to_wif(binprivkey, compressed: false, network: :testnet)
{:ok, %{
  wif: "91bRE5Duv5h8kYhhTLhYRXijCiXWSpWwFNX6nndfuntBdPV2idD",
  privkey_bin: <<1, 35, 69, 103, 137, 171, 205, 239, 1, 35, 69, 103, 137, 171, 205, 239, 1, 35, 69, 103, 137, 171, 205, 239, 1, 35, 69, 103, 137, 171, 205, 239>>,
  privkey_hex: "0123456789ABCDEF0123456789ABCDEF0123456789ABCDEF0123456789ABCDEF",
  compressed: false,
  network: :testnet
}}

When binary private key has an unexpected length (not 64 bytes for hex format or 32 bytes for binary format) returns error:

iex> privkey_to_wif(<<1, 35, 69>>)
{:error, :incorrect_privkey_length}

When private key is out of recommended range will return error:

iex> maxprivkey = "FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEBAAEDCE6AF48A03BBFD25E8CD0364140" |> Base.decode16!()
iex> privkey_to_wif(maxprivkey)
{:error, :ecc_out_range}

When private key is hexadecimal and have an unexpected case:

iex> privkey_to_wif("0123456789ABCDEF0123456789ABCDEF0123456789ABCDEF0123456789ABCDEF", case: :lower)
{:error, :unexpected_hexadecimal_case}
Link to this function wif_to_address(wif, options \\ []) View Source
wif_to_address(wif_type(), [{atom(), any()}]) ::
  {:ok, %{address: binary(), type: address_type()}} |
  {:error, atom()}

Generate the address corresponding to the Wallet Import Format (WIP) string where to receive funds.

The result will include the address, the type of the address and if public key used was compressed or not.

Options

Available options are:

  • type: Which tells the type of address. Currently the only valid value is only :p2pkh. Default value will be the most modern available, so be sure to specify a type if you don’t want your address to suddenly change.

Examples

iex> wif_to_address("L3p8oAcQTtuokSCRHQ7i4MhjWc9zornvpJLfmg62sYpLRJF9woSu")
{:ok, %{
  address: "1C7zdTfnkzmr13HfA2vNm5SJYRK6nEKyq8",
  compressed: true,
  type: :p2pkh}}
iex> wif_to_address("5KJvsngHeMpm884wtkJNzQGaCErckhHJBGFsvd3VyK5qMZXj3hS", type: :p2pkh)
{:ok, %{
  address: "1JwSSubhmg6iPtRjtyqhUYYH7bZg3Lfy1T",
  compressed: false,
  type: :p2pkh}}
iex> wif_to_address("5KJvsngHeMpm884wtkJNzQGaCErckhHJBGFsvd3VyK5qMZXj3hS", type: :wofff)
{:error, :unknown_address_type}
Link to this function wif_to_privkey(wif) View Source
wif_to_privkey(wif_type()) ::
  {:ok, privkey_result()} |
  {:error, atom()}

Returns the raw private key from a Wallet Import Format (WIF) string. Including metadata from WIF telling:

  • compressed: If when generating the public key, should use compressed format or uncompressed.
  • network: Where the private key should be used. In mainnet, or testnet.

Examples

Converts from a WIF string to raw private key.

iex> wif_to_privkey("KwFvTne98E1t3mTNAr8pKx67eUzFJWdSNPqPSfxMEtrueW7PcQzL")
{:ok, %{
  wif: "KwFvTne98E1t3mTNAr8pKx67eUzFJWdSNPqPSfxMEtrueW7PcQzL",
  privkey_bin: <<1, 35, 69, 103, 137, 171, 205, 239, 1, 35, 69, 103, 137, 171, 205, 239, 1, 35, 69, 103, 137, 171, 205, 239, 1, 35, 69, 103, 137, 171, 205, 239>>,
  privkey_hex: "0123456789ABCDEF0123456789ABCDEF0123456789ABCDEF0123456789ABCDEF",
  compressed: true,
  network: :mainnet
}}

Expected errors:

Error if is base58check is another type other from WIF:

iex> wif_to_privkey("1CLrrRUwXswyF2EVAtuXyqdk4qb8DSUHCX")
{:error, :not_wif_version_prefix}

Error if checksum is not valid:

iex> wif_to_privkey("KyFvTne98E1t3mTNAr8pKx67eUzFJWdSNPqPSfxMEtrueW7PcQzL")
{:error, :checksum_incorrect}

Error if using an unvalid base58 character:

iex> wif_to_privkey("10Ol0Ol0Ol0Ol0Ol0Ol0OOlIIIIIII0OlI")
{:error, :incorrect_base58}
Link to this function wif_to_pubkey(wif) View Source
wif_to_pubkey(wif_type()) ::
  {:ok, %{pubkey_bin: BtcTool.pubkey_type(), pubkey_hex: <<_::528>> | <<_::1040>>, compressed: boolean()}} |
  {:error, atom()}

Generate public key from a Wallet Import Format string (WIF).

If succesfull, public key is returned in binary and hexadecimal format in a map.

For any private key, its public key can be presented in compressed or uncompressed format. The compressed format is usually the most used. Which one should be used, can be deducted from the WIF string, so this function finds out which one is requested without any user intervention. Returned result will include if public key was compressed or uncompressed.

Note: If you want to generate a public key directly from the binary private key, instead of a WIF string, you can use the function BtcTool.Pubkey.binprivkey_to_binpubkey/2, documented in the source.

Examples

With a compressed a WIF string (starts with K or L):

iex> wif_to_pubkey("KwFvTne98E1t3mTNAr8pKx67eUzFJWdSNPqPSfxMEtrueW7PcQzL")
{:ok, %{
  pubkey_bin: <<3, 70, 70, 174, 80, 71, 49, 107, 66, 48, 208, 8, 108, 138, 206, 198, 135, 240, 11, 28, 217, 209, 220, 99, 79, 108, 179, 88, 172, 10, 154, 143, 255>>,
  pubkey_hex: "034646AE5047316B4230D0086C8ACEC687F00B1CD9D1DC634F6CB358AC0A9A8FFF",
  compressed: true}
}

With a uncompressed WIF string (starts with 5):

iex> wif_to_pubkey("5HpneLQNKrcznVCQpzodYwAmZ4AoHeyjuRf9iAHAa498rP5kuWb")
{:ok, %{
  pubkey_bin: <<4, 70, 70, 174, 80, 71, 49, 107, 66, 48, 208, 8, 108, 138, 206, 198, 135, 240, 11, 28, 217, 209, 220, 99, 79, 108, 179, 88, 172, 10, 154, 143, 255, 254, 119, 180, 221, 10, 75, 251, 149, 133, 31, 59, 115, 85, 199, 129, 221, 96, 248, 65, 143, 200, 166, 93, 20, 144, 122, 255, 71, 201, 3, 165, 89>>,
  pubkey_hex: "044646AE5047316B4230D0086C8ACEC687F00B1CD9D1DC634F6CB358AC0A9A8FFFFE77B4DD0A4BFB95851F3B7355C781DD60F8418FC8A65D14907AFF47C903A559",
  compressed: false}
}

Will return an error and atom if any error is present. Some examples (among other errors):

iex> wif_to_pubkey("Not_a_WIF")
{:error, :incorrect_base58}
iex> wif_to_pubkey("2dqAyxFfwDYds")
{:error, :unexpected_length}