@atomiqlabs/chain-solana 13.5.13 → 13.5.14

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (131) hide show
  1. package/LICENSE +201 -201
  2. package/README.md +73 -73
  3. package/dist/index.d.ts +81 -81
  4. package/dist/index.js +102 -102
  5. package/dist/node/index.d.ts +9 -9
  6. package/dist/node/index.js +13 -13
  7. package/dist/solana/SolanaChainType.d.ts +15 -15
  8. package/dist/solana/SolanaChainType.js +2 -2
  9. package/dist/solana/SolanaChains.d.ts +12 -12
  10. package/dist/solana/SolanaChains.js +45 -45
  11. package/dist/solana/SolanaInitializer.d.ts +94 -94
  12. package/dist/solana/SolanaInitializer.js +174 -174
  13. package/dist/solana/btcrelay/SolanaBtcRelay.d.ts +222 -222
  14. package/dist/solana/btcrelay/SolanaBtcRelay.js +455 -455
  15. package/dist/solana/btcrelay/headers/SolanaBtcHeader.d.ts +84 -84
  16. package/dist/solana/btcrelay/headers/SolanaBtcHeader.js +70 -70
  17. package/dist/solana/btcrelay/headers/SolanaBtcStoredHeader.d.ts +92 -92
  18. package/dist/solana/btcrelay/headers/SolanaBtcStoredHeader.js +109 -109
  19. package/dist/solana/btcrelay/program/programIdl.json +671 -671
  20. package/dist/solana/chain/SolanaAction.d.ts +26 -26
  21. package/dist/solana/chain/SolanaAction.js +87 -87
  22. package/dist/solana/chain/SolanaChainInterface.d.ts +224 -224
  23. package/dist/solana/chain/SolanaChainInterface.js +275 -275
  24. package/dist/solana/chain/SolanaModule.d.ts +14 -14
  25. package/dist/solana/chain/SolanaModule.js +13 -13
  26. package/dist/solana/chain/modules/SolanaAddresses.d.ts +8 -8
  27. package/dist/solana/chain/modules/SolanaAddresses.js +22 -22
  28. package/dist/solana/chain/modules/SolanaBlocks.d.ts +32 -32
  29. package/dist/solana/chain/modules/SolanaBlocks.js +78 -78
  30. package/dist/solana/chain/modules/SolanaEvents.d.ts +68 -68
  31. package/dist/solana/chain/modules/SolanaEvents.js +238 -238
  32. package/dist/solana/chain/modules/SolanaFees.d.ts +189 -189
  33. package/dist/solana/chain/modules/SolanaFees.js +434 -434
  34. package/dist/solana/chain/modules/SolanaSignatures.d.ts +24 -24
  35. package/dist/solana/chain/modules/SolanaSignatures.js +39 -39
  36. package/dist/solana/chain/modules/SolanaSlots.d.ts +33 -33
  37. package/dist/solana/chain/modules/SolanaSlots.js +72 -72
  38. package/dist/solana/chain/modules/SolanaTokens.d.ts +123 -123
  39. package/dist/solana/chain/modules/SolanaTokens.js +242 -242
  40. package/dist/solana/chain/modules/SolanaTransactions.d.ts +149 -149
  41. package/dist/solana/chain/modules/SolanaTransactions.js +445 -445
  42. package/dist/solana/connection/ConnectionWithRetries.d.ts +35 -35
  43. package/dist/solana/connection/ConnectionWithRetries.js +86 -71
  44. package/dist/solana/events/SolanaChainEvents.d.ts +45 -45
  45. package/dist/solana/events/SolanaChainEvents.js +108 -108
  46. package/dist/solana/events/SolanaChainEventsBrowser.d.ts +205 -205
  47. package/dist/solana/events/SolanaChainEventsBrowser.js +404 -404
  48. package/dist/solana/program/SolanaProgramBase.d.ts +73 -73
  49. package/dist/solana/program/SolanaProgramBase.js +54 -54
  50. package/dist/solana/program/SolanaProgramModule.d.ts +8 -8
  51. package/dist/solana/program/SolanaProgramModule.js +11 -11
  52. package/dist/solana/program/modules/SolanaProgramEvents.d.ts +53 -53
  53. package/dist/solana/program/modules/SolanaProgramEvents.js +117 -117
  54. package/dist/solana/swaps/SolanaSwapData.d.ts +333 -333
  55. package/dist/solana/swaps/SolanaSwapData.js +535 -535
  56. package/dist/solana/swaps/SolanaSwapModule.d.ts +11 -11
  57. package/dist/solana/swaps/SolanaSwapModule.js +12 -12
  58. package/dist/solana/swaps/SolanaSwapProgram.d.ts +376 -376
  59. package/dist/solana/swaps/SolanaSwapProgram.js +769 -769
  60. package/dist/solana/swaps/SwapTypeEnum.d.ts +11 -11
  61. package/dist/solana/swaps/SwapTypeEnum.js +43 -43
  62. package/dist/solana/swaps/modules/SolanaDataAccount.d.ts +95 -95
  63. package/dist/solana/swaps/modules/SolanaDataAccount.js +232 -232
  64. package/dist/solana/swaps/modules/SolanaLpVault.d.ts +69 -69
  65. package/dist/solana/swaps/modules/SolanaLpVault.js +171 -171
  66. package/dist/solana/swaps/modules/SwapClaim.d.ts +126 -126
  67. package/dist/solana/swaps/modules/SwapClaim.js +294 -294
  68. package/dist/solana/swaps/modules/SwapInit.d.ts +213 -213
  69. package/dist/solana/swaps/modules/SwapInit.js +658 -658
  70. package/dist/solana/swaps/modules/SwapRefund.d.ts +87 -87
  71. package/dist/solana/swaps/modules/SwapRefund.js +293 -293
  72. package/dist/solana/swaps/programIdl.json +945 -945
  73. package/dist/solana/swaps/programTypes.d.ts +943 -943
  74. package/dist/solana/swaps/programTypes.js +945 -945
  75. package/dist/solana/swaps/v1/programIdl.json +945 -945
  76. package/dist/solana/swaps/v1/programTypes.d.ts +943 -943
  77. package/dist/solana/swaps/v1/programTypes.js +945 -945
  78. package/dist/solana/swaps/v2/programIdl.json +952 -952
  79. package/dist/solana/swaps/v2/programTypes.d.ts +950 -950
  80. package/dist/solana/swaps/v2/programTypes.js +952 -952
  81. package/dist/solana/wallet/SolanaKeypairWallet.d.ts +29 -29
  82. package/dist/solana/wallet/SolanaKeypairWallet.js +50 -50
  83. package/dist/solana/wallet/SolanaSigner.d.ts +30 -30
  84. package/dist/solana/wallet/SolanaSigner.js +30 -30
  85. package/dist/utils/Utils.d.ts +58 -58
  86. package/dist/utils/Utils.js +170 -170
  87. package/node/index.d.ts +1 -1
  88. package/node/index.js +3 -3
  89. package/package.json +46 -46
  90. package/src/index.ts +87 -87
  91. package/src/node/index.ts +9 -9
  92. package/src/solana/SolanaChainType.ts +32 -32
  93. package/src/solana/SolanaChains.ts +46 -46
  94. package/src/solana/SolanaInitializer.ts +278 -278
  95. package/src/solana/btcrelay/SolanaBtcRelay.ts +615 -615
  96. package/src/solana/btcrelay/headers/SolanaBtcHeader.ts +116 -116
  97. package/src/solana/btcrelay/headers/SolanaBtcStoredHeader.ts +148 -148
  98. package/src/solana/btcrelay/program/programIdl.json +670 -670
  99. package/src/solana/chain/SolanaAction.ts +109 -109
  100. package/src/solana/chain/SolanaChainInterface.ts +404 -404
  101. package/src/solana/chain/SolanaModule.ts +20 -20
  102. package/src/solana/chain/modules/SolanaAddresses.ts +20 -20
  103. package/src/solana/chain/modules/SolanaBlocks.ts +89 -89
  104. package/src/solana/chain/modules/SolanaEvents.ts +271 -271
  105. package/src/solana/chain/modules/SolanaFees.ts +522 -522
  106. package/src/solana/chain/modules/SolanaSignatures.ts +39 -39
  107. package/src/solana/chain/modules/SolanaSlots.ts +85 -85
  108. package/src/solana/chain/modules/SolanaTokens.ts +300 -300
  109. package/src/solana/chain/modules/SolanaTransactions.ts +503 -503
  110. package/src/solana/connection/ConnectionWithRetries.ts +113 -96
  111. package/src/solana/events/SolanaChainEvents.ts +127 -127
  112. package/src/solana/events/SolanaChainEventsBrowser.ts +495 -495
  113. package/src/solana/program/SolanaProgramBase.ts +119 -119
  114. package/src/solana/program/SolanaProgramModule.ts +15 -15
  115. package/src/solana/program/modules/SolanaProgramEvents.ts +157 -157
  116. package/src/solana/swaps/SolanaSwapData.ts +735 -735
  117. package/src/solana/swaps/SolanaSwapModule.ts +19 -19
  118. package/src/solana/swaps/SolanaSwapProgram.ts +1074 -1074
  119. package/src/solana/swaps/SwapTypeEnum.ts +30 -30
  120. package/src/solana/swaps/modules/SolanaDataAccount.ts +302 -302
  121. package/src/solana/swaps/modules/SolanaLpVault.ts +208 -208
  122. package/src/solana/swaps/modules/SwapClaim.ts +387 -387
  123. package/src/solana/swaps/modules/SwapInit.ts +785 -785
  124. package/src/solana/swaps/modules/SwapRefund.ts +353 -353
  125. package/src/solana/swaps/v1/programIdl.json +944 -944
  126. package/src/solana/swaps/v1/programTypes.ts +1885 -1885
  127. package/src/solana/swaps/v2/programIdl.json +951 -951
  128. package/src/solana/swaps/v2/programTypes.ts +1899 -1899
  129. package/src/solana/wallet/SolanaKeypairWallet.ts +56 -56
  130. package/src/solana/wallet/SolanaSigner.ts +43 -43
  131. package/src/utils/Utils.ts +194 -194
