@rosen-bridge/minimum-fee 0.1.13 → 1.0.1

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.
Files changed (37) hide show
  1. package/README.md +28 -0
  2. package/dist/lib/MinimumFeeBox.d.ts +56 -0
  3. package/dist/lib/MinimumFeeBox.d.ts.map +1 -0
  4. package/dist/lib/MinimumFeeBox.js +240 -0
  5. package/dist/lib/MinimumFeeBoxBuilder.d.ts +47 -0
  6. package/dist/lib/MinimumFeeBoxBuilder.d.ts.map +1 -0
  7. package/dist/lib/MinimumFeeBoxBuilder.js +149 -0
  8. package/dist/lib/MinimumFeeConfig.d.ts +22 -0
  9. package/dist/lib/MinimumFeeConfig.d.ts.map +1 -0
  10. package/dist/lib/MinimumFeeConfig.js +39 -0
  11. package/dist/lib/constants.d.ts +2 -0
  12. package/dist/lib/constants.d.ts.map +1 -0
  13. package/dist/lib/constants.js +2 -0
  14. package/dist/lib/errors.d.ts +16 -0
  15. package/dist/lib/errors.d.ts.map +1 -0
  16. package/dist/lib/errors.js +26 -0
  17. package/dist/lib/handleApiError.d.ts +18 -0
  18. package/dist/lib/handleApiError.d.ts.map +1 -0
  19. package/dist/lib/handleApiError.js +35 -0
  20. package/dist/lib/index.d.ts +5 -2
  21. package/dist/lib/index.d.ts.map +1 -1
  22. package/dist/lib/index.js +6 -2
  23. package/dist/lib/types.d.ts +21 -14
  24. package/dist/lib/types.d.ts.map +1 -1
  25. package/dist/lib/types.js +21 -2
  26. package/dist/tsconfig.build.tsbuildinfo +1 -0
  27. package/package.json +14 -9
  28. package/dist/lib/BridgeMinimumFee.d.ts +0 -27
  29. package/dist/lib/BridgeMinimumFee.d.ts.map +0 -1
  30. package/dist/lib/BridgeMinimumFee.js +0 -111
  31. package/dist/lib/consts.d.ts +0 -5
  32. package/dist/lib/consts.d.ts.map +0 -1
  33. package/dist/lib/consts.js +0 -5
  34. package/dist/lib/parser.d.ts +0 -12
  35. package/dist/lib/parser.d.ts.map +0 -1
  36. package/dist/lib/parser.js +0 -40
  37. package/dist/tsconfig.tsbuildinfo +0 -1
