forked from across-protocol/relayer
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathwithdrawFromOpStack.ts
143 lines (132 loc) · 5.55 KB
/
withdrawFromOpStack.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
131
132
133
134
135
136
137
138
139
140
141
142
143
// Submits a bridge from OpStack L2 to L1.
// For now, this script only supports ETH withdrawals.
import {
ethers,
retrieveSignerFromCLIArgs,
getProvider,
ERC20,
WETH9,
TOKEN_SYMBOLS_MAP,
assert,
getL1TokenInfo,
Contract,
fromWei,
blockExplorerLink,
CHAIN_IDs,
ZERO_ADDRESS,
} from "../src/utils";
import { CONTRACT_ADDRESSES } from "../src/common";
import { askYesNoQuestion, getOvmSpokePoolContract } from "./utils";
import minimist from "minimist";
const cliArgs = ["amount", "chainId", "token"];
const args = minimist(process.argv.slice(2), {
string: cliArgs,
});
// Example run:
// ts-node ./scripts/withdrawFromOpStack.ts
// \ --amount 3000000000000000000
// \ --chainId 1135
// \ --wallet gckms
// \ --token WETH
// \ --keys bot1
export async function run(): Promise<void> {
assert(
cliArgs.every((cliArg) => Object.keys(args).includes(cliArg)),
`Missing cliArg, expected: ${cliArgs}`
);
const baseSigner = await retrieveSignerFromCLIArgs();
const signerAddr = await baseSigner.getAddress();
const chainId = parseInt(args.chainId);
const connectedSigner = baseSigner.connect(await getProvider(chainId));
const l2Token = TOKEN_SYMBOLS_MAP[args.token]?.addresses[chainId];
assert(l2Token, `${args.token} not found on chain ${chainId} in TOKEN_SYMBOLS_MAP`);
const l1TokenInfo = getL1TokenInfo(l2Token, chainId);
console.log("Fetched L1 token info:", l1TokenInfo);
const amount = args.amount;
const amountFromWei = ethers.utils.formatUnits(amount, l1TokenInfo.decimals);
console.log(`Amount to bridge from chain ${chainId}: ${amountFromWei} ${l2Token}`);
const erc20 = new Contract(l2Token, ERC20.abi, connectedSigner);
const currentBalance = await erc20.balanceOf(signerAddr);
const currentEthBalance = await connectedSigner.getBalance();
console.log(
`Current ${l1TokenInfo.symbol} balance for account ${signerAddr}: ${fromWei(
currentBalance,
l1TokenInfo.decimals
)} ${l2Token}`
);
console.log(`Current ETH balance for account ${signerAddr}: ${fromWei(currentEthBalance)}`);
// First offer user option to unwrap WETH into ETH
if (l1TokenInfo.symbol === "ETH") {
const weth = new Contract(l2Token, WETH9.abi, connectedSigner);
if (await askYesNoQuestion(`\nUnwrap ${amount} of WETH @ ${weth.address}?`)) {
const unwrap = await weth.withdraw(amount);
console.log(`Submitted transaction: ${blockExplorerLink(unwrap.hash, chainId)}.`);
const receipt = await unwrap.wait();
console.log("Unwrap complete...", receipt);
}
}
// Now, submit a withdrawal. This might fail if the ERC20 uses a non-standard OVM bridge to withdraw.
const ovmStandardBridgeObj = CONTRACT_ADDRESSES[chainId].ovmStandardBridge;
assert(CONTRACT_ADDRESSES[chainId].ovmStandardBridge, "ovmStandardBridge for chain not found in CONTRACT_ADDRESSES");
const ovmStandardBridge = new Contract(ovmStandardBridgeObj.address, ovmStandardBridgeObj.abi, connectedSigner);
const bridgeArgs =
l1TokenInfo.symbol === "ETH"
? [
signerAddr, // to
200_000, // minGasLimit
"0x", // extraData
{ value: amount }, // msg.value
]
: [
l2Token, // _localToken
TOKEN_SYMBOLS_MAP[args.token]?.addresses[CHAIN_IDs.MAINNET], // Remote token to be received on L1 side. If the
// remoteL1Token on the other chain does not recognize the local token as the correct
// pair token, the ERC20 bridge will fail and the tokens will be returned to sender on
// this chain.
signerAddr, // _to
amount, // _amount
200_000, // minGasLimit
"0x", // _data
];
const functionNameToCall = l1TokenInfo.symbol === "ETH" ? "bridgeETHTo" : "bridgeERC20To";
console.log(
`Submitting ${functionNameToCall} on the OVM standard bridge @ ${ovmStandardBridge.address} with the following args: `,
...bridgeArgs
);
// Sanity check that the ovmStandardBridge contract is the one we expect by comparing its stored addresses
// with the ones we have recorded.
const spokePool = await getOvmSpokePoolContract(chainId, connectedSigner);
const expectedL2Messenger = await spokePool.MESSENGER();
const l2Messenger = await ovmStandardBridge.MESSENGER();
assert(
l2Messenger === expectedL2Messenger,
`Unexpected L2 messenger address in ovmStandardBridge contract, expected: ${expectedL2Messenger}, got: ${l2Messenger}`
);
const l1StandardBridge = await ovmStandardBridge.l1TokenBridge();
const expectedL1StandardBridge = CONTRACT_ADDRESSES[CHAIN_IDs.MAINNET][`ovmStandardBridge_${chainId}`].address;
assert(
l1StandardBridge === expectedL1StandardBridge,
`Unexpected L1 standard bridge address in ovmStandardBridge contract, expected: ${expectedL1StandardBridge}, got: ${l1StandardBridge}`
);
const customTokenBridge = await spokePool.tokenBridges(l2Token);
assert(customTokenBridge === ZERO_ADDRESS, `Custom token bridge set for token ${l2Token} (${customTokenBridge})`);
if (!(await askYesNoQuestion("\nDo you want to proceed?"))) {
return;
}
const withdrawal = await ovmStandardBridge[functionNameToCall](...bridgeArgs);
console.log(`Submitted withdrawal: ${blockExplorerLink(withdrawal.hash, chainId)}.`);
const receipt = await withdrawal.wait();
console.log("Receipt", receipt);
}
if (require.main === module) {
run()
.then(async () => {
// eslint-disable-next-line no-process-exit
process.exit(0);
})
.catch(async (error) => {
console.error("Process exited with", error);
// eslint-disable-next-line no-process-exit
process.exit(1);
});
}