@atomiqlabs/chain-solana 10.0.0-dev.3 → 11.0.0
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.
- package/LICENSE +201 -201
- package/dist/index.d.ts +29 -29
- package/dist/index.js +45 -45
- package/dist/solana/SolanaChainType.d.ts +10 -10
- package/dist/solana/SolanaChainType.js +2 -2
- package/dist/solana/SolanaChains.d.ts +20 -20
- package/dist/solana/SolanaChains.js +25 -25
- package/dist/solana/SolanaInitializer.d.ts +18 -18
- package/dist/solana/SolanaInitializer.js +63 -63
- package/dist/solana/btcrelay/SolanaBtcRelay.d.ts +228 -228
- package/dist/solana/btcrelay/SolanaBtcRelay.js +441 -441
- package/dist/solana/btcrelay/headers/SolanaBtcHeader.d.ts +29 -29
- package/dist/solana/btcrelay/headers/SolanaBtcHeader.js +34 -34
- package/dist/solana/btcrelay/headers/SolanaBtcStoredHeader.d.ts +46 -46
- package/dist/solana/btcrelay/headers/SolanaBtcStoredHeader.js +78 -78
- package/dist/solana/btcrelay/program/programIdl.json +671 -671
- package/dist/solana/chain/SolanaAction.d.ts +26 -26
- package/dist/solana/chain/SolanaAction.js +86 -86
- package/dist/solana/chain/SolanaChainInterface.d.ts +58 -58
- package/dist/solana/chain/SolanaChainInterface.js +112 -112
- package/dist/solana/chain/SolanaModule.d.ts +14 -14
- package/dist/solana/chain/SolanaModule.js +13 -13
- package/dist/solana/chain/modules/SolanaAddresses.d.ts +8 -8
- package/dist/solana/chain/modules/SolanaAddresses.js +22 -22
- package/dist/solana/chain/modules/SolanaBlocks.d.ts +28 -28
- package/dist/solana/chain/modules/SolanaBlocks.js +72 -72
- package/dist/solana/chain/modules/SolanaEvents.d.ts +25 -25
- package/dist/solana/chain/modules/SolanaEvents.js +58 -58
- package/dist/solana/chain/modules/SolanaFees.d.ts +121 -121
- package/dist/solana/chain/modules/SolanaFees.js +379 -379
- package/dist/solana/chain/modules/SolanaSignatures.d.ts +23 -23
- package/dist/solana/chain/modules/SolanaSignatures.js +39 -39
- package/dist/solana/chain/modules/SolanaSlots.d.ts +31 -31
- package/dist/solana/chain/modules/SolanaSlots.js +68 -68
- package/dist/solana/chain/modules/SolanaTokens.d.ts +136 -136
- package/dist/solana/chain/modules/SolanaTokens.js +248 -248
- package/dist/solana/chain/modules/SolanaTransactions.d.ts +124 -124
- package/dist/solana/chain/modules/SolanaTransactions.js +332 -332
- package/dist/solana/events/SolanaChainEvents.d.ts +88 -88
- package/dist/solana/events/SolanaChainEvents.js +256 -256
- package/dist/solana/events/SolanaChainEventsBrowser.d.ts +85 -85
- package/dist/solana/events/SolanaChainEventsBrowser.js +194 -194
- package/dist/solana/program/SolanaProgramBase.d.ts +40 -40
- package/dist/solana/program/SolanaProgramBase.js +43 -43
- package/dist/solana/program/SolanaProgramModule.d.ts +8 -8
- package/dist/solana/program/SolanaProgramModule.js +11 -11
- package/dist/solana/program/modules/SolanaProgramEvents.d.ts +59 -59
- package/dist/solana/program/modules/SolanaProgramEvents.js +103 -103
- package/dist/solana/swaps/SolanaSwapData.d.ts +59 -59
- package/dist/solana/swaps/SolanaSwapData.js +267 -267
- package/dist/solana/swaps/SolanaSwapModule.d.ts +10 -10
- package/dist/solana/swaps/SolanaSwapModule.js +11 -11
- package/dist/solana/swaps/SolanaSwapProgram.d.ts +202 -202
- package/dist/solana/swaps/SolanaSwapProgram.js +470 -463
- package/dist/solana/swaps/SwapTypeEnum.d.ts +11 -11
- package/dist/solana/swaps/SwapTypeEnum.js +42 -42
- package/dist/solana/swaps/modules/SolanaDataAccount.d.ts +94 -94
- package/dist/solana/swaps/modules/SolanaDataAccount.js +231 -231
- package/dist/solana/swaps/modules/SolanaLpVault.d.ts +71 -71
- package/dist/solana/swaps/modules/SolanaLpVault.js +173 -173
- package/dist/solana/swaps/modules/SwapClaim.d.ts +129 -129
- package/dist/solana/swaps/modules/SwapClaim.js +291 -291
- package/dist/solana/swaps/modules/SwapInit.d.ts +217 -217
- package/dist/solana/swaps/modules/SwapInit.js +519 -519
- package/dist/solana/swaps/modules/SwapRefund.d.ts +82 -82
- package/dist/solana/swaps/modules/SwapRefund.js +252 -252
- package/dist/solana/swaps/programIdl.json +945 -945
- package/dist/solana/swaps/programTypes.d.ts +943 -943
- package/dist/solana/swaps/programTypes.js +945 -945
- package/dist/solana/wallet/SolanaKeypairWallet.d.ts +9 -9
- package/dist/solana/wallet/SolanaKeypairWallet.js +33 -33
- package/dist/solana/wallet/SolanaSigner.d.ts +10 -10
- package/dist/solana/wallet/SolanaSigner.js +16 -16
- package/dist/utils/Utils.d.ts +53 -53
- package/dist/utils/Utils.js +170 -170
- package/package.json +41 -41
- package/src/index.ts +36 -36
- package/src/solana/SolanaChainType.ts +25 -25
- package/src/solana/SolanaChains.ts +23 -23
- package/src/solana/SolanaInitializer.ts +102 -102
- package/src/solana/btcrelay/SolanaBtcRelay.ts +589 -588
- package/src/solana/btcrelay/headers/SolanaBtcHeader.ts +57 -57
- package/src/solana/btcrelay/headers/SolanaBtcStoredHeader.ts +102 -102
- package/src/solana/btcrelay/program/programIdl.json +670 -670
- package/src/solana/chain/SolanaAction.ts +108 -108
- package/src/solana/chain/SolanaChainInterface.ts +174 -174
- package/src/solana/chain/SolanaModule.ts +20 -20
- package/src/solana/chain/modules/SolanaAddresses.ts +20 -20
- package/src/solana/chain/modules/SolanaBlocks.ts +78 -78
- package/src/solana/chain/modules/SolanaEvents.ts +56 -56
- package/src/solana/chain/modules/SolanaFees.ts +450 -450
- package/src/solana/chain/modules/SolanaSignatures.ts +39 -39
- package/src/solana/chain/modules/SolanaSlots.ts +82 -82
- package/src/solana/chain/modules/SolanaTokens.ts +307 -307
- package/src/solana/chain/modules/SolanaTransactions.ts +370 -370
- package/src/solana/events/SolanaChainEvents.ts +299 -299
- package/src/solana/events/SolanaChainEventsBrowser.ts +256 -256
- package/src/solana/program/SolanaProgramBase.ts +79 -79
- package/src/solana/program/SolanaProgramModule.ts +15 -15
- package/src/solana/program/modules/SolanaProgramEvents.ts +140 -140
- package/src/solana/swaps/SolanaSwapData.ts +379 -379
- package/src/solana/swaps/SolanaSwapModule.ts +16 -16
- package/src/solana/swaps/SolanaSwapProgram.ts +697 -692
- package/src/solana/swaps/SwapTypeEnum.ts +29 -29
- package/src/solana/swaps/modules/SolanaDataAccount.ts +307 -307
- package/src/solana/swaps/modules/SolanaLpVault.ts +215 -215
- package/src/solana/swaps/modules/SwapClaim.ts +389 -389
- package/src/solana/swaps/modules/SwapInit.ts +663 -663
- package/src/solana/swaps/modules/SwapRefund.ts +312 -312
- package/src/solana/swaps/programIdl.json +944 -944
- package/src/solana/swaps/programTypes.ts +1885 -1885
- package/src/solana/wallet/SolanaKeypairWallet.ts +36 -36
- package/src/solana/wallet/SolanaSigner.ts +23 -23
- package/src/utils/Utils.ts +180 -180
|
@@ -1,256 +1,256 @@
|
|
|
1
|
-
import {ChainEvents, ClaimEvent, EventListener, InitializeEvent, RefundEvent, SwapEvent} from "@atomiqlabs/base";
|
|
2
|
-
import {SolanaSwapData} from "../swaps/SolanaSwapData";
|
|
3
|
-
import {IdlEvents} from "@coral-xyz/anchor";
|
|
4
|
-
import {SolanaSwapProgram} from "../swaps/SolanaSwapProgram";
|
|
5
|
-
import {
|
|
6
|
-
getLogger,
|
|
7
|
-
onceAsync, toEscrowHash, tryWithRetries
|
|
8
|
-
} from "../../utils/Utils";
|
|
9
|
-
import {Connection, ParsedTransactionWithMeta, PublicKey} from "@solana/web3.js";
|
|
10
|
-
import * as BN from "bn.js";
|
|
11
|
-
import {SwapTypeEnum} from "../swaps/SwapTypeEnum";
|
|
12
|
-
import {
|
|
13
|
-
InstructionWithAccounts,
|
|
14
|
-
ProgramEvent,
|
|
15
|
-
SingleInstructionWithAccounts
|
|
16
|
-
} from "../program/modules/SolanaProgramEvents";
|
|
17
|
-
import {SwapProgram} from "../swaps/programTypes";
|
|
18
|
-
import {Buffer} from "buffer";
|
|
19
|
-
|
|
20
|
-
export type EventObject = {
|
|
21
|
-
events: ProgramEvent<SwapProgram>[],
|
|
22
|
-
instructions: InstructionWithAccounts<SwapProgram>[],
|
|
23
|
-
blockTime: number,
|
|
24
|
-
signature: string
|
|
25
|
-
};
|
|
26
|
-
|
|
27
|
-
export type InitInstruction = SingleInstructionWithAccounts<SwapProgram["instructions"][2 | 3], SwapProgram>;
|
|
28
|
-
|
|
29
|
-
/**
|
|
30
|
-
* Solana on-chain event handler for front-end systems without access to fs, uses pure WS to subscribe, might lose
|
|
31
|
-
* out on some events if the network is unreliable, front-end systems should take this into consideration and not
|
|
32
|
-
* rely purely on events
|
|
33
|
-
*/
|
|
34
|
-
export class SolanaChainEventsBrowser implements ChainEvents<SolanaSwapData> {
|
|
35
|
-
|
|
36
|
-
protected readonly listeners: EventListener<SolanaSwapData>[] = [];
|
|
37
|
-
protected readonly connection: Connection;
|
|
38
|
-
protected readonly solanaSwapProgram: SolanaSwapProgram;
|
|
39
|
-
protected eventListeners: number[] = [];
|
|
40
|
-
protected readonly logger = getLogger("SolanaChainEventsBrowser: ");
|
|
41
|
-
|
|
42
|
-
constructor(connection: Connection, solanaSwapContract: SolanaSwapProgram) {
|
|
43
|
-
this.connection = connection;
|
|
44
|
-
this.solanaSwapProgram = solanaSwapContract;
|
|
45
|
-
}
|
|
46
|
-
|
|
47
|
-
/**
|
|
48
|
-
* Fetches and parses transaction instructions
|
|
49
|
-
*
|
|
50
|
-
* @private
|
|
51
|
-
* @returns {Promise<InstructionWithAccounts<SwapProgram>[]>} array of parsed instructions
|
|
52
|
-
*/
|
|
53
|
-
private async getTransactionInstructions(signature: string): Promise<InstructionWithAccounts<SwapProgram>[]> {
|
|
54
|
-
const transaction = await tryWithRetries<ParsedTransactionWithMeta>(async () => {
|
|
55
|
-
const res = await this.connection.getParsedTransaction(signature, {
|
|
56
|
-
commitment: "confirmed",
|
|
57
|
-
maxSupportedTransactionVersion: 0
|
|
58
|
-
});
|
|
59
|
-
if(res==null) throw new Error("Transaction not found!");
|
|
60
|
-
return res;
|
|
61
|
-
});
|
|
62
|
-
if(transaction==null) return null;
|
|
63
|
-
if(transaction.meta.err!=null) return null;
|
|
64
|
-
return this.solanaSwapProgram.Events.decodeInstructions(transaction.transaction.message);
|
|
65
|
-
}
|
|
66
|
-
|
|
67
|
-
/**
|
|
68
|
-
* Converts initialize instruction data into {SolanaSwapData}
|
|
69
|
-
*
|
|
70
|
-
* @param initIx
|
|
71
|
-
* @param txoHash
|
|
72
|
-
* @private
|
|
73
|
-
* @returns {SolanaSwapData} converted and parsed swap data
|
|
74
|
-
*/
|
|
75
|
-
private instructionToSwapData(
|
|
76
|
-
initIx: InitInstruction,
|
|
77
|
-
txoHash: string
|
|
78
|
-
): SolanaSwapData {
|
|
79
|
-
const paymentHash: Buffer = Buffer.from(initIx.data.swapData.hash);
|
|
80
|
-
let securityDeposit: BN = new BN(0);
|
|
81
|
-
let claimerBounty: BN = new BN(0);
|
|
82
|
-
let payIn: boolean = true;
|
|
83
|
-
if(initIx.name === "offererInitialize") {
|
|
84
|
-
payIn = false;
|
|
85
|
-
securityDeposit = initIx.data.securityDeposit;
|
|
86
|
-
claimerBounty = initIx.data.claimerBounty;
|
|
87
|
-
}
|
|
88
|
-
|
|
89
|
-
return new SolanaSwapData(
|
|
90
|
-
initIx.accounts.offerer,
|
|
91
|
-
initIx.accounts.claimer,
|
|
92
|
-
initIx.accounts.mint,
|
|
93
|
-
initIx.data.swapData.amount,
|
|
94
|
-
paymentHash.toString("hex"),
|
|
95
|
-
initIx.data.swapData.sequence,
|
|
96
|
-
initIx.data.swapData.expiry,
|
|
97
|
-
initIx.data.swapData.nonce,
|
|
98
|
-
initIx.data.swapData.confirmations,
|
|
99
|
-
initIx.data.swapData.payOut,
|
|
100
|
-
SwapTypeEnum.toNumber(initIx.data.swapData.kind),
|
|
101
|
-
payIn,
|
|
102
|
-
initIx.name === "offererInitializePayIn" ? initIx.accounts.offererAta : PublicKey.default,
|
|
103
|
-
initIx.data.swapData.payOut ? initIx.accounts.claimerAta : PublicKey.default,
|
|
104
|
-
securityDeposit,
|
|
105
|
-
claimerBounty,
|
|
106
|
-
txoHash
|
|
107
|
-
);
|
|
108
|
-
}
|
|
109
|
-
|
|
110
|
-
/**
|
|
111
|
-
* Returns async getter for fetching on-demand initialize event swap data
|
|
112
|
-
*
|
|
113
|
-
* @param eventObject
|
|
114
|
-
* @param txoHash
|
|
115
|
-
* @private
|
|
116
|
-
* @returns {() => Promise<SolanaSwapData>} getter to be passed to InitializeEvent constructor
|
|
117
|
-
*/
|
|
118
|
-
private getSwapDataGetter(eventObject: EventObject, txoHash: string): () => Promise<SolanaSwapData> {
|
|
119
|
-
return async () => {
|
|
120
|
-
if(eventObject.instructions==null) eventObject.instructions = await this.getTransactionInstructions(eventObject.signature);
|
|
121
|
-
if(eventObject.instructions==null) return null;
|
|
122
|
-
|
|
123
|
-
const initIx = eventObject.instructions.find(
|
|
124
|
-
ix => ix!=null && (ix.name === "offererInitializePayIn" || ix.name === "offererInitialize")
|
|
125
|
-
) as InitInstruction;
|
|
126
|
-
if(initIx == null) return null;
|
|
127
|
-
|
|
128
|
-
return this.instructionToSwapData(initIx, txoHash);
|
|
129
|
-
}
|
|
130
|
-
}
|
|
131
|
-
|
|
132
|
-
protected parseInitializeEvent(data: IdlEvents<SwapProgram>["InitializeEvent"], eventObject: EventObject): InitializeEvent<SolanaSwapData> {
|
|
133
|
-
const paymentHash: string = Buffer.from(data.hash).toString("hex");
|
|
134
|
-
const txoHash: string = Buffer.from(data.txoHash).toString("hex");
|
|
135
|
-
const escrowHash = toEscrowHash(paymentHash, data.sequence);
|
|
136
|
-
this.logger.debug("InitializeEvent paymentHash: "+paymentHash+" sequence: "+data.sequence.toString(10)+
|
|
137
|
-
" txoHash: "+txoHash+" escrowHash: "+escrowHash);
|
|
138
|
-
return new InitializeEvent<SolanaSwapData>(
|
|
139
|
-
escrowHash,
|
|
140
|
-
SwapTypeEnum.toChainSwapType(data.kind),
|
|
141
|
-
onceAsync<SolanaSwapData>(this.getSwapDataGetter(eventObject, txoHash))
|
|
142
|
-
);
|
|
143
|
-
}
|
|
144
|
-
|
|
145
|
-
protected parseRefundEvent(data: IdlEvents<SwapProgram>["RefundEvent"]): RefundEvent<SolanaSwapData> {
|
|
146
|
-
const paymentHash: string = Buffer.from(data.hash).toString("hex");
|
|
147
|
-
const escrowHash = toEscrowHash(paymentHash, data.sequence);
|
|
148
|
-
this.logger.debug("RefundEvent paymentHash: "+paymentHash+" sequence: "+data.sequence.toString(10)+
|
|
149
|
-
" escrowHash: "+escrowHash);
|
|
150
|
-
return new RefundEvent<SolanaSwapData>(escrowHash);
|
|
151
|
-
}
|
|
152
|
-
|
|
153
|
-
protected parseClaimEvent(data: IdlEvents<SwapProgram>["ClaimEvent"]): ClaimEvent<SolanaSwapData> {
|
|
154
|
-
const secret: string = Buffer.from(data.secret).toString("hex");
|
|
155
|
-
const paymentHash: string = Buffer.from(data.hash).toString("hex");
|
|
156
|
-
const escrowHash = toEscrowHash(paymentHash, data.sequence);
|
|
157
|
-
this.logger.debug("ClaimEvent paymentHash: "+paymentHash+" sequence: "+data.sequence.toString(10)+
|
|
158
|
-
" secret: "+secret+" escrowHash: "+escrowHash);
|
|
159
|
-
return new ClaimEvent<SolanaSwapData>(escrowHash, secret);
|
|
160
|
-
}
|
|
161
|
-
|
|
162
|
-
/**
|
|
163
|
-
* Processes event as received from the chain, parses it & calls event listeners
|
|
164
|
-
*
|
|
165
|
-
* @param eventObject
|
|
166
|
-
* @protected
|
|
167
|
-
*/
|
|
168
|
-
protected async processEvent(eventObject : EventObject) {
|
|
169
|
-
let parsedEvents: SwapEvent<SolanaSwapData>[] = eventObject.events.map(event => {
|
|
170
|
-
let parsedEvent: SwapEvent<SolanaSwapData>;
|
|
171
|
-
switch(event.name) {
|
|
172
|
-
case "ClaimEvent":
|
|
173
|
-
parsedEvent = this.parseClaimEvent(event.data);
|
|
174
|
-
break;
|
|
175
|
-
case "RefundEvent":
|
|
176
|
-
parsedEvent = this.parseRefundEvent(event.data);
|
|
177
|
-
break;
|
|
178
|
-
case "InitializeEvent":
|
|
179
|
-
parsedEvent = this.parseInitializeEvent(event.data, eventObject);
|
|
180
|
-
break;
|
|
181
|
-
}
|
|
182
|
-
(parsedEvent as any).meta = {
|
|
183
|
-
blockTime: eventObject.blockTime,
|
|
184
|
-
timestamp: eventObject.blockTime,
|
|
185
|
-
txId: eventObject.signature
|
|
186
|
-
};
|
|
187
|
-
return parsedEvent;
|
|
188
|
-
}).filter(parsedEvent => parsedEvent!=null);
|
|
189
|
-
|
|
190
|
-
for(let listener of this.listeners) {
|
|
191
|
-
await listener(parsedEvents);
|
|
192
|
-
}
|
|
193
|
-
}
|
|
194
|
-
|
|
195
|
-
/**
|
|
196
|
-
* Returns websocket event handler for specific event type
|
|
197
|
-
*
|
|
198
|
-
* @param name
|
|
199
|
-
* @protected
|
|
200
|
-
* @returns event handler to be passed to program's addEventListener function
|
|
201
|
-
*/
|
|
202
|
-
protected getWsEventHandler<E extends "InitializeEvent" | "RefundEvent" | "ClaimEvent">(
|
|
203
|
-
name: E
|
|
204
|
-
): (data: IdlEvents<SwapProgram>[E], slotNumber: number, signature: string) => void {
|
|
205
|
-
return (data: IdlEvents<SwapProgram>[E], slotNumber: number, signature: string) => {
|
|
206
|
-
this.logger.debug("wsEventHandler: Process signature: ", signature);
|
|
207
|
-
|
|
208
|
-
this.processEvent({
|
|
209
|
-
events: [{name, data: data as any}],
|
|
210
|
-
instructions: null, //Instructions will be fetched on-demand if required
|
|
211
|
-
blockTime: Math.floor(Date.now()/1000),
|
|
212
|
-
signature
|
|
213
|
-
}).then(() => true).catch(e => {
|
|
214
|
-
this.logger.error("wsEventHandler: Error when processing signature: "+signature, e);
|
|
215
|
-
return false;
|
|
216
|
-
});
|
|
217
|
-
};
|
|
218
|
-
}
|
|
219
|
-
|
|
220
|
-
/**
|
|
221
|
-
* Sets up event handlers listening for swap events over websocket
|
|
222
|
-
*
|
|
223
|
-
* @protected
|
|
224
|
-
*/
|
|
225
|
-
protected setupWebsocket() {
|
|
226
|
-
const program = this.solanaSwapProgram.program;
|
|
227
|
-
this.eventListeners.push(program.addEventListener<"InitializeEvent">("InitializeEvent", this.getWsEventHandler("InitializeEvent")));
|
|
228
|
-
this.eventListeners.push(program.addEventListener<"ClaimEvent">("ClaimEvent", this.getWsEventHandler("ClaimEvent")));
|
|
229
|
-
this.eventListeners.push(program.addEventListener<"RefundEvent">("RefundEvent", this.getWsEventHandler("RefundEvent")));
|
|
230
|
-
}
|
|
231
|
-
|
|
232
|
-
init(): Promise<void> {
|
|
233
|
-
this.setupWebsocket();
|
|
234
|
-
return Promise.resolve();
|
|
235
|
-
}
|
|
236
|
-
|
|
237
|
-
async stop(): Promise<void> {
|
|
238
|
-
for(let num of this.eventListeners) {
|
|
239
|
-
await this.solanaSwapProgram.program.removeEventListener(num);
|
|
240
|
-
}
|
|
241
|
-
this.eventListeners = [];
|
|
242
|
-
}
|
|
243
|
-
|
|
244
|
-
registerListener(cbk: EventListener<SolanaSwapData>): void {
|
|
245
|
-
this.listeners.push(cbk);
|
|
246
|
-
}
|
|
247
|
-
|
|
248
|
-
unregisterListener(cbk: EventListener<SolanaSwapData>): boolean {
|
|
249
|
-
const index = this.listeners.indexOf(cbk);
|
|
250
|
-
if(index>=0) {
|
|
251
|
-
this.listeners.splice(index, 1);
|
|
252
|
-
return true;
|
|
253
|
-
}
|
|
254
|
-
return false;
|
|
255
|
-
}
|
|
256
|
-
}
|
|
1
|
+
import {ChainEvents, ClaimEvent, EventListener, InitializeEvent, RefundEvent, SwapEvent} from "@atomiqlabs/base";
|
|
2
|
+
import {SolanaSwapData} from "../swaps/SolanaSwapData";
|
|
3
|
+
import {IdlEvents} from "@coral-xyz/anchor";
|
|
4
|
+
import {SolanaSwapProgram} from "../swaps/SolanaSwapProgram";
|
|
5
|
+
import {
|
|
6
|
+
getLogger,
|
|
7
|
+
onceAsync, toEscrowHash, tryWithRetries
|
|
8
|
+
} from "../../utils/Utils";
|
|
9
|
+
import {Connection, ParsedTransactionWithMeta, PublicKey} from "@solana/web3.js";
|
|
10
|
+
import * as BN from "bn.js";
|
|
11
|
+
import {SwapTypeEnum} from "../swaps/SwapTypeEnum";
|
|
12
|
+
import {
|
|
13
|
+
InstructionWithAccounts,
|
|
14
|
+
ProgramEvent,
|
|
15
|
+
SingleInstructionWithAccounts
|
|
16
|
+
} from "../program/modules/SolanaProgramEvents";
|
|
17
|
+
import {SwapProgram} from "../swaps/programTypes";
|
|
18
|
+
import {Buffer} from "buffer";
|
|
19
|
+
|
|
20
|
+
export type EventObject = {
|
|
21
|
+
events: ProgramEvent<SwapProgram>[],
|
|
22
|
+
instructions: InstructionWithAccounts<SwapProgram>[],
|
|
23
|
+
blockTime: number,
|
|
24
|
+
signature: string
|
|
25
|
+
};
|
|
26
|
+
|
|
27
|
+
export type InitInstruction = SingleInstructionWithAccounts<SwapProgram["instructions"][2 | 3], SwapProgram>;
|
|
28
|
+
|
|
29
|
+
/**
|
|
30
|
+
* Solana on-chain event handler for front-end systems without access to fs, uses pure WS to subscribe, might lose
|
|
31
|
+
* out on some events if the network is unreliable, front-end systems should take this into consideration and not
|
|
32
|
+
* rely purely on events
|
|
33
|
+
*/
|
|
34
|
+
export class SolanaChainEventsBrowser implements ChainEvents<SolanaSwapData> {
|
|
35
|
+
|
|
36
|
+
protected readonly listeners: EventListener<SolanaSwapData>[] = [];
|
|
37
|
+
protected readonly connection: Connection;
|
|
38
|
+
protected readonly solanaSwapProgram: SolanaSwapProgram;
|
|
39
|
+
protected eventListeners: number[] = [];
|
|
40
|
+
protected readonly logger = getLogger("SolanaChainEventsBrowser: ");
|
|
41
|
+
|
|
42
|
+
constructor(connection: Connection, solanaSwapContract: SolanaSwapProgram) {
|
|
43
|
+
this.connection = connection;
|
|
44
|
+
this.solanaSwapProgram = solanaSwapContract;
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
/**
|
|
48
|
+
* Fetches and parses transaction instructions
|
|
49
|
+
*
|
|
50
|
+
* @private
|
|
51
|
+
* @returns {Promise<InstructionWithAccounts<SwapProgram>[]>} array of parsed instructions
|
|
52
|
+
*/
|
|
53
|
+
private async getTransactionInstructions(signature: string): Promise<InstructionWithAccounts<SwapProgram>[]> {
|
|
54
|
+
const transaction = await tryWithRetries<ParsedTransactionWithMeta>(async () => {
|
|
55
|
+
const res = await this.connection.getParsedTransaction(signature, {
|
|
56
|
+
commitment: "confirmed",
|
|
57
|
+
maxSupportedTransactionVersion: 0
|
|
58
|
+
});
|
|
59
|
+
if(res==null) throw new Error("Transaction not found!");
|
|
60
|
+
return res;
|
|
61
|
+
});
|
|
62
|
+
if(transaction==null) return null;
|
|
63
|
+
if(transaction.meta.err!=null) return null;
|
|
64
|
+
return this.solanaSwapProgram.Events.decodeInstructions(transaction.transaction.message);
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
/**
|
|
68
|
+
* Converts initialize instruction data into {SolanaSwapData}
|
|
69
|
+
*
|
|
70
|
+
* @param initIx
|
|
71
|
+
* @param txoHash
|
|
72
|
+
* @private
|
|
73
|
+
* @returns {SolanaSwapData} converted and parsed swap data
|
|
74
|
+
*/
|
|
75
|
+
private instructionToSwapData(
|
|
76
|
+
initIx: InitInstruction,
|
|
77
|
+
txoHash: string
|
|
78
|
+
): SolanaSwapData {
|
|
79
|
+
const paymentHash: Buffer = Buffer.from(initIx.data.swapData.hash);
|
|
80
|
+
let securityDeposit: BN = new BN(0);
|
|
81
|
+
let claimerBounty: BN = new BN(0);
|
|
82
|
+
let payIn: boolean = true;
|
|
83
|
+
if(initIx.name === "offererInitialize") {
|
|
84
|
+
payIn = false;
|
|
85
|
+
securityDeposit = initIx.data.securityDeposit;
|
|
86
|
+
claimerBounty = initIx.data.claimerBounty;
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
return new SolanaSwapData(
|
|
90
|
+
initIx.accounts.offerer,
|
|
91
|
+
initIx.accounts.claimer,
|
|
92
|
+
initIx.accounts.mint,
|
|
93
|
+
initIx.data.swapData.amount,
|
|
94
|
+
paymentHash.toString("hex"),
|
|
95
|
+
initIx.data.swapData.sequence,
|
|
96
|
+
initIx.data.swapData.expiry,
|
|
97
|
+
initIx.data.swapData.nonce,
|
|
98
|
+
initIx.data.swapData.confirmations,
|
|
99
|
+
initIx.data.swapData.payOut,
|
|
100
|
+
SwapTypeEnum.toNumber(initIx.data.swapData.kind),
|
|
101
|
+
payIn,
|
|
102
|
+
initIx.name === "offererInitializePayIn" ? initIx.accounts.offererAta : PublicKey.default,
|
|
103
|
+
initIx.data.swapData.payOut ? initIx.accounts.claimerAta : PublicKey.default,
|
|
104
|
+
securityDeposit,
|
|
105
|
+
claimerBounty,
|
|
106
|
+
txoHash
|
|
107
|
+
);
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
/**
|
|
111
|
+
* Returns async getter for fetching on-demand initialize event swap data
|
|
112
|
+
*
|
|
113
|
+
* @param eventObject
|
|
114
|
+
* @param txoHash
|
|
115
|
+
* @private
|
|
116
|
+
* @returns {() => Promise<SolanaSwapData>} getter to be passed to InitializeEvent constructor
|
|
117
|
+
*/
|
|
118
|
+
private getSwapDataGetter(eventObject: EventObject, txoHash: string): () => Promise<SolanaSwapData> {
|
|
119
|
+
return async () => {
|
|
120
|
+
if(eventObject.instructions==null) eventObject.instructions = await this.getTransactionInstructions(eventObject.signature);
|
|
121
|
+
if(eventObject.instructions==null) return null;
|
|
122
|
+
|
|
123
|
+
const initIx = eventObject.instructions.find(
|
|
124
|
+
ix => ix!=null && (ix.name === "offererInitializePayIn" || ix.name === "offererInitialize")
|
|
125
|
+
) as InitInstruction;
|
|
126
|
+
if(initIx == null) return null;
|
|
127
|
+
|
|
128
|
+
return this.instructionToSwapData(initIx, txoHash);
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
protected parseInitializeEvent(data: IdlEvents<SwapProgram>["InitializeEvent"], eventObject: EventObject): InitializeEvent<SolanaSwapData> {
|
|
133
|
+
const paymentHash: string = Buffer.from(data.hash).toString("hex");
|
|
134
|
+
const txoHash: string = Buffer.from(data.txoHash).toString("hex");
|
|
135
|
+
const escrowHash = toEscrowHash(paymentHash, data.sequence);
|
|
136
|
+
this.logger.debug("InitializeEvent paymentHash: "+paymentHash+" sequence: "+data.sequence.toString(10)+
|
|
137
|
+
" txoHash: "+txoHash+" escrowHash: "+escrowHash);
|
|
138
|
+
return new InitializeEvent<SolanaSwapData>(
|
|
139
|
+
escrowHash,
|
|
140
|
+
SwapTypeEnum.toChainSwapType(data.kind),
|
|
141
|
+
onceAsync<SolanaSwapData>(this.getSwapDataGetter(eventObject, txoHash))
|
|
142
|
+
);
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
protected parseRefundEvent(data: IdlEvents<SwapProgram>["RefundEvent"]): RefundEvent<SolanaSwapData> {
|
|
146
|
+
const paymentHash: string = Buffer.from(data.hash).toString("hex");
|
|
147
|
+
const escrowHash = toEscrowHash(paymentHash, data.sequence);
|
|
148
|
+
this.logger.debug("RefundEvent paymentHash: "+paymentHash+" sequence: "+data.sequence.toString(10)+
|
|
149
|
+
" escrowHash: "+escrowHash);
|
|
150
|
+
return new RefundEvent<SolanaSwapData>(escrowHash);
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
protected parseClaimEvent(data: IdlEvents<SwapProgram>["ClaimEvent"]): ClaimEvent<SolanaSwapData> {
|
|
154
|
+
const secret: string = Buffer.from(data.secret).toString("hex");
|
|
155
|
+
const paymentHash: string = Buffer.from(data.hash).toString("hex");
|
|
156
|
+
const escrowHash = toEscrowHash(paymentHash, data.sequence);
|
|
157
|
+
this.logger.debug("ClaimEvent paymentHash: "+paymentHash+" sequence: "+data.sequence.toString(10)+
|
|
158
|
+
" secret: "+secret+" escrowHash: "+escrowHash);
|
|
159
|
+
return new ClaimEvent<SolanaSwapData>(escrowHash, secret);
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
/**
|
|
163
|
+
* Processes event as received from the chain, parses it & calls event listeners
|
|
164
|
+
*
|
|
165
|
+
* @param eventObject
|
|
166
|
+
* @protected
|
|
167
|
+
*/
|
|
168
|
+
protected async processEvent(eventObject : EventObject) {
|
|
169
|
+
let parsedEvents: SwapEvent<SolanaSwapData>[] = eventObject.events.map(event => {
|
|
170
|
+
let parsedEvent: SwapEvent<SolanaSwapData>;
|
|
171
|
+
switch(event.name) {
|
|
172
|
+
case "ClaimEvent":
|
|
173
|
+
parsedEvent = this.parseClaimEvent(event.data);
|
|
174
|
+
break;
|
|
175
|
+
case "RefundEvent":
|
|
176
|
+
parsedEvent = this.parseRefundEvent(event.data);
|
|
177
|
+
break;
|
|
178
|
+
case "InitializeEvent":
|
|
179
|
+
parsedEvent = this.parseInitializeEvent(event.data, eventObject);
|
|
180
|
+
break;
|
|
181
|
+
}
|
|
182
|
+
(parsedEvent as any).meta = {
|
|
183
|
+
blockTime: eventObject.blockTime,
|
|
184
|
+
timestamp: eventObject.blockTime,
|
|
185
|
+
txId: eventObject.signature
|
|
186
|
+
};
|
|
187
|
+
return parsedEvent;
|
|
188
|
+
}).filter(parsedEvent => parsedEvent!=null);
|
|
189
|
+
|
|
190
|
+
for(let listener of this.listeners) {
|
|
191
|
+
await listener(parsedEvents);
|
|
192
|
+
}
|
|
193
|
+
}
|
|
194
|
+
|
|
195
|
+
/**
|
|
196
|
+
* Returns websocket event handler for specific event type
|
|
197
|
+
*
|
|
198
|
+
* @param name
|
|
199
|
+
* @protected
|
|
200
|
+
* @returns event handler to be passed to program's addEventListener function
|
|
201
|
+
*/
|
|
202
|
+
protected getWsEventHandler<E extends "InitializeEvent" | "RefundEvent" | "ClaimEvent">(
|
|
203
|
+
name: E
|
|
204
|
+
): (data: IdlEvents<SwapProgram>[E], slotNumber: number, signature: string) => void {
|
|
205
|
+
return (data: IdlEvents<SwapProgram>[E], slotNumber: number, signature: string) => {
|
|
206
|
+
this.logger.debug("wsEventHandler: Process signature: ", signature);
|
|
207
|
+
|
|
208
|
+
this.processEvent({
|
|
209
|
+
events: [{name, data: data as any}],
|
|
210
|
+
instructions: null, //Instructions will be fetched on-demand if required
|
|
211
|
+
blockTime: Math.floor(Date.now()/1000),
|
|
212
|
+
signature
|
|
213
|
+
}).then(() => true).catch(e => {
|
|
214
|
+
this.logger.error("wsEventHandler: Error when processing signature: "+signature, e);
|
|
215
|
+
return false;
|
|
216
|
+
});
|
|
217
|
+
};
|
|
218
|
+
}
|
|
219
|
+
|
|
220
|
+
/**
|
|
221
|
+
* Sets up event handlers listening for swap events over websocket
|
|
222
|
+
*
|
|
223
|
+
* @protected
|
|
224
|
+
*/
|
|
225
|
+
protected setupWebsocket() {
|
|
226
|
+
const program = this.solanaSwapProgram.program;
|
|
227
|
+
this.eventListeners.push(program.addEventListener<"InitializeEvent">("InitializeEvent", this.getWsEventHandler("InitializeEvent")));
|
|
228
|
+
this.eventListeners.push(program.addEventListener<"ClaimEvent">("ClaimEvent", this.getWsEventHandler("ClaimEvent")));
|
|
229
|
+
this.eventListeners.push(program.addEventListener<"RefundEvent">("RefundEvent", this.getWsEventHandler("RefundEvent")));
|
|
230
|
+
}
|
|
231
|
+
|
|
232
|
+
init(): Promise<void> {
|
|
233
|
+
this.setupWebsocket();
|
|
234
|
+
return Promise.resolve();
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
async stop(): Promise<void> {
|
|
238
|
+
for(let num of this.eventListeners) {
|
|
239
|
+
await this.solanaSwapProgram.program.removeEventListener(num);
|
|
240
|
+
}
|
|
241
|
+
this.eventListeners = [];
|
|
242
|
+
}
|
|
243
|
+
|
|
244
|
+
registerListener(cbk: EventListener<SolanaSwapData>): void {
|
|
245
|
+
this.listeners.push(cbk);
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
unregisterListener(cbk: EventListener<SolanaSwapData>): boolean {
|
|
249
|
+
const index = this.listeners.indexOf(cbk);
|
|
250
|
+
if(index>=0) {
|
|
251
|
+
this.listeners.splice(index, 1);
|
|
252
|
+
return true;
|
|
253
|
+
}
|
|
254
|
+
return false;
|
|
255
|
+
}
|
|
256
|
+
}
|