@nlabs/reaktor 0.5.4 → 0.6.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 (73) hide show
  1. package/README.md +7 -7
  2. package/lib/actions/conversations.js +36 -47
  3. package/lib/actions/dynamodb.js +22 -33
  4. package/lib/actions/email.js +32 -29
  5. package/lib/actions/files.js +45 -53
  6. package/lib/actions/groups.js +33 -28
  7. package/lib/actions/images.js +53 -63
  8. package/lib/actions/index.js +54 -37
  9. package/lib/actions/ios.js +21 -40
  10. package/lib/actions/locations.js +25 -22
  11. package/lib/actions/messages.js +28 -41
  12. package/lib/actions/notifications.d.ts +1 -1
  13. package/lib/actions/notifications.js +13 -20
  14. package/lib/actions/payments.js +37 -48
  15. package/lib/actions/posts.js +19 -42
  16. package/lib/actions/reactions.d.ts +1 -1
  17. package/lib/actions/reactions.js +17 -40
  18. package/lib/actions/s3.js +21 -18
  19. package/lib/actions/search.js +26 -40
  20. package/lib/actions/sms.js +24 -21
  21. package/lib/actions/statistics.js +14 -21
  22. package/lib/actions/subscription.js +27 -24
  23. package/lib/actions/tags.js +47 -54
  24. package/lib/actions/users.js +81 -94
  25. package/lib/actions/websockets.js +26 -39
  26. package/lib/config.js +21 -18
  27. package/lib/index.js +24 -22
  28. package/lib/lambdas/actions/websockets.js +27 -40
  29. package/lib/lambdas/authorizer.js +21 -18
  30. package/lib/lambdas/connection.js +26 -40
  31. package/lib/lambdas/utils/message.js +20 -17
  32. package/lib/lambdas/utils/websocket.js +33 -25
  33. package/lib/templates/email/layout.js +16 -9
  34. package/lib/templates/email/passwordForgot.js +16 -9
  35. package/lib/templates/email/passwordRecovery.js +16 -9
  36. package/lib/templates/email/verifyEmail.js +16 -9
  37. package/lib/templates/email/welcome.js +16 -9
  38. package/lib/templates/sms/passwordForgot.js +16 -9
  39. package/lib/templates/sms/passwordRecovery.js +16 -9
  40. package/lib/templates/sms/verifyEmail.js +16 -9
  41. package/lib/templates/sms/verifyPhone.js +16 -9
  42. package/lib/templates/sms/welcome.js +16 -9
  43. package/lib/types/apps.d.ts +1 -1
  44. package/lib/types/apps.js +14 -5
  45. package/lib/types/arangodb.js +14 -5
  46. package/lib/types/auth.js +14 -5
  47. package/lib/types/connections.js +14 -5
  48. package/lib/types/conversations.js +14 -5
  49. package/lib/types/email.js +14 -5
  50. package/lib/types/files.js +14 -5
  51. package/lib/types/google.js +14 -5
  52. package/lib/types/groups.js +14 -5
  53. package/lib/types/images.js +14 -5
  54. package/lib/types/index.js +50 -35
  55. package/lib/types/locations.js +14 -5
  56. package/lib/types/messages.js +14 -5
  57. package/lib/types/notifications.js +14 -5
  58. package/lib/types/payments.d.ts +1 -1
  59. package/lib/types/payments.js +14 -5
  60. package/lib/types/posts.js +14 -5
  61. package/lib/types/statistics.js +14 -5
  62. package/lib/types/tags.js +14 -5
  63. package/lib/types/users.js +14 -5
  64. package/lib/types/websocket.js +14 -5
  65. package/lib/utils/analytics.js +24 -44
  66. package/lib/utils/arangodb.js +14 -21
  67. package/lib/utils/auth.js +22 -36
  68. package/lib/utils/graphql.js +20 -17
  69. package/lib/utils/index.js +26 -23
  70. package/lib/utils/objects.js +22 -19
  71. package/lib/utils/session.js +20 -17
  72. package/package.json +30 -32
  73. package/.eslintrc +0 -18
@@ -1,59 +1,32 @@
1
1
  var __create = Object.create;
2
2
  var __defProp = Object.defineProperty;
3
- var __defProps = Object.defineProperties;
4
3
  var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
5
- var __getOwnPropDescs = Object.getOwnPropertyDescriptors;
6
4
  var __getOwnPropNames = Object.getOwnPropertyNames;
7
- var __getOwnPropSymbols = Object.getOwnPropertySymbols;
8
5
  var __getProtoOf = Object.getPrototypeOf;
9
6
  var __hasOwnProp = Object.prototype.hasOwnProperty;
10
- var __propIsEnum = Object.prototype.propertyIsEnumerable;
11
- var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
12
- var __spreadValues = (a, b) => {
13
- for (var prop in b || (b = {}))
14
- if (__hasOwnProp.call(b, prop))
15
- __defNormalProp(a, prop, b[prop]);
16
- if (__getOwnPropSymbols)
17
- for (var prop of __getOwnPropSymbols(b)) {
18
- if (__propIsEnum.call(b, prop))
19
- __defNormalProp(a, prop, b[prop]);
20
- }
21
- return a;
22
- };
23
- var __spreadProps = (a, b) => __defProps(a, __getOwnPropDescs(b));
24
- var __markAsModule = (target) => __defProp(target, "__esModule", { value: true });
25
- var __require = typeof require !== "undefined" ? require : (x) => {
26
- throw new Error('Dynamic require of "' + x + '" is not supported');
27
- };
28
- var __objRest = (source, exclude) => {
29
- var target = {};
30
- for (var prop in source)
31
- if (__hasOwnProp.call(source, prop) && exclude.indexOf(prop) < 0)
32
- target[prop] = source[prop];
33
- if (source != null && __getOwnPropSymbols)
34
- for (var prop of __getOwnPropSymbols(source)) {
35
- if (exclude.indexOf(prop) < 0 && __propIsEnum.call(source, prop))
36
- target[prop] = source[prop];
37
- }
38
- return target;
39
- };
40
7
  var __export = (target, all) => {
41
- __markAsModule(target);
42
8
  for (var name in all)
43
9
  __defProp(target, name, { get: all[name], enumerable: true });
44
10
  };
