@bitgo-beta/sdk-coin-ada 2.3.14-beta.182 → 2.3.14-beta.1821
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/ada.d.ts +15 -13
- package/dist/src/ada.d.ts.map +1 -1
- package/dist/src/ada.js +124 -42
- package/dist/src/adaToken.d.ts +16 -11
- package/dist/src/adaToken.d.ts.map +1 -1
- package/dist/src/adaToken.js +109 -3
- package/dist/src/index.js +6 -2
- package/dist/src/lib/index.d.ts +3 -1
- package/dist/src/lib/index.d.ts.map +1 -1
- package/dist/src/lib/index.js +29 -9
- package/dist/src/lib/keyPair.js +29 -16
- package/dist/src/lib/messages/cip8/cip8Message.d.ts +25 -0
- package/dist/src/lib/messages/cip8/cip8Message.d.ts.map +1 -0
- package/dist/src/lib/messages/cip8/cip8Message.js +140 -0
- package/dist/src/lib/messages/cip8/cip8MessageBuilder.d.ts +19 -0
- package/dist/src/lib/messages/cip8/cip8MessageBuilder.d.ts.map +1 -0
- package/dist/src/lib/messages/cip8/cip8MessageBuilder.js +27 -0
- package/dist/src/lib/messages/cip8/index.d.ts +4 -0
- package/dist/src/lib/messages/cip8/index.d.ts.map +1 -0
- package/dist/src/lib/messages/cip8/index.js +20 -0
- package/dist/src/lib/messages/cip8/utils.d.ts +24 -0
- package/dist/src/lib/messages/cip8/utils.d.ts.map +1 -0
- package/dist/src/lib/messages/cip8/utils.js +73 -0
- package/dist/src/lib/messages/index.d.ts +3 -0
- package/dist/src/lib/messages/index.d.ts.map +1 -0
- package/dist/src/lib/messages/index.js +19 -0
- package/dist/src/lib/messages/messageBuilderFactory.d.ts +7 -0
- package/dist/src/lib/messages/messageBuilderFactory.d.ts.map +1 -0
- package/dist/src/lib/messages/messageBuilderFactory.js +20 -0
- package/dist/src/lib/stakingActivateBuilder.d.ts +1 -1
- package/dist/src/lib/stakingActivateBuilder.d.ts.map +1 -1
- package/dist/src/lib/stakingActivateBuilder.js +30 -10
- package/dist/src/lib/stakingDeactivateBuilder.js +24 -10
- package/dist/src/lib/transaction.d.ts +24 -7
- package/dist/src/lib/transaction.d.ts.map +1 -1
- package/dist/src/lib/transaction.js +81 -17
- package/dist/src/lib/transactionBuilder.d.ts +114 -3
- package/dist/src/lib/transactionBuilder.d.ts.map +1 -1
- package/dist/src/lib/transactionBuilder.js +464 -20
- package/dist/src/lib/transactionBuilderFactory.d.ts +2 -0
- package/dist/src/lib/transactionBuilderFactory.d.ts.map +1 -1
- package/dist/src/lib/transactionBuilderFactory.js +7 -1
- package/dist/src/lib/utils.d.ts +22 -0
- package/dist/src/lib/utils.d.ts.map +1 -1
- package/dist/src/lib/utils.js +152 -17
- package/dist/src/lib/voteDelegationBuilder.d.ts +24 -0
- package/dist/src/lib/voteDelegationBuilder.d.ts.map +1 -0
- package/dist/src/lib/voteDelegationBuilder.js +84 -0
- package/dist/test/resources/cip8Resources.d.ts +25 -0
- package/dist/test/resources/cip8Resources.d.ts.map +1 -0
- package/dist/test/resources/cip8Resources.js +76 -0
- package/dist/test/resources/index.d.ts +458 -0
- package/dist/test/resources/index.d.ts.map +1 -0
- package/dist/test/resources/index.js +659 -0
- package/dist/test/unit/StakingActivateBuilder.d.ts +2 -0
- package/dist/test/unit/StakingActivateBuilder.d.ts.map +1 -0
- package/dist/test/unit/StakingActivateBuilder.js +213 -0
- package/dist/test/unit/StakingDeactivateBuilder.d.ts +2 -0
- package/dist/test/unit/StakingDeactivateBuilder.d.ts.map +1 -0
- package/dist/test/unit/StakingDeactivateBuilder.js +158 -0
- package/dist/test/unit/ada.d.ts +5 -0
- package/dist/test/unit/ada.d.ts.map +1 -0
- package/dist/test/unit/ada.js +1430 -0
- package/dist/test/unit/getBuilderFactory.d.ts +3 -0
- package/dist/test/unit/getBuilderFactory.d.ts.map +1 -0
- package/dist/test/unit/getBuilderFactory.js +10 -0
- package/dist/test/unit/index.d.ts +2 -0
- package/dist/test/unit/index.d.ts.map +1 -0
- package/dist/test/unit/index.js +32 -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 +88 -0
- package/dist/test/unit/messages/cip8/cip8Message.d.ts +2 -0
- package/dist/test/unit/messages/cip8/cip8Message.d.ts.map +1 -0
- package/dist/test/unit/messages/cip8/cip8Message.js +131 -0
- package/dist/test/unit/messages/cip8/cip8MessageBuilder.d.ts +2 -0
- package/dist/test/unit/messages/cip8/cip8MessageBuilder.d.ts.map +1 -0
- package/dist/test/unit/messages/cip8/cip8MessageBuilder.js +132 -0
- package/dist/test/unit/messages/cip8/utils.d.ts +2 -0
- package/dist/test/unit/messages/cip8/utils.d.ts.map +1 -0
- package/dist/test/unit/messages/cip8/utils.js +127 -0
- package/dist/test/unit/messages/messageBuilderFactory.d.ts +2 -0
- package/dist/test/unit/messages/messageBuilderFactory.d.ts.map +1 -0
- package/dist/test/unit/messages/messageBuilderFactory.js +35 -0
- package/dist/test/unit/stakingClaimRewardsBuilder.d.ts +2 -0
- package/dist/test/unit/stakingClaimRewardsBuilder.d.ts.map +1 -0
- package/dist/test/unit/stakingClaimRewardsBuilder.js +70 -0
- package/dist/test/unit/stakingPledgeBuilder.d.ts +2 -0
- package/dist/test/unit/stakingPledgeBuilder.d.ts.map +1 -0
- package/dist/test/unit/stakingPledgeBuilder.js +141 -0
- package/dist/test/unit/stakingWithdrawBuilder.d.ts +2 -0
- package/dist/test/unit/stakingWithdrawBuilder.d.ts.map +1 -0
- package/dist/test/unit/stakingWithdrawBuilder.js +167 -0
- package/dist/test/unit/tokenWithdrawal.d.ts +2 -0
- package/dist/test/unit/tokenWithdrawal.d.ts.map +1 -0
- package/dist/test/unit/tokenWithdrawal.js +959 -0
- package/dist/test/unit/transaction.d.ts +2 -0
- package/dist/test/unit/transaction.d.ts.map +1 -0
- package/dist/test/unit/transaction.js +100 -0
- package/dist/test/unit/transactionBuilder.d.ts +2 -0
- package/dist/test/unit/transactionBuilder.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder.js +545 -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 +138 -0
- package/dist/test/unit/voteDelegationBuilder.d.ts +2 -0
- package/dist/test/unit/voteDelegationBuilder.d.ts.map +1 -0
- package/dist/test/unit/voteDelegationBuilder.js +96 -0
- package/dist/tsconfig.tsbuildinfo +1 -0
- package/package.json +18 -13
- package/.eslintignore +0 -5
- package/.mocharc.yml +0 -8
- package/CHANGELOG.md +0 -450
|
@@ -0,0 +1,959 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
35
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
36
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
37
|
+
};
|
|
38
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
39
|
+
const should_1 = __importDefault(require("should"));
|
|
40
|
+
const sdk_core_1 = require("@bitgo-beta/sdk-core");
|
|
41
|
+
const testData = __importStar(require("../resources"));
|
|
42
|
+
const src_1 = require("../../src");
|
|
43
|
+
const statics_1 = require("@bitgo-beta/statics");
|
|
44
|
+
const CardanoWasm = __importStar(require("@emurgo/cardano-serialization-lib-nodejs"));
|
|
45
|
+
const sdk_test_1 = require("@bitgo-beta/sdk-test");
|
|
46
|
+
const sdk_api_1 = require("@bitgo-beta/sdk-api");
|
|
47
|
+
describe('ADA Token Operations', async () => {
|
|
48
|
+
const factory = new src_1.TransactionBuilderFactory(statics_1.coins.get('tada'));
|
|
49
|
+
const receiverAddress = testData.rawTx.outputAddress2.address;
|
|
50
|
+
const senderAddress = testData.rawTx.outputAddress1.address;
|
|
51
|
+
const policyId = 'e16c2dc8ae937e8d3790c7fd7168d7b994621ba14ca11415f39fed72';
|
|
52
|
+
const policyScriptHash = CardanoWasm.ScriptHash.from_hex(policyId);
|
|
53
|
+
const assetName = 'tada:water';
|
|
54
|
+
const name = 'WATER';
|
|
55
|
+
const asciiEncodedName = Buffer.from(name, 'ascii').toString('hex');
|
|
56
|
+
const fingerprint = 'asset1n69xf60d0760xvn8v2ffd5frvsm0cl2r8hfjf6';
|
|
57
|
+
const unsupportedPolicyId = '279c909f348e533da5808898f87f9a14bb2c3dfbbacccd631d927a3f';
|
|
58
|
+
const unsupportedPolicyScriptHash = CardanoWasm.ScriptHash.from_hex(unsupportedPolicyId);
|
|
59
|
+
const unsupportedName = 'BITGO';
|
|
60
|
+
const unsupportedAsciiEncodedName = Buffer.from(unsupportedName, 'ascii').toString('hex');
|
|
61
|
+
const unsupportedFingerprint = 'asset1m8h0gk7f6x5j5qg0x5m4q5f5j5qg0x5m4q5f5j';
|
|
62
|
+
it(`should build a transaction with token ${assetName} - ada + ${assetName}`, async () => {
|
|
63
|
+
const quantity = '20';
|
|
64
|
+
const totalInput = 20000000;
|
|
65
|
+
const totalAssetList = {
|
|
66
|
+
[fingerprint]: {
|
|
67
|
+
quantity: '100',
|
|
68
|
+
policy_id: policyId,
|
|
69
|
+
asset_name: asciiEncodedName,
|
|
70
|
+
},
|
|
71
|
+
};
|
|
72
|
+
const expectedChangeAda = (totalInput -
|
|
73
|
+
1500000 /* min ada for change token utxo */ -
|
|
74
|
+
1500000 /* min ada for recipient token utxo*/ -
|
|
75
|
+
173597) /* fee */
|
|
76
|
+
.toString();
|
|
77
|
+
const expectedChangeToken = '80';
|
|
78
|
+
const txBuilder = factory.getTransferBuilder();
|
|
79
|
+
txBuilder.input({
|
|
80
|
+
transaction_id: '3677e75c7ba699bfdc6cd57d42f246f86f63aefd76025006ac78313fad2bba21',
|
|
81
|
+
transaction_index: 1,
|
|
82
|
+
});
|
|
83
|
+
txBuilder.output({
|
|
84
|
+
address: receiverAddress,
|
|
85
|
+
amount: '0', // Set ADA amount to 0 for token transfer (min ADA is handled in sdk build)
|
|
86
|
+
multiAssets: {
|
|
87
|
+
asset_name: asciiEncodedName,
|
|
88
|
+
policy_id: policyId,
|
|
89
|
+
quantity,
|
|
90
|
+
fingerprint,
|
|
91
|
+
},
|
|
92
|
+
});
|
|
93
|
+
txBuilder.changeAddress(senderAddress, totalInput.toString(), totalAssetList);
|
|
94
|
+
txBuilder.ttl(800000000);
|
|
95
|
+
txBuilder.isTokenTransaction();
|
|
96
|
+
const tx = (await txBuilder.build());
|
|
97
|
+
should_1.default.equal(tx.type, sdk_core_1.TransactionType.Send);
|
|
98
|
+
const txData = tx.toJson();
|
|
99
|
+
// Check outputs (should include multi-asset and regular change)
|
|
100
|
+
txData.outputs.length.should.equal(3);
|
|
101
|
+
txData.inputs.length.should.equal(1);
|
|
102
|
+
// One output should have the multi-asset change and the other for the transfer
|
|
103
|
+
const assetOutput = txData.outputs.filter((output) => output.multiAssets !== undefined);
|
|
104
|
+
assetOutput.length.should.equal(2);
|
|
105
|
+
// Verify inputs and outputs
|
|
106
|
+
const input = txData.inputs[0];
|
|
107
|
+
input.transaction_id.should.equal('3677e75c7ba699bfdc6cd57d42f246f86f63aefd76025006ac78313fad2bba21');
|
|
108
|
+
input.transaction_index.should.equal(1);
|
|
109
|
+
// Validate receiver output
|
|
110
|
+
const receiverOutput = txData.outputs.filter((output) => output.address === receiverAddress);
|
|
111
|
+
receiverOutput.length.should.equal(1);
|
|
112
|
+
receiverOutput[0].amount.should.equal('1500000'); // Minimum ADA for asset output
|
|
113
|
+
receiverOutput[0].multiAssets
|
|
114
|
+
.get_asset(policyScriptHash, CardanoWasm.AssetName.new(Buffer.from(asciiEncodedName, 'hex')))
|
|
115
|
+
.to_str()
|
|
116
|
+
.should.equal(quantity);
|
|
117
|
+
// Validate change outputs (one with asset and one without)
|
|
118
|
+
const changeOutput = txData.outputs.filter((output) => output.address === senderAddress);
|
|
119
|
+
changeOutput.length.should.equal(2);
|
|
120
|
+
const changeWithAsset = changeOutput.find((output) => output.multiAssets !== undefined);
|
|
121
|
+
should_1.default.exist(changeWithAsset);
|
|
122
|
+
changeWithAsset.amount.should.equal('1500000'); // Minimum ADA for asset output
|
|
123
|
+
changeWithAsset.multiAssets
|
|
124
|
+
.get_asset(policyScriptHash, CardanoWasm.AssetName.new(Buffer.from(asciiEncodedName, 'hex')))
|
|
125
|
+
.to_str()
|
|
126
|
+
.should.equal(expectedChangeToken);
|
|
127
|
+
const changeWithoutAsset = changeOutput.find((output) => output.multiAssets === undefined);
|
|
128
|
+
should_1.default.exist(changeWithoutAsset);
|
|
129
|
+
changeWithoutAsset.amount.should.equal(expectedChangeAda); // Remaining ADA after fees and min ADAs
|
|
130
|
+
});
|
|
131
|
+
it(`should build a transaction with token ${assetName} - ada + ${assetName} + unsupported token`, async () => {
|
|
132
|
+
const quantity = '20';
|
|
133
|
+
const totalInput = 20000000;
|
|
134
|
+
const totalAssetList = {
|
|
135
|
+
[fingerprint]: {
|
|
136
|
+
quantity: '100',
|
|
137
|
+
policy_id: policyId,
|
|
138
|
+
asset_name: asciiEncodedName,
|
|
139
|
+
},
|
|
140
|
+
[unsupportedFingerprint]: {
|
|
141
|
+
quantity: '10000',
|
|
142
|
+
policy_id: unsupportedPolicyId,
|
|
143
|
+
asset_name: unsupportedAsciiEncodedName,
|
|
144
|
+
},
|
|
145
|
+
};
|
|
146
|
+
const expectedChangeAda = (totalInput -
|
|
147
|
+
1500000 /* min ada for change token utxo */ -
|
|
148
|
+
1500000 /* min ada for recipient token utxo*/ -
|
|
149
|
+
1500000 /* min ada for unsupported token change utxo */ -
|
|
150
|
+
179889) /* fee */
|
|
151
|
+
.toString();
|
|
152
|
+
const expectedChangeToken = '80';
|
|
153
|
+
const txBuilder = factory.getTransferBuilder();
|
|
154
|
+
txBuilder.input({
|
|
155
|
+
transaction_id: '3677e75c7ba699bfdc6cd57d42f246f86f63aefd76025006ac78313fad2bba21',
|
|
156
|
+
transaction_index: 1,
|
|
157
|
+
});
|
|
158
|
+
txBuilder.input({
|
|
159
|
+
transaction_id: '3677e75c7ba699bfdc6cd57d42f246f86f63aefd76025006ac78313fad2bba22',
|
|
160
|
+
transaction_index: 1,
|
|
161
|
+
});
|
|
162
|
+
txBuilder.output({
|
|
163
|
+
address: receiverAddress,
|
|
164
|
+
amount: '0', // Set ADA amount to 0 for token transfer (min ADA is handled in sdk build)
|
|
165
|
+
multiAssets: {
|
|
166
|
+
asset_name: asciiEncodedName,
|
|
167
|
+
policy_id: policyId,
|
|
168
|
+
quantity,
|
|
169
|
+
fingerprint,
|
|
170
|
+
},
|
|
171
|
+
});
|
|
172
|
+
txBuilder.changeAddress(senderAddress, totalInput.toString(), totalAssetList);
|
|
173
|
+
txBuilder.ttl(800000000);
|
|
174
|
+
txBuilder.isTokenTransaction();
|
|
175
|
+
const tx = (await txBuilder.build());
|
|
176
|
+
should_1.default.equal(tx.type, sdk_core_1.TransactionType.Send);
|
|
177
|
+
const txData = tx.toJson();
|
|
178
|
+
// Check outputs (should include multi-asset and regular change)
|
|
179
|
+
txData.outputs.length.should.equal(4);
|
|
180
|
+
txData.inputs.length.should.equal(2);
|
|
181
|
+
// One output should have the multi-asset change and the other for the transfer
|
|
182
|
+
const assetOutput = txData.outputs.filter((output) => output.multiAssets !== undefined);
|
|
183
|
+
assetOutput.length.should.equal(3);
|
|
184
|
+
// Verify inputs and outputs
|
|
185
|
+
txData.inputs[0].transaction_id.should.equal('3677e75c7ba699bfdc6cd57d42f246f86f63aefd76025006ac78313fad2bba21');
|
|
186
|
+
txData.inputs[0].transaction_index.should.equal(1);
|
|
187
|
+
txData.inputs[1].transaction_id.should.equal('3677e75c7ba699bfdc6cd57d42f246f86f63aefd76025006ac78313fad2bba22');
|
|
188
|
+
txData.inputs[1].transaction_index.should.equal(1);
|
|
189
|
+
// Validate receiver output
|
|
190
|
+
const receiverOutput = txData.outputs.filter((output) => output.address === receiverAddress);
|
|
191
|
+
receiverOutput.length.should.equal(1);
|
|
192
|
+
receiverOutput[0].amount.should.equal('1500000'); // Minimum ADA for asset output
|
|
193
|
+
receiverOutput[0].multiAssets
|
|
194
|
+
.get_asset(policyScriptHash, CardanoWasm.AssetName.new(Buffer.from(asciiEncodedName, 'hex')))
|
|
195
|
+
.to_str()
|
|
196
|
+
.should.equal(quantity);
|
|
197
|
+
// Validate change outputs (one with supported token, one with unsupported token and one for pure ada)
|
|
198
|
+
const changeOutput = txData.outputs.filter((output) => output.address === senderAddress);
|
|
199
|
+
changeOutput.length.should.equal(3);
|
|
200
|
+
const changeWithAsset = changeOutput.filter((output) => output.multiAssets !== undefined);
|
|
201
|
+
changeWithAsset.length.should.equal(2);
|
|
202
|
+
let tokens = 0;
|
|
203
|
+
changeWithAsset.forEach((output) => {
|
|
204
|
+
output.amount.should.equal('1500000'); // Minimum ADA for asset output
|
|
205
|
+
const supportedTokenChangeQty = output.multiAssets
|
|
206
|
+
.get_asset(policyScriptHash, CardanoWasm.AssetName.new(Buffer.from(asciiEncodedName, 'hex')))
|
|
207
|
+
.to_str();
|
|
208
|
+
const unsupportedTokenChangeQty = output.multiAssets
|
|
209
|
+
.get_asset(unsupportedPolicyScriptHash, CardanoWasm.AssetName.new(Buffer.from(unsupportedAsciiEncodedName, 'hex')))
|
|
210
|
+
.to_str();
|
|
211
|
+
if (supportedTokenChangeQty !== '0') {
|
|
212
|
+
supportedTokenChangeQty.should.equal(expectedChangeToken);
|
|
213
|
+
tokens++;
|
|
214
|
+
}
|
|
215
|
+
else {
|
|
216
|
+
unsupportedTokenChangeQty.should.equal('10000');
|
|
217
|
+
tokens++;
|
|
218
|
+
}
|
|
219
|
+
});
|
|
220
|
+
tokens.should.equal(2);
|
|
221
|
+
const changeWithoutAsset = changeOutput.find((output) => output.multiAssets === undefined);
|
|
222
|
+
should_1.default.exist(changeWithoutAsset);
|
|
223
|
+
changeWithoutAsset.amount.should.equal(expectedChangeAda); // Remaining ADA after fees and min ADAs
|
|
224
|
+
});
|
|
225
|
+
it(`should fail to build a transaction with ${assetName} token and insufficient minimum ADA`, async () => {
|
|
226
|
+
const quantity = '20';
|
|
227
|
+
const totalInput = 2000000;
|
|
228
|
+
const totalAssetList = {
|
|
229
|
+
[fingerprint]: {
|
|
230
|
+
quantity: '100',
|
|
231
|
+
policy_id: policyId,
|
|
232
|
+
asset_name: asciiEncodedName,
|
|
233
|
+
},
|
|
234
|
+
};
|
|
235
|
+
const txBuilder = factory.getTransferBuilder();
|
|
236
|
+
txBuilder.input({
|
|
237
|
+
transaction_id: '3677e75c7ba699bfdc6cd57d42f246f86f63aefd76025006ac78313fad2bba21',
|
|
238
|
+
transaction_index: 1,
|
|
239
|
+
});
|
|
240
|
+
txBuilder.output({
|
|
241
|
+
address: receiverAddress,
|
|
242
|
+
amount: '0', // Set ADA amount to 0 for token transfer (min ADA is handled in sdk build)
|
|
243
|
+
multiAssets: {
|
|
244
|
+
asset_name: asciiEncodedName,
|
|
245
|
+
policy_id: policyId,
|
|
246
|
+
quantity,
|
|
247
|
+
fingerprint,
|
|
248
|
+
},
|
|
249
|
+
});
|
|
250
|
+
txBuilder.changeAddress(senderAddress, totalInput.toString(), totalAssetList);
|
|
251
|
+
txBuilder.ttl(800000000);
|
|
252
|
+
txBuilder.isTokenTransaction();
|
|
253
|
+
await txBuilder
|
|
254
|
+
.build()
|
|
255
|
+
.should.rejectedWith('Insufficient funds: need a minimum of 1500000 lovelace per output to construct token transactions');
|
|
256
|
+
});
|
|
257
|
+
it(`should fail to build a transaction with ${assetName} and insufficient token qty`, async () => {
|
|
258
|
+
const quantity = '20';
|
|
259
|
+
const totalInput = 20000000;
|
|
260
|
+
const totalAssetList = {
|
|
261
|
+
[fingerprint]: {
|
|
262
|
+
quantity: '5',
|
|
263
|
+
policy_id: policyId,
|
|
264
|
+
asset_name: asciiEncodedName,
|
|
265
|
+
},
|
|
266
|
+
};
|
|
267
|
+
const txBuilder = factory.getTransferBuilder();
|
|
268
|
+
txBuilder.input({
|
|
269
|
+
transaction_id: '3677e75c7ba699bfdc6cd57d42f246f86f63aefd76025006ac78313fad2bba21',
|
|
270
|
+
transaction_index: 1,
|
|
271
|
+
});
|
|
272
|
+
txBuilder.output({
|
|
273
|
+
address: receiverAddress,
|
|
274
|
+
amount: '0', // Set ADA amount to 0 for token transfer (min ADA is handled in sdk build)
|
|
275
|
+
multiAssets: {
|
|
276
|
+
asset_name: asciiEncodedName,
|
|
277
|
+
policy_id: policyId,
|
|
278
|
+
quantity,
|
|
279
|
+
fingerprint,
|
|
280
|
+
},
|
|
281
|
+
});
|
|
282
|
+
txBuilder.changeAddress(senderAddress, totalInput.toString(), totalAssetList);
|
|
283
|
+
txBuilder.ttl(800000000);
|
|
284
|
+
txBuilder.isTokenTransaction();
|
|
285
|
+
await txBuilder.build().should.rejectedWith('Insufficient qty: not enough token qty to cover receiver output');
|
|
286
|
+
});
|
|
287
|
+
it(`should build a transaction with ${assetName} when the token qty is exactly the qty in withdrawal`, async () => {
|
|
288
|
+
const quantity = '1';
|
|
289
|
+
const totalInput = 20000000;
|
|
290
|
+
const totalAssetList = {
|
|
291
|
+
[fingerprint]: {
|
|
292
|
+
quantity: '1',
|
|
293
|
+
policy_id: policyId,
|
|
294
|
+
asset_name: asciiEncodedName,
|
|
295
|
+
},
|
|
296
|
+
};
|
|
297
|
+
const txBuilder = factory.getTransferBuilder();
|
|
298
|
+
txBuilder.input({
|
|
299
|
+
transaction_id: '3677e75c7ba699bfdc6cd57d42f246f86f63aefd76025006ac78313fad2bba21',
|
|
300
|
+
transaction_index: 1,
|
|
301
|
+
});
|
|
302
|
+
txBuilder.output({
|
|
303
|
+
address: receiverAddress,
|
|
304
|
+
amount: '0', // Set ADA amount to 0 for token transfer (min ADA is handled in sdk build)
|
|
305
|
+
multiAssets: {
|
|
306
|
+
asset_name: asciiEncodedName,
|
|
307
|
+
policy_id: policyId,
|
|
308
|
+
quantity,
|
|
309
|
+
fingerprint,
|
|
310
|
+
},
|
|
311
|
+
});
|
|
312
|
+
txBuilder.changeAddress(senderAddress, totalInput.toString(), totalAssetList);
|
|
313
|
+
txBuilder.ttl(800000000);
|
|
314
|
+
txBuilder.isTokenTransaction();
|
|
315
|
+
await txBuilder.build().should.not.be.rejected();
|
|
316
|
+
});
|
|
317
|
+
it(`should build a sponsored token transaction where fee address sponsors min ADA for receiver`, async () => {
|
|
318
|
+
const feeAddress = 'addr_test1qz2fxv2umyhttkxyxp8x0dlpdt3k6cwng5pxj3jhsydzer3jcu5d8ps7zex2k2xt3uqxgjqnnj83ws8lhrn648jjxtwq2ytjqp';
|
|
319
|
+
const quantity = '20';
|
|
320
|
+
const senderInputBalance = 5000000;
|
|
321
|
+
const feeAddressInputBalance = 20000000; // Fee address has enough ADA to sponsor
|
|
322
|
+
const totalAssetList = {
|
|
323
|
+
[fingerprint]: {
|
|
324
|
+
quantity: '100',
|
|
325
|
+
policy_id: policyId,
|
|
326
|
+
asset_name: asciiEncodedName,
|
|
327
|
+
},
|
|
328
|
+
};
|
|
329
|
+
const txBuilder = factory.getTransferBuilder();
|
|
330
|
+
// Sender input (has tokens)
|
|
331
|
+
txBuilder.input({
|
|
332
|
+
transaction_id: '3677e75c7ba699bfdc6cd57d42f246f86f63aefd76025006ac78313fad2bba21',
|
|
333
|
+
transaction_index: 1,
|
|
334
|
+
});
|
|
335
|
+
// Fee address input (sponsors fees and min ADA)
|
|
336
|
+
txBuilder.input({
|
|
337
|
+
transaction_id: '3677e75c7ba699bfdc6cd57d42f246f86f63aefd76025006ac78313fad2bba22',
|
|
338
|
+
transaction_index: 0,
|
|
339
|
+
});
|
|
340
|
+
txBuilder.output({
|
|
341
|
+
address: receiverAddress,
|
|
342
|
+
amount: '0', // Set ADA amount to 0 for token transfer (min ADA is handled by fee address)
|
|
343
|
+
multiAssets: {
|
|
344
|
+
asset_name: asciiEncodedName,
|
|
345
|
+
policy_id: policyId,
|
|
346
|
+
quantity,
|
|
347
|
+
fingerprint,
|
|
348
|
+
},
|
|
349
|
+
});
|
|
350
|
+
txBuilder.changeAddress(senderAddress, senderInputBalance.toString(), totalAssetList);
|
|
351
|
+
txBuilder.sponsorshipInfo({
|
|
352
|
+
feeAddress: feeAddress,
|
|
353
|
+
feeAddressInputBalance: feeAddressInputBalance.toString(),
|
|
354
|
+
});
|
|
355
|
+
txBuilder.ttl(800000000);
|
|
356
|
+
txBuilder.isTokenTransaction();
|
|
357
|
+
const tx = (await txBuilder.build());
|
|
358
|
+
should_1.default.equal(tx.type, sdk_core_1.TransactionType.Send);
|
|
359
|
+
const txData = tx.toJson();
|
|
360
|
+
txData.inputs.length.should.equal(2);
|
|
361
|
+
txData.outputs.length.should.equal(4);
|
|
362
|
+
// Validate receiver output - min ADA should be sponsored by fee address
|
|
363
|
+
const receiverOutput = txData.outputs.filter((output) => output.address === receiverAddress);
|
|
364
|
+
receiverOutput.length.should.equal(1);
|
|
365
|
+
receiverOutput[0].amount.should.equal('1500000'); // Minimum ADA for asset output (sponsored by fee address)
|
|
366
|
+
receiverOutput[0].multiAssets
|
|
367
|
+
.get_asset(policyScriptHash, CardanoWasm.AssetName.new(Buffer.from(asciiEncodedName, 'hex')))
|
|
368
|
+
.to_str()
|
|
369
|
+
.should.equal(quantity);
|
|
370
|
+
// Validate sender change output - should have full sender balance (tokens only, no ADA deduction for receiver)
|
|
371
|
+
const senderChangeOutput = txData.outputs.filter((output) => output.address === senderAddress);
|
|
372
|
+
senderChangeOutput.length.should.be.equal(2);
|
|
373
|
+
// Validate fee address change output - should have remaining ADA after fees and min ADA for receiver
|
|
374
|
+
const feeAddressChangeOutput = txData.outputs.filter((output) => output.address === feeAddress);
|
|
375
|
+
feeAddressChangeOutput.length.should.equal(1);
|
|
376
|
+
// Fee address change should not have any tokens
|
|
377
|
+
should_1.default.not.exist(feeAddressChangeOutput[0].multiAssets);
|
|
378
|
+
tx.getFee.should.equal('182485'); // Fee with two witnesses
|
|
379
|
+
});
|
|
380
|
+
it(`should rebuild a sponsored transaction from hex with isRebuild flag`, async () => {
|
|
381
|
+
const feeAddress = 'addr_test1qz2fxv2umyhttkxyxp8x0dlpdt3k6cwng5pxj3jhsydzer3jcu5d8ps7zex2k2xt3uqxgjqnnj83ws8lhrn648jjxtwq2ytjqp';
|
|
382
|
+
const quantity = '20';
|
|
383
|
+
const senderInputBalance = 5000000;
|
|
384
|
+
const feeAddressInputBalance = 20000000;
|
|
385
|
+
const totalAssetList = {
|
|
386
|
+
[fingerprint]: {
|
|
387
|
+
quantity: '100',
|
|
388
|
+
policy_id: policyId,
|
|
389
|
+
asset_name: asciiEncodedName,
|
|
390
|
+
},
|
|
391
|
+
};
|
|
392
|
+
// Step 1: Build the initial sponsored transaction
|
|
393
|
+
const txBuilder = factory.getTransferBuilder();
|
|
394
|
+
txBuilder.input({
|
|
395
|
+
transaction_id: '3677e75c7ba699bfdc6cd57d42f246f86f63aefd76025006ac78313fad2bba21',
|
|
396
|
+
transaction_index: 1,
|
|
397
|
+
});
|
|
398
|
+
txBuilder.input({
|
|
399
|
+
transaction_id: '3677e75c7ba699bfdc6cd57d42f246f86f63aefd76025006ac78313fad2bba22',
|
|
400
|
+
transaction_index: 0,
|
|
401
|
+
});
|
|
402
|
+
txBuilder.output({
|
|
403
|
+
address: receiverAddress,
|
|
404
|
+
amount: '0',
|
|
405
|
+
multiAssets: {
|
|
406
|
+
asset_name: asciiEncodedName,
|
|
407
|
+
policy_id: policyId,
|
|
408
|
+
quantity,
|
|
409
|
+
fingerprint,
|
|
410
|
+
},
|
|
411
|
+
});
|
|
412
|
+
txBuilder.changeAddress(senderAddress, senderInputBalance.toString(), totalAssetList);
|
|
413
|
+
txBuilder.sponsorshipInfo({
|
|
414
|
+
feeAddress: feeAddress,
|
|
415
|
+
feeAddressInputBalance: feeAddressInputBalance.toString(),
|
|
416
|
+
});
|
|
417
|
+
txBuilder.ttl(800000000);
|
|
418
|
+
txBuilder.isTokenTransaction();
|
|
419
|
+
const initialTx = (await txBuilder.build());
|
|
420
|
+
const initialFee = initialTx.getFee;
|
|
421
|
+
const initialTxData = initialTx.toJson();
|
|
422
|
+
// Step 2: Rebuild with isRebuild = true
|
|
423
|
+
// This simulates rebuilding from scratch but with isRebuild flag to add sponsor witness
|
|
424
|
+
const rebuildTxBuilder = factory.getTransferBuilder();
|
|
425
|
+
rebuildTxBuilder.input({
|
|
426
|
+
transaction_id: '3677e75c7ba699bfdc6cd57d42f246f86f63aefd76025006ac78313fad2bba21',
|
|
427
|
+
transaction_index: 1,
|
|
428
|
+
});
|
|
429
|
+
rebuildTxBuilder.input({
|
|
430
|
+
transaction_id: '3677e75c7ba699bfdc6cd57d42f246f86f63aefd76025006ac78313fad2bba22',
|
|
431
|
+
transaction_index: 0,
|
|
432
|
+
});
|
|
433
|
+
rebuildTxBuilder.output({
|
|
434
|
+
address: receiverAddress,
|
|
435
|
+
amount: '0',
|
|
436
|
+
multiAssets: {
|
|
437
|
+
asset_name: asciiEncodedName,
|
|
438
|
+
policy_id: policyId,
|
|
439
|
+
quantity,
|
|
440
|
+
fingerprint,
|
|
441
|
+
},
|
|
442
|
+
});
|
|
443
|
+
rebuildTxBuilder.changeAddress(senderAddress, senderInputBalance.toString(), totalAssetList);
|
|
444
|
+
rebuildTxBuilder.sponsorshipInfo({
|
|
445
|
+
feeAddress: feeAddress,
|
|
446
|
+
feeAddressInputBalance: feeAddressInputBalance.toString(),
|
|
447
|
+
isRebuild: true,
|
|
448
|
+
});
|
|
449
|
+
rebuildTxBuilder.ttl(800000000);
|
|
450
|
+
rebuildTxBuilder.isTokenTransaction();
|
|
451
|
+
const rebuiltTx = (await rebuildTxBuilder.build());
|
|
452
|
+
const rebuiltTxData = rebuiltTx.toJson();
|
|
453
|
+
// Verify the rebuilt transaction preserves the same structure
|
|
454
|
+
rebuiltTxData.inputs.length.should.equal(initialTxData.inputs.length);
|
|
455
|
+
rebuiltTxData.outputs.length.should.equal(initialTxData.outputs.length);
|
|
456
|
+
// Fee should be preserved from the original transaction
|
|
457
|
+
rebuiltTx.getFee.should.equal(initialFee);
|
|
458
|
+
// Verify receiver output is preserved
|
|
459
|
+
const receiverOutput = rebuiltTxData.outputs.filter((output) => output.address === receiverAddress);
|
|
460
|
+
receiverOutput.length.should.equal(1);
|
|
461
|
+
receiverOutput[0].amount.should.equal('1500000');
|
|
462
|
+
});
|
|
463
|
+
describe('AdaToken verifyTransaction', () => {
|
|
464
|
+
let bitgo;
|
|
465
|
+
let adaToken;
|
|
466
|
+
before(function () {
|
|
467
|
+
bitgo = sdk_test_1.TestBitGo.decorate(sdk_api_1.BitGoAPI, { env: 'mock' });
|
|
468
|
+
bitgo.initializeTestVars();
|
|
469
|
+
const tokenConfig = {
|
|
470
|
+
type: 'tada:water',
|
|
471
|
+
coin: 'tada',
|
|
472
|
+
network: 'Testnet',
|
|
473
|
+
name: 'WATER',
|
|
474
|
+
decimalPlaces: 0,
|
|
475
|
+
policyId: policyId,
|
|
476
|
+
assetName: name, // ASCII 'WATER', not hex-encoded
|
|
477
|
+
contractAddress: `${policyId}:${asciiEncodedName}`,
|
|
478
|
+
};
|
|
479
|
+
adaToken = new src_1.AdaToken(bitgo, tokenConfig);
|
|
480
|
+
});
|
|
481
|
+
it('should verify a token transaction with correct token amount', async () => {
|
|
482
|
+
const quantity = '20';
|
|
483
|
+
const totalInput = 20000000;
|
|
484
|
+
const totalAssetList = {
|
|
485
|
+
[fingerprint]: {
|
|
486
|
+
quantity: '100',
|
|
487
|
+
policy_id: policyId,
|
|
488
|
+
asset_name: asciiEncodedName,
|
|
489
|
+
},
|
|
490
|
+
};
|
|
491
|
+
const txBuilder = factory.getTransferBuilder();
|
|
492
|
+
txBuilder.input({
|
|
493
|
+
transaction_id: '3677e75c7ba699bfdc6cd57d42f246f86f63aefd76025006ac78313fad2bba21',
|
|
494
|
+
transaction_index: 1,
|
|
495
|
+
});
|
|
496
|
+
txBuilder.output({
|
|
497
|
+
address: receiverAddress,
|
|
498
|
+
amount: '0',
|
|
499
|
+
multiAssets: {
|
|
500
|
+
asset_name: asciiEncodedName,
|
|
501
|
+
policy_id: policyId,
|
|
502
|
+
quantity,
|
|
503
|
+
fingerprint,
|
|
504
|
+
},
|
|
505
|
+
});
|
|
506
|
+
txBuilder.changeAddress(senderAddress, totalInput.toString(), totalAssetList);
|
|
507
|
+
txBuilder.ttl(800000000);
|
|
508
|
+
txBuilder.isTokenTransaction();
|
|
509
|
+
const tx = (await txBuilder.build());
|
|
510
|
+
const txHex = tx.toBroadcastFormat();
|
|
511
|
+
// Verify transaction with correct token amount
|
|
512
|
+
const txParams = {
|
|
513
|
+
recipients: [
|
|
514
|
+
{
|
|
515
|
+
address: receiverAddress,
|
|
516
|
+
amount: quantity, // Token amount, not ADA amount
|
|
517
|
+
},
|
|
518
|
+
],
|
|
519
|
+
};
|
|
520
|
+
const txPrebuild = { txHex };
|
|
521
|
+
const mockWallet = { coinSpecific: () => ({ baseAddress: senderAddress }) };
|
|
522
|
+
const isVerified = await adaToken.verifyTransaction({ txParams, txPrebuild, wallet: mockWallet });
|
|
523
|
+
isVerified.should.equal(true);
|
|
524
|
+
});
|
|
525
|
+
it('should fail to verify a token transaction with incorrect token amount', async () => {
|
|
526
|
+
const quantity = '20';
|
|
527
|
+
const totalInput = 20000000;
|
|
528
|
+
const totalAssetList = {
|
|
529
|
+
[fingerprint]: {
|
|
530
|
+
quantity: '100',
|
|
531
|
+
policy_id: policyId,
|
|
532
|
+
asset_name: asciiEncodedName,
|
|
533
|
+
},
|
|
534
|
+
};
|
|
535
|
+
const txBuilder = factory.getTransferBuilder();
|
|
536
|
+
txBuilder.input({
|
|
537
|
+
transaction_id: '3677e75c7ba699bfdc6cd57d42f246f86f63aefd76025006ac78313fad2bba21',
|
|
538
|
+
transaction_index: 1,
|
|
539
|
+
});
|
|
540
|
+
txBuilder.output({
|
|
541
|
+
address: receiverAddress,
|
|
542
|
+
amount: '0',
|
|
543
|
+
multiAssets: {
|
|
544
|
+
asset_name: asciiEncodedName,
|
|
545
|
+
policy_id: policyId,
|
|
546
|
+
quantity,
|
|
547
|
+
fingerprint,
|
|
548
|
+
},
|
|
549
|
+
});
|
|
550
|
+
txBuilder.changeAddress(senderAddress, totalInput.toString(), totalAssetList);
|
|
551
|
+
txBuilder.ttl(800000000);
|
|
552
|
+
txBuilder.isTokenTransaction();
|
|
553
|
+
const tx = (await txBuilder.build());
|
|
554
|
+
const txHex = tx.toBroadcastFormat();
|
|
555
|
+
// Verify transaction with WRONG token amount (should fail)
|
|
556
|
+
const txParams = {
|
|
557
|
+
recipients: [
|
|
558
|
+
{
|
|
559
|
+
address: receiverAddress,
|
|
560
|
+
amount: '999', // Wrong amount
|
|
561
|
+
},
|
|
562
|
+
],
|
|
563
|
+
};
|
|
564
|
+
const txPrebuild = { txHex };
|
|
565
|
+
const mockWallet = { coinSpecific: () => ({ baseAddress: senderAddress }) };
|
|
566
|
+
await adaToken
|
|
567
|
+
.verifyTransaction({ txParams, txPrebuild, wallet: mockWallet })
|
|
568
|
+
.should.be.rejectedWith('cannot find recipient in expected output');
|
|
569
|
+
});
|
|
570
|
+
it('should fail to verify when address does not match', async () => {
|
|
571
|
+
const quantity = '20';
|
|
572
|
+
const totalInput = 20000000;
|
|
573
|
+
const totalAssetList = {
|
|
574
|
+
[fingerprint]: {
|
|
575
|
+
quantity: '100',
|
|
576
|
+
policy_id: policyId,
|
|
577
|
+
asset_name: asciiEncodedName,
|
|
578
|
+
},
|
|
579
|
+
};
|
|
580
|
+
const txBuilder = factory.getTransferBuilder();
|
|
581
|
+
txBuilder.input({
|
|
582
|
+
transaction_id: '3677e75c7ba699bfdc6cd57d42f246f86f63aefd76025006ac78313fad2bba21',
|
|
583
|
+
transaction_index: 1,
|
|
584
|
+
});
|
|
585
|
+
txBuilder.output({
|
|
586
|
+
address: receiverAddress,
|
|
587
|
+
amount: '0',
|
|
588
|
+
multiAssets: {
|
|
589
|
+
asset_name: asciiEncodedName,
|
|
590
|
+
policy_id: policyId,
|
|
591
|
+
quantity,
|
|
592
|
+
fingerprint,
|
|
593
|
+
},
|
|
594
|
+
});
|
|
595
|
+
txBuilder.changeAddress(senderAddress, totalInput.toString(), totalAssetList);
|
|
596
|
+
txBuilder.ttl(800000000);
|
|
597
|
+
txBuilder.isTokenTransaction();
|
|
598
|
+
const tx = (await txBuilder.build());
|
|
599
|
+
const txHex = tx.toBroadcastFormat();
|
|
600
|
+
// Verify with wrong address (should fail)
|
|
601
|
+
const txParams = {
|
|
602
|
+
recipients: [
|
|
603
|
+
{
|
|
604
|
+
address: 'addr_test1qqa86e3d7lfpwu0k2rhjz76ecmfxdr74s9kf9yfcp5hj5vmnh6xccjcclrk8jtaw9jgeuy99p2n8smtdpylmy45qjjfsfmp3g6',
|
|
605
|
+
amount: quantity,
|
|
606
|
+
},
|
|
607
|
+
],
|
|
608
|
+
};
|
|
609
|
+
const txPrebuild = { txHex };
|
|
610
|
+
const mockWallet = { coinSpecific: () => ({ baseAddress: senderAddress }) };
|
|
611
|
+
await adaToken
|
|
612
|
+
.verifyTransaction({ txParams, txPrebuild, wallet: mockWallet })
|
|
613
|
+
.should.be.rejectedWith('cannot find recipient in expected output');
|
|
614
|
+
});
|
|
615
|
+
it('should verify transaction when policyId has concatenated assetName (crypto compare format)', async () => {
|
|
616
|
+
// This tests the case where policyId in tokenConfig contains policyId + asciiEncodedAssetName
|
|
617
|
+
// which is consistent with crypto compare format
|
|
618
|
+
const concatenatedPolicyId = policyId + asciiEncodedName; // e.g., 'e16c2dc8ae937e8d3790c7fd7168d7b994621ba14ca11415f39fed725741544552'
|
|
619
|
+
const tokenConfigWithConcatenatedPolicyId = {
|
|
620
|
+
type: 'tada:water',
|
|
621
|
+
coin: 'tada',
|
|
622
|
+
network: 'Testnet',
|
|
623
|
+
name: 'WATER',
|
|
624
|
+
decimalPlaces: 0,
|
|
625
|
+
policyId: concatenatedPolicyId, // policyId + assetName hex
|
|
626
|
+
assetName: name, // ASCII name 'WATER' (not hex encoded)
|
|
627
|
+
contractAddress: `${policyId}:${asciiEncodedName}`,
|
|
628
|
+
};
|
|
629
|
+
const adaTokenWithConcatenatedPolicyId = new src_1.AdaToken(bitgo, tokenConfigWithConcatenatedPolicyId);
|
|
630
|
+
const quantity = '20';
|
|
631
|
+
const totalInput = 20000000;
|
|
632
|
+
const totalAssetList = {
|
|
633
|
+
[fingerprint]: {
|
|
634
|
+
quantity: '100',
|
|
635
|
+
policy_id: policyId,
|
|
636
|
+
asset_name: asciiEncodedName,
|
|
637
|
+
},
|
|
638
|
+
};
|
|
639
|
+
const txBuilder = factory.getTransferBuilder();
|
|
640
|
+
txBuilder.input({
|
|
641
|
+
transaction_id: '3677e75c7ba699bfdc6cd57d42f246f86f63aefd76025006ac78313fad2bba21',
|
|
642
|
+
transaction_index: 1,
|
|
643
|
+
});
|
|
644
|
+
txBuilder.output({
|
|
645
|
+
address: receiverAddress,
|
|
646
|
+
amount: '0',
|
|
647
|
+
multiAssets: {
|
|
648
|
+
asset_name: asciiEncodedName,
|
|
649
|
+
policy_id: policyId,
|
|
650
|
+
quantity,
|
|
651
|
+
fingerprint,
|
|
652
|
+
},
|
|
653
|
+
});
|
|
654
|
+
txBuilder.changeAddress(senderAddress, totalInput.toString(), totalAssetList);
|
|
655
|
+
txBuilder.ttl(800000000);
|
|
656
|
+
txBuilder.isTokenTransaction();
|
|
657
|
+
const tx = (await txBuilder.build());
|
|
658
|
+
const txHex = tx.toBroadcastFormat();
|
|
659
|
+
// Verify transaction - the verifyTransaction should strip the assetName from policyId
|
|
660
|
+
const txParams = {
|
|
661
|
+
recipients: [
|
|
662
|
+
{
|
|
663
|
+
address: receiverAddress,
|
|
664
|
+
amount: quantity,
|
|
665
|
+
},
|
|
666
|
+
],
|
|
667
|
+
};
|
|
668
|
+
const txPrebuild = { txHex };
|
|
669
|
+
const mockWallet = { coinSpecific: () => ({ baseAddress: senderAddress }) };
|
|
670
|
+
const isVerified = await adaTokenWithConcatenatedPolicyId.verifyTransaction({
|
|
671
|
+
txParams,
|
|
672
|
+
txPrebuild,
|
|
673
|
+
wallet: mockWallet,
|
|
674
|
+
});
|
|
675
|
+
isVerified.should.equal(true);
|
|
676
|
+
});
|
|
677
|
+
it('should verify transaction with policyId that does not have concatenated assetName', async () => {
|
|
678
|
+
// This tests the case where policyId is just the 28-byte policy ID (no assetName appended)
|
|
679
|
+
const tokenConfigWithPlainPolicyId = {
|
|
680
|
+
type: 'tada:water',
|
|
681
|
+
coin: 'tada',
|
|
682
|
+
network: 'Testnet',
|
|
683
|
+
name: 'WATER',
|
|
684
|
+
decimalPlaces: 0,
|
|
685
|
+
policyId: policyId, // Just the policy ID without assetName
|
|
686
|
+
assetName: name, // ASCII name 'WATER'
|
|
687
|
+
contractAddress: `${policyId}:${asciiEncodedName}`,
|
|
688
|
+
};
|
|
689
|
+
const adaTokenWithPlainPolicyId = new src_1.AdaToken(bitgo, tokenConfigWithPlainPolicyId);
|
|
690
|
+
const quantity = '20';
|
|
691
|
+
const totalInput = 20000000;
|
|
692
|
+
const totalAssetList = {
|
|
693
|
+
[fingerprint]: {
|
|
694
|
+
quantity: '100',
|
|
695
|
+
policy_id: policyId,
|
|
696
|
+
asset_name: asciiEncodedName,
|
|
697
|
+
},
|
|
698
|
+
};
|
|
699
|
+
const txBuilder = factory.getTransferBuilder();
|
|
700
|
+
txBuilder.input({
|
|
701
|
+
transaction_id: '3677e75c7ba699bfdc6cd57d42f246f86f63aefd76025006ac78313fad2bba21',
|
|
702
|
+
transaction_index: 1,
|
|
703
|
+
});
|
|
704
|
+
txBuilder.output({
|
|
705
|
+
address: receiverAddress,
|
|
706
|
+
amount: '0',
|
|
707
|
+
multiAssets: {
|
|
708
|
+
asset_name: asciiEncodedName,
|
|
709
|
+
policy_id: policyId,
|
|
710
|
+
quantity,
|
|
711
|
+
fingerprint,
|
|
712
|
+
},
|
|
713
|
+
});
|
|
714
|
+
txBuilder.changeAddress(senderAddress, totalInput.toString(), totalAssetList);
|
|
715
|
+
txBuilder.ttl(800000000);
|
|
716
|
+
txBuilder.isTokenTransaction();
|
|
717
|
+
const tx = (await txBuilder.build());
|
|
718
|
+
const txHex = tx.toBroadcastFormat();
|
|
719
|
+
// Verify transaction - should work with plain policyId as well
|
|
720
|
+
const txParams = {
|
|
721
|
+
recipients: [
|
|
722
|
+
{
|
|
723
|
+
address: receiverAddress,
|
|
724
|
+
amount: quantity,
|
|
725
|
+
},
|
|
726
|
+
],
|
|
727
|
+
};
|
|
728
|
+
const txPrebuild = { txHex };
|
|
729
|
+
const mockWallet = { coinSpecific: () => ({ baseAddress: senderAddress }) };
|
|
730
|
+
const isVerified = await adaTokenWithPlainPolicyId.verifyTransaction({
|
|
731
|
+
txParams,
|
|
732
|
+
txPrebuild,
|
|
733
|
+
wallet: mockWallet,
|
|
734
|
+
});
|
|
735
|
+
isVerified.should.equal(true);
|
|
736
|
+
});
|
|
737
|
+
it('should verify token consolidation transaction when all outputs go to base address', async () => {
|
|
738
|
+
const quantity = '100';
|
|
739
|
+
const totalInput = 20000000;
|
|
740
|
+
const totalAssetList = {
|
|
741
|
+
[fingerprint]: {
|
|
742
|
+
quantity: '100',
|
|
743
|
+
policy_id: policyId,
|
|
744
|
+
asset_name: asciiEncodedName,
|
|
745
|
+
},
|
|
746
|
+
};
|
|
747
|
+
// Build a consolidation transaction - all outputs go to sender (base) address
|
|
748
|
+
const txBuilder = factory.getTransferBuilder();
|
|
749
|
+
txBuilder.input({
|
|
750
|
+
transaction_id: '3677e75c7ba699bfdc6cd57d42f246f86f63aefd76025006ac78313fad2bba21',
|
|
751
|
+
transaction_index: 1,
|
|
752
|
+
});
|
|
753
|
+
// For consolidation, tokens go back to the sender's base address
|
|
754
|
+
txBuilder.output({
|
|
755
|
+
address: senderAddress,
|
|
756
|
+
amount: '0',
|
|
757
|
+
multiAssets: {
|
|
758
|
+
asset_name: asciiEncodedName,
|
|
759
|
+
policy_id: policyId,
|
|
760
|
+
quantity,
|
|
761
|
+
fingerprint,
|
|
762
|
+
},
|
|
763
|
+
});
|
|
764
|
+
txBuilder.changeAddress(senderAddress, totalInput.toString(), totalAssetList);
|
|
765
|
+
txBuilder.ttl(800000000);
|
|
766
|
+
txBuilder.isTokenTransaction();
|
|
767
|
+
const tx = (await txBuilder.build());
|
|
768
|
+
const txHex = tx.toBroadcastFormat();
|
|
769
|
+
// Mock wallet with coinSpecific returning base address
|
|
770
|
+
const mockWallet = {
|
|
771
|
+
coinSpecific: () => ({
|
|
772
|
+
baseAddress: senderAddress,
|
|
773
|
+
}),
|
|
774
|
+
};
|
|
775
|
+
// Verify consolidation transaction - no recipients, but consolidationToBaseAddress is true
|
|
776
|
+
const txParams = {
|
|
777
|
+
recipients: undefined,
|
|
778
|
+
};
|
|
779
|
+
const txPrebuild = { txHex };
|
|
780
|
+
const verification = { consolidationToBaseAddress: true };
|
|
781
|
+
const isVerified = await adaToken.verifyTransaction({
|
|
782
|
+
txParams,
|
|
783
|
+
txPrebuild,
|
|
784
|
+
verification,
|
|
785
|
+
wallet: mockWallet,
|
|
786
|
+
});
|
|
787
|
+
isVerified.should.equal(true);
|
|
788
|
+
});
|
|
789
|
+
it('should fail token consolidation when output address does not match base address', async () => {
|
|
790
|
+
const quantity = '100';
|
|
791
|
+
const totalInput = 20000000;
|
|
792
|
+
const totalAssetList = {
|
|
793
|
+
[fingerprint]: {
|
|
794
|
+
quantity: '100',
|
|
795
|
+
policy_id: policyId,
|
|
796
|
+
asset_name: asciiEncodedName,
|
|
797
|
+
},
|
|
798
|
+
};
|
|
799
|
+
// Build a transaction with output to receiver (not base address)
|
|
800
|
+
const txBuilder = factory.getTransferBuilder();
|
|
801
|
+
txBuilder.input({
|
|
802
|
+
transaction_id: '3677e75c7ba699bfdc6cd57d42f246f86f63aefd76025006ac78313fad2bba21',
|
|
803
|
+
transaction_index: 1,
|
|
804
|
+
});
|
|
805
|
+
txBuilder.output({
|
|
806
|
+
address: receiverAddress, // Output goes to receiver, not base address
|
|
807
|
+
amount: '0',
|
|
808
|
+
multiAssets: {
|
|
809
|
+
asset_name: asciiEncodedName,
|
|
810
|
+
policy_id: policyId,
|
|
811
|
+
quantity,
|
|
812
|
+
fingerprint,
|
|
813
|
+
},
|
|
814
|
+
});
|
|
815
|
+
txBuilder.changeAddress(senderAddress, totalInput.toString(), totalAssetList);
|
|
816
|
+
txBuilder.ttl(800000000);
|
|
817
|
+
txBuilder.isTokenTransaction();
|
|
818
|
+
const tx = (await txBuilder.build());
|
|
819
|
+
const txHex = tx.toBroadcastFormat();
|
|
820
|
+
// Mock wallet with different base address
|
|
821
|
+
const mockWallet = {
|
|
822
|
+
coinSpecific: () => ({
|
|
823
|
+
baseAddress: senderAddress, // Base address is sender, but output goes to receiver
|
|
824
|
+
}),
|
|
825
|
+
};
|
|
826
|
+
const txParams = {
|
|
827
|
+
recipients: undefined,
|
|
828
|
+
};
|
|
829
|
+
const txPrebuild = { txHex };
|
|
830
|
+
const verification = { consolidationToBaseAddress: true };
|
|
831
|
+
await adaToken
|
|
832
|
+
.verifyTransaction({
|
|
833
|
+
txParams,
|
|
834
|
+
txPrebuild,
|
|
835
|
+
verification,
|
|
836
|
+
wallet: mockWallet,
|
|
837
|
+
})
|
|
838
|
+
.should.be.rejectedWith('tx outputs does not match with expected address');
|
|
839
|
+
});
|
|
840
|
+
it('should verify sponsored token consolidation when fee address output is present', async () => {
|
|
841
|
+
const feeAddress = 'addr_test1qz2fxv2umyhttkxyxp8x0dlpdt3k6cwng5pxj3jhsydzer3jcu5d8ps7zex2k2xt3uqxgjqnnj83ws8lhrn648jjxtwq2ytjqp';
|
|
842
|
+
const quantity = '100';
|
|
843
|
+
const senderInputBalance = 5000000;
|
|
844
|
+
const feeAddressInputBalance = 20000000;
|
|
845
|
+
const totalAssetList = {
|
|
846
|
+
[fingerprint]: {
|
|
847
|
+
quantity: '100',
|
|
848
|
+
policy_id: policyId,
|
|
849
|
+
asset_name: asciiEncodedName,
|
|
850
|
+
},
|
|
851
|
+
};
|
|
852
|
+
// Build a sponsored token consolidation: tokens go back to senderAddress, fee sponsor gets ADA change
|
|
853
|
+
const txBuilder = factory.getTransferBuilder();
|
|
854
|
+
txBuilder.input({
|
|
855
|
+
transaction_id: '3677e75c7ba699bfdc6cd57d42f246f86f63aefd76025006ac78313fad2bba21',
|
|
856
|
+
transaction_index: 1,
|
|
857
|
+
});
|
|
858
|
+
txBuilder.input({
|
|
859
|
+
transaction_id: '3677e75c7ba699bfdc6cd57d42f246f86f63aefd76025006ac78313fad2bba22',
|
|
860
|
+
transaction_index: 0,
|
|
861
|
+
});
|
|
862
|
+
// Consolidation output: tokens go back to the sender's base address
|
|
863
|
+
txBuilder.output({
|
|
864
|
+
address: senderAddress,
|
|
865
|
+
amount: '0',
|
|
866
|
+
multiAssets: {
|
|
867
|
+
asset_name: asciiEncodedName,
|
|
868
|
+
policy_id: policyId,
|
|
869
|
+
quantity,
|
|
870
|
+
fingerprint,
|
|
871
|
+
},
|
|
872
|
+
});
|
|
873
|
+
txBuilder.changeAddress(senderAddress, senderInputBalance.toString(), totalAssetList);
|
|
874
|
+
txBuilder.sponsorshipInfo({
|
|
875
|
+
feeAddress: feeAddress,
|
|
876
|
+
feeAddressInputBalance: feeAddressInputBalance.toString(),
|
|
877
|
+
});
|
|
878
|
+
txBuilder.ttl(800000000);
|
|
879
|
+
txBuilder.isTokenTransaction();
|
|
880
|
+
const tx = (await txBuilder.build());
|
|
881
|
+
const txHex = tx.toBroadcastFormat();
|
|
882
|
+
const mockWallet = {
|
|
883
|
+
coinSpecific: () => ({
|
|
884
|
+
baseAddress: senderAddress,
|
|
885
|
+
}),
|
|
886
|
+
};
|
|
887
|
+
const txParams = { recipients: undefined };
|
|
888
|
+
const txPrebuild = { txHex, txInfo: { feeAddress } };
|
|
889
|
+
const verification = { consolidationToBaseAddress: true };
|
|
890
|
+
const isVerified = await adaToken.verifyTransaction({
|
|
891
|
+
txParams,
|
|
892
|
+
txPrebuild,
|
|
893
|
+
verification,
|
|
894
|
+
wallet: mockWallet,
|
|
895
|
+
});
|
|
896
|
+
isVerified.should.equal(true);
|
|
897
|
+
});
|
|
898
|
+
it('should fail sponsored token consolidation when output goes to unexpected third-party address', async () => {
|
|
899
|
+
const feeAddress = 'addr_test1qz2fxv2umyhttkxyxp8x0dlpdt3k6cwng5pxj3jhsydzer3jcu5d8ps7zex2k2xt3uqxgjqnnj83ws8lhrn648jjxtwq2ytjqp';
|
|
900
|
+
const quantity = '100';
|
|
901
|
+
const senderInputBalance = 5000000;
|
|
902
|
+
const feeAddressInputBalance = 20000000;
|
|
903
|
+
const totalAssetList = {
|
|
904
|
+
[fingerprint]: {
|
|
905
|
+
quantity: '100',
|
|
906
|
+
policy_id: policyId,
|
|
907
|
+
asset_name: asciiEncodedName,
|
|
908
|
+
},
|
|
909
|
+
};
|
|
910
|
+
// Build transaction where token output goes to an unexpected third-party address
|
|
911
|
+
const txBuilder = factory.getTransferBuilder();
|
|
912
|
+
txBuilder.input({
|
|
913
|
+
transaction_id: '3677e75c7ba699bfdc6cd57d42f246f86f63aefd76025006ac78313fad2bba21',
|
|
914
|
+
transaction_index: 1,
|
|
915
|
+
});
|
|
916
|
+
txBuilder.input({
|
|
917
|
+
transaction_id: '3677e75c7ba699bfdc6cd57d42f246f86f63aefd76025006ac78313fad2bba22',
|
|
918
|
+
transaction_index: 0,
|
|
919
|
+
});
|
|
920
|
+
// Output goes to receiverAddress (third party), not the base address
|
|
921
|
+
txBuilder.output({
|
|
922
|
+
address: receiverAddress,
|
|
923
|
+
amount: '0',
|
|
924
|
+
multiAssets: {
|
|
925
|
+
asset_name: asciiEncodedName,
|
|
926
|
+
policy_id: policyId,
|
|
927
|
+
quantity,
|
|
928
|
+
fingerprint,
|
|
929
|
+
},
|
|
930
|
+
});
|
|
931
|
+
txBuilder.changeAddress(senderAddress, senderInputBalance.toString(), totalAssetList);
|
|
932
|
+
txBuilder.sponsorshipInfo({
|
|
933
|
+
feeAddress: feeAddress,
|
|
934
|
+
feeAddressInputBalance: feeAddressInputBalance.toString(),
|
|
935
|
+
});
|
|
936
|
+
txBuilder.ttl(800000000);
|
|
937
|
+
txBuilder.isTokenTransaction();
|
|
938
|
+
const tx = (await txBuilder.build());
|
|
939
|
+
const txHex = tx.toBroadcastFormat();
|
|
940
|
+
const mockWallet = {
|
|
941
|
+
coinSpecific: () => ({
|
|
942
|
+
baseAddress: senderAddress,
|
|
943
|
+
}),
|
|
944
|
+
};
|
|
945
|
+
const txParams = { recipients: undefined };
|
|
946
|
+
const txPrebuild = { txHex, txInfo: { feeAddress } };
|
|
947
|
+
const verification = { consolidationToBaseAddress: true };
|
|
948
|
+
await adaToken
|
|
949
|
+
.verifyTransaction({
|
|
950
|
+
txParams,
|
|
951
|
+
txPrebuild,
|
|
952
|
+
verification,
|
|
953
|
+
wallet: mockWallet,
|
|
954
|
+
})
|
|
955
|
+
.should.be.rejectedWith('tx outputs does not match with expected address');
|
|
956
|
+
});
|
|
957
|
+
});
|
|
958
|
+
});
|
|
959
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidG9rZW5XaXRoZHJhd2FsLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vdGVzdC91bml0L3Rva2VuV2l0aGRyYXdhbC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLG9EQUE0QjtBQUM1QixtREFBdUQ7QUFDdkQsdURBQXlDO0FBQ3pDLG1DQUFnRTtBQUNoRSxpREFBNEM7QUFDNUMsc0ZBQXdFO0FBRXhFLG1EQUErRDtBQUMvRCxpREFBK0M7QUFFL0MsUUFBUSxDQUFDLHNCQUFzQixFQUFFLEtBQUssSUFBSSxFQUFFO0lBQzFDLE1BQU0sT0FBTyxHQUFHLElBQUksK0JBQXlCLENBQUMsZUFBSyxDQUFDLEdBQUcsQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDO0lBRWpFLE1BQU0sZUFBZSxHQUFHLFFBQVEsQ0FBQyxLQUFLLENBQUMsY0FBYyxDQUFDLE9BQU8sQ0FBQztJQUM5RCxNQUFNLGFBQWEsR0FBRyxRQUFRLENBQUMsS0FBSyxDQUFDLGNBQWMsQ0FBQyxPQUFPLENBQUM7SUFFNUQsTUFBTSxRQUFRLEdBQUcsMERBQTBELENBQUM7SUFDNUUsTUFBTSxnQkFBZ0IsR0FBRyxXQUFXLENBQUMsVUFBVSxDQUFDLFFBQVEsQ0FBQyxRQUFRLENBQUMsQ0FBQztJQUNuRSxNQUFNLFNBQVMsR0FBRyxZQUFZLENBQUM7SUFDL0IsTUFBTSxJQUFJLEdBQUcsT0FBTyxDQUFDO0lBQ3JCLE1BQU0sZ0JBQWdCLEdBQUcsTUFBTSxDQUFDLElBQUksQ0FBQyxJQUFJLEVBQUUsT0FBTyxDQUFDLENBQUMsUUFBUSxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQ3BFLE1BQU0sV0FBVyxHQUFHLDhDQUE4QyxDQUFDO0lBRW5FLE1BQU0sbUJBQW1CLEdBQUcsMERBQTBELENBQUM7SUFDdkYsTUFBTSwyQkFBMkIsR0FBRyxXQUFXLENBQUMsVUFBVSxDQUFDLFFBQVEsQ0FBQyxtQkFBbUIsQ0FBQyxDQUFDO0lBQ3pGLE1BQU0sZUFBZSxHQUFHLE9BQU8sQ0FBQztJQUNoQyxNQUFNLDJCQUEyQixHQUFHLE1BQU0sQ0FBQyxJQUFJLENBQUMsZUFBZSxFQUFFLE9BQU8sQ0FBQyxDQUFDLFFBQVEsQ0FBQyxLQUFLLENBQUMsQ0FBQztJQUMxRixNQUFNLHNCQUFzQixHQUFHLDhDQUE4QyxDQUFDO0lBRTlFLEVBQUUsQ0FBQyx5Q0FBeUMsU0FBUyxZQUFZLFNBQVMsRUFBRSxFQUFFLEtBQUssSUFBSSxFQUFFO1FBQ3ZGLE1BQU0sUUFBUSxHQUFHLElBQUksQ0FBQztRQUN0QixNQUFNLFVBQVUsR0FBRyxRQUFRLENBQUM7UUFDNUIsTUFBTSxjQUFjLEdBQUc7WUFDckIsQ0FBQyxXQUFXLENBQUMsRUFBRTtnQkFDYixRQUFRLEVBQUUsS0FBSztnQkFDZixTQUFTLEVBQUUsUUFBUTtnQkFDbkIsVUFBVSxFQUFFLGdCQUFnQjthQUM3QjtTQUNGLENBQUM7UUFDRixNQUFNLGlCQUFpQixHQUFHLENBQ3hCLFVBQVU7WUFDVixPQUFPLENBQUMsbUNBQW1DO1lBQzNDLE9BQU8sQ0FBQyxxQ0FBcUM7WUFDN0MsTUFBTSxDQUNQLENBQUMsU0FBUzthQUNSLFFBQVEsRUFBRSxDQUFDO1FBQ2QsTUFBTSxtQkFBbUIsR0FBRyxJQUFJLENBQUM7UUFFakMsTUFBTSxTQUFTLEdBQUcsT0FBTyxDQUFDLGtCQUFrQixFQUFFLENBQUM7UUFDL0MsU0FBUyxDQUFDLEtBQUssQ0FBQztZQUNkLGNBQWMsRUFBRSxrRUFBa0U7WUFDbEYsaUJBQWlCLEVBQUUsQ0FBQztTQUNyQixDQUFDLENBQUM7UUFFSCxTQUFTLENBQUMsTUFBTSxDQUFDO1lBQ2YsT0FBTyxFQUFFLGVBQWU7WUFDeEIsTUFBTSxFQUFFLEdBQUcsRUFBRSwyRUFBMkU7WUFDeEYsV0FBVyxFQUFFO2dCQUNYLFVBQVUsRUFBRSxnQkFBZ0I7Z0JBQzVCLFNBQVMsRUFBRSxRQUFRO2dCQUNuQixRQUFRO2dCQUNSLFdBQVc7YUFDWjtTQUNGLENBQUMsQ0FBQztRQUVILFNBQVMsQ0FBQyxhQUFhLENBQUMsYUFBYSxFQUFFLFVBQVUsQ0FBQyxRQUFRLEVBQUUsRUFBRSxjQUFjLENBQUMsQ0FBQztRQUM5RSxTQUFTLENBQUMsR0FBRyxDQUFDLFNBQVMsQ0FBQyxDQUFDO1FBQ3pCLFNBQVMsQ0FBQyxrQkFBa0IsRUFBRSxDQUFDO1FBQy9CLE1BQU0sRUFBRSxHQUFHLENBQUMsTUFBTSxTQUFTLENBQUMsS0FBSyxFQUFFLENBQWdCLENBQUM7UUFFcEQsZ0JBQU0sQ0FBQyxLQUFLLENBQUMsRUFBRSxDQUFDLElBQUksRUFBRSwwQkFBZSxDQUFDLElBQUksQ0FBQyxDQUFDO1FBQzVDLE1BQU0sTUFBTSxHQUFHLEVBQUUsQ0FBQyxNQUFNLEVBQUUsQ0FBQztRQUUzQixnRUFBZ0U7UUFDaEUsTUFBTSxDQUFDLE9BQU8sQ0FBQyxNQUFNLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUN0QyxNQUFNLENBQUMsTUFBTSxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBRXJDLCtFQUErRTtRQUMvRSxNQUFNLFdBQVcsR0FBRyxNQUFNLENBQUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxDQUFDLE1BQU0sRUFBRSxFQUFFLENBQUMsTUFBTSxDQUFDLFdBQVcsS0FBSyxTQUFTLENBQUMsQ0FBQztRQUN4RixXQUFZLENBQUMsTUFBTSxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFFcEMsNEJBQTRCO1FBQzVCLE1BQU0sS0FBSyxHQUFHLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDL0IsS0FBSyxDQUFDLGNBQWMsQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLGtFQUFrRSxDQUFDLENBQUM7UUFDdEcsS0FBSyxDQUFDLGlCQUFpQixDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFFeEMsMkJBQTJCO1FBQzNCLE1BQU0sY0FBYyxHQUFHLE1BQU0sQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUFDLENBQUMsTUFBTSxFQUFFLEVBQUUsQ0FBQyxNQUFNLENBQUMsT0FBTyxLQUFLLGVBQWUsQ0FBQyxDQUFDO1FBQzdGLGNBQWMsQ0FBQyxNQUFNLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUN0QyxjQUFjLENBQUMsQ0FBQyxDQUFDLENBQUMsTUFBTSxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQywrQkFBK0I7UUFDaEYsY0FBYyxDQUFDLENBQUMsQ0FBQyxDQUFDLFdBQXVDO2FBQ3ZELFNBQVMsQ0FBQyxnQkFBZ0IsRUFBRSxXQUFXLENBQUMsU0FBUyxDQUFDLEdBQUcsQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLGdCQUFnQixFQUFFLEtBQUssQ0FBQyxDQUFDLENBQUM7YUFDNUYsTUFBTSxFQUFFO2FBQ1IsTUFBTSxDQUFDLEtBQUssQ0FBQyxRQUFRLENBQUMsQ0FBQztRQUUxQiwyREFBMkQ7UUFDM0QsTUFBTSxZQUFZLEdBQUcsTUFBTSxDQUFDLE9BQU8sQ0FBQyxNQUFNLENBQUMsQ0FBQyxNQUFNLEVBQUUsRUFBRSxDQUFDLE1BQU0sQ0FBQyxPQUFPLEtBQUssYUFBYSxDQUFDLENBQUM7UUFDekYsWUFBWSxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQ3BDLE1BQU0sZUFBZSxHQUFHLFlBQVksQ0FBQyxJQUFJLENBQUMsQ0FBQyxNQUFNLEVBQUUsRUFBRSxDQUFDLE1BQU0sQ0FBQyxXQUFXLEtBQUssU0FBUyxDQUFDLENBQUM7UUFDeEYsZ0JBQU0sQ0FBQyxLQUFLLENBQUMsZUFBZSxDQUFDLENBQUM7UUFDOUIsZUFBZ0IsQ0FBQyxNQUFNLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxTQUFTLENBQUMsQ0FBQyxDQUFDLCtCQUErQjtRQUMvRSxlQUFnQixDQUFDLFdBQXVDO2FBQ3RELFNBQVMsQ0FBQyxnQkFBZ0IsRUFBRSxXQUFXLENBQUMsU0FBUyxDQUFDLEdBQUcsQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLGdCQUFnQixFQUFFLEtBQUssQ0FBQyxDQUFDLENBQUM7YUFDNUYsTUFBTSxFQUFFO2FBQ1IsTUFBTSxDQUFDLEtBQUssQ0FBQyxtQkFBbUIsQ0FBQyxDQUFDO1FBRXJDLE1BQU0sa0JBQWtCLEdBQUcsWUFBWSxDQUFDLElBQUksQ0FBQyxDQUFDLE1BQU0sRUFBRSxFQUFFLENBQUMsTUFBTSxDQUFDLFdBQVcsS0FBSyxTQUFTLENBQUMsQ0FBQztRQUMzRixnQkFBTSxDQUFDLEtBQUssQ0FBQyxrQkFBa0IsQ0FBQyxDQUFDO1FBQ2pDLGtCQUFtQixDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLGlCQUFpQixDQUFDLENBQUMsQ0FBQyx3Q0FBd0M7SUFDdEcsQ0FBQyxDQUFDLENBQUM7SUFFSCxFQUFFLENBQUMseUNBQXlDLFNBQVMsWUFBWSxTQUFTLHNCQUFzQixFQUFFLEtBQUssSUFBSSxFQUFFO1FBQzNHLE1BQU0sUUFBUSxHQUFHLElBQUksQ0FBQztRQUN0QixNQUFNLFVBQVUsR0FBRyxRQUFRLENBQUM7UUFDNUIsTUFBTSxjQUFjLEdBQUc7WUFDckIsQ0FBQyxXQUFXLENBQUMsRUFBRTtnQkFDYixRQUFRLEVBQUUsS0FBSztnQkFDZixTQUFTLEVBQUUsUUFBUTtnQkFDbkIsVUFBVSxFQUFFLGdCQUFnQjthQUM3QjtZQUNELENBQUMsc0JBQXNCLENBQUMsRUFBRTtnQkFDeEIsUUFBUSxFQUFFLE9BQU87Z0JBQ2pCLFNBQVMsRUFBRSxtQkFBbUI7Z0JBQzlCLFVBQVUsRUFBRSwyQkFBMkI7YUFDeEM7U0FDRixDQUFDO1FBQ0YsTUFBTSxpQkFBaUIsR0FBRyxDQUN4QixVQUFVO1lBQ1YsT0FBTyxDQUFDLG1DQUFtQztZQUMzQyxPQUFPLENBQUMscUNBQXFDO1lBQzdDLE9BQU8sQ0FBQywrQ0FBK0M7WUFDdkQsTUFBTSxDQUNQLENBQUMsU0FBUzthQUNSLFFBQVEsRUFBRSxDQUFDO1FBQ2QsTUFBTSxtQkFBbUIsR0FBRyxJQUFJLENBQUM7UUFFakMsTUFBTSxTQUFTLEdBQUcsT0FBTyxDQUFDLGtCQUFrQixFQUFFLENBQUM7UUFDL0MsU0FBUyxDQUFDLEtBQUssQ0FBQztZQUNkLGNBQWMsRUFBRSxrRUFBa0U7WUFDbEYsaUJBQWlCLEVBQUUsQ0FBQztTQUNyQixDQUFDLENBQUM7UUFDSCxTQUFTLENBQUMsS0FBSyxDQUFDO1lBQ2QsY0FBYyxFQUFFLGtFQUFrRTtZQUNsRixpQkFBaUIsRUFBRSxDQUFDO1NBQ3JCLENBQUMsQ0FBQztRQUVILFNBQVMsQ0FBQyxNQUFNLENBQUM7WUFDZixPQUFPLEVBQUUsZUFBZTtZQUN4QixNQUFNLEVBQUUsR0FBRyxFQUFFLDJFQUEyRTtZQUN4RixXQUFXLEVBQUU7Z0JBQ1gsVUFBVSxFQUFFLGdCQUFnQjtnQkFDNUIsU0FBUyxFQUFFLFFBQVE7Z0JBQ25CLFFBQVE7Z0JBQ1IsV0FBVzthQUNaO1NBQ0YsQ0FBQyxDQUFDO1FBRUgsU0FBUyxDQUFDLGFBQWEsQ0FBQyxhQUFhLEVBQUUsVUFBVSxDQUFDLFFBQVEsRUFBRSxFQUFFLGNBQWMsQ0FBQyxDQUFDO1FBQzlFLFNBQVMsQ0FBQyxHQUFHLENBQUMsU0FBUyxDQUFDLENBQUM7UUFDekIsU0FBUyxDQUFDLGtCQUFrQixFQUFFLENBQUM7UUFDL0IsTUFBTSxFQUFFLEdBQUcsQ0FBQyxNQUFNLFNBQVMsQ0FBQyxLQUFLLEVBQUUsQ0FBZ0IsQ0FBQztRQUVwRCxnQkFBTSxDQUFDLEtBQUssQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLDBCQUFlLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDNUMsTUFBTSxNQUFNLEdBQUcsRUFBRSxDQUFDLE1BQU0sRUFBRSxDQUFDO1FBRTNCLGdFQUFnRTtRQUNoRSxNQUFNLENBQUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQ3RDLE1BQU0sQ0FBQyxNQUFNLENBQUMsTUFBTSxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFFckMsK0VBQStFO1FBQy9FLE1BQU0sV0FBVyxHQUFHLE1BQU0sQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUFDLENBQUMsTUFBTSxFQUFFLEVBQUUsQ0FBQyxNQUFNLENBQUMsV0FBVyxLQUFLLFNBQVMsQ0FBQyxDQUFDO1FBQ3hGLFdBQVksQ0FBQyxNQUFNLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUVwQyw0QkFBNEI7UUFDNUIsTUFBTSxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsQ0FBQyxjQUFjLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxrRUFBa0UsQ0FBQyxDQUFDO1FBQ2pILE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLENBQUMsaUJBQWlCLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUVuRCxNQUFNLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxDQUFDLGNBQWMsQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLGtFQUFrRSxDQUFDLENBQUM7UUFDakgsTUFBTSxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsQ0FBQyxpQkFBaUIsQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBRW5ELDJCQUEyQjtRQUMzQixNQUFNLGNBQWMsR0FBRyxNQUFNLENBQUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxDQUFDLE1BQU0sRUFBRSxFQUFFLENBQUMsTUFBTSxDQUFDLE9BQU8sS0FBSyxlQUFlLENBQUMsQ0FBQztRQUM3RixjQUFjLENBQUMsTUFBTSxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDdEMsY0FBYyxDQUFDLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLFNBQVMsQ0FBQyxDQUFDLENBQUMsK0JBQStCO1FBQ2hGLGNBQWMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxXQUF1QzthQUN2RCxTQUFTLENBQUMsZ0JBQWdCLEVBQUUsV0FBVyxDQUFDLFNBQVMsQ0FBQyxHQUFHLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxnQkFBZ0IsRUFBRSxLQUFLLENBQUMsQ0FBQyxDQUFDO2FBQzVGLE1BQU0sRUFBRTthQUNSLE1BQU0sQ0FBQyxLQUFLLENBQUMsUUFBUSxDQUFDLENBQUM7UUFFMUIsc0dBQXNHO1FBQ3RHLE1BQU0sWUFBWSxHQUFHLE1BQU0sQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUFDLENBQUMsTUFBTSxFQUFFLEVBQUUsQ0FBQyxNQUFNLENBQUMsT0FBTyxLQUFLLGFBQWEsQ0FBQyxDQUFDO1FBQ3pGLFlBQVksQ0FBQyxNQUFNLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUVwQyxNQUFNLGVBQWUsR0FBRyxZQUFZLENBQUMsTUFBTSxDQUFDLENBQUMsTUFBTSxFQUFFLEVBQUUsQ0FBQyxNQUFNLENBQUMsV0FBVyxLQUFLLFNBQVMsQ0FBQyxDQUFDO1FBQzFGLGVBQWUsQ0FBQyxNQUFNLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUN2QyxJQUFJLE1BQU0sR0FBRyxDQUFDLENBQUM7UUFDZixlQUFlLENBQUMsT0FBTyxDQUFDLENBQUMsTUFBTSxFQUFFLEVBQUU7WUFDakMsTUFBTyxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLFNBQVMsQ0FBQyxDQUFDLENBQUMsK0JBQStCO1lBQ3ZFLE1BQU0sdUJBQXVCLEdBQUksTUFBTyxDQUFDLFdBQXVDO2lCQUM3RSxTQUFTLENBQUMsZ0JBQWdCLEVBQUUsV0FBVyxDQUFDLFNBQVMsQ0FBQyxHQUFHLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxnQkFBZ0IsRUFBRSxLQUFLLENBQUMsQ0FBQyxDQUFDO2lCQUM1RixNQUFNLEVBQUUsQ0FBQztZQUNaLE1BQU0seUJBQXlCLEdBQUksTUFBTyxDQUFDLFdBQXVDO2lCQUMvRSxTQUFTLENBQ1IsMkJBQTJCLEVBQzNCLFdBQVcsQ0FBQyxTQUFTLENBQUMsR0FBRyxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsMkJBQTJCLEVBQUUsS0FBSyxDQUFDLENBQUMsQ0FDM0U7aUJBQ0EsTUFBTSxFQUFFLENBQUM7WUFDWixJQUFJLHVCQUF1QixLQUFLLEdBQUcsRUFBRSxDQUFDO2dCQUNwQyx1QkFBdUIsQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLG1CQUFtQixDQUFDLENBQUM7Z0JBQzFELE1BQU0sRUFBRSxDQUFDO1lBQ1gsQ0FBQztpQkFBTSxDQUFDO2dCQUNOLHlCQUEwQixDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLENBQUM7Z0JBQ2pELE1BQU0sRUFBRSxDQUFDO1lBQ1gsQ0FBQztRQUNILENBQUMsQ0FBQyxDQUFDO1FBQ0gsTUFBTSxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFFdkIsTUFBTSxrQkFBa0IsR0FBRyxZQUFZLENBQUMsSUFBSSxDQUFDLENBQUMsTUFBTSxFQUFFLEVBQUUsQ0FBQyxNQUFNLENBQUMsV0FBVyxLQUFLLFNBQVMsQ0FBQyxDQUFDO1FBQzNGLGdCQUFNLENBQUMsS0FBSyxDQUFDLGtCQUFrQixDQUFDLENBQUM7UUFDakMsa0JBQW1CLENBQUMsTUFBTSxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsaUJBQWlCLENBQUMsQ0FBQyxDQUFDLHdDQUF3QztJQUN0RyxDQUFDLENBQUMsQ0FBQztJQUVILEVBQUUsQ0FBQywyQ0FBMkMsU0FBUyxxQ0FBcUMsRUFBRSxLQUFLLElBQUksRUFBRTtRQUN2RyxNQUFNLFFBQVEsR0FBRyxJQUFJLENBQUM7UUFDdEIsTUFBTSxVQUFVLEdBQUcsT0FBTyxDQUFDO1FBQzNCLE1BQU0sY0FBYyxHQUFHO1lBQ3JCLENBQUMsV0FBVyxDQUFDLEVBQUU7Z0JBQ2IsUUFBUSxFQUFFLEtBQUs7Z0JBQ2YsU0FBUyxFQUFFLFFBQVE7Z0JBQ25CLFVBQVUsRUFBRSxnQkFBZ0I7YUFDN0I7U0FDRixDQUFDO1FBRUYsTUFBTSxTQUFTLEdBQUcsT0FBTyxDQUFDLGtCQUFrQixFQUFFLENBQUM7UUFDL0MsU0FBUyxDQUFDLEtBQUssQ0FBQztZQUNkLGNBQWMsRUFBRSxrRUFBa0U7WUFDbEYsaUJBQWlCLEVBQUUsQ0FBQztTQUNyQixDQUFDLENBQUM7UUFFSCxTQUFTLENBQUMsTUFBTSxDQUFDO1lBQ2YsT0FBTyxFQUFFLGVBQWU7WUFDeEIsTUFBTSxFQUFFLEdBQUcsRUFBRSwyRUFBMkU7WUFDeEYsV0FBVyxFQUFFO2dCQUNYLFVBQVUsRUFBRSxnQkFBZ0I7Z0JBQzVCLFNBQVMsRUFBRSxRQUFRO2dCQUNuQixRQUFRO2dCQUNSLFdBQVc7YUFDWjtTQUNGLENBQUMsQ0FBQztRQUVILFNBQVMsQ0FBQyxhQUFhLENBQUMsYUFBYSxFQUFFLFVBQVUsQ0FBQyxRQUFRLEVBQUUsRUFBRSxjQUFjLENBQUMsQ0FBQztRQUM5RSxTQUFTLENBQUMsR0FBRyxDQUFDLFNBQVMsQ0FBQyxDQUFDO1FBQ3pCLFNBQVMsQ0FBQyxrQkFBa0IsRUFBRSxDQUFDO1FBRS9CLE1BQU0sU0FBUzthQUNaLEtBQUssRUFBRTthQUNQLE1BQU0sQ0FBQyxZQUFZLENBQ2xCLG1HQUFtRyxDQUNwRyxDQUFDO0lBQ04sQ0FBQyxDQUFDLENBQUM7SUFFSCxFQUFFLENBQUMsMkNBQTJDLFNBQVMsNkJBQTZCLEVBQUUsS0FBSyxJQUFJLEVBQUU7UUFDL0YsTUFBTSxRQUFRLEdBQUcsSUFBSSxDQUFDO1FBQ3RCLE1BQU0sVUFBVSxHQUFHLFFBQVEsQ0FBQztRQUM1QixNQUFNLGNBQWMsR0FBRztZQUNyQixDQUFDLFdBQVcsQ0FBQyxFQUFFO2dCQUNiLFFBQVEsRUFBRSxHQUFHO2dCQUNiLFNBQVMsRUFBRSxRQUFRO2dCQUNuQixVQUFVLEVBQUUsZ0JBQWdCO2FBQzdCO1NBQ0YsQ0FBQztRQUVGLE1BQU0sU0FBUyxHQUFHLE9BQU8sQ0FBQyxrQkFBa0IsRUFBRSxDQUFDO1FBQy9DLFNBQVMsQ0FBQyxLQUFLLENBQUM7WUFDZCxjQUFjLEVBQUUsa0VBQWtFO1lBQ2xGLGlCQUFpQixFQUFFLENBQUM7U0FDckIsQ0FBQyxDQUFDO1FBRUgsU0FBUyxDQUFDLE1BQU0sQ0FBQztZQUNmLE9BQU8sRUFBRSxlQUFlO1lBQ3hCLE1BQU0sRUFBRSxHQUFHLEVBQUUsMkVBQTJFO1lBQ3hGLFdBQVcsRUFBRTtnQkFDWCxVQUFVLEVBQUUsZ0JBQWdCO2dCQUM1QixTQUFTLEVBQUUsUUFBUTtnQkFDbkIsUUFBUTtnQkFDUixXQUFXO2FBQ1o7U0FDRixDQUFDLENBQUM7UUFFSCxTQUFTLENBQUMsYUFBYSxDQUFDLGFBQWEsRUFBRSxVQUFVLENBQUMsUUFBUSxFQUFFLEVBQUUsY0FBYyxDQUFDLENBQUM7UUFDOUUsU0FBUyxDQUFDLEdBQUcsQ0FBQyxTQUFTLENBQUMsQ0FBQztRQUN6QixTQUFTLENBQUMsa0JBQWtCLEVBQUUsQ0FBQztRQUUvQixNQUFNLFNBQVMsQ0FBQyxLQUFLLEVBQUUsQ0FBQyxNQUFNLENBQUMsWUFBWSxDQUFDLGlFQUFpRSxDQUFDLENBQUM7SUFDakgsQ0FBQyxDQUFDLENBQUM7SUFFSCxFQUFFLENBQUMsbUNBQW1DLFNBQVMsc0RBQXNELEVBQUUsS0FBSyxJQUFJLEVBQUU7UUFDaEgsTUFBTSxRQUFRLEdBQUcsR0FBRyxDQUFDO1FBQ3JCLE1BQU0sVUFBVSxHQUFHLFFBQVEsQ0FBQztRQUM1QixNQUFNLGNBQWMsR0FBRztZQUNyQixDQUFDLFdBQVcsQ0FBQyxFQUFFO2dCQUNiLFFBQVEsRUFBRSxHQUFHO2dCQUNiLFNBQVMsRUFBRSxRQUFRO2dCQUNuQixVQUFVLEVBQUUsZ0JBQWdCO2FBQzdCO1NBQ0YsQ0FBQztRQUVGLE1BQU0sU0FBUyxHQUFHLE9BQU8sQ0FBQyxrQkFBa0IsRUFBRSxDQUFDO1FBQy9DLFNBQVMsQ0FBQyxLQUFLLENBQUM7WUFDZCxjQUFjLEVBQUUsa0VBQWtFO1lBQ2xGLGlCQUFpQixFQUFFLENBQUM7U0FDckIsQ0FBQyxDQUFDO1FBRUgsU0FBUyxDQUFDLE1BQU0sQ0FBQztZQUNmLE9BQU8sRUFBRSxlQUFlO1lBQ3hCLE1BQU0sRUFBRSxHQUFHLEVBQUUsMkVBQTJFO1lBQ3hGLFdBQVcsRUFBRTtnQkFDWCxVQUFVLEVBQUUsZ0JBQWdCO2dCQUM1QixTQUFTLEVBQUUsUUFBUTtnQkFDbkIsUUFBUTtnQkFDUixXQUFXO2FBQ1o7U0FDRixDQUFDLENBQUM7UUFFSCxTQUFTLENBQUMsYUFBYSxDQUFDLGFBQWEsRUFBRSxVQUFVLENBQUMsUUFBUSxFQUFFLEVBQUUsY0FBYyxDQUFDLENBQUM7UUFDOUUsU0FBUyxDQUFDLEdBQUcsQ0FBQyxTQUFTLENBQUMsQ0FBQztRQUN6QixTQUFTLENBQUMsa0JBQWtCLEVBQUUsQ0FBQztRQUUvQixNQUFNLFNBQVMsQ0FBQyxLQUFLLEVBQUUsQ0FBQyxNQUFNLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQyxRQUFRLEVBQUUsQ0FBQztJQUNuRCxDQUFDLENBQUMsQ0FBQztJQUVILEVBQUUsQ0FBQyw0RkFBNEYsRUFBRSxLQUFLLElBQUksRUFBRTtRQUMxRyxNQUFNLFVBQVUsR0FDZCw4R0FBOEcsQ0FBQztRQUNqSCxNQUFNLFFBQVEsR0FBRyxJQUFJLENBQUM7UUFDdEIsTUFBTSxrQkFBa0IsR0FBRyxPQUFPLENBQUM7UUFDbkMsTUFBTSxzQkFBc0IsR0FBRyxRQUFRLENBQUMsQ0FBQyx3Q0FBd0M7UUFDakYsTUFBTSxjQUFjLEdBQUc7WUFDckIsQ0FBQyxXQUFXLENBQUMsRUFBRTtnQkFDYixRQUFRLEVBQUUsS0FBSztnQkFDZixTQUFTLEVBQUUsUUFBUTtnQkFDbkIsVUFBVSxFQUFFLGdCQUFnQjthQUM3QjtTQUNGLENBQUM7UUFFRixNQUFNLFNBQVMsR0FBRyxPQUFPLENBQUMsa0JBQWtCLEVBQUUsQ0FBQztRQUMvQyw0QkFBNEI7UUFDNUIsU0FBUyxDQUFDLEtBQUssQ0FBQztZQUNkLGNBQWMsRUFBRSxrRUFBa0U7WUFDbEYsaUJBQWlCLEVBQUUsQ0FBQztTQUNyQixDQUFDLENBQUM7UUFDSCxnREFBZ0Q7UUFDaEQsU0FBUyxDQUFDLEtBQUssQ0FBQztZQUNkLGNBQWMsRUFBRSxrRUFBa0U7WUFDbEYsaUJBQWlCLEVBQUUsQ0FBQztTQUNyQixDQUFDLENBQUM7UUFFSCxTQUFTLENBQUMsTUFBTSxDQUFDO1lBQ2YsT0FBTyxFQUFFLGVBQWU7WUFDeEIsTUFBTSxFQUFFLEdBQUcsRUFBRSw2RUFBNkU7WUFDMUYsV0FBVyxFQUFFO2dCQUNYLFVBQVUsRUFBRSxnQkFBZ0I7Z0JBQzVCLFNBQVMsRUFBRSxRQUFRO2dCQUNuQixRQUFRO2dCQUNSLFdBQVc7YUFDWjtTQUNGLENBQUMsQ0FBQztRQUVILFNBQVMsQ0FBQyxhQUFhLENBQUMsYUFBYSxFQUFFLGtCQUFrQixDQUFDLFFBQVEsRUFBRSxFQUFFLGNBQWMsQ0FBQyxDQUFDO1FBQ3RGLFNBQVMsQ0FBQyxlQUFlLENBQUM7WUFDeEIsVUFBVSxFQUFFLFVBQVU7WUFDdEIsc0JBQXNCLEVBQUUsc0JBQXNCLENBQUMsUUFBUSxFQUFFO1NBQzFELENBQUMsQ0FBQztRQUNILFNBQVMsQ0FBQyxHQUFHLENBQUMsU0FBUyxDQUFDLENBQUM7UUFDekIsU0FBUyxDQUFDLGtCQUFrQixFQUFFLENBQUM7UUFDL0IsTUFBTSxFQUFFLEdBQUcsQ0FBQyxNQUFNLFNBQVMsQ0FBQyxLQUFLLEVBQUUsQ0FBZ0IsQ0FBQztRQUVwRCxnQkFBTSxDQUFDLEtBQUssQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLDBCQUFlLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDNUMsTUFBTSxNQUFNLEdBQUcsRUFBRSxDQUFDLE1BQU0sRUFBRSxDQUFDO1FBRTNCLE1BQU0sQ0FBQyxNQUFNLENBQUMsTUFBTSxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDckMsTUFBTSxDQUFDLE9BQU8sQ0FBQyxNQUFNLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUV0Qyx3RUFBd0U7UUFDeEUsTUFBTSxjQUFjLEdBQUcsTUFBTSxDQUFDLE9BQU8sQ0FBQyxNQUFNLENBQUMsQ0FBQyxNQUFNLEVBQUUsRUFBRSxDQUFDLE1BQU0sQ0FBQyxPQUFPLEtBQUssZUFBZSxDQUFDLENBQUM7UUFDN0YsY0FBYyxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQ3RDLGNBQWMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxNQUFNLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxTQUFTLENBQUMsQ0FBQyxDQUFDLDBEQUEwRDtRQUMzRyxjQUFjLENBQUMsQ0FBQyxDQUFDLENBQUMsV0FBdUM7YUFDdkQsU0FBUyxDQUFDLGdCQUFnQixFQUFFLFdBQVcsQ0FBQyxTQUFTLENBQUMsR0FBRyxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsZ0JBQWdCLEVBQUUsS0FBSyxDQUFDLENBQUMsQ0FBQzthQUM1RixNQUFNLEVBQUU7YUFDUixNQUFNLENBQUMsS0FBSyxDQUFDLFFBQVEsQ0FBQyxDQUFDO1FBRTFCLCtHQUErRztRQUMvRyxNQUFNLGtCQUFrQixHQUFHLE1BQU0sQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUFDLENBQUMsTUFBTSxFQUFFLEVBQUUsQ0FBQyxNQUFNLENBQUMsT0FBTyxLQUFLLGFBQWEsQ0FBQyxDQUFDO1FBQy9GLGtCQUFrQixDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsRUFBRSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUU3QyxxR0FBcUc7UUFDckcsTUFBTSxzQkFBc0IsR0FBRyxNQUFNLENBQUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxDQUFDLE1BQU0sRUFBRSxFQUFFLENBQUMsTUFBTSxDQUFDLE9BQU8sS0FBSyxVQUFVLENBQUMsQ0FBQztRQUNoRyxzQkFBc0IsQ0FBQyxNQUFNLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUM5QyxnREFBZ0Q7UUFDaEQsZ0JBQU0sQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLHNCQUFzQixDQUFDLENBQUMsQ0FBQyxDQUFDLFdBQVcsQ0FBQyxDQUFDO1FBRXhELEVBQUUsQ0FBQyxNQUFNLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxRQUFRLENBQUMsQ0FBQyxDQUFDLHlCQUF5QjtJQUM3RCxDQUFDLENBQUMsQ0FBQztJQUVILEVBQUUsQ0FBQyxxRUFBcUUsRUFBRSxLQUFLLElBQUksRUFBRTtRQUNuRixNQUFNLFVBQVUsR0FDZCw4R0FBOEcsQ0FBQztRQUNqSCxNQUFNLFFBQVEsR0FBRyxJQUFJLENBQUM7UUFDdEIsTUFBTSxrQkFBa0IsR0FBRyxPQUFPLENBQUM7UUFDbkMsTUFBTSxzQkFBc0IsR0FBRyxRQUFRLENBQUM7UUFDeEMsTUFBTSxjQUFjLEdBQUc7WUFDckIsQ0FBQyxXQUFXLENBQUMsRUFBRTtnQkFDYixRQUFRLEVBQUUsS0FBSztnQkFDZixTQUFTLEVBQUUsUUFBUTtnQkFDbkIsVUFBVSxFQUFFLGdCQUFnQjthQUM3QjtTQUNGLENBQUM7UUFFRixrREFBa0Q7UUFDbEQsTUFBTSxTQUFTLEdBQUcsT0FBTyxDQUFDLGtCQUFrQixFQUFFLENBQUM7UUFDL0MsU0FBUyxDQUFDLEtBQUssQ0FBQztZQUNkLGNBQWMsRUFBRSxrRUFBa0U7WUFDbEYsaUJBQWlCLEVBQUUsQ0FBQztTQUNyQixDQUFDLENBQUM7UUFDSCxTQUFTLENBQUMsS0FBSyxDQUFDO1lBQ2QsY0FBYyxFQUFFLGtFQUFrRTtZQUNsRixpQkFBaUIsRUFBRSxDQUFDO1NBQ3JCLENBQUMsQ0FBQztRQUVILFNBQVMsQ0FBQyxNQUFNLENBQUM7WUFDZixPQUFPLEVBQUUsZUFBZTtZQUN4QixNQUFNLEVBQUUsR0FBRztZQUNYLFdBQVcsRUFBRTtnQkFDWCxVQUFVLEVBQUUsZ0JBQWdCO2dCQUM1QixTQUFTLEVBQUUsUUFBUTtnQkFDbkIsUUFBUTtnQkFDUixXQUFXO2FBQ1o7U0FDRixDQUFDLENBQUM7UUFFSCxTQUFTLENBQUMsYUFBYSxDQUFDLGFBQWEsRUFBRSxrQkFBa0IsQ0FBQyxRQUFRLEVBQUUsRUFBRSxjQUFjLENBQUMsQ0FBQztRQUN0RixTQUFTLENBQUMsZUFBZSxDQUFDO1lBQ3hCLFVBQVUsRUFBRSxVQUFVO1lBQ3RCLHNCQUFzQixFQUFFLHNCQUFzQixDQUFDLFFBQVEsRUFBRTtTQUMxRCxDQUFDLENBQUM7UUFDSCxTQUFTLENBQUMsR0FBRyxDQUFDLFNBQVMsQ0FBQyxDQUFDO1FBQ3pCLFNBQVMsQ0FBQyxrQkFBa0IsRUFBRSxDQUFDO1FBQy9CLE1BQU0sU0FBUyxHQUFHLENBQUMsTUFBTSxTQUFTLENBQUMsS0FBSyxFQUFFLENBQWdCLENBQUM7UUFDM0QsTUFBTSxVQUFVLEdBQUcsU0FBUyxDQUFDLE1BQU0sQ0FBQztRQUNwQyxNQUFNLGFBQWEsR0FBRyxTQUFTLENBQUMsTUFBTSxFQUFFLENBQUM7UUFFekMsd0NBQXdDO1FBQ3hDLHdGQUF3RjtRQUN4RixNQUFNLGdCQUFnQixHQUFHLE9BQU8sQ0FBQyxrQkFBa0IsRUFBRSxDQUFDO1FBQ3RELGdCQUFnQixDQUFDLEtBQUssQ0FBQztZQUNyQixjQUFjLEVBQUUsa0VBQWtFO1lBQ2xGLGlCQUFpQixFQUFFLENBQUM7U0FDckIsQ0FBQyxDQUFDO1FBQ0gsZ0JBQWdCLENBQUMsS0FBSyxDQUFDO1lBQ3JCLGNBQWMsRUFBRSxrRUFBa0U7WUFDbEYsaUJBQWlCLEVBQUUsQ0FBQztTQUNyQixDQUFDLENBQUM7UUFFSCxnQkFBZ0IsQ0FBQyxNQUFNLENBQUM7WUFDdEIsT0FBTyxFQUFFLGVBQWU7WUFDeEIsTUFBTSxFQUFFLEdBQUc7WUFDWCxXQUFXLEVBQUU7Z0JBQ1gsVUFBVSxFQUFFLGdCQUFnQjtnQkFDNUIsU0FBUyxFQUFFLFFBQVE7Z0JBQ25CLFFBQVE7Z0JBQ1IsV0FBVzthQUNaO1NBQ0YsQ0FBQyxDQUFDO1FBRUgsZ0JBQWdCLENBQUMsYUFBYSxDQUFDLGFBQWEsRUFBRSxrQkFBa0IsQ0FBQyxRQUFRLEVBQUUsRUFBRSxjQUFjLENBQUMsQ0FBQztRQUM3RixnQkFBZ0IsQ0FBQyxlQUFlLENBQUM7WUFDL0IsVUFBVSxFQUFFLFVBQVU7WUFDdEIsc0JBQXNCLEVBQUUsc0JBQXNCLENBQUMsUUFBUSxFQUFFO1lBQ3pELFNBQVMsRUFBRSxJQUFJO1NBQ2hCLENBQUMsQ0FBQztRQUNILGdCQUFnQixDQUFDLEdBQUcsQ0FBQyxTQUFTLENBQUMsQ0FBQztRQUNoQyxnQkFBZ0IsQ0FBQyxrQkFBa0IsRUFBRSxDQUFDO1FBRXRDLE1BQU0sU0FBUyxHQUFHLENBQUMsTUFBTSxnQkFBZ0IsQ0FBQyxLQUFLLEVBQUUsQ0FBZ0IsQ0FBQztRQUNsRSxNQUFNLGFBQWEsR0FBRyxTQUFTLENBQUMsTUFBTSxFQUFFLENBQUM7UUFFekMsOERBQThEO1FBQzlELGFBQWEsQ0FBQyxNQUFNLENBQUMsTUFBTSxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsYUFBYSxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQztRQUN0RSxhQUFhLENBQUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLGFBQWEsQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUFDLENBQUM7UUFFeEUsd0RBQXdEO1FBQ3hELFNBQVMsQ0FBQyxNQUFNLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxVQUFVLENBQUMsQ0FBQztRQUUxQyxzQ0FBc0M7UUFDdEMsTUFBTSxjQUFjLEdBQUcsYUFBYSxDQUFDLE9BQU8sQ0FBQyxNQUFNLENBQUMsQ0FBQyxNQUFNLEVBQUUsRUFBRSxDQUFDLE1BQU0sQ0FBQyxPQUFPLEtBQUssZUFBZSxDQUFDLENBQUM7UUFDcEcsY0FBYyxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQ3RDLGNBQWMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxNQUFNLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxTQUFTLENBQUMsQ0FBQztJQUNuRCxDQUFDLENBQUMsQ0FBQztJQUVILFFBQVEsQ0FBQyw0QkFBNEIsRUFBRSxHQUFHLEVBQUU7UUFDMUMsSUFBSSxLQUFtQixDQUFDO1FBQ3hCLElBQUksUUFBUSxDQUFDO1FBRWIsTUFBTSxDQUFDO1lBQ0wsS0FBSyxHQUFHLG9CQUFTLENBQUMsUUFBUSxDQUFDLGtCQUFRLEVBQUUsRUFBRSxHQUFHLEVBQUUsTUFBTSxFQUFFLENBQUMsQ0FBQztZQUN0RCxLQUFLLENBQUMsa0JBQWtCLEVBQUUsQ0FBQztZQUMzQixNQUFNLFdBQVcsR0FBRztnQkFDbEIsSUFBSSxFQUFFLFlBQVk7Z0JBQ2xCLElBQUksRUFBRSxNQUFNO2dCQUNaLE9BQU8sRUFBRSxTQUFrQjtnQkFDM0IsSUFBSSxFQUFFLE9BQU87Z0JBQ2IsYUFBYSxFQUFFLENBQUM7Z0JBQ2hCLFFBQVEsRUFBRSxRQUFRO2dCQUNsQixTQUFTLEVBQUUsSUFBSSxFQUFFLGlDQUFpQztnQkFDbEQsZUFBZSxFQUFFLEdBQUcsUUFBUSxJQUFJLGdCQUFnQixFQUFFO2FBQ25ELENBQUM7WUFDRixRQUFRLEdBQUcsSUFBSSxjQUFRLENBQUMsS0FBSyxFQUFFLFdBQVcsQ0FBQyxDQUFDO1FBQzlDLENBQUMsQ0FBQyxDQUFDO1FBRUgsRUFBRSxDQUFDLDZEQUE2RCxFQUFFLEtBQUssSUFBSSxFQUFFO1lBQzNFLE1BQU0sUUFBUSxHQUFHLElBQUksQ0FBQztZQUN0QixNQUFNLFVBQVUsR0FBRyxRQUFRLENBQUM7WUFDNUIsTUFBTSxjQUFjLEdBQUc7Z0JBQ3JCLENBQUMsV0FBVyxDQUFDLEVBQUU7b0JBQ2IsUUFBUSxFQUFFLEtBQUs7b0JBQ2YsU0FBUyxFQUFFLFFBQVE7b0JBQ25CLFVBQVUsRUFBRSxnQkFBZ0I7aUJBQzdCO2FBQ0YsQ0FBQztZQUVGLE1BQU0sU0FBUyxHQUFHLE9BQU8sQ0FBQyxrQkFBa0IsRUFBRSxDQUFDO1lBQy9DLFNBQVMsQ0FBQyxLQUFLLENBQUM7Z0JBQ2QsY0FBYyxFQUFFLGtFQUFrRTtnQkFDbEYsaUJBQWlCLEVBQUUsQ0FBQzthQUNyQixDQUFDLENBQUM7WUFFSCxTQUFTLENBQUMsTUFBTSxDQUFDO2dCQUNmLE9BQU8sRUFBRSxlQUFlO2dCQUN4QixNQUFNLEVBQUUsR0FBRztnQkFDWCxXQUFXLEVBQUU7b0JBQ1gsVUFBVSxFQUFFLGdCQUFnQjtvQkFDNUIsU0FBUyxFQUFFLFFBQVE7b0JBQ25CLFFBQVE7b0JBQ1IsV0FBVztpQkFDWjthQUNGLENBQUMsQ0FBQztZQUVILFNBQVMsQ0FBQyxhQUFhLENBQUMsYUFBYSxFQUFFLFVBQVUsQ0FBQyxRQUFRLEVBQUUsRUFBRSxjQUFjLENBQUMsQ0FBQztZQUM5RSxTQUFTLENBQUMsR0FBRyxDQUFDLFNBQVMsQ0FBQyxDQUFDO1lBQ3pCLFNBQVMsQ0FBQyxrQkFBa0IsRUFBRSxDQUFDO1lBQy9CLE1BQU0sRUFBRSxHQUFHLENBQUMsTUFBTSxTQUFTLENBQUMsS0FBSyxFQUFFLENBQWdCLENBQUM7WUFDcEQsTUFBTSxLQUFLLEdBQUcsRUFBRSxDQUFDLGlCQUFpQixFQUFFLENBQUM7WUFFckMsK0NBQStDO1lBQy9DLE1BQU0sUUFBUSxHQUFHO2dCQUNmLFVBQVUsRUFBRTtvQkFDVjt3QkFDRSxPQUFPLEVBQUUsZUFBZTt3QkFDeEIsTUFBTSxFQUFFLFFBQVEsRUFBRSwrQkFBK0I7cUJBQ2xEO2lCQUNGO2FBQ0YsQ0FBQztZQUVGLE1BQU0sVUFBVSxHQUFHLEVBQUUsS0FBSyxFQUFFLENBQUM7WUFDN0IsTUFBTSxVQUFVLEdBQUcsRUFBRSxZQUFZLEVBQUUsR0FBRyxFQUFFLENBQUMsQ0FBQyxFQUFFLFdBQVcsRUFBRSxhQUFhLEVBQUUsQ0FBQyxFQUFFLENBQUM7WUFDNUUsTUFBTSxVQUFVLEdBQUcsTUFBTSxRQUFRLENBQUMsaUJBQWlCLENBQUMsRUFBRSxRQUFRLEVBQUUsVUFBVSxFQUFFLE1BQU0sRUFBRSxVQUFpQixFQUFFLENBQUMsQ0FBQztZQUN6RyxVQUFVLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUNoQyxDQUFDLENBQUMsQ0FBQztRQUVILEVBQUUsQ0FBQyx1RUFBdUUsRUFBRSxLQUFLLElBQUksRUFBRTtZQUNyRixNQUFNLFFBQVEsR0FBRyxJQUFJLENBQUM7WUFDdEIsTUFBTSxVQUFVLEdBQUcsUUFBUSxDQUFDO1lBQzVCLE1BQU0sY0FBYyxHQUFHO2dCQUNyQixDQUFDLFdBQVcsQ0FBQyxFQUFFO29CQUNiLFFBQVEsRUFBRSxLQUFLO29CQUNmLFNBQVMsRUFBRSxRQUFRO29CQUNuQixVQUFVLEVBQUUsZ0JBQWdCO2lCQUM3QjthQUNGLENBQUM7WUFFRixNQUFNLFNBQVMsR0FBRyxPQUFPLENBQUMsa0JBQWtCLEVBQUUsQ0FBQztZQUMvQyxTQUFTLENBQUMsS0FBSyxDQUFDO2dCQUNkLGNBQWMsRUFBRSxrRUFBa0U7Z0JBQ2xGLGlCQUFpQixFQUFFLENBQUM7YUFDckIsQ0FBQyxDQUFDO1lBRUgsU0FBUyxDQUFDLE1BQU0sQ0FBQztnQkFDZixPQUFPLEVBQUUsZUFBZTtnQkFDeEIsTUFBTSxFQUFFLEdBQUc7Z0JBQ1gsV0FBVyxFQUFFO29CQUNYLFVBQVUsRUFBRSxnQkFBZ0I7b0JBQzVCLFNBQVMsRUFBRSxRQUFRO29CQUNuQixRQUFRO29CQUNSLFdBQVc7aUJBQ1o7YUFDRixDQUFDLENBQUM7WUFFSCxTQUFTLENBQUMsYUFBYSxDQUFDLGFBQWEsRUFBRSxVQUFVLENBQUMsUUFBUSxFQUFFLEVBQUUsY0FBYyxDQUFDLENBQUM7WUFDOUUsU0FBUyxDQUFDLEdBQUcsQ0FBQyxTQUFTLENBQUMsQ0FBQztZQUN6QixTQUFTLENBQUMsa0JBQWtCLEVBQUUsQ0FBQztZQUMvQixNQUFNLEVBQUUsR0FBRyxDQUFDLE1BQU0sU0FBUyxDQUFDLEtBQUssRUFBRSxDQUFnQixDQUFDO1lBQ3BELE1BQU0sS0FBSyxHQUFHLEVBQUUsQ0FBQyxpQkFBaUIsRUFBRSxDQUFDO1lBRXJDLDJEQUEyRDtZQUMzRCxNQUFNLFFBQVEsR0FBRztnQkFDZixVQUFVLEVBQUU7b0JBQ1Y7d0JBQ0UsT0FBTyxFQUFFLGVBQWU7d0JBQ3hCLE1BQU0sRUFBRSxLQUFLLEVBQUUsZUFBZTtxQkFDL0I7aUJBQ0Y7YUFDRixDQUFDO1lBRUYsTUFBTSxVQUFVLEdBQUcsRUFBRSxLQUFLLEVBQUUsQ0FBQztZQUM3QixNQUFNLFVBQVUsR0FBRyxFQUFFLFlBQVksRUFBRSxHQUFHLEVBQUUsQ0FBQyxDQUFDLEVBQUUsV0FBVyxFQUFFLGFBQWEsRUFBRSxDQUFDLEVBQUUsQ0FBQztZQUM1RSxNQUFNLFFBQVE7aUJBQ1gsaUJBQWlCLENBQUMsRUFBRSxRQUFRLEVBQUUsVUFBVSxFQUFFLE1BQU0sRUFBRSxVQUFpQixFQUFFLENBQUM7aUJBQ3RFLE1BQU0sQ0FBQyxFQUFFLENBQUMsWUFBWSxDQUFDLDBDQUEwQyxDQUFDLENBQUM7UUFDeEUsQ0FBQyxDQUFDLENBQUM7UUFFSCxFQUFFLENBQUMsbURBQW1ELEVBQUUsS0FBSyxJQUFJLEVBQUU7WUFDakUsTUFBTSxRQUFRLEdBQUcsSUFBSSxDQUFDO1lBQ3RCLE1BQU0sVUFBVSxHQUFHLFFBQVEsQ0FBQztZQUM1QixNQUFNLGNBQWMsR0FBRztnQkFDckIsQ0FBQyxXQUFXLENBQUMsRUFBRTtvQkFDYixRQUFRLEVBQUUsS0FBSztvQkFDZixTQUFTLEVBQUUsUUFBUTtvQkFDbkIsVUFBVSxFQUFFLGdCQUFnQjtpQkFDN0I7YUFDRixDQUFDO1lBRUYsTUFBTSxTQUFTLEdBQUcsT0FBTyxDQUFDLGtCQUFrQixFQUFFLENBQUM7WUFDL0MsU0FBUyxDQUFDLEtBQUssQ0FBQztnQkFDZCxjQUFjLEVBQUUsa0VBQWtFO2dCQUNsRixpQkFBaUIsRUFBRSxDQUFDO2FBQ3JCLENBQUMsQ0FBQztZQUVILFNBQVMsQ0FBQyxNQUFNLENBQUM7Z0JBQ2YsT0FBTyxFQUFFLGVBQWU7Z0JBQ3hCLE1BQU0sRUFBRSxHQUFHO2dCQUNYLFdBQVcsRUFBRTtvQkFDWCxVQUFVLEVBQUUsZ0JBQWdCO29CQUM1QixTQUFTLEVBQUUsUUFBUTtvQkFDbkIsUUFBUTtvQkFDUixXQUFXO2lCQUNaO2FBQ0YsQ0FBQyxDQUFDO1lBRUgsU0FBUyxDQUFDLGFBQWEsQ0FBQyxhQUFhLEVBQUUsVUFBVSxDQUFDLFFBQVEsRUFBRSxFQUFFLGNBQWMsQ0FBQyxDQUFDO1lBQzlFLFNBQVMsQ0FBQyxHQUFHLENBQUMsU0FBUyxDQUFDLENBQUM7WUFDekIsU0FBUyxDQUFDLGtCQUFrQixFQUFFLENBQUM7WUFDL0IsTUFBTSxFQUFFLEdBQUcsQ0FBQyxNQUFNLFNBQVMsQ0FBQyxLQUFLLEVBQUUsQ0FBZ0IsQ0FBQztZQUNwRCxNQUFNLEtBQUssR0FBRyxFQUFFLENBQUMsaUJBQWlCLEVBQUUsQ0FBQztZQUVyQywwQ0FBMEM7WUFDMUMsTUFBTSxRQUFRLEdBQUc7Z0JBQ2YsVUFBVSxFQUFFO29CQUNWO3dCQUNFLE9BQU8sRUFDTCw4R0FBOEc7d0JBQ2hILE1BQU0sRUFBRSxRQUFRO3FCQUNqQjtpQkFDRjthQUNGLENBQUM7WUFFRixNQUFNLFVBQVUsR0FBRyxFQUFFLEtBQUssRUFBRSxDQUFDO1lBQzdCLE1BQU0sVUFBVSxHQUFHLEVBQUUsWUFBWSxFQUFFLEdBQUcsRUFBRSxDQUFDLENBQUMsRUFBRSxXQUFXLEVBQUUsYUFBYSxFQUFFLENBQUMsRUFBRSxDQUFDO1lBQzVFLE1BQU0sUUFBUTtpQkFDWCxpQkFBaUIsQ0FBQyxFQUFFLFFBQVEsRUFBRSxVQUFVLEVBQUUsTUFBTSxFQUFFLFVBQWlCLEVBQUUsQ0FBQztpQkFDdEUsTUFBTSxDQUFDLEVBQUUsQ0FBQyxZQUFZLENBQUMsMENBQTBDLENBQUMsQ0FBQztRQUN4RSxDQUFDLENBQUMsQ0FBQztRQUVILEVBQUUsQ0FBQyw0RkFBNEYsRUFBRSxLQUFLLElBQUksRUFBRTtZQUMxRyw4RkFBOEY7WUFDOUYsaURBQWlEO1lBQ2pELE1BQU0sb0JBQW9CLEdBQUcsUUFBUSxHQUFHLGdCQUFnQixDQUFDLENBQUMsNkVBQTZFO1lBRXZJLE1BQU0sbUNBQW1DLEdBQUc7Z0JBQzFDLElBQUksRUFBRSxZQUFZO2dCQUNsQixJQUFJLEVBQUUsTUFBTTtnQkFDWixPQUFPLEVBQUUsU0FBa0I7Z0JBQzNCLElBQUksRUFBRSxPQUFPO2dCQUNiLGFBQWEsRUFBRSxDQUFDO2dCQUNoQixRQUFRLEVBQUUsb0JBQW9CLEVBQUUsMkJBQTJCO2dCQUMzRCxTQUFTLEVBQUUsSUFBSSxFQUFFLHVDQUF1QztnQkFDeEQsZUFBZSxFQUFFLEdBQUcsUUFBUSxJQUFJLGdCQUFnQixFQUFFO2FBQ25ELENBQUM7WUFDRixNQUFNLGdDQUFnQyxHQUFHLElBQUksY0FBUSxDQUFDLEtBQUssRUFBRSxtQ0FBbUMsQ0FBQyxDQUFDO1lBRWxHLE1BQU0sUUFBUSxHQUFHLElBQUksQ0FBQztZQUN0QixNQUFNLFVBQVUsR0FBRyxRQUFRLENBQUM7WUFDNUIsTUFBTSxjQUFjLEdBQUc7Z0JBQ3JCLENBQUMsV0FBVyxDQUFDLEVBQUU7b0JBQ2IsUUFBUSxFQUFFLEtBQUs7b0JBQ2YsU0FBUyxFQUFFLFFBQVE7b0JBQ25CLFVBQVUsRUFBRSxnQkFBZ0I7aUJBQzdCO2FBQ0YsQ0FBQztZQUVGLE1BQU0sU0FBUyxHQUFHLE9BQU8sQ0FBQyxrQkFBa0IsRUFBRSxDQUFDO1lBQy9DLFNBQVMsQ0FBQyxLQUFLLENBQUM7Z0JBQ2QsY0FBYyxFQUFFLGtFQUFrRTtnQkFDbEYsaUJBQWlCLEVBQUUsQ0FBQzthQUNyQixDQUFDLENBQUM7WUFFSCxTQUFTLENBQUMsTUFBTSxDQUFDO2dCQUNmLE9BQU8sRUFBRSxlQUFlO2dCQUN4QixNQUFNLEVBQUUsR0FBRztnQkFDWCxXQUFXLEVBQUU7b0JBQ1gsVUFBVSxFQUFFLGdCQUFnQjtvQkFDNUIsU0FBUyxFQUFFLFFBQVE7b0JBQ25CLFFBQVE7b0JBQ1IsV0FBVztpQkFDWjthQUNGLENBQUMsQ0FBQztZQUVILFNBQVMsQ0FBQyxhQUFhLENBQUMsYUFBYSxFQUFFLFVBQVUsQ0FBQyxRQUFRLEVBQUUsRUFBRSxjQUFjLENBQUMsQ0FBQztZQUM5RSxTQUFTLENBQUMsR0FBRyxDQUFDLFNBQVMsQ0FBQyxDQUFDO1lBQ3pCLFNBQVMsQ0FBQyxrQkFBa0IsRUFBRSxDQUFDO1lBQy9CLE1BQU0sRUFBRSxHQUFHLENBQUMsTUFBTSxTQUFTLENBQUMsS0FBSyxFQUFFLENBQWdCLENBQUM7WUFDcEQsTUFBTSxLQUFLLEdBQUcsRUFBRSxDQUFDLGlCQUFpQixFQUFFLENBQUM7WUFFckMsc0ZBQXNGO1lBQ3RGLE1BQU0sUUFBUSxHQUFHO2dCQUNmLFVBQVUsRUFBRTtvQkFDVjt3QkFDRSxPQUFPLEVBQUUsZUFBZTt3QkFDeEIsTUFBTSxFQUFFLFFBQVE7cUJBQ2pCO2lCQUNGO2FBQ0YsQ0FBQztZQUVGLE1BQU0sVUFBVSxHQUFHLEVBQUUsS0FBSyxFQUFFLENBQUM7WUFDN0IsTUFBTSxVQUFVLEdBQUcsRUFBRSxZQUFZLEVBQUUsR0FBRyxFQUFFLENBQUMsQ0FBQyxFQUFFLFdBQVcsRUFBRSxhQUFhLEVBQUUsQ0FBQyxFQUFFLENBQUM7WUFDNUUsTUFBTSxVQUFVLEdBQUcsTUFBTSxnQ0FBZ0MsQ0FBQyxpQkFBaUIsQ0FBQztnQkFDMUUsUUFBUTtnQkFDUixVQUFVO2dCQUNWLE1BQU0sRUFBRSxVQUFpQjthQUMxQixDQUFDLENBQUM7WUFDSCxVQUFVLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUNoQyxDQUFDLENBQUMsQ0FBQztRQUVILEVBQUUsQ0FBQyxtRkFBbUYsRUFBRSxLQUFLLElBQUksRUFBRTtZQUNqRywyRkFBMkY7WUFDM0YsTUFBTSw0QkFBNEIsR0FBRztnQkFDbkMsSUFBSSxFQUFFLFlBQVk7Z0JBQ2xCLElBQUksRUFBRSxNQUFNO2dCQUNaLE9BQU8sRUFBRSxTQUFrQjtnQkFDM0IsSUFBSSxFQUFFLE9BQU87Z0JBQ2IsYUFBYSxFQUFFLENBQUM7Z0JBQ2hCLFFBQVEsRUFBRSxRQUFRLEVBQUUsdUNBQXVDO2dCQUMzRCxTQUFTLEVBQUUsSUFBSSxFQUFFLHFCQUFxQjtnQkFDdEMsZUFBZSxFQUFFLEdBQUcsUUFBUSxJQUFJLGdCQUFnQixFQUFFO2FBQ25ELENBQUM7WUFDRixNQUFNLHlCQUF5QixHQUFHLElBQUksY0FBUSxDQUFDLEtBQUssRUFBRSw0QkFBNEIsQ0FBQyxDQUFDO1lBRXBGLE1BQU0sUUFBUSxHQUFHLElBQUksQ0FBQztZQUN0QixNQUFNLFVBQVUsR0FBRyxRQUFRLENBQUM7WUFDNUIsTUFBTSxjQUFjLEdBQUc7Z0JBQ3JCLENBQUMsV0FBVyxDQUFDLEVBQUU7b0JBQ2IsUUFBUSxFQUFFLEtBQUs7b0JBQ2YsU0FBUyxFQUFFLFFBQVE7b0JBQ25CLFVBQVUsRUFBRSxnQkFBZ0I7aUJBQzdCO2FBQ0YsQ0FBQztZQUVGLE1BQU0sU0FBUyxHQUFHLE9BQU8sQ0FBQyxrQkFBa0IsRUFBRSxDQUFDO1lBQy9DLFNBQVMsQ0FBQyxLQUFLLENBQUM7Z0JBQ2QsY0FBYyxFQUFFLGtFQUFrRTtnQkFDbEYsaUJBQWlCLEVBQUUsQ0FBQzthQUNyQixDQUFDLENBQUM7WUFFSCxTQUFTLENBQUMsTUFBTSxDQUFDO2dCQUNmLE9BQU8sRUFBRSxlQUFlO2dCQUN4QixNQUFNLEVBQUUsR0FBRztnQkFDWCxXQUFXLEVBQUU7b0JBQ1gsVUFBVSxFQUFFLGdCQUFnQjtvQkFDNUIsU0FBUyxFQUFFLFFBQVE7b0JBQ25CLFFBQVE7b0JBQ1IsV0FBVztpQkFDWjthQUNGLENBQUMsQ0FBQztZQUVILFNBQVMsQ0FBQyxhQUFhLENBQUMsYUFBYSxFQUFFLFVBQVUsQ0FBQyxRQUFRLEVBQUUsRUFBRSxjQUFjLENBQUMsQ0FBQztZQUM5RSxTQUFTLENBQUMsR0FBRyxDQUFDLFNBQVMsQ0FBQyxDQUFDO1lBQ3pCLFNBQVMsQ0FBQyxrQkFBa0IsRUFBRSxDQUFDO1lBQy9CLE1BQU0sRUFBRSxHQUFHLENBQUMsTUFBTSxTQUFTLENBQUMsS0FBSyxFQUFFLENBQWdCLENBQUM7WUFDcEQsTUFBTSxLQUFLLEdBQUcsRUFBRSxDQUFDLGlCQUFpQixFQUFFLENBQUM7WUFFckMsK0RBQStEO1lBQy9ELE1BQU0sUUFBUSxHQUFHO2dCQUNmLFVBQVUsRUFBRTtvQkFDVjt3QkFDRSxPQUFPLEVBQUUsZUFBZTt3QkFDeEIsTUFBTSxFQUFFLFFBQVE7cUJBQ2pCO2lCQUNGO2FBQ0YsQ0FBQztZQUVGLE1BQU0sVUFBVSxHQUFHLEVBQUUsS0FBSyxFQUFFLENBQUM7WUFDN0IsTUFBTSxVQUFVLEdBQUcsRUFBRSxZQUFZLEVBQUUsR0FBRyxFQUFFLENBQUMsQ0FBQyxFQUFFLFdBQVcsRUFBRSxhQUFhLEVBQUUsQ0FBQyxFQUFFLENBQUM7WUFDNUUsTUFBTSxVQUFVLEdBQUcsTUFBTSx5QkFBeUIsQ0FBQyxpQkFBaUIsQ0FBQztnQkFDbkUsUUFBUTtnQkFDUixVQUFVO2dCQUNWLE1BQU0sRUFBRSxVQUFpQjthQUMxQixDQUFDLENBQUM7WUFDSCxVQUFVLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUNoQyxDQUFDLENBQUMsQ0FBQztRQUVILEVBQUUsQ0FBQyxtRkFBbUYsRUFBRSxLQUFLLElBQUksRUFBRTtZQUNqRyxNQUFNLFFBQVEsR0FBRyxLQUFLLENBQUM7WUFDdkIsTUFBTSxVQUFVLEdBQUcsUUFBUSxDQUFDO1lBQzVCLE1BQU0sY0FBYyxHQUFHO2dCQUNyQixDQUFDLFdBQVcsQ0FBQyxFQUFFO29CQUNiLFFBQVEsRUFBRSxLQUFLO29CQUNmLFNBQVMsRUFBRSxRQUFRO29CQUNuQixVQUFVLEVBQUUsZ0JBQWdCO2lCQUM3QjthQUNGLENBQUM7WUFFRiw4RUFBOEU7WUFDOUUsTUFBTSxTQUFTLEdBQUcsT0FBTyxDQUFDLGtCQUFrQixFQUFFLENBQUM7WUFDL0MsU0FBUyxDQUFDLEtBQUssQ0FBQztnQkFDZCxjQUFjLEVBQUUsa0VBQWtFO2dCQUNsRixpQkFBaUIsRUFBRSxDQUFDO2FBQ3JCLENBQUMsQ0FBQztZQUVILGlFQUFpRTtZQUNqRSxTQUFTLENBQUMsTUFBTSxDQUFDO2dCQUNmLE9BQU8sRUFBRSxhQUFhO2dCQUN0QixNQUFNLEVBQUUsR0FBRztnQkFDWCxXQUFXLEVBQUU7b0JBQ1gsVUFBVSxFQUFFLGdCQUFnQjtvQkFDNUIsU0FBUyxFQUFFLFFBQVE7b0JBQ25CLFFBQVE7b0JBQ1IsV0FBVztpQkFDWjthQUNGLENBQUMsQ0FBQztZQUVILFNBQVMsQ0FBQyxhQUFhLENBQUMsYUFBYSxFQUFFLFVBQVUsQ0FBQyxRQUFRLEVBQUUsRUFBRSxjQUFjLENBQUMsQ0FBQztZQUM5RSxTQUFTLENBQUMsR0FBRyxDQUFDLFNBQVMsQ0FBQyxDQUFDO1lBQ3pCLFNBQVMsQ0FBQyxrQkFBa0IsRUFBRSxDQUFDO1lBQy9CLE1BQU0sRUFBRSxHQUFHLENBQUMsTUFBTSxTQUFTLENBQUMsS0FBSyxFQUFFLENBQWdCLENBQUM7WUFDcEQsTUFBTSxLQUFLLEdBQUcsRUFBRSxDQUFDLGlCQUFpQixFQUFFLENBQUM7WUFFckMsdURBQXVEO1lBQ3ZELE1BQU0sVUFBVSxHQUFHO2dCQUNqQixZQUFZLEVBQUUsR0FBRyxFQUFFLENBQUMsQ0FBQztvQkFDbkIsV0FBVyxFQUFFLGFBQWE7aUJBQzNCLENBQUM7YUFDSCxDQUFDO1lBRUYsMkZBQTJGO1lBQzNGLE1BQU0sUUFBUSxHQUFHO2dCQUNmLFVBQVUsRUFBRSxTQUFTO2FBQ3RCLENBQUM7WUFFRixNQUFNLFVBQVUsR0FBRyxFQUFFLEtBQUssRUFBRSxDQUFDO1lBQzdCLE1BQU0sWUFBWSxHQUFHLEVBQUUsMEJBQTBCLEVBQUUsSUFBSSxFQUFFLENBQUM7WUFFMUQsTUFBTSxVQUFVLEdBQUcsTUFBTSxRQUFRLENBQUMsaUJBQWlCLENBQUM7Z0JBQ2xELFFBQVE7Z0JBQ1IsVUFBVTtnQkFDVixZQUFZO2dCQUNaLE1BQU0sRUFBRSxVQUFpQjthQUMxQixDQUFDLENBQUM7WUFDSCxVQUFVLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUNoQyxDQUFDLENBQUMsQ0FBQztRQUVILEVBQUUsQ0FBQyxpRkFBaUYsRUFBRSxLQUFLLElBQUksRUFBRTtZQUMvRixNQUFNLFFBQVEsR0FBRyxLQUFLLENBQUM7WUFDdkIsTUFBTSxVQUFVLEdBQUcsUUFBUSxDQUFDO1lBQzVCLE1BQU0sY0FBYyxHQUFHO2dCQUNyQixDQUFDLFdBQVcsQ0FBQyxFQUFFO29CQUNiLFFBQVEsRUFBRSxLQUFLO29CQUNmLFNBQVMsRUFBRSxRQUFRO29CQUNuQixVQUFVLEVBQUUsZ0JBQWdCO2lCQUM3QjthQUNGLENBQUM7WUFFRixpRUFBaUU7WUFDakUsTUFBTSxTQUFTLEdBQUcsT0FBTyxDQUFDLGtCQUFrQixFQUFFLENBQUM7WUFDL0MsU0FBUyxDQUFDLEtBQUssQ0FBQztnQkFDZCxjQUFjLEVBQUUsa0VBQWtFO2dCQUNsRixpQkFBaUIsRUFBRSxDQUFDO2FBQ3JCLENBQUMsQ0FBQztZQUVILFNBQVMsQ0FBQyxNQUFNLENBQUM7Z0JBQ2YsT0FBTyxFQUFFLGVBQWUsRUFBRSw0Q0FBNEM7Z0JBQ3RFLE1BQU0sRUFBRSxHQUFHO2dCQUNYLFdBQVcsRUFBRTtvQkFDWCxVQUFVLEVBQUUsZ0JBQWdCO29CQUM1QixTQUFTLEVBQUUsUUFBUTtvQkFDbkIsUUFBUTtvQkFDUixXQUFXO2lCQUNaO2FBQ0YsQ0FBQyxDQUFDO1lBRUgsU0FBUyxDQUFDLGFBQWEsQ0FBQyxhQUFhLEVBQUUsVUFBVSxDQUFDLFFBQVEsRUFBRSxFQUFFLGNBQWMsQ0FBQyxDQUFDO1lBQzlFLFNBQVMsQ0FBQyxHQUFHLENBQUMsU0FBUyxDQUFDLENBQUM7WUFDekIsU0FBUyxDQUFDLGtCQUFrQixFQUFFLENBQUM7WUFDL0IsTUFBTSxFQUFFLEdBQUcsQ0FBQyxNQUFNLFNBQVMsQ0FBQyxLQUFLLEVBQUUsQ0FBZ0IsQ0FBQztZQUNwRCxNQUFNLEtBQUssR0FBRyxFQUFFLENBQUMsaUJBQWlCLEVBQUUsQ0FBQztZQUVyQywwQ0FBMEM7WUFDMUMsTUFBTSxVQUFVLEdBQUc7Z0JBQ2pCLFlBQVksRUFBRSxHQUFHLEVBQUUsQ0FBQyxDQUFDO29CQUNuQixXQUFXLEVBQUUsYUFBYSxFQUFFLHNEQUFzRDtpQkFDbkYsQ0FBQzthQUNILENBQUM7WUFFRixNQUFNLFFBQVEsR0FBRztnQkFDZixVQUFVLEVBQUUsU0FBUzthQUN0QixDQUFDO1lBRUYsTUFBTSxVQUFVLEdBQUcsRUFBRSxLQUFLLEVBQUUsQ0FBQztZQUM3QixNQUFNLFlBQVksR0FBRyxFQUFFLDBCQUEwQixFQUFFLElBQUksRUFBRSxDQUFDO1lBRTFELE1BQU0sUUFBUTtpQkFDWCxpQkFBaUIsQ0FBQztnQkFDakIsUUFBUTtnQkFDUixVQUFVO2dCQUNWLFlBQVk7Z0JBQ1osTUFBTSxFQUFFLFVBQWlCO2FBQzFCLENBQUM7aUJBQ0QsTUFBTSxDQUFDLEVBQUUsQ0FBQyxZQUFZLENBQUMsaURBQWlELENBQUMsQ0FBQztRQUMvRSxDQUFDLENBQUMsQ0FBQztRQUVILEVBQUUsQ0FBQyxnRkFBZ0YsRUFBRSxLQUFLLElBQUksRUFBRTtZQUM5RixNQUFNLFVBQVUsR0FDZCw4R0FBOEcsQ0FBQztZQUNqSCxNQUFNLFFBQVEsR0FBRyxLQUFLLENBQUM7WUFDdkIsTUFBTSxrQkFBa0IsR0FBRyxPQUFPLENBQUM7WUFDbkMsTUFBTSxzQkFBc0IsR0FBRyxRQUFRLENBQUM7WUFDeEMsTUFBTSxjQUFjLEdBQUc7Z0JBQ3JCLENBQUMsV0FBVyxDQUFDLEVBQUU7b0JBQ2IsUUFBUSxFQUFFLEtBQUs7b0JBQ2YsU0FBUyxFQUFFLFFBQVE7b0JBQ25CLFVBQVUsRUFBRSxnQkFBZ0I7aUJBQzdCO2FBQ0YsQ0FBQztZQUVGLHNHQUFzRztZQUN0RyxNQUFNLFNBQVMsR0FBRyxPQUFPLENBQUMsa0JBQWtCLEVBQUUsQ0FBQztZQUMvQyxTQUFTLENBQUMsS0FBSyxDQUFDO2dCQUNkLGNBQWMsRUFBRSxrRUFBa0U7Z0JBQ2xGLGlCQUFpQixFQUFFLENBQUM7YUFDckIsQ0FBQyxDQUFDO1lBQ0gsU0FBUyxDQUFDLEtBQUssQ0FBQztnQkFDZCxjQUFjLEVBQUUsa0VBQWtFO2dCQUNsRixpQkFBaUIsRUFBRSxDQUFDO2FBQ3JCLENBQUMsQ0FBQztZQUVILG9FQUFvRTtZQUNwRSxTQUFTLENBQUMsTUFBTSxDQUFDO2dCQUNmLE9BQU8sRUFBRSxhQUFhO2dCQUN0QixNQUFNLEVBQUUsR0FBRztnQkFDWCxXQUFXLEVBQUU7b0JBQ1gsVUFBVSxFQUFFLGdCQUFnQjtvQkFDNUIsU0FBUyxFQUFFLFFBQVE7b0JBQ25CLFFBQVE7b0JBQ1IsV0FBVztpQkFDWjthQUNGLENBQUMsQ0FBQztZQUVILFNBQVMsQ0FBQyxhQUFhLENBQUMsYUFBYSxFQUFFLGtCQUFrQixDQUFDLFFBQVEsRUFBRSxFQUFFLGNBQWMsQ0FBQyxDQUFDO1lBQ3RGLFNBQVMsQ0FBQyxlQUFlLENBQUM7Z0JBQ3hCLFVBQVUsRUFBRSxVQUFVO2dCQUN0QixzQkFBc0IsRUFBRSxzQkFBc0IsQ0FBQyxRQUFRLEVBQUU7YUFDMUQsQ0FBQyxDQUFDO1lBQ0gsU0FBUyxDQUFDLEdBQUcsQ0FBQyxTQUFTLENBQUMsQ0FBQztZQUN6QixTQUFTLENBQUMsa0JBQWtCLEVBQUUsQ0FBQztZQUMvQixNQUFNLEVBQUUsR0FBRyxDQUFDLE1BQU0sU0FBUyxDQUFDLEtBQUssRUFBRSxDQUFnQixDQUFDO1lBQ3BELE1BQU0sS0FBSyxHQUFHLEVBQUUsQ0FBQyxpQkFBaUIsRUFBRSxDQUFDO1lBRXJDLE1BQU0sVUFBVSxHQUFHO2dCQUNqQixZQUFZLEVBQUUsR0FBRyxFQUFFLENBQUMsQ0FBQztvQkFDbkIsV0FBVyxFQUFFLGFBQWE7aUJBQzNCLENBQUM7YUFDSCxDQUFDO1lBRUYsTUFBTSxRQUFRLEdBQUcsRUFBRSxVQUFVLEVBQUUsU0FBUyxFQUFFLENBQUM7WUFDM0MsTUFBTSxVQUFVLEdBQUcsRUFBRSxLQUFLLEVBQUUsTUFBTSxFQUFFLEVBQUUsVUFBVSxFQUFFLEVBQUUsQ0FBQztZQUNyRCxNQUFNLFlBQVksR0FBRyxFQUFFLDBCQUEwQixFQUFFLElBQUksRUFBRSxDQUFDO1lBRTFELE1BQU0sVUFBVSxHQUFHLE1BQU0sUUFBUSxDQUFDLGlCQUFpQixDQUFDO2dCQUNsRCxRQUFRO2dCQUNSLFVBQVU7Z0JBQ1YsWUFBWTtnQkFDWixNQUFNLEVBQUUsVUFBaUI7YUFDMUIsQ0FBQyxDQUFDO1lBQ0gsVUFBVSxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDaEMsQ0FBQyxDQUFDLENBQUM7UUFFSCxFQUFFLENBQUMsOEZBQThGLEVBQUUsS0FBSyxJQUFJLEVBQUU7WUFDNUcsTUFBTSxVQUFVLEdBQ2QsOEdBQThHLENBQUM7WUFDakgsTUFBTSxRQUFRLEdBQUcsS0FBSyxDQUFDO1lBQ3ZCLE1BQU0sa0JBQWtCLEdBQUcsT0FBTyxDQUFDO1lBQ25DLE1BQU0sc0JBQXNCLEdBQUcsUUFBUSxDQUFDO1lBQ3hDLE1BQU0sY0FBYyxHQUFHO2dCQUNyQixDQUFDLFdBQVcsQ0FBQyxFQUFFO29CQUNiLFFBQVEsRUFBRSxLQUFLO29CQUNmLFNBQVMsRUFBRSxRQUFRO29CQUNuQixVQUFVLEVBQUUsZ0JBQWdCO2lCQUM3QjthQUNGLENBQUM7WUFFRixpRkFBaUY7WUFDakYsTUFBTSxTQUFTLEdBQUcsT0FBTyxDQUFDLGtCQUFrQixFQUFFLENBQUM7WUFDL0MsU0FBUyxDQUFDLEtBQUssQ0FBQztnQkFDZCxjQUFjLEVBQUUsa0VBQWtFO2dCQUNsRixpQkFBaUIsRUFBRSxDQUFDO2FBQ3JCLENBQUMsQ0FBQztZQUNILFNBQVMsQ0FBQyxLQUFLLENBQUM7Z0JBQ2QsY0FBYyxFQUFFLGtFQUFrRTtnQkFDbEYsaUJBQWlCLEVBQUUsQ0FBQzthQUNyQixDQUFDLENBQUM7WUFFSCxxRUFBcUU7WUFDckUsU0FBUyxDQUFDLE1BQU0sQ0FBQztnQkFDZixPQUFPLEVBQUUsZUFBZTtnQkFDeEIsTUFBTSxFQUFFLEdBQUc7Z0JBQ1gsV0FBVyxFQUFFO29CQUNYLFVBQVUsRUFBRSxnQkFBZ0I7b0JBQzVCLFNBQVMsRUFBRSxRQUFRO29CQUNuQixRQUFRO29CQUNSLFdBQVc7aUJBQ1o7YUFDRixDQUFDLENBQUM7WUFFSCxTQUFTLENBQUMsYUFBYSxDQUFDLGFBQWEsRUFBRSxrQkFBa0IsQ0FBQyxRQUFRLEVBQUUsRUFBRSxjQUFjLENBQUMsQ0FBQztZQUN0RixTQUFTLENBQUMsZUFBZSxDQUFDO2dCQUN4QixVQUFVLEVBQUUsVUFBVTtnQkFDdEIsc0JBQXNCLEVBQUUsc0JBQXNCLENBQUMsUUFBUSxFQUFFO2FBQzFELENBQUMsQ0FBQztZQUNILFNBQVMsQ0FBQyxHQUFHLENBQUMsU0FBUyxDQUFDLENBQUM7WUFDekIsU0FBUyxDQUFDLGtCQUFrQixFQUFFLENBQUM7WUFDL0IsTUFBTSxFQUFFLEdBQUcsQ0FBQyxNQUFNLFNBQVMsQ0FBQyxLQUFLLEVBQUUsQ0FBZ0IsQ0FBQztZQUNwRCxNQUFNLEtBQUssR0FBRyxFQUFFLENBQUMsaUJBQWlCLEVBQUUsQ0FBQztZQUVyQyxNQUFNLFVBQVUsR0FBRztnQkFDakIsWUFBWSxFQUFFLEdBQUcsRUFBRSxDQUFDLENBQUM7b0JBQ25CLFdBQVcsRUFBRSxhQUFhO2lCQUMzQixDQUFDO2FBQ0gsQ0FBQztZQUVGLE1BQU0sUUFBUSxHQUFHLEVBQUUsVUFBVSxFQUFFLFNBQVMsRUFBRSxDQUFDO1lBQzNDLE1BQU0sVUFBVSxHQUFHLEVBQUUsS0FBSyxFQUFFLE1BQU0sRUFBRSxFQUFFLFVBQVUsRUFBRSxFQUFFLENBQUM7WUFDckQsTUFBTSxZQUFZLEdBQUcsRUFBRSwwQkFBMEIsRUFBRSxJQUFJLEVBQUUsQ0FBQztZQUUxRCxNQUFNLFFBQVE7aUJBQ1gsaUJBQWlCLENBQUM7Z0JBQ2pCLFFBQVE7Z0JBQ1IsVUFBVTtnQkFDVixZQUFZO2dCQUNaLE1BQU0sRUFBRSxVQUFpQjthQUMxQixDQUFDO2lCQUNELE1BQU0sQ0FBQyxFQUFFLENBQUMsWUFBWSxDQUFDLGlEQUFpRCxDQUFDLENBQUM7UUFDL0UsQ0FBQyxDQUFDLENBQUM7SUFDTCxDQUFDLENBQUMsQ0FBQztBQUNMLENBQUMsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHNob3VsZCBmcm9tICdzaG91bGQnO1xuaW1wb3J0IHsgVHJhbnNhY3Rpb25UeXBlIH0gZnJvbSAnQGJpdGdvLWJldGEvc2RrLWNvcmUnO1xuaW1wb3J0ICogYXMgdGVzdERhdGEgZnJvbSAnLi4vcmVzb3VyY2VzJztcbmltcG9ydCB7IFRyYW5zYWN0aW9uQnVpbGRlckZhY3RvcnksIEFkYVRva2VuIH0gZnJvbSAnLi4vLi4vc3JjJztcbmltcG9ydCB7IGNvaW5zIH0gZnJvbSAnQGJpdGdvLWJldGEvc3RhdGljcyc7XG5pbXBvcnQgKiBhcyBDYXJkYW5vV2FzbSBmcm9tICdAZW11cmdvL2NhcmRhbm8tc2VyaWFsaXphdGlvbi1saWItbm9kZWpzJztcbmltcG9ydCB7IFRyYW5zYWN0aW9uIH0gZnJvbSAnLi4vLi4vc3JjL2xpYi90cmFuc2FjdGlvbic7XG5pbXBvcnQgeyBUZXN0Qml0R28sIFRlc3RCaXRHb0FQSSB9IGZyb20gJ0BiaXRnby1iZXRhL3Nkay10ZXN0JztcbmltcG9ydCB7IEJpdEdvQVBJIH0gZnJvbSAnQGJpdGdvLWJldGEvc2RrLWFwaSc7XG5cbmRlc2NyaWJlKCdBREEgVG9rZW4gT3BlcmF0aW9ucycsIGFzeW5jICgpID0+IHtcbiAgY29uc3QgZmFjdG9yeSA9IG5ldyBUcmFuc2FjdGlvbkJ1aWxkZXJGYWN0b3J5KGNvaW5zLmdldCgndGFkYScpKTtcblxuICBjb25zdCByZWNlaXZlckFkZHJlc3MgPSB0ZXN0RGF0YS5yYXdUeC5vdXRwdXRBZGRyZXNzMi5hZGRyZXNzO1xuICBjb25zdCBzZW5kZXJBZGRyZXNzID0gdGVzdERhdGEucmF3VHgub3V0cHV0QWRkcmVzczEuYWRkcmVzcztcblxuICBjb25zdCBwb2xpY3lJZCA9ICdlMTZjMmRjOGFlOTM3ZThkMzc5MGM3ZmQ3MTY4ZDdiOTk0NjIxYmExNGNhMTE0MTVmMzlmZWQ3Mic7XG4gIGNvbnN0IHBvbGljeVNjcmlwdEhhc2ggPSBDYXJkYW5vV2FzbS5TY3JpcHRIYXNoLmZyb21faGV4KHBvbGljeUlkKTtcbiAgY29uc3QgYXNzZXROYW1lID0gJ3RhZGE6d2F0ZXInO1xuICBjb25zdCBuYW1lID0gJ1dBVEVSJztcbiAgY29uc3QgYXNjaWlFbmNvZGVkTmFtZSA9IEJ1ZmZlci5mcm9tKG5hbWUsICdhc2NpaScpLnRvU3RyaW5nKCdoZXgnKTtcbiAgY29uc3QgZmluZ2VycHJpbnQgPSAnYXNzZXQxbjY5eGY2MGQwNzYweHZuOHYyZmZkNWZydnNtMGNsMnI4aGZqZjYnO1xuXG4gIGNvbnN0IHVuc3VwcG9ydGVkUG9saWN5SWQgPSAnMjc5YzkwOWYzNDhlNTMzZGE1ODA4ODk4Zjg3ZjlhMTRiYjJjM2RmYmJhY2NjZDYzMWQ5MjdhM2YnO1xuICBjb25zdCB1bnN1cHBvcnRlZFBvbGljeVNjcmlwdEhhc2ggPSBDYXJkYW5vV2FzbS5TY3JpcHRIYXNoLmZyb21faGV4KHVuc3VwcG9ydGVkUG9saWN5SWQpO1xuICBjb25zdCB1bnN1cHBvcnRlZE5hbWUgPSAnQklUR08nO1xuICBjb25zdCB1bnN1cHBvcnRlZEFzY2lpRW5jb2RlZE5hbWUgPSBCdWZmZXIuZnJvbSh1bnN1cHBvcnRlZE5hbWUsICdhc2NpaScpLnRvU3RyaW5nKCdoZXgnKTtcbiAgY29uc3QgdW5zdXBwb3J0ZWRGaW5nZXJwcmludCA9ICdhc3NldDFtOGgwZ2s3ZjZ4NWo1cWcweDVtNHE1ZjVqNXFnMHg1bTRxNWY1aic7XG5cbiAgaXQoYHNob3VsZCBidWlsZCBhIHRyYW5zYWN0aW9uIHdpdGggdG9rZW4gJHthc3NldE5hbWV9IC0gYWRhICsgJHthc3NldE5hbWV9YCwgYXN5bmMgKCkgPT4ge1xuICAgIGNvbnN0IHF1YW50aXR5ID0gJzIwJztcbiAgICBjb25zdCB0b3RhbElucHV0ID0gMjAwMDAwMDA7XG4gICAgY29uc3QgdG90YWxBc3NldExpc3QgPSB7XG4gICAgICBbZmluZ2VycHJpbnRdOiB7XG4gICAgICAgIHF1YW50aXR5OiAnMTAwJyxcbiAgICAgICAgcG9saWN5X2lkOiBwb2xpY3lJZCxcbiAgICAgICAgYXNzZXRfbmFtZTogYXNjaWlFbmNvZGVkTmFtZSxcbiAgICAgIH0sXG4gICAgfTtcbiAgICBjb25zdCBleHBlY3RlZENoYW5nZUFkYSA9IChcbiAgICAgIHRvdGFsSW5wdXQgLVxuICAgICAgMTUwMDAwMCAvKiBtaW4gYWRhIGZvciBjaGFuZ2UgdG9rZW4gdXR4byAqLyAtXG4gICAgICAxNTAwMDAwIC8qIG1pbiBhZGEgZm9yIHJlY2lwaWVudCB0b2tlbiB1dHhvKi8gLVxuICAgICAgMTczNTk3XG4gICAgKSAvKiBmZWUgKi9cbiAgICAgIC50b1N0cmluZygpO1xuICAgIGNvbnN0IGV4cGVjdGVkQ2hhbmdlVG9rZW4gPSAnODAnO1xuXG4gICAgY29uc3QgdHhCdWlsZGVyID0gZmFjdG9yeS5nZXRUcmFuc2ZlckJ1aWxkZXIoKTtcbiAgICB0eEJ1aWxkZXIuaW5wdXQoe1xuICAgICAgdHJhbnNhY3Rpb25faWQ6ICczNjc3ZTc1YzdiYTY5OWJmZGM2Y2Q1N2Q0MmYyNDZmODZmNjNhZWZkNzYwMjUwMDZhYzc4MzEzZmFkMmJiYTIxJyxcbiAgICAgIHRyYW5zYWN0aW9uX2luZGV4OiAxLFxuICAgIH0pO1xuXG4gICAgdHhCdWlsZGVyLm91dHB1dCh7XG4gICAgICBhZGRyZXNzOiByZWNlaXZlckFkZHJlc3MsXG4gICAgICBhbW91bnQ6ICcwJywgLy8gU2V0IEFEQSBhbW91bnQgdG8gMCBmb3IgdG9rZW4gdHJhbnNmZXIgKG1pbiBBREEgaXMgaGFuZGxlZCBpbiBzZGsgYnVpbGQpXG4gICAgICBtdWx0aUFzc2V0czoge1xuICAgICAgICBhc3NldF9uYW1lOiBhc2NpaUVuY29kZWROYW1lLFxuICAgICAgICBwb2xpY3lfaWQ6IHBvbGljeUlkLFxuICAgICAgICBxdWFudGl0eSxcbiAgICAgICAgZmluZ2VycHJpbnQsXG4gICAgICB9LFxuICAgIH0pO1xuXG4gICAgdHhCdWlsZGVyLmNoYW5nZUFkZHJlc3Moc2VuZGVyQWRkcmVzcywgdG90YWxJbnB1dC50b1N0cmluZygpLCB0b3RhbEFzc2V0TGlzdCk7XG4gICAgdHhCdWlsZGVyLnR0bCg4MDAwMDAwMDApO1xuICAgIHR4QnVpbGRlci5pc1Rva2VuVHJhbnNhY3Rpb24oKTtcbiAgICBjb25zdCB0eCA9IChhd2FpdCB0eEJ1aWxkZXIuYnVpbGQoKSkgYXMgVHJhbnNhY3Rpb247XG5cbiAgICBzaG91bGQuZXF1YWwodHgudHlwZSwgVHJhbnNhY3Rpb25UeXBlLlNlbmQpO1xuICAgIGNvbnN0IHR4RGF0YSA9IHR4LnRvSnNvbigpO1xuXG4gICAgLy8gQ2hlY2sgb3V0cHV0cyAoc2hvdWxkIGluY2x1ZGUgbXVsdGktYXNzZXQgYW5kIHJlZ3VsYXIgY2hhbmdlKVxuICAgIHR4RGF0YS5vdXRwdXRzLmxlbmd0aC5zaG91bGQuZXF1YWwoMyk7XG4gICAgdHhEYXRhLmlucHV0cy5sZW5ndGguc2hvdWxkLmVxdWFsKDEpO1xuXG4gICAgLy8gT25lIG91dHB1dCBzaG91bGQgaGF2ZSB0aGUgbXVsdGktYXNzZXQgY2hhbmdlIGFuZCB0aGUgb3RoZXIgZm9yIHRoZSB0cmFuc2ZlclxuICAgIGNvbnN0IGFzc2V0T3V0cHV0ID0gdHhEYXRhLm91dHB1dHMuZmlsdGVyKChvdXRwdXQpID0+IG91dHB1dC5tdWx0aUFzc2V0cyAhPT0gdW5kZWZpbmVkKTtcbiAgICBhc3NldE91dHB1dCEubGVuZ3RoLnNob3VsZC5lcXVhbCgyKTtcblxuICAgIC8vIFZlcmlmeSBpbnB1dHMgYW5kIG91dHB1dHNcbiAgICBjb25zdCBpbnB1dCA9IHR4RGF0YS5pbnB1dHNbMF07XG4gICAgaW5wdXQudHJhbnNhY3Rpb25faWQuc2hvdWxkLmVxdWFsKCczNjc3ZTc1YzdiYTY5OWJmZGM2Y2Q1N2Q0MmYyNDZmODZmNjNhZWZkNzYwMjUwMDZhYzc4MzEzZmFkMmJiYTIxJyk7XG4gICAgaW5wdXQudHJhbnNhY3Rpb25faW5kZXguc2hvdWxkLmVxdWFsKDEpO1xuXG4gICAgLy8gVmFsaWRhdGUgcmVjZWl2ZXIgb3V0cHV0XG4gICAgY29uc3QgcmVjZWl2ZXJPdXRwdXQgPSB0eERhdGEub3V0cHV0cy5maWx0ZXIoKG91dHB1dCkgPT4gb3V0cHV0LmFkZHJlc3MgPT09IHJlY2VpdmVyQWRkcmVzcyk7XG4gICAgcmVjZWl2ZXJPdXRwdXQubGVuZ3RoLnNob3VsZC5lcXVhbCgxKTtcbiAgICByZWNlaXZlck91dHB1dFswXS5hbW91bnQuc2hvdWxkLmVxdWFsKCcxNTAwMDAwJyk7IC8vIE1pbmltdW0gQURBIGZvciBhc3NldCBvdXRwdXRcbiAgICAocmVjZWl2ZXJPdXRwdXRbMF0ubXVsdGlBc3NldHMhIGFzIENhcmRhbm9XYXNtLk11bHRpQXNzZXQpXG4gICAgICAuZ2V0X2Fzc2V0KHBvbGljeVNjcmlwdEhhc2gsIENhcmRhbm9XYXNtLkFzc2V0TmFtZS5uZXcoQnVmZmVyLmZyb20oYXNjaWlFbmNvZGVkTmFtZSwgJ2hleCcpKSlcbiAgICAgIC50b19zdHIoKVxuICAgICAgLnNob3VsZC5lcXVhbChxdWFudGl0eSk7XG5cbiAgICAvLyBWYWxpZGF0ZSBjaGFuZ2Ugb3V0cHV0cyAob25lIHdpdGggYXNzZXQgYW5kIG9uZSB3aXRob3V0KVxuICAgIGNvbnN0IGNoYW5nZU91dHB1dCA9IHR4RGF0YS5vdXRwdXRzLmZpbHRlcigob3V0cHV0KSA9PiBvdXRwdXQuYWRkcmVzcyA9PT0gc2VuZGVyQWRkcmVzcyk7XG4gICAgY2hhbmdlT3V0cHV0Lmxlbmd0aC5zaG91bGQuZXF1YWwoMik7XG4gICAgY29uc3QgY2hhbmdlV2l0aEFzc2V0ID0gY2hhbmdlT3V0cHV0LmZpbmQoKG91dHB1dCkgPT4gb3V0cHV0Lm11bHRpQXNzZXRzICE9PSB1bmRlZmluZWQpO1xuICAgIHNob3VsZC5leGlzdChjaGFuZ2VXaXRoQXNzZXQpO1xuICAgIGNoYW5nZVdpdGhBc3NldCEuYW1vdW50LnNob3VsZC5lcXVhbCgnMTUwMDAwMCcpOyAvLyBNaW5pbXVtIEFEQSBmb3IgYXNzZXQgb3V0cHV0XG4gICAgKGNoYW5nZVdpdGhBc3NldCEubXVsdGlBc3NldHMhIGFzIENhcmRhbm9XYXNtLk11bHRpQXNzZXQpXG4gICAgICAuZ2V0X2Fzc2V0KHBvbGljeVNjcmlwdEhhc2gsIENhcmRhbm9XYXNtLkFzc2V0TmFtZS5uZXcoQnVmZmVyLmZyb20oYXNjaWlFbmNvZGVkTmFtZSwgJ2hleCcpKSlcbiAgICAgIC50b19zdHIoKVxuICAgICAgLnNob3VsZC5lcXVhbChleHBlY3RlZENoYW5nZVRva2VuKTtcblxuICAgIGNvbnN0IGNoYW5nZVdpdGhvdXRBc3NldCA9IGNoYW5nZU91dHB1dC5maW5kKChvdXRwdXQpID0+IG91dHB1dC5tdWx0aUFzc2V0cyA9PT0gdW5kZWZpbmVkKTtcbiAgICBzaG91bGQuZXhpc3QoY2hhbmdlV2l0aG91dEFzc2V0KTtcbiAgICBjaGFuZ2VXaXRob3V0QXNzZXQhLmFtb3VudC5zaG91bGQuZXF1YWwoZXhwZWN0ZWRDaGFuZ2VBZGEpOyAvLyBSZW1haW5pbmcgQURBIGFmdGVyIGZlZXMgYW5kIG1pbiBBREFzXG4gIH0pO1xuXG4gIGl0KGBzaG91bGQgYnVpbGQgYSB0cmFuc2FjdGlvbiB3aXRoIHRva2VuICR7YXNzZXROYW1lfSAtIGFkYSArICR7YXNzZXROYW1lfSArIHVuc3VwcG9ydGVkIHRva2VuYCwgYXN5bmMgKCkgPT4ge1xuICAgIGNvbnN0IHF1YW50aXR5ID0gJzIwJztcbiAgICBjb25zdCB0b3RhbElucHV0ID0gMjAwMDAwMDA7XG4gICAgY29uc3QgdG90YWxBc3NldExpc3QgPSB7XG4gICAgICBbZmluZ2VycHJpbnRdOiB7XG4gICAgICAgIHF1YW50aXR5OiAnMTAwJyxcbiAgICAgICAgcG9saWN5X2lkOiBwb2xpY3lJZCxcbiAgICAgICAgYXNzZXRfbmFtZTogYXNjaWlFbmNvZGVkTmFtZSxcbiAgICAgIH0sXG4gICAgICBbdW5zdXBwb3J0ZWRGaW5nZXJwcmludF06IHtcbiAgICAgICAgcXVhbnRpdHk6ICcxMDAwMCcsXG4gICAgICAgIHBvbGljeV9pZDogdW5zdXBwb3J0ZWRQb2xpY3lJZCxcbiAgICAgICAgYXNzZXRfbmFtZTogdW5zdXBwb3J0ZWRBc2NpaUVuY29kZWROYW1lLFxuICAgICAgfSxcbiAgICB9O1xuICAgIGNvbnN0IGV4cGVjdGVkQ2hhbmdlQWRhID0gKFxuICAgICAgdG90YWxJbnB1dCAtXG4gICAgICAxNTAwMDAwIC8qIG1pbiBhZGEgZm9yIGNoYW5nZSB0b2tlbiB1dHhvICovIC1cbiAgICAgIDE1MDAwMDAgLyogbWluIGFkYSBmb3IgcmVjaXBpZW50IHRva2VuIHV0eG8qLyAtXG4gICAgICAxNTAwMDAwIC8qIG1pbiBhZGEgZm9yIHVuc3VwcG9ydGVkIHRva2VuIGNoYW5nZSB1dHhvICovIC1cbiAgICAgIDE3OTg4OVxuICAgICkgLyogZmVlICovXG4gICAgICAudG9TdHJpbmcoKTtcbiAgICBjb25zdCBleHBlY3RlZENoYW5nZVRva2VuID0gJzgwJztcblxuICAgIGNvbnN0IHR4QnVpbGRlciA9IGZhY3RvcnkuZ2V0VHJhbnNmZXJCdWlsZGVyKCk7XG4gICAgdHhCdWlsZGVyLmlucHV0KHtcbiAgICAgIHRyYW5zYWN0aW9uX2lkOiAnMzY3N2U3NWM3YmE2OTliZmRjNmNkNTdkNDJmMjQ2Zjg2ZjYzYWVmZDc2MDI1MDA2YWM3ODMxM2ZhZDJiYmEyMScsXG4gICAgICB0cmFuc2FjdGlvbl9pbmRleDogMSxcbiAgICB9KTtcbiAgICB0eEJ1aWxkZXIuaW5wdXQoe1xuICAgICAgdHJhbnNhY3Rpb25faWQ6ICczNjc3ZTc1YzdiYTY5OWJmZGM2Y2Q1N2Q0MmYyNDZmODZmNjNhZWZkNzYwMjUwMDZhYzc4MzEzZmFkMmJiYTIyJyxcbiAgICAgIHRyYW5zYWN0aW9uX2luZGV4OiAxLFxuICAgIH0pO1xuXG4gICAgdHhCdWlsZGVyLm91dHB1dCh7XG4gICAgICBhZGRyZXNzOiByZWNlaXZlckFkZHJlc3MsXG4gICAgICBhbW91bnQ6ICcwJywgLy8gU2V0IEFEQSBhbW91bnQgdG8gMCBmb3IgdG9rZW4gdHJhbnNmZXIgKG1pbiBBREEgaXMgaGFuZGxlZCBpbiBzZGsgYnVpbGQpXG4gICAgICBtdWx0aUFzc2V0czoge1xuICAgICAgICBhc3NldF9uYW1lOiBhc2NpaUVuY29kZWROYW1lLFxuICAgICAgICBwb2xpY3lfaWQ6IHBvbGljeUlkLFxuICAgICAgICBxdWFudGl0eSxcbiAgICAgICAgZmluZ2VycHJpbnQsXG4gICAgICB9LFxuICAgIH0pO1xuXG4gICAgdHhCdWlsZGVyLmNoYW5nZUFkZHJlc3Moc2VuZGVyQWRkcmVzcywgdG90YWxJbnB1dC50b1N0cmluZygpLCB0b3RhbEFzc2V0TGlzdCk7XG4gICAgdHhCdWlsZGVyLnR0bCg4MDAwMDAwMDApO1xuICAgIHR4QnVpbGRlci5pc1Rva2VuVHJhbnNhY3Rpb24oKTtcbiAgICBjb25zdCB0eCA9IChhd2FpdCB0eEJ1aWxkZXIuYnVpbGQoKSkgYXMgVHJhbnNhY3Rpb247XG5cbiAgICBzaG91bGQuZXF1YWwodHgudHlwZSwgVHJhbnNhY3Rpb25UeXBlLlNlbmQpO1xuICAgIGNvbnN0IHR4RGF0YSA9IHR4LnRvSnNvbigpO1xuXG4gICAgLy8gQ2hlY2sgb3V0cHV0cyAoc2hvdWxkIGluY2x1ZGUgbXVsdGktYXNzZXQgYW5kIHJlZ3VsYXIgY2hhbmdlKVxuICAgIHR4RGF0YS5vdXRwdXRzLmxlbmd0aC5zaG91bGQuZXF1YWwoNCk7XG4gICAgdHhEYXRhLmlucHV0cy5sZW5ndGguc2hvdWxkLmVxdWFsKDIpO1xuXG4gICAgLy8gT25lIG91dHB1dCBzaG91bGQgaGF2ZSB0aGUgbXVsdGktYXNzZXQgY2hhbmdlIGFuZCB0aGUgb3RoZXIgZm9yIHRoZSB0cmFuc2ZlclxuICAgIGNvbnN0IGFzc2V0T3V0cHV0ID0gdHhEYXRhLm91dHB1dHMuZmlsdGVyKChvdXRwdXQpID0+IG91dHB1dC5tdWx0aUFzc2V0cyAhPT0gdW5kZWZpbmVkKTtcbiAgICBhc3NldE91dHB1dCEubGVuZ3RoLnNob3VsZC5lcXVhbCgzKTtcblxuICAgIC8vIFZlcmlmeSBpbnB1dHMgYW5kIG91dHB1dHNcbiAgICB0eERhdGEuaW5wdXRzWzBdLnRyYW5zYWN0aW9uX2lkLnNob3VsZC5lcXVhbCgnMzY3N2U3NWM3YmE2OTliZmRjNmNkNTdkNDJmMjQ2Zjg2ZjYzYWVmZDc2MDI1MDA2YWM3ODMxM2ZhZDJiYmEyMScpO1xuICAgIHR4RGF0YS5pbnB1dHNbMF0udHJhbnNhY3Rpb25faW5kZXguc2hvdWxkLmVxdWFsKDEpO1xuXG4gICAgdHhEYXRhLmlucHV0c1sxXS50cmFuc2FjdGlvbl9pZC5zaG91bGQuZXF1YWwoJzM2NzdlNzVjN2JhNjk5YmZkYzZjZDU3ZDQyZjI0NmY4NmY2M2FlZmQ3NjAyNTAwNmFjNzgzMTNmYWQyYmJhMjInKTtcbiAgICB0eERhdGEuaW5wdXRzWzFdLnRyYW5zYWN0aW9uX2luZGV4LnNob3VsZC5lcXVhbCgxKTtcblxuICAgIC8vIFZhbGlkYXRlIHJlY2VpdmVyIG91dHB1dFxuICAgIGNvbnN0IHJlY2VpdmVyT3V0cHV0ID0gdHhEYXRhLm91dHB1dHMuZmlsdGVyKChvdXRwdXQpID0+IG91dHB1dC5hZGRyZXNzID09PSByZWNlaXZlckFkZHJlc3MpO1xuICAgIHJlY2VpdmVyT3V0cHV0Lmxlbmd0aC5zaG91bGQuZXF1YWwoMSk7XG4gICAgcmVjZWl2ZXJPdXRwdXRbMF0uYW1vdW50LnNob3VsZC5lcXVhbCgnMTUwMDAwMCcpOyAvLyBNaW5pbXVtIEFEQSBmb3IgYXNzZXQgb3V0cHV0XG4gICAgKHJlY2VpdmVyT3V0cHV0WzBdLm11bHRpQXNzZXRzISBhcyBDYXJkYW5vV2FzbS5NdWx0aUFzc2V0KVxuICAgICAgLmdldF9hc3NldChwb2xpY3lTY3JpcHRIYXNoLCBDYXJkYW5vV2FzbS5Bc3NldE5hbWUubmV3KEJ1ZmZlci5mcm9tKGFzY2lpRW5jb2RlZE5hbWUsICdoZXgnKSkpXG4gICAgICAudG9fc3RyKClcbiAgICAgIC5zaG91bGQuZXF1YWwocXVhbnRpdHkpO1xuXG4gICAgLy8gVmFsaWRhdGUgY2hhbmdlIG91dHB1dHMgKG9uZSB3aXRoIHN1cHBvcnRlZCB0b2tlbiwgb25lIHdpdGggdW5zdXBwb3J0ZWQgdG9rZW4gYW5kIG9uZSBmb3IgcHVyZSBhZGEpXG4gICAgY29uc3QgY2hhbmdlT3V0cHV0ID0gdHhEYXRhLm91dHB1dHMuZmlsdGVyKChvdXRwdXQpID0+IG91dHB1dC5hZGRyZXNzID09PSBzZW5kZXJBZGRyZXNzKTtcbiAgICBjaGFuZ2VPdXRwdXQubGVuZ3RoLnNob3VsZC5lcXVhbCgzKTtcblxuICAgIGNvbnN0IGNoYW5nZVdpdGhBc3NldCA9IGNoYW5nZU91dHB1dC5maWx0ZXIoKG91dHB1dCkgPT4gb3V0cHV0Lm11bHRpQXNzZXRzICE9PSB1bmRlZmluZWQpO1xuICAgIGNoYW5nZVdpdGhBc3NldC5sZW5ndGguc2hvdWxkLmVxdWFsKDIpO1xuICAgIGxldCB0b2tlbnMgPSAwO1xuICAgIGNoYW5nZVdpdGhBc3NldC5mb3JFYWNoKChvdXRwdXQpID0+IHtcbiAgICAgIG91dHB1dCEuYW1vdW50LnNob3VsZC5lcXVhbCgnMTUwMDAwMCcpOyAvLyBNaW5pbXVtIEFEQSBmb3IgYXNzZXQgb3V0cHV0XG4gICAgICBjb25zdCBzdXBwb3J0ZWRUb2tlbkNoYW5nZVF0eSA9IChvdXRwdXQhLm11bHRpQXNzZXRzISBhcyBDYXJkYW5vV2FzbS5NdWx0aUFzc2V0KVxuICAgICAgICAuZ2V0X2Fzc2V0KHBvbGljeVNjcmlwdEhhc2gsIENhcmRhbm9XYXNtLkFzc2V0TmFtZS5uZXcoQnVmZmVyLmZyb20oYXNjaWlFbmNvZGVkTmFtZSwgJ2hleCcpKSlcbiAgICAgICAgLnRvX3N0cigpO1xuICAgICAgY29uc3QgdW5zdXBwb3J0ZWRUb2tlbkNoYW5nZVF0eSA9IChvdXRwdXQhLm11bHRpQXNzZXRzISBhcyBDYXJkYW5vV2FzbS5NdWx0aUFzc2V0KVxuICAgICAgICAuZ2V0X2Fzc2V0KFxuICAgICAgICAgIHVuc3VwcG9ydGVkUG9saWN5U2NyaXB0SGFzaCxcbiAgICAgICAgICBDYXJkYW5vV2FzbS5Bc3NldE5hbWUubmV3KEJ1ZmZlci5mcm9tKHVuc3VwcG9ydGVkQXNjaWlFbmNvZGVkTmFtZSwgJ2hleCcpKVxuICAgICAgICApXG4gICAgICAgIC50b19zdHIoKTtcbiAgICAgIGlmIChzdXBwb3J0ZWRUb2tlbkNoYW5nZVF0eSAhPT0gJzAnKSB7XG4gICAgICAgIHN1cHBvcnRlZFRva2VuQ2hhbmdlUXR5LnNob3VsZC5lcXVhbChleHBlY3RlZENoYW5nZVRva2VuKTtcbiAgICAgICAgdG9rZW5zKys7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICB1bnN1cHBvcnRlZFRva2VuQ2hhbmdlUXR5IS5zaG91bGQuZXF1YWwoJzEwMDAwJyk7XG4gICAgICAgIHRva2VucysrO1xuICAgICAgfVxuICAgIH0pO1xuICAgIHRva2Vucy5zaG91bGQuZXF1YWwoMik7XG5cbiAgICBjb25zdCBjaGFuZ2VXaXRob3V0QXNzZXQgPSBjaGFuZ2VPdXRwdXQuZmluZCgob3V0cHV0KSA9PiBvdXRwdXQubXVsdGlBc3NldHMgPT09IHVuZGVmaW5lZCk7XG4gICAgc2hvdWxkLmV4aXN0KGNoYW5nZVdpdGhvdXRBc3NldCk7XG4gICAgY2hhbmdlV2l0aG91dEFzc2V0IS5hbW91bnQuc2hvdWxkLmVxdWFsKGV4cGVjdGVkQ2hhbmdlQWRhKTsgLy8gUmVtYWluaW5nIEFEQSBhZnRlciBmZWVzIGFuZCBtaW4gQURBc1xuICB9KTtcblxuICBpdChgc2hvdWxkIGZhaWwgdG8gYnVpbGQgYSB0cmFuc2FjdGlvbiB3aXRoICR7YXNzZXROYW1lfSB0b2tlbiBhbmQgaW5zdWZmaWNpZW50IG1pbmltdW0gQURBYCwgYXN5bmMgKCkgPT4ge1xuICAgIGNvbnN0IHF1YW50aXR5ID0gJzIwJztcbiAgICBjb25zdCB0b3RhbElucHV0ID0gMjAwMDAwMDtcbiAgICBjb25zdCB0b3RhbEFzc2V0TGlzdCA9IHtcbiAgICAgIFtmaW5nZXJwcmludF06IHtcbiAgICAgICAgcXVhbnRpdHk6ICcxMDAnLFxuICAgICAgICBwb2xpY3lfaWQ6IHBvbGljeUlkLFxuICAgICAgICBhc3NldF9uYW1lOiBhc2NpaUVuY29kZWROYW1lLFxuICAgICAgfSxcbiAgICB9O1xuXG4gICAgY29uc3QgdHhCdWlsZGVyID0gZmFjdG9yeS5nZXRUcmFuc2ZlckJ1aWxkZXIoKTtcbiAgICB0eEJ1aWxkZXIuaW5wdXQoe1xuICAgICAgdHJhbnNhY3Rpb25faWQ6ICczNjc3ZTc1YzdiYTY5OWJmZGM2Y2Q1N2Q0MmYyNDZmODZmNjNhZWZkNzYwMjUwMDZhYzc4MzEzZmFkMmJiYTIxJyxcbiAgICAgIHRyYW5zYWN0aW9uX2luZGV4OiAxLFxuICAgIH0pO1xuXG4gICAgdHhCdWlsZGVyLm91dHB1dCh7XG4gICAgICBhZGRyZXNzOiByZWNlaXZlckFkZHJlc3MsXG4gICAgICBhbW91bnQ6ICcwJywgLy8gU2V0IEFEQSBhbW91bnQgdG8gMCBmb3IgdG9rZW4gdHJhbnNmZXIgKG1pbiBBREEgaXMgaGFuZGxlZCBpbiBzZGsgYnVpbGQpXG4gICAgICBtdWx0aUFzc2V0czoge1xuICAgICAgICBhc3NldF9uYW1lOiBhc2NpaUVuY29kZWROYW1lLFxuICAgICAgICBwb2xpY3lfaWQ6IHBvbGljeUlkLFxuICAgICAgICBxdWFudGl0eSxcbiAgICAgICAgZmluZ2VycHJpbnQsXG4gICAgICB9LFxuICAgIH0pO1xuXG4gICAgdHhCdWlsZGVyLmNoYW5nZUFkZHJlc3Moc2VuZGVyQWRkcmVzcywgdG90YWxJbnB1dC50b1N0cmluZygpLCB0b3RhbEFzc2V0TGlzdCk7XG4gICAgdHhCdWlsZGVyLnR0bCg4MDAwMDAwMDApO1xuICAgIHR4QnVpbGRlci5pc1Rva2VuVHJhbnNhY3Rpb24oKTtcblxuICAgIGF3YWl0IHR4QnVpbGRlclxuICAgICAgLmJ1aWxkKClcbiAgICAgIC5zaG91bGQucmVqZWN0ZWRXaXRoKFxuICAgICAgICAnSW5zdWZmaWNpZW50IGZ1bmRzOiBuZWVkIGEgbWluaW11bSBvZiAxNTAwMDAwIGxvdmVsYWNlIHBlciBvdXRwdXQgdG8gY29uc3RydWN0IHRva2VuIHRyYW5zYWN0aW9ucydcbiAgICAgICk7XG4gIH0pO1xuXG4gIGl0KGBzaG91bGQgZmFpbCB0byBidWlsZCBhIHRyYW5zYWN0aW9uIHdpdGggJHthc3NldE5hbWV9IGFuZCBpbnN1ZmZpY2llbnQgdG9rZW4gcXR5YCwgYXN5bmMgKCkgPT4ge1xuICAgIGNvbnN0IHF1YW50aXR5ID0gJzIwJztcbiAgICBjb25zdCB0b3RhbElucHV0ID0gMjAwMDAwMDA7XG4gICAgY29uc3QgdG90YWxBc3NldExpc3QgPSB7XG4gICAgICBbZmluZ2VycHJpbnRdOiB7XG4gICAgICAgIHF1YW50aXR5OiAnNScsXG4gICAgICAgIHBvbGljeV9pZDogcG9saWN5SWQsXG4gICAgICAgIGFzc2V0X25hbWU6IGFzY2lpRW5jb2RlZE5hbWUsXG4gICAgICB9LFxuICAgIH07XG5cbiAgICBjb25zdCB0eEJ1aWxkZXIgPSBmYWN0b3J5LmdldFRyYW5zZmVyQnVpbGRlcigpO1xuICAgIHR4QnVpbGRlci5pbnB1dCh7XG4gICAgICB0cmFuc2FjdGlvbl9pZDogJzM2NzdlNzVjN2JhNjk5YmZkYzZjZDU3ZDQyZjI0NmY4NmY2M2FlZmQ3NjAyNTAwNmFjNzgzMTNmYWQyYmJhMjEnLFxuICAgICAgdHJhbnNhY3Rpb25faW5kZXg6IDEsXG4gICAgfSk7XG5cbiAgICB0eEJ1aWxkZXIub3V0cHV0KHtcbiAgICAgIGFkZHJlc3M6IHJlY2VpdmVyQWRkcmVzcyxcbiAgICAgIGFtb3VudDogJzAnLCAvLyBTZXQgQURBIGFtb3VudCB0byAwIGZvciB0b2tlbiB0cmFuc2ZlciAobWluIEFEQSBpcyBoYW5kbGVkIGluIHNkayBidWlsZClcbiAgICAgIG11bHRpQXNzZXRzOiB7XG4gICAgICAgIGFzc2V0X25hbWU6IGFzY2lpRW5jb2RlZE5hbWUsXG4gICAgICAgIHBvbGljeV9pZDogcG9saWN5SWQsXG4gICAgICAgIHF1YW50aXR5LFxuICAgICAgICBmaW5nZXJwcmludCxcbiAgICAgIH0sXG4gICAgfSk7XG5cbiAgICB0eEJ1aWxkZXIuY2hhbmdlQWRkcmVzcyhzZW5kZXJBZGRyZXNzLCB0b3RhbElucHV0LnRvU3RyaW5nKCksIHRvdGFsQXNzZXRMaXN0KTtcbiAgICB0eEJ1aWxkZXIudHRsKDgwMDAwMDAwMCk7XG4gICAgdHhCdWlsZGVyLmlzVG9rZW5UcmFuc2FjdGlvbigpO1xuXG4gICAgYXdhaXQgdHhCdWlsZGVyLmJ1aWxkKCkuc2hvdWxkLnJlamVjdGVkV2l0aCgnSW5zdWZmaWNpZW50IHF0eTogbm90IGVub3VnaCB0b2tlbiBxdHkgdG8gY292ZXIgcmVjZWl2ZXIgb3V0cHV0Jyk7XG4gIH0pO1xuXG4gIGl0KGBzaG91bGQgYnVpbGQgYSB0cmFuc2FjdGlvbiB3aXRoICR7YXNzZXROYW1lfSB3aGVuIHRoZSB0b2tlbiBxdHkgaXMgZXhhY3RseSB0aGUgcXR5IGluIHdpdGhkcmF3YWxgLCBhc3luYyAoKSA9PiB7XG4gICAgY29uc3QgcXVhbnRpdHkgPSAnMSc7XG4gICAgY29uc3QgdG90YWxJbnB1dCA9IDIwMDAwMDAwO1xuICAgIGNvbnN0IHRvdGFsQXNzZXRMaXN0ID0ge1xuICAgICAgW2ZpbmdlcnByaW50XToge1xuICAgICAgICBxdWFudGl0eTogJzEnLFxuICAgICAgICBwb2xpY3lfaWQ6IHBvbGljeUlkLFxuICAgICAgICBhc3NldF9uYW1lOiBhc2NpaUVuY29kZWROYW1lLFxuICAgICAgfSxcbiAgICB9O1xuXG4gICAgY29uc3QgdHhCdWlsZGVyID0gZmFjdG9yeS5nZXRUcmFuc2ZlckJ1aWxkZXIoKTtcbiAgICB0eEJ1aWxkZXIuaW5wdXQoe1xuICAgICAgdHJhbnNhY3Rpb25faWQ6ICczNjc3ZTc1YzdiYTY5OWJmZGM2Y2Q1N2Q0MmYyNDZmODZmNjNhZWZkNzYwMjUwMDZhYzc4MzEzZmFkMmJiYTIxJyxcbiAgICAgIHRyYW5zYWN0aW9uX2luZGV4OiAxLFxuICAgIH0pO1xuXG4gICAgdHhCdWlsZGVyLm91dHB1dCh7XG4gICAgICBhZGRyZXNzOiByZWNlaXZlckFkZHJlc3MsXG4gICAgICBhbW91bnQ6ICcwJywgLy8gU2V0IEFEQSBhbW91bnQgdG8gMCBmb3IgdG9rZW4gdHJhbnNmZXIgKG1pbiBBREEgaXMgaGFuZGxlZCBpbiBzZGsgYnVpbGQpXG4gICAgICBtdWx0aUFzc2V0czoge1xuICAgICAgICBhc3NldF9uYW1lOiBhc2NpaUVuY29kZWROYW1lLFxuICAgICAgICBwb2xpY3lfaWQ6IHBvbGljeUlkLFxuICAgICAgICBxdWFudGl0eSxcbiAgICAgICAgZmluZ2VycHJpbnQsXG4gICAgICB9LFxuICAgIH0pO1xuXG4gICAgdHhCdWlsZGVyLmNoYW5nZUFkZHJlc3Moc2VuZGVyQWRkcmVzcywgdG90YWxJbnB1dC50b1N0cmluZygpLCB0b3RhbEFzc2V0TGlzdCk7XG4gICAgdHhCdWlsZGVyLnR0bCg4MDAwMDAwMDApO1xuICAgIHR4QnVpbGRlci5pc1Rva2VuVHJhbnNhY3Rpb24oKTtcblxuICAgIGF3YWl0IHR4QnVpbGRlci5idWlsZCgpLnNob3VsZC5ub3QuYmUucmVqZWN0ZWQoKTtcbiAgfSk7XG5cbiAgaXQoYHNob3VsZCBidWlsZCBhIHNwb25zb3JlZCB0b2tlbiB0cmFuc2FjdGlvbiB3aGVyZSBmZWUgYWRkcmVzcyBzcG9uc29ycyBtaW4gQURBIGZvciByZWNlaXZlcmAsIGFzeW5jICgpID0+IHtcbiAgICBjb25zdCBmZWVBZGRyZXNzID1cbiAgICAgICdhZGRyX3Rlc3QxcXoyZnh2MnVteWh0dGt4eXhwOHgwZGxwZHQzazZjd25nNXB4ajNqaHN5ZHplcjNqY3U1ZDhwczd6ZXgyazJ4dDN1cXhnanFubmo4M3dzOGxocm42NDhqanh0d3EyeXRqcXAnO1xuICAgIGNvbnN0IHF1YW50aXR5ID0gJzIwJztcbiAgICBjb25zdCBzZW5kZXJJbnB1dEJhbGFuY2UgPSA1MDAwMDAwO1xuICAgIGNvbnN0IGZlZUFkZHJlc3NJbnB1dEJhbGFuY2UgPSAyMDAwMDAwMDsgLy8gRmVlIGFkZHJlc3MgaGFzIGVub3VnaCBBREEgdG8gc3BvbnNvclxuICAgIGNvbnN0IHRvdGFsQXNzZXRMaXN0ID0ge1xuICAgICAgW2ZpbmdlcnByaW50XToge1xuICAgICAgICBxdWFudGl0eTogJzEwMCcsXG4gICAgICAgIHBvbGljeV9pZDogcG9saWN5SWQsXG4gICAgICAgIGFzc2V0X25hbWU6IGFzY2lpRW5jb2RlZE5hbWUsXG4gICAgICB9LFxuICAgIH07XG5cbiAgICBjb25zdCB0eEJ1aWxkZXIgPSBmYWN0b3J5LmdldFRyYW5zZmVyQnVpbGRlcigpO1xuICAgIC8vIFNlbmRlciBpbnB1dCAoaGFzIHRva2VucylcbiAgICB0eEJ1aWxkZXIuaW5wdXQoe1xuICAgICAgdHJhbnNhY3Rpb25faWQ6ICczNjc3ZTc1YzdiYTY5OWJmZGM2Y2Q1N2Q0MmYyNDZmODZmNjNhZWZkNzYwMjUwMDZhYzc4MzEzZmFkMmJiYTIxJyxcbiAgICAgIHRyYW5zYWN0aW9uX2luZGV4OiAxLFxuICAgIH0pO1xuICAgIC8vIEZlZSBhZGRyZXNzIGlucHV0IChzcG9uc29ycyBmZWVzIGFuZCBtaW4gQURBKVxuICAgIHR4QnVpbGRlci5pbnB1dCh7XG4gICAgICB0cmFuc2FjdGlvbl9pZDogJzM2NzdlNzVjN2JhNjk5YmZkYzZjZDU3ZDQyZjI0NmY4NmY2M2FlZmQ3NjAyNTAwNmFjNzgzMTNmYWQyYmJhMjInLFxuICAgICAgdHJhbnNhY3Rpb25faW5kZXg6IDAsXG4gICAgfSk7XG5cbiAgICB0eEJ1aWxkZXIub3V0cHV0KHtcbiAgICAgIGFkZHJlc3M6IHJlY2VpdmVyQWRkcmVzcyxcbiAgICAgIGFtb3VudDogJzAnLCAvLyBTZXQgQURBIGFtb3VudCB0byAwIGZvciB0b2tlbiB0cmFuc2ZlciAobWluIEFEQSBpcyBoYW5kbGVkIGJ5IGZlZSBhZGRyZXNzKVxuICAgICAgbXVsdGlBc3NldHM6IHtcbiAgICAgICAgYXNzZXRfbmFtZTogYXNjaWlFbmNvZGVkTmFtZSxcbiAgICAgICAgcG9saWN5X2lkOiBwb2xpY3lJZCxcbiAgICAgICAgcXVhbnRpdHksXG4gICAgICAgIGZpbmdlcnByaW50LFxuICAgICAgfSxcbiAgICB9KTtcblxuICAgIHR4QnVpbGRlci5jaGFuZ2VBZGRyZXNzKHNlbmRlckFkZHJlc3MsIHNlbmRlcklucHV0QmFsYW5jZS50b1N0cmluZygpLCB0b3RhbEFzc2V0TGlzdCk7XG4gICAgdHhCdWlsZGVyLnNwb25zb3JzaGlwSW5mbyh7XG4gICAgICBmZWVBZGRyZXNzOiBmZWVBZGRyZXNzLFxuICAgICAgZmVlQWRkcmVzc0lucHV0QmFsYW5jZTogZmVlQWRkcmVzc0lucHV0QmFsYW5jZS50b1N0cmluZygpLFxuICAgIH0pO1xuICAgIHR4QnVpbGRlci50dGwoODAwMDAwMDAwKTtcbiAgICB0eEJ1aWxkZXIuaXNUb2tlblRyYW5zYWN0aW9uKCk7XG4gICAgY29uc3QgdHggPSAoYXdhaXQgdHhCdWlsZGVyLmJ1aWxkKCkpIGFzIFRyYW5zYWN0aW9uO1xuXG4gICAgc2hvdWxkLmVxdWFsKHR4LnR5cGUsIFRyYW5zYWN0aW9uVHlwZS5TZW5kKTtcbiAgICBjb25zdCB0eERhdGEgPSB0eC50b0pzb24oKTtcblxuICAgIHR4RGF0YS5pbnB1dHMubGVuZ3RoLnNob3VsZC5lcXVhbCgyKTtcbiAgICB0eERhdGEub3V0cHV0cy5sZW5ndGguc2hvdWxkLmVxdWFsKDQpO1xuXG4gICAgLy8gVmFsaWRhdGUgcmVjZWl2ZXIgb3V0cHV0IC0gbWluIEFEQSBzaG91bGQgYmUgc3BvbnNvcmVkIGJ5IGZlZSBhZGRyZXNzXG4gICAgY29uc3QgcmVjZWl2ZXJPdXRwdXQgPSB0eERhdGEub3V0cHV0cy5maWx0ZXIoKG91dHB1dCkgPT4gb3V0cHV0LmFkZHJlc3MgPT09IHJlY2VpdmVyQWRkcmVzcyk7XG4gICAgcmVjZWl2ZXJPdXRwdXQubGVuZ3RoLnNob3VsZC5lcXVhbCgxKTtcbiAgICByZWNlaXZlck91dHB1dFswXS5hbW91bnQuc2hvdWxkLmVxdWFsKCcxNTAwMDAwJyk7IC8vIE1pbmltdW0gQURBIGZvciBhc3NldCBvdXRwdXQgKHNwb25zb3JlZCBieSBmZWUgYWRkcmVzcylcbiAgICAocmVjZWl2ZXJPdXRwdXRbMF0ubXVsdGlBc3NldHMhIGFzIENhcmRhbm9XYXNtLk11bHRpQXNzZXQpXG4gICAgICAuZ2V0X2Fzc2V0KHBvbGljeVNjcmlwdEhhc2gsIENhcmRhbm9XYXNtLkFzc2V0TmFtZS5uZXcoQnVmZmVyLmZyb20oYXNjaWlFbmNvZGVkTmFtZSwgJ2hleCcpKSlcbiAgICAgIC50b19zdHIoKVxuICAgICAgLnNob3VsZC5lcXVhbChxdWFudGl0eSk7XG5cbiAgICAvLyBWYWxpZGF0ZSBzZW5kZXIgY2hhbmdlIG91dHB1dCAtIHNob3VsZCBoYXZlIGZ1bGwgc2VuZGVyIGJhbGFuY2UgKHRva2VucyBvbmx5LCBubyBBREEgZGVkdWN0aW9uIGZvciByZWNlaXZlcilcbiAgICBjb25zdCBzZW5kZXJDaGFuZ2VPdXRwdXQgPSB0eERhdGEub3V0cHV0cy5maWx0ZXIoKG91dHB1dCkgPT4gb3V0cHV0LmFkZHJlc3MgPT09IHNlbmRlckFkZHJlc3MpO1xuICAgIHNlbmRlckNoYW5nZU91dHB1dC5sZW5ndGguc2hvdWxkLmJlLmVxdWFsKDIpO1xuXG4gICAgLy8gVmFsaWRhdGUgZmVlIGFkZHJlc3MgY2hhbmdlIG91dHB1dCAtIHNob3VsZCBoYXZlIHJlbWFpbmluZyBBREEgYWZ0ZXIgZmVlcyBhbmQgbWluIEFEQSBmb3IgcmVjZWl2ZXJcbiAgICBjb25zdCBmZWVBZGRyZXNzQ2hhbmdlT3V0cHV0ID0gdHhEYXRhLm91dHB1dHMuZmlsdGVyKChvdXRwdXQpID0+IG91dHB1dC5hZGRyZXNzID09PSBmZWVBZGRyZXNzKTtcbiAgICBmZWVBZGRyZXNzQ2hhbmdlT3V0cHV0Lmxlbmd0aC5zaG91bGQuZXF1YWwoMSk7XG4gICAgLy8gRmVlIGFkZHJlc3MgY2hhbmdlIHNob3VsZCBub3QgaGF2ZSBhbnkgdG9rZW5zXG4gICAgc2hvdWxkLm5vdC5leGlzdChmZWVBZGRyZXNzQ2hhbmdlT3V0cHV0WzBdLm11bHRpQXNzZXRzKTtcblxuICAgIHR4LmdldEZlZS5zaG91bGQuZXF1YWwoJzE4MjQ4NScpOyAvLyBGZWUgd2l0aCB0d28gd2l0bmVzc2VzXG4gIH0pO1xuXG4gIGl0KGBzaG91bGQgcmVidWlsZCBhIHNwb25zb3JlZCB0cmFuc2FjdGlvbiBmcm9tIGhleCB3aXRoIGlzUmVidWlsZCBmbGFnYCwgYXN5bmMgKCkgPT4ge1xuICAgIGNvbnN0IGZlZUFkZHJlc3MgPVxuICAgICAgJ2FkZHJfdGVzdDFxejJmeHYydW15aHR0a3h5eHA4eDBkbHBkdDNrNmN3bmc1cHhqM2poc3lkemVyM2pjdTVkOHBzN3pleDJrMnh0M3VxeGdqcW5uajgzd3M4bGhybjY0OGpqeHR3cTJ5dGpxcCc7XG4gICAgY29uc3QgcXVhbnRpdHkgPSAnMjAnO1xuICAgIGNvbnN0IHNlbmRlcklucHV0QmFsYW5jZSA9IDUwMDAwMDA7XG4gICAgY29uc3QgZmVlQWRkcmVzc0lucHV0QmFsYW5jZSA9IDIwMDAwMDAwO1xuICAgIGNvbnN0IHRvdGFsQXNzZXRMaXN0ID0ge1xuICAgICAgW2ZpbmdlcnByaW50XToge1xuICAgICAgICBxdWFudGl0eTogJzEwMCcsXG4gICAgICAgIHBvbGljeV9pZDogcG9saWN5SWQsXG4gICAgICAgIGFzc2V0X25hbWU6IGFzY2lpRW5jb2RlZE5hbWUsXG4gICAgICB9LFxuICAgIH07XG5cbiAgICAvLyBTdGVwIDE6IEJ1aWxkIHRoZSBpbml0aWFsIHNwb25zb3JlZCB0cmFuc2FjdGlvblxuICAgIGNvbnN0IHR4QnVpbGRlciA9IGZhY3RvcnkuZ2V0VHJhbnNmZXJCdWlsZGVyKCk7XG4gICAgdHhCdWlsZGVyLmlucHV0KHtcbiAgICAgIHRyYW5zYWN0aW9uX2lkOiAnMzY3N2U3NWM3YmE2OTliZmRjNmNkNTdkNDJmMjQ2Zjg2ZjYzYWVmZDc2MDI1MDA2YWM3ODMxM2ZhZDJiYmEyMScsXG4gICAgICB0cmFuc2FjdGlvbl9pbmRleDogMSxcbiAgICB9KTtcbiAgICB0eEJ1aWxkZXIuaW5wdXQoe1xuICAgICAgdHJhbnNhY3Rpb25faWQ6ICczNjc3ZTc1YzdiYTY5OWJmZGM2Y2Q1N2Q0MmYyNDZmODZmNjNhZWZkNzYwMjUwMDZhYzc4MzEzZmFkMmJiYTIyJyxcbiAgICAgIHRyYW5zYWN0aW9uX2luZGV4OiAwLFxuICAgIH0pO1xuXG4gICAgdHhCdWlsZGVyLm91dHB1dCh7XG4gICAgICBhZGRyZXNzOiByZWNlaXZlckFkZHJlc3MsXG4gICAgICBhbW91bnQ6ICcwJyxcbiAgICAgIG11bHRpQXNzZXRzOiB7XG4gICAgICAgIGFzc2V0X25hbWU6IGFzY2lpRW5jb2RlZE5hbWUsXG4gICAgICAgIHBvbGljeV9pZDogcG9saWN5SWQsXG4gICAgICAgIHF1YW50aXR5LFxuICAgICAgICBmaW5nZXJwcmludCxcbiAgICAgIH0sXG4gICAgfSk7XG5cbiAgICB0eEJ1aWxkZXIuY2hhbmdlQWRkcmVzcyhzZW5kZXJBZGRyZXNzLCBzZW5kZXJJbnB1dEJhbGFuY2UudG9TdHJpbmcoKSwgdG90YWxBc3NldExpc3QpO1xuICAgIHR4QnVpbGRlci5zcG9uc29yc2hpcEluZm8oe1xuICAgICAgZmVlQWRkcmVzczogZmVlQWRkcmVzcyxcbiAgICAgIGZlZUFkZHJlc3NJbnB1dEJhbGFuY2U6IGZlZUFkZHJlc3NJbnB1dEJhbGFuY2UudG9TdHJpbmcoKSxcbiAgICB9KTtcbiAgICB0eEJ1aWxkZXIudHRsKDgwMDAwMDAwMCk7XG4gICAgdHhCdWlsZGVyLmlzVG9rZW5UcmFuc2FjdGlvbigpO1xuICAgIGNvbnN0IGluaXRpYWxUeCA9IChhd2FpdCB0eEJ1aWxkZXIuYnVpbGQoKSkgYXMgVHJhbnNhY3Rpb247XG4gICAgY29uc3QgaW5pdGlhbEZlZSA9IGluaXRpYWxUeC5nZXRGZWU7XG4gICAgY29uc3QgaW5pdGlhbFR4RGF0YSA9IGluaXRpYWxUeC50b0pzb24oKTtcblxuICAgIC8vIFN0ZXAgMjogUmVidWlsZCB3aXRoIGlzUmVidWlsZCA9IHRydWVcbiAgICAvLyBUaGlzIHNpbXVsYXRlcyByZWJ1aWxkaW5nIGZyb20gc2NyYXRjaCBidXQgd2l0aCBpc1JlYnVpbGQgZmxhZyB0byBhZGQgc3BvbnNvciB3aXRuZXNzXG4gICAgY29uc3QgcmVidWlsZFR4QnVpbGRlciA9IGZhY3RvcnkuZ2V0VHJhbnNmZXJCdWlsZGVyKCk7XG4gICAgcmVidWlsZFR4QnVpbGRlci5pbnB1dCh7XG4gICAgICB0cmFuc2FjdGlvbl9pZDogJzM2NzdlNzVjN2JhNjk5YmZkYzZjZDU3ZDQyZjI0NmY4NmY2M2FlZmQ3NjAyNTAwNmFjNzgzMTNmYWQyYmJhMjEnLFxuICAgICAgdHJhbnNhY3Rpb25faW5kZXg6IDEsXG4gICAgfSk7XG4gICAgcmVidWlsZFR4QnVpbGRlci5pbnB1dCh7XG4gICAgICB0cmFuc2FjdGlvbl9pZDogJzM2NzdlNzVjN2JhNjk5YmZkYzZjZDU3ZDQyZjI0NmY4NmY2M2FlZmQ3NjAyNTAwNmFjNzgzMTNmYWQyYmJhMjInLFxuICAgICAgdHJhbnNhY3Rpb25faW5kZXg6IDAsXG4gICAgfSk7XG5cbiAgICByZWJ1aWxkVHhCdWlsZGVyLm91dHB1dCh7XG4gICAgICBhZGRyZXNzOiByZWNlaXZlckFkZHJlc3MsXG4gICAgICBhbW91bnQ6ICcwJyxcbiAgICAgIG11bHRpQXNzZXRzOiB7XG4gICAgICAgIGFzc2V0X25hbWU6IGFzY2lpRW5jb2RlZE5hbWUsXG4gICAgICAgIHBvbGljeV9pZDogcG9saWN5SWQsXG4gICAgICAgIHF1YW50aXR5LFxuICAgICAgICBmaW5nZXJwcmludCxcbiAgICAgIH0sXG4gICAgfSk7XG5cbiAgICByZWJ1aWxkVHhCdWlsZGVyLmNoYW5nZUFkZHJlc3Moc2VuZGVyQWRkcmVzcywgc2VuZGVySW5wdXRCYWxhbmNlLnRvU3RyaW5nKCksIHRvdGFsQXNzZXRMaXN0KTtcbiAgICByZWJ1aWxkVHhCdWlsZGVyLnNwb25zb3JzaGlwSW5mbyh7XG4gICAgICBmZWVBZGRyZXNzOiBmZWVBZGRyZXNzLFxuICAgICAgZmVlQWRkcmVzc0lucHV0QmFsYW5jZTogZmVlQWRkcmVzc0lucHV0QmFsYW5jZS50b1N0cmluZygpLFxuICAgICAgaXNSZWJ1aWxkOiB0cnVlLFxuICAgIH0pO1xuICAgIHJlYnVpbGRUeEJ1aWxkZXIudHRsKDgwMDAwMDAwMCk7XG4gICAgcmVidWlsZFR4QnVpbGRlci5pc1Rva2VuVHJhbnNhY3Rpb24oKTtcblxuICAgIGNvbnN0IHJlYnVpbHRUeCA9IChhd2FpdCByZWJ1aWxkVHhCdWlsZGVyLmJ1aWxkKCkpIGFzIFRyYW5zYWN0aW9uO1xuICAgIGNvbnN0IHJlYnVpbHRUeERhdGEgPSByZWJ1aWx0VHgudG9Kc29uKCk7XG5cbiAgICAvLyBWZXJpZnkgdGhlIHJlYnVpbHQgdHJhbnNhY3Rpb24gcHJlc2VydmVzIHRoZSBzYW1lIHN0cnVjdHVyZVxuICAgIHJlYnVpbHRUeERhdGEuaW5wdXRzLmxlbmd0aC5zaG91bGQuZXF1YWwoaW5pdGlhbFR4RGF0YS5pbnB1dHMubGVuZ3RoKTtcbiAgICByZWJ1aWx0VHhEYXRhLm91dHB1dHMubGVuZ3RoLnNob3VsZC5lcXVhbChpbml0aWFsVHhEYXRhLm91dHB1dHMubGVuZ3RoKTtcblxuICAgIC8vIEZlZSBzaG91bGQgYmUgcHJlc2VydmVkIGZyb20gdGhlIG9yaWdpbmFsIHRyYW5zYWN0aW9uXG4gICAgcmVidWlsdFR4LmdldEZlZS5zaG91bGQuZXF1YWwoaW5pdGlhbEZlZSk7XG5cbiAgICAvLyBWZXJpZnkgcmVjZWl2ZXIgb3V0cHV0IGlzIHByZXNlcnZlZFxuICAgIGNvbnN0IHJlY2VpdmVyT3V0cHV0ID0gcmVidWlsdFR4RGF0YS5vdXRwdXRzLmZpbHRlcigob3V0cHV0KSA9PiBvdXRwdXQuYWRkcmVzcyA9PT0gcmVjZWl2ZXJBZGRyZXNzKTtcbiAgICByZWNlaXZlck91dHB1dC5sZW5ndGguc2hvdWxkLmVxdWFsKDEpO1xuICAgIHJlY2VpdmVyT3V0cHV0WzBdLmFtb3VudC5zaG91bGQuZXF1YWwoJzE1MDAwMDAnKTtcbiAgfSk7XG5cbiAgZGVzY3JpYmUoJ0FkYVRva2VuIHZlcmlmeVRyYW5zYWN0aW9uJywgKCkgPT4ge1xuICAgIGxldCBiaXRnbzogVGVzdEJpdEdvQVBJO1xuICAgIGxldCBhZGFUb2tlbjtcblxuICAgIGJlZm9yZShmdW5jdGlvbiAoKSB7XG4gICAgICBiaXRnbyA9IFRlc3RCaXRHby5kZWNvcmF0ZShCaXRHb0FQSSwgeyBlbnY6ICdtb2NrJyB9KTtcbiAgICAgIGJpdGdvLmluaXRpYWxpemVUZXN0VmFycygpO1xuICAgICAgY29uc3QgdG9rZW5Db25maWcgPSB7XG4gICAgICAgIHR5cGU6ICd0YWRhOndhdGVyJyxcbiAgICAgICAgY29pbjogJ3RhZGEnLFxuICAgICAgICBuZXR3b3JrOiAnVGVzdG5ldCcgYXMgY29uc3QsXG4gICAgICAgIG5hbWU6ICdXQVRFUicsXG4gICAgICAgIGRlY2ltYWxQbGFjZXM6IDAsXG4gICAgICAgIHBvbGljeUlkOiBwb2xpY3lJZCxcbiAgICAgICAgYXNzZXROYW1lOiBuYW1lLCAvLyBBU0NJSSAnV0FURVInLCBub3QgaGV4LWVuY29kZWRcbiAgICAgICAgY29udHJhY3RBZGRyZXNzOiBgJHtwb2xpY3lJZH06JHthc2NpaUVuY29kZWROYW1lfWAsXG4gICAgICB9O1xuICAgICAgYWRhVG9rZW4gPSBuZXcgQWRhVG9rZW4oYml0Z28sIHRva2VuQ29uZmlnKTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgdmVyaWZ5IGEgdG9rZW4gdHJhbnNhY3Rpb24gd2l0aCBjb3JyZWN0IHRva2VuIGFtb3VudCcsIGFzeW5jICgpID0+IHtcbiAgICAgIGNvbnN0IHF1YW50aXR5ID0gJzIwJztcbiAgICAgIGNvbnN0IHRvdGFsSW5wdXQgPSAyMDAwMDAwMDtcbiAgICAgIGNvbnN0IHRvdGFsQXNzZXRMaXN0ID0ge1xuICAgICAgICBbZmluZ2VycHJpbnRdOiB7XG4gICAgICAgICAgcXVhbnRpdHk6ICcxMDAnLFxuICAgICAgICAgIHBvbGljeV9pZDogcG9saWN5SWQsXG4gICAgICAgICAgYXNzZXRfbmFtZTogYXNjaWlFbmNvZGVkTmFtZSxcbiAgICAgICAgfSxcbiAgICAgIH07XG5cbiAgICAgIGNvbnN0IHR4QnVpbGRlciA9IGZhY3RvcnkuZ2V0VHJhbnNmZXJCdWlsZGVyKCk7XG4gICAgICB0eEJ1aWxkZXIuaW5wdXQoe1xuICAgICAgICB0cmFuc2FjdGlvbl9pZDogJzM2NzdlNzVjN2JhNjk5YmZkYzZjZDU3ZDQyZjI0NmY4NmY2M2FlZmQ3NjAyNTAwNmFjNzgzMTNmYWQyYmJhMjEnLFxuICAgICAgICB0cmFuc2FjdGlvbl9pbmRleDogMSxcbiAgICAgIH0pO1xuXG4gICAgICB0eEJ1aWxkZXIub3V0cHV0KHtcbiAgICAgICAgYWRkcmVzczogcmVjZWl2ZXJBZGRyZXNzLFxuICAgICAgICBhbW91bnQ6ICcwJyxcbiAgICAgICAgbXVsdGlBc3NldHM6IHtcbiAgICAgICAgICBhc3NldF9uYW1lOiBhc2NpaUVuY29kZWROYW1lLFxuICAgICAgICAgIHBvbGljeV9pZDogcG9saWN5SWQsXG4gICAgICAgICAgcXVhbnRpdHksXG4gICAgICAgICAgZmluZ2VycHJpbnQsXG4gICAgICAgIH0sXG4gICAgICB9KTtcblxuICAgICAgdHhCdWlsZGVyLmNoYW5nZUFkZHJlc3Moc2VuZGVyQWRkcmVzcywgdG90YWxJbnB1dC50b1N0cmluZygpLCB0b3RhbEFzc2V0TGlzdCk7XG4gICAgICB0eEJ1aWxkZXIudHRsKDgwMDAwMDAwMCk7XG4gICAgICB0eEJ1aWxkZXIuaXNUb2tlblRyYW5zYWN0aW9uKCk7XG4gICAgICBjb25zdCB0eCA9IChhd2FpdCB0eEJ1aWxkZXIuYnVpbGQoKSkgYXMgVHJhbnNhY3Rpb247XG4gICAgICBjb25zdCB0eEhleCA9IHR4LnRvQnJvYWRjYXN0Rm9ybWF0KCk7XG5cbiAgICAgIC8vIFZlcmlmeSB0cmFuc2FjdGlvbiB3aXRoIGNvcnJlY3QgdG9rZW4gYW1vdW50XG4gICAgICBjb25zdCB0eFBhcmFtcyA9IHtcbiAgICAgICAgcmVjaXBpZW50czogW1xuICAgICAgICAgIHtcbiAgICAgICAgICAgIGFkZHJlc3M6IHJlY2VpdmVyQWRkcmVzcyxcbiAgICAgICAgICAgIGFtb3VudDogcXVhbnRpdHksIC8vIFRva2VuIGFtb3VudCwgbm90IEFEQSBhbW91bnRcbiAgICAgICAgICB9LFxuICAgICAgICBdLFxuICAgICAgfTtcblxuICAgICAgY29uc3QgdHhQcmVidWlsZCA9IHsgdHhIZXggfTtcbiAgICAgIGNvbnN0IG1vY2tXYWxsZXQgPSB7IGNvaW5TcGVjaWZpYzogKCkgPT4gKHsgYmFzZUFkZHJlc3M6IHNlbmRlckFkZHJlc3MgfSkgfTtcbiAgICAgIGNvbnN0IGlzVmVyaWZpZWQgPSBhd2FpdCBhZGFUb2tlbi52ZXJpZnlUcmFuc2FjdGlvbih7IHR4UGFyYW1zLCB0eFByZWJ1aWxkLCB3YWxsZXQ6IG1vY2tXYWxsZXQgYXMgYW55IH0pO1xuICAgICAgaXNWZXJpZmllZC5zaG91bGQuZXF1YWwodHJ1ZSk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIGZhaWwgdG8gdmVyaWZ5IGEgdG9rZW4gdHJhbnNhY3Rpb24gd2l0aCBpbmNvcnJlY3QgdG9rZW4gYW1vdW50JywgYXN5bmMgKCkgPT4ge1xuICAgICAgY29uc3QgcXVhbnRpdHkgPSAnMjAnO1xuICAgICAgY29uc3QgdG90YWxJbnB1dCA9IDIwMDAwMDAwO1xuICAgICAgY29uc3QgdG90YWxBc3NldExpc3QgPSB7XG4gICAgICAgIFtmaW5nZXJwcmludF06IHtcbiAgICAgICAgICBxdWFudGl0eTogJzEwMCcsXG4gICAgICAgICAgcG9saWN5X2lkOiBwb2xpY3lJZCxcbiAgICAgICAgICBhc3NldF9uYW1lOiBhc2NpaUVuY29kZWROYW1lLFxuICAgICAgICB9LFxuICAgICAgfTtcblxuICAgICAgY29uc3QgdHhCdWlsZGVyID0gZmFjdG9yeS5nZXRUcmFuc2ZlckJ1aWxkZXIoKTtcbiAgICAgIHR4QnVpbGRlci5pbnB1dCh7XG4gICAgICAgIHRyYW5zYWN0aW9uX2lkOiAnMzY3N2U3NWM3YmE2OTliZmRjNmNkNTdkNDJmMjQ2Zjg2ZjYzYWVmZDc2MDI1MDA2YWM3ODMxM2ZhZDJiYmEyMScsXG4gICAgICAgIHRyYW5zYWN0aW9uX2luZGV4OiAxLFxuICAgICAgfSk7XG5cbiAgICAgIHR4QnVpbGRlci5vdXRwdXQoe1xuICAgICAgICBhZGRyZXNzOiByZWNlaXZlckFkZHJlc3MsXG4gICAgICAgIGFtb3VudDogJzAnLFxuICAgICAgICBtdWx0aUFzc2V0czoge1xuICAgICAgICAgIGFzc2V0X25hbWU6IGFzY2lpRW5jb2RlZE5hbWUsXG4gICAgICAgICAgcG9saWN5X2lkOiBwb2xpY3lJZCxcbiAgICAgICAgICBxdWFudGl0eSxcbiAgICAgICAgICBmaW5nZXJwcmludCxcbiAgICAgICAgfSxcbiAgICAgIH0pO1xuXG4gICAgICB0eEJ1aWxkZXIuY2hhbmdlQWRkcmVzcyhzZW5kZXJBZGRyZXNzLCB0b3RhbElucHV0LnRvU3RyaW5nKCksIHRvdGFsQXNzZXRMaXN0KTtcbiAgICAgIHR4QnVpbGRlci50dGwoODAwMDAwMDAwKTtcbiAgICAgIHR4QnVpbGRlci5pc1Rva2VuVHJhbnNhY3Rpb24oKTtcbiAgICAgIGNvbnN0IHR4ID0gKGF3YWl0IHR4QnVpbGRlci5idWlsZCgpKSBhcyBUcmFuc2FjdGlvbjtcbiAgICAgIGNvbnN0IHR4SGV4ID0gdHgudG9Ccm9hZGNhc3RGb3JtYXQoKTtcblxuICAgICAgLy8gVmVyaWZ5IHRyYW5zYWN0aW9uIHdpdGggV1JPTkcgdG9rZW4gYW1vdW50IChzaG91bGQgZmFpbClcbiAgICAgIGNvbnN0IHR4UGFyYW1zID0ge1xuICAgICAgICByZWNpcGllbnRzOiBbXG4gICAgICAgICAge1xuICAgICAgICAgICAgYWRkcmVzczogcmVjZWl2ZXJBZGRyZXNzLFxuICAgICAgICAgICAgYW1vdW50OiAnOTk5JywgLy8gV3JvbmcgYW1vdW50XG4gICAgICAgICAgfSxcbiAgICAgICAgXSxcbiAgICAgIH07XG5cbiAgICAgIGNvbnN0IHR4UHJlYnVpbGQgPSB7IHR4SGV4IH07XG4gICAgICBjb25zdCBtb2NrV2FsbGV0ID0geyBjb2luU3BlY2lmaWM6ICgpID0+ICh7IGJhc2VBZGRyZXNzOiBzZW5kZXJBZGRyZXNzIH0pIH07XG4gICAgICBhd2FpdCBhZGFUb2tlblxuICAgICAgICAudmVyaWZ5VHJhbnNhY3Rpb24oeyB0eFBhcmFtcywgdHhQcmVidWlsZCwgd2FsbGV0OiBtb2NrV2FsbGV0IGFzIGFueSB9KVxuICAgICAgICAuc2hvdWxkLmJlLnJlamVjdGVkV2l0aCgnY2Fubm90IGZpbmQgcmVjaXBpZW50IGluIGV4cGVjdGVkIG91dHB1dCcpO1xuICAgIH0pO1xuXG4gICAgaXQoJ3Nob3VsZCBmYWlsIHRvIHZlcmlmeSB3aGVuIGFkZHJlc3MgZG9lcyBub3QgbWF0Y2gnLCBhc3luYyAoKSA9PiB7XG4gICAgICBjb25zdCBxdWFudGl0eSA9ICcyMCc7XG4gICAgICBjb25zdCB0b3RhbElucHV0ID0gMjAwMDAwMDA7XG4gICAgICBjb25zdCB0b3RhbEFzc2V0TGlzdCA9IHtcbiAgICAgICAgW2ZpbmdlcnByaW50XToge1xuICAgICAgICAgIHF1YW50aXR5OiAnMTAwJyxcbiAgICAgICAgICBwb2xpY3lfaWQ6IHBvbGljeUlkLFxuICAgICAgICAgIGFzc2V0X25hbWU6IGFzY2lpRW5jb2RlZE5hbWUsXG4gICAgICAgIH0sXG4gICAgICB9O1xuXG4gICAgICBjb25zdCB0eEJ1aWxkZXIgPSBmYWN0b3J5LmdldFRyYW5zZmVyQnVpbGRlcigpO1xuICAgICAgdHhCdWlsZGVyLmlucHV0KHtcbiAgICAgICAgdHJhbnNhY3Rpb25faWQ6ICczNjc3ZTc1YzdiYTY5OWJmZGM2Y2Q1N2Q0MmYyNDZmODZmNjNhZWZkNzYwMjUwMDZhYzc4MzEzZmFkMmJiYTIxJyxcbiAgICAgICAgdHJhbnNhY3Rpb25faW5kZXg6IDEsXG4gICAgICB9KTtcblxuICAgICAgdHhCdWlsZGVyLm91dHB1dCh7XG4gICAgICAgIGFkZHJlc3M6IHJlY2VpdmVyQWRkcmVzcyxcbiAgICAgICAgYW1vdW50OiAnMCcsXG4gICAgICAgIG11bHRpQXNzZXRzOiB7XG4gICAgICAgICAgYXNzZXRfbmFtZTogYXNjaWlFbmNvZGVkTmFtZSxcbiAgICAgICAgICBwb2xpY3lfaWQ6IHBvbGljeUlkLFxuICAgICAgICAgIHF1YW50aXR5LFxuICAgICAgICAgIGZpbmdlcnByaW50LFxuICAgICAgICB9LFxuICAgICAgfSk7XG5cbiAgICAgIHR4QnVpbGRlci5jaGFuZ2VBZGRyZXNzKHNlbmRlckFkZHJlc3MsIHRvdGFsSW5wdXQudG9TdHJpbmcoKSwgdG90YWxBc3NldExpc3QpO1xuICAgICAgdHhCdWlsZGVyLnR0bCg4MDAwMDAwMDApO1xuICAgICAgdHhCdWlsZGVyLmlzVG9rZW5UcmFuc2FjdGlvbigpO1xuICAgICAgY29uc3QgdHggPSAoYXdhaXQgdHhCdWlsZGVyLmJ1aWxkKCkpIGFzIFRyYW5zYWN0aW9uO1xuICAgICAgY29uc3QgdHhIZXggPSB0eC50b0Jyb2FkY2FzdEZvcm1hdCgpO1xuXG4gICAgICAvLyBWZXJpZnkgd2l0aCB3cm9uZyBhZGRyZXNzIChzaG91bGQgZmFpbClcbiAgICAgIGNvbnN0IHR4UGFyYW1zID0ge1xuICAgICAgICByZWNpcGllbnRzOiBbXG4gICAgICAgICAge1xuICAgICAgICAgICAgYWRkcmVzczpcbiAgICAgICAgICAgICAgJ2FkZHJfdGVzdDFxcWE4NmUzZDdsZnB3dTBrMnJoano3NmVjbWZ4ZHI3NHM5a2Y5eWZjcDVoajV2bW5oNnhjY2pjY2xyazhqdGF3OWpnZXV5OTlwMm44c210ZHB5bG15NDVxampmc2ZtcDNnNicsXG4gICAgICAgICAgICBhbW91bnQ6IHF1YW50aXR5LFxuICAgICAgICAgIH0sXG4gICAgICAgIF0sXG4gICAgICB9O1xuXG4gICAgICBjb25zdCB0eFByZWJ1aWxkID0geyB0eEhleCB9O1xuICAgICAgY29uc3QgbW9ja1dhbGxldCA9IHsgY29pblNwZWNpZmljOiAoKSA9PiAoeyBiYXNlQWRkcmVzczogc2VuZGVyQWRkcmVzcyB9KSB9O1xuICAgICAgYXdhaXQgYWRhVG9rZW5cbiAgICAgICAgLnZlcmlmeVRyYW5zYWN0aW9uKHsgdHhQYXJhbXMsIHR4UHJlYnVpbGQsIHdhbGxldDogbW9ja1dhbGxldCBhcyBhbnkgfSlcbiAgICAgICAgLnNob3VsZC5iZS5yZWplY3RlZFdpdGgoJ2Nhbm5vdCBmaW5kIHJlY2lwaWVudCBpbiBleHBlY3RlZCBvdXRwdXQnKTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgdmVyaWZ5IHRyYW5zYWN0aW9uIHdoZW4gcG9saWN5SWQgaGFzIGNvbmNhdGVuYXRlZCBhc3NldE5hbWUgKGNyeXB0byBjb21wYXJlIGZvcm1hdCknLCBhc3luYyAoKSA9PiB7XG4gICAgICAvLyBUaGlzIHRlc3RzIHRoZSBjYXNlIHdoZXJlIHBvbGljeUlkIGluIHRva2VuQ29uZmlnIGNvbnRhaW5zIHBvbGljeUlkICsgYXNjaWlFbmNvZGVkQXNzZXROYW1lXG4gICAgICAvLyB3aGljaCBpcyBjb25zaXN0ZW50IHdpdGggY3J5cHRvIGNvbXBhcmUgZm9ybWF0XG4gICAgICBjb25zdCBjb25jYXRlbmF0ZWRQb2xpY3lJZCA9IHBvbGljeUlkICsgYXNjaWlFbmNvZGVkTmFtZTsgLy8gZS5nLiwgJ2UxNmMyZGM4YWU5MzdlOGQzNzkwYzdmZDcxNjhkN2I5OTQ2MjFiYTE0Y2ExMTQxNWYzOWZlZDcyNTc0MTU0NDU1MidcblxuICAgICAgY29uc3QgdG9rZW5Db25maWdXaXRoQ29uY2F0ZW5hdGVkUG9saWN5SWQgPSB7XG4gICAgICAgIHR5cGU6ICd0YWRhOndhdGVyJyxcbiAgICAgICAgY29pbjogJ3RhZGEnLFxuICAgICAgICBuZXR3b3JrOiAnVGVzdG5ldCcgYXMgY29uc3QsXG4gICAgICAgIG5hbWU6ICdXQVRFUicsXG4gICAgICAgIGRlY2ltYWxQbGFjZXM6IDAsXG4gICAgICAgIHBvbGljeUlkOiBjb25jYXRlbmF0ZWRQb2xpY3lJZCwgLy8gcG9saWN5SWQgKyBhc3NldE5hbWUgaGV4XG4gICAgICAgIGFzc2V0TmFtZTogbmFtZSwgLy8gQVNDSUkgbmFtZSAnV0FURVInIChub3QgaGV4IGVuY29kZWQpXG4gICAgICAgIGNvbnRyYWN0QWRkcmVzczogYCR7cG9saWN5SWR9OiR7YXNjaWlFbmNvZGVkTmFtZX1gLFxuICAgICAgfTtcbiAgICAgIGNvbnN0IGFkYVRva2VuV2l0aENvbmNhdGVuYXRlZFBvbGljeUlkID0gbmV3IEFkYVRva2VuKGJpdGdvLCB0b2tlbkNvbmZpZ1dpdGhDb25jYXRlbmF0ZWRQb2xpY3lJZCk7XG5cbiAgICAgIGNvbnN0IHF1YW50aXR5ID0gJzIwJztcbiAgICAgIGNvbnN0IHRvdGFsSW5wdXQgPSAyMDAwMDAwMDtcbiAgICAgIGNvbnN0IHRvdGFsQXNzZXRMaXN0ID0ge1xuICAgICAgICBbZmluZ2VycHJpbnRdOiB7XG4gICAgICAgICAgcXVhbnRpdHk6ICcxMDAnLFxuICAgICAgICAgIHBvbGljeV9pZDogcG9saWN5SWQsXG4gICAgICAgICAgYXNzZXRfbmFtZTogYXNjaWlFbmNvZGVkTmFtZSxcbiAgICAgICAgfSxcbiAgICAgIH07XG5cbiAgICAgIGNvbnN0IHR4QnVpbGRlciA9IGZhY3RvcnkuZ2V0VHJhbnNmZXJCdWlsZGVyKCk7XG4gICAgICB0eEJ1aWxkZXIuaW5wdXQoe1xuICAgICAgICB0cmFuc2FjdGlvbl9pZDogJzM2NzdlNzVjN2JhNjk5YmZkYzZjZDU3ZDQyZjI0NmY4NmY2M2FlZmQ3NjAyNTAwNmFjNzgzMTNmYWQyYmJhMjEnLFxuICAgICAgICB0cmFuc2FjdGlvbl9pbmRleDogMSxcbiAgICAgIH0pO1xuXG4gICAgICB0eEJ1aWxkZXIub3V0cHV0KHtcbiAgICAgICAgYWRkcmVzczogcmVjZWl2ZXJBZGRyZXNzLFxuICAgICAgICBhbW91bnQ6ICcwJyxcbiAgICAgICAgbXVsdGlBc3NldHM6IHtcbiAgICAgICAgICBhc3NldF9uYW1lOiBhc2NpaUVuY29kZWROYW1lLFxuICAgICAgICAgIHBvbGljeV9pZDogcG9saWN5SWQsXG4gICAgICAgICAgcXVhbnRpdHksXG4gICAgICAgICAgZmluZ2VycHJpbnQsXG4gICAgICAgIH0sXG4gICAgICB9KTtcblxuICAgICAgdHhCdWlsZGVyLmNoYW5nZUFkZHJlc3Moc2VuZGVyQWRkcmVzcywgdG90YWxJbnB1dC50b1N0cmluZygpLCB0b3RhbEFzc2V0TGlzdCk7XG4gICAgICB0eEJ1aWxkZXIudHRsKDgwMDAwMDAwMCk7XG4gICAgICB0eEJ1aWxkZXIuaXNUb2tlblRyYW5zYWN0aW9uKCk7XG4gICAgICBjb25zdCB0eCA9IChhd2FpdCB0eEJ1aWxkZXIuYnVpbGQoKSkgYXMgVHJhbnNhY3Rpb247XG4gICAgICBjb25zdCB0eEhleCA9IHR4LnRvQnJvYWRjYXN0Rm9ybWF0KCk7XG5cbiAgICAgIC8vIFZlcmlmeSB0cmFuc2FjdGlvbiAtIHRoZSB2ZXJpZnlUcmFuc2FjdGlvbiBzaG91bGQgc3RyaXAgdGhlIGFzc2V0TmFtZSBmcm9tIHBvbGljeUlkXG4gICAgICBjb25zdCB0eFBhcmFtcyA9IHtcbiAgICAgICAgcmVjaXBpZW50czogW1xuICAgICAgICAgIHtcbiAgICAgICAgICAgIGFkZHJlc3M6IHJlY2VpdmVyQWRkcmVzcyxcbiAgICAgICAgICAgIGFtb3VudDogcXVhbnRpdHksXG4gICAgICAgICAgfSxcbiAgICAgICAgXSxcbiAgICAgIH07XG5cbiAgICAgIGNvbnN0IHR4UHJlYnVpbGQgPSB7IHR4SGV4IH07XG4gICAgICBjb25zdCBtb2NrV2FsbGV0ID0geyBjb2luU3BlY2lmaWM6ICgpID0+ICh7IGJhc2VBZGRyZXNzOiBzZW5kZXJBZGRyZXNzIH0pIH07XG4gICAgICBjb25zdCBpc1ZlcmlmaWVkID0gYXdhaXQgYWRhVG9rZW5XaXRoQ29uY2F0ZW5hdGVkUG9saWN5SWQudmVyaWZ5VHJhbnNhY3Rpb24oe1xuICAgICAgICB0eFBhcmFtcyxcbiAgICAgICAgdHhQcmVidWlsZCxcbiAgICAgICAgd2FsbGV0OiBtb2NrV2FsbGV0IGFzIGFueSxcbiAgICAgIH0pO1xuICAgICAgaXNWZXJpZmllZC5zaG91bGQuZXF1YWwodHJ1ZSk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIHZlcmlmeSB0cmFuc2FjdGlvbiB3aXRoIHBvbGljeUlkIHRoYXQgZG9lcyBub3QgaGF2ZSBjb25jYXRlbmF0ZWQgYXNzZXROYW1lJywgYXN5bmMgKCkgPT4ge1xuICAgICAgLy8gVGhpcyB0ZXN0cyB0aGUgY2FzZSB3aGVyZSBwb2xpY3lJZCBpcyBqdXN0IHRoZSAyOC1ieXRlIHBvbGljeSBJRCAobm8gYXNzZXROYW1lIGFwcGVuZGVkKVxuICAgICAgY29uc3QgdG9rZW5Db25maWdXaXRoUGxhaW5Qb2xpY3lJZCA9IHtcbiAgICAgICAgdHlwZTogJ3RhZGE6d2F0ZXInLFxuICAgICAgICBjb2luOiAndGFkYScsXG4gICAgICAgIG5ldHdvcms6ICdUZXN0bmV0JyBhcyBjb25zdCxcbiAgICAgICAgbmFtZTogJ1dBVEVSJyxcbiAgICAgICAgZGVjaW1hbFBsYWNlczogMCxcbiAgICAgICAgcG9saWN5SWQ6IHBvbGljeUlkLCAvLyBKdXN0IHRoZSBwb2xpY3kgSUQgd2l0aG91dCBhc3NldE5hbWVcbiAgICAgICAgYXNzZXROYW1lOiBuYW1lLCAvLyBBU0NJSSBuYW1lICdXQVRFUidcbiAgICAgICAgY29udHJhY3RBZGRyZXNzOiBgJHtwb2xpY3lJZH06JHthc2NpaUVuY29kZWROYW1lfWAsXG4gICAgICB9O1xuICAgICAgY29uc3QgYWRhVG9rZW5XaXRoUGxhaW5Qb2xpY3lJZCA9IG5ldyBBZGFUb2tlbihiaXRnbywgdG9rZW5Db25maWdXaXRoUGxhaW5Qb2xpY3lJZCk7XG5cbiAgICAgIGNvbnN0IHF1YW50aXR5ID0gJzIwJztcbiAgICAgIGNvbnN0IHRvdGFsSW5wdXQgPSAyMDAwMDAwMDtcbiAgICAgIGNvbnN0IHRvdGFsQXNzZXRMaXN0ID0ge1xuICAgICAgICBbZmluZ2VycHJpbnRdOiB7XG4gICAgICAgICAgcXVhbnRpdHk6ICcxMDAnLFxuICAgICAgICAgIHBvbGljeV9pZDogcG9saWN5SWQsXG4gICAgICAgICAgYXNzZXRfbmFtZTogYXNjaWlFbmNvZGVkTmFtZSxcbiAgICAgICAgfSxcbiAgICAgIH07XG5cbiAgICAgIGNvbnN0IHR4QnVpbGRlciA9IGZhY3RvcnkuZ2V0VHJhbnNmZXJCdWlsZGVyKCk7XG4gICAgICB0eEJ1aWxkZXIuaW5wdXQoe1xuICAgICAgICB0cmFuc2FjdGlvbl9pZDogJzM2NzdlNzVjN2JhNjk5YmZkYzZjZDU3ZDQyZjI0NmY4NmY2M2FlZmQ3NjAyNTAwNmFjNzgzMTNmYWQyYmJhMjEnLFxuICAgICAgICB0cmFuc2FjdGlvbl9pbmRleDogMSxcbiAgICAgIH0pO1xuXG4gICAgICB0eEJ1aWxkZXIub3V0cHV0KHtcbiAgICAgICAgYWRkcmVzczogcmVjZWl2ZXJBZGRyZXNzLFxuICAgICAgICBhbW91bnQ6ICcwJyxcbiAgICAgICAgbXVsdGlBc3NldHM6IHtcbiAgICAgICAgICBhc3NldF9uYW1lOiBhc2NpaUVuY29kZWROYW1lLFxuICAgICAgICAgIHBvbGljeV9pZDogcG9saWN5SWQsXG4gICAgICAgICAgcXVhbnRpdHksXG4gICAgICAgICAgZmluZ2VycHJpbnQsXG4gICAgICAgIH0sXG4gICAgICB9KTtcblxuICAgICAgdHhCdWlsZGVyLmNoYW5nZUFkZHJlc3Moc2VuZGVyQWRkcmVzcywgdG90YWxJbnB1dC50b1N0cmluZygpLCB0b3RhbEFzc2V0TGlzdCk7XG4gICAgICB0eEJ1aWxkZXIudHRsKDgwMDAwMDAwMCk7XG4gICAgICB0eEJ1aWxkZXIuaXNUb2tlblRyYW5zYWN0aW9uKCk7XG4gICAgICBjb25zdCB0eCA9IChhd2FpdCB0eEJ1aWxkZXIuYnVpbGQoKSkgYXMgVHJhbnNhY3Rpb247XG4gICAgICBjb25zdCB0eEhleCA9IHR4LnRvQnJvYWRjYXN0Rm9ybWF0KCk7XG5cbiAgICAgIC8vIFZlcmlmeSB0cmFuc2FjdGlvbiAtIHNob3VsZCB3b3JrIHdpdGggcGxhaW4gcG9saWN5SWQgYXMgd2VsbFxuICAgICAgY29uc3QgdHhQYXJhbXMgPSB7XG4gICAgICAgIHJlY2lwaWVudHM6IFtcbiAgICAgICAgICB7XG4gICAgICAgICAgICBhZGRyZXNzOiByZWNlaXZlckFkZHJlc3MsXG4gICAgICAgICAgICBhbW91bnQ6IHF1YW50aXR5LFxuICAgICAgICAgIH0sXG4gICAgICAgIF0sXG4gICAgICB9O1xuXG4gICAgICBjb25zdCB0eFByZWJ1aWxkID0geyB0eEhleCB9O1xuICAgICAgY29uc3QgbW9ja1dhbGxldCA9IHsgY29pblNwZWNpZmljOiAoKSA9PiAoeyBiYXNlQWRkcmVzczogc2VuZGVyQWRkcmVzcyB9KSB9O1xuICAgICAgY29uc3QgaXNWZXJpZmllZCA9IGF3YWl0IGFkYVRva2VuV2l0aFBsYWluUG9saWN5SWQudmVyaWZ5VHJhbnNhY3Rpb24oe1xuICAgICAgICB0eFBhcmFtcyxcbiAgICAgICAgdHhQcmVidWlsZCxcbiAgICAgICAgd2FsbGV0OiBtb2NrV2FsbGV0IGFzIGFueSxcbiAgICAgIH0pO1xuICAgICAgaXNWZXJpZmllZC5zaG91bGQuZXF1YWwodHJ1ZSk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIHZlcmlmeSB0b2tlbiBjb25zb2xpZGF0aW9uIHRyYW5zYWN0aW9uIHdoZW4gYWxsIG91dHB1dHMgZ28gdG8gYmFzZSBhZGRyZXNzJywgYXN5bmMgKCkgPT4ge1xuICAgICAgY29uc3QgcXVhbnRpdHkgPSAnMTAwJztcbiAgICAgIGNvbnN0IHRvdGFsSW5wdXQgPSAyMDAwMDAwMDtcbiAgICAgIGNvbnN0IHRvdGFsQXNzZXRMaXN0ID0ge1xuICAgICAgICBbZmluZ2VycHJpbnRdOiB7XG4gICAgICAgICAgcXVhbnRpdHk6ICcxMDAnLFxuICAgICAgICAgIHBvbGljeV9pZDogcG9saWN5SWQsXG4gICAgICAgICAgYXNzZXRfbmFtZTogYXNjaWlFbmNvZGVkTmFtZSxcbiAgICAgICAgfSxcbiAgICAgIH07XG5cbiAgICAgIC8vIEJ1aWxkIGEgY29uc29saWRhdGlvbiB0cmFuc2FjdGlvbiAtIGFsbCBvdXRwdXRzIGdvIHRvIHNlbmRlciAoYmFzZSkgYWRkcmVzc1xuICAgICAgY29uc3QgdHhCdWlsZGVyID0gZmFjdG9yeS5nZXRUcmFuc2ZlckJ1aWxkZXIoKTtcbiAgICAgIHR4QnVpbGRlci5pbnB1dCh7XG4gICAgICAgIHRyYW5zYWN0aW9uX2lkOiAnMzY3N2U3NWM3YmE2OTliZmRjNmNkNTdkNDJmMjQ2Zjg2ZjYzYWVmZDc2MDI1MDA2YWM3ODMxM2ZhZDJiYmEyMScsXG4gICAgICAgIHRyYW5zYWN0aW9uX2luZGV4OiAxLFxuICAgICAgfSk7XG5cbiAgICAgIC8vIEZvciBjb25zb2xpZGF0aW9uLCB0b2tlbnMgZ28gYmFjayB0byB0aGUgc2VuZGVyJ3MgYmFzZSBhZGRyZXNzXG4gICAgICB0eEJ1aWxkZXIub3V0cHV0KHtcbiAgICAgICAgYWRkcmVzczogc2VuZGVyQWRkcmVzcyxcbiAgICAgICAgYW1vdW50OiAnMCcsXG4gICAgICAgIG11bHRpQXNzZXRzOiB7XG4gICAgICAgICAgYXNzZXRfbmFtZTogYXNjaWlFbmNvZGVkTmFtZSxcbiAgICAgICAgICBwb2xpY3lfaWQ6IHBvbGljeUlkLFxuICAgICAgICAgIHF1YW50aXR5LFxuICAgICAgICAgIGZpbmdlcnByaW50LFxuICAgICAgICB9LFxuICAgICAgfSk7XG5cbiAgICAgIHR4QnVpbGRlci5jaGFuZ2VBZGRyZXNzKHNlbmRlckFkZHJlc3MsIHRvdGFsSW5wdXQudG9TdHJpbmcoKSwgdG90YWxBc3NldExpc3QpO1xuICAgICAgdHhCdWlsZGVyLnR0bCg4MDAwMDAwMDApO1xuICAgICAgdHhCdWlsZGVyLmlzVG9rZW5UcmFuc2FjdGlvbigpO1xuICAgICAgY29uc3QgdHggPSAoYXdhaXQgdHhCdWlsZGVyLmJ1aWxkKCkpIGFzIFRyYW5zYWN0aW9uO1xuICAgICAgY29uc3QgdHhIZXggPSB0eC50b0Jyb2FkY2FzdEZvcm1hdCgpO1xuXG4gICAgICAvLyBNb2NrIHdhbGxldCB3aXRoIGNvaW5TcGVjaWZpYyByZXR1cm5pbmcgYmFzZSBhZGRyZXNzXG4gICAgICBjb25zdCBtb2NrV2FsbGV0ID0ge1xuICAgICAgICBjb2luU3BlY2lmaWM6ICgpID0+ICh7XG4gICAgICAgICAgYmFzZUFkZHJlc3M6IHNlbmRlckFkZHJlc3MsXG4gICAgICAgIH0pLFxuICAgICAgfTtcblxuICAgICAgLy8gVmVyaWZ5IGNvbnNvbGlkYXRpb24gdHJhbnNhY3Rpb24gLSBubyByZWNpcGllbnRzLCBidXQgY29uc29saWRhdGlvblRvQmFzZUFkZHJlc3MgaXMgdHJ1ZVxuICAgICAgY29uc3QgdHhQYXJhbXMgPSB7XG4gICAgICAgIHJlY2lwaWVudHM6IHVuZGVmaW5lZCxcbiAgICAgIH07XG5cbiAgICAgIGNvbnN0IHR4UHJlYnVpbGQgPSB7IHR4SGV4IH07XG4gICAgICBjb25zdCB2ZXJpZmljYXRpb24gPSB7IGNvbnNvbGlkYXRpb25Ub0Jhc2VBZGRyZXNzOiB0cnVlIH07XG5cbiAgICAgIGNvbnN0IGlzVmVyaWZpZWQgPSBhd2FpdCBhZGFUb2tlbi52ZXJpZnlUcmFuc2FjdGlvbih7XG4gICAgICAgIHR4UGFyYW1zLFxuICAgICAgICB0eFByZWJ1aWxkLFxuICAgICAgICB2ZXJpZmljYXRpb24sXG4gICAgICAgIHdhbGxldDogbW9ja1dhbGxldCBhcyBhbnksXG4gICAgICB9KTtcbiAgICAgIGlzVmVyaWZpZWQuc2hvdWxkLmVxdWFsKHRydWUpO1xuICAgIH0pO1xuXG4gICAgaXQoJ3Nob3VsZCBmYWlsIHRva2VuIGNvbnNvbGlkYXRpb24gd2hlbiBvdXRwdXQgYWRkcmVzcyBkb2VzIG5vdCBtYXRjaCBiYXNlIGFkZHJlc3MnLCBhc3luYyAoKSA9PiB7XG4gICAgICBjb25zdCBxdWFudGl0eSA9ICcxMDAnO1xuICAgICAgY29uc3QgdG90YWxJbnB1dCA9IDIwMDAwMDAwO1xuICAgICAgY29uc3QgdG90YWxBc3NldExpc3QgPSB7XG4gICAgICAgIFtmaW5nZXJwcmludF06IHtcbiAgICAgICAgICBxdWFudGl0eTogJzEwMCcsXG4gICAgICAgICAgcG9saWN5X2lkOiBwb2xpY3lJZCxcbiAgICAgICAgICBhc3NldF9uYW1lOiBhc2NpaUVuY29kZWROYW1lLFxuICAgICAgICB9LFxuICAgICAgfTtcblxuICAgICAgLy8gQnVpbGQgYSB0cmFuc2FjdGlvbiB3aXRoIG91dHB1dCB0byByZWNlaXZlciAobm90IGJhc2UgYWRkcmVzcylcbiAgICAgIGNvbnN0IHR4QnVpbGRlciA9IGZhY3RvcnkuZ2V0VHJhbnNmZXJCdWlsZGVyKCk7XG4gICAgICB0eEJ1aWxkZXIuaW5wdXQoe1xuICAgICAgICB0cmFuc2FjdGlvbl9pZDogJzM2NzdlNzVjN2JhNjk5YmZkYzZjZDU3ZDQyZjI0NmY4NmY2M2FlZmQ3NjAyNTAwNmFjNzgzMTNmYWQyYmJhMjEnLFxuICAgICAgICB0cmFuc2FjdGlvbl9pbmRleDogMSxcbiAgICAgIH0pO1xuXG4gICAgICB0eEJ1aWxkZXIub3V0cHV0KHtcbiAgICAgICAgYWRkcmVzczogcmVjZWl2ZXJBZGRyZXNzLCAvLyBPdXRwdXQgZ29lcyB0byByZWNlaXZlciwgbm90IGJhc2UgYWRkcmVzc1xuICAgICAgICBhbW91bnQ6ICcwJyxcbiAgICAgICAgbXVsdGlBc3NldHM6IHtcbiAgICAgICAgICBhc3NldF9uYW1lOiBhc2NpaUVuY29kZWROYW1lLFxuICAgICAgICAgIHBvbGljeV9pZDogcG9saWN5SWQsXG4gICAgICAgICAgcXVhbnRpdHksXG4gICAgICAgICAgZmluZ2VycHJpbnQsXG4gICAgICAgIH0sXG4gICAgICB9KTtcblxuICAgICAgdHhCdWlsZGVyLmNoYW5nZUFkZHJlc3Moc2VuZGVyQWRkcmVzcywgdG90YWxJbnB1dC50b1N0cmluZygpLCB0b3RhbEFzc2V0TGlzdCk7XG4gICAgICB0eEJ1aWxkZXIudHRsKDgwMDAwMDAwMCk7XG4gICAgICB0eEJ1aWxkZXIuaXNUb2tlblRyYW5zYWN0aW9uKCk7XG4gICAgICBjb25zdCB0eCA9IChhd2FpdCB0eEJ1aWxkZXIuYnVpbGQoKSkgYXMgVHJhbnNhY3Rpb247XG4gICAgICBjb25zdCB0eEhleCA9IHR4LnRvQnJvYWRjYXN0Rm9ybWF0KCk7XG5cbiAgICAgIC8vIE1vY2sgd2FsbGV0IHdpdGggZGlmZmVyZW50IGJhc2UgYWRkcmVzc1xuICAgICAgY29uc3QgbW9ja1dhbGxldCA9IHtcbiAgICAgICAgY29pblNwZWNpZmljOiAoKSA9PiAoe1xuICAgICAgICAgIGJhc2VBZGRyZXNzOiBzZW5kZXJBZGRyZXNzLCAvLyBCYXNlIGFkZHJlc3MgaXMgc2VuZGVyLCBidXQgb3V0cHV0IGdvZXMgdG8gcmVjZWl2ZXJcbiAgICAgICAgfSksXG4gICAgICB9O1xuXG4gICAgICBjb25zdCB0eFBhcmFtcyA9IHtcbiAgICAgICAgcmVjaXBpZW50czogdW5kZWZpbmVkLFxuICAgICAgfTtcblxuICAgICAgY29uc3QgdHhQcmVidWlsZCA9IHsgdHhIZXggfTtcbiAgICAgIGNvbnN0IHZlcmlmaWNhdGlvbiA9IHsgY29uc29saWRhdGlvblRvQmFzZUFkZHJlc3M6IHRydWUgfTtcblxuICAgICAgYXdhaXQgYWRhVG9rZW5cbiAgICAgICAgLnZlcmlmeVRyYW5zYWN0aW9uKHtcbiAgICAgICAgICB0eFBhcmFtcyxcbiAgICAgICAgICB0eFByZWJ1aWxkLFxuICAgICAgICAgIHZlcmlmaWNhdGlvbixcbiAgICAgICAgICB3YWxsZXQ6IG1vY2tXYWxsZXQgYXMgYW55LFxuICAgICAgICB9KVxuICAgICAgICAuc2hvdWxkLmJlLnJlamVjdGVkV2l0aCgndHggb3V0cHV0cyBkb2VzIG5vdCBtYXRjaCB3aXRoIGV4cGVjdGVkIGFkZHJlc3MnKTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgdmVyaWZ5IHNwb25zb3JlZCB0b2tlbiBjb25zb2xpZGF0aW9uIHdoZW4gZmVlIGFkZHJlc3Mgb3V0cHV0IGlzIHByZXNlbnQnLCBhc3luYyAoKSA9PiB7XG4gICAgICBjb25zdCBmZWVBZGRyZXNzID1cbiAgICAgICAgJ2FkZHJfdGVzdDFxejJmeHYydW15aHR0a3h5eHA4eDBkbHBkdDNrNmN3bmc1cHhqM2poc3lkemVyM2pjdTVkOHBzN3pleDJrMnh0M3VxeGdqcW5uajgzd3M4bGhybjY0OGpqeHR3cTJ5dGpxcCc7XG4gICAgICBjb25zdCBxdWFudGl0eSA9ICcxMDAnO1xuICAgICAgY29uc3Qgc2VuZGVySW5wdXRCYWxhbmNlID0gNTAwMDAwMDtcbiAgICAgIGNvbnN0IGZlZUFkZHJlc3NJbnB1dEJhbGFuY2UgPSAyMDAwMDAwMDtcbiAgICAgIGNvbnN0IHRvdGFsQXNzZXRMaXN0ID0ge1xuICAgICAgICBbZmluZ2VycHJpbnRdOiB7XG4gICAgICAgICAgcXVhbnRpdHk6ICcxMDAnLFxuICAgICAgICAgIHBvbGljeV9pZDogcG9saWN5SWQsXG4gICAgICAgICAgYXNzZXRfbmFtZTogYXNjaWlFbmNvZGVkTmFtZSxcbiAgICAgICAgfSxcbiAgICAgIH07XG5cbiAgICAgIC8vIEJ1aWxkIGEgc3BvbnNvcmVkIHRva2VuIGNvbnNvbGlkYXRpb246IHRva2VucyBnbyBiYWNrIHRvIHNlbmRlckFkZHJlc3MsIGZlZSBzcG9uc29yIGdldHMgQURBIGNoYW5nZVxuICAgICAgY29uc3QgdHhCdWlsZGVyID0gZmFjdG9yeS5nZXRUcmFuc2ZlckJ1aWxkZXIoKTtcbiAgICAgIHR4QnVpbGRlci5pbnB1dCh7XG4gICAgICAgIHRyYW5zYWN0aW9uX2lkOiAnMzY3N2U3NWM3YmE2OTliZmRjNmNkNTdkNDJmMjQ2Zjg2ZjYzYWVmZDc2MDI1MDA2YWM3ODMxM2ZhZDJiYmEyMScsXG4gICAgICAgIHRyYW5zYWN0aW9uX2luZGV4OiAxLFxuICAgICAgfSk7XG4gICAgICB0eEJ1aWxkZXIuaW5wdXQoe1xuICAgICAgICB0cmFuc2FjdGlvbl9pZDogJzM2NzdlNzVjN2JhNjk5YmZkYzZjZDU3ZDQyZjI0NmY4NmY2M2FlZmQ3NjAyNTAwNmFjNzgzMTNmYWQyYmJhMjInLFxuICAgICAgICB0cmFuc2FjdGlvbl9pbmRleDogMCxcbiAgICAgIH0pO1xuXG4gICAgICAvLyBDb25zb2xpZGF0aW9uIG91dHB1dDogdG9rZW5zIGdvIGJhY2sgdG8gdGhlIHNlbmRlcidzIGJhc2UgYWRkcmVzc1xuICAgICAgdHhCdWlsZGVyLm91dHB1dCh7XG4gICAgICAgIGFkZHJlc3M6IHNlbmRlckFkZHJlc3MsXG4gICAgICAgIGFtb3VudDogJzAnLFxuICAgICAgICBtdWx0aUFzc2V0czoge1xuICAgICAgICAgIGFzc2V0X25hbWU6IGFzY2lpRW5jb2RlZE5hbWUsXG4gICAgICAgICAgcG9saWN5X2lkOiBwb2xpY3lJZCxcbiAgICAgICAgICBxdWFudGl0eSxcbiAgICAgICAgICBmaW5nZXJwcmludCxcbiAgICAgICAgfSxcbiAgICAgIH0pO1xuXG4gICAgICB0eEJ1aWxkZXIuY2hhbmdlQWRkcmVzcyhzZW5kZXJBZGRyZXNzLCBzZW5kZXJJbnB1dEJhbGFuY2UudG9TdHJpbmcoKSwgdG90YWxBc3NldExpc3QpO1xuICAgICAgdHhCdWlsZGVyLnNwb25zb3JzaGlwSW5mbyh7XG4gICAgICAgIGZlZUFkZHJlc3M6IGZlZUFkZHJlc3MsXG4gICAgICAgIGZlZUFkZHJlc3NJbnB1dEJhbGFuY2U6IGZlZUFkZHJlc3NJbnB1dEJhbGFuY2UudG9TdHJpbmcoKSxcbiAgICAgIH0pO1xuICAgICAgdHhCdWlsZGVyLnR0bCg4MDAwMDAwMDApO1xuICAgICAgdHhCdWlsZGVyLmlzVG9rZW5UcmFuc2FjdGlvbigpO1xuICAgICAgY29uc3QgdHggPSAoYXdhaXQgdHhCdWlsZGVyLmJ1aWxkKCkpIGFzIFRyYW5zYWN0aW9uO1xuICAgICAgY29uc3QgdHhIZXggPSB0eC50b0Jyb2FkY2FzdEZvcm1hdCgpO1xuXG4gICAgICBjb25zdCBtb2NrV2FsbGV0ID0ge1xuICAgICAgICBjb2luU3BlY2lmaWM6ICgpID0+ICh7XG4gICAgICAgICAgYmFzZUFkZHJlc3M6IHNlbmRlckFkZHJlc3MsXG4gICAgICAgIH0pLFxuICAgICAgfTtcblxuICAgICAgY29uc3QgdHhQYXJhbXMgPSB7IHJlY2lwaWVudHM6IHVuZGVmaW5lZCB9O1xuICAgICAgY29uc3QgdHhQcmVidWlsZCA9IHsgdHhIZXgsIHR4SW5mbzogeyBmZWVBZGRyZXNzIH0gfTtcbiAgICAgIGNvbnN0IHZlcmlmaWNhdGlvbiA9IHsgY29uc29saWRhdGlvblRvQmFzZUFkZHJlc3M6IHRydWUgfTtcblxuICAgICAgY29uc3QgaXNWZXJpZmllZCA9IGF3YWl0IGFkYVRva2VuLnZlcmlmeVRyYW5zYWN0aW9uKHtcbiAgICAgICAgdHhQYXJhbXMsXG4gICAgICAgIHR4UHJlYnVpbGQsXG4gICAgICAgIHZlcmlmaWNhdGlvbixcbiAgICAgICAgd2FsbGV0OiBtb2NrV2FsbGV0IGFzIGFueSxcbiAgICAgIH0pO1xuICAgICAgaXNWZXJpZmllZC5zaG91bGQuZXF1YWwodHJ1ZSk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIGZhaWwgc3BvbnNvcmVkIHRva2VuIGNvbnNvbGlkYXRpb24gd2hlbiBvdXRwdXQgZ29lcyB0byB1bmV4cGVjdGVkIHRoaXJkLXBhcnR5IGFkZHJlc3MnLCBhc3luYyAoKSA9PiB7XG4gICAgICBjb25zdCBmZWVBZGRyZXNzID1cbiAgICAgICAgJ2FkZHJfdGVzdDFxejJmeHYydW15aHR0a3h5eHA4eDBkbHBkdDNrNmN3bmc1cHhqM2poc3lkemVyM2pjdTVkOHBzN3pleDJrMnh0M3VxeGdqcW5uajgzd3M4bGhybjY0OGpqeHR3cTJ5dGpxcCc7XG4gICAgICBjb25zdCBxdWFudGl0eSA9ICcxMDAnO1xuICAgICAgY29uc3Qgc2VuZGVySW5wdXRCYWxhbmNlID0gNTAwMDAwMDtcbiAgICAgIGNvbnN0IGZlZUFkZHJlc3NJbnB1dEJhbGFuY2UgPSAyMDAwMDAwMDtcbiAgICAgIGNvbnN0IHRvdGFsQXNzZXRMaXN0ID0ge1xuICAgICAgICBbZmluZ2VycHJpbnRdOiB7XG4gICAgICAgICAgcXVhbnRpdHk6ICcxMDAnLFxuICAgICAgICAgIHBvbGljeV9pZDogcG9saWN5SWQsXG4gICAgICAgICAgYXNzZXRfbmFtZTogYXNjaWlFbmNvZGVkTmFtZSxcbiAgICAgICAgfSxcbiAgICAgIH07XG5cbiAgICAgIC8vIEJ1aWxkIHRyYW5zYWN0aW9uIHdoZXJlIHRva2VuIG91dHB1dCBnb2VzIHRvIGFuIHVuZXhwZWN0ZWQgdGhpcmQtcGFydHkgYWRkcmVzc1xuICAgICAgY29uc3QgdHhCdWlsZGVyID0gZmFjdG9yeS5nZXRUcmFuc2ZlckJ1aWxkZXIoKTtcbiAgICAgIHR4QnVpbGRlci5pbnB1dCh7XG4gICAgICAgIHRyYW5zYWN0aW9uX2lkOiAnMzY3N2U3NWM3YmE2OTliZmRjNmNkNTdkNDJmMjQ2Zjg2ZjYzYWVmZDc2MDI1MDA2YWM3ODMxM2ZhZDJiYmEyMScsXG4gICAgICAgIHRyYW5zYWN0aW9uX2luZGV4OiAxLFxuICAgICAgfSk7XG4gICAgICB0eEJ1aWxkZXIuaW5wdXQoe1xuICAgICAgICB0cmFuc2FjdGlvbl9pZDogJzM2NzdlNzVjN2JhNjk5YmZkYzZjZDU3ZDQyZjI0NmY4NmY2M2FlZmQ3NjAyNTAwNmFjNzgzMTNmYWQyYmJhMjInLFxuICAgICAgICB0cmFuc2FjdGlvbl9pbmRleDogMCxcbiAgICAgIH0pO1xuXG4gICAgICAvLyBPdXRwdXQgZ29lcyB0byByZWNlaXZlckFkZHJlc3MgKHRoaXJkIHBhcnR5KSwgbm90IHRoZSBiYXNlIGFkZHJlc3NcbiAgICAgIHR4QnVpbGRlci5vdXRwdXQoe1xuICAgICAgICBhZGRyZXNzOiByZWNlaXZlckFkZHJlc3MsXG4gICAgICAgIGFtb3VudDogJzAnLFxuICAgICAgICBtdWx0aUFzc2V0czoge1xuICAgICAgICAgIGFzc2V0X25hbWU6IGFzY2lpRW5jb2RlZE5hbWUsXG4gICAgICAgICAgcG9saWN5X2lkOiBwb2xpY3lJZCxcbiAgICAgICAgICBxdWFudGl0eSxcbiAgICAgICAgICBmaW5nZXJwcmludCxcbiAgICAgICAgfSxcbiAgICAgIH0pO1xuXG4gICAgICB0eEJ1aWxkZXIuY2hhbmdlQWRkcmVzcyhzZW5kZXJBZGRyZXNzLCBzZW5kZXJJbnB1dEJhbGFuY2UudG9TdHJpbmcoKSwgdG90YWxBc3NldExpc3QpO1xuICAgICAgdHhCdWlsZGVyLnNwb25zb3JzaGlwSW5mbyh7XG4gICAgICAgIGZlZUFkZHJlc3M6IGZlZUFkZHJlc3MsXG4gICAgICAgIGZlZUFkZHJlc3NJbnB1dEJhbGFuY2U6IGZlZUFkZHJlc3NJbnB1dEJhbGFuY2UudG9TdHJpbmcoKSxcbiAgICAgIH0pO1xuICAgICAgdHhCdWlsZGVyLnR0bCg4MDAwMDAwMDApO1xuICAgICAgdHhCdWlsZGVyLmlzVG9rZW5UcmFuc2FjdGlvbigpO1xuICAgICAgY29uc3QgdHggPSAoYXdhaXQgdHhCdWlsZGVyLmJ1aWxkKCkpIGFzIFRyYW5zYWN0aW9uO1xuICAgICAgY29uc3QgdHhIZXggPSB0eC50b0Jyb2FkY2FzdEZvcm1hdCgpO1xuXG4gICAgICBjb25zdCBtb2NrV2FsbGV0ID0ge1xuICAgICAgICBjb2luU3BlY2lmaWM6ICgpID0+ICh7XG4gICAgICAgICAgYmFzZUFkZHJlc3M6IHNlbmRlckFkZHJlc3MsXG4gICAgICAgIH0pLFxuICAgICAgfTtcblxuICAgICAgY29uc3QgdHhQYXJhbXMgPSB7IHJlY2lwaWVudHM6IHVuZGVmaW5lZCB9O1xuICAgICAgY29uc3QgdHhQcmVidWlsZCA9IHsgdHhIZXgsIHR4SW5mbzogeyBmZWVBZGRyZXNzIH0gfTtcbiAgICAgIGNvbnN0IHZlcmlmaWNhdGlvbiA9IHsgY29uc29saWRhdGlvblRvQmFzZUFkZHJlc3M6IHRydWUgfTtcblxuICAgICAgYXdhaXQgYWRhVG9rZW5cbiAgICAgICAgLnZlcmlmeVRyYW5zYWN0aW9uKHtcbiAgICAgICAgICB0eFBhcmFtcyxcbiAgICAgICAgICB0eFByZWJ1aWxkLFxuICAgICAgICAgIHZlcmlmaWNhdGlvbixcbiAgICAgICAgICB3YWxsZXQ6IG1vY2tXYWxsZXQgYXMgYW55LFxuICAgICAgICB9KVxuICAgICAgICAuc2hvdWxkLmJlLnJlamVjdGVkV2l0aCgndHggb3V0cHV0cyBkb2VzIG5vdCBtYXRjaCB3aXRoIGV4cGVjdGVkIGFkZHJlc3MnKTtcbiAgICB9KTtcbiAgfSk7XG59KTtcbiJdfQ==
|