clients/js: Add support for parsing token bridge transfers

commit-id:00a4fb9e
This commit is contained in:
Csongor Kiss 2022-06-29 13:54:59 +01:00
parent 96560a98f8
commit 33fce26f35
4 changed files with 99 additions and 4 deletions

View File

@ -244,7 +244,7 @@ export async function execute_governance_evm(
console.log("Hash: " + (await tb.registerChain(vaa, overrides)).hash) console.log("Hash: " + (await tb.registerChain(vaa, overrides)).hash)
break break
default: default:
impossible(payload) throw Error(`VAA is of type ${payload.type}, which is not a governance action`)
} }
break break

View File

@ -63,7 +63,7 @@ export async function execute_governance_solana(
ix = token_bridge.register_chain_ix(token_bridge_id.toString(), bridge_id.toString(), from.publicKey.toString(), vaa) ix = token_bridge.register_chain_ix(token_bridge_id.toString(), bridge_id.toString(), from.publicKey.toString(), vaa)
break break
default: default:
ix = impossible(v.payload) throw Error(`VAA is of type ${v.payload.type}, which is not a governance action`)
} }
break break

View File

@ -91,8 +91,7 @@ export async function execute_governance_terra(
console.log("Registering chain"); console.log("Registering chain");
break; break;
default: default:
impossible(payload); throw Error(`VAA is of type ${payload.type}, which is not a governance action`)
execute_msg = impossible(payload);
} }
break; break;
default: default:

View File

@ -60,6 +60,7 @@ export type Payload =
| PortalContractUpgrade<"NFTBridge"> | PortalContractUpgrade<"NFTBridge">
| PortalRegisterChain<"TokenBridge"> | PortalRegisterChain<"TokenBridge">
| PortalRegisterChain<"NFTBridge"> | PortalRegisterChain<"NFTBridge">
| TokenBridgeTransfer
// TODO: add other types of payloads // TODO: add other types of payloads
export function parse(buffer: Buffer): VAA<Payload | null> { export function parse(buffer: Buffer): VAA<Payload | null> {
@ -70,6 +71,7 @@ export function parse(buffer: Buffer): VAA<Payload | null> {
.or(portalContractUpgradeParser("NFTBridge")) .or(portalContractUpgradeParser("NFTBridge"))
.or(portalRegisterChainParser("TokenBridge")) .or(portalRegisterChainParser("TokenBridge"))
.or(portalRegisterChainParser("NFTBridge")) .or(portalRegisterChainParser("NFTBridge"))
.or(tokenBridgeTransferParser())
const payload = parser.parse(vaa.payload) const payload = parser.parse(vaa.payload)
var myVAA = { ...vaa, payload } var myVAA = { ...vaa, payload }
@ -164,9 +166,24 @@ function vaaBody(vaa: VAA<Payload>) {
case "ContractUpgrade": case "ContractUpgrade":
payload_str = serialiseCoreContractUpgrade(payload) payload_str = serialiseCoreContractUpgrade(payload)
break break
default:
impossible(payload)
break
} }
break break
case "NFTBridge": case "NFTBridge":
switch (payload.type) {
case "ContractUpgrade":
payload_str = serialisePortalContractUpgrade(payload)
break
case "RegisterChain":
payload_str = serialisePortalRegisterChain(payload)
break
default:
impossible(payload)
break
}
break
case "TokenBridge": case "TokenBridge":
switch (payload.type) { switch (payload.type) {
case "ContractUpgrade": case "ContractUpgrade":
@ -175,8 +192,17 @@ function vaaBody(vaa: VAA<Payload>) {
case "RegisterChain": case "RegisterChain":
payload_str = serialisePortalRegisterChain(payload) payload_str = serialisePortalRegisterChain(payload)
break break
case "Transfer":
payload_str = serialiseTokenBridgeTransfer(payload)
break
default:
impossible(payload)
break
} }
break break
default:
impossible(payload)
break
} }
const body = [ const body = [
encode("uint32", vaa.timestamp), encode("uint32", vaa.timestamp),
@ -406,6 +432,76 @@ function serialisePortalRegisterChain<Module extends "NFTBridge" | "TokenBridge"
return body.join("") return body.join("")
} }
////////////////////////////////////////////////////////////////////////////////
// Token bridge
export interface TokenBridgeTransfer {
module: "TokenBridge"
type: "Transfer"
amount: bigint
tokenAddress: string
tokenChain: number
toAddress: string
chain: number
fee: bigint
}
function tokenBridgeTransferParser(): P<TokenBridgeTransfer> {
return new P(new Parser()
.endianess("big")
.string("module", {
length: (_) => 0,
formatter: (_) => "TokenBridge"
})
.uint8("type", {
assert: 1,
formatter: (_action) => "Transfer"
})
.array("amount", {
type: "uint8",
lengthInBytes: 32,
formatter: (bytes) => BigNumber.from(bytes).toBigInt()
})
.array("tokenAddress", {
type: "uint8",
lengthInBytes: 32,
formatter: (arr) => "0x" + Buffer.from(arr).toString("hex")
})
.uint16("tokenChain")
.array("toAddress", {
type: "uint8",
lengthInBytes: 32,
formatter: (arr) => "0x" + Buffer.from(arr).toString("hex")
})
.uint16("chain")
.array("fee", {
type: "uint8",
lengthInBytes: 32,
formatter: (bytes) => BigNumber.from(bytes).toBigInt()
})
.string("end", {
greedy: true,
assert: str => str === ""
})
)
}
function serialiseTokenBridgeTransfer(payload: TokenBridgeTransfer): string {
const body = [
encode("uint8", 1),
encode("uint256", payload.amount),
encode("bytes32", hex(payload.tokenAddress)),
encode("uint16", payload.tokenChain),
encode("bytes32", hex(payload.toAddress)),
encode("uint16", payload.chain),
encode("uint256", payload.fee),
]
return body.join("")
}
////////////////////////////////////////////////////////////////////////////////
// NFT bridge
// This function should be called after pattern matching on all possible options // This function should be called after pattern matching on all possible options
// of an enum (union) type, so that typescript can derive that no other options // of an enum (union) type, so that typescript can derive that no other options
// are possible. If (from JavaScript land) an unsupported argument is passed // are possible. If (from JavaScript land) an unsupported argument is passed