@bitgo-beta/sdk-coin-vet 1.0.1-beta.64 → 1.0.1-beta.640
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/src/index.d.ts +1 -0
- package/dist/src/index.d.ts.map +1 -1
- package/dist/src/index.js +2 -1
- package/dist/src/lib/constants.d.ts +28 -0
- package/dist/src/lib/constants.d.ts.map +1 -1
- package/dist/src/lib/constants.js +30 -2
- package/dist/src/lib/iface.d.ts +28 -2
- package/dist/src/lib/iface.d.ts.map +1 -1
- package/dist/src/lib/iface.js +1 -1
- package/dist/src/lib/index.d.ts +21 -0
- package/dist/src/lib/index.d.ts.map +1 -1
- package/dist/src/lib/index.js +44 -2
- package/dist/src/lib/transaction/addressInitializationTransaction.d.ts +31 -0
- package/dist/src/lib/transaction/addressInitializationTransaction.d.ts.map +1 -0
- package/dist/src/lib/transaction/addressInitializationTransaction.js +170 -0
- package/dist/src/lib/transaction/burnNftTransaction.d.ts +29 -0
- package/dist/src/lib/transaction/burnNftTransaction.d.ts.map +1 -0
- package/dist/src/lib/transaction/burnNftTransaction.js +138 -0
- package/dist/src/lib/transaction/claimRewards.d.ts +31 -0
- package/dist/src/lib/transaction/claimRewards.d.ts.map +1 -0
- package/dist/src/lib/transaction/claimRewards.js +151 -0
- package/dist/src/lib/transaction/delegateClauseTransaction.d.ts +27 -0
- package/dist/src/lib/transaction/delegateClauseTransaction.d.ts.map +1 -0
- package/dist/src/lib/transaction/delegateClauseTransaction.js +158 -0
- package/dist/src/lib/transaction/exitDelegation.d.ts +29 -0
- package/dist/src/lib/transaction/exitDelegation.d.ts.map +1 -0
- package/dist/src/lib/transaction/exitDelegation.js +143 -0
- package/dist/src/lib/transaction/flushTokenTransaction.d.ts +20 -0
- package/dist/src/lib/transaction/flushTokenTransaction.d.ts.map +1 -0
- package/dist/src/lib/transaction/flushTokenTransaction.js +98 -0
- package/dist/src/lib/transaction/nftTransaction.d.ts +17 -0
- package/dist/src/lib/transaction/nftTransaction.d.ts.map +1 -0
- package/dist/src/lib/transaction/nftTransaction.js +108 -0
- package/dist/src/lib/transaction/stakeClauseTransaction.d.ts +27 -0
- package/dist/src/lib/transaction/stakeClauseTransaction.d.ts.map +1 -0
- package/dist/src/lib/transaction/stakeClauseTransaction.js +158 -0
- package/dist/src/lib/transaction/stakingTransaction.d.ts +31 -0
- package/dist/src/lib/transaction/stakingTransaction.d.ts.map +1 -0
- package/dist/src/lib/transaction/stakingTransaction.js +169 -0
- package/dist/src/lib/transaction/tokenTransaction.d.ts +14 -0
- package/dist/src/lib/transaction/tokenTransaction.d.ts.map +1 -0
- package/dist/src/lib/transaction/tokenTransaction.js +95 -0
- package/dist/src/lib/transaction/transaction.d.ts +18 -28
- package/dist/src/lib/transaction/transaction.d.ts.map +1 -1
- package/dist/src/lib/transaction/transaction.js +99 -58
- package/dist/src/lib/transaction/validatorRegistrationTransaction.d.ts +32 -0
- package/dist/src/lib/transaction/validatorRegistrationTransaction.d.ts.map +1 -0
- package/dist/src/lib/transaction/validatorRegistrationTransaction.js +170 -0
- package/dist/src/lib/transactionBuilder/addressInitializationBuilder.d.ts +78 -0
- package/dist/src/lib/transactionBuilder/addressInitializationBuilder.d.ts.map +1 -0
- package/dist/src/lib/transactionBuilder/addressInitializationBuilder.js +158 -0
- package/dist/src/lib/transactionBuilder/burnNftBuilder.d.ts +59 -0
- package/dist/src/lib/transactionBuilder/burnNftBuilder.d.ts.map +1 -0
- package/dist/src/lib/transactionBuilder/burnNftBuilder.js +118 -0
- package/dist/src/lib/transactionBuilder/claimRewardsBuilder.d.ts +59 -0
- package/dist/src/lib/transactionBuilder/claimRewardsBuilder.d.ts.map +1 -0
- package/dist/src/lib/transactionBuilder/claimRewardsBuilder.js +117 -0
- package/dist/src/lib/transactionBuilder/delegateTxnBuilder.d.ts +72 -0
- package/dist/src/lib/transactionBuilder/delegateTxnBuilder.d.ts.map +1 -0
- package/dist/src/lib/transactionBuilder/delegateTxnBuilder.js +133 -0
- package/dist/src/lib/transactionBuilder/exitDelegationBuilder.d.ts +59 -0
- package/dist/src/lib/transactionBuilder/exitDelegationBuilder.d.ts.map +1 -0
- package/dist/src/lib/transactionBuilder/exitDelegationBuilder.js +118 -0
- package/dist/src/lib/transactionBuilder/flushTokenTransactionBuilder.d.ts +66 -0
- package/dist/src/lib/transactionBuilder/flushTokenTransactionBuilder.d.ts.map +1 -0
- package/dist/src/lib/transactionBuilder/flushTokenTransactionBuilder.js +129 -0
- package/dist/src/lib/transactionBuilder/nftTransactionBuilder.d.ts +23 -0
- package/dist/src/lib/transactionBuilder/nftTransactionBuilder.d.ts.map +1 -0
- package/dist/src/lib/transactionBuilder/nftTransactionBuilder.js +93 -0
- package/dist/src/lib/transactionBuilder/stakeClauseTxnBuilder.d.ts +73 -0
- package/dist/src/lib/transactionBuilder/stakeClauseTxnBuilder.d.ts.map +1 -0
- package/dist/src/lib/transactionBuilder/stakeClauseTxnBuilder.js +146 -0
- package/dist/src/lib/transactionBuilder/stakingBuilder.d.ts +73 -0
- package/dist/src/lib/transactionBuilder/stakingBuilder.d.ts.map +1 -0
- package/dist/src/lib/transactionBuilder/stakingBuilder.js +147 -0
- package/dist/src/lib/transactionBuilder/tokenTransactionBuilder.d.ts +21 -0
- package/dist/src/lib/transactionBuilder/tokenTransactionBuilder.d.ts.map +1 -0
- package/dist/src/lib/transactionBuilder/tokenTransactionBuilder.js +70 -0
- package/dist/src/lib/transactionBuilder/transactionBuilder.d.ts +10 -9
- package/dist/src/lib/transactionBuilder/transactionBuilder.d.ts.map +1 -1
- package/dist/src/lib/transactionBuilder/transactionBuilder.js +25 -19
- package/dist/src/lib/transactionBuilder/validatorRegistrationBuilder.d.ts +79 -0
- package/dist/src/lib/transactionBuilder/validatorRegistrationBuilder.d.ts.map +1 -0
- package/dist/src/lib/transactionBuilder/validatorRegistrationBuilder.js +150 -0
- package/dist/src/lib/transactionBuilderFactory.d.ts +55 -0
- package/dist/src/lib/transactionBuilderFactory.d.ts.map +1 -1
- package/dist/src/lib/transactionBuilderFactory.js +124 -1
- package/dist/src/lib/types.d.ts +21 -0
- package/dist/src/lib/types.d.ts.map +1 -1
- package/dist/src/lib/types.js +1 -1
- package/dist/src/lib/utils.d.ts +102 -1
- package/dist/src/lib/utils.d.ts.map +1 -1
- package/dist/src/lib/utils.js +277 -8
- package/dist/src/register.d.ts.map +1 -1
- package/dist/src/register.js +5 -1
- package/dist/src/vet.d.ts +94 -3
- package/dist/src/vet.d.ts.map +1 -1
- package/dist/src/vet.js +505 -10
- package/dist/src/vetNFTCollection.d.ts +18 -0
- package/dist/src/vetNFTCollection.d.ts.map +1 -0
- package/dist/src/vetNFTCollection.js +52 -0
- package/dist/src/vetToken.d.ts +1 -1
- package/dist/src/vetToken.d.ts.map +1 -1
- package/dist/src/vetToken.js +2 -2
- package/dist/test/resources/vet.d.ts +88 -0
- package/dist/test/resources/vet.d.ts.map +1 -0
- package/dist/test/resources/vet.js +143 -0
- package/dist/test/transactionBuilder/addressInitializationBuilder.d.ts +2 -0
- package/dist/test/transactionBuilder/addressInitializationBuilder.d.ts.map +1 -0
- package/dist/test/transactionBuilder/addressInitializationBuilder.js +141 -0
- package/dist/test/transactionBuilder/burnNftBuilder.d.ts +2 -0
- package/dist/test/transactionBuilder/burnNftBuilder.d.ts.map +1 -0
- package/dist/test/transactionBuilder/burnNftBuilder.js +150 -0
- package/dist/test/transactionBuilder/claimRewardsBuilder.d.ts +2 -0
- package/dist/test/transactionBuilder/claimRewardsBuilder.d.ts.map +1 -0
- package/dist/test/transactionBuilder/claimRewardsBuilder.js +183 -0
- package/dist/test/transactionBuilder/delegateClauseTxnBuilder.d.ts +2 -0
- package/dist/test/transactionBuilder/delegateClauseTxnBuilder.d.ts.map +1 -0
- package/dist/test/transactionBuilder/delegateClauseTxnBuilder.js +209 -0
- package/dist/test/transactionBuilder/exitDelegationBuilder.d.ts +2 -0
- package/dist/test/transactionBuilder/exitDelegationBuilder.d.ts.map +1 -0
- package/dist/test/transactionBuilder/exitDelegationBuilder.js +150 -0
- package/dist/test/transactionBuilder/flushTokenTransactionBuilder.d.ts +2 -0
- package/dist/test/transactionBuilder/flushTokenTransactionBuilder.d.ts.map +1 -0
- package/dist/test/transactionBuilder/flushTokenTransactionBuilder.js +132 -0
- package/dist/test/transactionBuilder/nftTransactionBuilder.d.ts +2 -0
- package/dist/test/transactionBuilder/nftTransactionBuilder.d.ts.map +1 -0
- package/dist/test/transactionBuilder/nftTransactionBuilder.js +242 -0
- package/dist/test/transactionBuilder/stakeClauseTransactionBuilder.d.ts +2 -0
- package/dist/test/transactionBuilder/stakeClauseTransactionBuilder.d.ts.map +1 -0
- package/dist/test/transactionBuilder/stakeClauseTransactionBuilder.js +247 -0
- package/dist/test/transactionBuilder/stakingTransaction.d.ts +2 -0
- package/dist/test/transactionBuilder/stakingTransaction.d.ts.map +1 -0
- package/dist/test/transactionBuilder/stakingTransaction.js +250 -0
- package/dist/test/transactionBuilder/tokenTransactionBuilder.d.ts +2 -0
- package/dist/test/transactionBuilder/tokenTransactionBuilder.d.ts.map +1 -0
- package/dist/test/transactionBuilder/tokenTransactionBuilder.js +249 -0
- package/dist/test/transactionBuilder/transferBuilder.d.ts +2 -0
- package/dist/test/transactionBuilder/transferBuilder.d.ts.map +1 -0
- package/dist/test/transactionBuilder/transferBuilder.js +244 -0
- package/dist/test/transactionBuilder/validatorRegistrationTxnBuilder.d.ts +2 -0
- package/dist/test/transactionBuilder/validatorRegistrationTxnBuilder.d.ts.map +1 -0
- package/dist/test/transactionBuilder/validatorRegistrationTxnBuilder.js +231 -0
- package/dist/test/unit/keyPair.d.ts +2 -0
- package/dist/test/unit/keyPair.d.ts.map +1 -0
- package/dist/test/unit/keyPair.js +181 -0
- package/dist/test/unit/stakingFlowE2E.d.ts +2 -0
- package/dist/test/unit/stakingFlowE2E.d.ts.map +1 -0
- package/dist/test/unit/stakingFlowE2E.js +292 -0
- package/dist/test/unit/utils.d.ts +2 -0
- package/dist/test/unit/utils.d.ts.map +1 -0
- package/dist/test/unit/utils.js +106 -0
- package/dist/test/unit/vet.d.ts +2 -0
- package/dist/test/unit/vet.d.ts.map +1 -0
- package/dist/test/unit/vet.js +336 -0
- package/dist/tsconfig.tsbuildinfo +1 -0
- package/package.json +16 -10
- package/.eslintignore +0 -5
- package/.mocharc.yml +0 -8
- package/CHANGELOG.md +0 -40
package/dist/src/vet.js
CHANGED
|
@@ -39,18 +39,25 @@ Object.defineProperty(exports, "__esModule", { value: true });
|
|
|
39
39
|
exports.Vet = void 0;
|
|
40
40
|
const _ = __importStar(require("lodash"));
|
|
41
41
|
const bignumber_js_1 = __importDefault(require("bignumber.js"));
|
|
42
|
-
const
|
|
42
|
+
const blake2b_1 = __importDefault(require("@bitgo-beta/blake2b"));
|
|
43
|
+
const assert_1 = __importDefault(require("assert"));
|
|
44
|
+
const axios_1 = __importDefault(require("axios"));
|
|
45
|
+
const sdk_core_1 = require("@vechain/sdk-core");
|
|
46
|
+
const sdk_core_2 = require("@bitgo-beta/sdk-core");
|
|
47
|
+
const mpc = __importStar(require("@bitgo-beta/sdk-lib-mpc"));
|
|
48
|
+
const statics_1 = require("@bitgo-beta/statics");
|
|
43
49
|
const utils_1 = __importDefault(require("./lib/utils"));
|
|
44
50
|
const secp256k1_1 = require("@bitgo-beta/secp256k1");
|
|
45
51
|
const crypto_1 = require("crypto");
|
|
46
52
|
const abstract_eth_1 = require("@bitgo-beta/abstract-eth");
|
|
47
53
|
const lib_1 = require("./lib");
|
|
54
|
+
const constants_1 = require("./lib/constants");
|
|
48
55
|
/**
|
|
49
56
|
* Full Name: Vechain
|
|
50
57
|
* Docs: https://docs.vechain.org/
|
|
51
58
|
* GitHub : https://github.com/vechain
|
|
52
59
|
*/
|
|
53
|
-
class Vet extends
|
|
60
|
+
class Vet extends sdk_core_2.BaseCoin {
|
|
54
61
|
constructor(bitgo, staticsCoin) {
|
|
55
62
|
super(bitgo);
|
|
56
63
|
if (!staticsCoin) {
|
|
@@ -76,13 +83,16 @@ class Vet extends sdk_core_1.BaseCoin {
|
|
|
76
83
|
getFullName() {
|
|
77
84
|
return 'VeChain';
|
|
78
85
|
}
|
|
86
|
+
valuelessTransferAllowed() {
|
|
87
|
+
return true;
|
|
88
|
+
}
|
|
79
89
|
/** @inheritDoc */
|
|
80
90
|
supportsTss() {
|
|
81
91
|
return true;
|
|
82
92
|
}
|
|
83
93
|
/** inherited doc */
|
|
84
94
|
getDefaultMultisigType() {
|
|
85
|
-
return
|
|
95
|
+
return sdk_core_2.multisigTypes.tss;
|
|
86
96
|
}
|
|
87
97
|
getMPCAlgorithm() {
|
|
88
98
|
return 'ecdsa';
|
|
@@ -100,16 +110,16 @@ class Vet extends sdk_core_1.BaseCoin {
|
|
|
100
110
|
if (!explainedTx) {
|
|
101
111
|
throw new Error('failed to explain transaction');
|
|
102
112
|
}
|
|
103
|
-
if (txParams.recipients !== undefined) {
|
|
113
|
+
if (txParams.recipients !== undefined && txParams.recipients.length > 0) {
|
|
104
114
|
const filteredRecipients = txParams.recipients?.map((recipient) => {
|
|
105
115
|
return {
|
|
106
|
-
address: recipient.address,
|
|
116
|
+
address: recipient.address.toLowerCase(),
|
|
107
117
|
amount: BigInt(recipient.amount),
|
|
108
118
|
};
|
|
109
119
|
});
|
|
110
120
|
const filteredOutputs = explainedTx.outputs.map((output) => {
|
|
111
121
|
return {
|
|
112
|
-
address: output.address,
|
|
122
|
+
address: output.address.toLowerCase(),
|
|
113
123
|
amount: BigInt(output.amount),
|
|
114
124
|
};
|
|
115
125
|
});
|
|
@@ -126,10 +136,35 @@ class Vet extends sdk_core_1.BaseCoin {
|
|
|
126
136
|
}
|
|
127
137
|
return true;
|
|
128
138
|
}
|
|
139
|
+
/**
|
|
140
|
+
* Verify that an address belongs to this wallet.
|
|
141
|
+
*
|
|
142
|
+
* @param {TssVerifyVetAddressOptions} params - Verification parameters
|
|
143
|
+
* @returns {Promise<boolean>} True if address belongs to wallet
|
|
144
|
+
* @throws {InvalidAddressError} If address format is invalid
|
|
145
|
+
* @throws {Error} If invalid wallet version or missing parameters
|
|
146
|
+
*/
|
|
129
147
|
async isWalletAddress(params) {
|
|
130
|
-
const { address
|
|
131
|
-
if (!this.isValidAddress(
|
|
132
|
-
throw new
|
|
148
|
+
const { address, baseAddress, walletVersion } = params;
|
|
149
|
+
if (address && !this.isValidAddress(address)) {
|
|
150
|
+
throw new sdk_core_2.InvalidAddressError(`invalid address: ${address}`);
|
|
151
|
+
}
|
|
152
|
+
if (walletVersion !== 6) {
|
|
153
|
+
throw new Error(`VET only supports wallet version 6, but got version ${walletVersion}`);
|
|
154
|
+
}
|
|
155
|
+
const isVerifyingBaseAddress = baseAddress && address.toLowerCase() === baseAddress.toLowerCase();
|
|
156
|
+
if (isVerifyingBaseAddress) {
|
|
157
|
+
const index = typeof params.index === 'string' ? parseInt(params.index, 10) : params.index;
|
|
158
|
+
if (index !== 0) {
|
|
159
|
+
throw new Error(`Base address verification requires index 0, but got index ${params.index}.`);
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
const result = await (0, sdk_core_2.verifyMPCWalletAddress)({ ...params, keyCurve: 'secp256k1' }, this.isValidAddress.bind(this), (pubKey) => {
|
|
163
|
+
const keyPair = new abstract_eth_1.KeyPair({ pub: pubKey });
|
|
164
|
+
return keyPair.getAddress();
|
|
165
|
+
});
|
|
166
|
+
if (!result) {
|
|
167
|
+
throw new sdk_core_2.InvalidAddressError(`invalid address: ${address}`);
|
|
133
168
|
}
|
|
134
169
|
return true;
|
|
135
170
|
}
|
|
@@ -215,6 +250,466 @@ class Vet extends sdk_core_1.BaseCoin {
|
|
|
215
250
|
/** https://bitgoinc.atlassian.net/browse/COIN-4213 */
|
|
216
251
|
throw new Error('Method not implemented.');
|
|
217
252
|
}
|
|
253
|
+
/**
|
|
254
|
+
* Function to get coin specific hash function used to generate transaction digests.
|
|
255
|
+
* @returns {@see Hash} hash function if implemented, otherwise throws exception
|
|
256
|
+
*/
|
|
257
|
+
getHashFunction() {
|
|
258
|
+
const blake = (0, blake2b_1.default)(32);
|
|
259
|
+
// We return an object that mimics the Hash interface
|
|
260
|
+
return {
|
|
261
|
+
update(data) {
|
|
262
|
+
blake.update(data);
|
|
263
|
+
return this;
|
|
264
|
+
},
|
|
265
|
+
digest() {
|
|
266
|
+
const uint8Result = blake.digest();
|
|
267
|
+
return Buffer.from(uint8Result);
|
|
268
|
+
},
|
|
269
|
+
};
|
|
270
|
+
}
|
|
271
|
+
buildNftTransferData(params) {
|
|
272
|
+
const { recipientAddress, fromAddress, tokenContractAddress } = params;
|
|
273
|
+
if (!utils_1.default.isValidAddress(recipientAddress)) {
|
|
274
|
+
throw new sdk_core_2.InvalidAddressError('Invalid recipient address');
|
|
275
|
+
}
|
|
276
|
+
if (!utils_1.default.isValidAddress(fromAddress)) {
|
|
277
|
+
throw new sdk_core_2.InvalidAddressError('Invalid from address');
|
|
278
|
+
}
|
|
279
|
+
if (!utils_1.default.isValidAddress(tokenContractAddress)) {
|
|
280
|
+
throw new sdk_core_2.InvalidAddressError('Invalid NFT contract address address');
|
|
281
|
+
}
|
|
282
|
+
switch (params.type) {
|
|
283
|
+
case 'ERC721': {
|
|
284
|
+
const tokenId = params.tokenId;
|
|
285
|
+
return {
|
|
286
|
+
tokenType: sdk_core_2.TokenType.ERC721,
|
|
287
|
+
tokenQuantity: '1', // This NFT standard will always have quantity of 1
|
|
288
|
+
tokenContractAddress,
|
|
289
|
+
tokenId,
|
|
290
|
+
};
|
|
291
|
+
}
|
|
292
|
+
default:
|
|
293
|
+
throw new sdk_core_2.NotImplementedError(`NFT type ${params.type} not supported on ${this.getChain()}`);
|
|
294
|
+
}
|
|
295
|
+
}
|
|
296
|
+
/**
|
|
297
|
+
* Broadcasts a signed transaction to the VeChain network.
|
|
298
|
+
*
|
|
299
|
+
* @param {BaseBroadcastTransactionOptions} payload - The payload containing the serialized signed transaction.
|
|
300
|
+
* @param {string} payload.serializedSignedTransaction - The serialized signed transaction to broadcast.
|
|
301
|
+
* @returns {Promise<BaseBroadcastTransactionResult>} A promise that resolves to an empty object if the broadcast is successful.
|
|
302
|
+
* @throws {Error} If the broadcast fails, an error is thrown with the failure message.
|
|
303
|
+
*/
|
|
304
|
+
async broadcastTransaction(payload) {
|
|
305
|
+
const baseUrl = this.getPublicNodeUrl();
|
|
306
|
+
const url = `${baseUrl}/transactions`;
|
|
307
|
+
// The body should be a JSON object with a 'raw' key
|
|
308
|
+
const requestBody = {
|
|
309
|
+
raw: payload.serializedSignedTransaction,
|
|
310
|
+
};
|
|
311
|
+
try {
|
|
312
|
+
await axios_1.default.post(url, requestBody);
|
|
313
|
+
return {};
|
|
314
|
+
}
|
|
315
|
+
catch (error) {
|
|
316
|
+
throw new Error(`Failed to broadcast transaction: ${error.message}`);
|
|
317
|
+
}
|
|
318
|
+
}
|
|
319
|
+
/** @inheritDoc */
|
|
320
|
+
async recover(params) {
|
|
321
|
+
if (params.tokenContractAddress) {
|
|
322
|
+
return this.recoverTokens(params);
|
|
323
|
+
}
|
|
324
|
+
try {
|
|
325
|
+
if (!params.recoveryDestination || !this.isValidAddress(params.recoveryDestination)) {
|
|
326
|
+
throw new Error('invalid recoveryDestination');
|
|
327
|
+
}
|
|
328
|
+
const isUnsignedSweep = !params.userKey && !params.backupKey && !params.walletPassphrase;
|
|
329
|
+
let publicKey;
|
|
330
|
+
let userKeyShare, backupKeyShare, commonKeyChain;
|
|
331
|
+
const MPC = new sdk_core_2.Ecdsa();
|
|
332
|
+
if (isUnsignedSweep) {
|
|
333
|
+
const bitgoKey = params.bitgoKey;
|
|
334
|
+
if (!bitgoKey) {
|
|
335
|
+
throw new Error('missing bitgoKey');
|
|
336
|
+
}
|
|
337
|
+
const hdTree = new mpc.Secp256k1Bip32HdTree();
|
|
338
|
+
const derivationPath = 'm/0';
|
|
339
|
+
const derivedPub = hdTree.publicDerive({
|
|
340
|
+
pk: mpc.bigIntFromBufferBE(Buffer.from(bitgoKey.slice(0, 66), 'hex')),
|
|
341
|
+
chaincode: mpc.bigIntFromBufferBE(Buffer.from(bitgoKey.slice(66), 'hex')),
|
|
342
|
+
}, derivationPath);
|
|
343
|
+
publicKey = mpc.bigIntToBufferBE(derivedPub.pk).toString('hex');
|
|
344
|
+
}
|
|
345
|
+
else {
|
|
346
|
+
if (!params.userKey) {
|
|
347
|
+
throw new Error('missing userKey');
|
|
348
|
+
}
|
|
349
|
+
if (!params.backupKey) {
|
|
350
|
+
throw new Error('missing backupKey');
|
|
351
|
+
}
|
|
352
|
+
if (!params.walletPassphrase) {
|
|
353
|
+
throw new Error('missing wallet passphrase');
|
|
354
|
+
}
|
|
355
|
+
const userKey = params.userKey.replace(/\s/g, '');
|
|
356
|
+
const backupKey = params.backupKey.replace(/\s/g, '');
|
|
357
|
+
({ userKeyShare, backupKeyShare, commonKeyChain } = await sdk_core_2.ECDSAUtils.getMpcV2RecoveryKeyShares(userKey, backupKey, params.walletPassphrase));
|
|
358
|
+
publicKey = MPC.deriveUnhardened(commonKeyChain, 'm/0').slice(0, 66);
|
|
359
|
+
}
|
|
360
|
+
if (!publicKey) {
|
|
361
|
+
throw new Error('failed to derive public key');
|
|
362
|
+
}
|
|
363
|
+
const backupKeyPair = new abstract_eth_1.KeyPair({ pub: publicKey });
|
|
364
|
+
const baseAddress = backupKeyPair.getAddress();
|
|
365
|
+
const tx = await this.buildRecoveryTransaction({
|
|
366
|
+
baseAddress,
|
|
367
|
+
params,
|
|
368
|
+
});
|
|
369
|
+
const signableHex = await tx.signablePayload;
|
|
370
|
+
const serializedTxHex = await tx.toBroadcastFormat();
|
|
371
|
+
if (isUnsignedSweep) {
|
|
372
|
+
return {
|
|
373
|
+
txHex: serializedTxHex,
|
|
374
|
+
coin: this.getChain(),
|
|
375
|
+
};
|
|
376
|
+
}
|
|
377
|
+
const signableMessage = this.getHashFunction().update(signableHex).digest();
|
|
378
|
+
const signatureObj = await sdk_core_2.ECDSAUtils.signRecoveryMpcV2(signableMessage, userKeyShare, backupKeyShare, commonKeyChain);
|
|
379
|
+
const signature = Buffer.from(signatureObj.r + signatureObj.s + (signatureObj.recid === 0 ? '00' : '01'), 'hex');
|
|
380
|
+
const txBuilder = this.getTxBuilderFactory().getTransferBuilder();
|
|
381
|
+
await txBuilder.from(serializedTxHex);
|
|
382
|
+
txBuilder.isRecovery(true);
|
|
383
|
+
await txBuilder.addSenderSignature(signature);
|
|
384
|
+
const signedTx = await txBuilder.build();
|
|
385
|
+
return {
|
|
386
|
+
id: signedTx.id,
|
|
387
|
+
tx: signedTx.toBroadcastFormat(),
|
|
388
|
+
};
|
|
389
|
+
}
|
|
390
|
+
catch (error) {
|
|
391
|
+
throw new Error(`Error during Vechain recovery: ${error.message || error}`);
|
|
392
|
+
}
|
|
393
|
+
}
|
|
394
|
+
/**
|
|
395
|
+
* Returns the public node URL for the VeChain network.
|
|
396
|
+
* @returns {string} The URL of the public VeChain node.
|
|
397
|
+
*/
|
|
398
|
+
getPublicNodeUrl() {
|
|
399
|
+
return sdk_core_2.Environments[this.bitgo.getEnv()].vetNodeUrl;
|
|
400
|
+
}
|
|
401
|
+
/**
|
|
402
|
+
* Calculates the transaction fee based on the estimated gas limit and fee estimate data.
|
|
403
|
+
* @param {FeeEstimateData} feeEstimateData - The fee estimate data.
|
|
404
|
+
* @param {BigNumber} estimatedGasLimit - The estimated gas limit for the transaction.
|
|
405
|
+
* @returns {BigNumber} The calculated transaction fee.
|
|
406
|
+
*/
|
|
407
|
+
calculateFee(feeEstimateData, estimatedGasLimit) {
|
|
408
|
+
const gasLimit = estimatedGasLimit;
|
|
409
|
+
const adjustmentFactor = new bignumber_js_1.default(1).plus(new bignumber_js_1.default(feeEstimateData.gasPriceCoef)
|
|
410
|
+
.dividedBy(new bignumber_js_1.default(feeEstimateData.coefDivisor))
|
|
411
|
+
.decimalPlaces(18, bignumber_js_1.default.ROUND_DOWN));
|
|
412
|
+
const adjustedGasPrice = new bignumber_js_1.default(feeEstimateData.gasUnitPrice).times(adjustmentFactor);
|
|
413
|
+
return gasLimit.times(adjustedGasPrice).integerValue(bignumber_js_1.default.ROUND_CEIL);
|
|
414
|
+
}
|
|
415
|
+
/**
|
|
416
|
+
* Ensures that the given address has sufficient VTHO balance to cover the transaction fee.
|
|
417
|
+
* @param {string} baseAddress - The address to check for VTHO balance.
|
|
418
|
+
* @param {BigNumber} requiredGasUnits - The required gas units for the transaction.
|
|
419
|
+
* @throws {Error} If the VTHO balance is insufficient or if there's an error checking the balance.
|
|
420
|
+
*/
|
|
421
|
+
async ensureVthoBalanceForFee(baseAddress, requiredGasUnits) {
|
|
422
|
+
const vthoTokenAddress = '0x0000000000000000000000000000456E65726779'; // VTHO token contract address
|
|
423
|
+
try {
|
|
424
|
+
const vthoBalance = await this.getBalance(baseAddress, vthoTokenAddress);
|
|
425
|
+
const requiredFee = this.calculateFee(constants_1.feeEstimateData, requiredGasUnits);
|
|
426
|
+
if (vthoBalance.isLessThan(requiredFee)) {
|
|
427
|
+
throw new Error(`Insufficient VTHO balance for fees. Required: ${requiredFee.toString()}, Available: ${vthoBalance.toString()}`);
|
|
428
|
+
}
|
|
429
|
+
}
|
|
430
|
+
catch (error) {
|
|
431
|
+
throw new Error(`Failed to ensure VTHO balance: ${error.message}`);
|
|
432
|
+
}
|
|
433
|
+
}
|
|
434
|
+
/**
|
|
435
|
+
* Fetches the balance for a given Vechain address.
|
|
436
|
+
*
|
|
437
|
+
* @param address The Vechain address (e.g., "0x...") to check.
|
|
438
|
+
* @param tokenContractAddress (Optional) The contract address of a VIP180 token.
|
|
439
|
+
* @returns A Promise that resolves to a BigNumber instance of the balance.
|
|
440
|
+
*/
|
|
441
|
+
async getBalance(address, tokenContractAddress) {
|
|
442
|
+
const baseUrl = this.getPublicNodeUrl();
|
|
443
|
+
if (!tokenContractAddress) {
|
|
444
|
+
const url = `${baseUrl}/accounts/${address}`;
|
|
445
|
+
try {
|
|
446
|
+
const response = await axios_1.default.get(url);
|
|
447
|
+
// The 'balance' is returned as a hex string.
|
|
448
|
+
const balance = new bignumber_js_1.default(response.data.balance);
|
|
449
|
+
return balance;
|
|
450
|
+
}
|
|
451
|
+
catch (error) {
|
|
452
|
+
throw new Error('Failed to get native balance.');
|
|
453
|
+
}
|
|
454
|
+
}
|
|
455
|
+
const url = `${baseUrl}/accounts/*`;
|
|
456
|
+
// Construct the ABI-encoded data for the 'balanceOf(address)' call
|
|
457
|
+
// 1. Function selector for 'balanceOf(address)': '0x70a08231'
|
|
458
|
+
// 2. Padded address: The address, stripped of '0x', left-padded with zeros to 64 chars
|
|
459
|
+
const paddedAddress = address.startsWith('0x') ? address.substring(2).padStart(64, '0') : address.padStart(64, '0');
|
|
460
|
+
const data = `0x70a08231${paddedAddress}`;
|
|
461
|
+
const requestBody = {
|
|
462
|
+
clauses: [
|
|
463
|
+
{
|
|
464
|
+
to: tokenContractAddress, // The token contract address
|
|
465
|
+
value: '0x0',
|
|
466
|
+
data: data, // The 'balanceOf' call
|
|
467
|
+
},
|
|
468
|
+
],
|
|
469
|
+
};
|
|
470
|
+
try {
|
|
471
|
+
const response = await axios_1.default.post(url, requestBody);
|
|
472
|
+
const simResponse = response.data;
|
|
473
|
+
// Validate response and extract the balance data
|
|
474
|
+
if (!simResponse || !Array.isArray(simResponse) || simResponse.length === 0 || !simResponse[0].data) {
|
|
475
|
+
throw new Error('Invalid simulation response from VeChain node');
|
|
476
|
+
}
|
|
477
|
+
// The returned data is the hex-encoded balance
|
|
478
|
+
return new bignumber_js_1.default(simResponse[0].data);
|
|
479
|
+
}
|
|
480
|
+
catch (error) {
|
|
481
|
+
console.error('Error fetching token balance:', error);
|
|
482
|
+
throw new Error(`Failed to get token balance: ${error.message}`);
|
|
483
|
+
}
|
|
484
|
+
}
|
|
485
|
+
/**
|
|
486
|
+
* Retrieves the block reference from the VeChain network.
|
|
487
|
+
* @returns {Promise<string>} A promise that resolves to the block reference string.
|
|
488
|
+
* @throws {Error} If there's an error fetching the block reference or if the response is invalid.
|
|
489
|
+
*/
|
|
490
|
+
async getBlockRef() {
|
|
491
|
+
const baseUrl = this.getPublicNodeUrl();
|
|
492
|
+
const url = `${baseUrl}/blocks/best`;
|
|
493
|
+
try {
|
|
494
|
+
const response = await axios_1.default.get(url);
|
|
495
|
+
const data = response.data;
|
|
496
|
+
// Validate the response data
|
|
497
|
+
if (!data || !data.id) {
|
|
498
|
+
throw new Error('Invalid response from the VeChain node');
|
|
499
|
+
}
|
|
500
|
+
// Return the first 18 characters of the block ID
|
|
501
|
+
return data.id.slice(0, 18);
|
|
502
|
+
}
|
|
503
|
+
catch (error) {
|
|
504
|
+
// Rethrow or return a sensible default
|
|
505
|
+
throw new Error('Failed to get block ref: ');
|
|
506
|
+
}
|
|
507
|
+
}
|
|
508
|
+
/**
|
|
509
|
+
* Generates a random nonce for use in transactions.
|
|
510
|
+
* @returns {string} A hexadecimal string representing the random nonce.
|
|
511
|
+
*/
|
|
512
|
+
getRandomNonce() {
|
|
513
|
+
return '0x' + (0, crypto_1.randomBytes)(8).toString('hex');
|
|
514
|
+
}
|
|
515
|
+
/**
|
|
516
|
+
* Estimates the gas required for a set of transaction clauses.
|
|
517
|
+
* @param {TransactionClause[]} clauses - An array of transaction clauses.
|
|
518
|
+
* @param {string} caller - The address of the transaction caller.
|
|
519
|
+
* @returns {Promise<BigNumber>} A promise that resolves to the estimated gas amount.
|
|
520
|
+
* @throws {Error} If the clauses are invalid, the caller is not provided, or if there's an error in gas estimation.
|
|
521
|
+
*/
|
|
522
|
+
async estimateGas(clauses, caller) {
|
|
523
|
+
if (!clauses || !Array.isArray(clauses) || clauses.length === 0) {
|
|
524
|
+
throw new Error('Clauses must be a non-empty array');
|
|
525
|
+
}
|
|
526
|
+
if (!caller) {
|
|
527
|
+
throw new Error('Caller address is required');
|
|
528
|
+
}
|
|
529
|
+
const baseUrl = this.getPublicNodeUrl();
|
|
530
|
+
const url = `${baseUrl}/accounts/*`;
|
|
531
|
+
const requestBody = {
|
|
532
|
+
clauses: clauses,
|
|
533
|
+
caller: caller,
|
|
534
|
+
};
|
|
535
|
+
try {
|
|
536
|
+
const response = await axios_1.default.post(url, requestBody);
|
|
537
|
+
const simResponse = response.data;
|
|
538
|
+
if (!simResponse || !Array.isArray(simResponse)) {
|
|
539
|
+
throw new Error('Invalid simulation response from VeChain node');
|
|
540
|
+
}
|
|
541
|
+
const totalSimulatedGas = simResponse.reduce((sum, result) => sum + (result.gasUsed || 0), 0);
|
|
542
|
+
const intrinsicGas = Number(sdk_core_1.Transaction.intrinsicGas(clauses).wei);
|
|
543
|
+
const totalGas = Math.ceil(intrinsicGas + (totalSimulatedGas !== 0 ? totalSimulatedGas + 15000 : 0));
|
|
544
|
+
return new bignumber_js_1.default(totalGas);
|
|
545
|
+
}
|
|
546
|
+
catch (error) {
|
|
547
|
+
throw new Error(`Failed to estimate gas: ${error.message}`);
|
|
548
|
+
}
|
|
549
|
+
}
|
|
550
|
+
/**
|
|
551
|
+
* Builds a recovery transaction for the given address.
|
|
552
|
+
* @param {Object} buildParams - The parameters for building the recovery transaction.
|
|
553
|
+
* @param {string} buildParams.baseAddress - The address to recover funds from.
|
|
554
|
+
* @param {RecoverOptions} buildParams.params - The recovery options.
|
|
555
|
+
* @returns {Promise<Transaction>} A promise that resolves to the built recovery transaction.
|
|
556
|
+
* @throws {Error} If there's no VET balance to recover or if there's an error building the transaction.
|
|
557
|
+
*/
|
|
558
|
+
async buildRecoveryTransaction(buildParams) {
|
|
559
|
+
const { baseAddress, params } = buildParams;
|
|
560
|
+
const balance = await this.getBalance(baseAddress);
|
|
561
|
+
if (balance.isLessThanOrEqualTo(0)) {
|
|
562
|
+
throw new Error(`no VET balance to recover for address ${baseAddress}`);
|
|
563
|
+
}
|
|
564
|
+
const recipients = [
|
|
565
|
+
{
|
|
566
|
+
address: params.recoveryDestination,
|
|
567
|
+
amount: balance.toString(),
|
|
568
|
+
},
|
|
569
|
+
];
|
|
570
|
+
const blockRef = await this.getBlockRef();
|
|
571
|
+
const txBuilder = this.getTxBuilderFactory().getTransferBuilder();
|
|
572
|
+
txBuilder.chainTag(this.bitgo.getEnv() === 'prod' ? 0x4a : 0x27);
|
|
573
|
+
txBuilder.recipients(recipients);
|
|
574
|
+
txBuilder.sender(baseAddress);
|
|
575
|
+
txBuilder.addFeePayerAddress(baseAddress);
|
|
576
|
+
txBuilder.gas(Number(constants_1.AVG_GAS_UNITS));
|
|
577
|
+
txBuilder.blockRef(blockRef);
|
|
578
|
+
txBuilder.expiration(constants_1.EXPIRATION);
|
|
579
|
+
txBuilder.gasPriceCoef(Number(constants_1.GAS_PRICE_COEF));
|
|
580
|
+
txBuilder.nonce(this.getRandomNonce());
|
|
581
|
+
txBuilder.isRecovery(true);
|
|
582
|
+
let tx = (await txBuilder.build());
|
|
583
|
+
const clauses = tx.clauses;
|
|
584
|
+
const actualGasUnits = await this.estimateGas(clauses, baseAddress);
|
|
585
|
+
await this.ensureVthoBalanceForFee(baseAddress, actualGasUnits);
|
|
586
|
+
txBuilder.gas(actualGasUnits.toNumber());
|
|
587
|
+
tx = (await txBuilder.build());
|
|
588
|
+
return tx;
|
|
589
|
+
}
|
|
590
|
+
async recoverTokens(params) {
|
|
591
|
+
try {
|
|
592
|
+
if (!params.recoveryDestination || !this.isValidAddress(params.recoveryDestination)) {
|
|
593
|
+
throw new Error('invalid recoveryDestination');
|
|
594
|
+
}
|
|
595
|
+
if (!params.tokenContractAddress || !this.isValidAddress(params.tokenContractAddress)) {
|
|
596
|
+
throw new Error('invalid tokenContractAddress');
|
|
597
|
+
}
|
|
598
|
+
const isUnsignedSweep = !params.userKey && !params.backupKey && !params.walletPassphrase;
|
|
599
|
+
let publicKey;
|
|
600
|
+
let userKeyShare, backupKeyShare, commonKeyChain;
|
|
601
|
+
const MPC = new sdk_core_2.Ecdsa();
|
|
602
|
+
if (isUnsignedSweep) {
|
|
603
|
+
const bitgoKey = params.bitgoKey;
|
|
604
|
+
if (!bitgoKey) {
|
|
605
|
+
throw new Error('missing bitgoKey');
|
|
606
|
+
}
|
|
607
|
+
const hdTree = new mpc.Secp256k1Bip32HdTree();
|
|
608
|
+
const derivationPath = 'm/0';
|
|
609
|
+
const derivedPub = hdTree.publicDerive({
|
|
610
|
+
pk: mpc.bigIntFromBufferBE(Buffer.from(bitgoKey.slice(0, 66), 'hex')),
|
|
611
|
+
chaincode: mpc.bigIntFromBufferBE(Buffer.from(bitgoKey.slice(66), 'hex')),
|
|
612
|
+
}, derivationPath);
|
|
613
|
+
publicKey = mpc.bigIntToBufferBE(derivedPub.pk).toString('hex');
|
|
614
|
+
}
|
|
615
|
+
else {
|
|
616
|
+
if (!params.userKey) {
|
|
617
|
+
throw new Error('missing userKey');
|
|
618
|
+
}
|
|
619
|
+
if (!params.backupKey) {
|
|
620
|
+
throw new Error('missing backupKey');
|
|
621
|
+
}
|
|
622
|
+
if (!params.walletPassphrase) {
|
|
623
|
+
throw new Error('missing wallet passphrase');
|
|
624
|
+
}
|
|
625
|
+
const userKey = params.userKey.replace(/\s/g, '');
|
|
626
|
+
const backupKey = params.backupKey.replace(/\s/g, '');
|
|
627
|
+
({ userKeyShare, backupKeyShare, commonKeyChain } = await sdk_core_2.ECDSAUtils.getMpcV2RecoveryKeyShares(userKey, backupKey, params.walletPassphrase));
|
|
628
|
+
publicKey = MPC.deriveUnhardened(commonKeyChain, 'm/0').slice(0, 66);
|
|
629
|
+
}
|
|
630
|
+
if (!publicKey) {
|
|
631
|
+
throw new Error('failed to derive public key');
|
|
632
|
+
}
|
|
633
|
+
const backupKeyPair = new abstract_eth_1.KeyPair({ pub: publicKey });
|
|
634
|
+
const baseAddress = backupKeyPair.getAddress();
|
|
635
|
+
const tx = await this.buildTokenRecoveryTransaction({
|
|
636
|
+
baseAddress,
|
|
637
|
+
params,
|
|
638
|
+
});
|
|
639
|
+
const signableHex = await tx.signablePayload;
|
|
640
|
+
const serializedTxHex = await tx.toBroadcastFormat();
|
|
641
|
+
if (isUnsignedSweep) {
|
|
642
|
+
return {
|
|
643
|
+
txHex: serializedTxHex,
|
|
644
|
+
coin: this.getChain(),
|
|
645
|
+
};
|
|
646
|
+
}
|
|
647
|
+
const signableMessage = this.getHashFunction().update(signableHex).digest();
|
|
648
|
+
const signatureObj = await sdk_core_2.ECDSAUtils.signRecoveryMpcV2(signableMessage, userKeyShare, backupKeyShare, commonKeyChain);
|
|
649
|
+
const signature = Buffer.from(signatureObj.r + signatureObj.s + (signatureObj.recid === 0 ? '00' : '01'), 'hex');
|
|
650
|
+
const tokenTransaction = new lib_1.TokenTransaction(statics_1.coins.get(this.getChain()));
|
|
651
|
+
const txBuilder = this.getTxBuilderFactory().getTokenTransactionBuilder(tokenTransaction);
|
|
652
|
+
await txBuilder.from(serializedTxHex);
|
|
653
|
+
txBuilder.isRecovery(true);
|
|
654
|
+
await txBuilder.addSenderSignature(signature);
|
|
655
|
+
const signedTx = await txBuilder.build();
|
|
656
|
+
return {
|
|
657
|
+
id: signedTx.id,
|
|
658
|
+
tx: signedTx.toBroadcastFormat(),
|
|
659
|
+
};
|
|
660
|
+
}
|
|
661
|
+
catch (error) {
|
|
662
|
+
throw new Error(`Error during Vechain token recovery: ${error.message || error}`);
|
|
663
|
+
}
|
|
664
|
+
}
|
|
665
|
+
async buildTokenRecoveryTransaction(buildParams) {
|
|
666
|
+
const { baseAddress, params } = buildParams;
|
|
667
|
+
const tokenContractAddress = params.tokenContractAddress;
|
|
668
|
+
(0, assert_1.default)(tokenContractAddress, 'tokenContractAddress is required for token recovery');
|
|
669
|
+
const balance = await this.getBalance(baseAddress, tokenContractAddress);
|
|
670
|
+
//replace with get balance function
|
|
671
|
+
if (balance.isLessThanOrEqualTo(0)) {
|
|
672
|
+
throw new Error(`no token balance to recover for address ${baseAddress} contract address ${tokenContractAddress}`);
|
|
673
|
+
}
|
|
674
|
+
// create the recipients here so that we can build the clauses for gas estimation
|
|
675
|
+
const roughFeeEstimate = this.calculateFee(constants_1.feeEstimateData, new bignumber_js_1.default(51390));
|
|
676
|
+
let recipients = [
|
|
677
|
+
{
|
|
678
|
+
address: params.recoveryDestination,
|
|
679
|
+
amount: balance.minus(roughFeeEstimate).toString(),
|
|
680
|
+
},
|
|
681
|
+
];
|
|
682
|
+
const blockRef = await this.getBlockRef();
|
|
683
|
+
const tokenTransaction = new lib_1.TokenTransaction(statics_1.coins.get(this.getChain()));
|
|
684
|
+
const txBuilder = this.getTxBuilderFactory().getTokenTransactionBuilder(tokenTransaction);
|
|
685
|
+
txBuilder.tokenAddress(tokenContractAddress);
|
|
686
|
+
txBuilder.chainTag(this.bitgo.getEnv() === 'prod' ? 0x4a : 0x27);
|
|
687
|
+
txBuilder.recipients(recipients);
|
|
688
|
+
txBuilder.sender(baseAddress);
|
|
689
|
+
txBuilder.addFeePayerAddress(baseAddress);
|
|
690
|
+
txBuilder.gas(Number(constants_1.AVG_GAS_UNITS));
|
|
691
|
+
txBuilder.blockRef(blockRef);
|
|
692
|
+
txBuilder.expiration(constants_1.EXPIRATION);
|
|
693
|
+
txBuilder.gasPriceCoef(Number(constants_1.GAS_PRICE_COEF));
|
|
694
|
+
txBuilder.nonce(this.getRandomNonce());
|
|
695
|
+
txBuilder.isRecovery(true);
|
|
696
|
+
let tx = (await txBuilder.build());
|
|
697
|
+
const clauses = tx.clauses;
|
|
698
|
+
const actualGasUnits = await this.estimateGas(clauses, baseAddress);
|
|
699
|
+
await this.ensureVthoBalanceForFee(baseAddress, actualGasUnits);
|
|
700
|
+
const requiredFee = this.calculateFee(constants_1.feeEstimateData, actualGasUnits);
|
|
701
|
+
// create the final recipients with the fee deducted
|
|
702
|
+
recipients = [
|
|
703
|
+
{
|
|
704
|
+
address: params.recoveryDestination,
|
|
705
|
+
amount: balance.minus(requiredFee).toString(),
|
|
706
|
+
},
|
|
707
|
+
];
|
|
708
|
+
txBuilder.recipients(recipients);
|
|
709
|
+
txBuilder.gas(actualGasUnits.toNumber());
|
|
710
|
+
tx = (await txBuilder.build());
|
|
711
|
+
return tx;
|
|
712
|
+
}
|
|
218
713
|
}
|
|
219
714
|
exports.Vet = Vet;
|
|
220
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmV0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3ZldC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSwwQ0FBNEI7QUFDNUIsZ0VBQXFDO0FBQ3JDLG1EQWU4QjtBQUU5Qix3REFBZ0M7QUFDaEMscURBQThDO0FBQzlDLG1DQUFxQztBQUNyQywyREFBaUU7QUFDakUsK0JBQWtEO0FBSWxEOzs7O0dBSUc7QUFDSCxNQUFhLEdBQUksU0FBUSxtQkFBUTtJQUUvQixZQUFzQixLQUFnQixFQUFFLFdBQXVDO1FBQzdFLEtBQUssQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUViLElBQUksQ0FBQyxXQUFXLEVBQUUsQ0FBQztZQUNqQixNQUFNLElBQUksS0FBSyxDQUFDLG9EQUFvRCxDQUFDLENBQUM7UUFDeEUsQ0FBQztRQUVELElBQUksQ0FBQyxZQUFZLEdBQUcsV0FBVyxDQUFDO0lBQ2xDLENBQUM7SUFFRCxNQUFNLENBQUMsY0FBYyxDQUFDLEtBQWdCLEVBQUUsV0FBdUM7UUFDN0UsT0FBTyxJQUFJLEdBQUcsQ0FBQyxLQUFLLEVBQUUsV0FBVyxDQUFDLENBQUM7SUFDckMsQ0FBQztJQUVEOztPQUVHO0lBQ0ksYUFBYTtRQUNsQixPQUFPLElBQUksQ0FBQztJQUNkLENBQUM7SUFFTSxRQUFRO1FBQ2IsT0FBTyxLQUFLLENBQUM7SUFDZixDQUFDO0lBRU0sU0FBUztRQUNkLE9BQU8sS0FBSyxDQUFDO0lBQ2YsQ0FBQztJQUVNLFdBQVc7UUFDaEIsT0FBTyxTQUFTLENBQUM7SUFDbkIsQ0FBQztJQUVELGtCQUFrQjtJQUNsQixXQUFXO1FBQ1QsT0FBTyxJQUFJLENBQUM7SUFDZCxDQUFDO0lBRUQsb0JBQW9CO0lBQ3BCLHNCQUFzQjtRQUNwQixPQUFPLHdCQUFhLENBQUMsR0FBRyxDQUFDO0lBQzNCLENBQUM7SUFFRCxlQUFlO1FBQ2IsT0FBTyxPQUFPLENBQUM7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QUNwQixJQUFJLEtBQUssR0FBRyxJQUFJLENBQUM7UUFDakIsSUFBSSxDQUFDO1lBQ0gsSUFBSSxzQkFBVSxDQUFDLEVBQUUsR0FBRyxFQUFFLENBQUMsQ0FBQztRQUMxQixDQUFDO1FBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQztZQUNYLEtBQUssR0FBRyxLQUFLLENBQUM7UUFDaEIsQ0FBQztRQUNELE9BQU8sS0FBSyxDQUFDO0lBQ2YsQ0FBQztJQUVELGNBQWMsQ0FBQyxPQUFlO1FBQzVCLE9BQU8sZUFBSyxDQUFDLGNBQWMsQ0FBQyxPQUFPLENBQUMsQ0FBQztJQUN2QyxDQUFDO0lBRUQsZUFBZSxDQUFDLE1BQThCO1FBQzVDLE1BQU0sSUFBSSxLQUFLLENBQUMseUJBQXlCLENBQUMsQ0FBQztJQUM3QyxDQUFDO0lBRVMsbUJBQW1CO1FBQzNCLE9BQU8sSUFBSSwrQkFBeUIsQ0FBQyxJQUFJLENBQUMsWUFBWSxDQUFDLENBQUM7SUFDMUQsQ0FBQztJQUVTLEtBQUssQ0FBQyxrQkFBa0IsQ0FBQyxLQUFhO1FBQzlDLE1BQU0sZ0JBQWdCLEdBQUcsSUFBSSxDQUFDLG1CQUFtQixFQUFFLENBQUM7UUFDcEQsSUFBSSxDQUFDO1lBQ0gsTUFBTSxTQUFTLEdBQUcsZ0JBQWdCLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDO1lBQy9DLE9BQU8sU0FBUyxDQUFDLFdBQVcsQ0FBQztRQUMvQixDQUFDO1FBQUMsTUFBTSxDQUFDO1lBQ1AsTUFBTSxJQUFJLEtBQUssQ0FBQywrQkFBK0IsQ0FBQyxDQUFDO1FBQ25ELENBQUM7SUFDSCxDQUFDO0lBRUQsa0JBQWtCO0lBQ2xCLGlCQUFpQixDQUFDLE1BQStCO1FBQy9DLHNEQUFzRDtRQUN0RCxNQUFNLElBQUksS0FBSyxDQUFDLHlCQUF5QixDQUFDLENBQUM7SUFDN0MsQ0FBQztDQUNGO0FBMUxELGtCQTBMQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIF8gZnJvbSAnbG9kYXNoJztcbmltcG9ydCBCaWdOdW1iZXIgZnJvbSAnYmlnbnVtYmVyLmpzJztcbmltcG9ydCB7XG4gIEF1ZGl0RGVjcnlwdGVkS2V5UGFyYW1zLFxuICBCYXNlQ29pbixcbiAgQmFzZVRyYW5zYWN0aW9uLFxuICBCaXRHb0Jhc2UsXG4gIEludmFsaWRBZGRyZXNzRXJyb3IsXG4gIEtleVBhaXIsXG4gIE1QQ0FsZ29yaXRobSxcbiAgTXVsdGlzaWdUeXBlLFxuICBtdWx0aXNpZ1R5cGVzLFxuICBQYXJzZWRUcmFuc2FjdGlvbixcbiAgU2lnbmVkVHJhbnNhY3Rpb24sXG4gIFNpZ25UcmFuc2FjdGlvbk9wdGlvbnMsXG4gIFZlcmlmeUFkZHJlc3NPcHRpb25zLFxuICBWZXJpZnlUcmFuc2FjdGlvbk9wdGlvbnMsXG59IGZyb20gJ0BiaXRnby1iZXRhL3Nkay1jb3JlJztcbmltcG9ydCB7IEJhc2VDb2luIGFzIFN0YXRpY3NCYXNlQ29pbiB9IGZyb20gJ0BiaXRnby1iZXRhL3N0YXRpY3MnO1xuaW1wb3J0IHV0aWxzIGZyb20gJy4vbGliL3V0aWxzJztcbmltcG9ydCB7IGJpcDMyIH0gZnJvbSAnQGJpdGdvLWJldGEvc2VjcDI1NmsxJztcbmltcG9ydCB7IHJhbmRvbUJ5dGVzIH0gZnJvbSAnY3J5cHRvJztcbmltcG9ydCB7IEtleVBhaXIgYXMgRXRoS2V5UGFpciB9IGZyb20gJ0BiaXRnby1iZXRhL2Fic3RyYWN0LWV0aCc7XG5pbXBvcnQgeyBUcmFuc2FjdGlvbkJ1aWxkZXJGYWN0b3J5IH0gZnJvbSAnLi9saWInO1xuaW1wb3J0IHsgRXhwbGFpblRyYW5zYWN0aW9uT3B0aW9ucywgVmV0UGFyc2VUcmFuc2FjdGlvbk9wdGlvbnMgfSBmcm9tICcuL2xpYi90eXBlcyc7XG5pbXBvcnQgeyBWZXRUcmFuc2FjdGlvbkV4cGxhbmF0aW9uIH0gZnJvbSAnLi9saWIvaWZhY2UnO1xuXG4vKipcbiAqIEZ1bGwgTmFtZTogVmVjaGFpblxuICogRG9jczogaHR0cHM6Ly9kb2NzLnZlY2hhaW4ub3JnL1xuICogR2l0SHViIDogaHR0cHM6Ly9naXRodWIuY29tL3ZlY2hhaW5cbiAqL1xuZXhwb3J0IGNsYXNzIFZldCBleHRlbmRzIEJhc2VDb2luIHtcbiAgcHJvdGVjdGVkIHJlYWRvbmx5IF9zdGF0aWNzQ29pbjogUmVhZG9ubHk8U3RhdGljc0Jhc2VDb2luPjtcbiAgcHJvdGVjdGVkIGNvbnN0cnVjdG9yKGJpdGdvOiBCaXRHb0Jhc2UsIHN0YXRpY3NDb2luPzogUmVhZG9ubHk8U3RhdGljc0Jhc2VDb2luPikge1xuICAgIHN1cGVyKGJpdGdvKTtcblxuICAgIGlmICghc3RhdGljc0NvaW4pIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcignbWlzc2luZyByZXF1aXJlZCBjb25zdHJ1Y3RvciBwYXJhbWV0ZXIgc3RhdGljc0NvaW4nKTtcbiAgICB9XG5cbiAgICB0aGlzLl9zdGF0aWNzQ29pbiA9IHN0YXRpY3NDb2luO1xuICB9XG5cbiAgc3RhdGljIGNyZWF0ZUluc3RhbmNlKGJpdGdvOiBCaXRHb0Jhc2UsIHN0YXRpY3NDb2luPzogUmVhZG9ubHk8U3RhdGljc0Jhc2VDb2luPik6IEJhc2VDb2luIHtcbiAgICByZXR1cm4gbmV3IFZldChiaXRnbywgc3RhdGljc0NvaW4pO1xuICB9XG5cbiAgLyoqXG4gICAqIEZhY3RvciBiZXR3ZWVuIHRoZSBjb2luJ3MgYmFzZSB1bml0IGFuZCBpdHMgc21hbGxlc3Qgc3ViIGRpdmlzaW9uXG4gICAqL1xuICBwdWJsaWMgZ2V0QmFzZUZhY3RvcigpOiBudW1iZXIge1xuICAgIHJldHVybiAxZTE4O1xuICB9XG5cbiAgcHVibGljIGdldENoYWluKCk6IHN0cmluZyB7XG4gICAgcmV0dXJuICd2ZXQnO1xuICB9XG5cbiAgcHVibGljIGdldEZhbWlseSgpOiBzdHJpbmcge1xuICAgIHJldHVybiAndmV0JztcbiAgfVxuXG4gIHB1YmxpYyBnZXRGdWxsTmFtZSgpOiBzdHJpbmcge1xuICAgIHJldHVybiAnVmVDaGFpbic7XG4gIH1cblxuICAvKiogQGluaGVyaXREb2MgKi9cbiAgc3VwcG9ydHNUc3MoKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIHRydWU7XG4gIH1cblxuICAvKiogaW5oZXJpdGVkIGRvYyAqL1xuICBnZXREZWZhdWx0TXVsdGlzaWdUeXBlKCk6IE11bHRpc2lnVHlwZSB7XG4gICAgcmV0dXJuIG11bHRpc2lnVHlwZXMudHNzO1xuICB9XG5cbiAgZ2V0TVBDQWxnb3JpdGhtKCk6IE1QQ0FsZ29yaXRobSB7XG4gICAgcmV0dXJuICdlY2RzYSc7XG4gIH1cblxuICBhbGxvd3NBY2NvdW50Q29uc29saWRhdGlvbnMoKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIHRydWU7XG4gIH1cblxuICBhc3luYyB2ZXJpZnlUcmFuc2FjdGlvbihwYXJhbXM6IFZlcmlmeVRyYW5zYWN0aW9uT3B0aW9ucyk6IFByb21pc2U8Ym9vbGVhbj4ge1xuICAgIGNvbnN0IHsgdHhQcmVidWlsZDogdHhQcmVidWlsZCwgdHhQYXJhbXM6IHR4UGFyYW1zIH0gPSBwYXJhbXM7XG4gICAgY29uc3QgdHhIZXggPSB0eFByZWJ1aWxkLnR4SGV4O1xuICAgIGlmICghdHhIZXgpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcignbWlzc2luZyByZXF1aXJlZCB0eCBwcmVidWlsZCBwcm9wZXJ0eSB0eEhleCcpO1xuICAgIH1cbiAgICBjb25zdCBleHBsYWluZWRUeCA9IGF3YWl0IHRoaXMuZXhwbGFpblRyYW5zYWN0aW9uKHsgdHhIZXggfSk7XG4gICAgaWYgKCFleHBsYWluZWRUeCkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKCdmYWlsZWQgdG8gZXhwbGFpbiB0cmFuc2FjdGlvbicpO1xuICAgIH1cbiAgICBpZiAodHhQYXJhbXMucmVjaXBpZW50cyAhPT0gdW5kZWZpbmVkKSB7XG4gICAgICBjb25zdCBmaWx0ZXJlZFJlY2lwaWVudHMgPSB0eFBhcmFtcy5yZWNpcGllbnRzPy5tYXAoKHJlY2lwaWVudCkgPT4ge1xuICAgICAgICByZXR1cm4ge1xuICAgICAgICAgIGFkZHJlc3M6IHJlY2lwaWVudC5hZGRyZXNzLFxuICAgICAgICAgIGFtb3VudDogQmlnSW50KHJlY2lwaWVudC5hbW91bnQpLFxuICAgICAgICB9O1xuICAgICAgfSk7XG4gICAgICBjb25zdCBmaWx0ZXJlZE91dHB1dHMgPSBleHBsYWluZWRUeC5vdXRwdXRzLm1hcCgob3V0cHV0KSA9PiB7XG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgYWRkcmVzczogb3V0cHV0LmFkZHJlc3MsXG4gICAgICAgICAgYW1vdW50OiBCaWdJbnQob3V0cHV0LmFtb3VudCksXG4gICAgICAgIH07XG4gICAgICB9KTtcbiAgICAgIGlmICghXy5pc0VxdWFsKGZpbHRlcmVkT3V0cHV0cywgZmlsdGVyZWRSZWNpcGllbnRzKSkge1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoJ1R4IG91dHB1dHMgZG9lcyBub3QgbWF0Y2ggd2l0aCBleHBlY3RlZCB0eFBhcmFtcyByZWNpcGllbnRzJyk7XG4gICAgICB9XG4gICAgICBsZXQgdG90YWxBbW91bnQgPSBuZXcgQmlnTnVtYmVyKDApO1xuICAgICAgZm9yIChjb25zdCByZWNpcGllbnRzIG9mIHR4UGFyYW1zLnJlY2lwaWVudHMpIHtcbiAgICAgICAgdG90YWxBbW91bnQgPSB0b3RhbEFtb3VudC5wbHVzKHJlY2lwaWVudHMuYW1vdW50KTtcbiAgICAgIH1cbiAgICAgIGlmICghdG90YWxBbW91bnQuaXNFcXVhbFRvKGV4cGxhaW5lZFR4Lm91dHB1dEFtb3VudCkpIHtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKCdUeCB0b3RhbCBhbW91bnQgZG9lcyBub3QgbWF0Y2ggd2l0aCBleHBlY3RlZCB0b3RhbCBhbW91bnQgZmllbGQnKTtcbiAgICAgIH1cbiAgICB9XG4gICAgcmV0dXJuIHRydWU7XG4gIH1cblxuICBhc3luYyBpc1dhbGxldEFkZHJlc3MocGFyYW1zOiBWZXJpZnlBZGRyZXNzT3B0aW9ucyk6IFByb21pc2U8Ym9vbGVhbj4ge1xuICAgIGNvbnN0IHsgYWRkcmVzczogbmV3QWRkcmVzcyB9ID0gcGFyYW1zO1xuXG4gICAgaWYgKCF0aGlzLmlzVmFsaWRBZGRyZXNzKG5ld0FkZHJlc3MpKSB7XG4gICAgICB0aHJvdyBuZXcgSW52YWxpZEFkZHJlc3NFcnJvcihgaW52YWxpZCBhZGRyZXNzOiAke25ld0FkZHJlc3N9YCk7XG4gICAgfVxuICAgIHJldHVybiB0cnVlO1xuICB9XG5cbiAgYXN5bmMgcGFyc2VUcmFuc2FjdGlvbihwYXJhbXM6IFZldFBhcnNlVHJhbnNhY3Rpb25PcHRpb25zKTogUHJvbWlzZTxQYXJzZWRUcmFuc2FjdGlvbj4ge1xuICAgIGNvbnN0IHRyYW5zYWN0aW9uRXhwbGFuYXRpb24gPSBhd2FpdCB0aGlzLmV4cGxhaW5UcmFuc2FjdGlvbih7IHR4SGV4OiBwYXJhbXMudHhIZXggfSk7XG4gICAgaWYgKCF0cmFuc2FjdGlvbkV4cGxhbmF0aW9uKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoJ0ludmFsaWQgdHJhbnNhY3Rpb24nKTtcbiAgICB9XG4gICAgcmV0dXJuIHtcbiAgICAgIGlucHV0czogW1xuICAgICAgICB7XG4gICAgICAgICAgYWRkcmVzczogdHJhbnNhY3Rpb25FeHBsYW5hdGlvbi5zZW5kZXIsXG4gICAgICAgICAgYW1vdW50OiB0cmFuc2FjdGlvbkV4cGxhbmF0aW9uLm91dHB1dEFtb3VudCxcbiAgICAgICAgfSxcbiAgICAgIF0sXG4gICAgICBvdXRwdXRzOiBbXG4gICAgICAgIHtcbiAgICAgICAgICBhZGRyZXNzOiB0cmFuc2FjdGlvbkV4cGxhbmF0aW9uLm91dHB1dHNbMF0uYWRkcmVzcyxcbiAgICAgICAgICBhbW91bnQ6IHRyYW5zYWN0aW9uRXhwbGFuYXRpb24ub3V0cHV0c1swXS5hbW91bnQsXG4gICAgICAgIH0sXG4gICAgICBdLFxuICAgIH07XG4gIH1cblxuICAvKipcbiAgICogRXhwbGFpbiBhIFZlY2hhaW4gdHJhbnNhY3Rpb25cbiAgICogQHBhcmFtIHBhcmFtc1xuICAgKi9cbiAgYXN5bmMgZXhwbGFpblRyYW5zYWN0aW9uKHBhcmFtczogRXhwbGFpblRyYW5zYWN0aW9uT3B0aW9ucyk6IFByb21pc2U8VmV0VHJhbnNhY3Rpb25FeHBsYW5hdGlvbiB8IHVuZGVmaW5lZD4ge1xuICAgIGxldCByZWJ1aWx0VHJhbnNhY3Rpb246IEJhc2VUcmFuc2FjdGlvbjtcbiAgICB0cnkge1xuICAgICAgcmVidWlsdFRyYW5zYWN0aW9uID0gYXdhaXQgdGhpcy5yZWJ1aWxkVHJhbnNhY3Rpb24ocGFyYW1zLnR4SGV4KTtcbiAgICB9IGNhdGNoIHtcbiAgICAgIHJldHVybiB1bmRlZmluZWQ7XG4gICAgfVxuICAgIHJldHVybiByZWJ1aWx0VHJhbnNhY3Rpb24uZXhwbGFpblRyYW5zYWN0aW9uKCk7XG4gIH1cblxuICBnZW5lcmF0ZUtleVBhaXIoc2VlZD86IEJ1ZmZlcik6IEtleVBhaXIge1xuICAgIGlmICghc2VlZCkge1xuICAgICAgLy8gQW4gZXh0ZW5kZWQgcHJpdmF0ZSBrZXkgaGFzIGJvdGggYSBub3JtYWwgMjU2IGJpdCBwcml2YXRlIGtleSBhbmQgYSAyNTZcbiAgICAgIC8vIGJpdCBjaGFpbiBjb2RlLCBib3RoIG9mIHdoaWNoIG11c3QgYmUgcmFuZG9tLiA1MTIgYml0cyBpcyB0aGVyZWZvcmUgdGhlXG4gICAgICAvLyBtYXhpbXVtIGVudHJvcHkgYW5kIGdpdmVzIHVzIG1heGltdW0gc2VjdXJpdHkgYWdhaW5zdCBjcmFja2luZy5cbiAgICAgIHNlZWQgPSByYW5kb21CeXRlcyg1MTIgLyA4KTtcbiAgICB9XG4gICAgY29uc3QgZXh0ZW5kZWRLZXkgPSBiaXAzMi5mcm9tU2VlZChzZWVkKTtcbiAgICBjb25zdCB4cHViID0gZXh0ZW5kZWRLZXkubmV1dGVyZWQoKS50b0Jhc2U1OCgpO1xuICAgIHJldHVybiB7XG4gICAgICBwdWI6IHhwdWIsXG4gICAgICBwcnY6IGV4dGVuZGVkS2V5LnRvQmFzZTU4KCksXG4gICAgfTtcbiAgfVxuXG4gIGlzVmFsaWRQdWIocHViOiBzdHJpbmcpOiBib29sZWFuIHtcbiAgICBsZXQgdmFsaWQgPSB0cnVlO1xuICAgIHRyeSB7XG4gICAgICBuZXcgRXRoS2V5UGFpcih7IHB1YiB9KTtcbiAgICB9IGNhdGNoIChlKSB7XG4gICAgICB2YWxpZCA9IGZhbHNlO1xuICAgIH1cbiAgICByZXR1cm4gdmFsaWQ7XG4gIH1cblxuICBpc1ZhbGlkQWRkcmVzcyhhZGRyZXNzOiBzdHJpbmcpOiBib29sZWFuIHtcbiAgICByZXR1cm4gdXRpbHMuaXNWYWxpZEFkZHJlc3MoYWRkcmVzcyk7XG4gIH1cblxuICBzaWduVHJhbnNhY3Rpb24ocGFyYW1zOiBTaWduVHJhbnNhY3Rpb25PcHRpb25zKTogUHJvbWlzZTxTaWduZWRUcmFuc2FjdGlvbj4ge1xuICAgIHRocm93IG5ldyBFcnJvcignTWV0aG9kIG5vdCBpbXBsZW1lbnRlZC4nKTtcbiAgfVxuXG4gIHByb3RlY3RlZCBnZXRUeEJ1aWxkZXJGYWN0b3J5KCk6IFRyYW5zYWN0aW9uQnVpbGRlckZhY3Rvcnkge1xuICAgIHJldHVybiBuZXcgVHJhbnNhY3Rpb25CdWlsZGVyRmFjdG9yeSh0aGlzLl9zdGF0aWNzQ29pbik7XG4gIH1cblxuICBwcm90ZWN0ZWQgYXN5bmMgcmVidWlsZFRyYW5zYWN0aW9uKHR4SGV4OiBzdHJpbmcpOiBQcm9taXNlPEJhc2VUcmFuc2FjdGlvbj4ge1xuICAgIGNvbnN0IHR4QnVpbGRlckZhY3RvcnkgPSB0aGlzLmdldFR4QnVpbGRlckZhY3RvcnkoKTtcbiAgICB0cnkge1xuICAgICAgY29uc3QgdHhCdWlsZGVyID0gdHhCdWlsZGVyRmFjdG9yeS5mcm9tKHR4SGV4KTtcbiAgICAgIHJldHVybiB0eEJ1aWxkZXIudHJhbnNhY3Rpb247XG4gICAgfSBjYXRjaCB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoJ0ZhaWxlZCB0byByZWJ1aWxkIHRyYW5zYWN0aW9uJyk7XG4gICAgfVxuICB9XG5cbiAgLyoqIEBpbmhlcml0RG9jICovXG4gIGF1ZGl0RGVjcnlwdGVkS2V5KHBhcmFtczogQXVkaXREZWNyeXB0ZWRLZXlQYXJhbXMpOiB2b2lkIHtcbiAgICAvKiogaHR0cHM6Ly9iaXRnb2luYy5hdGxhc3NpYW4ubmV0L2Jyb3dzZS9DT0lOLTQyMTMgKi9cbiAgICB0aHJvdyBuZXcgRXJyb3IoJ01ldGhvZCBub3QgaW1wbGVtZW50ZWQuJyk7XG4gIH1cbn1cbiJdfQ==
|
|
715
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmV0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3ZldC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSwwQ0FBNEI7QUFDNUIsZ0VBQXFDO0FBQ3JDLGtFQUEwQztBQUMxQyxvREFBNEI7QUFDNUIsa0RBQTBCO0FBQzFCLGdEQUFxRjtBQUNyRixtREF5QjhCO0FBQzlCLDZEQUErQztBQUMvQyxpREFBeUU7QUFDekUsd0RBQWdDO0FBQ2hDLHFEQUE4QztBQUM5QyxtQ0FBMkM7QUFDM0MsMkRBQWlFO0FBQ2pFLCtCQUFpRjtBQVNqRiwrQ0FBNkY7QUFlN0Y7Ozs7R0FJRztBQUNILE1BQWEsR0FBSSxTQUFRLG1CQUFRO0lBRS9CLFlBQXNCLEtBQWdCLEVBQUUsV0FBdUM7UUFDN0UsS0FBSyxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBRWIsSUFBSSxDQUFDLFdBQVcsRUFBRSxDQUFDO1lBQ2pCLE1BQU0sSUFBSSxLQUFLLENBQUMsb0RBQW9ELENBQUMsQ0FBQztRQUN4RSxDQUFDO1FBRUQsSUFBSSxDQUFDLFlBQVksR0FBRyxXQUFXLENBQUM7SUFDbEMsQ0FBQztJQUVELE1BQU0sQ0FBQyxjQUFjLENBQUMsS0FBZ0IsRUFBRSxXQUF1QztRQUM3RSxPQUFPLElBQUksR0FBRyxDQUFDLEtBQUssRUFBRSxXQUFXLENBQUMsQ0FBQztJQUNyQyxDQUFDO0lBRUQ7O09BRUc7SUFDSSxhQUFhO1FBQ2xCLE9BQU8sSUFBSSxDQUFDO0lBQ2QsQ0FBQztJQUVNLFFBQVE7UUFDYixPQUFPLEtBQUssQ0FBQztJQUNmLENBQUM7SUFFTSxTQUFTO1FBQ2QsT0FBTyxLQUFLLENBQUM7SUFDZixDQUFDO0lBRU0sV0FBVztRQUNoQixPQUFPLFNBQVMsQ0FBQztJQUNuQixDQUFDO0lBRUQsd0JBQXdCO1FBQ3RCLE9BQU8sSUFBSSxDQUFDO0lBQ2QsQ0FBQztJQUVELGtCQUFrQjtJQUNs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XG5pbXBvcnQgQmlnTnVtYmVyIGZyb20gJ2JpZ251bWJlci5qcyc7XG5pbXBvcnQgYmxha2UyYiBmcm9tICdAYml0Z28tYmV0YS9ibGFrZTJiJztcbmltcG9ydCBhc3NlcnQgZnJvbSAnYXNzZXJ0JztcbmltcG9ydCBheGlvcyBmcm9tICdheGlvcyc7XG5pbXBvcnQgeyBUcmFuc2FjdGlvbkNsYXVzZSwgVHJhbnNhY3Rpb24gYXMgVmV0VHJhbnNhY3Rpb24gfSBmcm9tICdAdmVjaGFpbi9zZGstY29yZSc7XG5pbXBvcnQge1xuICBBdWRpdERlY3J5cHRlZEtleVBhcmFtcyxcbiAgQmFzZUNvaW4sXG4gIEJhc2VUcmFuc2FjdGlvbixcbiAgQml0R29CYXNlLFxuICBCdWlsZE5mdFRyYW5zZmVyRGF0YU9wdGlvbnMsXG4gIEludmFsaWRBZGRyZXNzRXJyb3IsXG4gIEtleVBhaXIsXG4gIE1QQ0FsZ29yaXRobSxcbiAgTXVsdGlzaWdUeXBlLFxuICBtdWx0aXNpZ1R5cGVzLFxuICBOb3RJbXBsZW1lbnRlZEVycm9yLFxuICBQYXJzZWRUcmFuc2FjdGlvbixcbiAgU2lnbmVkVHJhbnNhY3Rpb24sXG4gIFNpZ25UcmFuc2FjdGlvbk9wdGlvbnMsXG4gIFRva2VuVHJhbnNmZXJSZWNpcGllbnRQYXJhbXMsXG4gIFRzc1ZlcmlmeUFkZHJlc3NPcHRpb25zLFxuICBWZXJpZnlUcmFuc2FjdGlvbk9wdGlvbnMsXG4gIFRva2VuVHlwZSxcbiAgRWNkc2EsXG4gIEVDRFNBVXRpbHMsXG4gIEVudmlyb25tZW50cyxcbiAgQmFzZUJyb2FkY2FzdFRyYW5zYWN0aW9uT3B0aW9ucyxcbiAgQmFzZUJyb2FkY2FzdFRyYW5zYWN0aW9uUmVzdWx0LFxuICB2ZXJpZnlNUENXYWxsZXRBZGRyZXNzLFxufSBmcm9tICdAYml0Z28tYmV0YS9zZGstY29yZSc7XG5pbXBvcnQgKiBhcyBtcGMgZnJvbSAnQGJpdGdvLWJldGEvc2RrLWxpYi1tcGMnO1xuaW1wb3J0IHsgQmFzZUNvaW4gYXMgU3RhdGljc0Jhc2VDb2luLCBjb2lucyB9IGZyb20gJ0BiaXRnby1iZXRhL3N0YXRpY3MnO1xuaW1wb3J0IHV0aWxzIGZyb20gJy4vbGliL3V0aWxzJztcbmltcG9ydCB7IGJpcDMyIH0gZnJvbSAnQGJpdGdvLWJldGEvc2VjcDI1NmsxJztcbmltcG9ydCB7IHJhbmRvbUJ5dGVzLCBIYXNoIH0gZnJvbSAnY3J5cHRvJztcbmltcG9ydCB7IEtleVBhaXIgYXMgRXRoS2V5UGFpciB9IGZyb20gJ0BiaXRnby1iZXRhL2Fic3RyYWN0LWV0aCc7XG5pbXBvcnQgeyBUb2tlblRyYW5zYWN0aW9uLCBUcmFuc2FjdGlvbiwgVHJhbnNhY3Rpb25CdWlsZGVyRmFjdG9yeSB9IGZyb20gJy4vbGliJztcbmltcG9ydCB7XG4gIEV4cGxhaW5UcmFuc2FjdGlvbk9wdGlvbnMsXG4gIFJlY292ZXJPcHRpb25zLFxuICBSZWNvdmVyeVRyYW5zYWN0aW9uLFxuICBVbnNpZ25lZFN3ZWVwUmVjb3ZlcnlUcmFuc2FjdGlvbixcbiAgVmV0UGFyc2VUcmFuc2FjdGlvbk9wdGlvbnMsXG59IGZyb20gJy4vbGliL3R5cGVzJztcbmltcG9ydCB7IFZldFRyYW5zYWN0aW9uRXhwbGFuYXRpb24gfSBmcm9tICcuL2xpYi9pZmFjZSc7XG5pbXBvcnQgeyBBVkdfR0FTX1VOSVRTLCBFWFBJUkFUSU9OLCBHQVNfUFJJQ0VfQ09FRiwgZmVlRXN0aW1hdGVEYXRhIH0gZnJvbSAnLi9saWIvY29uc3RhbnRzJztcblxuaW50ZXJmYWNlIEZlZUVzdGltYXRlRGF0YSB7XG4gIGdhczogc3RyaW5nO1xuICBnYXNVbml0UHJpY2U6IHN0cmluZztcbiAgZ2FzUHJpY2VDb2VmOiBzdHJpbmc7XG4gIGNvZWZEaXZpc29yOiBzdHJpbmc7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgVHNzVmVyaWZ5VmV0QWRkcmVzc09wdGlvbnMgZXh0ZW5kcyBUc3NWZXJpZnlBZGRyZXNzT3B0aW9ucyB7XG4gIGFkZHJlc3M6IHN0cmluZztcbiAgYmFzZUFkZHJlc3M/OiBzdHJpbmc7XG4gIHdhbGxldFZlcnNpb24/OiBudW1iZXI7XG59XG5cbi8qKlxuICogRnVsbCBOYW1lOiBWZWNoYWluXG4gKiBEb2NzOiBodHRwczovL2RvY3MudmVjaGFpbi5vcmcvXG4gKiBHaXRIdWIgOiBodHRwczovL2dpdGh1Yi5jb20vdmVjaGFpblxuICovXG5leHBvcnQgY2xhc3MgVmV0IGV4dGVuZHMgQmFzZUNvaW4ge1xuICBwcm90ZWN0ZWQgcmVhZG9ubHkgX3N0YXRpY3NDb2luOiBSZWFkb25seTxTdGF0aWNzQmFzZUNvaW4+O1xuICBwcm90ZWN0ZWQgY29uc3RydWN0b3IoYml0Z286IEJpdEdvQmFzZSwgc3RhdGljc0NvaW4/OiBSZWFkb25seTxTdGF0aWNzQmFzZUNvaW4+KSB7XG4gICAgc3VwZXIoYml0Z28pO1xuXG4gICAgaWYgKCFzdGF0aWNzQ29pbikge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKCdtaXNzaW5nIHJlcXVpcmVkIGNvbnN0cnVjdG9yIHBhcmFtZXRlciBzdGF0aWNzQ29pbicpO1xuICAgIH1cblxuICAgIHRoaXMuX3N0YXRpY3NDb2luID0gc3RhdGljc0NvaW47XG4gIH1cblxuICBzdGF0aWMgY3JlYXRlSW5zdGFuY2UoYml0Z286IEJpdEdvQmFzZSwgc3RhdGljc0NvaW4/OiBSZWFkb25seTxTdGF0aWNzQmFzZUNvaW4+KTogQmFzZUNvaW4ge1xuICAgIHJldHVybiBuZXcgVmV0KGJpdGdvLCBzdGF0aWNzQ29pbik7XG4gIH1cblxuICAvKipcbiAgICogRmFjdG9yIGJldHdlZW4gdGhlIGNvaW4ncyBiYXNlIHVuaXQgYW5kIGl0cyBzbWFsbGVzdCBzdWIgZGl2aXNpb25cbiAgICovXG4gIHB1YmxpYyBnZXRCYXNlRmFjdG9yKCk6IG51bWJlciB7XG4gICAgcmV0dXJuIDFlMTg7XG4gIH1cblxuICBwdWJsaWMgZ2V0Q2hhaW4oKTogc3RyaW5nIHtcbiAgICByZXR1cm4gJ3ZldCc7XG4gIH1cblxuICBwdWJsaWMgZ2V0RmFtaWx5KCk6IHN0cmluZyB7XG4gICAgcmV0dXJuICd2ZXQnO1xuICB9XG5cbiAgcHVibGljIGdldEZ1bGxOYW1lKCk6IHN0cmluZyB7XG4gICAgcmV0dXJuICdWZUNoYWluJztcbiAgfVxuXG4gIHZhbHVlbGVzc1RyYW5zZmVyQWxsb3dlZCgpOiBib29sZWFuIHtcbiAgICByZXR1cm4gdHJ1ZTtcbiAgfVxuXG4gIC8qKiBAaW5oZXJpdERvYyAqL1xuICBzdXBwb3J0c1RzcygpOiBib29sZWFuIHtcbiAgICByZXR1cm4gdHJ1ZTtcbiAgfVxuXG4gIC8qKiBpbmhlcml0ZWQgZG9jICovXG4gIGdldERlZmF1bHRNdWx0aXNpZ1R5cGUoKTogTXVsdGlzaWdUeXBlIHtcbiAgICByZXR1cm4gbXVsdGlzaWdUeXBlcy50c3M7XG4gIH1cblxuICBnZXRNUENBbGdvcml0aG0oKTogTVBDQWxnb3JpdGhtIHtcbiAgICByZXR1cm4gJ2VjZHNhJztcbiAgfVxuXG4gIGFsbG93c0FjY291bnRDb25zb2xpZGF0aW9ucygpOiBib29sZWFuIHtcbiAgICByZXR1cm4gdHJ1ZTtcbiAgfVxuXG4gIGFzeW5jIHZlcmlmeVRyYW5zYWN0aW9uKHBhcmFtczogVmVyaWZ5VHJhbnNhY3Rpb25PcHRpb25zKTogUHJvbWlzZTxib29sZWFuPiB7XG4gICAgY29uc3QgeyB0eFByZWJ1aWxkOiB0eFByZWJ1aWxkLCB0eFBhcmFtczogdHhQYXJhbXMgfSA9IHBhcmFtcztcbiAgICBjb25zdCB0eEhleCA9IHR4UHJlYnVpbGQudHhIZXg7XG4gICAgaWYgKCF0eEhleCkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKCdtaXNzaW5nIHJlcXVpcmVkIHR4IHByZWJ1aWxkIHByb3BlcnR5IHR4SGV4Jyk7XG4gICAgfVxuICAgIGNvbnN0IGV4cGxhaW5lZFR4ID0gYXdhaXQgdGhpcy5leHBsYWluVHJhbnNhY3Rpb24oeyB0eEhleCB9KTtcbiAgICBpZiAoIWV4cGxhaW5lZFR4KSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoJ2ZhaWxlZCB0byBleHBsYWluIHRyYW5zYWN0aW9uJyk7XG4gICAgfVxuICAgIGlmICh0eFBhcmFtcy5yZWNpcGllbnRzICE9PSB1bmRlZmluZWQgJiYgdHhQYXJhbXMucmVjaXBpZW50cy5sZW5ndGggPiAwKSB7XG4gICAgICBjb25zdCBmaWx0ZXJlZFJlY2lwaWVudHMgPSB0eFBhcmFtcy5yZWNpcGllbnRzPy5tYXAoKHJlY2lwaWVudCkgPT4ge1xuICAgICAgICByZXR1cm4ge1xuICAgICAgICAgIGFkZHJlc3M6IHJlY2lwaWVudC5hZGRyZXNzLnRvTG93ZXJDYXNlKCksXG4gICAgICAgICAgYW1vdW50OiBCaWdJbnQocmVjaXBpZW50LmFtb3VudCksXG4gICAgICAgIH07XG4gICAgICB9KTtcbiAgICAgIGNvbnN0IGZpbHRlcmVkT3V0cHV0cyA9IGV4cGxhaW5lZFR4Lm91dHB1dHMubWFwKChvdXRwdXQpID0+IHtcbiAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICBhZGRyZXNzOiBvdXRwdXQuYWRkcmVzcy50b0xvd2VyQ2FzZSgpLFxuICAgICAgICAgIGFtb3VudDogQmlnSW50KG91dHB1dC5hbW91bnQpLFxuICAgICAgICB9O1xuICAgICAgfSk7XG5cbiAgICAgIGlmICghXy5pc0VxdWFsKGZpbHRlcmVkT3V0cHV0cywgZmlsdGVyZWRSZWNpcGllbnRzKSkge1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoJ1R4IG91dHB1dHMgZG9lcyBub3QgbWF0Y2ggd2l0aCBleHBlY3RlZCB0eFBhcmFtcyByZWNpcGllbnRzJyk7XG4gICAgICB9XG4gICAgICBsZXQgdG90YWxBbW91bnQgPSBuZXcgQmlnTnVtYmVyKDApO1xuICAgICAgZm9yIChjb25zdCByZWNpcGllbnRzIG9mIHR4UGFyYW1zLnJlY2lwaWVudHMpIHtcbiAgICAgICAgdG90YWxBbW91bnQgPSB0b3RhbEFtb3VudC5wbHVzKHJlY2lwaWVudHMuYW1vdW50KTtcbiAgICAgIH1cbiAgICAgIGlmICghdG90YWxBbW91bnQuaXNFcXVhbFRvKGV4cGxhaW5lZFR4Lm91dHB1dEFtb3VudCkpIHtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKCdUeCB0b3RhbCBhbW91bnQgZG9lcyBub3QgbWF0Y2ggd2l0aCBleHBlY3RlZCB0b3RhbCBhbW91bnQgZmllbGQnKTtcbiAgICAgIH1cbiAgICB9XG4gICAgcmV0dXJuIHRydWU7XG4gIH1cblxuICAvKipcbiAgICogVmVyaWZ5IHRoYXQgYW4gYWRkcmVzcyBiZWxvbmdzIHRvIHRoaXMgd2FsbGV0LlxuICAgKlxuICAgKiBAcGFyYW0ge1Rzc1ZlcmlmeVZldEFkZHJlc3NPcHRpb25zfSBwYXJhbXMgLSBWZXJpZmljYXRpb24gcGFyYW1ldGVyc1xuICAgKiBAcmV0dXJucyB7UHJvbWlzZTxib29sZWFuPn0gVHJ1ZSBpZiBhZGRyZXNzIGJlbG9uZ3MgdG8gd2FsbGV0XG4gICAqIEB0aHJvd3Mge0ludmFsaWRBZGRyZXNzRXJyb3J9IElmIGFkZHJlc3MgZm9ybWF0IGlzIGludmFsaWRcbiAgICogQHRocm93cyB7RXJyb3J9IElmIGludmFsaWQgd2FsbGV0IHZlcnNpb24gb3IgbWlzc2luZyBwYXJhbWV0ZXJzXG4gICAqL1xuICBhc3luYyBpc1dhbGxldEFkZHJlc3MocGFyYW1zOiBUc3NWZXJpZnlWZXRBZGRyZXNzT3B0aW9ucyk6IFByb21pc2U8Ym9vbGVhbj4ge1xuICAgIGNvbnN0IHsgYWRkcmVzcywgYmFzZUFkZHJlc3MsIHdhbGxldFZlcnNpb24gfSA9IHBhcmFtcztcblxuICAgIGlmIChhZGRyZXNzICYmICF0aGlzLmlzVmFsaWRBZGRyZXNzKGFkZHJlc3MpKSB7XG4gICAgICB0aHJvdyBuZXcgSW52YWxpZEFkZHJlc3NFcnJvcihgaW52YWxpZCBhZGRyZXNzOiAke2FkZHJlc3N9YCk7XG4gICAgfVxuXG4gICAgaWYgKHdhbGxldFZlcnNpb24gIT09IDYpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihgVkVUIG9ubHkgc3VwcG9ydHMgd2FsbGV0IHZlcnNpb24gNiwgYnV0IGdvdCB2ZXJzaW9uICR7d2FsbGV0VmVyc2lvbn1gKTtcbiAgICB9XG5cbiAgICBjb25zdCBpc1ZlcmlmeWluZ0Jhc2VBZGRyZXNzID0gYmFzZUFkZHJlc3MgJiYgYWRkcmVzcy50b0xvd2VyQ2FzZSgpID09PSBiYXNlQWRkcmVzcy50b0xvd2VyQ2FzZSgpO1xuICAgIGlmIChpc1ZlcmlmeWluZ0Jhc2VBZGRyZXNzKSB7XG4gICAgICBjb25zdCBpbmRleCA9IHR5cGVvZiBwYXJhbXMuaW5kZXggPT09ICdzdHJpbmcnID8gcGFyc2VJbnQocGFyYW1zLmluZGV4LCAxMCkgOiBwYXJhbXMuaW5kZXg7XG4gICAgICBpZiAoaW5kZXggIT09IDApIHtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBCYXNlIGFkZHJlc3MgdmVyaWZpY2F0aW9uIHJlcXVpcmVzIGluZGV4IDAsIGJ1dCBnb3QgaW5kZXggJHtwYXJhbXMuaW5kZXh9LmApO1xuICAgICAgfVxuICAgIH1cblxuICAgIGNvbnN0IHJlc3VsdCA9IGF3YWl0IHZlcmlmeU1QQ1dhbGxldEFkZHJlc3MoXG4gICAgICB7IC4uLnBhcmFtcywga2V5Q3VydmU6ICdzZWNwMjU2azEnIH0sXG4gICAgICB0aGlzLmlzVmFsaWRBZGRyZXNzLmJpbmQodGhpcyksXG4gICAgICAocHViS2V5KSA9PiB7XG4gICAgICAgIGNvbnN0IGtleVBhaXIgPSBuZXcgRXRoS2V5UGFpcih7IHB1YjogcHViS2V5IH0pO1xuICAgICAgICByZXR1cm4ga2V5UGFpci5nZXRBZGRyZXNzKCk7XG4gICAgICB9XG4gICAgKTtcblxuICAgIGlmICghcmVzdWx0KSB7XG4gICAgICB0aHJvdyBuZXcgSW52YWxpZEFkZHJlc3NFcnJvcihgaW52YWxpZCBhZGRyZXNzOiAke2FkZHJlc3N9YCk7XG4gICAgfVxuXG4gICAgcmV0dXJuIHRydWU7XG4gIH1cblxuICBhc3luYyBwYXJzZVRyYW5zYWN0aW9uKHBhcmFtczogVmV0UGFyc2VUcmFuc2FjdGlvbk9wdGlvbnMpOiBQcm9taXNlPFBhcnNlZFRyYW5zYWN0aW9uPiB7XG4gICAgY29uc3QgdHJhbnNhY3Rpb25FeHBsYW5hdGlvbiA9IGF3YWl0IHRoaXMuZXhwbGFpblRyYW5zYWN0aW9uKHsgdHhIZXg6IHBhcmFtcy50eEhleCB9KTtcbiAgICBpZiAoIXRyYW5zYWN0aW9uRXhwbGFuYXRpb24pIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcignSW52YWxpZCB0cmFuc2FjdGlvbicpO1xuICAgIH1cbiAgICByZXR1cm4ge1xuICAgICAgaW5wdXRzOiBbXG4gICAgICAgIHtcbiAgICAgICAgICBhZGRyZXNzOiB0cmFuc2FjdGlvbkV4cGxhbmF0aW9uLnNlbmRlcixcbiAgICAgICAgICBhbW91bnQ6IHRyYW5zYWN0aW9uRXhwbGFuYXRpb24ub3V0cHV0QW1vdW50LFxuICAgICAgICB9LFxuICAgICAgXSxcbiAgICAgIG91dHB1dHM6IFtcbiAgICAgICAge1xuICAgICAgICAgIGFkZHJlc3M6IHRyYW5zYWN0aW9uRXhwbGFuYXRpb24ub3V0cHV0c1swXS5hZGRyZXNzLFxuICAgICAgICAgIGFtb3VudDogdHJhbnNhY3Rpb25FeHBsYW5hdGlvbi5vdXRwdXRzWzBdLmFtb3VudCxcbiAgICAgICAgfSxcbiAgICAgIF0sXG4gICAgfTtcbiAgfVxuXG4gIC8qKlxuICAgKiBFeHBsYWluIGEgVmVjaGFpbiB0cmFuc2FjdGlvblxuICAgKiBAcGFyYW0gcGFyYW1zXG4gICAqL1xuICBhc3luYyBleHBsYWluVHJhbnNhY3Rpb24ocGFyYW1zOiBFeHBsYWluVHJhbnNhY3Rpb25PcHRpb25zKTogUHJvbWlzZTxWZXRUcmFuc2FjdGlvbkV4cGxhbmF0aW9uIHwgdW5kZWZpbmVkPiB7XG4gICAgbGV0IHJlYnVpbHRUcmFuc2FjdGlvbjogQmFzZVRyYW5zYWN0aW9uO1xuICAgIHRyeSB7XG4gICAgICByZWJ1aWx0VHJhbnNhY3Rpb24gPSBhd2FpdCB0aGlzLnJlYnVpbGRUcmFuc2FjdGlvbihwYXJhbXMudHhIZXgpO1xuICAgIH0gY2F0Y2gge1xuICAgICAgcmV0dXJuIHVuZGVmaW5lZDtcbiAgICB9XG4gICAgcmV0dXJuIHJlYnVpbHRUcmFuc2FjdGlvbi5leHBsYWluVHJhbnNhY3Rpb24oKTtcbiAgfVxuXG4gIGdlbmVyYXRlS2V5UGFpcihzZWVkPzogQnVmZmVyKTogS2V5UGFpciB7XG4gICAgaWYgKCFzZWVkKSB7XG4gICAgICAvLyBBbiBleHRlbmRlZCBwcml2YXRlIGtleSBoYXMgYm90aCBhIG5vcm1hbCAyNTYgYml0IHByaXZhdGUga2V5IGFuZCBhIDI1NlxuICAgICAgLy8gYml0IGNoYWluIGNvZGUsIGJvdGggb2Ygd2hpY2ggbXVzdCBiZSByYW5kb20uIDUxMiBiaXRzIGlzIHRoZXJlZm9yZSB0aGVcbiAgICAgIC8vIG1heGltdW0gZW50cm9weSBhbmQgZ2l2ZXMgdXMgbWF4aW11bSBzZWN1cml0eSBhZ2FpbnN0IGNyYWNraW5nLlxuICAgICAgc2VlZCA9IHJhbmRvbUJ5dGVzKDUxMiAvIDgpO1xuICAgIH1cbiAgICBjb25zdCBleHRlbmRlZEtleSA9IGJpcDMyLmZyb21TZWVkKHNlZWQpO1xuICAgIGNvbnN0IHhwdWIgPSBleHRlbmRlZEtleS5uZXV0ZXJlZCgpLnRvQmFzZTU4KCk7XG4gICAgcmV0dXJuIHtcbiAgICAgIHB1YjogeHB1YixcbiAgICAgIHBydjogZXh0ZW5kZWRLZXkudG9CYXNlNTgoKSxcbiAgICB9O1xuICB9XG5cbiAgaXNWYWxpZFB1YihwdWI6IHN0cmluZyk6IGJvb2xlYW4ge1xuICAgIGxldCB2YWxpZCA9IHRydWU7XG4gICAgdHJ5IHtcbiAgICAgIG5ldyBFdGhLZXlQYWlyKHsgcHViIH0pO1xuICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgIHZhbGlkID0gZmFsc2U7XG4gICAgfVxuICAgIHJldHVybiB2YWxpZDtcbiAgfVxuXG4gIGlzVmFsaWRBZGRyZXNzKGFkZHJlc3M6IHN0cmluZyk6IGJvb2xlYW4ge1xuICAgIHJldHVybiB1dGlscy5pc1ZhbGlkQWRkcmVzcyhhZGRyZXNzKTtcbiAgfVxuXG4gIHNpZ25UcmFuc2FjdGlvbihwYXJhbXM6IFNpZ25UcmFuc2FjdGlvbk9wdGlvbnMpOiBQcm9taXNlPFNpZ25lZFRyYW5zYWN0aW9uPiB7XG4gICAgdGhyb3cgbmV3IEVycm9yKCdNZXRob2Qgbm90IGltcGxlbWVudGVkLicpO1xuICB9XG5cbiAgcHJvdGVjdGVkIGdldFR4QnVpbGRlckZhY3RvcnkoKTogVHJhbnNhY3Rpb25CdWlsZGVyRmFjdG9yeSB7XG4gICAgcmV0dXJuIG5ldyBUcmFuc2FjdGlvbkJ1aWxkZXJGYWN0b3J5KHRoaXMuX3N0YXRpY3NDb2luKTtcbiAgfVxuXG4gIHByb3RlY3RlZCBhc3luYyByZWJ1aWxkVHJhbnNhY3Rpb24odHhIZXg6IHN0cmluZyk6IFByb21pc2U8QmFzZVRyYW5zYWN0aW9uPiB7XG4gICAgY29uc3QgdHhCdWlsZGVyRmFjdG9yeSA9IHRoaXMuZ2V0VHhCdWlsZGVyRmFjdG9yeSgpO1xuICAgIHRyeSB7XG4gICAgICBjb25zdCB0eEJ1aWxkZXIgPSB0eEJ1aWxkZXJGYWN0b3J5LmZyb20odHhIZXgpO1xuICAgICAgcmV0dXJuIHR4QnVpbGRlci50cmFuc2FjdGlvbjtcbiAgICB9IGNhdGNoIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcignRmFpbGVkIHRvIHJlYnVpbGQgdHJhbnNhY3Rpb24nKTtcbiAgICB9XG4gIH1cblxuICAvKiogQGluaGVyaXREb2MgKi9cbiAgYXVkaXREZWNyeXB0ZWRLZXkocGFyYW1zOiBBdWRpdERlY3J5cHRlZEtleVBhcmFtcyk6IHZvaWQge1xuICAgIC8qKiBodHRwczovL2JpdGdvaW5jLmF0bGFzc2lhbi5uZXQvYnJvd3NlL0NPSU4tNDIxMyAqL1xuICAgIHRocm93IG5ldyBFcnJvcignTWV0aG9kIG5vdCBpbXBsZW1lbnRlZC4nKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBGdW5jdGlvbiB0byBnZXQgY29pbiBzcGVjaWZpYyBoYXNoIGZ1bmN0aW9uIHVzZWQgdG8gZ2VuZXJhdGUgdHJhbnNhY3Rpb24gZGlnZXN0cy5cbiAgICogQHJldHVybnMge0BzZWUgSGFzaH0gaGFzaCBmdW5jdGlvbiBpZiBpbXBsZW1lbnRlZCwgb3RoZXJ3aXNlIHRocm93cyBleGNlcHRpb25cbiAgICovXG4gIGdldEhhc2hGdW5jdGlvbigpOiBIYXNoIHtcbiAgICBjb25zdCBibGFrZSA9IGJsYWtlMmIoMzIpO1xuXG4gICAgLy8gV2UgcmV0dXJuIGFuIG9iamVjdCB0aGF0IG1pbWljcyB0aGUgSGFzaCBpbnRlcmZhY2VcbiAgICByZXR1cm4ge1xuICAgICAgdXBkYXRlKGRhdGE6IEJ1ZmZlciB8IFVpbnQ4QXJyYXkpIHtcbiAgICAgICAgYmxha2UudXBkYXRlKGRhdGEpO1xuICAgICAgICByZXR1cm4gdGhpcztcbiAgICAgIH0sXG4gICAgICBkaWdlc3QoKSB7XG4gICAgICAgIGNvbnN0IHVpbnQ4UmVzdWx0ID0gYmxha2UuZGlnZXN0KCk7XG4gICAgICAgIHJldHVybiBCdWZmZXIuZnJvbSh1aW50OFJlc3VsdCk7XG4gICAgICB9LFxuICAgIH0gYXMgdW5rbm93biBhcyBIYXNoO1xuICB9XG5cbiAgYnVpbGROZnRUcmFuc2ZlckRhdGEocGFyYW1zOiBCdWlsZE5mdFRyYW5zZmVyRGF0YU9wdGlvbnMpOiBUb2tlblRyYW5zZmVyUmVjaXBpZW50UGFyYW1zIHtcbiAgICBjb25zdCB7IHJlY2lwaWVudEFkZHJlc3MsIGZyb21BZGRyZXNzLCB0b2tlbkNvbnRyYWN0QWRkcmVzcyB9ID0gcGFyYW1zO1xuICAgIGlmICghdXRpbHMuaXNWYWxpZEFkZHJlc3MocmVjaXBpZW50QWRkcmVzcykpIHtcbiAgICAgIHRocm93IG5ldyBJbnZhbGlkQWRkcmVzc0Vycm9yKCdJbnZhbGlkIHJlY2lwaWVudCBhZGRyZXNzJyk7XG4gICAgfVxuICAgIGlmICghdXRpbHMuaXNWYWxpZEFkZHJlc3MoZnJvbUFkZHJlc3MpKSB7XG4gICAgICB0aHJvdyBuZXcgSW52YWxpZEFkZHJlc3NFcnJvcignSW52YWxpZCBmcm9tIGFkZHJlc3MnKTtcbiAgICB9XG4gICAgaWYgKCF1dGlscy5pc1ZhbGlkQWRkcmVzcyh0b2tlbkNvbnRyYWN0QWRkcmVzcykpIHtcbiAgICAgIHRocm93IG5ldyBJbnZhbGlkQWRkcmVzc0Vycm9yKCdJbnZhbGlkIE5GVCBjb250cmFjdCBhZGRyZXNzIGFkZHJlc3MnKTtcbiAgICB9XG4gICAgc3dpdGNoIChwYXJhbXMudHlwZSkge1xuICAgICAgY2FzZSAnRVJDNzIxJzoge1xuICAgICAgICBjb25zdCB0b2tlbklkID0gcGFyYW1zLnRva2VuSWQ7XG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgdG9rZW5UeXBlOiBUb2tlblR5cGUuRVJDNzIxLFxuICAgICAgICAgIHRva2VuUXVhbnRpdHk6ICcxJywgLy8gVGhpcyBORlQgc3RhbmRhcmQgd2lsbCBhbHdheXMgaGF2ZSBxdWFudGl0eSBvZiAxXG4gICAgICAgICAgdG9rZW5Db250cmFjdEFkZHJlc3MsXG4gICAgICAgICAgdG9rZW5JZCxcbiAgICAgICAgfTtcbiAgICAgIH1cbiAgICAgIGRlZmF1bHQ6XG4gICAgICAgIHRocm93IG5ldyBOb3RJbXBsZW1lbnRlZEVycm9yKGBORlQgdHlwZSAke3BhcmFtcy50eXBlfSBub3Qgc3VwcG9ydGVkIG9uICR7dGhpcy5nZXRDaGFpbigpfWApO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBCcm9hZGNhc3RzIGEgc2lnbmVkIHRyYW5zYWN0aW9uIHRvIHRoZSBWZUNoYWluIG5ldHdvcmsuXG4gICAqXG4gICAqIEBwYXJhbSB7QmFzZUJyb2FkY2FzdFRyYW5zYWN0aW9uT3B0aW9uc30gcGF5bG9hZCAtIFRoZSBwYXlsb2FkIGNvbnRhaW5pbmcgdGhlIHNlcmlhbGl6ZWQgc2lnbmVkIHRyYW5zYWN0aW9uLlxuICAgKiBAcGFyYW0ge3N0cmluZ30gcGF5bG9hZC5zZXJpYWxpemVkU2lnbmVkVHJhbnNhY3Rpb24gLSBUaGUgc2VyaWFsaXplZCBzaWduZWQgdHJhbnNhY3Rpb24gdG8gYnJvYWRjYXN0LlxuICAgKiBAcmV0dXJucyB7UHJvbWlzZTxCYXNlQnJvYWRjYXN0VHJhbnNhY3Rpb25SZXN1bHQ+fSBBIHByb21pc2UgdGhhdCByZXNvbHZlcyB0byBhbiBlbXB0eSBvYmplY3QgaWYgdGhlIGJyb2FkY2FzdCBpcyBzdWNjZXNzZnVsLlxuICAgKiBAdGhyb3dzIHtFcnJvcn0gSWYgdGhlIGJyb2FkY2FzdCBmYWlscywgYW4gZXJyb3IgaXMgdGhyb3duIHdpdGggdGhlIGZhaWx1cmUgbWVzc2FnZS5cbiAgICovXG4gIHB1YmxpYyBhc3luYyBicm9hZGNhc3RUcmFuc2FjdGlvbihwYXlsb2FkOiBCYXNlQnJvYWRjYXN0VHJhbnNhY3Rpb25PcHRpb25zKTogUHJvbWlzZTxCYXNlQnJvYWRjYXN0VHJhbnNhY3Rpb25SZXN1bHQ+IHtcbiAgICBjb25zdCBiYXNlVXJsID0gdGhpcy5nZXRQdWJsaWNOb2RlVXJsKCk7XG4gICAgY29uc3QgdXJsID0gYCR7YmFzZVVybH0vdHJhbnNhY3Rpb25zYDtcblxuICAgIC8vIFRoZSBib2R5IHNob3VsZCBiZSBhIEpTT04gb2JqZWN0IHdpdGggYSAncmF3JyBrZXlcbiAgICBjb25zdCByZXF1ZXN0Qm9keSA9IHtcbiAgICAgIHJhdzogcGF5bG9hZC5zZXJpYWxpemVkU2lnbmVkVHJhbnNhY3Rpb24sXG4gICAgfTtcblxuICAgIHRyeSB7XG4gICAgICBhd2FpdCBheGlvcy5wb3N0KHVybCwgcmVxdWVzdEJvZHkpO1xuICAgICAgcmV0dXJuIHt9O1xuICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoYEZhaWxlZCB0byBicm9hZGNhc3QgdHJhbnNhY3Rpb246ICR7ZXJyb3IubWVzc2FnZX1gKTtcbiAgICB9XG4gIH1cblxuICAvKiogQGluaGVyaXREb2MgKi9cbiAgYXN5bmMgcmVjb3ZlcihwYXJhbXM6IFJlY292ZXJPcHRpb25zKTogUHJvbWlzZTxSZWNvdmVyeVRyYW5zYWN0aW9uIHwgVW5zaWduZWRTd2VlcFJlY292ZXJ5VHJhbnNhY3Rpb24+IHtcbiAgICBpZiAocGFyYW1zLnRva2VuQ29udHJhY3RBZGRyZXNzKSB7XG4gICAgICByZXR1cm4gdGhpcy5yZWNvdmVyVG9rZW5zKHBhcmFtcyk7XG4gICAgfVxuICAgIHRyeSB7XG4gICAgICBpZiAoIXBhcmFtcy5yZWNvdmVyeURlc3RpbmF0aW9uIHx8ICF0aGlzLmlzVmFsaWRBZGRyZXNzKHBhcmFtcy5yZWNvdmVyeURlc3RpbmF0aW9uKSkge1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoJ2ludmFsaWQgcmVjb3ZlcnlEZXN0aW5hdGlvbicpO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBpc1Vuc2lnbmVkU3dlZXAgPSAhcGFyYW1zLnVzZXJLZXkgJiYgIXBhcmFtcy5iYWNrdXBLZXkgJiYgIXBhcmFtcy53YWxsZXRQYXNzcGhyYXNlO1xuXG4gICAgICBsZXQgcHVibGljS2V5OiBzdHJpbmcgfCB1bmRlZmluZWQ7XG4gICAgICBsZXQgdXNlcktleVNoYXJlLCBiYWNrdXBLZXlTaGFyZSwgY29tbW9uS2V5Q2hhaW47XG4gICAgICBjb25zdCBNUEMgPSBuZXcgRWNkc2EoKTtcblxuICAgICAgaWYgKGlzVW5zaWduZWRTd2VlcCkge1xuICAgICAgICBjb25zdCBiaXRnb0tleSA9IHBhcmFtcy5iaXRnb0tleTtcbiAgICAgICAgaWYgKCFiaXRnb0tleSkge1xuICAgICAgICAgIHRocm93IG5ldyBFcnJvcignbWlzc2luZyBiaXRnb0tleScpO1xuICAgICAgICB9XG5cbiAgICAgICAgY29uc3QgaGRUcmVlID0gbmV3IG1wYy5TZWNwMjU2azFCaXAzMkhkVHJlZSgpO1xuICAgICAgICBjb25zdCBkZXJpdmF0aW9uUGF0aCA9ICdtLzAnO1xuICAgICAgICBjb25zdCBkZXJpdmVkUHViID0gaGRUcmVlLnB1YmxpY0Rlcml2ZShcbiAgICAgICAgICB7XG4gICAgICAgICAgICBwazogbXBjLmJpZ0ludEZyb21CdWZmZXJCRShCdWZmZXIuZnJvbShiaXRnb0tleS5zbGljZSgwLCA2NiksICdoZXgnKSksXG4gICAgICAgICAgICBjaGFpbmNvZGU6IG1wYy5iaWdJbnRGcm9tQnVmZmVyQkUoQnVmZmVyLmZyb20oYml0Z29LZXkuc2xpY2UoNjYpLCAnaGV4JykpLFxuICAgICAgICAgIH0sXG4gICAgICAgICAgZGVyaXZhdGlvblBhdGhcbiAgICAgICAgKTtcblxuICAgICAgICBwdWJsaWNLZXkgPSBtcGMuYmlnSW50VG9CdWZmZXJCRShkZXJpdmVkUHViLnBrKS50b1N0cmluZygnaGV4Jyk7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICBpZiAoIXBhcmFtcy51c2VyS2V5KSB7XG4gICAgICAgICAgdGhyb3cgbmV3IEVycm9yKCdtaXNzaW5nIHVzZXJLZXknKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGlmICghcGFyYW1zLmJhY2t1cEtleSkge1xuICAgICAgICAgIHRocm93IG5ldyBFcnJvcignbWlzc2luZyBiYWNrdXBLZXknKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGlmICghcGFyYW1zLndhbGxldFBhc3NwaHJhc2UpIHtcbiAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoJ21pc3Npbmcgd2FsbGV0IHBhc3NwaHJhc2UnKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGNvbnN0IHVzZXJLZXkgPSBwYXJhbXMudXNlcktleS5yZXBsYWNlKC9cXHMvZywgJycpO1xuICAgICAgICBjb25zdCBiYWNrdXBLZXkgPSBwYXJhbXMuYmFja3VwS2V5LnJlcGxhY2UoL1xccy9nLCAnJyk7XG5cbiAgICAgICAgKHsgdXNlcktleVNoYXJlLCBiYWNrdXBLZXlTaGFyZSwgY29tbW9uS2V5Q2hhaW4gfSA9IGF3YWl0IEVDRFNBVXRpbHMuZ2V0TXBjVjJSZWNvdmVyeUtleVNoYXJlcyhcbiAgICAgICAgICB1c2VyS2V5LFxuICAgICAgICAgIGJhY2t1cEtleSxcbiAgICAgICAgICBwYXJhbXMud2FsbGV0UGFzc3BocmFzZVxuICAgICAgICApKTtcbiAgICAgICAgcHVibGljS2V5ID0gTVBDLmRlcml2ZVVuaGFyZGVuZWQoY29tbW9uS2V5Q2hhaW4sICdtLzAnKS5zbGljZSgwLCA2Nik7XG4gICAgICB9XG5cbiAgICAgIGlmICghcHVibGljS2V5KSB7XG4gICAgICAgIHRocm93IG5ldyBFcnJvcignZmFpbGVkIHRvIGRlcml2ZSBwdWJsaWMga2V5Jyk7XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IGJhY2t1cEtleVBhaXIgPSBuZXcgRXRoS2V5UGFpcih7IHB1YjogcHVibGljS2V5IH0pO1xuICAgICAgY29uc3QgYmFzZUFkZHJlc3MgPSBiYWNrdXBLZXlQYWlyLmdldEFkZHJlc3MoKTtcblxuICAgICAgY29uc3QgdHggPSBhd2FpdCB0aGlzLmJ1aWxkUmVjb3ZlcnlUcmFuc2FjdGlvbih7XG4gICAgICAgIGJhc2VBZGRyZXNzLFxuICAgICAgICBwYXJhbXMsXG4gICAgICB9KTtcblxuICAgICAgY29uc3Qgc2lnbmFibGVIZXggPSBhd2FpdCB0eC5zaWduYWJsZVBheWxvYWQ7XG4gICAgICBjb25zdCBzZXJpYWxpemVkVHhIZXggPSBhd2FpdCB0eC50b0Jyb2FkY2FzdEZvcm1hdCgpO1xuXG4gICAgICBpZiAoaXNVbnNpZ25lZFN3ZWVwKSB7XG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgdHhIZXg6IHNlcmlhbGl6ZWRUeEhleCxcbiAgICAgICAgICBjb2luOiB0aGlzLmdldENoYWluKCksXG4gICAgICAgIH07XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IHNpZ25hYmxlTWVzc2FnZSA9IHRoaXMuZ2V0SGFzaEZ1bmN0aW9uKCkudXBkYXRlKHNpZ25hYmxlSGV4KS5kaWdlc3QoKTtcblxuICAgICAgY29uc3Qgc2lnbmF0dXJlT2JqID0gYXdhaXQgRUNEU0FVdGlscy5zaWduUmVjb3ZlcnlNcGNWMihcbiAgICAgICAgc2lnbmFibGVNZXNzYWdlLFxuICAgICAgICB1c2VyS2V5U2hhcmUsXG4gICAgICAgIGJhY2t1cEtleVNoYXJlLFxuICAgICAgICBjb21tb25LZXlDaGFpblxuICAgICAgKTtcbiAgICAgIGNvbnN0IHNpZ25hdHVyZSA9IEJ1ZmZlci5mcm9tKHNpZ25hdHVyZU9iai5yICsgc2lnbmF0dXJlT2JqLnMgKyAoc2lnbmF0dXJlT2JqLnJlY2lkID09PSAwID8gJzAwJyA6ICcwMScpLCAnaGV4Jyk7XG4gICAgICBjb25zdCB0eEJ1aWxkZXIgPSB0aGlzLmdldFR4QnVpbGRlckZhY3RvcnkoKS5nZXRUcmFuc2ZlckJ1aWxkZXIoKTtcbiAgICAgIGF3YWl0IHR4QnVpbGRlci5mcm9tKHNlcmlhbGl6ZWRUeEhleCk7XG4gICAgICB0eEJ1aWxkZXIuaXNSZWNvdmVyeSh0cnVlKTtcbiAgICAgIGF3YWl0IHR4QnVpbGRlci5hZGRTZW5kZXJTaWduYXR1cmUoc2lnbmF0dXJlKTtcblxuICAgICAgY29uc3Qgc2lnbmVkVHggPSBhd2FpdCB0eEJ1aWxkZXIuYnVpbGQoKTtcblxuICAgICAgcmV0dXJuIHtcbiAgICAgICAgaWQ6IHNpZ25lZFR4LmlkLFxuICAgICAgICB0eDogc2lnbmVkVHgudG9Ccm9hZGNhc3RGb3JtYXQoKSxcbiAgICAgIH07XG4gICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihgRXJyb3IgZHVyaW5nIFZlY2hhaW4gcmVjb3Zlcnk6ICR7ZXJyb3IubWVzc2FnZSB8fCBlcnJvcn1gKTtcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogUmV0dXJucyB0aGUgcHVibGljIG5vZGUgVVJMIGZvciB0aGUgVmVDaGFpbiBuZXR3b3JrLlxuICAgKiBAcmV0dXJucyB7c3RyaW5nfSBUaGUgVVJMIG9mIHRoZSBwdWJsaWMgVmVDaGFpbiBub2RlLlxuICAgKi9cbiAgcHJpdmF0ZSBnZXRQdWJsaWNOb2RlVXJsKCk6IHN0cmluZyB7XG4gICAgcmV0dXJuIEVudmlyb25tZW50c1t0aGlzLmJpdGdvLmdldEVudigpXS52ZXROb2RlVXJsO1xuICB9XG5cbiAgLyoqXG4gICAqIENhbGN1bGF0ZXMgdGhlIHRyYW5zYWN0aW9uIGZlZSBiYXNlZCBvbiB0aGUgZXN0aW1hdGVkIGdhcyBsaW1pdCBhbmQgZmVlIGVzdGltYXRlIGRhdGEuXG4gICAqIEBwYXJhbSB7RmVlRXN0aW1hdGVEYXRhfSBmZWVFc3RpbWF0ZURhdGEgLSBUaGUgZmVlIGVzdGltYXRlIGRhdGEuXG4gICAqIEBwYXJhbSB7QmlnTnVtYmVyfSBlc3RpbWF0ZWRHYXNMaW1pdCAtIFRoZSBlc3RpbWF0ZWQgZ2FzIGxpbWl0IGZvciB0aGUgdHJhbnNhY3Rpb24uXG4gICAqIEByZXR1cm5zIHtCaWdOdW1iZXJ9IFRoZSBjYWxjdWxhdGVkIHRyYW5zYWN0aW9uIGZlZS5cbiAgICovXG4gIHByaXZhdGUgY2FsY3VsYXRlRmVlKGZlZUVzdGltYXRlRGF0YTogRmVlRXN0aW1hdGVEYXRhLCBlc3RpbWF0ZWRHYXNMaW1pdDogQmlnTnVtYmVyKTogQmlnTnVtYmVyIHtcbiAgICBjb25zdCBnYXNMaW1pdCA9IGVzdGltYXRlZEdhc0xpbWl0O1xuICAgIGNvbnN0IGFkanVzdG1lbnRGYWN0b3IgPSBuZXcgQmlnTnVtYmVyKDEpLnBsdXMoXG4gICAgICBuZXcgQmlnTnVtYmVyKGZlZUVzdGltYXRlRGF0YS5nYXNQcmljZUNvZWYpXG4gICAgICAgIC5kaXZpZGVkQnkobmV3IEJpZ051bWJlcihmZWVFc3RpbWF0ZURhdGEuY29lZkRpdmlzb3IpKVxuICAgICAgICAuZGVjaW1hbFBsYWNlcygxOCwgQmlnTnVtYmVyLlJPVU5EX0RPV04pXG4gICAgKTtcbiAgICBjb25zdCBhZGp1c3RlZEdhc1ByaWNlID0gbmV3IEJpZ051bWJlcihmZWVFc3RpbWF0ZURhdGEuZ2FzVW5pdFByaWNlKS50aW1lcyhhZGp1c3RtZW50RmFjdG9yKTtcbiAgICByZXR1cm4gZ2FzTGltaXQudGltZXMoYWRqdXN0ZWRHYXNQcmljZSkuaW50ZWdlclZhbHVlKEJpZ051bWJlci5ST1VORF9DRUlMKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBFbnN1cmVzIHRoYXQgdGhlIGdpdmVuIGFkZHJlc3MgaGFzIHN1ZmZpY2llbnQgVlRITyBiYWxhbmNlIHRvIGNvdmVyIHRoZSB0cmFuc2FjdGlvbiBmZWUuXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBiYXNlQWRkcmVzcyAtIFRoZSBhZGRyZXNzIHRvIGNoZWNrIGZvciBWVEhPIGJhbGFuY2UuXG4gICAqIEBwYXJhbSB7QmlnTnVtYmVyfSByZXF1aXJlZEdhc1VuaXRzIC0gVGhlIHJlcXVpcmVkIGdhcyB1bml0cyBmb3IgdGhlIHRyYW5zYWN0aW9uLlxuICAgKiBAdGhyb3dzIHtFcnJvcn0gSWYgdGhlIFZUSE8gYmFsYW5jZSBpcyBpbnN1ZmZpY2llbnQgb3IgaWYgdGhlcmUncyBhbiBlcnJvciBjaGVja2luZyB0aGUgYmFsYW5jZS5cbiAgICovXG4gIGFzeW5jIGVuc3VyZVZ0aG9CYWxhbmNlRm9yRmVlKGJhc2VBZGRyZXNzOiBzdHJpbmcsIHJlcXVpcmVkR2FzVW5pdHM6IEJpZ051bWJlcik6IFByb21pc2U8dm9pZD4ge1xuICAgIGNvbnN0IHZ0aG9Ub2tlbkFkZHJlc3MgPSAnMHgwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwMDAwNDU2RTY1NzI2Nzc5JzsgLy8gVlRITyB0b2tlbiBjb250cmFjdCBhZGRyZXNzXG4gICAgdHJ5IHtcbiAgICAgIGNvbnN0IHZ0aG9CYWxhbmNlID0gYXdhaXQgdGhpcy5nZXRCYWxhbmNlKGJhc2VBZGRyZXNzLCB2dGhvVG9rZW5BZGRyZXNzKTtcblxuICAgICAgY29uc3QgcmVxdWlyZWRGZWUgPSB0aGlzLmNhbGN1bGF0ZUZlZShmZWVFc3RpbWF0ZURhdGEsIHJlcXVpcmVkR2FzVW5pdHMpO1xuXG4gICAgICBpZiAodnRob0JhbGFuY2UuaXNMZXNzVGhhbihyZXF1aXJlZEZlZSkpIHtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgICAgIGBJbnN1ZmZpY2llbnQgVlRITyBiYWxhbmNlIGZvciBmZWVzLiBSZXF1aXJlZDogJHtyZXF1aXJlZEZlZS50b1N0cmluZygpfSwgQXZhaWxhYmxlOiAke3Z0aG9CYWxhbmNlLnRvU3RyaW5nKCl9YFxuICAgICAgICApO1xuICAgICAgfVxuICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoYEZhaWxlZCB0byBlbnN1cmUgVlRITyBiYWxhbmNlOiAke2Vycm9yLm1lc3NhZ2V9YCk7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIEZldGNoZXMgdGhlIGJhbGFuY2UgZm9yIGEgZ2l2ZW4gVmVjaGFpbiBhZGRyZXNzLlxuICAgKlxuICAgKiBAcGFyYW0gYWRkcmVzcyBUaGUgVmVjaGFpbiBhZGRyZXNzIChlLmcuLCBcIjB4Li4uXCIpIHRvIGNoZWNrLlxuICAgKiBAcGFyYW0gdG9rZW5Db250cmFjdEFkZHJlc3MgKE9wdGlvbmFsKSBUaGUgY29udHJhY3QgYWRkcmVzcyBvZiBhIFZJUDE4MCB0b2tlbi5cbiAgICogQHJldHVybnMgQSBQcm9taXNlIHRoYXQgcmVzb2x2ZXMgdG8gYSBCaWdOdW1iZXIgaW5zdGFuY2Ugb2YgdGhlIGJhbGFuY2UuXG4gICAqL1xuICBhc3luYyBnZXRCYWxhbmNlKGFkZHJlc3M6IHN0cmluZywgdG9rZW5Db250cmFjdEFkZHJlc3M/OiBzdHJpbmcpOiBQcm9taXNlPEJpZ051bWJlcj4ge1xuICAgIGNvbnN0IGJhc2VVcmwgPSB0aGlzLmdldFB1YmxpY05vZGVVcmwoKTtcblxuICAgIGlmICghdG9rZW5Db250cmFjdEFkZHJlc3MpIHtcbiAgICAgIGNvbnN0IHVybCA9IGAke2Jhc2VVcmx9L2FjY291bnRzLyR7YWRkcmVzc31gO1xuXG4gICAgICB0cnkge1xuICAgICAgICBjb25zdCByZXNwb25zZSA9IGF3YWl0IGF4aW9zLmdldCh1cmwpO1xuXG4gICAgICAgIC8vIFRoZSAnYmFsYW5jZScgaXMgcmV0dXJuZWQgYXMgYSBoZXggc3RyaW5nLlxuICAgICAgICBjb25zdCBiYWxhbmNlID0gbmV3IEJpZ051bWJlcihyZXNwb25zZS5kYXRhLmJhbGFuY2UpO1xuXG4gICAgICAgIHJldHVybiBiYWxhbmNlO1xuICAgICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKCdGYWlsZWQgdG8gZ2V0IG5hdGl2ZSBiYWxhbmNlLicpO1xuICAgICAgfVxuICAgIH1cblxuICAgIGNvbnN0IHVybCA9IGAke2Jhc2VVcmx9L2FjY291bnRzLypgO1xuXG4gICAgLy8gQ29uc3RydWN0IHRoZSBBQkktZW5jb2RlZCBkYXRhIGZvciB0aGUgJ2JhbGFuY2VPZihhZGRyZXNzKScgY2FsbFxuICAgIC8vIDEuIEZ1bmN0aW9uIHNlbGVjdG9yIGZvciAnYmFsYW5jZU9mKGFkZHJlc3MpJzogJzB4NzBhMDgyMzEnXG4gICAgLy8gMi4gUGFkZGVkIGFkZHJlc3M6IFRoZSBhZGRyZXNzLCBzdHJpcHBlZCBvZiAnMHgnLCBsZWZ0LXBhZGRlZCB3aXRoIHplcm9zIHRvIDY0IGNoYXJzXG4gICAgY29uc3QgcGFkZGVkQWRkcmVzcyA9IGFkZHJlc3Muc3RhcnRzV2l0aCgnMHgnKSA/IGFkZHJlc3Muc3Vic3RyaW5nKDIpLnBhZFN0YXJ0KDY0LCAnMCcpIDogYWRkcmVzcy5wYWRTdGFydCg2NCwgJzAnKTtcbiAgICBjb25zdCBkYXRhID0gYDB4NzBhMDgyMzEke3BhZGRlZEFkZHJlc3N9YDtcblxuICAgIGNvbnN0IHJlcXVlc3RCb2R5ID0ge1xuICAgICAgY2xhdXNlczogW1xuICAgICAgICB7XG4gICAgICAgICAgdG86IHRva2VuQ29udHJhY3RBZGRyZXNzLCAvLyBUaGUgdG9rZW4gY29udHJhY3QgYWRkcmVzc1xuICAgICAgICAgIHZhbHVlOiAnMHgwJyxcbiAgICAgICAgICBkYXRhOiBkYXRhLCAvLyBUaGUgJ2JhbGFuY2VPZicgY2FsbFxuICAgICAgICB9LFxuICAgICAgXSxcbiAgICB9O1xuXG4gICAgdHJ5IHtcbiAgICAgIGNvbnN0IHJlc3BvbnNlID0gYXdhaXQgYXhpb3MucG9zdCh1cmwsIHJlcXVlc3RCb2R5KTtcblxuICAgICAgY29uc3Qgc2ltUmVzcG9uc2UgPSByZXNwb25zZS5kYXRhO1xuXG4gICAgICAvLyBWYWxpZGF0ZSByZXNwb25zZSBhbmQgZXh0cmFjdCB0aGUgYmFsYW5jZSBkYXRhXG4gICAgICBpZiAoIXNpbVJlc3BvbnNlIHx8ICFBcnJheS5pc0FycmF5KHNpbVJlc3BvbnNlKSB8fCBzaW1SZXNwb25zZS5sZW5ndGggPT09IDAgfHwgIXNpbVJlc3BvbnNlWzBdLmRhdGEpIHtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKCdJbnZhbGlkIHNpbXVsYXRpb24gcmVzcG9uc2UgZnJvbSBWZUNoYWluIG5vZGUnKTtcbiAgICAgIH1cblxuICAgICAgLy8gVGhlIHJldHVybmVkIGRhdGEgaXMgdGhlIGhleC1lbmNvZGVkIGJhbGFuY2VcbiAgICAgIHJldHVybiBuZXcgQmlnTnVtYmVyKHNpbVJlc3BvbnNlWzBdLmRhdGEpO1xuICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICBjb25zb2xlLmVycm9yKCdFcnJvciBmZXRjaGluZyB0b2tlbiBiYWxhbmNlOicsIGVycm9yKTtcbiAgICAgIHRocm93IG5ldyBFcnJvcihgRmFpbGVkIHRvIGdldCB0b2tlbiBiYWxhbmNlOiAke2Vycm9yLm1lc3NhZ2V9YCk7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIFJldHJpZXZlcyB0aGUgYmxvY2sgcmVmZXJlbmNlIGZyb20gdGhlIFZlQ2hhaW4gbmV0d29yay5cbiAgICogQHJldHVybnMge1Byb21pc2U8c3RyaW5nPn0gQSBwcm9taXNlIHRoYXQgcmVzb2x2ZXMgdG8gdGhlIGJsb2NrIHJlZmVyZW5jZSBzdHJpbmcuXG4gICAqIEB0aHJvd3Mge0Vycm9yfSBJZiB0aGVyZSdzIGFuIGVycm9yIGZldGNoaW5nIHRoZSBibG9jayByZWZlcmVuY2Ugb3IgaWYgdGhlIHJlc3BvbnNlIGlzIGludmFsaWQuXG4gICAqL1xuICBwdWJsaWMgYXN5bmMgZ2V0QmxvY2tSZWYoKTogUHJvbWlzZTxzdHJpbmc+IHtcbiAgICBjb25zdCBiYXNlVXJsID0gdGhpcy5nZXRQdWJsaWNOb2RlVXJsKCk7XG4gICAgY29uc3QgdXJsID0gYCR7YmFzZVVybH0vYmxvY2tzL2Jlc3RgO1xuXG4gICAgdHJ5IHtcbiAgICAgIGNvbnN0IHJlc3BvbnNlID0gYXdhaXQgYXhpb3MuZ2V0KHVybCk7XG5cbiAgICAgIGNvbnN0IGRhdGEgPSByZXNwb25zZS5kYXRhO1xuXG4gICAgICAvLyBWYWxpZGF0ZSB0aGUgcmVzcG9uc2UgZGF0YVxuICAgICAgaWYgKCFkYXRhIHx8ICFkYXRhLmlkKSB7XG4gICAgICAgIHRocm93IG5ldyBFcnJvcignSW52YWxpZCByZXNwb25zZSBmcm9tIHRoZSBWZUNoYWluIG5vZGUnKTtcbiAgICAgIH1cblxuICAgICAgLy8gUmV0dXJuIHRoZSBmaXJzdCAxOCBjaGFyYWN0ZXJzIG9mIHRoZSBibG9jayBJRFxuICAgICAgcmV0dXJuIGRhdGEuaWQuc2xpY2UoMCwgMTgpO1xuICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICAvLyBSZXRocm93IG9yIHJldHVybiBhIHNlbnNpYmxlIGRlZmF1bHRcbiAgICAgIHRocm93IG5ldyBFcnJvcignRmFpbGVkIHRvIGdldCBibG9jayByZWY6ICcpO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBHZW5lcmF0ZXMgYSByYW5kb20gbm9uY2UgZm9yIHVzZSBpbiB0cmFuc2FjdGlvbnMuXG4gICAqIEByZXR1cm5zIHtzdHJpbmd9IEEgaGV4YWRlY2ltYWwgc3RyaW5nIHJlcHJlc2VudGluZyB0aGUgcmFuZG9tIG5vbmNlLlxuICAgKi9cbiAgZ2V0UmFuZG9tTm9uY2UoKTogc3RyaW5nIHtcbiAgICByZXR1cm4gJzB4JyArIHJhbmRvbUJ5dGVzKDgpLnRvU3RyaW5nKCdoZXgnKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBFc3RpbWF0ZXMgdGhlIGdhcyByZXF1aXJlZCBmb3IgYSBzZXQgb2YgdHJhbnNhY3Rpb24gY2xhdXNlcy5cbiAgICogQHBhcmFtIHtUcmFuc2FjdGlvbkNsYXVzZVtdfSBjbGF1c2VzIC0gQW4gYXJyYXkgb2YgdHJhbnNhY3Rpb24gY2xhdXNlcy5cbiAgICogQHBhcmFtIHtzdHJpbmd9IGNhbGxlciAtIFRoZSBhZGRyZXNzIG9mIHRoZSB0cmFuc2FjdGlvbiBjYWxsZXIuXG4gICAqIEByZXR1cm5zIHtQcm9taXNlPEJpZ051bWJlcj59IEEgcHJvbWlzZSB0aGF0IHJlc29sdmVzIHRvIHRoZSBlc3RpbWF0ZWQgZ2FzIGFtb3VudC5cbiAgICogQHRocm93cyB7RXJyb3J9IElmIHRoZSBjbGF1c2VzIGFyZSBpbnZhbGlkLCB0aGUgY2FsbGVyIGlzIG5vdCBwcm92aWRlZCwgb3IgaWYgdGhlcmUncyBhbiBlcnJvciBpbiBnYXMgZXN0aW1hdGlvbi5cbiAgICovXG4gIHB1YmxpYyBhc3luYyBlc3RpbWF0ZUdhcyhjbGF1c2VzOiBUcmFuc2FjdGlvbkNsYXVzZVtdLCBjYWxsZXI6IHN0cmluZyk6IFByb21pc2U8QmlnTnVtYmVyPiB7XG4gICAgaWYgKCFjbGF1c2VzIHx8ICFBcnJheS5pc0FycmF5KGNsYXVzZXMpIHx8IGNsYXVzZXMubGVuZ3RoID09PSAwKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoJ0NsYXVzZXMgbXVzdCBiZSBhIG5vbi1lbXB0eSBhcnJheScpO1xuICAgIH1cblxuICAgIGlmICghY2FsbGVyKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoJ0NhbGxlciBhZGRyZXNzIGlzIHJlcXVpcmVkJyk7XG4gICAgfVxuXG4gICAgY29uc3QgYmFzZVVybCA9IHRoaXMuZ2V0UHVibGljTm9kZVVybCgpO1xuICAgIGNvbnN0IHVybCA9IGAke2Jhc2VVcmx9L2FjY291bnRzLypgO1xuXG4gICAgY29uc3QgcmVxdWVzdEJvZHkgPSB7XG4gICAgICBjbGF1c2VzOiBjbGF1c2VzLFxuICAgICAgY2FsbGVyOiBjYWxsZXIsXG4gICAgfTtcblxuICAgIHRyeSB7XG4gICAgICBjb25zdCByZXNwb25zZSA9IGF3YWl0IGF4aW9zLnBvc3QodXJsLCByZXF1ZXN0Qm9keSk7XG5cbiAgICAgIGNvbnN0IHNpbVJlc3BvbnNlID0gcmVzcG9uc2UuZGF0YTtcblxuICAgICAgaWYgKCFzaW1SZXNwb25zZSB8fCAhQXJyYXkuaXNBcnJheShzaW1SZXNwb25zZSkpIHtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKCdJbnZhbGlkIHNpbXVsYXRpb24gcmVzcG9uc2UgZnJvbSBWZUNoYWluIG5vZGUnKTtcbiAgICAgIH1cblxuICAgICAgY29uc3QgdG90YWxTaW11bGF0ZWRHYXMgPSBzaW1SZXNwb25zZS5yZWR1Y2UoKHN1bSwgcmVzdWx0KSA9PiBzdW0gKyAocmVzdWx0Lmdhc1VzZWQgfHwgMCksIDApO1xuXG4gICAgICBjb25zdCBpbnRyaW5zaWNHYXMgPSBOdW1iZXIoVmV0VHJhbnNhY3Rpb24uaW50cmluc2ljR2FzKGNsYXVzZXMpLndlaSk7XG5cbiAgICAgIGNvbnN0IHRvdGFsR2FzID0gTWF0aC5jZWlsKGludHJpbnNpY0dhcyArICh0b3RhbFNpbXVsYXRlZEdhcyAhPT0gMCA/IHRvdGFsU2ltdWxhdGVkR2FzICsgMTUwMDAgOiAwKSk7XG5cbiAgICAgIHJldHVybiBuZXcgQmlnTnVtYmVyKHRvdGFsR2FzKTtcbiAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKGBGYWlsZWQgdG8gZXN0aW1hdGUgZ2FzOiAke2Vycm9yLm1lc3NhZ2V9YCk7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIEJ1aWxkcyBhIHJlY292ZXJ5IHRyYW5zYWN0aW9uIGZvciB0aGUgZ2l2ZW4gYWRkcmVzcy5cbiAgICogQHBhcmFtIHtPYmplY3R9IGJ1aWxkUGFyYW1zIC0gVGhlIHBhcmFtZXRlcnMgZm9yIGJ1aWxkaW5nIHRoZSByZWNvdmVyeSB0cmFuc2FjdGlvbi5cbiAgICogQHBhcmFtIHtzdHJpbmd9IGJ1aWxkUGFyYW1zLmJhc2VBZGRyZXNzIC0gVGhlIGFkZHJlc3MgdG8gcmVjb3ZlciBmdW5kcyBmcm9tLlxuICAgKiBAcGFyYW0ge1JlY292ZXJPcHRpb25zfSBidWlsZFBhcmFtcy5wYXJhbXMgLSBUaGUgcmVjb3Zlcnkgb3B0aW9ucy5cbiAgICogQHJldHVybnMge1Byb21pc2U8VHJhbnNhY3Rpb24+fSBBIHByb21pc2UgdGhhdCByZXNvbHZlcyB0byB0aGUgYnVpbHQgcmVjb3ZlcnkgdHJhbnNhY3Rpb24uXG4gICAqIEB0aHJvd3Mge0Vycm9yfSBJZiB0aGVyZSdzIG5vIFZFVCBiYWxhbmNlIHRvIHJlY292ZXIgb3IgaWYgdGhlcmUncyBhbiBlcnJvciBidWlsZGluZyB0aGUgdHJhbnNhY3Rpb24uXG4gICAqL1xuICBwcml2YXRlIGFzeW5jIGJ1aWxkUmVjb3ZlcnlUcmFuc2FjdGlvbihidWlsZFBhcmFtczoge1xuICAgIGJhc2VBZGRyZXNzOiBzdHJpbmc7XG4gICAgcGFyYW1zOiBSZWNvdmVyT3B0aW9ucztcbiAgfSk6IFByb21pc2U8VHJhbnNhY3Rpb24+IHtcbiAgICBjb25zdCB7IGJhc2VBZGRyZXNzLCBwYXJhbXMgfSA9IGJ1aWxkUGFyYW1zO1xuICAgIGNvbnN0IGJhbGFuY2UgPSBhd2FpdCB0aGlzLmdldEJhbGFuY2UoYmFzZUFkZHJlc3MpO1xuXG4gICAgaWYgKGJhbGFuY2UuaXNMZXNzVGhhbk9yRXF1YWxUbygwKSkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKGBubyBWRVQgYmFsYW5jZSB0byByZWNvdmVyIGZvciBhZGRyZXNzICR7YmFzZUFkZHJlc3N9YCk7XG4gICAgfVxuXG4gICAgY29uc3QgcmVjaXBpZW50cyA9IFtcbiAgICAgIHtcbiAgICAgICAgYWRkcmVzczogcGFyYW1zLnJlY292ZXJ5RGVzdGluYXRpb24sXG4gICAgICAgIGFtb3VudDogYmFsYW5jZS50b1N0cmluZygpLFxuICAgICAgfSxcbiAgICBdO1xuXG4gICAgY29uc3QgYmxvY2tSZWYgPSBhd2FpdCB0aGlzLmdldEJsb2NrUmVmKCk7XG5cbiAgICBjb25zdCB0eEJ1aWxkZXIgPSB0aGlzLmdldFR4QnVpbGRlckZhY3RvcnkoKS5nZXRUcmFuc2ZlckJ1aWxkZXIoKTtcblxuICAgIHR4QnVpbGRlci5jaGFpblRhZyh0aGlzLmJpdGdvLmdldEVudigpID09PSAncHJvZCcgPyAweDRhIDogMHgyNyk7XG4gICAgdHhCdWlsZGVyLnJlY2lwaWVudHMocmVjaXBpZW50cyk7XG4gICAgdHhCdWlsZGVyLnNlbmRlcihiYXNlQWRkcmVzcyk7XG4gICAgdHhCdWlsZGVyLmFkZEZlZVBheWVyQWRkcmVzcyhiYXNlQWRkcmVzcyk7XG4gICAgdHhCdWlsZGVyLmdhcyhOdW1iZXIoQVZHX0dBU19VTklUUykpO1xuICAgIHR4QnVpbGRlci5ibG9ja1JlZihibG9ja1JlZik7XG4gICAgdHhCdWlsZGVyLmV4cGlyYXRpb24oRVhQSVJBVElPTik7XG4gICAgdHhCdWlsZGVyLmdhc1ByaWNlQ29lZihOdW1iZXIoR0FTX1BSSUNFX0NPRUYpKTtcbiAgICB0eEJ1aWxkZXIubm9uY2UodGhpcy5nZXRSYW5kb21Ob25jZSgpKTtcbiAgICB0eEJ1aWxkZXIuaXNSZWNvdmVyeSh0cnVlKTtcblxuICAgIGxldCB0eCA9IChhd2FpdCB0eEJ1aWxkZXIuYnVpbGQoKSkgYXMgVHJhbnNhY3Rpb247XG5cbiAgICBjb25zdCBjbGF1c2VzID0gdHguY2xhdXNlcztcblxuICAgIGNvbnN0IGFjdHVhbEdhc1VuaXRzID0gYXdhaXQgdGhpcy5lc3RpbWF0ZUdhcyhjbGF1c2VzLCBiYXNlQWRkcmVzcyk7XG5cbiAgICBhd2FpdCB0aGlzLmVuc3VyZVZ0aG9CYWxhbmNlRm9yRmVlKGJhc2VBZGRyZXNzLCBhY3R1YWxHYXNVbml0cyk7XG5cbiAgICB0eEJ1aWxkZXIuZ2FzKGFjdHVhbEdhc1VuaXRzLnRvTnVtYmVyKCkpO1xuXG4gICAgdHggPSAoYXdhaXQgdHhCdWlsZGVyLmJ1aWxkKCkpIGFzIFRyYW5zYWN0aW9uO1xuXG4gICAgcmV0dXJuIHR4O1xuICB9XG5cbiAgYXN5bmMgcmVjb3ZlclRva2VucyhwYXJhbXM6IFJlY292ZXJPcHRpb25zKTogUHJvbWlzZTxSZWNvdmVyeVRyYW5zYWN0aW9uIHwgVW5zaWduZWRTd2VlcFJlY292ZXJ5VHJhbnNhY3Rpb24+IHtcbiAgICB0cnkge1xuICAgICAgaWYgKCFwYXJhbXMucmVjb3ZlcnlEZXN0aW5hdGlvbiB8fCAhdGhpcy5pc1ZhbGlkQWRkcmVzcyhwYXJhbXMucmVjb3ZlcnlEZXN0aW5hdGlvbikpIHtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKCdpbnZhbGlkIHJlY292ZXJ5RGVzdGluYXRpb24nKTtcbiAgICAgIH1cbiAgICAgIGlmICghcGFyYW1zLnRva2VuQ29udHJhY3RBZGRyZXNzIHx8ICF0aGlzLmlzVmFsaWRBZGRyZXNzKHBhcmFtcy50b2tlbkNvbnRyYWN0QWRkcmVzcykpIHtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKCdpbnZhbGlkIHRva2VuQ29udHJhY3RBZGRyZXNzJyk7XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IGlzVW5zaWduZWRTd2VlcCA9ICFwYXJhbXMudXNlcktleSAmJiAhcGFyYW1zLmJhY2t1cEtleSAmJiAhcGFyYW1zLndhbGxldFBhc3NwaHJhc2U7XG5cbiAgICAgIGxldCBwdWJsaWNLZXk6IHN0cmluZyB8IHVuZGVmaW5lZDtcbiAgICAgIGxldCB1c2VyS2V5U2hhcmUsIGJhY2t1cEtleVNoYXJlLCBjb21tb25LZXlDaGFpbjtcbiAgICAgIGNvbnN0IE1QQyA9IG5ldyBFY2RzYSgpO1xuXG4gICAgICBpZiAoaXNVbnNpZ25lZFN3ZWVwKSB7XG4gICAgICAgIGNvbnN0IGJpdGdvS2V5ID0gcGFyYW1zLmJpdGdvS2V5O1xuICAgICAgICBpZiAoIWJpdGdvS2V5KSB7XG4gICAgICAgICAgdGhyb3cgbmV3IEVycm9yKCdtaXNzaW5nIGJpdGdvS2V5Jyk7XG4gICAgICAgIH1cblxuICAgICAgICBjb25zdCBoZFRyZWUgPSBuZXcgbXBjLlNlY3AyNTZrMUJpcDMySGRUcmVlKCk7XG4gICAgICAgIGNvbnN0IGRlcml2YXRpb25QYXRoID0gJ20vMCc7XG4gICAgICAgIGNvbnN0IGRlcml2ZWRQdWIgPSBoZFRyZWUucHVibGljRGVyaXZlKFxuICAgICAgICAgIHtcbiAgICAgICAgICAgIHBrOiBtcGMuYmlnSW50RnJvbUJ1ZmZlckJFKEJ1ZmZlci5mcm9tKGJpdGdvS2V5LnNsaWNlKDAsIDY2KSwgJ2hleCcpKSxcbiAgICAgICAgICAgIGNoYWluY29kZTogbXBjLmJpZ0ludEZyb21CdWZmZXJCRShCdWZmZXIuZnJvbShiaXRnb0tleS5zbGljZSg2NiksICdoZXgnKSksXG4gICAgICAgICAgfSxcbiAgICAgICAgICBkZXJpdmF0aW9uUGF0aFxuICAgICAgICApO1xuXG4gICAgICAgIHB1YmxpY0tleSA9IG1wYy5iaWdJbnRUb0J1ZmZlckJFKGRlcml2ZWRQdWIucGspLnRvU3RyaW5nKCdoZXgnKTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIGlmICghcGFyYW1zLnVzZXJLZXkpIHtcbiAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoJ21pc3NpbmcgdXNlcktleScpO1xuICAgICAgICB9XG5cbiAgICAgICAgaWYgKCFwYXJhbXMuYmFja3VwS2V5KSB7XG4gICAgICAgICAgdGhyb3cgbmV3IEVycm9yKCdtaXNzaW5nIGJhY2t1cEtleScpO1xuICAgICAgICB9XG5cbiAgICAgICAgaWYgKCFwYXJhbXMud2FsbGV0UGFzc3BocmFzZSkge1xuICAgICAgICAgIHRocm93IG5ldyBFcnJvcignbWlzc2luZyB3YWxsZXQgcGFzc3BocmFzZScpO1xuICAgICAgICB9XG5cbiAgICAgICAgY29uc3QgdXNlcktleSA9IHBhcmFtcy51c2VyS2V5LnJlcGxhY2UoL1xccy9nLCAnJyk7XG4gICAgICAgIGNvbnN0IGJhY2t1cEtleSA9IHBhcmFtcy5iYWNrdXBLZXkucmVwbGFjZSgvXFxzL2csICcnKTtcblxuICAgICAgICAoeyB1c2VyS2V5U2hhcmUsIGJhY2t1cEtleVNoYXJlLCBjb21tb25LZXlDaGFpbiB9ID0gYXdhaXQgRUNEU0FVdGlscy5nZXRNcGNWMlJlY292ZXJ5S2V5U2hhcmVzKFxuICAgICAgICAgIHVzZXJLZXksXG4gICAgICAgICAgYmFja3VwS2V5LFxuICAgICAgICAgIHBhcmFtcy53YWxsZXRQYXNzcGhyYXNlXG4gICAgICAgICkpO1xuICAgICAgICBwdWJsaWNLZXkgPSBNUEMuZGVyaXZlVW5oYXJkZW5lZChjb21tb25LZXlDaGFpbiwgJ20vMCcpLnNsaWNlKDAsIDY2KTtcbiAgICAgIH1cblxuICAgICAgaWYgKCFwdWJsaWNLZXkpIHtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKCdmYWlsZWQgdG8gZGVyaXZlIHB1YmxpYyBrZXknKTtcbiAgICAgIH1cblxuICAgICAgY29uc3QgYmFja3VwS2V5UGFpciA9IG5ldyBFdGhLZXlQYWlyKHsgcHViOiBwdWJsaWNLZXkgfSk7XG4gICAgICBjb25zdCBiYXNlQWRkcmVzcyA9IGJhY2t1cEtleVBhaXIuZ2V0QWRkcmVzcygpO1xuXG4gICAgICBjb25zdCB0eCA9IGF3YWl0IHRoaXMuYnVpbGRUb2tlblJlY292ZXJ5VHJhbnNhY3Rpb24oe1xuICAgICAgICBiYXNlQWRkcmVzcyxcbiAgICAgICAgcGFyYW1zLFxuICAgICAgfSk7XG5cbiAgICAgIGNvbnN0IHNpZ25hYmxlSGV4ID0gYXdhaXQgdHguc2lnbmFibGVQYXlsb2FkO1xuICAgICAgY29uc3Qgc2VyaWFsaXplZFR4SGV4ID0gYXdhaXQgdHgudG9Ccm9hZGNhc3RGb3JtYXQoKTtcblxuICAgICAgaWYgKGlzVW5zaWduZWRTd2VlcCkge1xuICAgICAgICByZXR1cm4ge1xuICAgICAgICAgIHR4SGV4OiBzZXJpYWxpemVkVHhIZXgsXG4gICAgICAgICAgY29pbjogdGhpcy5nZXRDaGFpbigpLFxuICAgICAgICB9O1xuICAgICAgfVxuXG4gICAgICBjb25zdCBzaWduYWJsZU1lc3NhZ2UgPSB0aGlzLmdldEhhc2hGdW5jdGlvbigpLnVwZGF0ZShzaWduYWJsZUhleCkuZGlnZXN0KCk7XG5cbiAgICAgIGNvbnN0IHNpZ25hdHVyZU9iaiA9IGF3YWl0IEVDRFNBVXRpbHMuc2lnblJlY292ZXJ5TXBjVjIoXG4gICAgICAgIHNpZ25hYmxlTWVzc2FnZSxcbiAgICAgICAgdXNlcktleVNoYXJlLFxuICAgICAgICBiYWNrdXBLZXlTaGFyZSxcbiAgICAgICAgY29tbW9uS2V5Q2hhaW5cbiAgICAgICk7XG4gICAgICBjb25zdCBzaWduYXR1cmUgPSBCdWZmZXIuZnJvbShzaWduYXR1cmVPYmouciArIHNpZ25hdHVyZU9iai5zICsgKHNpZ25hdHVyZU9iai5yZWNpZCA9PT0gMCA/ICcwMCcgOiAnMDEnKSwgJ2hleCcpO1xuICAgICAgY29uc3QgdG9rZW5UcmFuc2FjdGlvbiA9IG5ldyBUb2tlblRyYW5zYWN0aW9uKGNvaW5zLmdldCh0aGlzLmdldENoYWluKCkpKTtcbiAgICAgIGNvbnN0IHR4QnVpbGRlciA9IHRoaXMuZ2V0VHhCdWlsZGVyRmFjdG9yeSgpLmdldFRva2VuVHJhbnNhY3Rpb25CdWlsZGVyKHRva2VuVHJhbnNhY3Rpb24pO1xuICAgICAgYXdhaXQgdHhCdWlsZGVyLmZyb20oc2VyaWFsaXplZFR4SGV4KTtcbiAgICAgIHR4QnVpbGRlci5pc1JlY292ZXJ5KHRydWUpO1xuICAgICAgYXdhaXQgdHhCdWlsZGVyLmFkZFNlbmRlclNpZ25hdHVyZShzaWduYXR1cmUpO1xuXG4gICAgICBjb25zdCBzaWduZWRUeCA9IGF3YWl0IHR4QnVpbGRlci5idWlsZCgpO1xuXG4gICAgICByZXR1cm4ge1xuICAgICAgICBpZDogc2lnbmVkVHguaWQsXG4gICAgICAgIHR4OiBzaWduZWRUeC50b0Jyb2FkY2FzdEZvcm1hdCgpLFxuICAgICAgfTtcbiAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKGBFcnJvciBkdXJpbmcgVmVjaGFpbiB0b2tlbiByZWNvdmVyeTogJHtlcnJvci5tZXNzYWdlIHx8IGVycm9yfWApO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgYXN5bmMgYnVpbGRUb2tlblJlY292ZXJ5VHJhbnNhY3Rpb24oYnVpbGRQYXJhbXM6IHtcbiAgICBiYXNlQWRkcmVzczogc3RyaW5nO1xuICAgIHBhcmFtczogUmVjb3Zlck9wdGlvbnM7XG4gIH0pOiBQcm9taXNlPFRyYW5zYWN0aW9uPiB7XG4gICAgY29uc3QgeyBiYXNlQWRkcmVzcywgcGFyYW1zIH0gPSBidWlsZFBhcmFtcztcbiAgICBjb25zdCB0b2tlbkNvbnRyYWN0QWRkcmVzcyA9IHBhcmFtcy50b2tlbkNvbnRyYWN0QWRkcmVzcztcbiAgICBhc3NlcnQodG9rZW5Db250cmFjdEFkZHJlc3MsICd0b2tlbkNvbnRyYWN0QWRkcmVzcyBpcyByZXF1aXJlZCBmb3IgdG9rZW4gcmVjb3ZlcnknKTtcblxuICAgIGNvbnN0IGJhbGFuY2UgPSBhd2FpdCB0aGlzLmdldEJhbGFuY2UoYmFzZUFkZHJlc3MsIHRva2VuQ29udHJhY3RBZGRyZXNzKTtcbiAgICAvL3JlcGxhY2Ugd2l0aCBnZXQgYmFsYW5jZSBmdW5jdGlvblxuXG4gICAgaWYgKGJhbGFuY2UuaXNMZXNzVGhhbk9yRXF1YWxUbygwKSkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgICBgbm8gdG9rZW4gYmFsYW5jZSB0byByZWNvdmVyIGZvciBhZGRyZXNzICR7YmFzZUFkZHJlc3N9IGNvbnRyYWN0IGFkZHJlc3MgJHt0b2tlbkNvbnRyYWN0QWRkcmVzc31gXG4gICAgICApO1xuICAgIH1cblxuICAgIC8vIGNyZWF0ZSB0aGUgcmVjaXBpZW50cyBoZXJlIHNvIHRoYXQgd2UgY2FuIGJ1aWxkIHRoZSBjbGF1c2VzIGZvciBnYXMgZXN0aW1hdGlvblxuICAgIGNvbnN0IHJvdWdoRmVlRXN0aW1hdGUgPSB0aGlzLmNhbGN1bGF0ZUZlZShmZWVFc3RpbWF0ZURhdGEsIG5ldyBCaWdOdW1iZXIoNTEzOTApKTtcbiAgICBsZXQgcmVjaXBpZW50cyA9IFtcbiAgICAgIHtcbiAgICAgICAgYWRkcmVzczogcGFyYW1zLnJlY292ZXJ5RGVzdGluYXRpb24sXG4gICAgICAgIGFtb3VudDogYmFsYW5jZS5taW51cyhyb3VnaEZlZUVzdGltYXRlKS50b1N0cmluZygpLFxuICAgICAgfSxcbiAgICBdO1xuXG4gICAgY29uc3QgYmxvY2tSZWYgPSBhd2FpdCB0aGlzLmdldEJsb2NrUmVmKCk7XG5cbiAgICBjb25zdCB0b2tlblRyYW5zYWN0aW9uID0gbmV3IFRva2VuVHJhbnNhY3Rpb24oY29pbnMuZ2V0KHRoaXMuZ2V0Q2hhaW4oKSkpO1xuICAgIGNvbnN0IHR4QnVpbGRlciA9IHRoaXMuZ2V0VHhCdWlsZGVyRmFjdG9yeSgpLmdldFRva2VuVHJhbnNhY3Rpb25CdWlsZGVyKHRva2VuVHJhbnNhY3Rpb24pO1xuXG4gICAgdHhCdWlsZGVyLnRva2VuQWRkcmVzcyh0b2tlbkNvbnRyYWN0QWRkcmVzcyk7XG4gICAgdHhCdWlsZGVyLmNoYWluVGFnKHRoaXMuYml0Z28uZ2V0RW52KCkgPT09ICdwcm9kJyA/IDB4NGEgOiAweDI3KTtcbiAgICB0eEJ1aWxkZXIucmVjaXBpZW50cyhyZWNpcGllbnRzKTtcbiAgICB0eEJ1aWxkZXIuc2VuZGVyKGJhc2VBZGRyZXNzKTtcbiAgICB0eEJ1aWxkZXIuYWRkRmVlUGF5ZXJBZGRyZXNzKGJhc2VBZGRyZXNzKTtcbiAgICB0eEJ1aWxkZXIuZ2FzKE51bWJlcihBVkdfR0FTX1VOSVRTKSk7XG4gICAgdHhCdWlsZGVyLmJsb2NrUmVmKGJsb2NrUmVmKTtcbiAgICB0eEJ1aWxkZXIuZXhwaXJhdGlvbihFWFBJUkFUSU9OKTtcbiAgICB0eEJ1aWxkZXIuZ2FzUHJpY2VDb2VmKE51bWJlcihHQVNfUFJJQ0VfQ09FRikpO1xuICAgIHR4QnVpbGRlci5ub25jZSh0aGlzLmdldFJhbmRvbU5vbmNlKCkpO1xuICAgIHR4QnVpbGRlci5pc1JlY292ZXJ5KHRydWUpO1xuXG4gICAgbGV0IHR4ID0gKGF3YWl0IHR4QnVpbGRlci5idWlsZCgpKSBhcyBUcmFuc2FjdGlvbjtcblxuICAgIGNvbnN0IGNsYXVzZXMgPSB0eC5jbGF1c2VzO1xuXG4gICAgY29uc3QgYWN0dWFsR2FzVW5pdHMgPSBhd2FpdCB0aGlzLmVzdGltYXRlR2FzKGNsYXVzZXMsIGJhc2VBZGRyZXNzKTtcblxuICAgIGF3YWl0IHRoaXMuZW5zdXJlVnRob0JhbGFuY2VGb3JGZWUoYmFzZUFkZHJlc3MsIGFjdHVhbEdhc1VuaXRzKTtcblxuICAgIGNvbnN0IHJlcXVpcmVkRmVlID0gdGhpcy5jYWxjdWxhdGVGZWUoZmVlRXN0aW1hdGVEYXRhLCBhY3R1YWxHYXNVbml0cyk7XG5cbiAgICAvLyBjcmVhdGUgdGhlIGZpbmFsIHJlY2lwaWVudHMgd2l0aCB0aGUgZmVlIGRlZHVjdGVkXG4gICAgcmVjaXBpZW50cyA9IFtcbiAgICAgIHtcbiAgICAgICAgYWRkcmVzczogcGFyYW1zLnJlY292ZXJ5RGVzdGluYXRpb24sXG4gICAgICAgIGFtb3VudDogYmFsYW5jZS5taW51cyhyZXF1aXJlZEZlZSkudG9TdHJpbmcoKSxcbiAgICAgIH0sXG4gICAgXTtcblxuICAgIHR4QnVpbGRlci5yZWNpcGllbnRzKHJlY2lwaWVudHMpO1xuICAgIHR4QnVpbGRlci5nYXMoYWN0dWFsR2FzVW5pdHMudG9OdW1iZXIoKSk7XG5cbiAgICB0eCA9IChhd2FpdCB0eEJ1aWxkZXIuYnVpbGQoKSkgYXMgVHJhbnNhY3Rpb247XG5cbiAgICByZXR1cm4gdHg7XG4gIH1cbn1cbiJdfQ==
|