A modern NIX Core REST and RPC client to execute administrative tasks, multiwallet operations and queries about network and the blockchain.
[![npm version][npm-image]][npm-url] [![build status][travis-image]][travis-url]
Install the package via yarn
:
yarn add nix-sdk
or via npm
:
Install the package via npm
:
npm install nix-sdk --save
[agentOptions]
(Object): Optionalagent
options to configure SSL/TLS.[headers=false]
(boolean): Whether to return the response headers.[host=localhost]
(string): The host to connect to.[logger=debugnyan('nix-sdk')]
(Function): Custom logger (by default,debugnyan
).[network=mainnet]
(string): The network[password]
(string): The RPC server user password.[port=[network]]
(string): The RPC server port (nix default: 6214).[ssl]
(boolean|Object): Whether to use SSL/TLS with strict checking (boolean) or an expanded config (Object).[ssl.enabled]
(boolean): Whether to use SSL/TLS.[ssl.strict]
(boolean): Whether to do strict SSL/TLS checking (certificate must match host).[timeout=30000]
(number): How long until the request times out (ms).[username]
(number): The RPC server user name.[wallet]
(string): Which wallet to manage (read more).
The network
will automatically determine the port to connect to, just like the nixd
and nix-cli
commands.
const Client = require('nix-sdk');
const client = new Client({ network: 'mainnet' });
const client = new Client({ port: 28332 });
By default, when ssl
is enabled, strict checking is implicitly enabled.
const fs = require('fs');
const client = new Client({
agentOptions: {
ca: fs.readFileSync('/etc/ssl/nixd/cert.pem')
},
ssl: true
});
const client = new Client({
ssl: {
enabled: true,
strict: false
}
});
client.getInfo().then((help) => console.log(help));
client.getInfo((error, help) => console.log(help));
For compatibility with other NIX Core clients.
const client = new Client({ headers: true });
// Promise style with headers enabled:
client.getInfo().then(([body, headers]) => console.log(body, headers));
// Await style based on promises with headers enabled:
const [body, headers] = await client.getInfo();
Due to JavaScript's limited floating point precision, all big numbers (numbers with more than 15 significant digits) are returned as strings to prevent precision loss. This includes both the RPC and REST APIs.
This enables use-cases such as managing a personal and a business wallet simultaneously in order to simplify accounting and accidental misuse of funds.
To enable Multi Wallet support, start by specifying the number of added wallets you would like to have available and loaded on the server using the -wallet
argument multiple times.
Notice the rpcauth
hash which has been previously generated for the password j1DuzF7QRUp-iSXjgewO9T_WT1Qgrtz_XWOHCMn_O-Y=
. Do not copy and paste this hash ever beyond this exercise.
Instantiate a client for each wallet and execute commands targeted at each wallet:
const Client = require('nix-sdk');
const wallet1 = new Client({
network: 'regtest',
wallet: 'wallet1.dat',
username: 'foo',
password: 'j1DuzF7QRUp-iSXjgewO9T_WT1Qgrtz_XWOHCMn_O-Y='
});
const wallet2 = new Client({
network: 'regtest',
wallet: 'wallet2.dat',
username: 'foo',
password: 'j1DuzF7QRUp-iSXjgewO9T_WT1Qgrtz_XWOHCMn_O-Y='
});
(async function() {
await wallet2.generate(100);
console.log(await wallet1.getBalance());
// => 0
console.log(await wallet2.getBalance());
// => 50
}());
Start the nixd
with the RPC server enabled and optionally configure a username and password:
These configuration values may also be set on the nix.conf
file of your platform installation.
By default, port 6214
is used to listen for requests in mainnet
mode, or 16214
in testnet
and regtest
modes. Use the network
property to initialize the client on the desired mode and automatically set the respective default port. You can optionally set a custom port of your choice too.
The RPC services binds to the localhost loopback network interface, so use rpcbind
to change where to bind to and rpcallowip
to whitelist source IP access.
An example nix.conf
may look like this:
server=1
par=1
rpcbind=127.0.0.1
rpcallowip=127.0.0.1
rpcport=3335
rpcuser=username1
rpcpassword=password1
rpcclienttimeout=30
rpcthreads=2
rpcworkqueue=1000
staking=0
enableaccounts=1
All RPC methods are exposed on the client interface as a camelcase'd version of those available on nixd
(see examples below). You can also issue any command you desire by using the command
command. Example for un-ghosting: nix.command('unghostamountv2', '1');
For a more complete reference about which methods are available, especially those unique to NIX, check the commands documentation on the NIX wiki.
/*
======================
used nix.conf:
server=1
par=1
rpcbind=127.0.0.1
rpcallowip=127.0.0.1
rpcport=3335
rpcuser=username1
rpcpassword=password1
rpcclienttimeout=30
rpcthreads=2
rpcworkqueue=1000
staking=0
enableaccounts=1
======================
*/
const Client = require('nix-sdk');
const client = new Client({
username: 'username1',
password: 'password1',
port: 3335,
network: 'mainnet' });
client.command('ghostamountv2', '0.1').then(value => console.log(value)); //ghost 0.1 NIX and log response
client.command('unghostamountv2', '0.1').then(value => console.log(value)); //un-ghost 0.1 NIX and log response
client.command('unghostamount', '1');
client.createRawTransaction([{ txid: '1eb590cd06127f78bf38ab4140c4cdce56ad9eb8886999eb898ddf4d3b28a91d', vout: 0 }], { 'mgnucj8nYqdrPFh2JfZSB1NmUThUGnmsqe': 0.13 });
client.sendMany('test1', { mjSk1Ny9spzU2fouzYgLqGUD8U41iR35QN: 0.1, mgnucj8nYqdrPFh2JfZSB1NmUThUGnmsqe: 0.2 }, 6, 'Example Transaction');
client.sendToAddress('mmXgiR6KAhZCyQ8ndr2BCfEq1wNG2UnyG6', 0.1, 'sendtoaddress example', 'Nemo From Example.com');
Batch requests are support by passing an array to the command
method with a method
and optionally, parameters
. The return value will be an array with all the responses.
const batch = [
{ method: 'getnewaddress', parameters: [] },
{ method: 'getnewaddress', parameters: [] }
]
new Client().command(batch).then((responses) => console.log(responses)));
// Or, using ES2015 destructuring.
new Client().command(batch).then(([firstAddress, secondAddress]) => console.log(firstAddress, secondAddress)));
Note that batched requests will only throw an error if the batch request itself cannot be processed. However, each individual response may contain an error akin to an individual request.
const batch = [
{ method: 'foobar', params: [] },
{ method: 'getnewaddress', params: [] }
]
new Client().command(batch).then(([address, error]) => console.log(address, error)));
// => `mkteeBFmGkraJaWN5WzqHCjmbQWVrPo5X3, { [RpcError: Method not found] message: 'Method not found', name: 'RpcError', code: -32601 }`.
Support for the REST interface is still experimental and the API is still subject to change. These endpoints are also unauthenticated so there are certain risks which you should be aware, specifically of leaking sensitive data of the node if not correctly protected.
Error handling is still fragile so avoid passing user input.
Start the nixd
with the REST server enabled:
These configuration values may also be set on the nix.conf
file of your platform installation. Use txindex=1
if you'd like to enable full transaction query support (note: this will take a considerable amount of time on the first run).
Unlike RPC methods which are automatically exposed on the client, REST ones are handled individually as each method has its own specificity. The following methods are supported:
- getBlockByHash
- getBlockHeadersByHash
- getBlockchainInformation
- getMemoryPoolContent
- getMemoryPoolInformation
- getTransactionByHash
- getUnspentTransactionOutputs
Given a block hash, returns a block, in binary, hex-encoded binary or JSON formats.
hash
(string): The block hash.[options]
(Object): The options object.[options.extension=json]
(string): Return in binary (bin
), hex-encoded binary (hex
) or JSON (json
) format.[callback]
(Function): An optional callback, otherwise a Promise is returned.
client.getBlockByHash('0f9188f13cb7b2c71f2a335e3a4fc328bf5beb436012afca590b1a11466e2206', { extension: 'json' });
Given a block hash, returns amount of block headers in upward direction.
hash
(string): The block hash.count
(number): The number of blocks to count in upward direction.[options]
(Object): The options object.[options.extension=json]
(string): Return in binary (bin
), hex-encoded binary (hex
) or JSON (json
) format.[callback]
(Function): An optional callback, otherwise a Promise is returned.
client.getBlockHeadersByHash('0f9188f13cb7b2c71f2a335e3a4fc328bf5beb436012afca590b1a11466e2206', 1, { extension: 'json' });
Returns various state info regarding block chain processing.
[callback]
(Function): An optional callback, otherwise a Promise is returned.
client.getBlockchainInformation([callback]);
Returns transactions in the transaction memory pool.
[callback]
(Function): An optional callback, otherwise a Promise is returned.
client.getMemoryPoolContent();
Returns various information about the transaction memory pool. Only supports JSON as output format.
- size: the number of transactions in the transaction memory pool.
- bytes: size of the transaction memory pool in bytes.
- usage: total transaction memory pool memory usage.
[callback]
(Function): An optional callback, otherwise a Promise is returned.
client.getMemoryPoolInformation();
Given a transaction hash, returns a transaction in binary, hex-encoded binary, or JSON formats.
hash
(string): The transaction hash.[options]
(Object): The options object.[options.summary=false]
(boolean): Whether to return just the transaction hash, thus saving memory.[options.extension=json]
(string): Return in binary (bin
), hex-encoded binary (hex
) or JSON (json
) format.[callback]
(Function): An optional callback, otherwise a Promise is returned.
client.getTransactionByHash('b4dd08f32be15d96b7166fd77afd18aece7480f72af6c9c7f9c5cbeb01e686fe', { extension: 'json', summary: false });
Query unspent transaction outputs (UTXO) for a given set of outpoints. See BIP64 for input and output serialisation.
outpoints
(array<Object>|Object): The outpoint to query in the format{ id: '<txid>', index: '<index>' }
.[options]
(Object): The options object.[options.extension=json]
(string): Return in binary (bin
), hex-encoded binary (hex
) or JSON (json
) format.[callback]
(Function): An optional callback, otherwise a Promise is returned.
client.getUnspentTransactionOutputs([{
id: '0f9188f13cb7b2c71f2a335e3a4fc328bf5beb436012afca590b1a11466e2206',
index: 0
}, {
id: '0f9188f13cb7b2c71f2a335e3a4fc328bf5beb436012afca590b1a11466e2206',
index: 1
}], { extension: 'json' }, [callback])
This client supports SSL out of the box. Simply pass the SSL public certificate to the client and optionally disable strict SSL checking which will bypass SSL validation (the connection is still encrypted but the server it is connecting to may not be trusted). This is, of course, discouraged unless for testing purposes when using something like self-signed certificates.
Please note that the following procedure should only be used for testing purposes.
Generate an self-signed certificate together with an unprotected private key:
openssl req -x509 -newkey rsa:4096 -keyout key.pem -out cert.pem -days 3650 -nodes
You must use stunnel
(brew install stunnel
or sudo apt-get install stunnel4
) or an HTTPS reverse proxy to configure SSL since the built-in support for SSL has been removed. The trade off with stunnel
is performance and simplicity versus features, as it lacks more powerful capacities such as Basic Authentication and caching which are standard in reverse proxies.
You can use stunnel
by configuring stunnel.conf
with the following service requirements:
[nix]
accept = 28332
connect = 18332
cert = /etc/ssl/nixd/cert.pem
key = /etc/ssl/nixd/key.pem
The key
option may be omitted if you concatenating your private and public certificates into a single stunnel.pem
file.
On some versions of stunnel
it is also possible to start a service using command line arguments. The equivalent would be:
stunnel -d 28332 -r 127.0.0.1:18332 -p stunnel.pem -P ''
Then pass the public certificate to the client:
const Client = require('nix-sdk');
const fs = require('fs');
const client = new Client({
agentOptions: {
ca: fs.readFileSync('/etc/ssl/nixd/cert.pem')
},
port: 28332,
ssl: true
});
By default, all requests made with nix-sdk
are logged using uphold/debugnyan with nix-sdk
as the logging namespace.
Please note that all sensitive data is obfuscated before calling the logger.
Example output defining the environment variable DEBUG=nix-sdk
:
const client = new Client();
client.getTransactionByHash('b4dd08f32be15d96b7166fd77afd18aece7480f72af6c9c7f9c5cbeb01e686fe');
// {
// "name": "nix-sdk",
// "hostname": "localhost",
// "pid": 57908,
// "level": 20,
// "request": {
// "headers": {
// "host": "localhost:8332",
// "accept": "application/json"
// },
// "id": "82cea4e5-2c85-4284-b9ec-e5876c84e67c",
// "method": "GET",
// "type": "request",
// "uri": "http://localhost:8332/rest/tx/b4dd08f32be15d96b7166fd77afd18aece7480f72af6c9c7f9c5cbeb01e686fe.json"
// },
// "msg": "Making request 82cea4e5-2c85-4284-b9ec-e5876c84e67c to GET http://localhost:8332/rest/tx/b4dd08f32be15d96b7166fd77afd18aece7480f72af6c9c7f9c5cbeb01e686fe.json",
// "time": "2017-02-07T14:40:35.020Z",
// "v": 0
// }
A custom logger can be passed via the logger
option and it should implement bunyan's log levels.
Currently the test suite is tailored for Docker (including docker-compose
) due to the multitude of different nixd
configurations that are required in order to get the test suite passing.
To test using a local installation of node.js
but with dependencies (e.g. nixd
) running inside Docker:
npm run dependencies
npm test
To test using Docker exclusively (similarly to what is done in Travis CI):
npm run testdocker
npm version [<newversion> | major | minor | patch] -m "Release %s"
MIT