@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.
Files changed (50) hide show
  1. package/lib/config.js +138 -0
  2. package/lib/data/conversations.js +228 -0
  3. package/lib/data/dynamodb.js +172 -0
  4. package/lib/data/email.js +194 -0
  5. package/lib/data/files.js +463 -0
  6. package/lib/data/groups.js +401 -0
  7. package/lib/data/images.js +841 -0
  8. package/lib/data/index.js +234 -0
  9. package/lib/data/ios.js +327 -0
  10. package/lib/data/locations.js +148 -0
  11. package/lib/data/messages.js +281 -0
  12. package/lib/data/notifications.js +59 -0
  13. package/lib/data/payments.js +798 -0
  14. package/lib/data/posts.js +637 -0
  15. package/lib/data/reactions.js +243 -0
  16. package/lib/data/s3.js +133 -0
  17. package/lib/data/search.js +111 -0
  18. package/lib/data/sms.js +79 -0
  19. package/lib/data/subscription.js +311 -0
  20. package/lib/data/tags.js +343 -0
  21. package/lib/data/users.js +415 -0
  22. package/lib/index.js +42 -0
  23. package/lib/types/apps.js +2 -0
  24. package/lib/types/arangodb.js +2 -0
  25. package/lib/types/auth.js +2 -0
  26. package/lib/types/conversations.js +2 -0
  27. package/lib/types/email.js +2 -0
  28. package/lib/types/files.js +2 -0
  29. package/lib/types/google.js +2 -0
  30. package/lib/types/groups.js +2 -0
  31. package/lib/types/images.js +2 -0
  32. package/lib/types/index.js +210 -0
  33. package/lib/types/locations.js +2 -0
  34. package/lib/types/messages.js +2 -0
  35. package/lib/types/notifications.js +2 -0
  36. package/lib/types/payments.js +2 -0
  37. package/lib/types/posts.js +2 -0
  38. package/lib/types/reactions.js +2 -0
  39. package/lib/types/tags.js +2 -0
  40. package/lib/types/users.js +2 -0
  41. package/lib/utils/analytics.js +59 -0
  42. package/lib/utils/arangodb.d.ts +0 -1
  43. package/lib/utils/arangodb.js +122 -0
  44. package/lib/utils/auth.d.ts +0 -2
  45. package/lib/utils/auth.js +55 -0
  46. package/lib/utils/graphql.js +19 -0
  47. package/lib/utils/index.js +78 -0
  48. package/lib/utils/objects.js +54 -0
  49. package/lib/utils/redis.js +28 -0
  50. 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,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kYXRhL3BheW1lbnRzLnRzIl0sIm5hbWVzIjpbImV2ZW50Q2F0ZWdvcnkiLCJhZGRDdXN0b21lckFjY291bnQiLCJjb250ZXh0IiwiYWN0aW9uIiwiZGF0YWJhc2UiLCJzZXNzaW9uSWQiLCJ1c2VySWQiLCJ1c2VybmFtZSIsInN0cmlwZUNsaWVudCIsInN0cmlwZSIsIkNvbmZpZyIsImdldCIsImN1c3RvbWVycyIsImNyZWF0ZSIsIm1ldGFkYXRhIiwidGhlbiIsImN1c3RvbWVyIiwibm93IiwiRGF0ZSIsInVwZGF0ZSIsIm1vZGlmaWVkIiwic3RyaXBlQ3VzdG9tZXJJZCIsImlkIiwiYXFsUXJ5IiwiYXFsIiwicXVlcnkiLCJjdXJzb3IiLCJuZXh0IiwidXBkYXRlZFVzZXIiLCJlcnJvciIsImNhdGVnb3J5IiwibGFiZWwiLCJhZGRQYXltZW50QWNjb3VudEJhbmsiLCJiYW5rQWNjb3VudCIsImFjY291bnROdW1iZXIiLCJmdWxsTmFtZSIsInJvdXRpbmciLCJmb3JtYXRBY2NvdW50IiwiVXNlckVycm9yIiwiZm9ybWF0RnVsbE5hbWUiLCJmb3JtYXRSb3V0aW5nIiwidXNlciIsImNyZWF0ZVNvdXJjZSIsInN0cmlwZUFjY291bnRJZCIsInNvdXJjZSIsImFjY291bnRfaG9sZGVyX25hbWUiLCJhY2NvdW50X2hvbGRlcl90eXBlIiwiYWNjb3VudF9udW1iZXIiLCJjb3VudHJ5IiwiY3VycmVuY3kiLCJvYmplY3QiLCJyb3V0aW5nX251bWJlciIsImFjY291bnQiLCJsYXN0NCIsImJhbmtGdWxsTmFtZSIsImJhbmtJZCIsImJhbmtSb3V0aW5nIiwibXNnIiwibWVzc2FnZSIsImFkZFBheW1lbnRBY2NvdW50Q2FyZCIsImNhcmQiLCJyZXF1ZXN0RGF0YSIsInVzZXJLZXkiLCJfa2V5IiwiY2l0eSIsImRvYiIsImVtYWlsIiwiZmlyc3QiLCJsYXN0IiwiZG9iRGF5IiwiRGF0ZVRpbWUiLCJmcm9tTWlsbGlzIiwidG9Gb3JtYXQiLCJkb2JNb250aCIsImRvYlllYXIiLCJhY2NlcHRlZFRlcm1zIiwiZXhwTW9udGgiLCJleHBZZWFyIiwic3RhdGUiLCJzdHJlZXQxIiwiemlwIiwiYWNjZXB0ZWQiLCJsb2NhbCIsIm1pbGxpc2Vjb25kIiwiYWNjb3VudHMiLCJleHRlcm5hbF9hY2NvdW50IiwiZXhwX21vbnRoIiwiZXhwX3llYXIiLCJudW1iZXIiLCJsZWdhbF9lbnRpdHkiLCJhZGRyZXNzIiwibGluZTEiLCJwb3N0YWxfY29kZSIsImRheSIsIm1vbnRoIiwieWVhciIsImZpcnN0X25hbWUiLCJsYXN0X25hbWUiLCJ0eXBlIiwibWFuYWdlZCIsInRvc19hY2NlcHRhbmNlIiwiZGF0ZSIsImlwIiwidXNlcl9hZ2VudCIsInVzZXJBZ2VudCIsImFkZENyZWRpdENhcmQiLCJjdmMiLCJmb3JtYXROdW1iZXIiLCJmb3JtYXRFeHBNb250aCIsImZvcm1hdEV4cFllYXIiLCJmb3JtYXRDdmMiLCJwYXltZW50Q2FyZCIsImZvcm1hdENpdHkiLCJmb3JtYXRDb3VudHJ5IiwiZm9ybWF0U3RyZWV0IiwiZm9ybWF0U3RhdGUiLCJmb3JtYXRaaXAiLCJhZGRyZXNzX2NpdHkiLCJhZGRyZXNzX2NvdW50cnkiLCJhZGRyZXNzX2xpbmUxIiwiYWRkcmVzc19zdGF0ZSIsImFkZHJlc3NfemlwIiwibmFtZSIsInN0cmlwZU9iaiIsImluc2VydCIsImFkZGVkIiwiYnJhbmQiLCJjdmNfY2hlY2siLCJpbnNlcnRBcWxRcnkiLCJuZXdDYXJkIiwiY2FyZElkIiwiX2lkIiwiY2FyZEtleSIsImVkZ2VDb2xsZWN0aW9uIiwiZWRnZUlkIiwiZWRnZSIsInNhdmUiLCJ1cGRhdGVDcmVkaXRDYXJkIiwiZm9ybWF0SWQiLCJmb3JtYXRTdHJlZXQxIiwicmVzdWx0cyIsInVwZGF0ZWRDYXJkIiwidXBkYXRlQ2FyZCIsInN0cmlwZUlkIiwiY29uc29sZSIsImxvZyIsImdldENyZWRpdENhcmRzIiwiYWxsIiwibGlzdCIsImRlbGV0ZUNyZWRpdENhcmQiLCJmb3JtYXRDYXJkSWQiLCJyZXN1bHQiLCJvdXRFZGdlcyIsImVkZ2VzIiwibGVuZ3RoIiwiUHJvbWlzZSIsIm1hcCIsImVkZ2VLZXkiLCJyZW1vdmVBcWxRcnkiLCJkZWxldGVDYXJkIiwiZGVsZXRlUGF5bWVudEFjY291bnRCYW5rIiwiZGVsZXRlU291cmNlIiwicmVzcG9uc2UiLCJkZWxldGVkIiwicmVzb2x2ZSIsImNyZWF0ZVBheW1lbnRUcmFuc2ZlciIsInRyYW5zZmVyIiwiYW1vdW50IiwiZm9ybWF0QW1vdW50IiwiZm9ybWF0Q3VycmVuY3kiLCJ0b1VwcGVyQ2FzZSIsInRyYW5zZmVycyIsImRlc3RpbmF0aW9uIiwic3RyaXBlVHJhbnNmZXIiLCJuZXdUcmFuc2ZlciIsImNyZWF0ZVBheW1lbnRIb2xkIiwicGF5bWVudCIsImNhcHR1cmUiLCJkZXNjcmlwdGlvbiIsImNoYXJnZXMiLCJzdHJpcGVDaGFyZ2UiLCJjaGFyZ2VGYWlsQ29kZSIsImZhaWx1cmVfY29kZSIsImNoYXJnZUZhaWxNc2ciLCJmYWlsdXJlX21lc3NhZ2UiLCJjaGFyZ2VJZCIsImNoYXJnZVN0YXR1cyIsInN0YXR1cyIsIm5ld1BheW1lbnQiXSwibWFwcGluZ3MiOiI7Ozs7Ozs7QUFBQTs7QUFDQTs7QUFHQTs7QUFDQTs7QUFDQTs7QUFDQTs7QUFFQTs7QUFFQTs7QUFDQTs7QUFDQTs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFFQTs7OztBQUlBLElBQU1BLGFBQXFCLEdBQUcsVUFBOUI7O0FBRU8sSUFBTUMsa0JBQWtCLEdBQUcsU0FBckJBLGtCQUFxQixDQUFDQyxPQUFELEVBQTJDO0FBQzNFLE1BQU1DLE1BQWMsR0FBRyxvQkFBdkI7QUFEMkUsTUFFcEVDLFFBRm9FLEdBRTNCRixPQUYyQixDQUVwRUUsUUFGb0U7QUFBQSxNQUVsREMsU0FGa0QsR0FFM0JILE9BRjJC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QUFoQixDQUFzQkYsTUFBdEIsRUFDSlIsSUFESSxDQUNDLFVBQUNXLE1BQUQ7QUFBQSxhQUF5QkEsTUFBTSxDQUFDQyxJQUFQLEVBQXpCO0FBQUEsS0FERCxFQUVKWixJQUZJLENBRUMsVUFBQzZLLFVBQUQ7QUFBQSxhQUErQkEsVUFBL0I7QUFBQSxLQUZELENBQVA7QUFHRCxHQTdCSSxXQThCRSxVQUFDL0osS0FBRCxFQUFrQjtBQUN2Qm1ILElBQUFBLE9BQU8sQ0FBQ0MsR0FBUixDQUFZLDZCQUFaLEVBQTJDcEgsS0FBM0M7QUFDQSxVQUFNLElBQUlTLHdCQUFKLENBQWMsZUFBZCxDQUFOO0FBQ0QsR0FqQ0ksQ0FBUDtBQWtDRCxDQXpDTSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7Y3JlYXRlSGFzaCwgcGFyc2VDaGFyLCBwYXJzZUlkLCBwYXJzZU51bSwgcGFyc2VTdHJpbmcsIHBhcnNlVmFyQ2hhcn0gZnJvbSAnQG5sYWJzL3V0aWxzJztcbmltcG9ydCB7YXFsfSBmcm9tICdhcmFuZ29qcyc7XG5pbXBvcnQge0FxbFF1ZXJ5fSBmcm9tICdhcmFuZ29qcy9saWIvY2pzL2FxbC1xdWVyeSc7XG5pbXBvcnQge0FycmF5Q3Vyc29yfSBmcm9tICdhcmFuZ29qcy9saWIvY2pzL2N1cnNvcic7XG5pbXBvcnQge1VzZXJFcnJvcn0gZnJvbSAnZ3JhcGhxbC1lcnJvcnMnO1xuaW1wb3J0IGlzRW1wdHkgZnJvbSAnbG9kYXNoL2lzRW1wdHknO1xuaW1wb3J0IHtEYXRlVGltZX0gZnJvbSAnbHV4b24nO1xuaW1wb3J0ICogYXMgc3RyaXBlIGZyb20gJ3N0cmlwZSc7XG5cbmltcG9ydCB7Q29uZmlnfSBmcm9tICcuLi9jb25maWcnO1xuaW1wb3J0IHtBcGlDb250ZXh0LCBQYXltZW50QmFua0FjY291bnQsIFBheW1lbnRDYXJkVHlwZSwgUGF5bWVudENoYXJnZSwgUGF5bWVudFRyYW5zZmVyLCBVc2VyVHlwZX0gZnJvbSAnLi4vdHlwZXMnO1xuaW1wb3J0IHtsb2dFcnJvcn0gZnJvbSAnLi4vdXRpbHMvYW5hbHl0aWNzJztcbmltcG9ydCB7dXNlRGJ9IGZyb20gJy4uL3V0aWxzL2FyYW5nb2RiJztcbmltcG9ydCB7Z2V0VXNlcn0gZnJvbSAnLi91c2Vycyc7XG5cbi8qKlxuICogQ29weXJpZ2h0IChjKSAyMDE5LVByZXNlbnQsIE5pdHJvZ2VuIExhYnMsIEluYy5cbiAqIENvcHlyaWdodHMgbGljZW5zZWQgdW5kZXIgdGhlIE1JVCBMaWNlbnNlLiBTZWUgdGhlIGFjY29tcGFueWluZyBMSUNFTlNFIGZpbGUgZm9yIHRlcm1zLlxuICovXG5jb25zdCBldmVudENhdGVnb3J5OiBzdHJpbmcgPSAncGF5bWVudHMnO1xuXG5leHBvcnQgY29uc3QgYWRkQ3VzdG9tZXJBY2NvdW50ID0gKGNvbnRleHQ6IEFwaUNvbnRleHQpOiBQcm9taXNlPGJvb2xlYW4+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnYWRkQ3VzdG9tZXJBY2NvdW50JztcbiAgY29uc3Qge2RhdGFiYXNlLCB1c2VySWQ6IHNlc3Npb25JZCwgdXNlcm5hbWV9ID0gY29udGV4dDtcblxuICAvLyBTdHJpcGVcbiAgY29uc3Qgc3RyaXBlQ2xpZW50ID0gc3RyaXBlKENvbmZpZy5nZXQoJ3N0cmlwZS50b2tlbicpKTtcblxuICByZXR1cm4gc3RyaXBlQ2xpZW50LmN1c3RvbWVyc1xuICAgIC5jcmVhdGUoe1xuICAgICAgbWV0YWRhdGE6IHtcbiAgICAgICAgdXNlcklkOiBzZXNzaW9uSWQsXG4gICAgICAgIHVzZXJuYW1lXG4gICAgICB9XG4gICAgfSlcbiAgICAudGhlbigoY3VzdG9tZXIpID0+IHtcbiAgICAgIC8vIENyZWF0ZSBzZXNzaW9uXG4gICAgICBjb25zdCBub3c6IG51bWJlciA9IERhdGUubm93KCk7XG4gICAgICBjb25zdCB1cGRhdGU6IFVzZXJUeXBlID0ge1xuICAgICAgICBtb2RpZmllZDogbm93LFxuICAgICAgICBzdHJpcGVDdXN0b21lcklkOiBjdXN0b21lci5pZFxuICAgICAgfTtcblxuICAgICAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBVUERBVEUgJHtzZXNzaW9uSWR9IFdJVEggJHt1cGRhdGV9IElOIHVzZXJzIExJTUlUIDEgUkVUVVJOIE5FV2A7XG5cbiAgICAgIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgICAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAgICAgLnRoZW4oKHVwZGF0ZWRVc2VyOiBVc2VyVHlwZSA9IHt9KSA9PiAhaXNFbXB0eSh1cGRhdGVkVXNlcikpXG4gICAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgICAgYWN0aW9uLFxuICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgIGxhYmVsOiAnZGJfZXJyb3InXG4gICAgICAgIH0sIGVycm9yLCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpKTtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBhZGRQYXltZW50QWNjb3VudEJhbmsgPSAoY29udGV4dDogQXBpQ29udGV4dCwgYmFua0FjY291bnQ6IFBheW1lbnRCYW5rQWNjb3VudCk6IFByb21pc2U8Ym9vbGVhbj4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdhZGRQYXltZW50QWNjb3VudEJhbmsnO1xuICBjb25zdCB7ZGF0YWJhc2UsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG5cbiAgLy8gUGFyYW1zXG4gIGNvbnN0IHtcbiAgICBhY2NvdW50TnVtYmVyLFxuICAgIGZ1bGxOYW1lLFxuICAgIHJvdXRpbmdcbiAgfSA9IGJhbmtBY2NvdW50O1xuXG4gIGNvbnN0IGZvcm1hdEFjY291bnQ6IHN0cmluZyA9IHBhcnNlU3RyaW5nKGFjY291bnROdW1iZXIsIDMyKTtcblxuICBpZihmb3JtYXRBY2NvdW50ID09PSAnJykge1xuICAgIHRocm93IG5ldyBVc2VyRXJyb3IoJ3JlcXVpcmVkX2FjY291bnRfbnVtYmVyJyk7XG4gIH1cblxuICBjb25zdCBmb3JtYXRGdWxsTmFtZTogc3RyaW5nID0gcGFyc2VWYXJDaGFyKGZ1bGxOYW1lLCAxMjgpO1xuXG4gIGlmKGZvcm1hdEZ1bGxOYW1lID09PSAnJykge1xuICAgIHRocm93IG5ldyBVc2VyRXJyb3IoJ3JlcXVpcmVkX2Z1bGxfbmFtZScpO1xuICB9XG5cbiAgY29uc3QgZm9ybWF0Um91dGluZzogc3RyaW5nID0gcGFyc2VTdHJpbmcocm91dGluZywgMzIpO1xuXG4gIGlmKGZvcm1hdFJvdXRpbmcgPT09ICcnKSB7XG4gICAgdGhyb3cgbmV3IFVzZXJFcnJvcigncmVxdWlyZWRfcm91dGluZ19udW1iZXInKTtcbiAgfVxuXG4gIHJldHVybiBnZXRVc2VyKGNvbnRleHQsIHNlc3Npb25JZClcbiAgICAudGhlbigodXNlcjogVXNlclR5cGUpID0+IHtcbiAgICAgIC8vIFN0cmlwZVxuICAgICAgY29uc3Qgc3RyaXBlQ2xpZW50ID0gc3RyaXBlKENvbmZpZy5nZXQoJ3N0cmlwZS50b2tlbicpKTtcblxuICAgICAgcmV0dXJuIHN0cmlwZUNsaWVudC5jdXN0b21lcnNcbiAgICAgICAgLmNyZWF0ZVNvdXJjZSh1c2VyLnN0cmlwZUFjY291bnRJZCwge1xuICAgICAgICAgIHNvdXJjZToge1xuICAgICAgICAgICAgYWNjb3VudF9ob2xkZXJfbmFtZTogZm9ybWF0RnVsbE5hbWUsXG4gICAgICAgICAgICBhY2NvdW50X2hvbGRlcl90eXBlOiAnaW5kaXZpZHVhbCcsXG4gICAgICAgICAgICBhY2NvdW50X251bWJlcjogZm9ybWF0QWNjb3VudCxcbiAgICAgICAgICAgIGNvdW50cnk6ICdVUycsXG4gICAgICAgICAgICBjdXJyZW5jeTogJ1VTRCcsXG4gICAgICAgICAgICBvYmplY3Q6ICdiYW5rX2FjY291bnQnLFxuICAgICAgICAgICAgcm91dGluZ19udW1iZXI6IGZvcm1hdFJvdXRpbmdcbiAgICAgICAgICB9XG4gICAgICAgIH0pXG4gICAgICAgIC50aGVuKChhY2NvdW50KSA9PiB7XG4gICAgICAgICAgY29uc3QgdXBkYXRlOiBhbnkgPSB7XG4gICAgICAgICAgICBiYW5rQWNjb3VudDogYWNjb3VudC5sYXN0NCxcbiAgICAgICAgICAgIGJhbmtGdWxsTmFtZTogZm9ybWF0RnVsbE5hbWUsXG4gICAgICAgICAgICBiYW5rSWQ6IGFjY291bnQuaWQsXG4gICAgICAgICAgICBiYW5rUm91dGluZzogYWNjb3VudC5yb3V0aW5nX251bWJlcixcbiAgICAgICAgICAgIG1vZGlmaWVkOiBEYXRlLm5vdygpXG4gICAgICAgICAgfTtcblxuICAgICAgICAgIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgVVBEQVRFICR7c2Vzc2lvbklkfSBXSVRIICR7dXBkYXRlfSBJTiB1c2VycyBMSU1JVCAxIFJFVFVSTiBORVdgO1xuXG4gICAgICAgICAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgICAgICAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAgICAgICAgIC50aGVuKCh1cGRhdGVkVXNlcjogVXNlclR5cGUgPSB7fSkgPT4gdXBkYXRlZFVzZXIpO1xuICAgICAgICB9KVxuICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgICAgIGNvbnN0IG1zZyA9IGVycm9yLm1lc3NhZ2U7XG5cbiAgICAgICAgICBpZihtc2cgPT09ICdBIGJhbmsgYWNjb3VudCB3aXRoIHRoYXQgcm91dGluZyBudW1iZXIgYW5kIGFjY291bnQgbnVtYmVyICcgK1xuICAgICAgICAgICAgJ2FscmVhZHkgZXhpc3RzIGZvciB0aGlzIGN1c3RvbWVyLicpIHtcbiAgICAgICAgICAgIHJldHVybiBsb2dFcnJvcih7XG4gICAgICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgICAgIGxhYmVsOiAnYmFua19hY2NvdW50X2V4aXN0cydcbiAgICAgICAgICAgIH0sIGVycm9yLCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpO1xuICAgICAgICAgIH1cbiAgICAgICAgICByZXR1cm4gbG9nRXJyb3Ioe1xuICAgICAgICAgICAgYWN0aW9uLFxuICAgICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgICBsYWJlbDogJ3BheW1lbnRfZXJyb3InXG4gICAgICAgICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCk7XG4gICAgICAgIH0pO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGFkZFBheW1lbnRBY2NvdW50Q2FyZCA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBjYXJkOiBQYXltZW50Q2FyZFR5cGUsIHJlcXVlc3REYXRhKTogUHJvbWlzZTxVc2VyVHlwZT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdhZGRQYXltZW50QWNjb3VudENhcmQnO1xuICBjb25zdCB7ZGF0YWJhc2UsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG5cbiAgcmV0dXJuIGdldFVzZXIoY29udGV4dCwgc2Vzc2lvbklkKVxuICAgIC50aGVuKCh1c2VyOiBVc2VyVHlwZSkgPT4ge1xuICAgICAgLy8gVXNlclxuICAgICAgY29uc3Qge1xuICAgICAgICBfa2V5OiB1c2VyS2V5LFxuICAgICAgICBjb3VudHJ5ID0gJ1VTJyxcbiAgICAgICAgY2l0eSxcbiAgICAgICAgY3VycmVuY3kgPSAnVVNEJyxcbiAgICAgICAgZG9iLFxuICAgICAgICBlbWFpbCxcbiAgICAgICAgZmlyc3QsXG4gICAgICAgIGxhc3QsXG4gICAgICAgIHVzZXJuYW1lXG4gICAgICB9OiBVc2VyVHlwZSA9IHVzZXI7XG5cbiAgICAgIC8vIEJpcnRoZGF5XG4gICAgICBjb25zdCBkb2JEYXk6IG51bWJlciA9ICsoRGF0ZVRpbWUuZnJvbU1pbGxpcyhkb2IpLnRvRm9ybWF0KCdkJykpO1xuICAgICAgY29uc3QgZG9iTW9udGg6IG51bWJlciA9ICsoRGF0ZVRpbWUuZnJvbU1pbGxpcyhkb2IpLnRvRm9ybWF0KCdNJykpO1xuICAgICAgY29uc3QgZG9iWWVhcjogbnVtYmVyID0gKyhEYXRlVGltZS5mcm9tTWlsbGlzKGRvYikudG9Gb3JtYXQoJ3knKSk7XG5cbiAgICAgIC8vIENhcmRcbiAgICAgIGNvbnN0IHtcbiAgICAgICAgYWNjb3VudE51bWJlcixcbiAgICAgICAgYWNjZXB0ZWRUZXJtcyxcbiAgICAgICAgZXhwTW9udGgsXG4gICAgICAgIGV4cFllYXIsXG4gICAgICAgIHN0YXRlLFxuICAgICAgICBzdHJlZXQxLFxuICAgICAgICB6aXBcbiAgICAgIH06IFBheW1lbnRDYXJkVHlwZSA9IGNhcmQ7XG5cbiAgICAgIC8vIEFjY2VwdGFuY2VcbiAgICAgIGNvbnN0IGFjY2VwdGVkOiBudW1iZXIgPSBhY2NlcHRlZFRlcm1zID8gRGF0ZVRpbWUubG9jYWwoKS5taWxsaXNlY29uZCA6IDA7XG5cbiAgICAgIC8vIFN0cmlwZVxuICAgICAgY29uc3Qgc3RyaXBlQ2xpZW50ID0gc3RyaXBlKENvbmZpZy5nZXQoJ3N0cmlwZS50b2tlbicpKTtcblxuICAgICAgcmV0dXJuIHN0cmlwZUNsaWVudC5hY2NvdW50c1xuICAgICAgICAuY3JlYXRlKHtcbiAgICAgICAgICBjb3VudHJ5LFxuICAgICAgICAgIGVtYWlsLFxuICAgICAgICAgIGV4dGVybmFsX2FjY291bnQ6IHtcbiAgICAgICAgICAgIGN1cnJlbmN5LFxuICAgICAgICAgICAgZXhwX21vbnRoOiBleHBNb250aCxcbiAgICAgICAgICAgIGV4cF95ZWFyOiBleHBZZWFyLFxuICAgICAgICAgICAgbnVtYmVyOiBhY2NvdW50TnVtYmVyLFxuICAgICAgICAgICAgb2JqZWN0OiAnY2FyZCdcbiAgICAgICAgICB9LFxuICAgICAgICAgIGxlZ2FsX2VudGl0eToge1xuICAgICAgICAgICAgYWRkcmVzczoge1xuICAgICAgICAgICAgICBjaXR5LFxuICAgICAgICAgICAgICBsaW5lMTogc3RyZWV0MSxcbiAgICAgICAgICAgICAgcG9zdGFsX2NvZGU6IHppcCxcbiAgICAgICAgICAgICAgc3RhdGVcbiAgICAgICAgICAgIH0sXG4gICAgICAgICAgICBkb2I6IHtcbiAgICAgICAgICAgICAgZGF5OiBkb2JEYXksXG4gICAgICAgICAgICAgIG1vbnRoOiBkb2JNb250aCxcbiAgICAgICAgICAgICAgeWVhcjogZG9iWWVhclxuICAgICAgICAgICAgfSxcbiAgICAgICAgICAgIGZpcnN0X25hbWU6IGZpcnN0LFxuICAgICAgICAgICAgbGFzdF9uYW1lOiBsYXN0LFxuICAgICAgICAgICAgLy8gc3NuX2xhc3RfNDogc3NuLnN1YnN0cigtNCksXG4gICAgICAgICAgICAvLyBwZXJzb25hbF9pZF9udW1iZXI6IHNzbixcbiAgICAgICAgICAgIHR5cGU6ICdpbmRpdmlkdWFsJ1xuICAgICAgICAgIH0sXG4gICAgICAgICAgbWFuYWdlZDogdHJ1ZSxcbiAgICAgICAgICBtZXRhZGF0YToge1xuICAgICAgICAgICAgdXNlcklkOiB1c2VyS2V5LFxuICAgICAgICAgICAgdXNlcm5hbWVcbiAgICAgICAgICB9LFxuICAgICAgICAgIHRvc19hY2NlcHRhbmNlOiB7XG4gICAgICAgICAgICBkYXRlOiBhY2NlcHRlZCxcbiAgICAgICAgICAgIGlwOiByZXF1ZXN0RGF0YS5pcCxcbiAgICAgICAgICAgIHVzZXJfYWdlbnQ6IHJlcXVlc3REYXRhLnVzZXJBZ2VudC5zb3VyY2VcbiAgICAgICAgICB9XG4gICAgICAgIH0pXG4gICAgICAgIC50aGVuKChjdXN0b21lcikgPT4ge1xuICAgICAgICAgIC8vIENyZWF0ZSBzZXNzaW9uXG4gICAgICAgICAgY29uc3Qgbm93OiBudW1iZXIgPSBEYXRlLm5vdygpO1xuICAgICAgICAgIGNvbnN0IHVwZGF0ZTogVXNlclR5cGUgPSB7XG4gICAgICAgICAgICBtb2RpZmllZDogbm93LFxuICAgICAgICAgICAgc3RyaXBlQWNjb3VudElkOiBjdXN0b21lci5pZFxuICAgICAgICAgIH07XG5cbiAgICAgICAgICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYFVQREFURSAke3VzZXJLZXl9IFdJVEggJHt1cGRhdGV9IElOIHVzZXJzIExJTUlUIDEgUkVUVVJOIE5FV2A7XG5cbiAgICAgICAgICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAgICAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgICAgICAgICAgLnRoZW4oKHVwZGF0ZWRVc2VyOiBVc2VyVHlwZSA9IHt9KSA9PiAhaXNFbXB0eSh1cGRhdGVkVXNlcikpXG4gICAgICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgICAgICBsYWJlbDogJ2RiX2Vycm9yJ1xuICAgICAgICAgICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCkpO1xuICAgICAgICB9KTtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBhZGRDcmVkaXRDYXJkID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGNhcmQ6IFBheW1lbnRDYXJkVHlwZSk6IFByb21pc2U8UGF5bWVudENhcmRUeXBlPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2FkZENyZWRpdENhcmQnO1xuICBjb25zdCB7ZGF0YWJhc2UsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG5cbiAgY29uc3Qge1xuICAgIGFjY291bnROdW1iZXIsXG4gICAgY2l0eSxcbiAgICBjb3VudHJ5LFxuICAgIGN2YyxcbiAgICBleHBNb250aCxcbiAgICBleHBZZWFyLFxuICAgIGZ1bGxOYW1lLFxuICAgIHN0cmVldDEsXG4gICAgc3RhdGUsXG4gICAgemlwXG4gIH06IFBheW1lbnRDYXJkVHlwZSA9IGNhcmQ7XG4gIGNvbnN0IGZvcm1hdE51bWJlcjogbnVtYmVyID0gcGFyc2VOdW0oYWNjb3VudE51bWJlciwgMTYpO1xuXG4gIGlmKCFmb3JtYXROdW1iZXIpIHtcbiAgICB0aHJvdyBuZXcgVXNlckVycm9yKCdyZXF1aXJlZF9jcmVkaXRfY2FyZF9udW1iZXInKTtcbiAgfVxuXG4gIGNvbnN0IGZvcm1hdEV4cE1vbnRoOiBudW1iZXIgPSBwYXJzZU51bShleHBNb250aCwgMik7XG5cbiAgaWYoIWZvcm1hdEV4cE1vbnRoKSB7XG4gICAgdGhyb3cgbmV3IFVzZXJFcnJvcigncmVxdWlyZWRfY3JlZGl0X2NhcmRfZXhwX21vbnRoJyk7XG4gIH1cblxuICBjb25zdCBmb3JtYXRFeHBZZWFyOiBudW1iZXIgPSBwYXJzZU51bShleHBZZWFyLCAyKTtcblxuICBpZighZm9ybWF0RXhwWWVhcikge1xuICAgIHRocm93IG5ldyBVc2VyRXJyb3IoJ3JlcXVpcmVkX2NyZWRpdF9jYXJkX2V4cF95ZWFyJyk7XG4gIH1cblxuICBjb25zdCBmb3JtYXRDdmM6IG51bWJlciA9IHBhcnNlTnVtKGN2YywgMyk7XG5cbiAgaWYoIWZvcm1hdEN2Yykge1xuICAgIHRocm93IG5ldyBVc2VyRXJyb3IoJ3JlcXVpcmVkX2NyZWRpdF9jYXJkX2N2YycpO1xuICB9XG5cbiAgY29uc3QgcGF5bWVudENhcmQ6IFBheW1lbnRDYXJkVHlwZSA9IHt9O1xuICBjb25zdCBmb3JtYXRDaXR5OiBzdHJpbmcgPSBwYXJzZVZhckNoYXIoY2l0eSwgMzIpO1xuXG4gIGlmKGZvcm1hdENpdHkpIHtcbiAgICBwYXltZW50Q2FyZC5jaXR5ID0gZm9ybWF0Q2l0eTtcbiAgfVxuXG4gIGNvbnN0IGZvcm1hdENvdW50cnk6IHN0cmluZyA9IHBhcnNlVmFyQ2hhcihjb3VudHJ5LCAzMik7XG5cbiAgaWYoZm9ybWF0Q291bnRyeSkge1xuICAgIHBheW1lbnRDYXJkLmNvdW50cnkgPSBmb3JtYXRDb3VudHJ5O1xuICB9XG5cbiAgY29uc3QgZm9ybWF0RnVsbE5hbWU6IHN0cmluZyA9IHBhcnNlVmFyQ2hhcihmdWxsTmFtZSwgMzIpO1xuXG4gIGlmKGZvcm1hdEZ1bGxOYW1lKSB7XG4gICAgcGF5bWVudENhcmQuZnVsbE5hbWUgPSBmb3JtYXRGdWxsTmFtZTtcbiAgfVxuXG4gIGNvbnN0IGZvcm1hdFN0cmVldDogc3RyaW5nID0gcGFyc2VWYXJDaGFyKHN0cmVldDEsIDMyKTtcblxuICBpZihmb3JtYXRTdHJlZXQpIHtcbiAgICBwYXltZW50Q2FyZC5zdHJlZXQxID0gZm9ybWF0U3RyZWV0O1xuICB9XG5cbiAgY29uc3QgZm9ybWF0U3RhdGU6IHN0cmluZyA9IHBhcnNlVmFyQ2hhcihzdGF0ZSwgMzIpO1xuXG4gIGlmKGZvcm1hdFN0YXRlKSB7XG4gICAgcGF5bWVudENhcmQuc3RhdGUgPSBmb3JtYXRTdGF0ZTtcbiAgfVxuXG4gIGNvbnN0IGZvcm1hdFppcDogc3RyaW5nID0gcGFyc2VWYXJDaGFyKHppcCwgMzIpO1xuXG4gIGlmKGZvcm1hdFppcCkge1xuICAgIHBheW1lbnRDYXJkLnppcCA9IGZvcm1hdFppcDtcbiAgfVxuXG4gIC8vIEdldCBzdHJpcGUgaWRcbiAgcmV0dXJuIGdldFVzZXIoY29udGV4dCwgc2Vzc2lvbklkKVxuICAgIC50aGVuKCh1c2VyOiBVc2VyVHlwZSA9IHt9KSA9PiB7XG4gICAgICAvLyBTdHJpcGVcbiAgICAgIGNvbnN0IHN0cmlwZUNsaWVudCA9IHN0cmlwZShDb25maWcuZ2V0KCdzdHJpcGUudG9rZW4nKSk7XG5cbiAgICAgIHJldHVybiBzdHJpcGVDbGllbnQuY3VzdG9tZXJzXG4gICAgICAgIC5jcmVhdGVTb3VyY2UodXNlci5zdHJpcGVDdXN0b21lcklkLCB7XG4gICAgICAgICAgc291cmNlOiB7XG4gICAgICAgICAgICBhZGRyZXNzX2NpdHk6IGZvcm1hdENpdHksXG4gICAgICAgICAgICBhZGRyZXNzX2NvdW50cnk6IGZvcm1hdENvdW50cnksXG4gICAgICAgICAgICBhZGRyZXNzX2xpbmUxOiBmb3JtYXRTdHJlZXQsXG4gICAgICAgICAgICBhZGRyZXNzX3N0YXRlOiBmb3JtYXRTdGF0ZSxcbiAgICAgICAgICAgIGFkZHJlc3NfemlwOiBmb3JtYXRaaXAsXG4gICAgICAgICAgICBjdXJyZW5jeTogJ3VzZCcsXG4gICAgICAgICAgICBjdmM6IGZvcm1hdEN2YyxcbiAgICAgICAgICAgIGV4cF9tb250aDogZm9ybWF0RXhwTW9udGgsXG4gICAgICAgICAgICBleHBfeWVhcjogZm9ybWF0RXhwWWVhcixcbiAgICAgICAgICAgIG5hbWU6IGZvcm1hdEZ1bGxOYW1lLFxuICAgICAgICAgICAgbnVtYmVyOiBmb3JtYXROdW1iZXIsXG4gICAgICAgICAgICBvYmplY3Q6ICdjYXJkJ1xuICAgICAgICAgIH1cbiAgICAgICAgfSlcbiAgICAgICAgLnRoZW4oKHN0cmlwZU9iaikgPT4ge1xuICAgICAgICAgIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcbiAgICAgICAgICBjb25zdCBpbnNlcnQgPSB7XG4gICAgICAgICAgICAuLi5wYXltZW50Q2FyZCxcbiAgICAgICAgICAgIF9rZXk6IGNyZWF0ZUhhc2goYHVzZXItcGF5bWVudC0ke3Nlc3Npb25JZH1gKSxcbiAgICAgICAgICAgIGFjY291bnROdW1iZXI6IHN0cmlwZU9iai5sYXN0NCxcbiAgICAgICAgICAgIGFkZGVkOiBub3csXG4gICAgICAgICAgICBicmFuZDogc3RyaXBlT2JqLmJyYW5kLFxuICAgICAgICAgICAgY3ZjOiBzdHJpcGVPYmouY3ZjX2NoZWNrLFxuICAgICAgICAgICAgZXhwTW9udGgsXG4gICAgICAgICAgICBleHBZZWFyLFxuICAgICAgICAgICAgbW9kaWZpZWQ6IG5vdyxcbiAgICAgICAgICAgIHVzZXJJZDogc2Vzc2lvbklkXG4gICAgICAgICAgfTtcbiAgICAgICAgICBjb25zdCBpbnNlcnRBcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYElOU0VSVCAke2luc2VydH0gSU4gY3JlZGl0Q2FyZHMgUkVUVVJOIE5FV2A7XG5cbiAgICAgICAgICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGluc2VydEFxbFFyeSlcbiAgICAgICAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgICAgICAgICAgLnRoZW4oKG5ld0NhcmQ6IFBheW1lbnRDYXJkVHlwZSA9IHt9KSA9PiB7XG4gICAgICAgICAgICAgIGlmKCFpc0VtcHR5KG5ld0NhcmQpKSB7XG4gICAgICAgICAgICAgICAgLy8gQWRkIGxpbmtlZCBlZGdlXG4gICAgICAgICAgICAgICAgY29uc3Qge19pZDogY2FyZElkLCBfa2V5OiBjYXJkS2V5fSA9IGNhcmQ7XG4gICAgICAgICAgICAgICAgY29uc3QgZWRnZUNvbGxlY3Rpb24gPSB1c2VEYihkYXRhYmFzZSkuZWRnZUNvbGxlY3Rpb24oJ2hhc1BheW1lbnQnKTtcbiAgICAgICAgICAgICAgICBjb25zdCBlZGdlSWQgPSBjcmVhdGVIYXNoKGBwYXltZW50LSR7Y2FyZEtleX1gKTtcbiAgICAgICAgICAgICAgICBjb25zdCBlZGdlOiBhbnkgPSB7XG4gICAgICAgICAgICAgICAgICBfa2V5OiBlZGdlSWRcbiAgICAgICAgICAgICAgICB9O1xuXG4gICAgICAgICAgICAgICAgcmV0dXJuIGVkZ2VDb2xsZWN0aW9uLnNhdmUoZWRnZSwgYHVzZXJzLyR7c2Vzc2lvbklkfWAsIGNhcmRJZCkudGhlbigoKSA9PiBjYXJkKTtcbiAgICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICAgIHJldHVybiBuZXdDYXJkO1xuICAgICAgICAgICAgfSk7XG4gICAgICAgIH0pXG4gICAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgICAgYWN0aW9uLFxuICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgIGxhYmVsOiAncGF5bWVudF9lcnJvcidcbiAgICAgICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCkpO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IHVwZGF0ZUNyZWRpdENhcmQgPSAoY29udGV4dDogQXBpQ29udGV4dCwgY2FyZDogUGF5bWVudENhcmRUeXBlKTogUHJvbWlzZTxQYXltZW50Q2FyZFR5cGU+ID0+IHtcbiAgY29uc3Qge2RhdGFiYXNlLCB1c2VySWQ6IHNlc3Npb25JZH0gPSBjb250ZXh0O1xuXG4gIGNvbnN0IHtcbiAgICBjaXR5LFxuICAgIGNvdW50cnksXG4gICAgZXhwTW9udGgsXG4gICAgZXhwWWVhcixcbiAgICBmdWxsTmFtZSxcbiAgICBpZCxcbiAgICBzdHJlZXQxLFxuICAgIHN0YXRlLFxuICAgIHppcFxuICB9OiBQYXltZW50Q2FyZFR5cGUgPSBjYXJkO1xuXG4gIGNvbnN0IGZvcm1hdElkOiBzdHJpbmcgPSBwYXJzZUlkKGlkKTtcblxuICBpZihmb3JtYXRJZCkge1xuICAgIHRocm93IG5ldyBVc2VyRXJyb3IoJ3JlcXVpcmVkX2NyZWRpdF9jYXJkX2lkJyk7XG4gIH1cblxuICBjb25zdCBwYXltZW50Q2FyZDogUGF5bWVudENhcmRUeXBlID0ge307XG4gIGNvbnN0IGZvcm1hdEV4cE1vbnRoOiBudW1iZXIgPSBwYXJzZU51bShleHBNb250aCwgMik7XG4gIGNvbnN0IGZvcm1hdEV4cFllYXI6IG51bWJlciA9IHBhcnNlTnVtKGV4cFllYXIsIDIpO1xuICBjb25zdCBmb3JtYXRDaXR5OiBzdHJpbmcgPSBwYXJzZVZhckNoYXIoY2l0eSwgMzIpO1xuICBjb25zdCBmb3JtYXRDb3VudHJ5OiBzdHJpbmcgPSBwYXJzZVZhckNoYXIoY291bnRyeSwgMik7XG4gIGNvbnN0IGZvcm1hdEZ1bGxOYW1lOiBzdHJpbmcgPSBwYXJzZVZhckNoYXIoZnVsbE5hbWUsIDMyKTtcbiAgY29uc3QgZm9ybWF0U3RyZWV0MTogc3RyaW5nID0gcGFyc2VTdHJpbmcoc3RyZWV0MSwgMzIpO1xuICBjb25zdCBmb3JtYXRTdGF0ZTogc3RyaW5nID0gcGFyc2VWYXJDaGFyKHN0YXRlLCAyKTtcbiAgY29uc3QgZm9ybWF0WmlwOiBzdHJpbmcgPSBwYXJzZVZhckNoYXIoemlwLCAxMCk7XG5cbiAgaWYoZm9ybWF0RXhwTW9udGgpIHtcbiAgICBwYXltZW50Q2FyZC5leHBNb250aCA9IGZvcm1hdEV4cE1vbnRoO1xuICB9XG5cbiAgaWYoZm9ybWF0RXhwWWVhcikge1xuICAgIHBheW1lbnRDYXJkLmV4cFllYXIgPSBmb3JtYXRFeHBZZWFyO1xuICB9XG5cbiAgaWYoZm9ybWF0Q2l0eSkge1xuICAgIHBheW1lbnRDYXJkLmNpdHkgPSBmb3JtYXRDaXR5O1xuICB9XG5cbiAgaWYoZm9ybWF0Q291bnRyeSkge1xuICAgIHBheW1lbnRDYXJkLmNvdW50cnkgPSBmb3JtYXRDb3VudHJ5O1xuICB9XG5cbiAgaWYoZm9ybWF0RnVsbE5hbWUpIHtcbiAgICBwYXltZW50Q2FyZC5mdWxsTmFtZSA9IGZvcm1hdEZ1bGxOYW1lO1xuICB9XG5cbiAgaWYoZm9ybWF0U3RyZWV0MSkge1xuICAgIHBheW1lbnRDYXJkLnN0cmVldDEgPSBmb3JtYXRTdHJlZXQxO1xuICB9XG5cbiAgaWYoZm9ybWF0U3RhdGUpIHtcbiAgICBwYXltZW50Q2FyZC5zdGF0ZSA9IGZvcm1hdFN0YXRlO1xuICB9XG5cbiAgaWYoZm9ybWF0WmlwKSB7XG4gICAgcGF5bWVudENhcmQuemlwID0gZm9ybWF0WmlwO1xuICB9XG5cbiAgY29uc3QgdXBkYXRlOiBhbnkgPSBwYXltZW50Q2FyZDtcbiAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBcbiAgICAgIExFVCB1cGRhdGVkQ2FyZCA9IEZJUlNUKFxuICAgICAgICBGT1IgYyBJTiBjcmVkaXRDYXJkc1xuICAgICAgICBGSUxURVIgYy5fa2V5ID09ICR7Zm9ybWF0SWR9ICYmIGMudXNlcklkID09ICR7c2Vzc2lvbklkfVxuICAgICAgICBVUERBVEUgYyBXSVRIICR7dXBkYXRlfSBJTiBjcmVkaXRDYXJkc1xuICAgICAgICBMSU1JVCAxXG4gICAgICAgIFJFVFVSTiBORVdcbiAgICAgIClcbiAgICAgIExFVCB1c2VyID0gRklSU1QoXG4gICAgICAgIEZPUiB1IElOIHVzZXJzXG4gICAgICAgIEZJTFRFUiB1Ll9rZXkgPT0gJHtzZXNzaW9uSWR9XG4gICAgICAgIExJTUlUIDFcbiAgICAgICAgUkVUVVJOIHVcbiAgICAgIClcbiAgICAgIFJFVFVSTiB7dXNlcjogdXNlciwgY2FyZDogdXBkYXRlZENhcmR9YDtcblxuICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAudGhlbigocmVzdWx0cyA9IHtjYXJkOiB7fSwgdXNlcjoge319KSA9PiB7XG4gICAgICBjb25zdCB1cGRhdGVkQ2FyZDogUGF5bWVudENhcmRUeXBlID0gcmVzdWx0cy5jYXJkO1xuICAgICAgY29uc3Qge3VzZXJ9ID0gcmVzdWx0cztcblxuICAgICAgaWYoaXNFbXB0eSh1cGRhdGVkQ2FyZCkpIHtcbiAgICAgICAgdGhyb3cgbmV3IFVzZXJFcnJvcignbm90X2ZvdW5kJyk7XG4gICAgICB9XG5cbiAgICAgIC8vIFN0cmlwZVxuICAgICAgY29uc3Qgc3RyaXBlQ2xpZW50ID0gc3RyaXBlKENvbmZpZy5nZXQoJ3N0cmlwZS50b2tlbicpKTtcblxuICAgICAgcmV0dXJuIHN0cmlwZUNsaWVudC5jdXN0b21lcnNcbiAgICAgICAgLnVwZGF0ZUNhcmQodXNlci5zdHJpcGVDdXN0b21lcklkLCBjYXJkLnN0cmlwZUlkLCB7XG4gICAgICAgICAgYWRkcmVzc19jaXR5OiBmb3JtYXRDaXR5LFxuICAgICAgICAgIGFkZHJlc3NfY291bnRyeTogZm9ybWF0Q291bnRyeSxcbiAgICAgICAgICBhZGRyZXNzX2xpbmUxOiBmb3JtYXRTdHJlZXQxLFxuICAgICAgICAgIGFkZHJlc3Nfc3RhdGU6IGZvcm1hdFN0YXRlLFxuICAgICAgICAgIGFkZHJlc3NfemlwOiBmb3JtYXRaaXAsXG4gICAgICAgICAgZXhwX21vbnRoOiBmb3JtYXRFeHBNb250aCxcbiAgICAgICAgICBleHBfeWVhcjogZm9ybWF0RXhwWWVhcixcbiAgICAgICAgICBuYW1lOiBmb3JtYXRGdWxsTmFtZVxuICAgICAgICB9KVxuICAgICAgICAudGhlbigoKSA9PiBjYXJkKVxuICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgICAgIGNvbnNvbGUubG9nKCdwYXltZW50czo6dXBkYXRlQ2FyZDo6ZXJyb3InLCBlcnJvcik7XG4gICAgICAgICAgdGhyb3cgbmV3IFVzZXJFcnJvcigncGF5bWVudF9lcnJvcicpO1xuICAgICAgICB9KTtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRDcmVkaXRDYXJkcyA9IChjb250ZXh0OiBBcGlDb250ZXh0KTogUHJvbWlzZTxQYXltZW50Q2FyZFR5cGVbXT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRDcmVkaXRDYXJkcyc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgdXNlcklkOiBzZXNzaW9uSWR9ID0gY29udGV4dDtcbiAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBGT1IgYyBJTiBjcmVkaXRDYXJkc1xuICAgICAgICBGSUxURVIgYy51c2VySWQgPT0gJHtzZXNzaW9uSWR9XG4gICAgICAgIFJFVFVSTiBjYDtcblxuICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLmFsbCgpKVxuICAgIC50aGVuKChsaXN0OiBQYXltZW50Q2FyZFR5cGVbXSA9IFtdKSA9PiBsaXN0KVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICBhY3Rpb24sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIGxhYmVsOiAnZGJfZXJyb3InXG4gICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCkpO1xufTtcblxuZXhwb3J0IGNvbnN0IGRlbGV0ZUNyZWRpdENhcmQgPSAoY29udGV4dDogQXBpQ29udGV4dCwgY2FyZElkOiBzdHJpbmcpOiBQcm9taXNlPGJvb2xlYW4+ID0+IHtcbiAgY29uc3Qge2RhdGFiYXNlLCB1c2VySWQ6IHNlc3Npb25JZH0gPSBjb250ZXh0O1xuICBjb25zdCBmb3JtYXRDYXJkSWQ6IHN0cmluZyA9IHBhcnNlSWQoY2FyZElkKTtcbiAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBcbiAgICAgIExFVCBjYXJkID0gRklSU1QoXG4gICAgICAgIEZPUiBjIElOIGNyZWRpdENhcmRzXG4gICAgICAgIEZJTFRFUiBjLl9rZXkgPT0gJHtmb3JtYXRDYXJkSWR9ICYmIGMudXNlcklkID09ICR7c2Vzc2lvbklkfVxuICAgICAgICBMSU1JVCAxXG4gICAgICAgIFJFTU9WRSBjIElOIGNyZWRpdENhcmRzXG4gICAgICAgIFJFVFVSTiBPTERcbiAgICAgIClcbiAgICAgIExFVCB1c2VyID0gRklSU1QoXG4gICAgICAgIEZPUiB1IElOIHVzZXJzXG4gICAgICAgIEZJTFRFUiB1Ll9rZXkgPT0gJHtzZXNzaW9uSWR9XG4gICAgICAgIExJTUlUIDFcbiAgICAgICAgUkVUVVJOIHVcbiAgICAgIClcbiAgICAgIFJFVFVSTiB7dXNlcjogdXNlciwgY2FyZDogY2FyZH1gO1xuXG4gIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgIC50aGVuKChyZXN1bHQgPSB7Y2FyZDoge30sIHVzZXI6IHt9fSkgPT4ge1xuICAgICAgaWYoaXNFbXB0eShyZXN1bHQpKSB7XG4gICAgICAgIHJldHVybiBmYWxzZTtcbiAgICAgIH1cblxuICAgICAgY29uc3Qge2NhcmQsIHVzZXJ9ID0gcmVzdWx0O1xuICAgICAgY29uc3Qge19rZXk6IGNhcmRLZXl9ID0gY2FyZDtcblxuICAgICAgLy8gUmVtb3ZlIGxpbmtlZCBlZGdlc1xuICAgICAgY29uc3QgZWRnZUNvbGxlY3Rpb24gPSB1c2VEYihkYXRhYmFzZSkuZWRnZUNvbGxlY3Rpb24oJ2hhc1BheW1lbnQnKTtcblxuICAgICAgcmV0dXJuIGVkZ2VDb2xsZWN0aW9uLm91dEVkZ2VzKGNhcmRLZXkpXG4gICAgICAgIC50aGVuKChlZGdlcykgPT4ge1xuICAgICAgICAgIGlmKGVkZ2VzLmxlbmd0aCkge1xuICAgICAgICAgICAgcmV0dXJuIFByb21pc2UuYWxsKFxuICAgICAgICAgICAgICBlZGdlcy5tYXAoKGVkZ2UpID0+IHtcbiAgICAgICAgICAgICAgICBjb25zdCB7X2tleTogZWRnZUtleX0gPSBlZGdlO1xuICAgICAgICAgICAgICAgIGNvbnN0IHJlbW92ZUFxbFFyeTogQXFsUXVlcnkgPSBhcWxgUkVNT1ZFIHtfa2V5OiR7ZWRnZUtleX19IElOIGhhc1BheW1lbnRgO1xuICAgICAgICAgICAgICAgIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkocmVtb3ZlQXFsUXJ5KTtcbiAgICAgICAgICAgICAgfSkpXG4gICAgICAgICAgICAgIC50aGVuKCgpID0+IHtcbiAgICAgICAgICAgICAgICAvLyBTdHJpcGVcbiAgICAgICAgICAgICAgICBjb25zdCBzdHJpcGVDbGllbnQgPSBzdHJpcGUoQ29uZmlnLmdldCgnc3RyaXBlLnRva2VuJykpO1xuXG4gICAgICAgICAgICAgICAgcmV0dXJuIHN0cmlwZUNsaWVudC5jdXN0b21lcnNcbiAgICAgICAgICAgICAgICAgIC5kZWxldGVDYXJkKHVzZXIuc3RyaXBlQ3VzdG9tZXJJZCwgY2FyZC5zdHJpcGVJZClcbiAgICAgICAgICAgICAgICAgIC50aGVuKCgpID0+IHRydWUpXG4gICAgICAgICAgICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgICAgICAgICAgICAgICBjb25zb2xlLmxvZygncGF5bWVudHM6OmRlbGV0ZUNhcmQ6OmVycm9yJywgZXJyb3IpO1xuICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgVXNlckVycm9yKCdwYXltZW50X2Vycm9yJyk7XG4gICAgICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgICAgfSk7XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgcmV0dXJuIGZhbHNlO1xuICAgICAgICB9KTtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBkZWxldGVQYXltZW50QWNjb3VudEJhbmsgPSAoY29udGV4dDogQXBpQ29udGV4dCwgYmFua0lkOiBzdHJpbmcpOiBQcm9taXNlPGJvb2xlYW4+ID0+IHtcbiAgY29uc3Qge2RhdGFiYXNlLCB1c2VySWQ6IHNlc3Npb25JZH0gPSBjb250ZXh0O1xuXG4gIC8vIENsZWFuIGRiXG4gIGNvbnN0IHVwZGF0ZTogVXNlclR5cGUgPSB7XG4gICAgYmFua0FjY291bnQ6ICcnLFxuICAgIGJhbmtGdWxsTmFtZTogJycsXG4gICAgYmFua0lkOiAnJyxcbiAgICBiYW5rUm91dGluZzogJycsXG4gICAgbW9kaWZpZWQ6IERhdGUubm93KClcbiAgfTtcbiAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBVUERBVEUgJHtzZXNzaW9uSWR9IFdJVEggJHt1cGRhdGV9IElOIHVzZXJzIExJTUlUIDEgUkVUVVJOIE5FV2A7XG5cbiAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgLnRoZW4oKHVzZXI6IFVzZXJUeXBlID0ge30pID0+IHtcbiAgICAgIC8vIFN0cmlwZVxuICAgICAgY29uc3Qgc3RyaXBlQ2xpZW50ID0gc3RyaXBlKENvbmZpZy5nZXQoJ3N0cmlwZS50b2tlbicpKTtcblxuICAgICAgcmV0dXJuIHN0cmlwZUNsaWVudC5jdXN0b21lcnNcbiAgICAgICAgLmRlbGV0ZVNvdXJjZSh1c2VyLnN0cmlwZUFjY291bnRJZCwgYmFua0lkKVxuICAgICAgICAudGhlbigocmVzcG9uc2UgPSB7ZGVsZXRlZDogZmFsc2V9KSA9PiByZXNwb25zZS5kZWxldGVkKVxuICAgICAgICAuY2F0Y2goKCkgPT4gUHJvbWlzZS5yZXNvbHZlKGZhbHNlKSk7XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgY3JlYXRlUGF5bWVudFRyYW5zZmVyID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIHRyYW5zZmVyOiBQYXltZW50VHJhbnNmZXIpOiBQcm9taXNlPFBheW1lbnRUcmFuc2Zlcj4gPT4ge1xuICBjb25zdCB7ZGF0YWJhc2UsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG4gIGNvbnN0IHthbW91bnQsIGN1cnJlbmN5fSA9IHRyYW5zZmVyO1xuICBjb25zdCBmb3JtYXRBbW91bnQ6IG51bWJlciA9IHBhcnNlTnVtKGFtb3VudCk7XG4gIGNvbnN0IGZvcm1hdEN1cnJlbmN5OiBzdHJpbmcgPSBwYXJzZUNoYXIoY3VycmVuY3ksIDMsICdVU0QnKS50b1VwcGVyQ2FzZSgpO1xuXG4gIHJldHVybiBnZXRVc2VyKGNvbnRleHQsIHNlc3Npb25JZClcbiAgICAudGhlbigodXNlcjogVXNlclR5cGUpID0+IHtcbiAgICAgIC8vIFN0cmlwZVxuICAgICAgY29uc3Qgc3RyaXBlQ2xpZW50ID0gc3RyaXBlKENvbmZpZy5nZXQoJ3N0cmlwZS50b2tlbicpKTtcblxuICAgICAgcmV0dXJuIHN0cmlwZUNsaWVudC50cmFuc2ZlcnNcbiAgICAgICAgLmNyZWF0ZSh7XG4gICAgICAgICAgYW1vdW50OiBmb3JtYXRBbW91bnQsXG4gICAgICAgICAgY3VycmVuY3k6IGZvcm1hdEN1cnJlbmN5LFxuICAgICAgICAgIGRlc3RpbmF0aW9uOiB1c2VyLnN0cmlwZUFjY291bnRJZFxuICAgICAgICB9KVxuICAgICAgICAudGhlbigoc3RyaXBlVHJhbnNmZXIpID0+IHtcbiAgICAgICAgICBjb25zb2xlLmxvZyhzdHJpcGVUcmFuc2Zlcik7XG4gICAgICAgICAgY29uc3Qgbm93OiBudW1iZXIgPSBEYXRlLm5vdygpO1xuICAgICAgICAgIGNvbnN0IGluc2VydDogUGF5bWVudFRyYW5zZmVyID0ge1xuICAgICAgICAgICAgYWRkZWQ6IG5vdyxcbiAgICAgICAgICAgIGFtb3VudDogZm9ybWF0QW1vdW50LFxuICAgICAgICAgICAgY3VycmVuY3k6IGZvcm1hdEN1cnJlbmN5LFxuICAgICAgICAgICAgbW9kaWZpZWQ6IG5vdyxcbiAgICAgICAgICAgIHVzZXJJZDogc2Vzc2lvbklkXG4gICAgICAgICAgfTtcbiAgICAgICAgICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYElOU0VSVCAke2luc2VydH0gSU4gdHJhbnNmZXJzIFJFVFVSTiBORVdgO1xuXG4gICAgICAgICAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgICAgICAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAgICAgICAgIC50aGVuKChuZXdUcmFuc2ZlcjogUGF5bWVudFRyYW5zZmVyKSA9PiBuZXdUcmFuc2Zlcik7XG4gICAgICAgIH0pO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGNyZWF0ZVBheW1lbnRIb2xkID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIHBheW1lbnQ6IFBheW1lbnRDaGFyZ2UpOiBQcm9taXNlPFBheW1lbnRDaGFyZ2U+ID0+IHtcbiAgY29uc3Qge2RhdGFiYXNlLCB1c2VySWQ6IHNlc3Npb25JZH0gPSBjb250ZXh0O1xuICBjb25zdCB7YW1vdW50LCBjYXB0dXJlLCBjYXJkSWQsIGN1cnJlbmN5LCBkZXNjcmlwdGlvbn0gPSBwYXltZW50O1xuICBjb25zdCBmb3JtYXRDdXJyZW5jeSA9IHBhcnNlQ2hhcihjdXJyZW5jeSwgMywgJ1VTRCcpLnRvVXBwZXJDYXNlKCk7XG5cbiAgY29uc3Qgc3RyaXBlQ2xpZW50ID0gc3RyaXBlKENvbmZpZy5nZXQoJ3N0cmlwZS50b2tlbicpKTtcblxuICByZXR1cm4gc3RyaXBlQ2xpZW50LmNoYXJnZXNcbiAgICAuY3JlYXRlKHtcbiAgICAgIGFtb3VudCxcbiAgICAgIGNhcHR1cmUsXG4gICAgICBjdXJyZW5jeTogZm9ybWF0Q3VycmVuY3ksXG4gICAgICBkZXNjcmlwdGlvbixcbiAgICAgIHNvdXJjZTogY2FyZElkXG4gICAgfSlcbiAgICAudGhlbigoc3RyaXBlQ2hhcmdlKSA9PiB7XG4gICAgICBjb25zdCBub3c6IG51bWJlciA9IERhdGUubm93KCk7XG4gICAgICBjb25zdCBpbnNlcnQ6IFBheW1lbnRDaGFyZ2UgPSB7XG4gICAgICAgIGFkZGVkOiBub3csXG4gICAgICAgIGFtb3VudCxcbiAgICAgICAgY2FwdHVyZSxcbiAgICAgICAgY2FyZElkLFxuICAgICAgICBjaGFyZ2VGYWlsQ29kZTogc3RyaXBlQ2hhcmdlLmZhaWx1cmVfY29kZSxcbiAgICAgICAgY2hhcmdlRmFpbE1zZzogc3RyaXBlQ2hhcmdlLmZhaWx1cmVfbWVzc2FnZSxcbiAgICAgICAgY2hhcmdlSWQ6IHN0cmlwZUNoYXJnZS5pZCxcbiAgICAgICAgY2hhcmdlU3RhdHVzOiBzdHJpcGVDaGFyZ2Uuc3RhdHVzLFxuICAgICAgICBjdXJyZW5jeTogZm9ybWF0Q3VycmVuY3ksXG4gICAgICAgIGRlc2NyaXB0aW9uLFxuICAgICAgICBtb2RpZmllZDogbm93LFxuICAgICAgICB1c2VySWQ6IHNlc3Npb25JZFxuICAgICAgfTtcbiAgICAgIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgSU5TRVJUICR7aW5zZXJ0fSBJTiBwYXltZW50cyBSRVRVUk4gTkVXYDtcblxuICAgICAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgICAgICAudGhlbigobmV3UGF5bWVudDogUGF5bWVudENoYXJnZSkgPT4gbmV3UGF5bWVudCk7XG4gICAgfSlcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgY29uc29sZS5sb2coJ3BheW1lbnRzOjpjcmVhdGVIb2xkOjplcnJvcicsIGVycm9yKTtcbiAgICAgIHRocm93IG5ldyBVc2VyRXJyb3IoJ3BheW1lbnRfZXJyb3InKTtcbiAgICB9KTtcbn07XG4iXX0=