mempool/backend/src/index.ts

206 lines
8.5 KiB
TypeScript
Raw Normal View History

2020-06-07 12:30:32 +02:00
import { Express, Request, Response, NextFunction } from 'express';
2019-07-21 16:59:47 +02:00
import * as express from 'express';
import * as http from 'http';
import * as https from 'https';
import * as WebSocket from 'ws';
2020-09-21 22:52:54 +02:00
import * as cluster from 'cluster';
import axios from 'axios';
2019-07-21 16:59:47 +02:00
2020-07-22 19:04:29 +02:00
import { checkDbConnection } from './database';
import config from './config';
import routes from './routes';
2019-07-21 16:59:47 +02:00
import blocks from './api/blocks';
import memPool from './api/mempool';
import diskCache from './api/disk-cache';
2019-07-21 16:59:47 +02:00
import statistics from './api/statistics';
import websocketHandler from './api/websocket-handler';
2019-07-21 16:59:47 +02:00
import fiatConversion from './api/fiat-conversion';
2020-09-10 09:46:23 +02:00
import bisq from './api/bisq/bisq';
import bisqMarkets from './api/bisq/markets';
import donations from './api/donations';
import logger from './logger';
import backendInfo from './api/backend-info';
2019-07-21 16:59:47 +02:00
class Server {
2020-09-21 22:52:54 +02:00
private wss: WebSocket.Server | undefined;
private server: https.Server | http.Server | undefined;
private app: Express;
private retryOnElectrsErrorAfterSeconds = 5;
2019-07-21 16:59:47 +02:00
constructor() {
this.app = express();
if (!config.MEMPOOL.SPAWN_CLUSTER_PROCS) {
2020-09-21 22:52:54 +02:00
this.startServer();
return;
}
if (cluster.isMaster) {
logger.notice(`Mempool Server (Master) is running on port ${config.MEMPOOL.HTTP_PORT} (${backendInfo.getShortCommitHash()})`);
2020-09-21 22:52:54 +02:00
const numCPUs = config.MEMPOOL.SPAWN_CLUSTER_PROCS;
2020-09-21 22:52:54 +02:00
for (let i = 0; i < numCPUs; i++) {
2020-09-29 19:25:43 +02:00
const env = { workerId: i };
const worker = cluster.fork(env);
worker.process['env'] = env;
2020-09-21 22:52:54 +02:00
}
cluster.on('exit', (worker, code, signal) => {
2020-09-29 19:25:43 +02:00
const workerId = worker.process['env'].workerId;
logger.warn(`Mempool Worker PID #${worker.process.pid} workerId: ${workerId} died. Restarting in 10 seconds... ${signal || code}`);
2020-09-29 19:25:43 +02:00
setTimeout(() => {
const env = { workerId: workerId };
const newWorker = cluster.fork(env);
newWorker.process['env'] = env;
}, 10000);
2020-09-21 22:52:54 +02:00
});
} else {
this.startServer(true);
}
}
startServer(worker = false) {
2019-07-21 16:59:47 +02:00
this.app
2020-06-07 12:30:32 +02:00
.use((req: Request, res: Response, next: NextFunction) => {
2019-07-21 16:59:47 +02:00
res.setHeader('Access-Control-Allow-Origin', '*');
next();
})
.use(express.urlencoded({ extended: true }))
.use(express.json());
this.server = http.createServer(this.app);
this.wss = new WebSocket.Server({ server: this.server });
2019-07-21 16:59:47 +02:00
if (config.DATABASE.ENABLED) {
2020-07-22 19:04:29 +02:00
checkDbConnection();
}
if (config.STATISTICS.ENABLED && config.DATABASE.ENABLED) {
2020-07-22 19:04:29 +02:00
statistics.startStatistics();
}
this.setUpHttpApiRoutes();
2019-07-21 16:59:47 +02:00
this.setUpWebsocketHandling();
this.runMainUpdateLoop();
2019-07-21 16:59:47 +02:00
fiatConversion.startService();
diskCache.loadMempoolCache();
2019-07-21 16:59:47 +02:00
if (config.BISQ_BLOCKS.ENABLED) {
bisq.startBisqService();
2020-07-14 16:26:02 +02:00
bisq.setPriceCallbackFunction((price) => websocketHandler.setExtraInitProperties('bsq-price', price));
blocks.setNewBlockCallback(bisq.handleNewBitcoinBlock.bind(bisq));
}
if (config.BISQ_MARKETS.ENABLED) {
2020-09-10 09:46:23 +02:00
bisqMarkets.startBisqService();
}
this.server.listen(config.MEMPOOL.HTTP_PORT, () => {
2020-09-21 22:52:54 +02:00
if (worker) {
logger.info(`Mempool Server worker #${process.pid} started`);
2020-09-21 22:52:54 +02:00
} else {
logger.notice(`Mempool Server is running on port ${config.MEMPOOL.HTTP_PORT} (${backendInfo.getShortCommitHash()})`);
2020-09-21 22:52:54 +02:00
}
2019-07-21 16:59:47 +02:00
});
}
async runMainUpdateLoop() {
try {
await memPool.$updateMemPoolInfo();
await blocks.$updateBlocks();
await memPool.$updateMempool();
setTimeout(this.runMainUpdateLoop.bind(this), config.ELECTRS.POLL_RATE_MS);
this.retryOnElectrsErrorAfterSeconds = 5;
} catch (e) {
const loggerMsg = `runMainLoop error: ${(e.message || e)}. Retrying in ${this.retryOnElectrsErrorAfterSeconds} sec.`;
if (this.retryOnElectrsErrorAfterSeconds > 5) {
logger.warn(loggerMsg);
} else {
logger.debug(loggerMsg);
}
setTimeout(this.runMainUpdateLoop.bind(this), 1000 * this.retryOnElectrsErrorAfterSeconds);
2020-10-18 17:48:15 +02:00
this.retryOnElectrsErrorAfterSeconds *= 2;
this.retryOnElectrsErrorAfterSeconds = Math.min(this.retryOnElectrsErrorAfterSeconds, 60);
}
2019-07-21 16:59:47 +02:00
}
setUpWebsocketHandling() {
2020-09-21 22:52:54 +02:00
if (this.wss) {
websocketHandler.setWebsocketServer(this.wss);
}
websocketHandler.setupConnectionHandling();
statistics.setNewStatisticsEntryCallback(websocketHandler.handleNewStatistic.bind(websocketHandler));
blocks.setNewBlockCallback(websocketHandler.handleNewBlock.bind(websocketHandler));
memPool.setMempoolChangedCallback(websocketHandler.handleMempoolChange.bind(websocketHandler));
donations.setNotfyDonationStatusCallback(websocketHandler.handleNewDonation.bind(websocketHandler));
2019-07-21 16:59:47 +02:00
}
setUpHttpApiRoutes() {
2019-07-21 16:59:47 +02:00
this.app
.get(config.MEMPOOL.API_URL_PREFIX + 'transaction-times', routes.getTransactionTimes)
.get(config.MEMPOOL.API_URL_PREFIX + 'fees/recommended', routes.getRecommendedFees)
.get(config.MEMPOOL.API_URL_PREFIX + 'fees/mempool-blocks', routes.getMempoolBlocks)
.get(config.MEMPOOL.API_URL_PREFIX + 'backend-info', routes.getBackendInfo)
;
2020-10-19 13:47:10 +02:00
if (config.STATISTICS.ENABLED && config.DATABASE.ENABLED) {
this.app
.get(config.MEMPOOL.API_URL_PREFIX + 'statistics/2h', routes.get2HStatistics)
.get(config.MEMPOOL.API_URL_PREFIX + 'statistics/24h', routes.get24HStatistics.bind(routes))
.get(config.MEMPOOL.API_URL_PREFIX + 'statistics/1w', routes.get1WHStatistics.bind(routes))
.get(config.MEMPOOL.API_URL_PREFIX + 'statistics/1m', routes.get1MStatistics.bind(routes))
.get(config.MEMPOOL.API_URL_PREFIX + 'statistics/3m', routes.get3MStatistics.bind(routes))
.get(config.MEMPOOL.API_URL_PREFIX + 'statistics/6m', routes.get6MStatistics.bind(routes))
.get(config.MEMPOOL.API_URL_PREFIX + 'statistics/1y', routes.get1YStatistics.bind(routes))
;
}
if (config.BISQ_BLOCKS.ENABLED) {
this.app
.get(config.MEMPOOL.API_URL_PREFIX + 'bisq/stats', routes.getBisqStats)
.get(config.MEMPOOL.API_URL_PREFIX + 'bisq/tx/:txId', routes.getBisqTransaction)
.get(config.MEMPOOL.API_URL_PREFIX + 'bisq/block/:hash', routes.getBisqBlock)
.get(config.MEMPOOL.API_URL_PREFIX + 'bisq/blocks/tip/height', routes.getBisqTip)
.get(config.MEMPOOL.API_URL_PREFIX + 'bisq/blocks/:index/:length', routes.getBisqBlocks)
.get(config.MEMPOOL.API_URL_PREFIX + 'bisq/address/:address', routes.getBisqAddress)
.get(config.MEMPOOL.API_URL_PREFIX + 'bisq/txs/:index/:length', routes.getBisqTransactions)
;
}
2020-09-10 09:46:23 +02:00
if (config.BISQ_MARKETS.ENABLED) {
2020-09-10 09:46:23 +02:00
this.app
.get(config.MEMPOOL.API_URL_PREFIX + 'bisq/markets/currencies', routes.getBisqMarketCurrencies.bind(routes))
.get(config.MEMPOOL.API_URL_PREFIX + 'bisq/markets/depth', routes.getBisqMarketDepth.bind(routes))
.get(config.MEMPOOL.API_URL_PREFIX + 'bisq/markets/hloc', routes.getBisqMarketHloc.bind(routes))
.get(config.MEMPOOL.API_URL_PREFIX + 'bisq/markets/markets', routes.getBisqMarketMarkets.bind(routes))
.get(config.MEMPOOL.API_URL_PREFIX + 'bisq/markets/offers', routes.getBisqMarketOffers.bind(routes))
.get(config.MEMPOOL.API_URL_PREFIX + 'bisq/markets/ticker', routes.getBisqMarketTicker.bind(routes))
.get(config.MEMPOOL.API_URL_PREFIX + 'bisq/markets/trades', routes.getBisqMarketTrades.bind(routes))
.get(config.MEMPOOL.API_URL_PREFIX + 'bisq/markets/volumes', routes.getBisqMarketVolumes.bind(routes))
2020-09-10 09:46:23 +02:00
;
}
if (config.SPONSORS.ENABLED) {
this.app
.get(config.MEMPOOL.API_URL_PREFIX + 'donations', routes.getDonations.bind(routes))
.get(config.MEMPOOL.API_URL_PREFIX + 'donations/images/:id', routes.getSponsorImage.bind(routes))
.post(config.MEMPOOL.API_URL_PREFIX + 'donations', routes.createDonationRequest.bind(routes))
.post(config.MEMPOOL.API_URL_PREFIX + 'donations-webhook', routes.donationWebhook.bind(routes))
;
} else {
this.app
.get(config.MEMPOOL.API_URL_PREFIX + 'donations', async (req, res) => {
const response = await axios.get('https://mempool.space/api/v1/donations', { responseType: 'stream' });
response.data.pipe(res);
2020-10-18 16:37:27 +02:00
})
.get(config.MEMPOOL.API_URL_PREFIX + 'donations/images/:id', async (req, res) => {
const response = await axios.get('https://mempool.space/api/v1/donations/images/' + req.params.id, { responseType: 'stream' });
response.data.pipe(res);
});
}
}
}
2019-11-06 08:35:02 +01:00
const server = new Server();