@nlabs/reaktor 0.1.12 → 0.1.14

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (50) hide show
  1. package/lib/config.js +138 -0
  2. package/lib/data/conversations.js +228 -0
  3. package/lib/data/dynamodb.js +172 -0
  4. package/lib/data/email.js +194 -0
  5. package/lib/data/files.js +463 -0
  6. package/lib/data/groups.js +401 -0
  7. package/lib/data/images.js +841 -0
  8. package/lib/data/index.js +234 -0
  9. package/lib/data/ios.js +327 -0
  10. package/lib/data/locations.js +148 -0
  11. package/lib/data/messages.js +281 -0
  12. package/lib/data/notifications.js +59 -0
  13. package/lib/data/payments.js +798 -0
  14. package/lib/data/posts.js +637 -0
  15. package/lib/data/reactions.js +243 -0
  16. package/lib/data/s3.js +133 -0
  17. package/lib/data/search.js +111 -0
  18. package/lib/data/sms.js +79 -0
  19. package/lib/data/subscription.js +311 -0
  20. package/lib/data/tags.js +343 -0
  21. package/lib/data/users.js +415 -0
  22. package/lib/index.js +42 -0
  23. package/lib/types/apps.js +2 -0
  24. package/lib/types/arangodb.js +2 -0
  25. package/lib/types/auth.js +2 -0
  26. package/lib/types/conversations.js +2 -0
  27. package/lib/types/email.js +2 -0
  28. package/lib/types/files.js +2 -0
  29. package/lib/types/google.js +2 -0
  30. package/lib/types/groups.js +2 -0
  31. package/lib/types/images.js +2 -0
  32. package/lib/types/index.js +210 -0
  33. package/lib/types/locations.js +2 -0
  34. package/lib/types/messages.js +2 -0
  35. package/lib/types/notifications.js +2 -0
  36. package/lib/types/payments.js +2 -0
  37. package/lib/types/posts.js +2 -0
  38. package/lib/types/reactions.js +2 -0
  39. package/lib/types/tags.js +2 -0
  40. package/lib/types/users.js +2 -0
  41. package/lib/utils/analytics.js +59 -0
  42. package/lib/utils/arangodb.d.ts +0 -1
  43. package/lib/utils/arangodb.js +122 -0
  44. package/lib/utils/auth.d.ts +0 -2
  45. package/lib/utils/auth.js +55 -0
  46. package/lib/utils/graphql.js +19 -0
  47. package/lib/utils/index.js +78 -0
  48. package/lib/utils/objects.js +54 -0
  49. package/lib/utils/redis.js +28 -0
  50. package/package.json +7 -7
