/birthday-attack

what are the odds?

Primary LanguagePython

birthday attack

build the bitcoin UTXO set and attempt to collide by randomizing private keys

it's basically impossible. #getatme

inspiration

ok sure, so there are liek just as many private keys as there are atoms in the known universe and stuff. everyone likes to be all smart and say things like "guessing a single private key is hard and stuff" but what if you could guess -any- private key? what are the odds?

the 'birthday paradox' is the idea that while you and i have a 1/366 chance of having the same birthday- it only takes 15 people in a room before the odds are more likely than not that any two people share a birthday. maths.

so what are the odds?

the odds

the following is a table of all UTXOs on the bitcoin blockchain as of January 28th, 2023

Total UTXOs: 83310739
Total BTC: 19266337.75311853

Script Types:

type count
non-standard 9,565
p2pk 47,584
p2pkh 47,777,074
p2sh 15,252,482
p2ms 440,424
p2wpkh 18,343,108
p2wsh 1,120,662
p2tr 319,840

out of the 'easy' to guess script types: p2pk, p2pkh, p2wpkh... this give us a total of 66,167,766 targets (based on a single pubkey) we could easily identify, add to a set, and randomly guess private keys to try to find a match.

therefore, for every private key we test- we have a 1 in 1,749,977,311,268,393,069,682,640,262,276,501,272,818,145,923,590,142,361,648,747,610,374,144 chance! bullish!

my fastest computer can do 30 million private keys an hour. sure, I could re-write the collision logic in C++ or Rust and do this much faster, but this is my thing and I can do it stupid if i want (and i do)

so, if i run my fastest computer non-stop, it will only take 6,658,969,981,995,408,494,363,561,289,680,424,209,196,855,797,736,955,445,248 years to find a match. bullish!

setup

bitcoin core configuration

verify your bitcoin.conf file contains

rpcuser=<some username>
rpcpassword=<some password>
rpcservertimeout=<some value>

I typically set my the rpcservertimeout value high, such as 1200. However, 30 (the default) is fine with the use of RpcController.

dependencies

https://github.com/jgarzik/python-bitcoinrpc
https://github.com/rsalmei/alive-progress

install using pip

pip install -r requirements.txt

or install manually

pip install bitcoinrpc
pip install python-bitcoinrpc
pip install secp256k1

importing

I strongly recommend using the import feature to bootstrap this program. reading every single block from rpc takes a long time.

in3rsha's bitcoin-utxo-dump can be used to dump the current state of the blockchain. the project can be found here:

https://github.com/in3rsha/bitcoin-utxo-dump

as of Jan 28, 2023- I've fixed a bug in this project, and until my pull request is merged by in3rsha, I have to recommend using my version:

https://github.com/turkycat/bitcoin-utxo-dump

follow the instructions in the repo to build or install bitcoin-utxo-dump, point it at a copy of your chainstate folder and run with the following parameter to get the proper format for importing: -f txid,vout,height,amount,script,type

the program will output a file named utxodump.csv, which you should then pass as an argument using the --import switch. IE

python birthday-attack.py --import ./utxodump.csv

importing will force clean the outputs folder (the same as running with --clean)- so any existing state you may have in that folder that you wish to preserve should be backed up before importing.

RpcController

This repository contains an RpcController class which wraps the behavior of python-bitcoinrpc making it much easier to use. An RPC connection will be established automatically and maintained. If the connection is severed either through timeout or failure, reconnection attempts will be made. All RPC exceptions are caught and managed by the controller, making usage in a primary application much easier and cleaner, although errors will not be immediately surfaced.

To configure RpcController, edit config.py in the module directory to define your authentication credentials.

note: any time the RPC connection times out, you will see this error. It is normal, as the connection will be re-established automatically.

IO Error [Errno 32] Broken pipe

pick a random starting point for keyring

paste output from the following command to pick a truly random starting point for target matching. either overwrite the saved value in cache.json or replace the value in the KeyRing initialization in __main__

TODO: add a switch to read this random value instead of hardcoding it

$ dd if=/dev/urandom bs=32 count=1 2>/dev/null | xxd -p | tr -d '\n'

output

an output directory will automatically be created containing a json cache file for storing program state, an errors.txt file, and an sqlite database containing the unspent transactions.

sqlite database

each row contains a single utxo.

column type description
hash text transaction hash for the output
vout integer index for the output
height integer block height containing this transaction
amount integer value of this output, in sats
script text the full scriptPubKey that 'locks' this transaction
type text the type as described by bitcoin core. IE- 'pubkeyhash', 'multisig', 'witness_v0_keyhash' etc
target text the target pubkey or pubkey hash, should always be a substring of the script

(hash, idx, height) is used as a composite primary key for this database. normally, hash and idx should suffice to identify a transaction (and only those two values are used when removing spent outputs from the database)- but height was also added as part of the primary key due to some early coinbase transactions having the same transaction hash.

example of a duplicate transaction hash:

e3bf3d07d4b0375638d5f1db5255fe07ba2c4cb067cd81b84ee974b6585fb468 from block 91722

e3bf3d07d4b0375638d5f1db5255fe07ba2c4cb067cd81b84ee974b6585fb468 from block 91880

tests

from tests/ directory, simply run

python -m unittest