@nlabs/reaktor 0.2.2 → 0.3.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 (63) hide show
  1. package/lib/data/conversations.d.ts +3 -1
  2. package/lib/data/dynamodb.d.ts +8 -8
  3. package/lib/data/email.d.ts +2 -2
  4. package/lib/data/images.d.ts +1 -1
  5. package/lib/data/index.d.ts +1 -0
  6. package/lib/data/ios.d.ts +5 -5
  7. package/lib/data/notifications.d.ts +1 -1
  8. package/lib/data/payments.d.ts +3 -4
  9. package/lib/data/reactions.d.ts +1 -1
  10. package/lib/data/s3.d.ts +6 -6
  11. package/lib/data/subscription.d.ts +1 -1
  12. package/lib/data/websockets.d.ts +6 -0
  13. package/lib/types/payments.d.ts +1 -0
  14. package/lib/types/posts.d.ts +2 -0
  15. package/lib/utils/arangodb.d.ts +2 -1
  16. package/package.json +26 -24
  17. package/lib/config.js +0 -145
  18. package/lib/data/conversations.js +0 -226
  19. package/lib/data/dynamodb.js +0 -174
  20. package/lib/data/email.js +0 -194
  21. package/lib/data/files.js +0 -465
  22. package/lib/data/groups.js +0 -401
  23. package/lib/data/images.js +0 -899
  24. package/lib/data/index.js +0 -234
  25. package/lib/data/ios.js +0 -329
  26. package/lib/data/locations.js +0 -148
  27. package/lib/data/messages.js +0 -283
  28. package/lib/data/notifications.js +0 -59
  29. package/lib/data/payments.js +0 -800
  30. package/lib/data/posts.js +0 -690
  31. package/lib/data/reactions.js +0 -487
  32. package/lib/data/s3.js +0 -147
  33. package/lib/data/search.js +0 -113
  34. package/lib/data/sms.js +0 -79
  35. package/lib/data/subscription.js +0 -311
  36. package/lib/data/tags.js +0 -345
  37. package/lib/data/users.js +0 -419
  38. package/lib/index.js +0 -42
  39. package/lib/types/apps.js +0 -2
  40. package/lib/types/arangodb.js +0 -2
  41. package/lib/types/auth.js +0 -2
  42. package/lib/types/conversations.js +0 -2
  43. package/lib/types/email.js +0 -2
  44. package/lib/types/files.js +0 -2
  45. package/lib/types/google.js +0 -2
  46. package/lib/types/groups.js +0 -2
  47. package/lib/types/images.js +0 -2
  48. package/lib/types/index.js +0 -210
  49. package/lib/types/locations.js +0 -2
  50. package/lib/types/messages.js +0 -2
  51. package/lib/types/notifications.js +0 -2
  52. package/lib/types/payments.js +0 -2
  53. package/lib/types/posts.js +0 -2
  54. package/lib/types/reactions.js +0 -2
  55. package/lib/types/tags.js +0 -2
  56. package/lib/types/users.js +0 -2
  57. package/lib/utils/analytics.js +0 -61
  58. package/lib/utils/arangodb.js +0 -122
  59. package/lib/utils/auth.js +0 -57
  60. package/lib/utils/graphql.js +0 -19
  61. package/lib/utils/index.js +0 -78
  62. package/lib/utils/objects.js +0 -54
  63. package/lib/utils/redis.js +0 -32
