mirror of
https://github.com/cerc-io/watcher-ts
synced 2025-01-08 12:28:05 +00:00
Watch for Factory PoolCreated event. (#119)
This commit is contained in:
parent
da758aceaa
commit
8d2a4c6b14
@ -92,7 +92,7 @@ export class Database {
|
||||
async getEventsByName ({ blockHash, token, eventName }: { blockHash: string, token: string, eventName: string }): Promise<Event[] | undefined> {
|
||||
return this._conn.getRepository(Event)
|
||||
.createQueryBuilder('event')
|
||||
.where('block_hash = :blockHash AND token = :token AND :eventName = :eventName', {
|
||||
.where('block_hash = :blockHash AND token = :token AND event_name = :eventName', {
|
||||
blockHash,
|
||||
token,
|
||||
eventName
|
||||
|
@ -22,10 +22,13 @@
|
||||
subscribersDir = "src/subscriber"
|
||||
|
||||
[upstream]
|
||||
gqlEndpoint = "http://127.0.0.1:8083/graphql"
|
||||
gqlEndpoint = "http://127.0.0.1:8082/graphql"
|
||||
gqlSubscriptionEndpoint = "http://127.0.0.1:5000/graphql"
|
||||
|
||||
[upstream.cache]
|
||||
name = "requests"
|
||||
enabled = false
|
||||
deleteOnStart = false
|
||||
|
||||
[contracts]
|
||||
factory = "0xF3eeD405056b4BB1ce100761325f82F2b6Da2D82"
|
||||
|
@ -19,6 +19,9 @@ export interface Config {
|
||||
gqlSubscriptionEndpoint: string;
|
||||
cache: CacheConfig
|
||||
}
|
||||
contracts: {
|
||||
factory: string;
|
||||
}
|
||||
}
|
||||
|
||||
export const getConfig = async (configFile: string): Promise<Config> => {
|
||||
|
@ -28,41 +28,41 @@ export class Database {
|
||||
}
|
||||
|
||||
// Returns true if events have already been synced for the (block, token) combination.
|
||||
async didSyncEvents ({ blockHash, token }: { blockHash: string, token: string }): Promise<boolean> {
|
||||
async didSyncEvents ({ blockHash, contract }: { blockHash: string, contract: string }): Promise<boolean> {
|
||||
const numRows = await this._conn.getRepository(EventSyncProgress)
|
||||
.createQueryBuilder()
|
||||
.where('block_hash = :blockHash AND token = :token', {
|
||||
.where('block_hash = :blockHash AND contract = :contract', {
|
||||
blockHash,
|
||||
token
|
||||
contract
|
||||
})
|
||||
.getCount();
|
||||
|
||||
return numRows > 0;
|
||||
}
|
||||
|
||||
async getEvents ({ blockHash, token }: { blockHash: string, token: string }): Promise<Event[]> {
|
||||
async getEvents ({ blockHash, contract }: { blockHash: string, contract: string }): Promise<Event[]> {
|
||||
return this._conn.getRepository(Event)
|
||||
.createQueryBuilder('event')
|
||||
.where('block_hash = :blockHash AND token = :token', {
|
||||
.where('block_hash = :blockHash AND contract = :contract', {
|
||||
blockHash,
|
||||
token
|
||||
contract
|
||||
})
|
||||
.addOrderBy('id', 'ASC')
|
||||
.getMany();
|
||||
}
|
||||
|
||||
async getEventsByName ({ blockHash, token, eventName }: { blockHash: string, token: string, eventName: string }): Promise<Event[] | undefined> {
|
||||
async getEventsByName ({ blockHash, contract, eventName }: { blockHash: string, contract: string, eventName: string }): Promise<Event[] | undefined> {
|
||||
return this._conn.getRepository(Event)
|
||||
.createQueryBuilder('event')
|
||||
.where('block_hash = :blockHash AND token = :token AND :eventName = :eventName', {
|
||||
.where('block_hash = :blockHash AND contract = :contract AND event_name = :eventName', {
|
||||
blockHash,
|
||||
token,
|
||||
contract,
|
||||
eventName
|
||||
})
|
||||
.getMany();
|
||||
}
|
||||
|
||||
async saveEvents ({ blockHash, token, events }: { blockHash: string, token: string, events: DeepPartial<Event>[] }): Promise<void> {
|
||||
async saveEvents ({ blockHash, contract, events }: { blockHash: string, contract: string, events: DeepPartial<Event>[] }): Promise<void> {
|
||||
// In a transaction:
|
||||
// (1) Save all the events in the database.
|
||||
// (2) Add an entry to the event progress table.
|
||||
@ -73,9 +73,9 @@ export class Database {
|
||||
// Check sync progress inside the transaction.
|
||||
const numRows = await repo
|
||||
.createQueryBuilder()
|
||||
.where('block_hash = :blockHash AND token = :token', {
|
||||
.where('block_hash = :blockHash AND contract = :contract', {
|
||||
blockHash,
|
||||
token
|
||||
contract
|
||||
})
|
||||
.getCount();
|
||||
|
||||
@ -88,7 +88,7 @@ export class Database {
|
||||
.execute();
|
||||
|
||||
// Update event sync progress.
|
||||
const progress = repo.create({ blockHash, token });
|
||||
const progress = repo.create({ blockHash, contract });
|
||||
await repo.save(progress);
|
||||
}
|
||||
});
|
||||
|
17
packages/uni-watcher/src/entity/Contract.ts
Normal file
17
packages/uni-watcher/src/entity/Contract.ts
Normal file
@ -0,0 +1,17 @@
|
||||
import { Entity, PrimaryGeneratedColumn, Column, Index } from 'typeorm';
|
||||
|
||||
@Entity()
|
||||
@Index(['address'], { unique: true })
|
||||
export class Contract {
|
||||
@PrimaryGeneratedColumn()
|
||||
id!: number;
|
||||
|
||||
@Column('varchar', { length: 42 })
|
||||
address!: string;
|
||||
|
||||
@Column('varchar', { length: 8 })
|
||||
kind!: string;
|
||||
|
||||
@Column('numeric')
|
||||
startingBlock!: number;
|
||||
}
|
@ -2,7 +2,7 @@ import { Entity, PrimaryGeneratedColumn, Column, Index } from 'typeorm';
|
||||
|
||||
@Entity()
|
||||
// Index to query all events for a contract efficiently.
|
||||
@Index(['blockHash', 'token'])
|
||||
@Index(['blockHash', 'contract'])
|
||||
export class Event {
|
||||
@PrimaryGeneratedColumn()
|
||||
id!: number;
|
||||
@ -11,11 +11,15 @@ export class Event {
|
||||
blockHash!: string;
|
||||
|
||||
@Column('varchar', { length: 42 })
|
||||
token!: string;
|
||||
contract!: string;
|
||||
|
||||
@Column('varchar', { length: 256 })
|
||||
eventName!: string;
|
||||
|
||||
// TODO: Polymorphic relationships?
|
||||
@Column('text')
|
||||
eventData!: string;
|
||||
|
||||
@Column('text')
|
||||
proof!: string;
|
||||
}
|
||||
|
@ -7,7 +7,7 @@ import { Entity, PrimaryGeneratedColumn, Column, Index } from 'typeorm';
|
||||
// yet been synced from upstream.
|
||||
//
|
||||
@Entity()
|
||||
@Index(['blockHash', 'token'], { unique: true })
|
||||
@Index(['blockHash', 'contract'], { unique: true })
|
||||
export class EventSyncProgress {
|
||||
@PrimaryGeneratedColumn()
|
||||
id!: number;
|
||||
@ -16,5 +16,5 @@ export class EventSyncProgress {
|
||||
blockHash!: string;
|
||||
|
||||
@Column('varchar', { length: 42 })
|
||||
token!: string;
|
||||
contract!: string;
|
||||
}
|
||||
|
@ -1,107 +1,73 @@
|
||||
import assert from 'assert';
|
||||
import debug from 'debug';
|
||||
import { invert } from 'lodash';
|
||||
import { JsonFragment } from '@ethersproject/abi';
|
||||
import _ from 'lodash';
|
||||
import { DeepPartial } from 'typeorm';
|
||||
import JSONbig from 'json-bigint';
|
||||
import { ethers } from 'ethers';
|
||||
import { PubSub } from 'apollo-server-express';
|
||||
|
||||
import { EthClient, topictoAddress } from '@vulcanize/ipld-eth-client';
|
||||
import { getEventNameTopics, getStorageValue, GetStorageAt, StorageLayout } from '@vulcanize/solidity-mapper';
|
||||
import { EthClient } from '@vulcanize/ipld-eth-client';
|
||||
import { GetStorageAt } from '@vulcanize/solidity-mapper';
|
||||
|
||||
import { Database } from './database';
|
||||
import { Event } from './entity/Event';
|
||||
import { Config } from './config';
|
||||
|
||||
import factoryABI from './artifacts/factory.json';
|
||||
|
||||
const log = debug('vulcanize:indexer');
|
||||
|
||||
interface Artifacts {
|
||||
abi: JsonFragment[];
|
||||
storageLayout: StorageLayout;
|
||||
}
|
||||
|
||||
export interface ValueResult {
|
||||
value: string | bigint;
|
||||
proof: {
|
||||
data: string;
|
||||
}
|
||||
}
|
||||
|
||||
type EventsResult = Array<{
|
||||
event: {
|
||||
from?: string;
|
||||
to?: string;
|
||||
owner?: string;
|
||||
spender?: string;
|
||||
value?: BigInt;
|
||||
__typename: string;
|
||||
}
|
||||
event: any;
|
||||
proof: string;
|
||||
}>
|
||||
|
||||
export class Indexer {
|
||||
_config: Config;
|
||||
_db: Database
|
||||
_ethClient: EthClient
|
||||
_pubsub: PubSub
|
||||
_getStorageAt: GetStorageAt
|
||||
|
||||
// _abi: JsonFragment[]
|
||||
// _storageLayout: StorageLayout
|
||||
// _contract: ethers.utils.Interface
|
||||
_factoryContract: ethers.utils.Interface
|
||||
|
||||
constructor (db: Database, ethClient: EthClient, pubsub: PubSub) {
|
||||
constructor (config: Config, db: Database, ethClient: EthClient, pubsub: PubSub) {
|
||||
assert(config);
|
||||
assert(db);
|
||||
assert(ethClient);
|
||||
assert(pubsub);
|
||||
|
||||
// const { abi, storageLayout } = artifacts;
|
||||
|
||||
// assert(abi);
|
||||
// assert(storageLayout);
|
||||
|
||||
this._config = config;
|
||||
this._db = db;
|
||||
this._ethClient = ethClient;
|
||||
this._pubsub = pubsub;
|
||||
this._getStorageAt = this._ethClient.getStorageAt.bind(this._ethClient);
|
||||
|
||||
// this._abi = abi;
|
||||
// this._storageLayout = storageLayout;
|
||||
|
||||
// this._contract = new ethers.utils.Interface(this._abi);
|
||||
this._factoryContract = new ethers.utils.Interface(factoryABI);
|
||||
}
|
||||
|
||||
getEventIterator (): AsyncIterator<any> {
|
||||
return this._pubsub.asyncIterator(['event']);
|
||||
}
|
||||
|
||||
async getEvents (blockHash: string, token: string, name: string | null): Promise<EventsResult> {
|
||||
const didSyncEvents = await this._db.didSyncEvents({ blockHash, token });
|
||||
async getEvents (blockHash: string, contract: string, name: string | null): Promise<EventsResult> {
|
||||
const didSyncEvents = await this._db.didSyncEvents({ blockHash, contract });
|
||||
if (!didSyncEvents) {
|
||||
// Fetch and save events first and make a note in the event sync progress table.
|
||||
await this._fetchAndSaveEvents({ blockHash, token });
|
||||
await this._fetchAndSaveEvents({ blockHash, contract });
|
||||
log('getEvents: db miss, fetching from upstream server');
|
||||
}
|
||||
|
||||
assert(await this._db.didSyncEvents({ blockHash, token }));
|
||||
assert(await this._db.didSyncEvents({ blockHash, contract }));
|
||||
|
||||
const events = await this._db.getEvents({ blockHash, token });
|
||||
log('getEvents: db hit');
|
||||
const events = await this._db.getEvents({ blockHash, contract });
|
||||
log(`getEvents: db hit, num events: ${events.length}`);
|
||||
|
||||
const result = events
|
||||
// TODO: Filter using db WHERE condition when name is not empty.
|
||||
.filter(event => !name || name === event.eventName)
|
||||
.map(e => {
|
||||
const eventFields: {
|
||||
from?: string,
|
||||
to?: string,
|
||||
value?: BigInt,
|
||||
owner?: string,
|
||||
spender?: string,
|
||||
} = {};
|
||||
|
||||
switch (e.eventName) {
|
||||
// TODO: Handle events.
|
||||
}
|
||||
const eventFields = JSON.parse(e.eventData);
|
||||
|
||||
return {
|
||||
event: {
|
||||
@ -118,112 +84,87 @@ export class Indexer {
|
||||
return result;
|
||||
}
|
||||
|
||||
async triggerIndexingOnEvent (blockHash: string, token: string, receipt: any, logIndex: number): Promise<void> {
|
||||
const topics = [];
|
||||
/* eslint-disable */
|
||||
async triggerIndexingOnEvent (blockHash: string, contract: string, receipt: any, logIndex: number): Promise<void> {
|
||||
|
||||
// We only care about the event type for now.
|
||||
const data = '0x0000000000000000000000000000000000000000000000000000000000000000';
|
||||
|
||||
topics.push(receipt.topic0S[logIndex]);
|
||||
topics.push(receipt.topic1S[logIndex]);
|
||||
topics.push(receipt.topic2S[logIndex]);
|
||||
|
||||
// const { name: eventName, args } = this._contract.parseLog({ topics, data });
|
||||
// log(`trigger indexing on event: ${eventName} ${args}`);
|
||||
|
||||
// What data we index depends on the kind of event.
|
||||
// switch (eventName) {
|
||||
// TODO: Index event.
|
||||
// }
|
||||
}
|
||||
/* eslint-enable */
|
||||
|
||||
async publishEventToSubscribers (blockHash: string, token: string, logIndex: number): Promise<void> {
|
||||
async publishEventToSubscribers (blockHash: string, contract: string, logIndex: number): Promise<void> {
|
||||
// TODO: Optimize this fetching of events.
|
||||
const events = await this.getEvents(blockHash, token, null);
|
||||
const events = await this.getEvents(blockHash, contract, null);
|
||||
|
||||
log(JSON.stringify(events, null, 2));
|
||||
log(logIndex);
|
||||
|
||||
const event = events[logIndex];
|
||||
|
||||
log(`pushing event to GQL subscribers: ${event.event.__typename}`);
|
||||
|
||||
// Publishing the event here will result in pushing the payload to GQL subscribers for `onTokenEvent`.
|
||||
// Publishing the event here will result in pushing the payload to GQL subscribers for `onEvent`.
|
||||
await this._pubsub.publish('event', {
|
||||
onTokenEvent: {
|
||||
onEvent: {
|
||||
blockHash,
|
||||
token,
|
||||
contract,
|
||||
event
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
async isUniswapContract (address: String): Promise<boolean> {
|
||||
async isUniswapContract (address: string): Promise<boolean> {
|
||||
// TODO: Return true for uniswap contracts of interest to the indexer (from config?).
|
||||
return address != null;
|
||||
}
|
||||
|
||||
async processEvent (blockHash: string, token: string, receipt: any, logIndex: number): Promise<void> {
|
||||
async processEvent (blockHash: string, contract: string, receipt: any, logIndex: number): Promise<void> {
|
||||
// Trigger indexing of data based on the event.
|
||||
await this.triggerIndexingOnEvent(blockHash, token, receipt, logIndex);
|
||||
await this.triggerIndexingOnEvent(blockHash, contract, receipt, logIndex);
|
||||
|
||||
// Also trigger downstream event watcher subscriptions.
|
||||
await this.publishEventToSubscribers(blockHash, token, logIndex);
|
||||
await this.publishEventToSubscribers(blockHash, contract, logIndex);
|
||||
}
|
||||
|
||||
// TODO: Move into base/class or framework package.
|
||||
async _getStorageValue (blockHash: string, token: string, variable: string, ...mappingKeys: string[]): Promise<ValueResult> {
|
||||
return {
|
||||
value: '',
|
||||
proof: {
|
||||
data: ''
|
||||
async _fetchAndSaveEvents ({ blockHash, contract }: { blockHash: string, contract: string }): Promise<void> {
|
||||
const logs = await this._ethClient.getLogs({ blockHash, contract });
|
||||
|
||||
const dbEvents = logs.map((logObj: any) => {
|
||||
const { topics, data, cid, ipldBlock } = logObj;
|
||||
|
||||
let eventName;
|
||||
let eventProps = {};
|
||||
|
||||
// TODO: Get contract kind from contracts table.
|
||||
if (contract === this._config.contracts.factory) {
|
||||
const logDescription = this._factoryContract.parseLog({ data, topics });
|
||||
const { token0, token1, fee, tickSpacing, pool } = logDescription.args;
|
||||
|
||||
eventName = logDescription.name;
|
||||
eventProps = { token0, token1, fee, tickSpacing, pool };
|
||||
}
|
||||
};
|
||||
|
||||
// return getStorageValue(
|
||||
// this._storageLayout,
|
||||
// this._getStorageAt,
|
||||
// blockHash,
|
||||
// token,
|
||||
// variable,
|
||||
// ...mappingKeys
|
||||
// );
|
||||
}
|
||||
|
||||
async _fetchAndSaveEvents ({ blockHash, token }: { blockHash: string, token: string }): Promise<void> {
|
||||
const logs = await this._ethClient.getLogs({ blockHash, contract: token });
|
||||
|
||||
const eventNameToTopic = {}; // getEventNameTopics(this._abi);
|
||||
const logTopicToEventName = invert(eventNameToTopic);
|
||||
|
||||
const dbEvents = logs.map((log: any) => {
|
||||
const { topics, data: value, cid, ipldBlock } = log;
|
||||
|
||||
const [topic0, topic1, topic2] = topics;
|
||||
|
||||
const eventName = logTopicToEventName[topic0];
|
||||
const address1 = topictoAddress(topic1);
|
||||
const address2 = topictoAddress(topic2);
|
||||
|
||||
const event: DeepPartial<Event> = {
|
||||
blockHash,
|
||||
token,
|
||||
eventName,
|
||||
|
||||
proof: JSONbig.stringify({
|
||||
data: JSONbig.stringify({
|
||||
blockHash,
|
||||
receipt: {
|
||||
cid,
|
||||
ipldBlock
|
||||
}
|
||||
let event: DeepPartial<Event> | undefined;
|
||||
if (eventName) {
|
||||
event = {
|
||||
blockHash,
|
||||
contract,
|
||||
eventName,
|
||||
eventData: JSONbig.stringify({ ...eventProps }),
|
||||
proof: JSONbig.stringify({
|
||||
data: JSONbig.stringify({
|
||||
blockHash,
|
||||
receipt: {
|
||||
cid,
|
||||
ipldBlock
|
||||
}
|
||||
})
|
||||
})
|
||||
})
|
||||
};
|
||||
|
||||
switch (eventName) {
|
||||
// TODO: Handle event.
|
||||
};
|
||||
}
|
||||
|
||||
return event;
|
||||
});
|
||||
|
||||
await this._db.saveEvents({ blockHash, token, events: dbEvents });
|
||||
const events: DeepPartial<Event>[] = _.compact(dbEvents);
|
||||
await this._db.saveEvents({ blockHash, contract, events });
|
||||
}
|
||||
}
|
||||
|
@ -2,7 +2,7 @@ import assert from 'assert';
|
||||
import BigInt from 'apollo-type-bigint';
|
||||
import debug from 'debug';
|
||||
|
||||
import { Indexer, ValueResult } from './indexer';
|
||||
import { Indexer } from './indexer';
|
||||
|
||||
const log = debug('vulcanize:resolver');
|
||||
|
||||
@ -13,32 +13,34 @@ export const createResolvers = async (indexer: Indexer): Promise<any> => {
|
||||
BigInt: new BigInt('bigInt'),
|
||||
|
||||
ERC20Event: {
|
||||
__resolveType() {
|
||||
__resolveType () {
|
||||
return null;
|
||||
}
|
||||
},
|
||||
|
||||
FactoryEvent: {
|
||||
__resolveType() {
|
||||
__resolveType () {
|
||||
return null;
|
||||
}
|
||||
},
|
||||
|
||||
NonFungiblePositionManagerEvent: {
|
||||
__resolveType() {
|
||||
__resolveType () {
|
||||
return null;
|
||||
}
|
||||
},
|
||||
|
||||
PoolEvent: {
|
||||
__resolveType() {
|
||||
__resolveType () {
|
||||
return null;
|
||||
}
|
||||
},
|
||||
|
||||
Event: {
|
||||
__resolveType() {
|
||||
return null;
|
||||
__resolveType: (obj: any) => {
|
||||
assert(obj.__typename);
|
||||
|
||||
return obj.__typename;
|
||||
}
|
||||
},
|
||||
|
||||
@ -50,9 +52,9 @@ export const createResolvers = async (indexer: Indexer): Promise<any> => {
|
||||
|
||||
Query: {
|
||||
|
||||
events: async (_: any, { blockHash, token, name }: { blockHash: string, token: string, name: string }) => {
|
||||
log('events', blockHash, token, name || '');
|
||||
return indexer.getEvents(blockHash, token, name);
|
||||
events: async (_: any, { blockHash, contract, name }: { blockHash: string, contract: string, name: string }) => {
|
||||
log('events', blockHash, contract, name || '');
|
||||
return indexer.getEvents(blockHash, contract, name);
|
||||
}
|
||||
}
|
||||
};
|
||||
|
@ -11,7 +11,6 @@ import { createServer } from 'http';
|
||||
import { getCache } from '@vulcanize/cache';
|
||||
import { EthClient } from '@vulcanize/ipld-eth-client';
|
||||
|
||||
import artifacts from './artifacts/ERC20.json';
|
||||
import typeDefs from './schema';
|
||||
|
||||
import { createResolvers as createMockResolvers } from './mock/resolvers';
|
||||
@ -58,7 +57,7 @@ export const main = async (): Promise<any> => {
|
||||
// Note: In-memory pubsub works fine for now, as each watcher is a single process anyway.
|
||||
// Later: https://www.apollographql.com/docs/apollo-server/data/subscriptions/#production-pubsub-libraries
|
||||
const pubsub = new PubSub();
|
||||
const indexer = new Indexer(db, ethClient, pubsub);
|
||||
const indexer = new Indexer(config, db, ethClient, pubsub);
|
||||
|
||||
const eventWatcher = new EventWatcher(ethClient, indexer);
|
||||
await eventWatcher.start();
|
||||
|
Loading…
Reference in New Issue
Block a user