@@ -0,0 +1,311 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.deleteSubscription = exports.addSubscription = exports.addPlan = exports.getSubscription = exports.getPlanList = void 0;
7
+
8
+ var _utils = require("@nlabs/utils");
9
+
10
+ var _arangojs = require("arangojs");
11
+
12
+ var _luxon = require("luxon");
13
+
14
+ var stripe = _interopRequireWildcard(require("stripe"));
15
+
16
+ var _config = require("../config");
17
+
18
+ var _utils2 = require("../utils");
19
+
20
+ var _users = require("./users");
21
+
22
+ function _interopRequireWildcard(obj) { if (obj && obj.__esModule) { return obj; } else { var newObj = {}; if (obj != null) { for (var key in obj) { if (Object.prototype.hasOwnProperty.call(obj, key)) { var desc = Object.defineProperty && Object.getOwnPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : {}; if (desc.get || desc.set) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } } newObj["default"] = obj; return newObj; } }
23
+
24
+ function _templateObject3() {
25
+ var data = _taggedTemplateLiteral(["UPDATE s WITH ", " IN subscriptions"]);
26
+
27
+ _templateObject3 = function _templateObject3() {
28
+ return data;
29
+ };
30
+
31
+ return data;
32
+ }
33
+
34
+ function _templateObject2() {
35
+ var data = _taggedTemplateLiteral(["INSERT ", " IN plans RETURN NEW"]);
36
+
37
+ _templateObject2 = function _templateObject2() {
38
+ return data;
39
+ };
40
+
41
+ return data;
42
+ }
43
+
44
+ function _templateObject() {
45
+ var data = _taggedTemplateLiteral(["INSERT ", " IN plans RETURN NEW"]);
46
+
47
+ _templateObject = function _templateObject() {
48
+ return data;
49
+ };
50
+
51
+ return data;
52
+ }
53
+
54
+ function _taggedTemplateLiteral(strings, raw) { if (!raw) { raw = strings.slice(0); } return Object.freeze(Object.defineProperties(strings, { raw: { value: Object.freeze(raw) } })); }
55
+
56
+ var eventCategory = 'subscription';
57
+
58
+ var getPlanList = function getPlanList(context) {
59
+ var from = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 0;
60
+ var to = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : 30;
61
+ var action = 'getList';
62
+ var database = context.database;
63
+ var limit = (0, _utils2.getLimit)(from, to);
64
+ var aqlQry = "FOR p IN plans\n ".concat(limit.aql, "\n SORT p.amount\n RETURN p");
65
+ return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
66
+ return cursor.all();
67
+ }).then(function () {
68
+ var list = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [];
69
+ return list;
70
+ })["catch"](function (error) {
71
+ return (0, _utils2.logError)({
72
+ action: action,
73
+ category: eventCategory,
74
+ label: 'db_error'
75
+ }, error, {}).then(function () {
76
+ return null;
77
+ })["catch"](function (error) {
78
+ return Promise.reject(error);
79
+ });
80
+ });
81
+ };
82
+
83
+ exports.getPlanList = getPlanList;
84
+
85
+ var getSubscription = function getSubscription(context) {
86
+ var action = 'getItem';
87
+ var database = context.database,
88
+ sessionId = context.userId;
89
+ 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})");
90
+ return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
91
+ return cursor.next();
92
+ }).then(function () {
93
+ var subscription = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
94
+ return subscription;
95
+ })["catch"](function (error) {
96
+ return (0, _utils2.logError)({
97
+ action: action,
98
+ category: eventCategory,
99
+ label: 'db_error'
100
+ }, error, {}).then(function () {
101
+ return null;
102
+ })["catch"](function (error) {
103
+ return Promise.reject(error);
104
+ });
105
+ });
106
+ };
107
+
108
+ exports.getSubscription = getSubscription;
109
+
110
+ var addPlan = function addPlan(context, item) {
111
+ var action = 'addPlan';
112
+ var database = context.database,
113
+ sessionId = context.userId,
114
+ userType = context.userType;
115
+ var isAdmin = userType > 2;
116
+
117
+ if (!isAdmin) {
118
+ return (0, _utils2.logException)({
119
+ action: action,
120
+ category: eventCategory,
121
+ label: 'unauthorized',
122
+ value: 'invalid_session'
123
+ }, context).then(function () {
124
+ return null;
125
+ });
126
+ }
127
+
128
+ var now = Date.now();
129
+ var formatId = (0, _utils.parseId)(item.id);
130
+ var planId = formatId === '' ? (0, _utils.createHash)("tag-".concat(sessionId)) : formatId;
131
+ var amount = item.amount,
132
+ _item$currency = item.currency,
133
+ currency = _item$currency === void 0 ? 'USD' : _item$currency,
134
+ description = item.description,
135
+ interval = item.interval,
136
+ intervalCount = item.intervalCount,
137
+ name = item.name;
138
+ var formatAmount = (0, _utils.parseNum)(amount);
139
+ var formatInterval = (0, _utils.parseChar)(interval, 5).toLowerCase();
140
+ var formatIntervalCnt = (0, _utils.parseNum)(intervalCount, 5);
141
+ var formatName = (0, _utils.parseVarChar)(name, 32);
142
+ var formatDesc = (0, _utils.parseVarChar)(description, 32);
143
+ var insert = {
144
+ _key: planId,
145
+ added: now,
146
+ amount: formatAmount,
147
+ currency: currency,
148
+ description: formatDesc,
149
+ interval: formatInterval,
150
+ intervalCount: formatIntervalCnt,
151
+ modified: now,
152
+ name: formatName
153
+ };
154
+ var aqlQry = (0, _arangojs.aql)(_templateObject(), insert);
155
+ return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
156
+ return cursor.next();
157
+ }).then(function () {
158
+ var plan = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
159
+ // Stripe
160
+ var stripeClient = stripe(_config.Config.get('stripe.token'));
161
+ return stripeClient.plans.create({
162
+ amount: formatAmount * 100,
163
+ currency: currency,
164
+ id: planId,
165
+ interval: formatInterval,
166
+ interval_count: formatIntervalCnt,
167
+ metadata: {},
168
+ name: formatName,
169
+ statement_descriptor: formatDesc
170
+ }).then(function () {
171
+ return plan;
172
+ });
173
+ })["catch"](function (error) {
174
+ return (0, _utils2.logError)({
175
+ action: action,
176
+ category: eventCategory,
177
+ label: 'db_error'
178
+ }, error, {}).then(function () {
179
+ return null;
180
+ })["catch"](function (error) {
181
+ return Promise.reject(error);
182
+ });
183
+ });
184
+ };
185
+
186
+ exports.addPlan = addPlan;
187
+
188
+ var addSubscription = function addSubscription(context, item) {
189
+ var action = 'addSubscription';
190
+ var database = context.database,
191
+ sessionId = context.userId,
192
+ userType = context.userType;
193
+ var isAdmin = userType > 2;
194
+
195
+ if (!isAdmin) {
196
+ return (0, _utils2.logException)({
197
+ action: action,
198
+ category: eventCategory,
199
+ label: 'unauthorized',
200
+ value: 'invalid_session'
201
+ }, context).then(function () {
202
+ return null;
203
+ });
204
+ }
205
+
206
+ var now = Date.now();
207
+ var formatId = (0, _utils.parseId)(item.id);
208
+ var subscriptionId = formatId === '' ? (0, _utils.createHash)("tag-".concat(sessionId)) : formatId;
209
+ var planId = item.planId,
210
+ tax = item.tax,
211
+ trialEnd = item.trialEnd,
212
+ trialPeriod = item.trialPeriod;
213
+ var formatPlanId = (0, _utils.parseId)(planId);
214
+ var formatTaxPercent = (0, _utils.parseNum)(tax) || 0;
215
+ var formatTrialPeriod = (0, _utils.parseNum)(tax) || 0;
216
+ var formatTrialEnd = trialEnd || Date.now();
217
+
218
+ if (formatTrialPeriod) {
219
+ formatTrialEnd = _luxon.DateTime.local().plus({
220
+ days: trialPeriod
221
+ }).toMillis();
222
+ }
223
+
224
+ return (0, _users.getUser)(context, sessionId).then(function (user) {
225
+ var insert = {
226
+ _key: subscriptionId,
227
+ added: now,
228
+ cancelDate: 0,
229
+ modified: now,
230
+ planId: formatPlanId,
231
+ tax: formatTaxPercent,
232
+ trialEnd: formatTrialEnd,
233
+ userId: sessionId
234
+ };
235
+ var aqlQry = (0, _arangojs.aql)(_templateObject2(), insert);
236
+ return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
237
+ return cursor.next();
238
+ }).then(function () {
239
+ var subscription = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
240
+ // Stripe
241
+ var stripeClient = stripe(_config.Config.get('stripe.token'));
242
+ return stripeClient.subscriptions.create({
243
+ customer: user.stripeCustomerId,
244
+ items: [{
245
+ plan: formatPlanId
246
+ }],
247
+ metadata: {
248
+ userId: sessionId
249
+ },
250
+ tax_percent: formatTaxPercent * 100,
251
+ trial_end: formatTrialEnd
252
+ }).then(function () {
253
+ return subscription;
254
+ });
255
+ })["catch"](function (error) {
256
+ return (0, _utils2.logError)({
257
+ action: action,
258
+ category: eventCategory,
259
+ label: 'db_error'
260
+ }, error, {}).then(function () {
261
+ return null;
262
+ })["catch"](function (error) {
263
+ return Promise.reject(error);
264
+ });
265
+ });
266
+ });
267
+ };
268
+
269
+ exports.addSubscription = addSubscription;
270
+
271
+ var deleteSubscription = function deleteSubscription(context, endDate) {
272
+ var action = 'deleteSubscription';
273
+ var database = context.database;
274
+ var now = Date.now();
275
+ var formatEndDate = (0, _utils.parseNum)(endDate) || now;
276
+ return getSubscription(context).then(function (subscription) {
277
+ // Stripe
278
+ var stripeClient = stripe(_config.Config.get('stripe.token'));
279
+ return stripeClient.subscriptions.del(subscription.transactionId, {
280
+ subscription: formatEndDate
281
+ }).then(function (stripeResponse) {
282
+ // Make sure we cancelled on Stripe before updating the db
283
+ if (stripeResponse.cancelled || stripeResponse.cancel_at_period_end) {
284
+ var update = {
285
+ cancelDate: formatEndDate,
286
+ modified: now,
287
+ planId: ''
288
+ };
289
+ var aqlQry = (0, _arangojs.aql)(_templateObject3(), update);
290
+ return (0, _utils2.useDb)(database).query(aqlQry).then(function () {
291
+ return true;
292
+ })["catch"](function (error) {
293
+ return (0, _utils2.logError)({
294
+ action: action,
295
+ category: eventCategory,
296
+ label: 'db_error'
297
+ }, error, {}).then(function () {
298
+ return null;
299
+ })["catch"](function (error) {
300
+ return Promise.reject(error);
301
+ });
302
+ });
303
+ }
304
+
305
+ return false;
306
+ });
307
+ });
308
+ };
309
+
310
+ exports.deleteSubscription = deleteSubscription;
311
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kYXRhL3N1YnNjcmlwdGlvbi50cyJdLCJuYW1lcyI6WyJldmVudENhdGVnb3J5IiwiZ2V0UGxhbkxpc3QiLCJjb250ZXh0IiwiZnJvbSIsInRvIiwiYWN0aW9uIiwiZGF0YWJhc2UiLCJsaW1pdCIsImFxbFFyeSIsImFxbCIsInF1ZXJ5IiwidGhlbiIsImN1cnNvciIsImFsbCIsImxpc3QiLCJlcnJvciIsImNhdGVnb3J5IiwibGFiZWwiLCJQcm9taXNlIiwicmVqZWN0IiwiZ2V0U3Vic2NyaXB0aW9uIiwic2Vzc2lvbklkIiwidXNlcklkIiwibmV4dCIsInN1YnNjcmlwdGlvbiIsImFkZFBsYW4iLCJpdGVtIiwidXNlclR5cGUiLCJpc0FkbWluIiwidmFsdWUiLCJub3ciLCJEYXRlIiwiZm9ybWF0SWQiLCJpZCIsInBsYW5JZCIsImFtb3VudCIsImN1cnJlbmN5IiwiZGVzY3JpcHRpb24iLCJpbnRlcnZhbCIsImludGVydmFsQ291bnQiLCJuYW1lIiwiZm9ybWF0QW1vdW50IiwiZm9ybWF0SW50ZXJ2YWwiLCJ0b0xvd2VyQ2FzZSIsImZvcm1hdEludGVydmFsQ250IiwiZm9ybWF0TmFtZSIsImZvcm1hdERlc2MiLCJpbnNlcnQiLCJfa2V5IiwiYWRkZWQiLCJtb2RpZmllZCIsInBsYW4iLCJzdHJpcGVDbGllbnQiLCJzdHJpcGUiLCJDb25maWciLCJnZXQiLCJwbGFucyIsImNyZWF0ZSIsImludGVydmFsX2NvdW50IiwibWV0YWRhdGEiLCJzdGF0ZW1lbnRfZGVzY3JpcHRvciIsImFkZFN1YnNjcmlwdGlvbiIsInN1YnNjcmlwdGlvbklkIiwidGF4IiwidHJpYWxFbmQiLCJ0cmlhbFBlcmlvZCIsImZvcm1hdFBsYW5JZCIsImZvcm1hdFRheFBlcmNlbnQiLCJmb3JtYXRUcmlhbFBlcmlvZCIsImZvcm1hdFRyaWFsRW5kIiwiRGF0ZVRpbWUiLCJsb2NhbCIsInBsdXMiLCJkYXlzIiwidG9NaWxsaXMiLCJ1c2VyIiwiY2FuY2VsRGF0ZSIsInN1YnNjcmlwdGlvbnMiLCJjdXN0b21lciIsInN0cmlwZUN1c3RvbWVySWQiLCJpdGVtcyIsInRheF9wZXJjZW50IiwidHJpYWxfZW5kIiwiZGVsZXRlU3Vic2NyaXB0aW9uIiwiZW5kRGF0ZSIsImZvcm1hdEVuZERhdGUiLCJkZWwiLCJ0cmFuc2FjdGlvbklkIiwic3RyaXBlUmVzcG9uc2UiLCJjYW5jZWxsZWQiLCJjYW5jZWxfYXRfcGVyaW9kX2VuZCIsInVwZGF0ZSJdLCJtYXBwaW5ncyI6Ijs7Ozs7OztBQUlBOztBQUNBOztBQUdBOztBQUNBOztBQUVBOztBQUlBOztBQUNBOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFFQSxJQUFNQSxhQUFxQixHQUFHLGNBQTlCOztBQUVPLElBQU1DLFdBQVcsR0FBRyxTQUFkQSxXQUFjLENBQUNDLE9BQUQsRUFBb0Y7QUFBQSxNQUE5REMsSUFBOEQsdUVBQS9DLENBQStDO0FBQUEsTUFBNUNDLEVBQTRDLHVFQUEvQixFQUErQjtBQUM3RyxNQUFNQyxNQUFjLEdBQUcsU0FBdkI7QUFENkcsTUFFdEdDLFFBRnNHLEdBRTFGSixPQUYwRixDQUV0R0ksUUFGc0c7QUFHN0csTUFBTUMsS0FBb0IsR0FBRyxzQkFBU0osSUFBVCxFQUFlQyxFQUFmLENBQTdCO0FBQ0EsTUFBTUksTUFBYyxtQ0FDZEQsS0FBSyxDQUFDRSxHQURRLDBDQUFwQjtBQUtBLFNBQU8sbUJBQU1ILFFBQU4sRUFBZ0JJLEtBQWhCLENBQXNCRixNQUF0QixFQUNKRyxJQURJLENBQ0MsVUFBQ0MsTUFBRDtBQUFBLFdBQXlCQSxNQUFNLENBQUNDLEdBQVAsRUFBekI7QUFBQSxHQURELEVBRUpGLElBRkksQ0FFQztBQUFBLFFBQUNHLElBQUQsdUVBQVEsRUFBUjtBQUFBLFdBQWVBLElBQWY7QUFBQSxHQUZELFdBR0UsVUFBQ0MsS0FBRDtBQUFBLFdBQWtCLHNCQUFTO0FBQ2hDVixNQUFBQSxNQUFNLEVBQU5BLE1BRGdDO0FBRWhDVyxNQUFBQSxRQUFRLEVBQUVoQixhQUZzQjtB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QVksQ0FBQ3lCLGFBQWIsQ0FDSlMsR0FESSxDQUNBOUQsWUFBWSxDQUFDK0QsYUFEYixFQUM0QjtBQUMvQi9ELE1BQUFBLFlBQVksRUFBRTZEO0FBRGlCLEtBRDVCLEVBSUoxRSxJQUpJLENBSUMsVUFBQzZFLGNBQUQsRUFBb0I7QUFDeEI7QUFDQSxVQUFHQSxjQUFjLENBQUNDLFNBQWYsSUFBNEJELGNBQWMsQ0FBQ0Usb0JBQTlDLEVBQW9FO0FBQ2xFLFlBQU1DLE1BQTJCLEdBQUc7QUFDbENmLFVBQUFBLFVBQVUsRUFBRVMsYUFEc0I7QUFFbENuQyxVQUFBQSxRQUFRLEVBQUVwQixHQUZ3QjtBQUdsQ0ksVUFBQUEsTUFBTSxFQUFFO0FBSDBCLFNBQXBDO0FBS0EsWUFBTTFCLE1BQWdCLE9BQUdDLGFBQUgsc0JBQXVCa0YsTUFBdkIsQ0FBdEI7QUFFQSxlQUFPLG1CQUFNckYsUUFBTixFQUFnQkksS0FBaEIsQ0FBc0JGLE1BQXRCLEVBQ0pHLElBREksQ0FDQztBQUFBLGlCQUFNLElBQU47QUFBQSxTQURELFdBRUUsVUFBQ0ksS0FBRDtBQUFBLGlCQUFrQixzQkFBUztBQUNoQ1YsWUFBQUEsTUFBTSxFQUFOQSxNQURnQztBQUVoQ1csWUFBQUEsUUFBUSxFQUFFaEIsYUFGc0I7QUFHaENpQixZQUFBQSxLQUFLLEVBQUU7QUFIeUIsV0FBVCxFQUl0QkYsS0FKc0IsRUFJZixFQUplLEVBSVhKLElBSlcsQ0FJTjtBQUFBLG1CQUFNLElBQU47QUFBQSxXQUpNLFdBSVksVUFBQ0ksS0FBRDtBQUFBLG1CQUFXRyxPQUFPLENBQUNDLE1BQVIsQ0FBZUosS0FBZixDQUFYO0FBQUEsV0FKWixDQUFsQjtBQUFBLFNBRkYsQ0FBUDtBQU9EOztBQUNELGFBQU8sS0FBUDtBQUNELEtBdkJJLENBQVA7QUF3QkQsR0E1QkksQ0FBUDtBQTZCRCxDQW5DTSIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQ29weXJpZ2h0IChjKSAyMDE5LVByZXNlbnQsIE5pdHJvZ2VuIExhYnMsIEluYy5cbiAqIENvcHlyaWdodHMgbGljZW5zZWQgdW5kZXIgdGhlIE1JVCBMaWNlbnNlLiBTZWUgdGhlIGFjY29tcGFueWluZyBMSUNFTlNFIGZpbGUgZm9yIHRlcm1zLlxuICovXG5pbXBvcnQge2NyZWF0ZUhhc2gsIHBhcnNlQ2hhciwgcGFyc2VJZCwgcGFyc2VOdW0sIHBhcnNlVmFyQ2hhcn0gZnJvbSAnQG5sYWJzL3V0aWxzJztcbmltcG9ydCB7YXFsfSBmcm9tICdhcmFuZ29qcyc7XG5pbXBvcnQge0FxbFF1ZXJ5fSBmcm9tICdhcmFuZ29qcy9saWIvY2pzL2FxbC1xdWVyeSc7XG5pbXBvcnQge0FycmF5Q3Vyc29yfSBmcm9tICdhcmFuZ29qcy9saWIvY2pzL2N1cnNvcic7XG5pbXBvcnQge0RhdGVUaW1lfSBmcm9tICdsdXhvbic7XG5pbXBvcnQgKiBhcyBzdHJpcGUgZnJvbSAnc3RyaXBlJztcblxuaW1wb3J0IHtDb25maWd9IGZyb20gJy4uL2NvbmZpZyc7XG5pbXBvcnQge0FyYW5nb0RCTGltaXR9IGZyb20gJy4uL3R5cGVzL2FyYW5nb2RiJztcbmltcG9ydCB7QXBpQ29udGV4dH0gZnJvbSAnLi4vdHlwZXMvYXV0aCc7XG5pbXBvcnQge1BheW1lbnRJbnRlcnZhbCwgUGF5bWVudFBsYW4sIFBheW1lbnRTdWJzY3JpcHRpb259IGZyb20gJy4uL3R5cGVzL3BheW1lbnRzJztcbmltcG9ydCB7Z2V0TGltaXQsIGxvZ0Vycm9yLCBsb2dFeGNlcHRpb24sIHVzZURifSBmcm9tICcuLi91dGlscyc7XG5pbXBvcnQge2dldFVzZXJ9IGZyb20gJy4vdXNlcnMnO1xuXG5jb25zdCBldmVudENhdGVnb3J5OiBzdHJpbmcgPSAnc3Vic2NyaXB0aW9uJztcblxuZXhwb3J0IGNvbnN0IGdldFBsYW5MaXN0ID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGZyb206IG51bWJlciA9IDAsIHRvOiBudW1iZXIgPSAzMCk6IFByb21pc2U8UGF5bWVudFBsYW5bXT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRMaXN0JztcbiAgY29uc3Qge2RhdGFiYXNlfSA9IGNvbnRleHQ7XG4gIGNvbnN0IGxpbWl0OiBBcmFuZ29EQkxpbWl0ID0gZ2V0TGltaXQoZnJvbSwgdG8pO1xuICBjb25zdCBhcWxRcnk6IHN0cmluZyA9IGBGT1IgcCBJTiBwbGFuc1xuICAgICAgJHtsaW1pdC5hcWx9XG4gICAgICBTT1JUIHAuYW1vdW50XG4gICAgICBSRVRVUk4gcGA7XG5cbiAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5hbGwoKSlcbiAgICAudGhlbigobGlzdCA9IFtdKSA9PiBsaXN0KVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICBhY3Rpb24sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIGxhYmVsOiAnZGJfZXJyb3InXG4gICAgfSwgZXJyb3IsIHt9KS50aGVuKCgpID0+IG51bGwpLmNhdGNoKChlcnJvcikgPT4gUHJvbWlzZS5yZWplY3QoZXJyb3IpKSk7XG59O1xuXG5leHBvcnQgY29uc3QgZ2V0U3Vic2NyaXB0aW9uID0gKGNvbnRleHQ6IEFwaUNvbnRleHQpOiBQcm9taXNlPFBheW1lbnRTdWJzY3JpcHRpb24+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnZ2V0SXRlbSc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgdXNlcklkOiBzZXNzaW9uSWR9ID0gY29udGV4dDtcbiAgY29uc3QgYXFsUXJ5OiBzdHJpbmcgPSBgRk9SIHMgSU4gc3Vic2NyaXB0aW9uc1xuICAgICAgRklMVEVSIHMudXNlcklkID09ICR7c2Vzc2lvbklkfSAmJiBzLmNhbmNlbGxlZCA+IDBcbiAgICAgIExFVCBwbGFuID0gRklSU1QoXG4gICAgICAgIEZPUiBwIElOIHBsYW5zXG4gICAgICAgIEZJTFRFUiBwLl9rZXkgPT0gcy5wbGFuSWRcbiAgICAgIClcbiAgICAgIExJTUlUIDFcbiAgICAgIFJFVFVSTiBNRVJHRShzLCB7cGxhbjogcGxhbn0pYDtcblxuICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAudGhlbigoc3Vic2NyaXB0aW9uID0ge30pID0+IHN1YnNjcmlwdGlvbilcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgYWN0aW9uLFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICBsYWJlbDogJ2RiX2Vycm9yJ1xuICAgIH0sIGVycm9yLCB7fSkudGhlbigoKSA9PiBudWxsKS5jYXRjaCgoZXJyb3IpID0+IFByb21pc2UucmVqZWN0KGVycm9yKSkpO1xufTtcblxuZXhwb3J0IGNvbnN0IGFkZFBsYW4gPSAoY29udGV4dDogQXBpQ29udGV4dCwgaXRlbTogUGF5bWVudFBsYW4pOiBQcm9taXNlPFBheW1lbnRQbGFuPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2FkZFBsYW4nO1xuICBjb25zdCB7ZGF0YWJhc2UsIHVzZXJJZDogc2Vzc2lvbklkLCB1c2VyVHlwZX0gPSBjb250ZXh0O1xuICBjb25zdCBpc0FkbWluOiBib29sZWFuID0gdXNlclR5cGUgPiAyO1xuXG4gIGlmKCFpc0FkbWluKSB7XG4gICAgcmV0dXJuIGxvZ0V4Y2VwdGlvbih7XG4gICAgICBhY3Rpb24sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIGxhYmVsOiAndW5hdXRob3JpemVkJyxcbiAgICAgIHZhbHVlOiAnaW52YWxpZF9zZXNzaW9uJ1xuICAgIH0sIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCk7XG4gIH1cblxuICBjb25zdCBub3c6IG51bWJlciA9IERhdGUubm93KCk7XG4gIGNvbnN0IGZvcm1hdElkOiBzdHJpbmcgPSBwYXJzZUlkKGl0ZW0uaWQpO1xuICBjb25zdCBwbGFuSWQ6IHN0cmluZyA9IGZvcm1hdElkID09PSAnJyA/IGNyZWF0ZUhhc2goYHRhZy0ke3Nlc3Npb25JZH1gKSA6IGZvcm1hdElkO1xuICBjb25zdCB7YW1vdW50LCBjdXJyZW5jeSA9ICdVU0QnLCBkZXNjcmlwdGlvbiwgaW50ZXJ2YWwsIGludGVydmFsQ291bnQsIG5hbWV9ID0gaXRlbTtcbiAgY29uc3QgZm9ybWF0QW1vdW50OiBudW1iZXIgPSBwYXJzZU51bShhbW91bnQpO1xuICBjb25zdCBmb3JtYXRJbnRlcnZhbDogUGF5bWVudEludGVydmFsID0gcGFyc2VDaGFyKGludGVydmFsLCA1KS50b0xvd2VyQ2FzZSgpIGFzIFBheW1lbnRJbnRlcnZhbDtcbiAgY29uc3QgZm9ybWF0SW50ZXJ2YWxDbnQ6IG51bWJlciA9IHBhcnNlTnVtKGludGVydmFsQ291bnQsIDUpO1xuICBjb25zdCBmb3JtYXROYW1lOiBzdHJpbmcgPSBwYXJzZVZhckNoYXIobmFtZSwgMzIpO1xuICBjb25zdCBmb3JtYXREZXNjOiBzdHJpbmcgPSBwYXJzZVZhckNoYXIoZGVzY3JpcHRpb24sIDMyKTtcblxuICBjb25zdCBpbnNlcnQ6IFBheW1lbnRQbGFuID0ge1xuICAgIF9rZXk6IHBsYW5JZCxcbiAgICBhZGRlZDogbm93LFxuICAgIGFtb3VudDogZm9ybWF0QW1vdW50LFxuICAgIGN1cnJlbmN5LFxuICAgIGRlc2NyaXB0aW9uOiBmb3JtYXREZXNjLFxuICAgIGludGVydmFsOiBmb3JtYXRJbnRlcnZhbCxcbiAgICBpbnRlcnZhbENvdW50OiBmb3JtYXRJbnRlcnZhbENudCxcbiAgICBtb2RpZmllZDogbm93LFxuICAgIG5hbWU6IGZvcm1hdE5hbWVcbiAgfTtcbiAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBJTlNFUlQgJHtpbnNlcnR9IElOIHBsYW5zIFJFVFVSTiBORVdgO1xuXG4gIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgIC50aGVuKChwbGFuOiBQYXltZW50UGxhbiA9IHt9KSA9PiB7XG4gICAgICAvLyBTdHJpcGVcbiAgICAgIGNvbnN0IHN0cmlwZUNsaWVudCA9IHN0cmlwZShDb25maWcuZ2V0KCdzdHJpcGUudG9rZW4nKSk7XG4gICAgICByZXR1cm4gc3RyaXBlQ2xpZW50LnBsYW5zXG4gICAgICAgIC5jcmVhdGUoe1xuICAgICAgICAgIGFtb3VudDogZm9ybWF0QW1vdW50ICogMTAwLFxuICAgICAgICAgIGN1cnJlbmN5LFxuICAgICAgICAgIGlkOiBwbGFuSWQsXG4gICAgICAgICAgaW50ZXJ2YWw6IGZvcm1hdEludGVydmFsLFxuICAgICAgICAgIGludGVydmFsX2NvdW50OiBmb3JtYXRJbnRlcnZhbENudCxcbiAgICAgICAgICBtZXRhZGF0YToge1xuICAgICAgICAgIH0sXG4gICAgICAgICAgbmFtZTogZm9ybWF0TmFtZSxcbiAgICAgICAgICBzdGF0ZW1lbnRfZGVzY3JpcHRvcjogZm9ybWF0RGVzY1xuICAgICAgICB9KVxuICAgICAgICAudGhlbigoKSA9PiBwbGFuKTtcbiAgICB9KVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICBhY3Rpb24sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIGxhYmVsOiAnZGJfZXJyb3InXG4gICAgfSwgZXJyb3IsIHt9KS50aGVuKCgpID0+IG51bGwpLmNhdGNoKChlcnJvcikgPT4gUHJvbWlzZS5yZWplY3QoZXJyb3IpKSk7XG59O1xuXG5leHBvcnQgY29uc3QgYWRkU3Vic2NyaXB0aW9uID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGl0ZW0pOiBQcm9taXNlPFBheW1lbnRTdWJzY3JpcHRpb24+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnYWRkU3Vic2NyaXB0aW9uJztcbiAgY29uc3Qge2RhdGFiYXNlLCB1c2VySWQ6IHNlc3Npb25JZCwgdXNlclR5cGV9ID0gY29udGV4dDtcbiAgY29uc3QgaXNBZG1pbjogYm9vbGVhbiA9IHVzZXJUeXBlID4gMjtcblxuICBpZighaXNBZG1pbikge1xuICAgIHJldHVybiBsb2dFeGNlcHRpb24oe1xuICAgICAgYWN0aW9uLFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICBsYWJlbDogJ3VuYXV0aG9yaXplZCcsXG4gICAgICB2YWx1ZTogJ2ludmFsaWRfc2Vzc2lvbidcbiAgICB9LCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpO1xuICB9XG5cbiAgY29uc3Qgbm93OiBudW1iZXIgPSBEYXRlLm5vdygpO1xuICBjb25zdCBmb3JtYXRJZDogc3RyaW5nID0gcGFyc2VJZChpdGVtLmlkKTtcbiAgY29uc3Qgc3Vic2NyaXB0aW9uSWQ6IHN0cmluZyA9IGZvcm1hdElkID09PSAnJyA/IGNyZWF0ZUhhc2goYHRhZy0ke3Nlc3Npb25JZH1gKSA6IGZvcm1hdElkO1xuICBjb25zdCB7cGxhbklkLCB0YXgsIHRyaWFsRW5kLCB0cmlhbFBlcmlvZH0gPSBpdGVtO1xuICBjb25zdCBmb3JtYXRQbGFuSWQ6IHN0cmluZyA9IHBhcnNlSWQocGxhbklkKTtcbiAgY29uc3QgZm9ybWF0VGF4UGVyY2VudDogbnVtYmVyID0gcGFyc2VOdW0odGF4KSB8fCAwO1xuICBjb25zdCBmb3JtYXRUcmlhbFBlcmlvZDogbnVtYmVyID0gcGFyc2VOdW0odGF4KSB8fCAwO1xuICBsZXQgZm9ybWF0VHJpYWxFbmQ6IG51bWJlciA9IHRyaWFsRW5kIHx8IERhdGUubm93KCk7XG5cbiAgaWYoZm9ybWF0VHJpYWxQZXJpb2QpIHtcbiAgICBmb3JtYXRUcmlhbEVuZCA9IERhdGVUaW1lLmxvY2FsKCkucGx1cyh7ZGF5czogdHJpYWxQZXJpb2R9KS50b01pbGxpcygpO1xuICB9XG5cbiAgcmV0dXJuIGdldFVzZXIoY29udGV4dCwgc2Vzc2lvbklkKVxuICAgIC50aGVuKCh1c2VyKSA9PiB7XG4gICAgICBjb25zdCBpbnNlcnQ6IFBheW1lbnRTdWJzY3JpcHRpb24gPSB7XG4gICAgICAgIF9rZXk6IHN1YnNjcmlwdGlvbklkLFxuICAgICAgICBhZGRlZDogbm93LFxuICAgICAgICBjYW5jZWxEYXRlOiAwLFxuICAgICAgICBtb2RpZmllZDogbm93LFxuICAgICAgICBwbGFuSWQ6IGZvcm1hdFBsYW5JZCxcbiAgICAgICAgdGF4OiBmb3JtYXRUYXhQZXJjZW50LFxuICAgICAgICB0cmlhbEVuZDogZm9ybWF0VHJpYWxFbmQsXG4gICAgICAgIHVzZXJJZDogc2Vzc2lvbklkXG4gICAgICB9O1xuICAgICAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBJTlNFUlQgJHtpbnNlcnR9IElOIHBsYW5zIFJFVFVSTiBORVdgO1xuXG4gICAgICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAgICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgICAgIC50aGVuKChzdWJzY3JpcHRpb246IFBheW1lbnRTdWJzY3JpcHRpb24gPSB7fSkgPT4ge1xuICAgICAgICAgIC8vIFN0cmlwZVxuICAgICAgICAgIGNvbnN0IHN0cmlwZUNsaWVudCA9IHN0cmlwZShDb25maWcuZ2V0KCdzdHJpcGUudG9rZW4nKSk7XG4gICAgICAgICAgcmV0dXJuIHN0cmlwZUNsaWVudC5zdWJzY3JpcHRpb25zXG4gICAgICAgICAgICAuY3JlYXRlKHtcbiAgICAgICAgICAgICAgY3VzdG9tZXI6IHVzZXIuc3RyaXBlQ3VzdG9tZXJJZCxcbiAgICAgICAgICAgICAgaXRlbXM6IFtcbiAgICAgICAgICAgICAgICB7cGxhbjogZm9ybWF0UGxhbklkfVxuICAgICAgICAgICAgICBdLFxuICAgICAgICAgICAgICBtZXRhZGF0YToge1xuICAgICAgICAgICAgICAgIHVzZXJJZDogc2Vzc2lvbklkXG4gICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICAgIHRheF9wZXJjZW50OiBmb3JtYXRUYXhQZXJjZW50ICogMTAwLFxuICAgICAgICAgICAgICB0cmlhbF9lbmQ6IGZvcm1hdFRyaWFsRW5kXG4gICAgICAgICAgICB9KVxuICAgICAgICAgICAgLnRoZW4oKCkgPT4gc3Vic2NyaXB0aW9uKTtcbiAgICAgICAgfSlcbiAgICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgbGFiZWw6ICdkYl9lcnJvcidcbiAgICAgICAgfSwgZXJyb3IsIHt9KS50aGVuKCgpID0+IG51bGwpLmNhdGNoKChlcnJvcikgPT4gUHJvbWlzZS5yZWplY3QoZXJyb3IpKSk7XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgZGVsZXRlU3Vic2NyaXB0aW9uID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGVuZERhdGU6IG51bWJlcik6IFByb21pc2U8Ym9vbGVhbj4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdkZWxldGVTdWJzY3JpcHRpb24nO1xuICBjb25zdCB7ZGF0YWJhc2V9ID0gY29udGV4dDtcbiAgY29uc3Qgbm93OiBudW1iZXIgPSBEYXRlLm5vdygpO1xuICBjb25zdCBmb3JtYXRFbmREYXRlOiBudW1iZXIgPSBwYXJzZU51bShlbmREYXRlKSB8fCBub3c7XG5cbiAgcmV0dXJuIGdldFN1YnNjcmlwdGlvbihjb250ZXh0KVxuICAgIC50aGVuKChzdWJzY3JpcHRpb24pID0+IHtcbiAgICAgIC8vIFN0cmlwZVxuICAgICAgY29uc3Qgc3RyaXBlQ2xpZW50ID0gc3RyaXBlKENvbmZpZy5nZXQoJ3N0cmlwZS50b2tlbicpKTtcbiAgICAgIHJldHVybiBzdHJpcGVDbGllbnQuc3Vic2NyaXB0aW9uc1xuICAgICAgICAuZGVsKHN1YnNjcmlwdGlvbi50cmFuc2FjdGlvbklkLCB7XG4gICAgICAgICAgc3Vic2NyaXB0aW9uOiBmb3JtYXRFbmREYXRlXG4gICAgICAgIH0pXG4gICAgICAgIC50aGVuKChzdHJpcGVSZXNwb25zZSkgPT4ge1xuICAgICAgICAgIC8vIE1ha2Ugc3VyZSB3ZSBjYW5jZWxsZWQgb24gU3RyaXBlIGJlZm9yZSB1cGRhdGluZyB0aGUgZGJcbiAgICAgICAgICBpZihzdHJpcGVSZXNwb25zZS5jYW5jZWxsZWQgfHwgc3RyaXBlUmVzcG9uc2UuY2FuY2VsX2F0X3BlcmlvZF9lbmQpIHtcbiAgICAgICAgICAgIGNvbnN0IHVwZGF0ZTogUGF5bWVudFN1YnNjcmlwdGlvbiA9IHtcbiAgICAgICAgICAgICAgY2FuY2VsRGF0ZTogZm9ybWF0RW5kRGF0ZSxcbiAgICAgICAgICAgICAgbW9kaWZpZWQ6IG5vdyxcbiAgICAgICAgICAgICAgcGxhbklkOiAnJ1xuICAgICAgICAgICAgfTtcbiAgICAgICAgICAgIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgVVBEQVRFIHMgV0lUSCAke3VwZGF0ZX0gSU4gc3Vic2NyaXB0aW9uc2A7XG5cbiAgICAgICAgICAgIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgICAgICAgICAgICAudGhlbigoKSA9PiB0cnVlKVxuICAgICAgICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgICAgICAgICBsYWJlbDogJ2RiX2Vycm9yJ1xuICAgICAgICAgICAgICB9LCBlcnJvciwge30pLnRoZW4oKCkgPT4gbnVsbCkuY2F0Y2goKGVycm9yKSA9PiBQcm9taXNlLnJlamVjdChlcnJvcikpKTtcbiAgICAgICAgICB9XG4gICAgICAgICAgcmV0dXJuIGZhbHNlO1xuICAgICAgICB9KTtcbiAgICB9KTtcbn07XG4iXX0=