lamassu-server/lib/plugins/exchange/ccxt/ccxt.js
2021-05-25 10:53:54 +01:00

57 lines
2.4 KiB
JavaScript

var ccxt = require('ccxt')
const coinUtils = require('../../../coin-utils')
const _ = require('lodash/fp')
const common = require('../../common/ccxt')
function trade (side, account, cryptoAtoms, fiatCode, cryptoCode, exchangeName) {
try {
const exchange = setUpExchange(account, exchangeName)
const symbol = common.verifyCurrencies(exchangeName, fiatCode, cryptoCode)
const amount = coinUtils.toUnit(cryptoAtoms, cryptoCode).toFixed(8)
if (exchangeName === 'itbit') {
return exchange.fetchOrderBook(symbol)
.then(orderBook => {
return exchange.createOrder(symbol, 'limit', side, amount, calculatePrice(side, amount, orderBook), { walletId: account.walletId })
})
}
return exchange.createOrder(symbol, 'market', side, amount)
} catch (e) {
return Promise.reject(e)
}
}
function setUpExchange (account, exchangeName) {
// map given credentials to cctx properties
if (!_.includes(exchangeName, ccxt.exchanges)) {
throw new Error(`Exchange ${exchangeName} not supported by ccxt.`)
}
switch (exchangeName) {
case 'itbit':
if (!account.clientKey || !account.clientSecret || !account.userId || !account.walletId)
throw new Error('Must provide user ID, wallet ID, client key, and client secret')
return new ccxt[exchangeName](_.mapKeys((key) => { return key === 'clientKey' ? 'apiKey' : key === 'clientSecret' ? 'secret' : key === 'userId' ? 'uid' : key }, _.omit(['walletId'], account)))
case 'kraken':
if (!account.apiKey || !account.privateKey)
throw new Error('Must provide key and private key')
return new ccxt[exchangeName](_.mapKeys((key) => { return key === 'privateKey' ? 'secret' : key }, account))
case 'bitstamp':
if (!account.key || !account.secret || !account.clientId)
throw new Error('Must provide key, secret and client ID')
return new ccxt[exchangeName](_.mapKeys((key) => { return key === 'key' ? 'apiKey' : key === 'clientId' ? 'uid' : key }, account))
default:
throw new Error(`Exchange ${exchangeName} not supported.`)
}
}
function calculatePrice (side, amount, orderBook) {
const book = side === 'buy' ? 'asks' : 'bids'
let collected = 0.0
for (const entry of orderBook[book]) {
collected += parseFloat(entry[1])
if (collected >= amount) return parseFloat(entry[0])
}
throw new Error('Insufficient market depth')
}
module.exports = { trade }