lamassu-server/lib/routes.js
2016-02-08 16:02:03 +02:00

220 lines
5.6 KiB
JavaScript

'use strict';
var logger = require('./logger');
var mock = false;
var plugins;
var lamassuConfig;
module.exports = {
init: init,
getFingerprint: getFingerprint
};
// Make sure these are higher than polling interval
// or there will be a lot of errors
var STALE_TICKER = 180000;
var STALE_BALANCE = 180000;
function poll(req, res) {
var rateRec = plugins.getDeviceRate();
var balanceRec = plugins.getBalance();
logger.debug('poll request from: %s', getFingerprint(req));
// `rateRec` and `balanceRec` are both objects, so there's no danger
// of misinterpreting rate or balance === 0 as 'Server initializing'.
if (!rateRec || !balanceRec) {
return res.json({err: 'Server initializing'});
}
var now = Date.now();
if (now - rateRec.timestamp > STALE_TICKER) {
return res.json({err: 'Stale ticker'});
}
if (now - balanceRec.timestamp > STALE_BALANCE) {
return res.json({err: 'Stale balance'});
}
var rate = rateRec.rates.ask;
var fiatRate = rateRec.rates.bid || rate;
if (rate === null) return res.json({err: 'No rate available'});
if (!fiatRate)
logger.warn('No bid rate, using ask rate');
var fiatBalance = plugins.fiatBalance();
if (fiatBalance === null) {
logger.warn('No balance available.');
return res.json({err: 'No balance available'});
}
var config = plugins.getConfig();
var settings = config.exchanges.settings;
var complianceSettings = settings.compliance;
var fiatCommission = settings.fiatCommission || settings.commission;
plugins.pollQueries(session(req), function(err, results) {
if (err) return logger.error(err);
var cartridges = results.cartridges;
var response = {
err: null,
rate: rate * settings.commission,
fiatRate: fiatRate / fiatCommission,
fiat: fiatBalance,
locale: config.brain.locale,
txLimit: parseInt(complianceSettings.maximum.limit, 10),
idVerificationEnabled: complianceSettings.idVerificationEnabled,
cartridges: cartridges,
twoWayMode: cartridges ? true : false,
zeroConfLimit: settings.zeroConfLimit,
fiatTxLimit: settings.fiatTxLimit
};
if (response.idVerificationEnabled)
response.idVerificationLimit = complianceSettings.idVerificationLimit;
res.json(response);
});
}
function trade(req, res) {
plugins.trade(session(req), req.body, function(err) {
var statusCode = err ? 500 : 201;
res.json(statusCode, {err: err});
});
}
function stateChange(req, res) {
plugins.stateChange(session(req), req.body, function(err) {
res.json(200)
})
}
function send(req, res) {
plugins.sendBitcoins(session(req), req.body, function(err, status) {
// TODO: use status.statusCode here after confirming machine compatibility
// FIX: (joshm) set txHash to status.txId instead of previous status.txHash which wasn't being set
// Need to clean up txHash vs txId
res.json({
errType: err && err.name,
err: err && err.message,
txHash: status && status.txHash,
txId: status && status.txId
});
});
}
function cashOut(req, res) {
logger.info({tx: req.body, cmd: 'cashOut'});
plugins.cashOut(session(req), req.body, function(err, bitcoinAddress) {
if (err) logger.error(err);
res.json({
err: err && err.message,
errType: err && err.name,
bitcoinAddress: bitcoinAddress
});
});
}
function dispenseAck(req, res) {
plugins.dispenseAck(session(req), req.body);
res.json(200);
}
function deviceEvent(req, res) {
plugins.logEvent(session(req), req.body);
res.json({err: null});
}
function verifyUser(req, res) {
if (mock) return res.json({success: true});
plugins.verifyUser(req.body, function (err, idResult) {
if (err) {
logger.error(err);
return res.json({err: 'Verification failed'});
}
res.json(idResult);
});
}
function verifyTx(req, res) {
if (mock) return res.json({success: true});
plugins.verifyTx(req.body, function (err, idResult) {
if (err) {
logger.error(err);
return res.json({err: 'Verification failed'});
}
res.json(idResult);
});
}
function pair(req, res) {
var token = req.body.token;
var name = req.body.name;
lamassuConfig.pair(
token,
getFingerprint(req),
name,
function(err) {
if (err) return res.json(500, { err: err.message });
res.json(200);
}
);
}
function raqia(req, res) {
var raqiaCreds;
try {
var raqiaRec = require('../raqia.json');
raqiaCreds = raqiaRec[getFingerprint(req)].apiKeys[0];
} catch(ex) {
raqiaCreds = null;
}
res.json(raqiaCreds || {});
}
function init(localConfig) {
lamassuConfig = localConfig.lamassuConfig;
plugins = localConfig.plugins;
mock = localConfig.mock;
var authMiddleware = localConfig.authMiddleware;
var reloadConfigMiddleware = localConfig.reloadConfigMiddleware;
var app = localConfig.app;
app.get('/poll', authMiddleware, reloadConfigMiddleware, poll);
app.post('/trade', authMiddleware, trade);
app.post('/state', authMiddleware, stateChange);
app.post('/send', authMiddleware, send);
app.post('/cash_out', authMiddleware, cashOut);
app.post('/dispense_ack', authMiddleware, dispenseAck);
app.post('/event', authMiddleware, deviceEvent);
app.post('/verify_user', authMiddleware, verifyUser);
app.post('/verify_transaction', authMiddleware, verifyTx);
app.post('/pair', pair);
app.get('/raqia', raqia);
return app;
}
function session(req) {
return {fingerprint: getFingerprint(req), id: req.get('session-id')};
}
function getFingerprint(req) {
return (typeof req.connection.getPeerCertificate === 'function' &&
req.connection.getPeerCertificate().fingerprint) || 'unknown';
}