@hashgraphonline/standards-sdk 0.1.136-fix-hol-env.canary.a285980.31 → 0.1.137-feat-hcs-21.canary.5802d65.32
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/dist/cjs/hcs-21/base-client.d.ts +31 -0
- package/dist/cjs/hcs-21/base-client.d.ts.map +1 -0
- package/dist/cjs/hcs-21/browser.d.ts +20 -0
- package/dist/cjs/hcs-21/browser.d.ts.map +1 -0
- package/dist/cjs/hcs-21/errors.d.ts +6 -0
- package/dist/cjs/hcs-21/errors.d.ts.map +1 -0
- package/dist/cjs/hcs-21/index.d.ts +7 -0
- package/dist/cjs/hcs-21/index.d.ts.map +1 -0
- package/dist/cjs/hcs-21/sdk.d.ts +48 -0
- package/dist/cjs/hcs-21/sdk.d.ts.map +1 -0
- package/dist/cjs/hcs-21/tx.d.ts +20 -0
- package/dist/cjs/hcs-21/tx.d.ts.map +1 -0
- package/dist/cjs/hcs-21/types.d.ts +72 -0
- package/dist/cjs/hcs-21/types.d.ts.map +1 -0
- package/dist/cjs/index.d.ts +1 -0
- package/dist/cjs/index.d.ts.map +1 -1
- package/dist/cjs/services/types.d.ts +1 -1
- package/dist/cjs/services/types.d.ts.map +1 -1
- package/dist/cjs/standards-sdk.cjs +2 -2
- package/dist/cjs/standards-sdk.cjs.map +1 -1
- package/dist/es/hcs-21/base-client.d.ts +31 -0
- package/dist/es/hcs-21/base-client.d.ts.map +1 -0
- package/dist/es/hcs-21/browser.d.ts +20 -0
- package/dist/es/hcs-21/browser.d.ts.map +1 -0
- package/dist/es/hcs-21/errors.d.ts +6 -0
- package/dist/es/hcs-21/errors.d.ts.map +1 -0
- package/dist/es/hcs-21/index.d.ts +7 -0
- package/dist/es/hcs-21/index.d.ts.map +1 -0
- package/dist/es/hcs-21/sdk.d.ts +48 -0
- package/dist/es/hcs-21/sdk.d.ts.map +1 -0
- package/dist/es/hcs-21/tx.d.ts +20 -0
- package/dist/es/hcs-21/tx.d.ts.map +1 -0
- package/dist/es/hcs-21/types.d.ts +72 -0
- package/dist/es/hcs-21/types.d.ts.map +1 -0
- package/dist/es/index.d.ts +1 -0
- package/dist/es/index.d.ts.map +1 -1
- package/dist/es/services/types.d.ts +1 -1
- package/dist/es/services/types.d.ts.map +1 -1
- package/dist/es/standards-sdk.es.js +66 -48
- package/dist/es/standards-sdk.es.js.map +1 -1
- package/dist/es/standards-sdk.es10.js +2 -2
- package/dist/es/standards-sdk.es100.js +231 -28
- package/dist/es/standards-sdk.es100.js.map +1 -1
- package/dist/es/standards-sdk.es101.js +109 -18
- package/dist/es/standards-sdk.es101.js.map +1 -1
- package/dist/es/standards-sdk.es102.js +32 -158
- package/dist/es/standards-sdk.es102.js.map +1 -1
- package/dist/es/standards-sdk.es103.js +80 -204
- package/dist/es/standards-sdk.es103.js.map +1 -1
- package/dist/es/standards-sdk.es104.js +21 -780
- package/dist/es/standards-sdk.es104.js.map +1 -1
- package/dist/es/standards-sdk.es105.js +140 -11
- package/dist/es/standards-sdk.es105.js.map +1 -1
- package/dist/es/standards-sdk.es106.js +29 -567
- package/dist/es/standards-sdk.es106.js.map +1 -1
- package/dist/es/standards-sdk.es107.js +17 -597
- package/dist/es/standards-sdk.es107.js.map +1 -1
- package/dist/es/standards-sdk.es108.js +159 -12
- package/dist/es/standards-sdk.es108.js.map +1 -1
- package/dist/es/standards-sdk.es109.js +208 -2
- package/dist/es/standards-sdk.es109.js.map +1 -1
- package/dist/es/standards-sdk.es110.js +774 -67
- package/dist/es/standards-sdk.es110.js.map +1 -1
- package/dist/es/standards-sdk.es111.js +11 -37
- package/dist/es/standards-sdk.es111.js.map +1 -1
- package/dist/es/standards-sdk.es112.js +567 -2
- package/dist/es/standards-sdk.es112.js.map +1 -1
- package/dist/es/standards-sdk.es113.js +576 -197
- package/dist/es/standards-sdk.es113.js.map +1 -1
- package/dist/es/standards-sdk.es114.js +12 -1139
- package/dist/es/standards-sdk.es114.js.map +1 -1
- package/dist/es/standards-sdk.es115.js +2 -306
- package/dist/es/standards-sdk.es115.js.map +1 -1
- package/dist/es/standards-sdk.es116.js +79 -418
- package/dist/es/standards-sdk.es116.js.map +1 -1
- package/dist/es/standards-sdk.es117.js +35 -351
- package/dist/es/standards-sdk.es117.js.map +1 -1
- package/dist/es/standards-sdk.es118.js +2 -761
- package/dist/es/standards-sdk.es118.js.map +1 -1
- package/dist/es/standards-sdk.es119.js +204 -185
- package/dist/es/standards-sdk.es119.js.map +1 -1
- package/dist/es/standards-sdk.es12.js +1 -1
- package/dist/es/standards-sdk.es120.js +1055 -1482
- package/dist/es/standards-sdk.es120.js.map +1 -1
- package/dist/es/standards-sdk.es121.js +265 -2042
- package/dist/es/standards-sdk.es121.js.map +1 -1
- package/dist/es/standards-sdk.es122.js +420 -50
- package/dist/es/standards-sdk.es122.js.map +1 -1
- package/dist/es/standards-sdk.es123.js +350 -82
- package/dist/es/standards-sdk.es123.js.map +1 -1
- package/dist/es/standards-sdk.es124.js +741 -139
- package/dist/es/standards-sdk.es124.js.map +1 -1
- package/dist/es/standards-sdk.es125.js +204 -7
- package/dist/es/standards-sdk.es125.js.map +1 -1
- package/dist/es/standards-sdk.es126.js +1533 -52
- package/dist/es/standards-sdk.es126.js.map +1 -1
- package/dist/es/standards-sdk.es127.js +2077 -59
- package/dist/es/standards-sdk.es127.js.map +1 -1
- package/dist/es/standards-sdk.es128.js +52 -28
- package/dist/es/standards-sdk.es128.js.map +1 -1
- package/dist/es/standards-sdk.es129.js +85 -692
- package/dist/es/standards-sdk.es129.js.map +1 -1
- package/dist/es/standards-sdk.es13.js +1 -1
- package/dist/es/standards-sdk.es130.js +135 -12250
- package/dist/es/standards-sdk.es130.js.map +1 -1
- package/dist/es/standards-sdk.es131.js +7 -138
- package/dist/es/standards-sdk.es131.js.map +1 -1
- package/dist/es/standards-sdk.es132.js +80 -36
- package/dist/es/standards-sdk.es132.js.map +1 -1
- package/dist/es/standards-sdk.es133.js +58 -49
- package/dist/es/standards-sdk.es133.js.map +1 -1
- package/dist/es/standards-sdk.es134.js +28 -64
- package/dist/es/standards-sdk.es134.js.map +1 -1
- package/dist/es/standards-sdk.es135.js +118 -152
- package/dist/es/standards-sdk.es135.js.map +1 -1
- package/dist/es/standards-sdk.es136.js +34 -314
- package/dist/es/standards-sdk.es136.js.map +1 -1
- package/dist/es/standards-sdk.es137.js +694 -346
- package/dist/es/standards-sdk.es137.js.map +1 -1
- package/dist/es/standards-sdk.es138.js +12216 -395
- package/dist/es/standards-sdk.es138.js.map +1 -1
- package/dist/es/standards-sdk.es139.js +55 -330
- package/dist/es/standards-sdk.es139.js.map +1 -1
- package/dist/es/standards-sdk.es14.js +1 -1
- package/dist/es/standards-sdk.es140.js +50 -62
- package/dist/es/standards-sdk.es140.js.map +1 -1
- package/dist/es/standards-sdk.es141.js +168 -13
- package/dist/es/standards-sdk.es141.js.map +1 -1
- package/dist/es/standards-sdk.es142.js +325 -0
- package/dist/es/standards-sdk.es142.js.map +1 -0
- package/dist/es/standards-sdk.es143.js +349 -0
- package/dist/es/standards-sdk.es143.js.map +1 -0
- package/dist/es/standards-sdk.es144.js +456 -0
- package/dist/es/standards-sdk.es144.js.map +1 -0
- package/dist/es/standards-sdk.es145.js +334 -0
- package/dist/es/standards-sdk.es145.js.map +1 -0
- package/dist/es/standards-sdk.es146.js +79 -0
- package/dist/es/standards-sdk.es146.js.map +1 -0
- package/dist/es/standards-sdk.es147.js +20 -0
- package/dist/es/standards-sdk.es147.js.map +1 -0
- package/dist/es/standards-sdk.es15.js +1 -1
- package/dist/es/standards-sdk.es16.js +1 -1
- package/dist/es/standards-sdk.es17.js +5 -5
- package/dist/es/standards-sdk.es19.js +12 -12
- package/dist/es/standards-sdk.es20.js +9 -9
- package/dist/es/standards-sdk.es21.js +1 -1
- package/dist/es/standards-sdk.es22.js +1 -1
- package/dist/es/standards-sdk.es23.js +1 -1
- package/dist/es/standards-sdk.es24.js +1 -1
- package/dist/es/standards-sdk.es25.js +1 -1
- package/dist/es/standards-sdk.es26.js +1 -1
- package/dist/es/standards-sdk.es27.js +1 -1
- package/dist/es/standards-sdk.es28.js +12 -12
- package/dist/es/standards-sdk.es3.js +2 -2
- package/dist/es/standards-sdk.es31.js +2 -2
- package/dist/es/standards-sdk.es32.js +4 -4
- package/dist/es/standards-sdk.es33.js +1 -1
- package/dist/es/standards-sdk.es36.js +6 -6
- package/dist/es/standards-sdk.es37.js +4 -4
- package/dist/es/standards-sdk.es38.js +2 -2
- package/dist/es/standards-sdk.es39.js +2 -2
- package/dist/es/standards-sdk.es40.js +1 -1
- package/dist/es/standards-sdk.es41.js +1 -1
- package/dist/es/standards-sdk.es42.js +2 -2
- package/dist/es/standards-sdk.es47.js +1 -1
- package/dist/es/standards-sdk.es5.js +2 -2
- package/dist/es/standards-sdk.es52.js +1 -1
- package/dist/es/standards-sdk.es54.js +1 -1
- package/dist/es/standards-sdk.es57.js +2 -2
- package/dist/es/standards-sdk.es59.js +1 -1
- package/dist/es/standards-sdk.es6.js +2 -2
- package/dist/es/standards-sdk.es60.js +1 -1
- package/dist/es/standards-sdk.es61.js +7 -7
- package/dist/es/standards-sdk.es63.js +1 -1
- package/dist/es/standards-sdk.es65.js +2 -2
- package/dist/es/standards-sdk.es66.js +3 -3
- package/dist/es/standards-sdk.es69.js +2 -2
- package/dist/es/standards-sdk.es7.js +2 -2
- package/dist/es/standards-sdk.es70.js +3 -3
- package/dist/es/standards-sdk.es71.js +2 -2
- package/dist/es/standards-sdk.es72.js +1 -1
- package/dist/es/standards-sdk.es75.js +2 -2
- package/dist/es/standards-sdk.es77.js +2 -2
- package/dist/es/standards-sdk.es78.js +4 -4
- package/dist/es/standards-sdk.es79.js +1 -1
- package/dist/es/standards-sdk.es8.js +1 -1
- package/dist/es/standards-sdk.es80.js +25 -27
- package/dist/es/standards-sdk.es80.js.map +1 -1
- package/dist/es/standards-sdk.es81.js +6 -78
- package/dist/es/standards-sdk.es81.js.map +1 -1
- package/dist/es/standards-sdk.es82.js +27 -64
- package/dist/es/standards-sdk.es82.js.map +1 -1
- package/dist/es/standards-sdk.es83.js +67 -23
- package/dist/es/standards-sdk.es83.js.map +1 -1
- package/dist/es/standards-sdk.es84.js +165 -23
- package/dist/es/standards-sdk.es84.js.map +1 -1
- package/dist/es/standards-sdk.es85.js +31 -166
- package/dist/es/standards-sdk.es85.js.map +1 -1
- package/dist/es/standards-sdk.es86.js +24 -127
- package/dist/es/standards-sdk.es86.js.map +1 -1
- package/dist/es/standards-sdk.es87.js +65 -142
- package/dist/es/standards-sdk.es87.js.map +1 -1
- package/dist/es/standards-sdk.es88.js +52 -127
- package/dist/es/standards-sdk.es88.js.map +1 -1
- package/dist/es/standards-sdk.es89.js +26 -41
- package/dist/es/standards-sdk.es89.js.map +1 -1
- package/dist/es/standards-sdk.es90.js +23 -261
- package/dist/es/standards-sdk.es90.js.map +1 -1
- package/dist/es/standards-sdk.es91.js +162 -93
- package/dist/es/standards-sdk.es91.js.map +1 -1
- package/dist/es/standards-sdk.es92.js +112 -83
- package/dist/es/standards-sdk.es92.js.map +1 -1
- package/dist/es/standards-sdk.es93.js +156 -28
- package/dist/es/standards-sdk.es93.js.map +1 -1
- package/dist/es/standards-sdk.es94.js +133 -225
- package/dist/es/standards-sdk.es94.js.map +1 -1
- package/dist/es/standards-sdk.es95.js +40 -108
- package/dist/es/standards-sdk.es95.js.map +1 -1
- package/dist/es/standards-sdk.es96.js +253 -24
- package/dist/es/standards-sdk.es96.js.map +1 -1
- package/dist/es/standards-sdk.es97.js +96 -80
- package/dist/es/standards-sdk.es97.js.map +1 -1
- package/dist/es/standards-sdk.es98.js +91 -21
- package/dist/es/standards-sdk.es98.js.map +1 -1
- package/dist/es/standards-sdk.es99.js +28 -141
- package/dist/es/standards-sdk.es99.js.map +1 -1
- package/package.json +2 -1
|
@@ -1,1570 +1,1143 @@
|
|
|
1
|
-
import { PublicKey, Timestamp, AccountId } from "@hashgraph/sdk";
|
|
2
|
-
import axios from "axios";
|
|
3
|
-
import { Logger } from "./standards-sdk.es99.js";
|
|
4
1
|
import { proto } from "@hashgraph/proto";
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
this.maxDelayMs = 3e4;
|
|
10
|
-
this.backoffFactor = 2;
|
|
11
|
-
this.network = network;
|
|
12
|
-
this.apiKey = config?.apiKey;
|
|
13
|
-
this.customHeaders = config?.headers || {};
|
|
14
|
-
this.baseUrl = config?.customUrl || this.getMirrorNodeUrl();
|
|
15
|
-
this.logger = logger || new Logger({
|
|
16
|
-
level: "debug",
|
|
17
|
-
module: "MirrorNode"
|
|
18
|
-
});
|
|
19
|
-
this.isServerEnvironment = typeof window === "undefined";
|
|
20
|
-
if (config?.customUrl) {
|
|
21
|
-
this.logger.info(`Using custom mirror node URL: ${config.customUrl}`);
|
|
22
|
-
}
|
|
23
|
-
if (config?.apiKey) {
|
|
24
|
-
this.logger.info("Using API key for mirror node requests");
|
|
25
|
-
}
|
|
26
|
-
}
|
|
27
|
-
/**
|
|
28
|
-
* Configures the retry mechanism for API requests.
|
|
29
|
-
* @param config The retry configuration.
|
|
30
|
-
*/
|
|
31
|
-
configureRetry(config) {
|
|
32
|
-
this.maxRetries = config.maxRetries ?? this.maxRetries;
|
|
33
|
-
this.initialDelayMs = config.initialDelayMs ?? this.initialDelayMs;
|
|
34
|
-
this.maxDelayMs = config.maxDelayMs ?? this.maxDelayMs;
|
|
35
|
-
this.backoffFactor = config.backoffFactor ?? this.backoffFactor;
|
|
36
|
-
this.logger.info(
|
|
37
|
-
`Retry configuration updated: maxRetries=${this.maxRetries}, initialDelayMs=${this.initialDelayMs}, maxDelayMs=${this.maxDelayMs}, backoffFactor=${this.backoffFactor}`
|
|
38
|
-
);
|
|
39
|
-
}
|
|
40
|
-
/**
|
|
41
|
-
* Updates the mirror node configuration.
|
|
42
|
-
* @param config The new mirror node configuration.
|
|
43
|
-
*/
|
|
44
|
-
configureMirrorNode(config) {
|
|
45
|
-
if (config.customUrl) {
|
|
46
|
-
this.baseUrl = config.customUrl;
|
|
47
|
-
this.logger.info(`Updated mirror node URL: ${config.customUrl}`);
|
|
48
|
-
}
|
|
49
|
-
if (config.apiKey) {
|
|
50
|
-
this.apiKey = config.apiKey;
|
|
51
|
-
this.logger.info("Updated API key for mirror node requests");
|
|
52
|
-
}
|
|
53
|
-
if (config.headers) {
|
|
54
|
-
this.customHeaders = { ...this.customHeaders, ...config.headers };
|
|
55
|
-
this.logger.info("Updated custom headers for mirror node requests");
|
|
56
|
-
}
|
|
57
|
-
}
|
|
58
|
-
/**
|
|
59
|
-
* Constructs a full URL for API requests, handling custom providers with API keys in the path.
|
|
60
|
-
* @param endpoint The API endpoint (e.g., '/api/v1/accounts/0.0.123')
|
|
61
|
-
* @returns The full URL for the request
|
|
62
|
-
*/
|
|
63
|
-
constructUrl(endpoint) {
|
|
64
|
-
if (this.baseUrl.includes("<API-KEY>") && this.apiKey) {
|
|
65
|
-
const baseUrlWithKey = this.baseUrl.replace("<API-KEY>", this.apiKey);
|
|
66
|
-
return endpoint.startsWith("/") ? `${baseUrlWithKey}${endpoint}` : `${baseUrlWithKey}/${endpoint}`;
|
|
67
|
-
}
|
|
68
|
-
return endpoint.startsWith("/") ? `${this.baseUrl}${endpoint}` : `${this.baseUrl}/${endpoint}`;
|
|
69
|
-
}
|
|
2
|
+
import { AccountId, Long, TokenId } from "@hashgraph/sdk";
|
|
3
|
+
import { parseKey, hasTransactionType, extractTransactionBody } from "./standards-sdk.es146.js";
|
|
4
|
+
import { Buffer } from "buffer";
|
|
5
|
+
class HTSParser {
|
|
70
6
|
/**
|
|
71
|
-
*
|
|
72
|
-
*
|
|
73
|
-
* @private
|
|
7
|
+
* Main entry point for parsing HTS transactions from a Transaction object
|
|
8
|
+
* This method bridges between the Transaction object and the protobuf parsers
|
|
74
9
|
*/
|
|
75
|
-
|
|
76
|
-
return this.network === "mainnet" ? "https://mainnet-public.mirrornode.hedera.com" : "https://testnet.mirrornode.hedera.com";
|
|
77
|
-
}
|
|
78
|
-
getBaseUrl() {
|
|
79
|
-
return this.baseUrl;
|
|
80
|
-
}
|
|
81
|
-
/**
|
|
82
|
-
* Retrieves the public key for a given account ID from the mirror node.
|
|
83
|
-
* @param accountId The ID of the account to retrieve the public key for.
|
|
84
|
-
* @returns A promise that resolves to the public key for the given account.
|
|
85
|
-
* @throws An error if the account ID is invalid or the public key cannot be retrieved.
|
|
86
|
-
*/
|
|
87
|
-
async getPublicKey(accountId) {
|
|
88
|
-
this.logger.debug(`Getting public key for account ${accountId}`);
|
|
89
|
-
const accountInfo = await this.requestAccount(accountId);
|
|
10
|
+
static parseHTSTransaction(transaction) {
|
|
90
11
|
try {
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
12
|
+
const transactionBody = transaction._transactionBody;
|
|
13
|
+
if (!transactionBody) {
|
|
14
|
+
return {};
|
|
15
|
+
}
|
|
16
|
+
if (transactionBody.tokenCreation) {
|
|
17
|
+
const tokenCreation = this.parseTokenCreate(
|
|
18
|
+
transactionBody.tokenCreation
|
|
94
19
|
);
|
|
20
|
+
if (tokenCreation) {
|
|
21
|
+
return {
|
|
22
|
+
type: "TOKENCREATE",
|
|
23
|
+
humanReadableType: "Token Creation",
|
|
24
|
+
tokenCreation
|
|
25
|
+
};
|
|
26
|
+
}
|
|
95
27
|
}
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
* Retrieves the memo for a given account ID from the mirror node.
|
|
106
|
-
* @param accountId The ID of the account to retrieve the memo for.
|
|
107
|
-
* @returns A promise that resolves to the memo for the given account.
|
|
108
|
-
* @throws An error if the account ID is invalid or the memo cannot be retrieved.
|
|
109
|
-
*/
|
|
110
|
-
async getAccountMemo(accountId) {
|
|
111
|
-
this.logger.debug(`Getting account memo for account ID: ${accountId}`);
|
|
112
|
-
try {
|
|
113
|
-
const accountInfo = await this._requestWithRetry(
|
|
114
|
-
`/api/v1/accounts/${accountId}`
|
|
115
|
-
);
|
|
116
|
-
if (accountInfo?.memo) {
|
|
117
|
-
return accountInfo.memo;
|
|
28
|
+
if (transactionBody.tokenMint) {
|
|
29
|
+
const tokenMint = this.parseTokenMint(transactionBody.tokenMint);
|
|
30
|
+
if (tokenMint) {
|
|
31
|
+
return {
|
|
32
|
+
type: "TOKENMINT",
|
|
33
|
+
humanReadableType: "Token Mint",
|
|
34
|
+
tokenMint
|
|
35
|
+
};
|
|
36
|
+
}
|
|
118
37
|
}
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
}
|
|
129
|
-
/**
|
|
130
|
-
* Retrieves topic information for a given topic ID from the mirror node.
|
|
131
|
-
* @param topicId The ID of the topic to retrieve information for.
|
|
132
|
-
* @returns A promise that resolves to the topic information.
|
|
133
|
-
* @throws An error if the topic ID is invalid or the information cannot be retrieved.
|
|
134
|
-
*/
|
|
135
|
-
async getTopicInfo(topicId) {
|
|
136
|
-
try {
|
|
137
|
-
this.logger.debug(`Fetching topic info for ${topicId}`);
|
|
138
|
-
const data = await this._requestWithRetry(
|
|
139
|
-
`/api/v1/topics/${topicId}`
|
|
140
|
-
);
|
|
141
|
-
return data;
|
|
142
|
-
} catch (e) {
|
|
143
|
-
const error = e;
|
|
144
|
-
const logMessage = `Error retrieving topic information for ${topicId} after retries: ${error.message}`;
|
|
145
|
-
this.logger.error(logMessage);
|
|
146
|
-
throw new Error(logMessage);
|
|
147
|
-
}
|
|
148
|
-
}
|
|
149
|
-
/**
|
|
150
|
-
* Retrieves custom fees for a given topic ID from the mirror node.
|
|
151
|
-
* @param topicId The ID of the topic to retrieve custom fees for.
|
|
152
|
-
* @returns A promise that resolves to the custom fees for the given topic.
|
|
153
|
-
* @throws An error if the topic ID is invalid or the custom fees cannot be retrieved.
|
|
154
|
-
*/
|
|
155
|
-
async getTopicFees(topicId) {
|
|
156
|
-
try {
|
|
157
|
-
const topicInfo = await this.getTopicInfo(topicId);
|
|
158
|
-
return topicInfo.custom_fees;
|
|
159
|
-
} catch (e) {
|
|
160
|
-
const error = e;
|
|
161
|
-
const logMessage = `Error retrieving topic fees: ${error.message}`;
|
|
162
|
-
this.logger.error(logMessage);
|
|
163
|
-
return null;
|
|
164
|
-
}
|
|
165
|
-
}
|
|
166
|
-
/**
|
|
167
|
-
* Retrieves the current HBAR price from the mirror node.
|
|
168
|
-
* @param date The date to retrieve the HBAR price for.
|
|
169
|
-
* @returns A promise that resolves to the HBAR price for the given date.
|
|
170
|
-
* @throws An error if the date is invalid or the price cannot be retrieved.
|
|
171
|
-
*/
|
|
172
|
-
async getHBARPrice(date) {
|
|
173
|
-
try {
|
|
174
|
-
const timestamp = Timestamp.fromDate(date).toString();
|
|
175
|
-
this.logger.debug(`Fetching HBAR price for timestamp ${timestamp}`);
|
|
176
|
-
const response = await this._requestWithRetry(
|
|
177
|
-
`/api/v1/network/exchangerate?timestamp=${timestamp}`
|
|
178
|
-
);
|
|
179
|
-
const usdPrice = Number(response?.current_rate?.cent_equivalent) / Number(response?.current_rate?.hbar_equivalent) / 100;
|
|
180
|
-
return usdPrice;
|
|
181
|
-
} catch (e) {
|
|
182
|
-
const error = e;
|
|
183
|
-
const logMessage = `Error retrieving HBAR price: ${error.message}`;
|
|
184
|
-
this.logger.error(logMessage);
|
|
185
|
-
return null;
|
|
186
|
-
}
|
|
187
|
-
}
|
|
188
|
-
/**
|
|
189
|
-
* Retrieves token information for a given token ID from the mirror node.
|
|
190
|
-
* @param tokenId The ID of the token to retrieve information for.
|
|
191
|
-
* @returns A promise that resolves to the token information.
|
|
192
|
-
* @throws An error if the token ID is invalid or the information cannot be retrieved.
|
|
193
|
-
*/
|
|
194
|
-
async getTokenInfo(tokenId) {
|
|
195
|
-
this.logger.debug(`Fetching token info for ${tokenId}`);
|
|
196
|
-
try {
|
|
197
|
-
const data = await this._requestWithRetry(
|
|
198
|
-
`/api/v1/tokens/${tokenId}`
|
|
199
|
-
);
|
|
200
|
-
if (data) {
|
|
201
|
-
this.logger.trace(`Token info found for ${tokenId}:`, data);
|
|
202
|
-
return data;
|
|
38
|
+
if (transactionBody.tokenBurn) {
|
|
39
|
+
const tokenBurn = this.parseTokenBurn(transactionBody.tokenBurn);
|
|
40
|
+
if (tokenBurn) {
|
|
41
|
+
return {
|
|
42
|
+
type: "TOKENBURN",
|
|
43
|
+
humanReadableType: "Token Burn",
|
|
44
|
+
tokenBurn
|
|
45
|
+
};
|
|
46
|
+
}
|
|
203
47
|
}
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
}
|
|
213
|
-
/**
|
|
214
|
-
* Retrieves messages for a given topic ID from the mirror node. Supports filtering by sequence number
|
|
215
|
-
* based on the OpenAPI specification.
|
|
216
|
-
* @param topicId The ID of the topic to retrieve messages for.
|
|
217
|
-
* @param options Optional filtering parameters.
|
|
218
|
-
* @returns A promise that resolves to the messages for the given topic.
|
|
219
|
-
*/
|
|
220
|
-
async getTopicMessages(topicId, options) {
|
|
221
|
-
this.logger.trace(
|
|
222
|
-
`Querying messages for topic ${topicId}${options ? " with filters" : ""}`
|
|
223
|
-
);
|
|
224
|
-
let endpoint = `/api/v1/topics/${topicId}/messages`;
|
|
225
|
-
const params = new URLSearchParams();
|
|
226
|
-
if (options) {
|
|
227
|
-
if (options.sequenceNumber !== void 0) {
|
|
228
|
-
const seqNum = typeof options.sequenceNumber === "number" ? options.sequenceNumber.toString() : options.sequenceNumber;
|
|
229
|
-
if (!seqNum.match(/^(gt|gte|lt|lte|eq|ne):/)) {
|
|
230
|
-
params.append("sequencenumber", `gt:${seqNum}`);
|
|
231
|
-
} else {
|
|
232
|
-
params.append("sequencenumber", seqNum);
|
|
48
|
+
if (transactionBody.tokenUpdate) {
|
|
49
|
+
const tokenUpdate = this.parseTokenUpdate(transactionBody.tokenUpdate);
|
|
50
|
+
if (tokenUpdate) {
|
|
51
|
+
return {
|
|
52
|
+
type: "TOKENUPDATE",
|
|
53
|
+
humanReadableType: "Token Update",
|
|
54
|
+
tokenUpdate
|
|
55
|
+
};
|
|
233
56
|
}
|
|
234
57
|
}
|
|
235
|
-
if (
|
|
236
|
-
|
|
58
|
+
if (transactionBody.tokenFreeze) {
|
|
59
|
+
const tokenFreeze = this.parseTokenFreeze(transactionBody.tokenFreeze);
|
|
60
|
+
if (tokenFreeze) {
|
|
61
|
+
return {
|
|
62
|
+
type: "TOKENFREEZE",
|
|
63
|
+
humanReadableType: "Token Freeze",
|
|
64
|
+
tokenFreeze
|
|
65
|
+
};
|
|
66
|
+
}
|
|
237
67
|
}
|
|
238
|
-
if (
|
|
239
|
-
|
|
68
|
+
if (transactionBody.tokenUnfreeze) {
|
|
69
|
+
const tokenUnfreeze = this.parseTokenUnfreeze(
|
|
70
|
+
transactionBody.tokenUnfreeze
|
|
71
|
+
);
|
|
72
|
+
if (tokenUnfreeze) {
|
|
73
|
+
return {
|
|
74
|
+
type: "TOKENUNFREEZE",
|
|
75
|
+
humanReadableType: "Token Unfreeze",
|
|
76
|
+
tokenUnfreeze
|
|
77
|
+
};
|
|
78
|
+
}
|
|
240
79
|
}
|
|
80
|
+
if (transactionBody.tokenGrantKyc) {
|
|
81
|
+
const tokenGrantKyc = this.parseTokenGrantKyc(
|
|
82
|
+
transactionBody.tokenGrantKyc
|
|
83
|
+
);
|
|
84
|
+
if (tokenGrantKyc) {
|
|
85
|
+
return {
|
|
86
|
+
type: "TOKENGRANTKYC",
|
|
87
|
+
humanReadableType: "Token Grant KYC",
|
|
88
|
+
tokenGrantKyc
|
|
89
|
+
};
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
if (transactionBody.tokenRevokeKyc) {
|
|
93
|
+
const tokenRevokeKyc = this.parseTokenRevokeKyc(
|
|
94
|
+
transactionBody.tokenRevokeKyc
|
|
95
|
+
);
|
|
96
|
+
if (tokenRevokeKyc) {
|
|
97
|
+
return {
|
|
98
|
+
type: "TOKENREVOKEKYC",
|
|
99
|
+
humanReadableType: "Token Revoke KYC",
|
|
100
|
+
tokenRevokeKyc
|
|
101
|
+
};
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
if (transactionBody.tokenPause) {
|
|
105
|
+
const tokenPause = this.parseTokenPause(transactionBody.tokenPause);
|
|
106
|
+
if (tokenPause) {
|
|
107
|
+
return {
|
|
108
|
+
type: "TOKENPAUSE",
|
|
109
|
+
humanReadableType: "Token Pause",
|
|
110
|
+
tokenPause
|
|
111
|
+
};
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
if (transactionBody.tokenUnpause) {
|
|
115
|
+
const tokenUnpause = this.parseTokenUnpause(
|
|
116
|
+
transactionBody.tokenUnpause
|
|
117
|
+
);
|
|
118
|
+
if (tokenUnpause) {
|
|
119
|
+
return {
|
|
120
|
+
type: "TOKENUNPAUSE",
|
|
121
|
+
humanReadableType: "Token Unpause",
|
|
122
|
+
tokenUnpause
|
|
123
|
+
};
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
if (transactionBody.tokenWipe) {
|
|
127
|
+
const tokenWipeAccount = this.parseTokenWipeAccount(
|
|
128
|
+
transactionBody.tokenWipe
|
|
129
|
+
);
|
|
130
|
+
if (tokenWipeAccount) {
|
|
131
|
+
return {
|
|
132
|
+
type: "TOKENWIPEACCOUNT",
|
|
133
|
+
humanReadableType: "Token Wipe Account",
|
|
134
|
+
tokenWipeAccount
|
|
135
|
+
};
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
if (transactionBody.tokenDeletion) {
|
|
139
|
+
const tokenDelete = this.parseTokenDelete(
|
|
140
|
+
transactionBody.tokenDeletion
|
|
141
|
+
);
|
|
142
|
+
if (tokenDelete) {
|
|
143
|
+
return {
|
|
144
|
+
type: "TOKENDELETE",
|
|
145
|
+
humanReadableType: "Token Delete",
|
|
146
|
+
tokenDelete
|
|
147
|
+
};
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
if (transactionBody.tokenAssociate) {
|
|
151
|
+
const tokenAssociate = this.parseTokenAssociate(
|
|
152
|
+
transactionBody.tokenAssociate
|
|
153
|
+
);
|
|
154
|
+
if (tokenAssociate) {
|
|
155
|
+
return {
|
|
156
|
+
type: "TOKENASSOCIATE",
|
|
157
|
+
humanReadableType: "Token Associate",
|
|
158
|
+
tokenAssociate
|
|
159
|
+
};
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
if (transactionBody.tokenDissociate) {
|
|
163
|
+
const tokenDissociate = this.parseTokenDissociate(
|
|
164
|
+
transactionBody.tokenDissociate
|
|
165
|
+
);
|
|
166
|
+
if (tokenDissociate) {
|
|
167
|
+
return {
|
|
168
|
+
type: "TOKENDISSOCIATE",
|
|
169
|
+
humanReadableType: "Token Dissociate",
|
|
170
|
+
tokenDissociate
|
|
171
|
+
};
|
|
172
|
+
}
|
|
173
|
+
}
|
|
174
|
+
if (transactionBody.tokenFeeScheduleUpdate) {
|
|
175
|
+
const tokenFeeScheduleUpdate = this.parseTokenFeeScheduleUpdate(
|
|
176
|
+
transactionBody.tokenFeeScheduleUpdate
|
|
177
|
+
);
|
|
178
|
+
if (tokenFeeScheduleUpdate) {
|
|
179
|
+
return {
|
|
180
|
+
type: "TOKENFEESCHEDULEUPDATE",
|
|
181
|
+
humanReadableType: "Token Fee Schedule Update",
|
|
182
|
+
tokenFeeScheduleUpdate
|
|
183
|
+
};
|
|
184
|
+
}
|
|
185
|
+
}
|
|
186
|
+
const airdrop = this.parseTokenAirdrop(transaction);
|
|
187
|
+
if (airdrop) {
|
|
188
|
+
return {
|
|
189
|
+
type: "TOKENAIRDROP",
|
|
190
|
+
humanReadableType: "Token Airdrop",
|
|
191
|
+
tokenAirdrop: airdrop
|
|
192
|
+
};
|
|
193
|
+
}
|
|
194
|
+
return {};
|
|
195
|
+
} catch (error) {
|
|
196
|
+
console.warn("[HTSParser] Failed to parse HTS transaction:", error);
|
|
197
|
+
return {};
|
|
241
198
|
}
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
}
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
|
|
252
|
-
|
|
253
|
-
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
|
|
267
|
-
|
|
268
|
-
|
|
269
|
-
|
|
270
|
-
|
|
271
|
-
|
|
272
|
-
|
|
273
|
-
|
|
274
|
-
|
|
275
|
-
|
|
276
|
-
|
|
277
|
-
|
|
278
|
-
|
|
279
|
-
|
|
280
|
-
|
|
281
|
-
|
|
282
|
-
|
|
283
|
-
|
|
284
|
-
|
|
285
|
-
|
|
286
|
-
|
|
287
|
-
|
|
288
|
-
|
|
289
|
-
|
|
290
|
-
|
|
291
|
-
|
|
292
|
-
|
|
293
|
-
|
|
294
|
-
|
|
295
|
-
|
|
296
|
-
|
|
297
|
-
|
|
298
|
-
|
|
199
|
+
}
|
|
200
|
+
static parseTokenCreate(body) {
|
|
201
|
+
if (!body) return void 0;
|
|
202
|
+
const data = {};
|
|
203
|
+
if (body.name) {
|
|
204
|
+
data.tokenName = body.name;
|
|
205
|
+
}
|
|
206
|
+
if (body.symbol) {
|
|
207
|
+
data.tokenSymbol = body.symbol;
|
|
208
|
+
}
|
|
209
|
+
if (body.treasury) {
|
|
210
|
+
data.treasuryAccountId = new AccountId(
|
|
211
|
+
body.treasury.shardNum ?? 0,
|
|
212
|
+
body.treasury.realmNum ?? 0,
|
|
213
|
+
body.treasury.accountNum ?? 0
|
|
214
|
+
).toString();
|
|
215
|
+
}
|
|
216
|
+
if (body.initialSupply) {
|
|
217
|
+
data.initialSupply = Long.fromValue(body.initialSupply).toString();
|
|
218
|
+
}
|
|
219
|
+
if (body.decimals !== void 0 && body.decimals !== null) {
|
|
220
|
+
data.decimals = Long.fromValue(body.decimals).toNumber();
|
|
221
|
+
}
|
|
222
|
+
if (body.maxSupply) {
|
|
223
|
+
data.maxSupply = Long.fromValue(body.maxSupply).toString();
|
|
224
|
+
}
|
|
225
|
+
if (body.memo) {
|
|
226
|
+
data.memo = body.memo;
|
|
227
|
+
}
|
|
228
|
+
if (body.tokenType !== null && body.tokenType !== void 0) {
|
|
229
|
+
data.tokenType = proto.TokenType[body.tokenType];
|
|
230
|
+
}
|
|
231
|
+
if (body.supplyType !== null && body.supplyType !== void 0) {
|
|
232
|
+
data.supplyType = proto.TokenSupplyType[body.supplyType];
|
|
233
|
+
}
|
|
234
|
+
data.adminKey = parseKey(body.adminKey);
|
|
235
|
+
data.kycKey = parseKey(body.kycKey);
|
|
236
|
+
data.freezeKey = parseKey(body.freezeKey);
|
|
237
|
+
data.wipeKey = parseKey(body.wipeKey);
|
|
238
|
+
data.supplyKey = parseKey(body.supplyKey);
|
|
239
|
+
data.feeScheduleKey = parseKey(body.feeScheduleKey);
|
|
240
|
+
data.pauseKey = parseKey(body.pauseKey);
|
|
241
|
+
if (body.autoRenewAccount) {
|
|
242
|
+
data.autoRenewAccount = new AccountId(
|
|
243
|
+
body.autoRenewAccount.shardNum ?? 0,
|
|
244
|
+
body.autoRenewAccount.realmNum ?? 0,
|
|
245
|
+
body.autoRenewAccount.accountNum ?? 0
|
|
246
|
+
).toString();
|
|
247
|
+
}
|
|
248
|
+
if (body.autoRenewPeriod?.seconds) {
|
|
249
|
+
data.autoRenewPeriod = Long.fromValue(
|
|
250
|
+
body.autoRenewPeriod.seconds
|
|
251
|
+
).toString();
|
|
252
|
+
}
|
|
253
|
+
if (body.customFees && body.customFees.length > 0) {
|
|
254
|
+
data.customFees = body.customFees.map((fee) => {
|
|
255
|
+
const feeCollectorAccountId = fee.feeCollectorAccountId ? new AccountId(
|
|
256
|
+
fee.feeCollectorAccountId.shardNum ?? 0,
|
|
257
|
+
fee.feeCollectorAccountId.realmNum ?? 0,
|
|
258
|
+
fee.feeCollectorAccountId.accountNum ?? 0
|
|
259
|
+
).toString() : "Not Set";
|
|
260
|
+
const commonFeeData = {
|
|
261
|
+
feeCollectorAccountId,
|
|
262
|
+
allCollectorsAreExempt: fee.allCollectorsAreExempt || false
|
|
263
|
+
};
|
|
264
|
+
if (fee.fixedFee) {
|
|
265
|
+
return {
|
|
266
|
+
...commonFeeData,
|
|
267
|
+
feeType: "FIXED_FEE",
|
|
268
|
+
fixedFee: {
|
|
269
|
+
amount: Long.fromValue(fee.fixedFee.amount || 0).toString(),
|
|
270
|
+
denominatingTokenId: fee.fixedFee.denominatingTokenId ? new TokenId(
|
|
271
|
+
fee.fixedFee.denominatingTokenId.shardNum ?? 0,
|
|
272
|
+
fee.fixedFee.denominatingTokenId.realmNum ?? 0,
|
|
273
|
+
fee.fixedFee.denominatingTokenId.tokenNum ?? 0
|
|
274
|
+
).toString() : void 0
|
|
299
275
|
}
|
|
276
|
+
};
|
|
277
|
+
} else if (fee.fractionalFee) {
|
|
278
|
+
return {
|
|
279
|
+
...commonFeeData,
|
|
280
|
+
feeType: "FRACTIONAL_FEE",
|
|
281
|
+
fractionalFee: {
|
|
282
|
+
numerator: Long.fromValue(
|
|
283
|
+
fee.fractionalFee.fractionalAmount?.numerator || 0
|
|
284
|
+
).toString(),
|
|
285
|
+
denominator: Long.fromValue(
|
|
286
|
+
fee.fractionalFee.fractionalAmount?.denominator || 1
|
|
287
|
+
).toString(),
|
|
288
|
+
minimumAmount: Long.fromValue(
|
|
289
|
+
fee.fractionalFee.minimumAmount || 0
|
|
290
|
+
).toString(),
|
|
291
|
+
maximumAmount: Long.fromValue(
|
|
292
|
+
fee.fractionalFee.maximumAmount || 0
|
|
293
|
+
).toString(),
|
|
294
|
+
netOfTransfers: fee.fractionalFee.netOfTransfers || false
|
|
295
|
+
}
|
|
296
|
+
};
|
|
297
|
+
} else if (fee.royaltyFee) {
|
|
298
|
+
let fallbackFeeData = void 0;
|
|
299
|
+
if (fee.royaltyFee.fallbackFee) {
|
|
300
|
+
fallbackFeeData = {
|
|
301
|
+
amount: Long.fromValue(
|
|
302
|
+
fee.royaltyFee.fallbackFee.amount || 0
|
|
303
|
+
).toString(),
|
|
304
|
+
denominatingTokenId: fee.royaltyFee.fallbackFee.denominatingTokenId ? new TokenId(
|
|
305
|
+
fee.royaltyFee.fallbackFee.denominatingTokenId.shardNum ?? 0,
|
|
306
|
+
fee.royaltyFee.fallbackFee.denominatingTokenId.realmNum ?? 0,
|
|
307
|
+
fee.royaltyFee.fallbackFee.denominatingTokenId.tokenNum ?? 0
|
|
308
|
+
).toString() : void 0
|
|
309
|
+
};
|
|
300
310
|
}
|
|
311
|
+
return {
|
|
312
|
+
...commonFeeData,
|
|
313
|
+
feeType: "ROYALTY_FEE",
|
|
314
|
+
royaltyFee: {
|
|
315
|
+
numerator: Long.fromValue(
|
|
316
|
+
fee.royaltyFee.exchangeValueFraction?.numerator || 0
|
|
317
|
+
).toString(),
|
|
318
|
+
denominator: Long.fromValue(
|
|
319
|
+
fee.royaltyFee.exchangeValueFraction?.denominator || 1
|
|
320
|
+
).toString(),
|
|
321
|
+
fallbackFee: fallbackFeeData
|
|
322
|
+
}
|
|
323
|
+
};
|
|
301
324
|
}
|
|
302
|
-
|
|
303
|
-
|
|
304
|
-
|
|
305
|
-
|
|
306
|
-
|
|
307
|
-
|
|
308
|
-
|
|
309
|
-
|
|
310
|
-
return messages;
|
|
325
|
+
return {
|
|
326
|
+
...commonFeeData,
|
|
327
|
+
feeType: "FIXED_FEE",
|
|
328
|
+
fixedFee: { amount: "0" }
|
|
329
|
+
};
|
|
330
|
+
});
|
|
331
|
+
}
|
|
332
|
+
return data;
|
|
311
333
|
}
|
|
312
|
-
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
318
|
-
|
|
319
|
-
|
|
320
|
-
|
|
321
|
-
|
|
322
|
-
|
|
334
|
+
static parseTokenMint(body) {
|
|
335
|
+
if (!body || !body.token || body.amount === null || body.amount === void 0) {
|
|
336
|
+
return void 0;
|
|
337
|
+
}
|
|
338
|
+
const data = {
|
|
339
|
+
tokenId: new TokenId(
|
|
340
|
+
body.token.shardNum ?? 0,
|
|
341
|
+
body.token.realmNum ?? 0,
|
|
342
|
+
body.token.tokenNum ?? 0
|
|
343
|
+
).toString(),
|
|
344
|
+
amount: Long.fromValue(body.amount).toNumber()
|
|
345
|
+
};
|
|
346
|
+
if (body.metadata && body.metadata.length > 0) {
|
|
347
|
+
data.metadata = body.metadata.map(
|
|
348
|
+
(meta) => Buffer.from(meta).toString("base64")
|
|
323
349
|
);
|
|
324
|
-
if (!data) {
|
|
325
|
-
throw new Error(
|
|
326
|
-
`No data received from mirror node for account: ${accountId}`
|
|
327
|
-
);
|
|
328
|
-
}
|
|
329
|
-
return data;
|
|
330
|
-
} catch (e) {
|
|
331
|
-
const error = e;
|
|
332
|
-
const logMessage = `Failed to fetch account ${accountId} after retries: ${error.message}`;
|
|
333
|
-
this.logger.error(logMessage);
|
|
334
|
-
throw new Error(logMessage);
|
|
335
350
|
}
|
|
351
|
+
return data;
|
|
336
352
|
}
|
|
337
|
-
|
|
338
|
-
|
|
339
|
-
|
|
340
|
-
|
|
341
|
-
|
|
342
|
-
|
|
343
|
-
|
|
344
|
-
|
|
345
|
-
|
|
346
|
-
|
|
347
|
-
|
|
348
|
-
|
|
349
|
-
|
|
350
|
-
|
|
351
|
-
|
|
353
|
+
static parseTokenBurn(body) {
|
|
354
|
+
if (!body || !body.token || body.amount === null || body.amount === void 0) {
|
|
355
|
+
return void 0;
|
|
356
|
+
}
|
|
357
|
+
const data = {
|
|
358
|
+
tokenId: new TokenId(
|
|
359
|
+
body.token.shardNum ?? 0,
|
|
360
|
+
body.token.realmNum ?? 0,
|
|
361
|
+
body.token.tokenNum ?? 0
|
|
362
|
+
).toString(),
|
|
363
|
+
amount: Long.fromValue(body.amount).toNumber()
|
|
364
|
+
};
|
|
365
|
+
if (body.serialNumbers && body.serialNumbers.length > 0) {
|
|
366
|
+
data.serialNumbers = body.serialNumbers.map(
|
|
367
|
+
(sn) => Long.fromValue(sn).toNumber()
|
|
368
|
+
);
|
|
352
369
|
}
|
|
370
|
+
return data;
|
|
353
371
|
}
|
|
354
|
-
|
|
355
|
-
|
|
356
|
-
|
|
357
|
-
|
|
358
|
-
|
|
359
|
-
|
|
360
|
-
|
|
361
|
-
|
|
362
|
-
|
|
363
|
-
}
|
|
364
|
-
if (
|
|
365
|
-
|
|
366
|
-
}
|
|
367
|
-
if (
|
|
368
|
-
|
|
369
|
-
}
|
|
370
|
-
|
|
372
|
+
static parseTokenUpdate(body) {
|
|
373
|
+
if (!body) return void 0;
|
|
374
|
+
const data = {};
|
|
375
|
+
if (body.token) {
|
|
376
|
+
data.tokenId = new TokenId(
|
|
377
|
+
body.token.shardNum ?? 0,
|
|
378
|
+
body.token.realmNum ?? 0,
|
|
379
|
+
body.token.tokenNum ?? 0
|
|
380
|
+
).toString();
|
|
381
|
+
}
|
|
382
|
+
if (body.name) {
|
|
383
|
+
data.name = body.name;
|
|
384
|
+
}
|
|
385
|
+
if (body.symbol) {
|
|
386
|
+
data.symbol = body.symbol;
|
|
387
|
+
}
|
|
388
|
+
if (body.treasury) {
|
|
389
|
+
data.treasuryAccountId = new AccountId(
|
|
390
|
+
body.treasury.shardNum ?? 0,
|
|
391
|
+
body.treasury.realmNum ?? 0,
|
|
392
|
+
body.treasury.accountNum ?? 0
|
|
393
|
+
).toString();
|
|
394
|
+
}
|
|
395
|
+
data.adminKey = parseKey(body.adminKey);
|
|
396
|
+
data.kycKey = parseKey(body.kycKey);
|
|
397
|
+
data.freezeKey = parseKey(body.freezeKey);
|
|
398
|
+
data.wipeKey = parseKey(body.wipeKey);
|
|
399
|
+
data.supplyKey = parseKey(body.supplyKey);
|
|
400
|
+
data.feeScheduleKey = parseKey(body.feeScheduleKey);
|
|
401
|
+
data.pauseKey = parseKey(body.pauseKey);
|
|
402
|
+
if (body.autoRenewAccount) {
|
|
403
|
+
data.autoRenewAccountId = new AccountId(
|
|
404
|
+
body.autoRenewAccount.shardNum ?? 0,
|
|
405
|
+
body.autoRenewAccount.realmNum ?? 0,
|
|
406
|
+
body.autoRenewAccount.accountNum ?? 0
|
|
407
|
+
).toString();
|
|
408
|
+
}
|
|
409
|
+
if (body.autoRenewPeriod?.seconds) {
|
|
410
|
+
data.autoRenewPeriod = Long.fromValue(
|
|
411
|
+
body.autoRenewPeriod.seconds
|
|
412
|
+
).toString();
|
|
413
|
+
}
|
|
414
|
+
if (body.memo?.value !== void 0) {
|
|
415
|
+
data.memo = body.memo.value;
|
|
416
|
+
}
|
|
417
|
+
if (body.expiry?.seconds) {
|
|
418
|
+
data.expiry = `${Long.fromValue(body.expiry.seconds).toString()}.${body.expiry.nanos}`;
|
|
419
|
+
}
|
|
420
|
+
return data;
|
|
371
421
|
}
|
|
372
|
-
|
|
373
|
-
|
|
374
|
-
|
|
375
|
-
|
|
376
|
-
|
|
377
|
-
|
|
378
|
-
|
|
379
|
-
|
|
380
|
-
|
|
381
|
-
|
|
382
|
-
|
|
383
|
-
|
|
384
|
-
|
|
385
|
-
|
|
386
|
-
|
|
387
|
-
|
|
388
|
-
|
|
389
|
-
|
|
390
|
-
|
|
391
|
-
|
|
392
|
-
|
|
393
|
-
|
|
394
|
-
|
|
395
|
-
|
|
396
|
-
|
|
397
|
-
|
|
422
|
+
static parseTokenFeeScheduleUpdate(body) {
|
|
423
|
+
if (!body) return void 0;
|
|
424
|
+
const data = {};
|
|
425
|
+
if (body.tokenId) {
|
|
426
|
+
data.tokenId = new TokenId(
|
|
427
|
+
body.tokenId.shardNum ?? 0,
|
|
428
|
+
body.tokenId.realmNum ?? 0,
|
|
429
|
+
body.tokenId.tokenNum ?? 0
|
|
430
|
+
).toString();
|
|
431
|
+
}
|
|
432
|
+
if (body.customFees && body.customFees.length > 0) {
|
|
433
|
+
data.customFees = body.customFees.map((fee) => {
|
|
434
|
+
const feeCollectorAccountId = fee.feeCollectorAccountId ? new AccountId(
|
|
435
|
+
fee.feeCollectorAccountId.shardNum ?? 0,
|
|
436
|
+
fee.feeCollectorAccountId.realmNum ?? 0,
|
|
437
|
+
fee.feeCollectorAccountId.accountNum ?? 0
|
|
438
|
+
).toString() : "Not Set";
|
|
439
|
+
const commonFeeData = {
|
|
440
|
+
feeCollectorAccountId,
|
|
441
|
+
allCollectorsAreExempt: fee.allCollectorsAreExempt || false
|
|
442
|
+
};
|
|
443
|
+
if (fee.fixedFee) {
|
|
444
|
+
return {
|
|
445
|
+
...commonFeeData,
|
|
446
|
+
feeType: "FIXED_FEE",
|
|
447
|
+
fixedFee: {
|
|
448
|
+
amount: Long.fromValue(fee.fixedFee.amount || 0).toString(),
|
|
449
|
+
denominatingTokenId: fee.fixedFee.denominatingTokenId ? new TokenId(
|
|
450
|
+
fee.fixedFee.denominatingTokenId.shardNum ?? 0,
|
|
451
|
+
fee.fixedFee.denominatingTokenId.realmNum ?? 0,
|
|
452
|
+
fee.fixedFee.denominatingTokenId.tokenNum ?? 0
|
|
453
|
+
).toString() : void 0
|
|
454
|
+
}
|
|
455
|
+
};
|
|
456
|
+
} else if (fee.fractionalFee) {
|
|
457
|
+
return {
|
|
458
|
+
...commonFeeData,
|
|
459
|
+
feeType: "FRACTIONAL_FEE",
|
|
460
|
+
fractionalFee: {
|
|
461
|
+
numerator: Long.fromValue(
|
|
462
|
+
fee.fractionalFee.fractionalAmount?.numerator || 0
|
|
463
|
+
).toString(),
|
|
464
|
+
denominator: Long.fromValue(
|
|
465
|
+
fee.fractionalFee.fractionalAmount?.denominator || 1
|
|
466
|
+
).toString(),
|
|
467
|
+
minimumAmount: Long.fromValue(
|
|
468
|
+
fee.fractionalFee.minimumAmount || 0
|
|
469
|
+
).toString(),
|
|
470
|
+
maximumAmount: Long.fromValue(
|
|
471
|
+
fee.fractionalFee.maximumAmount || 0
|
|
472
|
+
).toString(),
|
|
473
|
+
netOfTransfers: fee.fractionalFee.netOfTransfers || false
|
|
474
|
+
}
|
|
475
|
+
};
|
|
476
|
+
} else if (fee.royaltyFee) {
|
|
477
|
+
let fallbackFeeData = void 0;
|
|
478
|
+
if (fee.royaltyFee.fallbackFee) {
|
|
479
|
+
fallbackFeeData = {
|
|
480
|
+
amount: Long.fromValue(
|
|
481
|
+
fee.royaltyFee.fallbackFee.amount || 0
|
|
482
|
+
).toString(),
|
|
483
|
+
denominatingTokenId: fee.royaltyFee.fallbackFee.denominatingTokenId ? new TokenId(
|
|
484
|
+
fee.royaltyFee.fallbackFee.denominatingTokenId.shardNum ?? 0,
|
|
485
|
+
fee.royaltyFee.fallbackFee.denominatingTokenId.realmNum ?? 0,
|
|
486
|
+
fee.royaltyFee.fallbackFee.denominatingTokenId.tokenNum ?? 0
|
|
487
|
+
).toString() : void 0
|
|
488
|
+
};
|
|
398
489
|
}
|
|
399
|
-
|
|
400
|
-
|
|
401
|
-
|
|
402
|
-
|
|
490
|
+
return {
|
|
491
|
+
...commonFeeData,
|
|
492
|
+
feeType: "ROYALTY_FEE",
|
|
493
|
+
royaltyFee: {
|
|
494
|
+
numerator: Long.fromValue(
|
|
495
|
+
fee.royaltyFee.exchangeValueFraction?.numerator || 0
|
|
496
|
+
).toString(),
|
|
497
|
+
denominator: Long.fromValue(
|
|
498
|
+
fee.royaltyFee.exchangeValueFraction?.denominator || 1
|
|
499
|
+
).toString(),
|
|
500
|
+
fallbackFee: fallbackFeeData
|
|
501
|
+
}
|
|
502
|
+
};
|
|
403
503
|
}
|
|
404
|
-
|
|
405
|
-
|
|
406
|
-
|
|
504
|
+
return {
|
|
505
|
+
...commonFeeData,
|
|
506
|
+
feeType: "FIXED_FEE",
|
|
507
|
+
fixedFee: { amount: "0" }
|
|
508
|
+
};
|
|
509
|
+
});
|
|
510
|
+
}
|
|
511
|
+
return data;
|
|
407
512
|
}
|
|
408
|
-
|
|
409
|
-
|
|
410
|
-
|
|
411
|
-
|
|
412
|
-
|
|
413
|
-
|
|
414
|
-
|
|
415
|
-
|
|
416
|
-
|
|
417
|
-
|
|
418
|
-
|
|
419
|
-
|
|
420
|
-
|
|
421
|
-
|
|
422
|
-
|
|
423
|
-
|
|
513
|
+
static parseTokenFreeze(body) {
|
|
514
|
+
if (!body) return void 0;
|
|
515
|
+
const data = {};
|
|
516
|
+
if (body.token) {
|
|
517
|
+
data.tokenId = new TokenId(
|
|
518
|
+
body.token.shardNum ?? 0,
|
|
519
|
+
body.token.realmNum ?? 0,
|
|
520
|
+
body.token.tokenNum ?? 0
|
|
521
|
+
).toString();
|
|
522
|
+
}
|
|
523
|
+
if (body.account) {
|
|
524
|
+
data.accountId = new AccountId(
|
|
525
|
+
body.account.shardNum ?? 0,
|
|
526
|
+
body.account.realmNum ?? 0,
|
|
527
|
+
body.account.accountNum ?? 0
|
|
528
|
+
).toString();
|
|
529
|
+
}
|
|
530
|
+
return data;
|
|
424
531
|
}
|
|
425
|
-
|
|
426
|
-
|
|
427
|
-
|
|
428
|
-
|
|
429
|
-
|
|
430
|
-
|
|
431
|
-
|
|
432
|
-
|
|
433
|
-
|
|
434
|
-
|
|
435
|
-
|
|
436
|
-
|
|
437
|
-
|
|
438
|
-
|
|
439
|
-
|
|
440
|
-
|
|
441
|
-
|
|
442
|
-
|
|
443
|
-
);
|
|
444
|
-
return null;
|
|
445
|
-
} catch (error) {
|
|
446
|
-
this.logger.error(
|
|
447
|
-
`Error fetching schedule info for ${scheduleId} after retries: ${error.message}`
|
|
448
|
-
);
|
|
449
|
-
return null;
|
|
450
|
-
}
|
|
532
|
+
static parseTokenUnfreeze(body) {
|
|
533
|
+
if (!body) return void 0;
|
|
534
|
+
const data = {};
|
|
535
|
+
if (body.token) {
|
|
536
|
+
data.tokenId = new TokenId(
|
|
537
|
+
body.token.shardNum ?? 0,
|
|
538
|
+
body.token.realmNum ?? 0,
|
|
539
|
+
body.token.tokenNum ?? 0
|
|
540
|
+
).toString();
|
|
541
|
+
}
|
|
542
|
+
if (body.account) {
|
|
543
|
+
data.accountId = new AccountId(
|
|
544
|
+
body.account.shardNum ?? 0,
|
|
545
|
+
body.account.realmNum ?? 0,
|
|
546
|
+
body.account.accountNum ?? 0
|
|
547
|
+
).toString();
|
|
548
|
+
}
|
|
549
|
+
return data;
|
|
451
550
|
}
|
|
452
|
-
|
|
453
|
-
|
|
454
|
-
|
|
455
|
-
|
|
456
|
-
|
|
457
|
-
|
|
458
|
-
|
|
459
|
-
|
|
460
|
-
|
|
461
|
-
|
|
462
|
-
|
|
463
|
-
|
|
464
|
-
|
|
465
|
-
|
|
466
|
-
|
|
467
|
-
|
|
468
|
-
|
|
469
|
-
|
|
470
|
-
|
|
471
|
-
|
|
472
|
-
|
|
473
|
-
|
|
551
|
+
static parseTokenGrantKyc(body) {
|
|
552
|
+
if (!body) return void 0;
|
|
553
|
+
const data = {};
|
|
554
|
+
if (body.token) {
|
|
555
|
+
data.tokenId = new TokenId(
|
|
556
|
+
body.token.shardNum ?? 0,
|
|
557
|
+
body.token.realmNum ?? 0,
|
|
558
|
+
body.token.tokenNum ?? 0
|
|
559
|
+
).toString();
|
|
560
|
+
}
|
|
561
|
+
if (body.account) {
|
|
562
|
+
data.accountId = new AccountId(
|
|
563
|
+
body.account.shardNum ?? 0,
|
|
564
|
+
body.account.realmNum ?? 0,
|
|
565
|
+
body.account.accountNum ?? 0
|
|
566
|
+
).toString();
|
|
567
|
+
}
|
|
568
|
+
return data;
|
|
569
|
+
}
|
|
570
|
+
static parseTokenRevokeKyc(body) {
|
|
571
|
+
if (!body) return void 0;
|
|
572
|
+
const data = {};
|
|
573
|
+
if (body.token) {
|
|
574
|
+
data.tokenId = new TokenId(
|
|
575
|
+
body.token.shardNum ?? 0,
|
|
576
|
+
body.token.realmNum ?? 0,
|
|
577
|
+
body.token.tokenNum ?? 0
|
|
578
|
+
).toString();
|
|
579
|
+
}
|
|
580
|
+
if (body.account) {
|
|
581
|
+
data.accountId = new AccountId(
|
|
582
|
+
body.account.shardNum ?? 0,
|
|
583
|
+
body.account.realmNum ?? 0,
|
|
584
|
+
body.account.accountNum ?? 0
|
|
585
|
+
).toString();
|
|
586
|
+
}
|
|
587
|
+
return data;
|
|
588
|
+
}
|
|
589
|
+
static parseTokenPause(body) {
|
|
590
|
+
if (!body) return void 0;
|
|
591
|
+
const data = {};
|
|
592
|
+
if (body.token) {
|
|
593
|
+
data.tokenId = new TokenId(
|
|
594
|
+
body.token.shardNum ?? 0,
|
|
595
|
+
body.token.realmNum ?? 0,
|
|
596
|
+
body.token.tokenNum ?? 0
|
|
597
|
+
).toString();
|
|
598
|
+
}
|
|
599
|
+
return data;
|
|
600
|
+
}
|
|
601
|
+
static parseTokenUnpause(body) {
|
|
602
|
+
if (!body) return void 0;
|
|
603
|
+
const data = {};
|
|
604
|
+
if (body.token) {
|
|
605
|
+
data.tokenId = new TokenId(
|
|
606
|
+
body.token.shardNum ?? 0,
|
|
607
|
+
body.token.realmNum ?? 0,
|
|
608
|
+
body.token.tokenNum ?? 0
|
|
609
|
+
).toString();
|
|
610
|
+
}
|
|
611
|
+
return data;
|
|
612
|
+
}
|
|
613
|
+
static parseTokenWipeAccount(body) {
|
|
614
|
+
if (!body) return void 0;
|
|
615
|
+
const data = {};
|
|
616
|
+
if (body.token) {
|
|
617
|
+
data.tokenId = new TokenId(
|
|
618
|
+
body.token.shardNum ?? 0,
|
|
619
|
+
body.token.realmNum ?? 0,
|
|
620
|
+
body.token.tokenNum ?? 0
|
|
621
|
+
).toString();
|
|
622
|
+
}
|
|
623
|
+
if (body.account) {
|
|
624
|
+
data.accountId = new AccountId(
|
|
625
|
+
body.account.shardNum ?? 0,
|
|
626
|
+
body.account.realmNum ?? 0,
|
|
627
|
+
body.account.accountNum ?? 0
|
|
628
|
+
).toString();
|
|
629
|
+
}
|
|
630
|
+
if (body.serialNumbers && body.serialNumbers.length > 0) {
|
|
631
|
+
data.serialNumbers = body.serialNumbers.map(
|
|
632
|
+
(sn) => Long.fromValue(sn).toString()
|
|
474
633
|
);
|
|
475
|
-
throw error;
|
|
476
634
|
}
|
|
635
|
+
if (body.amount) {
|
|
636
|
+
data.amount = Long.fromValue(body.amount).toString();
|
|
637
|
+
}
|
|
638
|
+
return data;
|
|
477
639
|
}
|
|
478
|
-
|
|
479
|
-
|
|
480
|
-
|
|
481
|
-
|
|
482
|
-
|
|
483
|
-
|
|
484
|
-
|
|
485
|
-
|
|
486
|
-
|
|
487
|
-
|
|
488
|
-
|
|
489
|
-
|
|
490
|
-
|
|
491
|
-
|
|
492
|
-
|
|
493
|
-
|
|
494
|
-
|
|
495
|
-
|
|
496
|
-
|
|
497
|
-
|
|
498
|
-
|
|
499
|
-
|
|
500
|
-
|
|
501
|
-
|
|
502
|
-
|
|
503
|
-
|
|
504
|
-
|
|
640
|
+
static parseTokenDelete(body) {
|
|
641
|
+
if (!body) return void 0;
|
|
642
|
+
const data = {};
|
|
643
|
+
if (body.token) {
|
|
644
|
+
data.tokenId = new TokenId(
|
|
645
|
+
body.token.shardNum ?? 0,
|
|
646
|
+
body.token.realmNum ?? 0,
|
|
647
|
+
body.token.tokenNum ?? 0
|
|
648
|
+
).toString();
|
|
649
|
+
}
|
|
650
|
+
return data;
|
|
651
|
+
}
|
|
652
|
+
static parseTokenAssociate(body) {
|
|
653
|
+
if (!body) return void 0;
|
|
654
|
+
const data = {};
|
|
655
|
+
if (body.account) {
|
|
656
|
+
data.accountId = new AccountId(
|
|
657
|
+
body.account.shardNum ?? 0,
|
|
658
|
+
body.account.realmNum ?? 0,
|
|
659
|
+
body.account.accountNum ?? 0
|
|
660
|
+
).toString();
|
|
661
|
+
}
|
|
662
|
+
if (body.tokens && body.tokens.length > 0) {
|
|
663
|
+
data.tokenIds = body.tokens.map(
|
|
664
|
+
(t) => new TokenId(
|
|
665
|
+
t.shardNum ?? 0,
|
|
666
|
+
t.realmNum ?? 0,
|
|
667
|
+
t.tokenNum ?? 0
|
|
668
|
+
).toString()
|
|
505
669
|
);
|
|
506
|
-
return null;
|
|
507
670
|
}
|
|
671
|
+
return data;
|
|
508
672
|
}
|
|
509
|
-
|
|
510
|
-
|
|
511
|
-
|
|
512
|
-
|
|
513
|
-
|
|
514
|
-
|
|
515
|
-
|
|
516
|
-
|
|
517
|
-
|
|
518
|
-
|
|
519
|
-
|
|
520
|
-
|
|
521
|
-
|
|
522
|
-
|
|
523
|
-
|
|
524
|
-
|
|
525
|
-
|
|
526
|
-
|
|
527
|
-
"X-API-Key": this.apiKey
|
|
528
|
-
};
|
|
529
|
-
}
|
|
530
|
-
while (attempt < this.maxRetries) {
|
|
531
|
-
try {
|
|
532
|
-
const response = await axios.get(url, config);
|
|
533
|
-
return response.data;
|
|
534
|
-
} catch (error) {
|
|
535
|
-
attempt++;
|
|
536
|
-
const isLastAttempt = attempt >= this.maxRetries;
|
|
537
|
-
const statusCode = error.response?.status;
|
|
538
|
-
if (statusCode && statusCode > 404 && statusCode < 500 && statusCode !== 429) {
|
|
539
|
-
this.logger.error(
|
|
540
|
-
`Client error for ${url} (status ${statusCode}): ${error.message}. Not retrying.`
|
|
541
|
-
);
|
|
542
|
-
throw error;
|
|
543
|
-
}
|
|
544
|
-
if (isLastAttempt) {
|
|
545
|
-
this.logger.error(
|
|
546
|
-
`Max retries (${this.maxRetries}) reached for ${url}. Last error: ${error.message}`
|
|
547
|
-
);
|
|
548
|
-
throw error;
|
|
549
|
-
}
|
|
550
|
-
this.logger.warn(
|
|
551
|
-
`Attempt ${attempt}/${this.maxRetries} failed for ${url}: ${error.message}. Retrying in ${delay}ms...`
|
|
552
|
-
);
|
|
553
|
-
await new Promise((resolve) => setTimeout(resolve, delay));
|
|
554
|
-
delay = Math.min(delay * this.backoffFactor, this.maxDelayMs);
|
|
555
|
-
}
|
|
673
|
+
static parseTokenDissociate(body) {
|
|
674
|
+
if (!body) return void 0;
|
|
675
|
+
const data = {};
|
|
676
|
+
if (body.account) {
|
|
677
|
+
data.accountId = new AccountId(
|
|
678
|
+
body.account.shardNum ?? 0,
|
|
679
|
+
body.account.realmNum ?? 0,
|
|
680
|
+
body.account.accountNum ?? 0
|
|
681
|
+
).toString();
|
|
682
|
+
}
|
|
683
|
+
if (body.tokens && body.tokens.length > 0) {
|
|
684
|
+
data.tokenIds = body.tokens.map(
|
|
685
|
+
(t) => new TokenId(
|
|
686
|
+
t.shardNum ?? 0,
|
|
687
|
+
t.realmNum ?? 0,
|
|
688
|
+
t.tokenNum ?? 0
|
|
689
|
+
).toString()
|
|
690
|
+
);
|
|
556
691
|
}
|
|
557
|
-
|
|
558
|
-
`Failed to fetch data from ${url} after ${this.maxRetries} attempts.`
|
|
559
|
-
);
|
|
692
|
+
return data;
|
|
560
693
|
}
|
|
561
694
|
/**
|
|
562
|
-
*
|
|
695
|
+
* Parse token airdrop transaction for both fungible tokens and NFTs
|
|
696
|
+
* Extracts airdrop transfers from transaction protobuf data
|
|
563
697
|
*/
|
|
564
|
-
|
|
565
|
-
|
|
566
|
-
|
|
567
|
-
|
|
568
|
-
|
|
569
|
-
|
|
570
|
-
|
|
571
|
-
|
|
572
|
-
|
|
573
|
-
|
|
574
|
-
|
|
575
|
-
|
|
576
|
-
|
|
577
|
-
|
|
578
|
-
|
|
579
|
-
|
|
580
|
-
|
|
698
|
+
static parseTokenAirdrop(transaction) {
|
|
699
|
+
try {
|
|
700
|
+
const transactionBody = transaction._transactionBody;
|
|
701
|
+
if (transactionBody?.tokenAirdrop) {
|
|
702
|
+
const airdrop = transactionBody.tokenAirdrop;
|
|
703
|
+
return {
|
|
704
|
+
tokenTransfers: (airdrop.tokenTransfers || []).map(
|
|
705
|
+
(transfer) => ({
|
|
706
|
+
tokenId: transfer.token?.toString() || "Unknown",
|
|
707
|
+
transfers: (transfer.transfers || []).map((t) => ({
|
|
708
|
+
accountId: t.accountID?.toString() || "Unknown",
|
|
709
|
+
amount: t.amount?.toString() || "0",
|
|
710
|
+
serialNumbers: t.serialNumbers?.map((sn) => sn.toString())
|
|
711
|
+
}))
|
|
712
|
+
})
|
|
713
|
+
)
|
|
714
|
+
};
|
|
581
715
|
}
|
|
582
|
-
|
|
583
|
-
|
|
584
|
-
|
|
585
|
-
|
|
586
|
-
}
|
|
587
|
-
const options = {
|
|
588
|
-
...fetchOptions,
|
|
589
|
-
headers
|
|
590
|
-
};
|
|
591
|
-
while (attempt < this.maxRetries) {
|
|
592
|
-
try {
|
|
593
|
-
const request = await fetch(url, options);
|
|
594
|
-
if (!request.ok) {
|
|
595
|
-
if (request.status >= 400 && request.status < 500 && request.status !== 429) {
|
|
596
|
-
this.logger.error(
|
|
597
|
-
`Client error for ${url} (status ${request.status}): ${request.statusText}. Not retrying.`
|
|
598
|
-
);
|
|
599
|
-
throw new Error(
|
|
600
|
-
`Fetch failed with status ${request.status}: ${request.statusText} for URL: ${url}`
|
|
601
|
-
);
|
|
602
|
-
}
|
|
603
|
-
throw new Error(
|
|
604
|
-
`Fetch failed with status ${request.status}: ${request.statusText} for URL: ${url}`
|
|
605
|
-
);
|
|
606
|
-
}
|
|
607
|
-
const response = await request.json();
|
|
608
|
-
return response;
|
|
609
|
-
} catch (error) {
|
|
610
|
-
attempt++;
|
|
611
|
-
if (attempt >= this.maxRetries) {
|
|
612
|
-
this.logger.error(
|
|
613
|
-
`Max retries (${this.maxRetries}) reached for ${url}. Last error: ${error.message}`
|
|
614
|
-
);
|
|
615
|
-
throw error;
|
|
716
|
+
if (hasTransactionType(transaction, "tokenAirdrop")) {
|
|
717
|
+
const txBody = extractTransactionBody(transaction);
|
|
718
|
+
if (txBody?.tokenAirdrop) {
|
|
719
|
+
return this.parseTokenAirdropFromProto(txBody.tokenAirdrop);
|
|
616
720
|
}
|
|
617
|
-
this.logger.warn(
|
|
618
|
-
`Attempt ${attempt}/${this.maxRetries} failed for ${url}: ${error.message}. Retrying in ${delay}ms...`
|
|
619
|
-
);
|
|
620
|
-
await new Promise((resolve) => setTimeout(resolve, delay));
|
|
621
|
-
delay = Math.min(delay * this.backoffFactor, this.maxDelayMs);
|
|
622
721
|
}
|
|
623
|
-
|
|
624
|
-
|
|
625
|
-
|
|
626
|
-
|
|
627
|
-
|
|
628
|
-
|
|
629
|
-
|
|
630
|
-
|
|
631
|
-
|
|
632
|
-
|
|
633
|
-
|
|
634
|
-
|
|
635
|
-
|
|
636
|
-
const accountInfo = await this.requestAccount(accountId);
|
|
637
|
-
if (accountInfo && accountInfo.balance) {
|
|
638
|
-
const hbarBalance = accountInfo.balance.balance / 1e8;
|
|
639
|
-
return hbarBalance;
|
|
722
|
+
const tx = transaction;
|
|
723
|
+
const tokenTransfersList = tx._tokenTransfers || [];
|
|
724
|
+
if (tokenTransfersList.length > 0) {
|
|
725
|
+
return {
|
|
726
|
+
tokenTransfers: tokenTransfersList.map((transfer) => ({
|
|
727
|
+
tokenId: transfer.tokenId?.toString() || "Unknown",
|
|
728
|
+
transfers: (transfer.transfers || []).map((t) => ({
|
|
729
|
+
accountId: t.accountId?.toString() || "Unknown",
|
|
730
|
+
amount: t.amount?.toString() || "0",
|
|
731
|
+
serialNumbers: t.serialNumbers?.map((sn) => sn.toString())
|
|
732
|
+
}))
|
|
733
|
+
}))
|
|
734
|
+
};
|
|
640
735
|
}
|
|
641
|
-
this.logger.warn(
|
|
642
|
-
`Could not retrieve balance for account ${accountId} from account info.`
|
|
643
|
-
);
|
|
644
736
|
return null;
|
|
645
737
|
} catch (error) {
|
|
646
|
-
|
|
647
|
-
`Error fetching numerical balance for account ${accountId}: ${error.message}`
|
|
648
|
-
);
|
|
738
|
+
console.warn("[HTSParser] Failed to parse token airdrop:", error);
|
|
649
739
|
return null;
|
|
650
740
|
}
|
|
651
741
|
}
|
|
652
742
|
/**
|
|
653
|
-
*
|
|
654
|
-
* @param topicId The ID of the topic.
|
|
655
|
-
* @param sequenceNumber Filter by sequence number (e.g., "gt:10", "lte:20").
|
|
656
|
-
* @param startTime Filter by consensus timestamp (e.g., "gt:1629400000.000000000").
|
|
657
|
-
* @param endTime Filter by consensus timestamp (e.g., "lt:1629500000.000000000").
|
|
658
|
-
* @param limit The maximum number of messages to return.
|
|
659
|
-
* @returns A promise that resolves to an array of HCSMessages or null.
|
|
743
|
+
* Parse token airdrop from protobuf data for scheduled transactions
|
|
660
744
|
*/
|
|
661
|
-
|
|
662
|
-
|
|
663
|
-
|
|
664
|
-
|
|
665
|
-
|
|
666
|
-
|
|
667
|
-
|
|
668
|
-
|
|
669
|
-
|
|
670
|
-
|
|
671
|
-
|
|
672
|
-
|
|
673
|
-
|
|
674
|
-
|
|
675
|
-
|
|
676
|
-
|
|
677
|
-
|
|
678
|
-
|
|
679
|
-
|
|
680
|
-
|
|
681
|
-
|
|
682
|
-
|
|
683
|
-
|
|
684
|
-
|
|
685
|
-
if (queryString) {
|
|
686
|
-
nextUrl += `?${queryString}`;
|
|
687
|
-
}
|
|
688
|
-
const messages = [];
|
|
689
|
-
let pagesFetched = 0;
|
|
690
|
-
const maxPages = 10;
|
|
691
|
-
try {
|
|
692
|
-
while (nextUrl && pagesFetched < maxPages) {
|
|
693
|
-
pagesFetched++;
|
|
694
|
-
const data = await this._requestWithRetry(nextUrl);
|
|
695
|
-
if (data.messages && data.messages.length > 0) {
|
|
696
|
-
for (const message of data.messages) {
|
|
697
|
-
try {
|
|
698
|
-
if (!message.message) {
|
|
699
|
-
continue;
|
|
700
|
-
}
|
|
701
|
-
let messageContent;
|
|
702
|
-
if (this.isServerEnvironment) {
|
|
703
|
-
messageContent = Buffer.from(
|
|
704
|
-
message.message,
|
|
705
|
-
"base64"
|
|
706
|
-
).toString("utf-8");
|
|
707
|
-
} else {
|
|
708
|
-
messageContent = new TextDecoder().decode(
|
|
709
|
-
Uint8Array.from(atob(message.message), (c) => c.charCodeAt(0))
|
|
710
|
-
);
|
|
711
|
-
}
|
|
712
|
-
let messageJson = {};
|
|
713
|
-
try {
|
|
714
|
-
messageJson = JSON.parse(messageContent);
|
|
715
|
-
} catch (parseError) {
|
|
716
|
-
this.logger.debug(
|
|
717
|
-
`Message content is not valid JSON, using raw: ${messageContent}`
|
|
718
|
-
);
|
|
719
|
-
messageJson = { raw_content: messageContent };
|
|
720
|
-
}
|
|
721
|
-
const parsedContent = messageJson;
|
|
722
|
-
const hcsMsg = {
|
|
723
|
-
...parsedContent,
|
|
724
|
-
consensus_timestamp: message.consensus_timestamp,
|
|
725
|
-
sequence_number: message.sequence_number,
|
|
726
|
-
payer_account_id: message.payer_account_id,
|
|
727
|
-
topic_id: message.topic_id,
|
|
728
|
-
running_hash: message.running_hash,
|
|
729
|
-
running_hash_version: message.running_hash_version,
|
|
730
|
-
chunk_info: message.chunk_info,
|
|
731
|
-
created: new Date(
|
|
732
|
-
Number(message.consensus_timestamp.split(".")[0]) * 1e3 + Number(message.consensus_timestamp.split(".")[1] || 0) / 1e6
|
|
733
|
-
),
|
|
734
|
-
payer: message.payer_account_id
|
|
735
|
-
};
|
|
736
|
-
messages.push(hcsMsg);
|
|
737
|
-
} catch (error) {
|
|
738
|
-
this.logger.error(
|
|
739
|
-
`Error processing individual message: ${error.message}`
|
|
740
|
-
);
|
|
741
|
-
}
|
|
745
|
+
static parseTokenAirdropFromProto(airdrop) {
|
|
746
|
+
const tokenTransfers = [];
|
|
747
|
+
if (airdrop.tokenTransfers) {
|
|
748
|
+
for (const tokenTransfer of airdrop.tokenTransfers) {
|
|
749
|
+
const token = tokenTransfer.token ? new TokenId(
|
|
750
|
+
tokenTransfer.token.shardNum ?? 0,
|
|
751
|
+
tokenTransfer.token.realmNum ?? 0,
|
|
752
|
+
tokenTransfer.token.tokenNum ?? 0
|
|
753
|
+
) : null;
|
|
754
|
+
const transfers = [];
|
|
755
|
+
if (tokenTransfer.transfers) {
|
|
756
|
+
for (const transfer of tokenTransfer.transfers) {
|
|
757
|
+
const accountId = transfer.accountID ? new AccountId(
|
|
758
|
+
transfer.accountID.shardNum ?? 0,
|
|
759
|
+
transfer.accountID.realmNum ?? 0,
|
|
760
|
+
transfer.accountID.accountNum ?? 0
|
|
761
|
+
) : null;
|
|
762
|
+
transfers.push({
|
|
763
|
+
accountId: accountId?.toString() || "Unknown",
|
|
764
|
+
amount: transfer.amount ? Long.fromValue(transfer.amount).toString() : "0",
|
|
765
|
+
serialNumbers: transfer.serialNumbers?.map(
|
|
766
|
+
(sn) => Long.fromValue(sn).toString()
|
|
767
|
+
)
|
|
768
|
+
});
|
|
742
769
|
}
|
|
743
770
|
}
|
|
744
|
-
|
|
745
|
-
|
|
771
|
+
tokenTransfers.push({
|
|
772
|
+
tokenId: token?.toString() || "Unknown",
|
|
773
|
+
transfers
|
|
774
|
+
});
|
|
746
775
|
}
|
|
747
|
-
return messages;
|
|
748
|
-
} catch (e) {
|
|
749
|
-
const error = e;
|
|
750
|
-
this.logger.error(
|
|
751
|
-
`Error querying filtered topic messages for ${topicId}: ${error.message}`
|
|
752
|
-
);
|
|
753
|
-
return null;
|
|
754
776
|
}
|
|
777
|
+
return { tokenTransfers };
|
|
755
778
|
}
|
|
756
779
|
/**
|
|
757
|
-
*
|
|
758
|
-
*
|
|
759
|
-
* @param limit The maximum number of tokens to return.
|
|
760
|
-
* @returns A promise that resolves to an array of AccountTokenBalance or null.
|
|
780
|
+
* Extract token creation data from Transaction object internal fields
|
|
781
|
+
* This handles the case where token creation data is stored in Transaction internals
|
|
761
782
|
*/
|
|
762
|
-
|
|
763
|
-
this.logger.info(`Getting tokens for account ${accountId}`);
|
|
764
|
-
let allTokens = [];
|
|
765
|
-
let endpoint = `/api/v1/accounts/${accountId}/tokens?limit=${limit}`;
|
|
783
|
+
static extractTokenCreationFromTransaction(transaction) {
|
|
766
784
|
try {
|
|
767
|
-
|
|
768
|
-
|
|
769
|
-
|
|
770
|
-
|
|
785
|
+
const tx = transaction;
|
|
786
|
+
if (tx._tokenName || tx._tokenSymbol) {
|
|
787
|
+
const result = {
|
|
788
|
+
tokenName: tx._tokenName || "Unknown Token",
|
|
789
|
+
tokenSymbol: tx._tokenSymbol || "UNKNOWN",
|
|
790
|
+
initialSupply: tx._initialSupply?.toString() || "0",
|
|
791
|
+
decimals: Number(tx._decimals || 0),
|
|
792
|
+
treasuryAccountId: tx._treasuryAccountId?.toString() || "Unknown"
|
|
793
|
+
};
|
|
794
|
+
if (tx._maxSupply) {
|
|
795
|
+
result.maxSupply = tx._maxSupply.toString();
|
|
771
796
|
}
|
|
772
|
-
|
|
773
|
-
|
|
774
|
-
if (limit && allTokens.length > limit) {
|
|
775
|
-
allTokens = allTokens.slice(0, limit);
|
|
776
|
-
}
|
|
777
|
-
break;
|
|
797
|
+
if (tx._tokenType) {
|
|
798
|
+
result.tokenType = tx._tokenType.toString ? tx._tokenType.toString() : String(tx._tokenType);
|
|
778
799
|
}
|
|
779
|
-
|
|
780
|
-
|
|
781
|
-
|
|
782
|
-
|
|
783
|
-
|
|
784
|
-
|
|
785
|
-
|
|
786
|
-
|
|
787
|
-
|
|
788
|
-
|
|
789
|
-
|
|
790
|
-
|
|
791
|
-
|
|
792
|
-
|
|
793
|
-
|
|
794
|
-
|
|
795
|
-
|
|
796
|
-
|
|
797
|
-
|
|
798
|
-
|
|
799
|
-
|
|
800
|
-
|
|
801
|
-
|
|
802
|
-
|
|
803
|
-
|
|
804
|
-
|
|
805
|
-
|
|
806
|
-
|
|
807
|
-
|
|
808
|
-
|
|
809
|
-
|
|
810
|
-
|
|
811
|
-
|
|
812
|
-
|
|
813
|
-
|
|
814
|
-
|
|
815
|
-
|
|
816
|
-
|
|
817
|
-
|
|
818
|
-
|
|
819
|
-
|
|
820
|
-
|
|
821
|
-
|
|
822
|
-
|
|
823
|
-
|
|
824
|
-
|
|
825
|
-
|
|
826
|
-
|
|
827
|
-
|
|
828
|
-
|
|
829
|
-
|
|
830
|
-
|
|
831
|
-
|
|
832
|
-
|
|
833
|
-
|
|
834
|
-
|
|
835
|
-
|
|
836
|
-
|
|
837
|
-
|
|
838
|
-
}
|
|
839
|
-
|
|
840
|
-
|
|
841
|
-
|
|
842
|
-
|
|
800
|
+
if (tx._supplyType) {
|
|
801
|
+
result.supplyType = tx._supplyType.toString ? tx._supplyType.toString() : String(tx._supplyType);
|
|
802
|
+
}
|
|
803
|
+
if (tx._tokenMemo) {
|
|
804
|
+
result.memo = tx._tokenMemo;
|
|
805
|
+
}
|
|
806
|
+
if (tx._adminKey) {
|
|
807
|
+
result.adminKey = tx._adminKey.toString();
|
|
808
|
+
}
|
|
809
|
+
if (tx._kycKey) {
|
|
810
|
+
result.kycKey = tx._kycKey.toString();
|
|
811
|
+
}
|
|
812
|
+
if (tx._freezeKey) {
|
|
813
|
+
result.freezeKey = tx._freezeKey.toString();
|
|
814
|
+
}
|
|
815
|
+
if (tx._wipeKey) {
|
|
816
|
+
result.wipeKey = tx._wipeKey.toString();
|
|
817
|
+
}
|
|
818
|
+
if (tx._supplyKey) {
|
|
819
|
+
result.supplyKey = tx._supplyKey.toString();
|
|
820
|
+
}
|
|
821
|
+
if (tx._feeScheduleKey) {
|
|
822
|
+
result.feeScheduleKey = tx._feeScheduleKey.toString();
|
|
823
|
+
}
|
|
824
|
+
if (tx._pauseKey) {
|
|
825
|
+
result.pauseKey = tx._pauseKey.toString();
|
|
826
|
+
}
|
|
827
|
+
if (tx._metadataKey) {
|
|
828
|
+
result.metadataKey = tx._metadataKey.toString();
|
|
829
|
+
}
|
|
830
|
+
if (tx._autoRenewAccountId) {
|
|
831
|
+
result.autoRenewAccount = tx._autoRenewAccountId.toString();
|
|
832
|
+
}
|
|
833
|
+
if (tx._autoRenewPeriod) {
|
|
834
|
+
result.autoRenewPeriod = tx._autoRenewPeriod.seconds?.toString() || tx._autoRenewPeriod.toString();
|
|
835
|
+
}
|
|
836
|
+
if (tx._expirationTime) {
|
|
837
|
+
result.expiry = tx._expirationTime.seconds?.toString() || tx._expirationTime.toString();
|
|
838
|
+
}
|
|
839
|
+
if (tx._customFees && Array.isArray(tx._customFees) && tx._customFees.length > 0) {
|
|
840
|
+
result.customFees = tx._customFees.map((fee) => {
|
|
841
|
+
const customFee = {
|
|
842
|
+
feeCollectorAccountId: fee.feeCollectorAccountId?.toString() || "",
|
|
843
|
+
feeType: "FIXED_FEE"
|
|
844
|
+
};
|
|
845
|
+
if (fee.fixedFee) {
|
|
846
|
+
customFee.feeType = "FIXED_FEE";
|
|
847
|
+
customFee.fixedFee = {
|
|
848
|
+
amount: fee.fixedFee.amount?.toString() || "0",
|
|
849
|
+
denominatingTokenId: fee.fixedFee.denominatingTokenId?.toString()
|
|
850
|
+
};
|
|
851
|
+
} else if (fee.fractionalFee) {
|
|
852
|
+
customFee.feeType = "FRACTIONAL_FEE";
|
|
853
|
+
customFee.fractionalFee = {
|
|
854
|
+
numerator: fee.fractionalFee.numerator?.toString() || "0",
|
|
855
|
+
denominator: fee.fractionalFee.denominator?.toString() || "1",
|
|
856
|
+
minimumAmount: fee.fractionalFee.minimumAmount?.toString() || "0",
|
|
857
|
+
maximumAmount: fee.fractionalFee.maximumAmount?.toString() || "0",
|
|
858
|
+
netOfTransfers: fee.fractionalFee.netOfTransfers || false
|
|
859
|
+
};
|
|
860
|
+
} else if (fee.royaltyFee) {
|
|
861
|
+
customFee.feeType = "ROYALTY_FEE";
|
|
862
|
+
customFee.royaltyFee = {
|
|
863
|
+
numerator: fee.royaltyFee.numerator?.toString() || "0",
|
|
864
|
+
denominator: fee.royaltyFee.denominator?.toString() || "1",
|
|
865
|
+
fallbackFee: fee.royaltyFee.fallbackFee ? {
|
|
866
|
+
amount: fee.royaltyFee.fallbackFee.amount?.toString() || "0",
|
|
867
|
+
denominatingTokenId: fee.royaltyFee.fallbackFee.denominatingTokenId?.toString()
|
|
868
|
+
} : void 0
|
|
869
|
+
};
|
|
843
870
|
}
|
|
844
|
-
|
|
871
|
+
customFee.allCollectorsAreExempt = fee.allCollectorsAreExempt || false;
|
|
872
|
+
return customFee;
|
|
845
873
|
});
|
|
846
|
-
allNfts = allNfts.concat(nftsWithUri);
|
|
847
874
|
}
|
|
848
|
-
|
|
849
|
-
if (!endpoint) break;
|
|
875
|
+
return result;
|
|
850
876
|
}
|
|
851
|
-
return
|
|
877
|
+
return null;
|
|
852
878
|
} catch (error) {
|
|
853
|
-
this.logger.error(
|
|
854
|
-
`Error fetching NFTs for account ${accountId}: ${error.message}`
|
|
855
|
-
);
|
|
856
879
|
return null;
|
|
857
880
|
}
|
|
858
881
|
}
|
|
859
882
|
/**
|
|
860
|
-
*
|
|
861
|
-
*
|
|
862
|
-
* @param tokenId The ID of the NFT's token.
|
|
863
|
-
* @param serialNumber The serial number of the NFT.
|
|
864
|
-
* @returns A promise that resolves to the NftDetail if owned, or null otherwise.
|
|
883
|
+
* Extract token airdrop data from Transaction object internal fields
|
|
884
|
+
* This handles the case where airdrop data is stored in Transaction internals
|
|
865
885
|
*/
|
|
866
|
-
|
|
867
|
-
this.logger.info(
|
|
868
|
-
`Validating ownership of NFT ${tokenId} SN ${serialNumber} for account ${accountId}`
|
|
869
|
-
);
|
|
886
|
+
static extractTokenAirdropFromTransaction(transaction) {
|
|
870
887
|
try {
|
|
871
|
-
const
|
|
872
|
-
if (
|
|
873
|
-
const
|
|
874
|
-
|
|
875
|
-
|
|
876
|
-
|
|
888
|
+
const tx = transaction;
|
|
889
|
+
if (tx._tokenAirdrops && Array.isArray(tx._tokenAirdrops)) {
|
|
890
|
+
const tokenTransfers = tx._tokenAirdrops.map((airdrop) => ({
|
|
891
|
+
tokenId: airdrop.tokenId?.toString() || "Unknown",
|
|
892
|
+
transfers: (airdrop.transfers || []).map((transfer) => ({
|
|
893
|
+
accountId: transfer.accountId?.toString() || "Unknown",
|
|
894
|
+
amount: transfer.amount?.toString() || "0",
|
|
895
|
+
serialNumbers: transfer.serialNumbers?.map((s) => s.toString()) || []
|
|
896
|
+
}))
|
|
897
|
+
}));
|
|
898
|
+
return { tokenTransfers };
|
|
877
899
|
}
|
|
878
900
|
return null;
|
|
879
901
|
} catch (error) {
|
|
880
|
-
this.logger.error(`Error validating NFT ownership: ${error.message}`);
|
|
881
902
|
return null;
|
|
882
903
|
}
|
|
883
904
|
}
|
|
884
905
|
/**
|
|
885
|
-
*
|
|
886
|
-
*
|
|
887
|
-
* @param functionSelector The function selector and encoded parameters (e.g., "0xabcdef12...").
|
|
888
|
-
* @param payerAccountId The account ID of the payer (not strictly payer for read-only, but often required as 'from').
|
|
889
|
-
* @param estimate Whether this is an estimate call. Mirror node might not support this directly in /contracts/call for true estimation.
|
|
890
|
-
* @param block Block parameter, e.g., "latest", "pending", or block number.
|
|
891
|
-
* @param value The value in tinybars to send with the call (for payable view/pure functions, usually 0).
|
|
892
|
-
* @returns A promise that resolves to the contract call query response or null.
|
|
906
|
+
* Parse HTS transaction from Transaction object with comprehensive extraction
|
|
907
|
+
* This is the unified entry point that handles both protobuf and internal field extraction
|
|
893
908
|
*/
|
|
894
|
-
|
|
895
|
-
this.logger.info(
|
|
896
|
-
`Reading smart contract ${contractIdOrAddress} with selector ${functionSelector}`
|
|
897
|
-
);
|
|
898
|
-
const toAddress = contractIdOrAddress.startsWith("0x") ? contractIdOrAddress : `0x${AccountId.fromString(contractIdOrAddress).toSolidityAddress()}`;
|
|
899
|
-
const fromAddress = payerAccountId.startsWith("0x") ? payerAccountId : `0x${AccountId.fromString(payerAccountId).toSolidityAddress()}`;
|
|
900
|
-
const body = {
|
|
901
|
-
block: options?.block || "latest",
|
|
902
|
-
data: functionSelector,
|
|
903
|
-
estimate: options?.estimate || false,
|
|
904
|
-
from: fromAddress,
|
|
905
|
-
to: toAddress,
|
|
906
|
-
gas: options?.gas,
|
|
907
|
-
gasPrice: options?.gasPrice,
|
|
908
|
-
value: options?.value || 0
|
|
909
|
-
};
|
|
910
|
-
Object.keys(body).forEach((key) => {
|
|
911
|
-
const K = key;
|
|
912
|
-
if (body[K] === void 0) {
|
|
913
|
-
delete body[K];
|
|
914
|
-
}
|
|
915
|
-
});
|
|
909
|
+
static parseFromTransactionObject(transaction) {
|
|
916
910
|
try {
|
|
917
|
-
|
|
918
|
-
|
|
919
|
-
|
|
920
|
-
|
|
921
|
-
|
|
922
|
-
|
|
923
|
-
|
|
924
|
-
|
|
911
|
+
try {
|
|
912
|
+
const bytes = transaction.toBytes ? transaction.toBytes() : void 0;
|
|
913
|
+
if (bytes) {
|
|
914
|
+
const decoded = proto.TransactionList.decode(bytes);
|
|
915
|
+
if (decoded.transactionList && decoded.transactionList.length > 0) {
|
|
916
|
+
const tx = decoded.transactionList[0];
|
|
917
|
+
let txBody = null;
|
|
918
|
+
if (tx.bodyBytes && tx.bodyBytes.length > 0) {
|
|
919
|
+
txBody = proto.TransactionBody.decode(tx.bodyBytes);
|
|
920
|
+
} else if (tx.signedTransactionBytes && tx.signedTransactionBytes.length > 0) {
|
|
921
|
+
const signedTx = proto.SignedTransaction.decode(
|
|
922
|
+
tx.signedTransactionBytes
|
|
923
|
+
);
|
|
924
|
+
if (signedTx.bodyBytes) {
|
|
925
|
+
txBody = proto.TransactionBody.decode(signedTx.bodyBytes);
|
|
926
|
+
}
|
|
927
|
+
}
|
|
928
|
+
if (txBody) {
|
|
929
|
+
if (txBody.tokenCreation) {
|
|
930
|
+
const tokenCreation2 = this.parseTokenCreate(
|
|
931
|
+
txBody.tokenCreation
|
|
932
|
+
);
|
|
933
|
+
if (tokenCreation2) {
|
|
934
|
+
return {
|
|
935
|
+
type: "TOKENCREATE",
|
|
936
|
+
humanReadableType: "Token Creation",
|
|
937
|
+
tokenCreation: tokenCreation2
|
|
938
|
+
};
|
|
939
|
+
}
|
|
940
|
+
}
|
|
941
|
+
if (txBody.tokenMint) {
|
|
942
|
+
const tokenMint = this.parseTokenMint(txBody.tokenMint);
|
|
943
|
+
if (tokenMint) {
|
|
944
|
+
return {
|
|
945
|
+
type: "TOKENMINT",
|
|
946
|
+
humanReadableType: "Token Mint",
|
|
947
|
+
tokenMint
|
|
948
|
+
};
|
|
949
|
+
}
|
|
950
|
+
}
|
|
951
|
+
if (txBody.tokenBurn) {
|
|
952
|
+
const tokenBurn = this.parseTokenBurn(txBody.tokenBurn);
|
|
953
|
+
if (tokenBurn) {
|
|
954
|
+
return {
|
|
955
|
+
type: "TOKENBURN",
|
|
956
|
+
humanReadableType: "Token Burn",
|
|
957
|
+
tokenBurn
|
|
958
|
+
};
|
|
959
|
+
}
|
|
960
|
+
}
|
|
961
|
+
if (txBody.tokenUpdate) {
|
|
962
|
+
const tokenUpdate = this.parseTokenUpdate(txBody.tokenUpdate);
|
|
963
|
+
if (tokenUpdate) {
|
|
964
|
+
return {
|
|
965
|
+
type: "TOKENUPDATE",
|
|
966
|
+
humanReadableType: "Token Update",
|
|
967
|
+
tokenUpdate
|
|
968
|
+
};
|
|
969
|
+
}
|
|
970
|
+
}
|
|
971
|
+
if (txBody.tokenFreeze) {
|
|
972
|
+
const tokenFreeze = this.parseTokenFreeze(txBody.tokenFreeze);
|
|
973
|
+
if (tokenFreeze) {
|
|
974
|
+
return {
|
|
975
|
+
type: "TOKENFREEZE",
|
|
976
|
+
humanReadableType: "Token Freeze",
|
|
977
|
+
tokenFreeze
|
|
978
|
+
};
|
|
979
|
+
}
|
|
980
|
+
}
|
|
981
|
+
if (txBody.tokenUnfreeze) {
|
|
982
|
+
const tokenUnfreeze = this.parseTokenUnfreeze(
|
|
983
|
+
txBody.tokenUnfreeze
|
|
984
|
+
);
|
|
985
|
+
if (tokenUnfreeze) {
|
|
986
|
+
return {
|
|
987
|
+
type: "TOKENUNFREEZE",
|
|
988
|
+
humanReadableType: "Token Unfreeze",
|
|
989
|
+
tokenUnfreeze
|
|
990
|
+
};
|
|
991
|
+
}
|
|
992
|
+
}
|
|
993
|
+
if (txBody.tokenGrantKyc) {
|
|
994
|
+
const tokenGrantKyc = this.parseTokenGrantKyc(
|
|
995
|
+
txBody.tokenGrantKyc
|
|
996
|
+
);
|
|
997
|
+
if (tokenGrantKyc) {
|
|
998
|
+
return {
|
|
999
|
+
type: "TOKENGRANTKYC",
|
|
1000
|
+
humanReadableType: "Token Grant KYC",
|
|
1001
|
+
tokenGrantKyc
|
|
1002
|
+
};
|
|
1003
|
+
}
|
|
1004
|
+
}
|
|
1005
|
+
if (txBody.tokenRevokeKyc) {
|
|
1006
|
+
const tokenRevokeKyc = this.parseTokenRevokeKyc(
|
|
1007
|
+
txBody.tokenRevokeKyc
|
|
1008
|
+
);
|
|
1009
|
+
if (tokenRevokeKyc) {
|
|
1010
|
+
return {
|
|
1011
|
+
type: "TOKENREVOKEKYC",
|
|
1012
|
+
humanReadableType: "Token Revoke KYC",
|
|
1013
|
+
tokenRevokeKyc
|
|
1014
|
+
};
|
|
1015
|
+
}
|
|
1016
|
+
}
|
|
1017
|
+
if (txBody.tokenPause) {
|
|
1018
|
+
const tokenPause = this.parseTokenPause(txBody.tokenPause);
|
|
1019
|
+
if (tokenPause) {
|
|
1020
|
+
return {
|
|
1021
|
+
type: "TOKENPAUSE",
|
|
1022
|
+
humanReadableType: "Token Pause",
|
|
1023
|
+
tokenPause
|
|
1024
|
+
};
|
|
1025
|
+
}
|
|
1026
|
+
}
|
|
1027
|
+
if (txBody.tokenUnpause) {
|
|
1028
|
+
const tokenUnpause = this.parseTokenUnpause(
|
|
1029
|
+
txBody.tokenUnpause
|
|
1030
|
+
);
|
|
1031
|
+
if (tokenUnpause) {
|
|
1032
|
+
return {
|
|
1033
|
+
type: "TOKENUNPAUSE",
|
|
1034
|
+
humanReadableType: "Token Unpause",
|
|
1035
|
+
tokenUnpause
|
|
1036
|
+
};
|
|
1037
|
+
}
|
|
1038
|
+
}
|
|
1039
|
+
if (txBody.tokenWipe) {
|
|
1040
|
+
const tokenWipeAccount = this.parseTokenWipeAccount(
|
|
1041
|
+
txBody.tokenWipe
|
|
1042
|
+
);
|
|
1043
|
+
if (tokenWipeAccount) {
|
|
1044
|
+
return {
|
|
1045
|
+
type: "TOKENWIPEACCOUNT",
|
|
1046
|
+
humanReadableType: "Token Wipe Account",
|
|
1047
|
+
tokenWipeAccount
|
|
1048
|
+
};
|
|
1049
|
+
}
|
|
1050
|
+
}
|
|
1051
|
+
if (txBody.tokenDeletion) {
|
|
1052
|
+
const tokenDelete = this.parseTokenDelete(txBody.tokenDeletion);
|
|
1053
|
+
if (tokenDelete) {
|
|
1054
|
+
return {
|
|
1055
|
+
type: "TOKENDELETE",
|
|
1056
|
+
humanReadableType: "Token Delete",
|
|
1057
|
+
tokenDelete
|
|
1058
|
+
};
|
|
1059
|
+
}
|
|
1060
|
+
}
|
|
1061
|
+
if (txBody.tokenAssociate) {
|
|
1062
|
+
const tokenAssociate = this.parseTokenAssociate(
|
|
1063
|
+
txBody.tokenAssociate
|
|
1064
|
+
);
|
|
1065
|
+
if (tokenAssociate) {
|
|
1066
|
+
return {
|
|
1067
|
+
type: "TOKENASSOCIATE",
|
|
1068
|
+
humanReadableType: "Token Associate",
|
|
1069
|
+
tokenAssociate
|
|
1070
|
+
};
|
|
1071
|
+
}
|
|
1072
|
+
}
|
|
1073
|
+
if (txBody.tokenDissociate) {
|
|
1074
|
+
const tokenDissociate = this.parseTokenDissociate(
|
|
1075
|
+
txBody.tokenDissociate
|
|
1076
|
+
);
|
|
1077
|
+
if (tokenDissociate) {
|
|
1078
|
+
return {
|
|
1079
|
+
type: "TOKENDISSOCIATE",
|
|
1080
|
+
humanReadableType: "Token Dissociate",
|
|
1081
|
+
tokenDissociate
|
|
1082
|
+
};
|
|
1083
|
+
}
|
|
1084
|
+
}
|
|
1085
|
+
if (txBody.tokenFeeScheduleUpdate) {
|
|
1086
|
+
const tokenFeeScheduleUpdate = this.parseTokenFeeScheduleUpdate(
|
|
1087
|
+
txBody.tokenFeeScheduleUpdate
|
|
1088
|
+
);
|
|
1089
|
+
if (tokenFeeScheduleUpdate) {
|
|
1090
|
+
return {
|
|
1091
|
+
type: "TOKENFEESCHEDULEUPDATE",
|
|
1092
|
+
humanReadableType: "Token Fee Schedule Update",
|
|
1093
|
+
tokenFeeScheduleUpdate
|
|
1094
|
+
};
|
|
1095
|
+
}
|
|
1096
|
+
}
|
|
1097
|
+
if (txBody.tokenAirdrop) {
|
|
1098
|
+
const tokenAirdrop2 = this.parseTokenAirdropFromProto(
|
|
1099
|
+
txBody.tokenAirdrop
|
|
1100
|
+
);
|
|
1101
|
+
if (tokenAirdrop2) {
|
|
1102
|
+
return {
|
|
1103
|
+
type: "TOKENAIRDROP",
|
|
1104
|
+
humanReadableType: "Token Airdrop",
|
|
1105
|
+
tokenAirdrop: tokenAirdrop2
|
|
1106
|
+
};
|
|
1107
|
+
}
|
|
1108
|
+
}
|
|
1109
|
+
}
|
|
925
1110
|
}
|
|
926
1111
|
}
|
|
927
|
-
)
|
|
928
|
-
|
|
929
|
-
|
|
930
|
-
|
|
931
|
-
|
|
932
|
-
|
|
933
|
-
|
|
934
|
-
|
|
935
|
-
|
|
936
|
-
|
|
937
|
-
|
|
938
|
-
|
|
939
|
-
|
|
940
|
-
|
|
941
|
-
|
|
942
|
-
|
|
943
|
-
|
|
944
|
-
|
|
945
|
-
|
|
946
|
-
|
|
947
|
-
|
|
948
|
-
|
|
949
|
-
|
|
950
|
-
}
|
|
951
|
-
if (options?.order) {
|
|
952
|
-
params.append("order", options.order);
|
|
953
|
-
}
|
|
954
|
-
if (options?.receiverId) {
|
|
955
|
-
params.append("receiver.id", options.receiverId);
|
|
956
|
-
}
|
|
957
|
-
if (options?.serialNumber) {
|
|
958
|
-
params.append("serialnumber", options.serialNumber);
|
|
959
|
-
}
|
|
960
|
-
if (options?.tokenId) {
|
|
961
|
-
params.append("token.id", options.tokenId);
|
|
962
|
-
}
|
|
963
|
-
const queryString = params.toString();
|
|
964
|
-
if (queryString) {
|
|
965
|
-
endpoint += `?${queryString}`;
|
|
966
|
-
}
|
|
967
|
-
try {
|
|
968
|
-
const response = await this._requestWithRetry(endpoint);
|
|
969
|
-
return response.airdrops || [];
|
|
970
|
-
} catch (error) {
|
|
971
|
-
this.logger.error(
|
|
972
|
-
`Error fetching outstanding token airdrops for account ${accountId}: ${error.message}`
|
|
973
|
-
);
|
|
974
|
-
return null;
|
|
975
|
-
}
|
|
976
|
-
}
|
|
977
|
-
/**
|
|
978
|
-
* Retrieves pending token airdrops received by an account.
|
|
979
|
-
* @param accountId The ID of the account that received the airdrops.
|
|
980
|
-
* @param options Optional parameters for filtering airdrops.
|
|
981
|
-
* @returns A promise that resolves to an array of TokenAirdrop or null.
|
|
982
|
-
*/
|
|
983
|
-
async getPendingTokenAirdrops(accountId, options) {
|
|
984
|
-
this.logger.info(
|
|
985
|
-
`Getting pending token airdrops received by account ${accountId}`
|
|
986
|
-
);
|
|
987
|
-
let endpoint = `/api/v1/accounts/${accountId}/airdrops/pending`;
|
|
988
|
-
const params = new URLSearchParams();
|
|
989
|
-
if (options?.limit) {
|
|
990
|
-
params.append("limit", options.limit.toString());
|
|
991
|
-
}
|
|
992
|
-
if (options?.order) {
|
|
993
|
-
params.append("order", options.order);
|
|
994
|
-
}
|
|
995
|
-
if (options?.senderId) {
|
|
996
|
-
params.append("sender.id", options.senderId);
|
|
997
|
-
}
|
|
998
|
-
if (options?.serialNumber) {
|
|
999
|
-
params.append("serialnumber", options.serialNumber);
|
|
1000
|
-
}
|
|
1001
|
-
if (options?.tokenId) {
|
|
1002
|
-
params.append("token.id", options.tokenId);
|
|
1003
|
-
}
|
|
1004
|
-
const queryString = params.toString();
|
|
1005
|
-
if (queryString) {
|
|
1006
|
-
endpoint += `?${queryString}`;
|
|
1007
|
-
}
|
|
1008
|
-
try {
|
|
1009
|
-
const response = await this._requestWithRetry(endpoint);
|
|
1010
|
-
return response.airdrops || [];
|
|
1011
|
-
} catch (error) {
|
|
1012
|
-
this.logger.error(
|
|
1013
|
-
`Error fetching pending token airdrops for account ${accountId}: ${error.message}`
|
|
1014
|
-
);
|
|
1015
|
-
return null;
|
|
1016
|
-
}
|
|
1017
|
-
}
|
|
1018
|
-
/**
|
|
1019
|
-
* Retrieves blocks from the network.
|
|
1020
|
-
* @param options Optional parameters for filtering blocks.
|
|
1021
|
-
* @returns A promise that resolves to an array of Block or null.
|
|
1022
|
-
*/
|
|
1023
|
-
async getBlocks(options) {
|
|
1024
|
-
this.logger.info("Getting blocks from the network");
|
|
1025
|
-
let endpoint = `/api/v1/blocks`;
|
|
1026
|
-
const params = new URLSearchParams();
|
|
1027
|
-
if (options?.limit) {
|
|
1028
|
-
params.append("limit", options.limit.toString());
|
|
1029
|
-
}
|
|
1030
|
-
if (options?.order) {
|
|
1031
|
-
params.append("order", options.order);
|
|
1032
|
-
}
|
|
1033
|
-
if (options?.timestamp) {
|
|
1034
|
-
params.append("timestamp", options.timestamp);
|
|
1035
|
-
}
|
|
1036
|
-
if (options?.blockNumber) {
|
|
1037
|
-
params.append("block.number", options.blockNumber);
|
|
1038
|
-
}
|
|
1039
|
-
const queryString = params.toString();
|
|
1040
|
-
if (queryString) {
|
|
1041
|
-
endpoint += `?${queryString}`;
|
|
1042
|
-
}
|
|
1043
|
-
try {
|
|
1044
|
-
const response = await this._requestWithRetry(endpoint);
|
|
1045
|
-
return response.blocks || [];
|
|
1046
|
-
} catch (error) {
|
|
1047
|
-
this.logger.error(`Error fetching blocks: ${error.message}`);
|
|
1048
|
-
return null;
|
|
1049
|
-
}
|
|
1050
|
-
}
|
|
1051
|
-
/**
|
|
1052
|
-
* Retrieves a specific block by number or hash.
|
|
1053
|
-
* @param blockNumberOrHash The block number or hash.
|
|
1054
|
-
* @returns A promise that resolves to a Block or null.
|
|
1055
|
-
*/
|
|
1056
|
-
async getBlock(blockNumberOrHash) {
|
|
1057
|
-
this.logger.info(`Getting block ${blockNumberOrHash}`);
|
|
1058
|
-
try {
|
|
1059
|
-
const response = await this._requestWithRetry(
|
|
1060
|
-
`/api/v1/blocks/${blockNumberOrHash}`
|
|
1061
|
-
);
|
|
1062
|
-
return response;
|
|
1063
|
-
} catch (error) {
|
|
1064
|
-
this.logger.error(
|
|
1065
|
-
`Error fetching block ${blockNumberOrHash}: ${error.message}`
|
|
1066
|
-
);
|
|
1067
|
-
return null;
|
|
1068
|
-
}
|
|
1069
|
-
}
|
|
1070
|
-
/**
|
|
1071
|
-
* Retrieves contract entities from the network.
|
|
1072
|
-
* @param options Optional parameters for filtering contracts.
|
|
1073
|
-
* @returns A promise that resolves to an array of ContractEntity or null.
|
|
1074
|
-
*/
|
|
1075
|
-
async getContracts(options) {
|
|
1076
|
-
this.logger.info("Getting contracts from the network");
|
|
1077
|
-
let url = `/api/v1/contracts`;
|
|
1078
|
-
const params = new URLSearchParams();
|
|
1079
|
-
if (options?.contractId) {
|
|
1080
|
-
params.append("contract.id", options.contractId);
|
|
1081
|
-
}
|
|
1082
|
-
if (options?.limit) {
|
|
1083
|
-
params.append("limit", options.limit.toString());
|
|
1084
|
-
}
|
|
1085
|
-
if (options?.order) {
|
|
1086
|
-
params.append("order", options.order);
|
|
1087
|
-
}
|
|
1088
|
-
const queryString = params.toString();
|
|
1089
|
-
if (queryString) {
|
|
1090
|
-
url += `?${queryString}`;
|
|
1091
|
-
}
|
|
1092
|
-
try {
|
|
1093
|
-
const response = await this._requestWithRetry(url);
|
|
1094
|
-
return response.contracts || [];
|
|
1095
|
-
} catch (error) {
|
|
1096
|
-
this.logger.error(`Error fetching contracts: ${error.message}`);
|
|
1097
|
-
return null;
|
|
1098
|
-
}
|
|
1099
|
-
}
|
|
1100
|
-
/**
|
|
1101
|
-
* Retrieves a specific contract by ID or address.
|
|
1102
|
-
* @param contractIdOrAddress The contract ID or EVM address.
|
|
1103
|
-
* @param timestamp Optional timestamp for historical data.
|
|
1104
|
-
* @returns A promise that resolves to a ContractEntity or null.
|
|
1105
|
-
*/
|
|
1106
|
-
async getContract(contractIdOrAddress, timestamp) {
|
|
1107
|
-
this.logger.info(`Getting contract ${contractIdOrAddress}`);
|
|
1108
|
-
let url = `/api/v1/contracts/${contractIdOrAddress}`;
|
|
1109
|
-
if (timestamp) {
|
|
1110
|
-
url += `?timestamp=${timestamp}`;
|
|
1111
|
-
}
|
|
1112
|
-
try {
|
|
1113
|
-
const response = await this._requestWithRetry(url);
|
|
1114
|
-
return response;
|
|
1115
|
-
} catch (error) {
|
|
1116
|
-
this.logger.error(
|
|
1117
|
-
`Error fetching contract ${contractIdOrAddress}: ${error.message}`
|
|
1118
|
-
);
|
|
1119
|
-
return null;
|
|
1120
|
-
}
|
|
1121
|
-
}
|
|
1122
|
-
/**
|
|
1123
|
-
* Retrieves contract results from the network.
|
|
1124
|
-
* @param options Optional parameters for filtering contract results.
|
|
1125
|
-
* @returns A promise that resolves to an array of ContractResult or null.
|
|
1126
|
-
*/
|
|
1127
|
-
async getContractResults(options) {
|
|
1128
|
-
this.logger.info("Getting contract results from the network");
|
|
1129
|
-
let url = `/api/v1/contracts/results`;
|
|
1130
|
-
const params = new URLSearchParams();
|
|
1131
|
-
if (options?.from) {
|
|
1132
|
-
params.append("from", options.from);
|
|
1133
|
-
}
|
|
1134
|
-
if (options?.blockHash) {
|
|
1135
|
-
params.append("block.hash", options.blockHash);
|
|
1136
|
-
}
|
|
1137
|
-
if (options?.blockNumber) {
|
|
1138
|
-
params.append("block.number", options.blockNumber);
|
|
1139
|
-
}
|
|
1140
|
-
if (options?.internal !== void 0) {
|
|
1141
|
-
params.append("internal", options.internal.toString());
|
|
1142
|
-
}
|
|
1143
|
-
if (options?.limit) {
|
|
1144
|
-
params.append("limit", options.limit.toString());
|
|
1145
|
-
}
|
|
1146
|
-
if (options?.order) {
|
|
1147
|
-
params.append("order", options.order);
|
|
1148
|
-
}
|
|
1149
|
-
if (options?.timestamp) {
|
|
1150
|
-
params.append("timestamp", options.timestamp);
|
|
1151
|
-
}
|
|
1152
|
-
if (options?.transactionIndex) {
|
|
1153
|
-
params.append("transaction.index", options.transactionIndex.toString());
|
|
1154
|
-
}
|
|
1155
|
-
const queryString = params.toString();
|
|
1156
|
-
if (queryString) {
|
|
1157
|
-
url += `?${queryString}`;
|
|
1158
|
-
}
|
|
1159
|
-
try {
|
|
1160
|
-
const response = await this._requestWithRetry(url);
|
|
1161
|
-
return response.results || [];
|
|
1162
|
-
} catch (error) {
|
|
1163
|
-
this.logger.error(`Error fetching contract results: ${error.message}`);
|
|
1164
|
-
return null;
|
|
1165
|
-
}
|
|
1166
|
-
}
|
|
1167
|
-
/**
|
|
1168
|
-
* Retrieves a specific contract result by transaction ID or hash.
|
|
1169
|
-
* @param transactionIdOrHash The transaction ID or hash.
|
|
1170
|
-
* @param nonce Optional nonce filter.
|
|
1171
|
-
* @returns A promise that resolves to a ContractResult or null.
|
|
1172
|
-
*/
|
|
1173
|
-
async getContractResult(transactionIdOrHash, nonce) {
|
|
1174
|
-
this.logger.info(`Getting contract result for ${transactionIdOrHash}`);
|
|
1175
|
-
let url = `/api/v1/contracts/results/${transactionIdOrHash}`;
|
|
1176
|
-
if (nonce !== void 0) {
|
|
1177
|
-
url += `?nonce=${nonce}`;
|
|
1178
|
-
}
|
|
1179
|
-
try {
|
|
1180
|
-
const response = await this._requestWithRetry(url);
|
|
1181
|
-
return response;
|
|
1182
|
-
} catch (error) {
|
|
1183
|
-
this.logger.error(
|
|
1184
|
-
`Error fetching contract result for ${transactionIdOrHash}: ${error.message}`
|
|
1185
|
-
);
|
|
1186
|
-
return null;
|
|
1187
|
-
}
|
|
1188
|
-
}
|
|
1189
|
-
/**
|
|
1190
|
-
* Retrieves contract results for a specific contract.
|
|
1191
|
-
* @param contractIdOrAddress The contract ID or EVM address.
|
|
1192
|
-
* @param options Optional parameters for filtering.
|
|
1193
|
-
* @returns A promise that resolves to an array of ContractResult or null.
|
|
1194
|
-
*/
|
|
1195
|
-
async getContractResultsByContract(contractIdOrAddress, options) {
|
|
1196
|
-
this.logger.info(
|
|
1197
|
-
`Getting contract results for contract ${contractIdOrAddress}`
|
|
1198
|
-
);
|
|
1199
|
-
let url = `/api/v1/contracts/${contractIdOrAddress}/results`;
|
|
1200
|
-
const params = new URLSearchParams();
|
|
1201
|
-
if (options?.blockHash) {
|
|
1202
|
-
params.append("block.hash", options.blockHash);
|
|
1203
|
-
}
|
|
1204
|
-
if (options?.blockNumber) {
|
|
1205
|
-
params.append("block.number", options.blockNumber);
|
|
1206
|
-
}
|
|
1207
|
-
if (options?.from) {
|
|
1208
|
-
params.append("from", options.from);
|
|
1209
|
-
}
|
|
1210
|
-
if (options?.internal !== void 0) {
|
|
1211
|
-
params.append("internal", options.internal.toString());
|
|
1212
|
-
}
|
|
1213
|
-
if (options?.limit) {
|
|
1214
|
-
params.append("limit", options.limit.toString());
|
|
1215
|
-
}
|
|
1216
|
-
if (options?.order) {
|
|
1217
|
-
params.append("order", options.order);
|
|
1218
|
-
}
|
|
1219
|
-
if (options?.timestamp) {
|
|
1220
|
-
params.append("timestamp", options.timestamp);
|
|
1221
|
-
}
|
|
1222
|
-
if (options?.transactionIndex) {
|
|
1223
|
-
params.append("transaction.index", options.transactionIndex.toString());
|
|
1224
|
-
}
|
|
1225
|
-
const queryString = params.toString();
|
|
1226
|
-
if (queryString) {
|
|
1227
|
-
url += `?${queryString}`;
|
|
1228
|
-
}
|
|
1229
|
-
try {
|
|
1230
|
-
const response = await this._requestWithRetry(url);
|
|
1231
|
-
return response.results || [];
|
|
1232
|
-
} catch (error) {
|
|
1233
|
-
this.logger.error(
|
|
1234
|
-
`Error fetching contract results for ${contractIdOrAddress}: ${error.message}`
|
|
1235
|
-
);
|
|
1236
|
-
return null;
|
|
1237
|
-
}
|
|
1238
|
-
}
|
|
1239
|
-
/**
|
|
1240
|
-
* Retrieves contract state for a specific contract.
|
|
1241
|
-
* @param contractIdOrAddress The contract ID or EVM address.
|
|
1242
|
-
* @param options Optional parameters for filtering.
|
|
1243
|
-
* @returns A promise that resolves to an array of ContractState or null.
|
|
1244
|
-
*/
|
|
1245
|
-
async getContractState(contractIdOrAddress, options) {
|
|
1246
|
-
this.logger.info(`Getting contract state for ${contractIdOrAddress}`);
|
|
1247
|
-
let url = `/api/v1/contracts/${contractIdOrAddress}/state`;
|
|
1248
|
-
const params = new URLSearchParams();
|
|
1249
|
-
if (options?.limit) {
|
|
1250
|
-
params.append("limit", options.limit.toString());
|
|
1251
|
-
}
|
|
1252
|
-
if (options?.order) {
|
|
1253
|
-
params.append("order", options.order);
|
|
1254
|
-
}
|
|
1255
|
-
if (options?.slot) {
|
|
1256
|
-
params.append("slot", options.slot);
|
|
1257
|
-
}
|
|
1258
|
-
if (options?.timestamp) {
|
|
1259
|
-
params.append("timestamp", options.timestamp);
|
|
1260
|
-
}
|
|
1261
|
-
const queryString = params.toString();
|
|
1262
|
-
if (queryString) {
|
|
1263
|
-
url += `?${queryString}`;
|
|
1264
|
-
}
|
|
1265
|
-
try {
|
|
1266
|
-
const response = await this._requestWithRetry(url);
|
|
1267
|
-
return response.state || [];
|
|
1268
|
-
} catch (error) {
|
|
1269
|
-
this.logger.error(
|
|
1270
|
-
`Error fetching contract state for ${contractIdOrAddress}: ${error.message}`
|
|
1271
|
-
);
|
|
1272
|
-
return null;
|
|
1273
|
-
}
|
|
1274
|
-
}
|
|
1275
|
-
/**
|
|
1276
|
-
* Retrieves contract actions for a specific transaction.
|
|
1277
|
-
* @param transactionIdOrHash The transaction ID or hash.
|
|
1278
|
-
* @param options Optional parameters for filtering.
|
|
1279
|
-
* @returns A promise that resolves to an array of ContractAction or null.
|
|
1280
|
-
*/
|
|
1281
|
-
async getContractActions(transactionIdOrHash, options) {
|
|
1282
|
-
this.logger.info(`Getting contract actions for ${transactionIdOrHash}`);
|
|
1283
|
-
let url = `/api/v1/contracts/results/${transactionIdOrHash}/actions`;
|
|
1284
|
-
const params = new URLSearchParams();
|
|
1285
|
-
if (options?.index) {
|
|
1286
|
-
params.append("index", options.index);
|
|
1287
|
-
}
|
|
1288
|
-
if (options?.limit) {
|
|
1289
|
-
params.append("limit", options.limit.toString());
|
|
1290
|
-
}
|
|
1291
|
-
if (options?.order) {
|
|
1292
|
-
params.append("order", options.order);
|
|
1293
|
-
}
|
|
1294
|
-
const queryString = params.toString();
|
|
1295
|
-
if (queryString) {
|
|
1296
|
-
url += `?${queryString}`;
|
|
1297
|
-
}
|
|
1298
|
-
try {
|
|
1299
|
-
const response = await this._requestWithRetry(url);
|
|
1300
|
-
return response.actions || [];
|
|
1301
|
-
} catch (error) {
|
|
1302
|
-
this.logger.error(
|
|
1303
|
-
`Error fetching contract actions for ${transactionIdOrHash}: ${error.message}`
|
|
1304
|
-
);
|
|
1305
|
-
return null;
|
|
1306
|
-
}
|
|
1307
|
-
}
|
|
1308
|
-
/**
|
|
1309
|
-
* Retrieves contract logs from the network.
|
|
1310
|
-
* @param options Optional parameters for filtering logs.
|
|
1311
|
-
* @returns A promise that resolves to an array of ContractLog or null.
|
|
1312
|
-
*/
|
|
1313
|
-
async getContractLogs(options) {
|
|
1314
|
-
this.logger.info("Getting contract logs from the network");
|
|
1315
|
-
let url = `/api/v1/contracts/results/logs`;
|
|
1316
|
-
const params = new URLSearchParams();
|
|
1317
|
-
if (options?.index) {
|
|
1318
|
-
params.append("index", options.index);
|
|
1319
|
-
}
|
|
1320
|
-
if (options?.limit) {
|
|
1321
|
-
params.append("limit", options.limit.toString());
|
|
1322
|
-
}
|
|
1323
|
-
if (options?.order) {
|
|
1324
|
-
params.append("order", options.order);
|
|
1325
|
-
}
|
|
1326
|
-
if (options?.timestamp) {
|
|
1327
|
-
params.append("timestamp", options.timestamp);
|
|
1328
|
-
}
|
|
1329
|
-
if (options?.topic0) {
|
|
1330
|
-
params.append("topic0", options.topic0);
|
|
1331
|
-
}
|
|
1332
|
-
if (options?.topic1) {
|
|
1333
|
-
params.append("topic1", options.topic1);
|
|
1334
|
-
}
|
|
1335
|
-
if (options?.topic2) {
|
|
1336
|
-
params.append("topic2", options.topic2);
|
|
1337
|
-
}
|
|
1338
|
-
if (options?.topic3) {
|
|
1339
|
-
params.append("topic3", options.topic3);
|
|
1340
|
-
}
|
|
1341
|
-
if (options?.transactionHash) {
|
|
1342
|
-
params.append("transaction.hash", options.transactionHash);
|
|
1343
|
-
}
|
|
1344
|
-
const queryString = params.toString();
|
|
1345
|
-
if (queryString) {
|
|
1346
|
-
url += `?${queryString}`;
|
|
1347
|
-
}
|
|
1348
|
-
try {
|
|
1349
|
-
const response = await this._requestWithRetry(url);
|
|
1350
|
-
return response.logs || [];
|
|
1351
|
-
} catch (error) {
|
|
1352
|
-
this.logger.error(`Error fetching contract logs: ${error.message}`);
|
|
1353
|
-
return null;
|
|
1354
|
-
}
|
|
1355
|
-
}
|
|
1356
|
-
/**
|
|
1357
|
-
* Retrieves contract logs for a specific contract.
|
|
1358
|
-
* @param contractIdOrAddress The contract ID or EVM address.
|
|
1359
|
-
* @param options Optional parameters for filtering logs.
|
|
1360
|
-
* @returns A promise that resolves to an array of ContractLog or null.
|
|
1361
|
-
*/
|
|
1362
|
-
async getContractLogsByContract(contractIdOrAddress, options) {
|
|
1363
|
-
this.logger.info(
|
|
1364
|
-
`Getting contract logs for contract ${contractIdOrAddress}`
|
|
1365
|
-
);
|
|
1366
|
-
let url = `/api/v1/contracts/${contractIdOrAddress}/results/logs`;
|
|
1367
|
-
const params = new URLSearchParams();
|
|
1368
|
-
if (options?.index) {
|
|
1369
|
-
params.append("index", options.index);
|
|
1370
|
-
}
|
|
1371
|
-
if (options?.limit) {
|
|
1372
|
-
params.append("limit", options.limit.toString());
|
|
1373
|
-
}
|
|
1374
|
-
if (options?.order) {
|
|
1375
|
-
params.append("order", options.order);
|
|
1376
|
-
}
|
|
1377
|
-
if (options?.timestamp) {
|
|
1378
|
-
params.append("timestamp", options.timestamp);
|
|
1379
|
-
}
|
|
1380
|
-
if (options?.topic0) {
|
|
1381
|
-
params.append("topic0", options.topic0);
|
|
1382
|
-
}
|
|
1383
|
-
if (options?.topic1) {
|
|
1384
|
-
params.append("topic1", options.topic1);
|
|
1385
|
-
}
|
|
1386
|
-
if (options?.topic2) {
|
|
1387
|
-
params.append("topic2", options.topic2);
|
|
1388
|
-
}
|
|
1389
|
-
if (options?.topic3) {
|
|
1390
|
-
params.append("topic3", options.topic3);
|
|
1391
|
-
}
|
|
1392
|
-
const queryString = params.toString();
|
|
1393
|
-
if (queryString) {
|
|
1394
|
-
url += `?${queryString}`;
|
|
1395
|
-
}
|
|
1396
|
-
try {
|
|
1397
|
-
const response = await this._requestWithRetry(url);
|
|
1398
|
-
return response.logs || [];
|
|
1399
|
-
} catch (error) {
|
|
1400
|
-
this.logger.error(
|
|
1401
|
-
`Error fetching contract logs for ${contractIdOrAddress}: ${error.message}`
|
|
1402
|
-
);
|
|
1403
|
-
return null;
|
|
1404
|
-
}
|
|
1405
|
-
}
|
|
1406
|
-
/**
|
|
1407
|
-
* Retrieves NFT information by token ID and serial number.
|
|
1408
|
-
* @param tokenId The token ID.
|
|
1409
|
-
* @param serialNumber The serial number of the NFT.
|
|
1410
|
-
* @returns A promise that resolves to an NftInfo or null.
|
|
1411
|
-
*/
|
|
1412
|
-
async getNftInfo(tokenId, serialNumber) {
|
|
1413
|
-
this.logger.info(`Getting NFT info for ${tokenId}/${serialNumber}`);
|
|
1414
|
-
const url = `/api/v1/tokens/${tokenId}/nfts/${serialNumber}`;
|
|
1415
|
-
try {
|
|
1416
|
-
const response = await this._requestWithRetry(url);
|
|
1417
|
-
return response;
|
|
1418
|
-
} catch (error) {
|
|
1419
|
-
this.logger.error(
|
|
1420
|
-
`Error fetching NFT info for ${tokenId}/${serialNumber}: ${error.message}`
|
|
1421
|
-
);
|
|
1422
|
-
return null;
|
|
1423
|
-
}
|
|
1424
|
-
}
|
|
1425
|
-
/**
|
|
1426
|
-
* Retrieves NFTs for a specific token.
|
|
1427
|
-
* @param tokenId The token ID.
|
|
1428
|
-
* @param options Optional parameters for filtering NFTs.
|
|
1429
|
-
* @returns A promise that resolves to an array of NftInfo or null.
|
|
1430
|
-
*/
|
|
1431
|
-
async getNftsByToken(tokenId, options) {
|
|
1432
|
-
this.logger.info(`Getting NFTs for token ${tokenId}`);
|
|
1433
|
-
let url = `/api/v1/tokens/${tokenId}/nfts`;
|
|
1434
|
-
const params = new URLSearchParams();
|
|
1435
|
-
if (options?.accountId) {
|
|
1436
|
-
params.append("account.id", options.accountId);
|
|
1437
|
-
}
|
|
1438
|
-
if (options?.limit) {
|
|
1439
|
-
params.append("limit", options.limit.toString());
|
|
1440
|
-
}
|
|
1441
|
-
if (options?.order) {
|
|
1442
|
-
params.append("order", options.order);
|
|
1443
|
-
}
|
|
1444
|
-
if (options?.serialNumber) {
|
|
1445
|
-
params.append("serialnumber", options.serialNumber);
|
|
1446
|
-
}
|
|
1447
|
-
const queryString = params.toString();
|
|
1448
|
-
if (queryString) {
|
|
1449
|
-
url += `?${queryString}`;
|
|
1450
|
-
}
|
|
1451
|
-
try {
|
|
1452
|
-
const response = await this._requestWithRetry(url);
|
|
1453
|
-
return response.nfts || [];
|
|
1454
|
-
} catch (error) {
|
|
1455
|
-
this.logger.error(
|
|
1456
|
-
`Error fetching NFTs for token ${tokenId}: ${error.message}`
|
|
1457
|
-
);
|
|
1458
|
-
return null;
|
|
1459
|
-
}
|
|
1460
|
-
}
|
|
1461
|
-
/**
|
|
1462
|
-
* Retrieves network information.
|
|
1463
|
-
* @returns A promise that resolves to NetworkInfo or null.
|
|
1464
|
-
*/
|
|
1465
|
-
async getNetworkInfo() {
|
|
1466
|
-
this.logger.info("Getting network information");
|
|
1467
|
-
const url = `/api/v1/network/nodes`;
|
|
1468
|
-
try {
|
|
1469
|
-
const response = await this._requestWithRetry(url);
|
|
1470
|
-
return response;
|
|
1471
|
-
} catch (error) {
|
|
1472
|
-
this.logger.error(`Error fetching network info: ${error.message}`);
|
|
1473
|
-
return null;
|
|
1474
|
-
}
|
|
1475
|
-
}
|
|
1476
|
-
/**
|
|
1477
|
-
* Retrieves network fees.
|
|
1478
|
-
* @param timestamp Optional timestamp for historical fees.
|
|
1479
|
-
* @returns A promise that resolves to NetworkFees or null.
|
|
1480
|
-
*/
|
|
1481
|
-
async getNetworkFees(timestamp) {
|
|
1482
|
-
this.logger.info("Getting network fees");
|
|
1483
|
-
let url = `/api/v1/network/fees`;
|
|
1484
|
-
if (timestamp) {
|
|
1485
|
-
url += `?timestamp=${timestamp}`;
|
|
1486
|
-
}
|
|
1487
|
-
try {
|
|
1488
|
-
const response = await this._requestWithRetry(url);
|
|
1489
|
-
return response;
|
|
1490
|
-
} catch (error) {
|
|
1491
|
-
this.logger.error(`Error fetching network fees: ${error.message}`);
|
|
1492
|
-
return null;
|
|
1493
|
-
}
|
|
1494
|
-
}
|
|
1495
|
-
/**
|
|
1496
|
-
* Retrieves network supply information.
|
|
1497
|
-
* @param timestamp Optional timestamp for historical supply data.
|
|
1498
|
-
* @returns A promise that resolves to NetworkSupply or null.
|
|
1499
|
-
*/
|
|
1500
|
-
async getNetworkSupply(timestamp) {
|
|
1501
|
-
this.logger.info("Getting network supply");
|
|
1502
|
-
let url = `/api/v1/network/supply`;
|
|
1503
|
-
if (timestamp) {
|
|
1504
|
-
url += `?timestamp=${timestamp}`;
|
|
1505
|
-
}
|
|
1506
|
-
try {
|
|
1507
|
-
const response = await this._requestWithRetry(url);
|
|
1508
|
-
return response;
|
|
1509
|
-
} catch (error) {
|
|
1510
|
-
this.logger.error(`Error fetching network supply: ${error.message}`);
|
|
1511
|
-
return null;
|
|
1512
|
-
}
|
|
1513
|
-
}
|
|
1514
|
-
/**
|
|
1515
|
-
* Retrieves network stake information.
|
|
1516
|
-
* @param timestamp Optional timestamp for historical stake data.
|
|
1517
|
-
* @returns A promise that resolves to NetworkStake or null.
|
|
1518
|
-
*/
|
|
1519
|
-
async getNetworkStake(timestamp) {
|
|
1520
|
-
this.logger.info("Getting network stake");
|
|
1521
|
-
let url = `/api/v1/network/stake`;
|
|
1522
|
-
if (timestamp) {
|
|
1523
|
-
url += `?timestamp=${timestamp}`;
|
|
1524
|
-
}
|
|
1525
|
-
try {
|
|
1526
|
-
const response = await this._requestWithRetry(url);
|
|
1527
|
-
return response;
|
|
1528
|
-
} catch (error) {
|
|
1529
|
-
this.logger.error(`Error fetching network stake: ${error.message}`);
|
|
1530
|
-
return null;
|
|
1531
|
-
}
|
|
1532
|
-
}
|
|
1533
|
-
/**
|
|
1534
|
-
* Retrieves opcode traces for a specific transaction.
|
|
1535
|
-
* @param transactionIdOrHash The transaction ID or hash.
|
|
1536
|
-
* @param options Optional parameters for trace details.
|
|
1537
|
-
* @returns A promise that resolves to an OpcodesResponse or null.
|
|
1538
|
-
*/
|
|
1539
|
-
async getOpcodeTraces(transactionIdOrHash, options) {
|
|
1540
|
-
this.logger.info(`Getting opcode traces for ${transactionIdOrHash}`);
|
|
1541
|
-
let url = `/api/v1/contracts/results/${transactionIdOrHash}/opcodes`;
|
|
1542
|
-
const params = new URLSearchParams();
|
|
1543
|
-
if (options?.stack !== void 0) {
|
|
1544
|
-
params.append("stack", options.stack.toString());
|
|
1545
|
-
}
|
|
1546
|
-
if (options?.memory !== void 0) {
|
|
1547
|
-
params.append("memory", options.memory.toString());
|
|
1548
|
-
}
|
|
1549
|
-
if (options?.storage !== void 0) {
|
|
1550
|
-
params.append("storage", options.storage.toString());
|
|
1551
|
-
}
|
|
1552
|
-
const queryString = params.toString();
|
|
1553
|
-
if (queryString) {
|
|
1554
|
-
url += `?${queryString}`;
|
|
1555
|
-
}
|
|
1556
|
-
try {
|
|
1557
|
-
const response = await this._requestWithRetry(url);
|
|
1558
|
-
return response;
|
|
1112
|
+
} catch (e) {
|
|
1113
|
+
}
|
|
1114
|
+
const protoResult = this.parseHTSTransaction(transaction);
|
|
1115
|
+
if (protoResult.type) {
|
|
1116
|
+
return protoResult;
|
|
1117
|
+
}
|
|
1118
|
+
const tokenCreation = this.extractTokenCreationFromTransaction(transaction);
|
|
1119
|
+
const tokenAirdrop = this.extractTokenAirdropFromTransaction(transaction);
|
|
1120
|
+
if (tokenCreation) {
|
|
1121
|
+
return {
|
|
1122
|
+
type: "TOKENCREATE",
|
|
1123
|
+
humanReadableType: "Token Creation",
|
|
1124
|
+
tokenCreation
|
|
1125
|
+
};
|
|
1126
|
+
}
|
|
1127
|
+
if (tokenAirdrop) {
|
|
1128
|
+
return {
|
|
1129
|
+
type: "TOKENAIRDROP",
|
|
1130
|
+
humanReadableType: "Token Airdrop",
|
|
1131
|
+
tokenAirdrop
|
|
1132
|
+
};
|
|
1133
|
+
}
|
|
1134
|
+
return {};
|
|
1559
1135
|
} catch (error) {
|
|
1560
|
-
|
|
1561
|
-
`Error fetching opcode traces for ${transactionIdOrHash}: ${error.message}`
|
|
1562
|
-
);
|
|
1563
|
-
return null;
|
|
1136
|
+
return {};
|
|
1564
1137
|
}
|
|
1565
1138
|
}
|
|
1566
1139
|
}
|
|
1567
1140
|
export {
|
|
1568
|
-
|
|
1141
|
+
HTSParser
|
|
1569
1142
|
};
|
|
1570
1143
|
//# sourceMappingURL=standards-sdk.es120.js.map
|