@nlabs/reaktor 0.1.10 → 0.1.12

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 (48) hide show
  1. package/package.json +14 -14
  2. package/lib/config.js +0 -139
  3. package/lib/data/conversations.js +0 -228
  4. package/lib/data/dynamodb.js +0 -172
  5. package/lib/data/email.js +0 -194
  6. package/lib/data/files.js +0 -463
  7. package/lib/data/groups.js +0 -401
  8. package/lib/data/images.js +0 -841
  9. package/lib/data/index.js +0 -234
  10. package/lib/data/ios.js +0 -327
  11. package/lib/data/locations.js +0 -148
  12. package/lib/data/messages.js +0 -281
  13. package/lib/data/notifications.js +0 -59
  14. package/lib/data/payments.js +0 -798
  15. package/lib/data/posts.js +0 -637
  16. package/lib/data/reactions.js +0 -243
  17. package/lib/data/s3.js +0 -133
  18. package/lib/data/search.js +0 -111
  19. package/lib/data/sms.js +0 -79
  20. package/lib/data/subscription.js +0 -311
  21. package/lib/data/tags.js +0 -343
  22. package/lib/data/users.js +0 -415
  23. package/lib/index.js +0 -42
  24. package/lib/types/apps.js +0 -2
  25. package/lib/types/arangodb.js +0 -2
  26. package/lib/types/auth.js +0 -2
  27. package/lib/types/conversations.js +0 -2
  28. package/lib/types/email.js +0 -2
  29. package/lib/types/files.js +0 -2
  30. package/lib/types/google.js +0 -2
  31. package/lib/types/groups.js +0 -2
  32. package/lib/types/images.js +0 -2
  33. package/lib/types/index.js +0 -210
  34. package/lib/types/locations.js +0 -2
  35. package/lib/types/messages.js +0 -2
  36. package/lib/types/notifications.js +0 -2
  37. package/lib/types/payments.js +0 -2
  38. package/lib/types/posts.js +0 -2
  39. package/lib/types/reactions.js +0 -2
  40. package/lib/types/tags.js +0 -2
  41. package/lib/types/users.js +0 -2
  42. package/lib/utils/analytics.js +0 -59
  43. package/lib/utils/arangodb.js +0 -131
  44. package/lib/utils/auth.js +0 -104
  45. package/lib/utils/graphql.js +0 -19
  46. package/lib/utils/index.js +0 -78
  47. package/lib/utils/objects.js +0 -54
  48. package/lib/utils/redis.js +0 -28
