@atomiqlabs/chain-solana 12.0.11 → 12.0.13

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 (114) hide show
  1. package/LICENSE +201 -201
  2. package/dist/index.d.ts +29 -29
  3. package/dist/index.js +45 -45
  4. package/dist/solana/SolanaChainType.d.ts +11 -11
  5. package/dist/solana/SolanaChainType.js +2 -2
  6. package/dist/solana/SolanaChains.d.ts +20 -20
  7. package/dist/solana/SolanaChains.js +25 -25
  8. package/dist/solana/SolanaInitializer.d.ts +18 -18
  9. package/dist/solana/SolanaInitializer.js +63 -63
  10. package/dist/solana/btcrelay/SolanaBtcRelay.d.ts +228 -228
  11. package/dist/solana/btcrelay/SolanaBtcRelay.js +441 -441
  12. package/dist/solana/btcrelay/headers/SolanaBtcHeader.d.ts +29 -29
  13. package/dist/solana/btcrelay/headers/SolanaBtcHeader.js +34 -34
  14. package/dist/solana/btcrelay/headers/SolanaBtcStoredHeader.d.ts +46 -46
  15. package/dist/solana/btcrelay/headers/SolanaBtcStoredHeader.js +78 -78
  16. package/dist/solana/btcrelay/program/programIdl.json +671 -671
  17. package/dist/solana/chain/SolanaAction.d.ts +26 -26
  18. package/dist/solana/chain/SolanaAction.js +86 -86
  19. package/dist/solana/chain/SolanaChainInterface.d.ts +65 -65
  20. package/dist/solana/chain/SolanaChainInterface.js +125 -125
  21. package/dist/solana/chain/SolanaModule.d.ts +14 -14
  22. package/dist/solana/chain/SolanaModule.js +13 -13
  23. package/dist/solana/chain/modules/SolanaAddresses.d.ts +8 -8
  24. package/dist/solana/chain/modules/SolanaAddresses.js +22 -22
  25. package/dist/solana/chain/modules/SolanaBlocks.d.ts +28 -28
  26. package/dist/solana/chain/modules/SolanaBlocks.js +72 -72
  27. package/dist/solana/chain/modules/SolanaEvents.d.ts +68 -68
  28. package/dist/solana/chain/modules/SolanaEvents.js +225 -225
  29. package/dist/solana/chain/modules/SolanaFees.d.ts +121 -121
  30. package/dist/solana/chain/modules/SolanaFees.js +379 -379
  31. package/dist/solana/chain/modules/SolanaSignatures.d.ts +23 -23
  32. package/dist/solana/chain/modules/SolanaSignatures.js +39 -39
  33. package/dist/solana/chain/modules/SolanaSlots.d.ts +31 -31
  34. package/dist/solana/chain/modules/SolanaSlots.js +68 -68
  35. package/dist/solana/chain/modules/SolanaTokens.d.ts +136 -136
  36. package/dist/solana/chain/modules/SolanaTokens.js +248 -248
  37. package/dist/solana/chain/modules/SolanaTransactions.d.ts +124 -124
  38. package/dist/solana/chain/modules/SolanaTransactions.js +323 -323
  39. package/dist/solana/events/SolanaChainEvents.d.ts +88 -88
  40. package/dist/solana/events/SolanaChainEvents.js +256 -256
  41. package/dist/solana/events/SolanaChainEventsBrowser.d.ts +75 -75
  42. package/dist/solana/events/SolanaChainEventsBrowser.js +172 -172
  43. package/dist/solana/program/SolanaProgramBase.d.ts +40 -40
  44. package/dist/solana/program/SolanaProgramBase.js +43 -43
  45. package/dist/solana/program/SolanaProgramModule.d.ts +8 -8
  46. package/dist/solana/program/SolanaProgramModule.js +11 -11
  47. package/dist/solana/program/modules/SolanaProgramEvents.d.ts +53 -53
  48. package/dist/solana/program/modules/SolanaProgramEvents.js +114 -114
  49. package/dist/solana/swaps/SolanaSwapData.d.ts +71 -71
  50. package/dist/solana/swaps/SolanaSwapData.js +292 -292
  51. package/dist/solana/swaps/SolanaSwapModule.d.ts +10 -10
  52. package/dist/solana/swaps/SolanaSwapModule.js +11 -11
  53. package/dist/solana/swaps/SolanaSwapProgram.d.ts +224 -224
  54. package/dist/solana/swaps/SolanaSwapProgram.js +570 -567
  55. package/dist/solana/swaps/SwapTypeEnum.d.ts +11 -11
  56. package/dist/solana/swaps/SwapTypeEnum.js +42 -42
  57. package/dist/solana/swaps/modules/SolanaDataAccount.d.ts +94 -94
  58. package/dist/solana/swaps/modules/SolanaDataAccount.js +231 -231
  59. package/dist/solana/swaps/modules/SolanaLpVault.d.ts +71 -71
  60. package/dist/solana/swaps/modules/SolanaLpVault.js +173 -173
  61. package/dist/solana/swaps/modules/SwapClaim.d.ts +129 -129
  62. package/dist/solana/swaps/modules/SwapClaim.js +291 -291
  63. package/dist/solana/swaps/modules/SwapInit.d.ts +217 -217
  64. package/dist/solana/swaps/modules/SwapInit.js +519 -519
  65. package/dist/solana/swaps/modules/SwapRefund.d.ts +82 -82
  66. package/dist/solana/swaps/modules/SwapRefund.js +262 -262
  67. package/dist/solana/swaps/programIdl.json +945 -945
  68. package/dist/solana/swaps/programTypes.d.ts +943 -943
  69. package/dist/solana/swaps/programTypes.js +945 -945
  70. package/dist/solana/wallet/SolanaKeypairWallet.d.ts +9 -9
  71. package/dist/solana/wallet/SolanaKeypairWallet.js +33 -33
  72. package/dist/solana/wallet/SolanaSigner.d.ts +11 -11
  73. package/dist/solana/wallet/SolanaSigner.js +17 -17
  74. package/dist/utils/Utils.d.ts +53 -53
  75. package/dist/utils/Utils.js +170 -170
  76. package/package.json +41 -41
  77. package/src/index.ts +36 -36
  78. package/src/solana/SolanaChainType.ts +27 -27
  79. package/src/solana/SolanaChains.ts +23 -23
  80. package/src/solana/SolanaInitializer.ts +102 -102
  81. package/src/solana/btcrelay/SolanaBtcRelay.ts +589 -589
  82. package/src/solana/btcrelay/headers/SolanaBtcHeader.ts +57 -57
  83. package/src/solana/btcrelay/headers/SolanaBtcStoredHeader.ts +102 -102
  84. package/src/solana/btcrelay/program/programIdl.json +670 -670
  85. package/src/solana/chain/SolanaAction.ts +108 -108
  86. package/src/solana/chain/SolanaChainInterface.ts +192 -192
  87. package/src/solana/chain/SolanaModule.ts +20 -20
  88. package/src/solana/chain/modules/SolanaAddresses.ts +20 -20
  89. package/src/solana/chain/modules/SolanaBlocks.ts +78 -78
  90. package/src/solana/chain/modules/SolanaEvents.ts +256 -256
  91. package/src/solana/chain/modules/SolanaFees.ts +450 -450
  92. package/src/solana/chain/modules/SolanaSignatures.ts +39 -39
  93. package/src/solana/chain/modules/SolanaSlots.ts +82 -82
  94. package/src/solana/chain/modules/SolanaTokens.ts +307 -307
  95. package/src/solana/chain/modules/SolanaTransactions.ts +365 -365
  96. package/src/solana/events/SolanaChainEvents.ts +299 -299
  97. package/src/solana/events/SolanaChainEventsBrowser.ts +209 -209
  98. package/src/solana/program/SolanaProgramBase.ts +79 -79
  99. package/src/solana/program/SolanaProgramModule.ts +15 -15
  100. package/src/solana/program/modules/SolanaProgramEvents.ts +155 -155
  101. package/src/solana/swaps/SolanaSwapData.ts +430 -430
  102. package/src/solana/swaps/SolanaSwapModule.ts +16 -16
  103. package/src/solana/swaps/SolanaSwapProgram.ts +854 -849
  104. package/src/solana/swaps/SwapTypeEnum.ts +29 -29
  105. package/src/solana/swaps/modules/SolanaDataAccount.ts +307 -307
  106. package/src/solana/swaps/modules/SolanaLpVault.ts +215 -215
  107. package/src/solana/swaps/modules/SwapClaim.ts +389 -389
  108. package/src/solana/swaps/modules/SwapInit.ts +663 -663
  109. package/src/solana/swaps/modules/SwapRefund.ts +323 -323
  110. package/src/solana/swaps/programIdl.json +944 -944
  111. package/src/solana/swaps/programTypes.ts +1885 -1885
  112. package/src/solana/wallet/SolanaKeypairWallet.ts +36 -36
  113. package/src/solana/wallet/SolanaSigner.ts +24 -24
  114. package/src/utils/Utils.ts +180 -180