@@ -1,311 +0,0 @@
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
- userAccess = context.userAccess;
115
- var isAdmin = userAccess > 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
- userAccess = context.userAccess;
193
- var isAdmin = userAccess > 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,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kYXRhL3N1YnNjcmlwdGlvbi50cyJdLCJuYW1lcyI6WyJldmVudENhdGVnb3J5IiwiZ2V0UGxhbkxpc3QiLCJjb250ZXh0IiwiZnJvbSIsInRvIiwiYWN0aW9uIiwiZGF0YWJhc2UiLCJsaW1pdCIsImFxbFFyeSIsImFxbCIsInF1ZXJ5IiwidGhlbiIsImN1cnNvciIsImFsbCIsImxpc3QiLCJlcnJvciIsImNhdGVnb3J5IiwibGFiZWwiLCJQcm9taXNlIiwicmVqZWN0IiwiZ2V0U3Vic2NyaXB0aW9uIiwic2Vzc2lvbklkIiwidXNlcklkIiwibmV4dCIsInN1YnNjcmlwdGlvbiIsImFkZFBsYW4iLCJpdGVtIiwidXNlckFjY2VzcyIsImlzQWRtaW4iLCJ2YWx1ZSIsIm5vdyIsIkRhdGUiLCJmb3JtYXRJZCIsImlkIiwicGxhbklkIiwiYW1vdW50IiwiY3VycmVuY3kiLCJkZXNjcmlwdGlvbiIsImludGVydmFsIiwiaW50ZXJ2YWxDb3VudCIsIm5hbWUiLCJmb3JtYXRBbW91bnQiLCJmb3JtYXRJbnRlcnZhbCIsInRvTG93ZXJDYXNlIiwiZm9ybWF0SW50ZXJ2YWxDbnQiLCJmb3JtYXROYW1lIiwiZm9ybWF0RGVzYyIsImluc2VydCIsIl9rZXkiLCJhZGRlZCIsIm1vZGlmaWVkIiwicGxhbiIsInN0cmlwZUNsaWVudCIsInN0cmlwZSIsIkNvbmZpZyIsImdldCIsInBsYW5zIiwiY3JlYXRlIiwiaW50ZXJ2YWxfY291bnQiLCJtZXRhZGF0YSIsInN0YXRlbWVudF9kZXNjcmlwdG9yIiwiYWRkU3Vic2NyaXB0aW9uIiwic3Vic2NyaXB0aW9uSWQiLCJ0YXgiLCJ0cmlhbEVuZCIsInRyaWFsUGVyaW9kIiwiZm9ybWF0UGxhbklkIiwiZm9ybWF0VGF4UGVyY2VudCIsImZvcm1hdFRyaWFsUGVyaW9kIiwiZm9ybWF0VHJpYWxFbmQiLCJEYXRlVGltZSIsImxvY2FsIiwicGx1cyIsImRheXMiLCJ0b01pbGxpcyIsInVzZXIiLCJjYW5jZWxEYXRlIiwic3Vic2NyaXB0aW9ucyIsImN1c3RvbWVyIiwic3RyaXBlQ3VzdG9tZXJJZCIsIml0ZW1zIiwidGF4X3BlcmNlbnQiLCJ0cmlhbF9lbmQiLCJkZWxldGVTdWJzY3JpcHRpb24iLCJlbmREYXRlIiwiZm9ybWF0RW5kRGF0ZSIsImRlbCIsInRyYW5zYWN0aW9uSWQiLCJzdHJpcGVSZXNwb25zZSIsImNhbmNlbGxlZCIsImNhbmNlbF9hdF9wZXJpb2RfZW5kIiwidXBkYXRlIl0sIm1hcHBpbmdzIjoiOzs7Ozs7O0FBSUE7O0FBQ0E7O0FBR0E7O0FBQ0E7O0FBRUE7O0FBSUE7O0FBQ0E7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUVBLElBQU1BLGFBQXFCLEdBQUcsY0FBOUI7O0FBRU8sSUFBTUMsV0FBVyxHQUFHLFNBQWRBLFdBQWMsQ0FBQ0MsT0FBRCxFQUFvRjtBQUFBLE1BQTlEQyxJQUE4RCx1RUFBL0MsQ0FBK0M7QUFBQSxNQUE1Q0MsRUFBNEMsdUVBQS9CLEVBQStCO0FBQzdHLE1BQU1DLE1BQWMsR0FBRyxTQUF2QjtBQUQ2RyxNQUV0R0MsUUFGc0csR0FFMUZKLE9BRjBGLENBRXRHSSxRQUZzRztBQUc3RyxNQUFNQyxLQUFvQixHQUFHLHNCQUFTSixJQUFULEVBQWVDLEVBQWYsQ0FBN0I7QUFDQSxNQUFNSSxNQUFjLG1DQUNkRCxLQUFLLENBQUNFLEdBRFEsMENBQXBCO0FBS0EsU0FBTyxtQkFBTUgsUUFBTixFQUFnQkksS0FBaEIsQ0FBc0JGLE1BQXRCLEVBQ0pHLElBREksQ0FDQyxVQUFDQyxNQUFEO0FBQUEsV0FBeUJBLE1BQU0sQ0FBQ0MsR0FBUCxFQUF6QjtBQUFBLEdBREQsRUFFSkYsSUFGSSxDQUVDO0FBQUEsUUFBQ0csSUFBRCx1RUFBUSxFQUFSO0FBQUEsV0FBZUEsSUFBZjtBQUFBLEdBRkQsV0FHRSxVQUFDQyxLQUFEO0FBQUEsV0FBa0Isc0JBQVM7QUFDaENWLE1BQUFBLE1BQU0sRUFBTkEsTUFEZ0M7QUFFaENXLE1BQUFBLFFBQVEsRUFBRWhCLGFBRnNC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WUFBWSxDQUFDeUIsYUFBYixDQUNKUyxHQURJLENBQ0E5RCxZQUFZLENBQUMrRCxhQURiLEVBQzRCO0FBQy9CL0QsTUFBQUEsWUFBWSxFQUFFNkQ7QUFEaUIsS0FENUIsRUFJSjFFLElBSkksQ0FJQyxVQUFDNkUsY0FBRCxFQUFvQjtBQUN4QjtBQUNBLFVBQUdBLGNBQWMsQ0FBQ0MsU0FBZixJQUE0QkQsY0FBYyxDQUFDRSxvQkFBOUMsRUFBb0U7QUFDbEUsWUFBTUMsTUFBMkIsR0FBRztBQUNsQ2YsVUFBQUEsVUFBVSxFQUFFUyxhQURzQjtBQUVsQ25DLFVBQUFBLFFBQVEsRUFBRXBCLEdBRndCO0FBR2xDSSxVQUFBQSxNQUFNLEVBQUU7QUFIMEIsU0FBcEM7QUFLQSxZQUFNMUIsTUFBZ0IsT0FBR0MsYUFBSCxzQkFBdUJrRixNQUF2QixDQUF0QjtBQUVBLGVBQU8sbUJBQU1yRixRQUFOLEVBQWdCSSxLQUFoQixDQUFzQkYsTUFBdEIsRUFDSkcsSUFESSxDQUNDO0FBQUEsaUJBQU0sSUFBTjtBQUFBLFNBREQsV0FFRSxVQUFDSSxLQUFEO0FBQUEsaUJBQWtCLHNCQUFTO0FBQ2hDVixZQUFBQSxNQUFNLEVBQU5BLE1BRGdDO0FBRWhDVyxZQUFBQSxRQUFRLEVBQUVoQixhQUZzQjtBQUdoQ2lCLFlBQUFBLEtBQUssRUFBRTtBQUh5QixXQUFULEVBSXRCRixLQUpzQixFQUlmLEVBSmUsRUFJWEosSUFKVyxDQUlOO0FBQUEsbUJBQU0sSUFBTjtBQUFBLFdBSk0sV0FJWSxVQUFDSSxLQUFEO0FBQUEsbUJBQVdHLE9BQU8sQ0FBQ0MsTUFBUixDQUFlSixLQUFmLENBQVg7QUFBQSxXQUpaLENBQWxCO0FBQUEsU0FGRixDQUFQO0FBT0Q7O0FBQ0QsYUFBTyxLQUFQO0FBQ0QsS0F2QkksQ0FBUDtBQXdCRCxHQTVCSSxDQUFQO0FBNkJELENBbkNNIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBDb3B5cmlnaHQgKGMpIDIwMTktUHJlc2VudCwgTml0cm9nZW4gTGFicywgSW5jLlxuICogQ29weXJpZ2h0cyBsaWNlbnNlZCB1bmRlciB0aGUgTUlUIExpY2Vuc2UuIFNlZSB0aGUgYWNjb21wYW55aW5nIExJQ0VOU0UgZmlsZSBmb3IgdGVybXMuXG4gKi9cbmltcG9ydCB7Y3JlYXRlSGFzaCwgcGFyc2VDaGFyLCBwYXJzZUlkLCBwYXJzZU51bSwgcGFyc2VWYXJDaGFyfSBmcm9tICdAbmxhYnMvdXRpbHMnO1xuaW1wb3J0IHthcWx9IGZyb20gJ2FyYW5nb2pzJztcbmltcG9ydCB7QXFsUXVlcnl9IGZyb20gJ2FyYW5nb2pzL2xpYi9janMvYXFsLXF1ZXJ5JztcbmltcG9ydCB7QXJyYXlDdXJzb3J9IGZyb20gJ2FyYW5nb2pzL2xpYi9janMvY3Vyc29yJztcbmltcG9ydCB7RGF0ZVRpbWV9IGZyb20gJ2x1eG9uJztcbmltcG9ydCAqIGFzIHN0cmlwZSBmcm9tICdzdHJpcGUnO1xuXG5pbXBvcnQge0NvbmZpZ30gZnJvbSAnLi4vY29uZmlnJztcbmltcG9ydCB7QXJhbmdvREJMaW1pdH0gZnJvbSAnLi4vdHlwZXMvYXJhbmdvZGInO1xuaW1wb3J0IHtBcGlDb250ZXh0fSBmcm9tICcuLi90eXBlcy9hdXRoJztcbmltcG9ydCB7UGF5bWVudEludGVydmFsLCBQYXltZW50UGxhbiwgUGF5bWVudFN1YnNjcmlwdGlvbn0gZnJvbSAnLi4vdHlwZXMvcGF5bWVudHMnO1xuaW1wb3J0IHtnZXRMaW1pdCwgbG9nRXJyb3IsIGxvZ0V4Y2VwdGlvbiwgdXNlRGJ9IGZyb20gJy4uL3V0aWxzJztcbmltcG9ydCB7Z2V0VXNlcn0gZnJvbSAnLi91c2Vycyc7XG5cbmNvbnN0IGV2ZW50Q2F0ZWdvcnk6IHN0cmluZyA9ICdzdWJzY3JpcHRpb24nO1xuXG5leHBvcnQgY29uc3QgZ2V0UGxhbkxpc3QgPSAoY29udGV4dDogQXBpQ29udGV4dCwgZnJvbTogbnVtYmVyID0gMCwgdG86IG51bWJlciA9IDMwKTogUHJvbWlzZTxQYXltZW50UGxhbltdPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2dldExpc3QnO1xuICBjb25zdCB7ZGF0YWJhc2V9ID0gY29udGV4dDtcbiAgY29uc3QgbGltaXQ6IEFyYW5nb0RCTGltaXQgPSBnZXRMaW1pdChmcm9tLCB0byk7XG4gIGNvbnN0IGFxbFFyeTogc3RyaW5nID0gYEZPUiBwIElOIHBsYW5zXG4gICAgICAke2xpbWl0LmFxbH1cbiAgICAgIFNPUlQgcC5hbW91bnRcbiAgICAgIFJFVFVSTiBwYDtcblxuICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLmFsbCgpKVxuICAgIC50aGVuKChsaXN0ID0gW10pID0+IGxpc3QpXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgbGFiZWw6ICdkYl9lcnJvcidcbiAgICB9LCBlcnJvciwge30pLnRoZW4oKCkgPT4gbnVsbCkuY2F0Y2goKGVycm9yKSA9PiBQcm9taXNlLnJlamVjdChlcnJvcikpKTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRTdWJzY3JpcHRpb24gPSAoY29udGV4dDogQXBpQ29udGV4dCk6IFByb21pc2U8UGF5bWVudFN1YnNjcmlwdGlvbj4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRJdGVtJztcbiAgY29uc3Qge2RhdGFiYXNlLCB1c2VySWQ6IHNlc3Npb25JZH0gPSBjb250ZXh0O1xuICBjb25zdCBhcWxRcnk6IHN0cmluZyA9IGBGT1IgcyBJTiBzdWJzY3JpcHRpb25zXG4gICAgICBGSUxURVIgcy51c2VySWQgPT0gJHtzZXNzaW9uSWR9ICYmIHMuY2FuY2VsbGVkID4gMFxuICAgICAgTEVUIHBsYW4gPSBGSVJTVChcbiAgICAgICAgRk9SIHAgSU4gcGxhbnNcbiAgICAgICAgRklMVEVSIHAuX2tleSA9PSBzLnBsYW5JZFxuICAgICAgKVxuICAgICAgTElNSVQgMVxuICAgICAgUkVUVVJOIE1FUkdFKHMsIHtwbGFuOiBwbGFufSlgO1xuXG4gIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgIC50aGVuKChzdWJzY3JpcHRpb24gPSB7fSkgPT4gc3Vic2NyaXB0aW9uKVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICBhY3Rpb24sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIGxhYmVsOiAnZGJfZXJyb3InXG4gICAgfSwgZXJyb3IsIHt9KS50aGVuKCgpID0+IG51bGwpLmNhdGNoKChlcnJvcikgPT4gUHJvbWlzZS5yZWplY3QoZXJyb3IpKSk7XG59O1xuXG5leHBvcnQgY29uc3QgYWRkUGxhbiA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBpdGVtOiBQYXltZW50UGxhbik6IFByb21pc2U8UGF5bWVudFBsYW4+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnYWRkUGxhbic7XG4gIGNvbnN0IHtkYXRhYmFzZSwgdXNlcklkOiBzZXNzaW9uSWQsIHVzZXJBY2Nlc3N9ID0gY29udGV4dDtcbiAgY29uc3QgaXNBZG1pbjogYm9vbGVhbiA9IHVzZXJBY2Nlc3MgPiAyO1xuXG4gIGlmKCFpc0FkbWluKSB7XG4gICAgcmV0dXJuIGxvZ0V4Y2VwdGlvbih7XG4gICAgICBhY3Rpb24sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIGxhYmVsOiAndW5hdXRob3JpemVkJyxcbiAgICAgIHZhbHVlOiAnaW52YWxpZF9zZXNzaW9uJ1xuICAgIH0sIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCk7XG4gIH1cblxuICBjb25zdCBub3c6IG51bWJlciA9IERhdGUubm93KCk7XG4gIGNvbnN0IGZvcm1hdElkOiBzdHJpbmcgPSBwYXJzZUlkKGl0ZW0uaWQpO1xuICBjb25zdCBwbGFuSWQ6IHN0cmluZyA9IGZvcm1hdElkID09PSAnJyA/IGNyZWF0ZUhhc2goYHRhZy0ke3Nlc3Npb25JZH1gKSA6IGZvcm1hdElkO1xuICBjb25zdCB7YW1vdW50LCBjdXJyZW5jeSA9ICdVU0QnLCBkZXNjcmlwdGlvbiwgaW50ZXJ2YWwsIGludGVydmFsQ291bnQsIG5hbWV9ID0gaXRlbTtcbiAgY29uc3QgZm9ybWF0QW1vdW50OiBudW1iZXIgPSBwYXJzZU51bShhbW91bnQpO1xuICBjb25zdCBmb3JtYXRJbnRlcnZhbDogUGF5bWVudEludGVydmFsID0gcGFyc2VDaGFyKGludGVydmFsLCA1KS50b0xvd2VyQ2FzZSgpIGFzIFBheW1lbnRJbnRlcnZhbDtcbiAgY29uc3QgZm9ybWF0SW50ZXJ2YWxDbnQ6IG51bWJlciA9IHBhcnNlTnVtKGludGVydmFsQ291bnQsIDUpO1xuICBjb25zdCBmb3JtYXROYW1lOiBzdHJpbmcgPSBwYXJzZVZhckNoYXIobmFtZSwgMzIpO1xuICBjb25zdCBmb3JtYXREZXNjOiBzdHJpbmcgPSBwYXJzZVZhckNoYXIoZGVzY3JpcHRpb24sIDMyKTtcblxuICBjb25zdCBpbnNlcnQ6IFBheW1lbnRQbGFuID0ge1xuICAgIF9rZXk6IHBsYW5JZCxcbiAgICBhZGRlZDogbm93LFxuICAgIGFtb3VudDogZm9ybWF0QW1vdW50LFxuICAgIGN1cnJlbmN5LFxuICAgIGRlc2NyaXB0aW9uOiBmb3JtYXREZXNjLFxuICAgIGludGVydmFsOiBmb3JtYXRJbnRlcnZhbCxcbiAgICBpbnRlcnZhbENvdW50OiBmb3JtYXRJbnRlcnZhbENudCxcbiAgICBtb2RpZmllZDogbm93LFxuICAgIG5hbWU6IGZvcm1hdE5hbWVcbiAgfTtcbiAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBJTlNFUlQgJHtpbnNlcnR9IElOIHBsYW5zIFJFVFVSTiBORVdgO1xuXG4gIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgIC50aGVuKChwbGFuOiBQYXltZW50UGxhbiA9IHt9KSA9PiB7XG4gICAgICAvLyBTdHJpcGVcbiAgICAgIGNvbnN0IHN0cmlwZUNsaWVudCA9IHN0cmlwZShDb25maWcuZ2V0KCdzdHJpcGUudG9rZW4nKSk7XG4gICAgICByZXR1cm4gc3RyaXBlQ2xpZW50LnBsYW5zXG4gICAgICAgIC5jcmVhdGUoe1xuICAgICAgICAgIGFtb3VudDogZm9ybWF0QW1vdW50ICogMTAwLFxuICAgICAgICAgIGN1cnJlbmN5LFxuICAgICAgICAgIGlkOiBwbGFuSWQsXG4gICAgICAgICAgaW50ZXJ2YWw6IGZvcm1hdEludGVydmFsLFxuICAgICAgICAgIGludGVydmFsX2NvdW50OiBmb3JtYXRJbnRlcnZhbENudCxcbiAgICAgICAgICBtZXRhZGF0YToge1xuICAgICAgICAgIH0sXG4gICAgICAgICAgbmFtZTogZm9ybWF0TmFtZSxcbiAgICAgICAgICBzdGF0ZW1lbnRfZGVzY3JpcHRvcjogZm9ybWF0RGVzY1xuICAgICAgICB9KVxuICAgICAgICAudGhlbigoKSA9PiBwbGFuKTtcbiAgICB9KVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICBhY3Rpb24sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIGxhYmVsOiAnZGJfZXJyb3InXG4gICAgfSwgZXJyb3IsIHt9KS50aGVuKCgpID0+IG51bGwpLmNhdGNoKChlcnJvcikgPT4gUHJvbWlzZS5yZWplY3QoZXJyb3IpKSk7XG59O1xuXG5leHBvcnQgY29uc3QgYWRkU3Vic2NyaXB0aW9uID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGl0ZW0pOiBQcm9taXNlPFBheW1lbnRTdWJzY3JpcHRpb24+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnYWRkU3Vic2NyaXB0aW9uJztcbiAgY29uc3Qge2RhdGFiYXNlLCB1c2VySWQ6IHNlc3Npb25JZCwgdXNlckFjY2Vzc30gPSBjb250ZXh0O1xuICBjb25zdCBpc0FkbWluOiBib29sZWFuID0gdXNlckFjY2VzcyA+IDI7XG5cbiAgaWYoIWlzQWRtaW4pIHtcbiAgICByZXR1cm4gbG9nRXhjZXB0aW9uKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgbGFiZWw6ICd1bmF1dGhvcml6ZWQnLFxuICAgICAgdmFsdWU6ICdpbnZhbGlkX3Nlc3Npb24nXG4gICAgfSwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKTtcbiAgfVxuXG4gIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcbiAgY29uc3QgZm9ybWF0SWQ6IHN0cmluZyA9IHBhcnNlSWQoaXRlbS5pZCk7XG4gIGNvbnN0IHN1YnNjcmlwdGlvbklkOiBzdHJpbmcgPSBmb3JtYXRJZCA9PT0gJycgPyBjcmVhdGVIYXNoKGB0YWctJHtzZXNzaW9uSWR9YCkgOiBmb3JtYXRJZDtcbiAgY29uc3Qge3BsYW5JZCwgdGF4LCB0cmlhbEVuZCwgdHJpYWxQZXJpb2R9ID0gaXRlbTtcbiAgY29uc3QgZm9ybWF0UGxhbklkOiBzdHJpbmcgPSBwYXJzZUlkKHBsYW5JZCk7XG4gIGNvbnN0IGZvcm1hdFRheFBlcmNlbnQ6IG51bWJlciA9IHBhcnNlTnVtKHRheCkgfHwgMDtcbiAgY29uc3QgZm9ybWF0VHJpYWxQZXJpb2Q6IG51bWJlciA9IHBhcnNlTnVtKHRheCkgfHwgMDtcbiAgbGV0IGZvcm1hdFRyaWFsRW5kOiBudW1iZXIgPSB0cmlhbEVuZCB8fCBEYXRlLm5vdygpO1xuXG4gIGlmKGZvcm1hdFRyaWFsUGVyaW9kKSB7XG4gICAgZm9ybWF0VHJpYWxFbmQgPSBEYXRlVGltZS5sb2NhbCgpLnBsdXMoe2RheXM6IHRyaWFsUGVyaW9kfSkudG9NaWxsaXMoKTtcbiAgfVxuXG4gIHJldHVybiBnZXRVc2VyKGNvbnRleHQsIHNlc3Npb25JZClcbiAgICAudGhlbigodXNlcikgPT4ge1xuICAgICAgY29uc3QgaW5zZXJ0OiBQYXltZW50U3Vic2NyaXB0aW9uID0ge1xuICAgICAgICBfa2V5OiBzdWJzY3JpcHRpb25JZCxcbiAgICAgICAgYWRkZWQ6IG5vdyxcbiAgICAgICAgY2FuY2VsRGF0ZTogMCxcbiAgICAgICAgbW9kaWZpZWQ6IG5vdyxcbiAgICAgICAgcGxhbklkOiBmb3JtYXRQbGFuSWQsXG4gICAgICAgIHRheDogZm9ybWF0VGF4UGVyY2VudCxcbiAgICAgICAgdHJpYWxFbmQ6IGZvcm1hdFRyaWFsRW5kLFxuICAgICAgICB1c2VySWQ6IHNlc3Npb25JZFxuICAgICAgfTtcbiAgICAgIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgSU5TRVJUICR7aW5zZXJ0fSBJTiBwbGFucyBSRVRVUk4gTkVXYDtcblxuICAgICAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgICAgICAudGhlbigoc3Vic2NyaXB0aW9uOiBQYXltZW50U3Vic2NyaXB0aW9uID0ge30pID0+IHtcbiAgICAgICAgICAvLyBTdHJpcGVcbiAgICAgICAgICBjb25zdCBzdHJpcGVDbGllbnQgPSBzdHJpcGUoQ29uZmlnLmdldCgnc3RyaXBlLnRva2VuJykpO1xuICAgICAgICAgIHJldHVybiBzdHJpcGVDbGllbnQuc3Vic2NyaXB0aW9uc1xuICAgICAgICAgICAgLmNyZWF0ZSh7XG4gICAgICAgICAgICAgIGN1c3RvbWVyOiB1c2VyLnN0cmlwZUN1c3RvbWVySWQsXG4gICAgICAgICAgICAgIGl0ZW1zOiBbXG4gICAgICAgICAgICAgICAge3BsYW46IGZvcm1hdFBsYW5JZH1cbiAgICAgICAgICAgICAgXSxcbiAgICAgICAgICAgICAgbWV0YWRhdGE6IHtcbiAgICAgICAgICAgICAgICB1c2VySWQ6IHNlc3Npb25JZFxuICAgICAgICAgICAgICB9LFxuICAgICAgICAgICAgICB0YXhfcGVyY2VudDogZm9ybWF0VGF4UGVyY2VudCAqIDEwMCxcbiAgICAgICAgICAgICAgdHJpYWxfZW5kOiBmb3JtYXRUcmlhbEVuZFxuICAgICAgICAgICAgfSlcbiAgICAgICAgICAgIC50aGVuKCgpID0+IHN1YnNjcmlwdGlvbik7XG4gICAgICAgIH0pXG4gICAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgICAgYWN0aW9uLFxuICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgIGxhYmVsOiAnZGJfZXJyb3InXG4gICAgICAgIH0sIGVycm9yLCB7fSkudGhlbigoKSA9PiBudWxsKS5jYXRjaCgoZXJyb3IpID0+IFByb21pc2UucmVqZWN0KGVycm9yKSkpO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGRlbGV0ZVN1YnNjcmlwdGlvbiA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBlbmREYXRlOiBudW1iZXIpOiBQcm9taXNlPGJvb2xlYW4+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnZGVsZXRlU3Vic2NyaXB0aW9uJztcbiAgY29uc3Qge2RhdGFiYXNlfSA9IGNvbnRleHQ7XG4gIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcbiAgY29uc3QgZm9ybWF0RW5kRGF0ZTogbnVtYmVyID0gcGFyc2VOdW0oZW5kRGF0ZSkgfHwgbm93O1xuXG4gIHJldHVybiBnZXRTdWJzY3JpcHRpb24oY29udGV4dClcbiAgICAudGhlbigoc3Vic2NyaXB0aW9uKSA9PiB7XG4gICAgICAvLyBTdHJpcGVcbiAgICAgIGNvbnN0IHN0cmlwZUNsaWVudCA9IHN0cmlwZShDb25maWcuZ2V0KCdzdHJpcGUudG9rZW4nKSk7XG4gICAgICByZXR1cm4gc3RyaXBlQ2xpZW50LnN1YnNjcmlwdGlvbnNcbiAgICAgICAgLmRlbChzdWJzY3JpcHRpb24udHJhbnNhY3Rpb25JZCwge1xuICAgICAgICAgIHN1YnNjcmlwdGlvbjogZm9ybWF0RW5kRGF0ZVxuICAgICAgICB9KVxuICAgICAgICAudGhlbigoc3RyaXBlUmVzcG9uc2UpID0+IHtcbiAgICAgICAgICAvLyBNYWtlIHN1cmUgd2UgY2FuY2VsbGVkIG9uIFN0cmlwZSBiZWZvcmUgdXBkYXRpbmcgdGhlIGRiXG4gICAgICAgICAgaWYoc3RyaXBlUmVzcG9uc2UuY2FuY2VsbGVkIHx8IHN0cmlwZVJlc3BvbnNlLmNhbmNlbF9hdF9wZXJpb2RfZW5kKSB7XG4gICAgICAgICAgICBjb25zdCB1cGRhdGU6IFBheW1lbnRTdWJzY3JpcHRpb24gPSB7XG4gICAgICAgICAgICAgIGNhbmNlbERhdGU6IGZvcm1hdEVuZERhdGUsXG4gICAgICAgICAgICAgIG1vZGlmaWVkOiBub3csXG4gICAgICAgICAgICAgIHBsYW5JZDogJydcbiAgICAgICAgICAgIH07XG4gICAgICAgICAgICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYFVQREFURSBzIFdJVEggJHt1cGRhdGV9IElOIHN1YnNjcmlwdGlvbnNgO1xuXG4gICAgICAgICAgICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAgICAgICAgICAgLnRoZW4oKCkgPT4gdHJ1ZSlcbiAgICAgICAgICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgICAgICAgbGFiZWw6ICdkYl9lcnJvcidcbiAgICAgICAgICAgICAgfSwgZXJyb3IsIHt9KS50aGVuKCgpID0+IG51bGwpLmNhdGNoKChlcnJvcikgPT4gUHJvbWlzZS5yZWplY3QoZXJyb3IpKSk7XG4gICAgICAgICAgfVxuICAgICAgICAgIHJldHVybiBmYWxzZTtcbiAgICAgICAgfSk7XG4gICAgfSk7XG59O1xuIl19