2022-02-19 22:09:35 +09:00
|
|
|
import { Common } from '../api/common';
|
2022-02-19 20:45:02 +09:00
|
|
|
import { DB } from '../database';
|
|
|
|
import logger from '../logger';
|
2022-02-24 16:55:18 +09:00
|
|
|
import PoolsRepository from './PoolsRepository';
|
2022-02-19 20:45:02 +09:00
|
|
|
|
|
|
|
class HashratesRepository {
|
|
|
|
/**
|
|
|
|
* Save indexed block data in the database
|
|
|
|
*/
|
2022-02-21 17:34:07 +09:00
|
|
|
public async $saveHashrates(hashrates: any) {
|
2022-03-06 12:32:16 +01:00
|
|
|
if (hashrates.length === 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-02-21 17:34:07 +09:00
|
|
|
let query = `INSERT INTO
|
2022-02-24 16:55:18 +09:00
|
|
|
hashrates(hashrate_timestamp, avg_hashrate, pool_id, share, type) VALUES`;
|
2022-02-19 20:45:02 +09:00
|
|
|
|
2022-02-21 17:34:07 +09:00
|
|
|
for (const hashrate of hashrates) {
|
2022-02-24 16:55:18 +09:00
|
|
|
query += ` (FROM_UNIXTIME(${hashrate.hashrateTimestamp}), ${hashrate.avgHashrate}, ${hashrate.poolId}, ${hashrate.share}, "${hashrate.type}"),`;
|
2022-02-21 17:34:07 +09:00
|
|
|
}
|
|
|
|
query = query.slice(0, -1);
|
2022-02-19 20:45:02 +09:00
|
|
|
|
2022-03-12 14:47:33 +01:00
|
|
|
let connection;
|
2022-02-21 17:34:07 +09:00
|
|
|
try {
|
2022-03-12 14:47:33 +01:00
|
|
|
connection = await DB.getConnection();
|
2022-02-21 17:34:07 +09:00
|
|
|
await connection.query(query);
|
2022-03-06 16:44:09 +01:00
|
|
|
connection.release();
|
2022-02-19 20:45:02 +09:00
|
|
|
} catch (e: any) {
|
2022-03-06 16:44:09 +01:00
|
|
|
connection.release();
|
2022-02-19 20:45:02 +09:00
|
|
|
logger.err('$saveHashrateInDatabase() error' + (e instanceof Error ? e.message : e));
|
2022-03-05 16:23:01 +01:00
|
|
|
throw e;
|
2022-02-19 20:45:02 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-24 16:55:18 +09:00
|
|
|
public async $getNetworkDailyHashrate(interval: string | null): Promise<any[]> {
|
2022-02-19 22:09:35 +09:00
|
|
|
interval = Common.getSqlInterval(interval);
|
|
|
|
|
2022-03-12 14:47:33 +01:00
|
|
|
const connection = await DB.getConnection();
|
2022-02-19 22:09:35 +09:00
|
|
|
|
|
|
|
let query = `SELECT UNIX_TIMESTAMP(hashrate_timestamp) as timestamp, avg_hashrate as avgHashrate
|
|
|
|
FROM hashrates`;
|
|
|
|
|
|
|
|
if (interval) {
|
2022-02-24 16:55:18 +09:00
|
|
|
query += ` WHERE hashrate_timestamp BETWEEN DATE_SUB(NOW(), INTERVAL ${interval}) AND NOW()
|
2022-03-12 14:47:33 +01:00
|
|
|
AND hashrates.type = 'daily'`;
|
2022-02-24 16:55:18 +09:00
|
|
|
} else {
|
2022-03-12 14:47:33 +01:00
|
|
|
query += ` WHERE hashrates.type = 'daily'`;
|
2022-02-19 22:09:35 +09:00
|
|
|
}
|
|
|
|
|
2022-02-22 00:17:41 +09:00
|
|
|
query += ` ORDER by hashrate_timestamp`;
|
2022-02-21 12:22:20 +09:00
|
|
|
|
2022-03-06 16:44:09 +01:00
|
|
|
try {
|
|
|
|
const [rows]: any[] = await connection.query(query);
|
|
|
|
connection.release();
|
2022-02-19 22:09:35 +09:00
|
|
|
|
2022-03-06 16:44:09 +01:00
|
|
|
return rows;
|
|
|
|
} catch (e) {
|
|
|
|
connection.release();
|
|
|
|
logger.err('$getNetworkDailyHashrate() error' + (e instanceof Error ? e.message : e));
|
|
|
|
throw e;
|
|
|
|
}
|
2022-02-19 20:45:02 +09:00
|
|
|
}
|
2022-02-21 12:22:20 +09:00
|
|
|
|
2022-03-06 12:32:16 +01:00
|
|
|
public async $getWeeklyHashrateTimestamps(): Promise<number[]> {
|
2022-03-12 14:47:33 +01:00
|
|
|
const connection = await DB.getConnection();
|
2022-03-06 12:32:16 +01:00
|
|
|
|
2022-03-12 14:47:33 +01:00
|
|
|
const query = `SELECT UNIX_TIMESTAMP(hashrate_timestamp) as timestamp
|
|
|
|
FROM hashrates
|
|
|
|
WHERE type = 'weekly'
|
|
|
|
GROUP BY hashrate_timestamp`;
|
2022-03-06 12:32:16 +01:00
|
|
|
|
2022-03-06 16:44:09 +01:00
|
|
|
try {
|
|
|
|
const [rows]: any[] = await connection.query(query);
|
|
|
|
connection.release();
|
|
|
|
|
|
|
|
return rows.map(row => row.timestamp);
|
|
|
|
} catch (e) {
|
|
|
|
connection.release();
|
|
|
|
logger.err('$getWeeklyHashrateTimestamps() error' + (e instanceof Error ? e.message : e));
|
|
|
|
throw e;
|
|
|
|
}
|
2022-03-06 12:32:16 +01:00
|
|
|
}
|
|
|
|
|
2022-02-24 16:55:18 +09:00
|
|
|
/**
|
|
|
|
* Returns the current biggest pool hashrate history
|
|
|
|
*/
|
|
|
|
public async $getPoolsWeeklyHashrate(interval: string | null): Promise<any[]> {
|
|
|
|
interval = Common.getSqlInterval(interval);
|
|
|
|
|
2022-03-12 14:47:33 +01:00
|
|
|
const connection = await DB.getConnection();
|
2022-02-24 16:55:18 +09:00
|
|
|
const topPoolsId = (await PoolsRepository.$getPoolsInfo('1w')).map((pool) => pool.poolId);
|
|
|
|
|
|
|
|
let query = `SELECT UNIX_TIMESTAMP(hashrate_timestamp) as timestamp, avg_hashrate as avgHashrate, share, pools.name as poolName
|
|
|
|
FROM hashrates
|
|
|
|
JOIN pools on pools.id = pool_id`;
|
|
|
|
|
|
|
|
if (interval) {
|
|
|
|
query += ` WHERE hashrate_timestamp BETWEEN DATE_SUB(NOW(), INTERVAL ${interval}) AND NOW()
|
|
|
|
AND hashrates.type = 'weekly'
|
|
|
|
AND pool_id IN (${topPoolsId})`;
|
|
|
|
} else {
|
|
|
|
query += ` WHERE hashrates.type = 'weekly'
|
|
|
|
AND pool_id IN (${topPoolsId})`;
|
|
|
|
}
|
|
|
|
|
|
|
|
query += ` ORDER by hashrate_timestamp, FIELD(pool_id, ${topPoolsId})`;
|
|
|
|
|
2022-03-06 16:44:09 +01:00
|
|
|
try {
|
|
|
|
const [rows]: any[] = await connection.query(query);
|
|
|
|
connection.release();
|
2022-02-24 16:55:18 +09:00
|
|
|
|
2022-03-06 16:44:09 +01:00
|
|
|
return rows;
|
|
|
|
} catch (e) {
|
|
|
|
connection.release();
|
|
|
|
logger.err('$getPoolsWeeklyHashrate() error' + (e instanceof Error ? e.message : e));
|
|
|
|
throw e;
|
|
|
|
}
|
2022-02-24 16:55:18 +09:00
|
|
|
}
|
|
|
|
|
2022-03-08 12:50:47 +01:00
|
|
|
/**
|
|
|
|
* Returns a pool hashrate history
|
|
|
|
*/
|
2022-03-08 13:54:04 +01:00
|
|
|
public async $getPoolWeeklyHashrate(poolId: number): Promise<any[]> {
|
2022-03-12 14:47:33 +01:00
|
|
|
const connection = await DB.getConnection();
|
2022-03-08 12:50:47 +01:00
|
|
|
|
2022-03-08 13:54:04 +01:00
|
|
|
// Find hashrate boundaries
|
|
|
|
let query = `SELECT MIN(hashrate_timestamp) as firstTimestamp, MAX(hashrate_timestamp) as lastTimestamp
|
|
|
|
FROM hashrates
|
|
|
|
JOIN pools on pools.id = pool_id
|
2022-03-08 16:55:49 +01:00
|
|
|
WHERE hashrates.type = 'weekly' AND pool_id = ? AND avg_hashrate != 0
|
|
|
|
ORDER by hashrate_timestamp LIMIT 1`;
|
2022-03-08 13:54:04 +01:00
|
|
|
|
|
|
|
let boundaries = {
|
|
|
|
firstTimestamp: '1970-01-01',
|
|
|
|
lastTimestamp: '9999-01-01'
|
|
|
|
};
|
|
|
|
try {
|
2022-03-08 16:55:49 +01:00
|
|
|
const [rows]: any[] = await connection.query(query, [poolId]);
|
2022-03-08 13:54:04 +01:00
|
|
|
boundaries = rows[0];
|
|
|
|
connection.release();
|
|
|
|
} catch (e) {
|
|
|
|
connection.release();
|
|
|
|
logger.err('$getPoolWeeklyHashrate() error' + (e instanceof Error ? e.message : e));
|
2022-03-08 12:50:47 +01:00
|
|
|
}
|
|
|
|
|
2022-03-08 13:54:04 +01:00
|
|
|
// Get hashrates entries between boundaries
|
|
|
|
query = `SELECT UNIX_TIMESTAMP(hashrate_timestamp) as timestamp, avg_hashrate as avgHashrate, share, pools.name as poolName
|
|
|
|
FROM hashrates
|
|
|
|
JOIN pools on pools.id = pool_id
|
|
|
|
WHERE hashrates.type = 'weekly' AND hashrate_timestamp BETWEEN ? AND ?
|
2022-03-08 16:55:49 +01:00
|
|
|
AND pool_id = ?
|
2022-03-08 13:54:04 +01:00
|
|
|
ORDER by hashrate_timestamp`;
|
2022-03-08 12:50:47 +01:00
|
|
|
|
|
|
|
try {
|
2022-03-08 16:55:49 +01:00
|
|
|
const [rows]: any[] = await connection.query(query, [boundaries.firstTimestamp, boundaries.lastTimestamp, poolId]);
|
2022-03-08 12:50:47 +01:00
|
|
|
connection.release();
|
|
|
|
|
|
|
|
return rows;
|
|
|
|
} catch (e) {
|
|
|
|
connection.release();
|
|
|
|
logger.err('$getPoolWeeklyHashrate() error' + (e instanceof Error ? e.message : e));
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-06 12:32:16 +01:00
|
|
|
public async $setLatestRunTimestamp(key: string, val: any = null) {
|
2022-03-12 14:47:33 +01:00
|
|
|
const connection = await DB.getConnection();
|
2022-03-06 12:32:16 +01:00
|
|
|
const query = `UPDATE state SET number = ? WHERE name = ?`;
|
2022-02-24 20:20:18 +09:00
|
|
|
|
2022-03-06 16:44:09 +01:00
|
|
|
try {
|
|
|
|
await connection.query<any>(query, (val === null) ? [Math.round(new Date().getTime() / 1000), key] : [val, key]);
|
|
|
|
connection.release();
|
|
|
|
} catch (e) {
|
|
|
|
connection.release();
|
|
|
|
}
|
2022-02-21 12:22:20 +09:00
|
|
|
}
|
|
|
|
|
2022-03-06 12:32:16 +01:00
|
|
|
public async $getLatestRunTimestamp(key: string): Promise<number> {
|
2022-03-12 14:47:33 +01:00
|
|
|
const connection = await DB.getConnection();
|
2022-03-06 12:32:16 +01:00
|
|
|
const query = `SELECT number FROM state WHERE name = ?`;
|
2022-03-06 16:44:09 +01:00
|
|
|
|
|
|
|
try {
|
|
|
|
const [rows] = await connection.query<any>(query, [key]);
|
|
|
|
connection.release();
|
|
|
|
|
2022-03-08 16:55:49 +01:00
|
|
|
if (rows.length === 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
2022-03-06 16:44:09 +01:00
|
|
|
return rows[0]['number'];
|
|
|
|
} catch (e) {
|
|
|
|
connection.release();
|
|
|
|
logger.err('$setLatestRunTimestamp() error' + (e instanceof Error ? e.message : e));
|
|
|
|
throw e;
|
|
|
|
}
|
2022-02-21 12:22:20 +09:00
|
|
|
}
|
2022-02-19 20:45:02 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
export default new HashratesRepository();
|