@nlabs/reaktor 0.1.11 → 0.1.12

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (48) hide show
  1. package/package.json +14 -14
  2. package/lib/config.js +0 -139
  3. package/lib/data/conversations.js +0 -228
  4. package/lib/data/dynamodb.js +0 -172
  5. package/lib/data/email.js +0 -194
  6. package/lib/data/files.js +0 -463
  7. package/lib/data/groups.js +0 -401
  8. package/lib/data/images.js +0 -841
  9. package/lib/data/index.js +0 -234
  10. package/lib/data/ios.js +0 -327
  11. package/lib/data/locations.js +0 -148
  12. package/lib/data/messages.js +0 -281
  13. package/lib/data/notifications.js +0 -59
  14. package/lib/data/payments.js +0 -798
  15. package/lib/data/posts.js +0 -637
  16. package/lib/data/reactions.js +0 -243
  17. package/lib/data/s3.js +0 -133
  18. package/lib/data/search.js +0 -111
  19. package/lib/data/sms.js +0 -79
  20. package/lib/data/subscription.js +0 -311
  21. package/lib/data/tags.js +0 -343
  22. package/lib/data/users.js +0 -415
  23. package/lib/index.js +0 -42
  24. package/lib/types/apps.js +0 -2
  25. package/lib/types/arangodb.js +0 -2
  26. package/lib/types/auth.js +0 -2
  27. package/lib/types/conversations.js +0 -2
  28. package/lib/types/email.js +0 -2
  29. package/lib/types/files.js +0 -2
  30. package/lib/types/google.js +0 -2
  31. package/lib/types/groups.js +0 -2
  32. package/lib/types/images.js +0 -2
  33. package/lib/types/index.js +0 -210
  34. package/lib/types/locations.js +0 -2
  35. package/lib/types/messages.js +0 -2
  36. package/lib/types/notifications.js +0 -2
  37. package/lib/types/payments.js +0 -2
  38. package/lib/types/posts.js +0 -2
  39. package/lib/types/reactions.js +0 -2
  40. package/lib/types/tags.js +0 -2
  41. package/lib/types/users.js +0 -2
  42. package/lib/utils/analytics.js +0 -59
  43. package/lib/utils/arangodb.js +0 -131
  44. package/lib/utils/auth.js +0 -104
  45. package/lib/utils/graphql.js +0 -19
  46. package/lib/utils/index.js +0 -78
  47. package/lib/utils/objects.js +0 -54
  48. package/lib/utils/redis.js +0 -28
