2019-07-21 16:59:47 +02:00
|
|
|
const config = require('../mempool-config.json');
|
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 fs from 'fs';
|
|
|
|
import * as express from 'express';
|
|
|
|
import * as compression from 'compression';
|
|
|
|
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';
|
2020-10-12 05:50:10 +02:00
|
|
|
import * as request from 'request';
|
2019-07-21 16:59:47 +02:00
|
|
|
|
2020-07-22 19:04:29 +02:00
|
|
|
import { checkDbConnection } from './database';
|
2020-02-16 16:15:07 +01:00
|
|
|
import routes from './routes';
|
2019-07-21 16:59:47 +02:00
|
|
|
import blocks from './api/blocks';
|
2020-02-16 16:15:07 +01:00
|
|
|
import memPool from './api/mempool';
|
|
|
|
import diskCache from './api/disk-cache';
|
2019-07-21 16:59:47 +02:00
|
|
|
import statistics from './api/statistics';
|
2020-02-26 11:49:53 +01:00
|
|
|
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';
|
2020-10-07 15:15:42 +02:00
|
|
|
import donations from './api/donations';
|
2020-10-13 10:27:52 +02:00
|
|
|
import logger from './logger';
|
2020-10-13 12:48:43 +02:00
|
|
|
import backendInfo from './api/backend-info';
|
2019-07-21 16:59:47 +02:00
|
|
|
|
2020-02-16 16:15:07 +01: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;
|
2019-07-21 16:59:47 +02:00
|
|
|
|
|
|
|
constructor() {
|
|
|
|
this.app = express();
|
2020-02-16 16:15:07 +01:00
|
|
|
|
2020-09-21 22:52:54 +02:00
|
|
|
if (!config.CLUSTER_NUM_CORES || config.CLUSTER_NUM_CORES === 1) {
|
|
|
|
this.startServer();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cluster.isMaster) {
|
2020-10-13 12:48:43 +02:00
|
|
|
logger.notice(`Mempool Server is running on port ${config.HTTP_PORT} (${backendInfo.getShortCommitHash()})`);
|
2020-09-21 22:52:54 +02:00
|
|
|
|
|
|
|
const numCPUs = config.CLUSTER_NUM_CORES;
|
|
|
|
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;
|
2020-10-13 11:43:09 +02:00
|
|
|
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();
|
|
|
|
})
|
2020-10-07 15:15:42 +02:00
|
|
|
.use(compression())
|
|
|
|
.use(express.urlencoded({ extended: true }))
|
|
|
|
.use(express.json());
|
2020-02-16 16:15:07 +01:00
|
|
|
|
|
|
|
if (config.SSL === true) {
|
2019-07-21 16:59:47 +02:00
|
|
|
const credentials = {
|
2020-02-16 16:15:07 +01:00
|
|
|
cert: fs.readFileSync(config.SSL_CERT_FILE_PATH),
|
|
|
|
key: fs.readFileSync(config.SSL_KEY_FILE_PATH),
|
2019-07-21 16:59:47 +02:00
|
|
|
};
|
|
|
|
this.server = https.createServer(credentials, this.app);
|
|
|
|
this.wss = new WebSocket.Server({ server: this.server });
|
2020-02-16 16:15:07 +01:00
|
|
|
} else {
|
|
|
|
this.server = http.createServer(this.app);
|
|
|
|
this.wss = new WebSocket.Server({ server: this.server });
|
2019-07-21 16:59:47 +02:00
|
|
|
}
|
|
|
|
|
2020-07-22 19:04:29 +02:00
|
|
|
if (!config.DB_DISABLED) {
|
|
|
|
checkDbConnection();
|
|
|
|
statistics.startStatistics();
|
|
|
|
}
|
|
|
|
|
2020-02-26 11:49:53 +01:00
|
|
|
this.setUpHttpApiRoutes();
|
2019-07-21 16:59:47 +02:00
|
|
|
this.setUpWebsocketHandling();
|
|
|
|
this.runMempoolIntervalFunctions();
|
|
|
|
|
|
|
|
fiatConversion.startService();
|
2020-02-26 11:49:53 +01:00
|
|
|
diskCache.loadMempoolCache();
|
2019-07-21 16:59:47 +02:00
|
|
|
|
2020-10-15 06:37:25 +02:00
|
|
|
if (config.NETWORK === 'bisq') {
|
2020-07-03 18:45:19 +02:00
|
|
|
bisq.startBisqService();
|
2020-07-14 16:26:02 +02:00
|
|
|
bisq.setPriceCallbackFunction((price) => websocketHandler.setExtraInitProperties('bsq-price', price));
|
2020-09-27 12:21:18 +02:00
|
|
|
blocks.setNewBlockCallback(bisq.handleNewBitcoinBlock.bind(bisq));
|
2020-07-03 18:45:19 +02:00
|
|
|
}
|
|
|
|
|
2020-09-10 09:46:23 +02:00
|
|
|
if (config.BISQ_MARKET_ENABLED) {
|
|
|
|
bisqMarkets.startBisqService();
|
|
|
|
}
|
|
|
|
|
2020-02-16 16:15:07 +01:00
|
|
|
this.server.listen(config.HTTP_PORT, () => {
|
2020-09-21 22:52:54 +02:00
|
|
|
if (worker) {
|
2020-10-13 10:27:52 +02:00
|
|
|
logger.info(`Mempool Server worker #${process.pid} started`);
|
2020-09-21 22:52:54 +02:00
|
|
|
} else {
|
2020-10-13 12:48:43 +02:00
|
|
|
logger.notice(`Mempool Server is running on port ${config.HTTP_PORT} (${backendInfo.getShortCommitHash()})`);
|
2020-09-21 22:52:54 +02:00
|
|
|
}
|
2019-07-21 16:59:47 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2020-02-26 11:49:53 +01:00
|
|
|
async runMempoolIntervalFunctions() {
|
2020-02-17 14:39:20 +01:00
|
|
|
await memPool.updateMemPoolInfo();
|
2019-07-21 16:59:47 +02:00
|
|
|
await blocks.updateBlocks();
|
|
|
|
await memPool.updateMempool();
|
2020-02-16 16:15:07 +01:00
|
|
|
setTimeout(this.runMempoolIntervalFunctions.bind(this), config.ELECTRS_POLL_RATE_MS);
|
2019-07-21 16:59:47 +02:00
|
|
|
}
|
|
|
|
|
2020-02-26 11:49:53 +01:00
|
|
|
setUpWebsocketHandling() {
|
2020-09-21 22:52:54 +02:00
|
|
|
if (this.wss) {
|
|
|
|
websocketHandler.setWebsocketServer(this.wss);
|
|
|
|
}
|
2020-02-26 11:49:53 +01:00
|
|
|
websocketHandler.setupConnectionHandling();
|
|
|
|
statistics.setNewStatisticsEntryCallback(websocketHandler.handleNewStatistic.bind(websocketHandler));
|
|
|
|
blocks.setNewBlockCallback(websocketHandler.handleNewBlock.bind(websocketHandler));
|
|
|
|
memPool.setMempoolChangedCallback(websocketHandler.handleMempoolChange.bind(websocketHandler));
|
2020-10-07 15:15:42 +02:00
|
|
|
donations.setNotfyDonationStatusCallback(websocketHandler.handleNewDonation.bind(websocketHandler));
|
2019-07-21 16:59:47 +02:00
|
|
|
}
|
|
|
|
|
2020-02-26 11:49:53 +01:00
|
|
|
setUpHttpApiRoutes() {
|
2019-07-21 16:59:47 +02:00
|
|
|
this.app
|
2020-02-27 19:09:07 +01:00
|
|
|
.get(config.API_ENDPOINT + 'transaction-times', routes.getTransactionTimes)
|
2019-07-21 16:59:47 +02:00
|
|
|
.get(config.API_ENDPOINT + 'fees/recommended', routes.getRecommendedFees)
|
2020-02-16 16:15:07 +01:00
|
|
|
.get(config.API_ENDPOINT + 'fees/mempool-blocks', routes.getMempoolBlocks)
|
2019-07-21 16:59:47 +02:00
|
|
|
.get(config.API_ENDPOINT + 'statistics/2h', routes.get2HStatistics)
|
2019-08-15 13:06:08 +02:00
|
|
|
.get(config.API_ENDPOINT + 'statistics/24h', routes.get24HStatistics.bind(routes))
|
|
|
|
.get(config.API_ENDPOINT + 'statistics/1w', routes.get1WHStatistics.bind(routes))
|
|
|
|
.get(config.API_ENDPOINT + 'statistics/1m', routes.get1MStatistics.bind(routes))
|
|
|
|
.get(config.API_ENDPOINT + 'statistics/3m', routes.get3MStatistics.bind(routes))
|
|
|
|
.get(config.API_ENDPOINT + 'statistics/6m', routes.get6MStatistics.bind(routes))
|
2020-02-16 18:26:57 +01:00
|
|
|
.get(config.API_ENDPOINT + 'statistics/1y', routes.get1YStatistics.bind(routes))
|
2020-05-26 13:06:14 +02:00
|
|
|
.get(config.API_ENDPOINT + 'backend-info', routes.getBackendInfo)
|
2020-02-26 11:49:53 +01:00
|
|
|
;
|
2020-07-03 18:45:19 +02:00
|
|
|
|
2020-10-15 06:37:25 +02:00
|
|
|
if (config.NETWORK === 'bisq') {
|
2020-07-03 18:45:19 +02:00
|
|
|
this.app
|
2020-07-14 09:38:52 +02:00
|
|
|
.get(config.API_ENDPOINT + 'bisq/stats', routes.getBisqStats)
|
2020-07-03 18:45:19 +02:00
|
|
|
.get(config.API_ENDPOINT + 'bisq/tx/:txId', routes.getBisqTransaction)
|
|
|
|
.get(config.API_ENDPOINT + 'bisq/block/:hash', routes.getBisqBlock)
|
2020-07-15 08:10:13 +02:00
|
|
|
.get(config.API_ENDPOINT + 'bisq/blocks/tip/height', routes.getBisqTip)
|
2020-07-13 10:16:12 +02:00
|
|
|
.get(config.API_ENDPOINT + 'bisq/blocks/:index/:length', routes.getBisqBlocks)
|
2020-07-13 16:46:25 +02:00
|
|
|
.get(config.API_ENDPOINT + 'bisq/address/:address', routes.getBisqAddress)
|
2020-07-03 18:45:19 +02:00
|
|
|
.get(config.API_ENDPOINT + 'bisq/txs/:index/:length', routes.getBisqTransactions)
|
|
|
|
;
|
|
|
|
}
|
2020-09-10 09:46:23 +02:00
|
|
|
|
|
|
|
if (config.BISQ_MARKET_ENABLED) {
|
|
|
|
this.app
|
|
|
|
.get(config.API_ENDPOINT + 'bisq/markets/currencies', routes.getBisqMarketCurrencies.bind(routes))
|
|
|
|
.get(config.API_ENDPOINT + 'bisq/markets/depth', routes.getBisqMarketDepth.bind(routes))
|
|
|
|
.get(config.API_ENDPOINT + 'bisq/markets/hloc', routes.getBisqMarketHloc.bind(routes))
|
|
|
|
.get(config.API_ENDPOINT + 'bisq/markets/markets', routes.getBisqMarketMarkets.bind(routes))
|
|
|
|
.get(config.API_ENDPOINT + 'bisq/markets/offers', routes.getBisqMarketOffers.bind(routes))
|
|
|
|
.get(config.API_ENDPOINT + 'bisq/markets/ticker', routes.getBisqMarketTicker.bind(routes))
|
|
|
|
.get(config.API_ENDPOINT + 'bisq/markets/trades', routes.getBisqMarketTrades.bind(routes))
|
|
|
|
.get(config.API_ENDPOINT + 'bisq/markets/volumes', routes.getBisqMarketVolumes.bind(routes))
|
|
|
|
;
|
|
|
|
}
|
2020-10-07 15:15:42 +02:00
|
|
|
|
|
|
|
if (config.BTCPAY_URL) {
|
|
|
|
this.app
|
|
|
|
.get(config.API_ENDPOINT + 'donations', routes.getDonations.bind(routes))
|
|
|
|
.post(config.API_ENDPOINT + 'donations', routes.createDonationRequest.bind(routes))
|
|
|
|
.post(config.API_ENDPOINT + 'donations-webhook', routes.donationWebhook.bind(routes))
|
|
|
|
;
|
2020-10-12 05:50:10 +02:00
|
|
|
} else {
|
|
|
|
this.app
|
|
|
|
.get(config.API_ENDPOINT + 'donations', (req, res) => {
|
2020-10-12 07:26:01 +02:00
|
|
|
req.pipe(request('https://mempool.space/api/v1/donations')).pipe(res);
|
2020-10-12 05:50:10 +02:00
|
|
|
});
|
2020-10-07 15:15:42 +02:00
|
|
|
}
|
2020-02-26 11:49:53 +01:00
|
|
|
}
|
2020-02-16 16:15:07 +01:00
|
|
|
}
|
2019-11-06 08:35:02 +01:00
|
|
|
|
2020-02-16 16:15:07 +01:00
|
|
|
const server = new Server();
|