@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,85 +1,85 @@
|
|
|
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, SingleInstructionWithAccounts } 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
|
-
export type InitInstruction = SingleInstructionWithAccounts<SwapProgram["instructions"][2 | 3], SwapProgram>;
|
|
15
|
-
/**
|
|
16
|
-
* Solana on-chain event handler for front-end systems without access to fs, uses pure WS to subscribe, might lose
|
|
17
|
-
* out on some events if the network is unreliable, front-end systems should take this into consideration and not
|
|
18
|
-
* rely purely on events
|
|
19
|
-
*/
|
|
20
|
-
export declare class SolanaChainEventsBrowser implements ChainEvents<SolanaSwapData> {
|
|
21
|
-
protected readonly listeners: EventListener<SolanaSwapData>[];
|
|
22
|
-
protected readonly connection: Connection;
|
|
23
|
-
protected readonly solanaSwapProgram: SolanaSwapProgram;
|
|
24
|
-
protected eventListeners: number[];
|
|
25
|
-
protected readonly logger: {
|
|
26
|
-
debug: (msg: any, ...args: any[]) => void;
|
|
27
|
-
info: (msg: any, ...args: any[]) => void;
|
|
28
|
-
warn: (msg: any, ...args: any[]) => void;
|
|
29
|
-
error: (msg: any, ...args: any[]) => void;
|
|
30
|
-
};
|
|
31
|
-
constructor(connection: Connection, solanaSwapContract: SolanaSwapProgram);
|
|
32
|
-
/**
|
|
33
|
-
* Fetches and parses transaction instructions
|
|
34
|
-
*
|
|
35
|
-
* @private
|
|
36
|
-
* @returns {Promise<InstructionWithAccounts<SwapProgram>[]>} array of parsed instructions
|
|
37
|
-
*/
|
|
38
|
-
private getTransactionInstructions;
|
|
39
|
-
/**
|
|
40
|
-
* Converts initialize instruction data into {SolanaSwapData}
|
|
41
|
-
*
|
|
42
|
-
* @param initIx
|
|
43
|
-
* @param txoHash
|
|
44
|
-
* @private
|
|
45
|
-
* @returns {SolanaSwapData} converted and parsed swap data
|
|
46
|
-
*/
|
|
47
|
-
private instructionToSwapData;
|
|
48
|
-
/**
|
|
49
|
-
* Returns async getter for fetching on-demand initialize event swap data
|
|
50
|
-
*
|
|
51
|
-
* @param eventObject
|
|
52
|
-
* @param txoHash
|
|
53
|
-
* @private
|
|
54
|
-
* @returns {() => Promise<SolanaSwapData>} getter to be passed to InitializeEvent constructor
|
|
55
|
-
*/
|
|
56
|
-
private getSwapDataGetter;
|
|
57
|
-
protected parseInitializeEvent(data: IdlEvents<SwapProgram>["InitializeEvent"], eventObject: EventObject): InitializeEvent<SolanaSwapData>;
|
|
58
|
-
protected parseRefundEvent(data: IdlEvents<SwapProgram>["RefundEvent"]): RefundEvent<SolanaSwapData>;
|
|
59
|
-
protected parseClaimEvent(data: IdlEvents<SwapProgram>["ClaimEvent"]): ClaimEvent<SolanaSwapData>;
|
|
60
|
-
/**
|
|
61
|
-
* Processes event as received from the chain, parses it & calls event listeners
|
|
62
|
-
*
|
|
63
|
-
* @param eventObject
|
|
64
|
-
* @protected
|
|
65
|
-
*/
|
|
66
|
-
protected processEvent(eventObject: EventObject): Promise<void>;
|
|
67
|
-
/**
|
|
68
|
-
* Returns websocket event handler for specific event type
|
|
69
|
-
*
|
|
70
|
-
* @param name
|
|
71
|
-
* @protected
|
|
72
|
-
* @returns event handler to be passed to program's addEventListener function
|
|
73
|
-
*/
|
|
74
|
-
protected getWsEventHandler<E extends "InitializeEvent" | "RefundEvent" | "ClaimEvent">(name: E): (data: IdlEvents<SwapProgram>[E], slotNumber: number, signature: string) => void;
|
|
75
|
-
/**
|
|
76
|
-
* Sets up event handlers listening for swap events over websocket
|
|
77
|
-
*
|
|
78
|
-
* @protected
|
|
79
|
-
*/
|
|
80
|
-
protected setupWebsocket(): void;
|
|
81
|
-
init(): Promise<void>;
|
|
82
|
-
stop(): Promise<void>;
|
|
83
|
-
registerListener(cbk: EventListener<SolanaSwapData>): void;
|
|
84
|
-
unregisterListener(cbk: EventListener<SolanaSwapData>): boolean;
|
|
85
|
-
}
|
|
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, SingleInstructionWithAccounts } 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
|
+
export type InitInstruction = SingleInstructionWithAccounts<SwapProgram["instructions"][2 | 3], SwapProgram>;
|
|
15
|
+
/**
|
|
16
|
+
* Solana on-chain event handler for front-end systems without access to fs, uses pure WS to subscribe, might lose
|
|
17
|
+
* out on some events if the network is unreliable, front-end systems should take this into consideration and not
|
|
18
|
+
* rely purely on events
|
|
19
|
+
*/
|
|
20
|
+
export declare class SolanaChainEventsBrowser implements ChainEvents<SolanaSwapData> {
|
|
21
|
+
protected readonly listeners: EventListener<SolanaSwapData>[];
|
|
22
|
+
protected readonly connection: Connection;
|
|
23
|
+
protected readonly solanaSwapProgram: SolanaSwapProgram;
|
|
24
|
+
protected eventListeners: number[];
|
|
25
|
+
protected readonly logger: {
|
|
26
|
+
debug: (msg: any, ...args: any[]) => void;
|
|
27
|
+
info: (msg: any, ...args: any[]) => void;
|
|
28
|
+
warn: (msg: any, ...args: any[]) => void;
|
|
29
|
+
error: (msg: any, ...args: any[]) => void;
|
|
30
|
+
};
|
|
31
|
+
constructor(connection: Connection, solanaSwapContract: SolanaSwapProgram);
|
|
32
|
+
/**
|
|
33
|
+
* Fetches and parses transaction instructions
|
|
34
|
+
*
|
|
35
|
+
* @private
|
|
36
|
+
* @returns {Promise<InstructionWithAccounts<SwapProgram>[]>} array of parsed instructions
|
|
37
|
+
*/
|
|
38
|
+
private getTransactionInstructions;
|
|
39
|
+
/**
|
|
40
|
+
* Converts initialize instruction data into {SolanaSwapData}
|
|
41
|
+
*
|
|
42
|
+
* @param initIx
|
|
43
|
+
* @param txoHash
|
|
44
|
+
* @private
|
|
45
|
+
* @returns {SolanaSwapData} converted and parsed swap data
|
|
46
|
+
*/
|
|
47
|
+
private instructionToSwapData;
|
|
48
|
+
/**
|
|
49
|
+
* Returns async getter for fetching on-demand initialize event swap data
|
|
50
|
+
*
|
|
51
|
+
* @param eventObject
|
|
52
|
+
* @param txoHash
|
|
53
|
+
* @private
|
|
54
|
+
* @returns {() => Promise<SolanaSwapData>} getter to be passed to InitializeEvent constructor
|
|
55
|
+
*/
|
|
56
|
+
private getSwapDataGetter;
|
|
57
|
+
protected parseInitializeEvent(data: IdlEvents<SwapProgram>["InitializeEvent"], eventObject: EventObject): InitializeEvent<SolanaSwapData>;
|
|
58
|
+
protected parseRefundEvent(data: IdlEvents<SwapProgram>["RefundEvent"]): RefundEvent<SolanaSwapData>;
|
|
59
|
+
protected parseClaimEvent(data: IdlEvents<SwapProgram>["ClaimEvent"]): ClaimEvent<SolanaSwapData>;
|
|
60
|
+
/**
|
|
61
|
+
* Processes event as received from the chain, parses it & calls event listeners
|
|
62
|
+
*
|
|
63
|
+
* @param eventObject
|
|
64
|
+
* @protected
|
|
65
|
+
*/
|
|
66
|
+
protected processEvent(eventObject: EventObject): Promise<void>;
|
|
67
|
+
/**
|
|
68
|
+
* Returns websocket event handler for specific event type
|
|
69
|
+
*
|
|
70
|
+
* @param name
|
|
71
|
+
* @protected
|
|
72
|
+
* @returns event handler to be passed to program's addEventListener function
|
|
73
|
+
*/
|
|
74
|
+
protected getWsEventHandler<E extends "InitializeEvent" | "RefundEvent" | "ClaimEvent">(name: E): (data: IdlEvents<SwapProgram>[E], slotNumber: number, signature: string) => void;
|
|
75
|
+
/**
|
|
76
|
+
* Sets up event handlers listening for swap events over websocket
|
|
77
|
+
*
|
|
78
|
+
* @protected
|
|
79
|
+
*/
|
|
80
|
+
protected setupWebsocket(): void;
|
|
81
|
+
init(): Promise<void>;
|
|
82
|
+
stop(): Promise<void>;
|
|
83
|
+
registerListener(cbk: EventListener<SolanaSwapData>): void;
|
|
84
|
+
unregisterListener(cbk: EventListener<SolanaSwapData>): boolean;
|
|
85
|
+
}
|
|
@@ -1,194 +1,194 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.SolanaChainEventsBrowser = void 0;
|
|
4
|
-
const base_1 = require("@atomiqlabs/base");
|
|
5
|
-
const SolanaSwapData_1 = require("../swaps/SolanaSwapData");
|
|
6
|
-
const Utils_1 = require("../../utils/Utils");
|
|
7
|
-
const web3_js_1 = require("@solana/web3.js");
|
|
8
|
-
const BN = require("bn.js");
|
|
9
|
-
const SwapTypeEnum_1 = require("../swaps/SwapTypeEnum");
|
|
10
|
-
const buffer_1 = require("buffer");
|
|
11
|
-
/**
|
|
12
|
-
* Solana on-chain event handler for front-end systems without access to fs, uses pure WS to subscribe, might lose
|
|
13
|
-
* out on some events if the network is unreliable, front-end systems should take this into consideration and not
|
|
14
|
-
* rely purely on events
|
|
15
|
-
*/
|
|
16
|
-
class SolanaChainEventsBrowser {
|
|
17
|
-
constructor(connection, solanaSwapContract) {
|
|
18
|
-
this.listeners = [];
|
|
19
|
-
this.eventListeners = [];
|
|
20
|
-
this.logger = (0, Utils_1.getLogger)("SolanaChainEventsBrowser: ");
|
|
21
|
-
this.connection = connection;
|
|
22
|
-
this.solanaSwapProgram = solanaSwapContract;
|
|
23
|
-
}
|
|
24
|
-
/**
|
|
25
|
-
* Fetches and parses transaction instructions
|
|
26
|
-
*
|
|
27
|
-
* @private
|
|
28
|
-
* @returns {Promise<InstructionWithAccounts<SwapProgram>[]>} array of parsed instructions
|
|
29
|
-
*/
|
|
30
|
-
async getTransactionInstructions(signature) {
|
|
31
|
-
const transaction = await (0, Utils_1.tryWithRetries)(async () => {
|
|
32
|
-
const res = await this.connection.getParsedTransaction(signature, {
|
|
33
|
-
commitment: "confirmed",
|
|
34
|
-
maxSupportedTransactionVersion: 0
|
|
35
|
-
});
|
|
36
|
-
if (res == null)
|
|
37
|
-
throw new Error("Transaction not found!");
|
|
38
|
-
return res;
|
|
39
|
-
});
|
|
40
|
-
if (transaction == null)
|
|
41
|
-
return null;
|
|
42
|
-
if (transaction.meta.err != null)
|
|
43
|
-
return null;
|
|
44
|
-
return this.solanaSwapProgram.Events.decodeInstructions(transaction.transaction.message);
|
|
45
|
-
}
|
|
46
|
-
/**
|
|
47
|
-
* Converts initialize instruction data into {SolanaSwapData}
|
|
48
|
-
*
|
|
49
|
-
* @param initIx
|
|
50
|
-
* @param txoHash
|
|
51
|
-
* @private
|
|
52
|
-
* @returns {SolanaSwapData} converted and parsed swap data
|
|
53
|
-
*/
|
|
54
|
-
instructionToSwapData(initIx, txoHash) {
|
|
55
|
-
const paymentHash = buffer_1.Buffer.from(initIx.data.swapData.hash);
|
|
56
|
-
let securityDeposit = new BN(0);
|
|
57
|
-
let claimerBounty = new BN(0);
|
|
58
|
-
let payIn = true;
|
|
59
|
-
if (initIx.name === "offererInitialize") {
|
|
60
|
-
payIn = false;
|
|
61
|
-
securityDeposit = initIx.data.securityDeposit;
|
|
62
|
-
claimerBounty = initIx.data.claimerBounty;
|
|
63
|
-
}
|
|
64
|
-
return new SolanaSwapData_1.SolanaSwapData(initIx.accounts.offerer, initIx.accounts.claimer, initIx.accounts.mint, initIx.data.swapData.amount, paymentHash.toString("hex"), initIx.data.swapData.sequence, initIx.data.swapData.expiry, initIx.data.swapData.nonce, initIx.data.swapData.confirmations, initIx.data.swapData.payOut, SwapTypeEnum_1.SwapTypeEnum.toNumber(initIx.data.swapData.kind), payIn, initIx.name === "offererInitializePayIn" ? initIx.accounts.offererAta : web3_js_1.PublicKey.default, initIx.data.swapData.payOut ? initIx.accounts.claimerAta : web3_js_1.PublicKey.default, securityDeposit, claimerBounty, txoHash);
|
|
65
|
-
}
|
|
66
|
-
/**
|
|
67
|
-
* Returns async getter for fetching on-demand initialize event swap data
|
|
68
|
-
*
|
|
69
|
-
* @param eventObject
|
|
70
|
-
* @param txoHash
|
|
71
|
-
* @private
|
|
72
|
-
* @returns {() => Promise<SolanaSwapData>} getter to be passed to InitializeEvent constructor
|
|
73
|
-
*/
|
|
74
|
-
getSwapDataGetter(eventObject, txoHash) {
|
|
75
|
-
return async () => {
|
|
76
|
-
if (eventObject.instructions == null)
|
|
77
|
-
eventObject.instructions = await this.getTransactionInstructions(eventObject.signature);
|
|
78
|
-
if (eventObject.instructions == null)
|
|
79
|
-
return null;
|
|
80
|
-
const initIx = eventObject.instructions.find(ix => ix != null && (ix.name === "offererInitializePayIn" || ix.name === "offererInitialize"));
|
|
81
|
-
if (initIx == null)
|
|
82
|
-
return null;
|
|
83
|
-
return this.instructionToSwapData(initIx, txoHash);
|
|
84
|
-
};
|
|
85
|
-
}
|
|
86
|
-
parseInitializeEvent(data, eventObject) {
|
|
87
|
-
const paymentHash = buffer_1.Buffer.from(data.hash).toString("hex");
|
|
88
|
-
const txoHash = buffer_1.Buffer.from(data.txoHash).toString("hex");
|
|
89
|
-
const escrowHash = (0, Utils_1.toEscrowHash)(paymentHash, data.sequence);
|
|
90
|
-
this.logger.debug("InitializeEvent paymentHash: " + paymentHash + " sequence: " + data.sequence.toString(10) +
|
|
91
|
-
" txoHash: " + txoHash + " escrowHash: " + escrowHash);
|
|
92
|
-
return new base_1.InitializeEvent(escrowHash, SwapTypeEnum_1.SwapTypeEnum.toChainSwapType(data.kind), (0, Utils_1.onceAsync)(this.getSwapDataGetter(eventObject, txoHash)));
|
|
93
|
-
}
|
|
94
|
-
parseRefundEvent(data) {
|
|
95
|
-
const paymentHash = buffer_1.Buffer.from(data.hash).toString("hex");
|
|
96
|
-
const escrowHash = (0, Utils_1.toEscrowHash)(paymentHash, data.sequence);
|
|
97
|
-
this.logger.debug("RefundEvent paymentHash: " + paymentHash + " sequence: " + data.sequence.toString(10) +
|
|
98
|
-
" escrowHash: " + escrowHash);
|
|
99
|
-
return new base_1.RefundEvent(escrowHash);
|
|
100
|
-
}
|
|
101
|
-
parseClaimEvent(data) {
|
|
102
|
-
const secret = buffer_1.Buffer.from(data.secret).toString("hex");
|
|
103
|
-
const paymentHash = buffer_1.Buffer.from(data.hash).toString("hex");
|
|
104
|
-
const escrowHash = (0, Utils_1.toEscrowHash)(paymentHash, data.sequence);
|
|
105
|
-
this.logger.debug("ClaimEvent paymentHash: " + paymentHash + " sequence: " + data.sequence.toString(10) +
|
|
106
|
-
" secret: " + secret + " escrowHash: " + escrowHash);
|
|
107
|
-
return new base_1.ClaimEvent(escrowHash, secret);
|
|
108
|
-
}
|
|
109
|
-
/**
|
|
110
|
-
* Processes event as received from the chain, parses it & calls event listeners
|
|
111
|
-
*
|
|
112
|
-
* @param eventObject
|
|
113
|
-
* @protected
|
|
114
|
-
*/
|
|
115
|
-
async processEvent(eventObject) {
|
|
116
|
-
let parsedEvents = eventObject.events.map(event => {
|
|
117
|
-
let parsedEvent;
|
|
118
|
-
switch (event.name) {
|
|
119
|
-
case "ClaimEvent":
|
|
120
|
-
parsedEvent = this.parseClaimEvent(event.data);
|
|
121
|
-
break;
|
|
122
|
-
case "RefundEvent":
|
|
123
|
-
parsedEvent = this.parseRefundEvent(event.data);
|
|
124
|
-
break;
|
|
125
|
-
case "InitializeEvent":
|
|
126
|
-
parsedEvent = this.parseInitializeEvent(event.data, eventObject);
|
|
127
|
-
break;
|
|
128
|
-
}
|
|
129
|
-
parsedEvent.meta = {
|
|
130
|
-
blockTime: eventObject.blockTime,
|
|
131
|
-
timestamp: eventObject.blockTime,
|
|
132
|
-
txId: eventObject.signature
|
|
133
|
-
};
|
|
134
|
-
return parsedEvent;
|
|
135
|
-
}).filter(parsedEvent => parsedEvent != null);
|
|
136
|
-
for (let listener of this.listeners) {
|
|
137
|
-
await listener(parsedEvents);
|
|
138
|
-
}
|
|
139
|
-
}
|
|
140
|
-
/**
|
|
141
|
-
* Returns websocket event handler for specific event type
|
|
142
|
-
*
|
|
143
|
-
* @param name
|
|
144
|
-
* @protected
|
|
145
|
-
* @returns event handler to be passed to program's addEventListener function
|
|
146
|
-
*/
|
|
147
|
-
getWsEventHandler(name) {
|
|
148
|
-
return (data, slotNumber, signature) => {
|
|
149
|
-
this.logger.debug("wsEventHandler: Process signature: ", signature);
|
|
150
|
-
this.processEvent({
|
|
151
|
-
events: [{ name, data: data }],
|
|
152
|
-
instructions: null,
|
|
153
|
-
blockTime: Math.floor(Date.now() / 1000),
|
|
154
|
-
signature
|
|
155
|
-
}).then(() => true).catch(e => {
|
|
156
|
-
this.logger.error("wsEventHandler: Error when processing signature: " + signature, e);
|
|
157
|
-
return false;
|
|
158
|
-
});
|
|
159
|
-
};
|
|
160
|
-
}
|
|
161
|
-
/**
|
|
162
|
-
* Sets up event handlers listening for swap events over websocket
|
|
163
|
-
*
|
|
164
|
-
* @protected
|
|
165
|
-
*/
|
|
166
|
-
setupWebsocket() {
|
|
167
|
-
const program = this.solanaSwapProgram.program;
|
|
168
|
-
this.eventListeners.push(program.addEventListener("InitializeEvent", this.getWsEventHandler("InitializeEvent")));
|
|
169
|
-
this.eventListeners.push(program.addEventListener("ClaimEvent", this.getWsEventHandler("ClaimEvent")));
|
|
170
|
-
this.eventListeners.push(program.addEventListener("RefundEvent", this.getWsEventHandler("RefundEvent")));
|
|
171
|
-
}
|
|
172
|
-
init() {
|
|
173
|
-
this.setupWebsocket();
|
|
174
|
-
return Promise.resolve();
|
|
175
|
-
}
|
|
176
|
-
async stop() {
|
|
177
|
-
for (let num of this.eventListeners) {
|
|
178
|
-
await this.solanaSwapProgram.program.removeEventListener(num);
|
|
179
|
-
}
|
|
180
|
-
this.eventListeners = [];
|
|
181
|
-
}
|
|
182
|
-
registerListener(cbk) {
|
|
183
|
-
this.listeners.push(cbk);
|
|
184
|
-
}
|
|
185
|
-
unregisterListener(cbk) {
|
|
186
|
-
const index = this.listeners.indexOf(cbk);
|
|
187
|
-
if (index >= 0) {
|
|
188
|
-
this.listeners.splice(index, 1);
|
|
189
|
-
return true;
|
|
190
|
-
}
|
|
191
|
-
return false;
|
|
192
|
-
}
|
|
193
|
-
}
|
|
194
|
-
exports.SolanaChainEventsBrowser = SolanaChainEventsBrowser;
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.SolanaChainEventsBrowser = void 0;
|
|
4
|
+
const base_1 = require("@atomiqlabs/base");
|
|
5
|
+
const SolanaSwapData_1 = require("../swaps/SolanaSwapData");
|
|
6
|
+
const Utils_1 = require("../../utils/Utils");
|
|
7
|
+
const web3_js_1 = require("@solana/web3.js");
|
|
8
|
+
const BN = require("bn.js");
|
|
9
|
+
const SwapTypeEnum_1 = require("../swaps/SwapTypeEnum");
|
|
10
|
+
const buffer_1 = require("buffer");
|
|
11
|
+
/**
|
|
12
|
+
* Solana on-chain event handler for front-end systems without access to fs, uses pure WS to subscribe, might lose
|
|
13
|
+
* out on some events if the network is unreliable, front-end systems should take this into consideration and not
|
|
14
|
+
* rely purely on events
|
|
15
|
+
*/
|
|
16
|
+
class SolanaChainEventsBrowser {
|
|
17
|
+
constructor(connection, solanaSwapContract) {
|
|
18
|
+
this.listeners = [];
|
|
19
|
+
this.eventListeners = [];
|
|
20
|
+
this.logger = (0, Utils_1.getLogger)("SolanaChainEventsBrowser: ");
|
|
21
|
+
this.connection = connection;
|
|
22
|
+
this.solanaSwapProgram = solanaSwapContract;
|
|
23
|
+
}
|
|
24
|
+
/**
|
|
25
|
+
* Fetches and parses transaction instructions
|
|
26
|
+
*
|
|
27
|
+
* @private
|
|
28
|
+
* @returns {Promise<InstructionWithAccounts<SwapProgram>[]>} array of parsed instructions
|
|
29
|
+
*/
|
|
30
|
+
async getTransactionInstructions(signature) {
|
|
31
|
+
const transaction = await (0, Utils_1.tryWithRetries)(async () => {
|
|
32
|
+
const res = await this.connection.getParsedTransaction(signature, {
|
|
33
|
+
commitment: "confirmed",
|
|
34
|
+
maxSupportedTransactionVersion: 0
|
|
35
|
+
});
|
|
36
|
+
if (res == null)
|
|
37
|
+
throw new Error("Transaction not found!");
|
|
38
|
+
return res;
|
|
39
|
+
});
|
|
40
|
+
if (transaction == null)
|
|
41
|
+
return null;
|
|
42
|
+
if (transaction.meta.err != null)
|
|
43
|
+
return null;
|
|
44
|
+
return this.solanaSwapProgram.Events.decodeInstructions(transaction.transaction.message);
|
|
45
|
+
}
|
|
46
|
+
/**
|
|
47
|
+
* Converts initialize instruction data into {SolanaSwapData}
|
|
48
|
+
*
|
|
49
|
+
* @param initIx
|
|
50
|
+
* @param txoHash
|
|
51
|
+
* @private
|
|
52
|
+
* @returns {SolanaSwapData} converted and parsed swap data
|
|
53
|
+
*/
|
|
54
|
+
instructionToSwapData(initIx, txoHash) {
|
|
55
|
+
const paymentHash = buffer_1.Buffer.from(initIx.data.swapData.hash);
|
|
56
|
+
let securityDeposit = new BN(0);
|
|
57
|
+
let claimerBounty = new BN(0);
|
|
58
|
+
let payIn = true;
|
|
59
|
+
if (initIx.name === "offererInitialize") {
|
|
60
|
+
payIn = false;
|
|
61
|
+
securityDeposit = initIx.data.securityDeposit;
|
|
62
|
+
claimerBounty = initIx.data.claimerBounty;
|
|
63
|
+
}
|
|
64
|
+
return new SolanaSwapData_1.SolanaSwapData(initIx.accounts.offerer, initIx.accounts.claimer, initIx.accounts.mint, initIx.data.swapData.amount, paymentHash.toString("hex"), initIx.data.swapData.sequence, initIx.data.swapData.expiry, initIx.data.swapData.nonce, initIx.data.swapData.confirmations, initIx.data.swapData.payOut, SwapTypeEnum_1.SwapTypeEnum.toNumber(initIx.data.swapData.kind), payIn, initIx.name === "offererInitializePayIn" ? initIx.accounts.offererAta : web3_js_1.PublicKey.default, initIx.data.swapData.payOut ? initIx.accounts.claimerAta : web3_js_1.PublicKey.default, securityDeposit, claimerBounty, txoHash);
|
|
65
|
+
}
|
|
66
|
+
/**
|
|
67
|
+
* Returns async getter for fetching on-demand initialize event swap data
|
|
68
|
+
*
|
|
69
|
+
* @param eventObject
|
|
70
|
+
* @param txoHash
|
|
71
|
+
* @private
|
|
72
|
+
* @returns {() => Promise<SolanaSwapData>} getter to be passed to InitializeEvent constructor
|
|
73
|
+
*/
|
|
74
|
+
getSwapDataGetter(eventObject, txoHash) {
|
|
75
|
+
return async () => {
|
|
76
|
+
if (eventObject.instructions == null)
|
|
77
|
+
eventObject.instructions = await this.getTransactionInstructions(eventObject.signature);
|
|
78
|
+
if (eventObject.instructions == null)
|
|
79
|
+
return null;
|
|
80
|
+
const initIx = eventObject.instructions.find(ix => ix != null && (ix.name === "offererInitializePayIn" || ix.name === "offererInitialize"));
|
|
81
|
+
if (initIx == null)
|
|
82
|
+
return null;
|
|
83
|
+
return this.instructionToSwapData(initIx, txoHash);
|
|
84
|
+
};
|
|
85
|
+
}
|
|
86
|
+
parseInitializeEvent(data, eventObject) {
|
|
87
|
+
const paymentHash = buffer_1.Buffer.from(data.hash).toString("hex");
|
|
88
|
+
const txoHash = buffer_1.Buffer.from(data.txoHash).toString("hex");
|
|
89
|
+
const escrowHash = (0, Utils_1.toEscrowHash)(paymentHash, data.sequence);
|
|
90
|
+
this.logger.debug("InitializeEvent paymentHash: " + paymentHash + " sequence: " + data.sequence.toString(10) +
|
|
91
|
+
" txoHash: " + txoHash + " escrowHash: " + escrowHash);
|
|
92
|
+
return new base_1.InitializeEvent(escrowHash, SwapTypeEnum_1.SwapTypeEnum.toChainSwapType(data.kind), (0, Utils_1.onceAsync)(this.getSwapDataGetter(eventObject, txoHash)));
|
|
93
|
+
}
|
|
94
|
+
parseRefundEvent(data) {
|
|
95
|
+
const paymentHash = buffer_1.Buffer.from(data.hash).toString("hex");
|
|
96
|
+
const escrowHash = (0, Utils_1.toEscrowHash)(paymentHash, data.sequence);
|
|
97
|
+
this.logger.debug("RefundEvent paymentHash: " + paymentHash + " sequence: " + data.sequence.toString(10) +
|
|
98
|
+
" escrowHash: " + escrowHash);
|
|
99
|
+
return new base_1.RefundEvent(escrowHash);
|
|
100
|
+
}
|
|
101
|
+
parseClaimEvent(data) {
|
|
102
|
+
const secret = buffer_1.Buffer.from(data.secret).toString("hex");
|
|
103
|
+
const paymentHash = buffer_1.Buffer.from(data.hash).toString("hex");
|
|
104
|
+
const escrowHash = (0, Utils_1.toEscrowHash)(paymentHash, data.sequence);
|
|
105
|
+
this.logger.debug("ClaimEvent paymentHash: " + paymentHash + " sequence: " + data.sequence.toString(10) +
|
|
106
|
+
" secret: " + secret + " escrowHash: " + escrowHash);
|
|
107
|
+
return new base_1.ClaimEvent(escrowHash, secret);
|
|
108
|
+
}
|
|
109
|
+
/**
|
|
110
|
+
* Processes event as received from the chain, parses it & calls event listeners
|
|
111
|
+
*
|
|
112
|
+
* @param eventObject
|
|
113
|
+
* @protected
|
|
114
|
+
*/
|
|
115
|
+
async processEvent(eventObject) {
|
|
116
|
+
let parsedEvents = eventObject.events.map(event => {
|
|
117
|
+
let parsedEvent;
|
|
118
|
+
switch (event.name) {
|
|
119
|
+
case "ClaimEvent":
|
|
120
|
+
parsedEvent = this.parseClaimEvent(event.data);
|
|
121
|
+
break;
|
|
122
|
+
case "RefundEvent":
|
|
123
|
+
parsedEvent = this.parseRefundEvent(event.data);
|
|
124
|
+
break;
|
|
125
|
+
case "InitializeEvent":
|
|
126
|
+
parsedEvent = this.parseInitializeEvent(event.data, eventObject);
|
|
127
|
+
break;
|
|
128
|
+
}
|
|
129
|
+
parsedEvent.meta = {
|
|
130
|
+
blockTime: eventObject.blockTime,
|
|
131
|
+
timestamp: eventObject.blockTime,
|
|
132
|
+
txId: eventObject.signature
|
|
133
|
+
};
|
|
134
|
+
return parsedEvent;
|
|
135
|
+
}).filter(parsedEvent => parsedEvent != null);
|
|
136
|
+
for (let listener of this.listeners) {
|
|
137
|
+
await listener(parsedEvents);
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
/**
|
|
141
|
+
* Returns websocket event handler for specific event type
|
|
142
|
+
*
|
|
143
|
+
* @param name
|
|
144
|
+
* @protected
|
|
145
|
+
* @returns event handler to be passed to program's addEventListener function
|
|
146
|
+
*/
|
|
147
|
+
getWsEventHandler(name) {
|
|
148
|
+
return (data, slotNumber, signature) => {
|
|
149
|
+
this.logger.debug("wsEventHandler: Process signature: ", signature);
|
|
150
|
+
this.processEvent({
|
|
151
|
+
events: [{ name, data: data }],
|
|
152
|
+
instructions: null,
|
|
153
|
+
blockTime: Math.floor(Date.now() / 1000),
|
|
154
|
+
signature
|
|
155
|
+
}).then(() => true).catch(e => {
|
|
156
|
+
this.logger.error("wsEventHandler: Error when processing signature: " + signature, e);
|
|
157
|
+
return false;
|
|
158
|
+
});
|
|
159
|
+
};
|
|
160
|
+
}
|
|
161
|
+
/**
|
|
162
|
+
* Sets up event handlers listening for swap events over websocket
|
|
163
|
+
*
|
|
164
|
+
* @protected
|
|
165
|
+
*/
|
|
166
|
+
setupWebsocket() {
|
|
167
|
+
const program = this.solanaSwapProgram.program;
|
|
168
|
+
this.eventListeners.push(program.addEventListener("InitializeEvent", this.getWsEventHandler("InitializeEvent")));
|
|
169
|
+
this.eventListeners.push(program.addEventListener("ClaimEvent", this.getWsEventHandler("ClaimEvent")));
|
|
170
|
+
this.eventListeners.push(program.addEventListener("RefundEvent", this.getWsEventHandler("RefundEvent")));
|
|
171
|
+
}
|
|
172
|
+
init() {
|
|
173
|
+
this.setupWebsocket();
|
|
174
|
+
return Promise.resolve();
|
|
175
|
+
}
|
|
176
|
+
async stop() {
|
|
177
|
+
for (let num of this.eventListeners) {
|
|
178
|
+
await this.solanaSwapProgram.program.removeEventListener(num);
|
|
179
|
+
}
|
|
180
|
+
this.eventListeners = [];
|
|
181
|
+
}
|
|
182
|
+
registerListener(cbk) {
|
|
183
|
+
this.listeners.push(cbk);
|
|
184
|
+
}
|
|
185
|
+
unregisterListener(cbk) {
|
|
186
|
+
const index = this.listeners.indexOf(cbk);
|
|
187
|
+
if (index >= 0) {
|
|
188
|
+
this.listeners.splice(index, 1);
|
|
189
|
+
return true;
|
|
190
|
+
}
|
|
191
|
+
return false;
|
|
192
|
+
}
|
|
193
|
+
}
|
|
194
|
+
exports.SolanaChainEventsBrowser = SolanaChainEventsBrowser;
|
|
@@ -1,40 +1,40 @@
|
|
|
1
|
-
/// <reference types="node" />
|
|
2
|
-
import { Idl, Program } from "@coral-xyz/anchor";
|
|
3
|
-
import { SolanaChainInterface } from "../chain/SolanaChainInterface";
|
|
4
|
-
import { SolanaProgramEvents } from "./modules/SolanaProgramEvents";
|
|
5
|
-
import { Keypair, PublicKey } from "@solana/web3.js";
|
|
6
|
-
import { Buffer } from "buffer";
|
|
7
|
-
/**
|
|
8
|
-
* Base class providing program specific utilities
|
|
9
|
-
*/
|
|
10
|
-
export declare class SolanaProgramBase<T extends Idl> {
|
|
11
|
-
protected readonly logger: {
|
|
12
|
-
debug: (msg: any, ...args: any[]) => void;
|
|
13
|
-
info: (msg: any, ...args: any[]) => void;
|
|
14
|
-
warn: (msg: any, ...args: any[]) => void;
|
|
15
|
-
error: (msg: any, ...args: any[]) => void;
|
|
16
|
-
};
|
|
17
|
-
program: Program<T>;
|
|
18
|
-
readonly Events: SolanaProgramEvents<T>;
|
|
19
|
-
readonly Chain: SolanaChainInterface;
|
|
20
|
-
constructor(chainInterface: SolanaChainInterface, programIdl: any, programAddress?: string);
|
|
21
|
-
/**
|
|
22
|
-
* Derives static PDA address from the seed
|
|
23
|
-
*
|
|
24
|
-
* @param seed
|
|
25
|
-
*/
|
|
26
|
-
pda(seed: string): PublicKey;
|
|
27
|
-
/**
|
|
28
|
-
* Returns a function for deriving a dynamic PDA address from seed + dynamic arguments
|
|
29
|
-
*
|
|
30
|
-
* @param seed
|
|
31
|
-
* @param func function translating the function argument to Buffer[]
|
|
32
|
-
*/
|
|
33
|
-
pda<T extends Array<any>>(seed: string, func: (...args: T) => Buffer[]): (...args: T) => PublicKey;
|
|
34
|
-
/**
|
|
35
|
-
* Returns a function for deriving a dynamic deterministic keypair from dynamic arguments
|
|
36
|
-
*
|
|
37
|
-
* @param func function translating the function argument to Buffer[] to be used for deriving the keypair
|
|
38
|
-
*/
|
|
39
|
-
keypair<T extends Array<any>>(func: (...args: T) => Buffer[]): (...args: T) => Keypair;
|
|
40
|
-
}
|
|
1
|
+
/// <reference types="node" />
|
|
2
|
+
import { Idl, Program } from "@coral-xyz/anchor";
|
|
3
|
+
import { SolanaChainInterface } from "../chain/SolanaChainInterface";
|
|
4
|
+
import { SolanaProgramEvents } from "./modules/SolanaProgramEvents";
|
|
5
|
+
import { Keypair, PublicKey } from "@solana/web3.js";
|
|
6
|
+
import { Buffer } from "buffer";
|
|
7
|
+
/**
|
|
8
|
+
* Base class providing program specific utilities
|
|
9
|
+
*/
|
|
10
|
+
export declare class SolanaProgramBase<T extends Idl> {
|
|
11
|
+
protected readonly logger: {
|
|
12
|
+
debug: (msg: any, ...args: any[]) => void;
|
|
13
|
+
info: (msg: any, ...args: any[]) => void;
|
|
14
|
+
warn: (msg: any, ...args: any[]) => void;
|
|
15
|
+
error: (msg: any, ...args: any[]) => void;
|
|
16
|
+
};
|
|
17
|
+
program: Program<T>;
|
|
18
|
+
readonly Events: SolanaProgramEvents<T>;
|
|
19
|
+
readonly Chain: SolanaChainInterface;
|
|
20
|
+
constructor(chainInterface: SolanaChainInterface, programIdl: any, programAddress?: string);
|
|
21
|
+
/**
|
|
22
|
+
* Derives static PDA address from the seed
|
|
23
|
+
*
|
|
24
|
+
* @param seed
|
|
25
|
+
*/
|
|
26
|
+
pda(seed: string): PublicKey;
|
|
27
|
+
/**
|
|
28
|
+
* Returns a function for deriving a dynamic PDA address from seed + dynamic arguments
|
|
29
|
+
*
|
|
30
|
+
* @param seed
|
|
31
|
+
* @param func function translating the function argument to Buffer[]
|
|
32
|
+
*/
|
|
33
|
+
pda<T extends Array<any>>(seed: string, func: (...args: T) => Buffer[]): (...args: T) => PublicKey;
|
|
34
|
+
/**
|
|
35
|
+
* Returns a function for deriving a dynamic deterministic keypair from dynamic arguments
|
|
36
|
+
*
|
|
37
|
+
* @param func function translating the function argument to Buffer[] to be used for deriving the keypair
|
|
38
|
+
*/
|
|
39
|
+
keypair<T extends Array<any>>(func: (...args: T) => Buffer[]): (...args: T) => Keypair;
|
|
40
|
+
}
|