UNPKG

@wormhole-foundation/sdk-solana-cctp

Version:

SDK for Solana, used in conjunction with @wormhole-foundation/sdk

114 lines 5.03 kB
import { TOKEN_PROGRAM_ID } from '@solana/spl-token'; import { PublicKey, SystemProgram } from '@solana/web3.js'; import { CircleBridge, encoding } from '@wormhole-foundation/sdk-connect'; import { SolanaAddress } from '@wormhole-foundation/sdk-solana'; import { createMessageTransmitterProgramInterface } from '../program.js'; import { findProgramAddress } from './../accounts/index.js'; const MAX_NONCES_PER_ACCOUNT = 6400n; export function calculateFirstNonce(nonce) { return (((nonce - BigInt(1)) / MAX_NONCES_PER_ACCOUNT) * MAX_NONCES_PER_ACCOUNT + BigInt(1)); } export function nonceAccount(nonce, sourceChain, messageTransmitterProgramId) { const srcDomain = sourceChain.toString(); const usedNonces = findProgramAddress('used_nonces', messageTransmitterProgramId, [srcDomain, calculateFirstNonce(nonce).toString()]).publicKey; return usedNonces; } export async function createReceiveMessageInstruction(messageTransmitterProgramId, tokenMessengerProgramId, usdcAddress, circleMessage, attestation, payer) { const messageBytes = Buffer.from(CircleBridge.serialize(circleMessage)); const attestationBytes = Buffer.from(encoding.hex.decode(attestation)); const solanaUsdcAddress = new PublicKey(usdcAddress); const sourceUsdcAddress = new PublicKey(circleMessage.payload.burnToken.toUint8Array()); const receiver = new SolanaAddress(circleMessage.payload.mintRecipient).unwrap(); const payerPubkey = payer ? new PublicKey(payer) : receiver; const srcDomain = circleMessage.sourceDomain.toString(); // Find pdas const messageTransmitterAccount = findProgramAddress('message_transmitter', messageTransmitterProgramId); const tokenMessenger = findProgramAddress('token_messenger', tokenMessengerProgramId); const tokenMinter = findProgramAddress('token_minter', tokenMessengerProgramId); const localToken = findProgramAddress('local_token', tokenMessengerProgramId, [solanaUsdcAddress]); const remoteTokenMessengerKey = findProgramAddress('remote_token_messenger', tokenMessengerProgramId, [srcDomain]); const tokenPair = findProgramAddress('token_pair', tokenMessengerProgramId, [ srcDomain, sourceUsdcAddress, ]); const custodyTokenAccount = findProgramAddress('custody', tokenMessengerProgramId, [solanaUsdcAddress]); const authorityPda = findProgramAddress('message_transmitter_authority', messageTransmitterProgramId, [tokenMessengerProgramId]).publicKey; // Calculate the nonce PDA. const usedNonces = nonceAccount(circleMessage.nonce, circleMessage.sourceDomain, messageTransmitterProgramId); const eventAuthority = findProgramAddress('__event_authority', messageTransmitterProgramId); const tokenMessengerEventAuthority = findProgramAddress('__event_authority', tokenMessengerProgramId); // Build the accountMetas list. These are passed as remainingAccounts for the TokenMessengerMinter CPI const accountMetas = []; accountMetas.push({ isSigner: false, isWritable: false, pubkey: tokenMessenger.publicKey, }); accountMetas.push({ isSigner: false, isWritable: false, pubkey: remoteTokenMessengerKey.publicKey, }); accountMetas.push({ isSigner: false, isWritable: true, pubkey: tokenMinter.publicKey, }); accountMetas.push({ isSigner: false, isWritable: true, pubkey: localToken.publicKey, }); accountMetas.push({ isSigner: false, isWritable: false, pubkey: tokenPair.publicKey, }); accountMetas.push({ isSigner: false, isWritable: true, pubkey: receiver, }); accountMetas.push({ isSigner: false, isWritable: true, pubkey: custodyTokenAccount.publicKey, }); accountMetas.push({ isSigner: false, isWritable: false, pubkey: TOKEN_PROGRAM_ID, }); accountMetas.push({ isSigner: false, isWritable: false, pubkey: tokenMessengerEventAuthority.publicKey, }); accountMetas.push({ isSigner: false, isWritable: false, pubkey: tokenMessengerProgramId, }); const messageTransmitterProgram = createMessageTransmitterProgramInterface(messageTransmitterProgramId); return (messageTransmitterProgram.methods .receiveMessage({ message: messageBytes, attestation: attestationBytes, }) .accounts({ payer: payerPubkey, caller: payerPubkey, authorityPda, messageTransmitter: messageTransmitterAccount.publicKey, usedNonces, receiver: tokenMessengerProgramId, systemProgram: SystemProgram.programId, eventAuthority: eventAuthority.publicKey, program: messageTransmitterProgram.programId, }) // Add remainingAccounts needed for TokenMessengerMinter CPI .remainingAccounts(accountMetas) .transaction()); } //# sourceMappingURL=receiveMessage.js.map