@nlabs/reaktor 0.1.12 → 0.1.14
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/lib/config.js +138 -0
- package/lib/data/conversations.js +228 -0
- package/lib/data/dynamodb.js +172 -0
- package/lib/data/email.js +194 -0
- package/lib/data/files.js +463 -0
- package/lib/data/groups.js +401 -0
- package/lib/data/images.js +841 -0
- package/lib/data/index.js +234 -0
- package/lib/data/ios.js +327 -0
- package/lib/data/locations.js +148 -0
- package/lib/data/messages.js +281 -0
- package/lib/data/notifications.js +59 -0
- package/lib/data/payments.js +798 -0
- package/lib/data/posts.js +637 -0
- package/lib/data/reactions.js +243 -0
- package/lib/data/s3.js +133 -0
- package/lib/data/search.js +111 -0
- package/lib/data/sms.js +79 -0
- package/lib/data/subscription.js +311 -0
- package/lib/data/tags.js +343 -0
- package/lib/data/users.js +415 -0
- package/lib/index.js +42 -0
- package/lib/types/apps.js +2 -0
- package/lib/types/arangodb.js +2 -0
- package/lib/types/auth.js +2 -0
- package/lib/types/conversations.js +2 -0
- package/lib/types/email.js +2 -0
- package/lib/types/files.js +2 -0
- package/lib/types/google.js +2 -0
- package/lib/types/groups.js +2 -0
- package/lib/types/images.js +2 -0
- package/lib/types/index.js +210 -0
- package/lib/types/locations.js +2 -0
- package/lib/types/messages.js +2 -0
- package/lib/types/notifications.js +2 -0
- package/lib/types/payments.js +2 -0
- package/lib/types/posts.js +2 -0
- package/lib/types/reactions.js +2 -0
- package/lib/types/tags.js +2 -0
- package/lib/types/users.js +2 -0
- package/lib/utils/analytics.js +59 -0
- package/lib/utils/arangodb.d.ts +0 -1
- package/lib/utils/arangodb.js +122 -0
- package/lib/utils/auth.d.ts +0 -2
- package/lib/utils/auth.js +55 -0
- package/lib/utils/graphql.js +19 -0
- package/lib/utils/index.js +78 -0
- package/lib/utils/objects.js +54 -0
- package/lib/utils/redis.js +28 -0
- package/package.json +7 -7
|
@@ -0,0 +1,798 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
Object.defineProperty(exports, "__esModule", {
|
|
4
|
+
value: true
|
|
5
|
+
});
|
|
6
|
+
exports.createPaymentHold = exports.createPaymentTransfer = exports.deletePaymentAccountBank = exports.deleteCreditCard = exports.getCreditCards = exports.updateCreditCard = exports.addCreditCard = exports.addPaymentAccountCard = exports.addPaymentAccountBank = exports.addCustomerAccount = void 0;
|
|
7
|
+
|
|
8
|
+
var _utils = require("@nlabs/utils");
|
|
9
|
+
|
|
10
|
+
var _arangojs = require("arangojs");
|
|
11
|
+
|
|
12
|
+
var _graphqlErrors = require("graphql-errors");
|
|
13
|
+
|
|
14
|
+
var _isEmpty = _interopRequireDefault(require("lodash/isEmpty"));
|
|
15
|
+
|
|
16
|
+
var _luxon = require("luxon");
|
|
17
|
+
|
|
18
|
+
var stripe = _interopRequireWildcard(require("stripe"));
|
|
19
|
+
|
|
20
|
+
var _config = require("../config");
|
|
21
|
+
|
|
22
|
+
var _analytics = require("../utils/analytics");
|
|
23
|
+
|
|
24
|
+
var _arangodb = require("../utils/arangodb");
|
|
25
|
+
|
|
26
|
+
var _users = require("./users");
|
|
27
|
+
|
|
28
|
+
function _interopRequireWildcard(obj) { if (obj && obj.__esModule) { return obj; } else { var newObj = {}; if (obj != null) { for (var key in obj) { if (Object.prototype.hasOwnProperty.call(obj, key)) { var desc = Object.defineProperty && Object.getOwnPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : {}; if (desc.get || desc.set) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } } newObj["default"] = obj; return newObj; } }
|
|
29
|
+
|
|
30
|
+
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { "default": obj }; }
|
|
31
|
+
|
|
32
|
+
function _templateObject11() {
|
|
33
|
+
var data = _taggedTemplateLiteral(["INSERT ", " IN payments RETURN NEW"]);
|
|
34
|
+
|
|
35
|
+
_templateObject11 = function _templateObject11() {
|
|
36
|
+
return data;
|
|
37
|
+
};
|
|
38
|
+
|
|
39
|
+
return data;
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
function _templateObject10() {
|
|
43
|
+
var data = _taggedTemplateLiteral(["INSERT ", " IN transfers RETURN NEW"]);
|
|
44
|
+
|
|
45
|
+
_templateObject10 = function _templateObject10() {
|
|
46
|
+
return data;
|
|
47
|
+
};
|
|
48
|
+
|
|
49
|
+
return data;
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
function _templateObject9() {
|
|
53
|
+
var data = _taggedTemplateLiteral(["UPDATE ", " WITH ", " IN users LIMIT 1 RETURN NEW"]);
|
|
54
|
+
|
|
55
|
+
_templateObject9 = function _templateObject9() {
|
|
56
|
+
return data;
|
|
57
|
+
};
|
|
58
|
+
|
|
59
|
+
return data;
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
function _templateObject8() {
|
|
63
|
+
var data = _taggedTemplateLiteral(["REMOVE {_key:", "} IN hasPayment"]);
|
|
64
|
+
|
|
65
|
+
_templateObject8 = function _templateObject8() {
|
|
66
|
+
return data;
|
|
67
|
+
};
|
|
68
|
+
|
|
69
|
+
return data;
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
function _templateObject7() {
|
|
73
|
+
var data = _taggedTemplateLiteral(["\n LET card = FIRST(\n FOR c IN creditCards\n FILTER c._key == ", " && c.userId == ", "\n LIMIT 1\n REMOVE c IN creditCards\n RETURN OLD\n )\n LET user = FIRST(\n FOR u IN users\n FILTER u._key == ", "\n LIMIT 1\n RETURN u\n )\n RETURN {user: user, card: card}"]);
|
|
74
|
+
|
|
75
|
+
_templateObject7 = function _templateObject7() {
|
|
76
|
+
return data;
|
|
77
|
+
};
|
|
78
|
+
|
|
79
|
+
return data;
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
function _templateObject6() {
|
|
83
|
+
var data = _taggedTemplateLiteral(["FOR c IN creditCards\n FILTER c.userId == ", "\n RETURN c"]);
|
|
84
|
+
|
|
85
|
+
_templateObject6 = function _templateObject6() {
|
|
86
|
+
return data;
|
|
87
|
+
};
|
|
88
|
+
|
|
89
|
+
return data;
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
function _templateObject5() {
|
|
93
|
+
var data = _taggedTemplateLiteral(["\n LET updatedCard = FIRST(\n FOR c IN creditCards\n FILTER c._key == ", " && c.userId == ", "\n UPDATE c WITH ", " IN creditCards\n LIMIT 1\n RETURN NEW\n )\n LET user = FIRST(\n FOR u IN users\n FILTER u._key == ", "\n LIMIT 1\n RETURN u\n )\n RETURN {user: user, card: updatedCard}"]);
|
|
94
|
+
|
|
95
|
+
_templateObject5 = function _templateObject5() {
|
|
96
|
+
return data;
|
|
97
|
+
};
|
|
98
|
+
|
|
99
|
+
return data;
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
function _templateObject4() {
|
|
103
|
+
var data = _taggedTemplateLiteral(["INSERT ", " IN creditCards RETURN NEW"]);
|
|
104
|
+
|
|
105
|
+
_templateObject4 = function _templateObject4() {
|
|
106
|
+
return data;
|
|
107
|
+
};
|
|
108
|
+
|
|
109
|
+
return data;
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; var ownKeys = Object.keys(source); if (typeof Object.getOwnPropertySymbols === 'function') { ownKeys = ownKeys.concat(Object.getOwnPropertySymbols(source).filter(function (sym) { return Object.getOwnPropertyDescriptor(source, sym).enumerable; })); } ownKeys.forEach(function (key) { _defineProperty(target, key, source[key]); }); } return target; }
|
|
113
|
+
|
|
114
|
+
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
|
|
115
|
+
|
|
116
|
+
function _templateObject3() {
|
|
117
|
+
var data = _taggedTemplateLiteral(["UPDATE ", " WITH ", " IN users LIMIT 1 RETURN NEW"]);
|
|
118
|
+
|
|
119
|
+
_templateObject3 = function _templateObject3() {
|
|
120
|
+
return data;
|
|
121
|
+
};
|
|
122
|
+
|
|
123
|
+
return data;
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
function _templateObject2() {
|
|
127
|
+
var data = _taggedTemplateLiteral(["UPDATE ", " WITH ", " IN users LIMIT 1 RETURN NEW"]);
|
|
128
|
+
|
|
129
|
+
_templateObject2 = function _templateObject2() {
|
|
130
|
+
return data;
|
|
131
|
+
};
|
|
132
|
+
|
|
133
|
+
return data;
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
function _templateObject() {
|
|
137
|
+
var data = _taggedTemplateLiteral(["UPDATE ", " WITH ", " IN users LIMIT 1 RETURN NEW"]);
|
|
138
|
+
|
|
139
|
+
_templateObject = function _templateObject() {
|
|
140
|
+
return data;
|
|
141
|
+
};
|
|
142
|
+
|
|
143
|
+
return data;
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
function _taggedTemplateLiteral(strings, raw) { if (!raw) { raw = strings.slice(0); } return Object.freeze(Object.defineProperties(strings, { raw: { value: Object.freeze(raw) } })); }
|
|
147
|
+
|
|
148
|
+
/**
|
|
149
|
+
* Copyright (c) 2019-Present, Nitrogen Labs, Inc.
|
|
150
|
+
* Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.
|
|
151
|
+
*/
|
|
152
|
+
var eventCategory = 'payments';
|
|
153
|
+
|
|
154
|
+
var addCustomerAccount = function addCustomerAccount(context) {
|
|
155
|
+
var action = 'addCustomerAccount';
|
|
156
|
+
var database = context.database,
|
|
157
|
+
sessionId = context.userId,
|
|
158
|
+
username = context.username; // Stripe
|
|
159
|
+
|
|
160
|
+
var stripeClient = stripe(_config.Config.get('stripe.token'));
|
|
161
|
+
return stripeClient.customers.create({
|
|
162
|
+
metadata: {
|
|
163
|
+
userId: sessionId,
|
|
164
|
+
username: username
|
|
165
|
+
}
|
|
166
|
+
}).then(function (customer) {
|
|
167
|
+
// Create session
|
|
168
|
+
var now = Date.now();
|
|
169
|
+
var update = {
|
|
170
|
+
modified: now,
|
|
171
|
+
stripeCustomerId: customer.id
|
|
172
|
+
};
|
|
173
|
+
var aqlQry = (0, _arangojs.aql)(_templateObject(), sessionId, update);
|
|
174
|
+
return (0, _arangodb.useDb)(database).query(aqlQry).then(function (cursor) {
|
|
175
|
+
return cursor.next();
|
|
176
|
+
}).then(function () {
|
|
177
|
+
var updatedUser = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
|
|
178
|
+
return !(0, _isEmpty["default"])(updatedUser);
|
|
179
|
+
})["catch"](function (error) {
|
|
180
|
+
return (0, _analytics.logError)({
|
|
181
|
+
action: action,
|
|
182
|
+
category: eventCategory,
|
|
183
|
+
label: 'db_error'
|
|
184
|
+
}, error, context).then(function () {
|
|
185
|
+
return null;
|
|
186
|
+
});
|
|
187
|
+
});
|
|
188
|
+
});
|
|
189
|
+
};
|
|
190
|
+
|
|
191
|
+
exports.addCustomerAccount = addCustomerAccount;
|
|
192
|
+
|
|
193
|
+
var addPaymentAccountBank = function addPaymentAccountBank(context, bankAccount) {
|
|
194
|
+
var action = 'addPaymentAccountBank';
|
|
195
|
+
var database = context.database,
|
|
196
|
+
sessionId = context.userId; // Params
|
|
197
|
+
|
|
198
|
+
var accountNumber = bankAccount.accountNumber,
|
|
199
|
+
fullName = bankAccount.fullName,
|
|
200
|
+
routing = bankAccount.routing;
|
|
201
|
+
var formatAccount = (0, _utils.parseString)(accountNumber, 32);
|
|
202
|
+
|
|
203
|
+
if (formatAccount === '') {
|
|
204
|
+
throw new _graphqlErrors.UserError('required_account_number');
|
|
205
|
+
}
|
|
206
|
+
|
|
207
|
+
var formatFullName = (0, _utils.parseVarChar)(fullName, 128);
|
|
208
|
+
|
|
209
|
+
if (formatFullName === '') {
|
|
210
|
+
throw new _graphqlErrors.UserError('required_full_name');
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
var formatRouting = (0, _utils.parseString)(routing, 32);
|
|
214
|
+
|
|
215
|
+
if (formatRouting === '') {
|
|
216
|
+
throw new _graphqlErrors.UserError('required_routing_number');
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
return (0, _users.getUser)(context, sessionId).then(function (user) {
|
|
220
|
+
// Stripe
|
|
221
|
+
var stripeClient = stripe(_config.Config.get('stripe.token'));
|
|
222
|
+
return stripeClient.customers.createSource(user.stripeAccountId, {
|
|
223
|
+
source: {
|
|
224
|
+
account_holder_name: formatFullName,
|
|
225
|
+
account_holder_type: 'individual',
|
|
226
|
+
account_number: formatAccount,
|
|
227
|
+
country: 'US',
|
|
228
|
+
currency: 'USD',
|
|
229
|
+
object: 'bank_account',
|
|
230
|
+
routing_number: formatRouting
|
|
231
|
+
}
|
|
232
|
+
}).then(function (account) {
|
|
233
|
+
var update = {
|
|
234
|
+
bankAccount: account.last4,
|
|
235
|
+
bankFullName: formatFullName,
|
|
236
|
+
bankId: account.id,
|
|
237
|
+
bankRouting: account.routing_number,
|
|
238
|
+
modified: Date.now()
|
|
239
|
+
};
|
|
240
|
+
var aqlQry = (0, _arangojs.aql)(_templateObject2(), sessionId, update);
|
|
241
|
+
return (0, _arangodb.useDb)(database).query(aqlQry).then(function (cursor) {
|
|
242
|
+
return cursor.next();
|
|
243
|
+
}).then(function () {
|
|
244
|
+
var updatedUser = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
|
|
245
|
+
return updatedUser;
|
|
246
|
+
});
|
|
247
|
+
})["catch"](function (error) {
|
|
248
|
+
var msg = error.message;
|
|
249
|
+
|
|
250
|
+
if (msg === 'A bank account with that routing number and account number ' + 'already exists for this customer.') {
|
|
251
|
+
return (0, _analytics.logError)({
|
|
252
|
+
action: action,
|
|
253
|
+
category: eventCategory,
|
|
254
|
+
label: 'bank_account_exists'
|
|
255
|
+
}, error, context).then(function () {
|
|
256
|
+
return null;
|
|
257
|
+
});
|
|
258
|
+
}
|
|
259
|
+
|
|
260
|
+
return (0, _analytics.logError)({
|
|
261
|
+
action: action,
|
|
262
|
+
category: eventCategory,
|
|
263
|
+
label: 'payment_error'
|
|
264
|
+
}, error, context).then(function () {
|
|
265
|
+
return null;
|
|
266
|
+
});
|
|
267
|
+
});
|
|
268
|
+
});
|
|
269
|
+
};
|
|
270
|
+
|
|
271
|
+
exports.addPaymentAccountBank = addPaymentAccountBank;
|
|
272
|
+
|
|
273
|
+
var addPaymentAccountCard = function addPaymentAccountCard(context, card, requestData) {
|
|
274
|
+
var action = 'addPaymentAccountCard';
|
|
275
|
+
var database = context.database,
|
|
276
|
+
sessionId = context.userId;
|
|
277
|
+
return (0, _users.getUser)(context, sessionId).then(function (user) {
|
|
278
|
+
// User
|
|
279
|
+
var userKey = user._key,
|
|
280
|
+
_user$country = user.country,
|
|
281
|
+
country = _user$country === void 0 ? 'US' : _user$country,
|
|
282
|
+
city = user.city,
|
|
283
|
+
_user$currency = user.currency,
|
|
284
|
+
currency = _user$currency === void 0 ? 'USD' : _user$currency,
|
|
285
|
+
dob = user.dob,
|
|
286
|
+
email = user.email,
|
|
287
|
+
first = user.first,
|
|
288
|
+
last = user.last,
|
|
289
|
+
username = user.username; // Birthday
|
|
290
|
+
|
|
291
|
+
var dobDay = +_luxon.DateTime.fromMillis(dob).toFormat('d');
|
|
292
|
+
var dobMonth = +_luxon.DateTime.fromMillis(dob).toFormat('M');
|
|
293
|
+
var dobYear = +_luxon.DateTime.fromMillis(dob).toFormat('y'); // Card
|
|
294
|
+
|
|
295
|
+
var accountNumber = card.accountNumber,
|
|
296
|
+
acceptedTerms = card.acceptedTerms,
|
|
297
|
+
expMonth = card.expMonth,
|
|
298
|
+
expYear = card.expYear,
|
|
299
|
+
state = card.state,
|
|
300
|
+
street1 = card.street1,
|
|
301
|
+
zip = card.zip; // Acceptance
|
|
302
|
+
|
|
303
|
+
var accepted = acceptedTerms ? _luxon.DateTime.local().millisecond : 0; // Stripe
|
|
304
|
+
|
|
305
|
+
var stripeClient = stripe(_config.Config.get('stripe.token'));
|
|
306
|
+
return stripeClient.accounts.create({
|
|
307
|
+
country: country,
|
|
308
|
+
email: email,
|
|
309
|
+
external_account: {
|
|
310
|
+
currency: currency,
|
|
311
|
+
exp_month: expMonth,
|
|
312
|
+
exp_year: expYear,
|
|
313
|
+
number: accountNumber,
|
|
314
|
+
object: 'card'
|
|
315
|
+
},
|
|
316
|
+
legal_entity: {
|
|
317
|
+
address: {
|
|
318
|
+
city: city,
|
|
319
|
+
line1: street1,
|
|
320
|
+
postal_code: zip,
|
|
321
|
+
state: state
|
|
322
|
+
},
|
|
323
|
+
dob: {
|
|
324
|
+
day: dobDay,
|
|
325
|
+
month: dobMonth,
|
|
326
|
+
year: dobYear
|
|
327
|
+
},
|
|
328
|
+
first_name: first,
|
|
329
|
+
last_name: last,
|
|
330
|
+
// ssn_last_4: ssn.substr(-4),
|
|
331
|
+
// personal_id_number: ssn,
|
|
332
|
+
type: 'individual'
|
|
333
|
+
},
|
|
334
|
+
managed: true,
|
|
335
|
+
metadata: {
|
|
336
|
+
userId: userKey,
|
|
337
|
+
username: username
|
|
338
|
+
},
|
|
339
|
+
tos_acceptance: {
|
|
340
|
+
date: accepted,
|
|
341
|
+
ip: requestData.ip,
|
|
342
|
+
user_agent: requestData.userAgent.source
|
|
343
|
+
}
|
|
344
|
+
}).then(function (customer) {
|
|
345
|
+
// Create session
|
|
346
|
+
var now = Date.now();
|
|
347
|
+
var update = {
|
|
348
|
+
modified: now,
|
|
349
|
+
stripeAccountId: customer.id
|
|
350
|
+
};
|
|
351
|
+
var aqlQry = (0, _arangojs.aql)(_templateObject3(), userKey, update);
|
|
352
|
+
return (0, _arangodb.useDb)(database).query(aqlQry).then(function (cursor) {
|
|
353
|
+
return cursor.next();
|
|
354
|
+
}).then(function () {
|
|
355
|
+
var updatedUser = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
|
|
356
|
+
return !(0, _isEmpty["default"])(updatedUser);
|
|
357
|
+
})["catch"](function (error) {
|
|
358
|
+
return (0, _analytics.logError)({
|
|
359
|
+
action: action,
|
|
360
|
+
category: eventCategory,
|
|
361
|
+
label: 'db_error'
|
|
362
|
+
}, error, context).then(function () {
|
|
363
|
+
return null;
|
|
364
|
+
});
|
|
365
|
+
});
|
|
366
|
+
});
|
|
367
|
+
});
|
|
368
|
+
};
|
|
369
|
+
|
|
370
|
+
exports.addPaymentAccountCard = addPaymentAccountCard;
|
|
371
|
+
|
|
372
|
+
var addCreditCard = function addCreditCard(context, card) {
|
|
373
|
+
var action = 'addCreditCard';
|
|
374
|
+
var database = context.database,
|
|
375
|
+
sessionId = context.userId;
|
|
376
|
+
var accountNumber = card.accountNumber,
|
|
377
|
+
city = card.city,
|
|
378
|
+
country = card.country,
|
|
379
|
+
cvc = card.cvc,
|
|
380
|
+
expMonth = card.expMonth,
|
|
381
|
+
expYear = card.expYear,
|
|
382
|
+
fullName = card.fullName,
|
|
383
|
+
street1 = card.street1,
|
|
384
|
+
state = card.state,
|
|
385
|
+
zip = card.zip;
|
|
386
|
+
var formatNumber = (0, _utils.parseNum)(accountNumber, 16);
|
|
387
|
+
|
|
388
|
+
if (!formatNumber) {
|
|
389
|
+
throw new _graphqlErrors.UserError('required_credit_card_number');
|
|
390
|
+
}
|
|
391
|
+
|
|
392
|
+
var formatExpMonth = (0, _utils.parseNum)(expMonth, 2);
|
|
393
|
+
|
|
394
|
+
if (!formatExpMonth) {
|
|
395
|
+
throw new _graphqlErrors.UserError('required_credit_card_exp_month');
|
|
396
|
+
}
|
|
397
|
+
|
|
398
|
+
var formatExpYear = (0, _utils.parseNum)(expYear, 2);
|
|
399
|
+
|
|
400
|
+
if (!formatExpYear) {
|
|
401
|
+
throw new _graphqlErrors.UserError('required_credit_card_exp_year');
|
|
402
|
+
}
|
|
403
|
+
|
|
404
|
+
var formatCvc = (0, _utils.parseNum)(cvc, 3);
|
|
405
|
+
|
|
406
|
+
if (!formatCvc) {
|
|
407
|
+
throw new _graphqlErrors.UserError('required_credit_card_cvc');
|
|
408
|
+
}
|
|
409
|
+
|
|
410
|
+
var paymentCard = {};
|
|
411
|
+
var formatCity = (0, _utils.parseVarChar)(city, 32);
|
|
412
|
+
|
|
413
|
+
if (formatCity) {
|
|
414
|
+
paymentCard.city = formatCity;
|
|
415
|
+
}
|
|
416
|
+
|
|
417
|
+
var formatCountry = (0, _utils.parseVarChar)(country, 32);
|
|
418
|
+
|
|
419
|
+
if (formatCountry) {
|
|
420
|
+
paymentCard.country = formatCountry;
|
|
421
|
+
}
|
|
422
|
+
|
|
423
|
+
var formatFullName = (0, _utils.parseVarChar)(fullName, 32);
|
|
424
|
+
|
|
425
|
+
if (formatFullName) {
|
|
426
|
+
paymentCard.fullName = formatFullName;
|
|
427
|
+
}
|
|
428
|
+
|
|
429
|
+
var formatStreet = (0, _utils.parseVarChar)(street1, 32);
|
|
430
|
+
|
|
431
|
+
if (formatStreet) {
|
|
432
|
+
paymentCard.street1 = formatStreet;
|
|
433
|
+
}
|
|
434
|
+
|
|
435
|
+
var formatState = (0, _utils.parseVarChar)(state, 32);
|
|
436
|
+
|
|
437
|
+
if (formatState) {
|
|
438
|
+
paymentCard.state = formatState;
|
|
439
|
+
}
|
|
440
|
+
|
|
441
|
+
var formatZip = (0, _utils.parseVarChar)(zip, 32);
|
|
442
|
+
|
|
443
|
+
if (formatZip) {
|
|
444
|
+
paymentCard.zip = formatZip;
|
|
445
|
+
} // Get stripe id
|
|
446
|
+
|
|
447
|
+
|
|
448
|
+
return (0, _users.getUser)(context, sessionId).then(function () {
|
|
449
|
+
var user = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
|
|
450
|
+
// Stripe
|
|
451
|
+
var stripeClient = stripe(_config.Config.get('stripe.token'));
|
|
452
|
+
return stripeClient.customers.createSource(user.stripeCustomerId, {
|
|
453
|
+
source: {
|
|
454
|
+
address_city: formatCity,
|
|
455
|
+
address_country: formatCountry,
|
|
456
|
+
address_line1: formatStreet,
|
|
457
|
+
address_state: formatState,
|
|
458
|
+
address_zip: formatZip,
|
|
459
|
+
currency: 'usd',
|
|
460
|
+
cvc: formatCvc,
|
|
461
|
+
exp_month: formatExpMonth,
|
|
462
|
+
exp_year: formatExpYear,
|
|
463
|
+
name: formatFullName,
|
|
464
|
+
number: formatNumber,
|
|
465
|
+
object: 'card'
|
|
466
|
+
}
|
|
467
|
+
}).then(function (stripeObj) {
|
|
468
|
+
var now = Date.now();
|
|
469
|
+
|
|
470
|
+
var insert = _objectSpread({}, paymentCard, {
|
|
471
|
+
_key: (0, _utils.createHash)("user-payment-".concat(sessionId)),
|
|
472
|
+
accountNumber: stripeObj.last4,
|
|
473
|
+
added: now,
|
|
474
|
+
brand: stripeObj.brand,
|
|
475
|
+
cvc: stripeObj.cvc_check,
|
|
476
|
+
expMonth: expMonth,
|
|
477
|
+
expYear: expYear,
|
|
478
|
+
modified: now,
|
|
479
|
+
userId: sessionId
|
|
480
|
+
});
|
|
481
|
+
|
|
482
|
+
var insertAqlQry = (0, _arangojs.aql)(_templateObject4(), insert);
|
|
483
|
+
return (0, _arangodb.useDb)(database).query(insertAqlQry).then(function (cursor) {
|
|
484
|
+
return cursor.next();
|
|
485
|
+
}).then(function () {
|
|
486
|
+
var newCard = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
|
|
487
|
+
|
|
488
|
+
if (!(0, _isEmpty["default"])(newCard)) {
|
|
489
|
+
// Add linked edge
|
|
490
|
+
var cardId = card._id,
|
|
491
|
+
cardKey = card._key;
|
|
492
|
+
var edgeCollection = (0, _arangodb.useDb)(database).edgeCollection('hasPayment');
|
|
493
|
+
var edgeId = (0, _utils.createHash)("payment-".concat(cardKey));
|
|
494
|
+
var edge = {
|
|
495
|
+
_key: edgeId
|
|
496
|
+
};
|
|
497
|
+
return edgeCollection.save(edge, "users/".concat(sessionId), cardId).then(function () {
|
|
498
|
+
return card;
|
|
499
|
+
});
|
|
500
|
+
}
|
|
501
|
+
|
|
502
|
+
return newCard;
|
|
503
|
+
});
|
|
504
|
+
})["catch"](function (error) {
|
|
505
|
+
return (0, _analytics.logError)({
|
|
506
|
+
action: action,
|
|
507
|
+
category: eventCategory,
|
|
508
|
+
label: 'payment_error'
|
|
509
|
+
}, error, context).then(function () {
|
|
510
|
+
return null;
|
|
511
|
+
});
|
|
512
|
+
});
|
|
513
|
+
});
|
|
514
|
+
};
|
|
515
|
+
|
|
516
|
+
exports.addCreditCard = addCreditCard;
|
|
517
|
+
|
|
518
|
+
var updateCreditCard = function updateCreditCard(context, card) {
|
|
519
|
+
var database = context.database,
|
|
520
|
+
sessionId = context.userId;
|
|
521
|
+
var city = card.city,
|
|
522
|
+
country = card.country,
|
|
523
|
+
expMonth = card.expMonth,
|
|
524
|
+
expYear = card.expYear,
|
|
525
|
+
fullName = card.fullName,
|
|
526
|
+
id = card.id,
|
|
527
|
+
street1 = card.street1,
|
|
528
|
+
state = card.state,
|
|
529
|
+
zip = card.zip;
|
|
530
|
+
var formatId = (0, _utils.parseId)(id);
|
|
531
|
+
|
|
532
|
+
if (formatId) {
|
|
533
|
+
throw new _graphqlErrors.UserError('required_credit_card_id');
|
|
534
|
+
}
|
|
535
|
+
|
|
536
|
+
var paymentCard = {};
|
|
537
|
+
var formatExpMonth = (0, _utils.parseNum)(expMonth, 2);
|
|
538
|
+
var formatExpYear = (0, _utils.parseNum)(expYear, 2);
|
|
539
|
+
var formatCity = (0, _utils.parseVarChar)(city, 32);
|
|
540
|
+
var formatCountry = (0, _utils.parseVarChar)(country, 2);
|
|
541
|
+
var formatFullName = (0, _utils.parseVarChar)(fullName, 32);
|
|
542
|
+
var formatStreet1 = (0, _utils.parseString)(street1, 32);
|
|
543
|
+
var formatState = (0, _utils.parseVarChar)(state, 2);
|
|
544
|
+
var formatZip = (0, _utils.parseVarChar)(zip, 10);
|
|
545
|
+
|
|
546
|
+
if (formatExpMonth) {
|
|
547
|
+
paymentCard.expMonth = formatExpMonth;
|
|
548
|
+
}
|
|
549
|
+
|
|
550
|
+
if (formatExpYear) {
|
|
551
|
+
paymentCard.expYear = formatExpYear;
|
|
552
|
+
}
|
|
553
|
+
|
|
554
|
+
if (formatCity) {
|
|
555
|
+
paymentCard.city = formatCity;
|
|
556
|
+
}
|
|
557
|
+
|
|
558
|
+
if (formatCountry) {
|
|
559
|
+
paymentCard.country = formatCountry;
|
|
560
|
+
}
|
|
561
|
+
|
|
562
|
+
if (formatFullName) {
|
|
563
|
+
paymentCard.fullName = formatFullName;
|
|
564
|
+
}
|
|
565
|
+
|
|
566
|
+
if (formatStreet1) {
|
|
567
|
+
paymentCard.street1 = formatStreet1;
|
|
568
|
+
}
|
|
569
|
+
|
|
570
|
+
if (formatState) {
|
|
571
|
+
paymentCard.state = formatState;
|
|
572
|
+
}
|
|
573
|
+
|
|
574
|
+
if (formatZip) {
|
|
575
|
+
paymentCard.zip = formatZip;
|
|
576
|
+
}
|
|
577
|
+
|
|
578
|
+
var update = paymentCard;
|
|
579
|
+
var aqlQry = (0, _arangojs.aql)(_templateObject5(), formatId, sessionId, update, sessionId);
|
|
580
|
+
return (0, _arangodb.useDb)(database).query(aqlQry).then(function (cursor) {
|
|
581
|
+
return cursor.next();
|
|
582
|
+
}).then(function () {
|
|
583
|
+
var results = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {
|
|
584
|
+
card: {},
|
|
585
|
+
user: {}
|
|
586
|
+
};
|
|
587
|
+
var updatedCard = results.card;
|
|
588
|
+
var user = results.user;
|
|
589
|
+
|
|
590
|
+
if ((0, _isEmpty["default"])(updatedCard)) {
|
|
591
|
+
throw new _graphqlErrors.UserError('not_found');
|
|
592
|
+
} // Stripe
|
|
593
|
+
|
|
594
|
+
|
|
595
|
+
var stripeClient = stripe(_config.Config.get('stripe.token'));
|
|
596
|
+
return stripeClient.customers.updateCard(user.stripeCustomerId, card.stripeId, {
|
|
597
|
+
address_city: formatCity,
|
|
598
|
+
address_country: formatCountry,
|
|
599
|
+
address_line1: formatStreet1,
|
|
600
|
+
address_state: formatState,
|
|
601
|
+
address_zip: formatZip,
|
|
602
|
+
exp_month: formatExpMonth,
|
|
603
|
+
exp_year: formatExpYear,
|
|
604
|
+
name: formatFullName
|
|
605
|
+
}).then(function () {
|
|
606
|
+
return card;
|
|
607
|
+
})["catch"](function (error) {
|
|
608
|
+
console.log('payments::updateCard::error', error);
|
|
609
|
+
throw new _graphqlErrors.UserError('payment_error');
|
|
610
|
+
});
|
|
611
|
+
});
|
|
612
|
+
};
|
|
613
|
+
|
|
614
|
+
exports.updateCreditCard = updateCreditCard;
|
|
615
|
+
|
|
616
|
+
var getCreditCards = function getCreditCards(context) {
|
|
617
|
+
var action = 'getCreditCards';
|
|
618
|
+
var database = context.database,
|
|
619
|
+
sessionId = context.userId;
|
|
620
|
+
var aqlQry = (0, _arangojs.aql)(_templateObject6(), sessionId);
|
|
621
|
+
return (0, _arangodb.useDb)(database).query(aqlQry).then(function (cursor) {
|
|
622
|
+
return cursor.all();
|
|
623
|
+
}).then(function () {
|
|
624
|
+
var list = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [];
|
|
625
|
+
return list;
|
|
626
|
+
})["catch"](function (error) {
|
|
627
|
+
return (0, _analytics.logError)({
|
|
628
|
+
action: action,
|
|
629
|
+
category: eventCategory,
|
|
630
|
+
label: 'db_error'
|
|
631
|
+
}, error, context).then(function () {
|
|
632
|
+
return null;
|
|
633
|
+
});
|
|
634
|
+
});
|
|
635
|
+
};
|
|
636
|
+
|
|
637
|
+
exports.getCreditCards = getCreditCards;
|
|
638
|
+
|
|
639
|
+
var deleteCreditCard = function deleteCreditCard(context, cardId) {
|
|
640
|
+
var database = context.database,
|
|
641
|
+
sessionId = context.userId;
|
|
642
|
+
var formatCardId = (0, _utils.parseId)(cardId);
|
|
643
|
+
var aqlQry = (0, _arangojs.aql)(_templateObject7(), formatCardId, sessionId, sessionId);
|
|
644
|
+
return (0, _arangodb.useDb)(database).query(aqlQry).then(function (cursor) {
|
|
645
|
+
return cursor.next();
|
|
646
|
+
}).then(function () {
|
|
647
|
+
var result = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {
|
|
648
|
+
card: {},
|
|
649
|
+
user: {}
|
|
650
|
+
};
|
|
651
|
+
|
|
652
|
+
if ((0, _isEmpty["default"])(result)) {
|
|
653
|
+
return false;
|
|
654
|
+
}
|
|
655
|
+
|
|
656
|
+
var card = result.card,
|
|
657
|
+
user = result.user;
|
|
658
|
+
var cardKey = card._key; // Remove linked edges
|
|
659
|
+
|
|
660
|
+
var edgeCollection = (0, _arangodb.useDb)(database).edgeCollection('hasPayment');
|
|
661
|
+
return edgeCollection.outEdges(cardKey).then(function (edges) {
|
|
662
|
+
if (edges.length) {
|
|
663
|
+
return Promise.all(edges.map(function (edge) {
|
|
664
|
+
var edgeKey = edge._key;
|
|
665
|
+
var removeAqlQry = (0, _arangojs.aql)(_templateObject8(), edgeKey);
|
|
666
|
+
return (0, _arangodb.useDb)(database).query(removeAqlQry);
|
|
667
|
+
})).then(function () {
|
|
668
|
+
// Stripe
|
|
669
|
+
var stripeClient = stripe(_config.Config.get('stripe.token'));
|
|
670
|
+
return stripeClient.customers.deleteCard(user.stripeCustomerId, card.stripeId).then(function () {
|
|
671
|
+
return true;
|
|
672
|
+
})["catch"](function (error) {
|
|
673
|
+
console.log('payments::deleteCard::error', error);
|
|
674
|
+
throw new _graphqlErrors.UserError('payment_error');
|
|
675
|
+
});
|
|
676
|
+
});
|
|
677
|
+
}
|
|
678
|
+
|
|
679
|
+
return false;
|
|
680
|
+
});
|
|
681
|
+
});
|
|
682
|
+
};
|
|
683
|
+
|
|
684
|
+
exports.deleteCreditCard = deleteCreditCard;
|
|
685
|
+
|
|
686
|
+
var deletePaymentAccountBank = function deletePaymentAccountBank(context, bankId) {
|
|
687
|
+
var database = context.database,
|
|
688
|
+
sessionId = context.userId; // Clean db
|
|
689
|
+
|
|
690
|
+
var update = {
|
|
691
|
+
bankAccount: '',
|
|
692
|
+
bankFullName: '',
|
|
693
|
+
bankId: '',
|
|
694
|
+
bankRouting: '',
|
|
695
|
+
modified: Date.now()
|
|
696
|
+
};
|
|
697
|
+
var aqlQry = (0, _arangojs.aql)(_templateObject9(), sessionId, update);
|
|
698
|
+
return (0, _arangodb.useDb)(database).query(aqlQry).then(function (cursor) {
|
|
699
|
+
return cursor.next();
|
|
700
|
+
}).then(function () {
|
|
701
|
+
var user = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
|
|
702
|
+
// Stripe
|
|
703
|
+
var stripeClient = stripe(_config.Config.get('stripe.token'));
|
|
704
|
+
return stripeClient.customers.deleteSource(user.stripeAccountId, bankId).then(function () {
|
|
705
|
+
var response = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {
|
|
706
|
+
deleted: false
|
|
707
|
+
};
|
|
708
|
+
return response.deleted;
|
|
709
|
+
})["catch"](function () {
|
|
710
|
+
return Promise.resolve(false);
|
|
711
|
+
});
|
|
712
|
+
});
|
|
713
|
+
};
|
|
714
|
+
|
|
715
|
+
exports.deletePaymentAccountBank = deletePaymentAccountBank;
|
|
716
|
+
|
|
717
|
+
var createPaymentTransfer = function createPaymentTransfer(context, transfer) {
|
|
718
|
+
var database = context.database,
|
|
719
|
+
sessionId = context.userId;
|
|
720
|
+
var amount = transfer.amount,
|
|
721
|
+
currency = transfer.currency;
|
|
722
|
+
var formatAmount = (0, _utils.parseNum)(amount);
|
|
723
|
+
var formatCurrency = (0, _utils.parseChar)(currency, 3, 'USD').toUpperCase();
|
|
724
|
+
return (0, _users.getUser)(context, sessionId).then(function (user) {
|
|
725
|
+
// Stripe
|
|
726
|
+
var stripeClient = stripe(_config.Config.get('stripe.token'));
|
|
727
|
+
return stripeClient.transfers.create({
|
|
728
|
+
amount: formatAmount,
|
|
729
|
+
currency: formatCurrency,
|
|
730
|
+
destination: user.stripeAccountId
|
|
731
|
+
}).then(function (stripeTransfer) {
|
|
732
|
+
console.log(stripeTransfer);
|
|
733
|
+
var now = Date.now();
|
|
734
|
+
var insert = {
|
|
735
|
+
added: now,
|
|
736
|
+
amount: formatAmount,
|
|
737
|
+
currency: formatCurrency,
|
|
738
|
+
modified: now,
|
|
739
|
+
userId: sessionId
|
|
740
|
+
};
|
|
741
|
+
var aqlQry = (0, _arangojs.aql)(_templateObject10(), insert);
|
|
742
|
+
return (0, _arangodb.useDb)(database).query(aqlQry).then(function (cursor) {
|
|
743
|
+
return cursor.next();
|
|
744
|
+
}).then(function (newTransfer) {
|
|
745
|
+
return newTransfer;
|
|
746
|
+
});
|
|
747
|
+
});
|
|
748
|
+
});
|
|
749
|
+
};
|
|
750
|
+
|
|
751
|
+
exports.createPaymentTransfer = createPaymentTransfer;
|
|
752
|
+
|
|
753
|
+
var createPaymentHold = function createPaymentHold(context, payment) {
|
|
754
|
+
var database = context.database,
|
|
755
|
+
sessionId = context.userId;
|
|
756
|
+
var amount = payment.amount,
|
|
757
|
+
capture = payment.capture,
|
|
758
|
+
cardId = payment.cardId,
|
|
759
|
+
currency = payment.currency,
|
|
760
|
+
description = payment.description;
|
|
761
|
+
var formatCurrency = (0, _utils.parseChar)(currency, 3, 'USD').toUpperCase();
|
|
762
|
+
var stripeClient = stripe(_config.Config.get('stripe.token'));
|
|
763
|
+
return stripeClient.charges.create({
|
|
764
|
+
amount: amount,
|
|
765
|
+
capture: capture,
|
|
766
|
+
currency: formatCurrency,
|
|
767
|
+
description: description,
|
|
768
|
+
source: cardId
|
|
769
|
+
}).then(function (stripeCharge) {
|
|
770
|
+
var now = Date.now();
|
|
771
|
+
var insert = {
|
|
772
|
+
added: now,
|
|
773
|
+
amount: amount,
|
|
774
|
+
capture: capture,
|
|
775
|
+
cardId: cardId,
|
|
776
|
+
chargeFailCode: stripeCharge.failure_code,
|
|
777
|
+
chargeFailMsg: stripeCharge.failure_message,
|
|
778
|
+
chargeId: stripeCharge.id,
|
|
779
|
+
chargeStatus: stripeCharge.status,
|
|
780
|
+
currency: formatCurrency,
|
|
781
|
+
description: description,
|
|
782
|
+
modified: now,
|
|
783
|
+
userId: sessionId
|
|
784
|
+
};
|
|
785
|
+
var aqlQry = (0, _arangojs.aql)(_templateObject11(), insert);
|
|
786
|
+
return (0, _arangodb.useDb)(database).query(aqlQry).then(function (cursor) {
|
|
787
|
+
return cursor.next();
|
|
788
|
+
}).then(function (newPayment) {
|
|
789
|
+
return newPayment;
|
|
790
|
+
});
|
|
791
|
+
})["catch"](function (error) {
|
|
792
|
+
console.log('payments::createHold::error', error);
|
|
793
|
+
throw new _graphqlErrors.UserError('payment_error');
|
|
794
|
+
});
|
|
795
|
+
};
|
|
796
|
+
|
|
797
|
+
exports.createPaymentHold = createPaymentHold;
|
|
798
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kYXRhL3BheW1lbnRzLnRzIl0sIm5hbWVzIjpbImV2ZW50Q2F0ZWdvcnkiLCJhZGRDdXN0b21lckFjY291bnQiLCJjb250ZXh0IiwiYWN0aW9uIiwiZGF0YWJhc2UiLCJzZXNzaW9uSWQiLCJ1c2VySWQiLCJ1c2VybmFtZSIsInN0cmlwZUNsaWVudCIsInN0cmlwZSIsIkNvbmZpZyIsImdldCIsImN1c3RvbWVycyIsImNyZWF0ZSIsIm1ldGFkYXRhIiwidGhlbiIsImN1c3RvbWVyIiwibm93IiwiRGF0ZSIsInVwZGF0ZSIsIm1vZGlmaWVkIiwic3RyaXBlQ3VzdG9tZXJJZCIsImlkIiwiYXFsUXJ5IiwiYXFsIiwicXVlcnkiLCJjdXJzb3IiLCJuZXh0IiwidXBkYXRlZFVzZXIiLCJlcnJvciIsImNhdGVnb3J5IiwibGFiZWwiLCJhZGRQYXltZW50QWNjb3VudEJhbmsiLCJiYW5rQWNjb3VudCIsImFjY291bnROdW1iZXIiLCJmdWxsTmFtZSIsInJvdXRpbmciLCJmb3JtYXRBY2NvdW50IiwiVXNlckVycm9yIiwiZm9ybWF0RnVsbE5hbWUiLCJmb3JtYXRSb3V0aW5nIiwidXNlciIsImNyZWF0ZVNvdXJjZSIsInN0cmlwZUFjY291bnRJZCIsInNvdXJjZSIsImFjY291bnRfaG9sZGVyX25hbWUiLCJhY2NvdW50X2hvbGRlcl90eXBlIiwiYWNjb3VudF9udW1iZXIiLCJjb3VudHJ5IiwiY3VycmVuY3kiLCJvYmplY3QiLCJyb3V0aW5nX251bWJlciIsImFjY291bnQiLCJsYXN0NCIsImJhbmtGdWxsTmFtZSIsImJhbmtJZCIsImJhbmtSb3V0aW5nIiwibXNnIiwibWVzc2FnZSIsImFkZFBheW1lbnRBY2NvdW50Q2FyZCIsImNhcmQiLCJyZXF1ZXN0RGF0YSIsInVzZXJLZXkiLCJfa2V5IiwiY2l0eSIsImRvYiIsImVtYWlsIiwiZmlyc3QiLCJsYXN0IiwiZG9iRGF5IiwiRGF0ZVRpbWUiLCJmcm9tTWlsbGlzIiwidG9Gb3JtYXQiLCJkb2JNb250aCIsImRvYlllYXIiLCJhY2NlcHRlZFRlcm1zIiwiZXhwTW9udGgiLCJleHBZZWFyIiwic3RhdGUiLCJzdHJlZXQxIiwiemlwIiwiYWNjZXB0ZWQiLCJsb2NhbCIsIm1pbGxpc2Vjb25kIiwiYWNjb3VudHMiLCJleHRlcm5hbF9hY2NvdW50IiwiZXhwX21vbnRoIiwiZXhwX3llYXIiLCJudW1iZXIiLCJsZWdhbF9lbnRpdHkiLCJhZGRyZXNzIiwibGluZTEiLCJwb3N0YWxfY29kZSIsImRheSIsIm1vbnRoIiwieWVhciIsImZpcnN0X25hbWUiLCJsYXN0X25hbWUiLCJ0eXBlIiwibWFuYWdlZCIsInRvc19hY2NlcHRhbmNlIiwiZGF0ZSIsImlwIiwidXNlcl9hZ2VudCIsInVzZXJBZ2VudCIsImFkZENyZWRpdENhcmQiLCJjdmMiLCJmb3JtYXROdW1iZXIiLCJmb3JtYXRFeHBNb250aCIsImZvcm1hdEV4cFllYXIiLCJmb3JtYXRDdmMiLCJwYXltZW50Q2FyZCIsImZvcm1hdENpdHkiLCJmb3JtYXRDb3VudHJ5IiwiZm9ybWF0U3RyZWV0IiwiZm9ybWF0U3RhdGUiLCJmb3JtYXRaaXAiLCJhZGRyZXNzX2NpdHkiLCJhZGRyZXNzX2NvdW50cnkiLCJhZGRyZXNzX2xpbmUxIiwiYWRkcmVzc19zdGF0ZSIsImFkZHJlc3NfemlwIiwibmFtZSIsInN0cmlwZU9iaiIsImluc2VydCIsImFkZGVkIiwiYnJhbmQiLCJjdmNfY2hlY2siLCJpbnNlcnRBcWxRcnkiLCJuZXdDYXJkIiwiY2FyZElkIiwiX2lkIiwiY2FyZEtleSIsImVkZ2VDb2xsZWN0aW9uIiwiZWRnZUlkIiwiZWRnZSIsInNhdmUiLCJ1cGRhdGVDcmVkaXRDYXJkIiwiZm9ybWF0SWQiLCJmb3JtYXRTdHJlZXQxIiwicmVzdWx0cyIsInVwZGF0ZWRDYXJkIiwidXBkYXRlQ2FyZCIsInN0cmlwZUlkIiwiY29uc29sZSIsImxvZyIsImdldENyZWRpdENhcmRzIiwiYWxsIiwibGlzdCIsImRlbGV0ZUNyZWRpdENhcmQiLCJmb3JtYXRDYXJkSWQiLCJyZXN1bHQiLCJvdXRFZGdlcyIsImVkZ2VzIiwibGVuZ3RoIiwiUHJvbWlzZSIsIm1hcCIsImVkZ2VLZXkiLCJyZW1vdmVBcWxRcnkiLCJkZWxldGVDYXJkIiwiZGVsZXRlUGF5bWVudEFjY291bnRCYW5rIiwiZGVsZXRlU291cmNlIiwicmVzcG9uc2UiLCJkZWxldGVkIiwicmVzb2x2ZSIsImNyZWF0ZVBheW1lbnRUcmFuc2ZlciIsInRyYW5zZmVyIiwiYW1vdW50IiwiZm9ybWF0QW1vdW50IiwiZm9ybWF0Q3VycmVuY3kiLCJ0b1VwcGVyQ2FzZSIsInRyYW5zZmVycyIsImRlc3RpbmF0aW9uIiwic3RyaXBlVHJhbnNmZXIiLCJuZXdUcmFuc2ZlciIsImNyZWF0ZVBheW1lbnRIb2xkIiwicGF5bWVudCIsImNhcHR1cmUiLCJkZXNjcmlwdGlvbiIsImNoYXJnZXMiLCJzdHJpcGVDaGFyZ2UiLCJjaGFyZ2VGYWlsQ29kZSIsImZhaWx1cmVfY29kZSIsImNoYXJnZUZhaWxNc2ciLCJmYWlsdXJlX21lc3NhZ2UiLCJjaGFyZ2VJZCIsImNoYXJnZVN0YXR1cyIsInN0YXR1cyIsIm5ld1BheW1lbnQiXSwibWFwcGluZ3MiOiI7Ozs7Ozs7QUFBQTs7QUFDQTs7QUFHQTs7QUFDQTs7QUFDQTs7QUFDQTs7QUFFQTs7QUFFQTs7QUFDQTs7QUFDQTs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFFQTs7OztBQUlBLElBQU1BLGFBQXFCLEdBQUcsVUFBOUI7O0FBRU8sSUFBTUMsa0JBQWtCLEdBQUcsU0FBckJBLGtCQUFxQixDQUFDQyxPQUFELEVBQTJDO0FBQzNFLE1BQU1DLE1BQWMsR0FBRyxvQkFBdkI7QUFEMkUsTUFFcEVDLFFBRm9FLEdBRTNCRixPQUYyQixDQUVwRUUsUUFGb0U7QUFBQSxNQUVsREMsU0FGa0QsR0FFM0JILE9BRjJCLENBRTFESSxNQUYwRDtBQUFBLE1BRXZDQyxRQUZ1QyxHQUUzQkwsT0FGMkIsQ0FFdkNLLFFBRnVDLEVBSTNFOztBQUNBLE1BQU1DLFlBQVksR0FBR0MsTUFBTSxDQUFDQyxlQUFPQyxHQUFQLENBQVcsY0FBWCxDQUFELENBQTNCO0FBRUEsU0FBT0gsWUFBWSxDQUFDSSxTQUFiLENBQ0pDLE1BREksQ0FDRztBQUNOQyxJQUFBQSxRQUFRLEVBQUU7QUFDUlIsTUFBQUEsTUFBTSxFQUFFRCxTQURBO0FBRVJFLE1BQUFBLFFBQVEsRUFBUkE7QUFGUTtBQURKLEdBREgsRUFPSlEsSUFQSSxDQU9DLFVBQUNDLFFBQUQsRUFBYztBQUNsQjtBQUNBLFFBQU1DLEdBQVcsR0FBR0MsSUFBSSxDQUFDRCxHQUFMLEVBQXBCO0FBQ0EsUUFBTUUsTUFBZ0IsR0FBRztBQUN2QkMsTUFBQUEsUUFBUSxFQUFFSCxHQURhO0FBRXZCSSxNQUFBQSxnQkFBZ0IsRUFBRUwsUUFBUSxDQUFDTTtBQUZKLEtBQXpCO0FBS0EsUUFBTUMsTUFBZ0IsT0FBR0MsYUFBSCxxQkFBZ0JuQixTQUFoQixFQUFrQ2MsTUFBbEMsQ0FBdEI7QUFFQSxXQUFPLHFCQUFNZixRQUFOLEVBQWdCcUIsS0FBaEIsQ0FBc0JGLE1BQXRCLEVBQ0pSLElBREksQ0FDQyxVQUFDVyxNQUFEO0FBQUEsYUFBeUJBLE1BQU0sQ0FBQ0MsSUFBUCxFQUF6QjtBQUFBLEtBREQsRUFFSlosSUFGSSxDQUVDO0FBQUEsVUFBQ2EsV0FBRCx1RUFBeUIsRUFBekI7QUFBQSxhQUFnQyxDQUFDLHlCQUFRQSxXQUFSLENBQWpDO0FBQUEsS0FGRCxXQUdFLFVBQUNDLEtBQUQ7QUFBQSxhQUFrQix5QkFBUztBQUNoQzFCLFFBQUFBLE1BQU0sRUFBTkEsTUFEZ0M7QUFFaEMyQixRQUFBQSxRQUFRLEVBQUU5QixhQUZzQjtBQUdoQytCLFFBQUFBLEtBQUssRUFBRTtBQUh5QixPQUFULEVBSXRCRixLQUpzQixFQUlmM0IsT0FKZSxFQUlOYSxJQUpNLENBSUQ7QUFBQSxlQUFNLElBQU47QUFBQSxPQUpDLENBQWxCO0FBQUEsS0FIRixDQUFQO0FBUUQsR0F6QkksQ0FBUDtBQTBCRCxDQWpDTTs7OztBQW1DQSxJQUFNaUIscUJBQXFCLEdBQUcsU0FBeEJBLHFCQUF3QixDQUFDOUIsT0FBRCxFQUFzQitCLFdBQXRCLEVBQTRFO0FBQy9HLE1BQU05QixNQUFjLEdBQUcsdUJBQXZCO0FBRCtHLE1BRXhHQyxRQUZ3RyxHQUV6RUYsT0FGeUUsQ0FFeEdFLFFBRndHO0FBQUEsTUFFdEZDLFNBRnNGLEdBRXpFSCxPQUZ5RSxDQUU5RkksTUFGOEYsRUFJL0c7O0FBSitHLE1BTTdHNEIsYUFONkcsR0FTM0dELFdBVDJHLENBTTdHQyxhQU42RztBQUFBLE1BTzdHQyxRQVA2RyxHQVMzR0YsV0FUMkcsQ0FPN0dFLFFBUDZHO0FBQUEsTUFRN0dDLE9BUjZHLEdBUzNHSCxXQVQyRyxDQVE3R0csT0FSNkc7QUFXL0csTUFBTUMsYUFBcUIsR0FBRyx3QkFBWUgsYUFBWixFQUEyQixFQUEzQixDQUE5Qjs7QUFFQSxNQUFHRyxhQUFhLEtBQUssRUFBckIsRUFBeUI7QUFDdkIsVUFBTSxJQUFJQyx3QkFBSixDQUFjLHlCQUFkLENBQU47QUFDRDs7QUFFRCxNQUFNQyxjQUFzQixHQUFHLHlCQUFhSixRQUFiLEVBQXVCLEdBQXZCLENBQS9COztBQUVBLE1BQUdJLGNBQWMsS0FBSyxFQUF0QixFQUEwQjtBQUN4QixVQUFNLElBQUlELHdCQUFKLENBQWMsb0JBQWQsQ0FBTjtBQUNEOztBQUVELE1BQU1FLGFBQXFCLEdBQUcsd0JBQVlKLE9BQVosRUFBcUIsRUFBckIsQ0FBOUI7O0FBRUEsTUFBR0ksYUFBYSxLQUFLLEVBQXJCLEVBQXlCO0FBQ3ZCLFVBQU0sSUFBSUYsd0JBQUosQ0FBYyx5QkFBZCxDQUFOO0FBQ0Q7O0FBRUQsU0FBTyxvQkFBUXBDLE9BQVIsRUFBaUJHLFNBQWpCLEVBQ0pVLElBREksQ0FDQyxVQUFDMEIsSUFBRCxFQUFvQjtBQUN4QjtBQUNBLFFBQU1qQyxZQUFZLEdBQUdDLE1BQU0sQ0FBQ0MsZUFBT0MsR0FBUCxDQUFXLGNBQVgsQ0FBRCxDQUEzQjtBQUVBLFdBQU9ILFlBQVksQ0FBQ0ksU0FBYixDQUNKOEIsWUFESSxDQUNTRCxJQUFJLENBQUNFLGVBRGQsRUFDK0I7QUFDbENDLE1BQUFBLE1BQU0sRUFBRTtBQUNOQyxRQUFBQSxtQkFBbUIsRUFBRU4sY0FEZjtBQUVOTyxRQUFBQSxtQkFBbUIsRUFBRSxZQUZmO0FBR05DLFFBQUFBLGNBQWMsRUFBRVYsYUFIVjtBQUlOVyxRQUFBQSxPQUFPLEVBQUUsSUFKSDtBQUtOQyxRQUFBQSxRQUFRLEVBQUUsS0FMSjtBQU1OQyxRQUFBQSxNQUFNLEVBQUUsY0FORjtBQU9OQyxRQUFBQSxjQUFjLEVBQUVYO0FBUFY7QUFEMEIsS0FEL0IsRUFZSnpCLElBWkksQ0FZQyxVQUFDcUMsT0FBRCxFQUFhO0FBQ2pCLFVBQU1qQyxNQUFXLEdBQUc7QUFDbEJjLFFBQUFBLFdBQVcsRUFBRW1CLE9BQU8sQ0FBQ0MsS0FESDtBQUVsQkMsUUFBQUEsWUFBWSxFQUFFZixjQUZJO0FBR2xCZ0IsUUFBQUEsTUFBTSxFQUFFSCxPQUFPLENBQUM5QixFQUhFO0FBSWxCa0MsUUFBQUEsV0FBVyxFQUFFSixPQUFPLENBQUNELGNBSkg7QUFLbEIvQixRQUFBQSxRQUFRLEVBQUVGLElBQUksQ0FBQ0QsR0FBTDtBQUxRLE9BQXBCO0FBUUEsVUFBTU0sTUFBZ0IsT0FBR0MsYUFBSCxzQkFBZ0JuQixTQUFoQixFQUFrQ2MsTUFBbEMsQ0FBdEI7QUFFQSxhQUFPLHFCQUFNZixRQUFOLEVBQWdCcUIsS0FBaEIsQ0FBc0JGLE1BQXRCLEVBQ0pSLElBREksQ0FDQyxVQUFDVyxNQUFEO0FBQUEsZUFBeUJBLE1BQU0sQ0FBQ0MsSUFBUCxFQUF6QjtBQUFBLE9BREQsRUFFSlosSUFGSSxDQUVDO0FBQUEsWUFBQ2EsV0FBRCx1RUFBeUIsRUFBekI7QUFBQSxlQUFnQ0EsV0FBaEM7QUFBQSxPQUZELENBQVA7QUFHRCxLQTFCSSxXQTJCRSxVQUFDQyxLQUFELEVBQWtCO0FBQ3ZCLFVBQU00QixHQUFHLEdBQUc1QixLQUFLLENBQUM2QixPQUFsQjs7QUFFQSxVQUFHRCxHQUFHLEtBQUssZ0VBQ1QsbUNBREYsRUFDdUM7QUFDckMsZUFBTyx5QkFBUztBQUNkdEQsVUFBQUEsTUFBTSxFQUFOQSxNQURjO0FBRWQyQixVQUFBQSxRQUFRLEVBQUU5QixhQUZJO0FBR2QrQixVQUFBQSxLQUFLLEVBQUU7QUFITyxTQUFULEVBSUpGLEtBSkksRUFJRzNCLE9BSkgsRUFJWWEsSUFKWixDQUlpQjtBQUFBLGlCQUFNLElBQU47QUFBQSxTQUpqQixDQUFQO0FBS0Q7O0FBQ0QsYUFBTyx5QkFBUztBQUNkWixRQUFBQSxNQUFNLEVBQU5BLE1BRGM7QUFFZDJCLFFBQUFBLFFBQVEsRUFBRTlCLGFBRkk7QUFHZCtCLFFBQUFBLEtBQUssRUFBRTtBQUhPLE9BQVQsRUFJSkYsS0FKSSxFQUlHM0IsT0FKSCxFQUlZYSxJQUpaLENBSWlCO0FBQUEsZUFBTSxJQUFOO0FBQUEsT0FKakIsQ0FBUDtBQUtELEtBM0NJLENBQVA7QUE0Q0QsR0FqREksQ0FBUDtBQWtERCxDQS9FTTs7OztBQWlGQSxJQUFNNEMscUJBQXFCLEdBQUcsU0FBeEJBLHFCQUF3QixDQUFDekQsT0FBRCxFQUFzQjBELElBQXRCLEVBQTZDQyxXQUE3QyxFQUFnRjtBQUNuSCxNQUFNMUQsTUFBYyxHQUFHLHVCQUF2QjtBQURtSCxNQUU1R0MsUUFGNEcsR0FFN0VGLE9BRjZFLENBRTVHRSxRQUY0RztBQUFBLE1BRTFGQyxTQUYwRixHQUU3RUgsT0FGNkUsQ0FFbEdJLE1BRmtHO0FBSW5ILFNBQU8sb0JBQVFKLE9BQVIsRUFBaUJHLFNBQWpCLEVBQ0pVLElBREksQ0FDQyxVQUFDMEIsSUFBRCxFQUFvQjtBQUN4QjtBQUR3QixRQUdoQnFCLE9BSGdCLEdBWVZyQixJQVpVLENBR3RCc0IsSUFIc0I7QUFBQSx3QkFZVnRCLElBWlUsQ0FJdEJPLE9BSnNCO0FBQUEsUUFJdEJBLE9BSnNCLDhCQUlaLElBSlk7QUFBQSxRQUt0QmdCLElBTHNCLEdBWVZ2QixJQVpVLENBS3RCdUIsSUFMc0I7QUFBQSx5QkFZVnZCLElBWlUsQ0FNdEJRLFFBTnNCO0FBQUEsUUFNdEJBLFFBTnNCLCtCQU1YLEtBTlc7QUFBQSxRQU90QmdCLEdBUHNCLEdBWVZ4QixJQVpVLENBT3RCd0IsR0FQc0I7QUFBQSxRQVF0QkMsS0FSc0IsR0FZVnpCLElBWlUsQ0FRdEJ5QixLQVJzQjtBQUFBLFFBU3RCQyxLQVRzQixHQVlWMUIsSUFaVSxDQVN0QjBCLEtBVHNCO0FBQUEsUUFVdEJDLElBVnNCLEdBWVYzQixJQVpVLENBVXRCMkIsSUFWc0I7QUFBQSxRQVd0QjdELFFBWHNCLEdBWVZrQyxJQVpVLENBV3RCbEMsUUFYc0IsRUFjeEI7O0FBQ0EsUUFBTThELE1BQWMsR0FBRyxDQUFFQyxnQkFBU0MsVUFBVCxDQUFvQk4sR0FBcEIsRUFBeUJPLFFBQXpCLENBQWtDLEdBQWxDLENBQXpCO0FBQ0EsUUFBTUMsUUFBZ0IsR0FBRyxDQUFFSCxnQkFBU0MsVUFBVCxDQUFvQk4sR0FBcEIsRUFBeUJPLFFBQXpCLENBQWtDLEdBQWxDLENBQTNCO0FBQ0EsUUFBTUUsT0FBZSxHQUFHLENBQUVKLGdCQUFTQyxVQUFULENBQW9CTixHQUFwQixFQUF5Qk8sUUFBekIsQ0FBa0MsR0FBbEMsQ0FBMUIsQ0FqQndCLENBbUJ4Qjs7QUFuQndCLFFBcUJ0QnRDLGFBckJzQixHQTRCSDBCLElBNUJHLENBcUJ0QjFCLGFBckJzQjtBQUFBLFFBc0J0QnlDLGFBdEJzQixHQTRCSGYsSUE1QkcsQ0FzQnRCZSxhQXRCc0I7QUFBQSxRQXVCdEJDLFFBdkJzQixHQTRCSGhCLElBNUJHLENBdUJ0QmdCLFFBdkJzQjtBQUFBLFFBd0J0QkMsT0F4QnNCLEdBNEJIakIsSUE1QkcsQ0F3QnRCaUIsT0F4QnNCO0FBQUEsUUF5QnRCQyxLQXpCc0IsR0E0QkhsQixJQTVCRyxDQXlCdEJrQixLQXpCc0I7QUFBQSxRQTBCdEJDLE9BMUJzQixHQTRCSG5CLElBNUJHLENBMEJ0Qm1CLE9BMUJzQjtBQUFBLFFBMkJ0QkMsR0EzQnNCLEdBNEJIcEIsSUE1QkcsQ0EyQnRCb0IsR0EzQnNCLEVBOEJ4Qjs7QUFDQSxRQUFNQyxRQUFnQixHQUFHTixhQUFhLEdBQUdMLGdCQUFTWSxLQUFULEdBQWlCQyxXQUFwQixHQUFrQyxDQUF4RSxDQS9Cd0IsQ0FpQ3hCOztBQUNBLFFBQU0zRSxZQUFZLEdBQUdDLE1BQU0sQ0FBQ0MsZUFBT0MsR0FBUCxDQUFXLGNBQVgsQ0FBRCxDQUEzQjtBQUVBLFdBQU9ILFlBQVksQ0FBQzRFLFFBQWIsQ0FDSnZFLE1BREksQ0FDRztBQUNObUMsTUFBQUEsT0FBTyxFQUFQQSxPQURNO0FBRU5rQixNQUFBQSxLQUFLLEVBQUxBLEtBRk07QUFHTm1CLE1BQUFBLGdCQUFnQixFQUFFO0FBQ2hCcEMsUUFBQUEsUUFBUSxFQUFSQSxRQURnQjtBQUVoQnFDLFFBQUFBLFNBQVMsRUFBRVYsUUFGSztBQUdoQlcsUUFBQUEsUUFBUSxFQUFFVixPQUhNO0FBSWhCVyxRQUFBQSxNQUFNLEVBQUV0RCxhQUpRO0FBS2hCZ0IsUUFBQUEsTUFBTSxFQUFFO0FBTFEsT0FIWjtBQVVOdUMsTUFBQUEsWUFBWSxFQUFFO0FBQ1pDLFFBQUFBLE9BQU8sRUFBRTtBQUNQMUIsVUFBQUEsSUFBSSxFQUFKQSxJQURPO0FBRVAyQixVQUFBQSxLQUFLLEVBQUVaLE9BRkE7QUFHUGEsVUFBQUEsV0FBVyxFQUFFWixHQUhOO0FBSVBGLFVBQUFBLEtBQUssRUFBTEE7QUFKTyxTQURHO0FBT1piLFFBQUFBLEdBQUcsRUFBRTtBQUNINEIsVUFBQUEsR0FBRyxFQUFFeEIsTUFERjtBQUVIeUIsVUFBQUEsS0FBSyxFQUFFckIsUUFGSjtBQUdIc0IsVUFBQUEsSUFBSSxFQUFFckI7QUFISCxTQVBPO0FBWVpzQixRQUFBQSxVQUFVLEVBQUU3QixLQVpBO0FBYVo4QixRQUFBQSxTQUFTLEVBQUU3QixJQWJDO0FBY1o7QUFDQTtBQUNBOEIsUUFBQUEsSUFBSSxFQUFFO0FBaEJNLE9BVlI7QUE0Qk5DLE1BQUFBLE9BQU8sRUFBRSxJQTVCSDtBQTZCTnJGLE1BQUFBLFFBQVEsRUFBRTtBQUNSUixRQUFBQSxNQUFNLEVBQUV3RCxPQURBO0FBRVJ2RCxRQUFBQSxRQUFRLEVBQVJBO0FBRlEsT0E3Qko7QUFpQ042RixNQUFBQSxjQUFjLEVBQUU7QUFDZEMsUUFBQUEsSUFBSSxFQUFFcEIsUUFEUTtBQUVkcUIsUUFBQUEsRUFBRSxFQUFFekMsV0FBVyxDQUFDeUMsRUFGRjtBQUdkQyxRQUFBQSxVQUFVLEVBQUUxQyxXQUFXLENBQUMyQyxTQUFaLENBQXNCNUQ7QUFIcEI7QUFqQ1YsS0FESCxFQXdDSjdCLElBeENJLENBd0NDLFVBQUNDLFFBQUQsRUFBYztBQUNsQjtBQUNBLFVBQU1DLEdBQVcsR0FBR0MsSUFBSSxDQUFDRCxHQUFMLEVBQXBCO0FBQ0EsVUFBTUUsTUFBZ0IsR0FBRztBQUN2QkMsUUFBQUEsUUFBUSxFQUFFSCxHQURhO0FBRXZCMEIsUUFBQUEsZUFBZSxFQUFFM0IsUUFBUSxDQUFDTTtBQUZILE9BQXpCO0FBS0EsVUFBTUMsTUFBZ0IsT0FBR0MsYUFBSCxzQkFBZ0JzQyxPQUFoQixFQUFnQzNDLE1BQWhDLENBQXRCO0FBRUEsYUFBTyxxQkFBTWYsUUFBTixFQUFnQnFCLEtBQWhCLENBQXNCRixNQUF0QixFQUNKUixJQURJLENBQ0MsVUFBQ1csTUFBRDtBQUFBLGVBQXlCQSxNQUFNLENBQUNDLElBQVAsRUFBekI7QUFBQSxPQURELEVBRUpaLElBRkksQ0FFQztBQUFBLFlBQUNhLFdBQUQsdUVBQXlCLEVBQXpCO0FBQUEsZUFBZ0MsQ0FBQyx5QkFBUUEsV0FBUixDQUFqQztBQUFBLE9BRkQsV0FHRSxVQUFDQyxLQUFEO0FBQUEsZUFBa0IseUJBQVM7QUFDaEMxQixVQUFBQSxNQUFNLEVBQU5BLE1BRGdDO0FBRWhDMkIsVUFBQUEsUUFBUSxFQUFFOUIsYUFGc0I7QUFHaEMrQixVQUFBQSxLQUFLLEVBQUU7QUFIeUIsU0FBVCxFQUl0QkYsS0FKc0IsRUFJZjNCLE9BSmUsRUFJTmEsSUFKTSxDQUlEO0FBQUEsaUJBQU0sSUFBTjtBQUFBLFNBSkMsQ0FBbEI7QUFBQSxPQUhGLENBQVA7QUFRRCxLQTFESSxDQUFQO0FBMkRELEdBaEdJLENBQVA7QUFpR0QsQ0FyR007Ozs7QUF1R0EsSUFBTTBGLGFBQWEsR0FBRyxTQUFoQkEsYUFBZ0IsQ0FBQ3ZHLE9BQUQsRUFBc0IwRCxJQUF0QixFQUEwRTtBQUNyRyxNQUFNekQsTUFBYyxHQUFHLGVBQXZCO0FBRHFHLE1BRTlGQyxRQUY4RixHQUUvREYsT0FGK0QsQ0FFOUZFLFFBRjhGO0FBQUEsTUFFNUVDLFNBRjRFLEdBRS9ESCxPQUYrRCxDQUVwRkksTUFGb0Y7QUFBQSxNQUtuRzRCLGFBTG1HLEdBZWhGMEIsSUFmZ0YsQ0FLbkcxQixhQUxtRztBQUFBLE1BTW5HOEIsSUFObUcsR0FlaEZKLElBZmdGLENBTW5HSSxJQU5tRztBQUFBLE1BT25HaEIsT0FQbUcsR0FlaEZZLElBZmdGLENBT25HWixPQVBtRztBQUFBLE1BUW5HMEQsR0FSbUcsR0FlaEY5QyxJQWZnRixDQVFuRzhDLEdBUm1HO0FBQUEsTUFTbkc5QixRQVRtRyxHQWVoRmhCLElBZmdGLENBU25HZ0IsUUFUbUc7QUFBQSxNQVVuR0MsT0FWbUcsR0FlaEZqQixJQWZnRixDQVVuR2lCLE9BVm1HO0FBQUEsTUFXbkcxQyxRQVhtRyxHQWVoRnlCLElBZmdGLENBV25HekIsUUFYbUc7QUFBQSxNQVluRzRDLE9BWm1HLEdBZWhGbkIsSUFmZ0YsQ0FZbkdtQixPQVptRztBQUFBLE1BYW5HRCxLQWJtRyxHQWVoRmxCLElBZmdGLENBYW5Ha0IsS0FibUc7QUFBQSxNQWNuR0UsR0FkbUcsR0FlaEZwQixJQWZnRixDQWNuR29CLEdBZG1HO0FBZ0JyRyxNQUFNMkIsWUFBb0IsR0FBRyxxQkFBU3pFLGFBQVQsRUFBd0IsRUFBeEIsQ0FBN0I7O0FBRUEsTUFBRyxDQUFDeUUsWUFBSixFQUFrQjtBQUNoQixVQUFNLElBQUlyRSx3QkFBSixDQUFjLDZCQUFkLENBQU47QUFDRDs7QUFFRCxNQUFNc0UsY0FBc0IsR0FBRyxxQkFBU2hDLFFBQVQsRUFBbUIsQ0FBbkIsQ0FBL0I7O0FBRUEsTUFBRyxDQUFDZ0MsY0FBSixFQUFvQjtBQUNsQixVQUFNLElBQUl0RSx3QkFBSixDQUFjLGdDQUFkLENBQU47QUFDRDs7QUFFRCxNQUFNdUUsYUFBcUIsR0FBRyxxQkFBU2hDLE9BQVQsRUFBa0IsQ0FBbEIsQ0FBOUI7O0FBRUEsTUFBRyxDQUFDZ0MsYUFBSixFQUFtQjtBQUNqQixVQUFNLElBQUl2RSx3QkFBSixDQUFjLCtCQUFkLENBQU47QUFDRDs7QUFFRCxNQUFNd0UsU0FBaUIsR0FBRyxxQkFBU0osR0FBVCxFQUFjLENBQWQsQ0FBMUI7O0FBRUEsTUFBRyxDQUFDSSxTQUFKLEVBQWU7QUFDYixVQUFNLElBQUl4RSx3QkFBSixDQUFjLDBCQUFkLENBQU47QUFDRDs7QUFFRCxNQUFNeUUsV0FBNEIsR0FBRyxFQUFyQztBQUNBLE1BQU1DLFVBQWtCLEdBQUcseUJBQWFoRCxJQUFiLEVBQW1CLEVBQW5CLENBQTNCOztBQUVBLE1BQUdnRCxVQUFILEVBQWU7QUFDYkQsSUFBQUEsV0FBVyxDQUFDL0MsSUFBWixHQUFtQmdELFVBQW5CO0FBQ0Q7O0FBRUQsTUFBTUMsYUFBcUIsR0FBRyx5QkFBYWpFLE9BQWIsRUFBc0IsRUFBdEIsQ0FBOUI7O0FBRUEsTUFBR2lFLGFBQUgsRUFBa0I7QUFDaEJGLElBQUFBLFdBQVcsQ0FBQy9ELE9BQVosR0FBc0JpRSxhQUF0QjtBQUNEOztBQUVELE1BQU0xRSxjQUFzQixHQUFHLHlCQUFhSixRQUFiLEVBQXVCLEVBQXZCLENBQS9COztBQUVBLE1BQUdJLGNBQUgsRUFBbUI7QUFDakJ3RSxJQUFBQSxXQUFXLENBQUM1RSxRQUFaLEdBQXVCSSxjQUF2QjtBQUNEOztBQUVELE1BQU0yRSxZQUFvQixHQUFHLHlCQUFhbkMsT0FBYixFQUFzQixFQUF0QixDQUE3Qjs7QUFFQSxNQUFHbUMsWUFBSCxFQUFpQjtBQUNmSCxJQUFBQSxXQUFXLENBQUNoQyxPQUFaLEdBQXNCbUMsWUFBdEI7QUFDRDs7QUFFRCxNQUFNQyxXQUFtQixHQUFHLHlCQUFhckMsS0FBYixFQUFvQixFQUFwQixDQUE1Qjs7QUFFQSxNQUFHcUMsV0FBSCxFQUFnQjtBQUNkSixJQUFBQSxXQUFXLENBQUNqQyxLQUFaLEdBQW9CcUMsV0FBcEI7QUFDRDs7QUFFRCxNQUFNQyxTQUFpQixHQUFHLHlCQUFhcEMsR0FBYixFQUFrQixFQUFsQixDQUExQjs7QUFFQSxNQUFHb0MsU0FBSCxFQUFjO0FBQ1pMLElBQUFBLFdBQVcsQ0FBQy9CLEdBQVosR0FBa0JvQyxTQUFsQjtBQUNELEdBM0VvRyxDQTZFckc7OztBQUNBLFNBQU8sb0JBQVFsSCxPQUFSLEVBQWlCRyxTQUFqQixFQUNKVSxJQURJLENBQ0MsWUFBeUI7QUFBQSxRQUF4QjBCLElBQXdCLHVFQUFQLEVBQU87QUFDN0I7QUFDQSxRQUFNakMsWUFBWSxHQUFHQyxNQUFNLENBQUNDLGVBQU9DLEdBQVAsQ0FBVyxjQUFYLENBQUQsQ0FBM0I7QUFFQSxXQUFPSCxZQUFZLENBQUNJLFNBQWIsQ0FDSjhCLFlBREksQ0FDU0QsSUFBSSxDQUFDcEIsZ0JBRGQsRUFDZ0M7QUFDbkN1QixNQUFBQSxNQUFNLEVBQUU7QUFDTnlFLFFBQUFBLFlBQVksRUFBRUwsVUFEUjtBQUVOTSxRQUFBQSxlQUFlLEVBQUVMLGFBRlg7QUFHTk0sUUFBQUEsYUFBYSxFQUFFTCxZQUhUO0FBSU5NLFFBQUFBLGFBQWEsRUFBRUwsV0FKVDtBQUtOTSxRQUFBQSxXQUFXLEVBQUVMLFNBTFA7QUFNTm5FLFFBQUFBLFFBQVEsRUFBRSxLQU5KO0FBT055RCxRQUFBQSxHQUFHLEVBQUVJLFNBUEM7QUFRTnhCLFFBQUFBLFNBQVMsRUFBRXNCLGNBUkw7QUFTTnJCLFFBQUFBLFFBQVEsRUFBRXNCLGFBVEo7QUFVTmEsUUFBQUEsSUFBSSxFQUFFbkYsY0FWQTtBQVdOaUQsUUFBQUEsTUFBTSxFQUFFbUIsWUFYRjtBQVlOekQsUUFBQUEsTUFBTSxFQUFFO0FBWkY7QUFEMkIsS0FEaEMsRUFpQkpuQyxJQWpCSSxDQWlCQyxVQUFDNEcsU0FBRCxFQUFlO0FBQ25CLFVBQU0xRyxHQUFXLEdBQUdDLElBQUksQ0FBQ0QsR0FBTCxFQUFwQjs7QUFDQSxVQUFNMkcsTUFBTSxxQkFDUGIsV0FETztBQUVWaEQsUUFBQUEsSUFBSSxFQUFFLDhDQUEyQjFELFNBQTNCLEVBRkk7QUFHVjZCLFFBQUFBLGFBQWEsRUFBRXlGLFNBQVMsQ0FBQ3RFLEtBSGY7QUFJVndFLFFBQUFBLEtBQUssRUFBRTVHLEdBSkc7QUFLVjZHLFFBQUFBLEtBQUssRUFBRUgsU0FBUyxDQUFDRyxLQUxQO0FBTVZwQixRQUFBQSxHQUFHLEVBQUVpQixTQUFTLENBQUNJLFNBTkw7QUFPVm5ELFFBQUFBLFFBQVEsRUFBUkEsUUFQVTtBQVFWQyxRQUFBQSxPQUFPLEVBQVBBLE9BUlU7QUFTVnpELFFBQUFBLFFBQVEsRUFBRUgsR0FUQTtBQVVWWCxRQUFBQSxNQUFNLEVBQUVEO0FBVkUsUUFBWjs7QUFZQSxVQUFNMkgsWUFBc0IsT0FBR3hHLGFBQUgsc0JBQWdCb0csTUFBaEIsQ0FBNUI7QUFFQSxhQUFPLHFCQUFNeEgsUUFBTixFQUFnQnFCLEtBQWhCLENBQXNCdUcsWUFBdEIsRUFDSmpILElBREksQ0FDQyxVQUFDVyxNQUFEO0FBQUEsZUFBeUJBLE1BQU0sQ0FBQ0MsSUFBUCxFQUF6QjtBQUFBLE9BREQsRUFFSlosSUFGSSxDQUVDLFlBQW1DO0FBQUEsWUFBbENrSCxPQUFrQyx1RUFBUCxFQUFPOztBQUN2QyxZQUFHLENBQUMseUJBQVFBLE9BQVIsQ0FBSixFQUFzQjtBQUNwQjtBQURvQixjQUVSQyxNQUZRLEdBRWlCdEUsSUFGakIsQ0FFYnVFLEdBRmE7QUFBQSxjQUVNQyxPQUZOLEdBRWlCeEUsSUFGakIsQ0FFQUcsSUFGQTtBQUdwQixjQUFNc0UsY0FBYyxHQUFHLHFCQUFNakksUUFBTixFQUFnQmlJLGNBQWhCLENBQStCLFlBQS9CLENBQXZCO0FBQ0EsY0FBTUMsTUFBTSxHQUFHLHlDQUFzQkYsT0FBdEIsRUFBZjtBQUNBLGNBQU1HLElBQVMsR0FBRztBQUNoQnhFLFlBQUFBLElBQUksRUFBRXVFO0FBRFUsV0FBbEI7QUFJQSxpQkFBT0QsY0FBYyxDQUFDRyxJQUFmLENBQW9CRCxJQUFwQixrQkFBbUNsSSxTQUFuQyxHQUFnRDZILE1BQWhELEVBQXdEbkgsSUFBeEQsQ0FBNkQ7QUFBQSxtQkFBTTZDLElBQU47QUFBQSxXQUE3RCxDQUFQO0FBQ0Q7O0FBRUQsZUFBT3FFLE9BQVA7QUFDRCxPQWhCSSxDQUFQO0FBaUJELEtBbERJLFdBbURFLFVBQUNwRyxLQUFEO0FBQUEsYUFBa0IseUJBQVM7QUFDaEMxQixRQUFBQSxNQUFNLEVBQU5BLE1BRGdDO0FBRWhDMkIsUUFBQUEsUUFBUSxFQUFFOUIsYUFGc0I7QUFHaEMrQixRQUFBQSxLQUFLLEVBQUU7QUFIeUIsT0FBVCxFQUl0QkYsS0FKc0IsRUFJZjNCLE9BSmUsRUFJTmEsSUFKTSxDQUlEO0FBQUEsZUFBTSxJQUFOO0FBQUEsT0FKQyxDQUFsQjtBQUFBLEtBbkRGLENBQVA7QUF3REQsR0E3REksQ0FBUDtBQThERCxDQTVJTTs7OztBQThJQSxJQUFNMEgsZ0JBQWdCLEdBQUcsU0FBbkJBLGdCQUFtQixDQUFDdkksT0FBRCxFQUFzQjBELElBQXRCLEVBQTBFO0FBQUEsTUFDakd4RCxRQURpRyxHQUNsRUYsT0FEa0UsQ0FDakdFLFFBRGlHO0FBQUEsTUFDL0VDLFNBRCtFLEdBQ2xFSCxPQURrRSxDQUN2RkksTUFEdUY7QUFBQSxNQUl0RzBELElBSnNHLEdBYW5GSixJQWJtRixDQUl0R0ksSUFKc0c7QUFBQSxNQUt0R2hCLE9BTHNHLEdBYW5GWSxJQWJtRixDQUt0R1osT0FMc0c7QUFBQSxNQU10RzRCLFFBTnNHLEdBYW5GaEIsSUFibUYsQ0FNdEdnQixRQU5zRztBQUFBLE1BT3RHQyxPQVBzRyxHQWFuRmpCLElBYm1GLENBT3RHaUIsT0FQc0c7QUFBQSxNQVF0RzFDLFFBUnNHLEdBYW5GeUIsSUFibUYsQ0FRdEd6QixRQVJzRztBQUFBLE1BU3RHYixFQVRzRyxHQWFuRnNDLElBYm1GLENBU3RHdEMsRUFUc0c7QUFBQSxNQVV0R3lELE9BVnNHLEdBYW5GbkIsSUFibUYsQ0FVdEdtQixPQVZzRztBQUFBLE1BV3RHRCxLQVhzRyxHQWFuRmxCLElBYm1GLENBV3RHa0IsS0FYc0c7QUFBQSxNQVl0R0UsR0Fac0csR0FhbkZwQixJQWJtRixDQVl0R29CLEdBWnNHO0FBZXhHLE1BQU0wRCxRQUFnQixHQUFHLG9CQUFRcEgsRUFBUixDQUF6Qjs7QUFFQSxNQUFHb0gsUUFBSCxFQUFhO0FBQ1gsVUFBTSxJQUFJcEcsd0JBQUosQ0FBYyx5QkFBZCxDQUFOO0FBQ0Q7O0FBRUQsTUFBTXlFLFdBQTRCLEdBQUcsRUFBckM7QUFDQSxNQUFNSCxjQUFzQixHQUFHLHFCQUFTaEMsUUFBVCxFQUFtQixDQUFuQixDQUEvQjtBQUNBLE1BQU1pQyxhQUFxQixHQUFHLHFCQUFTaEMsT0FBVCxFQUFrQixDQUFsQixDQUE5QjtBQUNBLE1BQU1tQyxVQUFrQixHQUFHLHlCQUFhaEQsSUFBYixFQUFtQixFQUFuQixDQUEzQjtBQUNBLE1BQU1pRCxhQUFxQixHQUFHLHlCQUFhakUsT0FBYixFQUFzQixDQUF0QixDQUE5QjtBQUNBLE1BQU1ULGNBQXNCLEdBQUcseUJBQWFKLFFBQWIsRUFBdUIsRUFBdkIsQ0FBL0I7QUFDQSxNQUFNd0csYUFBcUIsR0FBRyx3QkFBWTVELE9BQVosRUFBcUIsRUFBckIsQ0FBOUI7QUFDQSxNQUFNb0MsV0FBbUIsR0FBRyx5QkFBYXJDLEtBQWIsRUFBb0IsQ0FBcEIsQ0FBNUI7QUFDQSxNQUFNc0MsU0FBaUIsR0FBRyx5QkFBYXBDLEdBQWIsRUFBa0IsRUFBbEIsQ0FBMUI7O0FBRUEsTUFBRzRCLGNBQUgsRUFBbUI7QUFDakJHLElBQUFBLFdBQVcsQ0FBQ25DLFFBQVosR0FBdUJnQyxjQUF2QjtBQUNEOztBQUVELE1BQUdDLGFBQUgsRUFBa0I7QUFDaEJFLElBQUFBLFdBQVcsQ0FBQ2xDLE9BQVosR0FBc0JnQyxhQUF0QjtBQUNEOztBQUVELE1BQUdHLFVBQUgsRUFBZTtBQUNiRCxJQUFBQSxXQUFXLENBQUMvQyxJQUFaLEdBQW1CZ0QsVUFBbkI7QUFDRDs7QUFFRCxNQUFHQyxhQUFILEVBQWtCO0FBQ2hCRixJQUFBQSxXQUFXLENBQUMvRCxPQUFaLEdBQXNCaUUsYUFBdEI7QUFDRDs7QUFFRCxNQUFHMUUsY0FBSCxFQUFtQjtBQUNqQndFLElBQUFBLFdBQVcsQ0FBQzVFLFFBQVosR0FBdUJJLGNBQXZCO0FBQ0Q7O0FBRUQsTUFBR29HLGFBQUgsRUFBa0I7QUFDaEI1QixJQUFBQSxXQUFXLENBQUNoQyxPQUFaLEdBQXNCNEQsYUFBdEI7QUFDRDs7QUFFRCxNQUFHeEIsV0FBSCxFQUFnQjtBQUNkSixJQUFBQSxXQUFXLENBQUNqQyxLQUFaLEdBQW9CcUMsV0FBcEI7QUFDRDs7QUFFRCxNQUFHQyxTQUFILEVBQWM7QUFDWkwsSUFBQUEsV0FBVyxDQUFDL0IsR0FBWixHQUFrQm9DLFNBQWxCO0FBQ0Q7O0FBRUQsTUFBTWpHLE1BQVcsR0FBRzRGLFdBQXBCO0FBQ0EsTUFBTXhGLE1BQWdCLE9BQUdDLGFBQUgsc0JBR0drSCxRQUhILEVBRzhCckksU0FIOUIsRUFJQWMsTUFKQSxFQVVHZCxTQVZILENBQXRCO0FBZ0JBLFNBQU8scUJBQU1ELFFBQU4sRUFBZ0JxQixLQUFoQixDQUFzQkYsTUFBdEIsRUFDSlIsSUFESSxDQUNDLFVBQUNXLE1BQUQ7QUFBQSxXQUF5QkEsTUFBTSxDQUFDQyxJQUFQLEVBQXpCO0FBQUEsR0FERCxFQUVKWixJQUZJLENBRUMsWUFBb0M7QUFBQSxRQUFuQzZILE9BQW1DLHVFQUF6QjtBQUFDaEYsTUFBQUEsSUFBSSxFQUFFLEVBQVA7QUFBV25CLE1BQUFBLElBQUksRUFBRTtBQUFqQixLQUF5QjtBQUN4QyxRQUFNb0csV0FBNEIsR0FBR0QsT0FBTyxDQUFDaEYsSUFBN0M7QUFEd0MsUUFFakNuQixJQUZpQyxHQUV6Qm1HLE9BRnlCLENBRWpDbkcsSUFGaUM7O0FBSXhDLFFBQUcseUJBQVFvRyxXQUFSLENBQUgsRUFBeUI7QUFDdkIsWUFBTSxJQUFJdkcsd0JBQUosQ0FBYyxXQUFkLENBQU47QUFDRCxLQU51QyxDQVF4Qzs7O0FBQ0EsUUFBTTlCLFlBQVksR0FBR0MsTUFBTSxDQUFDQyxlQUFPQyxHQUFQLENBQVcsY0FBWCxDQUFELENBQTNCO0FBRUEsV0FBT0gsWUFBWSxDQUFDSSxTQUFiLENBQ0prSSxVQURJLENBQ09yRyxJQUFJLENBQUNwQixnQkFEWixFQUM4QnVDLElBQUksQ0FBQ21GLFFBRG5DLEVBQzZDO0FBQ2hEMUIsTUFBQUEsWUFBWSxFQUFFTCxVQURrQztBQUVoRE0sTUFBQUEsZUFBZSxFQUFFTCxhQUYrQjtBQUdoRE0sTUFBQUEsYUFBYSxFQUFFb0IsYUFIaUM7QUFJaERuQixNQUFBQSxhQUFhLEVBQUVMLFdBSmlDO0FBS2hETSxNQUFBQSxXQUFXLEVBQUVMLFNBTG1DO0FBTWhEOUIsTUFBQUEsU0FBUyxFQUFFc0IsY0FOcUM7QUFPaERyQixNQUFBQSxRQUFRLEVBQUVzQixhQVBzQztBQVFoRGEsTUFBQUEsSUFBSSxFQUFFbkY7QUFSMEMsS0FEN0MsRUFXSnhCLElBWEksQ0FXQztBQUFBLGFBQU02QyxJQUFOO0FBQUEsS0FYRCxXQVlFLFVBQUMvQixLQUFELEVBQWtCO0FBQ3ZCbUgsTUFBQUEsT0FBTyxDQUFDQyxHQUFSLENBQVksNkJBQVosRUFBMkNwSCxLQUEzQztBQUNBLFlBQU0sSUFBSVMsd0JBQUosQ0FBYyxlQUFkLENBQU47QUFDRCxLQWZJLENBQVA7QUFnQkQsR0E3QkksQ0FBUDtBQThCRCxDQTlHTTs7OztBQWdIQSxJQUFNNEcsY0FBYyxHQUFHLFNBQWpCQSxjQUFpQixDQUFDaEosT0FBRCxFQUFxRDtBQUNqRixNQUFNQyxNQUFjLEdBQUcsZ0JBQXZCO0FBRGlGLE1BRTFFQyxRQUYwRSxHQUUzQ0YsT0FGMkMsQ0FFMUVFLFFBRjBFO0FBQUEsTUFFeERDLFNBRndELEdBRTNDSCxPQUYyQyxDQUVoRUksTUFGZ0U7QUFHakYsTUFBTWlCLE1BQWdCLE9BQUdDLGFBQUgsc0JBQ0tuQixTQURMLENBQXRCO0FBSUEsU0FBTyxxQkFBTUQsUUFBTixFQUFnQnFCLEtBQWhCLENBQXNCRixNQUF0QixFQUNKUixJQURJLENBQ0MsVUFBQ1csTUFBRDtBQUFBLFdBQXlCQSxNQUFNLENBQUN5SCxHQUFQLEVBQXpCO0FBQUEsR0FERCxFQUVKcEksSUFGSSxDQUVDO0FBQUEsUUFBQ3FJLElBQUQsdUVBQTJCLEVBQTNCO0FBQUEsV0FBa0NBLElBQWxDO0FBQUEsR0FGRCxXQUdFLFVBQUN2SCxLQUFEO0FBQUEsV0FBa0IseUJBQVM7QUFDaEMxQixNQUFBQSxNQUFNLEVBQU5BLE1BRGdDO0FBRWhDMkIsTUFBQUEsUUFBUSxFQUFFOUIsYUFGc0I7QUFHaEMrQixNQUFBQSxLQUFLLEVBQUU7QUFIeUIsS0FBVCxFQUl0QkYsS0FKc0IsRUFJZjNCLE9BSmUsRUFJTmEsSUFKTSxDQUlEO0FBQUEsYUFBTSxJQUFOO0FBQUEsS0FKQyxDQUFsQjtBQUFBLEdBSEYsQ0FBUDtBQVFELENBZk07Ozs7QUFpQkEsSUFBTXNJLGdCQUFnQixHQUFHLFNBQW5CQSxnQkFBbUIsQ0FBQ25KLE9BQUQsRUFBc0JnSSxNQUF0QixFQUEyRDtBQUFBLE1BQ2xGOUgsUUFEa0YsR0FDbkRGLE9BRG1ELENBQ2xGRSxRQURrRjtBQUFBLE1BQ2hFQyxTQURnRSxHQUNuREgsT0FEbUQsQ0FDeEVJLE1BRHdFO0FBRXpGLE1BQU1nSixZQUFvQixHQUFHLG9CQUFRcEIsTUFBUixDQUE3QjtBQUNBLE1BQU0zRyxNQUFnQixPQUFHQyxhQUFILHNCQUdHOEgsWUFISCxFQUdrQ2pKLFNBSGxDLEVBVUdBLFNBVkgsQ0FBdEI7QUFnQkEsU0FBTyxxQkFBTUQsUUFBTixFQUFnQnFCLEtBQWhCLENBQXNCRixNQUF0QixFQUNKUixJQURJLENBQ0MsVUFBQ1csTUFBRDtBQUFBLFdBQXlCQSxNQUFNLENBQUNDLElBQVAsRUFBekI7QUFBQSxHQURELEVBRUpaLElBRkksQ0FFQyxZQUFtQztBQUFBLFFBQWxDd0ksTUFBa0MsdUVBQXpCO0FBQUMzRixNQUFBQSxJQUFJLEVBQUUsRUFBUDtBQUFXbkIsTUFBQUEsSUFBSSxFQUFFO0FBQWpCLEtBQXlCOztBQUN2QyxRQUFHLHlCQUFROEcsTUFBUixDQUFILEVBQW9CO0FBQ2xCLGFBQU8sS0FBUDtBQUNEOztBQUhzQyxRQUtoQzNGLElBTGdDLEdBS2xCMkYsTUFMa0IsQ0FLaEMzRixJQUxnQztBQUFBLFFBSzFCbkIsSUFMMEIsR0FLbEI4RyxNQUxrQixDQUsxQjlHLElBTDBCO0FBQUEsUUFNMUIyRixPQU4wQixHQU1meEUsSUFOZSxDQU1oQ0csSUFOZ0MsRUFRdkM7O0FBQ0EsUUFBTXNFLGNBQWMsR0FBRyxxQkFBTWpJLFFBQU4sRUFBZ0JpSSxjQUFoQixDQUErQixZQUEvQixDQUF2QjtBQUVBLFdBQU9BLGNBQWMsQ0FBQ21CLFFBQWYsQ0FBd0JwQixPQUF4QixFQUNKckgsSUFESSxDQUNDLFVBQUMwSSxLQUFELEVBQVc7QUFDZixVQUFHQSxLQUFLLENBQUNDLE1BQVQsRUFBaUI7QUFDZixlQUFPQyxPQUFPLENBQUNSLEdBQVIsQ0FDTE0sS0FBSyxDQUFDRyxHQUFOLENBQVUsVUFBQ3JCLElBQUQsRUFBVTtBQUFBLGNBQ0xzQixPQURLLEdBQ010QixJQUROLENBQ1h4RSxJQURXO0FBRWxCLGNBQU0rRixZQUFzQixPQUFHdEksYUFBSCxzQkFBc0JxSSxPQUF0QixDQUE1QjtBQUNBLGlCQUFPLHFCQUFNekosUUFBTixFQUFnQnFCLEtBQWhCLENBQXNCcUksWUFBdEIsQ0FBUDtBQUNELFNBSkQsQ0FESyxFQU1KL0ksSUFOSSxDQU1DLFlBQU07QUFDVjtBQUNBLGNBQU1QLFlBQVksR0FBR0MsTUFBTSxDQUFDQyxlQUFPQyxHQUFQLENBQVcsY0FBWCxDQUFELENBQTNCO0FBRUEsaUJBQU9ILFlBQVksQ0FBQ0ksU0FBYixDQUNKbUosVUFESSxDQUNPdEgsSUFBSSxDQUFDcEIsZ0JBRFosRUFDOEJ1QyxJQUFJLENBQUNtRixRQURuQyxFQUVKaEksSUFGSSxDQUVDO0FBQUEsbUJBQU0sSUFBTjtBQUFBLFdBRkQsV0FHRSxVQUFDYyxLQUFELEVBQWtCO0FBQ3ZCbUgsWUFBQUEsT0FBTyxDQUFDQyxHQUFSLENBQVksNkJBQVosRUFBMkNwSCxLQUEzQztBQUNBLGtCQUFNLElBQUlTLHdCQUFKLENBQWMsZUFBZCxDQUFOO0FBQ0QsV0FOSSxDQUFQO0FBT0QsU0FqQkksQ0FBUDtBQWtCRDs7QUFFRCxhQUFPLEtBQVA7QUFDRCxLQXhCSSxDQUFQO0FBeUJELEdBdENJLENBQVA7QUF1Q0QsQ0ExRE07Ozs7QUE0REEsSUFBTTBILHdCQUF3QixHQUFHLFNBQTNCQSx3QkFBMkIsQ0FBQzlKLE9BQUQsRUFBc0JxRCxNQUF0QixFQUEyRDtBQUFBLE1BQzFGbkQsUUFEMEYsR0FDM0RGLE9BRDJELENBQzFGRSxRQUQwRjtBQUFBLE1BQ3hFQyxTQUR3RSxHQUMzREgsT0FEMkQsQ0FDaEZJLE1BRGdGLEVBR2pHOztBQUNBLE1BQU1hLE1BQWdCLEdBQUc7QUFDdkJjLElBQUFBLFdBQVcsRUFBRSxFQURVO0FBRXZCcUIsSUFBQUEsWUFBWSxFQUFFLEVBRlM7QUFHdkJDLElBQUFBLE1BQU0sRUFBRSxFQUhlO0FBSXZCQyxJQUFBQSxXQUFXLEVBQUUsRUFKVTtBQUt2QnBDLElBQUFBLFFBQVEsRUFBRUYsSUFBSSxDQUFDRCxHQUFMO0FBTGEsR0FBekI7QUFPQSxNQUFNTSxNQUFnQixPQUFHQyxhQUFILHNCQUFnQm5CLFNBQWhCLEVBQWtDYyxNQUFsQyxDQUF0QjtBQUVBLFNBQU8scUJBQU1mLFFBQU4sRUFBZ0JxQixLQUFoQixDQUFzQkYsTUFBdEIsRUFDSlIsSUFESSxDQUNDLFVBQUNXLE1BQUQ7QUFBQSxXQUF5QkEsTUFBTSxDQUFDQyxJQUFQLEVBQXpCO0FBQUEsR0FERCxFQUVKWixJQUZJLENBRUMsWUFBeUI7QUFBQSxRQUF4QjBCLElBQXdCLHVFQUFQLEVBQU87QUFDN0I7QUFDQSxRQUFNakMsWUFBWSxHQUFHQyxNQUFNLENBQUNDLGVBQU9DLEdBQVAsQ0FBVyxjQUFYLENBQUQsQ0FBM0I7QUFFQSxXQUFPSCxZQUFZLENBQUNJLFNBQWIsQ0FDSnFKLFlBREksQ0FDU3hILElBQUksQ0FBQ0UsZUFEZCxFQUMrQlksTUFEL0IsRUFFSnhDLElBRkksQ0FFQztBQUFBLFVBQUNtSixRQUFELHVFQUFZO0FBQUNDLFFBQUFBLE9BQU8sRUFBRTtBQUFWLE9BQVo7QUFBQSxhQUFpQ0QsUUFBUSxDQUFDQyxPQUExQztBQUFBLEtBRkQsV0FHRTtBQUFBLGFBQU1SLE9BQU8sQ0FBQ1MsT0FBUixDQUFnQixLQUFoQixDQUFOO0FBQUEsS0FIRixDQUFQO0FBSUQsR0FWSSxDQUFQO0FBV0QsQ0F4Qk07Ozs7QUEwQkEsSUFBTUMscUJBQXFCLEdBQUcsU0FBeEJBLHFCQUF3QixDQUFDbkssT0FBRCxFQUFzQm9LLFFBQXRCLEVBQThFO0FBQUEsTUFDMUdsSyxRQUQwRyxHQUMzRUYsT0FEMkUsQ0FDMUdFLFFBRDBHO0FBQUEsTUFDeEZDLFNBRHdGLEdBQzNFSCxPQUQyRSxDQUNoR0ksTUFEZ0c7QUFBQSxNQUUxR2lLLE1BRjBHLEdBRXRGRCxRQUZzRixDQUUxR0MsTUFGMEc7QUFBQSxNQUVsR3RILFFBRmtHLEdBRXRGcUgsUUFGc0YsQ0FFbEdySCxRQUZrRztBQUdqSCxNQUFNdUgsWUFBb0IsR0FBRyxxQkFBU0QsTUFBVCxDQUE3QjtBQUNBLE1BQU1FLGNBQXNCLEdBQUcsc0JBQVV4SCxRQUFWLEVBQW9CLENBQXBCLEVBQXVCLEtBQXZCLEVBQThCeUgsV0FBOUIsRUFBL0I7QUFFQSxTQUFPLG9CQUFReEssT0FBUixFQUFpQkcsU0FBakIsRUFDSlUsSUFESSxDQUNDLFVBQUMwQixJQUFELEVBQW9CO0FBQ3hCO0FBQ0EsUUFBTWpDLFlBQVksR0FBR0MsTUFBTSxDQUFDQyxlQUFPQyxHQUFQLENBQVcsY0FBWCxDQUFELENBQTNCO0FBRUEsV0FBT0gsWUFBWSxDQUFDbUssU0FBYixDQUNKOUosTUFESSxDQUNHO0FBQ04wSixNQUFBQSxNQUFNLEVBQUVDLFlBREY7QUFFTnZILE1BQUFBLFFBQVEsRUFBRXdILGNBRko7QUFHTkcsTUFBQUEsV0FBVyxFQUFFbkksSUFBSSxDQUFDRTtBQUhaLEtBREgsRUFNSjVCLElBTkksQ0FNQyxVQUFDOEosY0FBRCxFQUFvQjtBQUN4QjdCLE1BQUFBLE9BQU8sQ0FBQ0MsR0FBUixDQUFZNEIsY0FBWjtBQUNBLFVBQU01SixHQUFXLEdBQUdDLElBQUksQ0FBQ0QsR0FBTCxFQUFwQjtBQUNBLFVBQU0yRyxNQUF1QixHQUFHO0FBQzlCQyxRQUFBQSxLQUFLLEVBQUU1RyxHQUR1QjtBQUU5QnNKLFFBQUFBLE1BQU0sRUFBRUMsWUFGc0I7QUFHOUJ2SCxRQUFBQSxRQUFRLEVBQUV3SCxjQUhvQjtBQUk5QnJKLFFBQUFBLFFBQVEsRUFBRUgsR0FKb0I7QUFLOUJYLFFBQUFBLE1BQU0sRUFBRUQ7QUFMc0IsT0FBaEM7QUFPQSxVQUFNa0IsTUFBZ0IsT0FBR0MsYUFBSCx1QkFBZ0JvRyxNQUFoQixDQUF0QjtBQUVBLGFBQU8scUJBQU14SCxRQUFOLEVBQWdCcUIsS0FBaEIsQ0FBc0JGLE1BQXRCLEVBQ0pSLElBREksQ0FDQyxVQUFDVyxNQUFEO0FBQUEsZUFBeUJBLE1BQU0sQ0FBQ0MsSUFBUCxFQUF6QjtBQUFBLE9BREQsRUFFSlosSUFGSSxDQUVDLFVBQUMrSixXQUFEO0FBQUEsZUFBa0NBLFdBQWxDO0FBQUEsT0FGRCxDQUFQO0FBR0QsS0FyQkksQ0FBUDtBQXNCRCxHQTNCSSxDQUFQO0FBNEJELENBbENNOzs7O0FBb0NBLElBQU1DLGlCQUFpQixHQUFHLFNBQXBCQSxpQkFBb0IsQ0FBQzdLLE9BQUQsRUFBc0I4SyxPQUF0QixFQUF5RTtBQUFBLE1BQ2pHNUssUUFEaUcsR0FDbEVGLE9BRGtFLENBQ2pHRSxRQURpRztBQUFBLE1BQy9FQyxTQUQrRSxHQUNsRUgsT0FEa0UsQ0FDdkZJLE1BRHVGO0FBQUEsTUFFakdpSyxNQUZpRyxHQUUvQ1MsT0FGK0MsQ0FFakdULE1BRmlHO0FBQUEsTUFFekZVLE9BRnlGLEdBRS9DRCxPQUYrQyxDQUV6RkMsT0FGeUY7QUFBQSxNQUVoRi9DLE1BRmdGLEdBRS9DOEMsT0FGK0MsQ0FFaEY5QyxNQUZnRjtBQUFBLE1BRXhFakYsUUFGd0UsR0FFL0MrSCxPQUYrQyxDQUV4RS9ILFFBRndFO0FBQUEsTUFFOURpSSxXQUY4RCxHQUUvQ0YsT0FGK0MsQ0FFOURFLFdBRjhEO0FBR3hHLE1BQU1ULGNBQWMsR0FBRyxzQkFBVXhILFFBQVYsRUFBb0IsQ0FBcEIsRUFBdUIsS0FBdkIsRUFBOEJ5SCxXQUE5QixFQUF2QjtBQUVBLE1BQU1sSyxZQUFZLEdBQUdDLE1BQU0sQ0FBQ0MsZUFBT0MsR0FBUCxDQUFXLGNBQVgsQ0FBRCxDQUEzQjtBQUVBLFNBQU9ILFlBQVksQ0FBQzJLLE9BQWIsQ0FDSnRLLE1BREksQ0FDRztBQUNOMEosSUFBQUEsTUFBTSxFQUFOQSxNQURNO0FBRU5VLElBQUFBLE9BQU8sRUFBUEEsT0FGTTtBQUdOaEksSUFBQUEsUUFBUSxFQUFFd0gsY0FISjtBQUlOUyxJQUFBQSxXQUFXLEVBQVhBLFdBSk07QUFLTnRJLElBQUFBLE1BQU0sRUFBRXNGO0FBTEYsR0FESCxFQVFKbkgsSUFSSSxDQVFDLFVBQUNxSyxZQUFELEVBQWtCO0FBQ3RCLFFBQU1uSyxHQUFXLEdBQUdDLElBQUksQ0FBQ0QsR0FBTCxFQUFwQjtBQUNBLFFBQU0yRyxNQUFxQixHQUFHO0FBQzVCQyxNQUFBQSxLQUFLLEVBQUU1RyxHQURxQjtBQUU1QnNKLE1BQUFBLE1BQU0sRUFBTkEsTUFGNEI7QUFHNUJVLE1BQUFBLE9BQU8sRUFBUEEsT0FINEI7QUFJNUIvQyxNQUFBQSxNQUFNLEVBQU5BLE1BSjRCO0FBSzVCbUQsTUFBQUEsY0FBYyxFQUFFRCxZQUFZLENBQUNFLFlBTEQ7QUFNNUJDLE1BQUFBLGFBQWEsRUFBRUgsWUFBWSxDQUFDSSxlQU5BO0FBTzVCQyxNQUFBQSxRQUFRLEVBQUVMLFlBQVksQ0FBQzlKLEVBUEs7QUFRNUJvSyxNQUFBQSxZQUFZLEVBQUVOLFlBQVksQ0FBQ08sTUFSQztBQVM1QjFJLE1BQUFBLFFBQVEsRUFBRXdILGNBVGtCO0FBVTVCUyxNQUFBQSxXQUFXLEVBQVhBLFdBVjRCO0FBVzVCOUosTUFBQUEsUUFBUSxFQUFFSCxHQVhrQjtBQVk1QlgsTUFBQUEsTUFBTSxFQUFFRDtBQVpvQixLQUE5QjtBQWNBLFFBQU1rQixNQUFnQixPQUFHQyxhQUFILHVCQUFnQm9HLE1BQWhCLENBQXRCO0FBRUEsV0FBTyxxQkFBTXhILFFBQU4sRUFBZ0JxQixLQUFoQixDQUFzQkYsTUFBdEIsRUFDSlIsSUFESSxDQUNDLFVBQUNXLE1BQUQ7QUFBQSxhQUF5QkEsTUFBTSxDQUFDQyxJQUFQLEVBQXpCO0FBQUEsS0FERCxFQUVKWixJQUZJLENBRUMsVUFBQzZLLFVBQUQ7QUFBQSxhQUErQkEsVUFBL0I7QUFBQSxLQUZELENBQVA7QUFHRCxHQTdCSSxXQThCRSxVQUFDL0osS0FBRCxFQUFrQjtBQUN2Qm1ILElBQUFBLE9BQU8sQ0FBQ0MsR0FBUixDQUFZLDZCQUFaLEVBQTJDcEgsS0FBM0M7QUFDQSxVQUFNLElBQUlTLHdCQUFKLENBQWMsZUFBZCxDQUFOO0FBQ0QsR0FqQ0ksQ0FBUDtBQWtDRCxDQXpDTSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7Y3JlYXRlSGFzaCwgcGFyc2VDaGFyLCBwYXJzZUlkLCBwYXJzZU51bSwgcGFyc2VTdHJpbmcsIHBhcnNlVmFyQ2hhcn0gZnJvbSAnQG5sYWJzL3V0aWxzJztcbmltcG9ydCB7YXFsfSBmcm9tICdhcmFuZ29qcyc7XG5pbXBvcnQge0FxbFF1ZXJ5fSBmcm9tICdhcmFuZ29qcy9saWIvY2pzL2FxbC1xdWVyeSc7XG5pbXBvcnQge0FycmF5Q3Vyc29yfSBmcm9tICdhcmFuZ29qcy9saWIvY2pzL2N1cnNvcic7XG5pbXBvcnQge1VzZXJFcnJvcn0gZnJvbSAnZ3JhcGhxbC1lcnJvcnMnO1xuaW1wb3J0IGlzRW1wdHkgZnJvbSAnbG9kYXNoL2lzRW1wdHknO1xuaW1wb3J0IHtEYXRlVGltZX0gZnJvbSAnbHV4b24nO1xuaW1wb3J0ICogYXMgc3RyaXBlIGZyb20gJ3N0cmlwZSc7XG5cbmltcG9ydCB7Q29uZmlnfSBmcm9tICcuLi9jb25maWcnO1xuaW1wb3J0IHtBcGlDb250ZXh0LCBQYXltZW50QmFua0FjY291bnQsIFBheW1lbnRDYXJkVHlwZSwgUGF5bWVudENoYXJnZSwgUGF5bWVudFRyYW5zZmVyLCBVc2VyVHlwZX0gZnJvbSAnLi4vdHlwZXMnO1xuaW1wb3J0IHtsb2dFcnJvcn0gZnJvbSAnLi4vdXRpbHMvYW5hbHl0aWNzJztcbmltcG9ydCB7dXNlRGJ9IGZyb20gJy4uL3V0aWxzL2FyYW5nb2RiJztcbmltcG9ydCB7Z2V0VXNlcn0gZnJvbSAnLi91c2Vycyc7XG5cbi8qKlxuICogQ29weXJpZ2h0IChjKSAyMDE5LVByZXNlbnQsIE5pdHJvZ2VuIExhYnMsIEluYy5cbiAqIENvcHlyaWdodHMgbGljZW5zZWQgdW5kZXIgdGhlIE1JVCBMaWNlbnNlLiBTZWUgdGhlIGFjY29tcGFueWluZyBMSUNFTlNFIGZpbGUgZm9yIHRlcm1zLlxuICovXG5jb25zdCBldmVudENhdGVnb3J5OiBzdHJpbmcgPSAncGF5bWVudHMnO1xuXG5leHBvcnQgY29uc3QgYWRkQ3VzdG9tZXJBY2NvdW50ID0gKGNvbnRleHQ6IEFwaUNvbnRleHQpOiBQcm9taXNlPGJvb2xlYW4+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnYWRkQ3VzdG9tZXJBY2NvdW50JztcbiAgY29uc3Qge2RhdGFiYXNlLCB1c2VySWQ6IHNlc3Npb25JZCwgdXNlcm5hbWV9ID0gY29udGV4dDtcblxuICAvLyBTdHJpcGVcbiAgY29uc3Qgc3RyaXBlQ2xpZW50ID0gc3RyaXBlKENvbmZpZy5nZXQoJ3N0cmlwZS50b2tlbicpKTtcblxuICByZXR1cm4gc3RyaXBlQ2xpZW50LmN1c3RvbWVyc1xuICAgIC5jcmVhdGUoe1xuICAgICAgbWV0YWRhdGE6IHtcbiAgICAgICAgdXNlcklkOiBzZXNzaW9uSWQsXG4gICAgICAgIHVzZXJuYW1lXG4gICAgICB9XG4gICAgfSlcbiAgICAudGhlbigoY3VzdG9tZXIpID0+IHtcbiAgICAgIC8vIENyZWF0ZSBzZXNzaW9uXG4gICAgICBjb25zdCBub3c6IG51bWJlciA9IERhdGUubm93KCk7XG4gICAgICBjb25zdCB1cGRhdGU6IFVzZXJUeXBlID0ge1xuICAgICAgICBtb2RpZmllZDogbm93LFxuICAgICAgICBzdHJpcGVDdXN0b21lcklkOiBjdXN0b21lci5pZFxuICAgICAgfTtcblxuICAgICAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBVUERBVEUgJHtzZXNzaW9uSWR9IFdJVEggJHt1cGRhdGV9IElOIHVzZXJzIExJTUlUIDEgUkVUVVJOIE5FV2A7XG5cbiAgICAgIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgICAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAgICAgLnRoZW4oKHVwZGF0ZWRVc2VyOiBVc2VyVHlwZSA9IHt9KSA9PiAhaXNFbXB0eSh1cGRhdGVkVXNlcikpXG4gICAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgICAgYWN0aW9uLFxuICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgIGxhYmVsOiAnZGJfZXJyb3InXG4gICAgICAgIH0sIGVycm9yLCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpKTtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBhZGRQYXltZW50QWNjb3VudEJhbmsgPSAoY29udGV4dDogQXBpQ29udGV4dCwgYmFua0FjY291bnQ6IFBheW1lbnRCYW5rQWNjb3VudCk6IFByb21pc2U8Ym9vbGVhbj4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdhZGRQYXltZW50QWNjb3VudEJhbmsnO1xuICBjb25zdCB7ZGF0YWJhc2UsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG5cbiAgLy8gUGFyYW1zXG4gIGNvbnN0IHtcbiAgICBhY2NvdW50TnVtYmVyLFxuICAgIGZ1bGxOYW1lLFxuICAgIHJvdXRpbmdcbiAgfSA9IGJhbmtBY2NvdW50O1xuXG4gIGNvbnN0IGZvcm1hdEFjY291bnQ6IHN0cmluZyA9IHBhcnNlU3RyaW5nKGFjY291bnROdW1iZXIsIDMyKTtcblxuICBpZihmb3JtYXRBY2NvdW50ID09PSAnJykge1xuICAgIHRocm93IG5ldyBVc2VyRXJyb3IoJ3JlcXVpcmVkX2FjY291bnRfbnVtYmVyJyk7XG4gIH1cblxuICBjb25zdCBmb3JtYXRGdWxsTmFtZTogc3RyaW5nID0gcGFyc2VWYXJDaGFyKGZ1bGxOYW1lLCAxMjgpO1xuXG4gIGlmKGZvcm1hdEZ1bGxOYW1lID09PSAnJykge1xuICAgIHRocm93IG5ldyBVc2VyRXJyb3IoJ3JlcXVpcmVkX2Z1bGxfbmFtZScpO1xuICB9XG5cbiAgY29uc3QgZm9ybWF0Um91dGluZzogc3RyaW5nID0gcGFyc2VTdHJpbmcocm91dGluZywgMzIpO1xuXG4gIGlmKGZvcm1hdFJvdXRpbmcgPT09ICcnKSB7XG4gICAgdGhyb3cgbmV3IFVzZXJFcnJvcigncmVxdWlyZWRfcm91dGluZ19udW1iZXInKTtcbiAgfVxuXG4gIHJldHVybiBnZXRVc2VyKGNvbnRleHQsIHNlc3Npb25JZClcbiAgICAudGhlbigodXNlcjogVXNlclR5cGUpID0+IHtcbiAgICAgIC8vIFN0cmlwZVxuICAgICAgY29uc3Qgc3RyaXBlQ2xpZW50ID0gc3RyaXBlKENvbmZpZy5nZXQoJ3N0cmlwZS50b2tlbicpKTtcblxuICAgICAgcmV0dXJuIHN0cmlwZUNsaWVudC5jdXN0b21lcnNcbiAgICAgICAgLmNyZWF0ZVNvdXJjZSh1c2VyLnN0cmlwZUFjY291bnRJZCwge1xuICAgICAgICAgIHNvdXJjZToge1xuICAgICAgICAgICAgYWNjb3VudF9ob2xkZXJfbmFtZTogZm9ybWF0RnVsbE5hbWUsXG4gICAgICAgICAgICBhY2NvdW50X2hvbGRlcl90eXBlOiAnaW5kaXZpZHVhbCcsXG4gICAgICAgICAgICBhY2NvdW50X251bWJlcjogZm9ybWF0QWNjb3VudCxcbiAgICAgICAgICAgIGNvdW50cnk6ICdVUycsXG4gICAgICAgICAgICBjdXJyZW5jeTogJ1VTRCcsXG4gICAgICAgICAgICBvYmplY3Q6ICdiYW5rX2FjY291bnQnLFxuICAgICAgICAgICAgcm91dGluZ19udW1iZXI6IGZvcm1hdFJvdXRpbmdcbiAgICAgICAgICB9XG4gICAgICAgIH0pXG4gICAgICAgIC50aGVuKChhY2NvdW50KSA9PiB7XG4gICAgICAgICAgY29uc3QgdXBkYXRlOiBhbnkgPSB7XG4gICAgICAgICAgICBiYW5rQWNjb3VudDogYWNjb3VudC5sYXN0NCxcbiAgICAgICAgICAgIGJhbmtGdWxsTmFtZTogZm9ybWF0RnVsbE5hbWUsXG4gICAgICAgICAgICBiYW5rSWQ6IGFjY291bnQuaWQsXG4gICAgICAgICAgICBiYW5rUm91dGluZzogYWNjb3VudC5yb3V0aW5nX251bWJlcixcbiAgICAgICAgICAgIG1vZGlmaWVkOiBEYXRlLm5vdygpXG4gICAgICAgICAgfTtcblxuICAgICAgICAgIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgVVBEQVRFICR7c2Vzc2lvbklkfSBXSVRIICR7dXBkYXRlfSBJTiB1c2VycyBMSU1JVCAxIFJFVFVSTiBORVdgO1xuXG4gICAgICAgICAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgICAgICAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAgICAgICAgIC50aGVuKCh1cGRhdGVkVXNlcjogVXNlclR5cGUgPSB7fSkgPT4gdXBkYXRlZFVzZXIpO1xuICAgICAgICB9KVxuICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgICAgIGNvbnN0IG1zZyA9IGVycm9yLm1lc3NhZ2U7XG5cbiAgICAgICAgICBpZihtc2cgPT09ICdBIGJhbmsgYWNjb3VudCB3aXRoIHRoYXQgcm91dGluZyBudW1iZXIgYW5kIGFjY291bnQgbnVtYmVyICcgK1xuICAgICAgICAgICAgJ2FscmVhZHkgZXhpc3RzIGZvciB0aGlzIGN1c3RvbWVyLicpIHtcbiAgICAgICAgICAgIHJldHVybiBsb2dFcnJvcih7XG4gICAgICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgICAgIGxhYmVsOiAnYmFua19hY2NvdW50X2V4aXN0cydcbiAgICAgICAgICAgIH0sIGVycm9yLCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpO1xuICAgICAgICAgIH1cbiAgICAgICAgICByZXR1cm4gbG9nRXJyb3Ioe1xuICAgICAgICAgICAgYWN0aW9uLFxuICAgICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgICBsYWJlbDogJ3BheW1lbnRfZXJyb3InXG4gICAgICAgICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCk7XG4gICAgICAgIH0pO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGFkZFBheW1lbnRBY2NvdW50Q2FyZCA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBjYXJkOiBQYXltZW50Q2FyZFR5cGUsIHJlcXVlc3REYXRhKTogUHJvbWlzZTxVc2VyVHlwZT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdhZGRQYXltZW50QWNjb3VudENhcmQnO1xuICBjb25zdCB7ZGF0YWJhc2UsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG5cbiAgcmV0dXJuIGdldFVzZXIoY29udGV4dCwgc2Vzc2lvbklkKVxuICAgIC50aGVuKCh1c2VyOiBVc2VyVHlwZSkgPT4ge1xuICAgICAgLy8gVXNlclxuICAgICAgY29uc3Qge1xuICAgICAgICBfa2V5OiB1c2VyS2V5LFxuICAgICAgICBjb3VudHJ5ID0gJ1VTJyxcbiAgICAgICAgY2l0eSxcbiAgICAgICAgY3VycmVuY3kgPSAnVVNEJyxcbiAgICAgICAgZG9iLFxuICAgICAgICBlbWFpbCxcbiAgICAgICAgZmlyc3QsXG4gICAgICAgIGxhc3QsXG4gICAgICAgIHVzZXJuYW1lXG4gICAgICB9OiBVc2VyVHlwZSA9IHVzZXI7XG5cbiAgICAgIC8vIEJpcnRoZGF5XG4gICAgICBjb25zdCBkb2JEYXk6IG51bWJlciA9ICsoRGF0ZVRpbWUuZnJvbU1pbGxpcyhkb2IpLnRvRm9ybWF0KCdkJykpO1xuICAgICAgY29uc3QgZG9iTW9udGg6IG51bWJlciA9ICsoRGF0ZVRpbWUuZnJvbU1pbGxpcyhkb2IpLnRvRm9ybWF0KCdNJykpO1xuICAgICAgY29uc3QgZG9iWWVhcjogbnVtYmVyID0gKyhEYXRlVGltZS5mcm9tTWlsbGlzKGRvYikudG9Gb3JtYXQoJ3knKSk7XG5cbiAgICAgIC8vIENhcmRcbiAgICAgIGNvbnN0IHtcbiAgICAgICAgYWNjb3VudE51bWJlcixcbiAgICAgICAgYWNjZXB0ZWRUZXJtcyxcbiAgICAgICAgZXhwTW9udGgsXG4gICAgICAgIGV4cFllYXIsXG4gICAgICAgIHN0YXRlLFxuICAgICAgICBzdHJlZXQxLFxuICAgICAgICB6aXBcbiAgICAgIH06IFBheW1lbnRDYXJkVHlwZSA9IGNhcmQ7XG5cbiAgICAgIC8vIEFjY2VwdGFuY2VcbiAgICAgIGNvbnN0IGFjY2VwdGVkOiBudW1iZXIgPSBhY2NlcHRlZFRlcm1zID8gRGF0ZVRpbWUubG9jYWwoKS5taWxsaXNlY29uZCA6IDA7XG5cbiAgICAgIC8vIFN0cmlwZVxuICAgICAgY29uc3Qgc3RyaXBlQ2xpZW50ID0gc3RyaXBlKENvbmZpZy5nZXQoJ3N0cmlwZS50b2tlbicpKTtcblxuICAgICAgcmV0dXJuIHN0cmlwZUNsaWVudC5hY2NvdW50c1xuICAgICAgICAuY3JlYXRlKHtcbiAgICAgICAgICBjb3VudHJ5LFxuICAgICAgICAgIGVtYWlsLFxuICAgICAgICAgIGV4dGVybmFsX2FjY291bnQ6IHtcbiAgICAgICAgICAgIGN1cnJlbmN5LFxuICAgICAgICAgICAgZXhwX21vbnRoOiBleHBNb250aCxcbiAgICAgICAgICAgIGV4cF95ZWFyOiBleHBZZWFyLFxuICAgICAgICAgICAgbnVtYmVyOiBhY2NvdW50TnVtYmVyLFxuICAgICAgICAgICAgb2JqZWN0OiAnY2FyZCdcbiAgICAgICAgICB9LFxuICAgICAgICAgIGxlZ2FsX2VudGl0eToge1xuICAgICAgICAgICAgYWRkcmVzczoge1xuICAgICAgICAgICAgICBjaXR5LFxuICAgICAgICAgICAgICBsaW5lMTogc3RyZWV0MSxcbiAgICAgICAgICAgICAgcG9zdGFsX2NvZGU6IHppcCxcbiAgICAgICAgICAgICAgc3RhdGVcbiAgICAgICAgICAgIH0sXG4gICAgICAgICAgICBkb2I6IHtcbiAgICAgICAgICAgICAgZGF5OiBkb2JEYXksXG4gICAgICAgICAgICAgIG1vbnRoOiBkb2JNb250aCxcbiAgICAgICAgICAgICAgeWVhcjogZG9iWWVhclxuICAgICAgICAgICAgfSxcbiAgICAgICAgICAgIGZpcnN0X25hbWU6IGZpcnN0LFxuICAgICAgICAgICAgbGFzdF9uYW1lOiBsYXN0LFxuICAgICAgICAgICAgLy8gc3NuX2xhc3RfNDogc3NuLnN1YnN0cigtNCksXG4gICAgICAgICAgICAvLyBwZXJzb25hbF9pZF9udW1iZXI6IHNzbixcbiAgICAgICAgICAgIHR5cGU6ICdpbmRpdmlkdWFsJ1xuICAgICAgICAgIH0sXG4gICAgICAgICAgbWFuYWdlZDogdHJ1ZSxcbiAgICAgICAgICBtZXRhZGF0YToge1xuICAgICAgICAgICAgdXNlcklkOiB1c2VyS2V5LFxuICAgICAgICAgICAgdXNlcm5hbWVcbiAgICAgICAgICB9LFxuICAgICAgICAgIHRvc19hY2NlcHRhbmNlOiB7XG4gICAgICAgICAgICBkYXRlOiBhY2NlcHRlZCxcbiAgICAgICAgICAgIGlwOiByZXF1ZXN0RGF0YS5pcCxcbiAgICAgICAgICAgIHVzZXJfYWdlbnQ6IHJlcXVlc3REYXRhLnVzZXJBZ2VudC5zb3VyY2VcbiAgICAgICAgICB9XG4gICAgICAgIH0pXG4gICAgICAgIC50aGVuKChjdXN0b21lcikgPT4ge1xuICAgICAgICAgIC8vIENyZWF0ZSBzZXNzaW9uXG4gICAgICAgICAgY29uc3Qgbm93OiBudW1iZXIgPSBEYXRlLm5vdygpO1xuICAgICAgICAgIGNvbnN0IHVwZGF0ZTogVXNlclR5cGUgPSB7XG4gICAgICAgICAgICBtb2RpZmllZDogbm93LFxuICAgICAgICAgICAgc3RyaXBlQWNjb3VudElkOiBjdXN0b21lci5pZFxuICAgICAgICAgIH07XG5cbiAgICAgICAgICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYFVQREFURSAke3VzZXJLZXl9IFdJVEggJHt1cGRhdGV9IElOIHVzZXJzIExJTUlUIDEgUkVUVVJOIE5FV2A7XG5cbiAgICAgICAgICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAgICAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgICAgICAgICAgLnRoZW4oKHVwZGF0ZWRVc2VyOiBVc2VyVHlwZSA9IHt9KSA9PiAhaXNFbXB0eSh1cGRhdGVkVXNlcikpXG4gICAgICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgICAgICBsYWJlbDogJ2RiX2Vycm9yJ1xuICAgICAgICAgICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCkpO1xuICAgICAgICB9KTtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBhZGRDcmVkaXRDYXJkID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGNhcmQ6IFBheW1lbnRDYXJkVHlwZSk6IFByb21pc2U8UGF5bWVudENhcmRUeXBlPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2FkZENyZWRpdENhcmQnO1xuICBjb25zdCB7ZGF0YWJhc2UsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG5cbiAgY29uc3Qge1xuICAgIGFjY291bnROdW1iZXIsXG4gICAgY2l0eSxcbiAgICBjb3VudHJ5LFxuICAgIGN2YyxcbiAgICBleHBNb250aCxcbiAgICBleHBZZWFyLFxuICAgIGZ1bGxOYW1lLFxuICAgIHN0cmVldDEsXG4gICAgc3RhdGUsXG4gICAgemlwXG4gIH06IFBheW1lbnRDYXJkVHlwZSA9IGNhcmQ7XG4gIGNvbnN0IGZvcm1hdE51bWJlcjogbnVtYmVyID0gcGFyc2VOdW0oYWNjb3VudE51bWJlciwgMTYpO1xuXG4gIGlmKCFmb3JtYXROdW1iZXIpIHtcbiAgICB0aHJvdyBuZXcgVXNlckVycm9yKCdyZXF1aXJlZF9jcmVkaXRfY2FyZF9udW1iZXInKTtcbiAgfVxuXG4gIGNvbnN0IGZvcm1hdEV4cE1vbnRoOiBudW1iZXIgPSBwYXJzZU51bShleHBNb250aCwgMik7XG5cbiAgaWYoIWZvcm1hdEV4cE1vbnRoKSB7XG4gICAgdGhyb3cgbmV3IFVzZXJFcnJvcigncmVxdWlyZWRfY3JlZGl0X2NhcmRfZXhwX21vbnRoJyk7XG4gIH1cblxuICBjb25zdCBmb3JtYXRFeHBZZWFyOiBudW1iZXIgPSBwYXJzZU51bShleHBZZWFyLCAyKTtcblxuICBpZighZm9ybWF0RXhwWWVhcikge1xuICAgIHRocm93IG5ldyBVc2VyRXJyb3IoJ3JlcXVpcmVkX2NyZWRpdF9jYXJkX2V4cF95ZWFyJyk7XG4gIH1cblxuICBjb25zdCBmb3JtYXRDdmM6IG51bWJlciA9IHBhcnNlTnVtKGN2YywgMyk7XG5cbiAgaWYoIWZvcm1hdEN2Yykge1xuICAgIHRocm93IG5ldyBVc2VyRXJyb3IoJ3JlcXVpcmVkX2NyZWRpdF9jYXJkX2N2YycpO1xuICB9XG5cbiAgY29uc3QgcGF5bWVudENhcmQ6IFBheW1lbnRDYXJkVHlwZSA9IHt9O1xuICBjb25zdCBmb3JtYXRDaXR5OiBzdHJpbmcgPSBwYXJzZVZhckNoYXIoY2l0eSwgMzIpO1xuXG4gIGlmKGZvcm1hdENpdHkpIHtcbiAgICBwYXltZW50Q2FyZC5jaXR5ID0gZm9ybWF0Q2l0eTtcbiAgfVxuXG4gIGNvbnN0IGZvcm1hdENvdW50cnk6IHN0cmluZyA9IHBhcnNlVmFyQ2hhcihjb3VudHJ5LCAzMik7XG5cbiAgaWYoZm9ybWF0Q291bnRyeSkge1xuICAgIHBheW1lbnRDYXJkLmNvdW50cnkgPSBmb3JtYXRDb3VudHJ5O1xuICB9XG5cbiAgY29uc3QgZm9ybWF0RnVsbE5hbWU6IHN0cmluZyA9IHBhcnNlVmFyQ2hhcihmdWxsTmFtZSwgMzIpO1xuXG4gIGlmKGZvcm1hdEZ1bGxOYW1lKSB7XG4gICAgcGF5bWVudENhcmQuZnVsbE5hbWUgPSBmb3JtYXRGdWxsTmFtZTtcbiAgfVxuXG4gIGNvbnN0IGZvcm1hdFN0cmVldDogc3RyaW5nID0gcGFyc2VWYXJDaGFyKHN0cmVldDEsIDMyKTtcblxuICBpZihmb3JtYXRTdHJlZXQpIHtcbiAgICBwYXltZW50Q2FyZC5zdHJlZXQxID0gZm9ybWF0U3RyZWV0O1xuICB9XG5cbiAgY29uc3QgZm9ybWF0U3RhdGU6IHN0cmluZyA9IHBhcnNlVmFyQ2hhcihzdGF0ZSwgMzIpO1xuXG4gIGlmKGZvcm1hdFN0YXRlKSB7XG4gICAgcGF5bWVudENhcmQuc3RhdGUgPSBmb3JtYXRTdGF0ZTtcbiAgfVxuXG4gIGNvbnN0IGZvcm1hdFppcDogc3RyaW5nID0gcGFyc2VWYXJDaGFyKHppcCwgMzIpO1xuXG4gIGlmKGZvcm1hdFppcCkge1xuICAgIHBheW1lbnRDYXJkLnppcCA9IGZvcm1hdFppcDtcbiAgfVxuXG4gIC8vIEdldCBzdHJpcGUgaWRcbiAgcmV0dXJuIGdldFVzZXIoY29udGV4dCwgc2Vzc2lvbklkKVxuICAgIC50aGVuKCh1c2VyOiBVc2VyVHlwZSA9IHt9KSA9PiB7XG4gICAgICAvLyBTdHJpcGVcbiAgICAgIGNvbnN0IHN0cmlwZUNsaWVudCA9IHN0cmlwZShDb25maWcuZ2V0KCdzdHJpcGUudG9rZW4nKSk7XG5cbiAgICAgIHJldHVybiBzdHJpcGVDbGllbnQuY3VzdG9tZXJzXG4gICAgICAgIC5jcmVhdGVTb3VyY2UodXNlci5zdHJpcGVDdXN0b21lcklkLCB7XG4gICAgICAgICAgc291cmNlOiB7XG4gICAgICAgICAgICBhZGRyZXNzX2NpdHk6IGZvcm1hdENpdHksXG4gICAgICAgICAgICBhZGRyZXNzX2NvdW50cnk6IGZvcm1hdENvdW50cnksXG4gICAgICAgICAgICBhZGRyZXNzX2xpbmUxOiBmb3JtYXRTdHJlZXQsXG4gICAgICAgICAgICBhZGRyZXNzX3N0YXRlOiBmb3JtYXRTdGF0ZSxcbiAgICAgICAgICAgIGFkZHJlc3NfemlwOiBmb3JtYXRaaXAsXG4gICAgICAgICAgICBjdXJyZW5jeTogJ3VzZCcsXG4gICAgICAgICAgICBjdmM6IGZvcm1hdEN2YyxcbiAgICAgICAgICAgIGV4cF9tb250aDogZm9ybWF0RXhwTW9udGgsXG4gICAgICAgICAgICBleHBfeWVhcjogZm9ybWF0RXhwWWVhcixcbiAgICAgICAgICAgIG5hbWU6IGZvcm1hdEZ1bGxOYW1lLFxuICAgICAgICAgICAgbnVtYmVyOiBmb3JtYXROdW1iZXIsXG4gICAgICAgICAgICBvYmplY3Q6ICdjYXJkJ1xuICAgICAgICAgIH1cbiAgICAgICAgfSlcbiAgICAgICAgLnRoZW4oKHN0cmlwZU9iaikgPT4ge1xuICAgICAgICAgIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcbiAgICAgICAgICBjb25zdCBpbnNlcnQgPSB7XG4gICAgICAgICAgICAuLi5wYXltZW50Q2FyZCxcbiAgICAgICAgICAgIF9rZXk6IGNyZWF0ZUhhc2goYHVzZXItcGF5bWVudC0ke3Nlc3Npb25JZH1gKSxcbiAgICAgICAgICAgIGFjY291bnROdW1iZXI6IHN0cmlwZU9iai5sYXN0NCxcbiAgICAgICAgICAgIGFkZGVkOiBub3csXG4gICAgICAgICAgICBicmFuZDogc3RyaXBlT2JqLmJyYW5kLFxuICAgICAgICAgICAgY3ZjOiBzdHJpcGVPYmouY3ZjX2NoZWNrLFxuICAgICAgICAgICAgZXhwTW9udGgsXG4gICAgICAgICAgICBleHBZZWFyLFxuICAgICAgICAgICAgbW9kaWZpZWQ6IG5vdyxcbiAgICAgICAgICAgIHVzZXJJZDogc2Vzc2lvbklkXG4gICAgICAgICAgfTtcbiAgICAgICAgICBjb25zdCBpbnNlcnRBcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYElOU0VSVCAke2luc2VydH0gSU4gY3JlZGl0Q2FyZHMgUkVUVVJOIE5FV2A7XG5cbiAgICAgICAgICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGluc2VydEFxbFFyeSlcbiAgICAgICAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgICAgICAgICAgLnRoZW4oKG5ld0NhcmQ6IFBheW1lbnRDYXJkVHlwZSA9IHt9KSA9PiB7XG4gICAgICAgICAgICAgIGlmKCFpc0VtcHR5KG5ld0NhcmQpKSB7XG4gICAgICAgICAgICAgICAgLy8gQWRkIGxpbmtlZCBlZGdlXG4gICAgICAgICAgICAgICAgY29uc3Qge19pZDogY2FyZElkLCBfa2V5OiBjYXJkS2V5fSA9IGNhcmQ7XG4gICAgICAgICAgICAgICAgY29uc3QgZWRnZUNvbGxlY3Rpb24gPSB1c2VEYihkYXRhYmFzZSkuZWRnZUNvbGxlY3Rpb24oJ2hhc1BheW1lbnQnKTtcbiAgICAgICAgICAgICAgICBjb25zdCBlZGdlSWQgPSBjcmVhdGVIYXNoKGBwYXltZW50LSR7Y2FyZEtleX1gKTtcbiAgICAgICAgICAgICAgICBjb25zdCBlZGdlOiBhbnkgPSB7XG4gICAgICAgICAgICAgICAgICBfa2V5OiBlZGdlSWRcbiAgICAgICAgICAgICAgICB9O1xuXG4gICAgICAgICAgICAgICAgcmV0dXJuIGVkZ2VDb2xsZWN0aW9uLnNhdmUoZWRnZSwgYHVzZXJzLyR7c2Vzc2lvbklkfWAsIGNhcmRJZCkudGhlbigoKSA9PiBjYXJkKTtcbiAgICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICAgIHJldHVybiBuZXdDYXJkO1xuICAgICAgICAgICAgfSk7XG4gICAgICAgIH0pXG4gICAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgICAgYWN0aW9uLFxuICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgIGxhYmVsOiAncGF5bWVudF9lcnJvcidcbiAgICAgICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCkpO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IHVwZGF0ZUNyZWRpdENhcmQgPSAoY29udGV4dDogQXBpQ29udGV4dCwgY2FyZDogUGF5bWVudENhcmRUeXBlKTogUHJvbWlzZTxQYXltZW50Q2FyZFR5cGU+ID0+IHtcbiAgY29uc3Qge2RhdGFiYXNlLCB1c2VySWQ6IHNlc3Npb25JZH0gPSBjb250ZXh0O1xuXG4gIGNvbnN0IHtcbiAgICBjaXR5LFxuICAgIGNvdW50cnksXG4gICAgZXhwTW9udGgsXG4gICAgZXhwWWVhcixcbiAgICBmdWxsTmFtZSxcbiAgICBpZCxcbiAgICBzdHJlZXQxLFxuICAgIHN0YXRlLFxuICAgIHppcFxuICB9OiBQYXltZW50Q2FyZFR5cGUgPSBjYXJkO1xuXG4gIGNvbnN0IGZvcm1hdElkOiBzdHJpbmcgPSBwYXJzZUlkKGlkKTtcblxuICBpZihmb3JtYXRJZCkge1xuICAgIHRocm93IG5ldyBVc2VyRXJyb3IoJ3JlcXVpcmVkX2NyZWRpdF9jYXJkX2lkJyk7XG4gIH1cblxuICBjb25zdCBwYXltZW50Q2FyZDogUGF5bWVudENhcmRUeXBlID0ge307XG4gIGNvbnN0IGZvcm1hdEV4cE1vbnRoOiBudW1iZXIgPSBwYXJzZU51bShleHBNb250aCwgMik7XG4gIGNvbnN0IGZvcm1hdEV4cFllYXI6IG51bWJlciA9IHBhcnNlTnVtKGV4cFllYXIsIDIpO1xuICBjb25zdCBmb3JtYXRDaXR5OiBzdHJpbmcgPSBwYXJzZVZhckNoYXIoY2l0eSwgMzIpO1xuICBjb25zdCBmb3JtYXRDb3VudHJ5OiBzdHJpbmcgPSBwYXJzZVZhckNoYXIoY291bnRyeSwgMik7XG4gIGNvbnN0IGZvcm1hdEZ1bGxOYW1lOiBzdHJpbmcgPSBwYXJzZVZhckNoYXIoZnVsbE5hbWUsIDMyKTtcbiAgY29uc3QgZm9ybWF0U3RyZWV0MTogc3RyaW5nID0gcGFyc2VTdHJpbmcoc3RyZWV0MSwgMzIpO1xuICBjb25zdCBmb3JtYXRTdGF0ZTogc3RyaW5nID0gcGFyc2VWYXJDaGFyKHN0YXRlLCAyKTtcbiAgY29uc3QgZm9ybWF0WmlwOiBzdHJpbmcgPSBwYXJzZVZhckNoYXIoemlwLCAxMCk7XG5cbiAgaWYoZm9ybWF0RXhwTW9udGgpIHtcbiAgICBwYXltZW50Q2FyZC5leHBNb250aCA9IGZvcm1hdEV4cE1vbnRoO1xuICB9XG5cbiAgaWYoZm9ybWF0RXhwWWVhcikge1xuICAgIHBheW1lbnRDYXJkLmV4cFllYXIgPSBmb3JtYXRFeHBZZWFyO1xuICB9XG5cbiAgaWYoZm9ybWF0Q2l0eSkge1xuICAgIHBheW1lbnRDYXJkLmNpdHkgPSBmb3JtYXRDaXR5O1xuICB9XG5cbiAgaWYoZm9ybWF0Q291bnRyeSkge1xuICAgIHBheW1lbnRDYXJkLmNvdW50cnkgPSBmb3JtYXRDb3VudHJ5O1xuICB9XG5cbiAgaWYoZm9ybWF0RnVsbE5hbWUpIHtcbiAgICBwYXltZW50Q2FyZC5mdWxsTmFtZSA9IGZvcm1hdEZ1bGxOYW1lO1xuICB9XG5cbiAgaWYoZm9ybWF0U3RyZWV0MSkge1xuICAgIHBheW1lbnRDYXJkLnN0cmVldDEgPSBmb3JtYXRTdHJlZXQxO1xuICB9XG5cbiAgaWYoZm9ybWF0U3RhdGUpIHtcbiAgICBwYXltZW50Q2FyZC5zdGF0ZSA9IGZvcm1hdFN0YXRlO1xuICB9XG5cbiAgaWYoZm9ybWF0WmlwKSB7XG4gICAgcGF5bWVudENhcmQuemlwID0gZm9ybWF0WmlwO1xuICB9XG5cbiAgY29uc3QgdXBkYXRlOiBhbnkgPSBwYXltZW50Q2FyZDtcbiAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBcbiAgICAgIExFVCB1cGRhdGVkQ2FyZCA9IEZJUlNUKFxuICAgICAgICBGT1IgYyBJTiBjcmVkaXRDYXJkc1xuICAgICAgICBGSUxURVIgYy5fa2V5ID09ICR7Zm9ybWF0SWR9ICYmIGMudXNlcklkID09ICR7c2Vzc2lvbklkfVxuICAgICAgICBVUERBVEUgYyBXSVRIICR7dXBkYXRlfSBJTiBjcmVkaXRDYXJkc1xuICAgICAgICBMSU1JVCAxXG4gICAgICAgIFJFVFVSTiBORVdcbiAgICAgIClcbiAgICAgIExFVCB1c2VyID0gRklSU1QoXG4gICAgICAgIEZPUiB1IElOIHVzZXJzXG4gICAgICAgIEZJTFRFUiB1Ll9rZXkgPT0gJHtzZXNzaW9uSWR9XG4gICAgICAgIExJTUlUIDFcbiAgICAgICAgUkVUVVJOIHVcbiAgICAgIClcbiAgICAgIFJFVFVSTiB7dXNlcjogdXNlciwgY2FyZDogdXBkYXRlZENhcmR9YDtcblxuICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAudGhlbigocmVzdWx0cyA9IHtjYXJkOiB7fSwgdXNlcjoge319KSA9PiB7XG4gICAgICBjb25zdCB1cGRhdGVkQ2FyZDogUGF5bWVudENhcmRUeXBlID0gcmVzdWx0cy5jYXJkO1xuICAgICAgY29uc3Qge3VzZXJ9ID0gcmVzdWx0cztcblxuICAgICAgaWYoaXNFbXB0eSh1cGRhdGVkQ2FyZCkpIHtcbiAgICAgICAgdGhyb3cgbmV3IFVzZXJFcnJvcignbm90X2ZvdW5kJyk7XG4gICAgICB9XG5cbiAgICAgIC8vIFN0cmlwZVxuICAgICAgY29uc3Qgc3RyaXBlQ2xpZW50ID0gc3RyaXBlKENvbmZpZy5nZXQoJ3N0cmlwZS50b2tlbicpKTtcblxuICAgICAgcmV0dXJuIHN0cmlwZUNsaWVudC5jdXN0b21lcnNcbiAgICAgICAgLnVwZGF0ZUNhcmQodXNlci5zdHJpcGVDdXN0b21lcklkLCBjYXJkLnN0cmlwZUlkLCB7XG4gICAgICAgICAgYWRkcmVzc19jaXR5OiBmb3JtYXRDaXR5LFxuICAgICAgICAgIGFkZHJlc3NfY291bnRyeTogZm9ybWF0Q291bnRyeSxcbiAgICAgICAgICBhZGRyZXNzX2xpbmUxOiBmb3JtYXRTdHJlZXQxLFxuICAgICAgICAgIGFkZHJlc3Nfc3RhdGU6IGZvcm1hdFN0YXRlLFxuICAgICAgICAgIGFkZHJlc3NfemlwOiBmb3JtYXRaaXAsXG4gICAgICAgICAgZXhwX21vbnRoOiBmb3JtYXRFeHBNb250aCxcbiAgICAgICAgICBleHBfeWVhcjogZm9ybWF0RXhwWWVhcixcbiAgICAgICAgICBuYW1lOiBmb3JtYXRGdWxsTmFtZVxuICAgICAgICB9KVxuICAgICAgICAudGhlbigoKSA9PiBjYXJkKVxuICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgICAgIGNvbnNvbGUubG9nKCdwYXltZW50czo6dXBkYXRlQ2FyZDo6ZXJyb3InLCBlcnJvcik7XG4gICAgICAgICAgdGhyb3cgbmV3IFVzZXJFcnJvcigncGF5bWVudF9lcnJvcicpO1xuICAgICAgICB9KTtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRDcmVkaXRDYXJkcyA9IChjb250ZXh0OiBBcGlDb250ZXh0KTogUHJvbWlzZTxQYXltZW50Q2FyZFR5cGVbXT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRDcmVkaXRDYXJkcyc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgdXNlcklkOiBzZXNzaW9uSWR9ID0gY29udGV4dDtcbiAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBGT1IgYyBJTiBjcmVkaXRDYXJkc1xuICAgICAgICBGSUxURVIgYy51c2VySWQgPT0gJHtzZXNzaW9uSWR9XG4gICAgICAgIFJFVFVSTiBjYDtcblxuICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLmFsbCgpKVxuICAgIC50aGVuKChsaXN0OiBQYXltZW50Q2FyZFR5cGVbXSA9IFtdKSA9PiBsaXN0KVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICBhY3Rpb24sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIGxhYmVsOiAnZGJfZXJyb3InXG4gICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCkpO1xufTtcblxuZXhwb3J0IGNvbnN0IGRlbGV0ZUNyZWRpdENhcmQgPSAoY29udGV4dDogQXBpQ29udGV4dCwgY2FyZElkOiBzdHJpbmcpOiBQcm9taXNlPGJvb2xlYW4+ID0+IHtcbiAgY29uc3Qge2RhdGFiYXNlLCB1c2VySWQ6IHNlc3Npb25JZH0gPSBjb250ZXh0O1xuICBjb25zdCBmb3JtYXRDYXJkSWQ6IHN0cmluZyA9IHBhcnNlSWQoY2FyZElkKTtcbiAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBcbiAgICAgIExFVCBjYXJkID0gRklSU1QoXG4gICAgICAgIEZPUiBjIElOIGNyZWRpdENhcmRzXG4gICAgICAgIEZJTFRFUiBjLl9rZXkgPT0gJHtmb3JtYXRDYXJkSWR9ICYmIGMudXNlcklkID09ICR7c2Vzc2lvbklkfVxuICAgICAgICBMSU1JVCAxXG4gICAgICAgIFJFTU9WRSBjIElOIGNyZWRpdENhcmRzXG4gICAgICAgIFJFVFVSTiBPTERcbiAgICAgIClcbiAgICAgIExFVCB1c2VyID0gRklSU1QoXG4gICAgICAgIEZPUiB1IElOIHVzZXJzXG4gICAgICAgIEZJTFRFUiB1Ll9rZXkgPT0gJHtzZXNzaW9uSWR9XG4gICAgICAgIExJTUlUIDFcbiAgICAgICAgUkVUVVJOIHVcbiAgICAgIClcbiAgICAgIFJFVFVSTiB7dXNlcjogdXNlciwgY2FyZDogY2FyZH1gO1xuXG4gIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgIC50aGVuKChyZXN1bHQgPSB7Y2FyZDoge30sIHVzZXI6IHt9fSkgPT4ge1xuICAgICAgaWYoaXNFbXB0eShyZXN1bHQpKSB7XG4gICAgICAgIHJldHVybiBmYWxzZTtcbiAgICAgIH1cblxuICAgICAgY29uc3Qge2NhcmQsIHVzZXJ9ID0gcmVzdWx0O1xuICAgICAgY29uc3Qge19rZXk6IGNhcmRLZXl9ID0gY2FyZDtcblxuICAgICAgLy8gUmVtb3ZlIGxpbmtlZCBlZGdlc1xuICAgICAgY29uc3QgZWRnZUNvbGxlY3Rpb24gPSB1c2VEYihkYXRhYmFzZSkuZWRnZUNvbGxlY3Rpb24oJ2hhc1BheW1lbnQnKTtcblxuICAgICAgcmV0dXJuIGVkZ2VDb2xsZWN0aW9uLm91dEVkZ2VzKGNhcmRLZXkpXG4gICAgICAgIC50aGVuKChlZGdlcykgPT4ge1xuICAgICAgICAgIGlmKGVkZ2VzLmxlbmd0aCkge1xuICAgICAgICAgICAgcmV0dXJuIFByb21pc2UuYWxsKFxuICAgICAgICAgICAgICBlZGdlcy5tYXAoKGVkZ2UpID0+IHtcbiAgICAgICAgICAgICAgICBjb25zdCB7X2tleTogZWRnZUtleX0gPSBlZGdlO1xuICAgICAgICAgICAgICAgIGNvbnN0IHJlbW92ZUFxbFFyeTogQXFsUXVlcnkgPSBhcWxgUkVNT1ZFIHtfa2V5OiR7ZWRnZUtleX19IElOIGhhc1BheW1lbnRgO1xuICAgICAgICAgICAgICAgIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkocmVtb3ZlQXFsUXJ5KTtcbiAgICAgICAgICAgICAgfSkpXG4gICAgICAgICAgICAgIC50aGVuKCgpID0+IHtcbiAgICAgICAgICAgICAgICAvLyBTdHJpcGVcbiAgICAgICAgICAgICAgICBjb25zdCBzdHJpcGVDbGllbnQgPSBzdHJpcGUoQ29uZmlnLmdldCgnc3RyaXBlLnRva2VuJykpO1xuXG4gICAgICAgICAgICAgICAgcmV0dXJuIHN0cmlwZUNsaWVudC5jdXN0b21lcnNcbiAgICAgICAgICAgICAgICAgIC5kZWxldGVDYXJkKHVzZXIuc3RyaXBlQ3VzdG9tZXJJZCwgY2FyZC5zdHJpcGVJZClcbiAgICAgICAgICAgICAgICAgIC50aGVuKCgpID0+IHRydWUpXG4gICAgICAgICAgICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgICAgICAgICAgICAgICBjb25zb2xlLmxvZygncGF5bWVudHM6OmRlbGV0ZUNhcmQ6OmVycm9yJywgZXJyb3IpO1xuICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgVXNlckVycm9yKCdwYXltZW50X2Vycm9yJyk7XG4gICAgICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgICAgfSk7XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgcmV0dXJuIGZhbHNlO1xuICAgICAgICB9KTtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBkZWxldGVQYXltZW50QWNjb3VudEJhbmsgPSAoY29udGV4dDogQXBpQ29udGV4dCwgYmFua0lkOiBzdHJpbmcpOiBQcm9taXNlPGJvb2xlYW4+ID0+IHtcbiAgY29uc3Qge2RhdGFiYXNlLCB1c2VySWQ6IHNlc3Npb25JZH0gPSBjb250ZXh0O1xuXG4gIC8vIENsZWFuIGRiXG4gIGNvbnN0IHVwZGF0ZTogVXNlclR5cGUgPSB7XG4gICAgYmFua0FjY291bnQ6ICcnLFxuICAgIGJhbmtGdWxsTmFtZTogJycsXG4gICAgYmFua0lkOiAnJyxcbiAgICBiYW5rUm91dGluZzogJycsXG4gICAgbW9kaWZpZWQ6IERhdGUubm93KClcbiAgfTtcbiAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBVUERBVEUgJHtzZXNzaW9uSWR9IFdJVEggJHt1cGRhdGV9IElOIHVzZXJzIExJTUlUIDEgUkVUVVJOIE5FV2A7XG5cbiAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgLnRoZW4oKHVzZXI6IFVzZXJUeXBlID0ge30pID0+IHtcbiAgICAgIC8vIFN0cmlwZVxuICAgICAgY29uc3Qgc3RyaXBlQ2xpZW50ID0gc3RyaXBlKENvbmZpZy5nZXQoJ3N0cmlwZS50b2tlbicpKTtcblxuICAgICAgcmV0dXJuIHN0cmlwZUNsaWVudC5jdXN0b21lcnNcbiAgICAgICAgLmRlbGV0ZVNvdXJjZSh1c2VyLnN0cmlwZUFjY291bnRJZCwgYmFua0lkKVxuICAgICAgICAudGhlbigocmVzcG9uc2UgPSB7ZGVsZXRlZDogZmFsc2V9KSA9PiByZXNwb25zZS5kZWxldGVkKVxuICAgICAgICAuY2F0Y2goKCkgPT4gUHJvbWlzZS5yZXNvbHZlKGZhbHNlKSk7XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgY3JlYXRlUGF5bWVudFRyYW5zZmVyID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIHRyYW5zZmVyOiBQYXltZW50VHJhbnNmZXIpOiBQcm9taXNlPFBheW1lbnRUcmFuc2Zlcj4gPT4ge1xuICBjb25zdCB7ZGF0YWJhc2UsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG4gIGNvbnN0IHthbW91bnQsIGN1cnJlbmN5fSA9IHRyYW5zZmVyO1xuICBjb25zdCBmb3JtYXRBbW91bnQ6IG51bWJlciA9IHBhcnNlTnVtKGFtb3VudCk7XG4gIGNvbnN0IGZvcm1hdEN1cnJlbmN5OiBzdHJpbmcgPSBwYXJzZUNoYXIoY3VycmVuY3ksIDMsICdVU0QnKS50b1VwcGVyQ2FzZSgpO1xuXG4gIHJldHVybiBnZXRVc2VyKGNvbnRleHQsIHNlc3Npb25JZClcbiAgICAudGhlbigodXNlcjogVXNlclR5cGUpID0+IHtcbiAgICAgIC8vIFN0cmlwZVxuICAgICAgY29uc3Qgc3RyaXBlQ2xpZW50ID0gc3RyaXBlKENvbmZpZy5nZXQoJ3N0cmlwZS50b2tlbicpKTtcblxuICAgICAgcmV0dXJuIHN0cmlwZUNsaWVudC50cmFuc2ZlcnNcbiAgICAgICAgLmNyZWF0ZSh7XG4gICAgICAgICAgYW1vdW50OiBmb3JtYXRBbW91bnQsXG4gICAgICAgICAgY3VycmVuY3k6IGZvcm1hdEN1cnJlbmN5LFxuICAgICAgICAgIGRlc3RpbmF0aW9uOiB1c2VyLnN0cmlwZUFjY291bnRJZFxuICAgICAgICB9KVxuICAgICAgICAudGhlbigoc3RyaXBlVHJhbnNmZXIpID0+IHtcbiAgICAgICAgICBjb25zb2xlLmxvZyhzdHJpcGVUcmFuc2Zlcik7XG4gICAgICAgICAgY29uc3Qgbm93OiBudW1iZXIgPSBEYXRlLm5vdygpO1xuICAgICAgICAgIGNvbnN0IGluc2VydDogUGF5bWVudFRyYW5zZmVyID0ge1xuICAgICAgICAgICAgYWRkZWQ6IG5vdyxcbiAgICAgICAgICAgIGFtb3VudDogZm9ybWF0QW1vdW50LFxuICAgICAgICAgICAgY3VycmVuY3k6IGZvcm1hdEN1cnJlbmN5LFxuICAgICAgICAgICAgbW9kaWZpZWQ6IG5vdyxcbiAgICAgICAgICAgIHVzZXJJZDogc2Vzc2lvbklkXG4gICAgICAgICAgfTtcbiAgICAgICAgICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYElOU0VSVCAke2luc2VydH0gSU4gdHJhbnNmZXJzIFJFVFVSTiBORVdgO1xuXG4gICAgICAgICAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgICAgICAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAgICAgICAgIC50aGVuKChuZXdUcmFuc2ZlcjogUGF5bWVudFRyYW5zZmVyKSA9PiBuZXdUcmFuc2Zlcik7XG4gICAgICAgIH0pO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGNyZWF0ZVBheW1lbnRIb2xkID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIHBheW1lbnQ6IFBheW1lbnRDaGFyZ2UpOiBQcm9taXNlPFBheW1lbnRDaGFyZ2U+ID0+IHtcbiAgY29uc3Qge2RhdGFiYXNlLCB1c2VySWQ6IHNlc3Npb25JZH0gPSBjb250ZXh0O1xuICBjb25zdCB7YW1vdW50LCBjYXB0dXJlLCBjYXJkSWQsIGN1cnJlbmN5LCBkZXNjcmlwdGlvbn0gPSBwYXltZW50O1xuICBjb25zdCBmb3JtYXRDdXJyZW5jeSA9IHBhcnNlQ2hhcihjdXJyZW5jeSwgMywgJ1VTRCcpLnRvVXBwZXJDYXNlKCk7XG5cbiAgY29uc3Qgc3RyaXBlQ2xpZW50ID0gc3RyaXBlKENvbmZpZy5nZXQoJ3N0cmlwZS50b2tlbicpKTtcblxuICByZXR1cm4gc3RyaXBlQ2xpZW50LmNoYXJnZXNcbiAgICAuY3JlYXRlKHtcbiAgICAgIGFtb3VudCxcbiAgICAgIGNhcHR1cmUsXG4gICAgICBjdXJyZW5jeTogZm9ybWF0Q3VycmVuY3ksXG4gICAgICBkZXNjcmlwdGlvbixcbiAgICAgIHNvdXJjZTogY2FyZElkXG4gICAgfSlcbiAgICAudGhlbigoc3RyaXBlQ2hhcmdlKSA9PiB7XG4gICAgICBjb25zdCBub3c6IG51bWJlciA9IERhdGUubm93KCk7XG4gICAgICBjb25zdCBpbnNlcnQ6IFBheW1lbnRDaGFyZ2UgPSB7XG4gICAgICAgIGFkZGVkOiBub3csXG4gICAgICAgIGFtb3VudCxcbiAgICAgICAgY2FwdHVyZSxcbiAgICAgICAgY2FyZElkLFxuICAgICAgICBjaGFyZ2VGYWlsQ29kZTogc3RyaXBlQ2hhcmdlLmZhaWx1cmVfY29kZSxcbiAgICAgICAgY2hhcmdlRmFpbE1zZzogc3RyaXBlQ2hhcmdlLmZhaWx1cmVfbWVzc2FnZSxcbiAgICAgICAgY2hhcmdlSWQ6IHN0cmlwZUNoYXJnZS5pZCxcbiAgICAgICAgY2hhcmdlU3RhdHVzOiBzdHJpcGVDaGFyZ2Uuc3RhdHVzLFxuICAgICAgICBjdXJyZW5jeTogZm9ybWF0Q3VycmVuY3ksXG4gICAgICAgIGRlc2NyaXB0aW9uLFxuICAgICAgICBtb2RpZmllZDogbm93LFxuICAgICAgICB1c2VySWQ6IHNlc3Npb25JZFxuICAgICAgfTtcbiAgICAgIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgSU5TRVJUICR7aW5zZXJ0fSBJTiBwYXltZW50cyBSRVRVUk4gTkVXYDtcblxuICAgICAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgICAgICAudGhlbigobmV3UGF5bWVudDogUGF5bWVudENoYXJnZSkgPT4gbmV3UGF5bWVudCk7XG4gICAgfSlcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgY29uc29sZS5sb2coJ3BheW1lbnRzOjpjcmVhdGVIb2xkOjplcnJvcicsIGVycm9yKTtcbiAgICAgIHRocm93IG5ldyBVc2VyRXJyb3IoJ3BheW1lbnRfZXJyb3InKTtcbiAgICB9KTtcbn07XG4iXX0=
|