@nlabs/reaktor 0.3.0 → 0.4.0

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