Skip to content

Commit

Permalink
ft_watcher: add logic to persist token info
Browse files Browse the repository at this point in the history
Signed-off-by: bingyuyap <bingyu.yap.21@gmail.com>
  • Loading branch information
bingyuyap committed Sep 2, 2024
1 parent 3b59063 commit b40ccac
Show file tree
Hide file tree
Showing 5 changed files with 200 additions and 20 deletions.
11 changes: 11 additions & 0 deletions database/fast-transfer-schema.sql
Original file line number Diff line number Diff line change
Expand Up @@ -107,3 +107,14 @@ CREATE TABLE redeem_swaps (
relaying_fee BIGINT NOT NULL,
redeem_time TIMESTAMP NOT NULL
);

-- Token Infos table to store information about different tokens
-- A normalized table for us to reference token info for analytics purposes
CREATE TABLE token_infos (
name VARCHAR(255) NOT NULL,
chain_id INTEGER NOT NULL,
decimals INTEGER NOT NULL,
symbol VARCHAR(50) NOT NULL,
token_address VARCHAR(255) NOT NULL,
PRIMARY KEY (token_address, chain_id)
);
22 changes: 20 additions & 2 deletions watcher/src/fastTransfer/swapLayer/parser.ts
Original file line number Diff line number Diff line change
@@ -1,18 +1,31 @@
import { ethers } from 'ethers';
import { TransferCompletion } from '../types';
import { parseVaa } from '@wormhole-foundation/wormhole-monitor-common';
import { Knex } from 'knex';
import { Chain, chainToChainId } from '@wormhole-foundation/sdk-base';
import { TokenInfoManager } from '../../utils/TokenInfoManager';

