mirror of
https://github.com/cerc-io/watcher-ts
synced 2025-01-09 12:58:06 +00:00
Save watched contracts in db. (#122)
This commit is contained in:
parent
0b6a33561b
commit
9c60895352
@ -29,6 +29,3 @@
|
|||||||
name = "requests"
|
name = "requests"
|
||||||
enabled = false
|
enabled = false
|
||||||
deleteOnStart = false
|
deleteOnStart = false
|
||||||
|
|
||||||
[contracts]
|
|
||||||
factory = "0xECfCae19742EE4644b209d75cF15ee6D02A726Ff"
|
|
||||||
|
51
packages/uni-watcher/src/cli/watch-contract.ts
Normal file
51
packages/uni-watcher/src/cli/watch-contract.ts
Normal file
@ -0,0 +1,51 @@
|
|||||||
|
import assert from 'assert';
|
||||||
|
import yargs from 'yargs';
|
||||||
|
import 'reflect-metadata';
|
||||||
|
import { ethers } from 'ethers';
|
||||||
|
|
||||||
|
import { Config, getConfig } from '../config';
|
||||||
|
import { Database } from '../database';
|
||||||
|
|
||||||
|
(async () => {
|
||||||
|
const argv = await yargs.parserConfiguration({
|
||||||
|
'parse-numbers': false
|
||||||
|
}).options({
|
||||||
|
configFile: {
|
||||||
|
type: 'string',
|
||||||
|
require: true,
|
||||||
|
demandOption: true,
|
||||||
|
describe: 'configuration file path (toml)'
|
||||||
|
},
|
||||||
|
address: {
|
||||||
|
type: 'string',
|
||||||
|
require: true,
|
||||||
|
demandOption: true,
|
||||||
|
describe: 'Address of the deployed contract'
|
||||||
|
},
|
||||||
|
kind: {
|
||||||
|
type: 'string',
|
||||||
|
require: true,
|
||||||
|
demandOption: true,
|
||||||
|
describe: 'Kind of contract (factory|pool)'
|
||||||
|
},
|
||||||
|
startingBlock: {
|
||||||
|
type: 'number',
|
||||||
|
default: 1,
|
||||||
|
describe: 'Starting block'
|
||||||
|
}
|
||||||
|
}).argv;
|
||||||
|
|
||||||
|
const config: Config = await getConfig(argv.configFile);
|
||||||
|
const { database: dbConfig } = config;
|
||||||
|
|
||||||
|
assert(dbConfig);
|
||||||
|
|
||||||
|
const db = new Database(dbConfig);
|
||||||
|
await db.init();
|
||||||
|
|
||||||
|
// Always use the checksum address (https://docs.ethers.io/v5/api/utils/address/#utils-getAddress).
|
||||||
|
const address = ethers.utils.getAddress(argv.address);
|
||||||
|
|
||||||
|
await db.saveContract(address, argv.kind, argv.startingBlock);
|
||||||
|
await db.close();
|
||||||
|
})();
|
@ -19,9 +19,6 @@ export interface Config {
|
|||||||
gqlSubscriptionEndpoint: string;
|
gqlSubscriptionEndpoint: string;
|
||||||
cache: CacheConfig
|
cache: CacheConfig
|
||||||
}
|
}
|
||||||
contracts: {
|
|
||||||
factory: string;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export const getConfig = async (configFile: string): Promise<Config> => {
|
export const getConfig = async (configFile: string): Promise<Config> => {
|
||||||
|
@ -3,6 +3,7 @@ import { Connection, ConnectionOptions, createConnection, DeepPartial } from 'ty
|
|||||||
import { SnakeNamingStrategy } from 'typeorm-naming-strategies';
|
import { SnakeNamingStrategy } from 'typeorm-naming-strategies';
|
||||||
|
|
||||||
import { Event } from './entity/Event';
|
import { Event } from './entity/Event';
|
||||||
|
import { Contract } from './entity/Contract';
|
||||||
import { EventSyncProgress } from './entity/EventProgress';
|
import { EventSyncProgress } from './entity/EventProgress';
|
||||||
|
|
||||||
export class Database {
|
export class Database {
|
||||||
@ -93,4 +94,27 @@ export class Database {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async getContract (address: string): Promise<Contract | undefined> {
|
||||||
|
return this._conn.getRepository(Contract)
|
||||||
|
.createQueryBuilder('contract')
|
||||||
|
.where('address = :address', { address })
|
||||||
|
.getOne();
|
||||||
|
}
|
||||||
|
|
||||||
|
async saveContract (address: string, kind: string, startingBlock: number): Promise<void> {
|
||||||
|
await this._conn.transaction(async (tx) => {
|
||||||
|
const repo = tx.getRepository(Contract);
|
||||||
|
|
||||||
|
const numRows = await repo
|
||||||
|
.createQueryBuilder()
|
||||||
|
.where('address = :address', { address })
|
||||||
|
.getCount();
|
||||||
|
|
||||||
|
if (numRows === 0) {
|
||||||
|
const entity = repo.create({ address, kind, startingBlock });
|
||||||
|
await repo.save(entity);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,12 +1,12 @@
|
|||||||
import { Entity, PrimaryGeneratedColumn, Column, Index } from 'typeorm';
|
import { Entity, PrimaryColumn, Column } from 'typeorm';
|
||||||
|
|
||||||
|
export const KIND_FACTORY = 'factory';
|
||||||
|
|
||||||
|
export const KIND_POOL = 'pool';
|
||||||
|
|
||||||
@Entity()
|
@Entity()
|
||||||
@Index(['address'], { unique: true })
|
|
||||||
export class Contract {
|
export class Contract {
|
||||||
@PrimaryGeneratedColumn()
|
@PrimaryColumn('varchar', { length: 42 })
|
||||||
id!: number;
|
|
||||||
|
|
||||||
@Column('varchar', { length: 42 })
|
|
||||||
address!: string;
|
address!: string;
|
||||||
|
|
||||||
@Column('varchar', { length: 8 })
|
@Column('varchar', { length: 8 })
|
||||||
|
@ -35,15 +35,14 @@ export class EventWatcher {
|
|||||||
if (logContracts && logContracts.length) {
|
if (logContracts && logContracts.length) {
|
||||||
for (let logIndex = 0; logIndex < logContracts.length; logIndex++) {
|
for (let logIndex = 0; logIndex < logContracts.length; logIndex++) {
|
||||||
const contractAddress = logContracts[logIndex];
|
const contractAddress = logContracts[logIndex];
|
||||||
const isWatchedContract = await this._indexer.isUniswapContract(contractAddress);
|
const uniContract = await this._indexer.isUniswapContract(contractAddress);
|
||||||
if (isWatchedContract) {
|
if (uniContract) {
|
||||||
// TODO: Move processing to background task runner.
|
|
||||||
|
|
||||||
const { ethTransactionCidByTxId: { ethHeaderCidByHeaderId: { blockHash, blockNumber } } } = receipt;
|
const { ethTransactionCidByTxId: { ethHeaderCidByHeaderId: { blockHash, blockNumber } } } = receipt;
|
||||||
await this._indexer.getEvents(blockHash, contractAddress, null);
|
const events = await this._indexer.getEvents(blockHash, contractAddress, null);
|
||||||
|
const event = events[logIndex];
|
||||||
|
|
||||||
// Trigger other indexer methods based on event topic.
|
// Trigger other indexer methods based on event topic.
|
||||||
await this._indexer.processEvent(blockHash, blockNumber, contractAddress, receipt, logIndex);
|
await this._indexer.processEvent(blockHash, blockNumber, uniContract, receipt, event);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -11,16 +11,19 @@ import { GetStorageAt } from '@vulcanize/solidity-mapper';
|
|||||||
|
|
||||||
import { Database } from './database';
|
import { Database } from './database';
|
||||||
import { Event } from './entity/Event';
|
import { Event } from './entity/Event';
|
||||||
|
import { Contract, KIND_FACTORY, KIND_POOL } from './entity/Contract';
|
||||||
import { Config } from './config';
|
import { Config } from './config';
|
||||||
|
|
||||||
import factoryABI from './artifacts/factory.json';
|
import factoryABI from './artifacts/factory.json';
|
||||||
|
|
||||||
const log = debug('vulcanize:indexer');
|
const log = debug('vulcanize:indexer');
|
||||||
|
|
||||||
type EventsResult = Array<{
|
type EventResult = {
|
||||||
event: any;
|
event: any;
|
||||||
proof: string;
|
proof: string;
|
||||||
}>
|
};
|
||||||
|
|
||||||
|
type EventsResult = Array<EventResult>;
|
||||||
|
|
||||||
export class Indexer {
|
export class Indexer {
|
||||||
_config: Config;
|
_config: Config;
|
||||||
@ -51,10 +54,15 @@ export class Indexer {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async getEvents (blockHash: string, contract: string, name: string | null): Promise<EventsResult> {
|
async getEvents (blockHash: string, contract: string, name: string | null): Promise<EventsResult> {
|
||||||
|
const uniContract = await this.isUniswapContract(contract);
|
||||||
|
if (!uniContract) {
|
||||||
|
throw new Error('Not a uniswap contract');
|
||||||
|
}
|
||||||
|
|
||||||
const didSyncEvents = await this._db.didSyncEvents({ blockHash, contract });
|
const didSyncEvents = await this._db.didSyncEvents({ blockHash, contract });
|
||||||
if (!didSyncEvents) {
|
if (!didSyncEvents) {
|
||||||
// Fetch and save events first and make a note in the event sync progress table.
|
// Fetch and save events first and make a note in the event sync progress table.
|
||||||
await this._fetchAndSaveEvents({ blockHash, contract });
|
await this._fetchAndSaveEvents({ blockHash, contract, uniContract });
|
||||||
log('getEvents: db miss, fetching from upstream server');
|
log('getEvents: db miss, fetching from upstream server');
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -84,21 +92,16 @@ export class Indexer {
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* eslint-disable */
|
async triggerIndexingOnEvent (blockNumber: number, event: EventResult): Promise<void> {
|
||||||
async triggerIndexingOnEvent (blockHash: string, contract: string, receipt: any, logIndex: number): Promise<void> {
|
switch (event.event.__typename) {
|
||||||
|
case 'PoolCreatedEvent': {
|
||||||
|
const poolContract = ethers.utils.getAddress(event.event.pool);
|
||||||
|
await this._db.saveContract(poolContract, KIND_POOL, blockNumber);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
/* eslint-enable */
|
|
||||||
|
|
||||||
async publishEventToSubscribers (blockHash: string, blockNumber: number, contract: string, logIndex: number): Promise<void> {
|
|
||||||
// TODO: Optimize this fetching of events.
|
|
||||||
const events = await this.getEvents(blockHash, contract, null);
|
|
||||||
|
|
||||||
log(JSON.stringify(events, null, 2));
|
|
||||||
log(logIndex);
|
|
||||||
|
|
||||||
const event = events[logIndex];
|
|
||||||
|
|
||||||
|
async publishEventToSubscribers (blockHash: string, blockNumber: number, contract: string, event: EventResult): Promise<void> {
|
||||||
log(`pushing event to GQL subscribers: ${event.event.__typename}`);
|
log(`pushing event to GQL subscribers: ${event.event.__typename}`);
|
||||||
|
|
||||||
// Publishing the event here will result in pushing the payload to GQL subscribers for `onEvent`.
|
// Publishing the event here will result in pushing the payload to GQL subscribers for `onEvent`.
|
||||||
@ -112,20 +115,21 @@ export class Indexer {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
async isUniswapContract (address: string): Promise<boolean> {
|
async isUniswapContract (address: string): Promise<Contract | undefined> {
|
||||||
// TODO: Return true for uniswap contracts of interest to the indexer (from config?).
|
return this._db.getContract(ethers.utils.getAddress(address));
|
||||||
return address != null;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async processEvent (blockHash: string, blockNumber: number, contract: string, receipt: any, logIndex: number): Promise<void> {
|
async processEvent (blockHash: string, blockNumber: number, contract: Contract, receipt: any, event: EventResult): Promise<void> {
|
||||||
// Trigger indexing of data based on the event.
|
// Trigger indexing of data based on the event.
|
||||||
await this.triggerIndexingOnEvent(blockHash, contract, receipt, logIndex);
|
await this.triggerIndexingOnEvent(blockNumber, event);
|
||||||
|
|
||||||
// Also trigger downstream event watcher subscriptions.
|
// Also trigger downstream event watcher subscriptions.
|
||||||
await this.publishEventToSubscribers(blockHash, blockNumber, contract, logIndex);
|
await this.publishEventToSubscribers(blockHash, blockNumber, contract.address, event);
|
||||||
}
|
}
|
||||||
|
|
||||||
async _fetchAndSaveEvents ({ blockHash, contract }: { blockHash: string, contract: string }): Promise<void> {
|
async _fetchAndSaveEvents ({ blockHash, contract, uniContract }: { blockHash: string, contract: string, uniContract: Contract }): Promise<void> {
|
||||||
|
assert(uniContract);
|
||||||
|
|
||||||
const logs = await this._ethClient.getLogs({ blockHash, contract });
|
const logs = await this._ethClient.getLogs({ blockHash, contract });
|
||||||
|
|
||||||
const dbEvents = logs.map((logObj: any) => {
|
const dbEvents = logs.map((logObj: any) => {
|
||||||
@ -134,8 +138,7 @@ export class Indexer {
|
|||||||
let eventName;
|
let eventName;
|
||||||
let eventProps = {};
|
let eventProps = {};
|
||||||
|
|
||||||
// TODO: Get contract kind from contracts table.
|
if (uniContract.kind === KIND_FACTORY) {
|
||||||
if (contract === this._config.contracts.factory) {
|
|
||||||
const logDescription = this._factoryContract.parseLog({ data, topics });
|
const logDescription = this._factoryContract.parseLog({ data, topics });
|
||||||
const { token0, token1, fee, tickSpacing, pool } = logDescription.args;
|
const { token0, token1, fee, tickSpacing, pool } = logDescription.args;
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user