45
- var __reExport = (target, module2, desc) => {
46
- if (module2 && typeof module2 === "object" || typeof module2 === "function") {
47
- for (let key of __getOwnPropNames(module2))
48
- if (!__hasOwnProp.call(target, key) && key !== "default")
49
- __defProp(target, key, { get: () => module2[key], enumerable: !(desc = __getOwnPropDesc(module2, key)) || desc.enumerable });
11
+ var __copyProps = (to, from, except, desc) => {
12
+ if (from && typeof from === "object" || typeof from === "function") {
13
+ for (let key of __getOwnPropNames(from))
14
+ if (!__hasOwnProp.call(to, key) && key !== except)
15
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
50
16
  }
51
- return target;
52
- };
53
- var __toModule = (module2) => {
54
- return __reExport(__markAsModule(__defProp(module2 != null ? __create(__getProtoOf(module2)) : {}, "default", module2 && module2.__esModule && "default" in module2 ? { get: () => module2.default, enumerable: true } : { value: module2, enumerable: true })), module2);
17
+ return to;
55
18
  };
56
- __export(exports, {
19
+ var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
20
+ // If the importer is in node compatibility mode or this is not an ESM
21
+ // file that has been converted to a CommonJS file using a Babel-
22
+ // compatible transform (i.e. "__esModule" has not been set), then set
23
+ // "default" to the CommonJS "module.exports" for node compatibility.
24
+ isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
25
+ mod
26
+ ));
27
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
28
+ var users_exports = {};
29
+ __export(users_exports, {
57
30
  UserAccess: () => UserAccess,
58
31
  addUser: () => addUser,
59
32
  confirmCode: () => confirmCode,
@@ -75,24 +48,25 @@ __export(exports, {
75
48
  signOut: () => signOut,
76
49
  updateUser: () => updateUser
77
50
  });
78
- var import_utils = __toModule(require("@nlabs/utils"));
79
- var import_arangojs = __toModule(require("arangojs"));
80
- var import_isEmpty = __toModule(require("lodash/isEmpty"));
81
- var import_luxon = __toModule(require("luxon"));
82
- var import_stripe = __toModule(require("stripe"));
83
- var import_config = __toModule(require("../config"));
84
- var import_utils2 = __toModule(require("../utils"));
85
- var import_session = __toModule(require("../utils/session"));
51
+ module.exports = __toCommonJS(users_exports);
52
+ var import_utils = require("@nlabs/utils");
53
+ var import_arangojs = require("arangojs");
54
+ var import_isEmpty = __toESM(require("lodash/isEmpty"));
55
+ var import_luxon = require("luxon");
56
+ var import_stripe = __toESM(require("stripe"));
57
+ var import_config = require("../config");
58
+ var import_utils2 = require("../utils");
59
+ var import_session = require("../utils/session");
86
60
  const eventCategory = "users";
87
61
  const apiVersion = "2020-03-02";
88
- var UserAccess;
89
- (function(UserAccess2) {
62
+ var UserAccess = /* @__PURE__ */ ((UserAccess2) => {
90
63
  UserAccess2[UserAccess2["DEACTIVATED"] = 0] = "DEACTIVATED";
91
64
  UserAccess2[UserAccess2["ACTIVE"] = 1] = "ACTIVE";
92
65
  UserAccess2[UserAccess2["PREMIUM"] = 2] = "PREMIUM";
93
66
  UserAccess2[UserAccess2["CONTENT_ADMIN"] = 3] = "CONTENT_ADMIN";
94
67
  UserAccess2[UserAccess2["ADMIN"] = 4] = "ADMIN";
95
- })(UserAccess || (UserAccess = {}));
68
+ return UserAccess2;
69
+ })(UserAccess || {});
96
70
  const createToken = (userId, username, userAccess, expires = 15) => {
97
71
  const now = import_luxon.DateTime.local();
98
72
  const sessionExpires = now.plus({ minutes: expires });
@@ -122,9 +96,10 @@ const parseUserOptions = (options = {}) => {
122
96
  from = 0,
123
97
  to = 30
124
98
  } = options;
125
- return __spreadProps(__spreadValues({}, options), {
99
+ return {
100
+ ...options,
126
101
  limit: (0, import_utils2.getLimit)(from, to)
127
- });
102
+ };
128
103
  };
129
104
  const addUser = async (context, args) => {
130
105
  const action = "addUser";
@@ -154,12 +129,14 @@ const addUser = async (context, args) => {
154
129
  const checkQuery = `FOR u IN users
155
130
  FILTER ${filters.join(" || ")}
156
131
  RETURN u`;
157
- const existingUsers = await database.query(checkQuery).then((cursor) => cursor.all()).catch((error) => (0, import_utils2.logError)({
158
- action,
159
- args: { username },
160
- category: eventCategory,
161
- label: import_utils2.ErrorTypes.DATABASE_ERROR
162
- }, error, context).then(() => null));
132
+ const existingUsers = await database.query(checkQuery).then((cursor) => cursor.all()).catch(
133
+ (error) => (0, import_utils2.logError)({
134
+ action,
135
+ args: { username },
136
+ category: eventCategory,
137
+ label: import_utils2.ErrorTypes.DATABASE_ERROR
138
+ }, error, context).then(() => null)
139
+ );
163
140
  if (existingUsers.length) {
164
141
  return (0, import_utils2.logException)({
165
142
  action,
@@ -186,17 +163,19 @@ const addUser = async (context, args) => {
186
163
  verifiedPhoneCode
187
164
  };
188
165
  const insertQuery = import_arangojs.aql`INSERT ${insert} IN users RETURN NEW`;
189
- return await database.query(insertQuery).then((cursor) => cursor.next() || {}).catch((error) => (0, import_utils2.logError)({
190
- action,
191
- args: { username },
192
- category: eventCategory,
193
- label: import_utils2.ErrorTypes.DATABASE_ERROR
194
- }, error, context).then(() => null));
166
+ return await database.query(insertQuery).then((cursor) => cursor.next() || {}).catch(
167
+ (error) => (0, import_utils2.logError)({
168
+ action,
169
+ args: { username },
170
+ category: eventCategory,
171
+ label: import_utils2.ErrorTypes.DATABASE_ERROR
172
+ }, error, context).then(() => null)
173
+ );
195
174
  };
196
175
  const updateUser = async (context, user) => {
197
176
  const action = "updateUser";
198
177
  const { database } = context;
199
- const _a = user, { _key, _id, id, tags = [], userId } = _a, updated = __objRest(_a, ["_key", "_id", "id", "tags", "userId"]);
178
+ const { _key, _id, id, tags = [], userId, ...updated } = user;
200
179
  let userDocId;
201
180
  if (_id || id) {
202
181
  userDocId = _id || id;
@@ -252,12 +231,14 @@ const confirmCode = async (context, args) => {
252
231
  default:
253
232
  return false;
254
233
  }
255
- }).catch((error) => (0, import_utils2.logError)({
256
- action,
257
- args: { code, type, userId: sessionId },
258
- category: eventCategory,
259
- label: import_utils2.ErrorTypes.DATABASE_ERROR
260
- }, error, context));
234
+ }).catch(
235
+ (error) => (0, import_utils2.logError)({
236
+ action,
237
+ args: { code, type, userId: sessionId },
238
+ category: eventCategory,
239
+ label: import_utils2.ErrorTypes.DATABASE_ERROR
240
+ }, error, context)
241
+ );
261
242
  } catch (error) {
262
243
  return false;
263
244
  }
@@ -483,12 +464,14 @@ const signIn = async (context, args) => {
483
464
  FILTER u.username == ${formatUsername}
484
465
  LIMIT 1
485
466
  RETURN u`;
486
- const checkUser = await database.query(checkQuery).then((cursor) => cursor.next()).catch((error) => (0, import_utils2.logError)({
487
- action,
488
- args: { username: formatUsername },
489
- category: eventCategory,
490
- label: import_utils2.ErrorTypes.DATABASE_ERROR
491
- }, error, context).then(() => null));
467
+ const checkUser = await database.query(checkQuery).then((cursor) => cursor.next()).catch(
468
+ (error) => (0, import_utils2.logError)({
469
+ action,
470
+ args: { username: formatUsername },
471
+ category: eventCategory,
472
+ label: import_utils2.ErrorTypes.DATABASE_ERROR
473
+ }, error, context).then(() => null)
474
+ );
492
475
  if (!checkUser) {
493
476
  return (0, import_utils2.logException)({
494
477
  action,
@@ -522,12 +505,14 @@ const signOut = async (context, args) => {
522
505
  LIMIT 1
523
506
  RETURN NEW`;
524
507
  try {
525
- await database.query(sessionQuery).then((cursor) => cursor.next()).catch((error) => (0, import_utils2.logError)({
526
- action,
527
- args: { username, userId: sessionId },
528
- category: eventCategory,
529
- label: import_utils2.ErrorTypes.DATABASE_ERROR
530
- }, error, context));
508
+ await database.query(sessionQuery).then((cursor) => cursor.next()).catch(
509
+ (error) => (0, import_utils2.logError)({
510
+ action,
511
+ args: { username, userId: sessionId },
512
+ category: eventCategory,
513
+ label: import_utils2.ErrorTypes.DATABASE_ERROR
514
+ }, error, context)
515
+ );
531
516
  } catch (error) {
532
517
  return false;
533
518
  }
@@ -542,11 +527,13 @@ const getActiveUserCount = (context) => {
542
527
  RETURN u
543
528
  )
544
529
  RETURN LENGTH(docs)`;
545
- return database.query(countQuery).then((cursor) => cursor.next()).catch((error) => (0, import_utils2.logError)({
546
- action,
547
- category: eventCategory,
548
- label: import_utils2.ErrorTypes.DATABASE_ERROR
549
- }, error, context).then(() => 0));
530
+ return database.query(countQuery).then((cursor) => cursor.next()).catch(
531
+ (error) => (0, import_utils2.logError)({
532
+ action,
533
+ category: eventCategory,
534
+ label: import_utils2.ErrorTypes.DATABASE_ERROR
535
+ }, error, context).then(() => 0)
536
+ );
550
537
  };
551
538
  // Annotate the CommonJS export names for ESM import in node:
552
539
  0 && (module.exports = {
@@ -571,4 +558,4 @@ const getActiveUserCount = (context) => {
571
558
  signOut,
572
559
  updateUser
573
560
  });
574
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/users.ts"],
  "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 {\n  createHash,\n  createPassword,\n  parseChar,\n  parseEmail,\n  parseId,\n  parseNum,\n  parsePassword,\n  parsePhone,\n  parseUsername\n} from '@nlabs/utils';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {EdgeCollection} from 'arangojs/collection';\nimport {ArrayCursor} from 'arangojs/cursor';\nimport isEmpty from 'lodash/isEmpty';\nimport {DateTime} from 'luxon';\nimport Stripe from 'stripe';\n\nimport {Config} from '../config';\nimport {ApiContext} from '../types/auth';\nimport {User} from '../types/users';\nimport {ErrorTypes, getLimit, logError, logException, selectReactionCountByType} from '../utils';\nimport {getSession, SessionError, SessionToken, setSession} from '../utils/session';\n\nconst eventCategory: string = 'users';\nconst apiVersion: any = '2020-03-02';\n\nexport interface UserOptions {\n  readonly from?: number;\n  readonly to?: number;\n  readonly username?: string;\n}\n\nexport enum UserAccess {\n  DEACTIVATED = 0,\n  ACTIVE = 1,\n  PREMIUM = 2,\n  CONTENT_ADMIN = 3,\n  ADMIN = 4\n}\n\nexport const createToken = (\n  userId: string,\n  username: string,\n  userAccess: number,\n  expires: number = 15\n): SessionToken => {\n  const now: DateTime = DateTime.local();\n  const sessionExpires: DateTime = now.plus({minutes: expires});\n  const iat: number = Math.floor(now.toSeconds());\n  const exp: number = Math.floor(sessionExpires.toSeconds());\n\n  const token = setSession({\n    exp,\n    iat,\n    username,\n    userAccess,\n    userId\n  });\n\n  return {\n    expires: sessionExpires.toMillis(),\n    issued: now.toMillis(),\n    token\n  };\n};\n\nexport const getUserOptional = (fields: string[] = []) =>\n  fields.reduce((selects: any, field: string) => {\n    if(field.includes('Count')) {\n      return selectReactionCountByType('users', 'u', field, selects);\n    }\n\n    return selects;\n  }, {objects: [], queries: []});\n\nexport const parseUserOptions = (options: UserOptions = {}) => {\n  const {\n    from = 0,\n    to = 30\n  } = options;\n\n  return {\n    ...options,\n    limit: getLimit(from, to)\n  };\n};\n\nexport const addUser = async (context: ApiContext, args: any): Promise<User> => {\n  const action: string = 'addUser';\n  const {database} = context;\n  const {user} = args;\n  const {email, password, phone, username} = user;\n  const salt: string = createHash(`${username}${password}`, null);\n  const encryptedPassword = createPassword(password, salt);\n  const formatUsername: string = parseUsername(username);\n  const formatEmail: string = parseEmail(email);\n  const formatPhone: string = parsePhone(phone);\n\n  if(isEmpty(formatUsername) || isEmpty(password) || (!isEmpty(formatPhone) && !isEmpty(formatEmail))) {\n    return logException({\n      action,\n      args: {username},\n      category: eventCategory,\n      value: ErrorTypes.INVALID_ARGUMENTS\n    }, context);\n  }\n\n  const filters: string[] = [`u.username == \"${formatUsername}\"`];\n\n  if(!isEmpty(formatEmail)) {\n    filters.push(`u.email == \"${formatEmail}\"`);\n  }\n\n  if(!isEmpty(formatPhone)) {\n    filters.push(`u.phone == ${formatPhone}`);\n  }\n\n  const checkQuery: string = `FOR u IN users\n    FILTER ${filters.join(' || ')}\n    RETURN u`;\n  const existingUsers = await database.query(checkQuery)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error) => logError({\n      action,\n      args: {username},\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context)\n      .then(() => null)\n    );\n\n  if(existingUsers.length) {\n    return logException({\n      action,\n      args: {username},\n      category: eventCategory,\n      value: ErrorTypes.EXISTING_USERNAME\n    }, context);\n  }\n\n  const verifiedEmailCode: number = Math.floor(100000 + (Math.random() * 900000));\n  const verifiedPhoneCode: number = Math.floor(100000 + (Math.random() * 900000));\n\n  // Add new user properties\n  const insert: User = {\n    _key: createHash(username, null),\n    added: Date.now(),\n    email: formatEmail,\n    modified: Date.now(),\n    password: encryptedPassword,\n    phone: formatPhone,\n    salt,\n    username: formatUsername,\n    userAccess: 1,\n    verifiedEmail: false,\n    verifiedEmailCode,\n    verifiedPhone: false,\n    verifiedPhoneCode\n  };\n\n  // Add new user in ArangoDB\n  const insertQuery: AqlQuery = aql`INSERT ${insert} IN users RETURN NEW`;\n  return await database.query(insertQuery)\n    .then((cursor: ArrayCursor) => cursor.next() || {})\n    .catch((error) => logError({\n      action,\n      args: {username},\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context)\n      .then(() => null)\n    );\n};\n\nexport const updateUser = async (context: ApiContext, user: User): Promise<any> => {\n  const action: string = 'updateUser';\n  const {database} = context;\n  const {_key, _id, id, tags = [], userId, ...updated} = user;\n  let userDocId: string;\n\n  if(_id || id) {\n    userDocId = _id || id;\n  } else if(_key || userId) {\n    userDocId = `users/${_key || userId}`;\n  }\n\n  const userQuery: AqlQuery = aql`LET u = DOCUMENT(${userDocId})\n    UPDATE u WITH ${updated} IN users\n    RETURN NEW`;\n\n  const updatedUser = await database.query(userQuery)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => {\n      console.log(error);\n      throw error;\n    });\n\n  const tagCollection: EdgeCollection = database.collection('isTagged');\n  await Promise.all(tags.map(({id: tagDocId, name}) => {\n    const edge = {\n      _from: tagDocId,\n      _key: createHash(`isTagged-${tagDocId}-${userDocId}`),\n      _to: userDocId,\n      added: Date.now(),\n      name\n    };\n    const tagQuery: AqlQuery = aql`FOR it IN isTagged\n      FILTER it._from == ${tagDocId} && it._to == ${userDocId} && it.name == ${name}\n      LIMIT 1\n      RETURN it`;\n\n    return database.query(tagQuery)\n      .then((cursor: ArrayCursor) => cursor.next())\n      .then((tagEdge) => {\n        if(!!tagEdge) {\n          return tagEdge;\n        }\n\n        return tagCollection.save(edge, {returnNew: true}).then(() => edge);\n      })\n      .catch((error: Error) => logError({\n        action,\n        category: eventCategory,\n        label: 'db_error'\n      }, error, context).then(() => null));\n  }));\n\n  return updatedUser;\n};\n\nexport const confirmCode = async (context: ApiContext, args): Promise<boolean> => {\n  const action: string = 'confirmEmail';\n  const {database, session: {userId: sessionId}} = context;\n  const {code, type} = args;\n  const userDocId: string = `users/${sessionId}`;\n\n  const aqlQuery: AqlQuery = aql`LET u = DOCUMENT(${userDocId}) RETURN u`;\n\n  try {\n    return database.query(aqlQuery)\n      .then((cursor) => cursor.next() || {})\n      .then(({verifiedEmailCode, verifiedPhoneCode}: User) => {\n        switch(type) {\n          case 'email':\n            return code === verifiedEmailCode;\n          case 'phone':\n            return code === verifiedPhoneCode;\n          default:\n            return false;\n        }\n      })\n      .catch((error) => logError({\n        action,\n        args: {code, type, userId: sessionId},\n        category: eventCategory,\n        label: ErrorTypes.DATABASE_ERROR\n      }, error, context)\n      );\n  } catch(error) {\n    return false;\n  }\n};\n\nexport const deleteUser = (context: ApiContext, args: any): Promise<any> => {\n  const action: string = 'deleteUser';\n  const {database, session: {userId: sessionId, userAccess: sessionAccess}} = context;\n  const {userId} = args;\n  const isAdmin: boolean = sessionAccess > 2;\n\n  if(!isAdmin && (sessionId !== userId)) {\n    logException({\n      action,\n      args,\n      category: eventCategory,\n      label: 'unauthorized',\n      value: 'invalid_session'\n    }, context);\n    return null;\n  }\n\n  const aqlQuery: AqlQuery = aql`FOR u IN users\n    FILTER u._key == ${userId}\n    LIMIT 1\n    REMOVE u IN users\n    RETURN OLD`;\n\n  // Stripe\n  const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion, typescript: true});\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((user: User = {}) => stripeClient.customers.del(user.stripeCustomerId)\n      .then(() => stripeClient.accounts.del(user.stripeAccountId))\n      .then(() => user))\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const deactivateUser = (context: ApiContext, userId: string): Promise<User> => {\n  const action: string = 'delete';\n  const {database, session: {userId: sessionId, userAccess: sessionAccess}} = context;\n  const isAdmin: boolean = sessionAccess > 2;\n\n  if(!isAdmin && (sessionId !== userId)) {\n    logException({\n      action,\n      category: eventCategory,\n      label: 'unauthorized',\n      value: 'invalid_session'\n    }, context);\n    return null;\n  }\n\n  const updated: User = {\n    userAccess: 0\n  };\n  const aqlQuery: AqlQuery = aql`UPDATE ${userId} WITH ${updated} IN users LIMIT 1 RETURN NEW`;\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const getDisplayName = (user: User = {}): string => {\n  const {first, last, name = '', username = ''} = user;\n  const fullname: string = ([first, last]).join(' ').trim();\n\n  if(!isEmpty(name)) {\n    return name;\n  } else if(fullname !== '') {\n    return fullname;\n  } else if(!isEmpty(username)) {\n    return username;\n  }\n\n  return 'Unknown';\n};\n\nexport const getSessionUser = (context: ApiContext): Promise<User> => {\n  const action: string = 'getSessionUser';\n  const {database, fields, session: {userId: sessionId, username}} = context;\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  const aqlQuery: string = `LET u = DOCUMENT(\"users/${sessionId}\")\n  ${selectQueries.join('\\n')}\n  RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((user = {}) => user)\n    .catch((error: Error) => logError({\n      action,\n      args: {username, userId: sessionId},\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getUser = (context: ApiContext, args: any): Promise<User> => {\n  const action: string = 'getUser';\n  const {userId} = args;\n  const {database, fields} = context;\n  const formatUserId: string = parseId(userId);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  // Get data from database\n  const aqlQuery: string = `LET u = DOCUMENT(\"users/${formatUserId}\")\n    ${selectQueries.join('\\n')}\n    FILTER u.userAccess > 0\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((user = {}) => user)\n    .catch((error: Error) => logError({\n      action,\n      args,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context)\n      .then(() => { }));\n};\n\nexport const getUsers = (context: ApiContext, options?: UserOptions): Promise<User[]> => {\n  const action: string = 'getUserList';\n  const {database, fields} = context;\n  const {limit, username} = parseUserOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n  const filterBy: string[] = ['u.userAccess > 0'];\n\n  if(!isEmpty(username)) {\n    filterBy.push(`CONTAINS(u.username, \"${parseUsername(username)}\")`);\n  }\n\n  // Get data from database\n  const aqlQuery: string = `FOR u IN users\n    FILTER ${filterBy.join(' && ')}\n    ${selectQueries.join('\\n')}\n    ${limit.aql}\n    SORT u.username\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context).then(() => []));\n};\n\nexport const getUsersByReactions = (\n  context: ApiContext,\n  {reactions = [], username}: any,\n  options?: UserOptions\n): Promise<User[]> => {\n  const action: string = 'getUsersByReactions';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const formatReactions: string[] =  reactions.map((reactionName: string) => parseChar(reactionName, 32).toLowerCase());\n  const {limit} = parseUserOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  const formatSessionId: string = `users/${sessionId}`;\n  const formatUsername: string = parseUsername(username);\n  const filterBy: string[] = [`POSITION(${JSON.stringify(formatReactions)}, LOWER(r.name))`];\n\n  if(!isEmpty(username)) {\n    filterBy.push(`CONTAINS(u.username, \"${formatUsername}\")`);\n  }\n\n  // Get data from database\n  const aqlQuery: string = `FOR u, r IN OUTBOUND \"${formatSessionId}\" hasReactions\n    OPTIONS {vertexCollections: \"users\"}\n    ${selectQueries.join('\\n')}\n    ${filterBy.length ? `FILTER ${filterBy.join(' && ')}` : ''}\n    ${limit.aql}\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  console.log({aqlQuery});\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context).then(() => []));\n};\n\nexport const getUsersByTags = (\n  context: ApiContext,\n  {tags, username}: any,\n  options?: UserOptions\n): Promise<User[]> => {\n  const action: string = 'getUsersByTags';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const formatTags: string[] =  tags.reduce((list: string[], tagName: string) => {\n    if(!isEmpty(tagName)) {\n      list.push(parseChar(tagName, 32).toLowerCase());\n    }\n\n    return list;\n  }, []);\n  const {limit} = parseUserOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  const formatUsername: string = parseUsername(username);\n  const filterBy: string[] = [`u._key != \"${sessionId}\"`];\n\n  if(!isEmpty(username)) {\n    filterBy.push(`CONTAINS(u.username, \"${formatUsername}\")`);\n  }\n\n  // Get data from database\n  const aqlQuery: string = `FOR t IN tags\n    FILTER POSITION(${JSON.stringify(formatTags)}, LOWER(t.name))\n    FOR u, it IN OUTBOUND t isTagged\n    OPTIONS {bfs: true, uniqueVertices: \"global\", vertexCollections: \"users\"}\n    ${selectQueries.join('\\n')}\n    ${filterBy.length ? `FILTER ${filterBy.join(' && ')}` : ''}\n    ${limit.aql}\n    RETURN DISTINCT MERGE(u, {${selectObjects.join(', ')}})`;\n\n  console.log('getUsersByTags', aqlQuery);\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context).then(() => []));\n};\n\nexport const getUsersByLatest = (context: ApiContext, {username}, options?: UserOptions): Promise<User[]> => {\n  const action: string = 'getUsersByLatest';\n  const {database, fields, session: {userId}} = context;\n  const {limit} = parseUserOptions(options);\n  const filter = ['u._id != session._id'];\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  if(!isEmpty(username)) {\n    filter.push(`CONTAINS(u.username, \"${parseUsername(username)}\")`);\n  }\n\n  // Get data from database\n  const aqlQuery: string = `FOR u IN users\n    LET session = DOCUMENT(\"users/${userId}\")\n    FILTER ${filter.join(' && ')}\n    ${selectQueries.join('\\n')}\n    LET distance = DISTANCE(u.latitude || 0, u.longitude || 0, session.latitude || 0, session.longitude || 0)\n    ${limit.aql}\n    SORT distance ASC, u.added DESC\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context).then(() => []));\n};\n\nexport const refreshSession = async (context: ApiContext, {expires, token}): Promise<SessionToken | SessionError> => {\n  const {error} = getSession(token);\n\n  if(error) {\n    return {errors: [error]};\n  }\n\n  const {exp, userId, username, userAccess} = getSession(token);\n  const now: number = Math.ceil(DateTime.local().toSeconds());\n\n  if(exp > now) {\n    return createToken(userId, username, userAccess, expires);\n  }\n\n  return {errors: ['session_expired']};\n};\n\nexport const signIn = async (context: ApiContext, args): Promise<SessionToken> => {\n  const action: string = 'signIn';\n  const {expires, password, username} = args;\n  const formatUsername: string = parseUsername(username);\n  const formatPassword: string = parsePassword(password);\n  const formatExpires: number = parseNum(expires) || 15;\n  const {database} = context;\n\n  if(isEmpty(formatUsername) || isEmpty(formatPassword)) {\n    return logException({\n      action,\n      args: {username},\n      category: eventCategory,\n      value: ErrorTypes.INVALID_ARGUMENTS\n    }, context);\n  }\n\n  const checkQuery: AqlQuery = aql`FOR u IN users\n    FILTER u.username == ${formatUsername}\n    LIMIT 1\n    RETURN u`;\n  const checkUser: User = await database.query(checkQuery)\n    .then((cursor) => cursor.next())\n    .catch((error) => logError({\n      action,\n      args: {username: formatUsername},\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context)\n      .then(() => null)\n    );\n\n  if(!checkUser) {\n    return logException({\n      action,\n      args: {username},\n      category: eventCategory,\n      value: ErrorTypes.INVALID_AUTHENTICATION\n    }, context);\n  }\n\n  const {_key: userId, salt, userAccess} = checkUser;\n  const authPassword: string = createPassword(formatPassword, salt);\n\n  if(checkUser.password !== authPassword) {\n    return logException({\n      action,\n      args: {username},\n      category: eventCategory,\n      value: ErrorTypes.INVALID_AUTHENTICATION\n    }, context);\n  }\n\n  return createToken(userId, username, userAccess, formatExpires);\n};\n\nexport const signOut = async (context: ApiContext, args): Promise<boolean> => {\n  const action: string = 'signOut';\n  const {database, session: {userId: sessionId, username}} = context;\n  const userDocId: string = `users/${sessionId}`;\n\n  const update = {\n    lastOnline: Date.now(),\n    sessionId: null\n  };\n  const sessionQuery: AqlQuery = aql`LET u = DOCUMENT(${userDocId})\n    UPDATE u WITH ${update} IN users\n    LIMIT 1\n    RETURN NEW`;\n\n  try {\n    await database.query(sessionQuery)\n      .then((cursor) => cursor.next())\n      .catch((error) => logError({\n        action,\n        args: {username, userId: sessionId},\n        category: eventCategory,\n        label: ErrorTypes.DATABASE_ERROR\n      }, error, context)\n      );\n  } catch(error) {\n    return false;\n  }\n\n  return true;\n};\n\nexport const getActiveUserCount = (context: ApiContext) => {\n  const action: string = 'getActiveUserCount';\n  const {database} = context;\n  const countQuery: AqlQuery = aql`LET docs = (\n    FOR u IN users\n    FILTER u.active == true\n    RETURN u\n  )\n  RETURN LENGTH(docs)`;\n\n  return database.query(countQuery)\n    .then((cursor) => cursor.next())\n    .catch((error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context)\n      .then(() => 0)\n    );\n};\n"],
  "mappings": "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",
  "names": []
}

561
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/users.ts"],
  "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 {\n  createHash,\n  createPassword,\n  parseChar,\n  parseEmail,\n  parseId,\n  parseNum,\n  parsePassword,\n  parsePhone,\n  parseUsername\n} from '@nlabs/utils';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {EdgeCollection} from 'arangojs/collection';\nimport {ArrayCursor} from 'arangojs/cursor';\nimport isEmpty from 'lodash/isEmpty';\nimport {DateTime} from 'luxon';\nimport Stripe from 'stripe';\n\nimport {Config} from '../config';\nimport {ApiContext} from '../types/auth';\nimport {User} from '../types/users';\nimport {ErrorTypes, getLimit, logError, logException, selectReactionCountByType} from '../utils';\nimport {getSession, SessionError, SessionToken, setSession} from '../utils/session';\n\nconst eventCategory: string = 'users';\nconst apiVersion: any = '2020-03-02';\n\nexport interface UserOptions {\n  readonly from?: number;\n  readonly to?: number;\n  readonly username?: string;\n}\n\nexport enum UserAccess {\n  DEACTIVATED = 0,\n  ACTIVE = 1,\n  PREMIUM = 2,\n  CONTENT_ADMIN = 3,\n  ADMIN = 4\n}\n\nexport const createToken = (\n  userId: string,\n  username: string,\n  userAccess: number,\n  expires: number = 15\n): SessionToken => {\n  const now: DateTime = DateTime.local();\n  const sessionExpires: DateTime = now.plus({minutes: expires});\n  const iat: number = Math.floor(now.toSeconds());\n  const exp: number = Math.floor(sessionExpires.toSeconds());\n\n  const token = setSession({\n    exp,\n    iat,\n    username,\n    userAccess,\n    userId\n  });\n\n  return {\n    expires: sessionExpires.toMillis(),\n    issued: now.toMillis(),\n    token\n  };\n};\n\nexport const getUserOptional = (fields: string[] = []) =>\n  fields.reduce((selects: any, field: string) => {\n    if(field.includes('Count')) {\n      return selectReactionCountByType('users', 'u', field, selects);\n    }\n\n    return selects;\n  }, {objects: [], queries: []});\n\nexport const parseUserOptions = (options: UserOptions = {}) => {\n  const {\n    from = 0,\n    to = 30\n  } = options;\n\n  return {\n    ...options,\n    limit: getLimit(from, to)\n  };\n};\n\nexport const addUser = async (context: ApiContext, args: any): Promise<User> => {\n  const action: string = 'addUser';\n  const {database} = context;\n  const {user} = args;\n  const {email, password, phone, username} = user;\n  const salt: string = createHash(`${username}${password}`, null);\n  const encryptedPassword = createPassword(password, salt);\n  const formatUsername: string = parseUsername(username);\n  const formatEmail: string = parseEmail(email);\n  const formatPhone: string = parsePhone(phone);\n\n  if(isEmpty(formatUsername) || isEmpty(password) || (!isEmpty(formatPhone) && !isEmpty(formatEmail))) {\n    return logException({\n      action,\n      args: {username},\n      category: eventCategory,\n      value: ErrorTypes.INVALID_ARGUMENTS\n    }, context);\n  }\n\n  const filters: string[] = [`u.username == \"${formatUsername}\"`];\n\n  if(!isEmpty(formatEmail)) {\n    filters.push(`u.email == \"${formatEmail}\"`);\n  }\n\n  if(!isEmpty(formatPhone)) {\n    filters.push(`u.phone == ${formatPhone}`);\n  }\n\n  const checkQuery: string = `FOR u IN users\n    FILTER ${filters.join(' || ')}\n    RETURN u`;\n  const existingUsers = await database.query(checkQuery)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error) => logError({\n      action,\n      args: {username},\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context)\n      .then(() => null)\n    );\n\n  if(existingUsers.length) {\n    return logException({\n      action,\n      args: {username},\n      category: eventCategory,\n      value: ErrorTypes.EXISTING_USERNAME\n    }, context);\n  }\n\n  const verifiedEmailCode: number = Math.floor(100000 + (Math.random() * 900000));\n  const verifiedPhoneCode: number = Math.floor(100000 + (Math.random() * 900000));\n\n  // Add new user properties\n  const insert: User = {\n    _key: createHash(username, null),\n    added: Date.now(),\n    email: formatEmail,\n    modified: Date.now(),\n    password: encryptedPassword,\n    phone: formatPhone,\n    salt,\n    username: formatUsername,\n    userAccess: 1,\n    verifiedEmail: false,\n    verifiedEmailCode,\n    verifiedPhone: false,\n    verifiedPhoneCode\n  };\n\n  // Add new user in ArangoDB\n  const insertQuery: AqlQuery = aql`INSERT ${insert} IN users RETURN NEW`;\n  return await database.query(insertQuery)\n    .then((cursor: ArrayCursor) => cursor.next() || {})\n    .catch((error) => logError({\n      action,\n      args: {username},\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context)\n      .then(() => null)\n    );\n};\n\nexport const updateUser = async (context: ApiContext, user: User): Promise<any> => {\n  const action: string = 'updateUser';\n  const {database} = context;\n  const {_key, _id, id, tags = [], userId, ...updated} = user;\n  let userDocId: string;\n\n  if(_id || id) {\n    userDocId = _id || id;\n  } else if(_key || userId) {\n    userDocId = `users/${_key || userId}`;\n  }\n\n  const userQuery: AqlQuery = aql`LET u = DOCUMENT(${userDocId})\n    UPDATE u WITH ${updated} IN users\n    RETURN NEW`;\n\n  const updatedUser = await database.query(userQuery)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => {\n      console.log(error);\n      throw error;\n    });\n\n  const tagCollection: EdgeCollection = database.collection('isTagged');\n  await Promise.all(tags.map(({id: tagDocId, name}) => {\n    const edge = {\n      _from: tagDocId,\n      _key: createHash(`isTagged-${tagDocId}-${userDocId}`),\n      _to: userDocId,\n      added: Date.now(),\n      name\n    };\n    const tagQuery: AqlQuery = aql`FOR it IN isTagged\n      FILTER it._from == ${tagDocId} && it._to == ${userDocId} && it.name == ${name}\n      LIMIT 1\n      RETURN it`;\n\n    return database.query(tagQuery)\n      .then((cursor: ArrayCursor) => cursor.next())\n      .then((tagEdge) => {\n        if(!!tagEdge) {\n          return tagEdge;\n        }\n\n        return tagCollection.save(edge, {returnNew: true}).then(() => edge);\n      })\n      .catch((error: Error) => logError({\n        action,\n        category: eventCategory,\n        label: 'db_error'\n      }, error, context).then(() => null));\n  }));\n\n  return updatedUser;\n};\n\nexport const confirmCode = async (context: ApiContext, args): Promise<boolean> => {\n  const action: string = 'confirmEmail';\n  const {database, session: {userId: sessionId}} = context;\n  const {code, type} = args;\n  const userDocId: string = `users/${sessionId}`;\n\n  const aqlQuery: AqlQuery = aql`LET u = DOCUMENT(${userDocId}) RETURN u`;\n\n  try {\n    return database.query(aqlQuery)\n      .then((cursor) => cursor.next() || {})\n      .then(({verifiedEmailCode, verifiedPhoneCode}: User) => {\n        switch(type) {\n          case 'email':\n            return code === verifiedEmailCode;\n          case 'phone':\n            return code === verifiedPhoneCode;\n          default:\n            return false;\n        }\n      })\n      .catch((error) => logError({\n        action,\n        args: {code, type, userId: sessionId},\n        category: eventCategory,\n        label: ErrorTypes.DATABASE_ERROR\n      }, error, context)\n      );\n  } catch(error) {\n    return false;\n  }\n};\n\nexport const deleteUser = (context: ApiContext, args: any): Promise<any> => {\n  const action: string = 'deleteUser';\n  const {database, session: {userId: sessionId, userAccess: sessionAccess}} = context;\n  const {userId} = args;\n  const isAdmin: boolean = sessionAccess > 2;\n\n  if(!isAdmin && (sessionId !== userId)) {\n    logException({\n      action,\n      args,\n      category: eventCategory,\n      label: 'unauthorized',\n      value: 'invalid_session'\n    }, context);\n    return null;\n  }\n\n  const aqlQuery: AqlQuery = aql`FOR u IN users\n    FILTER u._key == ${userId}\n    LIMIT 1\n    REMOVE u IN users\n    RETURN OLD`;\n\n  // Stripe\n  const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion, typescript: true});\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((user: User = {}) => stripeClient.customers.del(user.stripeCustomerId)\n      .then(() => stripeClient.accounts.del(user.stripeAccountId))\n      .then(() => user))\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const deactivateUser = (context: ApiContext, userId: string): Promise<User> => {\n  const action: string = 'delete';\n  const {database, session: {userId: sessionId, userAccess: sessionAccess}} = context;\n  const isAdmin: boolean = sessionAccess > 2;\n\n  if(!isAdmin && (sessionId !== userId)) {\n    logException({\n      action,\n      category: eventCategory,\n      label: 'unauthorized',\n      value: 'invalid_session'\n    }, context);\n    return null;\n  }\n\n  const updated: User = {\n    userAccess: 0\n  };\n  const aqlQuery: AqlQuery = aql`UPDATE ${userId} WITH ${updated} IN users LIMIT 1 RETURN NEW`;\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const getDisplayName = (user: User = {}): string => {\n  const {first, last, name = '', username = ''} = user;\n  const fullname: string = ([first, last]).join(' ').trim();\n\n  if(!isEmpty(name)) {\n    return name;\n  } else if(fullname !== '') {\n    return fullname;\n  } else if(!isEmpty(username)) {\n    return username;\n  }\n\n  return 'Unknown';\n};\n\nexport const getSessionUser = (context: ApiContext): Promise<User> => {\n  const action: string = 'getSessionUser';\n  const {database, fields, session: {userId: sessionId, username}} = context;\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  const aqlQuery: string = `LET u = DOCUMENT(\"users/${sessionId}\")\n  ${selectQueries.join('\\n')}\n  RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((user = {}) => user)\n    .catch((error: Error) => logError({\n      action,\n      args: {username, userId: sessionId},\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getUser = (context: ApiContext, args: any): Promise<User> => {\n  const action: string = 'getUser';\n  const {userId} = args;\n  const {database, fields} = context;\n  const formatUserId: string = parseId(userId);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  // Get data from database\n  const aqlQuery: string = `LET u = DOCUMENT(\"users/${formatUserId}\")\n    ${selectQueries.join('\\n')}\n    FILTER u.userAccess > 0\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((user = {}) => user)\n    .catch((error: Error) => logError({\n      action,\n      args,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context)\n      .then(() => { }));\n};\n\nexport const getUsers = (context: ApiContext, options?: UserOptions): Promise<User[]> => {\n  const action: string = 'getUserList';\n  const {database, fields} = context;\n  const {limit, username} = parseUserOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n  const filterBy: string[] = ['u.userAccess > 0'];\n\n  if(!isEmpty(username)) {\n    filterBy.push(`CONTAINS(u.username, \"${parseUsername(username)}\")`);\n  }\n\n  // Get data from database\n  const aqlQuery: string = `FOR u IN users\n    FILTER ${filterBy.join(' && ')}\n    ${selectQueries.join('\\n')}\n    ${limit.aql}\n    SORT u.username\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context).then(() => []));\n};\n\nexport const getUsersByReactions = (\n  context: ApiContext,\n  {reactions = [], username}: any,\n  options?: UserOptions\n): Promise<User[]> => {\n  const action: string = 'getUsersByReactions';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const formatReactions: string[] =  reactions.map((reactionName: string) => parseChar(reactionName, 32).toLowerCase());\n  const {limit} = parseUserOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  const formatSessionId: string = `users/${sessionId}`;\n  const formatUsername: string = parseUsername(username);\n  const filterBy: string[] = [`POSITION(${JSON.stringify(formatReactions)}, LOWER(r.name))`];\n\n  if(!isEmpty(username)) {\n    filterBy.push(`CONTAINS(u.username, \"${formatUsername}\")`);\n  }\n\n  // Get data from database\n  const aqlQuery: string = `FOR u, r IN OUTBOUND \"${formatSessionId}\" hasReactions\n    OPTIONS {vertexCollections: \"users\"}\n    ${selectQueries.join('\\n')}\n    ${filterBy.length ? `FILTER ${filterBy.join(' && ')}` : ''}\n    ${limit.aql}\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  console.log({aqlQuery});\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context).then(() => []));\n};\n\nexport const getUsersByTags = (\n  context: ApiContext,\n  {tags, username}: any,\n  options?: UserOptions\n): Promise<User[]> => {\n  const action: string = 'getUsersByTags';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const formatTags: string[] =  tags.reduce((list: string[], tagName: string) => {\n    if(!isEmpty(tagName)) {\n      list.push(parseChar(tagName, 32).toLowerCase());\n    }\n\n    return list;\n  }, []);\n  const {limit} = parseUserOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  const formatUsername: string = parseUsername(username);\n  const filterBy: string[] = [`u._key != \"${sessionId}\"`];\n\n  if(!isEmpty(username)) {\n    filterBy.push(`CONTAINS(u.username, \"${formatUsername}\")`);\n  }\n\n  // Get data from database\n  const aqlQuery: string = `FOR t IN tags\n    FILTER POSITION(${JSON.stringify(formatTags)}, LOWER(t.name))\n    FOR u, it IN OUTBOUND t isTagged\n    OPTIONS {bfs: true, uniqueVertices: \"global\", vertexCollections: \"users\"}\n    ${selectQueries.join('\\n')}\n    ${filterBy.length ? `FILTER ${filterBy.join(' && ')}` : ''}\n    ${limit.aql}\n    RETURN DISTINCT MERGE(u, {${selectObjects.join(', ')}})`;\n\n  console.log('getUsersByTags', aqlQuery);\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context).then(() => []));\n};\n\nexport const getUsersByLatest = (context: ApiContext, {username}, options?: UserOptions): Promise<User[]> => {\n  const action: string = 'getUsersByLatest';\n  const {database, fields, session: {userId}} = context;\n  const {limit} = parseUserOptions(options);\n  const filter = ['u._id != session._id'];\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  if(!isEmpty(username)) {\n    filter.push(`CONTAINS(u.username, \"${parseUsername(username)}\")`);\n  }\n\n  // Get data from database\n  const aqlQuery: string = `FOR u IN users\n    LET session = DOCUMENT(\"users/${userId}\")\n    FILTER ${filter.join(' && ')}\n    ${selectQueries.join('\\n')}\n    LET distance = DISTANCE(u.latitude || 0, u.longitude || 0, session.latitude || 0, session.longitude || 0)\n    ${limit.aql}\n    SORT distance ASC, u.added DESC\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context).then(() => []));\n};\n\nexport const refreshSession = async (context: ApiContext, {expires, token}): Promise<SessionToken | SessionError> => {\n  const {error} = getSession(token);\n\n  if(error) {\n    return {errors: [error]};\n  }\n\n  const {exp, userId, username, userAccess} = getSession(token);\n  const now: number = Math.ceil(DateTime.local().toSeconds());\n\n  if(exp > now) {\n    return createToken(userId, username, userAccess, expires);\n  }\n\n  return {errors: ['session_expired']};\n};\n\nexport const signIn = async (context: ApiContext, args): Promise<SessionToken> => {\n  const action: string = 'signIn';\n  const {expires, password, username} = args;\n  const formatUsername: string = parseUsername(username);\n  const formatPassword: string = parsePassword(password);\n  const formatExpires: number = parseNum(expires) || 15;\n  const {database} = context;\n\n  if(isEmpty(formatUsername) || isEmpty(formatPassword)) {\n    return logException({\n      action,\n      args: {username},\n      category: eventCategory,\n      value: ErrorTypes.INVALID_ARGUMENTS\n    }, context);\n  }\n\n  const checkQuery: AqlQuery = aql`FOR u IN users\n    FILTER u.username == ${formatUsername}\n    LIMIT 1\n    RETURN u`;\n  const checkUser: User = await database.query(checkQuery)\n    .then((cursor) => cursor.next())\n    .catch((error) => logError({\n      action,\n      args: {username: formatUsername},\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context)\n      .then(() => null)\n    );\n\n  if(!checkUser) {\n    return logException({\n      action,\n      args: {username},\n      category: eventCategory,\n      value: ErrorTypes.INVALID_AUTHENTICATION\n    }, context);\n  }\n\n  const {_key: userId, salt, userAccess} = checkUser;\n  const authPassword: string = createPassword(formatPassword, salt);\n\n  if(checkUser.password !== authPassword) {\n    return logException({\n      action,\n      args: {username},\n      category: eventCategory,\n      value: ErrorTypes.INVALID_AUTHENTICATION\n    }, context);\n  }\n\n  return createToken(userId, username, userAccess, formatExpires);\n};\n\nexport const signOut = async (context: ApiContext, args): Promise<boolean> => {\n  const action: string = 'signOut';\n  const {database, session: {userId: sessionId, username}} = context;\n  const userDocId: string = `users/${sessionId}`;\n\n  const update = {\n    lastOnline: Date.now(),\n    sessionId: null\n  };\n  const sessionQuery: AqlQuery = aql`LET u = DOCUMENT(${userDocId})\n    UPDATE u WITH ${update} IN users\n    LIMIT 1\n    RETURN NEW`;\n\n  try {\n    await database.query(sessionQuery)\n      .then((cursor) => cursor.next())\n      .catch((error) => logError({\n        action,\n        args: {username, userId: sessionId},\n        category: eventCategory,\n        label: ErrorTypes.DATABASE_ERROR\n      }, error, context)\n      );\n  } catch(error) {\n    return false;\n  }\n\n  return true;\n};\n\nexport const getActiveUserCount = (context: ApiContext) => {\n  const action: string = 'getActiveUserCount';\n  const {database} = context;\n  const countQuery: AqlQuery = aql`LET docs = (\n    FOR u IN users\n    FILTER u.active == true\n    RETURN u\n  )\n  RETURN LENGTH(docs)`;\n\n  return database.query(countQuery)\n    .then((cursor) => cursor.next())\n    .catch((error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context)\n      .then(() => 0)\n    );\n};\n"],
  "mappings": "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",
  "names": ["import_utils", "UserAccess", "isEmpty", "Stripe"]
}

@@ -1,57 +1,43 @@
1
1
  var __create = Object.create;
2
2
  var __defProp = Object.defineProperty;
3
- var __defProps = Object.defineProperties;
4
3
  var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
5
- var __getOwnPropDescs = Object.getOwnPropertyDescriptors;
6
4
  var __getOwnPropNames = Object.getOwnPropertyNames;
7
- var __getOwnPropSymbols = Object.getOwnPropertySymbols;
8
5
  var __getProtoOf = Object.getPrototypeOf;
9
6
  var __hasOwnProp = Object.prototype.hasOwnProperty;
10
- var __propIsEnum = Object.prototype.propertyIsEnumerable;
11
- var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
12
- var __spreadValues = (a, b) => {
13
- for (var prop in b || (b = {}))
14
- if (__hasOwnProp.call(b, prop))
15
- __defNormalProp(a, prop, b[prop]);
16
- if (__getOwnPropSymbols)
17
- for (var prop of __getOwnPropSymbols(b)) {
18
- if (__propIsEnum.call(b, prop))
19
- __defNormalProp(a, prop, b[prop]);
20
- }
21
- return a;
22
- };
23
- var __spreadProps = (a, b) => __defProps(a, __getOwnPropDescs(b));
24
- var __markAsModule = (target) => __defProp(target, "__esModule", { value: true });
25
- var __require = typeof require !== "undefined" ? require : (x) => {
26
- throw new Error('Dynamic require of "' + x + '" is not supported');
27
- };
28
7
  var __export = (target, all) => {
29
- __markAsModule(target);
30
8
  for (var name in all)
31
9
  __defProp(target, name, { get: all[name], enumerable: true });
32
10
  };
33
- var __reExport = (target, module2, desc) => {
34
- if (module2 && typeof module2 === "object" || typeof module2 === "function") {
35
- for (let key of __getOwnPropNames(module2))
36
- if (!__hasOwnProp.call(target, key) && key !== "default")
37
- __defProp(target, key, { get: () => module2[key], enumerable: !(desc = __getOwnPropDesc(module2, key)) || desc.enumerable });
11
+ var __copyProps = (to, from, except, desc) => {
12
+ if (from && typeof from === "object" || typeof from === "function") {
13
+ for (let key of __getOwnPropNames(from))
14
+ if (!__hasOwnProp.call(to, key) && key !== except)
15
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
38
16
  }
39
- return target;
40
- };
41
- var __toModule = (module2) => {
42
- return __reExport(__markAsModule(__defProp(module2 != null ? __create(__getProtoOf(module2)) : {}, "default", module2 && module2.__esModule && "default" in module2 ? { get: () => module2.default, enumerable: true } : { value: module2, enumerable: true })), module2);
17
+ return to;
43
18
  };
44
- __export(exports, {
19
+ var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
20
+ // If the importer is in node compatibility mode or this is not an ESM
21
+ // file that has been converted to a CommonJS file using a Babel-
22
+ // compatible transform (i.e. "__esModule" has not been set), then set
23
+ // "default" to the CommonJS "module.exports" for node compatibility.
24
+ isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
25
+ mod
26
+ ));
27
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
28
+ var websockets_exports = {};
29
+ __export(websockets_exports, {
45
30
  addConnection: () => addConnection,
46
31
  getConnectionByConvo: () => getConnectionByConvo,
47
32
  getConnectionById: () => getConnectionById,
48
33
  getConnectionByUser: () => getConnectionByUser,
49
34
  removeConnection: () => removeConnection
50
35
  });
51
- var import_utils = __toModule(require("@nlabs/utils"));
52
- var import_arangojs = __toModule(require("arangojs"));
53
- var import_aws_sdk = __toModule(require("aws-sdk"));
54
- var import_utils2 = __toModule(require("../utils"));
36
+ module.exports = __toCommonJS(websockets_exports);
37
+ var import_utils = require("@nlabs/utils");
38
+ var import_arangojs = require("arangojs");
39
+ var import_aws_sdk = __toESM(require("aws-sdk"));
40
+ var import_utils2 = require("../utils");
55
41
  const cognito = new import_aws_sdk.default.CognitoIdentityServiceProvider({
56
42
  apiVersion: "2016-04-18",
57
43
  region: "us-east-1"
@@ -74,10 +60,11 @@ const addConnection = async (connectionId, token) => {
74
60
  connectionId: formatConnId,
75
61
  modified: Date.now()
76
62
  };
77
- const insert = __spreadProps(__spreadValues({}, update), {
63
+ const insert = {
64
+ ...update,
78
65
  _key: (0, import_utils.createHash)(`connection-${formatConnId}`),
79
66
  sub: userAttributes.sub
80
- });
67
+ };
81
68
  const aqlQry = import_arangojs.aql`UPSERT {sub: ${insert.sub}}
82
69
  INSERT ${insert}
83
70
  UPDATE ${update}
@@ -156,4 +143,4 @@ const removeConnection = (connectionId) => {
156
143
  getConnectionByUser,
157
144
  removeConnection
158
145
  });
159
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/websockets.ts"],
  "sourcesContent": ["import {createHash, parseId, parseString} from '@nlabs/utils';\nimport {aql, Database} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {ArrayCursor} from 'arangojs/cursor';\nimport AWS from 'aws-sdk';\n\nimport {ApiContext} from '../types/auth';\nimport {useDb} from '../utils';\n\nconst cognito = new AWS.CognitoIdentityServiceProvider({\n  apiVersion: '2016-04-18',\n  region: 'us-east-1'\n});\n\nexport const addConnection = async (connectionId: string, token: string): Promise<any> => {\n  // const action: string = 'delete';\n  const formatConnId: string = parseString(connectionId, 32);\n  const formatToken: string = parseString(token);\n\n  // Get user id from token\n  try {\n    const {UserAttributes: cognitoAttributes} = await cognito.getUser({AccessToken: formatToken}).promise();\n\n    // username = cognitoUsername;\n    const userAttributes: any = cognitoAttributes.reduce((userObj, attributes: any) => {\n      const {Name: key, Value: value} = attributes;\n\n      if(key !== undefined) {\n        const formatKey: string = key.replace('custom:', '');\n        userObj[formatKey] = value;\n      }\n\n      return userObj;\n    }, {});\n\n    const db: Database = useDb(process.env.ARANGODB_DATABASE);\n    const update = {\n      connectionId: formatConnId,\n      modified: Date.now()\n    };\n    const insert = {\n      ...update,\n      _key: createHash(`connection-${formatConnId}`),\n      sub: userAttributes.sub\n    };\n    const aqlQry: AqlQuery = aql`UPSERT {sub: ${insert.sub}}\n      INSERT ${insert}\n      UPDATE ${update}\n      IN connections RETURN NEW`;\n\n    return db.query(aqlQry)\n      .then((cursor) => cursor.next())\n      .catch((error: Error) => {\n        throw error;\n      });\n  } catch(error) {\n    throw new Error('Unauthorized Context');\n  }\n};\n\nexport const getConnectionByUser = (userId: string): Promise<any> => {\n  const formatUserId: string = parseId(userId);\n  const db: Database = useDb(process.env.ARANGODB_DATABASE);\n  const aqlQry = aql`FOR c IN connections\n    FILTER c.userId == ${formatUserId}\n    LIMIT 1\n    RETURN c`;\n\n  return db.query(aqlQry)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const getConnectionById = (connectionId: string): Promise<any> => {\n  const formatConnId: string = parseString(connectionId, 32);\n  const db: Database = useDb(process.env.ARANGODB_DATABASE);\n  const aqlQry = aql`FOR c IN connections\n    FILTER c.connectionId == ${formatConnId}\n    LET user = FIRST(\n      FOR u IN users\n      FILTER u.sub == c.sub\n      LIMIT 1\n      RETURN u;\n    )\n    LIMIT 1\n    RETURN MERGE(c, {user:user}`;\n\n  return db.query(aqlQry)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const getConnectionByConvo = (context: ApiContext, convoId: string): Promise<any> => {\n  const {database, session: {userId: sessionId}} = context;\n  const sessionDocId: string = `users/${parseId(sessionId)}`;\n  const convoDocId: string = `conversations/${parseId(convoId)}`;\n  const db: Database = database;\n  const aqlQry = aql`FOR c, e IN 1..1 OUTBOUND ${sessionDocId} hasConversations\n  FILTER c._id == ${convoDocId}\n  LIMIT 1\n  LET u = (\n    FOR inUser, inEdge IN 1..1 INBOUND ${convoDocId} hasConversations\n    FOR n IN connections\n    FILTER inUser.sub == n.sub\n    RETURN n.connectionId\n  )\n  RETURN u`;\n\n  return db.query(aqlQry)\n    .then((cursor) => cursor.all())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const removeConnection = (connectionId: string): Promise<any> => {\n  // const action: string = 'delete';\n  const formatConnId: string = parseString(connectionId, 32);\n  const db: Database = useDb(process.env.ARANGODB_DATABASE);\n  const aqlQry = aql`FOR c IN connections\n      FILTER c.connectionId == ${formatConnId}\n      LIMIT 1\n      REMOVE c IN connections\n      RETURN OLD`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,mBAA+C;AAC/C,sBAA4B;AAG5B,qBAAgB;AAGhB,oBAAoB;AAEpB,MAAM,UAAU,IAAI,uBAAI,+BAA+B;AAAA,EACrD,YAAY;AAAA,EACZ,QAAQ;AAAA;AAGH,MAAM,gBAAgB,OAAO,cAAsB,UAAgC;AAExF,QAAM,eAAuB,8BAAY,cAAc;AACvD,QAAM,cAAsB,8BAAY;AAGxC,MAAI;AACF,UAAM,EAAC,gBAAgB,sBAAqB,MAAM,QAAQ,QAAQ,EAAC,aAAa,eAAc;AAG9F,UAAM,iBAAsB,kBAAkB,OAAO,CAAC,SAAS,eAAoB;AACjF,YAAM,EAAC,MAAM,KAAK,OAAO,UAAS;AAElC,UAAG,QAAQ,QAAW;AACpB,cAAM,YAAoB,IAAI,QAAQ,WAAW;AACjD,gBAAQ,aAAa;AAAA;AAGvB,aAAO;AAAA,OACN;AAEH,UAAM,KAAe,yBAAM,QAAQ,IAAI;AACvC,UAAM,SAAS;AAAA,MACb,cAAc;AAAA,MACd,UAAU,KAAK;AAAA;AAEjB,UAAM,SAAS,iCACV,SADU;AAAA,MAEb,MAAM,6BAAW,cAAc;AAAA,MAC/B,KAAK,eAAe;AAAA;AAEtB,UAAM,SAAmB,mCAAmB,OAAO;AAAA,eACxC;AAAA,eACA;AAAA;AAGX,WAAO,GAAG,MAAM,QACb,KAAK,CAAC,WAAW,OAAO,QACxB,MAAM,CAAC,UAAiB;AACvB,YAAM;AAAA;AAAA,WAEJ,OAAN;AACA,UAAM,IAAI,MAAM;AAAA;AAAA;AAIb,MAAM,sBAAsB,CAAC,WAAiC;AACnE,QAAM,eAAuB,0BAAQ;AACrC,QAAM,KAAe,yBAAM,QAAQ,IAAI;AACvC,QAAM,SAAS;AAAA,yBACQ;AAAA;AAAA;AAIvB,SAAO,GAAG,MAAM,QACb,KAAK,CAAC,WAAW,OAAO,QACxB,MAAM,CAAC,UAAiB;AACvB,UAAM;AAAA;AAAA;AAIL,MAAM,oBAAoB,CAAC,iBAAuC;AACvE,QAAM,eAAuB,8BAAY,cAAc;AACvD,QAAM,KAAe,yBAAM,QAAQ,IAAI;AACvC,QAAM,SAAS;AAAA,+BACc;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAU7B,SAAO,GAAG,MAAM,QACb,KAAK,CAAC,WAAW,OAAO,QACxB,MAAM,CAAC,UAAiB;AACvB,UAAM;AAAA;AAAA;AAIL,MAAM,uBAAuB,CAAC,SAAqB,YAAkC;AAC1F,QAAM,EAAC,UAAU,SAAS,EAAC,QAAQ,gBAAc;AACjD,QAAM,eAAuB,SAAS,0BAAQ;AAC9C,QAAM,aAAqB,iBAAiB,0BAAQ;AACpD,QAAM,KAAe;AACrB,QAAM,SAAS,gDAAgC;AAAA,oBAC7B;AAAA;AAAA;AAAA,yCAGqB;AAAA;AAAA;AAAA;AAAA;AAAA;AAOvC,SAAO,GAAG,MAAM,QACb,KAAK,CAAC,WAAW,OAAO,OACxB,MAAM,CAAC,UAAiB;AACvB,UAAM;AAAA;AAAA;AAIL,MAAM,mBAAmB,CAAC,iBAAuC;AAEtE,QAAM,eAAuB,8BAAY,cAAc;AACvD,QAAM,KAAe,yBAAM,QAAQ,IAAI;AACvC,QAAM,SAAS;AAAA,iCACgB;AAAA;AAAA;AAAA;AAK/B,SAAO,GAAG,MAAM,QACb,KAAK,CAAC,WAAwB,OAAO,QACrC,MAAM,CAAC,UAAiB;AACvB,UAAM;AAAA;AAAA;",
  "names": []
}

146
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/websockets.ts"],
  "sourcesContent": ["import {createHash, parseId, parseString} from '@nlabs/utils';\nimport {aql, Database} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {ArrayCursor} from 'arangojs/cursor';\nimport AWS from 'aws-sdk';\n\nimport {ApiContext} from '../types/auth';\nimport {useDb} from '../utils';\n\nconst cognito = new AWS.CognitoIdentityServiceProvider({\n  apiVersion: '2016-04-18',\n  region: 'us-east-1'\n});\n\nexport const addConnection = async (connectionId: string, token: string): Promise<any> => {\n  // const action: string = 'delete';\n  const formatConnId: string = parseString(connectionId, 32);\n  const formatToken: string = parseString(token);\n\n  // Get user id from token\n  try {\n    const {UserAttributes: cognitoAttributes} = await cognito.getUser({AccessToken: formatToken}).promise();\n\n    // username = cognitoUsername;\n    const userAttributes: any = cognitoAttributes.reduce((userObj, attributes: any) => {\n      const {Name: key, Value: value} = attributes;\n\n      if(key !== undefined) {\n        const formatKey: string = key.replace('custom:', '');\n        userObj[formatKey] = value;\n      }\n\n      return userObj;\n    }, {});\n\n    const db: Database = useDb(process.env.ARANGODB_DATABASE);\n    const update = {\n      connectionId: formatConnId,\n      modified: Date.now()\n    };\n    const insert = {\n      ...update,\n      _key: createHash(`connection-${formatConnId}`),\n      sub: userAttributes.sub\n    };\n    const aqlQry: AqlQuery = aql`UPSERT {sub: ${insert.sub}}\n      INSERT ${insert}\n      UPDATE ${update}\n      IN connections RETURN NEW`;\n\n    return db.query(aqlQry)\n      .then((cursor) => cursor.next())\n      .catch((error: Error) => {\n        throw error;\n      });\n  } catch(error) {\n    throw new Error('Unauthorized Context');\n  }\n};\n\nexport const getConnectionByUser = (userId: string): Promise<any> => {\n  const formatUserId: string = parseId(userId);\n  const db: Database = useDb(process.env.ARANGODB_DATABASE);\n  const aqlQry = aql`FOR c IN connections\n    FILTER c.userId == ${formatUserId}\n    LIMIT 1\n    RETURN c`;\n\n  return db.query(aqlQry)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const getConnectionById = (connectionId: string): Promise<any> => {\n  const formatConnId: string = parseString(connectionId, 32);\n  const db: Database = useDb(process.env.ARANGODB_DATABASE);\n  const aqlQry = aql`FOR c IN connections\n    FILTER c.connectionId == ${formatConnId}\n    LET user = FIRST(\n      FOR u IN users\n      FILTER u.sub == c.sub\n      LIMIT 1\n      RETURN u;\n    )\n    LIMIT 1\n    RETURN MERGE(c, {user:user}`;\n\n  return db.query(aqlQry)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const getConnectionByConvo = (context: ApiContext, convoId: string): Promise<any> => {\n  const {database, session: {userId: sessionId}} = context;\n  const sessionDocId: string = `users/${parseId(sessionId)}`;\n  const convoDocId: string = `conversations/${parseId(convoId)}`;\n  const db: Database = database;\n  const aqlQry = aql`FOR c, e IN 1..1 OUTBOUND ${sessionDocId} hasConversations\n  FILTER c._id == ${convoDocId}\n  LIMIT 1\n  LET u = (\n    FOR inUser, inEdge IN 1..1 INBOUND ${convoDocId} hasConversations\n    FOR n IN connections\n    FILTER inUser.sub == n.sub\n    RETURN n.connectionId\n  )\n  RETURN u`;\n\n  return db.query(aqlQry)\n    .then((cursor) => cursor.all())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const removeConnection = (connectionId: string): Promise<any> => {\n  // const action: string = 'delete';\n  const formatConnId: string = parseString(connectionId, 32);\n  const db: Database = useDb(process.env.ARANGODB_DATABASE);\n  const aqlQry = aql`FOR c IN connections\n      FILTER c.connectionId == ${formatConnId}\n      LIMIT 1\n      REMOVE c IN connections\n      RETURN OLD`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n"],
  "mappings": "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",
  "names": ["import_utils", "AWS"]
}