class SwapLayerParser {
private provider: ethers.providers.JsonRpcProvider;
private swapLayerAddress: string;
private swapLayerInterface: ethers.utils.Interface;

constructor(provider: ethers.providers.JsonRpcProvider, swapLayerAddress: string) {
private tokenInfoManager: TokenInfoManager | null = null;

constructor(
provider: ethers.providers.JsonRpcProvider,
swapLayerAddress: string,
db: Knex | null,
chain: Chain
) {
this.provider = provider;
this.swapLayerAddress = swapLayerAddress;
this.swapLayerInterface = new ethers.utils.Interface([
'event Redeemed(address indexed recipient, address outputToken, uint256 outputAmount, uint256 relayingFee)',
]);

if (db) {
this.tokenInfoManager = new TokenInfoManager(db, chainToChainId(chain), provider);
}
}

async parseSwapLayerTransaction(
Expand Down Expand Up @@ -59,6 +72,11 @@ class SwapLayerParser {

if (!swapEvent) return null;

// if we have the tokenInfoManager inited, persist the token info
// this ensures we have the token decimal and name for analytics purposes
if (this.tokenInfoManager)
await this.tokenInfoManager.saveTokenInfoIfNotExist(swapEvent.args.outputToken);

return {
tx_hash: txHash,
recipient: swapEvent.args.recipient,
Expand Down
8 changes: 8 additions & 0 deletions watcher/src/fastTransfer/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -156,3 +156,11 @@ export type TransferCompletion = {
// on Solana Swap Layer, this acts as a link between complete_{transfer, swap}_payload and release_inbound
staged_inbound?: string;
};

export type TokenInfo = {
name: string;
chain_id: number;
decimals: number;
symbol: string;
token_address: string;
};
145 changes: 145 additions & 0 deletions watcher/src/utils/TokenInfoManager.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,145 @@
import { Knex } from 'knex';
import { ethers, providers } from 'ethers';
import { ChainId } from '@wormhole-foundation/sdk-base';
import { Connection, PublicKey } from '@solana/web3.js';
import { TokenInfo } from 'src/fastTransfer/types';

const minABI = [
{
inputs: [],
name: 'name',
outputs: [{ internalType: 'string', name: '', type: 'string' }],
stateMutability: 'view',
type: 'function',
},
{
inputs: [],
name: 'symbol',
outputs: [{ internalType: 'string', name: '', type: 'string' }],
stateMutability: 'view',
type: 'function',
},
{
inputs: [],
name: 'decimals',
outputs: [{ internalType: 'uint8', name: '', type: 'uint8' }],
stateMutability: 'view',
type: 'function',
},
];
// TokenInfoManager class for managing token information across different chains
// This class is to ensure that token info (e.g. decimal, name) for tokens that we see on Swap Layer is persisted for analytics purposes
export class TokenInfoManager {
private tokenInfoMap: Map<string, TokenInfo>;
private db: Knex;
private chainId: ChainId;
private provider: providers.JsonRpcProvider | Connection;

constructor(db: Knex, chainId: ChainId, provider: providers.JsonRpcProvider | Connection) {
this.tokenInfoMap = new Map();
this.db = db;
this.chainId = chainId;
this.provider = provider;
}

// Retrieve token information from the database
private async getTokenInfoFromDB(tokenAddress: string): Promise<TokenInfo | null> {
return await this.db('token_infos')
.select('token_address', 'name', 'symbol', 'decimals')
.where('token_address', tokenAddress)
.andWhere('chain_id', this.chainId)
.first();
}

private async saveTokenInfo(tokenAddress: string, tokenInfo: TokenInfo): Promise<void> {
await this.db('token_infos').insert({
token_address: tokenAddress,
name: tokenInfo.name,
symbol: tokenInfo.symbol,
decimals: tokenInfo.decimals,
chain_id: this.chainId,
});
}

// Save token information if it doesn't exist in the cache or database
public async saveTokenInfoIfNotExist(tokenAddress: string): Promise<TokenInfo | null> {
if (this.tokenInfoMap.has(tokenAddress)) {
return this.tokenInfoMap.get(tokenAddress) || null;
}
// Check if token info is in the database
const tokenInfo = await this.getTokenInfoFromDB(tokenAddress);
if (tokenInfo) {
this.tokenInfoMap.set(tokenAddress, tokenInfo);
return tokenInfo;
}
// If not in database, fetch from RPC
const fetchedTokenInfo = await this.fetchTokenInfoFromRPC(tokenAddress);
if (fetchedTokenInfo) {
await this.saveTokenInfo(tokenAddress, fetchedTokenInfo);
this.tokenInfoMap.set(tokenAddress, fetchedTokenInfo);
return fetchedTokenInfo;
}
return null;
}

// Fetch token information from RPC based on the chain ID
private async fetchTokenInfoFromRPC(tokenAddress: string): Promise<TokenInfo | null> {
if (this.chainId === 1) {
return this.fetchSolanaTokenInfo(tokenAddress);
}
return this.fetchEVMTokenInfo(tokenAddress);
}

// Fetch Solana token information
private async fetchSolanaTokenInfo(tokenAddress: string): Promise<TokenInfo | null> {
try {
const connection = this.provider as Connection;
const tokenPublicKey = new PublicKey(tokenAddress);
const accountInfo = await connection.getParsedAccountInfo(tokenPublicKey);

if (accountInfo.value && accountInfo.value.data && 'parsed' in accountInfo.value.data) {
const parsedData = accountInfo.value.data.parsed;
if (parsedData.type === 'mint' && 'info' in parsedData) {
const { name, symbol, decimals } = parsedData.info;
if (
typeof name === 'string' &&
typeof symbol === 'string' &&
typeof decimals === 'number'
) {
return { name, symbol, decimals, chain_id: 1, token_address: tokenAddress };
}
}
}
throw new Error('Invalid token account');
} catch (error) {
console.error('Error fetching Solana token info:', error);
return null;
}
}

// Fetch EVM token information
private async fetchEVMTokenInfo(tokenAddress: string): Promise<TokenInfo | null> {
// If it's null address, it's Ether. If this chain is not Ethereum then it's Wrapped Ether
if (tokenAddress.toLowerCase() === '0x0000000000000000000000000000000000000000') {
return {
name: this.chainId === 2 || this.chainId === 5 ? 'Ether' : 'Wrapped Ether',
symbol: this.chainId === 2 || this.chainId === 5 ? 'ETH' : 'WETH',
decimals: 18,
chain_id: this.chainId,
token_address: tokenAddress,
};
}

const provider = this.provider as providers.JsonRpcProvider;
const tokenContract = new ethers.Contract(tokenAddress, minABI, provider);
try {
const name = await tokenContract.name();
const symbol = await tokenContract.symbol();
const decimals = await tokenContract.decimals();
return { name, symbol, decimals, chain_id: this.chainId, token_address: tokenAddress };
} catch (error) {
console.error('Error fetching EVM token info:', error, tokenAddress);
return null;
}
}
}
34 changes: 16 additions & 18 deletions watcher/src/watchers/FTEVMWatcher.ts
Original file line number Diff line number Diff line change
Expand Up @@ -42,27 +42,25 @@ export class FTEVMWatcher extends Watcher {
this.provider = new ethers.providers.JsonRpcProvider(RPCS_BY_CHAIN[network][chain]);
this.rpc = RPCS_BY_CHAIN[this.network][this.chain]!;
this.tokenRouterParser = new TokenRouterParser(this.network, chain, this.provider);

// Initialize database connection before creating swap layer parser
if (!isTest) {
this.pg = knex({
client: 'pg',
connection: {
user: assertEnvironmentVariable('PG_FT_USER'),
password: assertEnvironmentVariable('PG_FT_PASSWORD'),
database: assertEnvironmentVariable('PG_FT_DATABASE'),
host: assertEnvironmentVariable('PG_FT_HOST'),
port: Number(assertEnvironmentVariable('PG_FT_PORT')),
},
});
}

this.swapLayerParser = this.swapLayerAddress
? new SwapLayerParser(this.provider, this.swapLayerAddress)
? new SwapLayerParser(this.provider, this.swapLayerAddress, this.pg, chain)
: null;
this.logger.debug('FTWatcher', network, chain, finalizedBlockTag);
// hacky way to not connect to the db in tests
// this is to allow ci to run without a db
if (isTest) {
// Components needed for testing is complete
return;
}

this.pg = knex({
client: 'pg',
connection: {
user: assertEnvironmentVariable('PG_FT_USER'),
password: assertEnvironmentVariable('PG_FT_PASSWORD'),
database: assertEnvironmentVariable('PG_FT_DATABASE'),
host: assertEnvironmentVariable('PG_FT_HOST'),
port: Number(assertEnvironmentVariable('PG_FT_PORT')),
},
});
}

async getBlock(blockNumberOrTag: number | BlockTag): Promise<Block> {
Expand Down

0 comments on commit b40ccac

Please sign in to comment.