@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,337 @@
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.slice");
8
+
9
+ require("core-js/modules/es.date.now");
10
+
11
+ require("core-js/modules/es.date.to-string");
12
+
13
+ require("core-js/modules/es.function.name");
14
+
15
+ require("core-js/modules/es.object.define-properties");
16
+
17
+ require("core-js/modules/es.object.freeze");
18
+
19
+ require("core-js/modules/es.object.to-string");
20
+
21
+ require("core-js/modules/es.promise");
22
+
23
+ Object.defineProperty(exports, "__esModule", {
24
+ value: true
25
+ });
26
+ exports.deleteSubscription = exports.addSubscription = exports.addPlan = exports.getSubscription = exports.getPlanList = void 0;
27
+
28
+ var _utils = require("@nlabs/utils");
29
+
30
+ var _arangojs = require("arangojs");
31
+
32
+ var _luxon = require("luxon");
33
+
34
+ var _stripe = _interopRequireDefault(require("stripe"));
35
+
36
+ var _config = require("../config");
37
+
38
+ var _utils2 = require("../utils");
39
+
40
+ var _users = require("./users");
41
+
42
+ function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { "default": obj }; }
43
+
44
+ function _templateObject3() {
45
+ var data = _taggedTemplateLiteral(["UPDATE s WITH ", " IN subscriptions"]);
46
+
47
+ _templateObject3 = function _templateObject3() {
48
+ return data;
49
+ };
50
+
51
+ return data;
52
+ }
53
+
54
+ function _templateObject2() {
55
+ var data = _taggedTemplateLiteral(["INSERT ", " IN plans RETURN NEW"]);
56
+
57
+ _templateObject2 = function _templateObject2() {
58
+ return data;
59
+ };
60
+
61
+ return data;
62
+ }
63
+
64
+ function _templateObject() {
65
+ var data = _taggedTemplateLiteral(["INSERT ", " IN plans RETURN NEW"]);
66
+
67
+ _templateObject = function _templateObject() {
68
+ return data;
69
+ };
70
+
71
+ return data;
72
+ }
73
+
74
+ function _taggedTemplateLiteral(strings, raw) { if (!raw) { raw = strings.slice(0); } return Object.freeze(Object.defineProperties(strings, { raw: { value: Object.freeze(raw) } })); }
75
+
76
+ var eventCategory = 'subscription';
77
+ var apiVersion = '2020-03-02';
78
+
79
+ var getPlanList = function getPlanList(context) {
80
+ var from = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 0;
81
+ var to = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : 30;
82
+ var action = 'getList';
83
+ var database = context.database;
84
+ var limit = (0, _utils2.getLimit)(from, to);
85
+ var aqlQry = "FOR p IN plans\n ".concat(limit.aql, "\n SORT p.amount\n RETURN p");
86
+ return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
87
+ return cursor.all();
88
+ }).then(function () {
89
+ var list = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [];
90
+ return list;
91
+ })["catch"](function (error) {
92
+ return (0, _utils2.logError)({
93
+ action: action,
94
+ category: eventCategory,
95
+ label: 'db_error'
96
+ }, error, {}).then(function () {
97
+ return null;
98
+ })["catch"](function (error) {
99
+ return Promise.reject(error);
100
+ });
101
+ });
102
+ };
103
+
104
+ exports.getPlanList = getPlanList;
105
+
106
+ var getSubscription = function getSubscription(context) {
107
+ var action = 'getItem';
108
+ var database = context.database,
109
+ sessionId = context.userId;
110
+ var aqlQry = "FOR s IN subscriptions\n FILTER s.userId == ".concat(sessionId, " && s.cancelled > 0\n LET plan = FIRST(\n FOR p IN plans\n FILTER p._key == s.planId\n )\n LIMIT 1\n RETURN MERGE(s, {plan: plan})");
111
+ return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
112
+ return cursor.next();
113
+ }).then(function () {
114
+ var subscription = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
115
+ return subscription;
116
+ })["catch"](function (error) {
117
+ return (0, _utils2.logError)({
118
+ action: action,
119
+ category: eventCategory,
120
+ label: 'db_error'
121
+ }, error, {}).then(function () {
122
+ return null;
123
+ })["catch"](function (error) {
124
+ return Promise.reject(error);
125
+ });
126
+ });
127
+ };
128
+
129
+ exports.getSubscription = getSubscription;
130
+
131
+ var addPlan = function addPlan(context, item) {
132
+ var action = 'addPlan';
133
+ var database = context.database,
134
+ sessionId = context.userId,
135
+ userAccess = context.userAccess;
136
+ var isAdmin = userAccess > 2;
137
+
138
+ if (!isAdmin) {
139
+ return (0, _utils2.logException)({
140
+ action: action,
141
+ category: eventCategory,
142
+ label: 'unauthorized',
143
+ value: 'invalid_session'
144
+ }, context).then(function () {
145
+ return null;
146
+ });
147
+ }
148
+
149
+ var now = Date.now();
150
+ var formatId = (0, _utils.parseId)(item.id);
151
+ var planId = formatId === '' ? (0, _utils.createHash)("tag-".concat(sessionId)) : formatId;
152
+ var amount = item.amount,
153
+ _item$currency = item.currency,
154
+ currency = _item$currency === void 0 ? 'USD' : _item$currency,
155
+ description = item.description,
156
+ interval = item.interval,
157
+ intervalCount = item.intervalCount,
158
+ name = item.name;
159
+ var formatAmount = (0, _utils.parseNum)(amount);
160
+ var formatInterval = (0, _utils.parseChar)(interval, 5).toLowerCase();
161
+ var formatIntervalCnt = (0, _utils.parseNum)(intervalCount, 5);
162
+ var formatName = (0, _utils.parseVarChar)(name, 32);
163
+ var formatDesc = (0, _utils.parseVarChar)(description, 32);
164
+ var insert = {
165
+ _key: planId,
166
+ added: now,
167
+ amount: formatAmount,
168
+ currency: currency,
169
+ description: formatDesc,
170
+ interval: formatInterval,
171
+ intervalCount: formatIntervalCnt,
172
+ modified: now,
173
+ name: formatName
174
+ };
175
+ var aqlQry = (0, _arangojs.aql)(_templateObject(), insert);
176
+ return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
177
+ return cursor.next();
178
+ }).then(function () {
179
+ var plan = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
180
+ // Stripe
181
+ var stripeClient = new _stripe["default"](_config.Config.get('stripe.token'), {
182
+ apiVersion: apiVersion,
183
+ typescript: true
184
+ });
185
+ var stripePlan = {
186
+ amount: formatAmount * 100,
187
+ currency: currency,
188
+ id: planId,
189
+ interval: formatInterval,
190
+ interval_count: formatIntervalCnt,
191
+ metadata: {},
192
+ name: formatName,
193
+ statement_descriptor: formatDesc
194
+ };
195
+ return stripeClient.plans.create(stripePlan).then(function () {
196
+ return plan;
197
+ });
198
+ })["catch"](function (error) {
199
+ return (0, _utils2.logError)({
200
+ action: action,
201
+ category: eventCategory,
202
+ label: 'db_error'
203
+ }, error, {}).then(function () {
204
+ return null;
205
+ })["catch"](function (error) {
206
+ return Promise.reject(error);
207
+ });
208
+ });
209
+ };
210
+
211
+ exports.addPlan = addPlan;
212
+
213
+ var addSubscription = function addSubscription(context, item) {
214
+ var action = 'addSubscription';
215
+ var database = context.database,
216
+ sessionId = context.userId,
217
+ userAccess = context.userAccess;
218
+ var isAdmin = userAccess > 2;
219
+
220
+ if (!isAdmin) {
221
+ return (0, _utils2.logException)({
222
+ action: action,
223
+ category: eventCategory,
224
+ label: 'unauthorized',
225
+ value: 'invalid_session'
226
+ }, context).then(function () {
227
+ return null;
228
+ });
229
+ }
230
+
231
+ var now = Date.now();
232
+ var formatId = (0, _utils.parseId)(item.id);
233
+ var subscriptionId = formatId === '' ? (0, _utils.createHash)("tag-".concat(sessionId)) : formatId;
234
+ var planId = item.planId,
235
+ trialEnd = item.trialEnd,
236
+ trialPeriod = item.trialPeriod;
237
+ var formatPlanId = (0, _utils.parseId)(planId);
238
+ var formatTrialPeriod = 0;
239
+ var formatTrialEnd = trialEnd || Date.now();
240
+
241
+ if (formatTrialPeriod) {
242
+ formatTrialEnd = _luxon.DateTime.local().plus({
243
+ days: trialPeriod
244
+ }).toMillis();
245
+ }
246
+
247
+ return (0, _users.getUser)(context, sessionId).then(function (user) {
248
+ var insert = {
249
+ _key: subscriptionId,
250
+ added: now,
251
+ cancelDate: 0,
252
+ modified: now,
253
+ planId: formatPlanId,
254
+ trialEnd: formatTrialEnd,
255
+ userId: sessionId
256
+ };
257
+ var aqlQry = (0, _arangojs.aql)(_templateObject2(), insert);
258
+ return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
259
+ return cursor.next();
260
+ }).then(function () {
261
+ var subscription = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
262
+ // Stripe
263
+ var stripeClient = new _stripe["default"](_config.Config.get('stripe.token'), {
264
+ apiVersion: apiVersion,
265
+ typescript: true
266
+ });
267
+ return stripeClient.subscriptions.create({
268
+ customer: user.stripeCustomerId,
269
+ items: [{
270
+ plan: formatPlanId
271
+ }],
272
+ metadata: {
273
+ userId: sessionId
274
+ },
275
+ trial_end: formatTrialEnd
276
+ }).then(function () {
277
+ return subscription;
278
+ });
279
+ })["catch"](function (error) {
280
+ return (0, _utils2.logError)({
281
+ action: action,
282
+ category: eventCategory,
283
+ label: 'db_error'
284
+ }, error, {}).then(function () {
285
+ return null;
286
+ })["catch"](function (error) {
287
+ return Promise.reject(error);
288
+ });
289
+ });
290
+ });
291
+ };
292
+
293
+ exports.addSubscription = addSubscription;
294
+
295
+ var deleteSubscription = function deleteSubscription(context) {
296
+ var action = 'deleteSubscription';
297
+ var database = context.database;
298
+ var now = Date.now();
299
+ return getSubscription(context).then(function (subscription) {
300
+ // Stripe
301
+ var stripeClient = new _stripe["default"](_config.Config.get('stripe.token'), {
302
+ apiVersion: apiVersion,
303
+ typescript: true
304
+ });
305
+ return stripeClient.subscriptions.del(subscription.transactionId).then(function (_ref) {
306
+ var cancelDate = _ref.canceled_at;
307
+
308
+ // Make sure we cancelled on Stripe before updating the db
309
+ if (cancelDate) {
310
+ var update = {
311
+ cancelDate: cancelDate,
312
+ modified: now,
313
+ planId: ''
314
+ };
315
+ var aqlQry = (0, _arangojs.aql)(_templateObject3(), update);
316
+ return (0, _utils2.useDb)(database).query(aqlQry).then(function () {
317
+ return true;
318
+ })["catch"](function (error) {
319
+ return (0, _utils2.logError)({
320
+ action: action,
321
+ category: eventCategory,
322
+ label: 'db_error'
323
+ }, error, {}).then(function () {
324
+ return null;
325
+ })["catch"](function (error) {
326
+ return Promise.reject(error);
327
+ });
328
+ });
329
+ }
330
+
331
+ return false;
332
+ });
333
+ });
334
+ };
335
+
336
+ exports.deleteSubscription = deleteSubscription;
337
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kYXRhL3N1YnNjcmlwdGlvbi50cyJdLCJuYW1lcyI6WyJldmVudENhdGVnb3J5IiwiYXBpVmVyc2lvbiIsImdldFBsYW5MaXN0IiwiY29udGV4dCIsImZyb20iLCJ0byIsImFjdGlvbiIsImRhdGFiYXNlIiwibGltaXQiLCJhcWxRcnkiLCJhcWwiLCJxdWVyeSIsInRoZW4iLCJjdXJzb3IiLCJhbGwiLCJsaXN0IiwiZXJyb3IiLCJjYXRlZ29yeSIsImxhYmVsIiwiUHJvbWlzZSIsInJlamVjdCIsImdldFN1YnNjcmlwdGlvbiIsInNlc3Npb25JZCIsInVzZXJJZCIsIm5leHQiLCJzdWJzY3JpcHRpb24iLCJhZGRQbGFuIiwiaXRlbSIsInVzZXJBY2Nlc3MiLCJpc0FkbWluIiwidmFsdWUiLCJub3ciLCJEYXRlIiwiZm9ybWF0SWQiLCJpZCIsInBsYW5JZCIsImFtb3VudCIsImN1cnJlbmN5IiwiZGVzY3JpcHRpb24iLCJpbnRlcnZhbCIsImludGVydmFsQ291bnQiLCJuYW1lIiwiZm9ybWF0QW1vdW50IiwiZm9ybWF0SW50ZXJ2YWwiLCJ0b0xvd2VyQ2FzZSIsImZvcm1hdEludGVydmFsQ250IiwiZm9ybWF0TmFtZSIsImZvcm1hdERlc2MiLCJpbnNlcnQiLCJfa2V5IiwiYWRkZWQiLCJtb2RpZmllZCIsInBsYW4iLCJzdHJpcGVDbGllbnQiLCJTdHJpcGUiLCJDb25maWciLCJnZXQiLCJ0eXBlc2NyaXB0Iiwic3RyaXBlUGxhbiIsImludGVydmFsX2NvdW50IiwibWV0YWRhdGEiLCJzdGF0ZW1lbnRfZGVzY3JpcHRvciIsInBsYW5zIiwiY3JlYXRlIiwiYWRkU3Vic2NyaXB0aW9uIiwic3Vic2NyaXB0aW9uSWQiLCJ0cmlhbEVuZCIsInRyaWFsUGVyaW9kIiwiZm9ybWF0UGxhbklkIiwiZm9ybWF0VHJpYWxQZXJpb2QiLCJmb3JtYXRUcmlhbEVuZCIsIkRhdGVUaW1lIiwibG9jYWwiLCJwbHVzIiwiZGF5cyIsInRvTWlsbGlzIiwidXNlciIsImNhbmNlbERhdGUiLCJzdWJzY3JpcHRpb25zIiwiY3VzdG9tZXIiLCJzdHJpcGVDdXN0b21lcklkIiwiaXRlbXMiLCJ0cmlhbF9lbmQiLCJkZWxldGVTdWJzY3JpcHRpb24iLCJkZWwiLCJ0cmFuc2FjdGlvbklkIiwiY2FuY2VsZWRfYXQiLCJ1cGRhdGUiXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUlBOztBQUNBOztBQUdBOztBQUNBOztBQUVBOztBQUlBOztBQUNBOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFFQSxJQUFNQSxhQUFxQixHQUFHLGNBQTlCO0FBQ0EsSUFBTUMsVUFBZSxHQUFHLFlBQXhCOztBQUVPLElBQU1DLFdBQVcsR0FBRyxTQUFkQSxXQUFjLENBQUNDLE9BQUQsRUFBb0Y7QUFBQSxNQUE5REMsSUFBOEQsdUVBQS9DLENBQStDO0FBQUEsTUFBNUNDLEVBQTRDLHVFQUEvQixFQUErQjtBQUM3RyxNQUFNQyxNQUFjLEdBQUcsU0FBdkI7QUFENkcsTUFFdEdDLFFBRnNHLEdBRTFGSixPQUYwRixDQUV0R0ksUUFGc0c7QUFHN0csTUFBTUMsS0FBb0IsR0FBRyxzQkFBU0osSUFBVCxFQUFlQyxFQUFmLENBQTdCO0FBQ0EsTUFBTUksTUFBYyxtQ0FDZEQsS0FBSyxDQUFDRSxHQURRLDBDQUFwQjtBQUtBLFNBQU8sbUJBQU1ILFFBQU4sRUFBZ0JJLEtBQWhCLENBQXNCRixNQUF0QixFQUNKRyxJQURJLENBQ0MsVUFBQ0MsTUFBRDtBQUFBLFdBQXlCQSxNQUFNLENBQUNDLEdBQVAsRUFBekI7QUFBQSxHQURELEVBRUpGLElBRkksQ0FFQztBQUFBLFFBQUNHLElBQUQsdUVBQVEsRUFBUjtBQUFBLFdBQWVBLElBQWY7QUFBQSxHQUZELFdBR0UsVUFBQ0MsS0FBRDtBQUFBLFdBQWtCLHNCQUFTO0FBQ2hDVixNQUFBQSxNQUFNLEVBQU5BLE1BRGdDO0FBRWhDVyxNQUFBQSxRQUFRLEVBQUVqQixhQUZzQjtBQUdoQ2tCLE1BQUFBLEtB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LENBQUN5QixhQUFiLENBQ0pNLEdBREksQ0FDQTNELFlBQVksQ0FBQzRELGFBRGIsRUFFSnpFLElBRkksQ0FFQyxnQkFBK0I7QUFBQSxVQUFoQmlFLFVBQWdCLFFBQTdCUyxXQUE2Qjs7QUFDbkM7QUFDQSxVQUFHVCxVQUFILEVBQWU7QUFDYixZQUFNVSxNQUEyQixHQUFHO0FBQ2xDVixVQUFBQSxVQUFVLEVBQVZBLFVBRGtDO0FBRWxDMUIsVUFBQUEsUUFBUSxFQUFFcEIsR0FGd0I7QUFHbENJLFVBQUFBLE1BQU0sRUFBRTtBQUgwQixTQUFwQztBQUtBLFlBQU0xQixNQUFnQixPQUFHQyxhQUFILHNCQUF1QjZFLE1BQXZCLENBQXRCO0FBRUEsZUFBTyxtQkFBTWhGLFFBQU4sRUFBZ0JJLEtBQWhCLENBQXNCRixNQUF0QixFQUNKRyxJQURJLENBQ0M7QUFBQSxpQkFBTSxJQUFOO0FBQUEsU0FERCxXQUVFLFVBQUNJLEtBQUQ7QUFBQSxpQkFBa0Isc0JBQVM7QUFDaENWLFlBQUFBLE1BQU0sRUFBTkEsTUFEZ0M7QUFFaENXLFlBQUFBLFFBQVEsRUFBRWpCLGFBRnNCO0FBR2hDa0IsWUFBQUEsS0FBSyxFQUFFO0FBSHlCLFdBQVQsRUFJdEJGLEtBSnNCLEVBSWYsRUFKZSxFQUlYSixJQUpXLENBSU47QUFBQSxtQkFBTSxJQUFOO0FBQUEsV0FKTSxXQUlZLFVBQUNJLEtBQUQ7QUFBQSxtQkFBV0csT0FBTyxDQUFDQyxNQUFSLENBQWVKLEtBQWYsQ0FBWDtBQUFBLFdBSlosQ0FBbEI7QUFBQSxTQUZGLENBQVA7QUFPRDs7QUFDRCxhQUFPLEtBQVA7QUFDRCxLQXJCSSxDQUFQO0FBc0JELEdBMUJJLENBQVA7QUEyQkQsQ0FoQ00iLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIENvcHlyaWdodCAoYykgMjAxOS1QcmVzZW50LCBOaXRyb2dlbiBMYWJzLCBJbmMuXG4gKiBDb3B5cmlnaHRzIGxpY2Vuc2VkIHVuZGVyIHRoZSBNSVQgTGljZW5zZS4gU2VlIHRoZSBhY2NvbXBhbnlpbmcgTElDRU5TRSBmaWxlIGZvciB0ZXJtcy5cbiAqL1xuaW1wb3J0IHtjcmVhdGVIYXNoLCBwYXJzZUNoYXIsIHBhcnNlSWQsIHBhcnNlTnVtLCBwYXJzZVZhckNoYXJ9IGZyb20gJ0BubGFicy91dGlscyc7XG5pbXBvcnQge2FxbH0gZnJvbSAnYXJhbmdvanMnO1xuaW1wb3J0IHtBcWxRdWVyeX0gZnJvbSAnYXJhbmdvanMvYXFsJztcbmltcG9ydCB7QXJyYXlDdXJzb3J9IGZyb20gJ2FyYW5nb2pzL2N1cnNvcic7XG5pbXBvcnQge0RhdGVUaW1lfSBmcm9tICdsdXhvbic7XG5pbXBvcnQgU3RyaXBlIGZyb20gJ3N0cmlwZSc7XG5cbmltcG9ydCB7Q29uZmlnfSBmcm9tICcuLi9jb25maWcnO1xuaW1wb3J0IHtBcmFuZ29EQkxpbWl0fSBmcm9tICcuLi90eXBlcy9hcmFuZ29kYic7XG5pbXBvcnQge0FwaUNvbnRleHR9IGZyb20gJy4uL3R5cGVzL2F1dGgnO1xuaW1wb3J0IHtQYXltZW50SW50ZXJ2YWwsIFBheW1lbnRQbGFuLCBQYXltZW50U3Vic2NyaXB0aW9ufSBmcm9tICcuLi90eXBlcy9wYXltZW50cyc7XG5pbXBvcnQge2dldExpbWl0LCBsb2dFcnJvciwgbG9nRXhjZXB0aW9uLCB1c2VEYn0gZnJvbSAnLi4vdXRpbHMnO1xuaW1wb3J0IHtnZXRVc2VyfSBmcm9tICcuL3VzZXJzJztcblxuY29uc3QgZXZlbnRDYXRlZ29yeTogc3RyaW5nID0gJ3N1YnNjcmlwdGlvbic7XG5jb25zdCBhcGlWZXJzaW9uOiBhbnkgPSAnMjAyMC0wMy0wMic7XG5cbmV4cG9ydCBjb25zdCBnZXRQbGFuTGlzdCA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBmcm9tOiBudW1iZXIgPSAwLCB0bzogbnVtYmVyID0gMzApOiBQcm9taXNlPFBheW1lbnRQbGFuW10+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnZ2V0TGlzdCc7XG4gIGNvbnN0IHtkYXRhYmFzZX0gPSBjb250ZXh0O1xuICBjb25zdCBsaW1pdDogQXJhbmdvREJMaW1pdCA9IGdldExpbWl0KGZyb20sIHRvKTtcbiAgY29uc3QgYXFsUXJ5OiBzdHJpbmcgPSBgRk9SIHAgSU4gcGxhbnNcbiAgICAgICR7bGltaXQuYXFsfVxuICAgICAgU09SVCBwLmFtb3VudFxuICAgICAgUkVUVVJOIHBgO1xuXG4gIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IuYWxsKCkpXG4gICAgLnRoZW4oKGxpc3QgPSBbXSkgPT4gbGlzdClcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgYWN0aW9uLFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICBsYWJlbDogJ2RiX2Vycm9yJ1xuICAgIH0sIGVycm9yLCB7fSkudGhlbigoKSA9PiBudWxsKS5jYXRjaCgoZXJyb3IpID0+IFByb21pc2UucmVqZWN0KGVycm9yKSkpO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldFN1YnNjcmlwdGlvbiA9IChjb250ZXh0OiBBcGlDb250ZXh0KTogUHJvbWlzZTxQYXltZW50U3Vic2NyaXB0aW9uPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2dldEl0ZW0nO1xuICBjb25zdCB7ZGF0YWJhc2UsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG4gIGNvbnN0IGFxbFFyeTogc3RyaW5nID0gYEZPUiBzIElOIHN1YnNjcmlwdGlvbnNcbiAgICAgIEZJTFRFUiBzLnVzZXJJZCA9PSAke3Nlc3Npb25JZH0gJiYgcy5jYW5jZWxsZWQgPiAwXG4gICAgICBMRVQgcGxhbiA9IEZJUlNUKFxuICAgICAgICBGT1IgcCBJTiBwbGFuc1xuICAgICAgICBGSUxURVIgcC5fa2V5ID09IHMucGxhbklkXG4gICAgICApXG4gICAgICBMSU1JVCAxXG4gICAgICBSRVRVUk4gTUVSR0Uocywge3BsYW46IHBsYW59KWA7XG5cbiAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgLnRoZW4oKHN1YnNjcmlwdGlvbiA9IHt9KSA9PiBzdWJzY3JpcHRpb24pXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgbGFiZWw6ICdkYl9lcnJvcidcbiAgICB9LCBlcnJvciwge30pLnRoZW4oKCkgPT4gbnVsbCkuY2F0Y2goKGVycm9yKSA9PiBQcm9taXNlLnJlamVjdChlcnJvcikpKTtcbn07XG5cbmV4cG9ydCBjb25zdCBhZGRQbGFuID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGl0ZW06IFBheW1lbnRQbGFuKTogUHJvbWlzZTxQYXltZW50UGxhbj4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdhZGRQbGFuJztcbiAgY29uc3Qge2RhdGFiYXNlLCB1c2VySWQ6IHNlc3Npb25JZCwgdXNlckFjY2Vzc30gPSBjb250ZXh0O1xuICBjb25zdCBpc0FkbWluOiBib29sZWFuID0gdXNlckFjY2VzcyA+IDI7XG5cbiAgaWYoIWlzQWRtaW4pIHtcbiAgICByZXR1cm4gbG9nRXhjZXB0aW9uKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgbGFiZWw6ICd1bmF1dGhvcml6ZWQnLFxuICAgICAgdmFsdWU6ICdpbnZhbGlkX3Nlc3Npb24nXG4gICAgfSwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKTtcbiAgfVxuXG4gIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcbiAgY29uc3QgZm9ybWF0SWQ6IHN0cmluZyA9IHBhcnNlSWQoaXRlbS5pZCk7XG4gIGNvbnN0IHBsYW5JZDogc3RyaW5nID0gZm9ybWF0SWQgPT09ICcnID8gY3JlYXRlSGFzaChgdGFnLSR7c2Vzc2lvbklkfWApIDogZm9ybWF0SWQ7XG4gIGNvbnN0IHthbW91bnQsIGN1cnJlbmN5ID0gJ1VTRCcsIGRlc2NyaXB0aW9uLCBpbnRlcnZhbCwgaW50ZXJ2YWxDb3VudCwgbmFtZX0gPSBpdGVtO1xuICBjb25zdCBmb3JtYXRBbW91bnQ6IG51bWJlciA9IHBhcnNlTnVtKGFtb3VudCk7XG4gIGNvbnN0IGZvcm1hdEludGVydmFsOiBQYXltZW50SW50ZXJ2YWwgPSBwYXJzZUNoYXIoaW50ZXJ2YWwsIDUpLnRvTG93ZXJDYXNlKCkgYXMgUGF5bWVudEludGVydmFsO1xuICBjb25zdCBmb3JtYXRJbnRlcnZhbENudDogbnVtYmVyID0gcGFyc2VOdW0oaW50ZXJ2YWxDb3VudCwgNSk7XG4gIGNvbnN0IGZvcm1hdE5hbWU6IHN0cmluZyA9IHBhcnNlVmFyQ2hhcihuYW1lLCAzMik7XG4gIGNvbnN0IGZvcm1hdERlc2M6IHN0cmluZyA9IHBhcnNlVmFyQ2hhcihkZXNjcmlwdGlvbiwgMzIpO1xuXG4gIGNvbnN0IGluc2VydDogUGF5bWVudFBsYW4gPSB7XG4gICAgX2tleTogcGxhbklkLFxuICAgIGFkZGVkOiBub3csXG4gICAgYW1vdW50OiBmb3JtYXRBbW91bnQsXG4gICAgY3VycmVuY3ksXG4gICAgZGVzY3JpcHRpb246IGZvcm1hdERlc2MsXG4gICAgaW50ZXJ2YWw6IGZvcm1hdEludGVydmFsLFxuICAgIGludGVydmFsQ291bnQ6IGZvcm1hdEludGVydmFsQ250LFxuICAgIG1vZGlmaWVkOiBub3csXG4gICAgbmFtZTogZm9ybWF0TmFtZVxuICB9O1xuICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYElOU0VSVCAke2luc2VydH0gSU4gcGxhbnMgUkVUVVJOIE5FV2A7XG5cbiAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgLnRoZW4oKHBsYW46IFBheW1lbnRQbGFuID0ge30pID0+IHtcbiAgICAgIC8vIFN0cmlwZVxuICAgICAgY29uc3Qgc3RyaXBlQ2xpZW50ID0gbmV3IFN0cmlwZShDb25maWcuZ2V0KCdzdHJpcGUudG9rZW4nKSwge2FwaVZlcnNpb24sIHR5cGVzY3JpcHQ6IHRydWV9KTtcbiAgICAgIGNvbnN0IHN0cmlwZVBsYW46IGFueSA9IHtcbiAgICAgICAgYW1vdW50OiBmb3JtYXRBbW91bnQgKiAxMDAsXG4gICAgICAgIGN1cnJlbmN5LFxuICAgICAgICBpZDogcGxhbklkLFxuICAgICAgICBpbnRlcnZhbDogZm9ybWF0SW50ZXJ2YWwsXG4gICAgICAgIGludGVydmFsX2NvdW50OiBmb3JtYXRJbnRlcnZhbENudCxcbiAgICAgICAgbWV0YWRhdGE6IHtcbiAgICAgICAgfSxcbiAgICAgICAgbmFtZTogZm9ybWF0TmFtZSxcbiAgICAgICAgc3RhdGVtZW50X2Rlc2NyaXB0b3I6IGZvcm1hdERlc2NcbiAgICAgIH07XG5cbiAgICAgIHJldHVybiBzdHJpcGVDbGllbnQucGxhbnNcbiAgICAgICAgLmNyZWF0ZShzdHJpcGVQbGFuKVxuICAgICAgICAudGhlbigoKSA9PiBwbGFuKTtcbiAgICB9KVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICBhY3Rpb24sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIGxhYmVsOiAnZGJfZXJyb3InXG4gICAgfSwgZXJyb3IsIHt9KS50aGVuKCgpID0+IG51bGwpLmNhdGNoKChlcnJvcikgPT4gUHJvbWlzZS5yZWplY3QoZXJyb3IpKSk7XG59O1xuXG5leHBvcnQgY29uc3QgYWRkU3Vic2NyaXB0aW9uID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGl0ZW0pOiBQcm9taXNlPFBheW1lbnRTdWJzY3JpcHRpb24+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnYWRkU3Vic2NyaXB0aW9uJztcbiAgY29uc3Qge2RhdGFiYXNlLCB1c2VySWQ6IHNlc3Npb25JZCwgdXNlckFjY2Vzc30gPSBjb250ZXh0O1xuICBjb25zdCBpc0FkbWluOiBib29sZWFuID0gdXNlckFjY2VzcyA+IDI7XG5cbiAgaWYoIWlzQWRtaW4pIHtcbiAgICByZXR1cm4gbG9nRXhjZXB0aW9uKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgbGFiZWw6ICd1bmF1dGhvcml6ZWQnLFxuICAgICAgdmFsdWU6ICdpbnZhbGlkX3Nlc3Npb24nXG4gICAgfSwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKTtcbiAgfVxuXG4gIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcbiAgY29uc3QgZm9ybWF0SWQ6IHN0cmluZyA9IHBhcnNlSWQoaXRlbS5pZCk7XG4gIGNvbnN0IHN1YnNjcmlwdGlvbklkOiBzdHJpbmcgPSBmb3JtYXRJZCA9PT0gJycgPyBjcmVhdGVIYXNoKGB0YWctJHtzZXNzaW9uSWR9YCkgOiBmb3JtYXRJZDtcbiAgY29uc3Qge3BsYW5JZCwgdHJpYWxFbmQsIHRyaWFsUGVyaW9kfSA9IGl0ZW07XG4gIGNvbnN0IGZvcm1hdFBsYW5JZDogc3RyaW5nID0gcGFyc2VJZChwbGFuSWQpO1xuICBjb25zdCBmb3JtYXRUcmlhbFBlcmlvZDogbnVtYmVyID0gMDtcbiAgbGV0IGZvcm1hdFRyaWFsRW5kOiBudW1iZXIgPSB0cmlhbEVuZCB8fCBEYXRlLm5vdygpO1xuXG4gIGlmKGZvcm1hdFRyaWFsUGVyaW9kKSB7XG4gICAgZm9ybWF0VHJpYWxFbmQgPSBEYXRlVGltZS5sb2NhbCgpLnBsdXMoe2RheXM6IHRyaWFsUGVyaW9kfSkudG9NaWxsaXMoKTtcbiAgfVxuXG4gIHJldHVybiBnZXRVc2VyKGNvbnRleHQsIHNlc3Npb25JZClcbiAgICAudGhlbigodXNlcikgPT4ge1xuICAgICAgY29uc3QgaW5zZXJ0OiBQYXltZW50U3Vic2NyaXB0aW9uID0ge1xuICAgICAgICBfa2V5OiBzdWJzY3JpcHRpb25JZCxcbiAgICAgICAgYWRkZWQ6IG5vdyxcbiAgICAgICAgY2FuY2VsRGF0ZTogMCxcbiAgICAgICAgbW9kaWZpZWQ6IG5vdyxcbiAgICAgICAgcGxhbklkOiBmb3JtYXRQbGFuSWQsXG4gICAgICAgIHRyaWFsRW5kOiBmb3JtYXRUcmlhbEVuZCxcbiAgICAgICAgdXNlcklkOiBzZXNzaW9uSWRcbiAgICAgIH07XG4gICAgICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYElOU0VSVCAke2luc2VydH0gSU4gcGxhbnMgUkVUVVJOIE5FV2A7XG5cbiAgICAgIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgICAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAgICAgLnRoZW4oKHN1YnNjcmlwdGlvbjogUGF5bWVudFN1YnNjcmlwdGlvbiA9IHt9KSA9PiB7XG4gICAgICAgICAgLy8gU3RyaXBlXG4gICAgICAgICAgY29uc3Qgc3RyaXBlQ2xpZW50ID0gbmV3IFN0cmlwZShDb25maWcuZ2V0KCdzdHJpcGUudG9rZW4nKSwge2FwaVZlcnNpb24sIHR5cGVzY3JpcHQ6IHRydWV9KTtcbiAgICAgICAgICByZXR1cm4gc3RyaXBlQ2xpZW50LnN1YnNjcmlwdGlvbnNcbiAgICAgICAgICAgIC5jcmVhdGUoe1xuICAgICAgICAgICAgICBjdXN0b21lcjogdXNlci5zdHJpcGVDdXN0b21lcklkLFxuICAgICAgICAgICAgICBpdGVtczogW1xuICAgICAgICAgICAgICAgIHtcbiAgICAgICAgICAgICAgICAgIHBsYW46IGZvcm1hdFBsYW5JZFxuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgXSxcbiAgICAgICAgICAgICAgbWV0YWRhdGE6IHtcbiAgICAgICAgICAgICAgICB1c2VySWQ6IHNlc3Npb25JZFxuICAgICAgICAgICAgICB9LFxuICAgICAgICAgICAgICB0cmlhbF9lbmQ6IGZvcm1hdFRyaWFsRW5kXG4gICAgICAgICAgICB9KVxuICAgICAgICAgICAgLnRoZW4oKCkgPT4gc3Vic2NyaXB0aW9uKTtcbiAgICAgICAgfSlcbiAgICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgbGFiZWw6ICdkYl9lcnJvcidcbiAgICAgICAgfSwgZXJyb3IsIHt9KS50aGVuKCgpID0+IG51bGwpLmNhdGNoKChlcnJvcikgPT4gUHJvbWlzZS5yZWplY3QoZXJyb3IpKSk7XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgZGVsZXRlU3Vic2NyaXB0aW9uID0gKGNvbnRleHQ6IEFwaUNvbnRleHQpOiBQcm9taXNlPGJvb2xlYW4+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnZGVsZXRlU3Vic2NyaXB0aW9uJztcbiAgY29uc3Qge2RhdGFiYXNlfSA9IGNvbnRleHQ7XG4gIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcblxuICByZXR1cm4gZ2V0U3Vic2NyaXB0aW9uKGNvbnRleHQpXG4gICAgLnRoZW4oKHN1YnNjcmlwdGlvbikgPT4ge1xuICAgICAgLy8gU3RyaXBlXG4gICAgICBjb25zdCBzdHJpcGVDbGllbnQgPSBuZXcgU3RyaXBlKENvbmZpZy5nZXQoJ3N0cmlwZS50b2tlbicpLCB7YXBpVmVyc2lvbiwgdHlwZXNjcmlwdDogdHJ1ZX0pO1xuICAgICAgcmV0dXJuIHN0cmlwZUNsaWVudC5zdWJzY3JpcHRpb25zXG4gICAgICAgIC5kZWwoc3Vic2NyaXB0aW9uLnRyYW5zYWN0aW9uSWQpXG4gICAgICAgIC50aGVuKCh7Y2FuY2VsZWRfYXQ6IGNhbmNlbERhdGV9KSA9PiB7XG4gICAgICAgICAgLy8gTWFrZSBzdXJlIHdlIGNhbmNlbGxlZCBvbiBTdHJpcGUgYmVmb3JlIHVwZGF0aW5nIHRoZSBkYlxuICAgICAgICAgIGlmKGNhbmNlbERhdGUpIHtcbiAgICAgICAgICAgIGNvbnN0IHVwZGF0ZTogUGF5bWVudFN1YnNjcmlwdGlvbiA9IHtcbiAgICAgICAgICAgICAgY2FuY2VsRGF0ZSxcbiAgICAgICAgICAgICAgbW9kaWZpZWQ6IG5vdyxcbiAgICAgICAgICAgICAgcGxhbklkOiAnJ1xuICAgICAgICAgICAgfTtcbiAgICAgICAgICAgIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgVVBEQVRFIHMgV0lUSCAke3VwZGF0ZX0gSU4gc3Vic2NyaXB0aW9uc2A7XG5cbiAgICAgICAgICAgIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgICAgICAgICAgICAudGhlbigoKSA9PiB0cnVlKVxuICAgICAgICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgICAgICAgICBsYWJlbDogJ2RiX2Vycm9yJ1xuICAgICAgICAgICAgICB9LCBlcnJvciwge30pLnRoZW4oKCkgPT4gbnVsbCkuY2F0Y2goKGVycm9yKSA9PiBQcm9taXNlLnJlamVjdChlcnJvcikpKTtcbiAgICAgICAgICB9XG4gICAgICAgICAgcmV0dXJuIGZhbHNlO1xuICAgICAgICB9KTtcbiAgICB9KTtcbn07XG4iXX0=