2022-01-18 09:37:04 +01:00
|
|
|
import { BlockExtended, PoolTag } from '../mempool.interfaces';
|
|
|
|
import { DB } from '../database';
|
|
|
|
import logger from '../logger';
|
2022-01-06 11:59:33 +01:00
|
|
|
|
|
|
|
export interface EmptyBlocks {
|
2022-01-18 09:37:04 +01:00
|
|
|
emptyBlocks: number;
|
|
|
|
poolId: number;
|
2022-01-06 11:59:33 +01:00
|
|
|
}
|
2022-01-05 07:41:14 +01:00
|
|
|
|
|
|
|
class BlocksRepository {
|
|
|
|
/**
|
|
|
|
* Save indexed block data in the database
|
|
|
|
*/
|
2022-02-08 07:47:43 +01:00
|
|
|
public async $saveBlockInDatabase(block: BlockExtended) {
|
2022-01-05 07:41:14 +01:00
|
|
|
const connection = await DB.pool.getConnection();
|
|
|
|
|
|
|
|
try {
|
|
|
|
const query = `INSERT INTO blocks(
|
2022-01-18 09:37:04 +01:00
|
|
|
height, hash, blockTimestamp, size,
|
|
|
|
weight, tx_count, coinbase_raw, difficulty,
|
2022-02-08 07:47:43 +01:00
|
|
|
pool_id, fees, fee_span, median_fee,
|
|
|
|
reward
|
2022-01-05 07:41:14 +01:00
|
|
|
) VALUE (
|
2022-01-06 11:59:33 +01:00
|
|
|
?, ?, FROM_UNIXTIME(?), ?,
|
2022-01-05 07:41:14 +01:00
|
|
|
?, ?, ?, ?,
|
2022-02-08 07:47:43 +01:00
|
|
|
?, ?, ?, ?,
|
|
|
|
?
|
2022-01-05 07:41:14 +01:00
|
|
|
)`;
|
|
|
|
|
|
|
|
const params: any[] = [
|
2022-02-04 04:51:45 +01:00
|
|
|
block.height,
|
2022-02-08 07:47:43 +01:00
|
|
|
block.id,
|
2022-02-04 04:51:45 +01:00
|
|
|
block.timestamp,
|
|
|
|
block.size,
|
|
|
|
block.weight,
|
|
|
|
block.tx_count,
|
2022-02-08 07:47:43 +01:00
|
|
|
block.extras?.coinbaseHex ?? '',
|
2022-02-04 04:51:45 +01:00
|
|
|
block.difficulty,
|
2022-02-08 07:47:43 +01:00
|
|
|
block.extras?.pool?.id, // Should always be set to something
|
2022-02-04 04:51:45 +01:00
|
|
|
0,
|
|
|
|
'[]',
|
2022-02-08 07:47:43 +01:00
|
|
|
block.extras.medianFee ?? 0,
|
|
|
|
block.extras?.reward ?? 0,
|
2022-01-05 07:41:14 +01:00
|
|
|
];
|
|
|
|
|
|
|
|
await connection.query(query, params);
|
2022-02-05 07:50:57 +01:00
|
|
|
} catch (e: any) {
|
|
|
|
if (e.errno === 1062) { // ER_DUP_ENTRY
|
|
|
|
logger.debug(`$saveBlockInDatabase() - Block ${block.height} has already been indexed, ignoring`);
|
|
|
|
} else {
|
|
|
|
logger.err('$saveBlockInDatabase() error' + (e instanceof Error ? e.message : e));
|
|
|
|
}
|
2022-01-05 07:41:14 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
connection.release();
|
|
|
|
}
|
|
|
|
|
2022-01-18 09:37:04 +01:00
|
|
|
/**
|
|
|
|
* Get all block height that have not been indexed between [startHeight, endHeight]
|
|
|
|
*/
|
|
|
|
public async $getMissingBlocksBetweenHeights(startHeight: number, endHeight: number): Promise<number[]> {
|
2022-01-24 09:43:11 +01:00
|
|
|
if (startHeight < endHeight) {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
2022-01-18 09:37:04 +01:00
|
|
|
const connection = await DB.pool.getConnection();
|
|
|
|
const [rows] : any[] = await connection.query(`
|
|
|
|
SELECT height
|
|
|
|
FROM blocks
|
|
|
|
WHERE height <= ${startHeight} AND height >= ${endHeight}
|
|
|
|
ORDER BY height DESC;
|
|
|
|
`);
|
|
|
|
connection.release();
|
|
|
|
|
|
|
|
const indexedBlockHeights: number[] = [];
|
|
|
|
rows.forEach((row: any) => { indexedBlockHeights.push(row.height); });
|
|
|
|
const seekedBlocks: number[] = Array.from(Array(startHeight - endHeight + 1).keys(), n => n + endHeight).reverse();
|
|
|
|
const missingBlocksHeights = seekedBlocks.filter(x => indexedBlockHeights.indexOf(x) === -1);
|
|
|
|
|
|
|
|
return missingBlocksHeights;
|
|
|
|
}
|
|
|
|
|
2022-01-06 11:59:33 +01:00
|
|
|
/**
|
|
|
|
* Count empty blocks for all pools
|
|
|
|
*/
|
2022-01-25 10:33:46 +01:00
|
|
|
public async $countEmptyBlocks(interval: string | null): Promise<EmptyBlocks[]> {
|
|
|
|
const query = `
|
2022-01-06 11:59:33 +01:00
|
|
|
SELECT pool_id as poolId
|
|
|
|
FROM blocks
|
2022-01-25 10:33:46 +01:00
|
|
|
WHERE tx_count = 1` +
|
|
|
|
(interval != null ? ` AND blockTimestamp BETWEEN DATE_SUB(NOW(), INTERVAL ${interval}) AND NOW()` : ``)
|
|
|
|
;
|
|
|
|
|
|
|
|
const connection = await DB.pool.getConnection();
|
|
|
|
const [rows] = await connection.query(query);
|
2022-01-06 11:59:33 +01:00
|
|
|
connection.release();
|
|
|
|
|
|
|
|
return <EmptyBlocks[]>rows;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get blocks count for a period
|
|
|
|
*/
|
2022-01-25 10:33:46 +01:00
|
|
|
public async $blockCount(interval: string | null): Promise<number> {
|
|
|
|
const query = `
|
2022-01-06 11:59:33 +01:00
|
|
|
SELECT count(height) as blockCount
|
2022-01-25 10:33:46 +01:00
|
|
|
FROM blocks` +
|
|
|
|
(interval != null ? ` WHERE blockTimestamp BETWEEN DATE_SUB(NOW(), INTERVAL ${interval}) AND NOW()` : ``)
|
|
|
|
;
|
|
|
|
|
|
|
|
const connection = await DB.pool.getConnection();
|
|
|
|
const [rows] = await connection.query(query);
|
|
|
|
connection.release();
|
|
|
|
|
|
|
|
return <number>rows[0].blockCount;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the oldest indexed block
|
|
|
|
*/
|
|
|
|
public async $oldestBlockTimestamp(): Promise<number> {
|
|
|
|
const connection = await DB.pool.getConnection();
|
|
|
|
const [rows]: any[] = await connection.query(`
|
|
|
|
SELECT blockTimestamp
|
2022-01-06 11:59:33 +01:00
|
|
|
FROM blocks
|
2022-01-25 10:33:46 +01:00
|
|
|
ORDER BY height
|
|
|
|
LIMIT 1;
|
2022-01-06 11:59:33 +01:00
|
|
|
`);
|
|
|
|
connection.release();
|
|
|
|
|
2022-01-25 10:33:46 +01:00
|
|
|
if (rows.length <= 0) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return <number>rows[0].blockTimestamp;
|
2022-01-06 11:59:33 +01:00
|
|
|
}
|
2022-02-08 07:47:43 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Get one block by height
|
|
|
|
*/
|
|
|
|
public async $getBlockByHeight(height: number): Promise<object | null> {
|
|
|
|
const connection = await DB.pool.getConnection();
|
|
|
|
const [rows]: any[] = await connection.query(`
|
|
|
|
SELECT *, UNIX_TIMESTAMP(blocks.blockTimestamp) as blockTimestamp, pools.id as pool_id, pools.name as pool_name, pools.link as pool_link, pools.addresses as pool_addresses, pools.regexes as pool_regexes
|
|
|
|
FROM blocks
|
|
|
|
JOIN pools ON blocks.pool_id = pools.id
|
|
|
|
WHERE height = ${height};
|
|
|
|
`);
|
|
|
|
connection.release();
|
|
|
|
|
|
|
|
if (rows.length <= 0) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
return rows[0];
|
|
|
|
}
|
2022-01-05 07:41:14 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
export default new BlocksRepository();
|