// // 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 { ResultObject } from '@vulcanize/assemblyscript/lib/loader'; import { EthClient } from '@vulcanize/ipld-eth-client'; import { IndexerInterface, getFullBlock, BlockHeight } from '@vulcanize/util'; import { createBlock, createEvent, getSubgraphConfig, resolveEntityFieldConflicts } from './utils'; import { Context, instantiate } from './loader'; import { Database } from './database'; const log = debug('vulcanize:graph-watcher'); interface DataSource { instance: ResultObject & { exports: any }, contractInterface: utils.Interface } export class GraphWatcher { _database: Database; _indexer?: IndexerInterface; _postgraphileClient: EthClient; _ethProvider: providers.BaseProvider; _subgraphPath: string; _dataSources: any[] = []; _dataSourceMap: { [key: string]: DataSource } = {}; _context: Context = { event: {} } constructor (database: Database, postgraphileClient: EthClient, ethProvider: providers.BaseProvider, subgraphPath: string) { this._database = database; this._postgraphileClient = postgraphileClient; this._ethProvider = ethProvider; this._subgraphPath = subgraphPath; } async init () { const { dataSources } = await getSubgraphConfig(this._subgraphPath); this._dataSources = dataSources; // Create wasm instance and contract interface for each dataSource in subgraph yaml. const dataPromises = this._dataSources.map(async (dataSource: any) => { const { source: { address, abi }, mapping } = 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: { address } }; const filePath = path.join(this._subgraphPath, file); assert(this._indexer); return { instance: await instantiate(this._database, this._indexer, this._context, filePath, data), contractInterface }; }, {}); 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 { source: { address } } = dataSource; acc[address] = data[index]; return acc; }, {}); } async addContracts () { assert(this._indexer?.watchContract); // Watching the contract(s). for (const dataSource of this._dataSources) { const { source: { address, startBlock }, name } = dataSource; await this._indexer.watchContract(address, name, true, startBlock); } } async handleEvent (eventData: any) { const { contract, event, eventSignature, block, tx, eventIndex } = eventData; // TODO: Use blockData fetched in handleBlock. const blockData = await getFullBlock(this._postgraphileClient, this._ethProvider, block.hash); this._context.event.block = blockData; // Get dataSource in subgraph yaml based on contract address. const dataSource = this._dataSources.find(dataSource => dataSource.source.address === contract); if (!dataSource) { log(`Subgraph doesnt have configuration for contract ${contract}`); return; } // Get event handler based on event signature. const eventHandler = dataSource.mapping.eventHandlers.find((eventHandler: any) => eventHandler.event === eventSignature); if (!eventHandler) { log(`No handler configured in subgraph for event ${eventSignature}`); return; } const { instance: { exports: instanceExports }, contractInterface } = this._dataSourceMap[contract]; const eventFragment = contractInterface.getEvent(eventSignature); 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 instanceExports[eventHandler.handler](ethereumEvent); } async handleBlock (blockHash: string) { const blockData = await getFullBlock(this._postgraphileClient, this._ethProvider, blockHash); this._context.event.block = blockData; // Call block handler(s) for each contract. for (const dataSource of this._dataSources) { // Check if block handler(s) are configured. if (!dataSource.mapping.blockHandlers) { continue; } const { instance: { exports: instanceExports } } = this._dataSourceMap[dataSource.source.address]; // Create ethereum block to be passed to a wasm block handler. const ethereumBlock = await createBlock(instanceExports, blockData); // Call all the block handlers one after the another for a contract. const blockHandlerPromises = dataSource.mapping.blockHandlers.map(async (blockHandler: any): Promise => { await instanceExports[blockHandler.handler](ethereumBlock); }); await Promise.all(blockHandlerPromises); } } setIndexer (indexer: IndexerInterface): void { this._indexer = indexer; } async getEntity (entity: new () => Entity, id: string, relations: { [key: string]: any }, block?: BlockHeight): Promise { // Get entity from the database. const result = await this._database.getEntityWithRelations(entity, id, relations, block) as any; // Resolve any field name conflicts in the entity result. return resolveEntityFieldConflicts(result); } }