SDK: Auto relayer better status command (#3406)
* update clients.js version * updates to clients/js * update clients/js * Generate modification * prettier * status change * docs change * add back the 'impossible's * update sdk version * better status check WIP * WIP * Improved status check and stringify function with times * improvements to status printing * prettier format * Remove last console log * prettier * readme modify * readme fix * Readme fix * readme changes * don't rely on wormscan for status in devnet * prettier * Remove status check from integration testing - this is a helper that shouldn't interfere with contract testing * prettier * update clients.js version * readme revert changes * base default rpcs * script improvements * Add manual delivery helper * remove console logs * arbitrum needs a custom block range * fix bug in testing if blocknumber is 0 * deliver fixes for manual delivery * prettier * fix default block tag * pre-pend scripts with test * review comments
This commit is contained in:
parent
5b9b10ada4
commit
9aa4d0329d
|
@ -255,7 +255,8 @@ Options:
|
||||||
"avalanche", "oasis", "algorand", "aurora", "fantom", "karura", "acala",
|
"avalanche", "oasis", "algorand", "aurora", "fantom", "karura", "acala",
|
||||||
"klaytn", "celo", "near", "moonbeam", "neon", "terra2", "injective",
|
"klaytn", "celo", "near", "moonbeam", "neon", "terra2", "injective",
|
||||||
"osmosis", "sui", "aptos", "arbitrum", "optimism", "gnosis", "pythnet",
|
"osmosis", "sui", "aptos", "arbitrum", "optimism", "gnosis", "pythnet",
|
||||||
"xpla", "btc", "base", "sei", "wormchain", "sepolia"]
|
"xpla", "btc", "base", "sei", "rootstock", "wormchain", "cosmoshub", "evmos",
|
||||||
|
"kujira", "sepolia"]
|
||||||
-n, --network Network
|
-n, --network Network
|
||||||
[required] [choices: "mainnet", "testnet", "devnet"]
|
[required] [choices: "mainnet", "testnet", "devnet"]
|
||||||
-a, --contract-address Contract to submit VAA to (override config) [string]
|
-a, --contract-address Contract to submit VAA to (override config) [string]
|
||||||
|
@ -310,13 +311,15 @@ Options:
|
||||||
"avalanche", "oasis", "algorand", "aurora", "fantom", "karura", "acala",
|
"avalanche", "oasis", "algorand", "aurora", "fantom", "karura", "acala",
|
||||||
"klaytn", "celo", "near", "moonbeam", "neon", "terra2", "injective",
|
"klaytn", "celo", "near", "moonbeam", "neon", "terra2", "injective",
|
||||||
"osmosis", "sui", "aptos", "arbitrum", "optimism", "gnosis", "pythnet",
|
"osmosis", "sui", "aptos", "arbitrum", "optimism", "gnosis", "pythnet",
|
||||||
"xpla", "btc", "base", "sei", "wormchain", "sepolia"]
|
"xpla", "btc", "base", "sei", "rootstock", "wormchain", "cosmoshub", "evmos",
|
||||||
|
"kujira", "sepolia"]
|
||||||
--dst-chain destination chain
|
--dst-chain destination chain
|
||||||
[required] [choices: "solana", "ethereum", "terra", "bsc", "polygon",
|
[required] [choices: "solana", "ethereum", "terra", "bsc", "polygon",
|
||||||
"avalanche", "oasis", "algorand", "aurora", "fantom", "karura", "acala",
|
"avalanche", "oasis", "algorand", "aurora", "fantom", "karura", "acala",
|
||||||
"klaytn", "celo", "near", "moonbeam", "neon", "terra2", "injective",
|
"klaytn", "celo", "near", "moonbeam", "neon", "terra2", "injective",
|
||||||
"osmosis", "sui", "aptos", "arbitrum", "optimism", "gnosis", "pythnet",
|
"osmosis", "sui", "aptos", "arbitrum", "optimism", "gnosis", "pythnet",
|
||||||
"xpla", "btc", "base", "sei", "wormchain", "sepolia"]
|
"xpla", "btc", "base", "sei", "rootstock", "wormchain", "cosmoshub", "evmos",
|
||||||
|
"kujira", "sepolia"]
|
||||||
--dst-addr destination address [string] [required]
|
--dst-addr destination address [string] [required]
|
||||||
--token-addr token address [string] [default: native token]
|
--token-addr token address [string] [default: native token]
|
||||||
--amount token amount [string] [required]
|
--amount token amount [string] [required]
|
||||||
|
@ -342,16 +345,15 @@ Options:
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
Positionals:
|
Positionals:
|
||||||
network Network [choices: "mainnet", "testnet", "devnet"]
|
network Network [choices: "mainnet", "testnet", "devnet"]
|
||||||
chain Source chain
|
chain Source chain
|
||||||
[choices: "unset", "solana", "ethereum", "terra", "bsc", "polygon",
|
[choices: "unset", "solana", "ethereum", "terra", "bsc", "polygon",
|
||||||
"avalanche", "oasis", "algorand", "aurora", "fantom", "karura", "acala",
|
"avalanche", "oasis", "algorand", "aurora", "fantom", "karura", "acala",
|
||||||
"klaytn", "celo", "near", "moonbeam", "neon", "terra2", "injective",
|
"klaytn", "celo", "near", "moonbeam", "neon", "terra2", "injective",
|
||||||
"osmosis", "sui", "aptos", "arbitrum", "optimism", "gnosis", "pythnet",
|
"osmosis", "sui", "aptos", "arbitrum", "optimism", "gnosis", "pythnet",
|
||||||
"xpla", "btc", "base", "sei", "wormchain", "sepolia"]
|
"xpla", "btc", "base", "sei", "rootstock", "wormchain", "cosmoshub", "evmos",
|
||||||
tx Source transaction hash [string]
|
"kujira", "sepolia"]
|
||||||
block-start Starting Block Range, i.e. -2048 [string]
|
tx Source transaction hash [string]
|
||||||
block-end Ending Block Range, i.e. latest [string]
|
|
||||||
|
|
||||||
Options:
|
Options:
|
||||||
--help Show help [boolean]
|
--help Show help [boolean]
|
||||||
|
|
|
@ -10,7 +10,7 @@
|
||||||
"license": "Apache-2.0",
|
"license": "Apache-2.0",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@celo-tools/celo-ethers-wrapper": "^0.1.0",
|
"@celo-tools/celo-ethers-wrapper": "^0.1.0",
|
||||||
"@certusone/wormhole-sdk": "^0.9.24-beta.0",
|
"@certusone/wormhole-sdk": "^0.10.5-beta.3",
|
||||||
"@cosmjs/encoding": "^0.26.2",
|
"@cosmjs/encoding": "^0.26.2",
|
||||||
"@improbable-eng/grpc-web-node-http-transport": "^0.15.0",
|
"@improbable-eng/grpc-web-node-http-transport": "^0.15.0",
|
||||||
"@injectivelabs/networks": "^1.10.7",
|
"@injectivelabs/networks": "^1.10.7",
|
||||||
|
@ -494,9 +494,9 @@
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@certusone/wormhole-sdk": {
|
"node_modules/@certusone/wormhole-sdk": {
|
||||||
"version": "0.9.24-beta.0",
|
"version": "0.10.5-beta.3",
|
||||||
"resolved": "https://registry.npmjs.org/@certusone/wormhole-sdk/-/wormhole-sdk-0.9.24-beta.0.tgz",
|
"resolved": "https://registry.npmjs.org/@certusone/wormhole-sdk/-/wormhole-sdk-0.10.5-beta.3.tgz",
|
||||||
"integrity": "sha512-SiUd9EAgPM5RfwPJ56I49K4Rgzkc5pVMqXDk2udRX+Q9dbvlaTW4d/Cu1dk+x1n6uSjAjact0pNz19C1yDebLA==",
|
"integrity": "sha512-Q1IrWYQ/NPpLcuPs3J8MbUeRwBxBOTZFwLs3jahohgUFCQhGuBwhgpv02CdEYBR/iZpgaDrj2Eo3HOe3uNlYVA==",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@certusone/wormhole-sdk-proto-web": "0.0.6",
|
"@certusone/wormhole-sdk-proto-web": "0.0.6",
|
||||||
"@certusone/wormhole-sdk-wasm": "^0.0.1",
|
"@certusone/wormhole-sdk-wasm": "^0.0.1",
|
||||||
|
@ -550,9 +550,9 @@
|
||||||
"integrity": "sha512-lcHwpNoggQTObv5apGNCTdJrO69eHOZMi4BNC+rTLER8iHAqGrUVeLh/irVIM7zTw2bOXA8T6uNPeujwOLg/2Q=="
|
"integrity": "sha512-lcHwpNoggQTObv5apGNCTdJrO69eHOZMi4BNC+rTLER8iHAqGrUVeLh/irVIM7zTw2bOXA8T6uNPeujwOLg/2Q=="
|
||||||
},
|
},
|
||||||
"node_modules/@certusone/wormhole-sdk-proto-web/node_modules/protobufjs": {
|
"node_modules/@certusone/wormhole-sdk-proto-web/node_modules/protobufjs": {
|
||||||
"version": "7.2.3",
|
"version": "7.2.5",
|
||||||
"resolved": "https://registry.npmjs.org/protobufjs/-/protobufjs-7.2.3.tgz",
|
"resolved": "https://registry.npmjs.org/protobufjs/-/protobufjs-7.2.5.tgz",
|
||||||
"integrity": "sha512-TtpvOqwB5Gdz/PQmOjgsrGH1nHjAQVCN7JG4A6r1sXRWESL5rNMAiRcBQlCAdKxZcAbstExQePYG8xof/JVRgg==",
|
"integrity": "sha512-gGXRSXvxQ7UiPgfw8gevrfRWcTlSbOFg+p/N+JVJEK5VhueL2miT6qTymqAmjr1Q5WbOCyJbyrk6JfWKwlFn6A==",
|
||||||
"hasInstallScript": true,
|
"hasInstallScript": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@protobufjs/aspromise": "^1.1.2",
|
"@protobufjs/aspromise": "^1.1.2",
|
||||||
|
@ -2995,11 +2995,11 @@
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@project-serum/anchor/node_modules/cross-fetch": {
|
"node_modules/@project-serum/anchor/node_modules/cross-fetch": {
|
||||||
"version": "3.1.5",
|
"version": "3.1.8",
|
||||||
"resolved": "https://registry.npmjs.org/cross-fetch/-/cross-fetch-3.1.5.tgz",
|
"resolved": "https://registry.npmjs.org/cross-fetch/-/cross-fetch-3.1.8.tgz",
|
||||||
"integrity": "sha512-lvb1SBsI0Z7GDwmuid+mU3kWVBwTVUbe7S0H52yaaAdQOXq2YktTCZdlAcNKFzE6QtRz0snpw9bNiPeOIkkQvw==",
|
"integrity": "sha512-cvA+JwZoU0Xq+h6WkMvAUqPEYy92Obet6UdKLfW60qn99ftItKjB5T+BkyWOFWe2pUyfQ+IJHmpOTznqk1M6Kg==",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"node-fetch": "2.6.7"
|
"node-fetch": "^2.6.12"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@project-serum/anchor/node_modules/superstruct": {
|
"node_modules/@project-serum/anchor/node_modules/superstruct": {
|
||||||
|
@ -3710,25 +3710,6 @@
|
||||||
"node-fetch": "^2.6.12"
|
"node-fetch": "^2.6.12"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/algosdk/node_modules/node-fetch": {
|
|
||||||
"version": "2.6.12",
|
|
||||||
"resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.6.12.tgz",
|
|
||||||
"integrity": "sha512-C/fGU2E8ToujUivIO0H+tpQ6HWo4eEmchoPIoXtxCrVghxdKq+QOHqEZW7tuP3KlV3bC8FRMO5nMCC7Zm1VP6g==",
|
|
||||||
"dependencies": {
|
|
||||||
"whatwg-url": "^5.0.0"
|
|
||||||
},
|
|
||||||
"engines": {
|
|
||||||
"node": "4.x || >=6.0.0"
|
|
||||||
},
|
|
||||||
"peerDependencies": {
|
|
||||||
"encoding": "^0.1.0"
|
|
||||||
},
|
|
||||||
"peerDependenciesMeta": {
|
|
||||||
"encoding": {
|
|
||||||
"optional": true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"node_modules/ansi-regex": {
|
"node_modules/ansi-regex": {
|
||||||
"version": "5.0.1",
|
"version": "5.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz",
|
||||||
|
@ -6687,9 +6668,9 @@
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
"node_modules/node-fetch": {
|
"node_modules/node-fetch": {
|
||||||
"version": "2.6.7",
|
"version": "2.7.0",
|
||||||
"resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.6.7.tgz",
|
"resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.7.0.tgz",
|
||||||
"integrity": "sha512-ZjMPFEfVx5j+y2yF35Kzx5sF7kDzxuDj6ziH4FFbOp87zKDZNx8yExJIb05OGF4Nlt9IHFIMBkRl41VdvcNdbQ==",
|
"integrity": "sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A==",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"whatwg-url": "^5.0.0"
|
"whatwg-url": "^5.0.0"
|
||||||
},
|
},
|
||||||
|
@ -8386,9 +8367,9 @@
|
||||||
"requires": {}
|
"requires": {}
|
||||||
},
|
},
|
||||||
"@certusone/wormhole-sdk": {
|
"@certusone/wormhole-sdk": {
|
||||||
"version": "0.9.24-beta.0",
|
"version": "0.10.5-beta.3",
|
||||||
"resolved": "https://registry.npmjs.org/@certusone/wormhole-sdk/-/wormhole-sdk-0.9.24-beta.0.tgz",
|
"resolved": "https://registry.npmjs.org/@certusone/wormhole-sdk/-/wormhole-sdk-0.10.5-beta.3.tgz",
|
||||||
"integrity": "sha512-SiUd9EAgPM5RfwPJ56I49K4Rgzkc5pVMqXDk2udRX+Q9dbvlaTW4d/Cu1dk+x1n6uSjAjact0pNz19C1yDebLA==",
|
"integrity": "sha512-Q1IrWYQ/NPpLcuPs3J8MbUeRwBxBOTZFwLs3jahohgUFCQhGuBwhgpv02CdEYBR/iZpgaDrj2Eo3HOe3uNlYVA==",
|
||||||
"requires": {
|
"requires": {
|
||||||
"@certusone/wormhole-sdk-proto-web": "0.0.6",
|
"@certusone/wormhole-sdk-proto-web": "0.0.6",
|
||||||
"@certusone/wormhole-sdk-wasm": "^0.0.1",
|
"@certusone/wormhole-sdk-wasm": "^0.0.1",
|
||||||
|
@ -8444,9 +8425,9 @@
|
||||||
"integrity": "sha512-lcHwpNoggQTObv5apGNCTdJrO69eHOZMi4BNC+rTLER8iHAqGrUVeLh/irVIM7zTw2bOXA8T6uNPeujwOLg/2Q=="
|
"integrity": "sha512-lcHwpNoggQTObv5apGNCTdJrO69eHOZMi4BNC+rTLER8iHAqGrUVeLh/irVIM7zTw2bOXA8T6uNPeujwOLg/2Q=="
|
||||||
},
|
},
|
||||||
"protobufjs": {
|
"protobufjs": {
|
||||||
"version": "7.2.3",
|
"version": "7.2.5",
|
||||||
"resolved": "https://registry.npmjs.org/protobufjs/-/protobufjs-7.2.3.tgz",
|
"resolved": "https://registry.npmjs.org/protobufjs/-/protobufjs-7.2.5.tgz",
|
||||||
"integrity": "sha512-TtpvOqwB5Gdz/PQmOjgsrGH1nHjAQVCN7JG4A6r1sXRWESL5rNMAiRcBQlCAdKxZcAbstExQePYG8xof/JVRgg==",
|
"integrity": "sha512-gGXRSXvxQ7UiPgfw8gevrfRWcTlSbOFg+p/N+JVJEK5VhueL2miT6qTymqAmjr1Q5WbOCyJbyrk6JfWKwlFn6A==",
|
||||||
"requires": {
|
"requires": {
|
||||||
"@protobufjs/aspromise": "^1.1.2",
|
"@protobufjs/aspromise": "^1.1.2",
|
||||||
"@protobufjs/base64": "^1.1.2",
|
"@protobufjs/base64": "^1.1.2",
|
||||||
|
@ -10288,11 +10269,11 @@
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"cross-fetch": {
|
"cross-fetch": {
|
||||||
"version": "3.1.5",
|
"version": "3.1.8",
|
||||||
"resolved": "https://registry.npmjs.org/cross-fetch/-/cross-fetch-3.1.5.tgz",
|
"resolved": "https://registry.npmjs.org/cross-fetch/-/cross-fetch-3.1.8.tgz",
|
||||||
"integrity": "sha512-lvb1SBsI0Z7GDwmuid+mU3kWVBwTVUbe7S0H52yaaAdQOXq2YktTCZdlAcNKFzE6QtRz0snpw9bNiPeOIkkQvw==",
|
"integrity": "sha512-cvA+JwZoU0Xq+h6WkMvAUqPEYy92Obet6UdKLfW60qn99ftItKjB5T+BkyWOFWe2pUyfQ+IJHmpOTznqk1M6Kg==",
|
||||||
"requires": {
|
"requires": {
|
||||||
"node-fetch": "2.6.7"
|
"node-fetch": "^2.6.12"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"superstruct": {
|
"superstruct": {
|
||||||
|
@ -10891,14 +10872,6 @@
|
||||||
"requires": {
|
"requires": {
|
||||||
"node-fetch": "^2.6.12"
|
"node-fetch": "^2.6.12"
|
||||||
}
|
}
|
||||||
},
|
|
||||||
"node-fetch": {
|
|
||||||
"version": "2.6.12",
|
|
||||||
"resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.6.12.tgz",
|
|
||||||
"integrity": "sha512-C/fGU2E8ToujUivIO0H+tpQ6HWo4eEmchoPIoXtxCrVghxdKq+QOHqEZW7tuP3KlV3bC8FRMO5nMCC7Zm1VP6g==",
|
|
||||||
"requires": {
|
|
||||||
"whatwg-url": "^5.0.0"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@ -13396,9 +13369,9 @@
|
||||||
"version": "2.0.2"
|
"version": "2.0.2"
|
||||||
},
|
},
|
||||||
"node-fetch": {
|
"node-fetch": {
|
||||||
"version": "2.6.7",
|
"version": "2.7.0",
|
||||||
"resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.6.7.tgz",
|
"resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.7.0.tgz",
|
||||||
"integrity": "sha512-ZjMPFEfVx5j+y2yF35Kzx5sF7kDzxuDj6ziH4FFbOp87zKDZNx8yExJIb05OGF4Nlt9IHFIMBkRl41VdvcNdbQ==",
|
"integrity": "sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A==",
|
||||||
"requires": {
|
"requires": {
|
||||||
"whatwg-url": "^5.0.0"
|
"whatwg-url": "^5.0.0"
|
||||||
}
|
}
|
||||||
|
|
|
@ -30,7 +30,7 @@
|
||||||
],
|
],
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@celo-tools/celo-ethers-wrapper": "^0.1.0",
|
"@celo-tools/celo-ethers-wrapper": "^0.1.0",
|
||||||
"@certusone/wormhole-sdk": "^0.9.24-beta.0",
|
"@certusone/wormhole-sdk": "^0.10.5-beta.3",
|
||||||
"@cosmjs/encoding": "^0.26.2",
|
"@cosmjs/encoding": "^0.26.2",
|
||||||
"@improbable-eng/grpc-web-node-http-transport": "^0.15.0",
|
"@improbable-eng/grpc-web-node-http-transport": "^0.15.0",
|
||||||
"@injectivelabs/networks": "^1.10.7",
|
"@injectivelabs/networks": "^1.10.7",
|
||||||
|
|
|
@ -110,6 +110,10 @@ export const getOriginalAsset = async (
|
||||||
case "osmosis":
|
case "osmosis":
|
||||||
case "pythnet":
|
case "pythnet":
|
||||||
case "wormchain":
|
case "wormchain":
|
||||||
|
case "cosmoshub":
|
||||||
|
case "evmos":
|
||||||
|
case "kujira":
|
||||||
|
case "rootstock":
|
||||||
throw new Error(`${chainName} not supported`);
|
throw new Error(`${chainName} not supported`);
|
||||||
default:
|
default:
|
||||||
impossible(chainName);
|
impossible(chainName);
|
||||||
|
|
|
@ -156,6 +156,10 @@ export const getWrappedAssetAddress = async (
|
||||||
case "osmosis":
|
case "osmosis":
|
||||||
case "pythnet":
|
case "pythnet":
|
||||||
case "wormchain":
|
case "wormchain":
|
||||||
|
case "cosmoshub":
|
||||||
|
case "evmos":
|
||||||
|
case "kujira":
|
||||||
|
case "rootstock":
|
||||||
throw new Error(`${chainName} not supported`);
|
throw new Error(`${chainName} not supported`);
|
||||||
default:
|
default:
|
||||||
impossible(chainName);
|
impossible(chainName);
|
||||||
|
|
|
@ -154,6 +154,10 @@ export const getProviderForChain = <T extends ChainId | ChainName>(
|
||||||
case "osmosis":
|
case "osmosis":
|
||||||
case "pythnet":
|
case "pythnet":
|
||||||
case "wormchain":
|
case "wormchain":
|
||||||
|
case "cosmoshub":
|
||||||
|
case "evmos":
|
||||||
|
case "kujira":
|
||||||
|
case "rootstock":
|
||||||
throw new Error(`${chainName} not supported`);
|
throw new Error(`${chainName} not supported`);
|
||||||
default:
|
default:
|
||||||
impossible(chainName);
|
impossible(chainName);
|
||||||
|
|
|
@ -311,8 +311,6 @@ function parseAddress(chain: ChainName, address: string): string {
|
||||||
return "0x" + evm_address(address);
|
return "0x" + evm_address(address);
|
||||||
} else if (chain === "near") {
|
} else if (chain === "near") {
|
||||||
return "0x" + evm_address(address);
|
return "0x" + evm_address(address);
|
||||||
} else if (chain === "osmosis") {
|
|
||||||
throw Error("OSMOSIS is not supported yet");
|
|
||||||
} else if (chain === "sui") {
|
} else if (chain === "sui") {
|
||||||
return "0x" + evm_address(address);
|
return "0x" + evm_address(address);
|
||||||
} else if (chain === "aptos") {
|
} else if (chain === "aptos") {
|
||||||
|
@ -321,10 +319,16 @@ function parseAddress(chain: ChainName, address: string): string {
|
||||||
}
|
}
|
||||||
|
|
||||||
return sha3_256(Buffer.from(address)); // address is hash of fully qualified type
|
return sha3_256(Buffer.from(address)); // address is hash of fully qualified type
|
||||||
} else if (chain === "wormchain") {
|
|
||||||
return "0x" + tryNativeToHexString(address, chain);
|
|
||||||
} else if (chain === "btc") {
|
} else if (chain === "btc") {
|
||||||
throw Error("btc is not supported yet");
|
throw Error("btc is not supported yet");
|
||||||
|
} else if (chain === "cosmoshub") {
|
||||||
|
throw Error("cosmoshub is not supported yet");
|
||||||
|
} else if (chain === "evmos") {
|
||||||
|
throw Error("evmos is not supported yet");
|
||||||
|
} else if (chain === "kujira") {
|
||||||
|
throw Error("kujira is not supported yet");
|
||||||
|
} else if (chain === "rootstock") {
|
||||||
|
throw Error("rootstock is not supported yet");
|
||||||
} else {
|
} else {
|
||||||
impossible(chain);
|
impossible(chain);
|
||||||
}
|
}
|
||||||
|
|
|
@ -29,14 +29,6 @@ export const builder = (y: typeof yargs) =>
|
||||||
describe: "Source transaction hash",
|
describe: "Source transaction hash",
|
||||||
type: "string",
|
type: "string",
|
||||||
demandOption: true,
|
demandOption: true,
|
||||||
} as const)
|
|
||||||
.positional("block-start", {
|
|
||||||
describe: "Starting Block Range, i.e. -2048",
|
|
||||||
type: "string",
|
|
||||||
} as const)
|
|
||||||
.positional("block-end", {
|
|
||||||
describe: "Ending Block Range, i.e. latest",
|
|
||||||
type: "string",
|
|
||||||
} as const);
|
} as const);
|
||||||
export const handler = async (
|
export const handler = async (
|
||||||
argv: Awaited<ReturnType<typeof builder>["argv"]>
|
argv: Awaited<ReturnType<typeof builder>["argv"]>
|
||||||
|
@ -59,39 +51,16 @@ export const handler = async (
|
||||||
targetChainProviders.set(
|
targetChainProviders.set(
|
||||||
key as ChainName,
|
key as ChainName,
|
||||||
new ethers.providers.JsonRpcProvider(
|
new ethers.providers.JsonRpcProvider(
|
||||||
NETWORKS[network][key as ChainName].rpc
|
NETWORKS[network as Network][key as ChainName].rpc
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
const targetChainBlockRanges = new Map<
|
|
||||||
ChainName,
|
|
||||||
[ethers.providers.BlockTag, ethers.providers.BlockTag]
|
|
||||||
>();
|
|
||||||
const getBlockTag = (tagString: string): ethers.providers.BlockTag => {
|
|
||||||
if (+tagString) return parseInt(tagString);
|
|
||||||
return tagString;
|
|
||||||
};
|
|
||||||
for (const key in NETWORKS[network]) {
|
|
||||||
targetChainBlockRanges.set(key as ChainName, [
|
|
||||||
getBlockTag(argv["block-start"] || "-2048"),
|
|
||||||
getBlockTag(argv["block-end"] || "latest"),
|
|
||||||
]);
|
|
||||||
}
|
|
||||||
|
|
||||||
const info = await relayer.getWormholeRelayerInfo(chain, argv.tx, {
|
const info = await relayer.getWormholeRelayerInfo(chain, argv.tx, {
|
||||||
environment: network,
|
environment: network,
|
||||||
sourceChainProvider,
|
sourceChainProvider,
|
||||||
targetChainProviders,
|
targetChainProviders,
|
||||||
targetChainBlockRanges,
|
|
||||||
});
|
});
|
||||||
|
|
||||||
console.log(relayer.stringifyWormholeRelayerInfo(info));
|
console.log(relayer.stringifyWormholeRelayerInfo(info));
|
||||||
if (
|
|
||||||
info.targetChainStatus.events[0].status ===
|
|
||||||
relayer.DeliveryStatus.DeliveryDidntHappenWithinRange
|
|
||||||
) {
|
|
||||||
console.log(
|
|
||||||
"Try using the '--block-start' and '--block-end' flags to specify a different block range"
|
|
||||||
);
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
|
@ -184,6 +184,14 @@ async function executeSubmit(
|
||||||
throw Error("Wormchain is not supported yet");
|
throw Error("Wormchain is not supported yet");
|
||||||
} else if (chain === "btc") {
|
} else if (chain === "btc") {
|
||||||
throw Error("btc is not supported yet");
|
throw Error("btc is not supported yet");
|
||||||
|
} else if (chain === "cosmoshub") {
|
||||||
|
throw Error("Cosmoshub is not supported yet");
|
||||||
|
} else if (chain === "evmos") {
|
||||||
|
throw Error("Evmos is not supported yet");
|
||||||
|
} else if (chain === "kujira") {
|
||||||
|
throw Error("kujira is not supported yet");
|
||||||
|
} else if (chain === "rootstock") {
|
||||||
|
throw Error("rootstock is not supported yet");
|
||||||
} else {
|
} else {
|
||||||
// If you get a type error here, hover over `chain`'s type and it tells you
|
// If you get a type error here, hover over `chain`'s type and it tells you
|
||||||
// which cases are not handled
|
// which cases are not handled
|
||||||
|
|
|
@ -138,6 +138,14 @@ export const handler = async (
|
||||||
throw Error("Wormchain is not supported yet");
|
throw Error("Wormchain is not supported yet");
|
||||||
} else if (srcChain === "btc") {
|
} else if (srcChain === "btc") {
|
||||||
throw Error("btc is not supported yet");
|
throw Error("btc is not supported yet");
|
||||||
|
} else if (srcChain === "cosmoshub") {
|
||||||
|
throw Error("cosmoshub is not supported yet");
|
||||||
|
} else if (srcChain === "evmos") {
|
||||||
|
throw Error("evmos is not supported yet");
|
||||||
|
} else if (srcChain === "kujira") {
|
||||||
|
throw Error("kujira is not supported yet");
|
||||||
|
} else if (srcChain === "rootstock") {
|
||||||
|
throw Error("rootstock is not supported yet");
|
||||||
} else {
|
} else {
|
||||||
// If you get a type error here, hover over `chain`'s type and it tells you
|
// If you get a type error here, hover over `chain`'s type and it tells you
|
||||||
// which cases are not handled
|
// which cases are not handled
|
||||||
|
|
|
@ -178,6 +178,21 @@ const MAINNET = {
|
||||||
key: undefined,
|
key: undefined,
|
||||||
chain_id: undefined,
|
chain_id: undefined,
|
||||||
},
|
},
|
||||||
|
cosmoshub: {
|
||||||
|
rpc: undefined,
|
||||||
|
key: undefined,
|
||||||
|
chain_id: undefined,
|
||||||
|
},
|
||||||
|
evmos: {
|
||||||
|
rpc: undefined,
|
||||||
|
key: undefined,
|
||||||
|
chain_id: undefined,
|
||||||
|
},
|
||||||
|
kujira: {
|
||||||
|
rpc: undefined,
|
||||||
|
key: undefined,
|
||||||
|
chain_id: undefined,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
const TESTNET = {
|
const TESTNET = {
|
||||||
|
@ -343,6 +358,21 @@ const TESTNET = {
|
||||||
key: getEnvVar("ETH_KEY_TESTNET"),
|
key: getEnvVar("ETH_KEY_TESTNET"),
|
||||||
chain_id: 31,
|
chain_id: 31,
|
||||||
},
|
},
|
||||||
|
cosmoshub: {
|
||||||
|
rpc: undefined,
|
||||||
|
key: undefined,
|
||||||
|
chain_id: undefined,
|
||||||
|
},
|
||||||
|
evmos: {
|
||||||
|
rpc: undefined,
|
||||||
|
key: undefined,
|
||||||
|
chain_id: undefined,
|
||||||
|
},
|
||||||
|
kujira: {
|
||||||
|
rpc: undefined,
|
||||||
|
key: undefined,
|
||||||
|
chain_id: undefined,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
const DEVNET = {
|
const DEVNET = {
|
||||||
|
@ -488,6 +518,18 @@ const DEVNET = {
|
||||||
rpc: undefined,
|
rpc: undefined,
|
||||||
key: undefined,
|
key: undefined,
|
||||||
},
|
},
|
||||||
|
cosmoshub: {
|
||||||
|
rpc: undefined,
|
||||||
|
key: undefined,
|
||||||
|
},
|
||||||
|
evmos: {
|
||||||
|
rpc: undefined,
|
||||||
|
key: undefined,
|
||||||
|
},
|
||||||
|
kujira: {
|
||||||
|
rpc: undefined,
|
||||||
|
key: undefined,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -28,7 +28,9 @@
|
||||||
"prepublishOnly": "echo \"disabled: npm test && npm run lint\"",
|
"prepublishOnly": "echo \"disabled: npm test && npm run lint\"",
|
||||||
"preversion": "npm run lint",
|
"preversion": "npm run lint",
|
||||||
"version": "npm run format && git add -A src",
|
"version": "npm run format && git add -A src",
|
||||||
"postversion": "git push && git push --tags"
|
"postversion": "git push && git push --tags",
|
||||||
|
"test-relayer-status": "npm run test-relayer-testnet -- ./src/relayer/__tests__/wormhole_relayer.ts -t 'Checks the status of a message'",
|
||||||
|
"test-relayer-manual-delivery": "npm run test-relayer-testnet -- ./src/relayer/__tests__/wormhole_relayer.ts -t 'custom manual delivery'"
|
||||||
},
|
},
|
||||||
"keywords": [
|
"keywords": [
|
||||||
"wormhole",
|
"wormhole",
|
||||||
|
|
|
@ -25,9 +25,8 @@ import {
|
||||||
} from "../../../";
|
} from "../../../";
|
||||||
import { GovernanceEmitter, MockGuardians } from "../../../src/mock";
|
import { GovernanceEmitter, MockGuardians } from "../../../src/mock";
|
||||||
import { Implementation__factory } from "../../ethers-contracts";
|
import { Implementation__factory } from "../../ethers-contracts";
|
||||||
import { deliver } from "../relayer";
|
import { manualDelivery } from "../relayer";
|
||||||
import { NodeHttpTransport } from "@improbable-eng/grpc-web-node-http-transport";
|
import { NodeHttpTransport } from "@improbable-eng/grpc-web-node-http-transport";
|
||||||
import { getSignedVAAWithRetry } from "../../rpc";
|
|
||||||
import { packEVMExecutionInfoV1 } from "../structs";
|
import { packEVMExecutionInfoV1 } from "../structs";
|
||||||
|
|
||||||
const network: Network = getNetwork();
|
const network: Network = getNetwork();
|
||||||
|
@ -37,6 +36,7 @@ const sourceChain = network == "DEVNET" ? "ethereum" : "celo";
|
||||||
const targetChain = network == "DEVNET" ? "bsc" : "avalanche";
|
const targetChain = network == "DEVNET" ? "bsc" : "avalanche";
|
||||||
|
|
||||||
const testIfDevnet = () => (network == "DEVNET" ? test : test.skip);
|
const testIfDevnet = () => (network == "DEVNET" ? test : test.skip);
|
||||||
|
const testIfNotDevnet = () => (network != "DEVNET" ? test : test.skip);
|
||||||
|
|
||||||
type TestChain = {
|
type TestChain = {
|
||||||
chainId: ChainId;
|
chainId: ChainId;
|
||||||
|
@ -178,10 +178,6 @@ describe("Wormhole Relayer Tests", () => {
|
||||||
|
|
||||||
await waitForRelay();
|
await waitForRelay();
|
||||||
|
|
||||||
console.log("Checking status using SDK");
|
|
||||||
const status = await getStatus(rx.transactionHash);
|
|
||||||
expect(status).toBe("Delivery Success");
|
|
||||||
|
|
||||||
console.log("Checking if message was relayed");
|
console.log("Checking if message was relayed");
|
||||||
const message = await target.mockIntegration.getMessage();
|
const message = await target.mockIntegration.getMessage();
|
||||||
expect(message).toBe(arbitraryPayload);
|
expect(message).toBe(arbitraryPayload);
|
||||||
|
@ -228,10 +224,6 @@ describe("Wormhole Relayer Tests", () => {
|
||||||
|
|
||||||
await waitForRelay();
|
await waitForRelay();
|
||||||
|
|
||||||
console.log("Checking status using SDK");
|
|
||||||
const status = await getStatus(tx.hash);
|
|
||||||
expect(status).toBe("Delivery Success");
|
|
||||||
|
|
||||||
console.log("Checking if message was relayed");
|
console.log("Checking if message was relayed");
|
||||||
const message = (await target.mockIntegration.getDeliveryData())
|
const message = (await target.mockIntegration.getDeliveryData())
|
||||||
.additionalVaas[0];
|
.additionalVaas[0];
|
||||||
|
@ -239,87 +231,86 @@ describe("Wormhole Relayer Tests", () => {
|
||||||
expect(parsedMessage.payload).toBe(arbitraryPayload);
|
expect(parsedMessage.payload).toBe(arbitraryPayload);
|
||||||
});
|
});
|
||||||
|
|
||||||
test("Executes a Delivery Success with manual delivery", async () => {
|
testIfNotDevnet()(
|
||||||
const arbitraryPayload = getArbitraryBytes32();
|
"Executes a Delivery Success with manual delivery",
|
||||||
console.log(`Sent message: ${arbitraryPayload}`);
|
async () => {
|
||||||
|
const arbitraryPayload = getArbitraryBytes32();
|
||||||
|
console.log(`Sent message: ${arbitraryPayload}`);
|
||||||
|
|
||||||
const deliverySeq = await Implementation__factory.connect(
|
const deliverySeq = await Implementation__factory.connect(
|
||||||
CONTRACTS[network][sourceChain].core || "",
|
CONTRACTS[network][sourceChain].core || "",
|
||||||
source.provider
|
source.provider
|
||||||
).nextSequence(source.wormholeRelayerAddress);
|
).nextSequence(source.wormholeRelayerAddress);
|
||||||
|
|
||||||
const rx = await testSend(arbitraryPayload, false, true);
|
const rx = await testSend(arbitraryPayload, false, true);
|
||||||
|
|
||||||
await waitForRelay();
|
await waitForRelay();
|
||||||
|
|
||||||
// confirm that the message was not relayed successfully
|
// confirm that the message was not relayed successfully
|
||||||
{
|
|
||||||
const message = await target.mockIntegration.getMessage();
|
|
||||||
expect(message).not.toBe(arbitraryPayload);
|
|
||||||
|
|
||||||
console.log("Checking status using SDK");
|
|
||||||
const status = await getStatus(rx.transactionHash);
|
|
||||||
expect(status).toBe("Receiver Failure");
|
|
||||||
}
|
|
||||||
const [value, refundPerGasUnused] = await relayer.getPriceAndRefundInfo(
|
|
||||||
sourceChain,
|
|
||||||
targetChain,
|
|
||||||
REASONABLE_GAS_LIMIT,
|
|
||||||
optionalParams
|
|
||||||
);
|
|
||||||
|
|
||||||
const info = (await relayer.getWormholeRelayerInfo(
|
|
||||||
sourceChain,
|
|
||||||
rx.transactionHash,
|
|
||||||
{ wormholeRelayerAddresses, ...optionalParams }
|
|
||||||
)) as relayer.DeliveryInfo;
|
|
||||||
|
|
||||||
const rpc = getGuardianRPC(network, ci);
|
|
||||||
const emitterAddress = Buffer.from(
|
|
||||||
tryNativeToUint8Array(source.wormholeRelayerAddress, "ethereum")
|
|
||||||
);
|
|
||||||
const deliveryVaa = await getSignedVAAWithRetry(
|
|
||||||
[rpc],
|
|
||||||
source.chainId,
|
|
||||||
emitterAddress.toString("hex"),
|
|
||||||
deliverySeq.toBigInt().toString(),
|
|
||||||
{ transport: NodeHttpTransport() }
|
|
||||||
);
|
|
||||||
|
|
||||||
console.log(`Got delivery VAA: ${deliveryVaa}`);
|
|
||||||
const deliveryRx = await deliver(
|
|
||||||
deliveryVaa.vaaBytes,
|
|
||||||
target.wallet,
|
|
||||||
getGuardianRPC(network, ci),
|
|
||||||
network,
|
|
||||||
{
|
{
|
||||||
newExecutionInfo: Buffer.from(
|
const message = await target.mockIntegration.getMessage();
|
||||||
packEVMExecutionInfoV1({
|
expect(message).not.toBe(arbitraryPayload);
|
||||||
gasLimit: ethers.BigNumber.from(REASONABLE_GAS_LIMIT),
|
|
||||||
targetChainRefundPerGasUnused:
|
|
||||||
ethers.BigNumber.from(refundPerGasUnused),
|
|
||||||
}).substring(2),
|
|
||||||
"hex"
|
|
||||||
),
|
|
||||||
newReceiverValue: ethers.BigNumber.from(0),
|
|
||||||
redeliveryHash: Buffer.from(
|
|
||||||
ethers.utils.keccak256("0x1234").substring(2),
|
|
||||||
"hex"
|
|
||||||
), // fake a redelivery
|
|
||||||
}
|
}
|
||||||
);
|
const [value, refundPerGasUnused] = await relayer.getPriceAndRefundInfo(
|
||||||
console.log("Manual delivery tx hash", deliveryRx.transactionHash);
|
sourceChain,
|
||||||
console.log("Manual delivery tx status", deliveryRx.status);
|
targetChain,
|
||||||
|
REASONABLE_GAS_LIMIT,
|
||||||
|
optionalParams
|
||||||
|
);
|
||||||
|
|
||||||
console.log("Checking status using SDK");
|
const priceInfo = await manualDelivery(
|
||||||
// Get the status of the second delivery (index 1)
|
sourceChain,
|
||||||
const status = await getStatus(rx.transactionHash, undefined, 1);
|
rx.transactionHash,
|
||||||
expect(status).toBe("Delivery Success");
|
{ wormholeRelayerAddresses, ...optionalParams },
|
||||||
|
true,
|
||||||
|
{
|
||||||
|
newExecutionInfo: Buffer.from(
|
||||||
|
packEVMExecutionInfoV1({
|
||||||
|
gasLimit: ethers.BigNumber.from(REASONABLE_GAS_LIMIT),
|
||||||
|
targetChainRefundPerGasUnused:
|
||||||
|
ethers.BigNumber.from(refundPerGasUnused),
|
||||||
|
}).substring(2),
|
||||||
|
"hex"
|
||||||
|
),
|
||||||
|
newReceiverValue: ethers.BigNumber.from(0),
|
||||||
|
redeliveryHash: Buffer.from(
|
||||||
|
ethers.utils.keccak256("0x1234").substring(2),
|
||||||
|
"hex"
|
||||||
|
), // fake a redelivery
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
console.log("Checking if message was relayed");
|
console.log(`Price: ${priceInfo.quote} of ${priceInfo.targetChain} wei`);
|
||||||
const message = await target.mockIntegration.getMessage();
|
|
||||||
expect(message).toBe(arbitraryPayload);
|
const deliveryRx = await manualDelivery(
|
||||||
});
|
sourceChain,
|
||||||
|
rx.transactionHash,
|
||||||
|
{ wormholeRelayerAddresses, ...optionalParams },
|
||||||
|
false,
|
||||||
|
{
|
||||||
|
newExecutionInfo: Buffer.from(
|
||||||
|
packEVMExecutionInfoV1({
|
||||||
|
gasLimit: ethers.BigNumber.from(REASONABLE_GAS_LIMIT),
|
||||||
|
targetChainRefundPerGasUnused:
|
||||||
|
ethers.BigNumber.from(refundPerGasUnused),
|
||||||
|
}).substring(2),
|
||||||
|
"hex"
|
||||||
|
),
|
||||||
|
newReceiverValue: ethers.BigNumber.from(0),
|
||||||
|
redeliveryHash: Buffer.from(
|
||||||
|
ethers.utils.keccak256("0x1234").substring(2),
|
||||||
|
"hex"
|
||||||
|
), // fake a redelivery
|
||||||
|
},
|
||||||
|
target.wallet
|
||||||
|
);
|
||||||
|
console.log("Manual delivery tx hash", deliveryRx.txHash);
|
||||||
|
|
||||||
|
console.log("Checking if message was relayed");
|
||||||
|
const message = await target.mockIntegration.getMessage();
|
||||||
|
expect(message).toBe(arbitraryPayload);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
testIfDevnet()("Test getPrice in Typescript SDK", async () => {
|
testIfDevnet()("Test getPrice in Typescript SDK", async () => {
|
||||||
const price = await relayer.getPrice(
|
const price = await relayer.getPrice(
|
||||||
|
@ -360,10 +351,6 @@ describe("Wormhole Relayer Tests", () => {
|
||||||
|
|
||||||
await waitForRelay();
|
await waitForRelay();
|
||||||
|
|
||||||
console.log("Checking status using SDK");
|
|
||||||
const status = await getStatus(tx.hash);
|
|
||||||
expect(status).toBe("Receiver Failure");
|
|
||||||
|
|
||||||
const info = (await relayer.getWormholeRelayerInfo(sourceChain, tx.hash, {
|
const info = (await relayer.getWormholeRelayerInfo(sourceChain, tx.hash, {
|
||||||
wormholeRelayerAddresses,
|
wormholeRelayerAddresses,
|
||||||
...optionalParams,
|
...optionalParams,
|
||||||
|
@ -373,14 +360,6 @@ describe("Wormhole Relayer Tests", () => {
|
||||||
|
|
||||||
const newEndingBalance = await source.wallet.getBalance();
|
const newEndingBalance = await source.wallet.getBalance();
|
||||||
|
|
||||||
console.log("Checking status of refund using SDK");
|
|
||||||
console.log(relayer.stringifyWormholeRelayerInfo(info));
|
|
||||||
const statusOfRefund = await getStatus(
|
|
||||||
info.targetChainStatus.events[0].transactionHash || "",
|
|
||||||
targetChain
|
|
||||||
);
|
|
||||||
expect(statusOfRefund).toBe("Delivery Success");
|
|
||||||
|
|
||||||
console.log(`Quoted gas delivery fee: ${value}`);
|
console.log(`Quoted gas delivery fee: ${value}`);
|
||||||
console.log(
|
console.log(
|
||||||
`Cost (including gas) ${startingBalance.sub(endingBalance).toString()}`
|
`Cost (including gas) ${startingBalance.sub(endingBalance).toString()}`
|
||||||
|
@ -408,9 +387,6 @@ describe("Wormhole Relayer Tests", () => {
|
||||||
|
|
||||||
const message = await target.mockIntegration.getMessage();
|
const message = await target.mockIntegration.getMessage();
|
||||||
expect(message).not.toBe(arbitraryPayload);
|
expect(message).not.toBe(arbitraryPayload);
|
||||||
|
|
||||||
const status = await getStatus(rx.transactionHash);
|
|
||||||
expect(status).toBe("Receiver Failure");
|
|
||||||
});
|
});
|
||||||
|
|
||||||
test("Executes a receiver failure and then redelivery through SDK", async () => {
|
test("Executes a receiver failure and then redelivery through SDK", async () => {
|
||||||
|
@ -424,10 +400,6 @@ describe("Wormhole Relayer Tests", () => {
|
||||||
const message = await target.mockIntegration.getMessage();
|
const message = await target.mockIntegration.getMessage();
|
||||||
expect(message).not.toBe(arbitraryPayload);
|
expect(message).not.toBe(arbitraryPayload);
|
||||||
|
|
||||||
console.log("Checking status using SDK");
|
|
||||||
const status = await getStatus(rx.transactionHash);
|
|
||||||
expect(status).toBe("Receiver Failure");
|
|
||||||
|
|
||||||
const value = await relayer.getPrice(
|
const value = await relayer.getPrice(
|
||||||
sourceChain,
|
sourceChain,
|
||||||
targetChain,
|
targetChain,
|
||||||
|
@ -625,6 +597,68 @@ describe("Wormhole Relayer Tests", () => {
|
||||||
ethers.utils.getAddress((await getImplementationAddress()).substring(26))
|
ethers.utils.getAddress((await getImplementationAddress()).substring(26))
|
||||||
).toBe(ethers.utils.getAddress(newWormholeRelayerImplementationAddress));
|
).toBe(ethers.utils.getAddress(newWormholeRelayerImplementationAddress));
|
||||||
});
|
});
|
||||||
|
|
||||||
|
testIfNotDevnet()("Checks the status of a message", async () => {
|
||||||
|
const txHash =
|
||||||
|
"0xa75e4100240e9b498a48fa29de32c9e62ec241bf4071a3c93fde0df5de53c507";
|
||||||
|
const mySourceChain: ChainName = "celo";
|
||||||
|
const environment: Network = "TESTNET";
|
||||||
|
|
||||||
|
const info = await relayer.getWormholeRelayerInfo(mySourceChain, txHash, {
|
||||||
|
environment,
|
||||||
|
});
|
||||||
|
console.log(info.stringified);
|
||||||
|
});
|
||||||
|
|
||||||
|
testIfNotDevnet()("Tests custom manual delivery", async () => {
|
||||||
|
const txHash =
|
||||||
|
"0xc57d12cc789e4e9fa50d496cea62c2a0f11a7557c8adf42b3420e0585ba1f911";
|
||||||
|
const mySourceChain: ChainName = "arbitrum";
|
||||||
|
const targetProvider = undefined;
|
||||||
|
const environment: Network = "TESTNET";
|
||||||
|
|
||||||
|
const info = await relayer.getWormholeRelayerInfo(mySourceChain, txHash, {
|
||||||
|
environment,
|
||||||
|
});
|
||||||
|
console.log(info.stringified);
|
||||||
|
|
||||||
|
const priceInfo = await manualDelivery(
|
||||||
|
mySourceChain,
|
||||||
|
txHash,
|
||||||
|
{ environment },
|
||||||
|
true
|
||||||
|
);
|
||||||
|
console.log(`Price info: ${JSON.stringify(priceInfo)}`);
|
||||||
|
|
||||||
|
const signer = new ethers.Wallet(
|
||||||
|
PRIVATE_KEY,
|
||||||
|
targetProvider
|
||||||
|
? new ethers.providers.JsonRpcProvider(targetProvider)
|
||||||
|
: getDefaultProvider(environment, priceInfo.targetChain)
|
||||||
|
);
|
||||||
|
|
||||||
|
console.log(
|
||||||
|
`Price: ${ethers.utils.formatEther(priceInfo.quote)} of ${
|
||||||
|
priceInfo.targetChain
|
||||||
|
} currency`
|
||||||
|
);
|
||||||
|
const balance = await signer.getBalance();
|
||||||
|
console.log(
|
||||||
|
`My balance: ${ethers.utils.formatEther(balance)} of ${
|
||||||
|
priceInfo.targetChain
|
||||||
|
} currency`
|
||||||
|
);
|
||||||
|
|
||||||
|
const deliveryRx = await manualDelivery(
|
||||||
|
mySourceChain,
|
||||||
|
txHash,
|
||||||
|
{ environment },
|
||||||
|
false,
|
||||||
|
undefined,
|
||||||
|
signer
|
||||||
|
);
|
||||||
|
console.log("Manual delivery tx hash", deliveryRx.txHash);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
function sleep(ms: number): Promise<void> {
|
function sleep(ms: number): Promise<void> {
|
||||||
|
|
|
@ -12,6 +12,11 @@ type AddressInfo = {
|
||||||
};
|
};
|
||||||
|
|
||||||
const TESTNET: { [K in ChainName]?: AddressInfo } = {
|
const TESTNET: { [K in ChainName]?: AddressInfo } = {
|
||||||
|
ethereum: {
|
||||||
|
wormholeRelayerAddress: "0x28D8F1Be96f97C1387e94A53e00eCcFb4E75175a",
|
||||||
|
mockDeliveryProviderAddress: "0xD1463B4fe86166768d2ff51B1A928beBB5c9f375",
|
||||||
|
mockIntegrationAddress: "0xb81bc199b73AB34c393a4192C163252116a03370",
|
||||||
|
},
|
||||||
bsc: {
|
bsc: {
|
||||||
wormholeRelayerAddress: "0x80aC94316391752A193C1c47E27D382b507c93F3",
|
wormholeRelayerAddress: "0x80aC94316391752A193C1c47E27D382b507c93F3",
|
||||||
mockDeliveryProviderAddress: "0x60a86b97a7596eBFd25fb769053894ed0D9A8366",
|
mockDeliveryProviderAddress: "0x60a86b97a7596eBFd25fb769053894ed0D9A8366",
|
||||||
|
@ -37,6 +42,16 @@ const TESTNET: { [K in ChainName]?: AddressInfo } = {
|
||||||
mockDeliveryProviderAddress: "0x60a86b97a7596eBFd25fb769053894ed0D9A8366",
|
mockDeliveryProviderAddress: "0x60a86b97a7596eBFd25fb769053894ed0D9A8366",
|
||||||
mockIntegrationAddress: "0x3bF0c43d88541BBCF92bE508ec41e540FbF28C56",
|
mockIntegrationAddress: "0x3bF0c43d88541BBCF92bE508ec41e540FbF28C56",
|
||||||
},
|
},
|
||||||
|
arbitrum: {
|
||||||
|
wormholeRelayerAddress: "0xAd753479354283eEE1b86c9470c84D42f229FF43",
|
||||||
|
mockDeliveryProviderAddress: "0x90995DBd1aae85872451b50A569dE947D34ac4ee",
|
||||||
|
mockIntegrationAddress: "0x0de48f34E14d08934DA1eA2286Be1b2BED5c062a",
|
||||||
|
},
|
||||||
|
optimism: {
|
||||||
|
wormholeRelayerAddress: "0x01A957A525a5b7A72808bA9D10c389674E459891",
|
||||||
|
mockDeliveryProviderAddress: "0xfCe1Df3EF22fe5Cb7e2f5988b7d58fF633a313a7",
|
||||||
|
mockIntegrationAddress: "0x421e0bb71dDeeC727Af79766423d33D8FD7dB963",
|
||||||
|
},
|
||||||
base: {
|
base: {
|
||||||
wormholeRelayerAddress: "0xea8029CD7FCAEFFcD1F53686430Db0Fc8ed384E1",
|
wormholeRelayerAddress: "0xea8029CD7FCAEFFcD1F53686430Db0Fc8ed384E1",
|
||||||
mockDeliveryProviderAddress: "0x60a86b97a7596eBFd25fb769053894ed0D9A8366",
|
mockDeliveryProviderAddress: "0x60a86b97a7596eBFd25fb769053894ed0D9A8366",
|
||||||
|
@ -163,6 +178,7 @@ export const RPCS_BY_CHAIN: {
|
||||||
terra: "https://columbus-fcd.terra.dev",
|
terra: "https://columbus-fcd.terra.dev",
|
||||||
injective: "https://k8s.mainnet.lcd.injective.network",
|
injective: "https://k8s.mainnet.lcd.injective.network",
|
||||||
solana: "https://api.mainnet-beta.solana.com",
|
solana: "https://api.mainnet-beta.solana.com",
|
||||||
|
base: "https://mainnet.base.org",
|
||||||
},
|
},
|
||||||
TESTNET: {
|
TESTNET: {
|
||||||
solana: "https://api.devnet.solana.com",
|
solana: "https://api.devnet.solana.com",
|
||||||
|
@ -191,6 +207,7 @@ export const RPCS_BY_CHAIN: {
|
||||||
optimism: "https://goerli.optimism.io",
|
optimism: "https://goerli.optimism.io",
|
||||||
gnosis: "https://sokol.poa.network/",
|
gnosis: "https://sokol.poa.network/",
|
||||||
rootstock: "https://public-node.rsk.co",
|
rootstock: "https://public-node.rsk.co",
|
||||||
|
base: "https://goerli.base.org",
|
||||||
},
|
},
|
||||||
DEVNET: {
|
DEVNET: {
|
||||||
ethereum: "http://localhost:8545",
|
ethereum: "http://localhost:8545",
|
||||||
|
@ -205,3 +222,30 @@ export const GUARDIAN_RPC_HOSTS = [
|
||||||
"https://wormhole-v2-mainnet-api.chainlayer.network",
|
"https://wormhole-v2-mainnet-api.chainlayer.network",
|
||||||
"https://wormhole-v2-mainnet-api.staking.fund",
|
"https://wormhole-v2-mainnet-api.staking.fund",
|
||||||
];
|
];
|
||||||
|
|
||||||
|
export const getCircleAPI = (environment: Network) => {
|
||||||
|
return (environment === "TESTNET"
|
||||||
|
? "https://iris-api-sandbox.circle.com/v1/attestations/"
|
||||||
|
: "https://iris-api.circle.com/v1/attestations/");
|
||||||
|
}
|
||||||
|
|
||||||
|
export const getWormscanAPI = (_network: Network) => {
|
||||||
|
switch (_network) {
|
||||||
|
case "MAINNET":
|
||||||
|
return "https://api.wormscan.io/";
|
||||||
|
case "TESTNET":
|
||||||
|
return "https://api.testnet.wormscan.io/";
|
||||||
|
default:
|
||||||
|
// possible extension for tilt/ci - search through the guardian api
|
||||||
|
// at localhost:7071 (tilt) or guardian:7071 (ci)
|
||||||
|
throw new Error("Not testnet or mainnet - so no wormscan api access");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export const CCTP_DOMAIN_TO_NAME = [
|
||||||
|
"ethereum",
|
||||||
|
"avalanche",
|
||||||
|
"optimism",
|
||||||
|
"arbitrum",
|
||||||
|
"base"
|
||||||
|
];
|
|
@ -1,6 +1,12 @@
|
||||||
import { BigNumber, ethers, ContractReceipt } from "ethers";
|
import { BigNumber, ethers, ContractReceipt } from "ethers";
|
||||||
import { IWormholeRelayer__factory } from "../../ethers-contracts";
|
import { IWormholeRelayer__factory } from "../../ethers-contracts";
|
||||||
import { ChainName, toChainName, ChainId, Network } from "../../utils";
|
import {
|
||||||
|
ChainName,
|
||||||
|
toChainName,
|
||||||
|
ChainId,
|
||||||
|
Network,
|
||||||
|
CHAIN_ID_TO_NAME,
|
||||||
|
} from "../../utils";
|
||||||
import { SignedVaa, parseVaa } from "../../vaa";
|
import { SignedVaa, parseVaa } from "../../vaa";
|
||||||
import { getWormholeRelayerAddress } from "../consts";
|
import { getWormholeRelayerAddress } from "../consts";
|
||||||
import {
|
import {
|
||||||
|
@ -11,23 +17,56 @@ import {
|
||||||
parseEVMExecutionInfoV1,
|
parseEVMExecutionInfoV1,
|
||||||
parseWormholeRelayerPayloadType,
|
parseWormholeRelayerPayloadType,
|
||||||
parseWormholeRelayerSend,
|
parseWormholeRelayerSend,
|
||||||
VaaKey,
|
|
||||||
KeyType,
|
|
||||||
parseVaaKey,
|
parseVaaKey,
|
||||||
|
MessageKey,
|
||||||
|
parseCCTPKey,
|
||||||
} from "../structs";
|
} from "../structs";
|
||||||
import { DeliveryTargetInfo } from "./helpers";
|
import {
|
||||||
import { getSignedVAAWithRetry } from "../../rpc";
|
DeliveryTargetInfo,
|
||||||
|
getCCTPMessageLogURL,
|
||||||
|
getDefaultProvider,
|
||||||
|
getWormscanInfo,
|
||||||
|
} from "./helpers";
|
||||||
|
import { InfoRequestParams, getWormholeRelayerInfo } from "./info";
|
||||||
|
|
||||||
|
export type CCTPTransferParsed = {
|
||||||
|
amount: bigint; // decimals is 6
|
||||||
|
mintRecipient: string;
|
||||||
|
destinationDomain: number;
|
||||||
|
estimatedAttestationSeconds: number;
|
||||||
|
attested: boolean;
|
||||||
|
};
|
||||||
|
export type TokenTransferParsed = {
|
||||||
|
amount: bigint;
|
||||||
|
originAddress: string;
|
||||||
|
originChain: number;
|
||||||
|
targetAddress: string;
|
||||||
|
targetChain: number;
|
||||||
|
fromAddress: string | undefined;
|
||||||
|
name?: string;
|
||||||
|
symbol?: string;
|
||||||
|
decimals?: number;
|
||||||
|
signedVaaTimestamp?: number;
|
||||||
|
};
|
||||||
|
export type AdditionalMessageParsed =
|
||||||
|
| CCTPTransferParsed
|
||||||
|
| TokenTransferParsed
|
||||||
|
| undefined;
|
||||||
|
|
||||||
export type DeliveryInfo = {
|
export type DeliveryInfo = {
|
||||||
type: RelayerPayloadId.Delivery;
|
type: RelayerPayloadId.Delivery;
|
||||||
sourceChain: ChainName;
|
sourceChain: ChainName;
|
||||||
sourceTransactionHash: string;
|
sourceTransactionHash: string;
|
||||||
sourceDeliverySequenceNumber: number;
|
sourceDeliverySequenceNumber: number;
|
||||||
|
sourceTimestamp: number;
|
||||||
|
signingOfVaaTimestamp: number | undefined;
|
||||||
deliveryInstruction: DeliveryInstruction;
|
deliveryInstruction: DeliveryInstruction;
|
||||||
|
additionalMessageInformation: AdditionalMessageParsed[];
|
||||||
targetChainStatus: {
|
targetChainStatus: {
|
||||||
chain: ChainName;
|
chain: ChainName;
|
||||||
events: DeliveryTargetInfo[];
|
events: DeliveryTargetInfo[];
|
||||||
};
|
};
|
||||||
|
stringified?: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
export type DeliveryArguments = {
|
export type DeliveryArguments = {
|
||||||
|
@ -36,25 +75,81 @@ export type DeliveryArguments = {
|
||||||
deliveryHash: string;
|
deliveryHash: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export async function manualDelivery(
|
||||||
|
sourceChain: ChainName,
|
||||||
|
sourceTransaction: string,
|
||||||
|
infoRequest?: InfoRequestParams,
|
||||||
|
getQuoteOnly?: boolean,
|
||||||
|
overrides?: DeliveryOverrideArgs,
|
||||||
|
signer?: ethers.Signer
|
||||||
|
): Promise<{ quote: BigNumber; targetChain: ChainName; txHash?: string }> {
|
||||||
|
const info = await getWormholeRelayerInfo(
|
||||||
|
sourceChain,
|
||||||
|
sourceTransaction,
|
||||||
|
infoRequest
|
||||||
|
);
|
||||||
|
const environment = infoRequest?.environment || "MAINNET";
|
||||||
|
const sourceProvider =
|
||||||
|
infoRequest?.sourceChainProvider ||
|
||||||
|
getDefaultProvider(environment, sourceChain);
|
||||||
|
const receipt = await sourceProvider.getTransactionReceipt(sourceTransaction);
|
||||||
|
const wormholeRelayerAddress =
|
||||||
|
infoRequest?.wormholeRelayerAddresses?.get(sourceChain) ||
|
||||||
|
getWormholeRelayerAddress(sourceChain, environment);
|
||||||
|
const response = await (
|
||||||
|
await getWormscanInfo(
|
||||||
|
environment,
|
||||||
|
info.sourceChain,
|
||||||
|
info.sourceDeliverySequenceNumber,
|
||||||
|
wormholeRelayerAddress
|
||||||
|
)
|
||||||
|
).json();
|
||||||
|
|
||||||
|
const signedVaa = response.data.vaa;
|
||||||
|
const signedVaaBuffer = Buffer.from(signedVaa, "base64");
|
||||||
|
const result: { quote: BigNumber; targetChain: ChainName; txHash?: string } =
|
||||||
|
{
|
||||||
|
quote: deliveryBudget(info.deliveryInstruction, overrides),
|
||||||
|
targetChain:
|
||||||
|
CHAIN_ID_TO_NAME[info.deliveryInstruction.targetChainId as ChainId],
|
||||||
|
txHash: undefined,
|
||||||
|
};
|
||||||
|
if (getQuoteOnly) {
|
||||||
|
return result;
|
||||||
|
} else {
|
||||||
|
if (!signer) {
|
||||||
|
throw new Error("no signer provided");
|
||||||
|
}
|
||||||
|
const deliveryReceipt = await deliver(
|
||||||
|
signedVaaBuffer,
|
||||||
|
signer,
|
||||||
|
environment,
|
||||||
|
overrides,
|
||||||
|
sourceChain,
|
||||||
|
receipt
|
||||||
|
);
|
||||||
|
result.txHash = deliveryReceipt.transactionHash;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export async function deliver(
|
export async function deliver(
|
||||||
deliveryVaa: SignedVaa,
|
deliveryVaa: SignedVaa,
|
||||||
signer: ethers.Signer,
|
signer: ethers.Signer,
|
||||||
wormholeRPCs: string | string[],
|
|
||||||
environment: Network = "MAINNET",
|
environment: Network = "MAINNET",
|
||||||
overrides?: DeliveryOverrideArgs
|
overrides?: DeliveryOverrideArgs,
|
||||||
|
sourceChain?: ChainName,
|
||||||
|
sourceReceipt?: ethers.providers.TransactionReceipt
|
||||||
): Promise<ContractReceipt> {
|
): Promise<ContractReceipt> {
|
||||||
const { budget, deliveryInstruction, deliveryHash } =
|
const { budget, deliveryInstruction, deliveryHash } =
|
||||||
extractDeliveryArguments(deliveryVaa, overrides);
|
extractDeliveryArguments(deliveryVaa, overrides);
|
||||||
|
|
||||||
const vaaKeys = deliveryInstruction.messageKeys.map((key) => {
|
const additionalMessages = await fetchAdditionalMessages(
|
||||||
if (key.keyType !== KeyType.VAA) {
|
deliveryInstruction.messageKeys,
|
||||||
throw new Error(
|
environment,
|
||||||
"Only VAA keys are supported by manual delivery. Found: " + key.keyType
|
sourceChain,
|
||||||
);
|
sourceReceipt
|
||||||
}
|
);
|
||||||
return parseVaaKey(key.key);
|
|
||||||
});
|
|
||||||
const additionalVaas = await fetchAdditionalVaas(wormholeRPCs, vaaKeys);
|
|
||||||
|
|
||||||
const wormholeRelayerAddress = getWormholeRelayerAddress(
|
const wormholeRelayerAddress = getWormholeRelayerAddress(
|
||||||
toChainName(deliveryInstruction.targetChainId as ChainId),
|
toChainName(deliveryInstruction.targetChainId as ChainId),
|
||||||
|
@ -65,21 +160,20 @@ export async function deliver(
|
||||||
signer
|
signer
|
||||||
);
|
);
|
||||||
const gasEstimate = await wormholeRelayer.estimateGas.deliver(
|
const gasEstimate = await wormholeRelayer.estimateGas.deliver(
|
||||||
additionalVaas,
|
additionalMessages,
|
||||||
deliveryVaa,
|
deliveryVaa,
|
||||||
signer.getAddress(),
|
signer.getAddress(),
|
||||||
overrides ? packOverrides(overrides) : new Uint8Array(),
|
overrides ? packOverrides(overrides) : new Uint8Array(),
|
||||||
{ value: budget }
|
{ value: budget }
|
||||||
);
|
);
|
||||||
const tx = await wormholeRelayer.deliver(
|
const tx = await wormholeRelayer.deliver(
|
||||||
additionalVaas,
|
additionalMessages,
|
||||||
deliveryVaa,
|
deliveryVaa,
|
||||||
signer.getAddress(),
|
signer.getAddress(),
|
||||||
overrides ? packOverrides(overrides) : new Uint8Array(),
|
overrides ? packOverrides(overrides) : new Uint8Array(),
|
||||||
{ value: budget, gasLimit: gasEstimate.mul(2) }
|
{ value: budget, gasLimit: gasEstimate.mul(2) }
|
||||||
);
|
);
|
||||||
const rx = await tx.wait();
|
const rx = await tx.wait();
|
||||||
console.log(`Delivered ${deliveryHash} on ${rx.blockNumber}`);
|
|
||||||
return rx;
|
return rx;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -125,20 +219,79 @@ export function extractDeliveryArguments(
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function fetchAdditionalVaas(
|
export async function fetchAdditionalMessages(
|
||||||
wormholeRPCs: string | string[],
|
additionalMessageKeys: MessageKey[],
|
||||||
additionalVaaKeys: VaaKey[]
|
environment: Network,
|
||||||
): Promise<SignedVaa[]> {
|
sourceChain?: ChainName,
|
||||||
const rpcs = typeof wormholeRPCs === "string" ? [wormholeRPCs] : wormholeRPCs;
|
sourceReceipt?: ethers.providers.TransactionReceipt
|
||||||
const vaas = await Promise.all(
|
): Promise<(Uint8Array | Buffer)[]> {
|
||||||
additionalVaaKeys.map(async (vaaKey) =>
|
const messages = await Promise.all(
|
||||||
getSignedVAAWithRetry(
|
additionalMessageKeys.map(async (messageKey) => {
|
||||||
rpcs,
|
if (messageKey.keyType === 1) {
|
||||||
vaaKey.chainId as ChainId,
|
const vaaKey = parseVaaKey(messageKey.key);
|
||||||
vaaKey.emitterAddress.toString("hex"),
|
const signedVaa = (
|
||||||
vaaKey.sequence.toBigInt().toString()
|
await await (
|
||||||
)
|
await getWormscanInfo(
|
||||||
)
|
environment,
|
||||||
|
CHAIN_ID_TO_NAME[vaaKey.chainId as ChainId],
|
||||||
|
vaaKey.sequence.toNumber(),
|
||||||
|
"0x" + vaaKey.emitterAddress.toString("hex")
|
||||||
|
)
|
||||||
|
).json()
|
||||||
|
).data?.vaa;
|
||||||
|
if (!signedVaa) {
|
||||||
|
throw new Error(
|
||||||
|
`No signed VAA available on WormScan for vaaKey ${JSON.stringify(
|
||||||
|
vaaKey
|
||||||
|
)}`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return Buffer.from(signedVaa, "base64");
|
||||||
|
} else if (messageKey.keyType === 2) {
|
||||||
|
const cctpKey = parseCCTPKey(messageKey.key);
|
||||||
|
if (!sourceReceipt)
|
||||||
|
throw new Error(
|
||||||
|
"No source receipt provided - needed to obtain CCTP message"
|
||||||
|
);
|
||||||
|
if (!environment)
|
||||||
|
throw new Error(
|
||||||
|
"No environment provided - needed to obtain CCTP message"
|
||||||
|
);
|
||||||
|
if (!sourceChain)
|
||||||
|
throw new Error(
|
||||||
|
"No source chain provided - needed to obtain CCTP message"
|
||||||
|
);
|
||||||
|
|
||||||
|
const response = await getCCTPMessageLogURL(
|
||||||
|
cctpKey,
|
||||||
|
sourceChain,
|
||||||
|
sourceReceipt,
|
||||||
|
environment
|
||||||
|
);
|
||||||
|
|
||||||
|
// Try to get attestation
|
||||||
|
const attestationResponse = await fetch(response?.url || "");
|
||||||
|
const attestationResponseJson = await attestationResponse.json();
|
||||||
|
const attestation = attestationResponseJson.attestation;
|
||||||
|
if (!attestation) {
|
||||||
|
throw new Error(
|
||||||
|
`Unable to get attestation from Circle, for cctp key ${JSON.stringify(
|
||||||
|
cctpKey
|
||||||
|
)}, message ${response?.message}`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return Buffer.from(
|
||||||
|
new ethers.utils.AbiCoder()
|
||||||
|
.encode(["bytes", "bytes"], [response?.message || [], attestation])
|
||||||
|
.substring(2),
|
||||||
|
"hex"
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
throw new Error(
|
||||||
|
`Message key type unknown: ${messageKey.keyType} (messageKey ${messageKey.key})`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
})
|
||||||
);
|
);
|
||||||
return vaas.map((vaa) => vaa.vaaBytes);
|
return messages;
|
||||||
}
|
}
|
||||||
|
|
|
@ -13,6 +13,10 @@ import {
|
||||||
getWormholeRelayer,
|
getWormholeRelayer,
|
||||||
RPCS_BY_CHAIN,
|
RPCS_BY_CHAIN,
|
||||||
RELAYER_CONTRACTS,
|
RELAYER_CONTRACTS,
|
||||||
|
getWormholeRelayerAddress,
|
||||||
|
getCircleAPI,
|
||||||
|
getWormscanAPI,
|
||||||
|
CCTP_DOMAIN_TO_NAME
|
||||||
} from "../consts";
|
} from "../consts";
|
||||||
import {
|
import {
|
||||||
parseWormholeRelayerPayloadType,
|
parseWormholeRelayerPayloadType,
|
||||||
|
@ -25,12 +29,15 @@ import {
|
||||||
VaaKey,
|
VaaKey,
|
||||||
DeliveryOverrideArgs,
|
DeliveryOverrideArgs,
|
||||||
parseRefundStatus,
|
parseRefundStatus,
|
||||||
|
RedeliveryInstruction,
|
||||||
|
parseWormholeRelayerResend,
|
||||||
|
CCTPKey,
|
||||||
} from "../structs";
|
} from "../structs";
|
||||||
|
import { InfoRequestParams } from "./info";
|
||||||
import {
|
import {
|
||||||
DeliveryProvider,
|
DeliveryProvider,
|
||||||
DeliveryProvider__factory,
|
DeliveryProvider__factory,
|
||||||
Implementation__factory,
|
Implementation__factory,
|
||||||
IWormholeRelayerDelivery__factory,
|
|
||||||
} from "../../ethers-contracts/";
|
} from "../../ethers-contracts/";
|
||||||
import { DeliveryEvent } from "../../ethers-contracts/WormholeRelayer";
|
import { DeliveryEvent } from "../../ethers-contracts/WormholeRelayer";
|
||||||
import { VaaKeyStruct } from "../../ethers-contracts/IWormholeRelayer.sol/IWormholeRelayer";
|
import { VaaKeyStruct } from "../../ethers-contracts/IWormholeRelayer.sol/IWormholeRelayer";
|
||||||
|
@ -39,17 +46,18 @@ export type DeliveryTargetInfo = {
|
||||||
status: DeliveryStatus | string;
|
status: DeliveryStatus | string;
|
||||||
transactionHash: string | null;
|
transactionHash: string | null;
|
||||||
vaaHash: string | null;
|
vaaHash: string | null;
|
||||||
sourceChain: ChainName;
|
sourceChain: ChainName | null;
|
||||||
sourceVaaSequence: BigNumber | null;
|
sourceVaaSequence: BigNumber | null;
|
||||||
gasUsed: BigNumber;
|
gasUsed: BigNumber;
|
||||||
refundStatus: RefundStatus;
|
refundStatus: RefundStatus;
|
||||||
|
timestamp?: number;
|
||||||
revertString?: string; // Only defined if status is RECEIVER_FAILURE
|
revertString?: string; // Only defined if status is RECEIVER_FAILURE
|
||||||
overrides?: DeliveryOverrideArgs;
|
overrides?: DeliveryOverrideArgs;
|
||||||
};
|
};
|
||||||
|
|
||||||
export function parseWormholeLog(log: ethers.providers.Log): {
|
export function parseWormholeLog(log: ethers.providers.Log): {
|
||||||
type: RelayerPayloadId;
|
type: RelayerPayloadId;
|
||||||
parsed: DeliveryInstruction | string;
|
parsed: DeliveryInstruction | RedeliveryInstruction | string;
|
||||||
} {
|
} {
|
||||||
const abi = [
|
const abi = [
|
||||||
"event LogMessagePublished(address indexed sender, uint64 sequence, uint32 nonce, bytes payload, uint8 consistencyLevel)",
|
"event LogMessagePublished(address indexed sender, uint64 sequence, uint32 nonce, bytes payload, uint8 consistencyLevel)",
|
||||||
|
@ -60,6 +68,8 @@ export function parseWormholeLog(log: ethers.providers.Log): {
|
||||||
const type = parseWormholeRelayerPayloadType(payload);
|
const type = parseWormholeRelayerPayloadType(payload);
|
||||||
if (type == RelayerPayloadId.Delivery) {
|
if (type == RelayerPayloadId.Delivery) {
|
||||||
return { type, parsed: parseWormholeRelayerSend(payload) };
|
return { type, parsed: parseWormholeRelayerSend(payload) };
|
||||||
|
} else if (type == RelayerPayloadId.Redelivery) {
|
||||||
|
return { type, parsed: parseWormholeRelayerResend(payload) };
|
||||||
} else {
|
} else {
|
||||||
throw Error("Invalid wormhole log");
|
throw Error("Invalid wormhole log");
|
||||||
}
|
}
|
||||||
|
@ -71,6 +81,21 @@ export function printChain(chainId: number) {
|
||||||
return `${CHAIN_ID_TO_NAME[chainId as ChainId]} (Chain ${chainId})`;
|
return `${CHAIN_ID_TO_NAME[chainId as ChainId]} (Chain ${chainId})`;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function printCCTPDomain(domain: number) {
|
||||||
|
if (domain >= CCTP_DOMAIN_TO_NAME.length)
|
||||||
|
throw Error(`Invalid cctp domain: ${domain}`);
|
||||||
|
return `${CCTP_DOMAIN_TO_NAME[domain]} (Domain ${domain})`;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const estimatedAttestationTimeInSeconds = (
|
||||||
|
sourceChain: string,
|
||||||
|
environment: Network
|
||||||
|
): number => {
|
||||||
|
const testnetTime = sourceChain === "avalanche" ? 20 : 60;
|
||||||
|
const mainnetTime = sourceChain === "avalanche" ? 20 : 60 * 13;
|
||||||
|
return environment === "TESTNET" ? testnetTime : mainnetTime;
|
||||||
|
};
|
||||||
|
|
||||||
export function getDefaultProvider(
|
export function getDefaultProvider(
|
||||||
network: Network,
|
network: Network,
|
||||||
chain: ChainName,
|
chain: ChainName,
|
||||||
|
@ -98,74 +123,47 @@ export function getDeliveryProvider(
|
||||||
return contract;
|
return contract;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getBlockRange(
|
|
||||||
provider: ethers.providers.Provider,
|
|
||||||
timestamp?: number
|
|
||||||
): [ethers.providers.BlockTag, ethers.providers.BlockTag] {
|
|
||||||
return [-2040, "latest"];
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function getWormholeRelayerInfoBySourceSequence(
|
export async function getWormholeRelayerInfoBySourceSequence(
|
||||||
environment: Network,
|
environment: Network,
|
||||||
targetChain: ChainName,
|
targetChain: ChainName,
|
||||||
targetChainProvider: ethers.providers.Provider,
|
targetChainProvider: ethers.providers.Provider,
|
||||||
sourceChain: ChainName,
|
sourceChain: ChainName | undefined,
|
||||||
sourceVaaSequence: BigNumber,
|
sourceVaaSequence: BigNumber | undefined,
|
||||||
blockStartNumber: ethers.providers.BlockTag,
|
blockRange:
|
||||||
blockEndNumber: ethers.providers.BlockTag,
|
| [ethers.providers.BlockTag, ethers.providers.BlockTag]
|
||||||
|
| undefined,
|
||||||
targetWormholeRelayerAddress: string
|
targetWormholeRelayerAddress: string
|
||||||
): Promise<{ chain: ChainName; events: DeliveryTargetInfo[] }> {
|
): Promise<DeliveryTargetInfo[]> {
|
||||||
const deliveryEvents = await getWormholeRelayerDeliveryEventsBySourceSequence(
|
const deliveryEvents = await getWormholeRelayerDeliveryEventsBySourceSequence(
|
||||||
environment,
|
environment,
|
||||||
targetChain,
|
targetChain,
|
||||||
targetChainProvider,
|
targetChainProvider,
|
||||||
sourceChain,
|
sourceChain,
|
||||||
sourceVaaSequence,
|
sourceVaaSequence,
|
||||||
blockStartNumber,
|
blockRange,
|
||||||
blockEndNumber,
|
|
||||||
targetWormholeRelayerAddress
|
targetWormholeRelayerAddress
|
||||||
);
|
);
|
||||||
if (deliveryEvents.length == 0) {
|
|
||||||
let status = `Delivery didn't happen on ${targetChain} within blocks ${blockStartNumber} to ${blockEndNumber}.`;
|
|
||||||
try {
|
|
||||||
const blockStart = await targetChainProvider.getBlock(blockStartNumber);
|
|
||||||
const blockEnd = await targetChainProvider.getBlock(blockEndNumber);
|
|
||||||
status = `Delivery didn't happen on ${targetChain} within blocks ${
|
|
||||||
blockStart.number
|
|
||||||
} to ${blockEnd.number} (within times ${new Date(
|
|
||||||
blockStart.timestamp * 1000
|
|
||||||
).toString()} to ${new Date(blockEnd.timestamp * 1000).toString()})`;
|
|
||||||
} catch (e) {}
|
|
||||||
deliveryEvents.push({
|
|
||||||
status,
|
|
||||||
transactionHash: null,
|
|
||||||
vaaHash: null,
|
|
||||||
sourceChain: sourceChain,
|
|
||||||
sourceVaaSequence,
|
|
||||||
gasUsed: BigNumber.from(0),
|
|
||||||
refundStatus: RefundStatus.RefundFail,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
const targetChainStatus = {
|
|
||||||
chain: targetChain,
|
|
||||||
events: deliveryEvents,
|
|
||||||
};
|
|
||||||
|
|
||||||
return targetChainStatus;
|
return deliveryEvents;
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function getWormholeRelayerDeliveryEventsBySourceSequence(
|
export async function getWormholeRelayerDeliveryEventsBySourceSequence(
|
||||||
environment: Network,
|
environment: Network,
|
||||||
targetChain: ChainName,
|
targetChain: ChainName,
|
||||||
targetChainProvider: ethers.providers.Provider,
|
targetChainProvider: ethers.providers.Provider,
|
||||||
sourceChain: ChainName,
|
sourceChain: ChainName | undefined,
|
||||||
sourceVaaSequence: BigNumber,
|
sourceVaaSequence: BigNumber | undefined,
|
||||||
blockStartNumber: ethers.providers.BlockTag,
|
blockRange:
|
||||||
blockEndNumber: ethers.providers.BlockTag,
|
| [ethers.providers.BlockTag, ethers.providers.BlockTag]
|
||||||
|
| undefined,
|
||||||
targetWormholeRelayerAddress: string
|
targetWormholeRelayerAddress: string
|
||||||
): Promise<DeliveryTargetInfo[]> {
|
): Promise<DeliveryTargetInfo[]> {
|
||||||
const sourceChainId = CHAINS[sourceChain];
|
let sourceChainId = undefined;
|
||||||
if (!sourceChainId) throw Error(`Invalid source chain: ${sourceChain}`);
|
if (sourceChain) {
|
||||||
|
sourceChainId = CHAINS[sourceChain];
|
||||||
|
if (!sourceChainId) throw Error(`Invalid source chain: ${sourceChain}`);
|
||||||
|
}
|
||||||
|
|
||||||
const wormholeRelayer = getWormholeRelayer(
|
const wormholeRelayer = getWormholeRelayer(
|
||||||
targetChain,
|
targetChain,
|
||||||
environment,
|
environment,
|
||||||
|
@ -173,71 +171,27 @@ export async function getWormholeRelayerDeliveryEventsBySourceSequence(
|
||||||
targetWormholeRelayerAddress
|
targetWormholeRelayerAddress
|
||||||
);
|
);
|
||||||
|
|
||||||
const deliveryEvents = wormholeRelayer.filters.Delivery(
|
const deliveryEventsFilter = wormholeRelayer.filters.Delivery(
|
||||||
null,
|
null,
|
||||||
sourceChainId,
|
sourceChainId,
|
||||||
sourceVaaSequence
|
sourceVaaSequence
|
||||||
);
|
);
|
||||||
|
|
||||||
const deliveryEventsPreFilter: DeliveryEvent[] =
|
const deliveryEvents: DeliveryEvent[] = await wormholeRelayer.queryFilter(
|
||||||
await wormholeRelayer.queryFilter(
|
deliveryEventsFilter,
|
||||||
deliveryEvents,
|
blockRange ? blockRange[0] : -2000,
|
||||||
blockStartNumber,
|
blockRange ? blockRange[1] : "latest"
|
||||||
blockEndNumber
|
);
|
||||||
);
|
|
||||||
|
|
||||||
const isValid: boolean[] = await Promise.all(
|
const timestamps = await Promise.all(
|
||||||
deliveryEventsPreFilter.map((deliveryEvent) =>
|
deliveryEvents.map(
|
||||||
areSignaturesValid(
|
async (e) =>
|
||||||
deliveryEvent.getTransaction(),
|
(await targetChainProvider.getBlock(e.blockNumber)).timestamp * 1000
|
||||||
targetChain,
|
|
||||||
targetChainProvider,
|
|
||||||
environment
|
|
||||||
)
|
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
|
||||||
// There is a max limit on RPCs sometimes for how many blocks to query
|
// There is a max limit on RPCs sometimes for how many blocks to query
|
||||||
return await transformDeliveryEvents(
|
return await transformDeliveryEvents(deliveryEvents, timestamps);
|
||||||
deliveryEventsPreFilter.filter((deliveryEvent, i) => isValid[i])
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
async function areSignaturesValid(
|
|
||||||
transaction: Promise<ethers.Transaction>,
|
|
||||||
targetChain: ChainName,
|
|
||||||
targetChainProvider: ethers.providers.Provider,
|
|
||||||
environment: Network
|
|
||||||
) {
|
|
||||||
const coreAddress = CONTRACTS[environment][targetChain].core;
|
|
||||||
if (!coreAddress)
|
|
||||||
throw Error(
|
|
||||||
`No Wormhole Address for chain ${targetChain}, network ${environment}`
|
|
||||||
);
|
|
||||||
|
|
||||||
const wormhole = Implementation__factory.connect(
|
|
||||||
coreAddress,
|
|
||||||
targetChainProvider
|
|
||||||
);
|
|
||||||
const decodedData =
|
|
||||||
IWormholeRelayerDelivery__factory.createInterface().parseTransaction(
|
|
||||||
await transaction
|
|
||||||
);
|
|
||||||
|
|
||||||
const vaaIsValid = async (vaa: ethers.utils.BytesLike): Promise<boolean> => {
|
|
||||||
const [, result, reason] = await wormhole.parseAndVerifyVM(vaa);
|
|
||||||
if (!result) console.log(`Invalid vaa! Reason: ${reason}`);
|
|
||||||
return result;
|
|
||||||
};
|
|
||||||
|
|
||||||
const vaas = decodedData.args[0];
|
|
||||||
for (let i = 0; i < vaas.length; i++) {
|
|
||||||
if (!(await vaaIsValid(vaas[i]))) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export function deliveryStatus(status: number) {
|
export function deliveryStatus(status: number) {
|
||||||
|
@ -251,20 +205,23 @@ export function deliveryStatus(status: number) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export function transformDeliveryLog(log: {
|
export function transformDeliveryLog(
|
||||||
args: [
|
log: {
|
||||||
string,
|
args: [
|
||||||
number,
|
string,
|
||||||
BigNumber,
|
number,
|
||||||
string,
|
BigNumber,
|
||||||
number,
|
string,
|
||||||
BigNumber,
|
number,
|
||||||
number,
|
BigNumber,
|
||||||
string,
|
number,
|
||||||
string
|
string,
|
||||||
];
|
string
|
||||||
transactionHash: string;
|
];
|
||||||
}): DeliveryTargetInfo {
|
transactionHash: string;
|
||||||
|
},
|
||||||
|
timestamp: number
|
||||||
|
): DeliveryTargetInfo {
|
||||||
const status = deliveryStatus(log.args[4]);
|
const status = deliveryStatus(log.args[4]);
|
||||||
if (!isChain(log.args[1]))
|
if (!isChain(log.args[1]))
|
||||||
throw Error(`Invalid source chain id: ${log.args[1]}`);
|
throw Error(`Invalid source chain id: ${log.args[1]}`);
|
||||||
|
@ -279,6 +236,7 @@ export function transformDeliveryLog(log: {
|
||||||
refundStatus: parseRefundStatus(log.args[6]),
|
refundStatus: parseRefundStatus(log.args[6]),
|
||||||
revertString:
|
revertString:
|
||||||
status == DeliveryStatus.ReceiverFailure ? log.args[7] : undefined,
|
status == DeliveryStatus.ReceiverFailure ? log.args[7] : undefined,
|
||||||
|
timestamp,
|
||||||
overrides:
|
overrides:
|
||||||
Buffer.from(log.args[8].substring(2), "hex").length > 0
|
Buffer.from(log.args[8].substring(2), "hex").length > 0
|
||||||
? parseOverrideInfoFromDeliveryEvent(
|
? parseOverrideInfoFromDeliveryEvent(
|
||||||
|
@ -289,17 +247,19 @@ export function transformDeliveryLog(log: {
|
||||||
}
|
}
|
||||||
|
|
||||||
async function transformDeliveryEvents(
|
async function transformDeliveryEvents(
|
||||||
events: DeliveryEvent[]
|
events: DeliveryEvent[],
|
||||||
|
timestamps: number[]
|
||||||
): Promise<DeliveryTargetInfo[]> {
|
): Promise<DeliveryTargetInfo[]> {
|
||||||
return events.map((x) => transformDeliveryLog(x));
|
return events.map((x, i) => transformDeliveryLog(x, timestamps[i]));
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getWormholeRelayerLog(
|
export function getWormholeLog(
|
||||||
receipt: ContractReceipt,
|
receipt: ContractReceipt,
|
||||||
bridgeAddress: string,
|
bridgeAddress: string,
|
||||||
emitterAddress: string,
|
emitterAddress: string,
|
||||||
index: number
|
index: number,
|
||||||
): { log: ethers.providers.Log; sequence: string } {
|
sequence?: number
|
||||||
|
): { log: ethers.providers.Log; sequence: string; payload: string } {
|
||||||
const bridgeLogs = receipt.logs.filter((l) => {
|
const bridgeLogs = receipt.logs.filter((l) => {
|
||||||
return l.address === bridgeAddress;
|
return l.address === bridgeAddress;
|
||||||
});
|
});
|
||||||
|
@ -314,17 +274,23 @@ export function getWormholeRelayerLog(
|
||||||
sequence: log.args[1].toString(),
|
sequence: log.args[1].toString(),
|
||||||
nonce: log.args[2].toString(),
|
nonce: log.args[2].toString(),
|
||||||
emitterAddress: tryNativeToHexString(log.args[0].toString(), "ethereum"),
|
emitterAddress: tryNativeToHexString(log.args[0].toString(), "ethereum"),
|
||||||
|
payload: log.args[3],
|
||||||
log: bridgeLog,
|
log: bridgeLog,
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
const filtered = parsed.filter(
|
const filtered = parsed.filter((x) => {
|
||||||
(x) => x.emitterAddress == emitterAddress.toLowerCase()
|
return (
|
||||||
);
|
x.emitterAddress == emitterAddress.toLowerCase() &&
|
||||||
|
(sequence === undefined ? true : x.sequence + "" === sequence + "")
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
if (filtered.length == 0) {
|
if (filtered.length == 0) {
|
||||||
throw Error(
|
throw Error(
|
||||||
"No WormholeRelayer contract interactions found for this transaction."
|
`No wormhole contract interactions found for this transaction, with emitter address ${emitterAddress} ${
|
||||||
|
sequence === undefined ? "" : `and sequence ${sequence}`
|
||||||
|
}`
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -334,6 +300,7 @@ export function getWormholeRelayerLog(
|
||||||
return {
|
return {
|
||||||
log: filtered[index].log,
|
log: filtered[index].log,
|
||||||
sequence: filtered[index].sequence,
|
sequence: filtered[index].sequence,
|
||||||
|
payload: filtered[index].payload,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -348,6 +315,145 @@ export function vaaKeyToVaaKeyStruct(vaaKey: VaaKey): VaaKeyStruct {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export async function getWormholeRelayerInfoByHash(
|
||||||
|
deliveryHash: string,
|
||||||
|
targetChain: ChainName,
|
||||||
|
sourceChain: ChainName | undefined,
|
||||||
|
sourceVaaSequence: number | undefined,
|
||||||
|
infoRequest?: InfoRequestParams
|
||||||
|
): Promise<DeliveryTargetInfo[]> {
|
||||||
|
const environment = infoRequest?.environment || "MAINNET";
|
||||||
|
const targetChainProvider =
|
||||||
|
infoRequest?.targetChainProviders?.get(targetChain) ||
|
||||||
|
getDefaultProvider(environment, targetChain);
|
||||||
|
|
||||||
|
if (!targetChainProvider) {
|
||||||
|
throw Error(
|
||||||
|
"No default RPC for this chain; pass in your own provider (as targetChainProvider)"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
const targetWormholeRelayerAddress =
|
||||||
|
infoRequest?.wormholeRelayerAddresses?.get(targetChain) ||
|
||||||
|
getWormholeRelayerAddress(targetChain, environment);
|
||||||
|
const wormholeRelayer = getWormholeRelayer(
|
||||||
|
targetChain,
|
||||||
|
environment,
|
||||||
|
targetChainProvider,
|
||||||
|
targetWormholeRelayerAddress
|
||||||
|
);
|
||||||
|
|
||||||
|
const blockNumberSuccess = await wormholeRelayer.deliverySuccessBlock(
|
||||||
|
deliveryHash
|
||||||
|
);
|
||||||
|
const blockNumberFailure = await wormholeRelayer.deliveryFailureBlock(
|
||||||
|
deliveryHash
|
||||||
|
);
|
||||||
|
const blockNumber = blockNumberSuccess.gt(0)
|
||||||
|
? blockNumberSuccess
|
||||||
|
: blockNumberFailure;
|
||||||
|
|
||||||
|
if (blockNumber.toNumber() === 0) return [];
|
||||||
|
|
||||||
|
// There is weirdness with arbitrum where if you call 'block.number', it gives you the L1 block number (the ethereum one) - and this is what is stored in the 'replay protection mapping' - so basically that value isn't useful in finding the delivery here
|
||||||
|
const blockRange =
|
||||||
|
infoRequest?.targetBlockRange ||
|
||||||
|
(targetChain === "arbitrum"
|
||||||
|
? undefined
|
||||||
|
: [blockNumber.toNumber(), blockNumber.toNumber()]);
|
||||||
|
|
||||||
|
return await getWormholeRelayerInfoBySourceSequence(
|
||||||
|
environment,
|
||||||
|
targetChain,
|
||||||
|
targetChainProvider,
|
||||||
|
sourceChain,
|
||||||
|
BigNumber.from(sourceVaaSequence),
|
||||||
|
blockRange,
|
||||||
|
targetWormholeRelayerAddress
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getDeliveryHashFromVaaFields(
|
||||||
|
sourceChain: number,
|
||||||
|
emitterAddress: string,
|
||||||
|
sequence: number,
|
||||||
|
timestamp: number,
|
||||||
|
nonce: number,
|
||||||
|
consistencyLevel: number,
|
||||||
|
deliveryVaaPayload: string
|
||||||
|
): string {
|
||||||
|
const body = ethers.utils.solidityPack(
|
||||||
|
["uint32", "uint32", "uint16", "bytes32", "uint64", "uint8", "bytes"],
|
||||||
|
|
||||||
|
[
|
||||||
|
timestamp,
|
||||||
|
nonce,
|
||||||
|
sourceChain,
|
||||||
|
emitterAddress,
|
||||||
|
sequence,
|
||||||
|
consistencyLevel,
|
||||||
|
deliveryVaaPayload,
|
||||||
|
]
|
||||||
|
);
|
||||||
|
const deliveryHash = ethers.utils.keccak256(ethers.utils.keccak256(body));
|
||||||
|
return deliveryHash;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getWormscanInfo(
|
||||||
|
network: Network,
|
||||||
|
sourceChain: ChainName,
|
||||||
|
sequence: number,
|
||||||
|
emitterAddress: string
|
||||||
|
) {
|
||||||
|
const wormscanAPI = getWormscanAPI(network);
|
||||||
|
const emitterAddressBytes32 = tryNativeToHexString(
|
||||||
|
emitterAddress,
|
||||||
|
sourceChain
|
||||||
|
);
|
||||||
|
const sourceChainId = CHAINS[sourceChain];
|
||||||
|
const result = await fetch(
|
||||||
|
`${wormscanAPI}api/v1/vaas/${sourceChainId}/${emitterAddressBytes32}/${sequence}`
|
||||||
|
);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getWormscanRelayerInfo(
|
||||||
|
sourceChain: ChainName,
|
||||||
|
sequence: number,
|
||||||
|
optionalParams?: {
|
||||||
|
network?: Network;
|
||||||
|
provider?: ethers.providers.Provider;
|
||||||
|
wormholeRelayerAddress?: string;
|
||||||
|
}
|
||||||
|
): Promise<Response> {
|
||||||
|
const network = optionalParams?.network || "MAINNET";
|
||||||
|
const wormholeRelayerAddress =
|
||||||
|
optionalParams?.wormholeRelayerAddress ||
|
||||||
|
getWormholeRelayerAddress(sourceChain, network);
|
||||||
|
return getWormscanInfo(
|
||||||
|
network,
|
||||||
|
sourceChain,
|
||||||
|
sequence,
|
||||||
|
wormholeRelayerAddress
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getRelayerTransactionHashFromWormscan(
|
||||||
|
sourceChain: ChainName,
|
||||||
|
sequence: number,
|
||||||
|
optionalParams?: {
|
||||||
|
network?: Network;
|
||||||
|
provider?: ethers.providers.Provider;
|
||||||
|
wormholeRelayerAddress?: string;
|
||||||
|
}
|
||||||
|
): Promise<string> {
|
||||||
|
const wormscanData = (
|
||||||
|
await (
|
||||||
|
await getWormscanRelayerInfo(sourceChain, sequence, optionalParams)
|
||||||
|
).json()
|
||||||
|
).data;
|
||||||
|
return "0x" + wormscanData.txHash;
|
||||||
|
}
|
||||||
|
|
||||||
export async function getDeliveryHash(
|
export async function getDeliveryHash(
|
||||||
rx: ethers.ContractReceipt,
|
rx: ethers.ContractReceipt,
|
||||||
sourceChain: ChainName,
|
sourceChain: ChainName,
|
||||||
|
@ -355,6 +461,7 @@ export async function getDeliveryHash(
|
||||||
network?: Network;
|
network?: Network;
|
||||||
provider?: ethers.providers.Provider;
|
provider?: ethers.providers.Provider;
|
||||||
index?: number;
|
index?: number;
|
||||||
|
wormholeRelayerAddress?: string;
|
||||||
}
|
}
|
||||||
): Promise<string> {
|
): Promise<string> {
|
||||||
const network: Network = optionalParams?.network || "MAINNET";
|
const network: Network = optionalParams?.network || "MAINNET";
|
||||||
|
@ -365,6 +472,7 @@ export async function getDeliveryHash(
|
||||||
throw Error(`No wormhole contract on ${sourceChain} for ${network}`);
|
throw Error(`No wormhole contract on ${sourceChain} for ${network}`);
|
||||||
}
|
}
|
||||||
const wormholeRelayerAddress =
|
const wormholeRelayerAddress =
|
||||||
|
optionalParams?.wormholeRelayerAddress ||
|
||||||
RELAYER_CONTRACTS[network][sourceChain]?.wormholeRelayerAddress;
|
RELAYER_CONTRACTS[network][sourceChain]?.wormholeRelayerAddress;
|
||||||
if (!wormholeRelayerAddress) {
|
if (!wormholeRelayerAddress) {
|
||||||
throw Error(
|
throw Error(
|
||||||
|
@ -385,23 +493,71 @@ export async function getDeliveryHash(
|
||||||
index > 0 ? ` (the ${index}-th wormhole relayer log was requested)` : ""
|
index > 0 ? ` (the ${index}-th wormhole relayer log was requested)` : ""
|
||||||
}`
|
}`
|
||||||
);
|
);
|
||||||
const log = logs[index];
|
return getDeliveryHashFromLog(
|
||||||
const wormholePublishedMessage =
|
logs[index],
|
||||||
Implementation__factory.createInterface().parseLog(log);
|
CHAINS[sourceChain],
|
||||||
const block = await provider.getBlock(rx.blockHash);
|
provider,
|
||||||
const body = ethers.utils.solidityPack(
|
rx.blockHash
|
||||||
["uint32", "uint32", "uint16", "bytes32", "uint64", "uint8", "bytes"],
|
|
||||||
|
|
||||||
[
|
|
||||||
block.timestamp,
|
|
||||||
wormholePublishedMessage.args["nonce"],
|
|
||||||
CHAINS[sourceChain],
|
|
||||||
log.topics[1],
|
|
||||||
wormholePublishedMessage.args["sequence"],
|
|
||||||
wormholePublishedMessage.args["consistencyLevel"],
|
|
||||||
wormholePublishedMessage.args["payload"],
|
|
||||||
]
|
|
||||||
);
|
);
|
||||||
const deliveryHash = ethers.utils.keccak256(ethers.utils.keccak256(body));
|
}
|
||||||
return deliveryHash;
|
|
||||||
|
export async function getDeliveryHashFromLog(
|
||||||
|
wormholeLog: ethers.providers.Log,
|
||||||
|
sourceChain: ChainId,
|
||||||
|
provider: ethers.providers.Provider,
|
||||||
|
blockHash: string
|
||||||
|
): Promise<string> {
|
||||||
|
const wormholePublishedMessage =
|
||||||
|
Implementation__factory.createInterface().parseLog(wormholeLog);
|
||||||
|
|
||||||
|
const block = await provider.getBlock(blockHash);
|
||||||
|
|
||||||
|
return getDeliveryHashFromVaaFields(
|
||||||
|
sourceChain,
|
||||||
|
wormholeLog.topics[1],
|
||||||
|
wormholePublishedMessage.args["sequence"],
|
||||||
|
block.timestamp,
|
||||||
|
wormholePublishedMessage.args["nonce"],
|
||||||
|
wormholePublishedMessage.args["consistencyLevel"],
|
||||||
|
wormholePublishedMessage.args["payload"]
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getCCTPMessageLogURL(
|
||||||
|
cctpKey: CCTPKey,
|
||||||
|
sourceChain: ChainName,
|
||||||
|
receipt: ethers.providers.TransactionReceipt,
|
||||||
|
environment: Network
|
||||||
|
) {
|
||||||
|
let cctpLog;
|
||||||
|
let messageSentLog;
|
||||||
|
const DepositForBurnTopic =
|
||||||
|
ethers.utils.keccak256("DepositForBurn(uint64,address,uint256,address,bytes32,uint32,bytes32,bytes32)");
|
||||||
|
const MessageSentTopic = ethers.utils.keccak256("MessageSent(bytes)")
|
||||||
|
try {
|
||||||
|
if (CCTP_DOMAIN_TO_NAME[cctpKey.domain] === sourceChain) {
|
||||||
|
const cctpLogFilter = (log: ethers.providers.Log) => {
|
||||||
|
return (
|
||||||
|
log.topics[0] === DepositForBurnTopic &&
|
||||||
|
parseInt(log.topics[1]) === cctpKey.nonce.toNumber()
|
||||||
|
);
|
||||||
|
};
|
||||||
|
cctpLog = receipt.logs.find(cctpLogFilter);
|
||||||
|
const index = receipt.logs.findIndex(cctpLogFilter);
|
||||||
|
const messageSentLogs = receipt.logs.filter((log, i) => {
|
||||||
|
return log.topics[0] === MessageSentTopic && i <= index;
|
||||||
|
});
|
||||||
|
messageSentLog = messageSentLogs[messageSentLogs.length - 1];
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
console.log(e);
|
||||||
|
}
|
||||||
|
if (!cctpLog || !messageSentLog) return undefined;
|
||||||
|
|
||||||
|
const message = new ethers.utils.Interface([
|
||||||
|
"event MessageSent(bytes message)",
|
||||||
|
]).parseLog(messageSentLog).args.message;
|
||||||
|
const msgHash = ethers.utils.keccak256(message);
|
||||||
|
const url = getCircleAPI(environment) + msgHash;
|
||||||
|
return { message, cctpLog, url };
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,6 +8,7 @@ import {
|
||||||
tryNativeToHexString,
|
tryNativeToHexString,
|
||||||
Network,
|
Network,
|
||||||
ethers_contracts,
|
ethers_contracts,
|
||||||
|
parseTransferPayload,
|
||||||
} from "../..";
|
} from "../..";
|
||||||
import { BigNumber, ethers } from "ethers";
|
import { BigNumber, ethers } from "ethers";
|
||||||
import { getWormholeRelayerAddress } from "../consts";
|
import { getWormholeRelayerAddress } from "../consts";
|
||||||
|
@ -20,27 +21,37 @@ import {
|
||||||
KeyType,
|
KeyType,
|
||||||
parseVaaKey,
|
parseVaaKey,
|
||||||
parseCCTPKey,
|
parseCCTPKey,
|
||||||
|
RedeliveryInstruction,
|
||||||
} from "../structs";
|
} from "../structs";
|
||||||
import {
|
import {
|
||||||
getDefaultProvider,
|
getDefaultProvider,
|
||||||
printChain,
|
printChain,
|
||||||
getWormholeRelayerLog,
|
printCCTPDomain,
|
||||||
|
getWormholeLog,
|
||||||
parseWormholeLog,
|
parseWormholeLog,
|
||||||
getBlockRange,
|
getDeliveryHashFromLog,
|
||||||
getWormholeRelayerInfoBySourceSequence,
|
getRelayerTransactionHashFromWormscan,
|
||||||
|
getWormholeRelayerInfoByHash,
|
||||||
|
getWormscanRelayerInfo,
|
||||||
|
getWormscanInfo,
|
||||||
|
estimatedAttestationTimeInSeconds,
|
||||||
|
getCCTPMessageLogURL,
|
||||||
} from "./helpers";
|
} from "./helpers";
|
||||||
import { DeliveryInfo } from "./deliver";
|
import {
|
||||||
|
AdditionalMessageParsed,
|
||||||
|
CCTPTransferParsed,
|
||||||
|
DeliveryInfo,
|
||||||
|
TokenTransferParsed,
|
||||||
|
} from "./deliver";
|
||||||
|
import { ERC20__factory } from "../../ethers-contracts";
|
||||||
|
|
||||||
export type InfoRequestParams = {
|
export type InfoRequestParams = {
|
||||||
environment?: Network;
|
environment?: Network;
|
||||||
sourceChainProvider?: ethers.providers.Provider;
|
sourceChainProvider?: ethers.providers.Provider;
|
||||||
targetChainProviders?: Map<ChainName, ethers.providers.Provider>;
|
targetChainProviders?: Map<ChainName, ethers.providers.Provider>;
|
||||||
targetChainBlockRanges?: Map<
|
|
||||||
ChainName,
|
|
||||||
[ethers.providers.BlockTag, ethers.providers.BlockTag]
|
|
||||||
>;
|
|
||||||
wormholeRelayerWhMessageIndex?: number;
|
wormholeRelayerWhMessageIndex?: number;
|
||||||
wormholeRelayerAddresses?: Map<ChainName, string>;
|
wormholeRelayerAddresses?: Map<ChainName, string>;
|
||||||
|
targetBlockRange?: [ethers.providers.BlockTag, ethers.providers.BlockTag];
|
||||||
};
|
};
|
||||||
|
|
||||||
export type GetPriceOptParams = {
|
export type GetPriceOptParams = {
|
||||||
|
@ -119,7 +130,12 @@ export async function getWormholeRelayerInfo(
|
||||||
const receipt = await sourceChainProvider.getTransactionReceipt(
|
const receipt = await sourceChainProvider.getTransactionReceipt(
|
||||||
sourceTransaction
|
sourceTransaction
|
||||||
);
|
);
|
||||||
if (!receipt) throw Error("Transaction has not been mined");
|
if (!receipt)
|
||||||
|
throw Error(
|
||||||
|
`Transaction has not been mined: ${sourceTransaction} on ${sourceChain} (${environment})`
|
||||||
|
);
|
||||||
|
const sourceTimestamp =
|
||||||
|
(await sourceChainProvider.getBlock(receipt.blockNumber)).timestamp * 1000;
|
||||||
const bridgeAddress = CONTRACTS[environment][sourceChain].core;
|
const bridgeAddress = CONTRACTS[environment][sourceChain].core;
|
||||||
const wormholeRelayerAddress =
|
const wormholeRelayerAddress =
|
||||||
infoRequest?.wormholeRelayerAddresses?.get(sourceChain) ||
|
infoRequest?.wormholeRelayerAddresses?.get(sourceChain) ||
|
||||||
|
@ -129,7 +145,7 @@ export async function getWormholeRelayerInfo(
|
||||||
`Invalid chain ID or network: Chain ${sourceChain}, ${environment}`
|
`Invalid chain ID or network: Chain ${sourceChain}, ${environment}`
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
const deliveryLog = getWormholeRelayerLog(
|
const deliveryLog = getWormholeLog(
|
||||||
receipt,
|
receipt,
|
||||||
bridgeAddress,
|
bridgeAddress,
|
||||||
tryNativeToHexString(wormholeRelayerAddress, "ethereum"),
|
tryNativeToHexString(wormholeRelayerAddress, "ethereum"),
|
||||||
|
@ -140,9 +156,54 @@ export async function getWormholeRelayerInfo(
|
||||||
|
|
||||||
const { type, parsed } = parseWormholeLog(deliveryLog.log);
|
const { type, parsed } = parseWormholeLog(deliveryLog.log);
|
||||||
|
|
||||||
|
if (type === RelayerPayloadId.Redelivery) {
|
||||||
|
const redeliveryInstruction = parsed as RedeliveryInstruction;
|
||||||
|
|
||||||
|
if (!isChain(redeliveryInstruction.deliveryVaaKey.chainId)) {
|
||||||
|
throw new Error(
|
||||||
|
`The chain ID specified by this redelivery is invalid: ${redeliveryInstruction.deliveryVaaKey.chainId}`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
if (!isChain(redeliveryInstruction.targetChainId)) {
|
||||||
|
throw new Error(
|
||||||
|
`The target chain ID specified by this redelivery is invalid: ${redeliveryInstruction.targetChainId}`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const originalSourceChainName =
|
||||||
|
CHAIN_ID_TO_NAME[redeliveryInstruction.deliveryVaaKey.chainId as ChainId];
|
||||||
|
|
||||||
|
const modifiedInfoRequest = infoRequest;
|
||||||
|
if (modifiedInfoRequest?.sourceChainProvider) {
|
||||||
|
modifiedInfoRequest.sourceChainProvider =
|
||||||
|
modifiedInfoRequest?.targetChainProviders?.get(originalSourceChainName);
|
||||||
|
}
|
||||||
|
|
||||||
|
const transactionHash = await getRelayerTransactionHashFromWormscan(
|
||||||
|
originalSourceChainName,
|
||||||
|
redeliveryInstruction.deliveryVaaKey.sequence.toNumber(),
|
||||||
|
{
|
||||||
|
network: infoRequest?.environment,
|
||||||
|
provider: infoRequest?.targetChainProviders?.get(
|
||||||
|
originalSourceChainName
|
||||||
|
),
|
||||||
|
wormholeRelayerAddress: infoRequest?.wormholeRelayerAddresses?.get(
|
||||||
|
originalSourceChainName
|
||||||
|
),
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
return getWormholeRelayerInfo(
|
||||||
|
originalSourceChainName,
|
||||||
|
transactionHash,
|
||||||
|
modifiedInfoRequest
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
const instruction = parsed as DeliveryInstruction;
|
const instruction = parsed as DeliveryInstruction;
|
||||||
|
|
||||||
const targetChainId = instruction.targetChainId as ChainId;
|
const targetChainId = instruction.targetChainId;
|
||||||
|
|
||||||
if (!isChain(targetChainId)) throw Error(`Invalid Chain: ${targetChainId}`);
|
if (!isChain(targetChainId)) throw Error(`Invalid Chain: ${targetChainId}`);
|
||||||
const targetChain = CHAIN_ID_TO_NAME[targetChainId];
|
const targetChain = CHAIN_ID_TO_NAME[targetChainId];
|
||||||
const targetChainProvider =
|
const targetChainProvider =
|
||||||
|
@ -154,32 +215,195 @@ export async function getWormholeRelayerInfo(
|
||||||
"No default RPC for this chain; pass in your own provider (as targetChainProvider)"
|
"No default RPC for this chain; pass in your own provider (as targetChainProvider)"
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
const [blockStartNumber, blockEndNumber] =
|
|
||||||
infoRequest?.targetChainBlockRanges?.get(targetChain) ||
|
|
||||||
getBlockRange(targetChainProvider);
|
|
||||||
|
|
||||||
const targetChainStatus = await getWormholeRelayerInfoBySourceSequence(
|
const sourceSequence = BigNumber.from(deliveryLog.sequence);
|
||||||
environment,
|
|
||||||
targetChain,
|
const deliveryHash = await getDeliveryHashFromLog(
|
||||||
targetChainProvider,
|
deliveryLog.log,
|
||||||
sourceChain,
|
CHAINS[sourceChain],
|
||||||
BigNumber.from(deliveryLog.sequence),
|
sourceChainProvider,
|
||||||
blockStartNumber,
|
receipt.blockHash
|
||||||
blockEndNumber,
|
|
||||||
infoRequest?.wormholeRelayerAddresses?.get(targetChain) ||
|
|
||||||
getWormholeRelayerAddress(targetChain, environment)
|
|
||||||
);
|
);
|
||||||
|
|
||||||
return {
|
let signingOfVaaTimestamp;
|
||||||
|
try {
|
||||||
|
const vaa = await getWormscanRelayerInfo(
|
||||||
|
sourceChain,
|
||||||
|
sourceSequence.toNumber(),
|
||||||
|
{
|
||||||
|
network: infoRequest?.environment,
|
||||||
|
provider: infoRequest?.sourceChainProvider,
|
||||||
|
wormholeRelayerAddress:
|
||||||
|
infoRequest?.wormholeRelayerAddresses?.get(sourceChain),
|
||||||
|
}
|
||||||
|
);
|
||||||
|
signingOfVaaTimestamp = new Date(
|
||||||
|
(await vaa.json()).data?.indexedAt
|
||||||
|
).getTime();
|
||||||
|
} catch {
|
||||||
|
// wormscan won't work for devnet - so let's hardcode this
|
||||||
|
if (environment === "DEVNET") {
|
||||||
|
signingOfVaaTimestamp = sourceTimestamp;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// obtain additional message info
|
||||||
|
const additionalMessageInformation: AdditionalMessageParsed[] =
|
||||||
|
await Promise.all(
|
||||||
|
instruction.messageKeys.map(async (messageKey) => {
|
||||||
|
if (messageKey.keyType === 1) {
|
||||||
|
// check receipt
|
||||||
|
const vaaKey = parseVaaKey(messageKey.key);
|
||||||
|
|
||||||
|
// if token bridge transfer in logs, parse it
|
||||||
|
let tokenBridgeLog;
|
||||||
|
const tokenBridgeEmitterAddress = tryNativeToHexString(
|
||||||
|
CONTRACTS[environment][sourceChain].token_bridge || "",
|
||||||
|
sourceChain
|
||||||
|
);
|
||||||
|
try {
|
||||||
|
if (
|
||||||
|
vaaKey.chainId === CHAINS[sourceChain] &&
|
||||||
|
vaaKey.emitterAddress.toString("hex") ===
|
||||||
|
tokenBridgeEmitterAddress
|
||||||
|
) {
|
||||||
|
tokenBridgeLog = getWormholeLog(
|
||||||
|
receipt,
|
||||||
|
CONTRACTS[environment][sourceChain].core || "",
|
||||||
|
tokenBridgeEmitterAddress,
|
||||||
|
0,
|
||||||
|
vaaKey.sequence.toNumber()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
console.log(e);
|
||||||
|
}
|
||||||
|
if (!tokenBridgeLog) return undefined;
|
||||||
|
const parsedTokenInfo = parseTransferPayload(
|
||||||
|
Buffer.from(tokenBridgeLog.payload.substring(2), "hex")
|
||||||
|
);
|
||||||
|
const originChainName =
|
||||||
|
CHAIN_ID_TO_NAME[parsedTokenInfo.originChain as ChainId];
|
||||||
|
let signedVaaTimestamp = undefined;
|
||||||
|
let tokenName = undefined;
|
||||||
|
let tokenSymbol = undefined;
|
||||||
|
let tokenDecimals = undefined;
|
||||||
|
|
||||||
|
// Try to get additional token information, assuming it is an ERC20
|
||||||
|
try {
|
||||||
|
const tokenProvider =
|
||||||
|
(parsedTokenInfo.originChain === CHAINS[sourceChain]
|
||||||
|
? infoRequest?.sourceChainProvider
|
||||||
|
: infoRequest?.targetChainProviders?.get(originChainName)) ||
|
||||||
|
getDefaultProvider(environment, originChainName);
|
||||||
|
const tokenContract = ERC20__factory.connect(
|
||||||
|
"0x" + parsedTokenInfo.originAddress.substring(24),
|
||||||
|
tokenProvider
|
||||||
|
);
|
||||||
|
tokenName = await tokenContract.name();
|
||||||
|
tokenSymbol = await tokenContract.symbol();
|
||||||
|
tokenDecimals = await tokenContract.decimals();
|
||||||
|
} catch (e) {
|
||||||
|
console.log(e);
|
||||||
|
}
|
||||||
|
// Try to get wormscan information on if the tokens have been signed
|
||||||
|
try {
|
||||||
|
const tokenVaa = await getWormscanInfo(
|
||||||
|
environment,
|
||||||
|
sourceChain,
|
||||||
|
parseInt(tokenBridgeLog.sequence),
|
||||||
|
CONTRACTS[environment][sourceChain].token_bridge || ""
|
||||||
|
);
|
||||||
|
signedVaaTimestamp = new Date(
|
||||||
|
(await tokenVaa.json()).data?.indexedAt
|
||||||
|
).getTime();
|
||||||
|
} catch {}
|
||||||
|
|
||||||
|
const parsed: TokenTransferParsed = {
|
||||||
|
amount: BigNumber.from(parsedTokenInfo.amount)
|
||||||
|
.mul(
|
||||||
|
BigNumber.from(10).pow(
|
||||||
|
tokenDecimals && tokenDecimals > 8 ? tokenDecimals - 8 : 1
|
||||||
|
)
|
||||||
|
)
|
||||||
|
.toBigInt(),
|
||||||
|
originAddress: parsedTokenInfo.originAddress,
|
||||||
|
originChain: parsedTokenInfo.originChain,
|
||||||
|
targetAddress: parsedTokenInfo.targetAddress,
|
||||||
|
targetChain: parsedTokenInfo.targetChain,
|
||||||
|
fromAddress: parsedTokenInfo.fromAddress,
|
||||||
|
name: tokenName,
|
||||||
|
symbol: tokenSymbol,
|
||||||
|
decimals: tokenDecimals,
|
||||||
|
signedVaaTimestamp,
|
||||||
|
};
|
||||||
|
return parsed;
|
||||||
|
} else if (messageKey.keyType === 2) {
|
||||||
|
// check receipt
|
||||||
|
const cctpKey = parseCCTPKey(messageKey.key);
|
||||||
|
const cctpInfo = await getCCTPMessageLogURL(
|
||||||
|
cctpKey,
|
||||||
|
sourceChain,
|
||||||
|
receipt,
|
||||||
|
environment
|
||||||
|
);
|
||||||
|
const url = cctpInfo?.url || "";
|
||||||
|
|
||||||
|
// Try to get attestation information on if the tokens have been signed
|
||||||
|
let attested = false;
|
||||||
|
try {
|
||||||
|
const attestation = await fetch(url);
|
||||||
|
attested = (await attestation.json()).status === "complete";
|
||||||
|
} catch (e) {
|
||||||
|
console.log(e);
|
||||||
|
}
|
||||||
|
const cctpLog = cctpInfo?.cctpLog!;
|
||||||
|
const parsed: CCTPTransferParsed = {
|
||||||
|
amount: BigNumber.from(
|
||||||
|
Buffer.from(cctpLog.data.substring(2, 2 + 64), "hex")
|
||||||
|
).toBigInt(),
|
||||||
|
mintRecipient: "0x" + cctpLog.data.substring(2 + 64 + 24, 2 + 128),
|
||||||
|
destinationDomain: BigNumber.from(
|
||||||
|
Buffer.from(cctpLog.data.substring(2 + 128, 2 + 192), "hex")
|
||||||
|
).toNumber(),
|
||||||
|
attested,
|
||||||
|
estimatedAttestationSeconds: estimatedAttestationTimeInSeconds(
|
||||||
|
sourceChain,
|
||||||
|
environment
|
||||||
|
),
|
||||||
|
};
|
||||||
|
return parsed;
|
||||||
|
} else {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
|
const targetChainDeliveries = await getWormholeRelayerInfoByHash(
|
||||||
|
deliveryHash,
|
||||||
|
targetChain,
|
||||||
|
sourceChain,
|
||||||
|
sourceSequence.toNumber(),
|
||||||
|
infoRequest
|
||||||
|
);
|
||||||
|
|
||||||
|
const result: DeliveryInfo = {
|
||||||
type: RelayerPayloadId.Delivery,
|
type: RelayerPayloadId.Delivery,
|
||||||
sourceChain: sourceChain,
|
sourceChain: sourceChain,
|
||||||
sourceTransactionHash: sourceTransaction,
|
sourceTransactionHash: sourceTransaction,
|
||||||
sourceDeliverySequenceNumber: BigNumber.from(
|
sourceDeliverySequenceNumber: sourceSequence.toNumber(),
|
||||||
deliveryLog.sequence
|
|
||||||
).toNumber(),
|
|
||||||
deliveryInstruction: instruction,
|
deliveryInstruction: instruction,
|
||||||
targetChainStatus,
|
sourceTimestamp,
|
||||||
|
signingOfVaaTimestamp,
|
||||||
|
additionalMessageInformation,
|
||||||
|
targetChainStatus: {
|
||||||
|
chain: targetChain,
|
||||||
|
events: targetChainDeliveries,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
const stringified = stringifyWormholeRelayerInfo(result);
|
||||||
|
result.stringified = stringified;
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function printWormholeRelayerInfo(info: DeliveryInfo) {
|
export function printWormholeRelayerInfo(info: DeliveryInfo) {
|
||||||
|
@ -198,17 +422,16 @@ export function stringifyWormholeRelayerInfo(
|
||||||
"0000000000000000000000000000000000000000000000000000000000000000"
|
"0000000000000000000000000000000000000000000000000000000000000000"
|
||||||
) {
|
) {
|
||||||
if (!excludeSourceInformation) {
|
if (!excludeSourceInformation) {
|
||||||
stringifiedInfo += `Found delivery request in transaction ${
|
stringifiedInfo += `Source chain: ${info.sourceChain}\n`;
|
||||||
info.sourceTransactionHash
|
|
||||||
} on ${
|
stringifiedInfo += `Source Transaction Hash: ${info.sourceTransactionHash}\n`;
|
||||||
info.sourceChain
|
stringifiedInfo += `Sender: ${
|
||||||
}\nfrom sender ${info.deliveryInstruction.senderAddress.toString(
|
"0x" +
|
||||||
"hex"
|
info.deliveryInstruction.senderAddress.toString("hex").substring(24)
|
||||||
)} from ${info.sourceChain} with delivery sequence number ${
|
|
||||||
info.sourceDeliverySequenceNumber
|
|
||||||
}\n`;
|
}\n`;
|
||||||
|
stringifiedInfo += `Delivery sequence number: ${info.sourceDeliverySequenceNumber}\n`;
|
||||||
} else {
|
} else {
|
||||||
stringifiedInfo += `Found delivery request from sender ${info.deliveryInstruction.senderAddress.toString(
|
stringifiedInfo += `Sender: ${info.deliveryInstruction.senderAddress.toString(
|
||||||
"hex"
|
"hex"
|
||||||
)}\n`;
|
)}\n`;
|
||||||
}
|
}
|
||||||
|
@ -216,30 +439,81 @@ export function stringifyWormholeRelayerInfo(
|
||||||
|
|
||||||
const payload = info.deliveryInstruction.payload.toString("hex");
|
const payload = info.deliveryInstruction.payload.toString("hex");
|
||||||
if (payload.length > 0) {
|
if (payload.length > 0) {
|
||||||
stringifiedInfo += `\nPayload to be relayed (as hex string): 0x${payload}`;
|
stringifiedInfo += `\nPayload to be relayed: 0x${payload}\n`;
|
||||||
}
|
}
|
||||||
if (numMsgs > 0) {
|
if (numMsgs > 0) {
|
||||||
stringifiedInfo += `\nThe following ${numMsgs} wormhole messages (VAAs) were ${
|
stringifiedInfo += `\nThe following ${
|
||||||
|
numMsgs === 1 ? "" : `${numMsgs} `
|
||||||
|
}message${numMsgs === 1 ? " was" : "s were"} ${
|
||||||
payload.length > 0 ? "also " : ""
|
payload.length > 0 ? "also " : ""
|
||||||
}requested to be relayed:\n`;
|
}requested to be relayed with this delivery:\n`;
|
||||||
stringifiedInfo += info.deliveryInstruction.messageKeys
|
stringifiedInfo += info.deliveryInstruction.messageKeys
|
||||||
.map((msgKey, i) => {
|
.map((msgKey, i) => {
|
||||||
let result = "";
|
let result = "";
|
||||||
if (msgKey.keyType == KeyType.VAA) {
|
if (msgKey.keyType == KeyType.VAA) {
|
||||||
const vaaKey = parseVaaKey(msgKey.key);
|
const vaaKey = parseVaaKey(msgKey.key);
|
||||||
result += `(VAA ${i}): `;
|
result += `(Message ${i + 1}): `;
|
||||||
result += `Message from ${
|
result += `Wormhole VAA from ${
|
||||||
vaaKey.chainId ? printChain(vaaKey.chainId) : ""
|
vaaKey.chainId ? printChain(vaaKey.chainId) : ""
|
||||||
}, with emitter address ${vaaKey.emitterAddress?.toString(
|
}, with emitter address ${vaaKey.emitterAddress?.toString(
|
||||||
"hex"
|
"hex"
|
||||||
)} and sequence number ${vaaKey.sequence}`;
|
)} and sequence number ${vaaKey.sequence}`;
|
||||||
|
if (info.additionalMessageInformation[i]) {
|
||||||
|
const tokenTransferInfo = info.additionalMessageInformation[
|
||||||
|
i
|
||||||
|
] as TokenTransferParsed;
|
||||||
|
result += `\nThis is a token bridge transfer of ${
|
||||||
|
tokenTransferInfo.decimals
|
||||||
|
? `${ethers.utils.formatUnits(
|
||||||
|
tokenTransferInfo.amount,
|
||||||
|
tokenTransferInfo.decimals
|
||||||
|
)} `
|
||||||
|
: `${tokenTransferInfo.amount} normalized units of `
|
||||||
|
}${
|
||||||
|
tokenTransferInfo.name
|
||||||
|
? `${tokenTransferInfo.name} (${tokenTransferInfo.symbol})`
|
||||||
|
: `token ${tokenTransferInfo.originAddress.substring(
|
||||||
|
24
|
||||||
|
)} (which is native to ${printChain(
|
||||||
|
tokenTransferInfo.originChain
|
||||||
|
)})`
|
||||||
|
}`;
|
||||||
|
if (tokenTransferInfo.signedVaaTimestamp) {
|
||||||
|
result += `\ntransfer signed by guardians: ${new Date(
|
||||||
|
tokenTransferInfo.signedVaaTimestamp
|
||||||
|
).toString()}`;
|
||||||
|
} else {
|
||||||
|
result += `\ntransfer not yet signed by guardians`;
|
||||||
|
}
|
||||||
|
}
|
||||||
} else if (msgKey.keyType == KeyType.CCTP) {
|
} else if (msgKey.keyType == KeyType.CCTP) {
|
||||||
const cctpKey = parseCCTPKey(msgKey.key);
|
const cctpKey = parseCCTPKey(msgKey.key);
|
||||||
result += `(CCTP ${i}): `;
|
result += `(Message ${i + 1}): `;
|
||||||
result += `Transfer from cctp domain ${printChain(cctpKey.domain)}`;
|
result += `CCTP Transfer from domain ${printCCTPDomain(
|
||||||
|
cctpKey.domain
|
||||||
|
)}`;
|
||||||
result += `, with nonce ${cctpKey.nonce}`;
|
result += `, with nonce ${cctpKey.nonce}`;
|
||||||
|
if (info.additionalMessageInformation[i]) {
|
||||||
|
const cctpTransferInfo = info.additionalMessageInformation[
|
||||||
|
i
|
||||||
|
] as CCTPTransferParsed;
|
||||||
|
result += `\nThis is a CCTP transfer of ${`${ethers.utils.formatUnits(
|
||||||
|
cctpTransferInfo.amount,
|
||||||
|
6
|
||||||
|
)}`} USDC ${
|
||||||
|
cctpTransferInfo.attested
|
||||||
|
? "(Attestation is complete"
|
||||||
|
: "(Attestation currently pending"
|
||||||
|
}, typically takes ${
|
||||||
|
cctpTransferInfo.estimatedAttestationSeconds < 60
|
||||||
|
? `${cctpTransferInfo.estimatedAttestationSeconds} seconds`
|
||||||
|
: `${
|
||||||
|
cctpTransferInfo.estimatedAttestationSeconds / 60
|
||||||
|
} minutes`
|
||||||
|
})`;
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
result += `(Unknown key type${i}): ${msgKey.keyType}`;
|
result += `(Unknown key type ${i}): ${msgKey.keyType}`;
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
})
|
})
|
||||||
|
@ -254,18 +528,13 @@ export function stringifyWormholeRelayerInfo(
|
||||||
instruction.requestedReceiverValue = overrides.newReceiverValue;
|
instruction.requestedReceiverValue = overrides.newReceiverValue;
|
||||||
instruction.encodedExecutionInfo = overrides.newExecutionInfo;
|
instruction.encodedExecutionInfo = overrides.newExecutionInfo;
|
||||||
}
|
}
|
||||||
|
|
||||||
const targetChainName =
|
const targetChainName =
|
||||||
CHAIN_ID_TO_NAME[instruction.targetChainId as ChainId];
|
CHAIN_ID_TO_NAME[instruction.targetChainId as ChainId];
|
||||||
stringifiedInfo += `${
|
stringifiedInfo += `\n\nDestination chain: ${printChain(
|
||||||
numMsgs == 0
|
|
||||||
? payload.length == 0
|
|
||||||
? ""
|
|
||||||
: "\n\nPayload was requested to be relayed"
|
|
||||||
: "\n\nThese were requested to be sent"
|
|
||||||
} to 0x${instruction.targetAddress.toString("hex")} on ${printChain(
|
|
||||||
instruction.targetChainId
|
instruction.targetChainId
|
||||||
)}\n`;
|
)}\nDestination address: 0x${instruction.targetAddress
|
||||||
|
.toString("hex")
|
||||||
|
.substring(24)}\n\n`;
|
||||||
const totalReceiverValue = instruction.requestedReceiverValue.add(
|
const totalReceiverValue = instruction.requestedReceiverValue.add(
|
||||||
instruction.extraReceiverValue
|
instruction.extraReceiverValue
|
||||||
);
|
);
|
||||||
|
@ -289,42 +558,95 @@ export function stringifyWormholeRelayerInfo(
|
||||||
stringifiedInfo += `Gas limit: ${executionInfo.gasLimit} ${targetChainName} gas\n`;
|
stringifiedInfo += `Gas limit: ${executionInfo.gasLimit} ${targetChainName} gas\n`;
|
||||||
|
|
||||||
const refundAddressChosen =
|
const refundAddressChosen =
|
||||||
instruction.refundAddress !== instruction.refundDeliveryProvider;
|
instruction.refundAddress.toString("hex") !==
|
||||||
|
"0000000000000000000000000000000000000000000000000000000000000000";
|
||||||
if (refundAddressChosen) {
|
if (refundAddressChosen) {
|
||||||
stringifiedInfo += `Refund rate: ${ethers.utils.formatEther(
|
stringifiedInfo += `Refund rate: ${ethers.utils.formatEther(
|
||||||
executionInfo.targetChainRefundPerGasUnused
|
executionInfo.targetChainRefundPerGasUnused
|
||||||
)} of ${targetChainName} currency per unit of gas unused\n`;
|
)} of ${targetChainName} currency per unit of gas unused\n`;
|
||||||
stringifiedInfo += `Refund address: ${instruction.refundAddress.toString(
|
stringifiedInfo += `Refund address: ${instruction.refundAddress.toString(
|
||||||
"hex"
|
"hex"
|
||||||
)}\n`;
|
)} on ${printChain(instruction.refundChainId)}\n`;
|
||||||
}
|
}
|
||||||
stringifiedInfo += `\n`;
|
stringifiedInfo += `\n`;
|
||||||
|
if (info.sourceTimestamp) {
|
||||||
|
stringifiedInfo += `Sent: ${new Date(info.sourceTimestamp).toString()}\n`;
|
||||||
|
}
|
||||||
|
if (info.signingOfVaaTimestamp) {
|
||||||
|
stringifiedInfo += `Delivery vaa signed by guardians: ${new Date(
|
||||||
|
info.signingOfVaaTimestamp
|
||||||
|
).toString()}\n`;
|
||||||
|
} else {
|
||||||
|
stringifiedInfo += `Delivery not yet signed by guardians - check https://wormhole-foundation.github.io/wormhole-dashboard/#/ for status\n`;
|
||||||
|
}
|
||||||
|
stringifiedInfo += `\n`;
|
||||||
|
if (info.targetChainStatus.events.length === 0) {
|
||||||
|
stringifiedInfo += "Delivery has not occured yet\n";
|
||||||
|
}
|
||||||
stringifiedInfo += info.targetChainStatus.events
|
stringifiedInfo += info.targetChainStatus.events
|
||||||
|
|
||||||
.map(
|
.map((e, i) => {
|
||||||
(e, i) =>
|
let override = e.overrides || false;
|
||||||
`Delivery attempt ${i + 1}: ${
|
let overriddenExecutionInfo = e.overrides
|
||||||
e.transactionHash
|
? parseEVMExecutionInfoV1(e.overrides.newExecutionInfo, 0)[0]
|
||||||
? ` ${targetChainName} transaction hash: ${e.transactionHash}`
|
: executionInfo;
|
||||||
: ""
|
let overriddenReceiverValue = e.overrides
|
||||||
}\nStatus: ${e.status}\n${
|
? e.overrides.newReceiverValue
|
||||||
e.revertString
|
: totalReceiverValue;
|
||||||
? `Failure reason: ${
|
const overriddenGasLimit = override
|
||||||
e.gasUsed.eq(executionInfo.gasLimit)
|
? overriddenExecutionInfo.gasLimit
|
||||||
? "Gas limit hit"
|
: executionInfo.gasLimit;
|
||||||
: e.revertString
|
|
||||||
}\n`
|
// Add information about any override applied to the delivery
|
||||||
: ""
|
let overrideStringifiedInfo = "";
|
||||||
}Gas used: ${e.gasUsed.toString()}\nTransaction fee used: ${ethers.utils.formatEther(
|
if (override) {
|
||||||
executionInfo.targetChainRefundPerGasUnused.mul(e.gasUsed)
|
overrideStringifiedInfo += !overriddenReceiverValue.eq(
|
||||||
)} of ${targetChainName} currency\n${`Refund amount: ${ethers.utils.formatEther(
|
totalReceiverValue
|
||||||
executionInfo.targetChainRefundPerGasUnused.mul(
|
)
|
||||||
executionInfo.gasLimit.sub(e.gasUsed)
|
? `Overridden amount to pass into target address: ${ethers.utils.formatEther(
|
||||||
)
|
overriddenReceiverValue
|
||||||
)} of ${targetChainName} currency \nRefund status: ${
|
)} of ${targetChainName} currency\n`
|
||||||
e.refundStatus
|
: ``;
|
||||||
}\n`}`
|
overrideStringifiedInfo += !(
|
||||||
)
|
overriddenGasLimit === executionInfo.gasLimit
|
||||||
|
)
|
||||||
|
? `Overridden gas limit: ${overriddenExecutionInfo.gasLimit} ${targetChainName} gas\n`
|
||||||
|
: "";
|
||||||
|
if (
|
||||||
|
refundAddressChosen &&
|
||||||
|
executionInfo.targetChainRefundPerGasUnused !==
|
||||||
|
overriddenExecutionInfo.targetChainRefundPerGasUnused
|
||||||
|
) {
|
||||||
|
overrideStringifiedInfo += `Overridden refund rate: ${ethers.utils.formatEther(
|
||||||
|
overriddenExecutionInfo.targetChainRefundPerGasUnused
|
||||||
|
)} of ${targetChainName} currency per unit of gas unused\n`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return `Delivery attempt: ${
|
||||||
|
e.transactionHash
|
||||||
|
? ` ${targetChainName} transaction hash: ${e.transactionHash}`
|
||||||
|
: ""
|
||||||
|
}\nDelivery Time: ${new Date(
|
||||||
|
e.timestamp as number
|
||||||
|
).toString()}\n${overrideStringifiedInfo}Status: ${e.status}\n${
|
||||||
|
e.revertString
|
||||||
|
? `Failure reason: ${
|
||||||
|
e.gasUsed.eq(overriddenExecutionInfo.gasLimit)
|
||||||
|
? "Gas limit hit"
|
||||||
|
: e.revertString
|
||||||
|
}\n`
|
||||||
|
: ""
|
||||||
|
}Gas used: ${e.gasUsed.toString()}\nTransaction fee used: ${ethers.utils.formatEther(
|
||||||
|
overriddenExecutionInfo.targetChainRefundPerGasUnused.mul(e.gasUsed)
|
||||||
|
)} of ${targetChainName} currency\n${`Refund amount: ${ethers.utils.formatEther(
|
||||||
|
overriddenExecutionInfo.targetChainRefundPerGasUnused.mul(
|
||||||
|
overriddenExecutionInfo.gasLimit.sub(e.gasUsed)
|
||||||
|
)
|
||||||
|
)} of ${targetChainName} currency \nRefund status: ${
|
||||||
|
e.refundStatus
|
||||||
|
}\n`}`;
|
||||||
|
})
|
||||||
.join("\n");
|
.join("\n");
|
||||||
} else if (
|
} else if (
|
||||||
info.type == RelayerPayloadId.Delivery &&
|
info.type == RelayerPayloadId.Delivery &&
|
||||||
|
@ -347,7 +669,7 @@ export function stringifyWormholeRelayerInfo(
|
||||||
|
|
||||||
.map(
|
.map(
|
||||||
(e, i) =>
|
(e, i) =>
|
||||||
`Delivery attempt ${i + 1}: ${
|
`Delivery attempt: ${
|
||||||
e.transactionHash
|
e.transactionHash
|
||||||
? ` ${targetChainName} transaction hash: ${e.transactionHash}`
|
? ` ${targetChainName} transaction hash: ${e.transactionHash}`
|
||||||
: ""
|
: ""
|
||||||
|
|
|
@ -1,5 +1,4 @@
|
||||||
import { BigNumber, ethers } from "ethers";
|
import { BigNumber, ethers } from "ethers";
|
||||||
import { arrayify } from "ethers/lib/utils";
|
|
||||||
|
|
||||||
export enum RelayerPayloadId {
|
export enum RelayerPayloadId {
|
||||||
Delivery = 1,
|
Delivery = 1,
|
||||||
|
@ -16,7 +15,6 @@ export enum DeliveryStatus {
|
||||||
DeliverySuccess = "Delivery Success",
|
DeliverySuccess = "Delivery Success",
|
||||||
ReceiverFailure = "Receiver Failure",
|
ReceiverFailure = "Receiver Failure",
|
||||||
ThisShouldNeverHappen = "This should never happen. Contact Support.",
|
ThisShouldNeverHappen = "This should never happen. Contact Support.",
|
||||||
DeliveryDidntHappenWithinRange = "Delivery didn't happen within given block range",
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export enum RefundStatus {
|
export enum RefundStatus {
|
||||||
|
@ -25,6 +23,8 @@ export enum RefundStatus {
|
||||||
CrossChainRefundSent = "Cross Chain Refund Sent",
|
CrossChainRefundSent = "Cross Chain Refund Sent",
|
||||||
CrossChainRefundFailProviderNotSupported = "Cross Chain Refund Fail - Provider does not support the refund chain",
|
CrossChainRefundFailProviderNotSupported = "Cross Chain Refund Fail - Provider does not support the refund chain",
|
||||||
CrossChainRefundFailNotEnough = "Cross Chain Refund Fail - Refund too low for cross chain refund",
|
CrossChainRefundFailNotEnough = "Cross Chain Refund Fail - Refund too low for cross chain refund",
|
||||||
|
RefundAddressNotProvided = "No refund address provided",
|
||||||
|
InvalidRefundStatus = "Invalid refund status",
|
||||||
}
|
}
|
||||||
|
|
||||||
export function parseRefundStatus(index: number) {
|
export function parseRefundStatus(index: number) {
|
||||||
|
@ -38,7 +38,9 @@ export function parseRefundStatus(index: number) {
|
||||||
? RefundStatus.CrossChainRefundFailProviderNotSupported
|
? RefundStatus.CrossChainRefundFailProviderNotSupported
|
||||||
: index === 4
|
: index === 4
|
||||||
? RefundStatus.CrossChainRefundFailNotEnough
|
? RefundStatus.CrossChainRefundFailNotEnough
|
||||||
: RefundStatus.CrossChainRefundFailProviderNotSupported;
|
: index === 5
|
||||||
|
? RefundStatus.RefundAddressNotProvided
|
||||||
|
: RefundStatus.InvalidRefundStatus;
|
||||||
}
|
}
|
||||||
|
|
||||||
export enum KeyType {
|
export enum KeyType {
|
||||||
|
@ -121,7 +123,9 @@ export function parseWormholeRelayerPayloadType(
|
||||||
stringPayload: string | Buffer | Uint8Array
|
stringPayload: string | Buffer | Uint8Array
|
||||||
): RelayerPayloadId {
|
): RelayerPayloadId {
|
||||||
const payload =
|
const payload =
|
||||||
typeof stringPayload === "string" ? arrayify(stringPayload) : stringPayload;
|
typeof stringPayload === "string"
|
||||||
|
? ethers.utils.arrayify(stringPayload)
|
||||||
|
: stringPayload;
|
||||||
if (
|
if (
|
||||||
payload[0] != RelayerPayloadId.Delivery &&
|
payload[0] != RelayerPayloadId.Delivery &&
|
||||||
payload[0] != RelayerPayloadId.Redelivery
|
payload[0] != RelayerPayloadId.Redelivery
|
||||||
|
@ -152,28 +156,22 @@ export function parseWormholeRelayerSend(bytes: Buffer): DeliveryInstruction {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
idx += 1;
|
idx += 1;
|
||||||
|
|
||||||
const targetChainId = bytes.readUInt16BE(idx);
|
const targetChainId = bytes.readUInt16BE(idx);
|
||||||
idx += 2;
|
idx += 2;
|
||||||
const targetAddress = bytes.slice(idx, idx + 32);
|
const targetAddress = bytes.slice(idx, idx + 32);
|
||||||
idx += 32;
|
idx += 32;
|
||||||
|
|
||||||
let payload: Buffer;
|
let payload: Buffer;
|
||||||
[payload, idx] = parsePayload(bytes, idx);
|
[payload, idx] = parsePayload(bytes, idx);
|
||||||
|
|
||||||
const requestedReceiverValue = ethers.BigNumber.from(
|
const requestedReceiverValue = ethers.BigNumber.from(
|
||||||
Uint8Array.prototype.subarray.call(bytes, idx, idx + 32)
|
Uint8Array.prototype.subarray.call(bytes, idx, idx + 32)
|
||||||
);
|
);
|
||||||
idx += 32;
|
idx += 32;
|
||||||
|
|
||||||
const extraReceiverValue = ethers.BigNumber.from(
|
const extraReceiverValue = ethers.BigNumber.from(
|
||||||
Uint8Array.prototype.subarray.call(bytes, idx, idx + 32)
|
Uint8Array.prototype.subarray.call(bytes, idx, idx + 32)
|
||||||
);
|
);
|
||||||
idx += 32;
|
idx += 32;
|
||||||
|
|
||||||
let encodedExecutionInfo;
|
let encodedExecutionInfo;
|
||||||
[encodedExecutionInfo, idx] = parsePayload(bytes, idx);
|
[encodedExecutionInfo, idx] = parsePayload(bytes, idx);
|
||||||
|
|
||||||
const refundChainId = bytes.readUInt16BE(idx);
|
const refundChainId = bytes.readUInt16BE(idx);
|
||||||
idx += 2;
|
idx += 2;
|
||||||
const refundAddress = bytes.slice(idx, idx + 32);
|
const refundAddress = bytes.slice(idx, idx + 32);
|
||||||
|
@ -186,7 +184,6 @@ export function parseWormholeRelayerSend(bytes: Buffer): DeliveryInstruction {
|
||||||
idx += 32;
|
idx += 32;
|
||||||
const numMessages = bytes.readUInt8(idx);
|
const numMessages = bytes.readUInt8(idx);
|
||||||
idx += 1;
|
idx += 1;
|
||||||
|
|
||||||
let messageKeys = [] as MessageKey[];
|
let messageKeys = [] as MessageKey[];
|
||||||
for (let i = 0; i < numMessages; ++i) {
|
for (let i = 0; i < numMessages; ++i) {
|
||||||
const res = parseMessageKey(bytes, idx);
|
const res = parseMessageKey(bytes, idx);
|
||||||
|
|
Loading…
Reference in New Issue