@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,{"version":3,"sources":["../../src/data/subscription.ts"],"names":["eventCategory","apiVersion","getPlanList","context","from","to","action","database","limit","aqlQry","aql","query","then","cursor","all","list","error","category","label","Promise","reject","getSubscription","sessionId","userId","next","subscription","addPlan","item","userAccess","isAdmin","value","now","Date","formatId","id","planId","amount","currency","description","interval","intervalCount","name","formatAmount","formatInterval","toLowerCase","formatIntervalCnt","formatName","formatDesc","insert","_key","added","modified","plan","stripeClient","Stripe","Config","get","typescript","stripePlan","interval_count","metadata","statement_descriptor","plans","create","addSubscription","subscriptionId","trialEnd","trialPeriod","formatPlanId","formatTrialPeriod","formatTrialEnd","DateTime","local","plus","days","toMillis","user","cancelDate","subscriptions","customer","stripeCustomerId","items","trial_end","deleteSubscription","del","transactionId","canceled_at","update"],"mappings":"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","sourcesContent":["/**\n * Copyright (c) 2019-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n */\nimport {createHash, parseChar, parseId, parseNum, parseVarChar} from '@nlabs/utils';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {ArrayCursor} from 'arangojs/cursor';\nimport {DateTime} from 'luxon';\nimport Stripe from 'stripe';\n\nimport {Config} from '../config';\nimport {ArangoDBLimit} from '../types/arangodb';\nimport {ApiContext} from '../types/auth';\nimport {PaymentInterval, PaymentPlan, PaymentSubscription} from '../types/payments';\nimport {getLimit, logError, logException, useDb} from '../utils';\nimport {getUser} from './users';\n\nconst eventCategory: string = 'subscription';\nconst apiVersion: any = '2020-03-02';\n\nexport const getPlanList = (context: ApiContext, from: number = 0, to: number = 30): Promise<PaymentPlan[]> => {\n  const action: string = 'getList';\n  const {database} = context;\n  const limit: ArangoDBLimit = getLimit(from, to);\n  const aqlQry: string = `FOR p IN plans\n      ${limit.aql}\n      SORT p.amount\n      RETURN p`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .then((list = []) => list)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, {}).then(() => null).catch((error) => Promise.reject(error)));\n};\n\nexport const getSubscription = (context: ApiContext): Promise<PaymentSubscription> => {\n  const action: string = 'getItem';\n  const {database, userId: sessionId} = context;\n  const aqlQry: string = `FOR s IN subscriptions\n      FILTER s.userId == ${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})`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((subscription = {}) => subscription)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, {}).then(() => null).catch((error) => Promise.reject(error)));\n};\n\nexport const addPlan = (context: ApiContext, item: PaymentPlan): Promise<PaymentPlan> => {\n  const action: string = 'addPlan';\n  const {database, userId: sessionId, userAccess} = context;\n  const isAdmin: boolean = userAccess > 2;\n\n  if(!isAdmin) {\n    return logException({\n      action,\n      category: eventCategory,\n      label: 'unauthorized',\n      value: 'invalid_session'\n    }, context).then(() => null);\n  }\n\n  const now: number = Date.now();\n  const formatId: string = parseId(item.id);\n  const planId: string = formatId === '' ? createHash(`tag-${sessionId}`) : formatId;\n  const {amount, currency = 'USD', description, interval, intervalCount, name} = item;\n  const formatAmount: number = parseNum(amount);\n  const formatInterval: PaymentInterval = parseChar(interval, 5).toLowerCase() as PaymentInterval;\n  const formatIntervalCnt: number = parseNum(intervalCount, 5);\n  const formatName: string = parseVarChar(name, 32);\n  const formatDesc: string = parseVarChar(description, 32);\n\n  const insert: PaymentPlan = {\n    _key: planId,\n    added: now,\n    amount: formatAmount,\n    currency,\n    description: formatDesc,\n    interval: formatInterval,\n    intervalCount: formatIntervalCnt,\n    modified: now,\n    name: formatName\n  };\n  const aqlQry: AqlQuery = aql`INSERT ${insert} IN plans RETURN NEW`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((plan: PaymentPlan = {}) => {\n      // Stripe\n      const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion, typescript: true});\n      const stripePlan: any = {\n        amount: formatAmount * 100,\n        currency,\n        id: planId,\n        interval: formatInterval,\n        interval_count: formatIntervalCnt,\n        metadata: {\n        },\n        name: formatName,\n        statement_descriptor: formatDesc\n      };\n\n      return stripeClient.plans\n        .create(stripePlan)\n        .then(() => plan);\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, {}).then(() => null).catch((error) => Promise.reject(error)));\n};\n\nexport const addSubscription = (context: ApiContext, item): Promise<PaymentSubscription> => {\n  const action: string = 'addSubscription';\n  const {database, userId: sessionId, userAccess} = context;\n  const isAdmin: boolean = userAccess > 2;\n\n  if(!isAdmin) {\n    return logException({\n      action,\n      category: eventCategory,\n      label: 'unauthorized',\n      value: 'invalid_session'\n    }, context).then(() => null);\n  }\n\n  const now: number = Date.now();\n  const formatId: string = parseId(item.id);\n  const subscriptionId: string = formatId === '' ? createHash(`tag-${sessionId}`) : formatId;\n  const {planId, trialEnd, trialPeriod} = item;\n  const formatPlanId: string = parseId(planId);\n  const formatTrialPeriod: number = 0;\n  let formatTrialEnd: number = trialEnd || Date.now();\n\n  if(formatTrialPeriod) {\n    formatTrialEnd = DateTime.local().plus({days: trialPeriod}).toMillis();\n  }\n\n  return getUser(context, sessionId)\n    .then((user) => {\n      const insert: PaymentSubscription = {\n        _key: subscriptionId,\n        added: now,\n        cancelDate: 0,\n        modified: now,\n        planId: formatPlanId,\n        trialEnd: formatTrialEnd,\n        userId: sessionId\n      };\n      const aqlQry: AqlQuery = aql`INSERT ${insert} IN plans RETURN NEW`;\n\n      return useDb(database).query(aqlQry)\n        .then((cursor: ArrayCursor) => cursor.next())\n        .then((subscription: PaymentSubscription = {}) => {\n          // Stripe\n          const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion, typescript: true});\n          return stripeClient.subscriptions\n            .create({\n              customer: user.stripeCustomerId,\n              items: [\n                {\n                  plan: formatPlanId\n                }\n              ],\n              metadata: {\n                userId: sessionId\n              },\n              trial_end: formatTrialEnd\n            })\n            .then(() => subscription);\n        })\n        .catch((error: Error) => logError({\n          action,\n          category: eventCategory,\n          label: 'db_error'\n        }, error, {}).then(() => null).catch((error) => Promise.reject(error)));\n    });\n};\n\nexport const deleteSubscription = (context: ApiContext): Promise<boolean> => {\n  const action: string = 'deleteSubscription';\n  const {database} = context;\n  const now: number = Date.now();\n\n  return getSubscription(context)\n    .then((subscription) => {\n      // Stripe\n      const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion, typescript: true});\n      return stripeClient.subscriptions\n        .del(subscription.transactionId)\n        .then(({canceled_at: cancelDate}) => {\n          // Make sure we cancelled on Stripe before updating the db\n          if(cancelDate) {\n            const update: PaymentSubscription = {\n              cancelDate,\n              modified: now,\n              planId: ''\n            };\n            const aqlQry: AqlQuery = aql`UPDATE s WITH ${update} IN subscriptions`;\n\n            return useDb(database).query(aqlQry)\n              .then(() => true)\n              .catch((error: Error) => logError({\n                action,\n                category: eventCategory,\n                label: 'db_error'\n              }, error, {}).then(() => null).catch((error) => Promise.reject(error)));\n          }\n          return false;\n        });\n    });\n};\n"]}