@nlabs/reaktor 0.1.2 → 0.1.4

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