@@ -1,798 +0,0 @@
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,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kYXRhL3BheW1lbnRzLnRzIl0sIm5hbWVzIjpbImV2ZW50Q2F0ZWdvcnkiLCJhZGRDdXN0b21lckFjY291bnQiLCJjb250ZXh0IiwiYWN0aW9uIiwiZGF0YWJhc2UiLCJzZXNzaW9uSWQiLCJ1c2VySWQiLCJ1c2VybmFtZSIsInN0cmlwZUNsaWVudCIsInN0cmlwZSIsIkNvbmZpZyIsImdldCIsImN1c3RvbWVycyIsImNyZWF0ZSIsIm1ldGFkYXRhIiwidGhlbiIsImN1c3RvbWVyIiwibm93IiwiRGF0ZSIsInVwZGF0ZSIsIm1vZGlmaWVkIiwic3RyaXBlQ3VzdG9tZXJJZCIsImlkIiwiYXFsUXJ5IiwiYXFsIiwicXVlcnkiLCJjdXJzb3IiLCJuZXh0IiwidXBkYXRlZFVzZXIiLCJjYXRjaCIsImVycm9yIiwiY2F0ZWdvcnkiLCJsYWJlbCIsImFkZFBheW1lbnRBY2NvdW50QmFuayIsImJhbmtBY2NvdW50IiwiYWNjb3VudE51bWJlciIsImZ1bGxOYW1lIiwicm91dGluZyIsImZvcm1hdEFjY291bnQiLCJVc2VyRXJyb3IiLCJmb3JtYXRGdWxsTmFtZSIsImZvcm1hdFJvdXRpbmciLCJ1c2VyIiwiY3JlYXRlU291cmNlIiwic3RyaXBlQWNjb3VudElkIiwic291cmNlIiwiYWNjb3VudF9ob2xkZXJfbmFtZSIsImFjY291bnRfaG9sZGVyX3R5cGUiLCJhY2NvdW50X251bWJlciIsImNvdW50cnkiLCJjdXJyZW5jeSIsIm9iamVjdCIsInJvdXRpbmdfbnVtYmVyIiwiYWNjb3VudCIsImxhc3Q0IiwiYmFua0Z1bGxOYW1lIiwiYmFua0lkIiwiYmFua1JvdXRpbmciLCJtc2ciLCJtZXNzYWdlIiwiYWRkUGF5bWVudEFjY291bnRDYXJkIiwiY2FyZCIsInJlcXVlc3REYXRhIiwidXNlcktleSIsIl9rZXkiLCJjaXR5IiwiZG9iIiwiZW1haWwiLCJmaXJzdCIsImxhc3QiLCJkb2JEYXkiLCJEYXRlVGltZSIsImZyb21NaWxsaXMiLCJ0b0Zvcm1hdCIsImRvYk1vbnRoIiwiZG9iWWVhciIsImFjY2VwdGVkVGVybXMiLCJleHBNb250aCIsImV4cFllYXIiLCJzdGF0ZSIsInN0cmVldDEiLCJ6aXAiLCJhY2NlcHRlZCIsImxvY2FsIiwibWlsbGlzZWNvbmQiLCJhY2NvdW50cyIsImV4dGVybmFsX2FjY291bnQiLCJleHBfbW9udGgiLCJleHBfeWVhciIsIm51bWJlciIsImxlZ2FsX2VudGl0eSIsImFkZHJlc3MiLCJsaW5lMSIsInBvc3RhbF9jb2RlIiwiZGF5IiwibW9udGgiLCJ5ZWFyIiwiZmlyc3RfbmFtZSIsImxhc3RfbmFtZSIsInR5cGUiLCJtYW5hZ2VkIiwidG9zX2FjY2VwdGFuY2UiLCJkYXRlIiwiaXAiLCJ1c2VyX2FnZW50IiwidXNlckFnZW50IiwiYWRkQ3JlZGl0Q2FyZCIsImN2YyIsImZvcm1hdE51bWJlciIsImZvcm1hdEV4cE1vbnRoIiwiZm9ybWF0RXhwWWVhciIsImZvcm1hdEN2YyIsInBheW1lbnRDYXJkIiwiZm9ybWF0Q2l0eSIsImZvcm1hdENvdW50cnkiLCJmb3JtYXRTdHJlZXQiLCJmb3JtYXRTdGF0ZSIsImZvcm1hdFppcCIsImFkZHJlc3NfY2l0eSIsImFkZHJlc3NfY291bnRyeSIsImFkZHJlc3NfbGluZTEiLCJhZGRyZXNzX3N0YXRlIiwiYWRkcmVzc196aXAiLCJuYW1lIiwic3RyaXBlT2JqIiwiaW5zZXJ0IiwiYWRkZWQiLCJicmFuZCIsImN2Y19jaGVjayIsImluc2VydEFxbFFyeSIsIm5ld0NhcmQiLCJjYXJkSWQiLCJfaWQiLCJjYXJkS2V5IiwiZWRnZUNvbGxlY3Rpb24iLCJlZGdlSWQiLCJlZGdlIiwic2F2ZSIsInVwZGF0ZUNyZWRpdENhcmQiLCJmb3JtYXRJZCIsImZvcm1hdFN0cmVldDEiLCJyZXN1bHRzIiwidXBkYXRlZENhcmQiLCJ1cGRhdGVDYXJkIiwic3RyaXBlSWQiLCJjb25zb2xlIiwibG9nIiwiZ2V0Q3JlZGl0Q2FyZHMiLCJhbGwiLCJsaXN0IiwiZGVsZXRlQ3JlZGl0Q2FyZCIsImZvcm1hdENhcmRJZCIsInJlc3VsdCIsIm91dEVkZ2VzIiwiZWRnZXMiLCJsZW5ndGgiLCJQcm9taXNlIiwibWFwIiwiZWRnZUtleSIsInJlbW92ZUFxbFFyeSIsImRlbGV0ZUNhcmQiLCJkZWxldGVQYXltZW50QWNjb3VudEJhbmsiLCJkZWxldGVTb3VyY2UiLCJyZXNwb25zZSIsImRlbGV0ZWQiLCJyZXNvbHZlIiwiY3JlYXRlUGF5bWVudFRyYW5zZmVyIiwidHJhbnNmZXIiLCJhbW91bnQiLCJmb3JtYXRBbW91bnQiLCJmb3JtYXRDdXJyZW5jeSIsInRvVXBwZXJDYXNlIiwidHJhbnNmZXJzIiwiZGVzdGluYXRpb24iLCJzdHJpcGVUcmFuc2ZlciIsIm5ld1RyYW5zZmVyIiwiY3JlYXRlUGF5bWVudEhvbGQiLCJwYXltZW50IiwiY2FwdHVyZSIsImRlc2NyaXB0aW9uIiwiY2hhcmdlcyIsInN0cmlwZUNoYXJnZSIsImNoYXJnZUZhaWxDb2RlIiwiZmFpbHVyZV9jb2RlIiwiY2hhcmdlRmFpbE1zZyIsImZhaWx1cmVfbWVzc2FnZSIsImNoYXJnZUlkIiwiY2hhcmdlU3RhdHVzIiwic3RhdHVzIiwibmV3UGF5bWVudCJdLCJtYXBwaW5ncyI6Ijs7Ozs7OztBQUFBOztBQUNBOztBQUdBOztBQUNBOztBQUNBOztBQUNBOztBQUVBOztBQUVBOztBQUNBOztBQUNBOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUVBOzs7O0FBSUEsSUFBTUEsYUFBcUIsR0FBRyxVQUE5Qjs7QUFFTyxJQUFNQyxrQkFBa0IsR0FBRyxTQUFyQkEsa0JBQXFCLENBQUNDLE9BQUQsRUFBMkM7QUFDM0UsTUFBTUMsTUFBYyxHQUFHLG9CQUF2QjtBQUQyRSxNQUVwRUMsUUFGb0UsR0FFM0JGLE9BRjJCLENBRXBFRSxRQUZvRTtBQUFBLE1BRWxEQyxTQUZrRCxHQUUz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TSxFQUFFRDtBQVpvQixLQUE5QjtBQWNBLFFBQU1rQixNQUFnQixPQUFHQyxhQUFILHVCQUFnQnFHLE1BQWhCLENBQXRCO0FBRUEsV0FBTyxxQkFBTXpILFFBQU4sRUFBZ0JxQixLQUFoQixDQUFzQkYsTUFBdEIsRUFDSlIsSUFESSxDQUNDLFVBQUNXLE1BQUQ7QUFBQSxhQUF5QkEsTUFBTSxDQUFDQyxJQUFQLEVBQXpCO0FBQUEsS0FERCxFQUVKWixJQUZJLENBRUMsVUFBQzhLLFVBQUQ7QUFBQSxhQUErQkEsVUFBL0I7QUFBQSxLQUZELENBQVA7QUFHRCxHQTdCSSxFQThCSmhLLEtBOUJJLENBOEJFLFVBQUNDLEtBQUQsRUFBa0I7QUFDdkJtSCxJQUFBQSxPQUFPLENBQUNDLEdBQVIsQ0FBWSw2QkFBWixFQUEyQ3BILEtBQTNDO0FBQ0EsVUFBTSxJQUFJUyx3QkFBSixDQUFjLGVBQWQsQ0FBTjtBQUNELEdBakNJLENBQVA7QUFrQ0QsQ0F6Q00iLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge2NyZWF0ZUhhc2gsIHBhcnNlQ2hhciwgcGFyc2VJZCwgcGFyc2VOdW0sIHBhcnNlU3RyaW5nLCBwYXJzZVZhckNoYXJ9IGZyb20gJ0BubGFicy91dGlscyc7XG5pbXBvcnQge2FxbH0gZnJvbSAnYXJhbmdvanMnO1xuaW1wb3J0IHtBcWxRdWVyeX0gZnJvbSAnYXJhbmdvanMvbGliL2Nqcy9hcWwtcXVlcnknO1xuaW1wb3J0IHtBcnJheUN1cnNvcn0gZnJvbSAnYXJhbmdvanMvbGliL2Nqcy9jdXJzb3InO1xuaW1wb3J0IHtVc2VyRXJyb3J9IGZyb20gJ2dyYXBocWwtZXJyb3JzJztcbmltcG9ydCBpc0VtcHR5IGZyb20gJ2xvZGFzaC9pc0VtcHR5JztcbmltcG9ydCB7RGF0ZVRpbWV9IGZyb20gJ2x1eG9uJztcbmltcG9ydCAqIGFzIHN0cmlwZSBmcm9tICdzdHJpcGUnO1xuXG5pbXBvcnQge0NvbmZpZ30gZnJvbSAnLi4vY29uZmlnJztcbmltcG9ydCB7QXBpQ29udGV4dCwgUGF5bWVudEJhbmtBY2NvdW50LCBQYXltZW50Q2FyZFR5cGUsIFBheW1lbnRDaGFyZ2UsIFBheW1lbnRUcmFuc2ZlciwgVXNlclR5cGV9IGZyb20gJy4uL3R5cGVzJztcbmltcG9ydCB7bG9nRXJyb3J9IGZyb20gJy4uL3V0aWxzL2FuYWx5dGljcyc7XG5pbXBvcnQge3VzZURifSBmcm9tICcuLi91dGlscy9hcmFuZ29kYic7XG5pbXBvcnQge2dldFVzZXJ9IGZyb20gJy4vdXNlcnMnO1xuXG4vKipcbiAqIENvcHlyaWdodCAoYykgMjAxOS1QcmVzZW50LCBOaXRyb2dlbiBMYWJzLCBJbmMuXG4gKiBDb3B5cmlnaHRzIGxpY2Vuc2VkIHVuZGVyIHRoZSBNSVQgTGljZW5zZS4gU2VlIHRoZSBhY2NvbXBhbnlpbmcgTElDRU5TRSBmaWxlIGZvciB0ZXJtcy5cbiAqL1xuY29uc3QgZXZlbnRDYXRlZ29yeTogc3RyaW5nID0gJ3BheW1lbnRzJztcblxuZXhwb3J0IGNvbnN0IGFkZEN1c3RvbWVyQWNjb3VudCA9IChjb250ZXh0OiBBcGlDb250ZXh0KTogUHJvbWlzZTxib29sZWFuPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2FkZEN1c3RvbWVyQWNjb3VudCc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgdXNlcklkOiBzZXNzaW9uSWQsIHVzZXJuYW1lfSA9IGNvbnRleHQ7XG5cbiAgLy8gU3RyaXBlXG4gIGNvbnN0IHN0cmlwZUNsaWVudCA9IHN0cmlwZShDb25maWcuZ2V0KCdzdHJpcGUudG9rZW4nKSk7XG5cbiAgcmV0dXJuIHN0cmlwZUNsaWVudC5jdXN0b21lcnNcbiAgICAuY3JlYXRlKHtcbiAgICAgIG1ldGFkYXRhOiB7XG4gICAgICAgIHVzZXJJZDogc2Vzc2lvbklkLFxuICAgICAgICB1c2VybmFtZVxuICAgICAgfVxuICAgIH0pXG4gICAgLnRoZW4oKGN1c3RvbWVyKSA9PiB7XG4gICAgICAvLyBDcmVhdGUgc2Vzc2lvblxuICAgICAgY29uc3Qgbm93OiBudW1iZXIgPSBEYXRlLm5vdygpO1xuICAgICAgY29uc3QgdXBkYXRlOiBVc2VyVHlwZSA9IHtcbiAgICAgICAgbW9kaWZpZWQ6IG5vdyxcbiAgICAgICAgc3RyaXBlQ3VzdG9tZXJJZDogY3VzdG9tZXIuaWRcbiAgICAgIH07XG5cbiAgICAgIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgVVBEQVRFICR7c2Vzc2lvbklkfSBXSVRIICR7dXBkYXRlfSBJTiB1c2VycyBMSU1JVCAxIFJFVFVSTiBORVdgO1xuXG4gICAgICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAgICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgICAgIC50aGVuKCh1cGRhdGVkVXNlcjogVXNlclR5cGUgPSB7fSkgPT4gIWlzRW1wdHkodXBkYXRlZFVzZXIpKVxuICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgICBsYWJlbDogJ2RiX2Vycm9yJ1xuICAgICAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKSk7XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgYWRkUGF5bWVudEFjY291bnRCYW5rID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGJhbmtBY2NvdW50OiBQYXltZW50QmFua0FjY291bnQpOiBQcm9taXNlPGJvb2xlYW4+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnYWRkUGF5bWVudEFjY291bnRCYW5rJztcbiAgY29uc3Qge2RhdGFiYXNlLCB1c2VySWQ6IHNlc3Npb25JZH0gPSBjb250ZXh0O1xuXG4gIC8vIFBhcmFtc1xuICBjb25zdCB7XG4gICAgYWNjb3VudE51bWJlcixcbiAgICBmdWxsTmFtZSxcbiAgICByb3V0aW5nXG4gIH0gPSBiYW5rQWNjb3VudDtcblxuICBjb25zdCBmb3JtYXRBY2NvdW50OiBzdHJpbmcgPSBwYXJzZVN0cmluZyhhY2NvdW50TnVtYmVyLCAzMik7XG5cbiAgaWYoZm9ybWF0QWNjb3VudCA9PT0gJycpIHtcbiAgICB0aHJvdyBuZXcgVXNlckVycm9yKCdyZXF1aXJlZF9hY2NvdW50X251bWJlcicpO1xuICB9XG5cbiAgY29uc3QgZm9ybWF0RnVsbE5hbWU6IHN0cmluZyA9IHBhcnNlVmFyQ2hhcihmdWxsTmFtZSwgMTI4KTtcblxuICBpZihmb3JtYXRGdWxsTmFtZSA9PT0gJycpIHtcbiAgICB0aHJvdyBuZXcgVXNlckVycm9yKCdyZXF1aXJlZF9mdWxsX25hbWUnKTtcbiAgfVxuXG4gIGNvbnN0IGZvcm1hdFJvdXRpbmc6IHN0cmluZyA9IHBhcnNlU3RyaW5nKHJvdXRpbmcsIDMyKTtcblxuICBpZihmb3JtYXRSb3V0aW5nID09PSAnJykge1xuICAgIHRocm93IG5ldyBVc2VyRXJyb3IoJ3JlcXVpcmVkX3JvdXRpbmdfbnVtYmVyJyk7XG4gIH1cblxuICByZXR1cm4gZ2V0VXNlcihjb250ZXh0LCBzZXNzaW9uSWQpXG4gICAgLnRoZW4oKHVzZXI6IFVzZXJUeXBlKSA9PiB7XG4gICAgICAvLyBTdHJpcGVcbiAgICAgIGNvbnN0IHN0cmlwZUNsaWVudCA9IHN0cmlwZShDb25maWcuZ2V0KCdzdHJpcGUudG9rZW4nKSk7XG5cbiAgICAgIHJldHVybiBzdHJpcGVDbGllbnQuY3VzdG9tZXJzXG4gICAgICAgIC5jcmVhdGVTb3VyY2UodXNlci5zdHJpcGVBY2NvdW50SWQsIHtcbiAgICAgICAgICBzb3VyY2U6IHtcbiAgICAgICAgICAgIGFjY291bnRfaG9sZGVyX25hbWU6IGZvcm1hdEZ1bGxOYW1lLFxuICAgICAgICAgICAgYWNjb3VudF9ob2xkZXJfdHlwZTogJ2luZGl2aWR1YWwnLFxuICAgICAgICAgICAgYWNjb3VudF9udW1iZXI6IGZvcm1hdEFjY291bnQsXG4gICAgICAgICAgICBjb3VudHJ5OiAnVVMnLFxuICAgICAgICAgICAgY3VycmVuY3k6ICdVU0QnLFxuICAgICAgICAgICAgb2JqZWN0OiAnYmFua19hY2NvdW50JyxcbiAgICAgICAgICAgIHJvdXRpbmdfbnVtYmVyOiBmb3JtYXRSb3V0aW5nXG4gICAgICAgICAgfVxuICAgICAgICB9KVxuICAgICAgICAudGhlbigoYWNjb3VudCkgPT4ge1xuICAgICAgICAgIGNvbnN0IHVwZGF0ZTogYW55ID0ge1xuICAgICAgICAgICAgYmFua0FjY291bnQ6IGFjY291bnQubGFzdDQsXG4gICAgICAgICAgICBiYW5rRnVsbE5hbWU6IGZvcm1hdEZ1bGxOYW1lLFxuICAgICAgICAgICAgYmFua0lkOiBhY2NvdW50LmlkLFxuICAgICAgICAgICAgYmFua1JvdXRpbmc6IGFjY291bnQucm91dGluZ19udW1iZXIsXG4gICAgICAgICAgICBtb2RpZmllZDogRGF0ZS5ub3coKVxuICAgICAgICAgIH07XG5cbiAgICAgICAgICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYFVQREFURSAke3Nlc3Npb25JZH0gV0lUSCAke3VwZGF0ZX0gSU4gdXNlcnMgTElNSVQgMSBSRVRVUk4gTkVXYDtcblxuICAgICAgICAgIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgICAgICAgICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgICAgICAgICAudGhlbigodXBkYXRlZFVzZXI6IFVzZXJUeXBlID0ge30pID0+IHVwZGF0ZWRVc2VyKTtcbiAgICAgICAgfSlcbiAgICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IHtcbiAgICAgICAgICBjb25zdCBtc2cgPSBlcnJvci5tZXNzYWdlO1xuXG4gICAgICAgICAgaWYobXNnID09PSAnQSBiYW5rIGFjY291bnQgd2l0aCB0aGF0IHJvdXRpbmcgbnVtYmVyIGFuZCBhY2NvdW50IG51bWJlciAnICtcbiAgICAgICAgICAgICdhbHJlYWR5IGV4aXN0cyBmb3IgdGhpcyBjdXN0b21lci4nKSB7XG4gICAgICAgICAgICByZXR1cm4gbG9nRXJyb3Ioe1xuICAgICAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgICAgICBsYWJlbDogJ2JhbmtfYWNjb3VudF9leGlzdHMnXG4gICAgICAgICAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKTtcbiAgICAgICAgICB9XG4gICAgICAgICAgcmV0dXJuIGxvZ0Vycm9yKHtcbiAgICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgICAgbGFiZWw6ICdwYXltZW50X2Vycm9yJ1xuICAgICAgICAgIH0sIGVycm9yLCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpO1xuICAgICAgICB9KTtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBhZGRQYXltZW50QWNjb3VudENhcmQgPSAoY29udGV4dDogQXBpQ29udGV4dCwgY2FyZDogUGF5bWVudENhcmRUeXBlLCByZXF1ZXN0RGF0YSk6IFByb21pc2U8VXNlclR5cGU+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnYWRkUGF5bWVudEFjY291bnRDYXJkJztcbiAgY29uc3Qge2RhdGFiYXNlLCB1c2VySWQ6IHNlc3Npb25JZH0gPSBjb250ZXh0O1xuXG4gIHJldHVybiBnZXRVc2VyKGNvbnRleHQsIHNlc3Npb25JZClcbiAgICAudGhlbigodXNlcjogVXNlclR5cGUpID0+IHtcbiAgICAgIC8vIFVzZXJcbiAgICAgIGNvbnN0IHtcbiAgICAgICAgX2tleTogdXNlcktleSxcbiAgICAgICAgY291bnRyeSA9ICdVUycsXG4gICAgICAgIGNpdHksXG4gICAgICAgIGN1cnJlbmN5ID0gJ1VTRCcsXG4gICAgICAgIGRvYixcbiAgICAgICAgZW1haWwsXG4gICAgICAgIGZpcnN0LFxuICAgICAgICBsYXN0LFxuICAgICAgICB1c2VybmFtZVxuICAgICAgfTogVXNlclR5cGUgPSB1c2VyO1xuXG4gICAgICAvLyBCaXJ0aGRheVxuICAgICAgY29uc3QgZG9iRGF5OiBudW1iZXIgPSArKERhdGVUaW1lLmZyb21NaWxsaXMoZG9iKS50b0Zvcm1hdCgnZCcpKTtcbiAgICAgIGNvbnN0IGRvYk1vbnRoOiBudW1iZXIgPSArKERhdGVUaW1lLmZyb21NaWxsaXMoZG9iKS50b0Zvcm1hdCgnTScpKTtcbiAgICAgIGNvbnN0IGRvYlllYXI6IG51bWJlciA9ICsoRGF0ZVRpbWUuZnJvbU1pbGxpcyhkb2IpLnRvRm9ybWF0KCd5JykpO1xuXG4gICAgICAvLyBDYXJkXG4gICAgICBjb25zdCB7XG4gICAgICAgIGFjY291bnROdW1iZXIsXG4gICAgICAgIGFjY2VwdGVkVGVybXMsXG4gICAgICAgIGV4cE1vbnRoLFxuICAgICAgICBleHBZZWFyLFxuICAgICAgICBzdGF0ZSxcbiAgICAgICAgc3RyZWV0MSxcbiAgICAgICAgemlwXG4gICAgICB9OiBQYXltZW50Q2FyZFR5cGUgPSBjYXJkO1xuXG4gICAgICAvLyBBY2NlcHRhbmNlXG4gICAgICBjb25zdCBhY2NlcHRlZDogbnVtYmVyID0gYWNjZXB0ZWRUZXJtcyA/IERhdGVUaW1lLmxvY2FsKCkubWlsbGlzZWNvbmQgOiAwO1xuXG4gICAgICAvLyBTdHJpcGVcbiAgICAgIGNvbnN0IHN0cmlwZUNsaWVudCA9IHN0cmlwZShDb25maWcuZ2V0KCdzdHJpcGUudG9rZW4nKSk7XG5cbiAgICAgIHJldHVybiBzdHJpcGVDbGllbnQuYWNjb3VudHNcbiAgICAgICAgLmNyZWF0ZSh7XG4gICAgICAgICAgY291bnRyeSxcbiAgICAgICAgICBlbWFpbCxcbiAgICAgICAgICBleHRlcm5hbF9hY2NvdW50OiB7XG4gICAgICAgICAgICBjdXJyZW5jeSxcbiAgICAgICAgICAgIGV4cF9tb250aDogZXhwTW9udGgsXG4gICAgICAgICAgICBleHBfeWVhcjogZXhwWWVhcixcbiAgICAgICAgICAgIG51bWJlcjogYWNjb3VudE51bWJlcixcbiAgICAgICAgICAgIG9iamVjdDogJ2NhcmQnXG4gICAgICAgICAgfSxcbiAgICAgICAgICBsZWdhbF9lbnRpdHk6IHtcbiAgICAgICAgICAgIGFkZHJlc3M6IHtcbiAgICAgICAgICAgICAgY2l0eSxcbiAgICAgICAgICAgICAgbGluZTE6IHN0cmVldDEsXG4gICAgICAgICAgICAgIHBvc3RhbF9jb2RlOiB6aXAsXG4gICAgICAgICAgICAgIHN0YXRlXG4gICAgICAgICAgICB9LFxuICAgICAgICAgICAgZG9iOiB7XG4gICAgICAgICAgICAgIGRheTogZG9iRGF5LFxuICAgICAgICAgICAgICBtb250aDogZG9iTW9udGgsXG4gICAgICAgICAgICAgIHllYXI6IGRvYlllYXJcbiAgICAgICAgICAgIH0sXG4gICAgICAgICAgICBmaXJzdF9uYW1lOiBmaXJzdCxcbiAgICAgICAgICAgIGxhc3RfbmFtZTogbGFzdCxcbiAgICAgICAgICAgIC8vIHNzbl9sYXN0XzQ6IHNzbi5zdWJzdHIoLTQpLFxuICAgICAgICAgICAgLy8gcGVyc29uYWxfaWRfbnVtYmVyOiBzc24sXG4gICAgICAgICAgICB0eXBlOiAnaW5kaXZpZHVhbCdcbiAgICAgICAgICB9LFxuICAgICAgICAgIG1hbmFnZWQ6IHRydWUsXG4gICAgICAgICAgbWV0YWRhdGE6IHtcbiAgICAgICAgICAgIHVzZXJJZDogdXNlcktleSxcbiAgICAgICAgICAgIHVzZXJuYW1lXG4gICAgICAgICAgfSxcbiAgICAgICAgICB0b3NfYWNjZXB0YW5jZToge1xuICAgICAgICAgICAgZGF0ZTogYWNjZXB0ZWQsXG4gICAgICAgICAgICBpcDogcmVxdWVzdERhdGEuaXAsXG4gICAgICAgICAgICB1c2VyX2FnZW50OiByZXF1ZXN0RGF0YS51c2VyQWdlbnQuc291cmNlXG4gICAgICAgICAgfVxuICAgICAgICB9KVxuICAgICAgICAudGhlbigoY3VzdG9tZXIpID0+IHtcbiAgICAgICAgICAvLyBDcmVhdGUgc2Vzc2lvblxuICAgICAgICAgIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcbiAgICAgICAgICBjb25zdCB1cGRhdGU6IFVzZXJUeXBlID0ge1xuICAgICAgICAgICAgbW9kaWZpZWQ6IG5vdyxcbiAgICAgICAgICAgIHN0cmlwZUFjY291bnRJZDogY3VzdG9tZXIuaWRcbiAgICAgICAgICB9O1xuXG4gICAgICAgICAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBVUERBVEUgJHt1c2VyS2V5fSBXSVRIICR7dXBkYXRlfSBJTiB1c2VycyBMSU1JVCAxIFJFVFVSTiBORVdgO1xuXG4gICAgICAgICAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgICAgICAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAgICAgICAgIC50aGVuKCh1cGRhdGVkVXNlcjogVXNlclR5cGUgPSB7fSkgPT4gIWlzRW1wdHkodXBkYXRlZFVzZXIpKVxuICAgICAgICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgICAgICAgYWN0aW9uLFxuICAgICAgICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgICAgICAgbGFiZWw6ICdkYl9lcnJvcidcbiAgICAgICAgICAgIH0sIGVycm9yLCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpKTtcbiAgICAgICAgfSk7XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgYWRkQ3JlZGl0Q2FyZCA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBjYXJkOiBQYXltZW50Q2FyZFR5cGUpOiBQcm9taXNlPFBheW1lbnRDYXJkVHlwZT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdhZGRDcmVkaXRDYXJkJztcbiAgY29uc3Qge2RhdGFiYXNlLCB1c2VySWQ6IHNlc3Npb25JZH0gPSBjb250ZXh0O1xuXG4gIGNvbnN0IHtcbiAgICBhY2NvdW50TnVtYmVyLFxuICAgIGNpdHksXG4gICAgY291bnRyeSxcbiAgICBjdmMsXG4gICAgZXhwTW9udGgsXG4gICAgZXhwWWVhcixcbiAgICBmdWxsTmFtZSxcbiAgICBzdHJlZXQxLFxuICAgIHN0YXRlLFxuICAgIHppcFxuICB9OiBQYXltZW50Q2FyZFR5cGUgPSBjYXJkO1xuICBjb25zdCBmb3JtYXROdW1iZXI6IG51bWJlciA9IHBhcnNlTnVtKGFjY291bnROdW1iZXIsIDE2KTtcblxuICBpZighZm9ybWF0TnVtYmVyKSB7XG4gICAgdGhyb3cgbmV3IFVzZXJFcnJvcigncmVxdWlyZWRfY3JlZGl0X2NhcmRfbnVtYmVyJyk7XG4gIH1cblxuICBjb25zdCBmb3JtYXRFeHBNb250aDogbnVtYmVyID0gcGFyc2VOdW0oZXhwTW9udGgsIDIpO1xuXG4gIGlmKCFmb3JtYXRFeHBNb250aCkge1xuICAgIHRocm93IG5ldyBVc2VyRXJyb3IoJ3JlcXVpcmVkX2NyZWRpdF9jYXJkX2V4cF9tb250aCcpO1xuICB9XG5cbiAgY29uc3QgZm9ybWF0RXhwWWVhcjogbnVtYmVyID0gcGFyc2VOdW0oZXhwWWVhciwgMik7XG5cbiAgaWYoIWZvcm1hdEV4cFllYXIpIHtcbiAgICB0aHJvdyBuZXcgVXNlckVycm9yKCdyZXF1aXJlZF9jcmVkaXRfY2FyZF9leHBfeWVhcicpO1xuICB9XG5cbiAgY29uc3QgZm9ybWF0Q3ZjOiBudW1iZXIgPSBwYXJzZU51bShjdmMsIDMpO1xuXG4gIGlmKCFmb3JtYXRDdmMpIHtcbiAgICB0aHJvdyBuZXcgVXNlckVycm9yKCdyZXF1aXJlZF9jcmVkaXRfY2FyZF9jdmMnKTtcbiAgfVxuXG4gIGNvbnN0IHBheW1lbnRDYXJkOiBQYXltZW50Q2FyZFR5cGUgPSB7fTtcbiAgY29uc3QgZm9ybWF0Q2l0eTogc3RyaW5nID0gcGFyc2VWYXJDaGFyKGNpdHksIDMyKTtcblxuICBpZihmb3JtYXRDaXR5KSB7XG4gICAgcGF5bWVudENhcmQuY2l0eSA9IGZvcm1hdENpdHk7XG4gIH1cblxuICBjb25zdCBmb3JtYXRDb3VudHJ5OiBzdHJpbmcgPSBwYXJzZVZhckNoYXIoY291bnRyeSwgMzIpO1xuXG4gIGlmKGZvcm1hdENvdW50cnkpIHtcbiAgICBwYXltZW50Q2FyZC5jb3VudHJ5ID0gZm9ybWF0Q291bnRyeTtcbiAgfVxuXG4gIGNvbnN0IGZvcm1hdEZ1bGxOYW1lOiBzdHJpbmcgPSBwYXJzZVZhckNoYXIoZnVsbE5hbWUsIDMyKTtcblxuICBpZihmb3JtYXRGdWxsTmFtZSkge1xuICAgIHBheW1lbnRDYXJkLmZ1bGxOYW1lID0gZm9ybWF0RnVsbE5hbWU7XG4gIH1cblxuICBjb25zdCBmb3JtYXRTdHJlZXQ6IHN0cmluZyA9IHBhcnNlVmFyQ2hhcihzdHJlZXQxLCAzMik7XG5cbiAgaWYoZm9ybWF0U3RyZWV0KSB7XG4gICAgcGF5bWVudENhcmQuc3RyZWV0MSA9IGZvcm1hdFN0cmVldDtcbiAgfVxuXG4gIGNvbnN0IGZvcm1hdFN0YXRlOiBzdHJpbmcgPSBwYXJzZVZhckNoYXIoc3RhdGUsIDMyKTtcblxuICBpZihmb3JtYXRTdGF0ZSkge1xuICAgIHBheW1lbnRDYXJkLnN0YXRlID0gZm9ybWF0U3RhdGU7XG4gIH1cblxuICBjb25zdCBmb3JtYXRaaXA6IHN0cmluZyA9IHBhcnNlVmFyQ2hhcih6aXAsIDMyKTtcblxuICBpZihmb3JtYXRaaXApIHtcbiAgICBwYXltZW50Q2FyZC56aXAgPSBmb3JtYXRaaXA7XG4gIH1cblxuICAvLyBHZXQgc3RyaXBlIGlkXG4gIHJldHVybiBnZXRVc2VyKGNvbnRleHQsIHNlc3Npb25JZClcbiAgICAudGhlbigodXNlcjogVXNlclR5cGUgPSB7fSkgPT4ge1xuICAgICAgLy8gU3RyaXBlXG4gICAgICBjb25zdCBzdHJpcGVDbGllbnQgPSBzdHJpcGUoQ29uZmlnLmdldCgnc3RyaXBlLnRva2VuJykpO1xuXG4gICAgICByZXR1cm4gc3RyaXBlQ2xpZW50LmN1c3RvbWVyc1xuICAgICAgICAuY3JlYXRlU291cmNlKHVzZXIuc3RyaXBlQ3VzdG9tZXJJZCwge1xuICAgICAgICAgIHNvdXJjZToge1xuICAgICAgICAgICAgYWRkcmVzc19jaXR5OiBmb3JtYXRDaXR5LFxuICAgICAgICAgICAgYWRkcmVzc19jb3VudHJ5OiBmb3JtYXRDb3VudHJ5LFxuICAgICAgICAgICAgYWRkcmVzc19saW5lMTogZm9ybWF0U3RyZWV0LFxuICAgICAgICAgICAgYWRkcmVzc19zdGF0ZTogZm9ybWF0U3RhdGUsXG4gICAgICAgICAgICBhZGRyZXNzX3ppcDogZm9ybWF0WmlwLFxuICAgICAgICAgICAgY3VycmVuY3k6ICd1c2QnLFxuICAgICAgICAgICAgY3ZjOiBmb3JtYXRDdmMsXG4gICAgICAgICAgICBleHBfbW9udGg6IGZvcm1hdEV4cE1vbnRoLFxuICAgICAgICAgICAgZXhwX3llYXI6IGZvcm1hdEV4cFllYXIsXG4gICAgICAgICAgICBuYW1lOiBmb3JtYXRGdWxsTmFtZSxcbiAgICAgICAgICAgIG51bWJlcjogZm9ybWF0TnVtYmVyLFxuICAgICAgICAgICAgb2JqZWN0OiAnY2FyZCdcbiAgICAgICAgICB9XG4gICAgICAgIH0pXG4gICAgICAgIC50aGVuKChzdHJpcGVPYmopID0+IHtcbiAgICAgICAgICBjb25zdCBub3c6IG51bWJlciA9IERhdGUubm93KCk7XG4gICAgICAgICAgY29uc3QgaW5zZXJ0ID0ge1xuICAgICAgICAgICAgLi4ucGF5bWVudENhcmQsXG4gICAgICAgICAgICBfa2V5OiBjcmVhdGVIYXNoKGB1c2VyLXBheW1lbnQtJHtzZXNzaW9uSWR9YCksXG4gICAgICAgICAgICBhY2NvdW50TnVtYmVyOiBzdHJpcGVPYmoubGFzdDQsXG4gICAgICAgICAgICBhZGRlZDogbm93LFxuICAgICAgICAgICAgYnJhbmQ6IHN0cmlwZU9iai5icmFuZCxcbiAgICAgICAgICAgIGN2Yzogc3RyaXBlT2JqLmN2Y19jaGVjayxcbiAgICAgICAgICAgIGV4cE1vbnRoLFxuICAgICAgICAgICAgZXhwWWVhcixcbiAgICAgICAgICAgIG1vZGlmaWVkOiBub3csXG4gICAgICAgICAgICB1c2VySWQ6IHNlc3Npb25JZFxuICAgICAgICAgIH07XG4gICAgICAgICAgY29uc3QgaW5zZXJ0QXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBJTlNFUlQgJHtpbnNlcnR9IElOIGNyZWRpdENhcmRzIFJFVFVSTiBORVdgO1xuXG4gICAgICAgICAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShpbnNlcnRBcWxRcnkpXG4gICAgICAgICAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAgICAgICAgIC50aGVuKChuZXdDYXJkOiBQYXltZW50Q2FyZFR5cGUgPSB7fSkgPT4ge1xuICAgICAgICAgICAgICBpZighaXNFbXB0eShuZXdDYXJkKSkge1xuICAgICAgICAgICAgICAgIC8vIEFkZCBsaW5rZWQgZWRnZVxuICAgICAgICAgICAgICAgIGNvbnN0IHtfaWQ6IGNhcmRJZCwgX2tleTogY2FyZEtleX0gPSBjYXJkO1xuICAgICAgICAgICAgICAgIGNvbnN0IGVkZ2VDb2xsZWN0aW9uID0gdXNlRGIoZGF0YWJhc2UpLmVkZ2VDb2xsZWN0aW9uKCdoYXNQYXltZW50Jyk7XG4gICAgICAgICAgICAgICAgY29uc3QgZWRnZUlkID0gY3JlYXRlSGFzaChgcGF5bWVudC0ke2NhcmRLZXl9YCk7XG4gICAgICAgICAgICAgICAgY29uc3QgZWRnZTogYW55ID0ge1xuICAgICAgICAgICAgICAgICAgX2tleTogZWRnZUlkXG4gICAgICAgICAgICAgICAgfTtcblxuICAgICAgICAgICAgICAgIHJldHVybiBlZGdlQ29sbGVjdGlvbi5zYXZlKGVkZ2UsIGB1c2Vycy8ke3Nlc3Npb25JZH1gLCBjYXJkSWQpLnRoZW4oKCkgPT4gY2FyZCk7XG4gICAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgICByZXR1cm4gbmV3Q2FyZDtcbiAgICAgICAgICAgIH0pO1xuICAgICAgICB9KVxuICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgICBsYWJlbDogJ3BheW1lbnRfZXJyb3InXG4gICAgICAgIH0sIGVycm9yLCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpKTtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCB1cGRhdGVDcmVkaXRDYXJkID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGNhcmQ6IFBheW1lbnRDYXJkVHlwZSk6IFByb21pc2U8UGF5bWVudENhcmRUeXBlPiA9PiB7XG4gIGNvbnN0IHtkYXRhYmFzZSwgdXNlcklkOiBzZXNzaW9uSWR9ID0gY29udGV4dDtcblxuICBjb25zdCB7XG4gICAgY2l0eSxcbiAgICBjb3VudHJ5LFxuICAgIGV4cE1vbnRoLFxuICAgIGV4cFllYXIsXG4gICAgZnVsbE5hbWUsXG4gICAgaWQsXG4gICAgc3RyZWV0MSxcbiAgICBzdGF0ZSxcbiAgICB6aXBcbiAgfTogUGF5bWVudENhcmRUeXBlID0gY2FyZDtcblxuICBjb25zdCBmb3JtYXRJZDogc3RyaW5nID0gcGFyc2VJZChpZCk7XG5cbiAgaWYoZm9ybWF0SWQpIHtcbiAgICB0aHJvdyBuZXcgVXNlckVycm9yKCdyZXF1aXJlZF9jcmVkaXRfY2FyZF9pZCcpO1xuICB9XG5cbiAgY29uc3QgcGF5bWVudENhcmQ6IFBheW1lbnRDYXJkVHlwZSA9IHt9O1xuICBjb25zdCBmb3JtYXRFeHBNb250aDogbnVtYmVyID0gcGFyc2VOdW0oZXhwTW9udGgsIDIpO1xuICBjb25zdCBmb3JtYXRFeHBZZWFyOiBudW1iZXIgPSBwYXJzZU51bShleHBZZWFyLCAyKTtcbiAgY29uc3QgZm9ybWF0Q2l0eTogc3RyaW5nID0gcGFyc2VWYXJDaGFyKGNpdHksIDMyKTtcbiAgY29uc3QgZm9ybWF0Q291bnRyeTogc3RyaW5nID0gcGFyc2VWYXJDaGFyKGNvdW50cnksIDIpO1xuICBjb25zdCBmb3JtYXRGdWxsTmFtZTogc3RyaW5nID0gcGFyc2VWYXJDaGFyKGZ1bGxOYW1lLCAzMik7XG4gIGNvbnN0IGZvcm1hdFN0cmVldDE6IHN0cmluZyA9IHBhcnNlU3RyaW5nKHN0cmVldDEsIDMyKTtcbiAgY29uc3QgZm9ybWF0U3RhdGU6IHN0cmluZyA9IHBhcnNlVmFyQ2hhcihzdGF0ZSwgMik7XG4gIGNvbnN0IGZvcm1hdFppcDogc3RyaW5nID0gcGFyc2VWYXJDaGFyKHppcCwgMTApO1xuXG4gIGlmKGZvcm1hdEV4cE1vbnRoKSB7XG4gICAgcGF5bWVudENhcmQuZXhwTW9udGggPSBmb3JtYXRFeHBNb250aDtcbiAgfVxuXG4gIGlmKGZvcm1hdEV4cFllYXIpIHtcbiAgICBwYXltZW50Q2FyZC5leHBZZWFyID0gZm9ybWF0RXhwWWVhcjtcbiAgfVxuXG4gIGlmKGZvcm1hdENpdHkpIHtcbiAgICBwYXltZW50Q2FyZC5jaXR5ID0gZm9ybWF0Q2l0eTtcbiAgfVxuXG4gIGlmKGZvcm1hdENvdW50cnkpIHtcbiAgICBwYXltZW50Q2FyZC5jb3VudHJ5ID0gZm9ybWF0Q291bnRyeTtcbiAgfVxuXG4gIGlmKGZvcm1hdEZ1bGxOYW1lKSB7XG4gICAgcGF5bWVudENhcmQuZnVsbE5hbWUgPSBmb3JtYXRGdWxsTmFtZTtcbiAgfVxuXG4gIGlmKGZvcm1hdFN0cmVldDEpIHtcbiAgICBwYXltZW50Q2FyZC5zdHJlZXQxID0gZm9ybWF0U3RyZWV0MTtcbiAgfVxuXG4gIGlmKGZvcm1hdFN0YXRlKSB7XG4gICAgcGF5bWVudENhcmQuc3RhdGUgPSBmb3JtYXRTdGF0ZTtcbiAgfVxuXG4gIGlmKGZvcm1hdFppcCkge1xuICAgIHBheW1lbnRDYXJkLnppcCA9IGZvcm1hdFppcDtcbiAgfVxuXG4gIGNvbnN0IHVwZGF0ZTogYW55ID0gcGF5bWVudENhcmQ7XG4gIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgXG4gICAgICBMRVQgdXBkYXRlZENhcmQgPSBGSVJTVChcbiAgICAgICAgRk9SIGMgSU4gY3JlZGl0Q2FyZHNcbiAgICAgICAgRklMVEVSIGMuX2tleSA9PSAke2Zvcm1hdElkfSAmJiBjLnVzZXJJZCA9PSAke3Nlc3Npb25JZH1cbiAgICAgICAgVVBEQVRFIGMgV0lUSCAke3VwZGF0ZX0gSU4gY3JlZGl0Q2FyZHNcbiAgICAgICAgTElNSVQgMVxuICAgICAgICBSRVRVUk4gTkVXXG4gICAgICApXG4gICAgICBMRVQgdXNlciA9IEZJUlNUKFxuICAgICAgICBGT1IgdSBJTiB1c2Vyc1xuICAgICAgICBGSUxURVIgdS5fa2V5ID09ICR7c2Vzc2lvbklkfVxuICAgICAgICBMSU1JVCAxXG4gICAgICAgIFJFVFVSTiB1XG4gICAgICApXG4gICAgICBSRVRVUk4ge3VzZXI6IHVzZXIsIGNhcmQ6IHVwZGF0ZWRDYXJkfWA7XG5cbiAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgLnRoZW4oKHJlc3VsdHMgPSB7Y2FyZDoge30sIHVzZXI6IHt9fSkgPT4ge1xuICAgICAgY29uc3QgdXBkYXRlZENhcmQ6IFBheW1lbnRDYXJkVHlwZSA9IHJlc3VsdHMuY2FyZDtcbiAgICAgIGNvbnN0IHt1c2VyfSA9IHJlc3VsdHM7XG5cbiAgICAgIGlmKGlzRW1wdHkodXBkYXRlZENhcmQpKSB7XG4gICAgICAgIHRocm93IG5ldyBVc2VyRXJyb3IoJ25vdF9mb3VuZCcpO1xuICAgICAgfVxuXG4gICAgICAvLyBTdHJpcGVcbiAgICAgIGNvbnN0IHN0cmlwZUNsaWVudCA9IHN0cmlwZShDb25maWcuZ2V0KCdzdHJpcGUudG9rZW4nKSk7XG5cbiAgICAgIHJldHVybiBzdHJpcGVDbGllbnQuY3VzdG9tZXJzXG4gICAgICAgIC51cGRhdGVDYXJkKHVzZXIuc3RyaXBlQ3VzdG9tZXJJZCwgY2FyZC5zdHJpcGVJZCwge1xuICAgICAgICAgIGFkZHJlc3NfY2l0eTogZm9ybWF0Q2l0eSxcbiAgICAgICAgICBhZGRyZXNzX2NvdW50cnk6IGZvcm1hdENvdW50cnksXG4gICAgICAgICAgYWRkcmVzc19saW5lMTogZm9ybWF0U3RyZWV0MSxcbiAgICAgICAgICBhZGRyZXNzX3N0YXRlOiBmb3JtYXRTdGF0ZSxcbiAgICAgICAgICBhZGRyZXNzX3ppcDogZm9ybWF0WmlwLFxuICAgICAgICAgIGV4cF9tb250aDogZm9ybWF0RXhwTW9udGgsXG4gICAgICAgICAgZXhwX3llYXI6IGZvcm1hdEV4cFllYXIsXG4gICAgICAgICAgbmFtZTogZm9ybWF0RnVsbE5hbWVcbiAgICAgICAgfSlcbiAgICAgICAgLnRoZW4oKCkgPT4gY2FyZClcbiAgICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IHtcbiAgICAgICAgICBjb25zb2xlLmxvZygncGF5bWVudHM6OnVwZGF0ZUNhcmQ6OmVycm9yJywgZXJyb3IpO1xuICAgICAgICAgIHRocm93IG5ldyBVc2VyRXJyb3IoJ3BheW1lbnRfZXJyb3InKTtcbiAgICAgICAgfSk7XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgZ2V0Q3JlZGl0Q2FyZHMgPSAoY29udGV4dDogQXBpQ29udGV4dCk6IFByb21pc2U8UGF5bWVudENhcmRUeXBlW10+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnZ2V0Q3JlZGl0Q2FyZHMnO1xuICBjb25zdCB7ZGF0YWJhc2UsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG4gIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgRk9SIGMgSU4gY3JlZGl0Q2FyZHNcbiAgICAgICAgRklMVEVSIGMudXNlcklkID09ICR7c2Vzc2lvbklkfVxuICAgICAgICBSRVRVUk4gY2A7XG5cbiAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5hbGwoKSlcbiAgICAudGhlbigobGlzdDogUGF5bWVudENhcmRUeXBlW10gPSBbXSkgPT4gbGlzdClcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgYWN0aW9uLFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICBsYWJlbDogJ2RiX2Vycm9yJ1xuICAgIH0sIGVycm9yLCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpKTtcbn07XG5cbmV4cG9ydCBjb25zdCBkZWxldGVDcmVkaXRDYXJkID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGNhcmRJZDogc3RyaW5nKTogUHJvbWlzZTxib29sZWFuPiA9PiB7XG4gIGNvbnN0IHtkYXRhYmFzZSwgdXNlcklkOiBzZXNzaW9uSWR9ID0gY29udGV4dDtcbiAgY29uc3QgZm9ybWF0Q2FyZElkOiBzdHJpbmcgPSBwYXJzZUlkKGNhcmRJZCk7XG4gIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgXG4gICAgICBMRVQgY2FyZCA9IEZJUlNUKFxuICAgICAgICBGT1IgYyBJTiBjcmVkaXRDYXJkc1xuICAgICAgICBGSUxURVIgYy5fa2V5ID09ICR7Zm9ybWF0Q2FyZElkfSAmJiBjLnVzZXJJZCA9PSAke3Nlc3Npb25JZH1cbiAgICAgICAgTElNSVQgMVxuICAgICAgICBSRU1PVkUgYyBJTiBjcmVkaXRDYXJkc1xuICAgICAgICBSRVRVUk4gT0xEXG4gICAgICApXG4gICAgICBMRVQgdXNlciA9IEZJUlNUKFxuICAgICAgICBGT1IgdSBJTiB1c2Vyc1xuICAgICAgICBGSUxURVIgdS5fa2V5ID09ICR7c2Vzc2lvbklkfVxuICAgICAgICBMSU1JVCAxXG4gICAgICAgIFJFVFVSTiB1XG4gICAgICApXG4gICAgICBSRVRVUk4ge3VzZXI6IHVzZXIsIGNhcmQ6IGNhcmR9YDtcblxuICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAudGhlbigocmVzdWx0ID0ge2NhcmQ6IHt9LCB1c2VyOiB7fX0pID0+IHtcbiAgICAgIGlmKGlzRW1wdHkocmVzdWx0KSkge1xuICAgICAgICByZXR1cm4gZmFsc2U7XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IHtjYXJkLCB1c2VyfSA9IHJlc3VsdDtcbiAgICAgIGNvbnN0IHtfa2V5OiBjYXJkS2V5fSA9IGNhcmQ7XG5cbiAgICAgIC8vIFJlbW92ZSBsaW5rZWQgZWRnZXNcbiAgICAgIGNvbnN0IGVkZ2VDb2xsZWN0aW9uID0gdXNlRGIoZGF0YWJhc2UpLmVkZ2VDb2xsZWN0aW9uKCdoYXNQYXltZW50Jyk7XG5cbiAgICAgIHJldHVybiBlZGdlQ29sbGVjdGlvbi5vdXRFZGdlcyhjYXJkS2V5KVxuICAgICAgICAudGhlbigoZWRnZXMpID0+IHtcbiAgICAgICAgICBpZihlZGdlcy5sZW5ndGgpIHtcbiAgICAgICAgICAgIHJldHVybiBQcm9taXNlLmFsbChcbiAgICAgICAgICAgICAgZWRnZXMubWFwKChlZGdlKSA9PiB7XG4gICAgICAgICAgICAgICAgY29uc3Qge19rZXk6IGVkZ2VLZXl9ID0gZWRnZTtcbiAgICAgICAgICAgICAgICBjb25zdCByZW1vdmVBcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYFJFTU9WRSB7X2tleToke2VkZ2VLZXl9fSBJTiBoYXNQYXltZW50YDtcbiAgICAgICAgICAgICAgICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KHJlbW92ZUFxbFFyeSk7XG4gICAgICAgICAgICAgIH0pKVxuICAgICAgICAgICAgICAudGhlbigoKSA9PiB7XG4gICAgICAgICAgICAgICAgLy8gU3RyaXBlXG4gICAgICAgICAgICAgICAgY29uc3Qgc3RyaXBlQ2xpZW50ID0gc3RyaXBlKENvbmZpZy5nZXQoJ3N0cmlwZS50b2tlbicpKTtcblxuICAgICAgICAgICAgICAgIHJldHVybiBzdHJpcGVDbGllbnQuY3VzdG9tZXJzXG4gICAgICAgICAgICAgICAgICAuZGVsZXRlQ2FyZCh1c2VyLnN0cmlwZUN1c3RvbWVySWQsIGNhcmQuc3RyaXBlSWQpXG4gICAgICAgICAgICAgICAgICAudGhlbigoKSA9PiB0cnVlKVxuICAgICAgICAgICAgICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IHtcbiAgICAgICAgICAgICAgICAgICAgY29uc29sZS5sb2coJ3BheW1lbnRzOjpkZWxldGVDYXJkOjplcnJvcicsIGVycm9yKTtcbiAgICAgICAgICAgICAgICAgICAgdGhyb3cgbmV3IFVzZXJFcnJvcigncGF5bWVudF9lcnJvcicpO1xuICAgICAgICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICAgIH0pO1xuICAgICAgICAgIH1cblxuICAgICAgICAgIHJldHVybiBmYWxzZTtcbiAgICAgICAgfSk7XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgZGVsZXRlUGF5bWVudEFjY291bnRCYW5rID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGJhbmtJZDogc3RyaW5nKTogUHJvbWlzZTxib29sZWFuPiA9PiB7XG4gIGNvbnN0IHtkYXRhYmFzZSwgdXNlcklkOiBzZXNzaW9uSWR9ID0gY29udGV4dDtcblxuICAvLyBDbGVhbiBkYlxuICBjb25zdCB1cGRhdGU6IFVzZXJUeXBlID0ge1xuICAgIGJhbmtBY2NvdW50OiAnJyxcbiAgICBiYW5rRnVsbE5hbWU6ICcnLFxuICAgIGJhbmtJZDogJycsXG4gICAgYmFua1JvdXRpbmc6ICcnLFxuICAgIG1vZGlmaWVkOiBEYXRlLm5vdygpXG4gIH07XG4gIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgVVBEQVRFICR7c2Vzc2lvbklkfSBXSVRIICR7dXBkYXRlfSBJTiB1c2VycyBMSU1JVCAxIFJFVFVSTiBORVdgO1xuXG4gIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgIC50aGVuKCh1c2VyOiBVc2VyVHlwZSA9IHt9KSA9PiB7XG4gICAgICAvLyBTdHJpcGVcbiAgICAgIGNvbnN0IHN0cmlwZUNsaWVudCA9IHN0cmlwZShDb25maWcuZ2V0KCdzdHJpcGUudG9rZW4nKSk7XG5cbiAgICAgIHJldHVybiBzdHJpcGVDbGllbnQuY3VzdG9tZXJzXG4gICAgICAgIC5kZWxldGVTb3VyY2UodXNlci5zdHJpcGVBY2NvdW50SWQsIGJhbmtJZClcbiAgICAgICAgLnRoZW4oKHJlc3BvbnNlID0ge2RlbGV0ZWQ6IGZhbHNlfSkgPT4gcmVzcG9uc2UuZGVsZXRlZClcbiAgICAgICAgLmNhdGNoKCgpID0+IFByb21pc2UucmVzb2x2ZShmYWxzZSkpO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGNyZWF0ZVBheW1lbnRUcmFuc2ZlciA9IChjb250ZXh0OiBBcGlDb250ZXh0LCB0cmFuc2ZlcjogUGF5bWVudFRyYW5zZmVyKTogUHJvbWlzZTxQYXltZW50VHJhbnNmZXI+ID0+IHtcbiAgY29uc3Qge2RhdGFiYXNlLCB1c2VySWQ6IHNlc3Npb25JZH0gPSBjb250ZXh0O1xuICBjb25zdCB7YW1vdW50LCBjdXJyZW5jeX0gPSB0cmFuc2ZlcjtcbiAgY29uc3QgZm9ybWF0QW1vdW50OiBudW1iZXIgPSBwYXJzZU51bShhbW91bnQpO1xuICBjb25zdCBmb3JtYXRDdXJyZW5jeTogc3RyaW5nID0gcGFyc2VDaGFyKGN1cnJlbmN5LCAzLCAnVVNEJykudG9VcHBlckNhc2UoKTtcblxuICByZXR1cm4gZ2V0VXNlcihjb250ZXh0LCBzZXNzaW9uSWQpXG4gICAgLnRoZW4oKHVzZXI6IFVzZXJUeXBlKSA9PiB7XG4gICAgICAvLyBTdHJpcGVcbiAgICAgIGNvbnN0IHN0cmlwZUNsaWVudCA9IHN0cmlwZShDb25maWcuZ2V0KCdzdHJpcGUudG9rZW4nKSk7XG5cbiAgICAgIHJldHVybiBzdHJpcGVDbGllbnQudHJhbnNmZXJzXG4gICAgICAgIC5jcmVhdGUoe1xuICAgICAgICAgIGFtb3VudDogZm9ybWF0QW1vdW50LFxuICAgICAgICAgIGN1cnJlbmN5OiBmb3JtYXRDdXJyZW5jeSxcbiAgICAgICAgICBkZXN0aW5hdGlvbjogdXNlci5zdHJpcGVBY2NvdW50SWRcbiAgICAgICAgfSlcbiAgICAgICAgLnRoZW4oKHN0cmlwZVRyYW5zZmVyKSA9PiB7XG4gICAgICAgICAgY29uc29sZS5sb2coc3RyaXBlVHJhbnNmZXIpO1xuICAgICAgICAgIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcbiAgICAgICAgICBjb25zdCBpbnNlcnQ6IFBheW1lbnRUcmFuc2ZlciA9IHtcbiAgICAgICAgICAgIGFkZGVkOiBub3csXG4gICAgICAgICAgICBhbW91bnQ6IGZvcm1hdEFtb3VudCxcbiAgICAgICAgICAgIGN1cnJlbmN5OiBmb3JtYXRDdXJyZW5jeSxcbiAgICAgICAgICAgIG1vZGlmaWVkOiBub3csXG4gICAgICAgICAgICB1c2VySWQ6IHNlc3Npb25JZFxuICAgICAgICAgIH07XG4gICAgICAgICAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBJTlNFUlQgJHtpbnNlcnR9IElOIHRyYW5zZmVycyBSRVRVUk4gTkVXYDtcblxuICAgICAgICAgIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgICAgICAgICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgICAgICAgICAudGhlbigobmV3VHJhbnNmZXI6IFBheW1lbnRUcmFuc2ZlcikgPT4gbmV3VHJhbnNmZXIpO1xuICAgICAgICB9KTtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBjcmVhdGVQYXltZW50SG9sZCA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBwYXltZW50OiBQYXltZW50Q2hhcmdlKTogUHJvbWlzZTxQYXltZW50Q2hhcmdlPiA9PiB7XG4gIGNvbnN0IHtkYXRhYmFzZSwgdXNlcklkOiBzZXNzaW9uSWR9ID0gY29udGV4dDtcbiAgY29uc3Qge2Ftb3VudCwgY2FwdHVyZSwgY2FyZElkLCBjdXJyZW5jeSwgZGVzY3JpcHRpb259ID0gcGF5bWVudDtcbiAgY29uc3QgZm9ybWF0Q3VycmVuY3kgPSBwYXJzZUNoYXIoY3VycmVuY3ksIDMsICdVU0QnKS50b1VwcGVyQ2FzZSgpO1xuXG4gIGNvbnN0IHN0cmlwZUNsaWVudCA9IHN0cmlwZShDb25maWcuZ2V0KCdzdHJpcGUudG9rZW4nKSk7XG5cbiAgcmV0dXJuIHN0cmlwZUNsaWVudC5jaGFyZ2VzXG4gICAgLmNyZWF0ZSh7XG4gICAgICBhbW91bnQsXG4gICAgICBjYXB0dXJlLFxuICAgICAgY3VycmVuY3k6IGZvcm1hdEN1cnJlbmN5LFxuICAgICAgZGVzY3JpcHRpb24sXG4gICAgICBzb3VyY2U6IGNhcmRJZFxuICAgIH0pXG4gICAgLnRoZW4oKHN0cmlwZUNoYXJnZSkgPT4ge1xuICAgICAgY29uc3Qgbm93OiBudW1iZXIgPSBEYXRlLm5vdygpO1xuICAgICAgY29uc3QgaW5zZXJ0OiBQYXltZW50Q2hhcmdlID0ge1xuICAgICAgICBhZGRlZDogbm93LFxuICAgICAgICBhbW91bnQsXG4gICAgICAgIGNhcHR1cmUsXG4gICAgICAgIGNhcmRJZCxcbiAgICAgICAgY2hhcmdlRmFpbENvZGU6IHN0cmlwZUNoYXJnZS5mYWlsdXJlX2NvZGUsXG4gICAgICAgIGNoYXJnZUZhaWxNc2c6IHN0cmlwZUNoYXJnZS5mYWlsdXJlX21lc3NhZ2UsXG4gICAgICAgIGNoYXJnZUlkOiBzdHJpcGVDaGFyZ2UuaWQsXG4gICAgICAgIGNoYXJnZVN0YXR1czogc3RyaXBlQ2hhcmdlLnN0YXR1cyxcbiAgICAgICAgY3VycmVuY3k6IGZvcm1hdEN1cnJlbmN5LFxuICAgICAgICBkZXNjcmlwdGlvbixcbiAgICAgICAgbW9kaWZpZWQ6IG5vdyxcbiAgICAgICAgdXNlcklkOiBzZXNzaW9uSWRcbiAgICAgIH07XG4gICAgICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYElOU0VSVCAke2luc2VydH0gSU4gcGF5bWVudHMgUkVUVVJOIE5FV2A7XG5cbiAgICAgIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgICAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAgICAgLnRoZW4oKG5ld1BheW1lbnQ6IFBheW1lbnRDaGFyZ2UpID0+IG5ld1BheW1lbnQpO1xuICAgIH0pXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IHtcbiAgICAgIGNvbnNvbGUubG9nKCdwYXltZW50czo6Y3JlYXRlSG9sZDo6ZXJyb3InLCBlcnJvcik7XG4gICAgICB0aHJvdyBuZXcgVXNlckVycm9yKCdwYXltZW50X2Vycm9yJyk7XG4gICAgfSk7XG59O1xuIl19