developed with :heart: by chainside
btcnodejs
is a Segwit-compliant bitcoin library which provides tools for managing bitcoin data structures. It is the NodeJS version of btcpy.
This library is a work in progress and its usage in a production environment is highly discouraged. Also, as long as the version is 0.*, API breaking changes might occur
Some of the functionalities are a wrapping around bitcoinjs-lib, and their development is still in progress.
It makes usage of bytebuffer.js for representing most of the data structures
To install the library, run npm install btcnodejs
This library can be used in the browser with browserify. If you are familiar with browserify, you can skip this.
Assuming the entry point of the project is a file main.js like:
const btcnodejs = require("btcnodejs");
const net = btcnodejs.network;
net.setup("testnet");
var k = new btcnodejs.HDPrivateKey();
console.log(k.privkey.toWIF());
Go into the main.js folder and run:
browserify main.js > browser_main.js
Now you can load the 'browserified' main.js into your html:
<!DOCTYPE html>
<html>
<head>
<title>Example Browserify</title>
</head>
<body>
<script src="https://github.com/chainside/btcnodejs/raw/master/browser_main.js"></script>
</body>
</html>
This library aims to manage bitcoin data structures. It offers functionalities for
Transactions and block headers deserialization
Scripts creation
Privatekeys, Publickeys and HDkeys management
Transactions signing
This library does not implement the following functionalities:
Validation : when transactions and scripts are parsed, only format errors are reported. No proof-of-work validation, script execution, transaction validation and signature verification is performed
Networking : this library does not provide functionalities to communicate with bitcoin nodes. Separates networking modules will be released soon.
In order to run tests, cd in the package directory and run
npm test
To run tests in the browser, you first need brfs installed. Then you can run browserify on the tests file, by doing (within the package directory):
browserify -t brfs test/test.js > test/browser_tests.js
Now you can create the .html file with the test script:
<!DOCTYPE html>
<html>
<head>
<title>Mocha Tests</title>
<link rel="stylesheet" href="https://github.com/chainside/btcnodejs/blob/master/node_modules/mocha/mocha.css">
</head>
<body>
<div id="mocha"></div>
<script src="https://github.com/chainside/btcnodejs/raw/master/node_modules/mocha/mocha.js"></script>
<script>mocha.setup('bdd')</script>
<script src="https://github.com/chainside/btcnodejs/raw/master/test/browser_tests.js"></script>
<script>
mocha.run();
</script>
</body>
</html>
On the first import, network setup must be executed. This is achieved by doing:
const network = require('btcnodejs').network
network.setup('testnet') //network can be either 'testnet' or 'mainnet'
Once the network setup is executed, every subsequent setup will throw an exception.
The network module also exposes functionalities to get the current setup network
const network = require('btcnodejs').network
network.setup('testnet')
network.net_name() //outputs 'testnet'
network.is_mainnet() //returns 'false'
btcnodejs.Transaction
Transactions are immutable objects representing bitcoin transactions.
version : Integer
inputs : list of Input objects
outputs : list of Output objects
locktime : Locktime object
segwit : Boolean
txid : String
Returns the JSON representation of the transaction
Computes the double sha256 on the serialized transaction. Returns the hex string representing the hash
Computes the txid of a segwit transaction Returns the hex string representing the id of the transaction.
Returns a Transaction object
const btcnodejs = require('btcnodejs')
const Transaction = btcnodejs.Transaction
btcnodejs.network.setup('testnet')
const tx = Transaction.fromHex("0100000001e4da173fbefe5e60ff63dfd38566ade407532294db655463b77a783f379ce605000000006b4" +
"83045022100af246c27890c2bc07a0b7450d3d82509702a44a4defdff766355240b114ee2ac02207bb67b" +
"468452fa1b325dd5583879f5c1412e0bb4dae1c2c96c7a408796ab76f1012102ab9e8575536a1e99604a1" +
"58fc60fe2ebd1cb1839e919b4ca42b8d050cfad71b2ffffffff0100c2eb0b000000001976a914df76c017" +
"354ac39bde796abe4294d31de8b5788a88ac00000000")
console.log(tx.txid) //'e977c07090c2a1dcaefd3f3c4ebf4e231f4116cb272f805b0b22a85e7eece09c'
Returns the hexadecimal representation of the transactions
Returns a ByteBuffer containing the serialized transaction. If called with no parameter, the serialization is performed based on the transaction type. To perform a non-segwit serialization of a segwit transaction(i.e. to compute the segwit txid), false can be passed to the function.
Returns a Transaction object from a ByteBuffer object representing the serialized transaction
Returns a MutableTransaction object
btcnodejs.MutableTransaction
Mutable Transaction objects
version : Integer
inputs : list of Input objects
outputs : list of Output objects
locktime : Locktime object
segwit : Boolean
txid : String
Returns a MutableTransaction object
Returns an immutable Transaction object
txouts : List of Output objects
solvers : List of Solver objects
Returns a Transaction object where the scriptSigs of its inputs are computed.
const btcnodejs = require('./lib/index')
btcnodejs.network.setup(network)
var t = new btcnodejs.Transaction(...)
var tospend = btcnodejs.Transaction.fromHex('...')
var key = btcnodejs.Privatekey.fromWIF(wif_key)
var solver = new btcnodejs.P2pkhSolver(key)
var unsigned = t.toMutable()
var spent = unsigned.spend([tospend.outputs[1]], [solver])
btcnodejs.Sighash
Sighash object
sighash : String
anyonecanpay : Boolean
Returns a Sighash object
btcnodejs.Input
Transaction Input object
txid : String
out : Integer
scriptSig : ScriptSig object
sequence : Sequence object
witness : Witness object
Returns an Input object
Returns the JSON representation of the Input
btcnodejs.Output
Transaction Output object
amount : Integer
scriptPubKey : ScriptPubKey object
Returns an Output object
Returns the JSON representation of the Output
btcnodejs.Witness
Input Witness Object
Returns a Witness Object where data represents the required data to sign a transaction Input
Returns a ByteBuffer representing the Witness serialization as it appears in a bitcoin transaction
Returns a ScriptSig object
Returns the hexadecimal representation of the Witness object
Returns a Witness object from an array of hexadecimal strings representing the siganture and the public key
btcnodejs.Sequence
Sequence object representing the sequence number of a transaction Input
Returns a Sequence object
Returns a Boolean which tells if the Sequence is measured in time
Returns a Boolean which tells if the Sequence is measured in blocks
Returns a Boolean which tells if a Sequence restriction is active
btcnodejs.Locktime
Locktime object representing the locktime on a transaction
Returns a Locktime object
Returns a Boolean which tells if the Locktime is measured in time
Returns a Boolean which tells if the Locktime is measured in blocks
Returns a Boolean which tells if a Locktime restriction is active
btcnodejs.Script
Base Script object representing a general script as a ByteBuffer. Every Script class extends Script.
Returns a Script object initialized from a bytebuffer representing the script
Returns the body of the script_code
btcnodejs.ScriptSig
ScriptSig object
Returns a string representing the ASM of the script
Returns a ScriptSig from an ASM string
Returns the hexadecimal representation of the ScriptSig
Returns a ScriptSig object from an hexadecimal string representing the body of the script
Returns a Witness object where its data is retrieved from the ScriptSig body removing the push operations
const btcnodejs = require('btcnodejs')
const ScriptSig = btcnodejs.ScriptSig
btcnodejs.network.setup('testnet')
const sig = ScriptSig.fromHex("483045022100b7bf286e5f6ac6fa308e8876a8a59b289094851a26cf62c20abd174917eb7762022069b5269e584e4c7" +
"6f207d1b789bff7171a663d795e49751c12cf07dceb2a94c70121024a0dcb0527c2751ea4dda3aa98f6eface16e978d" +
"ba8062bcbed623f158c07691")
sig.toWitness().toHex()
// "02483045022100b7bf286e5f6ac6fa308e8876a8a59b289094851a26cf62c20abd174917eb7762022069b5269e584e4c76f207d1b789bff7171a663d795e49751c12c f07dceb2a94c70121024a0dcb0527c2751ea4dda3aa98f6eface16e978dba8062bcbed623f158c07691"
btcnodejs.ScriptPubKey
ScriptPubkey object representing a general script pubkey. It extends Script and is extended by specific ScriptPubkey types.
Returns the hexadecimal representation of the ScriptPubKey
Returns a ScriptPubKey object from an hexadecimal string representing the body of the script. If the hex is representing an identifiable script, the fromHex() will return an instance of the specific ScriptPubKey. At the moment, identifiables scripts are P2pkh, P2sh, P2wpkhV0, P2wshV0, MultiSig
const btcnodejs = require('btcnodejs')
const ScriptPubKey = btcnodejs.ScriptPubKey
btcnodejs.network.setup('testnet')
const spk = ScriptPubKey.fromHex('76a9148b4912ec0496b5f759f3af5ab24d6f4779a52f9e88ac')
spk instanceof btcnodejs.P2pkhScript //true
network : String
segwitV : Integer
Returns the p2sh/p2wsh address of the Script.
If network is not specified, the initial setup network will be considered as the address network.
If segwitV is specified, the address will be of type 'p2wsh', with segwitV value as the segwit version
const btcnodejs = require('btcnodejs')
const ScriptPubKey = btcnodejs.ScriptPubKey
btcnodejs.network.setup('testnet')
const spk = ScriptPubKey.fromHex('76a9148b4912ec0496b5f759f3af5ab24d6f4779a52f9e88ac')
const p2pkh_address = spk.toAddress()
p2pkh_address.hash // "029c09b86e1e4c3822bc71859af3300520d577c2"
p2pkh_address.toBase58() // "2MsV2GNkfjxPjsp9ux2vwxW5HYaZh1HDtXJ
btcnodejs.P2pkhScript
P2pkhScript object
Returns a P2pkhScript object. This can be obtained from an Address, a Publickey or a ByteBuffer representing a pubkeyhash
type : String
pubkeyhash : ByteBuffer
Returns an Address object representing the script Address
btcnodejs.P2wpkhV0Script
Segwit version of P2pkhScript. It has the same interface of P2pkhScript but the source Address must be a Segwit Address object
Returns the ScriptCode of the P2wpkhV0Script
btcnodejs.P2shScript
P2shScript object
type : String
scripthash : ByteBuffer
Returns a P2pkhScript object. This can be obtained from an Address, a ScriptPubKey or a ByteBuffer representing a scripthash
Returns an Address object representing the script Address
btcnodejs.P2wshV0Script
Segwit version of P2shScript. It has the same interface of P2shScript but the source Address must be a Segwit Address object
btcnodejs.IfElseScript
IfElseScript object
type : String
if_script : ScriptPubKey object
else_script : ScriptPubKey object
const btcnodejs = require('btcnodejs')
btcnodejs.network.setup('testnet')
const p2pkh = new btcnodejs.P2pkhScript(btcnodejs.Publickey.fromHex("026263992eda6538202047f1514e0f6155a229c3d61b066807664e9ef73d406d95"))
const multisig = new btcnodejs.MultiSigScript([
2,
btcnodejs.Publickey.fromHex(
"02c08786d63f78bd0a6777ffe9c978cf5899756cfc32bfad09a89e211aeb926242"
),
btcnodejs.Publickey.fromHex(
"033e81519ecf373ea3a5c7e1c051b71a898fb3438c9550e274d980f147eb4d069d"
),
btcnodejs.Publickey.fromHex(
"036d568125a969dc78b963b494fa7ed5f20ee9c2f2fc2c57f86c5df63089f2ed3a"
),
3
])
const ie_script = new btcnodejs.IfElseScript([p2pkh, multisig])
btcnodejs.RelativeTimelockScript
RelativeTimelockScript object
Returns a RelativeTimelockScript object
sequence : Sequence object
locked_script : ScriptPubKey object
type : String
btcnodejs.MultiSigScript
MultisigScript object
Returns a MultiSigScript object
type : String
m : Integer
n : Integer
pubkeys : Array of Publickey objects
const btcnodejs = require('btcnodejs')
btcnodejs.network.setup('testnet')
//Creating a 2-of-3 Multisig Script
const multisig = new btcnodejs.MultiSigScript([
2,
btcnodejs.Publickey.fromHex(
"02c08786d63f78bd0a6777ffe9c978cf5899756cfc32bfad09a89e211aeb926242"
),
btcnodejs.Publickey.fromHex(
"033e81519ecf373ea3a5c7e1c051b71a898fb3438c9550e274d980f147eb4d069d"
),
btcnodejs.Publickey.fromHex(
"036d568125a969dc78b963b494fa7ed5f20ee9c2f2fc2c57f86c5df63089f2ed3a"
),
3
])
Solvers are objects which are able to compute the scriptSig and Witness from a given array of digests. They are an easy way to compute transaction input's signatures.
They all provide the method:
which returns an object :
{scriptSig : ScriptSig object , witness: Witness object}
const btcnodejs = require('btcnodejs')
btcnodejs.network.setup('testnet')
const private_key = btcnodejs.Privatekey.fromHex('9b1b400e3b1211c6a56695cf1742f0a94ea38b995c1e1fb910458baa8a0874c4')
const p2pkh_solver = new btcnodejs.P2pkhSolver(private_key)
p2pkh_solver.solve(["0e12bda8a692aefa29651e87af9f47127ab098be1c189284e41d8e17a0516add"]).scriptSig.toHex()
//
'47304402202409f1f966c382f02e023ac828d7653e9268777bd1030e7101338f36a383fde302207ced2d14ff131d3b349bb00d3010a16f08831548e68750223cb8117cab553cab01210330e8ca46b7e5aa07d975ee152214431e419fac34e50becaf7e46db9a9c97d244'
btcnodejs.P2pkhSolver
Solver for P2pkh Scripts
privkey : Privatekey object
sighash : Sighash object
Returns a P2pkhSolver object
const btcnodejs = require('btcnodejs')
btcnodejs.network.setup('testnet')
const private_key = btcnodejs.Privatekey.fromHex('9b1b400e3b1211c6a56695cf1742f0a94ea38b995c1e1fb910458baa8a0874c4')
const p2pkh_solver = new btcnodejs.P2pkhSolver(private_key)
btcnodejs.P2wpkhV0Solver
Solver for P2wpkh version 0 scripts. It extends P2pkhSolver
btcnodejs.P2shSolver
Solver for P2sh scripts
redeemScript : ScriptPubKey object
redeemScriptSolver : Solver object
btcnodejs.P2wshV0Solver
Solver for P2wsh version 0 Scripts
witnessScript : ScriptPubKey object
witnessScriptSolver : Solver object
btcnodejs.MultiSigSolver
Solver for MultiSig Scripts
privkeys : Array of Privatekey objects
sighashes : Array of Sighash objects
The number of sigashes must be equal to the number of privatekeys.
btcnodejs.IfElseSolver
Solver for If Else Scripts
branch : Integer
innerSolver : Solver object
btcnodejs.RelativeTimelockSolver
The library provides structures and methods to handle Private and Public key objects
btcnodejs.Privatekey
Returns a Privatekey object where the body is a bytebuffer representing the private key
Returns a Privatekey object from an hexadecimal string representing a private key
Returns the Hexadecimal representation of the Privatekey
Returns the body of the Privatekey
Returns a Publickey object representing the public key associated with this Privatekey. By passing compressed = false
, the public key will be of uncompressed type.
Computes the signature of a message, using the elliptic nodejs library using the secp256k1
curve.
Returns the signature of the message in DER
encoding
Returns the Wallet import format string representing the private key. If true
is passed in input, the WIF string will represent a private key associated with a compressed Publickey.
Return a Privatekey object from its Wallet import format string.
Returns a Privatekey object from its Bip32 format string.
btcnodejs.Publickey
Publickey object
type: String
uncompressed : ByteBuffer
compressed : ByteBuffer
Returns a Publickey object. Its type will be odd
, even
, uncompressed
based on the the input bytebuffer data. It will keep both the uncompressed and compressed versions as its body, but its type will decide which version to use for any operation.
Returns the hash of the Publickey body.
Returns a Publickey object from its hexadecimal representation
Returns the hexadecimal representation of its body. If false
is passed as input it will return the hexadecimal representing its uncompressed version.
network : String
segwit : Boolean
Returns an Address object created from the Publickey hash.
If network is undefined, the first network name setup will be used.
If segwit is true, the Address type will be p2wpkh, otherwise it will be p2pkh
Returns the Publickey body
This library exposes functionalities to manage Hierarchical deterministic keys. It makes usage of bitcoinjs-lib for some functionalities
btcnodejs.HDPrivateKey
Hierarchical deterministic PrivateKey object
privkey : Privatekey object
depth : Integer
fingerPrint : Integer
parentFingerPrint : Integer
childIndex : Integer
chainCode : String
checksum : Integer
xprivkey : String
Returns an HDPrivateKey object. If no parameter is given as input, a random HDPrivateKey is returned. Otherwise, a bip32 representation of an hd key can be passed
Returns a child HDPrivateKey derived as specified in BIP32. The path must be a string starting with 'm/'. To derive an hardened child, its index in the path is followed by '
i.e derive('m/0'')
Returns the corresponding HDPublicKey
Returns a master HDPrivateKey generated from the hexadecimal string representing a seed
btcnodejs.HDPublicKey
Hierarchical deterministic Public key object
pubkey : Privatekey object
depth : Integer
fingerPrint : Integer
parentFingerPrint : Integer
childIndex : Integer
chainCode : String
checksum : Integer
xpubkey : String
Returns an HDPublicKey object. If no parameter is given as input, a random HDPublicKey is returned. Otherwise, a bip32 representation of an hd key can be passed.
Returns a child HDPublicKey derived as specified in BIP32.
This library exposes functionalities to manage bitcoin addresses. It wraps bitcoinjs-lib for addresses encodings.
btcnodejs.Address
network : String
type : String
hash : ByteBuffer
Returns an Address object. If network is not specified, the first setup network name will be used.
Returns an Address object from its base58 encoding
Returns a Base58 encoded string representing the bitcoin address
btcnodejs.SegwitAddress
SegwitAddress object extending Address. It has an extra version
attribute specifying the segwit version.
Returns a Segwit Address object from its bech32 encoding
Returns a Bech32 encoded string representing the bitcoin Segwit address
btcnodejs.BlockHeader
BlockHeader object
version : Integer
prev_block : String
merkle_root : String
timestamp : Integer
bits : Integer
nonce : Integer
Returns a BlockHeader objects
Returns a BlockHeader object from an hexadecimal string representing the associated BlockHeader
Returns a ByteBuffer representing the serialized BlockHeader
Returns an hexadecimal string representing the hash of the associated Block
Expand the test vectors
Add docstrings to code
Manage Block and MerkleBlock structures
Add caching in segwit digests computation
Add further helpers for creating transactions
Implement the functionalities which are now wrappings around external libraries
Manage OP_CODESEPARATOR
in transaction signatures