Typescript
Token2022 com Web3.js

Token2022 com Web3.js

A Extensão Memo Transfer

A extensão MemoTransfer é uma extensão de conta de Token que exige que todas as transferências recebidas em uma conta de token incluam um memo, facilitando o rastreamento aprimorado de transações e a identificação de usuários.

Inicializando a Conta de Token

Para inicializar a extensão MemoTransfer em uma conta de Token, vamos precisar da função enableRequiredMemoTransfers().

Veja como criar um token com a extensão Memo Transfer:

ts
import { Keypair, SystemProgram, Transaction, sendAndConfirmTransaction } from "@solana/web3.js";
import {
  createInitializeAccountInstruction,
  createEnableRequiredMemoTransfersInstruction,
  getAccountLen,
  ExtensionType,
  TOKEN_2022_PROGRAM_ID,
} from "@solana/spl-token";

const tokenAccount = Keypair.generate();

// Calcula o tamanho necessário para uma conta de Token com a extensão Memo Transfer
const accountLen = getAccountLen([ExtensionType.MemoTransfer]);

// Calcula o mínimo de lamports necessários para isenção de aluguel
const lamports = await connection.getMinimumBalanceForRentExemption(accountLen);

// Cria a conta com o tamanho e owner corretos
const createAccountInstruction = SystemProgram.createAccount({
  fromPubkey: keypair.publicKey,
  newAccountPubkey: tokenAccount.publicKey,
  space: accountLen,
  lamports,
  programId: TOKEN_2022_PROGRAM_ID,
});

// Inicializa a extensão Memo Transfer
const enableMemoTransferInstruction = createEnableRequiredMemoTransfersInstruction(
  destinationTokenAccount.publicKey,
  destinationKeypair.publicKey,
  undefined,
  TOKEN_2022_PROGRAM_ID,
);

// Inicializa a conta de Token em si
const initializeAccountInstruction = createInitializeAccountInstruction(
  tokenAccount.publicKey,
  mint.publicKey,
  keypair.publicKey,
  TOKEN_2022_PROGRAM_ID,
);

const transaction = new Transaction().add(
  createAccountInstruction,
  initializeAccountInstruction,
  enableMemoTransferInstruction,
);

const signature = await sendAndConfirmTransaction(
  connection,
  transaction,
  [keypair, tokenAccount],
  { skipPreflight: false },
);

console.log(
  `Contas de token criadas! Confira sua transação aqui: https://explorer.solana.com/tx/${signature}?cluster=devnet`,
);
Expand
[43 more lines]

Transferindo Token com um Memo

Para usar o programa Memo na Solana, temos duas opções viáveis.

Construindo nossa própria instrução memo "raw" assim:

ts
const message = "Hello, Solana";

new TransactionInstruction({
    keys: [{ pubkey: keypair.publicKey, isSigner: true, isWritable: true }],
    data: Buffer.from(message, "utf-8"), // Mensagem do memo. Neste caso é "Hello, Solana"
    programId: new PublicKey("MemoSq4gqABAXKb96qnH8TysNcWxMyWCqXgDLGmfcHr"), // Programa Memo que valida as chaves e a mensagem do memo
}),

Ou podemos usar o SDK do programa Memo após instalar o pacote assim:

text
npm i @solana/spl-memo

No nosso exemplo, vamos usar a segunda opção e ficará assim:

ts
const memoInstruction = createMemoInstruction("Hello, world!", [keypair.publicKey]);

const transferInstruction = createTransferCheckedInstruction(
  tokenAccount,
  mint.publicKey,
  destinationTokenAccount.publicKey,
  keypair.publicKey,
  BigInt(100e6),
  6,
  undefined,
  TOKEN_2022_PROGRAM_ID,
);

const transferTransaction = new Transaction().add(memoInstruction, transferInstruction);

const transferSignature = await sendAndConfirmTransaction(connection, transferTransaction, [
  keypair,
]);

console.log(
  `Tokens transferidos com memo! Confira sua transação aqui: https://explorer.solana.com/tx/${transferSignature}?cluster=devnet`,
);
Expand
[7 more lines]

Desabilitando e Habilitando o Memo

Se não quisermos exigir que a transferência venha com um memo, podemos fazer isso usando a função disableRequiredMemoTransfer e ficaria assim:

ts
const disableRequiredMemoTransfersInstruction = createDisableRequiredMemoTransfersInstruction(
  destinationTokenAccount,
  destinationKeypair.publicKey,
  undefined,
  TOKEN_2022_PROGRAM_ID,
);

const transferInstruction = createTransferCheckedInstruction(
  tokenAccount,
  mint.publicKey,
  destinationTokenAccount,
  keypair.publicKey,
  BigInt(100e6),
  6,
  undefined,
  TOKEN_2022_PROGRAM_ID,
);

const transaction = new Transaction().add(
  disableRequiredMemoTransfersInstruction,
  transferInstruction,
);

const signature = await sendAndConfirmTransaction(connection, transaction, [
  keypair,
  destinationKeypair,
]);

console.log(
  `Tokens transferidos e estado da conta alterado! Confira sua transação aqui: https://explorer.solana.com/tx/${signature}?cluster=devnet`,
);
Expand
[16 more lines]

E se quisermos habilitá-la novamente, podemos simplesmente usar a função enableRequiredMemoTransfers() assim:

ts
const enableRequiredMemoTransfersInstruction = createEnableRequiredMemoTransfersInstruction(
  destinationTokenAccount,
  destinationKeypair.publicKey,
  undefined,
  TOKEN_2022_PROGRAM_ID,
);

const transferInstruction = createTransferCheckedInstruction(
  tokenAccount,
  mint.publicKey,
  destinationTokenAccount,
  keypair.publicKey,
  BigInt(100e6),
  6,
  undefined,
  TOKEN_2022_PROGRAM_ID,
);

const transaction = new Transaction().add(
  enableRequiredMemoTransfersInstruction,
  transferInstruction,
);

const signature = await sendAndConfirmTransaction(connection, transaction, [
  keypair,
  destinationKeypair,
]);

console.log(
  `Tokens transferidos e estado da conta alterado! Confira sua transação aqui: https://explorer.solana.com/tx/${signature}?cluster=devnet`,
);
Expand
[16 more lines]
Blueshift © 2026Commit: 3c44267