From 58212d247d2d6648d10e8d78a9b1860d92fcfbd3 Mon Sep 17 00:00:00 2001 From: Yolley Date: Wed, 27 Dec 2023 22:19:07 +0900 Subject: [PATCH] STREAM-1060: check all token accounts ATA (#118) * STREAM-1060: check all token accounts ATA --- lerna.json | 2 +- package-lock.json | 2 +- packages/stream/package.json | 2 +- packages/stream/solana/StreamClient.ts | 59 +++++++++++++++++--------- packages/stream/solana/types.ts | 9 +++- 5 files changed, 51 insertions(+), 23 deletions(-) diff --git a/lerna.json b/lerna.json index 18865450..5aafebaf 100644 --- a/lerna.json +++ b/lerna.json @@ -2,6 +2,6 @@ "packages": [ "packages/*" ], - "version": "5.9.0", + "version": "5.9.1", "$schema": "node_modules/lerna/schemas/lerna-schema.json" } \ No newline at end of file diff --git a/package-lock.json b/package-lock.json index db491c43..f71b5642 100644 --- a/package-lock.json +++ b/package-lock.json @@ -22169,7 +22169,7 @@ }, "packages/stream": { "name": "@streamflow/stream", - "version": "5.8.1", + "version": "5.9.1", "dependencies": { "@manahippo/aptos-wallet-adapter": "1.0.6", "@mysten/sui.js": "^0.40.0", diff --git a/packages/stream/package.json b/packages/stream/package.json index 84a10674..02003d7d 100644 --- a/packages/stream/package.json +++ b/packages/stream/package.json @@ -1,6 +1,6 @@ { "name": "@streamflow/stream", - "version": "5.9.0", + "version": "5.9.1", "description": "JavaScript SDK to interact with Streamflow protocol.", "main": "dist/index.js", "homepage": "https://github.com/streamflow-finance/js-sdk/", diff --git a/packages/stream/solana/StreamClient.ts b/packages/stream/solana/StreamClient.ts index c3c5f3f8..7c19cd50 100644 --- a/packages/stream/solana/StreamClient.ts +++ b/packages/stream/solana/StreamClient.ts @@ -31,7 +31,7 @@ import { ICreateStreamSolanaExt, IInteractStreamSolanaExt, ITopUpStreamSolanaExt, - CheckAssociatedTokenAccountData, + CheckAssociatedTokenAccountsData, } from "./types"; import { ata, @@ -479,7 +479,7 @@ export default class SolanaStreamClient extends BaseStreamClient { const streamflowTreasuryTokens = await ata(data.mint, STREAMFLOW_TREASURY_PUBLIC_KEY); const partnerTokens = await ata(data.mint, data.partner); - await this.checkAssociatedTokenAccount(data, { invoker }, ixs); + await this.checkAssociatedTokenAccounts(data, { invoker }, ixs); ixs.push( withdrawStreamInstruction(amount, this.programId, { @@ -534,7 +534,7 @@ export default class SolanaStreamClient extends BaseStreamClient { const partnerTokens = await ata(data.mint, data.partner); const ixs: TransactionInstruction[] = []; - await this.checkAssociatedTokenAccount(data, { invoker }, ixs); + await this.checkAssociatedTokenAccounts(data, { invoker }, ixs); ixs.push( cancelStreamInstruction(this.programId, { @@ -947,25 +947,46 @@ export default class SolanaStreamClient extends BaseStreamClient { } /** - * Utility function that checks whether associated token account for the recipient exists and adds an instruction to add if not + * Utility function that checks whether associated token accounts still exist and adds instructions to add them if not */ - private async checkAssociatedTokenAccount( - data: CheckAssociatedTokenAccountData, - { invoker }: IInteractStreamSolanaExt, + private async checkAssociatedTokenAccounts( + data: CheckAssociatedTokenAccountsData, + { invoker, checkTokenAccounts }: IInteractStreamSolanaExt, ixs: TransactionInstruction[] ) { - const accountExists = await this.connection.getAccountInfo(data.recipientTokens); - if (!accountExists?.data) { - ixs.push( - createAssociatedTokenAccountInstruction( - invoker.publicKey!, - data.recipientTokens, - data.recipient, - data.mint, - TOKEN_PROGRAM_ID, - ASSOCIATED_TOKEN_PROGRAM_ID - ) - ); + if (!checkTokenAccounts) { + return; + } + const checkedKeys: Set = new Set(); + // TODO: optimize fetching and maps/arrays + const accountArrays = [ + [data.sender, data.senderTokens], + [data.recipient, data.recipientTokens], + [data.partner, data.partnerTokens], + [data.streamflowTreasury, data.streamflowTreasuryTokens], + ].filter((value) => { + if (checkedKeys.has(value[1])) { + return false; + } + checkedKeys.add(value[1]); + return true; + }); + const response = await this.connection.getMultipleAccountsInfo( + accountArrays.map((item) => item[1]) + ); + for (let i = 0; i < response.length; i++) { + if (!response[i]) { + ixs.push( + createAssociatedTokenAccountInstruction( + invoker.publicKey!, + accountArrays[i][0], + accountArrays[i][1], + data.mint, + TOKEN_PROGRAM_ID, + ASSOCIATED_TOKEN_PROGRAM_ID + ) + ); + } } } diff --git a/packages/stream/solana/types.ts b/packages/stream/solana/types.ts index 277f786a..894b06bc 100644 --- a/packages/stream/solana/types.ts +++ b/packages/stream/solana/types.ts @@ -31,6 +31,7 @@ export interface ICreateStreamSolanaExt { export interface IInteractStreamSolanaExt { invoker: SignerWalletAdapter | Keypair; + checkTokenAccounts?: boolean; } export interface ITopUpStreamSolanaExt { @@ -426,9 +427,15 @@ export interface CreateMultipleStreamsValues { recipients: Recipient[]; } -export interface CheckAssociatedTokenAccountData { +export interface CheckAssociatedTokenAccountsData { + sender: PublicKey; + senderTokens: PublicKey; recipient: PublicKey; recipientTokens: PublicKey; + partner: PublicKey; + partnerTokens: PublicKey; + streamflowTreasury: PublicKey; + streamflowTreasuryTokens: PublicKey; mint: PublicKey; }