autobounty/index.js

123 lines
3.7 KiB
JavaScript
Raw Normal View History

2017-06-10 20:37:15 +05:30
/*
2018-01-22 12:24:19 +01:00
* Bot that receives a POST request (from a GitHub issue comment webhook)
* and in case it's a comment that has "@autobounty <decimal> <currency>"
* awards that bounty to the address posted earlier in the thread (by the
* commiteth bot).
* TODO tests
* REVIEW parsing, non-persisting storage of addresses, hardcoded string length.
* Depends on commiteth version as of 2017-06-10.
*/
2018-01-17 16:48:56 +00:00
const config = require('./config');
const bot = require('./bot');
2018-02-10 19:55:24 +01:00
const crypto = require('crypto');
2018-01-17 16:48:56 +00:00
2017-06-09 12:15:53 +02:00
var express = require('express'),
2018-01-23 16:36:34 +01:00
cors = require('cors'),
helmet = require('helmet'),
app = express(),
bodyParser = require('body-parser'),
jsonParser = bodyParser.json();
2017-03-07 12:24:01 +01:00
app.use(cors());
2018-01-19 11:42:58 +01:00
app.use(helmet());
2018-01-17 16:48:56 +00:00
// Receive a POST request at the url specified by an env. var.
2018-01-23 16:36:34 +01:00
app.post(`${config.urlEndpoint}`, jsonParser, function (req, res, next) {
2018-02-10 19:55:24 +01:00
if (!req.body || !req.body.action) {
return res.sendStatus(400);
} else if (!bot.needsFunding(req)) {
return res.sendStatus(204);
}
2018-02-10 19:55:24 +01:00
validation = validateRequest(req);
2018-02-10 19:55:24 +01:00
if (validation.correct) {
setTimeout(() => {
processRequest(req)
.then(() => {
2018-02-13 11:43:28 +01:00
bot.info('issue well funded: ' + req.body.issue.url);
2018-02-10 19:55:24 +01:00
})
.catch((err) => {
2018-02-13 11:43:28 +01:00
bot.error('Error processing request: ' + req.body.issue.url);
2018-02-27 11:24:56 +01:00
bot.error('Error: ' + err);
bot.error('Dump: ', req.body);
2018-02-10 19:55:24 +01:00
});
}, config.delayInMiliSeconds);
} else {
2018-02-13 11:43:28 +01:00
bot.error('Error validating issue: ' + req.body.issue.url);
2018-02-27 11:24:56 +01:00
bot.error('Error: ' + validation.error);
2018-02-10 19:55:24 +01:00
}
return res.sendStatus(200);
});
2018-02-10 19:55:24 +01:00
const validateRequest = function (req) {
validation = {correct: false, error: ''};
webhookSecret = process.env.WEBHOOK_SECRET;
if(!webhookSecret) {
validation.error = 'Github Webhook Secret key not found. ' +
'Please set env variable WEBHOOK_SECRET to github\'s webhook secret value';
} else {
2018-02-13 11:43:28 +01:00
const blob = JSON.stringify(req.body);
2018-02-10 19:55:24 +01:00
const hmac = crypto.createHmac('sha1', webhookSecret);
const ourSignature = `sha1=${hmac.update(blob).digest('hex')}`;
const theirSignature = req.get('X-Hub-Signature');
const bufferA = Buffer.from(ourSignature, 'utf8');
const bufferB = Buffer.from(theirSignature, 'utf8');
const safe = crypto.timingSafeEqual(bufferA, bufferB);
if (safe) {
validation.correct = true;
} else {
validation.error = 'Invalid signature. Check that WEBHOOK_SECRET ' +
'env variable matches github\'s webhook secret value';
}
}
return validation;
}
const processRequest = function (req) {
2018-02-27 11:10:17 +01:00
// const wallet = bot.wallet;
2018-02-10 19:55:24 +01:00
const from = config.sourceAddress;
const to = bot.getAddress(req);
// Asynchronous requests for Gas Price and Amount
const amountPromise = bot.getAmount(req);
const gasPricePromise = bot.getGasPrice();
return new Promise((resolve, reject) => {
2018-01-25 13:02:47 +01:00
Promise.all([amountPromise, gasPricePromise])
.then(function (results) {
let amount = results[0];
let gasPrice = results[1];
2018-02-13 13:05:39 +01:00
bot.sendTransaction(to, amount, gasPrice)
.then(function () {
resolve();
})
.catch(function (err) {
reject(err);
});
2018-01-23 16:36:34 +01:00
})
.catch(function (err) {
reject(err);
});
2018-01-25 13:02:47 +01:00
});
}
2017-03-07 12:24:01 +01:00
const port = process.env.PORT || 8181
2018-01-23 16:36:34 +01:00
app.listen(port, function () {
2018-02-13 11:43:28 +01:00
bot.info('Autobounty listening on port', port);
2017-06-10 23:54:34 +05:30
});