forked from across-protocol/relayer
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathutils.ts
130 lines (114 loc) · 4.41 KB
/
utils.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
import assert from "assert";
import { Contract, utils as ethersUtils, Signer } from "ethers";
import readline from "readline";
import * as contracts from "@across-protocol/contracts";
import { utils as sdkUtils } from "@across-protocol/sdk";
import { getDeployedContract, getProvider, CHAIN_IDs } from "../src/utils";
// https://nodejs.org/api/process.html#exit-codes
export const NODE_SUCCESS = 0;
export const NODE_INPUT_ERR = 9;
export const NODE_APP_ERR = 127; // user-defined
export type ERC20 = {
address: string;
decimals: number;
symbol: string;
};
async function askQuestion(query: string) {
const rl = readline.createInterface({ input: process.stdin, output: process.stdout });
return new Promise((resolve) =>
rl.question(query, (ans) => {
rl.close();
resolve(ans);
})
);
}
export async function askYesNoQuestion(query: string): Promise<boolean> {
const ans = (await askQuestion(`${query} (y/n) `)) as string;
if (ans.toLowerCase() === "y") {
return true;
}
if (ans.toLowerCase() === "n") {
return false;
}
return askYesNoQuestion(query);
}
/**
* Resolves an ERC20 type from a chain ID, and symbol or address.
* @param token The address or symbol of the token to resolve.
* @param chainId The chain ID to resolve the token on.
* @returns The ERC20 attributes of the token.
*/
export function resolveToken(token: string, chainId: number): ERC20 {
// `token` may be an address or a symbol. Normalise it to a symbol for easy lookup.
const symbol = !ethersUtils.isAddress(token)
? token.toUpperCase()
: Object.values(contracts.TOKEN_SYMBOLS_MAP).find(({ addresses }) => addresses[chainId] === token)?.symbol;
const _token = contracts.TOKEN_SYMBOLS_MAP[symbol];
if (_token === undefined) {
throw new Error(`Token ${token} on chain ID ${chainId} unrecognised`);
}
return {
address: _token.addresses[chainId],
decimals: _token.decimals,
symbol: _token.symbol,
};
}
/**
* @description Verify that an array of chain IDs have known Across deployments.
* @dev This function does not detect if the test and production chain IDs have been mixed.
* @param chainIds Array of chain IDs to validate.
* @returns True if all chainIds are known.
*/
export function validateChainIds(chainIds: number[]): boolean {
return (
chainIds.every((chainId) => sdkUtils.chainIsProd(chainId)) ||
chainIds.every((chainId) => sdkUtils.chainIsTestnet(chainId))
);
}
/**
* @description For a SpokePool chain ID, resolve its corresponding HubPool chain ID.
* @param spokeChainId Chain ID of the SpokePool.
* @returns Chain ID for the corresponding HubPool.
*/
export function resolveHubChainId(spokeChainId: number): number {
if (sdkUtils.chainIsProd(spokeChainId)) {
return CHAIN_IDs.MAINNET;
}
assert(sdkUtils.chainIsTestnet(spokeChainId), `Unsupported testnet SpokePool chain ID: ${spokeChainId}`);
return CHAIN_IDs.SEPOLIA;
}
/**
* @description Instantiate an ethers Contract instance.
* @param chainId Chain ID for the contract deployment.
* @param contractName Name of the deployed contract.
* @returns ethers Contract instance.
*/
export async function getContract(chainId: number, contractName: string): Promise<Contract> {
const contract = getDeployedContract(contractName, chainId);
const provider = await getProvider(chainId);
return contract.connect(provider);
}
/**
* @description Instantiate an Across SpokePool contract instance.
* @param chainId Chain ID for the SpokePool deployment.
* @returns SpokePool contract instance.
*/
export async function getSpokePoolContract(chainId: number): Promise<Contract> {
const hubChainId = resolveHubChainId(chainId);
const hubPool = await getContract(hubChainId, "HubPool");
const spokePoolAddr = (await hubPool.crossChainContracts(chainId))[1];
const contract = new Contract(spokePoolAddr, contracts.SpokePool__factory.abi);
return contract;
}
/**
* @description Instantiate an Across OVM SpokePool contract instance.
* @param chainId Chain ID for the SpokePool deployment.
* @returns SpokePool contract instance.
*/
export async function getOvmSpokePoolContract(chainId: number, signer?: Signer): Promise<Contract> {
const hubChainId = resolveHubChainId(chainId);
const hubPool = await getContract(hubChainId, "HubPool");
const spokePoolAddr = (await hubPool.crossChainContracts(chainId))[1];
const contract = new Contract(spokePoolAddr, contracts.Ovm_SpokePool__factory.abi, signer);
return contract;
}