2022-10-26 07:28:46 -07:00
|
|
|
import { expect } from "chai";
|
|
|
|
import * as web3 from "@solana/web3.js";
|
2023-05-15 12:55:03 -07:00
|
|
|
import {
|
|
|
|
Metadata,
|
|
|
|
PROGRAM_ID as TOKEN_METADATA_PROGRAM_ID,
|
|
|
|
createCreateMetadataAccountV3Instruction,
|
|
|
|
} from "@metaplex-foundation/mpl-token-metadata";
|
2022-10-26 07:28:46 -07:00
|
|
|
import {
|
|
|
|
ASSOCIATED_TOKEN_PROGRAM_ID,
|
|
|
|
createMint,
|
|
|
|
getAccount,
|
|
|
|
getAssociatedTokenAddressSync,
|
|
|
|
getMint,
|
|
|
|
getOrCreateAssociatedTokenAccount,
|
|
|
|
mintTo,
|
|
|
|
NATIVE_MINT,
|
|
|
|
TOKEN_PROGRAM_ID,
|
|
|
|
} from "@solana/spl-token";
|
|
|
|
import {
|
|
|
|
MockGuardians,
|
|
|
|
GovernanceEmitter,
|
|
|
|
MockEthereumNftBridge,
|
|
|
|
} from "../../../sdk/js/src/mock";
|
|
|
|
import { postVaa } from "../../../sdk/js/src/solana/sendAndConfirmPostVaa";
|
|
|
|
import {
|
|
|
|
BpfLoaderUpgradeable,
|
|
|
|
NodeWallet,
|
2023-05-15 12:55:03 -07:00
|
|
|
deriveTokenMetadataKey,
|
2022-10-26 07:28:46 -07:00
|
|
|
} from "../../../sdk/js/src/solana";
|
|
|
|
import {
|
|
|
|
deriveWormholeEmitterKey,
|
|
|
|
getPostedMessage,
|
|
|
|
getPostedVaa,
|
|
|
|
} from "../../../sdk/js/src/solana/wormhole";
|
|
|
|
import {
|
|
|
|
parseNftBridgeRegisterChainVaa,
|
|
|
|
parseNftTransferPayload,
|
|
|
|
parseVaa,
|
|
|
|
} from "../../../sdk/js/src/vaa";
|
|
|
|
|
|
|
|
import {
|
|
|
|
CORE_BRIDGE_ADDRESS,
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
ETHEREUM_NFT_BRIDGE_ADDRESS,
|
|
|
|
GOVERNANCE_EMITTER_ADDRESS,
|
|
|
|
GUARDIAN_KEYS,
|
|
|
|
GUARDIAN_SET_INDEX,
|
|
|
|
LOCALHOST,
|
|
|
|
WETH_ADDRESS,
|
|
|
|
} from "./helpers/consts";
|
|
|
|
import { ethAddressToBuffer, makeErc721Token, now } from "./helpers/utils";
|
|
|
|
import {
|
|
|
|
createApproveAuthoritySignerInstruction,
|
|
|
|
createCompleteTransferNativeInstruction,
|
|
|
|
createCompleteTransferWrappedInstruction,
|
|
|
|
createCompleteWrappedMetaInstruction,
|
|
|
|
createRegisterChainInstruction,
|
|
|
|
createTransferNativeInstruction,
|
|
|
|
createTransferWrappedInstruction,
|
|
|
|
deriveCustodyKey,
|
|
|
|
deriveEndpointKey,
|
|
|
|
deriveWrappedMintKey,
|
|
|
|
getCompleteTransferNativeAccounts,
|
|
|
|
getCompleteTransferWrappedAccounts,
|
|
|
|
getCompleteWrappedMetaAccounts,
|
|
|
|
getEndpointRegistration,
|
|
|
|
getInitializeAccounts,
|
|
|
|
getRegisterChainAccounts,
|
|
|
|
getTransferNativeAccounts,
|
|
|
|
getTransferWrappedAccounts,
|
|
|
|
getUpgradeContractAccounts,
|
|
|
|
getWrappedMeta,
|
|
|
|
mintToTokenId,
|
|
|
|
NFT_TRANSFER_NATIVE_TOKEN_ADDRESS,
|
|
|
|
} from "../../../sdk/js/src/solana/nftBridge";
|
|
|
|
import {
|
|
|
|
getForeignAssetSolana,
|
|
|
|
getIsWrappedAssetSolana,
|
|
|
|
getOriginalAssetSolana,
|
|
|
|
} from "../../../sdk/js/src/nft_bridge";
|
|
|
|
import { ChainId } from "../../../sdk/js/src";
|
|
|
|
|
|
|
|
describe("NFT Bridge", () => {
|
|
|
|
const connection = new web3.Connection(LOCALHOST, "processed");
|
|
|
|
|
|
|
|
const wallet = new NodeWallet(web3.Keypair.generate());
|
|
|
|
|
|
|
|
// for signing wormhole messages
|
|
|
|
const guardians = new MockGuardians(GUARDIAN_SET_INDEX + 1, GUARDIAN_KEYS);
|
|
|
|
|
|
|
|
const erc721Token = makeErc721Token(
|
|
|
|
WETH_ADDRESS,
|
|
|
|
6969n,
|
|
|
|
"Wetherean",
|
|
|
|
"WETH",
|
|
|
|
"https://ethereum.org/en/developers/tutorials/how-to-write-and-deploy-an-nft/"
|
|
|
|
);
|
|
|
|
|
|
|
|
const localVariables: any = {};
|
|
|
|
|
|
|
|
before("Airdrop SOL", async () => {
|
|
|
|
await connection
|
|
|
|
.requestAirdrop(wallet.key(), 1000 * web3.LAMPORTS_PER_SOL)
|
|
|
|
.then(async (signature) => connection.confirmTransaction(signature));
|
|
|
|
});
|
|
|
|
|
|
|
|
before("Create NFT", async () => {
|
|
|
|
localVariables.mint = await createMint(
|
|
|
|
connection,
|
|
|
|
wallet.signer(),
|
|
|
|
wallet.key(),
|
|
|
|
null,
|
|
|
|
0
|
|
|
|
);
|
|
|
|
|
|
|
|
localVariables.nftMeta = {
|
|
|
|
name: "Space Cadet",
|
|
|
|
symbol: "CADET",
|
|
|
|
uri: "https://spl.solana.com/token#example-create-a-non-fungible-token",
|
|
|
|
};
|
|
|
|
|
2023-05-15 12:55:03 -07:00
|
|
|
const mint: web3.PublicKey = localVariables.mint;
|
|
|
|
const name: string = localVariables.nftMeta.name;
|
|
|
|
const symbol: string = localVariables.nftMeta.symbol;
|
|
|
|
const uri: string = localVariables.nftMeta.uri;
|
|
|
|
|
|
|
|
const accounts = {
|
|
|
|
metadata: deriveTokenMetadataKey(mint),
|
2022-10-26 07:28:46 -07:00
|
|
|
mint,
|
2023-05-15 12:55:03 -07:00
|
|
|
mintAuthority: wallet.key(),
|
|
|
|
payer: wallet.key(),
|
|
|
|
updateAuthority: wallet.key(),
|
|
|
|
};
|
|
|
|
const args = {
|
|
|
|
createMetadataAccountArgsV3: {
|
|
|
|
data: {
|
|
|
|
name,
|
|
|
|
symbol,
|
|
|
|
uri,
|
|
|
|
sellerFeeBasisPoints: 0,
|
|
|
|
creators: null,
|
|
|
|
collection: null,
|
|
|
|
uses: null,
|
|
|
|
},
|
|
|
|
isMutable: false,
|
|
|
|
collectionDetails: null,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
const createMetadataIx = createCreateMetadataAccountV3Instruction(
|
|
|
|
accounts,
|
|
|
|
args,
|
|
|
|
TOKEN_METADATA_PROGRAM_ID
|
2022-10-26 07:28:46 -07:00
|
|
|
);
|
|
|
|
|
|
|
|
const createMetadataTx = await web3.sendAndConfirmTransaction(
|
|
|
|
connection,
|
|
|
|
new web3.Transaction().add(createMetadataIx),
|
|
|
|
[wallet.signer()]
|
|
|
|
);
|
|
|
|
// console.log("createMatadataTx", createMetadataTx);
|
|
|
|
|
|
|
|
localVariables.mintAta = await getOrCreateAssociatedTokenAccount(
|
|
|
|
connection,
|
|
|
|
wallet.signer(),
|
|
|
|
localVariables.mint,
|
|
|
|
wallet.key()
|
|
|
|
).then((account) => account.address);
|
|
|
|
|
|
|
|
const mintToTx = await mintTo(
|
|
|
|
connection,
|
|
|
|
wallet.signer(),
|
|
|
|
localVariables.mint,
|
|
|
|
localVariables.mintAta,
|
|
|
|
wallet.key(),
|
|
|
|
1
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("Accounts", () => {
|
|
|
|
// for generating governance wormhole messages
|
|
|
|
const governance = new GovernanceEmitter(
|
|
|
|
GOVERNANCE_EMITTER_ADDRESS.toBuffer().toString("hex")
|
|
|
|
);
|
|
|
|
|
|
|
|
// nft bridge on Ethereum
|
|
|
|
const ethereumNftBridge = new MockEthereumNftBridge(
|
|
|
|
ETHEREUM_NFT_BRIDGE_ADDRESS
|
|
|
|
);
|
|
|
|
|
|
|
|
const payer = new web3.PublicKey(
|
|
|
|
"6sbzC1eH4FTujJXWj51eQe25cYvr4xfXbJ1vAj7j2k5J"
|
|
|
|
);
|
|
|
|
|
|
|
|
it("Instruction 0: Initialize", () => {
|
|
|
|
const accounts = getInitializeAccounts(NFT_BRIDGE_ADDRESS, payer);
|
|
|
|
|
|
|
|
// verify accounts
|
|
|
|
expect(accounts.payer.equals(payer)).is.true;
|
|
|
|
expect(accounts.config.toString()).to.equal(
|
|
|
|
"J1oLBQPejgP75y9mKAAfftaQtmLhLkuQzbCufmYKMSQz"
|
|
|
|
);
|
|
|
|
expect(accounts.rent.equals(web3.SYSVAR_RENT_PUBKEY)).is.true;
|
|
|
|
expect(accounts.systemProgram.equals(web3.SystemProgram.programId)).to.be
|
|
|
|
.true;
|
|
|
|
});
|
|
|
|
|
|
|
|
it("Instruction 1: Complete Native", () => {
|
|
|
|
const timestamp = 12345678;
|
|
|
|
const mint = NATIVE_MINT;
|
|
|
|
const mintAta = getAssociatedTokenAddressSync(mint, payer);
|
|
|
|
|
|
|
|
const nftMeta = localVariables.nftMeta;
|
|
|
|
const tokenId = mintToTokenId(mint);
|
|
|
|
const nonce = 420;
|
|
|
|
const message = ethereumNftBridge.publishTransferNft(
|
|
|
|
NFT_TRANSFER_NATIVE_TOKEN_ADDRESS.toString("hex"),
|
|
|
|
1,
|
|
|
|
nftMeta.name,
|
|
|
|
nftMeta.symbol,
|
|
|
|
tokenId,
|
|
|
|
nftMeta.uri,
|
|
|
|
1,
|
|
|
|
mintAta.toBuffer().toString("hex"),
|
|
|
|
nonce,
|
|
|
|
timestamp
|
|
|
|
);
|
|
|
|
|
|
|
|
const signedVaa = guardians.addSignatures(
|
|
|
|
message,
|
|
|
|
[0, 1, 2, 3, 5, 7, 8, 9, 10, 12, 15, 16, 18]
|
|
|
|
);
|
|
|
|
|
|
|
|
const accounts = getCompleteTransferNativeAccounts(
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
CORE_BRIDGE_ADDRESS,
|
|
|
|
payer,
|
|
|
|
signedVaa
|
|
|
|
);
|
|
|
|
|
|
|
|
// verify accounts
|
|
|
|
expect(accounts.payer.equals(payer)).is.true;
|
|
|
|
expect(accounts.config.toString()).to.equal(
|
|
|
|
"J1oLBQPejgP75y9mKAAfftaQtmLhLkuQzbCufmYKMSQz"
|
|
|
|
);
|
|
|
|
expect(accounts.vaa.toString()).to.equal(
|
|
|
|
"8b6y2t5NngJxhyicDkMQGWbrFagKEiiuGz2bqsVtf6Ks"
|
|
|
|
);
|
|
|
|
expect(accounts.claim.toString()).to.equal(
|
|
|
|
"G3CvkGY9Pf7zMKmxxfN4UdaqDYcfQoKCoEZeCqh6ZQLR"
|
|
|
|
);
|
|
|
|
expect(accounts.endpoint.toString()).to.equal(
|
|
|
|
"GGobvHkLNgwD7qnMRLFniLjoAtr12H4bqPD6AEHWzCou"
|
|
|
|
);
|
|
|
|
expect(accounts.to.equals(mintAta)).is.true;
|
|
|
|
expect(accounts.toAuthority.equals(payer)).is.true;
|
|
|
|
expect(accounts.custody.toString()).to.equal(
|
|
|
|
"3ju9P66Ng9PEPhjY9HUDiC6taExZgWGULTERcP8RzT2j"
|
|
|
|
);
|
|
|
|
expect(accounts.mint.equals(mint)).is.true;
|
|
|
|
expect(accounts.custodySigner.toString()).to.equal(
|
|
|
|
"HHJPgZGoLrh8VmpmR4kPWmVoo8SZyAWQAVe15UtFJKQ1"
|
|
|
|
);
|
|
|
|
expect(accounts.rent.equals(web3.SYSVAR_RENT_PUBKEY)).is.true;
|
|
|
|
expect(accounts.systemProgram.equals(web3.SystemProgram.programId)).to.be
|
|
|
|
.true;
|
|
|
|
expect(accounts.tokenProgram.equals(TOKEN_PROGRAM_ID)).is.true;
|
|
|
|
expect(accounts.wormholeProgram.equals(CORE_BRIDGE_ADDRESS)).is.true;
|
|
|
|
});
|
|
|
|
|
|
|
|
it("Instruction 2: Complete Wrapped", () => {
|
|
|
|
const timestamp = 23456789;
|
|
|
|
const tokenAddress = ethAddressToBuffer(erc721Token.address);
|
|
|
|
const tokenChain = ethereumNftBridge.chain;
|
|
|
|
const tokenId = erc721Token.tokenId;
|
|
|
|
const mint = deriveWrappedMintKey(
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
tokenChain,
|
|
|
|
tokenAddress,
|
|
|
|
tokenId
|
|
|
|
);
|
|
|
|
const mintAta = getAssociatedTokenAddressSync(mint, payer);
|
|
|
|
|
|
|
|
const name = erc721Token.name;
|
|
|
|
const symbol = erc721Token.symbol;
|
|
|
|
const uri = erc721Token.uri;
|
|
|
|
|
|
|
|
const recipientChain = 1;
|
|
|
|
const nonce = 420;
|
|
|
|
const message = ethereumNftBridge.publishTransferNft(
|
|
|
|
tokenAddress.toString("hex"),
|
|
|
|
tokenChain,
|
|
|
|
name,
|
|
|
|
symbol,
|
|
|
|
tokenId,
|
|
|
|
uri,
|
|
|
|
recipientChain,
|
|
|
|
mintAta.toBuffer().toString("hex"),
|
|
|
|
nonce,
|
|
|
|
timestamp
|
|
|
|
);
|
|
|
|
|
|
|
|
const signedVaa = guardians.addSignatures(
|
|
|
|
message,
|
|
|
|
[0, 1, 2, 3, 5, 7, 8, 9, 10, 12, 15, 16, 18]
|
|
|
|
);
|
|
|
|
|
|
|
|
const accounts = getCompleteTransferWrappedAccounts(
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
CORE_BRIDGE_ADDRESS,
|
|
|
|
payer,
|
|
|
|
signedVaa
|
|
|
|
);
|
|
|
|
|
|
|
|
// verify accounts
|
|
|
|
expect(accounts.payer.equals(payer)).is.true;
|
|
|
|
expect(accounts.config.toString()).to.equal(
|
|
|
|
"J1oLBQPejgP75y9mKAAfftaQtmLhLkuQzbCufmYKMSQz"
|
|
|
|
);
|
|
|
|
expect(accounts.vaa.toString()).to.equal(
|
|
|
|
"2bNynrs1qu3tRah8ztXG1eY6XaiP1xPXSDxYim8DGVsQ"
|
|
|
|
);
|
|
|
|
expect(accounts.claim.toString()).to.equal(
|
|
|
|
"5svagh2zqoHkBewBq2KRKQDJHYHwUzLJNt2sfgpRWLuj"
|
|
|
|
);
|
|
|
|
expect(accounts.endpoint.toString()).to.equal(
|
|
|
|
"GGobvHkLNgwD7qnMRLFniLjoAtr12H4bqPD6AEHWzCou"
|
|
|
|
);
|
|
|
|
expect(accounts.to.equals(mintAta)).is.true;
|
|
|
|
expect(accounts.toAuthority.equals(payer)).is.true;
|
|
|
|
expect(accounts.mint.equals(mint)).is.true;
|
|
|
|
expect(accounts.wrappedMeta.toString()).to.equal(
|
|
|
|
"GjxBVsD3fHPa3R97B4uA6TaMbuKXrQ9So8ktdNA6agXs"
|
|
|
|
);
|
|
|
|
expect(accounts.mintAuthority.toString()).to.equal(
|
|
|
|
"ESeW7kNyP8mvfeqKkZdWkuFsKYeZnQUbrxYnNL8N11hi"
|
|
|
|
);
|
|
|
|
expect(accounts.rent.equals(web3.SYSVAR_RENT_PUBKEY)).is.true;
|
|
|
|
expect(accounts.systemProgram.equals(web3.SystemProgram.programId)).to.be
|
|
|
|
.true;
|
|
|
|
expect(accounts.tokenProgram.equals(TOKEN_PROGRAM_ID)).is.true;
|
2023-05-15 12:55:03 -07:00
|
|
|
expect(accounts.splMetadataProgram.equals(TOKEN_METADATA_PROGRAM_ID)).is
|
|
|
|
.true;
|
2022-10-26 07:28:46 -07:00
|
|
|
expect(
|
|
|
|
accounts.associatedTokenProgram.equals(ASSOCIATED_TOKEN_PROGRAM_ID)
|
|
|
|
).is.true;
|
|
|
|
expect(accounts.wormholeProgram.equals(CORE_BRIDGE_ADDRESS)).is.true;
|
|
|
|
});
|
|
|
|
|
|
|
|
it("Instruction 3: Complete Wrapped Meta", () => {
|
|
|
|
const timestamp = 34567890;
|
|
|
|
const tokenAddress = ethAddressToBuffer(erc721Token.address);
|
|
|
|
const tokenChain = ethereumNftBridge.chain;
|
|
|
|
const tokenId = erc721Token.tokenId;
|
|
|
|
const mint = deriveWrappedMintKey(
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
tokenChain,
|
|
|
|
tokenAddress,
|
|
|
|
tokenId
|
|
|
|
);
|
|
|
|
const mintAta = getAssociatedTokenAddressSync(mint, payer);
|
|
|
|
|
|
|
|
const name = erc721Token.name;
|
|
|
|
const symbol = erc721Token.symbol;
|
|
|
|
const uri = erc721Token.uri;
|
|
|
|
|
|
|
|
const recipientChain = 1;
|
|
|
|
const nonce = 420;
|
|
|
|
const message = ethereumNftBridge.publishTransferNft(
|
|
|
|
tokenAddress.toString("hex"),
|
|
|
|
tokenChain,
|
|
|
|
name,
|
|
|
|
symbol,
|
|
|
|
tokenId,
|
|
|
|
uri,
|
|
|
|
recipientChain,
|
|
|
|
mintAta.toBuffer().toString("hex"),
|
|
|
|
nonce,
|
|
|
|
timestamp
|
|
|
|
);
|
|
|
|
|
|
|
|
const signedVaa = guardians.addSignatures(
|
|
|
|
message,
|
|
|
|
[0, 1, 2, 3, 5, 7, 8, 9, 10, 12, 15, 16, 18]
|
|
|
|
);
|
|
|
|
|
|
|
|
const accounts = getCompleteWrappedMetaAccounts(
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
CORE_BRIDGE_ADDRESS,
|
|
|
|
payer,
|
|
|
|
signedVaa
|
|
|
|
);
|
|
|
|
|
|
|
|
// verify accounts
|
|
|
|
expect(accounts.payer.equals(payer)).is.true;
|
|
|
|
expect(accounts.config.toString()).to.equal(
|
|
|
|
"J1oLBQPejgP75y9mKAAfftaQtmLhLkuQzbCufmYKMSQz"
|
|
|
|
);
|
|
|
|
expect(accounts.vaa.toString()).to.equal(
|
|
|
|
"4ZxueqFJAomm96wiDcxgedvyGTGVSp8HMxCz731xUGFd"
|
|
|
|
);
|
|
|
|
expect(accounts.endpoint.toString()).to.equal(
|
|
|
|
"GGobvHkLNgwD7qnMRLFniLjoAtr12H4bqPD6AEHWzCou"
|
|
|
|
);
|
|
|
|
expect(accounts.mint.equals(mint)).is.true;
|
|
|
|
expect(accounts.wrappedMeta.toString()).to.equal(
|
|
|
|
"GjxBVsD3fHPa3R97B4uA6TaMbuKXrQ9So8ktdNA6agXs"
|
|
|
|
);
|
|
|
|
expect(accounts.splMetadata.toString()).to.equal(
|
|
|
|
"HgFdrXZJt1LzGuowL7KFP5JH7dgp2r22wRHijpgH4x9s"
|
|
|
|
);
|
|
|
|
expect(accounts.mintAuthority.toString()).to.equal(
|
|
|
|
"ESeW7kNyP8mvfeqKkZdWkuFsKYeZnQUbrxYnNL8N11hi"
|
|
|
|
);
|
|
|
|
expect(accounts.rent.equals(web3.SYSVAR_RENT_PUBKEY)).is.true;
|
|
|
|
expect(accounts.systemProgram.equals(web3.SystemProgram.programId)).to.be
|
|
|
|
.true;
|
|
|
|
expect(accounts.tokenProgram.equals(TOKEN_PROGRAM_ID)).is.true;
|
2023-05-15 12:55:03 -07:00
|
|
|
expect(accounts.splMetadataProgram.equals(TOKEN_METADATA_PROGRAM_ID)).is
|
|
|
|
.true;
|
2022-10-26 07:28:46 -07:00
|
|
|
expect(accounts.wormholeProgram.equals(CORE_BRIDGE_ADDRESS)).is.true;
|
|
|
|
});
|
|
|
|
|
|
|
|
it("Instruction 4: Transfer Wrapped", () => {
|
|
|
|
const tokenAddress = ethAddressToBuffer(erc721Token.address);
|
|
|
|
const tokenChain = ethereumNftBridge.chain;
|
|
|
|
const tokenId = erc721Token.tokenId;
|
|
|
|
const mint = deriveWrappedMintKey(
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
tokenChain,
|
|
|
|
tokenAddress,
|
|
|
|
tokenId
|
|
|
|
);
|
|
|
|
const mintAta = getAssociatedTokenAddressSync(mint, payer);
|
|
|
|
|
|
|
|
const message = web3.Keypair.generate();
|
|
|
|
const accounts = getTransferWrappedAccounts(
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
CORE_BRIDGE_ADDRESS,
|
|
|
|
payer,
|
|
|
|
message.publicKey,
|
|
|
|
mintAta,
|
|
|
|
payer,
|
|
|
|
tokenChain,
|
|
|
|
tokenAddress,
|
|
|
|
tokenId
|
|
|
|
);
|
|
|
|
|
|
|
|
// verify accounts
|
|
|
|
expect(accounts.payer.equals(payer)).is.true;
|
|
|
|
expect(accounts.config.toString()).to.equal(
|
|
|
|
"J1oLBQPejgP75y9mKAAfftaQtmLhLkuQzbCufmYKMSQz"
|
|
|
|
);
|
|
|
|
expect(accounts.from.equals(mintAta)).is.true;
|
|
|
|
expect(accounts.fromOwner.equals(payer)).is.true;
|
|
|
|
expect(accounts.mint.equals(mint)).is.true;
|
|
|
|
expect(accounts.wrappedMeta.toString()).to.equal(
|
|
|
|
"GjxBVsD3fHPa3R97B4uA6TaMbuKXrQ9So8ktdNA6agXs"
|
|
|
|
);
|
|
|
|
expect(accounts.splMetadata.toString()).to.equal(
|
|
|
|
"HgFdrXZJt1LzGuowL7KFP5JH7dgp2r22wRHijpgH4x9s"
|
|
|
|
);
|
|
|
|
expect(accounts.authoritySigner.toString()).to.equal(
|
|
|
|
"9xMX62GupB5AhucZyD3oC6aBd1NHsBCA6e1x9fez1zHe"
|
|
|
|
);
|
|
|
|
expect(accounts.wormholeBridge.toString()).to.equal(
|
|
|
|
"DNN2VhmrGTGj6QVnPz4NVfsiSk64cRHzKBLP5kUaQrf8"
|
|
|
|
);
|
|
|
|
expect(accounts.wormholeMessage.equals(message.publicKey)).is.true;
|
|
|
|
expect(accounts.wormholeEmitter.toString()).to.equal(
|
|
|
|
"6rbfWsCH5vFbwKAkaAvBSP1Mom6ZkCaCk2pGAbxWt1CH"
|
|
|
|
);
|
|
|
|
expect(accounts.wormholeSequence.toString()).to.equal(
|
|
|
|
"J1hwmV7YfVygE96jVFgWRE8F1niWRQ9pkQDr61551XpG"
|
|
|
|
);
|
|
|
|
expect(accounts.wormholeFeeCollector.toString()).to.equal(
|
|
|
|
"Cxt3Uka7X8vyHYjU6szcuYVPPFyg1fAtoeVy7eyzPjGV"
|
|
|
|
);
|
|
|
|
expect(accounts.clock.equals(web3.SYSVAR_CLOCK_PUBKEY)).is.true;
|
|
|
|
expect(accounts.rent.equals(web3.SYSVAR_RENT_PUBKEY)).is.true;
|
|
|
|
expect(accounts.systemProgram.equals(web3.SystemProgram.programId)).to.be
|
|
|
|
.true;
|
|
|
|
expect(accounts.tokenProgram.equals(TOKEN_PROGRAM_ID)).is.true;
|
2023-05-15 12:55:03 -07:00
|
|
|
expect(accounts.splMetadataProgram.equals(TOKEN_METADATA_PROGRAM_ID)).is
|
|
|
|
.true;
|
2022-10-26 07:28:46 -07:00
|
|
|
expect(accounts.wormholeProgram.equals(CORE_BRIDGE_ADDRESS)).is.true;
|
|
|
|
});
|
|
|
|
|
|
|
|
it("Instruction 5: Transfer Native", () => {
|
|
|
|
const mint = NATIVE_MINT;
|
|
|
|
const mintAta = getAssociatedTokenAddressSync(mint, payer);
|
|
|
|
|
|
|
|
const message = web3.Keypair.generate();
|
|
|
|
const accounts = getTransferNativeAccounts(
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
CORE_BRIDGE_ADDRESS,
|
|
|
|
payer,
|
|
|
|
message.publicKey,
|
|
|
|
mintAta,
|
|
|
|
mint
|
|
|
|
);
|
|
|
|
|
|
|
|
// verify accounts
|
|
|
|
expect(accounts.payer.equals(payer)).is.true;
|
|
|
|
expect(accounts.config.toString()).to.equal(
|
|
|
|
"J1oLBQPejgP75y9mKAAfftaQtmLhLkuQzbCufmYKMSQz"
|
|
|
|
);
|
|
|
|
expect(accounts.from.equals(mintAta)).is.true;
|
|
|
|
expect(accounts.mint.equals(mint)).is.true;
|
|
|
|
expect(accounts.splMetadata.toString()).to.equal(
|
|
|
|
"6dM4TqWyWJsbx7obrdLcviBkTafD5E8av61zfU6jq57X"
|
|
|
|
);
|
|
|
|
expect(accounts.custody.toString()).to.equal(
|
|
|
|
"3ju9P66Ng9PEPhjY9HUDiC6taExZgWGULTERcP8RzT2j"
|
|
|
|
);
|
|
|
|
expect(accounts.authoritySigner.toString()).to.equal(
|
|
|
|
"9xMX62GupB5AhucZyD3oC6aBd1NHsBCA6e1x9fez1zHe"
|
|
|
|
);
|
|
|
|
expect(accounts.custodySigner.toString()).to.equal(
|
|
|
|
"HHJPgZGoLrh8VmpmR4kPWmVoo8SZyAWQAVe15UtFJKQ1"
|
|
|
|
);
|
|
|
|
expect(accounts.wormholeBridge.toString()).to.equal(
|
|
|
|
"DNN2VhmrGTGj6QVnPz4NVfsiSk64cRHzKBLP5kUaQrf8"
|
|
|
|
);
|
|
|
|
expect(accounts.wormholeMessage.equals(message.publicKey)).is.true;
|
|
|
|
expect(accounts.wormholeEmitter.toString()).to.equal(
|
|
|
|
"6rbfWsCH5vFbwKAkaAvBSP1Mom6ZkCaCk2pGAbxWt1CH"
|
|
|
|
);
|
|
|
|
expect(accounts.wormholeSequence.toString()).to.equal(
|
|
|
|
"J1hwmV7YfVygE96jVFgWRE8F1niWRQ9pkQDr61551XpG"
|
|
|
|
);
|
|
|
|
expect(accounts.wormholeFeeCollector.toString()).to.equal(
|
|
|
|
"Cxt3Uka7X8vyHYjU6szcuYVPPFyg1fAtoeVy7eyzPjGV"
|
|
|
|
);
|
|
|
|
expect(accounts.clock.equals(web3.SYSVAR_CLOCK_PUBKEY)).is.true;
|
|
|
|
expect(accounts.rent.equals(web3.SYSVAR_RENT_PUBKEY)).is.true;
|
|
|
|
expect(accounts.systemProgram.equals(web3.SystemProgram.programId)).to.be
|
|
|
|
.true;
|
|
|
|
expect(accounts.tokenProgram.equals(TOKEN_PROGRAM_ID)).is.true;
|
2023-05-15 12:55:03 -07:00
|
|
|
expect(accounts.splMetadataProgram.equals(TOKEN_METADATA_PROGRAM_ID)).is
|
|
|
|
.true;
|
2022-10-26 07:28:46 -07:00
|
|
|
expect(accounts.wormholeProgram.equals(CORE_BRIDGE_ADDRESS)).is.true;
|
|
|
|
});
|
|
|
|
|
|
|
|
it("Instruction 6: Register Chain", () => {
|
|
|
|
const timestamp = 45678901;
|
|
|
|
const message = governance.publishNftBridgeRegisterChain(
|
|
|
|
timestamp,
|
|
|
|
2,
|
|
|
|
ETHEREUM_NFT_BRIDGE_ADDRESS
|
|
|
|
);
|
|
|
|
const signedVaa = guardians.addSignatures(
|
|
|
|
message,
|
|
|
|
[0, 1, 2, 3, 5, 7, 8, 9, 10, 12, 15, 16, 18]
|
|
|
|
);
|
|
|
|
|
|
|
|
const accounts = getRegisterChainAccounts(
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
CORE_BRIDGE_ADDRESS,
|
|
|
|
payer,
|
|
|
|
signedVaa
|
|
|
|
);
|
|
|
|
|
|
|
|
// verify accounts
|
|
|
|
expect(accounts.payer.equals(payer)).is.true;
|
|
|
|
expect(accounts.config.toString()).to.equal(
|
|
|
|
"J1oLBQPejgP75y9mKAAfftaQtmLhLkuQzbCufmYKMSQz"
|
|
|
|
);
|
|
|
|
expect(accounts.endpoint.toString()).to.equal(
|
|
|
|
"GGobvHkLNgwD7qnMRLFniLjoAtr12H4bqPD6AEHWzCou"
|
|
|
|
);
|
|
|
|
expect(accounts.vaa.toString()).to.equal(
|
|
|
|
"DU2VB93gzJ7Qb8xskdHXF4u3nFoAHD4L5DE1kLtXBHCJ"
|
|
|
|
);
|
|
|
|
expect(accounts.claim.toString()).to.equal(
|
|
|
|
"6QuJAFuXYpz8WvzbMoS41mFki5mdLQswhkxoccg2tmTS"
|
|
|
|
);
|
|
|
|
expect(accounts.rent.equals(web3.SYSVAR_RENT_PUBKEY)).is.true;
|
|
|
|
expect(accounts.systemProgram.equals(web3.SystemProgram.programId)).to.be
|
|
|
|
.true;
|
|
|
|
expect(accounts.wormholeProgram.equals(CORE_BRIDGE_ADDRESS)).is.true;
|
|
|
|
});
|
|
|
|
|
|
|
|
it("Instruction 7: Upgrade Contract", () => {
|
|
|
|
const timestamp = 56789012;
|
|
|
|
const chain = 1;
|
|
|
|
const implementation = new web3.PublicKey(
|
|
|
|
"2B5wMnErS8oKWV1wPTNQQhM1WLyxee2obtBMDtsYeHgA"
|
|
|
|
);
|
|
|
|
const message = governance.publishNftBridgeUpgradeContract(
|
|
|
|
timestamp,
|
|
|
|
chain,
|
|
|
|
implementation.toString()
|
|
|
|
);
|
|
|
|
const signedVaa = guardians.addSignatures(message, [0]);
|
|
|
|
|
|
|
|
const accounts = getUpgradeContractAccounts(
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
CORE_BRIDGE_ADDRESS,
|
|
|
|
payer,
|
|
|
|
signedVaa
|
|
|
|
);
|
|
|
|
|
|
|
|
// verify accounts
|
|
|
|
expect(accounts.payer.equals(payer)).is.true;
|
|
|
|
expect(accounts.vaa.toString()).to.equal(
|
|
|
|
"Evar3arhnjy84wPDUpKPifxFRT9oRJFzwYZAVUKpsTnd"
|
|
|
|
);
|
|
|
|
expect(accounts.claim.toString()).to.equal(
|
|
|
|
"3gHw5uPbhk1dDoCUxtK5VosaqTgV9H7wbNJsttswB4At"
|
|
|
|
);
|
|
|
|
expect(accounts.upgradeAuthority.toString()).to.equal(
|
|
|
|
"8GUsAHTGAjJv5XgdQrHprwVnzqARbWaxZ3GHvquhZhpp"
|
|
|
|
);
|
|
|
|
expect(accounts.spill.equals(payer)).is.true;
|
|
|
|
expect(accounts.implementation.equals(implementation)).is.true;
|
|
|
|
expect(accounts.programData.toString()).to.equal(
|
|
|
|
"2oC7qvaYxBLg1msKS8rPgEab5VSQ7TUFfhJBv75BzdSS"
|
|
|
|
);
|
|
|
|
expect(accounts.nftBridgeProgram.equals(NFT_BRIDGE_ADDRESS)).is.true;
|
|
|
|
expect(accounts.rent.equals(web3.SYSVAR_RENT_PUBKEY)).is.true;
|
|
|
|
expect(accounts.clock.equals(web3.SYSVAR_CLOCK_PUBKEY)).is.true;
|
|
|
|
expect(
|
|
|
|
accounts.bpfLoaderUpgradeable.equals(BpfLoaderUpgradeable.programId)
|
|
|
|
).is.true;
|
|
|
|
expect(accounts.systemProgram.equals(web3.SystemProgram.programId)).to.be
|
|
|
|
.true;
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("NFT Bridge Program Interaction", () => {
|
|
|
|
// for generating governance wormhole messages
|
|
|
|
const governance = new GovernanceEmitter(
|
|
|
|
GOVERNANCE_EMITTER_ADDRESS.toBuffer().toString("hex"),
|
|
|
|
30
|
|
|
|
);
|
|
|
|
|
|
|
|
// nft bridge on Ethereum
|
|
|
|
const ethereumNftBridge = new MockEthereumNftBridge(
|
|
|
|
ETHEREUM_NFT_BRIDGE_ADDRESS
|
|
|
|
);
|
|
|
|
|
|
|
|
describe("Setup NFT Bridge", () => {
|
|
|
|
it("Register Ethereum NFT Bridge", async () => {
|
|
|
|
const timestamp = now();
|
|
|
|
const message = governance.publishNftBridgeRegisterChain(
|
|
|
|
timestamp,
|
|
|
|
2,
|
|
|
|
ETHEREUM_NFT_BRIDGE_ADDRESS
|
|
|
|
);
|
|
|
|
const signedVaa = guardians.addSignatures(
|
|
|
|
message,
|
|
|
|
[0, 1, 2, 3, 5, 7, 8, 9, 10, 12, 15, 16, 18]
|
|
|
|
);
|
|
|
|
|
|
|
|
const txSignatures = await postVaa(
|
|
|
|
connection,
|
|
|
|
wallet.signTransaction,
|
|
|
|
CORE_BRIDGE_ADDRESS,
|
|
|
|
wallet.key(),
|
|
|
|
signedVaa
|
|
|
|
).then((results) => results.map((result) => result.signature));
|
|
|
|
const postTx = txSignatures.pop()!;
|
|
|
|
for (const verifyTx of txSignatures) {
|
|
|
|
// console.log(`verifySignatures: ${verifyTx}`);
|
|
|
|
}
|
|
|
|
// console.log(`postVaa: ${postTx}`);
|
|
|
|
|
|
|
|
const registerChainTx = await web3.sendAndConfirmTransaction(
|
|
|
|
connection,
|
|
|
|
new web3.Transaction().add(
|
|
|
|
createRegisterChainInstruction(
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
CORE_BRIDGE_ADDRESS,
|
|
|
|
wallet.key(),
|
|
|
|
signedVaa
|
|
|
|
)
|
|
|
|
),
|
|
|
|
[wallet.signer()]
|
|
|
|
);
|
|
|
|
// console.log(`registerChainTx: ${registerChainTx}`);
|
|
|
|
|
|
|
|
// verify data
|
|
|
|
const parsed = parseNftBridgeRegisterChainVaa(signedVaa);
|
|
|
|
const endpoint = deriveEndpointKey(
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
parsed.foreignChain,
|
|
|
|
parsed.foreignAddress
|
|
|
|
);
|
|
|
|
const endpointRegistration = await getEndpointRegistration(
|
|
|
|
connection,
|
|
|
|
endpoint
|
|
|
|
);
|
|
|
|
expect(endpointRegistration.chain).to.equal(2);
|
|
|
|
const expectedEmitter = ethAddressToBuffer(ETHEREUM_NFT_BRIDGE_ADDRESS);
|
|
|
|
expect(
|
|
|
|
Buffer.compare(endpointRegistration.contract, expectedEmitter)
|
|
|
|
).to.equal(0);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("Native Token Handling", () => {
|
|
|
|
it("Send NFT", async () => {
|
|
|
|
const mint: web3.PublicKey = localVariables.mint;
|
|
|
|
const mintAta: web3.PublicKey = localVariables.mintAta;
|
|
|
|
const custodyAccount = deriveCustodyKey(NFT_BRIDGE_ADDRESS, mint);
|
|
|
|
|
|
|
|
const walletBalanceBefore = await getAccount(connection, mintAta).then(
|
|
|
|
(account) => account.amount
|
|
|
|
);
|
|
|
|
const custodyBalanceBefore = 0n;
|
|
|
|
|
|
|
|
const nonce = 69;
|
|
|
|
const targetAddress = Buffer.alloc(32, "deadbeef", "hex");
|
|
|
|
const targetChain = 2;
|
|
|
|
|
|
|
|
const approveIx = createApproveAuthoritySignerInstruction(
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
mintAta,
|
|
|
|
wallet.key()
|
|
|
|
);
|
|
|
|
|
|
|
|
const message = web3.Keypair.generate();
|
|
|
|
const transferNativeIx = createTransferNativeInstruction(
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
CORE_BRIDGE_ADDRESS,
|
|
|
|
wallet.key(),
|
|
|
|
message.publicKey,
|
|
|
|
mintAta,
|
|
|
|
mint,
|
|
|
|
nonce,
|
|
|
|
targetAddress,
|
|
|
|
targetChain
|
|
|
|
);
|
|
|
|
|
|
|
|
const approveAndTransferTx = await web3.sendAndConfirmTransaction(
|
|
|
|
connection,
|
|
|
|
new web3.Transaction().add(approveIx, transferNativeIx),
|
|
|
|
[wallet.signer(), message]
|
|
|
|
);
|
|
|
|
// console.log(`approveAndTransferTx: ${approveAndTransferTx}`);
|
|
|
|
|
|
|
|
const walletBalanceAfter = await getAccount(connection, mintAta).then(
|
|
|
|
(account) => account.amount
|
|
|
|
);
|
|
|
|
const custodyBalanceAfter = await getAccount(
|
|
|
|
connection,
|
|
|
|
custodyAccount
|
|
|
|
).then((account) => account.amount);
|
|
|
|
|
|
|
|
// check balance changes
|
|
|
|
expect(walletBalanceBefore - walletBalanceAfter).to.equal(1n);
|
|
|
|
expect(custodyBalanceAfter - custodyBalanceBefore).to.equal(1n);
|
|
|
|
|
|
|
|
// verify data
|
|
|
|
const messageData = await getPostedMessage(
|
|
|
|
connection,
|
|
|
|
message.publicKey
|
|
|
|
).then((posted) => posted.message);
|
|
|
|
expect(messageData.consistencyLevel).to.equal(32);
|
|
|
|
expect(
|
|
|
|
Buffer.compare(
|
|
|
|
messageData.emitterAddress,
|
|
|
|
deriveWormholeEmitterKey(NFT_BRIDGE_ADDRESS).toBuffer()
|
|
|
|
)
|
|
|
|
).to.equal(0);
|
|
|
|
expect(messageData.emitterChain).to.equal(1);
|
|
|
|
expect(messageData.nonce).to.equal(nonce);
|
|
|
|
expect(messageData.sequence).to.equal(0n);
|
|
|
|
expect(messageData.vaaTime).to.equal(0);
|
|
|
|
expect(messageData.vaaSignatureAccount.equals(web3.PublicKey.default))
|
|
|
|
.is.true;
|
|
|
|
expect(messageData.vaaVersion).to.equal(0);
|
|
|
|
|
|
|
|
const nftTransfer = parseNftTransferPayload(messageData.payload);
|
|
|
|
const nftMeta = localVariables.nftMeta;
|
|
|
|
expect(nftTransfer.payloadType).to.equal(1);
|
|
|
|
expect(
|
|
|
|
Buffer.compare(
|
|
|
|
nftTransfer.tokenAddress,
|
|
|
|
NFT_TRANSFER_NATIVE_TOKEN_ADDRESS
|
|
|
|
)
|
|
|
|
).to.equal(0);
|
|
|
|
expect(nftTransfer.tokenChain).to.equal(1);
|
|
|
|
expect(nftTransfer.name).to.equal(nftMeta.name);
|
|
|
|
expect(nftTransfer.symbol).to.equal(nftMeta.symbol);
|
|
|
|
expect(nftTransfer.tokenId).to.equal(mintToTokenId(mint));
|
|
|
|
const expectedUri = Buffer.alloc(200);
|
|
|
|
expectedUri.write(nftMeta.uri, 0);
|
|
|
|
expect(nftTransfer.uri).to.equal(expectedUri.toString());
|
|
|
|
expect(Buffer.compare(nftTransfer.to, targetAddress)).to.equal(0);
|
|
|
|
expect(nftTransfer.toChain).to.equal(targetChain);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("Receive NFT", async () => {
|
|
|
|
const mint: web3.PublicKey = localVariables.mint;
|
|
|
|
const mintAta: web3.PublicKey = localVariables.mintAta;
|
|
|
|
const custodyAccount = deriveCustodyKey(NFT_BRIDGE_ADDRESS, mint);
|
|
|
|
const walletBalanceBefore = await getAccount(connection, mintAta).then(
|
|
|
|
(account) => account.amount
|
|
|
|
);
|
|
|
|
const custodyBalanceBefore = await getAccount(
|
|
|
|
connection,
|
|
|
|
custodyAccount
|
|
|
|
).then((account) => account.amount);
|
|
|
|
|
2023-05-15 12:55:03 -07:00
|
|
|
const metadata = await Metadata.fromAccountAddress(
|
|
|
|
connection,
|
|
|
|
deriveTokenMetadataKey(mint)
|
2022-10-26 07:28:46 -07:00
|
|
|
);
|
|
|
|
|
|
|
|
const tokenChain = 1;
|
|
|
|
const tokenId = mintToTokenId(mint);
|
|
|
|
const recipientChain = 1;
|
|
|
|
const nonce = 420;
|
|
|
|
const message = ethereumNftBridge.publishTransferNft(
|
|
|
|
NFT_TRANSFER_NATIVE_TOKEN_ADDRESS.toString("hex"),
|
|
|
|
tokenChain,
|
2023-05-15 12:55:03 -07:00
|
|
|
metadata.data.name,
|
|
|
|
metadata.data.symbol,
|
2022-10-26 07:28:46 -07:00
|
|
|
tokenId,
|
2023-05-15 12:55:03 -07:00
|
|
|
metadata.data.uri,
|
2022-10-26 07:28:46 -07:00
|
|
|
recipientChain,
|
|
|
|
mintAta.toBuffer().toString("hex"),
|
|
|
|
nonce
|
|
|
|
);
|
|
|
|
|
|
|
|
const signedVaa = guardians.addSignatures(
|
|
|
|
message,
|
|
|
|
[0, 1, 2, 3, 5, 7, 8, 9, 10, 12, 15, 16, 18]
|
|
|
|
);
|
|
|
|
|
|
|
|
const txSignatures = await postVaa(
|
|
|
|
connection,
|
|
|
|
wallet.signTransaction,
|
|
|
|
CORE_BRIDGE_ADDRESS,
|
|
|
|
wallet.key(),
|
|
|
|
signedVaa
|
|
|
|
).then((results) => results.map((result) => result.signature));
|
|
|
|
const postTx = txSignatures.pop()!;
|
|
|
|
for (const verifyTx of txSignatures) {
|
|
|
|
// console.log(`verifySignatures: ${verifyTx}`);
|
|
|
|
}
|
|
|
|
// console.log(`postVaa: ${postTx}`);
|
|
|
|
|
|
|
|
const completeNativeTransferIx =
|
|
|
|
createCompleteTransferNativeInstruction(
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
CORE_BRIDGE_ADDRESS,
|
|
|
|
wallet.key(),
|
|
|
|
signedVaa
|
|
|
|
);
|
|
|
|
|
|
|
|
const completeNativeTransferTx = await web3.sendAndConfirmTransaction(
|
|
|
|
connection,
|
|
|
|
new web3.Transaction().add(completeNativeTransferIx),
|
|
|
|
[wallet.signer()]
|
|
|
|
);
|
|
|
|
// console.log(`completeNativeTransferTx: ${completeNativeTransferTx}`);
|
|
|
|
|
|
|
|
const walletBalanceAfter = await getAccount(connection, mintAta).then(
|
|
|
|
(account) => account.amount
|
|
|
|
);
|
|
|
|
const custodyBalanceAfter = await getAccount(
|
|
|
|
connection,
|
|
|
|
custodyAccount
|
|
|
|
).then((account) => account.amount);
|
|
|
|
|
|
|
|
// check balance changes
|
|
|
|
expect(walletBalanceAfter - walletBalanceBefore).to.equal(1n);
|
|
|
|
expect(custodyBalanceBefore - custodyBalanceAfter).to.equal(1n);
|
|
|
|
|
|
|
|
// verify data
|
|
|
|
const messageData = await getPostedVaa(
|
|
|
|
connection,
|
|
|
|
CORE_BRIDGE_ADDRESS,
|
|
|
|
parseVaa(signedVaa).hash
|
|
|
|
).then((posted) => posted.message);
|
|
|
|
expect(messageData.consistencyLevel).to.equal(
|
|
|
|
ethereumNftBridge.consistencyLevel
|
|
|
|
);
|
|
|
|
expect(
|
|
|
|
Buffer.compare(
|
|
|
|
messageData.emitterAddress,
|
|
|
|
ethAddressToBuffer(ETHEREUM_NFT_BRIDGE_ADDRESS)
|
|
|
|
)
|
|
|
|
).to.equal(0);
|
|
|
|
expect(messageData.emitterChain).to.equal(ethereumNftBridge.chain);
|
|
|
|
expect(messageData.nonce).to.equal(nonce);
|
|
|
|
expect(messageData.sequence).to.equal(1n);
|
|
|
|
expect(messageData.vaaTime).to.equal(0);
|
|
|
|
expect(messageData.vaaVersion).to.equal(1);
|
|
|
|
expect(
|
|
|
|
Buffer.compare(parseVaa(signedVaa).payload, messageData.payload)
|
|
|
|
).to.equal(0);
|
|
|
|
|
|
|
|
const nftTransfer = parseNftTransferPayload(messageData.payload);
|
|
|
|
const nftMeta = localVariables.nftMeta;
|
|
|
|
expect(nftTransfer.payloadType).to.equal(1);
|
|
|
|
expect(
|
|
|
|
Buffer.compare(
|
|
|
|
nftTransfer.tokenAddress,
|
|
|
|
NFT_TRANSFER_NATIVE_TOKEN_ADDRESS
|
|
|
|
)
|
|
|
|
).to.equal(0);
|
|
|
|
expect(nftTransfer.tokenChain).to.equal(tokenChain);
|
|
|
|
expect(nftTransfer.name).to.equal(nftMeta.name);
|
|
|
|
expect(nftTransfer.symbol).to.equal(nftMeta.symbol);
|
|
|
|
expect(nftTransfer.tokenId).to.equal(mintToTokenId(mint));
|
|
|
|
const expectedUri = Buffer.alloc(200);
|
|
|
|
expectedUri.write(nftMeta.uri, 0);
|
|
|
|
expect(nftTransfer.uri).to.equal(expectedUri.toString());
|
|
|
|
expect(Buffer.compare(nftTransfer.to, mintAta.toBuffer())).to.equal(0);
|
|
|
|
expect(nftTransfer.toChain).to.equal(recipientChain);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("NFT Bridge Wrapped Token Handling", () => {
|
|
|
|
it("Receive NFT and Create Metadata", async () => {
|
|
|
|
const tokenAddress = ethAddressToBuffer(erc721Token.address);
|
|
|
|
const tokenChain = ethereumNftBridge.chain;
|
|
|
|
const tokenId = erc721Token.tokenId;
|
|
|
|
const mint = deriveWrappedMintKey(
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
tokenChain,
|
|
|
|
tokenAddress,
|
|
|
|
tokenId
|
|
|
|
);
|
|
|
|
const mintAta = getAssociatedTokenAddressSync(mint, wallet.key());
|
|
|
|
|
|
|
|
const name = erc721Token.name;
|
|
|
|
const symbol = erc721Token.symbol;
|
|
|
|
const uri = erc721Token.uri;
|
|
|
|
|
|
|
|
// token account and mint don't exist yet, so there is no balance
|
|
|
|
const walletBalanceBefore = 0n;
|
|
|
|
const supplyBefore = 0n;
|
|
|
|
|
|
|
|
const recipientChain = 1;
|
|
|
|
const nonce = 420;
|
|
|
|
const message = ethereumNftBridge.publishTransferNft(
|
|
|
|
tokenAddress.toString("hex"),
|
|
|
|
tokenChain,
|
|
|
|
name,
|
|
|
|
symbol,
|
|
|
|
tokenId,
|
|
|
|
uri,
|
|
|
|
recipientChain,
|
|
|
|
mintAta.toBuffer().toString("hex"),
|
|
|
|
nonce
|
|
|
|
);
|
|
|
|
|
|
|
|
const signedVaa = guardians.addSignatures(
|
|
|
|
message,
|
|
|
|
[0, 1, 2, 3, 5, 7, 8, 9, 10, 12, 15, 16, 18]
|
|
|
|
);
|
|
|
|
|
|
|
|
const txSignatures = await postVaa(
|
|
|
|
connection,
|
|
|
|
wallet.signTransaction,
|
|
|
|
CORE_BRIDGE_ADDRESS,
|
|
|
|
wallet.key(),
|
|
|
|
signedVaa
|
|
|
|
).then((results) => results.map((result) => result.signature));
|
|
|
|
const postTx = txSignatures.pop()!;
|
|
|
|
for (const verifyTx of txSignatures) {
|
|
|
|
// console.log(`verifySignatures: ${verifyTx}`);
|
|
|
|
}
|
|
|
|
// console.log(`postVaa: ${postTx}`);
|
|
|
|
|
|
|
|
const completeWrappedTransferIx =
|
|
|
|
createCompleteTransferWrappedInstruction(
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
CORE_BRIDGE_ADDRESS,
|
|
|
|
wallet.key(),
|
|
|
|
signedVaa
|
|
|
|
);
|
|
|
|
|
|
|
|
const completeWrappedTransferTx = await web3.sendAndConfirmTransaction(
|
|
|
|
connection,
|
|
|
|
new web3.Transaction().add(completeWrappedTransferIx),
|
|
|
|
[wallet.signer()]
|
|
|
|
);
|
|
|
|
// console.log(`completeWrappedTransferTx: ${completeWrappedTransferTx}`);
|
|
|
|
|
|
|
|
const walletBalanceAfter = await getAccount(connection, mintAta).then(
|
|
|
|
(account) => account.amount
|
|
|
|
);
|
|
|
|
const supplyAfter = await getMint(connection, mint).then(
|
|
|
|
(info) => info.supply
|
|
|
|
);
|
|
|
|
|
|
|
|
// check balance changes
|
|
|
|
expect(walletBalanceAfter - walletBalanceBefore).to.equal(1n);
|
|
|
|
expect(supplyAfter - supplyBefore).to.equal(1n);
|
|
|
|
|
|
|
|
// we need a separate transaction to execute complete_wrapped_meta instruction
|
|
|
|
// following complete_wrapped because... ???
|
|
|
|
const completeWrappedMetaIx = createCompleteWrappedMetaInstruction(
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
CORE_BRIDGE_ADDRESS,
|
|
|
|
wallet.key(),
|
|
|
|
signedVaa
|
|
|
|
);
|
|
|
|
|
|
|
|
const completeWrappedMetaTx = await web3.sendAndConfirmTransaction(
|
|
|
|
connection,
|
|
|
|
new web3.Transaction().add(completeWrappedMetaIx),
|
|
|
|
[wallet.signer()]
|
|
|
|
);
|
|
|
|
// console.log(`completeWrappedMetaTx: ${completeWrappedMetaTx}`);
|
|
|
|
|
|
|
|
// verify data
|
|
|
|
const messageData = await getPostedVaa(
|
|
|
|
connection,
|
|
|
|
CORE_BRIDGE_ADDRESS,
|
|
|
|
parseVaa(signedVaa).hash
|
|
|
|
).then((posted) => posted.message);
|
|
|
|
expect(messageData.consistencyLevel).to.equal(
|
|
|
|
ethereumNftBridge.consistencyLevel
|
|
|
|
);
|
|
|
|
expect(
|
|
|
|
Buffer.compare(
|
|
|
|
messageData.emitterAddress,
|
|
|
|
ethAddressToBuffer(ETHEREUM_NFT_BRIDGE_ADDRESS)
|
|
|
|
)
|
|
|
|
).to.equal(0);
|
|
|
|
expect(messageData.emitterChain).to.equal(ethereumNftBridge.chain);
|
|
|
|
expect(messageData.nonce).to.equal(nonce);
|
|
|
|
expect(messageData.sequence).to.equal(2n);
|
|
|
|
expect(messageData.vaaTime).to.equal(0);
|
|
|
|
expect(messageData.vaaVersion).to.equal(1);
|
|
|
|
expect(
|
|
|
|
Buffer.compare(parseVaa(signedVaa).payload, messageData.payload)
|
|
|
|
).to.equal(0);
|
|
|
|
|
|
|
|
const nftTransfer = parseNftTransferPayload(messageData.payload);
|
|
|
|
expect(nftTransfer.payloadType).to.equal(1);
|
|
|
|
expect(Buffer.compare(nftTransfer.tokenAddress, tokenAddress)).to.equal(
|
|
|
|
0
|
|
|
|
);
|
|
|
|
expect(nftTransfer.tokenChain).to.equal(tokenChain);
|
|
|
|
expect(nftTransfer.name).to.equal(name);
|
|
|
|
expect(nftTransfer.symbol).to.equal(symbol);
|
|
|
|
expect(nftTransfer.tokenId).to.equal(tokenId);
|
|
|
|
expect(nftTransfer.uri).to.equal(uri);
|
|
|
|
expect(Buffer.compare(nftTransfer.to, mintAta.toBuffer())).to.equal(0);
|
|
|
|
expect(nftTransfer.toChain).to.equal(recipientChain);
|
|
|
|
|
|
|
|
// check wrapped meta
|
|
|
|
const wrappedMeta = await getWrappedMeta(
|
|
|
|
connection,
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
mint
|
|
|
|
);
|
|
|
|
expect(wrappedMeta.chain).to.equal(tokenChain);
|
|
|
|
expect(Buffer.compare(wrappedMeta.tokenAddress, tokenAddress)).to.equal(
|
|
|
|
0
|
|
|
|
);
|
|
|
|
expect(wrappedMeta.tokenId).to.equal(tokenId);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("Send NFT", async () => {
|
|
|
|
const tokenAddress = ethAddressToBuffer(erc721Token.address);
|
|
|
|
const tokenChain = ethereumNftBridge.chain;
|
|
|
|
const tokenId = erc721Token.tokenId;
|
|
|
|
const mint = deriveWrappedMintKey(
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
tokenChain,
|
|
|
|
tokenAddress,
|
|
|
|
tokenId
|
|
|
|
);
|
|
|
|
const mintAta = getAssociatedTokenAddressSync(mint, wallet.key());
|
|
|
|
|
|
|
|
const walletBalanceBefore = await getAccount(connection, mintAta).then(
|
|
|
|
(account) => account.amount
|
|
|
|
);
|
|
|
|
const supplyBefore = await getMint(connection, mint).then(
|
|
|
|
(info) => info.supply
|
|
|
|
);
|
|
|
|
|
|
|
|
const nonce = 69;
|
|
|
|
const targetAddress = Buffer.alloc(32, "deadbeef", "hex");
|
|
|
|
const targetChain = 2;
|
|
|
|
|
|
|
|
const approveIx = createApproveAuthoritySignerInstruction(
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
mintAta,
|
|
|
|
wallet.key()
|
|
|
|
);
|
|
|
|
|
|
|
|
const message = web3.Keypair.generate();
|
|
|
|
const transferWrappedIx = createTransferWrappedInstruction(
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
CORE_BRIDGE_ADDRESS,
|
|
|
|
wallet.key(),
|
|
|
|
message.publicKey,
|
|
|
|
mintAta,
|
|
|
|
wallet.key(),
|
|
|
|
tokenChain,
|
|
|
|
tokenAddress,
|
|
|
|
tokenId,
|
|
|
|
nonce,
|
|
|
|
targetAddress,
|
|
|
|
targetChain
|
|
|
|
);
|
|
|
|
|
|
|
|
const approveAndTransferTx = await web3.sendAndConfirmTransaction(
|
|
|
|
connection,
|
|
|
|
new web3.Transaction().add(approveIx, transferWrappedIx),
|
|
|
|
[wallet.signer(), message]
|
|
|
|
);
|
|
|
|
// console.log(`approveAndTransferTx: ${approveAndTransferTx}`);
|
|
|
|
|
|
|
|
const walletBalanceAfter = await getAccount(connection, mintAta).then(
|
|
|
|
(account) => account.amount
|
|
|
|
);
|
|
|
|
const supplyAfter = await getMint(connection, mint).then(
|
|
|
|
(info) => info.supply
|
|
|
|
);
|
|
|
|
|
|
|
|
// check balance changes
|
|
|
|
expect(walletBalanceBefore - walletBalanceAfter).to.equal(1n);
|
|
|
|
expect(supplyBefore - supplyAfter).to.equal(1n);
|
|
|
|
|
|
|
|
// verify data
|
|
|
|
const messageData = await getPostedMessage(
|
|
|
|
connection,
|
|
|
|
message.publicKey
|
|
|
|
).then((posted) => posted.message);
|
|
|
|
expect(messageData.consistencyLevel).to.equal(32);
|
|
|
|
expect(
|
|
|
|
Buffer.compare(
|
|
|
|
messageData.emitterAddress,
|
|
|
|
deriveWormholeEmitterKey(NFT_BRIDGE_ADDRESS).toBuffer()
|
|
|
|
)
|
|
|
|
).to.equal(0);
|
|
|
|
expect(messageData.emitterChain).to.equal(1);
|
|
|
|
expect(messageData.nonce).to.equal(nonce);
|
|
|
|
expect(messageData.sequence).to.equal(1n);
|
|
|
|
expect(messageData.vaaTime).to.equal(0);
|
|
|
|
expect(messageData.vaaSignatureAccount.equals(web3.PublicKey.default))
|
|
|
|
.is.true;
|
|
|
|
expect(messageData.vaaVersion).to.equal(0);
|
|
|
|
|
|
|
|
const nftTransfer = parseNftTransferPayload(messageData.payload);
|
|
|
|
expect(nftTransfer.payloadType).to.equal(1);
|
|
|
|
expect(Buffer.compare(nftTransfer.tokenAddress, tokenAddress)).to.equal(
|
|
|
|
0
|
|
|
|
);
|
|
|
|
expect(nftTransfer.tokenChain).to.equal(tokenChain);
|
|
|
|
expect(nftTransfer.name).to.equal(erc721Token.name);
|
|
|
|
expect(nftTransfer.symbol).to.equal(erc721Token.symbol);
|
|
|
|
expect(nftTransfer.tokenId).to.equal(tokenId);
|
|
|
|
// bridge does this cool thing of adding padding to the uri when being
|
|
|
|
// transferred out when it's wrapped
|
|
|
|
const expectedUri = Buffer.alloc(200);
|
|
|
|
expectedUri.write(erc721Token.uri, 0);
|
|
|
|
expect(nftTransfer.uri).to.equal(expectedUri.toString());
|
|
|
|
expect(Buffer.compare(nftTransfer.to, targetAddress)).to.equal(0);
|
|
|
|
expect(nftTransfer.toChain).to.equal(targetChain);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("Asset Queries", () => {
|
|
|
|
// nft bridge on Ethereum
|
|
|
|
const ethereumNftBridge = new MockEthereumNftBridge(
|
|
|
|
ETHEREUM_NFT_BRIDGE_ADDRESS
|
|
|
|
);
|
|
|
|
|
|
|
|
describe("getOriginalAssetSolana", () => {
|
|
|
|
it("Non-existent NFT", async () => {
|
|
|
|
const mint = "wot m8?";
|
|
|
|
const asset = await getOriginalAssetSolana(
|
|
|
|
connection,
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
mint
|
|
|
|
);
|
|
|
|
|
|
|
|
// verify results
|
|
|
|
expect(asset.isWrapped).to.be.false;
|
|
|
|
expect(asset.chainId).to.equal(1);
|
|
|
|
expect(
|
|
|
|
Buffer.compare(Buffer.from(asset.assetAddress), Buffer.alloc(32))
|
|
|
|
).to.equal(0);
|
|
|
|
expect(asset.tokenId).is.undefined;
|
|
|
|
});
|
|
|
|
|
|
|
|
it("Native NFT", async () => {
|
|
|
|
const mint = localVariables.mint;
|
|
|
|
|
|
|
|
const asset = await getOriginalAssetSolana(
|
|
|
|
connection,
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
mint
|
|
|
|
);
|
|
|
|
|
|
|
|
// verify results
|
|
|
|
expect(asset.isWrapped).to.be.false;
|
|
|
|
expect(asset.chainId).to.equal(1);
|
|
|
|
expect(
|
|
|
|
Buffer.compare(Buffer.from(asset.assetAddress), mint.toBuffer())
|
|
|
|
).to.equal(0);
|
|
|
|
expect(asset.tokenId).is.undefined;
|
|
|
|
});
|
|
|
|
|
|
|
|
it("Wrapped NFT", async () => {
|
|
|
|
const tokenAddress = ethAddressToBuffer(erc721Token.address);
|
|
|
|
const tokenChain = ethereumNftBridge.chain;
|
|
|
|
const tokenId = erc721Token.tokenId;
|
|
|
|
const mint = deriveWrappedMintKey(
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
tokenChain,
|
|
|
|
tokenAddress,
|
|
|
|
tokenId
|
|
|
|
);
|
|
|
|
|
|
|
|
const asset = await getOriginalAssetSolana(
|
|
|
|
connection,
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
mint
|
|
|
|
);
|
|
|
|
|
|
|
|
// verify results
|
|
|
|
expect(asset.isWrapped).is.true;
|
|
|
|
expect(asset.chainId).to.equal(tokenChain);
|
|
|
|
expect(
|
|
|
|
Buffer.compare(Buffer.from(asset.assetAddress), tokenAddress)
|
|
|
|
).to.equal(0);
|
|
|
|
expect(asset.tokenId).to.equal(tokenId.toString());
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("getForeignAssetSolana", () => {
|
|
|
|
it("Wrapped NFT", async () => {
|
|
|
|
const tokenAddress = ethAddressToBuffer(erc721Token.address);
|
|
|
|
const tokenChain = ethereumNftBridge.chain;
|
|
|
|
const tokenId = erc721Token.tokenId;
|
|
|
|
|
|
|
|
const asset = await getForeignAssetSolana(
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
tokenChain as ChainId,
|
|
|
|
tokenAddress,
|
|
|
|
tokenId
|
|
|
|
);
|
|
|
|
|
|
|
|
// verify results
|
|
|
|
expect(asset).to.equal("GrWwR7tTfJvCLuNbcHAyhqBhuuCr8kG7xg47x47GshF4");
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("getIsWrappedAsset", () => {
|
|
|
|
it("Non-existent NFT", async () => {
|
|
|
|
const mint = null;
|
|
|
|
|
|
|
|
const isWrapped = await getIsWrappedAssetSolana(
|
|
|
|
connection,
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
// @ts-ignore
|
|
|
|
mint
|
|
|
|
);
|
|
|
|
|
|
|
|
// verify results
|
|
|
|
expect(isWrapped).to.be.false;
|
|
|
|
});
|
|
|
|
|
|
|
|
it("Native NFT", async () => {
|
|
|
|
const mint = localVariables.mint;
|
|
|
|
|
|
|
|
const isWrapped = await getIsWrappedAssetSolana(
|
|
|
|
connection,
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
mint
|
|
|
|
);
|
|
|
|
|
|
|
|
// verify results
|
|
|
|
expect(isWrapped).to.be.false;
|
|
|
|
});
|
|
|
|
|
|
|
|
it("Wrapped NFT", async () => {
|
|
|
|
const tokenAddress = ethAddressToBuffer(erc721Token.address);
|
|
|
|
const tokenChain = ethereumNftBridge.chain;
|
|
|
|
const tokenId = erc721Token.tokenId;
|
|
|
|
const mint = deriveWrappedMintKey(
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
tokenChain,
|
|
|
|
tokenAddress,
|
|
|
|
tokenId
|
|
|
|
);
|
|
|
|
|
|
|
|
const isWrapped = await getIsWrappedAssetSolana(
|
|
|
|
connection,
|
|
|
|
NFT_BRIDGE_ADDRESS,
|
|
|
|
mint
|
|
|
|
);
|
|
|
|
|
|
|
|
// verify results
|
|
|
|
expect(isWrapped).is.true;
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|