UNPKG

ripple-rest-dinex

Version:

A RESTful API for submitting payments and monitoring accounts on the Ripple network.

263 lines (231 loc) 8.27 kB
/* globals Promise: true */ /* eslint-disable valid-jsdoc */ 'use strict'; var _ = require('lodash'); var Promise = require('bluebird'); var ripple = require('ripple-lib'); var transactions = require('./transactions.js'); var SubmitTransactionHooks = require('./lib/submit_transaction_hooks.js'); var utils = require('./lib/utils'); var errors = require('./lib/errors.js'); var validator = require('./lib/schema-validator'); var TxToRestConverter = require('./lib/tx-to-rest-converter.js'); var validate = require('./lib/validate'); var TrustSetFlags = { SetAuth: {name: 'authorized', set: 'SetAuth'}, ClearNoRipple: {name: 'account_allows_rippling', set: 'ClearNoRipple', unset: 'NoRipple'}, SetFreeze: {name: 'account_trustline_frozen', set: 'SetFreeze', unset: 'ClearFreeze'} }; var DefaultPageLimit = 200; /** * Retrieves all trustlines for a given account * * Notes: * In order to use paging, you must provide at least ledger as a query parameter * Additionally, any limit lower than 10 will be bumped up to 10. * * @url * @param {String} request.params.account - account to retrieve trustlines for * * @query * @param {String ISO 4217 Currency Code} [request.query.currency] * - only request trustlines with given currency * @param {RippleAddress} [request.query.counterparty] * - only request trustlines with given counterparty * @param {String} [request.query.marker] - start position in response paging * @param {Number String} [request.query.limit] - max results per response * @param {Number String} [request.query.ledger] - identifier * */ function getTrustLines(account, options, callback) { if (validate.fail([ validate.account(account), validate.currency(options.currency, true), validate.counterparty(options.counterparty, true), validate.ledger(options.ledger, true), validate.limit(options.limit, true), validate.paging(options) ], callback)) { return; } var self = this; var currencyRE = new RegExp(options.currency ? ('^' + options.currency.toUpperCase() + '$') : /./); function getAccountLines(prevResult) { var isAggregate = options.limit === 'all'; if (prevResult && (!isAggregate || !prevResult.marker)) { return Promise.resolve(prevResult); } var promise = new Promise(function(resolve, reject) { var accountLinesRequest; var marker; var ledger; var limit; if (prevResult) { marker = prevResult.marker; limit = prevResult.limit; ledger = prevResult.ledger_index; } else { marker = options.marker; limit = validator.isValid(options.limit, 'UINT32') ? Number(options.limit) : DefaultPageLimit; ledger = utils.parseLedger(options.ledger); } accountLinesRequest = self.remote.requestAccountLines({ account: account, marker: marker, limit: limit, ledger: ledger }); if (options.counterparty) { accountLinesRequest.message.peer = options.counterparty; } accountLinesRequest.once('error', reject); accountLinesRequest.once('success', function(nextResult) { var lines = [ ]; nextResult.lines.forEach(function(line) { if (!currencyRE.test(line.currency)) { return; } lines.push({ account: account, counterparty: line.account, currency: line.currency, limit: line.limit, reciprocated_limit: line.limit_peer, account_allows_rippling: line.no_ripple ? !line.no_ripple : true, counterparty_allows_rippling: line.no_ripple_peer ? !line.no_ripple_peer : true, account_trustline_frozen: line.freeze ? line.freeze : false, counterparty_trustline_frozen: line.freeze_peer ? line.freeze_peer : false }); }); nextResult.lines = prevResult ? prevResult.lines.concat(lines) : lines; resolve([nextResult]); }); accountLinesRequest.request(); }); return promise.spread(getAccountLines); } function respondWithTrustlines(result) { var promise = new Promise(function (resolve) { var trustlines = {}; if (result.marker) { trustlines.marker = result.marker; } trustlines.limit = result.limit; trustlines.ledger = result.ledger_index; trustlines.validated = result.validated; trustlines.trustlines = result.lines; resolve(callback(null, trustlines)); }); return promise; } getAccountLines() .then(respondWithTrustlines) .catch(callback); } /** * Grant a trustline to a counterparty * * @body * @param {Trustline} request.body.trustline * @param {String} request.body.secret * * @query * @param {String "true"|"false"} request.query.validated Used to force request * to wait until rippled has finished validating the submitted transaction * */ function addTrustLine(account, trustline, secret, options, callback) { var params = { secret: secret, validated: options.validated }; function validateParams(_callback) { if (!ripple.UInt160.is_valid(account)) { return _callback(new errors.InvalidRequestError( 'Parameter is not a valid Ripple address: account')); } if (typeof trustline !== 'object') { return _callback(new errors.InvalidRequestError( 'Parameter missing: trustline')); } if (_.isUndefined(trustline.limit)) { return _callback(new errors.InvalidRequestError( 'Parameter missing: trustline.limit')); } trustline.limit = String(trustline.limit); // TODO: does not belong here if (isNaN(trustline.limit)) { return _callback(new errors.InvalidRequestError( 'Parameter is not a number: trustline.limit')); } if (!trustline.currency) { return _callback(new errors.InvalidRequestError( 'Parameter missing: trustline.currency')); } if (!validator.isValid(trustline.currency, 'Currency')) { return _callback(new errors.InvalidRequestError( 'Parameter is not a valid currency: trustline.currency')); } if (!trustline.counterparty) { return _callback(new errors.InvalidRequestError( 'Parameter missing: trustline.counterparty')); } if (!ripple.UInt160.is_valid(trustline.counterparty)) { return _callback(new errors.InvalidRequestError( 'Parameter is not a Ripple address: trustline.counterparty')); } if (!/^(undefined|number)$/.test(typeof trustline.quality_in)) { return _callback(new errors.InvalidRequestError( 'Parameter must be a number: trustline.quality_in')); } if (!/^(undefined|number)$/.test(typeof trustline.quality_out)) { return _callback(new errors.InvalidRequestError( 'Parameter must be a number: trustline.quality_out')); } if (!/^(undefined|boolean)$/.test( typeof trustline.account_allows_rippling)) { return _callback(new errors.InvalidRequestError( 'Parameter must be a boolean: trustline.allow_rippling')); } _callback(); } function setTransactionParameters(transaction) { var limit = [ trustline.limit, trustline.currency, trustline.counterparty ].join('/'); transaction.trustSet(account, limit); transaction.secret(secret); if (typeof trustline.quality_in === 'number') { transaction.tx_json.QualityIn = trustline.quality_in; } if (typeof trustline.quality_out === 'number') { transaction.tx_json.QualityOut = trustline.quality_out; } transactions.setTransactionBitFlags(transaction, { input: trustline, flags: TrustSetFlags, clear_setting: '' }); } var hooks = { validateParams: validateParams, formatTransactionResponse: TxToRestConverter.parseTrustResponseFromTx, setTransactionParameters: setTransactionParameters }; transactions.submit(this, params, new SubmitTransactionHooks(hooks), function(err, trustlineResult) { if (err) { return callback(err); } callback(null, trustlineResult); }); } module.exports.get = getTrustLines; module.exports.add = addTrustLine;