2022-01-11 12:43:59 +01:00
import { PoolConnection } from 'mysql2/promise' ;
2021-12-11 01:27:58 +01:00
import config from '../config' ;
import { DB } from '../database' ;
import logger from '../logger' ;
2022-01-19 10:50:52 +01:00
const sleep = ( ms : number ) = > new Promise ( res = > setTimeout ( res , ms ) ) ;
2022-01-12 09:26:10 +01:00
2021-12-11 01:27:58 +01:00
class DatabaseMigration {
2022-03-12 14:47:33 +01:00
private static currentVersion = 14 ;
2021-12-11 01:27:58 +01:00
private queryTimeout = 120000 ;
2022-01-12 06:10:16 +01:00
private statisticsAddedIndexed = false ;
2021-12-11 01:27:58 +01:00
constructor ( ) { }
2022-01-11 12:43:59 +01:00
/ * *
* Entry point
* /
2021-12-11 01:27:58 +01:00
public async $initializeOrMigrateDatabase ( ) : Promise < void > {
2022-02-21 15:57:44 +01:00
logger . debug ( 'MIGRATIONS: Running migrations' ) ;
2022-01-11 12:43:59 +01:00
2022-01-12 08:06:45 +01:00
await this . $printDatabaseVersion ( ) ;
2022-01-11 12:43:59 +01:00
// First of all, if the `state` database does not exist, create it so we can track migration version
if ( ! await this . $checkIfTableExists ( 'state' ) ) {
2022-02-21 15:57:44 +01:00
logger . debug ( 'MIGRATIONS: `state` table does not exist. Creating it.' ) ;
2022-01-11 12:43:59 +01:00
try {
await this . $createMigrationStateTable ( ) ;
} catch ( e ) {
2022-01-12 09:43:32 +01:00
logger . err ( 'MIGRATIONS: Unable to create `state` table, aborting in 10 seconds. ' + e ) ;
2022-01-12 09:26:10 +01:00
await sleep ( 10000 ) ;
2022-01-11 12:43:59 +01:00
process . exit ( - 1 ) ;
}
2022-02-21 15:57:44 +01:00
logger . debug ( 'MIGRATIONS: `state` table initialized.' ) ;
2022-01-11 12:43:59 +01:00
}
let databaseSchemaVersion = 0 ;
try {
databaseSchemaVersion = await this . $getSchemaVersionFromDatabase ( ) ;
} catch ( e ) {
2022-01-12 09:43:32 +01:00
logger . err ( 'MIGRATIONS: Unable to get current database migration version, aborting in 10 seconds. ' + e ) ;
2022-01-12 09:26:10 +01:00
await sleep ( 10000 ) ;
2022-01-11 12:43:59 +01:00
process . exit ( - 1 ) ;
}
2022-02-21 15:57:44 +01:00
logger . debug ( 'MIGRATIONS: Current state.schema_version ' + databaseSchemaVersion ) ;
logger . debug ( 'MIGRATIONS: Latest DatabaseMigration.version is ' + DatabaseMigration . currentVersion ) ;
2022-01-12 06:10:16 +01:00
if ( databaseSchemaVersion >= DatabaseMigration . currentVersion ) {
2022-02-21 15:57:44 +01:00
logger . debug ( 'MIGRATIONS: Nothing to do.' ) ;
2022-01-11 12:43:59 +01:00
return ;
2021-12-11 01:27:58 +01:00
}
2022-01-12 08:41:27 +01:00
// Now, create missing tables. Those queries cannot be wrapped into a transaction unfortunately
try {
await this . $createMissingTablesAndIndexes ( databaseSchemaVersion ) ;
} catch ( e ) {
2022-01-12 09:43:32 +01:00
logger . err ( 'MIGRATIONS: Unable to create required tables, aborting in 10 seconds. ' + e ) ;
2022-01-12 09:26:10 +01:00
await sleep ( 10000 ) ;
2022-01-12 08:41:27 +01:00
process . exit ( - 1 ) ;
}
2022-01-12 06:10:16 +01:00
2022-01-11 12:43:59 +01:00
if ( DatabaseMigration . currentVersion > databaseSchemaVersion ) {
2022-02-21 15:57:44 +01:00
logger . notice ( 'MIGRATIONS: Upgrading datababse schema' ) ;
2022-01-11 12:43:59 +01:00
try {
2021-12-11 01:27:58 +01:00
await this . $migrateTableSchemaFromVersion ( databaseSchemaVersion ) ;
2022-02-21 15:57:44 +01:00
logger . notice ( ` MIGRATIONS: OK. Database schema have been migrated from version ${ databaseSchemaVersion } to ${ DatabaseMigration . currentVersion } (latest version) ` ) ;
2022-01-11 12:43:59 +01:00
} catch ( e ) {
2022-01-12 09:43:32 +01:00
logger . err ( 'MIGRATIONS: Unable to migrate database, aborting. ' + e ) ;
2021-12-11 01:27:58 +01:00
}
}
2022-01-11 12:43:59 +01:00
return ;
2021-12-11 01:27:58 +01:00
}
2022-01-12 08:41:27 +01:00
/ * *
* Create all missing tables
* /
private async $createMissingTablesAndIndexes ( databaseSchemaVersion : number ) {
await this . $setStatisticsAddedIndexedFlag ( databaseSchemaVersion ) ;
2022-02-16 07:19:16 +01:00
const isBitcoin = [ 'mainnet' , 'testnet' , 'signet' ] . includes ( config . MEMPOOL . NETWORK ) ;
2022-03-12 14:47:33 +01:00
const connection = await DB . getConnection ( ) ;
2022-01-12 08:41:27 +01:00
try {
await this . $executeQuery ( connection , this . getCreateElementsTableQuery ( ) , await this . $checkIfTableExists ( 'elements_pegs' ) ) ;
await this . $executeQuery ( connection , this . getCreateStatisticsQuery ( ) , await this . $checkIfTableExists ( 'statistics' ) ) ;
if ( databaseSchemaVersion < 2 && this . statisticsAddedIndexed === false ) {
await this . $executeQuery ( connection , ` CREATE INDEX added ON statistics (added); ` ) ;
}
2022-01-19 10:50:52 +01:00
if ( databaseSchemaVersion < 3 ) {
await this . $executeQuery ( connection , this . getCreatePoolsTableQuery ( ) , await this . $checkIfTableExists ( 'pools' ) ) ;
2022-01-18 09:37:04 +01:00
}
if ( databaseSchemaVersion < 4 ) {
2022-01-20 16:08:51 +01:00
await this . $executeQuery ( connection , 'DROP table IF EXISTS blocks;' ) ;
2022-01-13 04:18:39 +01:00
await this . $executeQuery ( connection , this . getCreateBlocksTableQuery ( ) , await this . $checkIfTableExists ( 'blocks' ) ) ;
2022-01-19 10:50:52 +01:00
}
2022-02-16 07:19:16 +01:00
if ( databaseSchemaVersion < 5 && isBitcoin === true ) {
2022-03-09 19:18:51 +01:00
logger . warn ( ` 'blocks' table has been truncated. Re-indexing from scratch. ` ) ;
2022-02-17 10:02:55 +01:00
await this . $executeQuery ( connection , 'TRUNCATE blocks;' ) ; // Need to re-index
2022-02-16 07:19:16 +01:00
await this . $executeQuery ( connection , 'ALTER TABLE blocks ADD `reward` double unsigned NOT NULL DEFAULT "0"' ) ;
}
if ( databaseSchemaVersion < 6 && isBitcoin === true ) {
2022-03-09 19:18:51 +01:00
logger . warn ( ` 'blocks' table has been truncated. Re-indexing from scratch. ` ) ;
2022-02-17 10:02:55 +01:00
await this . $executeQuery ( connection , 'TRUNCATE blocks;' ) ; // Need to re-index
2022-02-16 07:19:16 +01:00
// Cleanup original blocks fields type
await this . $executeQuery ( connection , 'ALTER TABLE blocks MODIFY `height` integer unsigned NOT NULL DEFAULT "0"' ) ;
await this . $executeQuery ( connection , 'ALTER TABLE blocks MODIFY `tx_count` smallint unsigned NOT NULL DEFAULT "0"' ) ;
await this . $executeQuery ( connection , 'ALTER TABLE blocks MODIFY `size` integer unsigned NOT NULL DEFAULT "0"' ) ;
await this . $executeQuery ( connection , 'ALTER TABLE blocks MODIFY `weight` integer unsigned NOT NULL DEFAULT "0"' ) ;
await this . $executeQuery ( connection , 'ALTER TABLE blocks MODIFY `difficulty` double NOT NULL DEFAULT "0"' ) ;
// We also fix the pools.id type so we need to drop/re-create the foreign key
await this . $executeQuery ( connection , 'ALTER TABLE blocks DROP FOREIGN KEY IF EXISTS `blocks_ibfk_1`' ) ;
await this . $executeQuery ( connection , 'ALTER TABLE pools MODIFY `id` smallint unsigned AUTO_INCREMENT' ) ;
await this . $executeQuery ( connection , 'ALTER TABLE blocks MODIFY `pool_id` smallint unsigned NULL' ) ;
await this . $executeQuery ( connection , 'ALTER TABLE blocks ADD FOREIGN KEY (`pool_id`) REFERENCES `pools` (`id`)' ) ;
// Add new block indexing fields
await this . $executeQuery ( connection , 'ALTER TABLE blocks ADD `version` integer unsigned NOT NULL DEFAULT "0"' ) ;
await this . $executeQuery ( connection , 'ALTER TABLE blocks ADD `bits` integer unsigned NOT NULL DEFAULT "0"' ) ;
await this . $executeQuery ( connection , 'ALTER TABLE blocks ADD `nonce` bigint unsigned NOT NULL DEFAULT "0"' ) ;
await this . $executeQuery ( connection , 'ALTER TABLE blocks ADD `merkle_root` varchar(65) NOT NULL DEFAULT ""' ) ;
await this . $executeQuery ( connection , 'ALTER TABLE blocks ADD `previous_block_hash` varchar(65) NULL' ) ;
}
2022-02-19 12:45:02 +01:00
if ( databaseSchemaVersion < 7 && isBitcoin === true ) {
await this . $executeQuery ( connection , 'DROP table IF EXISTS hashrates;' ) ;
await this . $executeQuery ( connection , this . getCreateDailyStatsTableQuery ( ) , await this . $checkIfTableExists ( 'hashrates' ) ) ;
}
2022-02-24 08:55:18 +01:00
if ( databaseSchemaVersion < 8 && isBitcoin === true ) {
2022-03-09 19:18:51 +01:00
logger . warn ( ` 'hashrates' table has been truncated. Re-indexing from scratch. ` ) ;
2022-02-24 12:20:18 +01:00
await this . $executeQuery ( connection , 'TRUNCATE hashrates;' ) ; // Need to re-index
2022-02-24 08:55:18 +01:00
await this . $executeQuery ( connection , 'ALTER TABLE `hashrates` DROP INDEX `PRIMARY`' ) ;
await this . $executeQuery ( connection , 'ALTER TABLE `hashrates` ADD `id` int NOT NULL AUTO_INCREMENT PRIMARY KEY FIRST' ) ;
await this . $executeQuery ( connection , 'ALTER TABLE `hashrates` ADD `share` float NOT NULL DEFAULT "0"' ) ;
await this . $executeQuery ( connection , 'ALTER TABLE `hashrates` ADD `type` enum("daily", "weekly") DEFAULT "daily"' ) ;
}
2022-03-07 15:56:07 +01:00
if ( databaseSchemaVersion < 9 && isBitcoin === true ) {
2022-03-09 19:18:51 +01:00
logger . warn ( ` 'hashrates' table has been truncated. Re-indexing from scratch. ` ) ;
2022-03-06 16:48:14 +01:00
await this . $executeQuery ( connection , 'TRUNCATE hashrates;' ) ; // Need to re-index
2022-03-06 16:44:09 +01:00
await this . $executeQuery ( connection , 'ALTER TABLE `state` CHANGE `name` `name` varchar(100)' ) ;
await this . $executeQuery ( connection , 'ALTER TABLE `hashrates` ADD UNIQUE `hashrate_timestamp_pool_id` (`hashrate_timestamp`, `pool_id`)' ) ;
2022-03-06 12:32:16 +01:00
}
2022-03-08 17:10:29 +01:00
if ( databaseSchemaVersion < 10 && isBitcoin === true ) {
await this . $executeQuery ( connection , 'ALTER TABLE `blocks` ADD INDEX `blockTimestamp` (`blockTimestamp`)' ) ;
}
2022-03-09 19:18:51 +01:00
if ( databaseSchemaVersion < 11 && isBitcoin === true ) {
logger . warn ( ` 'blocks' table has been truncated. Re-indexing from scratch. ` ) ;
await this . $executeQuery ( connection , 'TRUNCATE blocks;' ) ; // Need to re-index
await this . $executeQuery ( connection , ` ALTER TABLE blocks
2022-03-10 14:21:11 +01:00
ADD avg_fee INT UNSIGNED NULL ,
ADD avg_fee_rate INT UNSIGNED NULL
2022-03-09 19:18:51 +01:00
` );
2022-03-10 14:21:11 +01:00
await this . $executeQuery ( connection , 'ALTER TABLE blocks MODIFY `reward` BIGINT UNSIGNED NOT NULL DEFAULT "0"' ) ;
await this . $executeQuery ( connection , 'ALTER TABLE blocks MODIFY `median_fee` INT UNSIGNED NOT NULL DEFAULT "0"' ) ;
await this . $executeQuery ( connection , 'ALTER TABLE blocks MODIFY `fees` INT UNSIGNED NOT NULL DEFAULT "0"' ) ;
2022-03-09 19:18:51 +01:00
}
2022-03-11 20:42:07 +01:00
if ( databaseSchemaVersion < 12 && isBitcoin === true ) {
// No need to re-index because the new data type can contain larger values
await this . $executeQuery ( connection , 'ALTER TABLE blocks MODIFY `fees` BIGINT UNSIGNED NOT NULL DEFAULT "0"' ) ;
}
2022-03-12 11:06:37 +01:00
if ( databaseSchemaVersion < 13 && isBitcoin === true ) {
await this . $executeQuery ( connection , 'ALTER TABLE blocks MODIFY `difficulty` DOUBLE UNSIGNED NOT NULL DEFAULT "0"' ) ;
await this . $executeQuery ( connection , 'ALTER TABLE blocks MODIFY `median_fee` BIGINT UNSIGNED NOT NULL DEFAULT "0"' ) ;
await this . $executeQuery ( connection , 'ALTER TABLE blocks MODIFY `avg_fee` BIGINT UNSIGNED NOT NULL DEFAULT "0"' ) ;
await this . $executeQuery ( connection , 'ALTER TABLE blocks MODIFY `avg_fee_rate` BIGINT UNSIGNED NOT NULL DEFAULT "0"' ) ;
}
2022-03-12 14:47:33 +01:00
if ( databaseSchemaVersion < 14 && isBitcoin === true ) {
logger . warn ( ` 'hashrates' table has been truncated. Re-indexing from scratch. ` ) ;
await this . $executeQuery ( connection , 'TRUNCATE hashrates;' ) ; // Need to re-index
await this . $executeQuery ( connection , 'ALTER TABLE `hashrates` DROP FOREIGN KEY `hashrates_ibfk_1`' ) ;
await this . $executeQuery ( connection , 'ALTER TABLE `hashrates` MODIFY `pool_id` SMALLINT UNSIGNED NOT NULL DEFAULT "0"' ) ;
}
2022-01-12 08:41:27 +01:00
connection . release ( ) ;
} catch ( e ) {
connection . release ( ) ;
throw e ;
}
}
2022-01-12 06:10:16 +01:00
/ * *
* Special case here for the ` statistics ` table - It appeared that somehow some dbs already had the ` added ` field indexed
* while it does not appear in previous schemas . The mariadb command "CREATE INDEX IF NOT EXISTS" is not supported on
* older mariadb version . Therefore we set a flag here in order to know if the index needs to be created or not before
* running the migration process
* /
private async $setStatisticsAddedIndexedFlag ( databaseSchemaVersion : number ) {
if ( databaseSchemaVersion >= 2 ) {
this . statisticsAddedIndexed = true ;
return ;
}
2022-03-12 14:47:33 +01:00
const connection = await DB . getConnection ( ) ;
2022-01-12 06:10:16 +01:00
try {
2022-01-12 08:41:27 +01:00
// We don't use "CREATE INDEX IF NOT EXISTS" because it is not supported on old mariadb version 5.X
2022-01-12 06:10:16 +01:00
const query = ` SELECT COUNT(1) hasIndex FROM INFORMATION_SCHEMA.STATISTICS
WHERE table_schema = DATABASE ( ) AND table_name = 'statistics' AND index_name = 'added' ; ` ;
2022-01-12 08:06:45 +01:00
const [ rows ] = await this . $executeQuery ( connection , query , true ) ;
2022-01-12 06:10:16 +01:00
if ( rows [ 0 ] . hasIndex === 0 ) {
2022-02-21 15:57:44 +01:00
logger . debug ( 'MIGRATIONS: `statistics.added` is not indexed' ) ;
2022-01-12 06:10:16 +01:00
this . statisticsAddedIndexed = false ;
} else if ( rows [ 0 ] . hasIndex === 1 ) {
2022-02-21 15:57:44 +01:00
logger . debug ( 'MIGRATIONS: `statistics.added` is already indexed' ) ;
2022-01-12 06:10:16 +01:00
this . statisticsAddedIndexed = true ;
}
} catch ( e ) {
// Should really never happen but just in case it fails, we just don't execute
// any query related to this indexing so it won't fail if the index actually already exists
logger . err ( 'MIGRATIONS: Unable to check if `statistics.added` INDEX exist or not.' ) ;
this . statisticsAddedIndexed = true ;
}
connection . release ( ) ;
}
2022-01-11 12:43:59 +01:00
/ * *
* Small query execution wrapper to log all executed queries
* /
2022-01-12 08:06:45 +01:00
private async $executeQuery ( connection : PoolConnection , query : string , silent : boolean = false ) : Promise < any > {
if ( ! silent ) {
2022-02-21 15:57:44 +01:00
logger . debug ( 'MIGRATIONS: Execute query:\n' + query ) ;
2022-01-12 08:06:45 +01:00
}
2022-01-11 12:43:59 +01:00
return connection . query < any > ( { sql : query , timeout : this.queryTimeout } ) ;
2021-12-11 01:27:58 +01:00
}
2022-01-11 12:43:59 +01:00
/ * *
* Check if 'table' exists in the database
* /
private async $checkIfTableExists ( table : string ) : Promise < boolean > {
2022-03-12 14:47:33 +01:00
const connection = await DB . getConnection ( ) ;
2022-01-11 12:43:59 +01:00
const query = ` SELECT COUNT(*) FROM information_schema.tables WHERE table_schema = ' ${ config . DATABASE . DATABASE } ' AND TABLE_NAME = ' ${ table } ' ` ;
const [ rows ] = await connection . query < any > ( { sql : query , timeout : this.queryTimeout } ) ;
2021-12-11 01:27:58 +01:00
connection . release ( ) ;
2022-01-11 12:43:59 +01:00
return rows [ 0 ] [ 'COUNT(*)' ] === 1 ;
2021-12-11 01:27:58 +01:00
}
2022-01-11 12:43:59 +01:00
/ * *
* Get current database version
* /
2021-12-11 01:27:58 +01:00
private async $getSchemaVersionFromDatabase ( ) : Promise < number > {
2022-03-12 14:47:33 +01:00
const connection = await DB . getConnection ( ) ;
2021-12-11 01:27:58 +01:00
const query = ` SELECT number FROM state WHERE name = 'schema_version'; ` ;
2022-01-12 08:06:45 +01:00
const [ rows ] = await this . $executeQuery ( connection , query , true ) ;
2021-12-11 01:27:58 +01:00
connection . release ( ) ;
return rows [ 0 ] [ 'number' ] ;
}
2022-01-11 12:43:59 +01:00
/ * *
* Create the ` state ` table
* /
private async $createMigrationStateTable ( ) : Promise < void > {
2022-03-12 14:47:33 +01:00
const connection = await DB . getConnection ( ) ;
2022-01-11 12:43:59 +01:00
try {
const query = ` CREATE TABLE IF NOT EXISTS state (
name varchar ( 25 ) NOT NULL ,
number int ( 11 ) NULL ,
string varchar ( 100 ) NULL ,
CONSTRAINT name_unique UNIQUE ( name )
) ENGINE = InnoDB DEFAULT CHARSET = utf8 ; ` ;
await this . $executeQuery ( connection , query ) ;
// Set initial values
await this . $executeQuery ( connection , ` INSERT INTO state VALUES('schema_version', 0, NULL); ` ) ;
await this . $executeQuery ( connection , ` INSERT INTO state VALUES('last_elements_block', 0, NULL); ` ) ;
2022-01-12 06:10:16 +01:00
connection . release ( ) ;
2022-01-11 12:43:59 +01:00
} catch ( e ) {
connection . release ( ) ;
throw e ;
}
2021-12-11 01:27:58 +01:00
}
2022-01-11 12:43:59 +01:00
/ * *
2022-01-12 08:41:27 +01:00
* We actually execute the migrations queries here
2022-01-11 12:43:59 +01:00
* /
private async $migrateTableSchemaFromVersion ( version : number ) : Promise < void > {
2022-01-12 06:10:16 +01:00
const transactionQueries : string [ ] = [ ] ;
2022-01-11 12:43:59 +01:00
for ( const query of this . getMigrationQueriesFromVersion ( version ) ) {
transactionQueries . push ( query ) ;
}
transactionQueries . push ( this . getUpdateToLatestSchemaVersionQuery ( ) ) ;
2022-03-12 14:47:33 +01:00
const connection = await DB . getConnection ( ) ;
2022-01-11 12:43:59 +01:00
try {
2022-01-12 06:10:16 +01:00
await this . $executeQuery ( connection , 'START TRANSACTION;' ) ;
2022-01-11 12:43:59 +01:00
for ( const query of transactionQueries ) {
await this . $executeQuery ( connection , query ) ;
}
2022-01-12 06:10:16 +01:00
await this . $executeQuery ( connection , 'COMMIT;' ) ;
connection . release ( ) ;
2022-01-11 12:43:59 +01:00
} catch ( e ) {
2022-01-12 06:10:16 +01:00
await this . $executeQuery ( connection , 'ROLLBACK;' ) ;
2022-01-11 12:43:59 +01:00
connection . release ( ) ;
throw e ;
}
2021-12-11 01:27:58 +01:00
}
2022-01-11 12:43:59 +01:00
/ * *
* Generate migration queries based on schema version
* /
private getMigrationQueriesFromVersion ( version : number ) : string [ ] {
2021-12-11 01:27:58 +01:00
const queries : string [ ] = [ ] ;
2022-03-11 21:58:25 +01:00
const isBitcoin = [ 'mainnet' , 'testnet' , 'signet' ] . includes ( config . MEMPOOL . NETWORK ) ;
2021-12-11 01:27:58 +01:00
2022-01-11 12:43:59 +01:00
if ( version < 1 ) {
if ( config . MEMPOOL . NETWORK !== 'liquid' && config . MEMPOOL . NETWORK !== 'liquidtestnet' ) {
2022-01-12 06:10:16 +01:00
queries . push ( this . getShiftStatisticsQuery ( ) ) ;
2022-01-11 12:43:59 +01:00
}
}
2022-03-11 21:58:25 +01:00
if ( version < 7 && isBitcoin === true ) {
2022-02-21 04:22:20 +01:00
queries . push ( ` INSERT INTO state(name, number, string) VALUES ('last_hashrates_indexing', 0, NULL) ` ) ;
}
2022-03-11 21:58:25 +01:00
if ( version < 9 && isBitcoin === true ) {
2022-03-06 12:32:16 +01:00
queries . push ( ` INSERT INTO state(name, number, string) VALUES ('last_weekly_hashrates_indexing', 0, NULL) ` ) ;
}
2022-01-11 12:43:59 +01:00
return queries ;
}
/ * *
* Save the schema version in the database
* /
2022-01-12 06:10:16 +01:00
private getUpdateToLatestSchemaVersionQuery ( ) : string {
2022-01-11 12:43:59 +01:00
return ` UPDATE state SET number = ${ DatabaseMigration . currentVersion } WHERE name = 'schema_version'; ` ;
}
2022-01-12 08:06:45 +01:00
/ * *
* Print current database version
* /
private async $printDatabaseVersion() {
2022-03-12 14:47:33 +01:00
const connection = await DB . getConnection ( ) ;
2022-01-12 08:06:45 +01:00
try {
const [ rows ] = await this . $executeQuery ( connection , 'SELECT VERSION() as version;' , true ) ;
2022-02-21 15:57:44 +01:00
logger . debug ( ` MIGRATIONS: Database engine version ' ${ rows [ 0 ] . version } ' ` ) ;
2022-01-12 08:06:45 +01:00
} catch ( e ) {
2022-02-21 15:57:44 +01:00
logger . debug ( ` MIGRATIONS: Could not fetch database engine version. ` + e ) ;
2022-01-12 08:06:45 +01:00
}
connection . release ( ) ;
}
2022-01-11 12:43:59 +01:00
// Couple of wrappers to clean the main logic
2022-01-12 08:41:27 +01:00
private getShiftStatisticsQuery ( ) : string {
return ` UPDATE statistics SET
vsize_1 = vsize_1 + vsize_2 , vsize_2 = vsize_3 ,
vsize_3 = vsize_4 , vsize_4 = vsize_5 ,
vsize_5 = vsize_6 , vsize_6 = vsize_8 ,
vsize_8 = vsize_10 , vsize_10 = vsize_12 ,
vsize_12 = vsize_15 , vsize_15 = vsize_20 ,
vsize_20 = vsize_30 , vsize_30 = vsize_40 ,
vsize_40 = vsize_50 , vsize_50 = vsize_60 ,
vsize_60 = vsize_70 , vsize_70 = vsize_80 ,
vsize_80 = vsize_90 , vsize_90 = vsize_100 ,
vsize_100 = vsize_125 , vsize_125 = vsize_150 ,
vsize_150 = vsize_175 , vsize_175 = vsize_200 ,
vsize_200 = vsize_250 , vsize_250 = vsize_300 ,
vsize_300 = vsize_350 , vsize_350 = vsize_400 ,
vsize_400 = vsize_500 , vsize_500 = vsize_600 ,
vsize_600 = vsize_700 , vsize_700 = vsize_800 ,
vsize_800 = vsize_900 , vsize_900 = vsize_1000 ,
vsize_1000 = vsize_1200 , vsize_1200 = vsize_1400 ,
vsize_1400 = vsize_1800 , vsize_1800 = vsize_2000 , vsize_2000 = 0 ; ` ;
}
2022-01-11 12:43:59 +01:00
private getCreateStatisticsQuery ( ) : string {
return ` CREATE TABLE IF NOT EXISTS statistics (
id int ( 11 ) NOT NULL AUTO_INCREMENT ,
2021-12-11 01:27:58 +01:00
added datetime NOT NULL ,
unconfirmed_transactions int ( 11 ) UNSIGNED NOT NULL ,
tx_per_second float UNSIGNED NOT NULL ,
vbytes_per_second int ( 10 ) UNSIGNED NOT NULL ,
mempool_byte_weight int ( 10 ) UNSIGNED NOT NULL ,
fee_data longtext NOT NULL ,
total_fee double UNSIGNED NOT NULL ,
vsize_1 int ( 11 ) NOT NULL ,
vsize_2 int ( 11 ) NOT NULL ,
vsize_3 int ( 11 ) NOT NULL ,
vsize_4 int ( 11 ) NOT NULL ,
vsize_5 int ( 11 ) NOT NULL ,
vsize_6 int ( 11 ) NOT NULL ,
vsize_8 int ( 11 ) NOT NULL ,
vsize_10 int ( 11 ) NOT NULL ,
vsize_12 int ( 11 ) NOT NULL ,
vsize_15 int ( 11 ) NOT NULL ,
vsize_20 int ( 11 ) NOT NULL ,
vsize_30 int ( 11 ) NOT NULL ,
vsize_40 int ( 11 ) NOT NULL ,
vsize_50 int ( 11 ) NOT NULL ,
vsize_60 int ( 11 ) NOT NULL ,
vsize_70 int ( 11 ) NOT NULL ,
vsize_80 int ( 11 ) NOT NULL ,
vsize_90 int ( 11 ) NOT NULL ,
vsize_100 int ( 11 ) NOT NULL ,
vsize_125 int ( 11 ) NOT NULL ,
vsize_150 int ( 11 ) NOT NULL ,
vsize_175 int ( 11 ) NOT NULL ,
vsize_200 int ( 11 ) NOT NULL ,
vsize_250 int ( 11 ) NOT NULL ,
vsize_300 int ( 11 ) NOT NULL ,
vsize_350 int ( 11 ) NOT NULL ,
vsize_400 int ( 11 ) NOT NULL ,
vsize_500 int ( 11 ) NOT NULL ,
vsize_600 int ( 11 ) NOT NULL ,
vsize_700 int ( 11 ) NOT NULL ,
vsize_800 int ( 11 ) NOT NULL ,
vsize_900 int ( 11 ) NOT NULL ,
vsize_1000 int ( 11 ) NOT NULL ,
vsize_1200 int ( 11 ) NOT NULL ,
vsize_1400 int ( 11 ) NOT NULL ,
vsize_1600 int ( 11 ) NOT NULL ,
vsize_1800 int ( 11 ) NOT NULL ,
2022-01-11 12:43:59 +01:00
vsize_2000 int ( 11 ) NOT NULL ,
CONSTRAINT PRIMARY KEY ( id )
2022-01-12 08:06:45 +01:00
) ENGINE = InnoDB DEFAULT CHARSET = utf8 ; ` ;
2021-12-11 01:27:58 +01:00
}
2022-01-12 08:41:27 +01:00
2022-01-11 12:43:59 +01:00
private getCreateElementsTableQuery ( ) : string {
return ` CREATE TABLE IF NOT EXISTS elements_pegs (
block int ( 11 ) NOT NULL ,
datetime int ( 11 ) NOT NULL ,
amount bigint ( 20 ) NOT NULL ,
txid varchar ( 65 ) NOT NULL ,
txindex int ( 11 ) NOT NULL ,
bitcoinaddress varchar ( 100 ) NOT NULL ,
bitcointxid varchar ( 65 ) NOT NULL ,
bitcoinindex int ( 11 ) NOT NULL ,
final_tx int ( 11 ) NOT NULL
2022-01-12 08:06:45 +01:00
) ENGINE = InnoDB DEFAULT CHARSET = utf8 ; ` ;
2021-12-11 01:27:58 +01:00
}
2022-01-19 10:50:52 +01:00
2022-01-13 04:18:39 +01:00
private getCreatePoolsTableQuery ( ) : string {
return ` CREATE TABLE IF NOT EXISTS pools (
id int ( 11 ) NOT NULL AUTO_INCREMENT ,
name varchar ( 50 ) NOT NULL ,
link varchar ( 255 ) NOT NULL ,
addresses text NOT NULL ,
regexes text NOT NULL ,
PRIMARY KEY ( id )
) ENGINE = InnoDB DEFAULT CHARSET = utf8mb4 ; ` ;
}
private getCreateBlocksTableQuery ( ) : string {
return ` CREATE TABLE IF NOT EXISTS blocks (
height int ( 11 ) unsigned NOT NULL ,
hash varchar ( 65 ) NOT NULL ,
blockTimestamp timestamp NOT NULL ,
size int ( 11 ) unsigned NOT NULL ,
weight int ( 11 ) unsigned NOT NULL ,
tx_count int ( 11 ) unsigned NOT NULL ,
coinbase_raw text ,
difficulty bigint ( 20 ) unsigned NOT NULL ,
pool_id int ( 11 ) DEFAULT - 1 ,
fees double unsigned NOT NULL ,
fee_span json NOT NULL ,
median_fee double unsigned NOT NULL ,
PRIMARY KEY ( height ) ,
INDEX ( pool_id ) ,
FOREIGN KEY ( pool_id ) REFERENCES pools ( id )
) ENGINE = InnoDB DEFAULT CHARSET = utf8 ; ` ;
}
2022-02-19 12:45:02 +01:00
private getCreateDailyStatsTableQuery ( ) : string {
return ` CREATE TABLE IF NOT EXISTS hashrates (
hashrate_timestamp timestamp NOT NULL ,
avg_hashrate double unsigned DEFAULT '0' ,
pool_id smallint unsigned NULL ,
PRIMARY KEY ( hashrate_timestamp ) ,
INDEX ( pool_id ) ,
FOREIGN KEY ( pool_id ) REFERENCES pools ( id )
) ENGINE = InnoDB DEFAULT CHARSET = utf8 ; ` ;
}
2022-02-21 08:38:18 +01:00
public async $truncateIndexedData ( tables : string [ ] ) {
const allowedTables = [ 'blocks' , 'hashrates' ] ;
2022-03-12 14:47:33 +01:00
const connection = await DB . getConnection ( ) ;
2022-02-21 08:38:18 +01:00
try {
for ( const table of tables ) {
if ( ! allowedTables . includes ( table ) ) {
2022-02-21 15:57:44 +01:00
logger . debug ( ` Table ${ table } cannot to be re-indexed (not allowed) ` ) ;
2022-02-21 08:38:18 +01:00
continue ;
} ;
await this . $executeQuery ( connection , ` TRUNCATE ${ table } ` , true ) ;
if ( table === 'hashrates' ) {
await this . $executeQuery ( connection , 'UPDATE state set number = 0 where name = "last_hashrates_indexing"' , true ) ;
}
2022-02-21 15:57:44 +01:00
logger . notice ( ` Table ${ table } has been truncated ` ) ;
2022-02-21 08:38:18 +01:00
}
} catch ( e ) {
logger . warn ( ` Unable to erase indexed data ` ) ;
}
connection . release ( ) ;
}
2021-12-11 01:27:58 +01:00
}
2022-01-05 18:26:07 +01:00
export default new DatabaseMigration ( ) ;