@hashgraphonline/standards-sdk 0.1.168 → 0.1.169
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/README.md +1 -0
- package/dist/browser/hcs-11/client.d.ts.map +1 -1
- package/dist/browser/hcs-21/sdk.d.ts.map +1 -1
- package/dist/browser/hcs-27/base-client.d.ts +41 -0
- package/dist/browser/hcs-27/base-client.d.ts.map +1 -0
- package/dist/browser/hcs-27/index.d.ts +6 -0
- package/dist/browser/hcs-27/index.d.ts.map +1 -0
- package/dist/browser/hcs-27/memos.d.ts +5 -0
- package/dist/browser/hcs-27/memos.d.ts.map +1 -0
- package/dist/browser/hcs-27/merkle.d.ts +23 -0
- package/dist/browser/hcs-27/merkle.d.ts.map +1 -0
- package/dist/browser/hcs-27/sdk.d.ts +23 -0
- package/dist/browser/hcs-27/sdk.d.ts.map +1 -0
- package/dist/browser/hcs-27/types.d.ts +1611 -0
- package/dist/browser/hcs-27/types.d.ts.map +1 -0
- package/dist/browser/index.d.ts +1 -0
- package/dist/browser/index.d.ts.map +1 -1
- package/dist/browser/standards-sdk.browser.js +29 -5
- package/dist/browser/standards-sdk.browser.js.map +1 -1
- package/dist/browser/utils/key-type-detector.d.ts.map +1 -1
- package/dist/cjs/hcs-11/client.d.ts.map +1 -1
- package/dist/cjs/hcs-21/sdk.d.ts.map +1 -1
- package/dist/cjs/hcs-27/base-client.d.ts +41 -0
- package/dist/cjs/hcs-27/base-client.d.ts.map +1 -0
- package/dist/cjs/hcs-27/index.d.ts +6 -0
- package/dist/cjs/hcs-27/index.d.ts.map +1 -0
- package/dist/cjs/hcs-27/memos.d.ts +5 -0
- package/dist/cjs/hcs-27/memos.d.ts.map +1 -0
- package/dist/cjs/hcs-27/merkle.d.ts +23 -0
- package/dist/cjs/hcs-27/merkle.d.ts.map +1 -0
- package/dist/cjs/hcs-27/sdk.d.ts +23 -0
- package/dist/cjs/hcs-27/sdk.d.ts.map +1 -0
- package/dist/cjs/hcs-27/types.d.ts +1611 -0
- package/dist/cjs/hcs-27/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/standards-sdk.cjs +2 -2
- package/dist/cjs/standards-sdk.cjs.map +1 -1
- package/dist/cjs/utils/key-type-detector.d.ts.map +1 -1
- package/dist/es/hcs-11/client.d.ts.map +1 -1
- package/dist/es/hcs-21/sdk.d.ts.map +1 -1
- package/dist/es/hcs-27/base-client.d.ts +41 -0
- package/dist/es/hcs-27/base-client.d.ts.map +1 -0
- package/dist/es/hcs-27/index.d.ts +6 -0
- package/dist/es/hcs-27/index.d.ts.map +1 -0
- package/dist/es/hcs-27/memos.d.ts +5 -0
- package/dist/es/hcs-27/memos.d.ts.map +1 -0
- package/dist/es/hcs-27/merkle.d.ts +23 -0
- package/dist/es/hcs-27/merkle.d.ts.map +1 -0
- package/dist/es/hcs-27/sdk.d.ts +23 -0
- package/dist/es/hcs-27/sdk.d.ts.map +1 -0
- package/dist/es/hcs-27/types.d.ts +1611 -0
- package/dist/es/hcs-27/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/standards-sdk.es.js +67 -38
- package/dist/es/standards-sdk.es.js.map +1 -1
- package/dist/es/standards-sdk.es101.js +2 -2
- package/dist/es/standards-sdk.es103.js +2 -2
- package/dist/es/standards-sdk.es104.js +1 -1
- package/dist/es/standards-sdk.es106.js +1 -1
- package/dist/es/standards-sdk.es108.js +2 -2
- package/dist/es/standards-sdk.es11.js +1 -1
- package/dist/es/standards-sdk.es110.js +1 -1
- package/dist/es/standards-sdk.es111.js +2 -2
- package/dist/es/standards-sdk.es112.js +151 -234
- package/dist/es/standards-sdk.es112.js.map +1 -1
- package/dist/es/standards-sdk.es113.js +20 -471
- package/dist/es/standards-sdk.es113.js.map +1 -1
- package/dist/es/standards-sdk.es114.js +263 -104
- package/dist/es/standards-sdk.es114.js.map +1 -1
- package/dist/es/standards-sdk.es115.js +167 -138
- package/dist/es/standards-sdk.es115.js.map +1 -1
- package/dist/es/standards-sdk.es116.js +315 -29
- package/dist/es/standards-sdk.es116.js.map +1 -1
- package/dist/es/standards-sdk.es117.js +250 -10
- package/dist/es/standards-sdk.es117.js.map +1 -1
- package/dist/es/standards-sdk.es118.js +448 -152
- package/dist/es/standards-sdk.es118.js.map +1 -1
- package/dist/es/standards-sdk.es119.js +101 -25
- 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 +155 -17
- package/dist/es/standards-sdk.es120.js.map +1 -1
- package/dist/es/standards-sdk.es121.js +29 -155
- package/dist/es/standards-sdk.es121.js.map +1 -1
- package/dist/es/standards-sdk.es122.js +9 -200
- package/dist/es/standards-sdk.es122.js.map +1 -1
- package/dist/es/standards-sdk.es123.js +146 -754
- package/dist/es/standards-sdk.es123.js.map +1 -1
- package/dist/es/standards-sdk.es124.js +27 -11
- package/dist/es/standards-sdk.es124.js.map +1 -1
- package/dist/es/standards-sdk.es125.js +19 -564
- package/dist/es/standards-sdk.es125.js.map +1 -1
- package/dist/es/standards-sdk.es126.js +140 -582
- package/dist/es/standards-sdk.es126.js.map +1 -1
- package/dist/es/standards-sdk.es127.js +202 -12
- package/dist/es/standards-sdk.es127.js.map +1 -1
- package/dist/es/standards-sdk.es128.js +790 -2
- package/dist/es/standards-sdk.es128.js.map +1 -1
- package/dist/es/standards-sdk.es129.js +10 -84
- 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 +567 -40
- package/dist/es/standards-sdk.es130.js.map +1 -1
- package/dist/es/standards-sdk.es131.js +626 -2
- package/dist/es/standards-sdk.es131.js.map +1 -1
- package/dist/es/standards-sdk.es132.js +12 -234
- package/dist/es/standards-sdk.es132.js.map +1 -1
- package/dist/es/standards-sdk.es133.js +2 -1140
- package/dist/es/standards-sdk.es133.js.map +1 -1
- package/dist/es/standards-sdk.es134.js +73 -292
- package/dist/es/standards-sdk.es134.js.map +1 -1
- package/dist/es/standards-sdk.es135.js +36 -418
- package/dist/es/standards-sdk.es135.js.map +1 -1
- package/dist/es/standards-sdk.es136.js +2 -355
- package/dist/es/standards-sdk.es136.js.map +1 -1
- package/dist/es/standards-sdk.es137.js +198 -1079
- package/dist/es/standards-sdk.es137.js.map +1 -1
- package/dist/es/standards-sdk.es138.js +1107 -175
- package/dist/es/standards-sdk.es138.js.map +1 -1
- package/dist/es/standards-sdk.es139.js +218 -1479
- 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 +422 -1500
- package/dist/es/standards-sdk.es140.js.map +1 -1
- package/dist/es/standards-sdk.es141.js +351 -13
- package/dist/es/standards-sdk.es141.js.map +1 -1
- package/dist/es/standards-sdk.es142.js +1102 -73
- package/dist/es/standards-sdk.es142.js.map +1 -1
- package/dist/es/standards-sdk.es143.js +203 -76
- package/dist/es/standards-sdk.es143.js.map +1 -1
- package/dist/es/standards-sdk.es144.js +1459 -830
- package/dist/es/standards-sdk.es144.js.map +1 -1
- package/dist/es/standards-sdk.es145.js +1499 -59
- package/dist/es/standards-sdk.es145.js.map +1 -1
- package/dist/es/standards-sdk.es146.js +14 -156
- package/dist/es/standards-sdk.es146.js.map +1 -1
- package/dist/es/standards-sdk.es147.js +87 -7
- package/dist/es/standards-sdk.es147.js.map +1 -1
- package/dist/es/standards-sdk.es148.js +74 -79
- package/dist/es/standards-sdk.es148.js.map +1 -1
- package/dist/es/standards-sdk.es149.js +934 -61
- package/dist/es/standards-sdk.es149.js.map +1 -1
- package/dist/es/standards-sdk.es15.js +1 -1
- package/dist/es/standards-sdk.es150.js +60 -30
- package/dist/es/standards-sdk.es150.js.map +1 -1
- package/dist/es/standards-sdk.es151.js +159 -34
- package/dist/es/standards-sdk.es151.js.map +1 -1
- package/dist/es/standards-sdk.es152.js +7 -48
- package/dist/es/standards-sdk.es152.js.map +1 -1
- package/dist/es/standards-sdk.es153.js +70 -122
- package/dist/es/standards-sdk.es153.js.map +1 -1
- package/dist/es/standards-sdk.es154.js +58 -35
- package/dist/es/standards-sdk.es154.js.map +1 -1
- package/dist/es/standards-sdk.es155.js +30 -56
- package/dist/es/standards-sdk.es155.js.map +1 -1
- package/dist/es/standards-sdk.es156.js +34 -84
- package/dist/es/standards-sdk.es156.js.map +1 -1
- package/dist/es/standards-sdk.es157.js +48 -81
- package/dist/es/standards-sdk.es157.js.map +1 -1
- package/dist/es/standards-sdk.es158.js +124 -186
- package/dist/es/standards-sdk.es158.js.map +1 -1
- package/dist/es/standards-sdk.es159.js +34 -12474
- package/dist/es/standards-sdk.es159.js.map +1 -1
- package/dist/es/standards-sdk.es16.js +5 -5
- package/dist/es/standards-sdk.es160.js +12477 -12
- package/dist/es/standards-sdk.es160.js.map +1 -1
- package/dist/es/standards-sdk.es161.js +15 -71
- package/dist/es/standards-sdk.es161.js.map +1 -1
- package/dist/es/standards-sdk.es162.js +48 -542
- package/dist/es/standards-sdk.es162.js.map +1 -1
- package/dist/es/standards-sdk.es163.js +72 -160
- package/dist/es/standards-sdk.es163.js.map +1 -1
- package/dist/es/standards-sdk.es164.js +71 -312
- package/dist/es/standards-sdk.es164.js.map +1 -1
- package/dist/es/standards-sdk.es165.js +187 -333
- package/dist/es/standards-sdk.es165.js.map +1 -1
- package/dist/es/standards-sdk.es166.js +538 -441
- package/dist/es/standards-sdk.es166.js.map +1 -1
- package/dist/es/standards-sdk.es167.js +65 -323
- package/dist/es/standards-sdk.es167.js.map +1 -1
- package/dist/es/standards-sdk.es168.js +162 -66
- package/dist/es/standards-sdk.es168.js.map +1 -1
- package/dist/es/standards-sdk.es169.js +310 -160
- package/dist/es/standards-sdk.es169.js.map +1 -1
- package/dist/es/standards-sdk.es170.js +336 -212
- package/dist/es/standards-sdk.es170.js.map +1 -1
- package/dist/es/standards-sdk.es171.js +437 -223
- package/dist/es/standards-sdk.es171.js.map +1 -1
- package/dist/es/standards-sdk.es172.js +324 -112
- package/dist/es/standards-sdk.es172.js.map +1 -1
- package/dist/es/standards-sdk.es173.js +65 -115
- package/dist/es/standards-sdk.es173.js.map +1 -1
- package/dist/es/standards-sdk.es174.js +151 -140
- package/dist/es/standards-sdk.es174.js.map +1 -1
- package/dist/es/standards-sdk.es175.js +193 -156
- package/dist/es/standards-sdk.es175.js.map +1 -1
- package/dist/es/standards-sdk.es176.js +221 -121
- package/dist/es/standards-sdk.es176.js.map +1 -1
- package/dist/es/standards-sdk.es177.js +81 -293
- package/dist/es/standards-sdk.es177.js.map +1 -1
- package/dist/es/standards-sdk.es178.js +114 -247
- package/dist/es/standards-sdk.es178.js.map +1 -1
- package/dist/es/standards-sdk.es179.js +119 -110
- package/dist/es/standards-sdk.es179.js.map +1 -1
- package/dist/es/standards-sdk.es18.js +12 -12
- package/dist/es/standards-sdk.es180.js +188 -0
- package/dist/es/standards-sdk.es180.js.map +1 -0
- package/dist/es/standards-sdk.es181.js +142 -0
- package/dist/es/standards-sdk.es181.js.map +1 -0
- package/dist/es/standards-sdk.es182.js +334 -0
- package/dist/es/standards-sdk.es182.js.map +1 -0
- package/dist/es/standards-sdk.es183.js +262 -0
- package/dist/es/standards-sdk.es183.js.map +1 -0
- package/dist/es/standards-sdk.es184.js +155 -0
- package/dist/es/standards-sdk.es184.js.map +1 -0
- package/dist/es/standards-sdk.es19.js +9 -9
- package/dist/es/standards-sdk.es2.js +2 -2
- package/dist/es/standards-sdk.es20.js +1 -1
- 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 +12 -12
- package/dist/es/standards-sdk.es27.js.map +1 -1
- package/dist/es/standards-sdk.es30.js +2 -2
- package/dist/es/standards-sdk.es31.js +4 -4
- package/dist/es/standards-sdk.es32.js +1 -1
- package/dist/es/standards-sdk.es35.js +6 -6
- package/dist/es/standards-sdk.es36.js +4 -4
- package/dist/es/standards-sdk.es37.js +2 -2
- package/dist/es/standards-sdk.es38.js +2 -2
- package/dist/es/standards-sdk.es39.js +1 -1
- package/dist/es/standards-sdk.es4.js +2 -2
- package/dist/es/standards-sdk.es40.js +1 -1
- package/dist/es/standards-sdk.es41.js +2 -2
- package/dist/es/standards-sdk.es46.js +1 -1
- package/dist/es/standards-sdk.es5.js +2 -2
- package/dist/es/standards-sdk.es51.js +1 -1
- package/dist/es/standards-sdk.es53.js +1 -1
- package/dist/es/standards-sdk.es56.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.es62.js +1 -1
- package/dist/es/standards-sdk.es63.js +2 -2
- package/dist/es/standards-sdk.es64.js +1 -1
- package/dist/es/standards-sdk.es65.js +1 -1
- package/dist/es/standards-sdk.es66.js +1 -1
- package/dist/es/standards-sdk.es67.js +7 -7
- package/dist/es/standards-sdk.es69.js +1 -1
- package/dist/es/standards-sdk.es7.js +1 -1
- package/dist/es/standards-sdk.es71.js +2 -2
- package/dist/es/standards-sdk.es72.js +3 -3
- package/dist/es/standards-sdk.es75.js +5 -5
- package/dist/es/standards-sdk.es76.js +3 -3
- package/dist/es/standards-sdk.es77.js +2 -2
- package/dist/es/standards-sdk.es78.js +1 -1
- package/dist/es/standards-sdk.es81.js +2 -2
- package/dist/es/standards-sdk.es83.js +2 -2
- package/dist/es/standards-sdk.es84.js +4 -4
- package/dist/es/standards-sdk.es85.js +1 -1
- package/dist/es/standards-sdk.es88.js +1 -1
- package/dist/es/standards-sdk.es89.js +2 -2
- package/dist/es/standards-sdk.es9.js +2 -2
- package/dist/es/standards-sdk.es90.js +4 -4
- package/dist/es/standards-sdk.es90.js.map +1 -1
- package/dist/es/standards-sdk.es94.js +3 -3
- package/dist/es/standards-sdk.es96.js +2 -2
- package/dist/es/standards-sdk.es98.js +1 -1
- package/dist/es/standards-sdk.es99.js +3 -3
- package/dist/es/utils/key-type-detector.d.ts.map +1 -1
- package/package.json +1 -1
|
@@ -1,1570 +1,309 @@
|
|
|
1
|
-
import { PublicKey, Timestamp, AccountId } from "@hashgraph/sdk";
|
|
2
|
-
import axios from "axios";
|
|
3
|
-
import { Logger } from "./standards-sdk.es118.js";
|
|
4
1
|
import { proto } from "@hashgraph/proto";
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
this.initialDelayMs = 2e3;
|
|
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
|
-
}
|
|
70
|
-
/**
|
|
71
|
-
* Returns the base URL for the Hedera mirror node based on the network type
|
|
72
|
-
* @returns The mirror node base URL
|
|
73
|
-
* @private
|
|
74
|
-
*/
|
|
75
|
-
getMirrorNodeUrl() {
|
|
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);
|
|
90
|
-
try {
|
|
91
|
-
if (!accountInfo || !accountInfo.key) {
|
|
92
|
-
throw new Error(
|
|
93
|
-
`Failed to retrieve public key for account ID: ${accountId}`
|
|
94
|
-
);
|
|
95
|
-
}
|
|
96
|
-
return PublicKey.fromString(accountInfo.key.key);
|
|
97
|
-
} catch (e) {
|
|
98
|
-
const error = e;
|
|
99
|
-
const logMessage = `Error fetching public key from Mirror Node: ${error.message}`;
|
|
100
|
-
this.logger.error(logMessage);
|
|
101
|
-
throw new Error(logMessage);
|
|
102
|
-
}
|
|
103
|
-
}
|
|
2
|
+
import { hasTransactionType, parseKey } from "./standards-sdk.es173.js";
|
|
3
|
+
import { AccountId, Long, ScheduleId } from "@hashgraph/sdk";
|
|
4
|
+
class ScheduleParser {
|
|
104
5
|
/**
|
|
105
|
-
*
|
|
106
|
-
*
|
|
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.
|
|
6
|
+
* Parse Schedule Service transaction using unified dual-branch approach
|
|
7
|
+
* This handles both regular transactions and signed transaction variants
|
|
109
8
|
*/
|
|
110
|
-
|
|
111
|
-
this.logger.debug(`Getting account memo for account ID: ${accountId}`);
|
|
9
|
+
static parseScheduleTransaction(transaction, originalBytes) {
|
|
112
10
|
try {
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
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;
|
|
203
|
-
}
|
|
204
|
-
this.logger.warn(`No token info found for ${tokenId}`);
|
|
205
|
-
return null;
|
|
206
|
-
} catch (e) {
|
|
207
|
-
const error = e;
|
|
208
|
-
const logMessage = `Error fetching token info for ${tokenId}: ${error.message}`;
|
|
209
|
-
this.logger.error(logMessage);
|
|
210
|
-
return null;
|
|
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);
|
|
233
|
-
}
|
|
234
|
-
}
|
|
235
|
-
if (options.limit) {
|
|
236
|
-
params.append("limit", options.limit.toString());
|
|
237
|
-
}
|
|
238
|
-
if (options.order) {
|
|
239
|
-
params.append("order", options.order);
|
|
240
|
-
}
|
|
241
|
-
}
|
|
242
|
-
const queryString = params.toString();
|
|
243
|
-
if (queryString) {
|
|
244
|
-
endpoint += `?${queryString}`;
|
|
245
|
-
}
|
|
246
|
-
const messages = [];
|
|
247
|
-
let nextEndpoint = endpoint;
|
|
248
|
-
while (nextEndpoint) {
|
|
249
|
-
try {
|
|
250
|
-
const data = await this._requestWithRetry(nextEndpoint);
|
|
251
|
-
if (data.messages && data.messages.length > 0) {
|
|
252
|
-
for (const message of data.messages) {
|
|
253
|
-
try {
|
|
254
|
-
if (!message.message) {
|
|
255
|
-
continue;
|
|
256
|
-
}
|
|
257
|
-
let messageContent;
|
|
258
|
-
try {
|
|
259
|
-
if (this.isServerEnvironment) {
|
|
260
|
-
messageContent = Buffer.from(
|
|
261
|
-
message.message,
|
|
262
|
-
"base64"
|
|
263
|
-
).toString("utf-8");
|
|
264
|
-
} else {
|
|
265
|
-
messageContent = new TextDecoder().decode(
|
|
266
|
-
Uint8Array.from(
|
|
267
|
-
atob(message.message),
|
|
268
|
-
(c) => c.charCodeAt(0)
|
|
269
|
-
)
|
|
270
|
-
);
|
|
271
|
-
}
|
|
272
|
-
} catch (error) {
|
|
273
|
-
const logMessage = `Error decoding message: ${error}`;
|
|
274
|
-
this.logger.error(logMessage);
|
|
275
|
-
continue;
|
|
11
|
+
if (originalBytes || transaction.toBytes) {
|
|
12
|
+
try {
|
|
13
|
+
const bytesToParse = originalBytes || transaction.toBytes();
|
|
14
|
+
const decoded = proto.TransactionList.decode(bytesToParse);
|
|
15
|
+
if (decoded.transactionList && decoded.transactionList.length > 0) {
|
|
16
|
+
const tx = decoded.transactionList[0];
|
|
17
|
+
let txBody = null;
|
|
18
|
+
if (tx.bodyBytes && tx.bodyBytes.length > 0) {
|
|
19
|
+
txBody = proto.TransactionBody.decode(tx.bodyBytes);
|
|
20
|
+
} else if (tx.signedTransactionBytes && tx.signedTransactionBytes.length > 0) {
|
|
21
|
+
const signedTx = proto.SignedTransaction.decode(
|
|
22
|
+
tx.signedTransactionBytes
|
|
23
|
+
);
|
|
24
|
+
if (signedTx.bodyBytes) {
|
|
25
|
+
txBody = proto.TransactionBody.decode(signedTx.bodyBytes);
|
|
276
26
|
}
|
|
277
|
-
|
|
278
|
-
|
|
279
|
-
|
|
280
|
-
|
|
281
|
-
|
|
282
|
-
this.logger.error(logMessage);
|
|
283
|
-
continue;
|
|
27
|
+
}
|
|
28
|
+
if (txBody) {
|
|
29
|
+
const protoResult = this.parseFromProtobufTxBody(txBody);
|
|
30
|
+
if (protoResult.type && protoResult.type !== "UNKNOWN") {
|
|
31
|
+
return protoResult;
|
|
284
32
|
}
|
|
285
|
-
messageJson.sequence_number = message.sequence_number;
|
|
286
|
-
messages.push({
|
|
287
|
-
...messageJson,
|
|
288
|
-
consensus_timestamp: message.consensus_timestamp,
|
|
289
|
-
sequence_number: message.sequence_number,
|
|
290
|
-
running_hash: message.running_hash,
|
|
291
|
-
running_hash_version: message.running_hash_version,
|
|
292
|
-
topic_id: message.topic_id,
|
|
293
|
-
payer: message.payer_account_id,
|
|
294
|
-
created: new Date(Number(message.consensus_timestamp) * 1e3)
|
|
295
|
-
});
|
|
296
|
-
} catch (error) {
|
|
297
|
-
const logMessage = `Error processing message: ${error.message}`;
|
|
298
|
-
this.logger.error(logMessage);
|
|
299
33
|
}
|
|
300
34
|
}
|
|
35
|
+
} catch (protoError) {
|
|
301
36
|
}
|
|
302
|
-
nextEndpoint = data.links?.next || "";
|
|
303
|
-
} catch (e) {
|
|
304
|
-
const error = e;
|
|
305
|
-
const logMessage = `Error querying topic messages for topic ${topicId} (endpoint: ${nextEndpoint}) after retries: ${error.message}`;
|
|
306
|
-
this.logger.error(logMessage);
|
|
307
|
-
throw new Error(logMessage);
|
|
308
37
|
}
|
|
309
|
-
|
|
310
|
-
return messages;
|
|
311
|
-
}
|
|
312
|
-
/**
|
|
313
|
-
* Requests account information for a given account ID from the mirror node.
|
|
314
|
-
* @param accountId The ID of the account to retrieve information for.
|
|
315
|
-
* @returns A promise that resolves to the account information.
|
|
316
|
-
* @throws An error if the account ID is invalid or the information cannot be retrieved.
|
|
317
|
-
*/
|
|
318
|
-
async requestAccount(accountId) {
|
|
319
|
-
try {
|
|
320
|
-
this.logger.debug(`Requesting account info for ${accountId}`);
|
|
321
|
-
const data = await this._requestWithRetry(
|
|
322
|
-
`/api/v1/accounts/${accountId}`
|
|
323
|
-
);
|
|
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
|
-
}
|
|
336
|
-
}
|
|
337
|
-
/**
|
|
338
|
-
* Checks if a user has access to a given key list.
|
|
339
|
-
* @param keyBytes The key list to check access for.
|
|
340
|
-
* @param userPublicKey The public key of the user to check access for.
|
|
341
|
-
* @returns A promise that resolves to true if the user has access, false otherwise.
|
|
342
|
-
*/
|
|
343
|
-
async checkKeyListAccess(keyBytes, userPublicKey) {
|
|
344
|
-
try {
|
|
345
|
-
const key = proto.Key.decode(keyBytes);
|
|
346
|
-
return this.evaluateKeyAccess(key, userPublicKey);
|
|
347
|
-
} catch (e) {
|
|
348
|
-
const error = e;
|
|
349
|
-
const logMessage = `Error decoding protobuf key: ${error.message}`;
|
|
350
|
-
this.logger.error(logMessage);
|
|
351
|
-
throw new Error(logMessage);
|
|
352
|
-
}
|
|
353
|
-
}
|
|
354
|
-
/**
|
|
355
|
-
* Evaluates the access of a given key to a user's public key.
|
|
356
|
-
* @param key The key to evaluate access for.
|
|
357
|
-
* @param userPublicKey The public key of the user to evaluate access for.
|
|
358
|
-
* @returns A promise that resolves to true if the key has access, false otherwise.
|
|
359
|
-
*/
|
|
360
|
-
async evaluateKeyAccess(key, userPublicKey) {
|
|
361
|
-
if (key.ed25519) {
|
|
362
|
-
return this.compareEd25519Key(key.ed25519, userPublicKey);
|
|
363
|
-
}
|
|
364
|
-
if (key.keyList) {
|
|
365
|
-
return this.evaluateKeyList(key.keyList, userPublicKey);
|
|
366
|
-
}
|
|
367
|
-
if (key.thresholdKey && key.thresholdKey.keys) {
|
|
368
|
-
return this.evaluateKeyList(key.thresholdKey.keys, userPublicKey);
|
|
369
|
-
}
|
|
370
|
-
return false;
|
|
371
|
-
}
|
|
372
|
-
/**
|
|
373
|
-
* Evaluates the access of a given key list to a user's public key.
|
|
374
|
-
* @param keyList The key list to evaluate access for.
|
|
375
|
-
* @param userPublicKey The public key of the user to evaluate access for.
|
|
376
|
-
* @returns A promise that resolves to true if the key list has access, false otherwise.
|
|
377
|
-
*/
|
|
378
|
-
async evaluateKeyList(keyList, userPublicKey) {
|
|
379
|
-
const keys = keyList.keys || [];
|
|
380
|
-
for (const listKey of keys) {
|
|
381
|
-
if (!listKey) continue;
|
|
382
|
-
if (listKey.ed25519) {
|
|
383
|
-
if (this.compareEd25519Key(listKey.ed25519, userPublicKey)) {
|
|
384
|
-
return true;
|
|
385
|
-
}
|
|
386
|
-
} else if (listKey.keyList || listKey.thresholdKey) {
|
|
387
|
-
try {
|
|
388
|
-
const nestedKeyBytes = proto.Key.encode({
|
|
389
|
-
...listKey.keyList ? { keyList: listKey.keyList } : {},
|
|
390
|
-
...listKey.thresholdKey ? { thresholdKey: listKey.thresholdKey } : {}
|
|
391
|
-
}).finish();
|
|
392
|
-
const hasNestedAccess = await this.checkKeyListAccess(
|
|
393
|
-
Buffer.from(nestedKeyBytes),
|
|
394
|
-
userPublicKey
|
|
395
|
-
);
|
|
396
|
-
if (hasNestedAccess) {
|
|
397
|
-
return true;
|
|
398
|
-
}
|
|
399
|
-
} catch (e) {
|
|
400
|
-
const error = e;
|
|
401
|
-
const logMessage = `Error in nested key: ${error.message}`;
|
|
402
|
-
this.logger.debug(logMessage);
|
|
403
|
-
}
|
|
404
|
-
}
|
|
405
|
-
}
|
|
406
|
-
return false;
|
|
407
|
-
}
|
|
408
|
-
/**
|
|
409
|
-
* Compares an Ed25519 key with a user's public key.
|
|
410
|
-
* @param keyData The Ed25519 key data to compare.
|
|
411
|
-
* @param userPublicKey The public key of the user to compare with.
|
|
412
|
-
* @returns A boolean indicating whether the key matches the user's public key.
|
|
413
|
-
*/
|
|
414
|
-
compareEd25519Key(keyData, userPublicKey) {
|
|
415
|
-
try {
|
|
416
|
-
const decodedKey = PublicKey.fromBytes(Buffer.from(keyData));
|
|
417
|
-
return decodedKey.toString() === userPublicKey.toString();
|
|
418
|
-
} catch (e) {
|
|
419
|
-
const error = e;
|
|
420
|
-
const logMessage = `Error comparing Ed25519 key: ${error.message}`;
|
|
421
|
-
this.logger.debug(logMessage);
|
|
422
|
-
return false;
|
|
423
|
-
}
|
|
424
|
-
}
|
|
425
|
-
/**
|
|
426
|
-
* Retrieves information about a scheduled transaction
|
|
427
|
-
* @param scheduleId The ID of the scheduled transaction
|
|
428
|
-
* @returns A promise that resolves to the scheduled transaction information
|
|
429
|
-
*/
|
|
430
|
-
async getScheduleInfo(scheduleId) {
|
|
431
|
-
try {
|
|
432
|
-
this.logger.info(
|
|
433
|
-
`Getting information for scheduled transaction ${scheduleId}`
|
|
434
|
-
);
|
|
435
|
-
const data = await this._requestWithRetry(
|
|
436
|
-
`/api/v1/schedules/${scheduleId}`
|
|
437
|
-
);
|
|
438
|
-
if (data) {
|
|
439
|
-
return data;
|
|
440
|
-
}
|
|
441
|
-
this.logger.warn(
|
|
442
|
-
`No schedule info found for ${scheduleId} after retries.`
|
|
443
|
-
);
|
|
444
|
-
return null;
|
|
38
|
+
return this.parseFromTransactionInternals(transaction);
|
|
445
39
|
} catch (error) {
|
|
446
|
-
this.logger.error(
|
|
447
|
-
`Error fetching schedule info for ${scheduleId} after retries: ${error.message}`
|
|
448
|
-
);
|
|
449
|
-
return null;
|
|
450
|
-
}
|
|
451
|
-
}
|
|
452
|
-
/**
|
|
453
|
-
* Checks the status of a scheduled transaction
|
|
454
|
-
* @param scheduleId The schedule ID to check
|
|
455
|
-
* @returns Status of the scheduled transaction
|
|
456
|
-
*/
|
|
457
|
-
async getScheduledTransactionStatus(scheduleId) {
|
|
458
|
-
try {
|
|
459
|
-
this.logger.info(
|
|
460
|
-
`Checking status of scheduled transaction ${scheduleId}`
|
|
461
|
-
);
|
|
462
|
-
const scheduleInfo = await this.getScheduleInfo(scheduleId);
|
|
463
|
-
if (!scheduleInfo) {
|
|
464
|
-
throw new Error(`Schedule ${scheduleId} not found`);
|
|
465
|
-
}
|
|
466
40
|
return {
|
|
467
|
-
|
|
468
|
-
|
|
469
|
-
deleted: scheduleInfo.deleted || false
|
|
41
|
+
type: "UNKNOWN",
|
|
42
|
+
humanReadableType: "Unknown Schedule Transaction"
|
|
470
43
|
};
|
|
471
|
-
} catch (error) {
|
|
472
|
-
this.logger.error(
|
|
473
|
-
`Error checking scheduled transaction status: ${error}`
|
|
474
|
-
);
|
|
475
|
-
throw error;
|
|
476
44
|
}
|
|
477
45
|
}
|
|
478
46
|
/**
|
|
479
|
-
*
|
|
480
|
-
*
|
|
481
|
-
* @returns A promise that resolves to the transaction details.
|
|
482
|
-
* @throws An error if the transaction ID/hash is invalid or details cannot be retrieved.
|
|
47
|
+
* Parse schedule transaction from protobuf TransactionBody
|
|
48
|
+
* Handles all schedule operations from decoded protobuf data
|
|
483
49
|
*/
|
|
484
|
-
|
|
485
|
-
|
|
486
|
-
|
|
487
|
-
|
|
488
|
-
try {
|
|
489
|
-
const response = await this._requestWithRetry(`/api/v1/transactions/${transactionIdOrHash}`);
|
|
490
|
-
if (response?.transactions?.length > 0) {
|
|
491
|
-
this.logger.trace(
|
|
492
|
-
`Transaction details found for ${transactionIdOrHash}:`,
|
|
493
|
-
response.transactions[0]
|
|
494
|
-
);
|
|
495
|
-
return response.transactions[0];
|
|
496
|
-
}
|
|
497
|
-
this.logger.warn(
|
|
498
|
-
`No transaction details found for ${transactionIdOrHash} or unexpected response structure.`
|
|
50
|
+
static parseFromProtobufTxBody(txBody) {
|
|
51
|
+
if (txBody.scheduleCreate) {
|
|
52
|
+
const scheduleCreate = this.parseScheduleCreateFromProto(
|
|
53
|
+
txBody.scheduleCreate
|
|
499
54
|
);
|
|
500
|
-
|
|
501
|
-
|
|
502
|
-
|
|
503
|
-
|
|
504
|
-
|
|
505
|
-
|
|
506
|
-
return null;
|
|
507
|
-
}
|
|
508
|
-
}
|
|
509
|
-
/**
|
|
510
|
-
* Private helper to make GET requests with retry logic using Axios.
|
|
511
|
-
*/
|
|
512
|
-
async _requestWithRetry(endpoint, axiosConfig) {
|
|
513
|
-
let attempt = 0;
|
|
514
|
-
let delay = this.initialDelayMs;
|
|
515
|
-
const url = this.constructUrl(endpoint);
|
|
516
|
-
const config = {
|
|
517
|
-
...axiosConfig,
|
|
518
|
-
headers: {
|
|
519
|
-
...this.customHeaders,
|
|
520
|
-
...axiosConfig?.headers
|
|
55
|
+
if (scheduleCreate) {
|
|
56
|
+
return {
|
|
57
|
+
type: "SCHEDULECREATE",
|
|
58
|
+
humanReadableType: "Schedule Create",
|
|
59
|
+
scheduleCreate
|
|
60
|
+
};
|
|
521
61
|
}
|
|
522
|
-
};
|
|
523
|
-
if (this.apiKey) {
|
|
524
|
-
config.headers = {
|
|
525
|
-
...config.headers,
|
|
526
|
-
Authorization: `Bearer ${this.apiKey}`,
|
|
527
|
-
"X-API-Key": this.apiKey
|
|
528
|
-
};
|
|
529
62
|
}
|
|
530
|
-
|
|
531
|
-
|
|
532
|
-
|
|
533
|
-
return
|
|
534
|
-
|
|
535
|
-
|
|
536
|
-
|
|
537
|
-
|
|
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
|
-
}
|
|
556
|
-
}
|
|
557
|
-
throw new Error(
|
|
558
|
-
`Failed to fetch data from ${url} after ${this.maxRetries} attempts.`
|
|
559
|
-
);
|
|
560
|
-
}
|
|
561
|
-
/**
|
|
562
|
-
* Private helper to make fetch requests with retry logic.
|
|
563
|
-
*/
|
|
564
|
-
async _fetchWithRetry(url, fetchOptions) {
|
|
565
|
-
let attempt = 0;
|
|
566
|
-
let delay = this.initialDelayMs;
|
|
567
|
-
const headers = {
|
|
568
|
-
...this.customHeaders
|
|
569
|
-
};
|
|
570
|
-
if (fetchOptions?.headers) {
|
|
571
|
-
if (fetchOptions.headers instanceof Headers) {
|
|
572
|
-
fetchOptions.headers.forEach((value, key) => {
|
|
573
|
-
headers[key] = value;
|
|
574
|
-
});
|
|
575
|
-
} else if (Array.isArray(fetchOptions.headers)) {
|
|
576
|
-
fetchOptions.headers.forEach(([key, value]) => {
|
|
577
|
-
headers[key] = value;
|
|
578
|
-
});
|
|
579
|
-
} else {
|
|
580
|
-
Object.assign(headers, fetchOptions.headers);
|
|
581
|
-
}
|
|
582
|
-
}
|
|
583
|
-
if (this.apiKey) {
|
|
584
|
-
headers["Authorization"] = `Bearer ${this.apiKey}`;
|
|
585
|
-
headers["X-API-Key"] = this.apiKey;
|
|
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;
|
|
616
|
-
}
|
|
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
|
-
}
|
|
623
|
-
}
|
|
624
|
-
throw new Error(
|
|
625
|
-
`Failed to fetch data from ${url} after ${this.maxRetries} attempts.`
|
|
626
|
-
);
|
|
627
|
-
}
|
|
628
|
-
/**
|
|
629
|
-
* Retrieves the numerical balance (in HBAR) for a given account ID.
|
|
630
|
-
* @param accountId The ID of the account.
|
|
631
|
-
* @returns A promise that resolves to the HBAR balance or null if an error occurs.
|
|
632
|
-
*/
|
|
633
|
-
async getAccountBalance(accountId) {
|
|
634
|
-
this.logger.info(`Getting balance for account ${accountId}`);
|
|
635
|
-
try {
|
|
636
|
-
const accountInfo = await this.requestAccount(accountId);
|
|
637
|
-
if (accountInfo && accountInfo.balance) {
|
|
638
|
-
const hbarBalance = accountInfo.balance.balance / 1e8;
|
|
639
|
-
return hbarBalance;
|
|
63
|
+
if (txBody.scheduleSign) {
|
|
64
|
+
const scheduleSign = this.parseScheduleSignFromProto(txBody.scheduleSign);
|
|
65
|
+
if (scheduleSign) {
|
|
66
|
+
return {
|
|
67
|
+
type: "SCHEDULESIGN",
|
|
68
|
+
humanReadableType: "Schedule Sign",
|
|
69
|
+
scheduleSign
|
|
70
|
+
};
|
|
640
71
|
}
|
|
641
|
-
this.logger.warn(
|
|
642
|
-
`Could not retrieve balance for account ${accountId} from account info.`
|
|
643
|
-
);
|
|
644
|
-
return null;
|
|
645
|
-
} catch (error) {
|
|
646
|
-
this.logger.error(
|
|
647
|
-
`Error fetching numerical balance for account ${accountId}: ${error.message}`
|
|
648
|
-
);
|
|
649
|
-
return null;
|
|
650
|
-
}
|
|
651
|
-
}
|
|
652
|
-
/**
|
|
653
|
-
* Retrieves messages for a given topic ID with optional filters.
|
|
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.
|
|
660
|
-
*/
|
|
661
|
-
async getTopicMessagesByFilter(topicId, options) {
|
|
662
|
-
this.logger.trace(
|
|
663
|
-
`Querying messages for topic ${topicId} with filters: ${JSON.stringify(
|
|
664
|
-
options
|
|
665
|
-
)}`
|
|
666
|
-
);
|
|
667
|
-
let nextUrl = `/api/v1/topics/${topicId}/messages`;
|
|
668
|
-
const params = new URLSearchParams();
|
|
669
|
-
if (options?.limit) {
|
|
670
|
-
params.append("limit", options.limit.toString());
|
|
671
|
-
}
|
|
672
|
-
if (options?.sequenceNumber) {
|
|
673
|
-
params.append("sequencenumber", options.sequenceNumber);
|
|
674
|
-
}
|
|
675
|
-
if (options?.startTime) {
|
|
676
|
-
params.append("timestamp", `gte:${options.startTime}`);
|
|
677
72
|
}
|
|
678
|
-
if (
|
|
679
|
-
|
|
680
|
-
|
|
681
|
-
if (options?.order) {
|
|
682
|
-
params.append("order", options.order);
|
|
683
|
-
}
|
|
684
|
-
const queryString = params.toString();
|
|
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
|
-
}
|
|
742
|
-
}
|
|
743
|
-
}
|
|
744
|
-
if (options?.limit && messages.length >= options.limit) break;
|
|
745
|
-
nextUrl = data.links?.next ? `${data.links.next}` : "";
|
|
746
|
-
}
|
|
747
|
-
return messages;
|
|
748
|
-
} catch (e) {
|
|
749
|
-
const error = e;
|
|
750
|
-
this.logger.error(
|
|
751
|
-
`Error querying filtered topic messages for ${topicId}: ${error.message}`
|
|
73
|
+
if (txBody.scheduleDelete) {
|
|
74
|
+
const scheduleDelete = this.parseScheduleDeleteFromProto(
|
|
75
|
+
txBody.scheduleDelete
|
|
752
76
|
);
|
|
753
|
-
|
|
754
|
-
|
|
755
|
-
|
|
756
|
-
|
|
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.
|
|
761
|
-
*/
|
|
762
|
-
async getAccountTokens(accountId, limit = 100) {
|
|
763
|
-
this.logger.info(`Getting tokens for account ${accountId}`);
|
|
764
|
-
let allTokens = [];
|
|
765
|
-
let endpoint = `/api/v1/accounts/${accountId}/tokens?limit=${limit}`;
|
|
766
|
-
try {
|
|
767
|
-
for (let i = 0; i < 10 && endpoint; i++) {
|
|
768
|
-
const response = await this._requestWithRetry(endpoint);
|
|
769
|
-
if (response && response.tokens) {
|
|
770
|
-
allTokens = allTokens.concat(response.tokens);
|
|
771
|
-
}
|
|
772
|
-
endpoint = response.links?.next || "";
|
|
773
|
-
if (!endpoint || limit && allTokens.length >= limit) {
|
|
774
|
-
if (limit && allTokens.length > limit) {
|
|
775
|
-
allTokens = allTokens.slice(0, limit);
|
|
776
|
-
}
|
|
777
|
-
break;
|
|
778
|
-
}
|
|
77
|
+
if (scheduleDelete) {
|
|
78
|
+
return {
|
|
79
|
+
type: "SCHEDULEDELETE",
|
|
80
|
+
humanReadableType: "Schedule Delete",
|
|
81
|
+
scheduleDelete
|
|
82
|
+
};
|
|
779
83
|
}
|
|
780
|
-
return allTokens;
|
|
781
|
-
} catch (error) {
|
|
782
|
-
this.logger.error(
|
|
783
|
-
`Error fetching tokens for account ${accountId}: ${error.message}`
|
|
784
|
-
);
|
|
785
|
-
return null;
|
|
786
84
|
}
|
|
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
|
-
async getAccountNfts(accountId, tokenId, limit = 100) {
|
|
813
|
-
this.logger.info(
|
|
814
|
-
`Getting NFTs for account ${accountId}${tokenId ? ` for token ${tokenId}` : ""}`
|
|
815
|
-
);
|
|
816
|
-
let allNfts = [];
|
|
817
|
-
let endpoint = `/api/v1/accounts/${accountId}/nfts?limit=${limit}`;
|
|
818
|
-
if (tokenId) {
|
|
819
|
-
endpoint += `&token.id=${tokenId}`;
|
|
820
|
-
}
|
|
821
|
-
try {
|
|
822
|
-
for (let i = 0; i < 10 && endpoint; i++) {
|
|
823
|
-
const response = await this._requestWithRetry(endpoint);
|
|
824
|
-
if (response && response.nfts) {
|
|
825
|
-
const nftsWithUri = response.nfts.map((nft) => {
|
|
826
|
-
let tokenUri = void 0;
|
|
827
|
-
if (nft.metadata) {
|
|
828
|
-
try {
|
|
829
|
-
if (this.isServerEnvironment) {
|
|
830
|
-
tokenUri = Buffer.from(nft.metadata, "base64").toString(
|
|
831
|
-
"utf-8"
|
|
832
|
-
);
|
|
833
|
-
} else {
|
|
834
|
-
tokenUri = new TextDecoder().decode(
|
|
835
|
-
Uint8Array.from(atob(nft.metadata), (c) => c.charCodeAt(0))
|
|
836
|
-
);
|
|
837
|
-
}
|
|
838
|
-
} catch (e) {
|
|
839
|
-
this.logger.warn(
|
|
840
|
-
`Failed to decode metadata for NFT ${nft.token_id} SN ${nft.serial_number}: ${e.message}`
|
|
841
|
-
);
|
|
842
|
-
}
|
|
843
|
-
}
|
|
844
|
-
return { ...nft, token_uri: tokenUri };
|
|
845
|
-
});
|
|
846
|
-
allNfts = allNfts.concat(nftsWithUri);
|
|
847
|
-
}
|
|
848
|
-
endpoint = response.links?.next || "";
|
|
849
|
-
if (!endpoint) break;
|
|
85
|
+
return {};
|
|
86
|
+
}
|
|
87
|
+
/**
|
|
88
|
+
* Extract schedule data from Transaction internal fields
|
|
89
|
+
* This handles cases where data is stored in Transaction object internals
|
|
90
|
+
*/
|
|
91
|
+
static parseFromTransactionInternals(transaction) {
|
|
92
|
+
try {
|
|
93
|
+
const tx = transaction;
|
|
94
|
+
if (hasTransactionType(transaction, "scheduleCreate")) {
|
|
95
|
+
const scheduleCreate = {
|
|
96
|
+
scheduledTransactionBody: Buffer.from(
|
|
97
|
+
tx._scheduledTransaction
|
|
98
|
+
).toString("base64"),
|
|
99
|
+
memo: tx._scheduleMemo,
|
|
100
|
+
adminKey: tx._adminKey ? parseKey(tx._adminKey) : void 0,
|
|
101
|
+
payerAccountId: tx._payerAccountId?.toString(),
|
|
102
|
+
expirationTime: tx._expirationTime?.toString(),
|
|
103
|
+
waitForExpiry: tx._waitForExpiry || false
|
|
104
|
+
};
|
|
105
|
+
return {
|
|
106
|
+
type: "SCHEDULECREATE",
|
|
107
|
+
humanReadableType: "Schedule Create",
|
|
108
|
+
scheduleCreate
|
|
109
|
+
};
|
|
850
110
|
}
|
|
851
|
-
|
|
852
|
-
|
|
853
|
-
|
|
854
|
-
|
|
855
|
-
|
|
856
|
-
|
|
857
|
-
|
|
858
|
-
|
|
859
|
-
|
|
860
|
-
* Validates NFT ownership by checking if a specific serial number of a token ID exists for an account.
|
|
861
|
-
* @param accountId The ID of the account.
|
|
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.
|
|
865
|
-
*/
|
|
866
|
-
async validateNFTOwnership(accountId, tokenId, serialNumber) {
|
|
867
|
-
this.logger.info(
|
|
868
|
-
`Validating ownership of NFT ${tokenId} SN ${serialNumber} for account ${accountId}`
|
|
869
|
-
);
|
|
870
|
-
try {
|
|
871
|
-
const nfts = await this.getAccountNfts(accountId, tokenId);
|
|
872
|
-
if (nfts) {
|
|
873
|
-
const foundNft = nfts.find(
|
|
874
|
-
(nft) => nft.token_id === tokenId && nft.serial_number === serialNumber
|
|
875
|
-
);
|
|
876
|
-
return foundNft || null;
|
|
111
|
+
if (hasTransactionType(transaction, "scheduleSign")) {
|
|
112
|
+
const scheduleSign = {
|
|
113
|
+
scheduleId: tx._scheduleId.toString()
|
|
114
|
+
};
|
|
115
|
+
return {
|
|
116
|
+
type: "SCHEDULESIGN",
|
|
117
|
+
humanReadableType: "Schedule Sign",
|
|
118
|
+
scheduleSign
|
|
119
|
+
};
|
|
877
120
|
}
|
|
878
|
-
|
|
879
|
-
|
|
880
|
-
|
|
881
|
-
|
|
882
|
-
|
|
883
|
-
|
|
884
|
-
|
|
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.
|
|
893
|
-
*/
|
|
894
|
-
async readSmartContractQuery(contractIdOrAddress, functionSelector, payerAccountId, options) {
|
|
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];
|
|
121
|
+
if (hasTransactionType(transaction, "scheduleDelete")) {
|
|
122
|
+
const scheduleDelete = {
|
|
123
|
+
scheduleId: tx._scheduleId.toString()
|
|
124
|
+
};
|
|
125
|
+
return {
|
|
126
|
+
type: "SCHEDULEDELETE",
|
|
127
|
+
humanReadableType: "Schedule Delete",
|
|
128
|
+
scheduleDelete
|
|
129
|
+
};
|
|
914
130
|
}
|
|
915
|
-
|
|
916
|
-
try {
|
|
917
|
-
const url = this.constructUrl("/api/v1/contracts/call");
|
|
918
|
-
const response = await this._fetchWithRetry(
|
|
919
|
-
url,
|
|
920
|
-
{
|
|
921
|
-
method: "POST",
|
|
922
|
-
body: JSON.stringify(body),
|
|
923
|
-
headers: {
|
|
924
|
-
"Content-Type": "application/json"
|
|
925
|
-
}
|
|
926
|
-
}
|
|
927
|
-
);
|
|
928
|
-
return response;
|
|
131
|
+
return {};
|
|
929
132
|
} catch (error) {
|
|
930
|
-
|
|
931
|
-
`Error reading smart contract ${contractIdOrAddress}: ${error.message}`
|
|
932
|
-
);
|
|
933
|
-
return null;
|
|
133
|
+
return {};
|
|
934
134
|
}
|
|
935
135
|
}
|
|
936
136
|
/**
|
|
937
|
-
*
|
|
938
|
-
* @param accountId The ID of the account that sent the airdrops.
|
|
939
|
-
* @param options Optional parameters for filtering airdrops.
|
|
940
|
-
* @returns A promise that resolves to an array of TokenAirdrop or null.
|
|
137
|
+
* Parse Schedule Create from protobuf data
|
|
941
138
|
*/
|
|
942
|
-
|
|
943
|
-
|
|
944
|
-
|
|
945
|
-
)
|
|
946
|
-
|
|
947
|
-
|
|
948
|
-
|
|
949
|
-
|
|
950
|
-
}
|
|
951
|
-
if (options?.order) {
|
|
952
|
-
params.append("order", options.order);
|
|
139
|
+
static parseScheduleCreateFromProto(body) {
|
|
140
|
+
if (!body) return void 0;
|
|
141
|
+
const data = {};
|
|
142
|
+
if (body.scheduledTransactionBody) {
|
|
143
|
+
const schedBytes = proto.SchedulableTransactionBody.encode(
|
|
144
|
+
proto.SchedulableTransactionBody.create(body.scheduledTransactionBody)
|
|
145
|
+
).finish();
|
|
146
|
+
data.scheduledTransactionBody = Buffer.from(schedBytes).toString("base64");
|
|
953
147
|
}
|
|
954
|
-
if (
|
|
955
|
-
|
|
148
|
+
if (body.memo) {
|
|
149
|
+
data.memo = body.memo;
|
|
956
150
|
}
|
|
957
|
-
if (
|
|
958
|
-
|
|
151
|
+
if (body.adminKey) {
|
|
152
|
+
data.adminKey = parseKey(body.adminKey);
|
|
959
153
|
}
|
|
960
|
-
if (
|
|
961
|
-
|
|
154
|
+
if (body.payerAccountID) {
|
|
155
|
+
data.payerAccountId = new AccountId(
|
|
156
|
+
body.payerAccountID.shardNum ?? 0,
|
|
157
|
+
body.payerAccountID.realmNum ?? 0,
|
|
158
|
+
body.payerAccountID.accountNum ?? 0
|
|
159
|
+
).toString();
|
|
962
160
|
}
|
|
963
|
-
|
|
964
|
-
|
|
965
|
-
|
|
161
|
+
if (body.expirationTime?.seconds) {
|
|
162
|
+
data.expirationTime = `${Long.fromValue(
|
|
163
|
+
body.expirationTime.seconds
|
|
164
|
+
).toString()}.${body.expirationTime.nanos ?? 0}`;
|
|
966
165
|
}
|
|
967
|
-
|
|
968
|
-
|
|
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;
|
|
166
|
+
if (body.waitForExpiry !== void 0) {
|
|
167
|
+
data.waitForExpiry = body.waitForExpiry;
|
|
975
168
|
}
|
|
169
|
+
return data;
|
|
976
170
|
}
|
|
977
171
|
/**
|
|
978
|
-
*
|
|
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.
|
|
172
|
+
* Parse Schedule Sign from protobuf data
|
|
982
173
|
*/
|
|
983
|
-
|
|
984
|
-
|
|
985
|
-
|
|
986
|
-
)
|
|
987
|
-
|
|
988
|
-
|
|
989
|
-
|
|
990
|
-
|
|
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;
|
|
174
|
+
static parseScheduleSignFromProto(body) {
|
|
175
|
+
if (!body) return void 0;
|
|
176
|
+
const data = {};
|
|
177
|
+
if (body.scheduleID) {
|
|
178
|
+
data.scheduleId = new ScheduleId(
|
|
179
|
+
body.scheduleID.shardNum ?? 0,
|
|
180
|
+
body.scheduleID.realmNum ?? 0,
|
|
181
|
+
body.scheduleID.scheduleNum ?? 0
|
|
182
|
+
).toString();
|
|
1016
183
|
}
|
|
184
|
+
return data;
|
|
1017
185
|
}
|
|
1018
186
|
/**
|
|
1019
|
-
*
|
|
1020
|
-
* @param options Optional parameters for filtering blocks.
|
|
1021
|
-
* @returns A promise that resolves to an array of Block or null.
|
|
187
|
+
* Parse Schedule Delete from protobuf data
|
|
1022
188
|
*/
|
|
1023
|
-
|
|
1024
|
-
|
|
1025
|
-
|
|
1026
|
-
|
|
1027
|
-
|
|
1028
|
-
|
|
1029
|
-
|
|
1030
|
-
|
|
1031
|
-
|
|
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;
|
|
189
|
+
static parseScheduleDeleteFromProto(body) {
|
|
190
|
+
if (!body) return void 0;
|
|
191
|
+
const data = {};
|
|
192
|
+
if (body.scheduleID) {
|
|
193
|
+
data.scheduleId = new ScheduleId(
|
|
194
|
+
body.scheduleID.shardNum ?? 0,
|
|
195
|
+
body.scheduleID.realmNum ?? 0,
|
|
196
|
+
body.scheduleID.scheduleNum ?? 0
|
|
197
|
+
).toString();
|
|
1049
198
|
}
|
|
199
|
+
return data;
|
|
1050
200
|
}
|
|
1051
201
|
/**
|
|
1052
|
-
*
|
|
1053
|
-
* @
|
|
1054
|
-
* @returns A promise that resolves to a Block or null.
|
|
202
|
+
* Legacy method: Parse schedule create transaction
|
|
203
|
+
* @deprecated Use parseScheduleTransaction instead
|
|
1055
204
|
*/
|
|
1056
|
-
|
|
1057
|
-
this.logger.info(`Getting block ${blockNumberOrHash}`);
|
|
205
|
+
static parseScheduleCreate(transaction) {
|
|
1058
206
|
try {
|
|
1059
|
-
const
|
|
1060
|
-
|
|
1061
|
-
);
|
|
1062
|
-
return response;
|
|
207
|
+
const result = this.parseScheduleTransaction(transaction);
|
|
208
|
+
return result.scheduleCreate || null;
|
|
1063
209
|
} catch (error) {
|
|
1064
|
-
|
|
1065
|
-
`Error fetching block ${blockNumberOrHash}: ${error.message}`
|
|
1066
|
-
);
|
|
210
|
+
console.warn("[ScheduleParser] Failed to parse schedule create:", error);
|
|
1067
211
|
return null;
|
|
1068
212
|
}
|
|
1069
213
|
}
|
|
1070
214
|
/**
|
|
1071
|
-
*
|
|
1072
|
-
* @
|
|
1073
|
-
* @returns A promise that resolves to an array of ContractEntity or null.
|
|
215
|
+
* Legacy method: Parse schedule sign transaction
|
|
216
|
+
* @deprecated Use parseScheduleTransaction instead
|
|
1074
217
|
*/
|
|
1075
|
-
|
|
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
|
-
}
|
|
218
|
+
static parseScheduleSign(transaction) {
|
|
1092
219
|
try {
|
|
1093
|
-
const
|
|
1094
|
-
return
|
|
220
|
+
const result = this.parseScheduleTransaction(transaction);
|
|
221
|
+
return result.scheduleSign || null;
|
|
1095
222
|
} catch (error) {
|
|
1096
|
-
|
|
223
|
+
console.warn("[ScheduleParser] Failed to parse schedule sign:", error);
|
|
1097
224
|
return null;
|
|
1098
225
|
}
|
|
1099
226
|
}
|
|
1100
227
|
/**
|
|
1101
|
-
*
|
|
1102
|
-
* @
|
|
1103
|
-
* @param timestamp Optional timestamp for historical data.
|
|
1104
|
-
* @returns A promise that resolves to a ContractEntity or null.
|
|
228
|
+
* Legacy method: Parse schedule delete transaction
|
|
229
|
+
* @deprecated Use parseScheduleTransaction instead
|
|
1105
230
|
*/
|
|
1106
|
-
|
|
1107
|
-
this.logger.info(`Getting contract ${contractIdOrAddress}`);
|
|
1108
|
-
let url = `/api/v1/contracts/${contractIdOrAddress}`;
|
|
1109
|
-
if (timestamp) {
|
|
1110
|
-
url += `?timestamp=${timestamp}`;
|
|
1111
|
-
}
|
|
231
|
+
static parseScheduleDelete(transaction) {
|
|
1112
232
|
try {
|
|
1113
|
-
const
|
|
1114
|
-
return
|
|
233
|
+
const result = this.parseScheduleTransaction(transaction);
|
|
234
|
+
return result.scheduleDelete || null;
|
|
1115
235
|
} catch (error) {
|
|
1116
|
-
|
|
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}`);
|
|
236
|
+
console.warn("[ScheduleParser] Failed to parse schedule delete:", error);
|
|
1164
237
|
return null;
|
|
1165
238
|
}
|
|
1166
239
|
}
|
|
1167
240
|
/**
|
|
1168
|
-
*
|
|
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.
|
|
241
|
+
* Parse schedule info from transaction body (for scheduled transactions)
|
|
1172
242
|
*/
|
|
1173
|
-
|
|
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
|
-
}
|
|
243
|
+
static extractScheduleInfo(transaction) {
|
|
1179
244
|
try {
|
|
1180
|
-
const
|
|
1181
|
-
|
|
1182
|
-
|
|
1183
|
-
|
|
1184
|
-
|
|
1185
|
-
|
|
1186
|
-
|
|
1187
|
-
|
|
1188
|
-
|
|
1189
|
-
|
|
1190
|
-
|
|
1191
|
-
|
|
1192
|
-
|
|
1193
|
-
|
|
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 || [];
|
|
245
|
+
const transactionBody = transaction._transactionBody;
|
|
246
|
+
if (transactionBody?.scheduleRef) {
|
|
247
|
+
return {
|
|
248
|
+
isScheduled: true,
|
|
249
|
+
scheduleRef: transactionBody.scheduleRef.toString()
|
|
250
|
+
};
|
|
251
|
+
}
|
|
252
|
+
const tx = transaction;
|
|
253
|
+
if (tx._scheduleId || tx.scheduleId) {
|
|
254
|
+
return {
|
|
255
|
+
isScheduled: true,
|
|
256
|
+
scheduleRef: (tx._scheduleId || tx.scheduleId)?.toString()
|
|
257
|
+
};
|
|
258
|
+
}
|
|
259
|
+
return { isScheduled: false };
|
|
1399
260
|
} catch (error) {
|
|
1400
|
-
|
|
1401
|
-
`Error fetching contract logs for ${contractIdOrAddress}: ${error.message}`
|
|
1402
|
-
);
|
|
1403
|
-
return null;
|
|
261
|
+
return { isScheduled: false };
|
|
1404
262
|
}
|
|
1405
263
|
}
|
|
1406
264
|
/**
|
|
1407
|
-
*
|
|
1408
|
-
*
|
|
1409
|
-
* @param serialNumber The serial number of the NFT.
|
|
1410
|
-
* @returns A promise that resolves to an NftInfo or null.
|
|
265
|
+
* Parse a scheduled transaction body to extract the inner transaction
|
|
266
|
+
* This is used when a schedule contains another transaction to be executed
|
|
1411
267
|
*/
|
|
1412
|
-
|
|
1413
|
-
this.logger.info(`Getting NFT info for ${tokenId}/${serialNumber}`);
|
|
1414
|
-
const url = `/api/v1/tokens/${tokenId}/nfts/${serialNumber}`;
|
|
268
|
+
static parseScheduledTransactionBody(scheduledTxBytes) {
|
|
1415
269
|
try {
|
|
1416
|
-
|
|
1417
|
-
|
|
1418
|
-
|
|
1419
|
-
|
|
1420
|
-
|
|
1421
|
-
|
|
270
|
+
let bytes;
|
|
271
|
+
if (scheduledTxBytes.startsWith("0x")) {
|
|
272
|
+
const hexString = scheduledTxBytes.slice(2);
|
|
273
|
+
bytes = new Uint8Array(Buffer.from(hexString, "hex"));
|
|
274
|
+
} else {
|
|
275
|
+
bytes = new Uint8Array(Buffer.from(scheduledTxBytes, "base64"));
|
|
276
|
+
}
|
|
277
|
+
const schedulableBody = proto.SchedulableTransactionBody.decode(bytes);
|
|
278
|
+
const txType = Object.keys(schedulableBody).find(
|
|
279
|
+
(key) => schedulableBody[key] !== null && key !== "transactionFee" && key !== "memo"
|
|
280
|
+
);
|
|
281
|
+
if (txType) {
|
|
282
|
+
return {
|
|
283
|
+
type: txType.toUpperCase(),
|
|
284
|
+
body: schedulableBody[txType],
|
|
285
|
+
memo: schedulableBody.memo,
|
|
286
|
+
transactionFee: schedulableBody.transactionFee?.toString()
|
|
287
|
+
};
|
|
288
|
+
}
|
|
1422
289
|
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
290
|
} catch (error) {
|
|
1455
|
-
|
|
1456
|
-
|
|
291
|
+
console.warn(
|
|
292
|
+
"[ScheduleParser] Failed to parse scheduled transaction body:",
|
|
293
|
+
error
|
|
1457
294
|
);
|
|
1458
295
|
return null;
|
|
1459
296
|
}
|
|
1460
297
|
}
|
|
1461
298
|
/**
|
|
1462
|
-
*
|
|
1463
|
-
*
|
|
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.
|
|
299
|
+
* Parse Schedule Service transaction from Transaction object
|
|
300
|
+
* This is the unified entry point that delegates to the comprehensive parsing logic
|
|
1499
301
|
*/
|
|
1500
|
-
|
|
1501
|
-
this.
|
|
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;
|
|
1559
|
-
} catch (error) {
|
|
1560
|
-
this.logger.error(
|
|
1561
|
-
`Error fetching opcode traces for ${transactionIdOrHash}: ${error.message}`
|
|
1562
|
-
);
|
|
1563
|
-
return null;
|
|
1564
|
-
}
|
|
302
|
+
static parseFromTransactionObject(transaction) {
|
|
303
|
+
return this.parseScheduleTransaction(transaction);
|
|
1565
304
|
}
|
|
1566
305
|
}
|
|
1567
306
|
export {
|
|
1568
|
-
|
|
307
|
+
ScheduleParser
|
|
1569
308
|
};
|
|
1570
309
|
//# sourceMappingURL=standards-sdk.es139.js.map
|