@@ -1,256 +1,256 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.SolanaChainEvents = void 0;
4
- const fs = require("fs/promises");
5
- const SolanaChainEventsBrowser_1 = require("./SolanaChainEventsBrowser");
6
- const BLOCKHEIGHT_FILENAME = "/blockheight.txt";
7
- const LOG_FETCH_INTERVAL = 5 * 1000;
8
- const LOG_FETCH_LIMIT = 500;
9
- const PROCESSED_SIGNATURES_BACKLOG = 100;
10
- /**
11
- * Event handler for backend Node.js systems with access to fs, uses HTTP polling in combination with WS to not miss
12
- * any events
13
- */
14
- class SolanaChainEvents extends SolanaChainEventsBrowser_1.SolanaChainEventsBrowser {
15
- constructor(directory, connection, solanaSwapProgram, logFetchInterval, logFetchLimit) {
16
- super(connection, solanaSwapProgram);
17
- this.signaturesProcessing = {};
18
- this.processedSignatures = [];
19
- this.processedSignaturesIndex = 0;
20
- this.directory = directory;
21
- this.logFetchInterval = logFetchInterval || LOG_FETCH_INTERVAL;
22
- this.logFetchLimit = logFetchLimit || LOG_FETCH_LIMIT;
23
- }
24
- addProcessedSignature(signature) {
25
- this.processedSignatures[this.processedSignaturesIndex] = signature;
26
- this.processedSignaturesIndex += 1;
27
- if (this.processedSignaturesIndex >= PROCESSED_SIGNATURES_BACKLOG)
28
- this.processedSignaturesIndex = 0;
29
- }
30
- isSignatureProcessed(signature) {
31
- return this.processedSignatures.includes(signature);
32
- }
33
- /**
34
- * Retrieves last signature & slot from filesystem
35
- *
36
- * @private
37
- */
38
- async getLastSignature() {
39
- try {
40
- const txt = (await fs.readFile(this.directory + BLOCKHEIGHT_FILENAME)).toString();
41
- const arr = txt.split(";");
42
- if (arr.length < 2)
43
- return {
44
- signature: txt,
45
- slot: 0
46
- };
47
- return {
48
- signature: arr[0],
49
- slot: parseInt(arr[1])
50
- };
51
- }
52
- catch (e) {
53
- return null;
54
- }
55
- }
56
- /**
57
- * Saves last signature & slot to the filesystem
58
- *
59
- * @private
60
- */
61
- saveLastSignature(lastSignature, slot) {
62
- return fs.writeFile(this.directory + BLOCKHEIGHT_FILENAME, lastSignature + ";" + slot);
63
- }
64
- /**
65
- * Parses EventObject from the transaction
66
- *
67
- * @param transaction
68
- * @private
69
- * @returns {EventObject} parsed event object
70
- */
71
- getEventObjectFromTransaction(transaction) {
72
- if (transaction.meta.err != null)
73
- return null;
74
- const instructions = this.solanaSwapProgram.Events.decodeInstructions(transaction.transaction.message);
75
- const events = this.solanaSwapProgram.Events.parseLogs(transaction.meta.logMessages);
76
- return {
77
- instructions,
78
- events,
79
- blockTime: transaction.blockTime,
80
- signature: transaction.transaction.signatures[0]
81
- };
82
- }
83
- /**
84
- * Fetches transaction from the RPC, parses it to even object & processes it through event handler
85
- *
86
- * @param signature
87
- * @private
88
- * @returns {boolean} whether the operation was successful
89
- */
90
- async fetchTxAndProcessEvent(signature) {
91
- try {
92
- const transaction = await this.connection.getParsedTransaction(signature, {
93
- commitment: "confirmed",
94
- maxSupportedTransactionVersion: 1
95
- });
96
- if (transaction == null)
97
- return false;
98
- const eventObject = this.getEventObjectFromTransaction(transaction);
99
- if (eventObject == null)
100
- return true;
101
- await this.processEvent(eventObject);
102
- return true;
103
- }
104
- catch (e) {
105
- this.logger.error("fetchTxAndProcessEvent(): Error fetching transaction and processing event, signature: " + signature, e);
106
- return false;
107
- }
108
- }
109
- /**
110
- * Returns websocket event handler for specific event type
111
- *
112
- * @param name
113
- * @protected
114
- * @returns event handler to be passed to program's addEventListener function
115
- */
116
- getWsEventHandler(name) {
117
- return (data, slotNumber, signature) => {
118
- if (this.signaturesProcessing[signature] != null)
119
- return;
120
- if (this.isSignatureProcessed(signature))
121
- return;
122
- this.logger.debug("getWsEventHandler(" + name + "): Process signature: ", signature);
123
- this.signaturesProcessing[signature] = this.processEvent({
124
- events: [{ name, data: data }],
125
- instructions: null,
126
- blockTime: Math.floor(Date.now() / 1000),
127
- signature
128
- }).then(() => true).catch(e => {
129
- this.logger.error("getWsEventHandler(" + name + "): Error processing signature: " + signature, e);
130
- return false;
131
- });
132
- };
133
- }
134
- /**
135
- * Gets all the new signatures from the last processed signature
136
- *
137
- * @param lastProcessedSignature
138
- * @private
139
- */
140
- async getNewSignatures(lastProcessedSignature) {
141
- let signatures = [];
142
- let fetched = null;
143
- while (fetched == null || fetched.length === this.logFetchLimit) {
144
- if (signatures.length === 0) {
145
- fetched = await this.connection.getSignaturesForAddress(this.solanaSwapProgram.program.programId, {
146
- until: lastProcessedSignature.signature,
147
- limit: this.logFetchLimit
148
- }, "confirmed");
149
- //Check if newest returned signature (index 0) is older than the latest signature's slot, this is a sanity check
150
- if (fetched.length > 0 && fetched[0].slot < lastProcessedSignature.slot) {
151
- this.logger.debug("getNewSignatures(): Sanity check triggered, returned signature slot height is older than latest!");
152
- return;
153
- }
154
- }
155
- else {
156
- fetched = await this.connection.getSignaturesForAddress(this.solanaSwapProgram.program.programId, {
157
- before: signatures[signatures.length - 1].signature,
158
- until: lastProcessedSignature.signature,
159
- limit: this.logFetchLimit
160
- }, "confirmed");
161
- }
162
- signatures = signatures.concat(fetched);
163
- }
164
- return signatures;
165
- }
166
- /**
167
- * Gets single latest known signature
168
- *
169
- * @private
170
- */
171
- async getFirstSignature() {
172
- return await this.connection.getSignaturesForAddress(this.solanaSwapProgram.program.programId, {
173
- limit: 1
174
- }, "confirmed");
175
- }
176
- /**
177
- * Processes signatures, fetches transactions & processes event through event handlers
178
- *
179
- * @param signatures
180
- * @private
181
- * @returns {Promise<{signature: string, slot: number}>} latest processed transaction signature and slot height
182
- */
183
- async processSignatures(signatures) {
184
- let lastSuccessfulSignature = null;
185
- try {
186
- for (let i = signatures.length - 1; i >= 0; i--) {
187
- const txSignature = signatures[i];
188
- //Check if signature is already being processed by the
189
- const signaturePromise = this.signaturesProcessing[txSignature.signature];
190
- if (signaturePromise != null) {
191
- const result = await signaturePromise;
192
- delete this.signaturesProcessing[txSignature.signature];
193
- if (result) {
194
- lastSuccessfulSignature = txSignature;
195
- this.addProcessedSignature(txSignature.signature);
196
- continue;
197
- }
198
- }
199
- this.logger.debug("processSignatures(): Process signature: ", txSignature);
200
- const processPromise = this.fetchTxAndProcessEvent(txSignature.signature);
201
- this.signaturesProcessing[txSignature.signature] = processPromise;
202
- const result = await processPromise;
203
- if (!result)
204
- throw new Error("Failed to process signature: " + txSignature);
205
- lastSuccessfulSignature = txSignature;
206
- this.addProcessedSignature(txSignature.signature);
207
- delete this.signaturesProcessing[txSignature.signature];
208
- }
209
- }
210
- catch (e) {
211
- this.logger.error("processSignatures(): Failed processing signatures: ", e);
212
- }
213
- return lastSuccessfulSignature;
214
- }
215
- /**
216
- * Polls for new events & processes them
217
- *
218
- * @private
219
- */
220
- async checkEvents() {
221
- const lastSignature = await this.getLastSignature();
222
- let signatures = lastSignature == null ? await this.getFirstSignature() : await this.getNewSignatures(lastSignature);
223
- let lastSuccessfulSignature = await this.processSignatures(signatures);
224
- if (lastSuccessfulSignature != null) {
225
- await this.saveLastSignature(lastSuccessfulSignature.signature, lastSuccessfulSignature.slot);
226
- }
227
- }
228
- async setupHttpPolling() {
229
- this.stopped = false;
230
- let func;
231
- func = async () => {
232
- await this.checkEvents().catch(e => {
233
- this.logger.error("setupHttpPolling(): Failed to fetch Solana log: ", e);
234
- });
235
- if (this.stopped)
236
- return;
237
- this.timeout = setTimeout(func, this.logFetchInterval);
238
- };
239
- await func();
240
- }
241
- async init() {
242
- try {
243
- await fs.mkdir(this.directory);
244
- }
245
- catch (e) { }
246
- await this.setupHttpPolling();
247
- this.setupWebsocket();
248
- }
249
- stop() {
250
- this.stopped = true;
251
- if (this.timeout != null)
252
- clearTimeout(this.timeout);
253
- return super.stop();
254
- }
255
- }
256
- exports.SolanaChainEvents = SolanaChainEvents;
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.SolanaChainEvents = void 0;
4
+ const fs = require("fs/promises");
5
+ const SolanaChainEventsBrowser_1 = require("./SolanaChainEventsBrowser");
6
+ const BLOCKHEIGHT_FILENAME = "/blockheight.txt";
7
+ const LOG_FETCH_INTERVAL = 5 * 1000;
8
+ const LOG_FETCH_LIMIT = 500;
9
+ const PROCESSED_SIGNATURES_BACKLOG = 100;
10
+ /**
11
+ * Event handler for backend Node.js systems with access to fs, uses HTTP polling in combination with WS to not miss
12
+ * any events
13
+ */
14
+ class SolanaChainEvents extends SolanaChainEventsBrowser_1.SolanaChainEventsBrowser {
15
+ constructor(directory, connection, solanaSwapProgram, logFetchInterval, logFetchLimit) {
16
+ super(connection, solanaSwapProgram);
17
+ this.signaturesProcessing = {};
18
+ this.processedSignatures = [];
19
+ this.processedSignaturesIndex = 0;
20
+ this.directory = directory;
21
+ this.logFetchInterval = logFetchInterval || LOG_FETCH_INTERVAL;
22
+ this.logFetchLimit = logFetchLimit || LOG_FETCH_LIMIT;
23
+ }
24
+ addProcessedSignature(signature) {
25
+ this.processedSignatures[this.processedSignaturesIndex] = signature;
26
+ this.processedSignaturesIndex += 1;
27
+ if (this.processedSignaturesIndex >= PROCESSED_SIGNATURES_BACKLOG)
28
+ this.processedSignaturesIndex = 0;
29
+ }
30
+ isSignatureProcessed(signature) {
31
+ return this.processedSignatures.includes(signature);
32
+ }
33
+ /**
34
+ * Retrieves last signature & slot from filesystem
35
+ *
36
+ * @private
37
+ */
38
+ async getLastSignature() {
39
+ try {
40
+ const txt = (await fs.readFile(this.directory + BLOCKHEIGHT_FILENAME)).toString();
41
+ const arr = txt.split(";");
42
+ if (arr.length < 2)
43
+ return {
44
+ signature: txt,
45
+ slot: 0
46
+ };
47
+ return {
48
+ signature: arr[0],
49
+ slot: parseInt(arr[1])
50
+ };
51
+ }
52
+ catch (e) {
53
+ return null;
54
+ }
55
+ }
56
+ /**
57
+ * Saves last signature & slot to the filesystem
58
+ *
59
+ * @private
60
+ */
61
+ saveLastSignature(lastSignature, slot) {
62
+ return fs.writeFile(this.directory + BLOCKHEIGHT_FILENAME, lastSignature + ";" + slot);
63
+ }
64
+ /**
65
+ * Parses EventObject from the transaction
66
+ *
67
+ * @param transaction
68
+ * @private
69
+ * @returns {EventObject} parsed event object
70
+ */
71
+ getEventObjectFromTransaction(transaction) {
72
+ if (transaction.meta.err != null)
73
+ return null;
74
+ const instructions = this.solanaSwapProgram.Events.decodeInstructions(transaction.transaction.message);
75
+ const events = this.solanaSwapProgram.Events.parseLogs(transaction.meta.logMessages);
76
+ return {
77
+ instructions,
78
+ events,
79
+ blockTime: transaction.blockTime,
80
+ signature: transaction.transaction.signatures[0]
81
+ };
82
+ }
83
+ /**
84
+ * Fetches transaction from the RPC, parses it to even object & processes it through event handler
85
+ *
86
+ * @param signature
87
+ * @private
88
+ * @returns {boolean} whether the operation was successful
89
+ */
90
+ async fetchTxAndProcessEvent(signature) {
91
+ try {
92
+ const transaction = await this.connection.getParsedTransaction(signature, {
93
+ commitment: "confirmed",
94
+ maxSupportedTransactionVersion: 1
95
+ });
96
+ if (transaction == null)
97
+ return false;
98
+ const eventObject = this.getEventObjectFromTransaction(transaction);
99
+ if (eventObject == null)
100
+ return true;
101
+ await this.processEvent(eventObject);
102
+ return true;
103
+ }
104
+ catch (e) {
105
+ this.logger.error("fetchTxAndProcessEvent(): Error fetching transaction and processing event, signature: " + signature, e);
106
+ return false;
107
+ }
108
+ }
109
+ /**
110
+ * Returns websocket event handler for specific event type
111
+ *
112
+ * @param name
113
+ * @protected
114
+ * @returns event handler to be passed to program's addEventListener function
115
+ */
116
+ getWsEventHandler(name) {
117
+ return (data, slotNumber, signature) => {
118
+ if (this.signaturesProcessing[signature] != null)
119
+ return;
120
+ if (this.isSignatureProcessed(signature))
121
+ return;
122
+ this.logger.debug("getWsEventHandler(" + name + "): Process signature: ", signature);
123
+ this.signaturesProcessing[signature] = this.processEvent({
124
+ events: [{ name, data: data }],
125
+ instructions: null,
126
+ blockTime: Math.floor(Date.now() / 1000),
127
+ signature
128
+ }).then(() => true).catch(e => {
129
+ this.logger.error("getWsEventHandler(" + name + "): Error processing signature: " + signature, e);
130
+ return false;
131
+ });
132
+ };
133
+ }
134
+ /**
135
+ * Gets all the new signatures from the last processed signature
136
+ *
137
+ * @param lastProcessedSignature
138
+ * @private
139
+ */
140
+ async getNewSignatures(lastProcessedSignature) {
141
+ let signatures = [];
142
+ let fetched = null;
143
+ while (fetched == null || fetched.length === this.logFetchLimit) {
144
+ if (signatures.length === 0) {
145
+ fetched = await this.connection.getSignaturesForAddress(this.solanaSwapProgram.program.programId, {
146
+ until: lastProcessedSignature.signature,
147
+ limit: this.logFetchLimit
148
+ }, "confirmed");
149
+ //Check if newest returned signature (index 0) is older than the latest signature's slot, this is a sanity check
150
+ if (fetched.length > 0 && fetched[0].slot < lastProcessedSignature.slot) {
151
+ this.logger.debug("getNewSignatures(): Sanity check triggered, returned signature slot height is older than latest!");
152
+ return;
153
+ }
154
+ }
155
+ else {
156
+ fetched = await this.connection.getSignaturesForAddress(this.solanaSwapProgram.program.programId, {
157
+ before: signatures[signatures.length - 1].signature,
158
+ until: lastProcessedSignature.signature,
159
+ limit: this.logFetchLimit
160
+ }, "confirmed");
161
+ }
162
+ signatures = signatures.concat(fetched);
163
+ }
164
+ return signatures;
165
+ }
166
+ /**
167
+ * Gets single latest known signature
168
+ *
169
+ * @private
170
+ */
171
+ async getFirstSignature() {
172
+ return await this.connection.getSignaturesForAddress(this.solanaSwapProgram.program.programId, {
173
+ limit: 1
174
+ }, "confirmed");
175
+ }
176
+ /**
177
+ * Processes signatures, fetches transactions & processes event through event handlers
178
+ *
179
+ * @param signatures
180
+ * @private
181
+ * @returns {Promise<{signature: string, slot: number}>} latest processed transaction signature and slot height
182
+ */
183
+ async processSignatures(signatures) {
184
+ let lastSuccessfulSignature = null;
185
+ try {
186
+ for (let i = signatures.length - 1; i >= 0; i--) {
187
+ const txSignature = signatures[i];
188
+ //Check if signature is already being processed by the
189
+ const signaturePromise = this.signaturesProcessing[txSignature.signature];
190
+ if (signaturePromise != null) {
191
+ const result = await signaturePromise;
192
+ delete this.signaturesProcessing[txSignature.signature];
193
+ if (result) {
194
+ lastSuccessfulSignature = txSignature;
195
+ this.addProcessedSignature(txSignature.signature);
196
+ continue;
197
+ }
198
+ }
199
+ this.logger.debug("processSignatures(): Process signature: ", txSignature);
200
+ const processPromise = this.fetchTxAndProcessEvent(txSignature.signature);
201
+ this.signaturesProcessing[txSignature.signature] = processPromise;
202
+ const result = await processPromise;
203
+ if (!result)
204
+ throw new Error("Failed to process signature: " + txSignature);
205
+ lastSuccessfulSignature = txSignature;
206
+ this.addProcessedSignature(txSignature.signature);
207
+ delete this.signaturesProcessing[txSignature.signature];
208
+ }
209
+ }
210
+ catch (e) {
211
+ this.logger.error("processSignatures(): Failed processing signatures: ", e);
212
+ }
213
+ return lastSuccessfulSignature;
214
+ }
215
+ /**
216
+ * Polls for new events & processes them
217
+ *
218
+ * @private
219
+ */
220
+ async checkEvents() {
221
+ const lastSignature = await this.getLastSignature();
222
+ let signatures = lastSignature == null ? await this.getFirstSignature() : await this.getNewSignatures(lastSignature);
223
+ let lastSuccessfulSignature = await this.processSignatures(signatures);
224
+ if (lastSuccessfulSignature != null) {
225
+ await this.saveLastSignature(lastSuccessfulSignature.signature, lastSuccessfulSignature.slot);
226
+ }
227
+ }
228
+ async setupHttpPolling() {
229
+ this.stopped = false;
230
+ let func;
231
+ func = async () => {
232
+ await this.checkEvents().catch(e => {
233
+ this.logger.error("setupHttpPolling(): Failed to fetch Solana log: ", e);
234
+ });
235
+ if (this.stopped)
236
+ return;
237
+ this.timeout = setTimeout(func, this.logFetchInterval);
238
+ };
239
+ await func();
240
+ }
241
+ async init() {
242
+ try {
243
+ await fs.mkdir(this.directory);
244
+ }
245
+ catch (e) { }
246
+ await this.setupHttpPolling();
247
+ this.setupWebsocket();
248
+ }
249
+ stop() {
250
+ this.stopped = true;
251
+ if (this.timeout != null)
252
+ clearTimeout(this.timeout);
253
+ return super.stop();
254
+ }
255
+ }
256
+ exports.SolanaChainEvents = SolanaChainEvents;
@@ -1,75 +1,75 @@
1
- import { ChainEvents, ClaimEvent, EventListener, InitializeEvent, RefundEvent } from "@atomiqlabs/base";
2
- import { SolanaSwapData } from "../swaps/SolanaSwapData";
3
- import { IdlEvents } from "@coral-xyz/anchor";
4
- import { SolanaSwapProgram } from "../swaps/SolanaSwapProgram";
5
- import { Connection } from "@solana/web3.js";
6
- import { InstructionWithAccounts, ProgramEvent } from "../program/modules/SolanaProgramEvents";
7
- import { SwapProgram } from "../swaps/programTypes";
8
- export type EventObject = {
9
- events: ProgramEvent<SwapProgram>[];
10
- instructions: InstructionWithAccounts<SwapProgram>[];
11
- blockTime: number;
12
- signature: string;
13
- };
14
- /**
15
- * Solana on-chain event handler for front-end systems without access to fs, uses pure WS to subscribe, might lose
16
- * out on some events if the network is unreliable, front-end systems should take this into consideration and not
17
- * rely purely on events
18
- */
19
- export declare class SolanaChainEventsBrowser implements ChainEvents<SolanaSwapData> {
20
- protected readonly listeners: EventListener<SolanaSwapData>[];
21
- protected readonly connection: Connection;
22
- protected readonly solanaSwapProgram: SolanaSwapProgram;
23
- protected eventListeners: number[];
24
- protected readonly logger: {
25
- debug: (msg: any, ...args: any[]) => void;
26
- info: (msg: any, ...args: any[]) => void;
27
- warn: (msg: any, ...args: any[]) => void;
28
- error: (msg: any, ...args: any[]) => void;
29
- };
30
- constructor(connection: Connection, solanaSwapContract: SolanaSwapProgram);
31
- /**
32
- * Fetches and parses transaction instructions
33
- *
34
- * @private
35
- * @returns {Promise<InstructionWithAccounts<SwapProgram>[]>} array of parsed instructions
36
- */
37
- private getTransactionInstructions;
38
- /**
39
- * Returns async getter for fetching on-demand initialize event swap data
40
- *
41
- * @param eventObject
42
- * @param txoHash
43
- * @private
44
- * @returns {() => Promise<SolanaSwapData>} getter to be passed to InitializeEvent constructor
45
- */
46
- private getSwapDataGetter;
47
- protected parseInitializeEvent(data: IdlEvents<SwapProgram>["InitializeEvent"], eventObject: EventObject): InitializeEvent<SolanaSwapData>;
48
- protected parseRefundEvent(data: IdlEvents<SwapProgram>["RefundEvent"]): RefundEvent<SolanaSwapData>;
49
- protected parseClaimEvent(data: IdlEvents<SwapProgram>["ClaimEvent"]): ClaimEvent<SolanaSwapData>;
50
- /**
51
- * Processes event as received from the chain, parses it & calls event listeners
52
- *
53
- * @param eventObject
54
- * @protected
55
- */
56
- protected processEvent(eventObject: EventObject): Promise<void>;
57
- /**
58
- * Returns websocket event handler for specific event type
59
- *
60
- * @param name
61
- * @protected
62
- * @returns event handler to be passed to program's addEventListener function
63
- */
64
- protected getWsEventHandler<E extends "InitializeEvent" | "RefundEvent" | "ClaimEvent">(name: E): (data: IdlEvents<SwapProgram>[E], slotNumber: number, signature: string) => void;
65
- /**
66
- * Sets up event handlers listening for swap events over websocket
67
- *
68
- * @protected
69
- */
70
- protected setupWebsocket(): void;
71
- init(): Promise<void>;
72
- stop(): Promise<void>;
73
- registerListener(cbk: EventListener<SolanaSwapData>): void;
74
- unregisterListener(cbk: EventListener<SolanaSwapData>): boolean;
75
- }
1
+ import { ChainEvents, ClaimEvent, EventListener, InitializeEvent, RefundEvent } from "@atomiqlabs/base";
2
+ import { SolanaSwapData } from "../swaps/SolanaSwapData";
3
+ import { IdlEvents } from "@coral-xyz/anchor";
4
+ import { SolanaSwapProgram } from "../swaps/SolanaSwapProgram";
5
+ import { Connection } from "@solana/web3.js";
6
+ import { InstructionWithAccounts, ProgramEvent } from "../program/modules/SolanaProgramEvents";
7
+ import { SwapProgram } from "../swaps/programTypes";
8
+ export type EventObject = {
9
+ events: ProgramEvent<SwapProgram>[];
10
+ instructions: InstructionWithAccounts<SwapProgram>[];
11
+ blockTime: number;
12
+ signature: string;
13
+ };
14
+ /**
15
+ * Solana on-chain event handler for front-end systems without access to fs, uses pure WS to subscribe, might lose
16
+ * out on some events if the network is unreliable, front-end systems should take this into consideration and not
17
+ * rely purely on events
18
+ */
19
+ export declare class SolanaChainEventsBrowser implements ChainEvents<SolanaSwapData> {
20
+ protected readonly listeners: EventListener<SolanaSwapData>[];
21
+ protected readonly connection: Connection;
22
+ protected readonly solanaSwapProgram: SolanaSwapProgram;
23
+ protected eventListeners: number[];
24
+ protected readonly logger: {
25
+ debug: (msg: any, ...args: any[]) => void;
26
+ info: (msg: any, ...args: any[]) => void;
27
+ warn: (msg: any, ...args: any[]) => void;
28
+ error: (msg: any, ...args: any[]) => void;
29
+ };
30
+ constructor(connection: Connection, solanaSwapContract: SolanaSwapProgram);
31
+ /**
32
+ * Fetches and parses transaction instructions
33
+ *
34
+ * @private
35
+ * @returns {Promise<InstructionWithAccounts<SwapProgram>[]>} array of parsed instructions
36
+ */
37
+ private getTransactionInstructions;
38
+ /**
39
+ * Returns async getter for fetching on-demand initialize event swap data
40
+ *
41
+ * @param eventObject
42
+ * @param txoHash
43
+ * @private
44
+ * @returns {() => Promise<SolanaSwapData>} getter to be passed to InitializeEvent constructor
45
+ */
46
+ private getSwapDataGetter;
47
+ protected parseInitializeEvent(data: IdlEvents<SwapProgram>["InitializeEvent"], eventObject: EventObject): InitializeEvent<SolanaSwapData>;
48
+ protected parseRefundEvent(data: IdlEvents<SwapProgram>["RefundEvent"]): RefundEvent<SolanaSwapData>;
49
+ protected parseClaimEvent(data: IdlEvents<SwapProgram>["ClaimEvent"]): ClaimEvent<SolanaSwapData>;
50
+ /**
51
+ * Processes event as received from the chain, parses it & calls event listeners
52
+ *
53
+ * @param eventObject
54
+ * @protected
55
+ */
56
+ protected processEvent(eventObject: EventObject): Promise<void>;
57
+ /**
58
+ * Returns websocket event handler for specific event type
59
+ *
60
+ * @param name
61
+ * @protected
62
+ * @returns event handler to be passed to program's addEventListener function
63
+ */
64
+ protected getWsEventHandler<E extends "InitializeEvent" | "RefundEvent" | "ClaimEvent">(name: E): (data: IdlEvents<SwapProgram>[E], slotNumber: number, signature: string) => void;
65
+ /**
66
+ * Sets up event handlers listening for swap events over websocket
67
+ *
68
+ * @protected
69
+ */
70
+ protected setupWebsocket(): void;
71
+ init(): Promise<void>;
72
+ stop(): Promise<void>;
73
+ registerListener(cbk: EventListener<SolanaSwapData>): void;
74
+ unregisterListener(cbk: EventListener<SolanaSwapData>): boolean;
75
+ }