2021-08-19 07:57:32 +00:00
|
|
|
//
|
|
|
|
// Copyright 2021 Vulcanize, Inc.
|
|
|
|
//
|
|
|
|
|
|
|
|
import assert from 'assert';
|
2021-09-21 11:13:55 +00:00
|
|
|
import { DeepPartial, FindConditions, Not } from 'typeorm';
|
2021-08-20 13:32:57 +00:00
|
|
|
import debug from 'debug';
|
2021-09-21 11:13:55 +00:00
|
|
|
import { ethers } from 'ethers';
|
2021-08-19 07:57:32 +00:00
|
|
|
|
2021-08-20 13:32:57 +00:00
|
|
|
import { EthClient } from '@vulcanize/ipld-eth-client';
|
2021-09-21 11:13:55 +00:00
|
|
|
import { GetStorageAt, getStorageValue, StorageLayout } from '@vulcanize/solidity-mapper';
|
2021-08-20 13:32:57 +00:00
|
|
|
|
2021-09-21 11:13:55 +00:00
|
|
|
import { BlockProgressInterface, DatabaseInterface, EventInterface, SyncStatusInterface, ContractInterface } from './types';
|
|
|
|
import { UNKNOWN_EVENT_NAME } from './constants';
|
2021-08-20 13:32:57 +00:00
|
|
|
|
2021-08-24 06:25:29 +00:00
|
|
|
const MAX_EVENTS_BLOCK_RANGE = 1000;
|
|
|
|
|
2021-08-20 13:32:57 +00:00
|
|
|
const log = debug('vulcanize:indexer');
|
2021-08-19 07:57:32 +00:00
|
|
|
|
2021-09-21 11:13:55 +00:00
|
|
|
export interface ValueResult {
|
|
|
|
value: any;
|
|
|
|
proof?: {
|
|
|
|
data: string;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-08-19 07:57:32 +00:00
|
|
|
export class Indexer {
|
|
|
|
_db: DatabaseInterface;
|
2021-12-03 10:53:11 +00:00
|
|
|
_postgraphileClient: EthClient;
|
2021-10-22 09:50:11 +00:00
|
|
|
_getStorageAt: GetStorageAt;
|
|
|
|
_ethProvider: ethers.providers.BaseProvider;
|
2021-08-19 07:57:32 +00:00
|
|
|
|
2021-10-22 09:50:11 +00:00
|
|
|
constructor (db: DatabaseInterface, ethClient: EthClient, ethProvider: ethers.providers.BaseProvider) {
|
2021-08-19 07:57:32 +00:00
|
|
|
this._db = db;
|
2021-12-03 10:53:11 +00:00
|
|
|
this._postgraphileClient = ethClient;
|
2021-10-22 09:50:11 +00:00
|
|
|
this._ethProvider = ethProvider;
|
2021-12-03 10:53:11 +00:00
|
|
|
this._getStorageAt = this._postgraphileClient.getStorageAt.bind(this._postgraphileClient);
|
2021-08-20 13:32:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
async getSyncStatus (): Promise<SyncStatusInterface | undefined> {
|
|
|
|
const dbTx = await this._db.createTransactionRunner();
|
|
|
|
let res;
|
|
|
|
|
|
|
|
try {
|
|
|
|
res = await this._db.getSyncStatus(dbTx);
|
|
|
|
await dbTx.commitTransaction();
|
|
|
|
} catch (error) {
|
|
|
|
await dbTx.rollbackTransaction();
|
|
|
|
throw error;
|
|
|
|
} finally {
|
|
|
|
await dbTx.release();
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
2021-08-19 07:57:32 +00:00
|
|
|
}
|
|
|
|
|
2021-10-20 10:36:03 +00:00
|
|
|
async updateSyncStatusIndexedBlock (blockHash: string, blockNumber: number, force = false): Promise<SyncStatusInterface> {
|
2021-08-19 07:57:32 +00:00
|
|
|
const dbTx = await this._db.createTransactionRunner();
|
|
|
|
let res;
|
|
|
|
|
|
|
|
try {
|
2021-10-20 10:36:03 +00:00
|
|
|
res = await this._db.updateSyncStatusIndexedBlock(dbTx, blockHash, blockNumber, force);
|
2021-08-19 07:57:32 +00:00
|
|
|
await dbTx.commitTransaction();
|
|
|
|
} catch (error) {
|
|
|
|
await dbTx.rollbackTransaction();
|
|
|
|
throw error;
|
|
|
|
} finally {
|
|
|
|
await dbTx.release();
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
async updateSyncStatusChainHead (blockHash: string, blockNumber: number): Promise<SyncStatusInterface> {
|
|
|
|
const dbTx = await this._db.createTransactionRunner();
|
|
|
|
let res;
|
|
|
|
|
|
|
|
try {
|
|
|
|
res = await this._db.updateSyncStatusChainHead(dbTx, blockHash, blockNumber);
|
|
|
|
await dbTx.commitTransaction();
|
|
|
|
} catch (error) {
|
|
|
|
await dbTx.rollbackTransaction();
|
|
|
|
throw error;
|
|
|
|
} finally {
|
|
|
|
await dbTx.release();
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2021-10-20 10:36:03 +00:00
|
|
|
async updateSyncStatusCanonicalBlock (blockHash: string, blockNumber: number, force = false): Promise<SyncStatusInterface> {
|
2021-08-19 07:57:32 +00:00
|
|
|
const dbTx = await this._db.createTransactionRunner();
|
|
|
|
let res;
|
|
|
|
|
|
|
|
try {
|
2021-10-20 10:36:03 +00:00
|
|
|
res = await this._db.updateSyncStatusCanonicalBlock(dbTx, blockHash, blockNumber, force);
|
2021-08-19 07:57:32 +00:00
|
|
|
await dbTx.commitTransaction();
|
|
|
|
} catch (error) {
|
|
|
|
await dbTx.rollbackTransaction();
|
|
|
|
throw error;
|
|
|
|
} finally {
|
|
|
|
await dbTx.release();
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2021-12-03 10:53:11 +00:00
|
|
|
async getBlocks (blockFilter: { blockNumber?: number, blockHash?: string }): Promise<any> {
|
|
|
|
assert(blockFilter.blockHash || blockFilter.blockNumber);
|
|
|
|
const result = await this._postgraphileClient.getBlocks(blockFilter);
|
|
|
|
const { allEthHeaderCids: { nodes: blocks } } = result;
|
|
|
|
|
|
|
|
if (!blocks.length) {
|
|
|
|
try {
|
|
|
|
const blockHashOrNumber = blockFilter.blockHash || blockFilter.blockNumber as string | number;
|
|
|
|
await this._ethProvider.getBlock(blockHashOrNumber);
|
|
|
|
} catch (error: any) {
|
|
|
|
// eth_getBlockByHash will update statediff but takes some time.
|
|
|
|
// The block is not returned immediately and an error is thrown so that it is fetched in the next job retry.
|
|
|
|
if (error.code !== ethers.utils.Logger.errors.SERVER_ERROR) {
|
2021-10-22 09:50:11 +00:00
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
|
2021-12-03 10:53:11 +00:00
|
|
|
log('Block not found. Fetching block after eth_call.');
|
|
|
|
}
|
2021-10-22 09:50:11 +00:00
|
|
|
}
|
2021-12-03 10:53:11 +00:00
|
|
|
|
|
|
|
return blocks;
|
2021-08-20 13:32:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
async getBlockProgress (blockHash: string): Promise<BlockProgressInterface | undefined> {
|
|
|
|
return this._db.getBlockProgress(blockHash);
|
|
|
|
}
|
|
|
|
|
2021-08-19 07:57:32 +00:00
|
|
|
async getBlocksAtHeight (height: number, isPruned: boolean): Promise<BlockProgressInterface[]> {
|
|
|
|
return this._db.getBlocksAtHeight(height, isPruned);
|
|
|
|
}
|
|
|
|
|
2021-08-23 11:36:35 +00:00
|
|
|
async markBlocksAsPruned (blocks: BlockProgressInterface[]): Promise<void> {
|
2021-08-19 07:57:32 +00:00
|
|
|
const dbTx = await this._db.createTransactionRunner();
|
|
|
|
|
|
|
|
try {
|
2021-08-23 11:36:35 +00:00
|
|
|
await this._db.markBlocksAsPruned(dbTx, blocks);
|
2021-08-19 07:57:32 +00:00
|
|
|
await dbTx.commitTransaction();
|
|
|
|
} catch (error) {
|
|
|
|
await dbTx.rollbackTransaction();
|
|
|
|
throw error;
|
|
|
|
} finally {
|
|
|
|
await dbTx.release();
|
|
|
|
}
|
|
|
|
}
|
2021-08-20 13:32:57 +00:00
|
|
|
|
|
|
|
async updateBlockProgress (blockHash: string, lastProcessedEventIndex: number): Promise<void> {
|
|
|
|
const dbTx = await this._db.createTransactionRunner();
|
|
|
|
let res;
|
|
|
|
|
|
|
|
try {
|
|
|
|
res = await this._db.updateBlockProgress(dbTx, blockHash, lastProcessedEventIndex);
|
|
|
|
await dbTx.commitTransaction();
|
|
|
|
} catch (error) {
|
|
|
|
await dbTx.rollbackTransaction();
|
|
|
|
throw error;
|
|
|
|
} finally {
|
|
|
|
await dbTx.release();
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
async getEvent (id: string): Promise<EventInterface | undefined> {
|
|
|
|
return this._db.getEvent(id);
|
|
|
|
}
|
|
|
|
|
|
|
|
async getOrFetchBlockEvents (block: DeepPartial<BlockProgressInterface>, fetchAndSaveEvents: (block: DeepPartial<BlockProgressInterface>) => Promise<void>): Promise<Array<EventInterface>> {
|
|
|
|
assert(block.blockHash);
|
|
|
|
const blockProgress = await this._db.getBlockProgress(block.blockHash);
|
|
|
|
if (!blockProgress) {
|
|
|
|
// Fetch and save events first and make a note in the event sync progress table.
|
|
|
|
log(`getBlockEvents: db miss, fetching from upstream server ${block.blockHash}`);
|
|
|
|
await fetchAndSaveEvents(block);
|
|
|
|
}
|
|
|
|
|
|
|
|
const events = await this._db.getBlockEvents(block.blockHash);
|
|
|
|
log(`getBlockEvents: db hit, ${block.blockHash} num events: ${events.length}`);
|
|
|
|
|
|
|
|
return events;
|
|
|
|
}
|
|
|
|
|
|
|
|
async getBlockEvents (blockHash: string): Promise<Array<EventInterface>> {
|
|
|
|
return this._db.getBlockEvents(blockHash);
|
|
|
|
}
|
2021-08-23 11:36:35 +00:00
|
|
|
|
2021-09-21 11:13:55 +00:00
|
|
|
async getEventsByFilter (blockHash: string, contract: string, name: string | null): Promise<Array<EventInterface>> {
|
|
|
|
if (contract) {
|
|
|
|
const watchedContract = await this.isWatchedContract(contract);
|
|
|
|
if (!watchedContract) {
|
|
|
|
throw new Error('Not a watched contract');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const where: FindConditions<EventInterface> = {
|
|
|
|
eventName: Not(UNKNOWN_EVENT_NAME)
|
|
|
|
};
|
|
|
|
|
|
|
|
if (contract) {
|
|
|
|
where.contract = contract;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (name) {
|
|
|
|
where.eventName = name;
|
|
|
|
}
|
|
|
|
|
|
|
|
const events = await this._db.getBlockEvents(blockHash, where);
|
|
|
|
log(`getEvents: db hit, num events: ${events.length}`);
|
|
|
|
|
|
|
|
return events;
|
|
|
|
}
|
|
|
|
|
2021-10-20 12:19:44 +00:00
|
|
|
async removeUnknownEvents (eventEntityClass: new () => EventInterface, block: BlockProgressInterface): Promise<void> {
|
|
|
|
const dbTx = await this._db.createTransactionRunner();
|
|
|
|
|
|
|
|
try {
|
|
|
|
await this._db.removeEntities(
|
|
|
|
dbTx,
|
|
|
|
eventEntityClass,
|
|
|
|
{
|
|
|
|
where: {
|
|
|
|
block: { id: block.id },
|
|
|
|
eventName: UNKNOWN_EVENT_NAME
|
|
|
|
},
|
|
|
|
relations: ['block']
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
await dbTx.commitTransaction();
|
|
|
|
} catch (error) {
|
|
|
|
await dbTx.rollbackTransaction();
|
|
|
|
throw error;
|
|
|
|
} finally {
|
|
|
|
await dbTx.release();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-08-23 11:36:35 +00:00
|
|
|
async getAncestorAtDepth (blockHash: string, depth: number): Promise<string> {
|
|
|
|
return this._db.getAncestorAtDepth(blockHash, depth);
|
|
|
|
}
|
2021-08-24 06:25:29 +00:00
|
|
|
|
|
|
|
async saveEventEntity (dbEvent: EventInterface): Promise<EventInterface> {
|
|
|
|
const dbTx = await this._db.createTransactionRunner();
|
|
|
|
let res;
|
|
|
|
|
|
|
|
try {
|
2021-12-02 09:58:03 +00:00
|
|
|
res = await this._db.saveEventEntity(dbTx, dbEvent);
|
2021-08-24 06:25:29 +00:00
|
|
|
await dbTx.commitTransaction();
|
|
|
|
} catch (error) {
|
|
|
|
await dbTx.rollbackTransaction();
|
|
|
|
throw error;
|
|
|
|
} finally {
|
|
|
|
await dbTx.release();
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
async getProcessedBlockCountForRange (fromBlockNumber: number, toBlockNumber: number): Promise<{ expected: number, actual: number }> {
|
|
|
|
return this._db.getProcessedBlockCountForRange(fromBlockNumber, toBlockNumber);
|
|
|
|
}
|
|
|
|
|
|
|
|
async getEventsInRange (fromBlockNumber: number, toBlockNumber: number): Promise<Array<EventInterface>> {
|
|
|
|
if (toBlockNumber <= fromBlockNumber) {
|
|
|
|
throw new Error('toBlockNumber should be greater than fromBlockNumber');
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((toBlockNumber - fromBlockNumber) > MAX_EVENTS_BLOCK_RANGE) {
|
|
|
|
throw new Error(`Max range (${MAX_EVENTS_BLOCK_RANGE}) exceeded`);
|
|
|
|
}
|
|
|
|
|
|
|
|
return this._db.getEventsInRange(fromBlockNumber, toBlockNumber);
|
|
|
|
}
|
2021-09-21 11:13:55 +00:00
|
|
|
|
|
|
|
async isWatchedContract (address : string): Promise<ContractInterface | undefined> {
|
|
|
|
assert(this._db.getContract);
|
|
|
|
|
|
|
|
return this._db.getContract(ethers.utils.getAddress(address));
|
|
|
|
}
|
|
|
|
|
|
|
|
async getStorageValue (storageLayout: StorageLayout, blockHash: string, token: string, variable: string, ...mappingKeys: any[]): Promise<ValueResult> {
|
|
|
|
return getStorageValue(
|
|
|
|
storageLayout,
|
|
|
|
this._getStorageAt,
|
|
|
|
blockHash,
|
|
|
|
token,
|
|
|
|
variable,
|
|
|
|
...mappingKeys
|
|
|
|
);
|
|
|
|
}
|
2021-08-19 07:57:32 +00:00
|
|
|
}
|