// // Copyright 2021 Vulcanize, Inc. // import assert from 'assert'; import 'reflect-metadata'; import debug from 'debug'; import path from 'path'; import fs from 'fs'; import { ContractInterface, utils, providers } from 'ethers'; import { SelectionNode } from 'graphql'; import { ResultObject } from '@vulcanize/assemblyscript/lib/loader'; import { EthClient } from '@cerc-io/ipld-eth-client'; import { getFullBlock, BlockHeight, ServerConfig, getFullTransaction, QueryOptions, IPLDBlockInterface, IPLDIndexerInterface, BlockProgressInterface } from '@cerc-io/util'; import { createBlock, createEvent, getSubgraphConfig, resolveEntityFieldConflicts, Transaction } from './utils'; import { Context, GraphData, instantiate } from './loader'; import { Database, DEFAULT_LIMIT } from './database'; const log = debug('vulcanize:graph-watcher'); interface DataSource { instance?: ResultObject & { exports: any }, contractInterface: utils.Interface, data: GraphData, } export class GraphWatcher { _database: Database; _indexer?: IPLDIndexerInterface; _ethClient: EthClient; _ethProvider: providers.BaseProvider; _subgraphPath: string; _wasmRestartBlocksInterval: number; _dataSources: any[] = []; _dataSourceMap: { [key: string]: DataSource } = {}; _transactionsMap: Map = new Map() _context: Context = {}; constructor (database: Database, ethClient: EthClient, ethProvider: providers.BaseProvider, serverConfig: ServerConfig) { this._database = database; this._ethClient = ethClient; this._ethProvider = ethProvider; this._subgraphPath = serverConfig.subgraphPath; this._wasmRestartBlocksInterval = serverConfig.wasmRestartBlocksInterval; } async init () { const { dataSources, templates = [] } = await getSubgraphConfig(this._subgraphPath); this._dataSources = dataSources.concat(templates); // Create wasm instance and contract interface for each dataSource and template in subgraph yaml. const dataPromises = this._dataSources.map(async (dataSource: any) => { const { source: { abi }, mapping, network, name } = dataSource; const { abis, file } = mapping; const abisMap = abis.reduce((acc: {[key: string]: ContractInterface}, abi: any) => { const { name, file } = abi; const abiFilePath = path.join(this._subgraphPath, file); acc[name] = JSON.parse(fs.readFileSync(abiFilePath).toString()); return acc; }, {}); const contractInterface = new utils.Interface(abisMap[abi]); const data = { abis: abisMap, dataSource: { network, name } }; const filePath = path.join(this._subgraphPath, file); assert(this._indexer); return { instance: await instantiate(this._database, this._indexer, this._ethProvider, this._context, filePath, data), contractInterface, data }; }, {}); const data = await Promise.all(dataPromises); // Create a map from dataSource contract address to instance and contract interface. this._dataSourceMap = this._dataSources.reduce((acc: { [key: string]: DataSource }, dataSource: any, index: number) => { const { instance } = data[index]; // Important to call _start for built subgraphs on instantiation! // TODO: Check api version https://github.com/graphprotocol/graph-node/blob/6098daa8955bdfac597cec87080af5449807e874/runtime/wasm/src/module/mod.rs#L533 instance.exports._start(); const { name } = dataSource; acc[name] = data[index]; return acc; }, {}); } get dataSources (): any[] { return this._dataSources; } async addContracts () { assert(this._indexer); assert(this._indexer.watchContract); assert(this._indexer.isWatchedContract); // Watching the contract(s) if not watched already. for (const dataSource of this._dataSources) { const { source: { address, startBlock }, name } = dataSource; // Skip for templates as they are added dynamically. if (address) { const watchedContract = await this._indexer.isWatchedContract(address); if (!watchedContract) { await this._indexer.watchContract(address, name, true, startBlock); } } } } async handleEvent (eventData: any) { const { contract, event, eventSignature, block, tx: { hash: txHash }, eventIndex } = eventData; // Check if block data is already fetched by a previous event in the same block. if (!this._context.block || this._context.block.blockHash !== block.hash) { this._context.block = await getFullBlock(this._ethClient, this._ethProvider, block.hash); } const blockData = this._context.block; assert(blockData); assert(this._indexer && this._indexer.isWatchedContract); const watchedContract = await this._indexer.isWatchedContract(contract); assert(watchedContract); // Get dataSource in subgraph yaml based on contract address. const dataSource = this._dataSources.find(dataSource => dataSource.name === watchedContract.kind); if (!dataSource) { log(`Subgraph doesn't have configuration for contract ${contract}`); return; } this._context.contractAddress = contract; const { instance, contractInterface } = this._dataSourceMap[watchedContract.kind]; assert(instance); const { exports: instanceExports } = instance; // Get event handler based on event topic (from event signature). const eventTopic = contractInterface.getEventTopic(eventSignature); const eventHandler = dataSource.mapping.eventHandlers.find((eventHandler: any) => { // The event signature we get from logDescription is different than that given in the subgraph yaml file. // For eg. event in subgraph.yaml: Stake(indexed address,uint256); from logDescription: Stake(address,uint256) // ethers.js doesn't recognize the subgraph event signature with indexed keyword before param type. // Match event topics from cleaned subgraph event signature (Stake(indexed address,uint256) -> Stake(address,uint256)). const subgraphEventTopic = contractInterface.getEventTopic(eventHandler.event.replace(/indexed /g, '')); return subgraphEventTopic === eventTopic; }); if (!eventHandler) { log(`No handler configured in subgraph for event ${eventSignature}`); return; } const eventFragment = contractInterface.getEvent(eventSignature); const tx = await this._getTransactionData(txHash); const data = { block: blockData, inputs: eventFragment.inputs, event, tx, eventIndex }; // Create ethereum event to be passed to the wasm event handler. const ethereumEvent = await createEvent(instanceExports, contract, data); await this._handleMemoryError(instanceExports[eventHandler.handler](ethereumEvent), dataSource.name); } async handleBlock (blockHash: string) { // Check if block data is already fetched in handleEvent method for the same block. if (!this._context.block || this._context.block.blockHash !== blockHash) { this._context.block = await getFullBlock(this._ethClient, this._ethProvider, blockHash); } const blockData = this._context.block; assert(blockData); // Clear transactions map on handling new block. this._transactionsMap.clear(); // Call block handler(s) for each contract. for (const dataSource of this._dataSources) { // Reinstantiate WASM after every N blocks. if (Number(blockData.blockNumber) % this._wasmRestartBlocksInterval === 0) { // The WASM instance allocates memory as required and the limit is 4GB. // https://stackoverflow.com/a/40453962 // https://github.com/AssemblyScript/assemblyscript/pull/1268#issue-618411291 // https://github.com/WebAssembly/memory64/blob/main/proposals/memory64/Overview.md#motivation await this._reInitWasm(dataSource.name); } // Check if block handler(s) are configured. if (!dataSource.mapping.blockHandlers) { continue; } const { instance } = this._dataSourceMap[dataSource.name]; assert(instance); const { exports: instanceExports } = instance; // Create ethereum block to be passed to a wasm block handler. const ethereumBlock = await createBlock(instanceExports, blockData); let contractAddressList: string[] = []; if (dataSource.source.address) { // Check if start block has been reached. if (blockData.blockNumber >= dataSource.source.startBlock) { contractAddressList.push(dataSource.source.address); } } else { // Data source templates will have multiple watched contracts. assert(this._indexer?.getContractsByKind); const watchedContracts = this._indexer.getContractsByKind(dataSource.name); contractAddressList = watchedContracts.filter(contract => Number(blockData.blockNumber) >= contract.startingBlock) .map(contract => contract.address); } for (const contractAddress of contractAddressList) { this._context.contractAddress = contractAddress; // Call all the block handlers one after another for a contract. const blockHandlerPromises = dataSource.mapping.blockHandlers.map(async (blockHandler: any): Promise => { await instanceExports[blockHandler.handler](ethereumBlock); }); await this._handleMemoryError(Promise.all(blockHandlerPromises), dataSource.name); } } } setIndexer (indexer: IPLDIndexerInterface): void { this._indexer = indexer; } async getEntity ( entity: new () => Entity, id: string, relationsMap: Map, block: BlockHeight, selections: ReadonlyArray = [] ): Promise { const dbTx = await this._database.createTransactionRunner(); try { // Get entity from the database. const result = await this._database.getEntityWithRelations(dbTx, entity, id, relationsMap, block, selections); await dbTx.commitTransaction(); // Resolve any field name conflicts in the entity result. return resolveEntityFieldConflicts(result); } catch (error) { await dbTx.rollbackTransaction(); throw error; } finally { await dbTx.release(); } } async getEntities ( entity: new () => Entity, relationsMap: Map, block: BlockHeight, where: { [key: string]: any } = {}, queryOptions: QueryOptions, selections: ReadonlyArray = [] ): Promise { const dbTx = await this._database.createTransactionRunner(); try { where = Object.entries(where).reduce((acc: { [key: string]: any }, [fieldWithSuffix, value]) => { const [field, ...suffix] = fieldWithSuffix.split('_'); if (!acc[field]) { acc[field] = []; } const filter = { value, not: false, operator: 'equals' }; let operator = suffix.shift(); if (operator === 'not') { filter.not = true; operator = suffix.shift(); } if (operator) { filter.operator = operator; } acc[field].push(filter); return acc; }, {}); if (!queryOptions.limit) { queryOptions.limit = DEFAULT_LIMIT; } // Get entities from the database. const entities = await this._database.getEntities(dbTx, entity, relationsMap, block, where, queryOptions, selections); await dbTx.commitTransaction(); // Resolve any field name conflicts in the entity result. return entities.map(entity => resolveEntityFieldConflicts(entity)); } catch (error) { await dbTx.rollbackTransaction(); throw error; } finally { await dbTx.release(); } } async updateEntitiesFromIPLDState (ipldBlock: IPLDBlockInterface) { assert(this._indexer); const data = this._indexer.getIPLDData(ipldBlock); for (const [entityName, entities] of Object.entries(data.state)) { // Get relations for subgraph entity assert(this._indexer.getRelationsMap); const relationsMap = this._indexer.getRelationsMap(); const result = Array.from(relationsMap.entries()) .find(([key]) => key.name === entityName); const relations = result ? result[1] : {}; log(`Updating entities from IPLD state for entity ${entityName}`); console.time(`time:watcher#GraphWatcher-updateEntitiesFromIPLDState-IPLD-update-entity-${entityName}`); for (const [id, entityData] of Object.entries(entities as any)) { const dbData = this._database.fromIPLDState(ipldBlock.block, entityName, entityData, relations); await this._database.saveEntity(entityName, dbData); } console.timeEnd(`time:watcher#GraphWatcher-updateEntitiesFromIPLDState-IPLD-update-entity-${entityName}`); } } updateEntityCacheFrothyBlocks (blockProgress: BlockProgressInterface): void { // Set latest block in frothy region to cachedEntities.frothyBlocks map. if (!this._database.cachedEntities.frothyBlocks.has(blockProgress.blockHash)) { this._database.cachedEntities.frothyBlocks.set( blockProgress.blockHash, { blockNumber: blockProgress.blockNumber, parentHash: blockProgress.parentHash, entities: new Map() } ); log(`Size of cachedEntities.frothyBlocks map: ${this._database.cachedEntities.frothyBlocks.size}`); } } pruneEntityCacheFrothyBlocks (canonicalBlockHash: string, canonicalBlockNumber: number) { const canonicalBlock = this._database.cachedEntities.frothyBlocks.get(canonicalBlockHash); if (canonicalBlock) { // Update latestPrunedEntities map with entities from latest canonical block. canonicalBlock.entities.forEach((entityIdMap, entityTableName) => { entityIdMap.forEach((data, id) => { let entityIdMap = this._database.cachedEntities.latestPrunedEntities.get(entityTableName); if (!entityIdMap) { entityIdMap = new Map(); } entityIdMap.set(id, data); this._database.cachedEntities.latestPrunedEntities.set(entityTableName, entityIdMap); }); }); } // Remove pruned blocks from frothyBlocks. const prunedBlockHashes = Array.from(this._database.cachedEntities.frothyBlocks.entries()) .filter(([, value]) => value.blockNumber <= canonicalBlockNumber) .map(([blockHash]) => blockHash); prunedBlockHashes.forEach(blockHash => this._database.cachedEntities.frothyBlocks.delete(blockHash)); } /** * Method to reinstantiate WASM instance for specified dataSource. * @param dataSourceName */ async _reInitWasm (dataSourceName: string): Promise { const { data, instance } = this._dataSourceMap[dataSourceName]; assert(instance); const { module } = instance; delete this._dataSourceMap[dataSourceName].instance; assert(this._indexer); // Reinstantiate with existing module. this._dataSourceMap[dataSourceName].instance = await instantiate( this._database, this._indexer, this._ethProvider, this._context, module, data ); // Important to call _start for built subgraphs on instantiation! // TODO: Check api version https://github.com/graphprotocol/graph-node/blob/6098daa8955bdfac597cec87080af5449807e874/runtime/wasm/src/module/mod.rs#L533 this._dataSourceMap[dataSourceName].instance!.exports._start(); } async _handleMemoryError (handlerPromise: Promise, dataSourceName: string): Promise { try { await handlerPromise; } catch (error) { if (error instanceof WebAssembly.RuntimeError && error instanceof Error) { if (error.message === 'unreachable') { // Reintantiate WASM for out of memory error. this._reInitWasm(dataSourceName); } } // Job will retry after throwing error. throw error; } } async _getTransactionData (txHash: string): Promise { let transaction = this._transactionsMap.get(txHash); if (transaction) { return transaction; } transaction = await getFullTransaction(this._ethClient, txHash); assert(transaction); this._transactionsMap.set(txHash, transaction); return transaction; } }