diff --git a/cli/package.json b/cli/package.json index cc995e5f49..51a6205a1a 100644 --- a/cli/package.json +++ b/cli/package.json @@ -1,6 +1,6 @@ { "name": "@lightprotocol/zk-compression-cli", - "version": "0.4.2", + "version": "0.4.3", "description": "ZK Compression: Secure Scaling on Solana", "maintainers": [ { diff --git a/js/compressed-token/package.json b/js/compressed-token/package.json index 7a7a81eb52..9bca7b7594 100644 --- a/js/compressed-token/package.json +++ b/js/compressed-token/package.json @@ -1,6 +1,6 @@ { "name": "@lightprotocol/compressed-token", - "version": "0.3.0", + "version": "0.3.1", "description": "JS client to interact with the compressed-token program", "sideEffects": false, "type": "module", diff --git a/js/compressed-token/tests/e2e/rpc-token-interop.test.ts b/js/compressed-token/tests/e2e/rpc-token-interop.test.ts index c5f60c457c..67e62832e8 100644 --- a/js/compressed-token/tests/e2e/rpc-token-interop.test.ts +++ b/js/compressed-token/tests/e2e/rpc-token-interop.test.ts @@ -168,4 +168,34 @@ describe('rpc-interop token', () => { assert.equal(accs.length, 0); }); + + it('[rpc] getCompressedTokenAccountsByOwner with 2 mints should return both mints', async () => { + // additional mint + const mint2 = ( + await createMint( + rpc, + payer, + mintAuthority.publicKey, + TEST_TOKEN_DECIMALS, + ) + ).mint; + + await mintTo(rpc, payer, mint2, bob.publicKey, mintAuthority, bn(1000)); + + const senderAccounts = await rpc.getCompressedTokenAccountsByOwner( + bob.publicKey, + ); + + // check that mint and mint2 exist in list of senderaccounts at least once + assert.isTrue( + senderAccounts.some( + account => account.parsed.mint.toBase58() === mint.toBase58(), + ), + ); + assert.isTrue( + senderAccounts.some( + account => account.parsed.mint.toBase58() === mint2.toBase58(), + ), + ); + }); }); diff --git a/js/stateless.js/package.json b/js/stateless.js/package.json index ca1d1c2cdd..49fc2a3d2f 100644 --- a/js/stateless.js/package.json +++ b/js/stateless.js/package.json @@ -1,6 +1,6 @@ { "name": "@lightprotocol/stateless.js", - "version": "0.4.0", + "version": "0.4.1", "description": "JavaScript API for Light and ZK Compression", "sideEffects": false, "main": "dist/cjs/node/index.cjs", diff --git a/js/stateless.js/src/rpc-interface.ts b/js/stateless.js/src/rpc-interface.ts index 6f6d1db394..ec7054184f 100644 --- a/js/stateless.js/src/rpc-interface.ts +++ b/js/stateless.js/src/rpc-interface.ts @@ -25,6 +25,21 @@ import { } from './state'; import { BN } from '@coral-xyz/anchor'; +export interface LatestNonVotingSignatures { + context: { slot: number }; + value: { + items: { signature: string; slot: number; blockTime: number }[]; + }; +} + +export interface LatestNonVotingSignaturesPaginated { + context: { slot: number }; + value: { + items: { signature: string; slot: number; blockTime: number }[]; + cursor: string | null; + }; +} + export interface SignatureWithMetadata { blockTime: number; signature: string; @@ -68,7 +83,7 @@ export type CompressedProofWithContext = { }; export interface GetCompressedTokenAccountsByOwnerOrDelegateOptions { - mint: PublicKey; + mint?: PublicKey; cursor?: string; limit?: BN; } @@ -259,6 +274,33 @@ export const SlotResult = number(); */ export const HealthResult = string(); +/** + * @internal + */ +export const LatestNonVotingSignaturesResult = pick({ + items: array( + pick({ + signature: string(), + slot: number(), + blockTime: number(), + }), + ), +}); + +/** + * @internal + */ +export const LatestNonVotingSignaturesResultPaginated = pick({ + items: array( + pick({ + signature: string(), + slot: number(), + blockTime: number(), + }), + ), + cursor: nullable(string()), +}); + /** * @internal */ @@ -386,10 +428,11 @@ export const CompressedTransactionResult = pick({ export interface CompressionApiInterface { getCompressedAccount( - hash: BN254, + address?: BN254, + hash?: BN254, ): Promise; - getCompressedBalance(hash: BN254): Promise; + getCompressedBalance(address?: BN254, hash?: BN254): Promise; getCompressedBalanceByOwner(owner: PublicKey): Promise; @@ -405,6 +448,11 @@ export interface CompressionApiInterface { hashes: BN254[], ): Promise; + getValidityProof( + hashes: BN254[], + newAddresses: BN254[], + ): Promise; + getCompressedAccountsByOwner( owner: PublicKey, ): Promise; @@ -426,14 +474,14 @@ export interface CompressionApiInterface { options: GetCompressedTokenAccountsByOwnerOrDelegateOptions, ): Promise<{ balance: BN; mint: PublicKey }[]>; - getSignaturesForCompressedAccount( - hash: BN254, - ): Promise; - getTransactionWithCompressionInfo( signature: string, ): Promise; + getCompressionSignaturesForAccount( + hash: BN254, + ): Promise; + getCompressionSignaturesForAddress( address: PublicKey, ): Promise; @@ -446,12 +494,16 @@ export interface CompressionApiInterface { owner: PublicKey, ): Promise; + getLatestNonVotingSignatures( + limit?: number, + ): Promise; + + getLatestCompressionSignatures( + cursor?: string, + limit?: number, + ): Promise; + getIndexerHealth(): Promise; getIndexerSlot(): Promise; - - getValidityProof( - hashes: BN254[], - newAddresses: BN254[], - ): Promise; } diff --git a/js/stateless.js/src/rpc.ts b/js/stateless.js/src/rpc.ts index 8433f1a91b..1c615fc310 100644 --- a/js/stateless.js/src/rpc.ts +++ b/js/stateless.js/src/rpc.ts @@ -28,9 +28,12 @@ import { TokenBalanceListResult, jsonRpcResult, jsonRpcResultAndContext, - HexBatchInputsForProver, ValidityProofResult, NewAddressProofResult, + LatestNonVotingSignaturesResult, + LatestNonVotingSignatures, + LatestNonVotingSignaturesResultPaginated, + LatestNonVotingSignaturesPaginated, } from './rpc-interface'; import { MerkleContextWithMerkleProof, @@ -85,7 +88,9 @@ async function getCompressedTokenAccountsByOwnerOrDelegate( const unsafeRes = await rpcRequest(rpc.compressionApiEndpoint, endpoint, { [propertyToCheck]: ownerOrDelegate.toBase58(), - mint: options.mint.toBase58(), + mint: options.mint?.toBase58(), + limit: options.limit, + cursor: options.cursor, }); const res = create( @@ -283,7 +288,6 @@ export const proverRequest = async ( if (!response.ok) { throw new Error(`Error fetching proof: ${response.statusText}`); } - /// TODO: Move compression into the gnark prover to save bandwidth. const data: any = await response.json(); const parsed = proofFromJsonStruct(data); const compressedProof = negateAndCompressProof(parsed); @@ -403,15 +407,25 @@ export class Rpc extends Connection implements CompressionApiInterface { } /** - * Fetch the compressed account for the specified account hash + * Fetch the compressed account for the specified account address or hash */ async getCompressedAccount( - hash: BN254, + address?: BN254, + hash?: BN254, ): Promise { + if (!hash && !address) { + throw new Error('Either hash or address must be provided'); + } + if (hash && address) { + throw new Error('Only one of hash or address must be provided'); + } const unsafeRes = await rpcRequest( this.compressionApiEndpoint, 'getCompressedAccount', - { hash: encodeBN254toBase58(hash) }, + { + hash: hash ? encodeBN254toBase58(hash) : undefined, + address: address ? encodeBN254toBase58(address) : undefined, + }, ); const res = create( unsafeRes, @@ -420,7 +434,7 @@ export class Rpc extends Connection implements CompressionApiInterface { if ('error' in res) { throw new SolanaJSONRPCError( res.error, - `failed to get info for compressed account ${hash.toString()}`, + `failed to get info for compressed account ${hash ? hash.toString() : address ? address.toString() : ''}`, ); } if (res.result.value === null) { @@ -443,13 +457,22 @@ export class Rpc extends Connection implements CompressionApiInterface { } /** - * Fetch the compressed balance for the specified account hash + * Fetch the compressed balance for the specified account address or hash */ - async getCompressedBalance(hash: BN254): Promise { + async getCompressedBalance(address?: BN254, hash?: BN254): Promise { + if (!hash && !address) { + throw new Error('Either hash or address must be provided'); + } + if (hash && address) { + throw new Error('Only one of hash or address must be provided'); + } const unsafeRes = await rpcRequest( this.compressionApiEndpoint, 'getCompressedBalance', - { hash: encodeBN254toBase58(hash) }, + { + hash: hash ? encodeBN254toBase58(hash) : undefined, + address: address ? encodeBN254toBase58(address) : undefined, + }, ); const res = create( unsafeRes, @@ -458,7 +481,7 @@ export class Rpc extends Connection implements CompressionApiInterface { if ('error' in res) { throw new SolanaJSONRPCError( res.error, - `failed to get balance for compressed account ${hash.toString()}`, + `failed to get balance for compressed account ${hash ? hash.toString() : address ? address.toString() : ''}`, ); } if (res.result.value === null) { @@ -522,18 +545,14 @@ export class Rpc extends Connection implements CompressionApiInterface { ); } - // const proofWithoutRoot = res.result.value.proof.slice(0, -1); - - // const root = res.result.value.proof[res.result.value.proof.length - 1]; - const value: MerkleContextWithMerkleProof = { hash: res.result.value.hash.toArray('be', 32), merkleTree: res.result.value.merkleTree, leafIndex: res.result.value.leafIndex, - merkleProof: res.result.value.proof, //proofWithoutRoot, - nullifierQueue: mockNullifierQueue, // TODO: use nullifierQueue from indexer - rootIndex: res.result.value.rootSeq % 2400, // TODO: rootSeq % rootHistoryArray.length - root: res.result.value.root, // TODO: validate correct root + merkleProof: res.result.value.proof, + nullifierQueue: mockNullifierQueue, // TODO(photon): support nullifierQueue in response. + rootIndex: res.result.value.rootSeq % 2400, + root: res.result.value.root, }; return value; } @@ -685,8 +704,10 @@ export class Rpc extends Connection implements CompressionApiInterface { */ async getCompressedTokenAccountsByOwner( owner: PublicKey, - options: GetCompressedTokenAccountsByOwnerOrDelegateOptions, + options?: GetCompressedTokenAccountsByOwnerOrDelegateOptions, ): Promise { + if (!options) options = {}; + return await getCompressedTokenAccountsByOwnerOrDelegate( this, owner, @@ -700,8 +721,10 @@ export class Rpc extends Connection implements CompressionApiInterface { */ async getCompressedTokenAccountsByDelegate( delegate: PublicKey, - options: GetCompressedTokenAccountsByOwnerOrDelegateOptions, + options?: GetCompressedTokenAccountsByOwnerOrDelegateOptions, ): Promise { + if (!options) options = {}; + return getCompressedTokenAccountsByOwnerOrDelegate( this, delegate, @@ -743,14 +766,18 @@ export class Rpc extends Connection implements CompressionApiInterface { */ async getCompressedTokenBalancesByOwner( owner: PublicKey, - options: GetCompressedTokenAccountsByOwnerOrDelegateOptions, + options?: GetCompressedTokenAccountsByOwnerOrDelegateOptions, ): Promise<{ balance: BN; mint: PublicKey }[]> { + if (!options) options = {}; + const unsafeRes = await rpcRequest( this.compressionApiEndpoint, 'getCompressedTokenBalancesByOwner', { owner: owner.toBase58(), - mint: options.mint.toBase58(), + mint: options.mint?.toBase58(), + limit: options.limit, + cursor: options.cursor, }, ); @@ -770,28 +797,29 @@ export class Rpc extends Connection implements CompressionApiInterface { ); } - /// filter by mint - const filtered = res.result.value.tokenBalances.filter( - tokenBalance => - tokenBalance.mint.toBase58() === options.mint.toBase58(), - ); + const maybeFiltered = options.mint + ? res.result.value.tokenBalances.filter( + tokenBalance => + tokenBalance.mint.toBase58() === options.mint!.toBase58(), + ) + : res.result.value.tokenBalances; - return filtered; + return maybeFiltered; } /** - * Returns confirmed signatures for transactions involving the specified + * Returns confirmed compression signatures for transactions involving the specified * account hash forward in time from genesis to the most recent confirmed * block * * @param hash queried account hash */ - async getSignaturesForCompressedAccount( + async getCompressionSignaturesForAccount( hash: BN254, ): Promise { const unsafeRes = await rpcRequest( this.compressionApiEndpoint, - 'getCompressionSignaturesForAccount', // TODO: update + 'getCompressionSignaturesForAccount', { hash: encodeBN254toBase58(hash) }, ); const res = create( @@ -855,9 +883,11 @@ export class Rpc extends Connection implements CompressionApiInterface { } /** + * @deprecated This method is currently not available. Please use + * {@link getCompressionSignaturesForAccount} instead. + * * Returns confirmed signatures for transactions involving the specified - * address forward in time from genesis to the most recent confirmed - * block + * address forward in time from genesis to the most recent confirmed block * * @param address queried compressed account address */ @@ -924,7 +954,7 @@ export class Rpc extends Connection implements CompressionApiInterface { return res.result.value.items; } - /// TODO: needs mint + /// TODO(photon): needs mint /** * Returns confirmed signatures for compression transactions involving the * specified token account owner forward in time from genesis to the most @@ -988,6 +1018,55 @@ export class Rpc extends Connection implements CompressionApiInterface { return res.result; } + /** + * Fetch the latest compression signatures on the cluster. Results are + * paginated. + */ + async getLatestCompressionSignatures( + cursor?: string, + limit?: number, + ): Promise { + const unsafeRes = await rpcRequest( + this.compressionApiEndpoint, + 'getLatestCompressionSignatures', + { limit, cursor }, + ); + const res = create( + unsafeRes, + jsonRpcResultAndContext(LatestNonVotingSignaturesResultPaginated), + ); + if ('error' in res) { + throw new SolanaJSONRPCError( + res.error, + 'failed to get latest non-voting signatures', + ); + } + return res.result; + } + /** + * Fetch all non-voting signatures + */ + async getLatestNonVotingSignatures( + limit?: number, + ): Promise { + const unsafeRes = await rpcRequest( + this.compressionApiEndpoint, + 'getLatestNonVotingSignatures', + { limit }, + ); + const res = create( + unsafeRes, + jsonRpcResultAndContext(LatestNonVotingSignaturesResult), + ); + if ('error' in res) { + throw new SolanaJSONRPCError( + res.error, + 'failed to get latest non-voting signatures', + ); + } + return res.result; + } + /** * Fetch the latest address proofs for new unique addresses specified by an * array of addresses. @@ -1041,6 +1120,9 @@ export class Rpc extends Connection implements CompressionApiInterface { } /** + * @deprecated This method is not available. Please use + * {@link getValidityProof} instead. + * * Fetch the latest validity proof for (1) compressed accounts specified by * an array of account hashes. (2) new unique addresses specified by an * array of addresses. @@ -1112,8 +1194,7 @@ export class Rpc extends Connection implements CompressionApiInterface { validityProof = { compressedProof, roots: newAddressProofs.map(proof => proof.root), - // TODO(crank): make dynamic to enable forester support in - // test-rpc.ts. Currently this is a static root because the + // This is a static root because the // address tree doesn't advance. rootIndices: newAddressProofs.map(_ => 3), leafIndices: newAddressProofs.map( @@ -1152,8 +1233,7 @@ export class Rpc extends Connection implements CompressionApiInterface { .concat(newAddressProofs.map(proof => proof.root)), rootIndices: merkleProofsWithContext .map(proof => proof.rootIndex) - // TODO(crank): make dynamic to enable forester support in - // test-rpc.ts. Currently this is a static root because the + // This is a static root because the // address tree doesn't advance. .concat(newAddressProofs.map(_ => 3)), leafIndices: merkleProofsWithContext @@ -1194,8 +1274,6 @@ export class Rpc extends Connection implements CompressionApiInterface { * @param newAddresses Array of BN254 new addresses. * @returns validity proof with context */ - // FIXME: debug photon zkp. For debugging use either - // testRpc.getValidityProof or rpc.getValidityProof to test against async getValidityProof( hashes: BN254[] = [], newAddresses: BN254[] = [], diff --git a/js/stateless.js/src/test-helpers/test-rpc/test-rpc.ts b/js/stateless.js/src/test-helpers/test-rpc/test-rpc.ts index e39ffe9f36..8f74ce1daa 100644 --- a/js/stateless.js/src/test-helpers/test-rpc/test-rpc.ts +++ b/js/stateless.js/src/test-helpers/test-rpc/test-rpc.ts @@ -14,10 +14,10 @@ import { import { MerkleTree } from '../merkle-tree/merkle-tree'; import { getParsedEvents } from './get-parsed-events'; import { defaultTestStateTreeAccounts } from '../../constants'; -import { toHex } from '../../utils/conversion'; import { CompressedTransaction, - HexInputsForProver, + LatestNonVotingSignatures, + LatestNonVotingSignaturesPaginated, SignatureWithMetadata, } from '../../rpc-interface'; import { @@ -29,12 +29,10 @@ import { import { BN254, CompressedAccountWithMerkleContext, - CompressedProof, MerkleContextWithMerkleProof, PublicTransactionEvent, bn, } from '../../state'; -import { proofFromJsonStruct, negateAndCompressProof } from '../../utils'; import { IndexedArray } from '../merkle-tree'; import { MerkleContextWithNewAddressProof, @@ -195,8 +193,15 @@ export class TestRpc extends Connection implements CompressionApiInterface { * Fetch the compressed account for the specified account hash */ async getCompressedAccount( - hash: BN254, + address?: BN254, + hash?: BN254, ): Promise { + if (address) { + throw new Error('address is not supported in test-rpc'); + } + if (!hash) { + throw new Error('hash is required'); + } const account = await getCompressedAccountByHashTest(this, hash); return account ?? null; } @@ -204,7 +209,13 @@ export class TestRpc extends Connection implements CompressionApiInterface { /** * Fetch the compressed balance for the specified account hash */ - async getCompressedBalance(hash: BN254): Promise { + async getCompressedBalance(address?: BN254, hash?: BN254): Promise { + if (address) { + throw new Error('address is not supported in test-rpc'); + } + if (!hash) { + throw new Error('hash is required'); + } const account = await getCompressedAccountByHashTest(this, hash); if (!account) { throw new Error('Account not found'); @@ -321,6 +332,29 @@ export class TestRpc extends Connection implements CompressionApiInterface { return accounts; } + /** + * Fetch the latest compression signatures on the cluster. Results are + * paginated. + */ + async getLatestCompressionSignatures( + _cursor?: string, + _limit?: number, + ): Promise { + throw new Error( + 'getLatestNonVotingSignaturesWithContext not supported in test-rpc', + ); + } + /** + * Fetch the latest non-voting signatures on the cluster. Results are + * not paginated. + */ + async getLatestNonVotingSignatures( + _limit?: number, + ): Promise { + throw new Error( + 'getLatestNonVotingSignaturesWithContext not supported in test-rpc', + ); + } /** * Fetch all the compressed token accounts owned by the specified public * key. Owner can be a program or user account @@ -386,11 +420,11 @@ export class TestRpc extends Connection implements CompressionApiInterface { * * @param hash queried account hash */ - async getSignaturesForCompressedAccount( + async getCompressionSignaturesForAccount( hash: BN254, ): Promise { throw new Error( - 'getSignaturesForCompressedAccount not implemented in test-rpc', + 'getCompressionSignaturesForAccount not implemented in test-rpc', ); } @@ -412,7 +446,7 @@ export class TestRpc extends Connection implements CompressionApiInterface { * @param address queried compressed account address */ async getCompressionSignaturesForAddress( - address: PublicKey, + _address: PublicKey, ): Promise { throw new Error('getSignaturesForAddress3 not implemented'); } @@ -518,7 +552,10 @@ export class TestRpc extends Connection implements CompressionApiInterface { return newAddressProofs; } - /// TODO: remove once photon 'getValidityProof' is fixed + /** + * @deprecated This method is not available. Please use + * {@link getValidityProof} instead. + */ async getValidityProof_direct( hashes: BN254[] = [], newAddresses: BN254[] = [], diff --git a/js/stateless.js/tests/e2e/rpc-interop.test.ts b/js/stateless.js/tests/e2e/rpc-interop.test.ts index 50059b1310..3408314fd9 100644 --- a/js/stateless.js/tests/e2e/rpc-interop.test.ts +++ b/js/stateless.js/tests/e2e/rpc-interop.test.ts @@ -1,5 +1,5 @@ -import { describe, it, assert, beforeAll } from 'vitest'; -import { Signer } from '@solana/web3.js'; +import { describe, it, assert, beforeAll, expect } from 'vitest'; +import { PublicKey, Signer } from '@solana/web3.js'; import { newAccountWithLamports } from '../../src/utils/test-utils'; import { Rpc, createRpc } from '../../src/rpc'; import { @@ -8,11 +8,13 @@ import { compress, createAccount, createAccountWithLamports, + defaultTestStateTreeAccounts, deriveAddress, } from '../../src'; import { getTestRpc, TestRpc } from '../../src/test-helpers/test-rpc'; import { transfer } from '../../src/actions/transfer'; import { WasmFactory } from '@lightprotocol/hasher.rs'; +import { randomBytes } from 'tweetnacl'; describe('rpc-interop', () => { let payer: Signer; @@ -451,9 +453,11 @@ describe('rpc-interop', () => { ); const compressedAccount = await rpc.getCompressedAccount( + undefined, bn(senderAccounts[0].hash), ); const compressedAccountTest = await testRpc.getCompressedAccount( + undefined, bn(senderAccounts[0].hash), ); @@ -498,11 +502,11 @@ describe('rpc-interop', () => { }); }); - it('[test-rpc missing] getSignaturesForCompressedAccount should match', async () => { + it('[test-rpc missing] getCompressionSignaturesForAccount should match', async () => { const senderAccounts = await rpc.getCompressedAccountsByOwner( payer.publicKey, ); - const signaturesUnspent = await rpc.getSignaturesForCompressedAccount( + const signaturesUnspent = await rpc.getCompressionSignaturesForAccount( bn(senderAccounts[0].hash), ); @@ -513,11 +517,11 @@ describe('rpc-interop', () => { const largestAccount = senderAccounts.reduce((acc, account) => account.lamports.gt(acc.lamports) ? account : acc, ); - /// assert for Rpc once getValidityProof is working again. - await transfer(testRpc, payer, 1, payer, bob.publicKey); + + await transfer(rpc, payer, 1, payer, bob.publicKey); executedTxs++; - const signaturesSpent = await rpc.getSignaturesForCompressedAccount( + const signaturesSpent = await rpc.getCompressionSignaturesForAccount( bn(largestAccount.hash), ); @@ -532,7 +536,38 @@ describe('rpc-interop', () => { assert.equal(signatures.length, executedTxs); }); - /// TODO: add getCompressedTransaction, getSignaturesForAddress3 + it('[test-rpc missing] getLatestNonVotingSignatures should match', async () => { + const testEnvSetupTxs = 2; + + let signatures = (await rpc.getLatestNonVotingSignatures()).value.items; + assert.isAtLeast(signatures.length, executedTxs + testEnvSetupTxs); + + signatures = (await rpc.getLatestNonVotingSignatures(2)).value.items; + assert.equal(signatures.length, 2); + }); + + it('[test-rpc missing] getLatestCompressionSignatures should match', async () => { + const { items: signatures } = ( + await rpc.getLatestCompressionSignatures() + ).value; + + assert.isAtLeast(signatures.length, executedTxs); + + /// Shoudl return 1 using limit param + const { items: signatures2, cursor } = ( + await rpc.getLatestCompressionSignatures(undefined, 1) + ).value; + + assert.equal(signatures2.length, 1); + + const signatures3 = ( + await rpc.getLatestCompressionSignatures(cursor!, 1) + ).value.items; + + /// cursor should work + assert.notEqual(signatures2[0].signature, signatures3[0].signature); + }); + it('[test-rpc missing] getCompressedTransaction should match', async () => { const signatures = await rpc.getCompressionSignaturesForOwner( payer.publicKey, @@ -546,4 +581,64 @@ describe('rpc-interop', () => { assert.equal(compressedTx?.compressionInfo.closedAccounts.length, 1); assert.equal(compressedTx?.compressionInfo.openedAccounts.length, 2); }); + + // TODO(photon): Fix 'internal server error' + it.skip('[test-rpc missing] getCompressionSignaturesForAddress should work', async () => { + const seed = new Uint8Array(randomBytes(32)); + const addressTree = defaultTestStateTreeAccounts().addressTree; + const address = await deriveAddress(seed, addressTree); + + await createAccount(rpc, payer, seed, LightSystemProgram.programId); + + // fetch the owners latest account + const accounts = await rpc.getCompressedAccountsByOwner( + payer.publicKey, + ); + const latestAccount = accounts[0]; + + // assert the address was indexed + assert.isTrue(new PublicKey(latestAccount.address!).equals(address)); + + const signaturesUnspent = await rpc.getCompressionSignaturesForAddress( + new PublicKey(latestAccount.address!), + ); + + /// most recent therefore unspent account + assert.equal(signaturesUnspent.length, 1); + }); + + it('getCompressedAccount with address param should work ', async () => { + const seed = new Uint8Array(randomBytes(32)); + const addressTree = defaultTestStateTreeAccounts().addressTree; + const address = await deriveAddress(seed, addressTree); + + await createAccount(rpc, payer, seed, LightSystemProgram.programId); + + // fetch the owners latest account + const accounts = await rpc.getCompressedAccountsByOwner( + payer.publicKey, + ); + const latestAccount = accounts[0]; + + assert.isTrue(new PublicKey(latestAccount.address!).equals(address)); + + const compressedAccountByHash = await rpc.getCompressedAccount( + undefined, + bn(latestAccount.hash), + ); + const compressedAccountByAddress = await rpc.getCompressedAccount( + bn(latestAccount.address!), + undefined, + ); + + await expect( + testRpc.getCompressedAccount(bn(latestAccount.address!), undefined), + ).rejects.toThrow(); + + assert.isTrue( + bn(compressedAccountByHash!.address!).eq( + bn(compressedAccountByAddress!.address!), + ), + ); + }); }); diff --git a/js/stateless.js/tests/e2e/test-rpc.test.ts b/js/stateless.js/tests/e2e/test-rpc.test.ts index 67f9764883..9487bb0135 100644 --- a/js/stateless.js/tests/e2e/test-rpc.test.ts +++ b/js/stateless.js/tests/e2e/test-rpc.test.ts @@ -150,7 +150,10 @@ describe('test-rpc', () => { ); const refHash = compressedAccounts[0].hash; /// getCompressedAccount - const compressedAccount = await rpc.getCompressedAccount(bn(refHash)); + const compressedAccount = await rpc.getCompressedAccount( + undefined, + bn(refHash), + ); assert(compressedAccount !== null); assert.equal( compressedAccount.owner.toBase58(), @@ -165,7 +168,15 @@ describe('test-rpc', () => { ); const refHash = compressedAccounts[0].hash; /// getCompressedBalance - const compressedBalance = await rpc.getCompressedBalance(bn(refHash)); + await expect(rpc.getCompressedBalance(bn(refHash))).rejects.toThrow( + 'address is not supported in test-rpc', + ); + + const compressedBalance = await rpc.getCompressedBalance( + undefined, + bn(refHash), + ); + expect(compressedBalance?.eq(bn(refCompressLamports))).toBeTruthy(); }); });