@@ -1,272 +1,272 @@
1
- import {SolanaModule} from "../SolanaModule";
2
- import {ConfirmedSignatureInfo, ParsedTransactionWithMeta, PublicKey} from "@solana/web3.js";
3
- import {sign} from "tweetnacl";
4
- import {ProgramEvent} from "../../program/modules/SolanaProgramEvents";
5
- import {tryWithRetries} from "../../../utils/Utils";
6
-
7
- export class SolanaEvents extends SolanaModule {
8
-
9
- public readonly LOG_FETCH_LIMIT = 500;
10
-
11
- private usingHeliusTFA: "yes" | "no" | "auto" = "auto";
12
-
13
- /**
14
- * Gets the signatures for a given topicKey public key, if lastProcessedSignature is specified, it fetches only
15
- * the signatures before this signature
16
- *
17
- * @param topicKey
18
- * @param logFetchLimit
19
- * @param lastProcessedSignature
20
- * @private
21
- */
22
- private getSignatures(topicKey: PublicKey, logFetchLimit: number, lastProcessedSignature?: string): Promise<ConfirmedSignatureInfo[]> {
23
- if(lastProcessedSignature==null) {
24
- return this.connection.getSignaturesForAddress(topicKey, {
25
- limit: logFetchLimit,
26
- }, "confirmed");
27
- } else {
28
- return this.connection.getSignaturesForAddress(topicKey, {
29
- before: lastProcessedSignature,
30
- limit: logFetchLimit
31
- }, "confirmed");
32
- }
33
- }
34
-
35
- /**
36
- * Implements Helius getTransactionsForAddress RPC API
37
- *
38
- * @param account
39
- * @param options
40
- * @param commitment
41
- */
42
- async getTransactionsForAddress(
43
- account: PublicKey,
44
- options?: {
45
- paginationToken?: string,
46
- filters?: {
47
- slot?: {gte?: number, lte?: number, gt?: number, lt?: number},
48
- blockTime?: {gte?: number, lte?: number, gt?: number, lt?: number, eq?: number},
49
- signature?: {gte?: number, lte?: number, gt?: number, lt?: number, eq?: number},
50
- status?: "succeeded" | "failed" | "any"
51
- }
52
- },
53
- commitment?: "finalized" | "confirmed" | "processed"
54
- ): Promise<{
55
- data: ParsedTransactionWithMeta[],
56
- paginationToken?: string
57
- } | null> {
58
- const limit = 100;
59
-
60
- //Try to use getPriorityFeeEstimate api of Helius
61
- const response = await (this.connection as any)._rpcRequest("getTransactionsForAddress", [
62
- account.toString(),
63
- {
64
- ...options,
65
- transactionDetails: "full",
66
- sortOrder: "desc",
67
- limit,
68
- commitment: commitment ?? "confirmed",
69
- encoding: "jsonParsed",
70
- maxSupportedTransactionVersion: 0
71
- }
72
- ]).catch((e: any) => {
73
- //Catching not supported errors
74
- if(e.message!=null && (e.message.includes("-32601") || e.message.includes("-32600") || e.message.includes("-32403"))) {
75
- return {
76
- error: {
77
- code: -32601,
78
- message: e.message
79
- }
80
- };
81
- }
82
- throw e;
83
- });
84
-
85
- if(response.error!=null) {
86
- //Catching not supported errors
87
- if(response.error.code!==-32601 && response.error.code!==-32600 && response.error.code!==-32403) throw new Error(response.error.message);
88
- return null;
89
- }
90
-
91
- return {
92
- data: response.result.data.map((val: any) => {
93
- return {
94
- ...val, //slot, blockTime, version
95
- meta: val.meta==null ? undefined : {
96
- //ParsedTransactionMeta
97
- ...val.meta,
98
- innerInstructions: val.meta.innerInstructions==null ? undefined : val.meta.innerInstructions.map((innerIx: any) => ({
99
- //ParsedInnerInstruction
100
- ...innerIx, //index
101
- instructions: innerIx.instructions.map((ix: any) => {
102
- if(ix.program!=null && ix.programId!=null) {
103
- return {
104
- //ParsedInstruction
105
- ...ix,
106
- programId: new PublicKey(ix.programId)
107
- }
108
- } else {
109
- return {
110
- //PartiallyDecodedInstruction
111
- data: ix.data,
112
- programId: new PublicKey(ix.programId),
113
- accounts: ix.accounts.map((pubkey: string) => new PublicKey(pubkey))
114
- }
115
- }
116
- })
117
- })),
118
- loadedAddresses: val.meta.loadedAddresses==null ? undefined : {
119
- writable: val.meta.loadedAddresses.writable.map((pubkey: string) => new PublicKey(pubkey)),
120
- readonly: val.meta.loadedAddresses.readonly.map((pubkey: string) => new PublicKey(pubkey)),
121
- }
122
- },
123
- transaction: {
124
- //ParsedTransaction
125
- ...val.transaction, //signatures
126
- message: {
127
- //ParsedMessage
128
- ...val.transaction.message, //recentBlockhash
129
- accountKeys: val.transaction.message.accountKeys.map((accountKey: any) => ({
130
- //ParsedMessageAccount
131
- ...accountKey,
132
- pubkey: new PublicKey(accountKey.pubkey)
133
- })),
134
- instructions: val.transaction.message.instructions.map((ix: any) => {
135
- if(ix.program!=null && ix.programId!=null) {
136
- return {
137
- //ParsedInstruction
138
- ...ix,
139
- programId: new PublicKey(ix.programId)
140
- }
141
- } else {
142
- return {
143
- //PartiallyDecodedInstruction
144
- data: ix.data,
145
- programId: new PublicKey(ix.programId),
146
- accounts: ix.accounts.map((pubkey: string) => new PublicKey(pubkey))
147
- }
148
- }
149
- }),
150
- addressTableLookups: val.transaction.message.addressTableLookups==null
151
- ? undefined
152
- : val.transaction.message.addressTableLookups.map((addressTableLookup: any) => ({
153
- //ParsedAddressTableLookup
154
- ...addressTableLookup,
155
- accountKey: new PublicKey(addressTableLookup.accountKey)
156
- }))
157
- }
158
- }
159
- }
160
- }),
161
- paginationToken: response.result.paginationToken
162
- };
163
- }
164
-
165
- private async _findInTxsTFA<T>(
166
- topicKey: PublicKey,
167
- processor: (data: {signatures?: ConfirmedSignatureInfo[], txs?: ParsedTransactionWithMeta[]}) => Promise<T>,
168
- abortSignal?: AbortSignal,
169
- startBlockheight?: number
170
- ): Promise<T | undefined | null> {
171
- let paginationToken: string | undefined;
172
- let txs: ParsedTransactionWithMeta[] | undefined;
173
- while(txs==null || txs.length>0) {
174
- let filters = startBlockheight!=null ? {
175
- slot: {gte: startBlockheight}
176
- } : {};
177
- const tfaResult = await tryWithRetries(
178
- () => this.getTransactionsForAddress(topicKey, {
179
- paginationToken,
180
- filters: {
181
- ...filters,
182
- status: "succeeded"
183
- }
184
- }, "confirmed"),
185
- undefined, undefined, abortSignal
186
- );
187
-
188
- if(tfaResult==null) {
189
- //Not supported
190
- return undefined;
191
- }
192
-
193
- txs = tfaResult.data;
194
- paginationToken = tfaResult.paginationToken;
195
-
196
- if(txs.length===0) {
197
- this.logger.debug(`_findInTxsTFA(): Got ${txs.length} txns (empty response), paginationToken: ${paginationToken}`);
198
- } else {
199
- this.logger.debug(`_findInTxsTFA(): Got ${txs.length} txns (${txs[0].transaction.signatures[0]}..${txs[txs.length-1].transaction.signatures[0]}), paginationToken: ${paginationToken}`);
200
- }
201
-
202
- if(abortSignal!=null) abortSignal.throwIfAborted();
203
- const result: T = await processor({txs});
204
- if(result!=null) return result;
205
- if(paginationToken==null) break;
206
- }
207
- return null;
208
-
209
- }
210
-
211
- /**
212
- * Runs a search backwards in time, processing transaction signatures for a specific topic public key
213
- *
214
- * @param topicKey
215
- * @param processor called for every batch of returned signatures, should return a value if the correct signature
216
- * was found, or null if the search should continue
217
- * @param abortSignal
218
- * @param logFetchLimit
219
- * @param startBlockheight
220
- */
221
- private async _findInSignatures<T>(
222
- topicKey: PublicKey,
223
- processor: (data: {signatures?: ConfirmedSignatureInfo[], txs?: ParsedTransactionWithMeta[]}) => Promise<T | null | undefined>,
224
- abortSignal?: AbortSignal,
225
- logFetchLimit?: number,
226
- startBlockheight?: number
227
- ): Promise<T | null> {
228
- if(logFetchLimit==null || logFetchLimit>this.LOG_FETCH_LIMIT) logFetchLimit = this.LOG_FETCH_LIMIT;
229
- let signatures: ConfirmedSignatureInfo[] | undefined;
230
- do {
231
- signatures = await this.getSignatures(topicKey, logFetchLimit, signatures!=null ? signatures?.[signatures.length-1].signature : undefined);
232
- if(startBlockheight!=null) {
233
- const endIndex = signatures.findIndex(val => val.slot < startBlockheight);
234
- if(endIndex===0) return null;
235
- if(endIndex!==-1) signatures = signatures.slice(0, endIndex - 1);
236
- }
237
-
238
- if(signatures.length===0) {
239
- this.logger.debug(`_findInSignatures(): Got ${signatures.length} txns (empty response)`);
240
- } else {
241
- this.logger.debug(`_findInSignatures(): Got ${signatures.length} txns (${signatures[0].signature}..${signatures[signatures.length-1].signature})`);
242
- }
243
-
244
- if(abortSignal!=null) abortSignal.throwIfAborted();
245
- const result = await processor({signatures});
246
- if(result!=null) return result;
247
- } while(signatures.length>=logFetchLimit); //Only fetch next one if this response is full
248
- return null;
249
- }
250
-
251
- public async findInSignatures<T>(
252
- topicKey: PublicKey,
253
- processor: (data: {signatures?: ConfirmedSignatureInfo[], txs?: ParsedTransactionWithMeta[]}) => Promise<T | undefined | null>,
254
- abortSignal?: AbortSignal,
255
- logFetchLimit?: number,
256
- startBlockheight?: number
257
- ): Promise<T | null> {
258
- if(this.usingHeliusTFA!=="no") {
259
- //Attempt to use Helius's gTFA
260
- const result = await this._findInTxsTFA(topicKey, processor, abortSignal, startBlockheight);
261
- if(result!==undefined) return result;
262
-
263
- //Not supported
264
- if(this.usingHeliusTFA==="yes") throw new Error("Helius gTFA is not supported with current provider!");
265
- //If set to auto, we can manually set to "no"
266
- this.usingHeliusTFA = "no";
267
- this.logger.warn("findInSignatures(): Helius gTFA is not supported, switching back to using gSFA!")
268
- }
269
- return await this._findInSignatures(topicKey, processor, abortSignal, logFetchLimit, startBlockheight);
270
- }
271
-
1
+ import {SolanaModule} from "../SolanaModule";
2
+ import {ConfirmedSignatureInfo, ParsedTransactionWithMeta, PublicKey} from "@solana/web3.js";
3
+ import {sign} from "tweetnacl";
4
+ import {ProgramEvent} from "../../program/modules/SolanaProgramEvents";
5
+ import {tryWithRetries} from "../../../utils/Utils";
6
+
7
+ export class SolanaEvents extends SolanaModule {
8
+
9
+ public readonly LOG_FETCH_LIMIT = 500;
10
+
11
+ private usingHeliusTFA: "yes" | "no" | "auto" = "auto";
12
+
13
+ /**
14
+ * Gets the signatures for a given topicKey public key, if lastProcessedSignature is specified, it fetches only
15
+ * the signatures before this signature
16
+ *
17
+ * @param topicKey
18
+ * @param logFetchLimit
19
+ * @param lastProcessedSignature
20
+ * @private
21
+ */
22
+ private getSignatures(topicKey: PublicKey, logFetchLimit: number, lastProcessedSignature?: string): Promise<ConfirmedSignatureInfo[]> {
23
+ if(lastProcessedSignature==null) {
24
+ return this.connection.getSignaturesForAddress(topicKey, {
25
+ limit: logFetchLimit,
26
+ }, "confirmed");
27
+ } else {
28
+ return this.connection.getSignaturesForAddress(topicKey, {
29
+ before: lastProcessedSignature,
30
+ limit: logFetchLimit
31
+ }, "confirmed");
32
+ }
33
+ }
34
+
35
+ /**
36
+ * Implements Helius getTransactionsForAddress RPC API
37
+ *
38
+ * @param account
39
+ * @param options
40
+ * @param commitment
41
+ */
42
+ async getTransactionsForAddress(
43
+ account: PublicKey,
44
+ options?: {
45
+ paginationToken?: string,
46
+ filters?: {
47
+ slot?: {gte?: number, lte?: number, gt?: number, lt?: number},
48
+ blockTime?: {gte?: number, lte?: number, gt?: number, lt?: number, eq?: number},
49
+ signature?: {gte?: number, lte?: number, gt?: number, lt?: number, eq?: number},
50
+ status?: "succeeded" | "failed" | "any"
51
+ }
52
+ },
53
+ commitment?: "finalized" | "confirmed" | "processed"
54
+ ): Promise<{
55
+ data: ParsedTransactionWithMeta[],
56
+ paginationToken?: string
57
+ } | null> {
58
+ const limit = 100;
59
+
60
+ //Try to use getPriorityFeeEstimate api of Helius
61
+ const response = await (this.connection as any)._rpcRequest("getTransactionsForAddress", [
62
+ account.toString(),
63
+ {
64
+ ...options,
65
+ transactionDetails: "full",
66
+ sortOrder: "desc",
67
+ limit,
68
+ commitment: commitment ?? "confirmed",
69
+ encoding: "jsonParsed",
70
+ maxSupportedTransactionVersion: 0
71
+ }
72
+ ]).catch((e: any) => {
73
+ //Catching not supported errors
74
+ if(e.message!=null && (e.message.includes("-32601") || e.message.includes("-32600") || e.message.includes("-32403"))) {
75
+ return {
76
+ error: {
77
+ code: -32601,
78
+ message: e.message
79
+ }
80
+ };
81
+ }
82
+ throw e;
83
+ });
84
+
85
+ if(response.error!=null) {
86
+ //Catching not supported errors
87
+ if(response.error.code!==-32601 && response.error.code!==-32600 && response.error.code!==-32403) throw new Error(response.error.message);
88
+ return null;
89
+ }
90
+
91
+ return {
92
+ data: response.result.data.map((val: any) => {
93
+ return {
94
+ ...val, //slot, blockTime, version
95
+ meta: val.meta==null ? undefined : {
96
+ //ParsedTransactionMeta
97
+ ...val.meta,
98
+ innerInstructions: val.meta.innerInstructions==null ? undefined : val.meta.innerInstructions.map((innerIx: any) => ({
99
+ //ParsedInnerInstruction
100
+ ...innerIx, //index
101
+ instructions: innerIx.instructions.map((ix: any) => {
102
+ if(ix.program!=null && ix.programId!=null) {
103
+ return {
104
+ //ParsedInstruction
105
+ ...ix,
106
+ programId: new PublicKey(ix.programId)
107
+ }
108
+ } else {
109
+ return {
110
+ //PartiallyDecodedInstruction
111
+ data: ix.data,
112
+ programId: new PublicKey(ix.programId),
113
+ accounts: ix.accounts.map((pubkey: string) => new PublicKey(pubkey))
114
+ }
115
+ }
116
+ })
117
+ })),
118
+ loadedAddresses: val.meta.loadedAddresses==null ? undefined : {
119
+ writable: val.meta.loadedAddresses.writable.map((pubkey: string) => new PublicKey(pubkey)),
120
+ readonly: val.meta.loadedAddresses.readonly.map((pubkey: string) => new PublicKey(pubkey)),
121
+ }
122
+ },
123
+ transaction: {
124
+ //ParsedTransaction
125
+ ...val.transaction, //signatures
126
+ message: {
127
+ //ParsedMessage
128
+ ...val.transaction.message, //recentBlockhash
129
+ accountKeys: val.transaction.message.accountKeys.map((accountKey: any) => ({
130
+ //ParsedMessageAccount
131
+ ...accountKey,
132
+ pubkey: new PublicKey(accountKey.pubkey)
133
+ })),
134
+ instructions: val.transaction.message.instructions.map((ix: any) => {
135
+ if(ix.program!=null && ix.programId!=null) {
136
+ return {
137
+ //ParsedInstruction
138
+ ...ix,
139
+ programId: new PublicKey(ix.programId)
140
+ }
141
+ } else {
142
+ return {
143
+ //PartiallyDecodedInstruction
144
+ data: ix.data,
145
+ programId: new PublicKey(ix.programId),
146
+ accounts: ix.accounts.map((pubkey: string) => new PublicKey(pubkey))
147
+ }
148
+ }
149
+ }),
150
+ addressTableLookups: val.transaction.message.addressTableLookups==null
151
+ ? undefined
152
+ : val.transaction.message.addressTableLookups.map((addressTableLookup: any) => ({
153
+ //ParsedAddressTableLookup
154
+ ...addressTableLookup,
155
+ accountKey: new PublicKey(addressTableLookup.accountKey)
156
+ }))
157
+ }
158
+ }
159
+ }
160
+ }),
161
+ paginationToken: response.result.paginationToken
162
+ };
163
+ }
164
+
165
+ private async _findInTxsTFA<T>(
166
+ topicKey: PublicKey,
167
+ processor: (data: {signatures?: ConfirmedSignatureInfo[], txs?: ParsedTransactionWithMeta[]}) => Promise<T>,
168
+ abortSignal?: AbortSignal,
169
+ startBlockheight?: number
170
+ ): Promise<T | undefined | null> {
171
+ let paginationToken: string | undefined;
172
+ let txs: ParsedTransactionWithMeta[] | undefined;
173
+ while(txs==null || txs.length>0) {
174
+ let filters = startBlockheight!=null ? {
175
+ slot: {gte: startBlockheight}
176
+ } : {};
177
+ const tfaResult = await tryWithRetries(
178
+ () => this.getTransactionsForAddress(topicKey, {
179
+ paginationToken,
180
+ filters: {
181
+ ...filters,
182
+ status: "succeeded"
183
+ }
184
+ }, "confirmed"),
185
+ undefined, undefined, abortSignal
186
+ );
187
+
188
+ if(tfaResult==null) {
189
+ //Not supported
190
+ return undefined;
191
+ }
192
+
193
+ txs = tfaResult.data;
194
+ paginationToken = tfaResult.paginationToken;
195
+
196
+ if(txs.length===0) {
197
+ this.logger.debug(`_findInTxsTFA(): Got ${txs.length} txns (empty response), paginationToken: ${paginationToken}`);
198
+ } else {
199
+ this.logger.debug(`_findInTxsTFA(): Got ${txs.length} txns (${txs[0].transaction.signatures[0]}..${txs[txs.length-1].transaction.signatures[0]}), paginationToken: ${paginationToken}`);
200
+ }
201
+
202
+ if(abortSignal!=null) abortSignal.throwIfAborted();
203
+ const result: T = await processor({txs});
204
+ if(result!=null) return result;
205
+ if(paginationToken==null) break;
206
+ }
207
+ return null;
208
+
209
+ }
210
+
211
+ /**
212
+ * Runs a search backwards in time, processing transaction signatures for a specific topic public key
213
+ *
214
+ * @param topicKey
215
+ * @param processor called for every batch of returned signatures, should return a value if the correct signature
216
+ * was found, or null if the search should continue
217
+ * @param abortSignal
218
+ * @param logFetchLimit
219
+ * @param startBlockheight
220
+ */
221
+ private async _findInSignatures<T>(
222
+ topicKey: PublicKey,
223
+ processor: (data: {signatures?: ConfirmedSignatureInfo[], txs?: ParsedTransactionWithMeta[]}) => Promise<T | null | undefined>,
224
+ abortSignal?: AbortSignal,
225
+ logFetchLimit?: number,
226
+ startBlockheight?: number
227
+ ): Promise<T | null> {
228
+ if(logFetchLimit==null || logFetchLimit>this.LOG_FETCH_LIMIT) logFetchLimit = this.LOG_FETCH_LIMIT;
229
+ let signatures: ConfirmedSignatureInfo[] | undefined;
230
+ do {
231
+ signatures = await this.getSignatures(topicKey, logFetchLimit, signatures!=null ? signatures?.[signatures.length-1].signature : undefined);
232
+ if(startBlockheight!=null) {
233
+ const endIndex = signatures.findIndex(val => val.slot < startBlockheight);
234
+ if(endIndex===0) return null;
235
+ if(endIndex!==-1) signatures = signatures.slice(0, endIndex - 1);
236
+ }
237
+
238
+ if(signatures.length===0) {
239
+ this.logger.debug(`_findInSignatures(): Got ${signatures.length} txns (empty response)`);
240
+ } else {
241
+ this.logger.debug(`_findInSignatures(): Got ${signatures.length} txns (${signatures[0].signature}..${signatures[signatures.length-1].signature})`);
242
+ }
243
+
244
+ if(abortSignal!=null) abortSignal.throwIfAborted();
245
+ const result = await processor({signatures});
246
+ if(result!=null) return result;
247
+ } while(signatures.length>=logFetchLimit); //Only fetch next one if this response is full
248
+ return null;
249
+ }
250
+
251
+ public async findInSignatures<T>(
252
+ topicKey: PublicKey,
253
+ processor: (data: {signatures?: ConfirmedSignatureInfo[], txs?: ParsedTransactionWithMeta[]}) => Promise<T | undefined | null>,
254
+ abortSignal?: AbortSignal,
255
+ logFetchLimit?: number,
256
+ startBlockheight?: number
257
+ ): Promise<T | null> {
258
+ if(this.usingHeliusTFA!=="no") {
259
+ //Attempt to use Helius's gTFA
260
+ const result = await this._findInTxsTFA(topicKey, processor, abortSignal, startBlockheight);
261
+ if(result!==undefined) return result;
262
+
263
+ //Not supported
264
+ if(this.usingHeliusTFA==="yes") throw new Error("Helius gTFA is not supported with current provider!");
265
+ //If set to auto, we can manually set to "no"
266
+ this.usingHeliusTFA = "no";
267
+ this.logger.warn("findInSignatures(): Helius gTFA is not supported, switching back to using gSFA!")
268
+ }
269
+ return await this._findInSignatures(topicKey, processor, abortSignal, logFetchLimit, startBlockheight);
270
+ }
271
+
272
272
  }