wormhole-explorer/blockchain-watcher/src/domain/actions/evm/HandleEvmLogs.ts

81 lines
2.2 KiB
TypeScript
Raw Normal View History

[Blockchain Watcher] Emit LogMessagePublished for Solana (#809) * adding gen-relayer watcher code to branch * folder rename * adding some new abstract types for processor redesign * added first cut of new process manager logic * large refactoring to environment configuration * Add basic default config * prettier run * modifying event handler interface * abstract handler typing changes * Blockchain watcher: adding domain and infra layers (#786) * reorg domain-infra * watch evm blocks action * adding evm log parser * wider prettier * renaming watch action * adding doc * persist latest metadata * gh action for blockchain-watcher * adding log-message-published mapper * deps: remove peers and nodemon * adding handler for LogMessagePublished * added parser for log message published --------- Co-authored-by: chase-45 <chasemoran45@gmail.com> * Blockchain Watcher: ethereum -> sns implementation (#790) * evm block repo implementation * adding sns publisher * adding external config and initial runner * fix: start from latest if no fromBlock configured * feat: add dryRun option * fix: handler filtering and eth_getLogs filter payload * local e2e * actual sns arns for testnet * smaller docker image * deployment changes * [Blockchain Watcher] Apply missing formatting for log message published (#791) * Apply missing formatting for log message published * keep extracting * emitterAddress -> emitter * using configured default winston logger * wait for new block * adding basic metrics * staging mainnet config * send hexa block number * simpler log format * group entities in different files * [Blockchain watcher] Adding light observability (#793) * using configured default winston logger * wait for new block * adding basic metrics * staging mainnet config * send hexa block number * simpler log format * better error logs * wait when no block available * grouping watchers * adding 429 handling http client * abstracting polling iteration * Load jobs dinamically * changing deployment strategy to jobs per pod * group actions per platform * [Blockchain Watcher] Add StartJob action + rate limit handling (#799) * using configured default winston logger * wait for new block * adding basic metrics * staging mainnet config * send hexa block number * simpler log format * group entities in different files * grouping watchers * adding 429 handling http client * abstracting polling iteration * Load jobs dinamically * changing deployment strategy to jobs per pod * adding poll solana txs action * adding solana repo implementation * solana: mapping to logmsgpublished * fixing solana mapper test * updating readme * adding some metrics * adding solana worker * adding finality param to get solana block * minor changes - using confirmed for solana log extraction * updating version * add solana env var * [Blockchain Watcher] Feature-813/fix-coverage-report (#815) * Fix coverage report * Run prettier * Rollback global coverage value * Set global coverage value to 55 * Add github PR template * Run prettier * Rename github template file --------- Co-authored-by: Julian Merlo <julianmerlo@MacBook-Pro-de-Julian.local> * private solana urls as secret * adding prod cfg files * Change repository name and cretae error foulder (#819) * Change repository name and cretae error foulder * Rename SolanaFailure file * Rename SolanaFailure file --------- Co-authored-by: Julian Merlo <julianmerlo@MacBook-Pro-de-Julian.local> * passing node_options for solana * adding missing envs templates * cru * ratelimited solana client + increase coverage * [Blockchain Watcher] Feature 817/add hexagonal prefix (#824) * Put hexadecimal prefix * Create id variable * Add error log * Add error log * Remove bigint in getBlock params --------- Co-authored-by: Julian Merlo <julianmerlo@MacBook-Pro-de-Julian.local> * fix never ending solana jobs * adding job counter metric * solana: add retry handling for 429s * cleanup * remove duplicated tests * cru + item count * minor readme update + do not use private solana rpc * fix items counter * fix solana job programID for devnet * log basic repo info * solana: passing finality everywhere * delete unnecesary files * adjusting solana cfgs * [Blockchain Watcher] Fix/re order some files (#828) * fix/re-order-some-files * run prettier * Run prettier * Resolve comment in PR * Remove import * Resolve comment * Remove prefix in sns log --------- Co-authored-by: Julian Merlo <julianmerlo@MacBook-Pro-de-Julian.local> --------- Co-authored-by: chase-45 <chasemoran45@gmail.com> Co-authored-by: Julian <52217955+julianmerlo95@users.noreply.github.com> Co-authored-by: Julian Merlo <julianmerlo@MacBook-Pro-de-Julian.local>
2023-11-30 07:05:43 -08:00
import { EvmLog, EvmTopicFilter } from "../../entities";
import { StatRepository } from "../../repositories";
import { ethers } from "ethers";
/**
* Handling means mapping and forward to a given target.
* As of today, only one type of event can be handled per each instance.
*/
export class HandleEvmLogs<T> {
cfg: HandleEvmConfig;
mapper: (log: EvmLog, parsedArgs: ReadonlyArray<any>) => T;
2024-02-14 06:14:25 -08:00
target: (parsed: T[], chain: string) => Promise<void>;
statsRepo: StatRepository;
constructor(
cfg: HandleEvmConfig,
mapper: (log: EvmLog, args: ReadonlyArray<any>) => T,
2024-02-14 06:14:25 -08:00
target: (parsed: T[], chain: string) => Promise<void>,
statsRepo: StatRepository
) {
this.cfg = this.normalizeCfg(cfg);
this.mapper = mapper;
this.target = target;
this.statsRepo = statsRepo;
}
public async handle(logs: EvmLog[]): Promise<T[]> {
const mappedItems = logs
.filter(
(log) =>
this.cfg.filter.addresses.includes(log.address.toLowerCase()) &&
this.cfg.filter.topics.includes(log.topics[0].toLowerCase())
)
.map((log) => {
const iface = new ethers.utils.Interface([this.cfg.abi]);
const parsedLog = iface.parseLog(log);
const logMap = this.mapper(log, parsedLog.args);
this.report();
return logMap;
});
2024-02-14 06:14:25 -08:00
await this.target(mappedItems, this.cfg.chain);
// TODO: return a result specifying failures if any
return mappedItems;
}
private report() {
const labels = {
job: this.cfg.id,
chain: this.cfg.chain ?? "",
commitment: this.cfg.commitment,
};
this.statsRepo.count(this.cfg.metricName, labels);
}
private normalizeCfg(cfg: HandleEvmConfig): HandleEvmConfig {
return {
filter: {
addresses: cfg.filter.addresses.map((addr) => addr.toLowerCase()),
topics: cfg.filter.topics.map((topic) => topic.toLowerCase()),
},
metricName: cfg.metricName,
commitment: cfg.commitment,
chainId: cfg.chainId,
chain: cfg.chain,
abi: cfg.abi,
id: cfg.id,
};
}
}
export type HandleEvmConfig = {
filter: EvmTopicFilter;
metricName: string;
commitment: string;
chainId: number;
chain: string;
abi: string;
id: string;
};