@atomiqlabs/chain-starknet 1.0.0-beta.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 -0
- package/dist/get_serialized_block.d.ts +1 -0
- package/dist/get_serialized_block.js +28 -0
- package/dist/index.d.ts +34 -0
- package/dist/index.js +50 -0
- package/dist/starknet/StarknetChainType.d.ts +9 -0
- package/dist/starknet/StarknetChainType.js +2 -0
- package/dist/starknet/StarknetInitializer.d.ts +18 -0
- package/dist/starknet/StarknetInitializer.js +49 -0
- package/dist/starknet/base/StarknetAction.d.ts +27 -0
- package/dist/starknet/base/StarknetAction.js +73 -0
- package/dist/starknet/base/StarknetBase.d.ts +34 -0
- package/dist/starknet/base/StarknetBase.js +29 -0
- package/dist/starknet/base/StarknetModule.d.ts +14 -0
- package/dist/starknet/base/StarknetModule.js +13 -0
- package/dist/starknet/base/modules/ERC20Abi.d.ts +755 -0
- package/dist/starknet/base/modules/ERC20Abi.js +1032 -0
- package/dist/starknet/base/modules/StarknetAccounts.d.ts +6 -0
- package/dist/starknet/base/modules/StarknetAccounts.js +24 -0
- package/dist/starknet/base/modules/StarknetAddresses.d.ts +9 -0
- package/dist/starknet/base/modules/StarknetAddresses.js +26 -0
- package/dist/starknet/base/modules/StarknetBlocks.d.ts +19 -0
- package/dist/starknet/base/modules/StarknetBlocks.js +49 -0
- package/dist/starknet/base/modules/StarknetEvents.d.ts +44 -0
- package/dist/starknet/base/modules/StarknetEvents.js +88 -0
- package/dist/starknet/base/modules/StarknetFees.d.ts +55 -0
- package/dist/starknet/base/modules/StarknetFees.js +102 -0
- package/dist/starknet/base/modules/StarknetSignatures.d.ts +30 -0
- package/dist/starknet/base/modules/StarknetSignatures.js +71 -0
- package/dist/starknet/base/modules/StarknetTokens.d.ts +67 -0
- package/dist/starknet/base/modules/StarknetTokens.js +97 -0
- package/dist/starknet/base/modules/StarknetTransactions.d.ts +87 -0
- package/dist/starknet/base/modules/StarknetTransactions.js +226 -0
- package/dist/starknet/btcrelay/BtcRelayAbi.d.ts +250 -0
- package/dist/starknet/btcrelay/BtcRelayAbi.js +341 -0
- package/dist/starknet/btcrelay/StarknetBtcRelay.d.ts +166 -0
- package/dist/starknet/btcrelay/StarknetBtcRelay.js +323 -0
- package/dist/starknet/btcrelay/headers/StarknetBtcHeader.d.ts +32 -0
- package/dist/starknet/btcrelay/headers/StarknetBtcHeader.js +74 -0
- package/dist/starknet/btcrelay/headers/StarknetBtcStoredHeader.d.ts +52 -0
- package/dist/starknet/btcrelay/headers/StarknetBtcStoredHeader.js +113 -0
- package/dist/starknet/contract/StarknetContractBase.d.ts +13 -0
- package/dist/starknet/contract/StarknetContractBase.js +18 -0
- package/dist/starknet/contract/modules/StarknetContractEvents.d.ts +40 -0
- package/dist/starknet/contract/modules/StarknetContractEvents.js +77 -0
- package/dist/starknet/events/StarknetChainEvents.d.ts +19 -0
- package/dist/starknet/events/StarknetChainEvents.js +51 -0
- package/dist/starknet/events/StarknetChainEventsBrowser.d.ts +73 -0
- package/dist/starknet/events/StarknetChainEventsBrowser.js +210 -0
- package/dist/starknet/swaps/EscrowManagerAbi.d.ts +445 -0
- package/dist/starknet/swaps/EscrowManagerAbi.js +601 -0
- package/dist/starknet/swaps/StarknetSwapContract.d.ts +215 -0
- package/dist/starknet/swaps/StarknetSwapContract.js +452 -0
- package/dist/starknet/swaps/StarknetSwapData.d.ts +74 -0
- package/dist/starknet/swaps/StarknetSwapData.js +316 -0
- package/dist/starknet/swaps/StarknetSwapModule.d.ts +9 -0
- package/dist/starknet/swaps/StarknetSwapModule.js +12 -0
- package/dist/starknet/swaps/handlers/IHandler.d.ts +13 -0
- package/dist/starknet/swaps/handlers/IHandler.js +2 -0
- package/dist/starknet/swaps/handlers/claim/ClaimHandlers.d.ts +13 -0
- package/dist/starknet/swaps/handlers/claim/ClaimHandlers.js +13 -0
- package/dist/starknet/swaps/handlers/claim/HashlockClaimHandler.d.ts +22 -0
- package/dist/starknet/swaps/handlers/claim/HashlockClaimHandler.js +44 -0
- package/dist/starknet/swaps/handlers/claim/btc/BitcoinNoncedOutputClaimHandler.d.ts +25 -0
- package/dist/starknet/swaps/handlers/claim/btc/BitcoinNoncedOutputClaimHandler.js +48 -0
- package/dist/starknet/swaps/handlers/claim/btc/BitcoinOutputClaimHandler.d.ts +26 -0
- package/dist/starknet/swaps/handlers/claim/btc/BitcoinOutputClaimHandler.js +40 -0
- package/dist/starknet/swaps/handlers/claim/btc/BitcoinTxIdClaimHandler.d.ts +20 -0
- package/dist/starknet/swaps/handlers/claim/btc/BitcoinTxIdClaimHandler.js +29 -0
- package/dist/starknet/swaps/handlers/claim/btc/IBitcoinClaimHandler.d.ts +64 -0
- package/dist/starknet/swaps/handlers/claim/btc/IBitcoinClaimHandler.js +86 -0
- package/dist/starknet/swaps/handlers/refund/TimelockRefundHandler.d.ts +17 -0
- package/dist/starknet/swaps/handlers/refund/TimelockRefundHandler.js +27 -0
- package/dist/starknet/swaps/modules/StarknetLpVault.d.ts +69 -0
- package/dist/starknet/swaps/modules/StarknetLpVault.js +122 -0
- package/dist/starknet/swaps/modules/StarknetSwapClaim.d.ts +53 -0
- package/dist/starknet/swaps/modules/StarknetSwapClaim.js +100 -0
- package/dist/starknet/swaps/modules/StarknetSwapInit.d.ts +84 -0
- package/dist/starknet/swaps/modules/StarknetSwapInit.js +164 -0
- package/dist/starknet/swaps/modules/StarknetSwapRefund.d.ts +64 -0
- package/dist/starknet/swaps/modules/StarknetSwapRefund.js +131 -0
- package/dist/starknet/swaps/modules/SwapClaim.d.ts +54 -0
- package/dist/starknet/swaps/modules/SwapClaim.js +115 -0
- package/dist/starknet/swaps/modules/SwapInit.d.ts +79 -0
- package/dist/starknet/swaps/modules/SwapInit.js +174 -0
- package/dist/starknet/swaps/modules/SwapRefund.d.ts +63 -0
- package/dist/starknet/swaps/modules/SwapRefund.js +149 -0
- package/dist/starknet/wallet/StarknetKeypairWallet.d.ts +6 -0
- package/dist/starknet/wallet/StarknetKeypairWallet.js +26 -0
- package/dist/starknet/wallet/StarknetSigner.d.ts +12 -0
- package/dist/starknet/wallet/StarknetSigner.js +46 -0
- package/dist/utils/Utils.d.ts +38 -0
- package/dist/utils/Utils.js +255 -0
- package/package.json +39 -0
- package/src/index.ts +41 -0
- package/src/starknet/StarknetChainType.ts +20 -0
- package/src/starknet/StarknetInitializer.ts +75 -0
- package/src/starknet/base/StarknetAction.ts +90 -0
- package/src/starknet/base/StarknetBase.ts +56 -0
- package/src/starknet/base/StarknetModule.ts +20 -0
- package/src/starknet/base/modules/ERC20Abi.ts +1029 -0
- package/src/starknet/base/modules/StarknetAccounts.ts +26 -0
- package/src/starknet/base/modules/StarknetAddresses.ts +23 -0
- package/src/starknet/base/modules/StarknetBlocks.ts +59 -0
- package/src/starknet/base/modules/StarknetEvents.ts +105 -0
- package/src/starknet/base/modules/StarknetFees.ts +136 -0
- package/src/starknet/base/modules/StarknetSignatures.ts +91 -0
- package/src/starknet/base/modules/StarknetTokens.ts +116 -0
- package/src/starknet/base/modules/StarknetTransactions.ts +254 -0
- package/src/starknet/btcrelay/BtcRelayAbi.ts +338 -0
- package/src/starknet/btcrelay/StarknetBtcRelay.ts +415 -0
- package/src/starknet/btcrelay/headers/StarknetBtcHeader.ts +101 -0
- package/src/starknet/btcrelay/headers/StarknetBtcStoredHeader.ts +142 -0
- package/src/starknet/contract/StarknetContractBase.ts +29 -0
- package/src/starknet/contract/modules/StarknetContractEvents.ts +108 -0
- package/src/starknet/events/StarknetChainEvents.ts +63 -0
- package/src/starknet/events/StarknetChainEventsBrowser.ts +289 -0
- package/src/starknet/swaps/EscrowManagerAbi.ts +600 -0
- package/src/starknet/swaps/StarknetSwapContract.ts +694 -0
- package/src/starknet/swaps/StarknetSwapData.ts +441 -0
- package/src/starknet/swaps/StarknetSwapModule.ts +17 -0
- package/src/starknet/swaps/handlers/IHandler.ts +20 -0
- package/src/starknet/swaps/handlers/claim/ClaimHandlers.ts +23 -0
- package/src/starknet/swaps/handlers/claim/HashlockClaimHandler.ts +54 -0
- package/src/starknet/swaps/handlers/claim/btc/BitcoinNoncedOutputClaimHandler.ts +73 -0
- package/src/starknet/swaps/handlers/claim/btc/BitcoinOutputClaimHandler.ts +67 -0
- package/src/starknet/swaps/handlers/claim/btc/BitcoinTxIdClaimHandler.ts +49 -0
- package/src/starknet/swaps/handlers/claim/btc/IBitcoinClaimHandler.ts +151 -0
- package/src/starknet/swaps/handlers/refund/TimelockRefundHandler.ts +39 -0
- package/src/starknet/swaps/modules/StarknetLpVault.ts +148 -0
- package/src/starknet/swaps/modules/StarknetSwapClaim.ts +142 -0
- package/src/starknet/swaps/modules/StarknetSwapInit.ts +226 -0
- package/src/starknet/swaps/modules/StarknetSwapRefund.ts +202 -0
- package/src/starknet/wallet/StarknetKeypairWallet.ts +34 -0
- package/src/starknet/wallet/StarknetSigner.ts +55 -0
- package/src/utils/Utils.ts +247 -0
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
import {Abi} from "abi-wan-kanabi";
|
|
2
|
+
import {EventToPrimitiveType, ExtractAbiEventNames} from "abi-wan-kanabi/dist/kanabi";
|
|
3
|
+
import {StarknetEvent, StarknetEvents} from "../../base/modules/StarknetEvents";
|
|
4
|
+
import {CallData, events, hash} from "starknet";
|
|
5
|
+
import {StarknetContractBase} from "../StarknetContractBase";
|
|
6
|
+
import {toHex} from "../../../utils/Utils";
|
|
7
|
+
|
|
8
|
+
export type StarknetAbiEvent<TAbi extends Abi, TEventName extends ExtractAbiEventNames<TAbi>> = {
|
|
9
|
+
name: TEventName,
|
|
10
|
+
params: EventToPrimitiveType<TAbi, TEventName>,
|
|
11
|
+
txHash: string,
|
|
12
|
+
blockHash: string,
|
|
13
|
+
blockNumber: number,
|
|
14
|
+
keys: string[],
|
|
15
|
+
data: string[]
|
|
16
|
+
};
|
|
17
|
+
|
|
18
|
+
export class StarknetContractEvents<TAbi extends Abi> extends StarknetEvents {
|
|
19
|
+
|
|
20
|
+
readonly root: StarknetContractBase<TAbi>;
|
|
21
|
+
readonly abi: TAbi;
|
|
22
|
+
|
|
23
|
+
constructor(root: StarknetContractBase<TAbi>, abi: TAbi) {
|
|
24
|
+
super(root);
|
|
25
|
+
this.abi = abi;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
private toStarknetAbiEvents<T extends ExtractAbiEventNames<TAbi>>(blockEvents: StarknetEvent[]): StarknetAbiEvent<TAbi, T>[] {
|
|
29
|
+
const abiEvents = events.getAbiEvents(this.abi);
|
|
30
|
+
const abiStructs = CallData.getAbiStruct(this.abi);
|
|
31
|
+
const abiEnums = CallData.getAbiEnum(this.abi);
|
|
32
|
+
|
|
33
|
+
const result = events.parseEvents(blockEvents, abiEvents, abiStructs, abiEnums);
|
|
34
|
+
if(result.length!==blockEvents.length) throw new Error("Invalid event detected, please check provided ABI");
|
|
35
|
+
return result.map((value, index) => {
|
|
36
|
+
const starknetEvent = blockEvents[index];
|
|
37
|
+
const name = Object.keys(value)[0];
|
|
38
|
+
return {
|
|
39
|
+
name: name as T,
|
|
40
|
+
txHash: starknetEvent.transaction_hash,
|
|
41
|
+
params: value[name] as any,
|
|
42
|
+
blockNumber: starknetEvent.block_number,
|
|
43
|
+
blockHash: starknetEvent.block_hash,
|
|
44
|
+
data: starknetEvent.data,
|
|
45
|
+
keys: starknetEvent.keys
|
|
46
|
+
}
|
|
47
|
+
});
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
private toFilter<T extends ExtractAbiEventNames<TAbi>>(
|
|
51
|
+
events: T[],
|
|
52
|
+
keys: string[],
|
|
53
|
+
): string[][] {
|
|
54
|
+
const filterArray: string[][] = [];
|
|
55
|
+
filterArray.push(events.map(name => {
|
|
56
|
+
const arr = name.split(":");
|
|
57
|
+
const eventName = arr[arr.length-1];
|
|
58
|
+
return toHex(hash.starknetKeccak(eventName))
|
|
59
|
+
}));
|
|
60
|
+
if(keys!=null) keys.forEach(key => filterArray.push(key==null ? [] : [key]));
|
|
61
|
+
return filterArray;
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
/**
|
|
65
|
+
* Returns the events occuring in a range of starknet block as identified by the contract and keys,
|
|
66
|
+
* returns pending events if no startHeight & endHeight is passed
|
|
67
|
+
*
|
|
68
|
+
* @param events
|
|
69
|
+
* @param keys
|
|
70
|
+
* @param startBlockHeight
|
|
71
|
+
* @param endBlockHeight
|
|
72
|
+
*/
|
|
73
|
+
public async getContractBlockEvents<T extends ExtractAbiEventNames<TAbi>>(
|
|
74
|
+
events: T[],
|
|
75
|
+
keys: string[],
|
|
76
|
+
startBlockHeight?: number,
|
|
77
|
+
endBlockHeight: number = startBlockHeight
|
|
78
|
+
): Promise<StarknetAbiEvent<TAbi, T>[]> {
|
|
79
|
+
const blockEvents = await super.getBlockEvents(this.root.contract.address, this.toFilter(events, keys), startBlockHeight, endBlockHeight);
|
|
80
|
+
return this.toStarknetAbiEvents(blockEvents);
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
/**
|
|
84
|
+
* Runs a search forawrds in time, processing the events for a specific topic public key
|
|
85
|
+
*
|
|
86
|
+
* @param events
|
|
87
|
+
* @param keys
|
|
88
|
+
* @param processor called for every event, should return a value if the correct event was found, or null
|
|
89
|
+
* if the search should continue
|
|
90
|
+
* @param abortSignal
|
|
91
|
+
*/
|
|
92
|
+
public async findInContractEvents<T, TEvent extends ExtractAbiEventNames<TAbi>>(
|
|
93
|
+
events: TEvent[],
|
|
94
|
+
keys: string[],
|
|
95
|
+
processor: (event: StarknetAbiEvent<TAbi, TEvent>) => Promise<T>,
|
|
96
|
+
abortSignal?: AbortSignal
|
|
97
|
+
) {
|
|
98
|
+
return this.findInEvents<T>(this.root.contract.address, this.toFilter(events, keys), async (events: StarknetEvent[]) => {
|
|
99
|
+
const parsedEvents = this.toStarknetAbiEvents<TEvent>(events);
|
|
100
|
+
for(let event of parsedEvents) {
|
|
101
|
+
const result: T = await processor(event);
|
|
102
|
+
if(result!=null) return result;
|
|
103
|
+
}
|
|
104
|
+
}, abortSignal);
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
}
|
|
108
|
+
|
|
@@ -0,0 +1,63 @@
|
|
|
1
|
+
import {StarknetChainEventsBrowser} from "./StarknetChainEventsBrowser";
|
|
2
|
+
import * as fs from "fs/promises";
|
|
3
|
+
import {StarknetSwapContract} from "../swaps/StarknetSwapContract";
|
|
4
|
+
|
|
5
|
+
const BLOCKHEIGHT_FILENAME = "/strk-blockheight.txt";
|
|
6
|
+
|
|
7
|
+
export class StarknetChainEvents extends StarknetChainEventsBrowser {
|
|
8
|
+
|
|
9
|
+
private readonly directory: string;
|
|
10
|
+
|
|
11
|
+
constructor(
|
|
12
|
+
directory: string,
|
|
13
|
+
starknetSwapContract: StarknetSwapContract,
|
|
14
|
+
pollIntervalSeconds?: number
|
|
15
|
+
) {
|
|
16
|
+
super(starknetSwapContract, pollIntervalSeconds);
|
|
17
|
+
this.directory = directory;
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
/**
|
|
21
|
+
* Retrieves last signature & slot from filesystem
|
|
22
|
+
*
|
|
23
|
+
* @private
|
|
24
|
+
*/
|
|
25
|
+
private async getLastEventData(): Promise<{blockNumber: number, txHash: string}> {
|
|
26
|
+
try {
|
|
27
|
+
const txt = (await fs.readFile(this.directory+BLOCKHEIGHT_FILENAME)).toString();
|
|
28
|
+
const arr = txt.split(";");
|
|
29
|
+
if(arr.length<2) return {
|
|
30
|
+
blockNumber: parseInt(arr[0]),
|
|
31
|
+
txHash: null
|
|
32
|
+
};
|
|
33
|
+
return {
|
|
34
|
+
blockNumber: parseInt(arr[0]),
|
|
35
|
+
txHash: arr[1]
|
|
36
|
+
};
|
|
37
|
+
} catch (e) {
|
|
38
|
+
return {
|
|
39
|
+
blockNumber: null,
|
|
40
|
+
txHash: null
|
|
41
|
+
};
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
/**
|
|
46
|
+
* Saves last signature & slot to the filesystem
|
|
47
|
+
*
|
|
48
|
+
* @private
|
|
49
|
+
*/
|
|
50
|
+
private saveLastEventData(blockNumber: number, txHash: string): Promise<void> {
|
|
51
|
+
return fs.writeFile(this.directory+BLOCKHEIGHT_FILENAME, blockNumber.toString()+";"+txHash);
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
async init(): Promise<void> {
|
|
55
|
+
const {blockNumber, txHash} = await this.getLastEventData();
|
|
56
|
+
await this.setupPoll(
|
|
57
|
+
blockNumber,
|
|
58
|
+
txHash,
|
|
59
|
+
(blockNumber: number, txHash: string) => this.saveLastEventData(blockNumber, txHash)
|
|
60
|
+
);
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
}
|
|
@@ -0,0 +1,289 @@
|
|
|
1
|
+
import {
|
|
2
|
+
ChainEvents,
|
|
3
|
+
ChainSwapType,
|
|
4
|
+
ClaimEvent,
|
|
5
|
+
EventListener,
|
|
6
|
+
InitializeEvent,
|
|
7
|
+
RefundEvent,
|
|
8
|
+
SwapEvent
|
|
9
|
+
} from "@atomiqlabs/base";
|
|
10
|
+
import {StarknetSwapData} from "../swaps/StarknetSwapData";
|
|
11
|
+
import {
|
|
12
|
+
bigNumberishToBuffer,
|
|
13
|
+
bytes31SpanToBuffer, findLastIndex,
|
|
14
|
+
getLogger,
|
|
15
|
+
onceAsync,
|
|
16
|
+
parseInitFunctionCalldata,
|
|
17
|
+
timeoutPromise,
|
|
18
|
+
toHex
|
|
19
|
+
} from "../../utils/Utils";
|
|
20
|
+
import {StarknetSwapContract} from "../swaps/StarknetSwapContract";
|
|
21
|
+
import {BigNumberish, hash, Provider} from "starknet";
|
|
22
|
+
import {StarknetAbiEvent} from "../contract/modules/StarknetContractEvents";
|
|
23
|
+
import {EscrowManagerAbiType} from "../swaps/EscrowManagerAbi";
|
|
24
|
+
import {ExtractAbiFunctionNames} from "abi-wan-kanabi/dist/kanabi";
|
|
25
|
+
import {IClaimHandler} from "../swaps/handlers/claim/ClaimHandlers";
|
|
26
|
+
|
|
27
|
+
export type StarknetTraceCall = {
|
|
28
|
+
calldata: string[],
|
|
29
|
+
contract_address: string,
|
|
30
|
+
entry_point_selector: string,
|
|
31
|
+
calls: StarknetTraceCall[]
|
|
32
|
+
};
|
|
33
|
+
|
|
34
|
+
/**
|
|
35
|
+
* Solana on-chain event handler for front-end systems without access to fs, uses pure WS to subscribe, might lose
|
|
36
|
+
* out on some events if the network is unreliable, front-end systems should take this into consideration and not
|
|
37
|
+
* rely purely on events
|
|
38
|
+
*/
|
|
39
|
+
export class StarknetChainEventsBrowser implements ChainEvents<StarknetSwapData> {
|
|
40
|
+
|
|
41
|
+
protected readonly listeners: EventListener<StarknetSwapData>[] = [];
|
|
42
|
+
protected readonly provider: Provider;
|
|
43
|
+
protected readonly starknetSwapContract: StarknetSwapContract;
|
|
44
|
+
protected eventListeners: number[] = [];
|
|
45
|
+
protected readonly logger = getLogger("StarknetChainEventsBrowser: ");
|
|
46
|
+
|
|
47
|
+
protected initFunctionName: ExtractAbiFunctionNames<EscrowManagerAbiType> = "initialize";
|
|
48
|
+
protected initEntryPointSelector = BigInt(hash.starknetKeccak(this.initFunctionName));
|
|
49
|
+
|
|
50
|
+
protected stopped: boolean;
|
|
51
|
+
protected pollIntervalSeconds: number;
|
|
52
|
+
|
|
53
|
+
private timeout: NodeJS.Timeout;
|
|
54
|
+
|
|
55
|
+
constructor(starknetSwapContract: StarknetSwapContract, pollIntervalSeconds: number = 5) {
|
|
56
|
+
this.provider = starknetSwapContract.provider;
|
|
57
|
+
this.starknetSwapContract = starknetSwapContract;
|
|
58
|
+
this.pollIntervalSeconds = pollIntervalSeconds;
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
findInitSwapData(call: StarknetTraceCall, escrowHash: BigNumberish, claimHandler: IClaimHandler<any, any>): StarknetSwapData {
|
|
62
|
+
if(
|
|
63
|
+
BigInt(call.contract_address)===BigInt(this.starknetSwapContract.contract.address) &&
|
|
64
|
+
BigInt(call.entry_point_selector)===this.initEntryPointSelector
|
|
65
|
+
) {
|
|
66
|
+
//Found, check correct escrow hash
|
|
67
|
+
const {escrow, extraData} = parseInitFunctionCalldata(call.calldata, claimHandler);
|
|
68
|
+
if("0x"+escrow.getEscrowHash()===toHex(escrowHash)) {
|
|
69
|
+
if(extraData.length!==0) {
|
|
70
|
+
escrow.setExtraData(bytes31SpanToBuffer(extraData, 42).toString("hex"));
|
|
71
|
+
}
|
|
72
|
+
return escrow;
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
for(let _call of call.calls) {
|
|
76
|
+
const found = this.findInitSwapData(_call, escrowHash, claimHandler);
|
|
77
|
+
if(found!=null) return found;
|
|
78
|
+
}
|
|
79
|
+
return null;
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
/**
|
|
83
|
+
* Returns async getter for fetching on-demand initialize event swap data
|
|
84
|
+
*
|
|
85
|
+
* @param event
|
|
86
|
+
* @param claimHandler
|
|
87
|
+
* @private
|
|
88
|
+
* @returns {() => Promise<StarknetSwapData>} getter to be passed to InitializeEvent constructor
|
|
89
|
+
*/
|
|
90
|
+
private getSwapDataGetter(
|
|
91
|
+
event: StarknetAbiEvent<EscrowManagerAbiType, "escrow_manager::events::Initialize">,
|
|
92
|
+
claimHandler: IClaimHandler<any, any>
|
|
93
|
+
): () => Promise<StarknetSwapData> {
|
|
94
|
+
return async () => {
|
|
95
|
+
const trace: any = await this.provider.getTransactionTrace(event.txHash);
|
|
96
|
+
if(trace==null) return null;
|
|
97
|
+
if(trace.execute_invocation.revert_reason!=null) return null;
|
|
98
|
+
return this.findInitSwapData(trace.execute_invocation as any, event.params.escrow_hash, claimHandler);
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
protected parseInitializeEvent(
|
|
103
|
+
event: StarknetAbiEvent<EscrowManagerAbiType, "escrow_manager::events::Initialize">
|
|
104
|
+
): InitializeEvent<StarknetSwapData> {
|
|
105
|
+
const escrowHashBuffer = bigNumberishToBuffer(event.params.escrow_hash, 32);
|
|
106
|
+
const escrowHash = escrowHashBuffer.toString("hex");
|
|
107
|
+
const claimHandlerHex = toHex(event.params.claim_handler);
|
|
108
|
+
const claimHandler = this.starknetSwapContract.claimHandlersByAddress[claimHandlerHex];
|
|
109
|
+
if(claimHandler==null) {
|
|
110
|
+
this.logger.warn("parseInitializeEvent("+escrowHash+"): Unknown claim handler with claim: "+claimHandlerHex);
|
|
111
|
+
return null;
|
|
112
|
+
}
|
|
113
|
+
const swapType: ChainSwapType = claimHandler.getType();
|
|
114
|
+
|
|
115
|
+
this.logger.debug("InitializeEvent claimHash: "+toHex(event.params.claim_data)+" escrowHash: "+escrowHash);
|
|
116
|
+
return new InitializeEvent<StarknetSwapData>(
|
|
117
|
+
escrowHash,
|
|
118
|
+
swapType,
|
|
119
|
+
onceAsync<StarknetSwapData>(this.getSwapDataGetter(event, claimHandler))
|
|
120
|
+
);
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
protected parseRefundEvent(
|
|
124
|
+
event: StarknetAbiEvent<EscrowManagerAbiType, "escrow_manager::events::Refund">
|
|
125
|
+
): RefundEvent<StarknetSwapData> {
|
|
126
|
+
const escrowHashBuffer = bigNumberishToBuffer(event.params.escrow_hash, 32);
|
|
127
|
+
const escrowHash = escrowHashBuffer.toString("hex");
|
|
128
|
+
this.logger.debug("RefundEvent claimHash: "+toHex(event.params.claim_data)+" escrowHash: "+escrowHash);
|
|
129
|
+
return new RefundEvent<StarknetSwapData>(escrowHash);
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
protected parseClaimEvent(
|
|
133
|
+
event: StarknetAbiEvent<EscrowManagerAbiType, "escrow_manager::events::Claim">
|
|
134
|
+
): ClaimEvent<StarknetSwapData> {
|
|
135
|
+
const escrowHashBuffer = bigNumberishToBuffer(event.params.escrow_hash, 32);
|
|
136
|
+
const escrowHash = escrowHashBuffer.toString("hex");
|
|
137
|
+
const claimHandlerHex = toHex(event.params.claim_handler);
|
|
138
|
+
const claimHandler = this.starknetSwapContract.claimHandlersByAddress[claimHandlerHex];
|
|
139
|
+
if(claimHandler==null) {
|
|
140
|
+
this.logger.warn("parseClaimEvent("+escrowHash+"): Unknown claim handler with claim: "+claimHandlerHex);
|
|
141
|
+
return null;
|
|
142
|
+
}
|
|
143
|
+
const witnessResult = claimHandler.parseWitnessResult(event.params.witness_result);
|
|
144
|
+
this.logger.debug("ClaimEvent claimHash: "+toHex(event.params.claim_data)+
|
|
145
|
+
" witnessResult: "+witnessResult+" escrowHash: "+escrowHash);
|
|
146
|
+
return new ClaimEvent<StarknetSwapData>(escrowHash, witnessResult);
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
/**
|
|
150
|
+
* Processes event as received from the chain, parses it & calls event listeners
|
|
151
|
+
*
|
|
152
|
+
* @param events
|
|
153
|
+
* @param currentBlockNumber
|
|
154
|
+
* @param currentBlockTimestamp
|
|
155
|
+
* @protected
|
|
156
|
+
*/
|
|
157
|
+
protected async processEvents(
|
|
158
|
+
events : StarknetAbiEvent<
|
|
159
|
+
EscrowManagerAbiType,
|
|
160
|
+
"escrow_manager::events::Initialize" | "escrow_manager::events::Refund" | "escrow_manager::events::Claim"
|
|
161
|
+
>[],
|
|
162
|
+
currentBlockNumber: number,
|
|
163
|
+
currentBlockTimestamp: number
|
|
164
|
+
) {
|
|
165
|
+
const blockTimestampsCache: {[blockNumber: string]: number} = {};
|
|
166
|
+
const getBlockTimestamp: (blockNumber: number) => Promise<number> = async (blockNumber: number)=> {
|
|
167
|
+
const blockNumberString = blockNumber.toString();
|
|
168
|
+
blockTimestampsCache[blockNumberString] ??= (await this.provider.getBlockWithTxHashes(blockNumber)).timestamp;
|
|
169
|
+
return blockTimestampsCache[blockNumberString];
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
const parsedEvents: SwapEvent<StarknetSwapData>[] = [];
|
|
173
|
+
|
|
174
|
+
for(let event of events) {
|
|
175
|
+
let parsedEvent: SwapEvent<StarknetSwapData>;
|
|
176
|
+
switch(event.name) {
|
|
177
|
+
case "escrow_manager::events::Claim":
|
|
178
|
+
parsedEvent = this.parseClaimEvent(event as any);
|
|
179
|
+
break;
|
|
180
|
+
case "escrow_manager::events::Refund":
|
|
181
|
+
parsedEvent = this.parseRefundEvent(event as any);
|
|
182
|
+
break;
|
|
183
|
+
case "escrow_manager::events::Initialize":
|
|
184
|
+
parsedEvent = this.parseInitializeEvent(event as any);
|
|
185
|
+
break;
|
|
186
|
+
}
|
|
187
|
+
const timestamp = (event.blockNumber==null || event.blockNumber===currentBlockNumber) ? currentBlockTimestamp : await getBlockTimestamp(event.blockNumber);
|
|
188
|
+
parsedEvent.meta = {
|
|
189
|
+
blockTime: timestamp,
|
|
190
|
+
txId: event.txHash,
|
|
191
|
+
timestamp //Maybe deprecated
|
|
192
|
+
} as any;
|
|
193
|
+
parsedEvents.push(parsedEvent);
|
|
194
|
+
}
|
|
195
|
+
|
|
196
|
+
for(let listener of this.listeners) {
|
|
197
|
+
await listener(parsedEvents);
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
protected async checkEvents(lastBlockNumber: number, lastTxHash: string): Promise<{txHash: string, blockNumber: number}> {
|
|
202
|
+
//Get pending events
|
|
203
|
+
let pendingEvents = await this.starknetSwapContract.Events.getContractBlockEvents(
|
|
204
|
+
["escrow_manager::events::Initialize", "escrow_manager::events::Claim", "escrow_manager::events::Refund"],
|
|
205
|
+
[]
|
|
206
|
+
);
|
|
207
|
+
if(lastTxHash!=null) {
|
|
208
|
+
const latestProcessedEventIndex = findLastIndex(pendingEvents, val => val.txHash===lastTxHash);
|
|
209
|
+
if(latestProcessedEventIndex!==-1) pendingEvents.splice(0, latestProcessedEventIndex+1);
|
|
210
|
+
}
|
|
211
|
+
if(pendingEvents.length>0) {
|
|
212
|
+
await this.processEvents(pendingEvents, null, Math.floor(Date.now()/1000));
|
|
213
|
+
lastTxHash = pendingEvents[pendingEvents.length-1].txHash;
|
|
214
|
+
}
|
|
215
|
+
|
|
216
|
+
const currentBlock = await this.provider.getBlockWithTxHashes("latest");
|
|
217
|
+
const currentBlockNumber: number = (currentBlock as any).block_number;
|
|
218
|
+
if(lastBlockNumber!=null && currentBlockNumber>lastBlockNumber) {
|
|
219
|
+
const events = await this.starknetSwapContract.Events.getContractBlockEvents(
|
|
220
|
+
["escrow_manager::events::Initialize", "escrow_manager::events::Claim", "escrow_manager::events::Refund"],
|
|
221
|
+
[],
|
|
222
|
+
lastBlockNumber+1,
|
|
223
|
+
currentBlockNumber
|
|
224
|
+
);
|
|
225
|
+
if(lastTxHash!=null) {
|
|
226
|
+
const latestProcessedEventIndex = findLastIndex(events, val => val.txHash === lastTxHash);
|
|
227
|
+
if (latestProcessedEventIndex !== -1) events.splice(0, latestProcessedEventIndex + 1);
|
|
228
|
+
}
|
|
229
|
+
if(events.length>0) {
|
|
230
|
+
await this.processEvents(events, currentBlockNumber, currentBlock.timestamp);
|
|
231
|
+
lastTxHash = events[events.length - 1].txHash;
|
|
232
|
+
}
|
|
233
|
+
}
|
|
234
|
+
return {
|
|
235
|
+
txHash: lastTxHash,
|
|
236
|
+
blockNumber: currentBlockNumber
|
|
237
|
+
};
|
|
238
|
+
}
|
|
239
|
+
|
|
240
|
+
/**
|
|
241
|
+
* Sets up event handlers listening for swap events over websocket
|
|
242
|
+
*
|
|
243
|
+
* @protected
|
|
244
|
+
*/
|
|
245
|
+
protected async setupPoll(
|
|
246
|
+
lastBlockNumber?: number,
|
|
247
|
+
lastTxHash?: string,
|
|
248
|
+
saveLatestProcessedBlockNumber?: (blockNumber: number, lastTxHash: string) => Promise<void>
|
|
249
|
+
) {
|
|
250
|
+
this.stopped = false;
|
|
251
|
+
let func;
|
|
252
|
+
func = async () => {
|
|
253
|
+
await this.checkEvents(lastBlockNumber, lastTxHash).then(({blockNumber, txHash}) => {
|
|
254
|
+
lastBlockNumber = blockNumber;
|
|
255
|
+
lastTxHash = txHash;
|
|
256
|
+
if(saveLatestProcessedBlockNumber!=null) return saveLatestProcessedBlockNumber(blockNumber, lastTxHash);
|
|
257
|
+
}).catch(e => {
|
|
258
|
+
this.logger.error("setupPoll(): Failed to fetch starknet log: ", e);
|
|
259
|
+
});
|
|
260
|
+
if(this.stopped) return;
|
|
261
|
+
this.timeout = setTimeout(func, this.pollIntervalSeconds*1000);
|
|
262
|
+
};
|
|
263
|
+
await func();
|
|
264
|
+
}
|
|
265
|
+
|
|
266
|
+
init(): Promise<void> {
|
|
267
|
+
this.setupPoll();
|
|
268
|
+
return Promise.resolve();
|
|
269
|
+
}
|
|
270
|
+
|
|
271
|
+
async stop(): Promise<void> {
|
|
272
|
+
this.stopped = true;
|
|
273
|
+
if(this.timeout!=null) clearTimeout(this.timeout);
|
|
274
|
+
this.eventListeners = [];
|
|
275
|
+
}
|
|
276
|
+
|
|
277
|
+
registerListener(cbk: EventListener<StarknetSwapData>): void {
|
|
278
|
+
this.listeners.push(cbk);
|
|
279
|
+
}
|
|
280
|
+
|
|
281
|
+
unregisterListener(cbk: EventListener<StarknetSwapData>): boolean {
|
|
282
|
+
const index = this.listeners.indexOf(cbk);
|
|
283
|
+
if(index>=0) {
|
|
284
|
+
this.listeners.splice(index, 1);
|
|
285
|
+
return true;
|
|
286
|
+
}
|
|
287
|
+
return false;
|
|
288
|
+
}
|
|
289
|
+
}
|