diff --git a/docs/guide/jsonrpc.md b/docs/guide/jsonrpc.md deleted file mode 100644 index 454cb87..0000000 --- a/docs/guide/jsonrpc.md +++ /dev/null @@ -1,44 +0,0 @@ -title: JSON-RPC -description: A simple interface to connect and make RPC calls to bitcoind. ---- -# JSON-RPC - -## Description - -Bitcoind provides a direct interface to the bitcoin network and it also exposes a `JSON-RPC` API. This class will connect to a local instance of a bitcoind server and make simple or batch RPC calls to it. - -## Connection to bitcoind - -First you will need a running instance of bitcoind, setting up a username and password to connect with it. For more information about running bitcoind please refer to the [official documentation](https://en.bitcoin.it/wiki/Running_Bitcoin). - -The code for creating and configuring an instance of the RPC client looks like this: - -``` -var bitcore = require('bitcore'); -var RPC = bitcore.transport.RPC; - -var client = new RPC('username', 'password', { - host: 'localhost', - port: 18332, - secure: false, - disableAgent: true, - rejectUnauthorized: true -}); -``` - - -## Examples - -For more information please refer to the [API reference](https://en.bitcoin.it/wiki/API_reference_%28JSON-RPC%29). - -``` -var bitcore = require('bitcore'); -var blockHash = '0000000000000000045d581af7fa3b6110266ece8131424d95bf490af828be1c'; - -var client = new bitcore.transport.RPC('username', 'password'); - -client.getBlock(blockHash, function(err, block) { - // do something with the block -}); - -``` diff --git a/lib/transport/index.js b/lib/transport/index.js index 185732e..7d25662 100644 --- a/lib/transport/index.js +++ b/lib/transport/index.js @@ -5,6 +5,5 @@ module.exports = { explorers: require('./explorers'), Messages: require('./messages'), Peer: require('./peer'), - Pool: require('./pool'), - RPC: require('./rpc') + Pool: require('./pool') }; diff --git a/lib/transport/rpc.js b/lib/transport/rpc.js deleted file mode 100644 index b297ae5..0000000 --- a/lib/transport/rpc.js +++ /dev/null @@ -1,249 +0,0 @@ -'use strict'; - -var http = require('http'); -var https = require('https'); - -/** - * A JSON RPC client for bitcoind. An instances of RPC connects to a bitcoind - * server and enables simple and batch RPC calls. - * - * @example - * ```javascript - * - * var client = new RPC('user', 'pass'); - * client.getInfo(function(err, info) { - * // do something with the info - * }); - * ``` - * - * @param {String} user - username used to connect bitcoind - * @param {String} password - password used to connect bitcoind - * @param {Object} opts - Connection options: host, port, secure, disableAgent, rejectUnauthorized - * @returns {RPC} - * @constructor - */ -function RPC(user, password, opts) { - if (!(this instanceof RPC)) { - return new RPC(user, password, opts); - } - - this.user = user; - this.pass = password; - - opts = opts || {}; - this.host = opts.host || '127.0.0.1'; - this.port = opts.port || 8332; - - this.secure = typeof opts.secure === 'undefined' ? true : opts.secure; - this._client = opts.secure ? https : http; - - this.batchedCalls = null; - this.disableAgent = opts.disableAgent || false; - this.rejectUnauthorized = opts.rejectUnauthorized || false; -} - -/** - * Allows to excecute RPC calls in batch. - * - * @param {Function} batchCallback - Function that makes all calls to be excecuted in bach - * @param {Function} resultCallbak - Function to be called on result - */ -RPC.prototype.batch = function(batchCallback, resultCallback) { - this.batchedCalls = []; - batchCallback(); - this._request(this.batchedCalls, resultCallback); - this.batchedCalls = null; -} - -/** - * Internal function to make an RPC call - * - * @param {Object} request - Object to be serialized and sent to bitcoind - * @param {Function} callbak - Function to be called on result - */ -RPC.prototype._request = function(request, callback) { - var self = this; - - var request = JSON.stringify(request); - var auth = Buffer(self.user + ':' + self.pass).toString('base64'); - - var options = { - host: self.host, - path: '/', - method: 'POST', - port: self.port, - rejectUnauthorized: self.rejectUnauthorized, - agent: self.disableAgent ? false : undefined - }; - - var req = this._client.request(options, function(res) { - var buf = ''; - res.on('data', function(data) { - buf += data; - }); - - res.on('end', function() { - if (res.statusCode == 401) { - var error = new Error('bitcoin JSON-RPC connection rejected: 401 unauthorized'); - return callback(error); - } - - if (res.statusCode == 403) { - var error = new Error('bitcoin JSON-RPC connection rejected: 403 forbidden'); - return callback(error); - } - - try { - var parsedBuf = JSON.parse(buf); - } catch (e) { - return callback(e); - } - - callback(parsedBuf.error, parsedBuf); - }); - }); - - req.on('error', function(e) { - var err = new Error('Could not connect to bitcoin via RPC at host: ' + self.host + ' port: ' + self.port + ' Error: ' + e.message); - callback(err); - }); - - req.setHeader('Content-Length', request.length); - req.setHeader('Content-Type', 'application/json'); - req.setHeader('Authorization', 'Basic ' + auth); - req.write(request); - req.end(); -}; - -var callspec = { - addMultiSigAddress: '', - addNode: '', - backupWallet: '', - createMultiSig: '', - createRawTransaction: '', - decodeRawTransaction: '', - dumpPrivKey: '', - encryptWallet: '', - getAccount: '', - getAccountAddress: 'str', - getAddedNodeInfo: '', - getAddressesByAccount: '', - getBalance: 'str int', - getBestBlockHash: '', - getBlock: '', - getBlockCount: '', - getBlockHash: 'int', - getBlockNumber: '', - getBlockTemplate: '', - getConnectionCount: '', - getDifficulty: '', - getGenerate: '', - getHashesPerSec: '', - getInfo: '', - getMemoryPool: '', - getMiningInfo: '', - getNewAddress: '', - getPeerInfo: '', - getRawMemPool: '', - getRawTransaction: 'str int', - getReceivedByAccount: 'str int', - getReceivedByAddress: 'str int', - getTransaction: '', - getTxOut: 'str int bool', - getTxOutSetInfo: '', - getWork: '', - help: '', - importAddress: 'str str bool', - importPrivKey: 'str str bool', - keyPoolRefill: '', - listAccounts: 'int', - listAddressGroupings: '', - listReceivedByAccount: 'int bool', - listReceivedByAddress: 'int bool', - listSinceBlock: 'str int', - listTransactions: 'str int int', - listUnspent: 'int int', - listLockUnspent: 'bool', - lockUnspent: '', - move: 'str str float int str', - sendFrom: 'str str float int str str', - sendMany: 'str str int str', //not sure this is will work - sendRawTransaction: '', - sendToAddress: 'str float str str', - setAccount: '', - setGenerate: 'bool int', - setTxFee: 'float', - signMessage: '', - signRawTransaction: '', - stop: '', - submitBlock: '', - validateAddress: '', - verifyMessage: '', - walletLock: '', - walletPassPhrase: 'string int', - walletPassphraseChange: '', -}; - - -var slice = function(arr, start, end) { - return Array.prototype.slice.call(arr, start, end); -}; - -function generateRPCMethods(constructor, apiCalls) { - function createRPCMethod(methodName, argMap) { - return function() { - var limit = arguments.length - 1; - if (this.batchedCalls) var limit = arguments.length; - for (var i = 0; i < limit; i++) { - if (argMap[i]) arguments[i] = argMap[i](arguments[i]); - }; - if (this.batchedCalls) { - this.batchedCalls.push({ - jsonrpc: '2.0', - method: methodName, - params: slice(arguments) - }); - } else { - this._request({ - method: methodName, - params: slice(arguments, 0, arguments.length - 1) - }, arguments[arguments.length - 1]); - } - }; - }; - - var types = { - str: function(arg) { - return arg.toString(); - }, - int: function(arg) { - return parseFloat(arg); - }, - float: function(arg) { - return parseFloat(arg); - }, - bool: function(arg) { - return (arg === true || arg == '1' || arg == 'true' || arg.toString().toLowerCase() == 'true'); - }, - }; - - for (var k in apiCalls) { - if (apiCalls.hasOwnProperty(k)) { - var spec = apiCalls[k].split(' '); - for (var i = 0; i < spec.length; i++) { - if (types[spec[i]]) { - spec[i] = types[spec[i]]; - } else { - spec[i] = types.string; - } - } - var methodName = k.toLowerCase(); - constructor.prototype[k] = createRPCMethod(methodName, spec); - constructor.prototype[methodName] = constructor.prototype[k]; - } - } -} - -generateRPCMethods(RPC, callspec); - -module.exports = RPC; diff --git a/test/transport/rpc.js b/test/transport/rpc.js deleted file mode 100644 index 0b29266..0000000 --- a/test/transport/rpc.js +++ /dev/null @@ -1,62 +0,0 @@ -'use strict'; - -var chai = require('chai'); -var should = chai.should(); - -var bitcore = require('../..'); -var RPC = bitcore.transport.RPC; - -describe('RPC', function() { - it('should be able to create instance', function() { - var client = new RPC('user', 'pass'); - should.exist(client); - }); - - it('should set default config', function() { - var client = new RPC('user', 'pass'); - client.user.should.be.equal('user'); - client.pass.should.be.equal('pass'); - - client.host.should.be.equal('127.0.0.1'); - client.port.should.be.equal(8332); - client.secure.should.be.equal(true); - client.disableAgent.should.be.equal(false); - client.rejectUnauthorized.should.be.equal(false); - }); - - it('should allow setting custom host and port', function() { - var client = new RPC('user', 'pass', { - host: 'localhost', - port: 18332 - }); - - client.host.should.be.equal('localhost'); - client.port.should.be.equal(18332); - }); - - it('should honor request options', function() { - var client = new RPC('user', 'pass', { - host: 'localhost', - port: 18332, - rejectUnauthorized: true, - disableAgent: true - }); - - client._client = {}; - client._client.request = function(options, callback) { - options.host.should.be.equal('localhost'); - options.port.should.be.equal(18332); - options.rejectUnauthorized.should.be.equal(true); - options.agent.should.be.false; - return { - on: function() {}, - setHeader: function() {}, - write: function() {}, - end: function() {} - }; - }; - - client._request({}, function() {}); - }); - -});