package/README.md ADDED
@@ -0,0 +1,28 @@
1
+ # @rosen-bridge/minimum-fee
2
+
3
+ ## Table of contents
4
+
5
+ - [Introduction](#introduction)
6
+ - [Installation](#installation)
7
+
8
+ ## Introduction
9
+
10
+ `@rosen-bridge/minimum-fee` Typescript package to build and get minimum fee of the bridge for supported tokens from blockchain
11
+
12
+ ## Installation
13
+
14
+ npm:
15
+
16
+ ```sh
17
+ npm i @rosen-bridge/minimum-fee
18
+ ```
19
+
20
+ yarn:
21
+
22
+ ```sh
23
+ yarn add @rosen-bridge/minimum-fee
24
+ ```
25
+
26
+ #### Examples
27
+
28
+ **TODO**
@@ -0,0 +1,56 @@
1
+ import { ErgoBox } from 'ergo-lib-wasm-nodejs';
2
+ import { ChainMinimumFee, ErgoNetworkType, Fee } from './types';
3
+ import ergoExplorerClientFactory from '@rosen-clients/ergo-explorer';
4
+ import ergoNodeClientFactory from '@rosen-clients/ergo-node';
5
+ import { AbstractLogger } from '@rosen-bridge/abstract-logger';
6
+ import { MinimumFeeBoxBuilder } from './MinimumFeeBoxBuilder';
7
+ export declare class MinimumFeeBox {
8
+ protected readonly BOX_FETCHING_PAGE_SIZE = 50;
9
+ protected logger: AbstractLogger;
10
+ protected box: ErgoBox;
11
+ protected tokenId: string;
12
+ protected minimumFeeNFT: string;
13
+ protected address: string;
14
+ protected ergoTree: string;
15
+ protected explorerClient: ReturnType<typeof ergoExplorerClientFactory>;
16
+ protected nodeClient: ReturnType<typeof ergoNodeClientFactory>;
17
+ constructor(tokenId: string, minimumFeeNFT: string, address: string, networkType: ErgoNetworkType, networkUrl: string, logger?: AbstractLogger);
18
+ /**
19
+ * fetches the box from the blockchain
20
+ */
21
+ fetchBox: () => Promise<void>;
22
+ /**
23
+ * returns fetched box or throws approprite error if found more or none
24
+ * @param eligibleBoxes
25
+ */
26
+ protected selectEligibleBox: (eligibleBoxes: Array<ErgoBox>) => ErgoBox;
27
+ /**
28
+ * fetches box from the blockchain using explorer client
29
+ */
30
+ protected fetchBoxesUsingExplorer: () => Promise<Array<ErgoBox>>;
31
+ /**
32
+ * fetches the box from the blockchain using node client
33
+ */
34
+ protected fetchBoxesUsingNode: () => Promise<Array<ErgoBox>>;
35
+ /**
36
+ * returns fetched config box
37
+ */
38
+ getBox: () => ErgoBox;
39
+ /**
40
+ * gets corresponding config for two chains and height
41
+ * @param fromChain
42
+ * @param height blockchain height for fromChain
43
+ * @param toChain
44
+ */
45
+ getFee: (fromChain: string, height: number, toChain: string) => ChainMinimumFee;
46
+ /**
47
+ * extracts Fee config from box registers
48
+ */
49
+ protected extractFeeFromBox: () => Array<Fee>;
50
+ /**
51
+ * generates a MinimumFeeBoxBuilder using current box
52
+ * note that 'height' parameter of builder won't be set
53
+ */
54
+ toBuilder: () => MinimumFeeBoxBuilder;
55
+ }
56
+ //# sourceMappingURL=MinimumFeeBox.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"MinimumFeeBox.d.ts","sourceRoot":"","sources":["../../lib/MinimumFeeBox.ts"],"names":[],"mappings":"AAAA,OAAO,EAAW,OAAO,EAAE,MAAM,sBAAsB,CAAC;AACxD,OAAO,EAAE,eAAe,EAAE,eAAe,EAAE,GAAG,EAAE,MAAM,SAAS,CAAC;AAEhE,OAAO,yBAAyB,MAAM,8BAA8B,CAAC;AACrE,OAAO,qBAAqB,MAAM,0BAA0B,CAAC;AAG7D,OAAO,EAAE,cAAc,EAAe,MAAM,+BAA+B,CAAC;AAC5E,OAAO,EAAE,oBAAoB,EAAE,MAAM,wBAAwB,CAAC;AAI9D,qBAAa,aAAa;IACxB,SAAS,CAAC,QAAQ,CAAC,sBAAsB,MAAM;IAC/C,SAAS,CAAC,MAAM,EAAE,cAAc,CAAC;IACjC,SAAS,CAAC,GAAG,EAAE,OAAO,CAAC;IACvB,SAAS,CAAC,OAAO,EAAE,MAAM,CAAC;IAC1B,SAAS,CAAC,aAAa,EAAE,MAAM,CAAC;IAChC,SAAS,CAAC,OAAO,EAAE,MAAM,CAAC;IAC1B,SAAS,CAAC,QAAQ,EAAE,MAAM,CAAC;IAC3B,SAAS,CAAC,cAAc,EAAE,UAAU,CAAC,OAAO,yBAAyB,CAAC,CAAC;IACvE,SAAS,CAAC,UAAU,EAAE,UAAU,CAAC,OAAO,qBAAqB,CAAC,CAAC;gBAG7D,OAAO,EAAE,MAAM,EACf,aAAa,EAAE,MAAM,EACrB,OAAO,EAAE,MAAM,EACf,WAAW,EAAE,eAAe,EAC5B,UAAU,EAAE,MAAM,EAClB,MAAM,CAAC,EAAE,cAAc;IAczB;;OAEG;IACH,QAAQ,QAAa,QAAQ,IAAI,CAAC,CA2BhC;IAEF;;;OAGG;IACH,SAAS,CAAC,iBAAiB,kBAAmB,MAAM,OAAO,CAAC,KAAG,OAAO,CAyBpE;IAEF;;OAEG;IACH,SAAS,CAAC,uBAAuB,QAAa,QAAQ,MAAM,OAAO,CAAC,CAAC,CA8CnE;IAEF;;OAEG;IACH,SAAS,CAAC,mBAAmB,QAAa,QAAQ,MAAM,OAAO,CAAC,CAAC,CAiD/D;IAEF;;OAEG;IACH,MAAM,QAAO,OAAO,CAAa;IAEjC;;;;;OAKG;IACH,MAAM,cACO,MAAM,UACT,MAAM,WACL,MAAM,KACd,eAAe,CA4BhB;IAEF;;OAEG;IACH,SAAS,CAAC,iBAAiB,QAAO,MAAM,GAAG,CAAC,CAuD1C;IAEF;;;OAGG;IACH,SAAS,QAAO,oBAAoB,CAmBlC;CACH"}
@@ -0,0 +1,240 @@
1
+ import { Address, ErgoBox } from 'ergo-lib-wasm-nodejs';
2
+ import { ChainMinimumFee, ErgoNetworkType } from './types';
3
+ import { FailedError, NotFoundError } from './errors';
4
+ import ergoExplorerClientFactory from '@rosen-clients/ergo-explorer';
5
+ import ergoNodeClientFactory from '@rosen-clients/ergo-node';
6
+ import JsonBigInt from '@rosen-bridge/json-bigint';
7
+ import handleApiError from './handleApiError';
8
+ import { DummyLogger } from '@rosen-bridge/abstract-logger';
9
+ import { MinimumFeeBoxBuilder } from './MinimumFeeBoxBuilder';
10
+ import { MinimumFeeConfig } from './MinimumFeeConfig';
11
+ import { ERGO_NATIVE_TOKEN } from './constants';
12
+ export class MinimumFeeBox {
13
+ BOX_FETCHING_PAGE_SIZE = 50;
14
+ logger;
15
+ box;
16
+ tokenId;
17
+ minimumFeeNFT;
18
+ address;
19
+ ergoTree;
20
+ explorerClient;
21
+ nodeClient;
22
+ constructor(tokenId, minimumFeeNFT, address, networkType, networkUrl, logger) {
23
+ this.tokenId = tokenId;
24
+ this.minimumFeeNFT = minimumFeeNFT;
25
+ this.address = address;
26
+ this.ergoTree = Address.from_base58(this.address)
27
+ .to_ergo_tree()
28
+ .to_base16_bytes();
29
+ if (networkType === ErgoNetworkType.explorer)
30
+ this.explorerClient = ergoExplorerClientFactory(networkUrl);
31
+ else
32
+ this.nodeClient = ergoNodeClientFactory(networkUrl);
33
+ this.logger = logger ? logger : new DummyLogger();
34
+ }
35
+ /**
36
+ * fetches the box from the blockchain
37
+ */
38
+ fetchBox = async () => {
39
+ const boxes = this.explorerClient
40
+ ? await this.fetchBoxesUsingExplorer()
41
+ : await this.fetchBoxesUsingNode();
42
+ const boxHasCorrectAddress = (box) => box.ergo_tree().to_base16_bytes() === this.ergoTree;
43
+ const boxHasAppropriateTokens = (box) => {
44
+ const tokenLen = box.tokens().len();
45
+ let hasMinimumFeeNFT = false;
46
+ let hasTargetToken = this.tokenId === ERGO_NATIVE_TOKEN ? true : false;
47
+ const hasCorrectTokens = this.tokenId === ERGO_NATIVE_TOKEN ? tokenLen === 1 : tokenLen === 2;
48
+ for (let i = 0; i < tokenLen; i++) {
49
+ const id = box.tokens().get(i).id().to_str();
50
+ if (id === this.minimumFeeNFT)
51
+ hasMinimumFeeNFT = true;
52
+ else if (id === this.tokenId)
53
+ hasTargetToken = true;
54
+ }
55
+ return hasCorrectTokens && hasMinimumFeeNFT && hasTargetToken;
56
+ };
57
+ this.box = this.selectEligibleBox(boxes.filter((box) => boxHasCorrectAddress(box) && boxHasAppropriateTokens(box)));
58
+ };
59
+ /**
60
+ * returns fetched box or throws approprite error if found more or none
61
+ * @param eligibleBoxes
62
+ */
63
+ selectEligibleBox = (eligibleBoxes) => {
64
+ this.logger.debug(`Found [${eligibleBoxes.length}] minimum-fee boxes: ${JsonBigInt.stringify(eligibleBoxes.map((box) => box.to_json()))}`);
65
+ if (eligibleBoxes.length === 0)
66
+ throw new NotFoundError(`Found no minimum-fee box for token [${this.tokenId}] and address [${this.address}]`);
67
+ else if (eligibleBoxes.length > 1)
68
+ throw new FailedError(`Found [${eligibleBoxes.length}] minimum-fee boxes for token [${this.tokenId}] and address [${this.address}]`);
69
+ else {
70
+ this.logger.debug(`Found minimum-fee box [${eligibleBoxes[0]
71
+ .box_id()
72
+ .to_str()}] for token [${this.tokenId}] and address [${this.address}]`);
73
+ return eligibleBoxes[0];
74
+ }
75
+ };
76
+ /**
77
+ * fetches box from the blockchain using explorer client
78
+ */
79
+ fetchBoxesUsingExplorer = async () => {
80
+ const boxes = [];
81
+ try {
82
+ let currentPage = 0;
83
+ let boxesPage = await this.explorerClient.v1.getApiV1BoxesUnspentByaddressP1(this.address, {
84
+ offset: currentPage * this.BOX_FETCHING_PAGE_SIZE,
85
+ limit: this.BOX_FETCHING_PAGE_SIZE,
86
+ });
87
+ this.logger.debug(`requested 'explorerClient.getApiV1BoxesUnspentByaddressP1' for address [${this.address}]. res: ${JsonBigInt.stringify(boxesPage)}`);
88
+ while (boxesPage.items?.length) {
89
+ boxes.push(...boxesPage.items.map((box) => ErgoBox.from_json(JsonBigInt.stringify(box))));
90
+ currentPage++;
91
+ boxesPage =
92
+ await this.explorerClient.v1.getApiV1BoxesUnspentByaddressP1(this.address, {
93
+ offset: currentPage * this.BOX_FETCHING_PAGE_SIZE,
94
+ limit: this.BOX_FETCHING_PAGE_SIZE,
95
+ });
96
+ this.logger.debug(`requested 'explorerClient.getApiV1BoxesUnspentByaddressP1' for address [${this.address}]. res: ${JsonBigInt.stringify(boxesPage)}`);
97
+ }
98
+ }
99
+ catch (error) {
100
+ return handleApiError(error, 'Failed to get boxes by token id from Ergo Explorer:');
101
+ }
102
+ return boxes;
103
+ };
104
+ /**
105
+ * fetches the box from the blockchain using node client
106
+ */
107
+ fetchBoxesUsingNode = async () => {
108
+ const boxes = [];
109
+ try {
110
+ let currentPage = 0;
111
+ let boxesPage = await this.nodeClient.getBoxesByAddressUnspent(this.address, {
112
+ offset: currentPage * this.BOX_FETCHING_PAGE_SIZE,
113
+ limit: this.BOX_FETCHING_PAGE_SIZE,
114
+ });
115
+ this.logger.debug(`requested 'nodeClient.getBoxesByAddressUnspent' for address [${this.address}]. res: ${JsonBigInt.stringify(boxesPage)}`);
116
+ while (boxesPage.length !== 0) {
117
+ boxes.push(...boxesPage.map((box) => ErgoBox.from_json(JsonBigInt.stringify(box))));
118
+ currentPage++;
119
+ boxesPage = await this.nodeClient.getBoxesByAddressUnspent(this.address, {
120
+ offset: currentPage * this.BOX_FETCHING_PAGE_SIZE,
121
+ limit: this.BOX_FETCHING_PAGE_SIZE,
122
+ });
123
+ this.logger.debug(`requested 'nodeClient.getBoxesByAddressUnspent' for address [${this.address}]. res: ${JsonBigInt.stringify(boxesPage)}`);
124
+ }
125
+ }
126
+ catch (error) {
127
+ const baseError = 'Failed to get boxes by token id from Ergo Node:';
128
+ handleApiError(error, baseError, {
129
+ handleRespondedState: (error) => {
130
+ if (error.response.status === 400)
131
+ return;
132
+ throw new FailedError(`${baseError} [${error.response.status}] ${error.response.data.reason}`);
133
+ },
134
+ });
135
+ }
136
+ return boxes;
137
+ };
138
+ /**
139
+ * returns fetched config box
140
+ */
141
+ getBox = () => this.box;
142
+ /**
143
+ * gets corresponding config for two chains and height
144
+ * @param fromChain
145
+ * @param height blockchain height for fromChain
146
+ * @param toChain
147
+ */
148
+ getFee = (fromChain, height, toChain) => {
149
+ if (!this.box)
150
+ throw Error(`Box is not fetched yet`);
151
+ const fees = this.extractFeeFromBox().reverse();
152
+ for (const fee of fees) {
153
+ if (!Object.hasOwn(fee.heights, fromChain))
154
+ throw new NotFoundError(`No fee found for chain [${fromChain}] in box [${this.box
155
+ .box_id()
156
+ .to_str()}]`);
157
+ if (fee.heights[fromChain] < height) {
158
+ const chainFee = fee.configs[toChain];
159
+ if (chainFee)
160
+ return new ChainMinimumFee(chainFee);
161
+ else
162
+ throw new Error(`Chain [${toChain}] is not supported at given height of fromChain [${height} of ${fromChain}] in box [${this.box
163
+ .box_id()
164
+ .to_str()}]`);
165
+ }
166
+ }
167
+ throw new NotFoundError(`Config does not support height [${height}] for chain [${fromChain}] in box [${this.box
168
+ .box_id()
169
+ .to_str()}]`);
170
+ };
171
+ /**
172
+ * extracts Fee config from box registers
173
+ */
174
+ extractFeeFromBox = () => {
175
+ const R4 = this.box.register_value(4);
176
+ const R5 = this.box.register_value(5);
177
+ const R6 = this.box.register_value(6);
178
+ const R7 = this.box.register_value(7);
179
+ const R8 = this.box.register_value(8);
180
+ const R9 = this.box.register_value(9);
181
+ if (!R4 || !R5 || !R6 || !R7 || !R8 || !R9)
182
+ throw Error(`Incomplete register data for minimum-fee config box [${this.box
183
+ .box_id()
184
+ .to_str()}]`);
185
+ const fees = [];
186
+ const chains = R4.to_coll_coll_byte().map((element) => Buffer.from(element).toString());
187
+ const heights = R5.to_js();
188
+ const bridgeFees = R6.to_js();
189
+ const networkFees = R7.to_js();
190
+ const rsnRatios = R8.to_js();
191
+ const feeRatios = R9.to_js();
192
+ for (let feeIdx = 0; feeIdx < heights.length; feeIdx++) {
193
+ const fee = {
194
+ heights: {},
195
+ configs: {},
196
+ };
197
+ for (let chainIdx = 0; chainIdx < chains.length; chainIdx++) {
198
+ const chain = chains[chainIdx];
199
+ if (heights[feeIdx][chainIdx] === -1)
200
+ continue;
201
+ fee.heights[chain] = heights[feeIdx][chainIdx];
202
+ if (bridgeFees[feeIdx][chainIdx] === '-1')
203
+ continue;
204
+ fee.configs[chain] = {
205
+ bridgeFee: BigInt(bridgeFees[feeIdx][chainIdx]),
206
+ networkFee: BigInt(networkFees[feeIdx][chainIdx]),
207
+ rsnRatio: BigInt(rsnRatios[feeIdx][chainIdx][0]),
208
+ rsnRatioDivisor: BigInt(rsnRatios[feeIdx][chainIdx][1]),
209
+ feeRatio: BigInt(feeRatios[feeIdx][chainIdx]),
210
+ };
211
+ }
212
+ fees.push(fee);
213
+ }
214
+ this.logger.debug(`Extracted fee config from box [${this.box
215
+ .box_id()
216
+ .to_str()}]: ${JsonBigInt.stringify(fees)}`);
217
+ return fees;
218
+ };
219
+ /**
220
+ * generates a MinimumFeeBoxBuilder using current box
221
+ * note that 'height' parameter of builder won't be set
222
+ */
223
+ toBuilder = () => {
224
+ const builder = new MinimumFeeBoxBuilder(this.minimumFeeNFT, this.address)
225
+ .setValue(BigInt(this.box.value().as_i64().to_str()))
226
+ .setToken(this.tokenId);
227
+ this.extractFeeFromBox().forEach((fee) => {
228
+ const chainFee = new MinimumFeeConfig();
229
+ Object.keys(fee.heights).forEach((chain) => {
230
+ if (Object.hasOwn(fee.configs, chain))
231
+ chainFee.setChainConfig(chain, fee.heights[chain], fee.configs[chain]);
232
+ else
233
+ chainFee.setChainConfig(chain, fee.heights[chain], undefined);
234
+ });
235
+ builder.addConfig(chainFee);
236
+ });
237
+ return builder;
238
+ };
239
+ }
240
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiTWluaW11bUZlZUJveC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL2xpYi9NaW5pbXVtRmVlQm94LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxPQUFPLEVBQUUsT0FBTyxFQUFFLE1BQU0sc0JBQXNCLENBQUM7QUFDeEQsT0FBTyxFQUFFLGVBQWUsRUFBRSxlQUFlLEVBQU8sTUFBTSxTQUFTLENBQUM7QUFDaEUsT0FBTyxFQUFFLFdBQVcsRUFBRSxhQUFhLEVBQUUsTUFBTSxVQUFVLENBQUM7QUFDdEQsT0FBTyx5QkFBeUIsTUFBTSw4QkFBOEIsQ0FBQztBQUNyRSxPQUFPLHFCQUFxQixNQUFNLDBCQUEwQixDQUFDO0FBQzdELE9BQU8sVUFBVSxNQUFNLDJCQUEyQixDQUFDO0FBQ25ELE9BQU8sY0FBYyxNQUFNLGtCQUFrQixDQUFDO0FBQzlDLE9BQU8sRUFBa0IsV0FBVyxFQUFFLE1BQU0sK0JBQStCLENBQUM7QUFDNUUsT0FBTyxFQUFFLG9CQUFvQixFQUFFLE1BQU0sd0JBQXdCLENBQUM7QUFDOUQsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sb0JBQW9CLENBQUM7QUFDdEQsT0FBTyxFQUFFLGlCQUFpQixFQUFFLE1BQU0sYUFBYSxDQUFDO0FBRWhELE1BQU0sT0FBTyxhQUFhO0lBQ0wsc0JBQXNCLEdBQUcsRUFBRSxDQUFDO0lBQ3JDLE1BQU0sQ0FBaUI7SUFDdkIsR0FBRyxDQUFVO0lBQ2IsT0FBTyxDQUFTO0lBQ2hCLGFBQWEsQ0FBUztJQUN0QixPQUFPLENBQVM7SUFDaEIsUUFBUSxDQUFTO0lBQ2pCLGNBQWMsQ0FBK0M7SUFDN0QsVUFBVSxDQUEyQztJQUUvRCxZQUNFLE9BQWUsRUFDZixhQUFxQixFQUNyQixPQUFlLEVBQ2YsV0FBNEIsRUFDNUIsVUFBa0IsRUFDbEIsTUFBdUI7UUFFdkIsSUFBSSxDQUFDLE9BQU8sR0FBRyxPQUFPLENBQUM7UUFDdkIsSUFBSSxDQUFDLGFBQWEsR0FBRyxh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QUFDLE1BQU0sQ0FBQyxHQUFHLENBQUMsT0FBTyxFQUFFLEtBQUssQ0FBQztvQkFDbkMsUUFBUSxDQUFDLGNBQWMsQ0FDckIsS0FBSyxFQUNMLEdBQUcsQ0FBQyxPQUFPLENBQUMsS0FBSyxDQUFDLEVBQ2xCLEdBQUcsQ0FBQyxPQUFPLENBQUMsS0FBSyxDQUFDLENBQ25CLENBQUM7O29CQUNDLFFBQVEsQ0FBQyxjQUFjLENBQUMsS0FBSyxFQUFFLEdBQUcsQ0FBQyxPQUFPLENBQUMsS0FBSyxDQUFDLEVBQUUsU0FBUyxDQUFDLENBQUM7WUFDckUsQ0FBQyxDQUFDLENBQUM7WUFDSCxPQUFPLENBQUMsU0FBUyxDQUFDLFFBQVEsQ0FBQyxDQUFDO1FBQzlCLENBQUMsQ0FBQyxDQUFDO1FBQ0gsT0FBTyxPQUFPLENBQUM7SUFDakIsQ0FBQyxDQUFDO0NBQ0giLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBBZGRyZXNzLCBFcmdvQm94IH0gZnJvbSAnZXJnby1saWItd2FzbS1ub2RlanMnO1xuaW1wb3J0IHsgQ2hhaW5NaW5pbXVtRmVlLCBFcmdvTmV0d29ya1R5cGUsIEZlZSB9IGZyb20gJy4vdHlwZXMnO1xuaW1wb3J0IHsgRmFpbGVkRXJyb3IsIE5vdEZvdW5kRXJyb3IgfSBmcm9tICcuL2Vycm9ycyc7XG5pbXBvcnQgZXJnb0V4cGxvcmVyQ2xpZW50RmFjdG9yeSBmcm9tICdAcm9zZW4tY2xpZW50cy9lcmdvLWV4cGxvcmVyJztcbmltcG9ydCBlcmdvTm9kZUNsaWVudEZhY3RvcnkgZnJvbSAnQHJvc2VuLWNsaWVudHMvZXJnby1ub2RlJztcbmltcG9ydCBKc29uQmlnSW50IGZyb20gJ0Byb3Nlbi1icmlkZ2UvanNvbi1iaWdpbnQnO1xuaW1wb3J0IGhhbmRsZUFwaUVycm9yIGZyb20gJy4vaGFuZGxlQXBpRXJyb3InO1xuaW1wb3J0IHsgQWJzdHJhY3RMb2dnZXIsIER1bW15TG9nZ2VyIH0gZnJvbSAnQHJvc2VuLWJyaWRnZS9hYnN0cmFjdC1sb2dnZXInO1xuaW1wb3J0IHsgTWluaW11bUZlZUJveEJ1aWxkZXIgfSBmcm9tICcuL01pbmltdW1GZWVCb3hCdWlsZGVyJztcbmltcG9ydCB7IE1pbmltdW1GZWVDb25maWcgfSBmcm9tICcuL01pbmltdW1GZWVDb25maWcnO1xuaW1wb3J0IHsgRVJHT19OQVRJVkVfVE9LRU4gfSBmcm9tICcuL2NvbnN0YW50cyc7XG5cbmV4cG9ydCBjbGFzcyBNaW5pbXVtRmVlQm94IHtcbiAgcHJvdGVjdGVkIHJlYWRvbmx5IEJPWF9GRVRDSElOR19QQUdFX1NJWkUgPSA1MDtcbiAgcHJvdGVjdGVkIGxvZ2dlcjogQWJzdHJhY3RMb2dnZXI7XG4gIHByb3RlY3RlZCBib3g6IEVyZ29Cb3g7XG4gIHByb3RlY3RlZCB0b2tlbklkOiBzdHJpbmc7XG4gIHByb3RlY3RlZCBtaW5pbXVtRmVlTkZUOiBzdHJpbmc7XG4gIHByb3RlY3RlZCBhZGRyZXNzOiBzdHJpbmc7XG4gIHByb3RlY3RlZCBlcmdvVHJlZTogc3RyaW5nO1xuICBwcm90ZWN0ZWQgZXhwbG9yZXJDbGllbnQ6IFJldHVyblR5cGU8dHlwZW9mIGVyZ29FeHBsb3JlckNsaWVudEZhY3Rvcnk+O1xuICBwcm90ZWN0ZWQgbm9kZUNsaWVudDogUmV0dXJuVHlwZTx0eXBlb2YgZXJnb05vZGVDbGllbnRGYWN0b3J5PjtcblxuICBjb25zdHJ1Y3RvcihcbiAgICB0b2tlbklkOiBzdHJpbmcsXG4gICAgbWluaW11bUZlZU5GVDogc3RyaW5nLFxuICAgIGFkZHJlc3M6IHN0cmluZyxcbiAgICBuZXR3b3JrVHlwZTogRXJnb05ldHdvcmtUeXBlLFxuICAgIG5ldHdvcmtVcmw6IHN0cmluZyxcbiAgICBsb2dnZXI/OiBBYnN0cmFjdExvZ2dlclxuICApIHtcbiAgICB0aGlzLnRva2VuSWQgPSB0b2tlbklkO1xuICAgIHRoaXMubWluaW11bUZlZU5GVCA9IG1pbmltdW1GZWVORlQ7XG4gICAgdGhpcy5hZGRyZXNzID0gYWRkcmVzcztcbiAgICB0aGlzLmVyZ29UcmVlID0gQWRkcmVzcy5mcm9tX2Jhc2U1OCh0aGlzLmFkZHJlc3MpXG4gICAgICAudG9fZXJnb190cmVlKClcbiAgICAgIC50b19iYXNlMTZfYnl0ZXMoKTtcbiAgICBpZiAobmV0d29ya1R5cGUgPT09IEVyZ29OZXR3b3JrVHlwZS5leHBsb3JlcilcbiAgICAgIHRoaXMuZXhwbG9yZXJDbGllbnQgPSBlcmdvRXhwbG9yZXJDbGllbnRGYWN0b3J5KG5ldHdvcmtVcmwpO1xuICAgIGVsc2UgdGhpcy5ub2RlQ2xpZW50ID0gZXJnb05vZGVDbGllbnRGYWN0b3J5KG5ldHdvcmtVcmwpO1xuICAgIHRoaXMubG9nZ2VyID0gbG9nZ2VyID8gbG9nZ2VyIDogbmV3IER1bW15TG9nZ2VyKCk7XG4gIH1cblxuICAvKipcbiAgICogZmV0Y2hlcyB0aGUgYm94IGZyb20gdGhlIGJsb2NrY2hhaW5cbiAgICovXG4gIGZldGNoQm94ID0gYXN5bmMgKCk6IFByb21pc2U8dm9pZD4gPT4ge1xuICAgIGNvbnN0IGJveGVzID0gdGhpcy5leHBsb3JlckNsaWVudFxuICAgICAgPyBhd2FpdCB0aGlzLmZldGNoQm94ZXNVc2luZ0V4cGxvcmVyKClcbiAgICAgIDogYXdhaXQgdGhpcy5mZXRjaEJveGVzVXNpbmdOb2RlKCk7XG5cbiAgICBjb25zdCBib3hIYXNDb3JyZWN0QWRkcmVzcyA9IChib3g6IEVyZ29Cb3gpID0+XG4gICAgICBib3guZXJnb190cmVlKCkudG9fYmFzZTE2X2J5dGVzKCkgPT09IHRoaXMuZXJnb1RyZWU7XG4gICAgY29uc3QgYm94SGFzQXBwcm9wcmlhdGVUb2tlbnMgPSAoYm94OiBFcmdvQm94KSA9PiB7XG4gICAgICBjb25zdCB0b2tlbkxlbiA9IGJveC50b2tlbnMoKS5sZW4oKTtcbiAgICAgIGxldCBoYXNNaW5pbXVtRmVlTkZUID0gZmFsc2U7XG4gICAgICBsZXQgaGFzVGFyZ2V0VG9rZW4gPSB0aGlzLnRva2VuSWQgPT09IEVSR09fTkFUSVZFX1RPS0VOID8gdHJ1ZSA6IGZhbHNlO1xuICAgICAgY29uc3QgaGFzQ29ycmVjdFRva2VucyA9XG4gICAgICAgIHRoaXMudG9rZW5JZCA9PT0gRVJHT19OQVRJVkVfVE9LRU4gPyB0b2tlbkxlbiA9PT0gMSA6IHRva2VuTGVuID09PSAyO1xuICAgICAgZm9yIChsZXQgaSA9IDA7IGkgPCB0b2tlbkxlbjsgaSsrKSB7XG4gICAgICAgIGNvbnN0IGlkID0gYm94LnRva2VucygpLmdldChpKS5pZCgpLnRvX3N0cigpO1xuICAgICAgICBpZiAoaWQgPT09IHRoaXMubWluaW11bUZlZU5GVCkgaGFzTWluaW11bUZlZU5GVCA9IHRydWU7XG4gICAgICAgIGVsc2UgaWYgKGlkID09PSB0aGlzLnRva2VuSWQpIGhhc1RhcmdldFRva2VuID0gdHJ1ZTtcbiAgICAgIH1cbiAgICAgIHJldHVybiBoYXNDb3JyZWN0VG9rZW5zICYmIGhhc01pbmltdW1GZWVORlQgJiYgaGFzVGFyZ2V0VG9rZW47XG4gICAgfTtcblxuICAgIHRoaXMuYm94ID0gdGhpcy5zZWxlY3RFbGlnaWJsZUJveChcbiAgICAgIGJveGVzLmZpbHRlcihcbiAgICAgICAgKGJveDogRXJnb0JveCkgPT5cbiAgICAgICAgICBib3hIYXNDb3JyZWN0QWRkcmVzcyhib3gpICYmIGJveEhhc0FwcHJvcHJpYXRlVG9rZW5zKGJveClcbiAgICAgIClcbiAgICApO1xuICB9O1xuXG4gIC8qKlxuICAgKiByZXR1cm5zIGZldGNoZWQgYm94IG9yIHRocm93cyBhcHByb3ByaXRlIGVycm9yIGlmIGZvdW5kIG1vcmUgb3Igbm9uZVxuICAgKiBAcGFyYW0gZWxpZ2libGVCb3hlc1xuICAgKi9cbiAgcHJvdGVjdGVkIHNlbGVjdEVsaWdpYmxlQm94ID0gKGVsaWdpYmxlQm94ZXM6IEFycmF5PEVyZ29Cb3g+KTogRXJnb0JveCA9PiB7XG4gICAgdGhpcy5sb2dnZXIuZGVidWcoXG4gICAgICBgRm91bmQgWyR7XG4gICAgICAgIGVsaWdpYmxlQm94ZXMubGVuZ3RoXG4gICAgICB9XSBtaW5pbXVtLWZlZSBib3hlczogJHtKc29uQmlnSW50LnN0cmluZ2lmeShcbiAgICAgICAgZWxpZ2libGVCb3hlcy5tYXAoKGJveCkgPT4gYm94LnRvX2pzb24oKSlcbiAgICAgICl9YFxuICAgICk7XG5cbiAgICBpZiAoZWxpZ2libGVCb3hlcy5sZW5ndGggPT09IDApXG4gICAgICB0aHJvdyBuZXcgTm90Rm91bmRFcnJvcihcbiAgICAgICAgYEZvdW5kIG5vIG1pbmltdW0tZmVlIGJveCBmb3IgdG9rZW4gWyR7dGhpcy50b2tlbklkfV0gYW5kIGFkZHJlc3MgWyR7dGhpcy5hZGRyZXNzfV1gXG4gICAgICApO1xuICAgIGVsc2UgaWYgKGVsaWdpYmxlQm94ZXMubGVuZ3RoID4gMSlcbiAgICAgIHRocm93IG5ldyBGYWlsZWRFcnJvcihcbiAgICAgICAgYEZvdW5kIFske2VsaWdpYmxlQm94ZXMubGVuZ3RofV0gbWluaW11bS1mZWUgYm94ZXMgZm9yIHRva2VuIFske3RoaXMudG9rZW5JZH1dIGFuZCBhZGRyZXNzIFske3RoaXMuYWRkcmVzc31dYFxuICAgICAgKTtcbiAgICBlbHNlIHtcbiAgICAgIHRoaXMubG9nZ2VyLmRlYnVnKFxuICAgICAgICBgRm91bmQgbWluaW11bS1mZWUgYm94IFske2VsaWdpYmxlQm94ZXNbMF1cbiAgICAgICAgICAuYm94X2lkKClcbiAgICAgICAgICAudG9fc3RyKCl9XSBmb3IgdG9rZW4gWyR7dGhpcy50b2tlbklkfV0gYW5kIGFkZHJlc3MgWyR7dGhpcy5hZGRyZXNzfV1gXG4gICAgICApO1xuICAgICAgcmV0dXJuIGVsaWdpYmxlQm94ZXNbMF07XG4gICAgfVxuICB9O1xuXG4gIC8qKlxuICAgKiBmZXRjaGVzIGJveCBmcm9tIHRoZSBibG9ja2NoYWluIHVzaW5nIGV4cGxvcmVyIGNsaWVudFxuICAgKi9cbiAgcHJvdGVjdGVkIGZldGNoQm94ZXNVc2luZ0V4cGxvcmVyID0gYXN5bmMgKCk6IFByb21pc2U8QXJyYXk8RXJnb0JveD4+ID0+IHtcbiAgICBjb25zdCBib3hlczogQXJyYXk8RXJnb0JveD4gPSBbXTtcbiAgICB0cnkge1xuICAgICAgbGV0IGN1cnJlbnRQYWdlID0gMDtcbiAgICAgIGxldCBib3hlc1BhZ2UgPVxuICAgICAgICBhd2FpdCB0aGlzLmV4cGxvcmVyQ2xpZW50LnYxLmdldEFwaVYxQm94ZXNVbnNwZW50QnlhZGRyZXNzUDEoXG4gICAgICAgICAgdGhpcy5hZGRyZXNzLFxuICAgICAgICAgIHtcbiAgICAgICAgICAgIG9mZnNldDogY3VycmVudFBhZ2UgKiB0aGlzLkJPWF9GRVRDSElOR19QQUdFX1NJWkUsXG4gICAgICAgICAgICBsaW1pdDogdGhpcy5CT1hfRkVUQ0hJTkdfUEFHRV9TSVpFLFxuICAgICAgICAgIH1cbiAgICAgICAgKTtcbiAgICAgIHRoaXMubG9nZ2VyLmRlYnVnKFxuICAgICAgICBgcmVxdWVzdGVkICdleHBsb3JlckNsaWVudC5nZXRBcGlWMUJveGVzVW5zcGVudEJ5YWRkcmVzc1AxJyBmb3IgYWRkcmVzcyBbJHtcbiAgICAgICAgICB0aGlzLmFkZHJlc3NcbiAgICAgICAgfV0uIHJlczogJHtKc29uQmlnSW50LnN0cmluZ2lmeShib3hlc1BhZ2UpfWBcbiAgICAgICk7XG4gICAgICB3aGlsZSAoYm94ZXNQYWdlLml0ZW1zPy5sZW5ndGgpIHtcbiAgICAgICAgYm94ZXMucHVzaChcbiAgICAgICAgICAuLi5ib3hlc1BhZ2UuaXRlbXMubWFwKChib3gpID0+XG4gICAgICAgICAgICBFcmdvQm94LmZyb21fanNvbihKc29uQmlnSW50LnN0cmluZ2lmeShib3gpKVxuICAgICAgICAgIClcbiAgICAgICAgKTtcbiAgICAgICAgY3VycmVudFBhZ2UrKztcbiAgICAgICAgYm94ZXNQYWdlID1cbiAgICAgICAgICBhd2FpdCB0aGlzLmV4cGxvcmVyQ2xpZW50LnYxLmdldEFwaVYxQm94ZXNVbnNwZW50QnlhZGRyZXNzUDEoXG4gICAgICAgICAgICB0aGlzLmFkZHJlc3MsXG4gICAgICAgICAgICB7XG4gICAgICAgICAgICAgIG9mZnNldDogY3VycmVudFBhZ2UgKiB0aGlzLkJPWF9GRVRDSElOR19QQUdFX1NJWkUsXG4gICAgICAgICAgICAgIGxpbWl0OiB0aGlzLkJPWF9GRVRDSElOR19QQUdFX1NJWkUsXG4gICAgICAgICAgICB9XG4gICAgICAgICAgKTtcbiAgICAgICAgdGhpcy5sb2dnZXIuZGVidWcoXG4gICAgICAgICAgYHJlcXVlc3RlZCAnZXhwbG9yZXJDbGllbnQuZ2V0QXBpVjFCb3hlc1Vuc3BlbnRCeWFkZHJlc3NQMScgZm9yIGFkZHJlc3MgWyR7XG4gICAgICAgICAgICB0aGlzLmFkZHJlc3NcbiAgICAgICAgICB9XS4gcmVzOiAke0pzb25CaWdJbnQuc3RyaW5naWZ5KGJveGVzUGFnZSl9YFxuICAgICAgICApO1xuICAgICAgfVxuICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICByZXR1cm4gaGFuZGxlQXBpRXJyb3IoXG4gICAgICAgIGVycm9yLFxuICAgICAgICAnRmFpbGVkIHRvIGdldCBib3hlcyBieSB0b2tlbiBpZCBmcm9tIEVyZ28gRXhwbG9yZXI6J1xuICAgICAgKTtcbiAgICB9XG5cbiAgICByZXR1cm4gYm94ZXM7XG4gIH07XG5cbiAgLyoqXG4gICAqIGZldGNoZXMgdGhlIGJveCBmcm9tIHRoZSBibG9ja2NoYWluIHVzaW5nIG5vZGUgY2xpZW50XG4gICAqL1xuICBwcm90ZWN0ZWQgZmV0Y2hCb3hlc1VzaW5nTm9kZSA9IGFzeW5jICgpOiBQcm9taXNlPEFycmF5PEVyZ29Cb3g+PiA9PiB7XG4gICAgY29uc3QgYm94ZXM6IEFycmF5PEVyZ29Cb3g+ID0gW107XG4gICAgdHJ5IHtcbiAgICAgIGxldCBjdXJyZW50UGFnZSA9IDA7XG4gICAgICBsZXQgYm94ZXNQYWdlID0gYXdhaXQgdGhpcy5ub2RlQ2xpZW50LmdldEJveGVzQnlBZGRyZXNzVW5zcGVudChcbiAgICAgICAgdGhpcy5hZGRyZXNzLFxuICAgICAgICB7XG4gICAgICAgICAgb2Zmc2V0OiBjdXJyZW50UGFnZSAqIHRoaXMuQk9YX0ZFVENISU5HX1BBR0VfU0laRSxcbiAgICAgICAgICBsaW1pdDogdGhpcy5CT1hfRkVUQ0hJTkdfUEFHRV9TSVpFLFxuICAgICAgICB9XG4gICAgICApO1xuICAgICAgdGhpcy5sb2dnZXIuZGVidWcoXG4gICAgICAgIGByZXF1ZXN0ZWQgJ25vZGVDbGllbnQuZ2V0Qm94ZXNCeUFkZHJlc3NVbnNwZW50JyBmb3IgYWRkcmVzcyBbJHtcbiAgICAgICAgICB0aGlzLmFkZHJlc3NcbiAgICAgICAgfV0uIHJlczogJHtKc29uQmlnSW50LnN0cmluZ2lmeShib3hlc1BhZ2UpfWBcbiAgICAgICk7XG4gICAgICB3aGlsZSAoYm94ZXNQYWdlLmxlbmd0aCAhPT0gMCkge1xuICAgICAgICBib3hlcy5wdXNoKFxuICAgICAgICAgIC4uLmJveGVzUGFnZS5tYXAoKGJveCkgPT5cbiAgICAgICAgICAgIEVyZ29Cb3guZnJvbV9qc29uKEpzb25CaWdJbnQuc3RyaW5naWZ5KGJveCkpXG4gICAgICAgICAgKVxuICAgICAgICApO1xuICAgICAgICBjdXJyZW50UGFnZSsrO1xuICAgICAgICBib3hlc1BhZ2UgPSBhd2FpdCB0aGlzLm5vZGVDbGllbnQuZ2V0Qm94ZXNCeUFkZHJlc3NVbnNwZW50KFxuICAgICAgICAgIHRoaXMuYWRkcmVzcyxcbiAgICAgICAgICB7XG4gICAgICAgICAgICBvZmZzZXQ6IGN1cnJlbnRQYWdlICogdGhpcy5CT1hfRkVUQ0hJTkdfUEFHRV9TSVpFLFxuICAgICAgICAgICAgbGltaXQ6IHRoaXMuQk9YX0ZFVENISU5HX1BBR0VfU0laRSxcbiAgICAgICAgICB9XG4gICAgICAgICk7XG4gICAgICAgIHRoaXMubG9nZ2VyLmRlYnVnKFxuICAgICAgICAgIGByZXF1ZXN0ZWQgJ25vZGVDbGllbnQuZ2V0Qm94ZXNCeUFkZHJlc3NVbnNwZW50JyBmb3IgYWRkcmVzcyBbJHtcbiAgICAgICAgICAgIHRoaXMuYWRkcmVzc1xuICAgICAgICAgIH1dLiByZXM6ICR7SnNvbkJpZ0ludC5zdHJpbmdpZnkoYm94ZXNQYWdlKX1gXG4gICAgICAgICk7XG4gICAgICB9XG4gICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgIGNvbnN0IGJhc2VFcnJvciA9ICdGYWlsZWQgdG8gZ2V0IGJveGVzIGJ5IHRva2VuIGlkIGZyb20gRXJnbyBOb2RlOic7XG4gICAgICBoYW5kbGVBcGlFcnJvcihlcnJvciwgYmFzZUVycm9yLCB7XG4gICAgICAgIGhhbmRsZVJlc3BvbmRlZFN0YXRlOiAoZXJyb3IpID0+IHtcbiAgICAgICAgICBpZiAoZXJyb3IucmVzcG9uc2Uuc3RhdHVzID09PSA0MDApIHJldHVybjtcbiAgICAgICAgICB0aHJvdyBuZXcgRmFpbGVkRXJyb3IoXG4gICAgICAgICAgICBgJHtiYXNlRXJyb3J9IFske2Vycm9yLnJlc3BvbnNlLnN0YXR1c31dICR7ZXJyb3IucmVzcG9uc2UuZGF0YS5yZWFzb259YFxuICAgICAgICAgICk7XG4gICAgICAgIH0sXG4gICAgICB9KTtcbiAgICB9XG5cbiAgICByZXR1cm4gYm94ZXM7XG4gIH07XG5cbiAgLyoqXG4gICAqIHJldHVybnMgZmV0Y2hlZCBjb25maWcgYm94XG4gICAqL1xuICBnZXRCb3ggPSAoKTogRXJnb0JveCA9PiB0aGlzLmJveDtcblxuICAvKipcbiAgICogZ2V0cyBjb3JyZXNwb25kaW5nIGNvbmZpZyBmb3IgdHdvIGNoYWlucyBhbmQgaGVpZ2h0XG4gICAqIEBwYXJhbSBmcm9tQ2hhaW5cbiAgICogQHBhcmFtIGhlaWdodCBibG9ja2NoYWluIGhlaWdodCBmb3IgZnJvbUNoYWluXG4gICAqIEBwYXJhbSB0b0NoYWluXG4gICAqL1xuICBnZXRGZWUgPSAoXG4gICAgZnJvbUNoYWluOiBzdHJpbmcsXG4gICAgaGVpZ2h0OiBudW1iZXIsXG4gICAgdG9DaGFpbjogc3RyaW5nXG4gICk6IENoYWluTWluaW11bUZlZSA9PiB7XG4gICAgaWYgKCF0aGlzLmJveCkgdGhyb3cgRXJyb3IoYEJveCBpcyBub3QgZmV0Y2hlZCB5ZXRgKTtcblxuICAgIGNvbnN0IGZlZXMgPSB0aGlzLmV4dHJhY3RGZWVGcm9tQm94KCkucmV2ZXJzZSgpO1xuICAgIGZvciAoY29uc3QgZmVlIG9mIGZlZXMpIHtcbiAgICAgIGlmICghT2JqZWN0Lmhhc093bihmZWUuaGVpZ2h0cywgZnJvbUNoYWluKSlcbiAgICAgICAgdGhyb3cgbmV3IE5vdEZvdW5kRXJyb3IoXG4gICAgICAgICAgYE5vIGZlZSBmb3VuZCBmb3IgY2hhaW4gWyR7ZnJvbUNoYWlufV0gaW4gYm94IFske3RoaXMuYm94XG4gICAgICAgICAgICAuYm94X2lkKClcbiAgICAgICAgICAgIC50b19zdHIoKX1dYFxuICAgICAgICApO1xuICAgICAgaWYgKGZlZS5oZWlnaHRzW2Zyb21DaGFpbl0gPCBoZWlnaHQpIHtcbiAgICAgICAgY29uc3QgY2hhaW5GZWUgPSBmZWUuY29uZmlnc1t0b0NoYWluXTtcbiAgICAgICAgaWYgKGNoYWluRmVlKSByZXR1cm4gbmV3IENoYWluTWluaW11bUZlZShjaGFpbkZlZSk7XG4gICAgICAgIGVsc2VcbiAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICAgICAgICBgQ2hhaW4gWyR7dG9DaGFpbn1dIGlzIG5vdCBzdXBwb3J0ZWQgYXQgZ2l2ZW4gaGVpZ2h0IG9mIGZyb21DaGFpbiBbJHtoZWlnaHR9IG9mICR7ZnJvbUNoYWlufV0gaW4gYm94IFske3RoaXMuYm94XG4gICAgICAgICAgICAgIC5ib3hfaWQoKVxuICAgICAgICAgICAgICAudG9fc3RyKCl9XWBcbiAgICAgICAgICApO1xuICAgICAgfVxuICAgIH1cblxuICAgIHRocm93IG5ldyBOb3RGb3VuZEVycm9yKFxuICAgICAgYENvbmZpZyBkb2VzIG5vdCBzdXBwb3J0IGhlaWdodCBbJHtoZWlnaHR9XSBmb3IgY2hhaW4gWyR7ZnJvbUNoYWlufV0gaW4gYm94IFske3RoaXMuYm94XG4gICAgICAgIC5ib3hfaWQoKVxuICAgICAgICAudG9fc3RyKCl9XWBcbiAgICApO1xuICB9O1xuXG4gIC8qKlxuICAgKiBleHRyYWN0cyBGZWUgY29uZmlnIGZyb20gYm94IHJlZ2lzdGVyc1xuICAgKi9cbiAgcHJvdGVjdGVkIGV4dHJhY3RGZWVGcm9tQm94ID0gKCk6IEFycmF5PEZlZT4gPT4ge1xuICAgIGNvbnN0IFI0ID0gdGhpcy5ib3gucmVnaXN0ZXJfdmFsdWUoNCk7XG4gICAgY29uc3QgUjUgPSB0aGlzLmJveC5yZWdpc3Rlcl92YWx1ZSg1KTtcbiAgICBjb25zdCBSNiA9IHRoaXMuYm94LnJlZ2lzdGVyX3ZhbHVlKDYpO1xuICAgIGNvbnN0IFI3ID0gdGhpcy5ib3gucmVnaXN0ZXJfdmFsdWUoNyk7XG4gICAgY29uc3QgUjggPSB0aGlzLmJveC5yZWdpc3Rlcl92YWx1ZSg4KTtcbiAgICBjb25zdCBSOSA9IHRoaXMuYm94LnJlZ2lzdGVyX3ZhbHVlKDkpO1xuXG4gICAgaWYgKCFSNCB8fCAhUjUgfHwgIVI2IHx8ICFSNyB8fCAhUjggfHwgIVI5KVxuICAgICAgdGhyb3cgRXJyb3IoXG4gICAgICAgIGBJbmNvbXBsZXRlIHJlZ2lzdGVyIGRhdGEgZm9yIG1pbmltdW0tZmVlIGNvbmZpZyBib3ggWyR7dGhpcy5ib3hcbiAgICAgICAgICAuYm94X2lkKClcbiAgICAgICAgICAudG9fc3RyKCl9XWBcbiAgICAgICk7XG5cbiAgICBjb25zdCBmZWVzOiBBcnJheTxGZWU+ID0gW107XG4gICAgY29uc3QgY2hhaW5zID0gUjQudG9fY29sbF9jb2xsX2J5dGUoKS5tYXAoKGVsZW1lbnQpID0+XG4gICAgICBCdWZmZXIuZnJvbShlbGVtZW50KS50b1N0cmluZygpXG4gICAgKTtcbiAgICBjb25zdCBoZWlnaHRzID0gUjUudG9fanMoKSBhcyBBcnJheTxBcnJheTxudW1iZXI+PjtcbiAgICBjb25zdCBicmlkZ2VGZWVzID0gUjYudG9fanMoKSBhcyBBcnJheTxBcnJheTxzdHJpbmc+PjtcbiAgICBjb25zdCBuZXR3b3JrRmVlcyA9IFI3LnRvX2pzKCkgYXMgQXJyYXk8QXJyYXk8c3RyaW5nPj47XG4gICAgY29uc3QgcnNuUmF0aW9zID0gUjgudG9fanMoKSBhcyBBcnJheTxBcnJheTxBcnJheTxzdHJpbmc+Pj47XG4gICAgY29uc3QgZmVlUmF0aW9zID0gUjkudG9fanMoKSBhcyBBcnJheTxBcnJheTxzdHJpbmc+PjtcblxuICAgIGZvciAobGV0IGZlZUlkeCA9IDA7IGZlZUlkeCA8IGhlaWdodHMubGVuZ3RoOyBmZWVJZHgrKykge1xuICAgICAgY29uc3QgZmVlOiBGZWUgPSB7XG4gICAgICAgIGhlaWdodHM6IHt9LFxuICAgICAgICBjb25maWdzOiB7fSxcbiAgICAgIH07XG4gICAgICBmb3IgKGxldCBjaGFpbklkeCA9IDA7IGNoYWluSWR4IDwgY2hhaW5zLmxlbmd0aDsgY2hhaW5JZHgrKykge1xuICAgICAgICBjb25zdCBjaGFpbiA9IGNoYWluc1tjaGFpbklkeF07XG5cbiAgICAgICAgaWYgKGhlaWdodHNbZmVlSWR4XVtjaGFpbklkeF0gPT09IC0xKSBjb250aW51ZTtcbiAgICAgICAgZmVlLmhlaWdodHNbY2hhaW5dID0gaGVpZ2h0c1tmZWVJZHhdW2NoYWluSWR4XTtcblxuICAgICAgICBpZiAoYnJpZGdlRmVlc1tmZWVJZHhdW2NoYWluSWR4XSA9PT0gJy0xJykgY29udGludWU7XG4gICAgICAgIGZlZS5jb25maWdzW2NoYWluXSA9IHtcbiAgICAgICAgICBicmlkZ2VGZWU6IEJpZ0ludChicmlkZ2VGZWVzW2ZlZUlkeF1bY2hhaW5JZHhdKSxcbiAgICAgICAgICBuZXR3b3JrRmVlOiBCaWdJbnQobmV0d29ya0ZlZXNbZmVlSWR4XVtjaGFpbklkeF0pLFxuICAgICAgICAgIHJzblJhdGlvOiBCaWdJbnQocnNuUmF0aW9zW2ZlZUlkeF1bY2hhaW5JZHhdWzBdKSxcbiAgICAgICAgICByc25SYXRpb0Rpdmlzb3I6IEJpZ0ludChyc25SYXRpb3NbZmVlSWR4XVtjaGFpbklkeF1bMV0pLFxuICAgICAgICAgIGZlZVJhdGlvOiBCaWdJbnQoZmVlUmF0aW9zW2ZlZUlkeF1bY2hhaW5JZHhdKSxcbiAgICAgICAgfTtcbiAgICAgIH1cbiAgICAgIGZlZXMucHVzaChmZWUpO1xuICAgIH1cblxuICAgIHRoaXMubG9nZ2VyLmRlYnVnKFxuICAgICAgYEV4dHJhY3RlZCBmZWUgY29uZmlnIGZyb20gYm94IFske3RoaXMuYm94XG4gICAgICAgIC5ib3hfaWQoKVxuICAgICAgICAudG9fc3RyKCl9XTogJHtKc29uQmlnSW50LnN0cmluZ2lmeShmZWVzKX1gXG4gICAgKTtcblxuICAgIHJldHVybiBmZWVzO1xuICB9O1xuXG4gIC8qKlxuICAgKiBnZW5lcmF0ZXMgYSBNaW5pbXVtRmVlQm94QnVpbGRlciB1c2luZyBjdXJyZW50IGJveFxuICAgKiAgbm90ZSB0aGF0ICdoZWlnaHQnIHBhcmFtZXRlciBvZiBidWlsZGVyIHdvbid0IGJlIHNldFxuICAgKi9cbiAgdG9CdWlsZGVyID0gKCk6IE1pbmltdW1GZWVCb3hCdWlsZGVyID0+IHtcbiAgICBjb25zdCBidWlsZGVyID0gbmV3IE1pbmltdW1GZWVCb3hCdWlsZGVyKHRoaXMubWluaW11bUZlZU5GVCwgdGhpcy5hZGRyZXNzKVxuICAgICAgLnNldFZhbHVlKEJpZ0ludCh0aGlzLmJveC52YWx1ZSgpLmFzX2k2NCgpLnRvX3N0cigpKSlcbiAgICAgIC5zZXRUb2tlbih0aGlzLnRva2VuSWQpO1xuXG4gICAgdGhpcy5leHRyYWN0RmVlRnJvbUJveCgpLmZvckVhY2goKGZlZSkgPT4ge1xuICAgICAgY29uc3QgY2hhaW5GZWUgPSBuZXcgTWluaW11bUZlZUNvbmZpZygpO1xuICAgICAgT2JqZWN0LmtleXMoZmVlLmhlaWdodHMpLmZvckVhY2goKGNoYWluKSA9PiB7XG4gICAgICAgIGlmIChPYmplY3QuaGFzT3duKGZlZS5jb25maWdzLCBjaGFpbikpXG4gICAgICAgICAgY2hhaW5GZWUuc2V0Q2hhaW5Db25maWcoXG4gICAgICAgICAgICBjaGFpbixcbiAgICAgICAgICAgIGZlZS5oZWlnaHRzW2NoYWluXSxcbiAgICAgICAgICAgIGZlZS5jb25maWdzW2NoYWluXVxuICAgICAgICAgICk7XG4gICAgICAgIGVsc2UgY2hhaW5GZWUuc2V0Q2hhaW5Db25maWcoY2hhaW4sIGZlZS5oZWlnaHRzW2NoYWluXSwgdW5kZWZpbmVkKTtcbiAgICAgIH0pO1xuICAgICAgYnVpbGRlci5hZGRDb25maWcoY2hhaW5GZWUpO1xuICAgIH0pO1xuICAgIHJldHVybiBidWlsZGVyO1xuICB9O1xufVxuIl19
@@ -0,0 +1,47 @@
1
+ import { BoxValue, ErgoBoxCandidate } from 'ergo-lib-wasm-nodejs';
2
+ import { MinimumFeeConfig } from './MinimumFeeConfig';
3
+ import { Fee } from './types';
4
+ export declare class MinimumFeeBoxBuilder {
5
+ protected fees: Array<Fee>;
6
+ protected boxValue: BoxValue;
7
+ protected boxheight: number;
8
+ protected tokenId: string;
9
+ protected minimumFeeNFT: string;
10
+ protected address: string;
11
+ constructor(minimumFeeNFT: string, address: string);
12
+ /**
13
+ * adds a feeConfig
14
+ * @param feeConfig
15
+ */
16
+ addConfig: (feeConfig: MinimumFeeConfig) => MinimumFeeBoxBuilder;
17
+ /**
18
+ * removes a config by index
19
+ * @param index
20
+ */
21
+ removeConfig: (index: number) => MinimumFeeBoxBuilder;
22
+ /**
23
+ * sets ErgoBox Erg value
24
+ * @param nanoErg
25
+ * @returns
26
+ */
27
+ setValue: (nanoErg: bigint) => MinimumFeeBoxBuilder;
28
+ /**
29
+ * sets ErgoBox creationheight
30
+ * @param currentHeight
31
+ */
32
+ setHeight: (currentHeight: number) => MinimumFeeBoxBuilder;
33
+ /**
34
+ * sets config token id
35
+ * @param tokenId
36
+ */
37
+ setToken: (tokenId: string) => MinimumFeeBoxBuilder;
38
+ /**
39
+ * validates some of specified configs
40
+ */
41
+ protected validate: () => void;
42
+ /**
43
+ * validates specified configs and builds ErgoBoxCandidate of config box using them
44
+ */
45
+ build: () => ErgoBoxCandidate;
46
+ }
47
+ //# sourceMappingURL=MinimumFeeBoxBuilder.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"MinimumFeeBoxBuilder.d.ts","sourceRoot":"","sources":["../../lib/MinimumFeeBoxBuilder.ts"],"names":[],"mappings":"AAAA,OAAO,EAEL,QAAQ,EAOR,gBAAgB,EACjB,MAAM,sBAAsB,CAAC;AAC9B,OAAO,EAAE,gBAAgB,EAAE,MAAM,oBAAoB,CAAC;AACtD,OAAO,EAAE,GAAG,EAAE,MAAM,SAAS,CAAC;AAI9B,qBAAa,oBAAoB;IAC/B,SAAS,CAAC,IAAI,EAAE,KAAK,CAAC,GAAG,CAAC,CAAC;IAC3B,SAAS,CAAC,QAAQ,EAAE,QAAQ,CAAC;IAC7B,SAAS,CAAC,SAAS,EAAE,MAAM,CAAC;IAC5B,SAAS,CAAC,OAAO,EAAE,MAAM,CAAC;IAC1B,SAAS,CAAC,aAAa,EAAE,MAAM,CAAC;IAChC,SAAS,CAAC,OAAO,EAAE,MAAM,CAAC;gBAEd,aAAa,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM;IAMlD;;;OAGG;IACH,SAAS,cAAe,gBAAgB,KAAG,oBAAoB,CAG7D;IAEF;;;OAGG;IACH,YAAY,UAAW,MAAM,KAAG,oBAAoB,CAGlD;IAEF;;;;OAIG;IACH,QAAQ,YAAa,MAAM,KAAG,oBAAoB,CAGhD;IAEF;;;OAGG;IACH,SAAS,kBAAmB,MAAM,KAAG,oBAAoB,CAGvD;IAEF;;;OAGG;IACH,QAAQ,YAAa,MAAM,KAAG,oBAAoB,CAGhD;IAEF;;OAEG;IACH,SAAS,CAAC,QAAQ,QAAO,IAAI,CA6B3B;IAEF;;OAEG;IACH,KAAK,QAAO,gBAAgB,CAmF1B;CACH"}
@@ -0,0 +1,149 @@
1
+ import { Address, BoxValue, Contract, ErgoBoxCandidateBuilder, I64, TokenAmount, TokenId, Constant, } from 'ergo-lib-wasm-nodejs';
2
+ import { InvalidConfig } from './errors';
3
+ import { ERGO_NATIVE_TOKEN } from './constants';
4
+ export class MinimumFeeBoxBuilder {
5
+ fees;
6
+ boxValue;
7
+ boxheight;
8
+ tokenId;
9
+ minimumFeeNFT;
10
+ address;
11
+ constructor(minimumFeeNFT, address) {
12
+ this.fees = [];
13
+ this.minimumFeeNFT = minimumFeeNFT;
14
+ this.address = address;
15
+ }
16
+ /**
17
+ * adds a feeConfig
18
+ * @param feeConfig
19
+ */
20
+ addConfig = (feeConfig) => {
21
+ this.fees.push(feeConfig.getConfig());
22
+ return this;
23
+ };
24
+ /**
25
+ * removes a config by index
26
+ * @param index
27
+ */
28
+ removeConfig = (index) => {
29
+ this.fees.splice(index, 1);
30
+ return this;
31
+ };
32
+ /**
33
+ * sets ErgoBox Erg value
34
+ * @param nanoErg
35
+ * @returns
36
+ */
37
+ setValue = (nanoErg) => {
38
+ this.boxValue = BoxValue.from_i64(I64.from_str(nanoErg.toString()));
39
+ return this;
40
+ };
41
+ /**
42
+ * sets ErgoBox creationheight
43
+ * @param currentHeight
44
+ */
45
+ setHeight = (currentHeight) => {
46
+ this.boxheight = currentHeight;
47
+ return this;
48
+ };
49
+ /**
50
+ * sets config token id
51
+ * @param tokenId
52
+ */
53
+ setToken = (tokenId) => {
54
+ this.tokenId = tokenId;
55
+ return this;
56
+ };
57
+ /**
58
+ * validates some of specified configs
59
+ */
60
+ validate = () => {
61
+ if (!this.boxValue)
62
+ throw new InvalidConfig(`Box value argument is not defined`);
63
+ if (!this.boxheight)
64
+ throw new InvalidConfig(`Box creation height argument is not defined`);
65
+ if (!this.tokenId)
66
+ throw new InvalidConfig(`Config token id is not defined`);
67
+ if (this.fees.length === 0)
68
+ throw new InvalidConfig(`No config added. Please add at least one config`);
69
+ for (let i = 0; i < this.fees.length - 1; i++) {
70
+ const chains = Object.keys(this.fees[0].heights);
71
+ chains.forEach((chain) => {
72
+ if (!this.fees[i + 1].heights[chain])
73
+ throw new InvalidConfig(`Expected chain [${chain}] at index [${i + 1}]`);
74
+ if (this.fees[i + 1].heights[chain] < this.fees[i].heights[chain])
75
+ throw new InvalidConfig(`All heights for a chain should be ascending. Heights of chain [${chain}] at indexes [${i},${i + 1}] are invalid [${this.fees[i + 1].heights[chain]} < ${this.fees[i].heights[chain]}]`);
76
+ });
77
+ }
78
+ };
79
+ /**
80
+ * validates specified configs and builds ErgoBoxCandidate of config box using them
81
+ */
82
+ build = () => {
83
+ this.validate();
84
+ // add box value, address and creation height
85
+ const boxBuilder = new ErgoBoxCandidateBuilder(this.boxValue, Contract.new(Address.from_base58(this.address).to_ergo_tree()), this.boxheight);
86
+ // add box tokens
87
+ boxBuilder.add_token(TokenId.from_str(this.minimumFeeNFT), TokenAmount.from_i64(I64.from_str('1')));
88
+ if (this.tokenId !== ERGO_NATIVE_TOKEN)
89
+ boxBuilder.add_token(TokenId.from_str(this.tokenId), TokenAmount.from_i64(I64.from_str('1')));
90
+ // generate register values
91
+ // extract chains
92
+ const chains = [];
93
+ this.fees.forEach((fee) => {
94
+ Object.keys(fee.heights).forEach((feeChain) => {
95
+ if (!chains.includes(feeChain))
96
+ chains.push(feeChain);
97
+ });
98
+ });
99
+ chains.sort();
100
+ // extract configs
101
+ const heights = [];
102
+ const brdigeFees = [];
103
+ const networkFees = [];
104
+ const rsnRatios = [];
105
+ const feeRatios = [];
106
+ this.fees.forEach((fee) => {
107
+ const heightsConfigs = [];
108
+ const brdigeFeesConfigs = [];
109
+ const networkFeesConfigs = [];
110
+ const rsnRatiosConfigs = [];
111
+ const feeRatiosConfigs = [];
112
+ chains.forEach((chain) => {
113
+ if (Object.hasOwn(fee.heights, chain))
114
+ heightsConfigs.push(fee.heights[chain]);
115
+ else
116
+ heightsConfigs.push(-1);
117
+ if (Object.hasOwn(fee.configs, chain)) {
118
+ brdigeFeesConfigs.push(fee.configs[chain].bridgeFee.toString());
119
+ networkFeesConfigs.push(fee.configs[chain].networkFee.toString());
120
+ rsnRatiosConfigs.push([
121
+ fee.configs[chain].rsnRatio.toString(),
122
+ fee.configs[chain].rsnRatioDivisor.toString(),
123
+ ]);
124
+ feeRatiosConfigs.push(fee.configs[chain].feeRatio.toString());
125
+ }
126
+ else {
127
+ brdigeFeesConfigs.push('-1');
128
+ networkFeesConfigs.push('-1');
129
+ rsnRatiosConfigs.push(['-1', '-1']);
130
+ feeRatiosConfigs.push('-1');
131
+ }
132
+ });
133
+ heights.push(heightsConfigs);
134
+ brdigeFees.push(brdigeFeesConfigs);
135
+ networkFees.push(networkFeesConfigs);
136
+ rsnRatios.push(rsnRatiosConfigs);
137
+ feeRatios.push(feeRatiosConfigs);
138
+ });
139
+ // add box registers
140
+ boxBuilder.set_register_value(4, Constant.from_coll_coll_byte(chains.map((chain) => Buffer.from(chain))));
141
+ boxBuilder.set_register_value(5, Constant.from_js(heights));
142
+ boxBuilder.set_register_value(6, Constant.from_js(brdigeFees));
143
+ boxBuilder.set_register_value(7, Constant.from_js(networkFees));
144
+ boxBuilder.set_register_value(8, Constant.from_js(rsnRatios));
145
+ boxBuilder.set_register_value(9, Constant.from_js(feeRatios));
146
+ return boxBuilder.build();
147
+ };
148
+ }
149
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiTWluaW11bUZlZUJveEJ1aWxkZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9saWIvTWluaW11bUZlZUJveEJ1aWxkZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUNMLE9BQU8sRUFDUCxRQUFRLEVBQ1IsUUFBUSxFQUNSLHVCQUF1QixFQUN2QixHQUFHLEVBQ0gsV0FBVyxFQUNYLE9BQU8sRUFDUCxRQUFRLEdBRVQsTUFBTSxzQkFBc0IsQ0FBQztBQUc5QixPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sVUFBVSxDQUFDO0FBQ3pDLE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxNQUFNLGFBQWEsQ0FBQztBQUVoRCxNQUFNLE9BQU8sb0JBQW9CO0lBQ3JCLElBQUksQ0FBYTtJQUNqQixRQUFRLENBQVc7SUFDbkIsU0FBUyxDQUFTO0lBQ2xCLE9BQU8sQ0FBUztJQUNoQixhQUFhLENBQVM7SUFDdEIsT0FBTyxDQUFTO0lBRTFCLFlBQVksYUFBcUIsRUFBRSxPQUFlO1FBQ2hELElBQUksQ0FBQyxJQUFJLEdBQUcsRUFBRSxDQUFDO1FBQ2YsSUFBSSxDQUFDLGFBQWEsR0FBRyxhQUFhLENBQUM7UUFDbkMsSUFBSSxDQUFDLE9BQU8sR0FBRyxPQUFPLENBQUM7SUFDekIsQ0FBQztJQUVEOzs7T0FHRztJQUNILFNBQVMsR0FBRyxDQUFDLFNBQTJCLEVBQXdCLEVBQUU7UUFDaEUsSUFBSSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLFNBQVMsRUFBRSxDQUFDLENBQUM7UUFDdEMsT0FBTyxJQUFJLENBQUM7SUFDZCxDQUFDLENBQUM7SUFFRjs7O09BR0c7SUFDSCxZQUFZLEdBQUcsQ0FBQyxLQUFhLEVBQXdCLEVBQUU7UUFDckQsSUFBSSxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsS0FBSyxFQUFFLENBQUMsQ0FBQyxDQUFDO1FBQzNCLE9BQU8sSUFBSSxDQUFDO0lBQ2QsQ0FBQyxDQUFDO0lBRUY7Ozs7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Q2hFLFVBQVUsQ0FBQyxrQkFBa0IsQ0FBQyxDQUFDLEVBQUUsUUFBUSxDQUFDLE9BQU8sQ0FBQyxTQUFTLENBQUMsQ0FBQyxDQUFDO1FBQzlELFVBQVUsQ0FBQyxrQkFBa0IsQ0FBQyxDQUFDLEVBQUUsUUFBUSxDQUFDLE9BQU8sQ0FBQyxTQUFTLENBQUMsQ0FBQyxDQUFDO1FBRTlELE9BQU8sVUFBVSxDQUFDLEtBQUssRUFBRSxDQUFDO0lBQzVCLENBQUMsQ0FBQztDQUNIIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtcbiAgQWRkcmVzcyxcbiAgQm94VmFsdWUsXG4gIENvbnRyYWN0LFxuICBFcmdvQm94Q2FuZGlkYXRlQnVpbGRlcixcbiAgSTY0LFxuICBUb2tlbkFtb3VudCxcbiAgVG9rZW5JZCxcbiAgQ29uc3RhbnQsXG4gIEVyZ29Cb3hDYW5kaWRhdGUsXG59IGZyb20gJ2VyZ28tbGliLXdhc20tbm9kZWpzJztcbmltcG9ydCB7IE1pbmltdW1GZWVDb25maWcgfSBmcm9tICcuL01pbmltdW1GZWVDb25maWcnO1xuaW1wb3J0IHsgRmVlIH0gZnJvbSAnLi90eXBlcyc7XG5pbXBvcnQgeyBJbnZhbGlkQ29uZmlnIH0gZnJvbSAnLi9lcnJvcnMnO1xuaW1wb3J0IHsgRVJHT19OQVRJVkVfVE9LRU4gfSBmcm9tICcuL2NvbnN0YW50cyc7XG5cbmV4cG9ydCBjbGFzcyBNaW5pbXVtRmVlQm94QnVpbGRlciB7XG4gIHByb3RlY3RlZCBmZWVzOiBBcnJheTxGZWU+O1xuICBwcm90ZWN0ZWQgYm94VmFsdWU6IEJveFZhbHVlO1xuICBwcm90ZWN0ZWQgYm94aGVpZ2h0OiBudW1iZXI7XG4gIHByb3RlY3RlZCB0b2tlbklkOiBzdHJpbmc7XG4gIHByb3RlY3RlZCBtaW5pbXVtRmVlTkZUOiBzdHJpbmc7XG4gIHByb3RlY3RlZCBhZGRyZXNzOiBzdHJpbmc7XG5cbiAgY29uc3RydWN0b3IobWluaW11bUZlZU5GVDogc3RyaW5nLCBhZGRyZXNzOiBzdHJpbmcpIHtcbiAgICB0aGlzLmZlZXMgPSBbXTtcbiAgICB0aGlzLm1pbmltdW1GZWVORlQgPSBtaW5pbXVtRmVlTkZUO1xuICAgIHRoaXMuYWRkcmVzcyA9IGFkZHJlc3M7XG4gIH1cblxuICAvKipcbiAgICogYWRkcyBhIGZlZUNvbmZpZ1xuICAgKiBAcGFyYW0gZmVlQ29uZmlnXG4gICAqL1xuICBhZGRDb25maWcgPSAoZmVlQ29uZmlnOiBNaW5pbXVtRmVlQ29uZmlnKTogTWluaW11bUZlZUJveEJ1aWxkZXIgPT4ge1xuICAgIHRoaXMuZmVlcy5wdXNoKGZlZUNvbmZpZy5nZXRDb25maWcoKSk7XG4gICAgcmV0dXJuIHRoaXM7XG4gIH07XG5cbiAgLyoqXG4gICAqIHJlbW92ZXMgYSBjb25maWcgYnkgaW5kZXhcbiAgICogQHBhcmFtIGluZGV4XG4gICAqL1xuICByZW1vdmVDb25maWcgPSAoaW5kZXg6IG51bWJlcik6IE1pbmltdW1GZWVCb3hCdWlsZGVyID0+IHtcbiAgICB0aGlzLmZlZXMuc3BsaWNlKGluZGV4LCAxKTtcbiAgICByZXR1cm4gdGhpcztcbiAgfTtcblxuICAvKipcbiAgICogc2V0cyBFcmdvQm94IEVyZyB2YWx1ZVxuICAgKiBAcGFyYW0gbmFub0VyZ1xuICAgKiBAcmV0dXJuc1xuICAgKi9cbiAgc2V0VmFsdWUgPSAobmFub0VyZzogYmlnaW50KTogTWluaW11bUZlZUJveEJ1aWxkZXIgPT4ge1xuICAgIHRoaXMuYm94VmFsdWUgPSBCb3hWYWx1ZS5mcm9tX2k2NChJNjQuZnJvbV9zdHIobmFub0VyZy50b1N0cmluZygpKSk7XG4gICAgcmV0dXJuIHRoaXM7XG4gIH07XG5cbiAgLyoqXG4gICAqIHNldHMgRXJnb0JveCBjcmVhdGlvbmhlaWdodFxuICAgKiBAcGFyYW0gY3VycmVudEhlaWdodFxuICAgKi9cbiAgc2V0SGVpZ2h0ID0gKGN1cnJlbnRIZWlnaHQ6IG51bWJlcik6IE1pbmltdW1GZWVCb3hCdWlsZGVyID0+IHtcbiAgICB0aGlzLmJveGhlaWdodCA9IGN1cnJlbnRIZWlnaHQ7XG4gICAgcmV0dXJuIHRoaXM7XG4gIH07XG5cbiAgLyoqXG4gICAqIHNldHMgY29uZmlnIHRva2VuIGlkXG4gICAqIEBwYXJhbSB0b2tlbklkXG4gICAqL1xuICBzZXRUb2tlbiA9ICh0b2tlbklkOiBzdHJpbmcpOiBNaW5pbXVtRmVlQm94QnVpbGRlciA9PiB7XG4gICAgdGhpcy50b2tlbklkID0gdG9rZW5JZDtcbiAgICByZXR1cm4gdGhpcztcbiAgfTtcblxuICAvKipcbiAgICogdmFsaWRhdGVzIHNvbWUgb2Ygc3BlY2lmaWVkIGNvbmZpZ3NcbiAgICovXG4gIHByb3RlY3RlZCB2YWxpZGF0ZSA9ICgpOiB2b2lkID0+IHtcbiAgICBpZiAoIXRoaXMuYm94VmFsdWUpXG4gICAgICB0aHJvdyBuZXcgSW52YWxpZENvbmZpZyhgQm94IHZhbHVlIGFyZ3VtZW50IGlzIG5vdCBkZWZpbmVkYCk7XG4gICAgaWYgKCF0aGlzLmJveGhlaWdodClcbiAgICAgIHRocm93IG5ldyBJbnZhbGlkQ29uZmlnKGBCb3ggY3JlYXRpb24gaGVpZ2h0IGFyZ3VtZW50IGlzIG5vdCBkZWZpbmVkYCk7XG4gICAgaWYgKCF0aGlzLnRva2VuSWQpXG4gICAgICB0aHJvdyBuZXcgSW52YWxpZENvbmZpZyhgQ29uZmlnIHRva2VuIGlkIGlzIG5vdCBkZWZpbmVkYCk7XG4gICAgaWYgKHRoaXMuZmVlcy5sZW5ndGggPT09IDApXG4gICAgICB0aHJvdyBuZXcgSW52YWxpZENvbmZpZyhcbiAgICAgICAgYE5vIGNvbmZpZyBhZGRlZC4gUGxlYXNlIGFkZCBhdCBsZWFzdCBvbmUgY29uZmlnYFxuICAgICAgKTtcblxuICAgIGZvciAobGV0IGkgPSAwOyBpIDwgdGhpcy5mZWVzLmxlbmd0aCAtIDE7IGkrKykge1xuICAgICAgY29uc3QgY2hhaW5zID0gT2JqZWN0LmtleXModGhpcy5mZWVzWzBdLmhlaWdodHMpO1xuICAgICAgY2hhaW5zLmZvckVhY2goKGNoYWluKSA9PiB7XG4gICAgICAgIGlmICghdGhpcy5mZWVzW2kgKyAxXS5oZWlnaHRzW2NoYWluXSlcbiAgICAgICAgICB0aHJvdyBuZXcgSW52YWxpZENvbmZpZyhcbiAgICAgICAgICAgIGBFeHBlY3RlZCBjaGFpbiBbJHtjaGFpbn1dIGF0IGluZGV4IFske2kgKyAxfV1gXG4gICAgICAgICAgKTtcbiAgICAgICAgaWYgKHRoaXMuZmVlc1tpICsgMV0uaGVpZ2h0c1tjaGFpbl0gPCB0aGlzLmZlZXNbaV0uaGVpZ2h0c1tjaGFpbl0pXG4gICAgICAgICAgdGhyb3cgbmV3IEludmFsaWRDb25maWcoXG4gICAgICAgICAgICBgQWxsIGhlaWdodHMgZm9yIGEgY2hhaW4gc2hvdWxkIGJlIGFzY2VuZGluZy4gSGVpZ2h0cyBvZiBjaGFpbiBbJHtjaGFpbn1dIGF0IGluZGV4ZXMgWyR7aX0sJHtcbiAgICAgICAgICAgICAgaSArIDFcbiAgICAgICAgICAgIH1dIGFyZSBpbnZhbGlkIFske3RoaXMuZmVlc1tpICsgMV0uaGVpZ2h0c1tjaGFpbl19IDwgJHtcbiAgICAgICAgICAgICAgdGhpcy5mZWVzW2ldLmhlaWdodHNbY2hhaW5dXG4gICAgICAgICAgICB9XWBcbiAgICAgICAgICApO1xuICAgICAgfSk7XG4gICAgfVxuICB9O1xuXG4gIC8qKlxuICAgKiB2YWxpZGF0ZXMgc3BlY2lmaWVkIGNvbmZpZ3MgYW5kIGJ1aWxkcyBFcmdvQm94Q2FuZGlkYXRlIG9mIGNvbmZpZyBib3ggdXNpbmcgdGhlbVxuICAgKi9cbiAgYnVpbGQgPSAoKTogRXJnb0JveENhbmRpZGF0ZSA9PiB7XG4gICAgdGhpcy52YWxpZGF0ZSgpO1xuXG4gICAgLy8gYWRkIGJveCB2YWx1ZSwgYWRkcmVzcyBhbmQgY3JlYXRpb24gaGVpZ2h0XG4gICAgY29uc3QgYm94QnVpbGRlciA9IG5ldyBFcmdvQm94Q2FuZGlkYXRlQnVpbGRlcihcbiAgICAgIHRoaXMuYm94VmFsdWUsXG4gICAgICBDb250cmFjdC5uZXcoQWRkcmVzcy5mcm9tX2Jhc2U1OCh0aGlzLmFkZHJlc3MpLnRvX2VyZ29fdHJlZSgpKSxcbiAgICAgIHRoaXMuYm94aGVpZ2h0XG4gICAgKTtcblxuICAgIC8vIGFkZCBib3ggdG9rZW5zXG4gICAgYm94QnVpbGRlci5hZGRfdG9rZW4oXG4gICAgICBUb2tlbklkLmZyb21fc3RyKHRoaXMubWluaW11bUZlZU5GVCksXG4gICAgICBUb2tlbkFtb3VudC5mcm9tX2k2NChJNjQuZnJvbV9zdHIoJzEnKSlcbiAgICApO1xuICAgIGlmICh0aGlzLnRva2VuSWQgIT09IEVSR09fTkFUSVZFX1RPS0VOKVxuICAgICAgYm94QnVpbGRlci5hZGRfdG9rZW4oXG4gICAgICAgIFRva2VuSWQuZnJvbV9zdHIodGhpcy50b2tlbklkKSxcbiAgICAgICAgVG9rZW5BbW91bnQuZnJvbV9pNjQoSTY0LmZyb21fc3RyKCcxJykpXG4gICAgICApO1xuXG4gICAgLy8gZ2VuZXJhdGUgcmVnaXN0ZXIgdmFsdWVzXG4gICAgLy8gIGV4dHJhY3QgY2hhaW5zXG4gICAgY29uc3QgY2hhaW5zOiBBcnJheTxzdHJpbmc+ID0gW107XG4gICAgdGhpcy5mZWVzLmZvckVhY2goKGZlZSkgPT4ge1xuICAgICAgT2JqZWN0LmtleXMoZmVlLmhlaWdodHMpLmZvckVhY2goKGZlZUNoYWluKSA9PiB7XG4gICAgICAgIGlmICghY2hhaW5zLmluY2x1ZGVzKGZlZUNoYWluKSkgY2hhaW5zLnB1c2goZmVlQ2hhaW4pO1xuICAgICAgfSk7XG4gICAgfSk7XG4gICAgY2hhaW5zLnNvcnQoKTtcbiAgICAvLyAgZXh0cmFjdCBjb25maWdzXG4gICAgY29uc3QgaGVpZ2h0czogQXJyYXk8QXJyYXk8bnVtYmVyPj4gPSBbXTtcbiAgICBjb25zdCBicmRpZ2VGZWVzOiBBcnJheTxBcnJheTxzdHJpbmc+PiA9IFtdO1xuICAgIGNvbnN0IG5ldHdvcmtGZWVzOiBBcnJheTxBcnJheTxzdHJpbmc+PiA9IFtdO1xuICAgIGNvbnN0IHJzblJhdGlvczogQXJyYXk8QXJyYXk8QXJyYXk8c3RyaW5nPj4+ID0gW107XG4gICAgY29uc3QgZmVlUmF0aW9zOiBBcnJheTxBcnJheTxzdHJpbmc+PiA9IFtdO1xuICAgIHRoaXMuZmVlcy5mb3JFYWNoKChmZWUpID0+IHtcbiAgICAgIGNvbnN0IGhlaWdodHNDb25maWdzOiBBcnJheTxudW1iZXI+ID0gW107XG4gICAgICBjb25zdCBicmRpZ2VGZWVzQ29uZmlnczogQXJyYXk8c3RyaW5nPiA9IFtdO1xuICAgICAgY29uc3QgbmV0d29ya0ZlZXNDb25maWdzOiBBcnJheTxzdHJpbmc+ID0gW107XG4gICAgICBjb25zdCByc25SYXRpb3NDb25maWdzOiBBcnJheTxBcnJheTxzdHJpbmc+PiA9IFtdO1xuICAgICAgY29uc3QgZmVlUmF0aW9zQ29uZmlnczogQXJyYXk8c3RyaW5nPiA9IFtdO1xuXG4gICAgICBjaGFpbnMuZm9yRWFjaCgoY2hhaW4pID0+IHtcbiAgICAgICAgaWYgKE9iamVjdC5oYXNPd24oZmVlLmhlaWdodHMsIGNoYWluKSlcbiAgICAgICAgICBoZWlnaHRzQ29uZmlncy5wdXNoKGZlZS5oZWlnaHRzW2NoYWluXSk7XG4gICAgICAgIGVsc2UgaGVpZ2h0c0NvbmZpZ3MucHVzaCgtMSk7XG5cbiAgICAgICAgaWYgKE9iamVjdC5oYXNPd24oZmVlLmNvbmZpZ3MsIGNoYWluKSkge1xuICAgICAgICAgIGJyZGlnZUZlZXNDb25maWdzLnB1c2goZmVlLmNvbmZpZ3NbY2hhaW5dLmJyaWRnZUZlZS50b1N0cmluZygpKTtcbiAgICAgICAgICBuZXR3b3JrRmVlc0NvbmZpZ3MucHVzaChmZWUuY29uZmlnc1tjaGFpbl0ubmV0d29ya0ZlZS50b1N0cmluZygpKTtcbiAgICAgICAgICByc25SYXRpb3NDb25maWdzLnB1c2goW1xuICAgICAgICAgICAgZmVlLmNvbmZpZ3NbY2hhaW5dLnJzblJhdGlvLnRvU3RyaW5nKCksXG4gICAgICAgICAgICBmZWUuY29uZmlnc1tjaGFpbl0ucnNuUmF0aW9EaXZpc29yLnRvU3RyaW5nKCksXG4gICAgICAgICAgXSk7XG4gICAgICAgICAgZmVlUmF0aW9zQ29uZmlncy5wdXNoKGZlZS5jb25maWdzW2NoYWluXS5mZWVSYXRpby50b1N0cmluZygpKTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBicmRpZ2VGZWVzQ29uZmlncy5wdXNoKCctMScpO1xuICAgICAgICAgIG5ldHdvcmtGZWVzQ29uZmlncy5wdXNoKCctMScpO1xuICAgICAgICAgIHJzblJhdGlvc0NvbmZpZ3MucHVzaChbJy0xJywgJy0xJ10pO1xuICAgICAgICAgIGZlZVJhdGlvc0NvbmZpZ3MucHVzaCgnLTEnKTtcbiAgICAgICAgfVxuICAgICAgfSk7XG5cbiAgICAgIGhlaWdodHMucHVzaChoZWlnaHRzQ29uZmlncyk7XG4gICAgICBicmRpZ2VGZWVzLnB1c2goYnJkaWdlRmVlc0NvbmZpZ3MpO1xuICAgICAgbmV0d29ya0ZlZXMucHVzaChuZXR3b3JrRmVlc0NvbmZpZ3MpO1xuICAgICAgcnNuUmF0aW9zLnB1c2gocnNuUmF0aW9zQ29uZmlncyk7XG4gICAgICBmZWVSYXRpb3MucHVzaChmZWVSYXRpb3NDb25maWdzKTtcbiAgICB9KTtcblxuICAgIC8vIGFkZCBib3ggcmVnaXN0ZXJzXG4gICAgYm94QnVpbGRlci5zZXRfcmVnaXN0ZXJfdmFsdWUoXG4gICAgICA0LFxuICAgICAgQ29uc3RhbnQuZnJvbV9jb2xsX2NvbGxfYnl0ZShjaGFpbnMubWFwKChjaGFpbikgPT4gQnVmZmVyLmZyb20oY2hhaW4pKSlcbiAgICApO1xuICAgIGJveEJ1aWxkZXIuc2V0X3JlZ2lzdGVyX3ZhbHVlKDUsIENvbnN0YW50LmZyb21fanMoaGVpZ2h0cykpO1xuICAgIGJveEJ1aWxkZXIuc2V0X3JlZ2lzdGVyX3ZhbHVlKDYsIENvbnN0YW50LmZyb21fanMoYnJkaWdlRmVlcykpO1xuICAgIGJveEJ1aWxkZXIuc2V0X3JlZ2lzdGVyX3ZhbHVlKDcsIENvbnN0YW50LmZyb21fanMobmV0d29ya0ZlZXMpKTtcbiAgICBib3hCdWlsZGVyLnNldF9yZWdpc3Rlcl92YWx1ZSg4LCBDb25zdGFudC5mcm9tX2pzKHJzblJhdGlvcykpO1xuICAgIGJveEJ1aWxkZXIuc2V0X3JlZ2lzdGVyX3ZhbHVlKDksIENvbnN0YW50LmZyb21fanMoZmVlUmF0aW9zKSk7XG5cbiAgICByZXR1cm4gYm94QnVpbGRlci5idWlsZCgpO1xuICB9O1xufVxuIl19
@@ -0,0 +1,22 @@
1
+ import { ChainFee, Fee } from './types';
2
+ export declare class MinimumFeeConfig {
3
+ protected fee: Fee;
4
+ constructor();
5
+ /**
6
+ * sets fee for a chain
7
+ * @param chain
8
+ * @param height
9
+ * @param chainFee
10
+ */
11
+ setChainConfig: (chain: string, height: number, chainFee: ChainFee | undefined) => MinimumFeeConfig;
12
+ /**
13
+ * removes fee for a chain
14
+ * @param chain
15
+ */
16
+ removeChainConfig: (chain: string) => MinimumFeeConfig;
17
+ /**
18
+ * returns generated fee
19
+ */
20
+ getConfig: () => Fee;
21
+ }
22
+ //# sourceMappingURL=MinimumFeeConfig.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"MinimumFeeConfig.d.ts","sourceRoot":"","sources":["../../lib/MinimumFeeConfig.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,GAAG,EAAE,MAAM,SAAS,CAAC;AAExC,qBAAa,gBAAgB;IAC3B,SAAS,CAAC,GAAG,EAAE,GAAG,CAAC;;IASnB;;;;;OAKG;IACH,cAAc,UACL,MAAM,UACL,MAAM,YACJ,QAAQ,GAAG,SAAS,KAC7B,gBAAgB,CAKjB;IAEF;;;OAGG;IACH,iBAAiB,UAAW,MAAM,KAAG,gBAAgB,CAInD;IAEF;;OAEG;IACH,SAAS,QAAO,GAAG,CAEjB;CACH"}