@@ -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
- 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,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kYXRhL3N1YnNjcmlwdGlvbi50cyJdLCJuYW1lcyI6WyJldmVudENhdGVnb3J5IiwiZ2V0UGxhbkxpc3QiLCJjb250ZXh0IiwiZnJvbSIsInRvIiwiYWN0aW9uIiwiZGF0YWJhc2UiLCJsaW1pdCIsImFxbFFyeSIsImFxbCIsInF1ZXJ5IiwidGhlbiIsImN1cnNvciIsImFsbCIsImxpc3QiLCJjYXRjaCIsImVycm9yIiwiY2F0ZWdvcnkiLCJsYWJlbCIsIlByb21pc2UiLCJyZWplY3QiLCJnZXRTdWJzY3JpcHRpb24iLCJzZXNzaW9uSWQiLCJ1c2VySWQiLCJuZXh0Iiwic3Vic2NyaXB0aW9uIiwiYWRkUGxhbiIsIml0ZW0iLCJ1c2VyVHlwZSIsImlzQWRtaW4iLCJ2YWx1ZSIsIm5vdyIsIkRhdGUiLCJmb3JtYXRJZCIsImlkIiwicGxhbklkIiwiYW1vdW50IiwiY3VycmVuY3kiLCJkZXNjcmlwdGlvbiIsImludGVydmFsIiwiaW50ZXJ2YWxDb3VudCIsIm5hbWUiLCJmb3JtYXRBbW91bnQiLCJmb3JtYXRJbnRlcnZhbCIsInRvTG93ZXJDYXNlIiwiZm9ybWF0SW50ZXJ2YWxDbnQiLCJmb3JtYXROYW1lIiwiZm9ybWF0RGVzYyIsImluc2VydCIsIl9rZXkiLCJhZGRlZCIsIm1vZGlmaWVkIiwicGxhbiIsInN0cmlwZUNsaWVudCIsInN0cmlwZSIsIkNvbmZpZyIsImdldCIsInBsYW5zIiwiY3JlYXRlIiwiaW50ZXJ2YWxfY291bnQiLCJtZXRhZGF0YSIsInN0YXRlbWVudF9kZXNjcmlwdG9yIiwiYWRkU3Vic2NyaXB0aW9uIiwic3Vic2NyaXB0aW9uSWQiLCJ0YXgiLCJ0cmlhbEVuZCIsInRyaWFsUGVyaW9kIiwiZm9ybWF0UGxhbklkIiwiZm9ybWF0VGF4UGVyY2VudCIsImZvcm1hdFRyaWFsUGVyaW9kIiwiZm9ybWF0VHJpYWxFbmQiLCJEYXRlVGltZSIsImxvY2FsIiwicGx1cyIsImRheXMiLCJ0b01pbGxpcyIsInVzZXIiLCJjYW5jZWxEYXRlIiwic3Vic2NyaXB0aW9ucyIsImN1c3RvbWVyIiwic3RyaXBlQ3VzdG9tZXJJZCIsIml0ZW1zIiwidGF4X3BlcmNlbnQiLCJ0cmlhbF9lbmQiLCJkZWxldGVTdWJzY3JpcHRpb24iLCJlbmREYXRlIiwiZm9ybWF0RW5kRGF0ZSIsImRlbCIsInRyYW5zYWN0aW9uSWQiLCJzdHJpcGVSZXNwb25zZSIsImNhbmNlbGxlZCIsImNhbmNlbF9hdF9wZXJpb2RfZW5kIiwidXBkYXRlIl0sIm1hcHBpbmdzIjoiOzs7Ozs7O0FBSUE7O0FBQ0E7O0FBR0E7O0FBQ0E7O0FBRUE7O0FBSUE7O0FBQ0E7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUVBLElBQU1BLGFBQXFCLEdBQUcsY0FBOUI7O0FBRU8sSUFBTUMsV0FBVyxHQUFHLFNBQWRBLFdBQWMsQ0FBQ0MsT0FBRCxFQUFvRjtBQUFBLE1BQTlEQyxJQUE4RCx1RUFBL0MsQ0FBK0M7QUFBQSxNQUE1Q0MsRUFBNEMsdUVBQS9CLEVBQStCO0FBQzdHLE1BQU1DLE1BQWMsR0FBRyxTQUF2QjtBQUQ2RyxNQUV0R0MsUUFGc0csR0FFMUZKLE9BRjBGLENBRXRHSSxRQUZzRztBQUc3RyxNQUFNQyxLQUFvQixHQUFHLHNCQUFTSixJQUFULEVBQWVDLEVBQWYsQ0FBN0I7QUFDQSxNQUFNSSxNQUFjLG1DQUNkRCxLQUFLLENBQUNFLEdBRFEsMENBQXBCO0FBS0EsU0FBTyxtQkFBTUgsUUFBTixFQUFnQkksS0FBaEIsQ0FBc0JGLE1BQXRCLEVBQ0pHLElBREksQ0FDQyxVQUFDQyxNQUFEO0FBQUEsV0FBeUJBLE1BQU0sQ0FBQ0MsR0FBUCxFQUF6QjtBQUFBLEdBREQsRUFFSkYsSUFGSSxDQUVDO0FBQUEsUUFBQ0csSUFBRCx1RUFBUSxFQUFSO0FBQUEsV0FBZUEsSUFBZjtBQUFBLEdBRkQsRUFHSkMsS0FISSxDQUdFLFVBQUNDLEtBQUQ7QUFBQSxXQUFrQixzQkFBUztBQUNoQ1gsTUFBQUEsTUFBTSxFQUFOQSxNQURnQztBQUVoQ1ksTUFBQUEs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QVA7QUF3QkQsR0E1QkksQ0FBUDtBQTZCRCxDQW5DTSIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQ29weXJpZ2h0IChjKSAyMDE5LVByZXNlbnQsIE5pdHJvZ2VuIExhYnMsIEluYy5cbiAqIENvcHlyaWdodHMgbGljZW5zZWQgdW5kZXIgdGhlIE1JVCBMaWNlbnNlLiBTZWUgdGhlIGFjY29tcGFueWluZyBMSUNFTlNFIGZpbGUgZm9yIHRlcm1zLlxuICovXG5pbXBvcnQge2NyZWF0ZUhhc2gsIHBhcnNlQ2hhciwgcGFyc2VJZCwgcGFyc2VOdW0sIHBhcnNlVmFyQ2hhcn0gZnJvbSAnQG5sYWJzL3V0aWxzJztcbmltcG9ydCB7YXFsfSBmcm9tICdhcmFuZ29qcyc7XG5pbXBvcnQge0FxbFF1ZXJ5fSBmcm9tICdhcmFuZ29qcy9saWIvY2pzL2FxbC1xdWVyeSc7XG5pbXBvcnQge0FycmF5Q3Vyc29yfSBmcm9tICdhcmFuZ29qcy9saWIvY2pzL2N1cnNvcic7XG5pbXBvcnQge0RhdGVUaW1lfSBmcm9tICdsdXhvbic7XG5pbXBvcnQgKiBhcyBzdHJpcGUgZnJvbSAnc3RyaXBlJztcblxuaW1wb3J0IHtDb25maWd9IGZyb20gJy4uL2NvbmZpZyc7XG5pbXBvcnQge0FyYW5nb0RCTGltaXR9IGZyb20gJy4uL3R5cGVzL2FyYW5nb2RiJztcbmltcG9ydCB7QXBpQ29udGV4dH0gZnJvbSAnLi4vdHlwZXMvYXV0aCc7XG5pbXBvcnQge1BheW1lbnRJbnRlcnZhbCwgUGF5bWVudFBsYW4sIFBheW1lbnRTdWJzY3JpcHRpb259IGZyb20gJy4uL3R5cGVzL3BheW1lbnRzJztcbmltcG9ydCB7Z2V0TGltaXQsIGxvZ0Vycm9yLCBsb2dFeGNlcHRpb24sIHVzZURifSBmcm9tICcuLi91dGlscyc7XG5pbXBvcnQge2dldFVzZXJ9IGZyb20gJy4vdXNlcnMnO1xuXG5jb25zdCBldmVudENhdGVnb3J5OiBzdHJpbmcgPSAnc3Vic2NyaXB0aW9uJztcblxuZXhwb3J0IGNvbnN0IGdldFBsYW5MaXN0ID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGZyb206IG51bWJlciA9IDAsIHRvOiBudW1iZXIgPSAzMCk6IFByb21pc2U8UGF5bWVudFBsYW5bXT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRMaXN0JztcbiAgY29uc3Qge2RhdGFiYXNlfSA9IGNvbnRleHQ7XG4gIGNvbnN0IGxpbWl0OiBBcmFuZ29EQkxpbWl0ID0gZ2V0TGltaXQoZnJvbSwgdG8pO1xuICBjb25zdCBhcWxRcnk6IHN0cmluZyA9IGBGT1IgcCBJTiBwbGFuc1xuICAgICAgJHtsaW1pdC5hcWx9XG4gICAgICBTT1JUIHAuYW1vdW50XG4gICAgICBSRVRVUk4gcGA7XG5cbiAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5hbGwoKSlcbiAgICAudGhlbigobGlzdCA9IFtdKSA9PiBsaXN0KVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICBhY3Rpb24sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIGxhYmVsOiAnZGJfZXJyb3InXG4gICAgfSwgZXJyb3IsIHt9KS50aGVuKCgpID0+IG51bGwpLmNhdGNoKChlcnJvcikgPT4gUHJvbWlzZS5yZWplY3QoZXJyb3IpKSk7XG59O1xuXG5leHBvcnQgY29uc3QgZ2V0U3Vic2NyaXB0aW9uID0gKGNvbnRleHQ6IEFwaUNvbnRleHQpOiBQcm9taXNlPFBheW1lbnRTdWJzY3JpcHRpb24+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnZ2V0SXRlbSc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgdXNlcklkOiBzZXNzaW9uSWR9ID0gY29udGV4dDtcbiAgY29uc3QgYXFsUXJ5OiBzdHJpbmcgPSBgRk9SIHMgSU4gc3Vic2NyaXB0aW9uc1xuICAgICAgRklMVEVSIHMudXNlcklkID09ICR7c2Vzc2lvbklkfSAmJiBzLmNhbmNlbGxlZCA+IDBcbiAgICAgIExFVCBwbGFuID0gRklSU1QoXG4gICAgICAgIEZPUiBwIElOIHBsYW5zXG4gICAgICAgIEZJTFRFUiBwLl9rZXkgPT0gcy5wbGFuSWRcbiAgICAgIClcbiAgICAgIExJTUlUIDFcbiAgICAgIFJFVFVSTiBNRVJHRShzLCB7cGxhbjogcGxhbn0pYDtcblxuICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAudGhlbigoc3Vic2NyaXB0aW9uID0ge30pID0+IHN1YnNjcmlwdGlvbilcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgYWN0aW9uLFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICBsYWJlbDogJ2RiX2Vycm9yJ1xuICAgIH0sIGVycm9yLCB7fSkudGhlbigoKSA9PiBudWxsKS5jYXRjaCgoZXJyb3IpID0+IFByb21pc2UucmVqZWN0KGVycm9yKSkpO1xufTtcblxuZXhwb3J0IGNvbnN0IGFkZFBsYW4gPSAoY29udGV4dDogQXBpQ29udGV4dCwgaXRlbTogUGF5bWVudFBsYW4pOiBQcm9taXNlPFBheW1lbnRQbGFuPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2FkZFBsYW4nO1xuICBjb25zdCB7ZGF0YWJhc2UsIHVzZXJJZDogc2Vzc2lvbklkLCB1c2VyVHlwZX0gPSBjb250ZXh0O1xuICBjb25zdCBpc0FkbWluOiBib29sZWFuID0gdXNlclR5cGUgPiAyO1xuXG4gIGlmKCFpc0FkbWluKSB7XG4gICAgcmV0dXJuIGxvZ0V4Y2VwdGlvbih7XG4gICAgICBhY3Rpb24sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIGxhYmVsOiAndW5hdXRob3JpemVkJyxcbiAgICAgIHZhbHVlOiAnaW52YWxpZF9zZXNzaW9uJ1xuICAgIH0sIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCk7XG4gIH1cblxuICBjb25zdCBub3c6IG51bWJlciA9IERhdGUubm93KCk7XG4gIGNvbnN0IGZvcm1hdElkOiBzdHJpbmcgPSBwYXJzZUlkKGl0ZW0uaWQpO1xuICBjb25zdCBwbGFuSWQ6IHN0cmluZyA9IGZvcm1hdElkID09PSAnJyA/IGNyZWF0ZUhhc2goYHRhZy0ke3Nlc3Npb25JZH1gKSA6IGZvcm1hdElkO1xuICBjb25zdCB7YW1vdW50LCBjdXJyZW5jeSA9ICdVU0QnLCBkZXNjcmlwdGlvbiwgaW50ZXJ2YWwsIGludGVydmFsQ291bnQsIG5hbWV9ID0gaXRlbTtcbiAgY29uc3QgZm9ybWF0QW1vdW50OiBudW1iZXIgPSBwYXJzZU51bShhbW91bnQpO1xuICBjb25zdCBmb3JtYXRJbnRlcnZhbDogUGF5bWVudEludGVydmFsID0gcGFyc2VDaGFyKGludGVydmFsLCA1KS50b0xvd2VyQ2FzZSgpIGFzIFBheW1lbnRJbnRlcnZhbDtcbiAgY29uc3QgZm9ybWF0SW50ZXJ2YWxDbnQ6IG51bWJlciA9IHBhcnNlTnVtKGludGVydmFsQ291bnQsIDUpO1xuICBjb25zdCBmb3JtYXROYW1lOiBzdHJpbmcgPSBwYXJzZVZhckNoYXIobmFtZSwgMzIpO1xuICBjb25zdCBmb3JtYXREZXNjOiBzdHJpbmcgPSBwYXJzZVZhckNoYXIoZGVzY3JpcHRpb24sIDMyKTtcblxuICBjb25zdCBpbnNlcnQ6IFBheW1lbnRQbGFuID0ge1xuICAgIF9rZXk6IHBsYW5JZCxcbiAgICBhZGRlZDogbm93LFxuICAgIGFtb3VudDogZm9ybWF0QW1vdW50LFxuICAgIGN1cnJlbmN5LFxuICAgIGRlc2NyaXB0aW9uOiBmb3JtYXREZXNjLFxuICAgIGludGVydmFsOiBmb3JtYXRJbnRlcnZhbCxcbiAgICBpbnRlcnZhbENvdW50OiBmb3JtYXRJbnRlcnZhbENudCxcbiAgICBtb2RpZmllZDogbm93LFxuICAgIG5hbWU6IGZvcm1hdE5hbWVcbiAgfTtcbiAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBJTlNFUlQgJHtpbnNlcnR9IElOIHBsYW5zIFJFVFVSTiBORVdgO1xuXG4gIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgIC50aGVuKChwbGFuOiBQYXltZW50UGxhbiA9IHt9KSA9PiB7XG4gICAgICAvLyBTdHJpcGVcbiAgICAgIGNvbnN0IHN0cmlwZUNsaWVudCA9IHN0cmlwZShDb25maWcuZ2V0KCdzdHJpcGUudG9rZW4nKSk7XG4gICAgICByZXR1cm4gc3RyaXBlQ2xpZW50LnBsYW5zXG4gICAgICAgIC5jcmVhdGUoe1xuICAgICAgICAgIGFtb3VudDogZm9ybWF0QW1vdW50ICogMTAwLFxuICAgICAgICAgIGN1cnJlbmN5LFxuICAgICAgICAgIGlkOiBwbGFuSWQsXG4gICAgICAgICAgaW50ZXJ2YWw6IGZvcm1hdEludGVydmFsLFxuICAgICAgICAgIGludGVydmFsX2NvdW50OiBmb3JtYXRJbnRlcnZhbENudCxcbiAgICAgICAgICBtZXRhZGF0YToge1xuICAgICAgICAgIH0sXG4gICAgICAgICAgbmFtZTogZm9ybWF0TmFtZSxcbiAgICAgICAgICBzdGF0ZW1lbnRfZGVzY3JpcHRvcjogZm9ybWF0RGVzY1xuICAgICAgICB9KVxuICAgICAgICAudGhlbigoKSA9PiBwbGFuKTtcbiAgICB9KVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICBhY3Rpb24sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIGxhYmVsOiAnZGJfZXJyb3InXG4gICAgfSwgZXJyb3IsIHt9KS50aGVuKCgpID0+IG51bGwpLmNhdGNoKChlcnJvcikgPT4gUHJvbWlzZS5yZWplY3QoZXJyb3IpKSk7XG59O1xuXG5leHBvcnQgY29uc3QgYWRkU3Vic2NyaXB0aW9uID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGl0ZW0pOiBQcm9taXNlPFBheW1lbnRTdWJzY3JpcHRpb24+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnYWRkU3Vic2NyaXB0aW9uJztcbiAgY29uc3Qge2RhdGFiYXNlLCB1c2VySWQ6IHNlc3Npb25JZCwgdXNlclR5cGV9ID0gY29udGV4dDtcbiAgY29uc3QgaXNBZG1pbjogYm9vbGVhbiA9IHVzZXJUeXBlID4gMjtcblxuICBpZighaXNBZG1pbikge1xuICAgIHJldHVybiBsb2dFeGNlcHRpb24oe1xuICAgICAgYWN0aW9uLFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICBsYWJlbDogJ3VuYXV0aG9yaXplZCcsXG4gICAgICB2YWx1ZTogJ2ludmFsaWRfc2Vzc2lvbidcbiAgICB9LCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpO1xuICB9XG5cbiAgY29uc3Qgbm93OiBudW1iZXIgPSBEYXRlLm5vdygpO1xuICBjb25zdCBmb3JtYXRJZDogc3RyaW5nID0gcGFyc2VJZChpdGVtLmlkKTtcbiAgY29uc3Qgc3Vic2NyaXB0aW9uSWQ6IHN0cmluZyA9IGZvcm1hdElkID09PSAnJyA/IGNyZWF0ZUhhc2goYHRhZy0ke3Nlc3Npb25JZH1gKSA6IGZvcm1hdElkO1xuICBjb25zdCB7cGxhbklkLCB0YXgsIHRyaWFsRW5kLCB0cmlhbFBlcmlvZH0gPSBpdGVtO1xuICBjb25zdCBmb3JtYXRQbGFuSWQ6IHN0cmluZyA9IHBhcnNlSWQocGxhbklkKTtcbiAgY29uc3QgZm9ybWF0VGF4UGVyY2VudDogbnVtYmVyID0gcGFyc2VOdW0odGF4KSB8fCAwO1xuICBjb25zdCBmb3JtYXRUcmlhbFBlcmlvZDogbnVtYmVyID0gcGFyc2VOdW0odGF4KSB8fCAwO1xuICBsZXQgZm9ybWF0VHJpYWxFbmQ6IG51bWJlciA9IHRyaWFsRW5kIHx8IERhdGUubm93KCk7XG5cbiAgaWYoZm9ybWF0VHJpYWxQZXJpb2QpIHtcbiAgICBmb3JtYXRUcmlhbEVuZCA9IERhdGVUaW1lLmxvY2FsKCkucGx1cyh7ZGF5czogdHJpYWxQZXJpb2R9KS50b01pbGxpcygpO1xuICB9XG5cbiAgcmV0dXJuIGdldFVzZXIoY29udGV4dCwgc2Vzc2lvbklkKVxuICAgIC50aGVuKCh1c2VyKSA9PiB7XG4gICAgICBjb25zdCBpbnNlcnQ6IFBheW1lbnRTdWJzY3JpcHRpb24gPSB7XG4gICAgICAgIF9rZXk6IHN1YnNjcmlwdGlvbklkLFxuICAgICAgICBhZGRlZDogbm93LFxuICAgICAgICBjYW5jZWxEYXRlOiAwLFxuICAgICAgICBtb2RpZmllZDogbm93LFxuICAgICAgICBwbGFuSWQ6IGZvcm1hdFBsYW5JZCxcbiAgICAgICAgdGF4OiBmb3JtYXRUYXhQZXJjZW50LFxuICAgICAgICB0cmlhbEVuZDogZm9ybWF0VHJpYWxFbmQsXG4gICAgICAgIHVzZXJJZDogc2Vzc2lvbklkXG4gICAgICB9O1xuICAgICAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBJTlNFUlQgJHtpbnNlcnR9IElOIHBsYW5zIFJFVFVSTiBORVdgO1xuXG4gICAgICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAgICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgICAgIC50aGVuKChzdWJzY3JpcHRpb246IFBheW1lbnRTdWJzY3JpcHRpb24gPSB7fSkgPT4ge1xuICAgICAgICAgIC8vIFN0cmlwZVxuICAgICAgICAgIGNvbnN0IHN0cmlwZUNsaWVudCA9IHN0cmlwZShDb25maWcuZ2V0KCdzdHJpcGUudG9rZW4nKSk7XG4gICAgICAgICAgcmV0dXJuIHN0cmlwZUNsaWVudC5zdWJzY3JpcHRpb25zXG4gICAgICAgICAgICAuY3JlYXRlKHtcbiAgICAgICAgICAgICAgY3VzdG9tZXI6IHVzZXIuc3RyaXBlQ3VzdG9tZXJJZCxcbiAgICAgICAgICAgICAgaXRlbXM6IFtcbiAgICAgICAgICAgICAgICB7cGxhbjogZm9ybWF0UGxhbklkfVxuICAgICAgICAgICAgICBdLFxuICAgICAgICAgICAgICBtZXRhZGF0YToge1xuICAgICAgICAgICAgICAgIHVzZXJJZDogc2Vzc2lvbklkXG4gICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICAgIHRheF9wZXJjZW50OiBmb3JtYXRUYXhQZXJjZW50ICogMTAwLFxuICAgICAgICAgICAgICB0cmlhbF9lbmQ6IGZvcm1hdFRyaWFsRW5kXG4gICAgICAgICAgICB9KVxuICAgICAgICAgICAgLnRoZW4oKCkgPT4gc3Vic2NyaXB0aW9uKTtcbiAgICAgICAgfSlcbiAgICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgbGFiZWw6ICdkYl9lcnJvcidcbiAgICAgICAgfSwgZXJyb3IsIHt9KS50aGVuKCgpID0+IG51bGwpLmNhdGNoKChlcnJvcikgPT4gUHJvbWlzZS5yZWplY3QoZXJyb3IpKSk7XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgZGVsZXRlU3Vic2NyaXB0aW9uID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGVuZERhdGU6IG51bWJlcik6IFByb21pc2U8Ym9vbGVhbj4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdkZWxldGVTdWJzY3JpcHRpb24nO1xuICBjb25zdCB7ZGF0YWJhc2V9ID0gY29udGV4dDtcbiAgY29uc3Qgbm93OiBudW1iZXIgPSBEYXRlLm5vdygpO1xuICBjb25zdCBmb3JtYXRFbmREYXRlOiBudW1iZXIgPSBwYXJzZU51bShlbmREYXRlKSB8fCBub3c7XG5cbiAgcmV0dXJuIGdldFN1YnNjcmlwdGlvbihjb250ZXh0KVxuICAgIC50aGVuKChzdWJzY3JpcHRpb24pID0+IHtcbiAgICAgIC8vIFN0cmlwZVxuICAgICAgY29uc3Qgc3RyaXBlQ2xpZW50ID0gc3RyaXBlKENvbmZpZy5nZXQoJ3N0cmlwZS50b2tlbicpKTtcbiAgICAgIHJldHVybiBzdHJpcGVDbGllbnQuc3Vic2NyaXB0aW9uc1xuICAgICAgICAuZGVsKHN1YnNjcmlwdGlvbi50cmFuc2FjdGlvbklkLCB7XG4gICAgICAgICAgc3Vic2NyaXB0aW9uOiBmb3JtYXRFbmREYXRlXG4gICAgICAgIH0pXG4gICAgICAgIC50aGVuKChzdHJpcGVSZXNwb25zZSkgPT4ge1xuICAgICAgICAgIC8vIE1ha2Ugc3VyZSB3ZSBjYW5jZWxsZWQgb24gU3RyaXBlIGJlZm9yZSB1cGRhdGluZyB0aGUgZGJcbiAgICAgICAgICBpZihzdHJpcGVSZXNwb25zZS5jYW5jZWxsZWQgfHwgc3RyaXBlUmVzcG9uc2UuY2FuY2VsX2F0X3BlcmlvZF9lbmQpIHtcbiAgICAgICAgICAgIGNvbnN0IHVwZGF0ZTogUGF5bWVudFN1YnNjcmlwdGlvbiA9IHtcbiAgICAgICAgICAgICAgY2FuY2VsRGF0ZTogZm9ybWF0RW5kRGF0ZSxcbiAgICAgICAgICAgICAgbW9kaWZpZWQ6IG5vdyxcbiAgICAgICAgICAgICAgcGxhbklkOiAnJ1xuICAgICAgICAgICAgfTtcbiAgICAgICAgICAgIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgVVBEQVRFIHMgV0lUSCAke3VwZGF0ZX0gSU4gc3Vic2NyaXB0aW9uc2A7XG5cbiAgICAgICAgICAgIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgICAgICAgICAgICAudGhlbigoKSA9PiB0cnVlKVxuICAgICAgICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgICAgICAgICBsYWJlbDogJ2RiX2Vycm9yJ1xuICAgICAgICAgICAgICB9LCBlcnJvciwge30pLnRoZW4oKCkgPT4gbnVsbCkuY2F0Y2goKGVycm9yKSA9PiBQcm9taXNlLnJlamVjdChlcnJvcikpKTtcbiAgICAgICAgICB9XG4gICAgICAgICAgcmV0dXJuIGZhbHNlO1xuICAgICAgICB9KTtcbiAgICB9KTtcbn07XG4iXX0=