@nlabs/reaktor 0.10.0 → 0.10.1

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 (314) hide show
  1. package/.env +1 -0
  2. package/.env.example +1 -0
  3. package/DATABASE_I18N_GUIDE.md +434 -0
  4. package/TEST_UTILITIES_GUIDE.md +360 -0
  5. package/coverage/index.html +46 -76
  6. package/index.js +1 -1
  7. package/jest.config.js +17 -0
  8. package/jest.setup.js +36 -0
  9. package/lex.config.cjs +2 -2
  10. package/lib/actions/apps.js +17 -239
  11. package/lib/actions/connections.js +6 -89
  12. package/lib/actions/content.js +17 -0
  13. package/lib/actions/conversations.js +19 -336
  14. package/lib/actions/dynamodb.js +2 -150
  15. package/lib/actions/email.js +2 -152
  16. package/lib/actions/files.js +5 -283
  17. package/lib/actions/groups.js +19 -259
  18. package/lib/actions/images.js +31 -700
  19. package/lib/actions/index.js +2 -66
  20. package/lib/actions/ios.js +9 -162
  21. package/lib/actions/locations.js +7 -122
  22. package/lib/actions/messages.js +21 -196
  23. package/lib/actions/notifications.js +2 -59
  24. package/lib/actions/payments.js +11 -464
  25. package/lib/actions/posts.js +75 -527
  26. package/lib/actions/profiles.js +8 -0
  27. package/lib/actions/reactions.js +25 -312
  28. package/lib/actions/s3.js +2 -133
  29. package/lib/actions/search.js +5 -90
  30. package/lib/actions/sms.js +2 -108
  31. package/lib/actions/statistics.js +6 -60
  32. package/lib/actions/subscriptions.js +12 -0
  33. package/lib/actions/tags.js +19 -287
  34. package/lib/actions/users.js +64 -764
  35. package/lib/actions/websockets.js +14 -158
  36. package/lib/adapters/arangoAdapter.js +2 -46
  37. package/lib/adapters/contentAdapter.js +2 -0
  38. package/lib/adapters/fileAdapter.js +2 -76
  39. package/lib/adapters/imageAdapter.js +2 -40
  40. package/lib/adapters/messageAdapter.js +2 -49
  41. package/lib/adapters/postAdapter.js +2 -70
  42. package/lib/adapters/reaktorAdapter.js +2 -44
  43. package/lib/adapters/tagAdapter.js +2 -50
  44. package/lib/adapters/userAdapter.js +2 -115
  45. package/lib/config.js +2 -125
  46. package/lib/handlers/graphqlHandler.js +2 -0
  47. package/lib/index.js +2 -66
  48. package/lib/lambdas/actions/websockets.js +14 -116
  49. package/lib/lambdas/authorizer.js +2 -67
  50. package/lib/lambdas/connection.js +2 -91
  51. package/lib/lambdas/utils/message.js +2 -42
  52. package/lib/lambdas/utils/websocket.js +2 -105
  53. package/lib/mocks/conversation.js +2 -35
  54. package/lib/mocks/file.js +2 -38
  55. package/lib/mocks/group.js +2 -47
  56. package/lib/mocks/image.js +2 -44
  57. package/lib/mocks/post.js +2 -55
  58. package/lib/mocks/tag.js +2 -37
  59. package/lib/mocks/user.js +2 -88
  60. package/lib/mutations/index.js +2 -26
  61. package/lib/mutations/locations.integration.js +2 -0
  62. package/lib/mutations/locations.js +2 -44
  63. package/lib/mutations/messages.integration.js +2 -0
  64. package/lib/mutations/messages.js +2 -86
  65. package/lib/mutations/posts.integration.js +2 -0
  66. package/lib/mutations/posts.js +2 -53
  67. package/lib/mutations/profiles.integration.js +2 -0
  68. package/lib/mutations/profiles.js +2 -0
  69. package/lib/mutations/reactions.integration.js +2 -0
  70. package/lib/mutations/reactions.js +2 -51
  71. package/lib/mutations/statistics.integration.js +2 -0
  72. package/lib/mutations/statistics.js +2 -39
  73. package/lib/mutations/subscriptions.integration.js +2 -0
  74. package/lib/mutations/subscriptions.js +2 -56
  75. package/lib/mutations/tags.integration.js +2 -0
  76. package/lib/mutations/tags.js +2 -120
  77. package/lib/mutations/users.integration.js +2 -0
  78. package/lib/mutations/users.js +2 -116
  79. package/lib/objectTypes/app.js +2 -173
  80. package/lib/objectTypes/bankAccount.js +2 -76
  81. package/lib/objectTypes/connection.js +2 -48
  82. package/lib/objectTypes/conversation.js +2 -77
  83. package/lib/objectTypes/creditCard.js +2 -86
  84. package/lib/objectTypes/document.js +2 -46
  85. package/lib/objectTypes/error.js +2 -46
  86. package/lib/objectTypes/external.js +2 -74
  87. package/lib/objectTypes/file.js +2 -100
  88. package/lib/objectTypes/filter.js +2 -43
  89. package/lib/objectTypes/group.js +2 -123
  90. package/lib/objectTypes/iapSubscription.js +2 -40
  91. package/lib/objectTypes/image.js +2 -129
  92. package/lib/objectTypes/index.js +2 -68
  93. package/lib/objectTypes/location.js +2 -109
  94. package/lib/objectTypes/message.js +2 -96
  95. package/lib/objectTypes/passcode.js +2 -42
  96. package/lib/objectTypes/plan.js +2 -95
  97. package/lib/objectTypes/post.js +2 -125
  98. package/lib/objectTypes/profile.js +2 -0
  99. package/lib/objectTypes/reaction.js +2 -61
  100. package/lib/objectTypes/relation.js +2 -49
  101. package/lib/objectTypes/search.js +2 -72
  102. package/lib/objectTypes/statistics.js +2 -39
  103. package/lib/objectTypes/subscription.js +2 -117
  104. package/lib/objectTypes/tag.js +2 -65
  105. package/lib/objectTypes/user.js +2 -144
  106. package/lib/queries/index.js +2 -33
  107. package/lib/queries/locations.integration.js +2 -0
  108. package/lib/queries/locations.js +2 -45
  109. package/lib/queries/messages.integration.js +2 -0
  110. package/lib/queries/messages.js +2 -52
  111. package/lib/queries/posts.integration.js +2 -0
  112. package/lib/queries/posts.js +2 -154
  113. package/lib/queries/reactions.integration.js +2 -0
  114. package/lib/queries/reactions.js +2 -56
  115. package/lib/queries/statistics.js +2 -39
  116. package/lib/queries/subscriptions.js +2 -44
  117. package/lib/queries/tags.integration.js +2 -0
  118. package/lib/queries/tags.js +2 -75
  119. package/lib/queries/users.integration.js +2 -0
  120. package/lib/queries/users.js +2 -64
  121. package/lib/templates/email/layout.js +3 -25
  122. package/lib/templates/email/passwordForgot.js +3 -25
  123. package/lib/templates/email/passwordRecovery.js +3 -25
  124. package/lib/templates/email/verifyEmail.js +3 -25
  125. package/lib/templates/email/welcome.js +3 -25
  126. package/lib/templates/sms/passwordForgot.js +2 -24
  127. package/lib/templates/sms/passwordRecovery.js +2 -24
  128. package/lib/templates/sms/verifyEmail.js +2 -24
  129. package/lib/templates/sms/verifyPhone.js +2 -24
  130. package/lib/templates/sms/welcome.js +2 -24
  131. package/lib/types/apps.types.js +2 -32
  132. package/lib/types/arangodb.types.js +1 -16
  133. package/lib/types/auth.types.js +1 -16
  134. package/lib/types/connections.types.js +1 -16
  135. package/lib/types/content.types.js +1 -0
  136. package/lib/types/conversations.types.js +1 -16
  137. package/lib/types/email.types.js +1 -16
  138. package/lib/types/error.types.js +2 -44
  139. package/lib/types/files.types.js +1 -16
  140. package/lib/types/google.types.js +1 -16
  141. package/lib/types/groups.types.js +1 -16
  142. package/lib/types/images.types.js +1 -16
  143. package/lib/types/index.js +2 -60
  144. package/lib/types/locations.types.js +1 -16
  145. package/lib/types/messages.types.js +1 -16
  146. package/lib/types/notifications.types.js +1 -16
  147. package/lib/types/payments.types.js +1 -16
  148. package/lib/types/posts.types.js +1 -16
  149. package/lib/types/profiles.types.js +1 -0
  150. package/lib/types/statistics.types.js +1 -16
  151. package/lib/types/tags.types.js +1 -16
  152. package/lib/types/users.types.js +1 -16
  153. package/lib/types/websockets.types.js +1 -16
  154. package/lib/utils/adapterUtils.js +2 -45
  155. package/lib/utils/analyticsUtils.js +2 -72
  156. package/lib/utils/arangodbUtils.js +5 -163
  157. package/lib/utils/authUtils.js +2 -0
  158. package/lib/utils/contextUtils.js +2 -0
  159. package/lib/utils/dbI18n.example.js +6 -0
  160. package/lib/utils/dbI18n.js +2 -0
  161. package/lib/utils/googleTranslate.js +2 -0
  162. package/lib/utils/graphqlUtils.js +2 -0
  163. package/lib/utils/index.js +2 -30
  164. package/lib/utils/languageDetection.js +2 -0
  165. package/lib/utils/localeUtils.example.js +2 -0
  166. package/lib/utils/localeUtils.js +2 -0
  167. package/lib/utils/middlewareUtils.js +2 -0
  168. package/lib/utils/sessionUtils.js +2 -0
  169. package/lib/utils/stripeUtils.js +2 -43
  170. package/lib/utils/templateUtils.js +2 -0
  171. package/lib/utils/testUtils.js +2 -0
  172. package/lib/utils/translationQueue.example.js +2 -0
  173. package/lib/utils/translationQueue.js +2 -0
  174. package/package.json +27 -24
  175. package/.prettierrc.js +0 -4
  176. package/lib/actions/apps.d.ts +0 -25
  177. package/lib/actions/connections.d.ts +0 -4
  178. package/lib/actions/conversations.d.ts +0 -14
  179. package/lib/actions/dynamodb.d.ts +0 -8
  180. package/lib/actions/email.d.ts +0 -5
  181. package/lib/actions/files.d.ts +0 -19
  182. package/lib/actions/groups.d.ts +0 -14
  183. package/lib/actions/images.d.ts +0 -26
  184. package/lib/actions/index.d.ts +0 -23
  185. package/lib/actions/ios.d.ts +0 -7
  186. package/lib/actions/locations.d.ts +0 -6
  187. package/lib/actions/messages.d.ts +0 -14
  188. package/lib/actions/notifications.d.ts +0 -5
  189. package/lib/actions/payments.d.ts +0 -10
  190. package/lib/actions/personas.d.ts +0 -3
  191. package/lib/actions/personas.js +0 -110
  192. package/lib/actions/posts.d.ts +0 -22
  193. package/lib/actions/reactions.d.ts +0 -30
  194. package/lib/actions/s3.d.ts +0 -7
  195. package/lib/actions/search.d.ts +0 -3
  196. package/lib/actions/sms.d.ts +0 -9
  197. package/lib/actions/statistics.d.ts +0 -3
  198. package/lib/actions/subscription.d.ts +0 -7
  199. package/lib/actions/subscription.js +0 -208
  200. package/lib/actions/tags.d.ts +0 -34
  201. package/lib/actions/users.d.ts +0 -72
  202. package/lib/actions/websockets.d.ts +0 -20
  203. package/lib/adapters/arangoAdapter.d.ts +0 -2
  204. package/lib/adapters/fileAdapter.d.ts +0 -3
  205. package/lib/adapters/imageAdapter.d.ts +0 -2
  206. package/lib/adapters/messageAdapter.d.ts +0 -2
  207. package/lib/adapters/postAdapter.d.ts +0 -2
  208. package/lib/adapters/reaktorAdapter.d.ts +0 -6
  209. package/lib/adapters/tagAdapter.d.ts +0 -2
  210. package/lib/adapters/userAdapter.d.ts +0 -2
  211. package/lib/config.d.ts +0 -20
  212. package/lib/index.d.ts +0 -12
  213. package/lib/lambdas/actions/websockets.d.ts +0 -7
  214. package/lib/lambdas/authorizer.d.ts +0 -20
  215. package/lib/lambdas/connection.d.ts +0 -12
  216. package/lib/lambdas/utils/message.d.ts +0 -1
  217. package/lib/lambdas/utils/websocket.d.ts +0 -7
  218. package/lib/mocks/conversation.d.ts +0 -8
  219. package/lib/mocks/file.d.ts +0 -11
  220. package/lib/mocks/group.d.ts +0 -17
  221. package/lib/mocks/image.d.ts +0 -3
  222. package/lib/mocks/post.d.ts +0 -38
  223. package/lib/mocks/tag.d.ts +0 -2
  224. package/lib/mocks/user.d.ts +0 -4
  225. package/lib/mutations/index.d.ts +0 -3
  226. package/lib/mutations/locations.d.ts +0 -2
  227. package/lib/mutations/messages.d.ts +0 -2
  228. package/lib/mutations/personas.d.ts +0 -2
  229. package/lib/mutations/personas.js +0 -100
  230. package/lib/mutations/posts.d.ts +0 -2
  231. package/lib/mutations/reactions.d.ts +0 -2
  232. package/lib/mutations/statistics.d.ts +0 -2
  233. package/lib/mutations/subscriptions.d.ts +0 -2
  234. package/lib/mutations/tags.d.ts +0 -2
  235. package/lib/mutations/users.d.ts +0 -1
  236. package/lib/objectTypes/app.d.ts +0 -3
  237. package/lib/objectTypes/bankAccount.d.ts +0 -1
  238. package/lib/objectTypes/connection.d.ts +0 -1
  239. package/lib/objectTypes/conversation.d.ts +0 -2
  240. package/lib/objectTypes/creditCard.d.ts +0 -1
  241. package/lib/objectTypes/document.d.ts +0 -1
  242. package/lib/objectTypes/error.d.ts +0 -1
  243. package/lib/objectTypes/external.d.ts +0 -1
  244. package/lib/objectTypes/file.d.ts +0 -2
  245. package/lib/objectTypes/filter.d.ts +0 -1
  246. package/lib/objectTypes/group.d.ts +0 -3
  247. package/lib/objectTypes/iapSubscription.d.ts +0 -1
  248. package/lib/objectTypes/image.d.ts +0 -2
  249. package/lib/objectTypes/index.d.ts +0 -24
  250. package/lib/objectTypes/location.d.ts +0 -2
  251. package/lib/objectTypes/message.d.ts +0 -2
  252. package/lib/objectTypes/passcode.d.ts +0 -1
  253. package/lib/objectTypes/persona.d.ts +0 -3
  254. package/lib/objectTypes/persona.js +0 -87
  255. package/lib/objectTypes/plan.d.ts +0 -2
  256. package/lib/objectTypes/post.d.ts +0 -2
  257. package/lib/objectTypes/reaction.d.ts +0 -2
  258. package/lib/objectTypes/relation.d.ts +0 -1
  259. package/lib/objectTypes/search.d.ts +0 -1
  260. package/lib/objectTypes/statistics.d.ts +0 -1
  261. package/lib/objectTypes/subscription.d.ts +0 -2
  262. package/lib/objectTypes/tag.d.ts +0 -2
  263. package/lib/objectTypes/user.d.ts +0 -4
  264. package/lib/queries/index.d.ts +0 -3
  265. package/lib/queries/locations.d.ts +0 -2
  266. package/lib/queries/messages.d.ts +0 -2
  267. package/lib/queries/posts.d.ts +0 -2
  268. package/lib/queries/reactions.d.ts +0 -2
  269. package/lib/queries/statistics.d.ts +0 -2
  270. package/lib/queries/subscriptions.d.ts +0 -2
  271. package/lib/queries/tags.d.ts +0 -2
  272. package/lib/queries/users.d.ts +0 -1
  273. package/lib/templates/email/layout.d.ts +0 -2
  274. package/lib/templates/email/passwordForgot.d.ts +0 -2
  275. package/lib/templates/email/passwordRecovery.d.ts +0 -2
  276. package/lib/templates/email/verifyEmail.d.ts +0 -2
  277. package/lib/templates/email/welcome.d.ts +0 -2
  278. package/lib/templates/sms/passwordForgot.d.ts +0 -2
  279. package/lib/templates/sms/passwordRecovery.d.ts +0 -2
  280. package/lib/templates/sms/verifyEmail.d.ts +0 -2
  281. package/lib/templates/sms/verifyPhone.d.ts +0 -2
  282. package/lib/templates/sms/welcome.d.ts +0 -2
  283. package/lib/types/apps.types.d.ts +0 -46
  284. package/lib/types/arangodb.types.d.ts +0 -34
  285. package/lib/types/auth.types.d.ts +0 -9
  286. package/lib/types/connections.types.d.ts +0 -5
  287. package/lib/types/conversations.types.d.ts +0 -27
  288. package/lib/types/email.types.d.ts +0 -13
  289. package/lib/types/error.types.d.ts +0 -20
  290. package/lib/types/files.types.d.ts +0 -23
  291. package/lib/types/google.types.d.ts +0 -29
  292. package/lib/types/groups.types.d.ts +0 -18
  293. package/lib/types/images.types.d.ts +0 -52
  294. package/lib/types/index.d.ts +0 -20
  295. package/lib/types/locations.types.d.ts +0 -18
  296. package/lib/types/messages.types.d.ts +0 -16
  297. package/lib/types/notifications.types.d.ts +0 -19
  298. package/lib/types/payments.types.d.ts +0 -109
  299. package/lib/types/personas.types.d.ts +0 -32
  300. package/lib/types/personas.types.js +0 -16
  301. package/lib/types/posts.types.d.ts +0 -28
  302. package/lib/types/statistics.types.d.ts +0 -3
  303. package/lib/types/tags.types.d.ts +0 -15
  304. package/lib/types/users.types.d.ts +0 -79
  305. package/lib/types/websockets.types.d.ts +0 -18
  306. package/lib/utils/adapterUtils.d.ts +0 -1
  307. package/lib/utils/analyticsUtils.d.ts +0 -21
  308. package/lib/utils/arangodbUtils.d.ts +0 -66
  309. package/lib/utils/auth.d.ts +0 -21
  310. package/lib/utils/auth.js +0 -57
  311. package/lib/utils/index.d.ts +0 -5
  312. package/lib/utils/session.d.ts +0 -18
  313. package/lib/utils/session.js +0 -60
  314. package/lib/utils/stripeUtils.d.ts +0 -3
@@ -1,784 +1,84 @@
1
- var __create = Object.create;
2
- var __defProp = Object.defineProperty;
3
- var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
4
- var __getOwnPropNames = Object.getOwnPropertyNames;
5
- var __getProtoOf = Object.getPrototypeOf;
6
- var __hasOwnProp = Object.prototype.hasOwnProperty;
7
- var __export = (target, all) => {
8
- for (var name in all)
9
- __defProp(target, name, { get: all[name], enumerable: true });
10
- };
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 });
16
- }
17
- return to;
18
- };
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, {
30
- UserAccess: () => UserAccess,
31
- addUser: () => addUser,
32
- confirmCode: () => confirmCode,
33
- createToken: () => createToken,
34
- deactivateUser: () => deactivateUser,
35
- deleteUser: () => deleteUser,
36
- forgotPassword: () => forgotPassword,
37
- getActiveUserCount: () => getActiveUserCount,
38
- getDisplayName: () => getDisplayName,
39
- getSessionUser: () => getSessionUser,
40
- getUser: () => getUser,
41
- getUserByToken: () => getUserByToken,
42
- getUserOptional: () => getUserOptional,
43
- getUsers: () => getUsers,
44
- getUsersByConnection: () => getUsersByConnection,
45
- getUsersByLatest: () => getUsersByLatest,
46
- getUsersByReactions: () => getUsersByReactions,
47
- getUsersByTags: () => getUsersByTags,
48
- parseUserOptions: () => parseUserOptions,
49
- refreshSession: () => refreshSession,
50
- resetPassword: () => resetPassword,
51
- signIn: () => signIn,
52
- signOut: () => signOut,
53
- updateUser: () => updateUser
54
- });
55
- module.exports = __toCommonJS(users_exports);
56
- var import_utils = require("@nlabs/utils");
57
- var import_arangojs = require("arangojs");
58
- var import_luxon = require("luxon");
59
- var import_stripe = __toESM(require("stripe"), 1);
60
- var import_userAdapter = require("../adapters/userAdapter");
61
- var import_config = require("../config");
62
- var import_email = require("./email");
63
- var import_sms = require("./sms");
64
- var import_error = require("../types/error.types");
65
- var import_analyticsUtils = require("../utils/analyticsUtils");
66
- var import_arangodbUtils = require("../utils/arangodbUtils");
67
- var import_session = require("../utils/session");
68
- const eventCategory = "users";
69
- const STRIPE_API_VERSION = "2025-05-28.basil";
70
- var UserAccess = /* @__PURE__ */ ((UserAccess2) => {
71
- UserAccess2[UserAccess2["DEACTIVATED"] = 0] = "DEACTIVATED";
72
- UserAccess2[UserAccess2["ACTIVE"] = 1] = "ACTIVE";
73
- UserAccess2[UserAccess2["PREMIUM"] = 2] = "PREMIUM";
74
- UserAccess2[UserAccess2["CONTENT_ADMIN"] = 3] = "CONTENT_ADMIN";
75
- UserAccess2[UserAccess2["ADMIN"] = 4] = "ADMIN";
76
- return UserAccess2;
77
- })(UserAccess || {});
78
- const createToken = (userId, username, userAccess, expiresInMinutes = 15) => {
79
- const now = import_luxon.DateTime.local();
80
- const sessionExpires = now.plus({ minutes: expiresInMinutes });
81
- const iat = Math.floor(now.toSeconds());
82
- const exp = Math.floor(sessionExpires.toSeconds());
83
- const token = (0, import_session.setSession)({
84
- exp,
85
- iat,
86
- userAccess,
87
- userId,
88
- username
89
- });
90
- return {
91
- expires: sessionExpires.toMillis(),
92
- issued: now.toMillis(),
93
- token,
94
- userId,
95
- username
96
- };
97
- };
98
- const getUserOptional = (fields = []) => fields.reduce((selects, field) => {
99
- if (field.includes("Count")) {
100
- return (0, import_arangodbUtils.selectReactionCountByType)("users", "u", field, selects);
101
- }
102
- return selects;
103
- }, { objects: [], queries: [] });
104
- const parseUserOptions = (options = {}) => {
105
- const {
106
- from = 0,
107
- to = 30
108
- } = options;
109
- const limit = (0, import_arangodbUtils.getLimit)(from, to);
110
- return {
111
- ...options,
112
- limit
113
- };
114
- };
115
- const addUser = async (context, user) => {
116
- const action = "addUser";
117
- const { database } = context;
118
- const { email, password, phone, username } = (0, import_userAdapter.parseUser)(user);
119
- const formatUsername = (0, import_utils.parseUsername)(username);
120
- const formatEmail = (0, import_utils.parseEmail)(email);
121
- const formatPhone = (0, import_utils.parsePhone)(phone);
122
- const formatPassword = (0, import_utils.parsePassword)(password);
123
- const hasPassword = !!formatPassword;
124
- const hasUsername = !!formatUsername || !!formatPhone || !!formatEmail;
125
- if (!hasPassword || !hasUsername) {
126
- return (0, import_analyticsUtils.logException)({
127
- action,
128
- category: eventCategory,
129
- params: { username },
130
- value: import_error.ErrorTypes.INVALID_ARGUMENTS
131
- }, context);
132
- }
133
- const hashId = formatUsername || formatPhone || formatEmail;
134
- const salt = (0, import_utils.createHash)(`${hashId}${formatPassword}`, null);
135
- const encryptedPassword = (0, import_utils.createPassword)(formatPassword, salt);
136
- const filters = [];
137
- if (formatUsername) {
138
- filters.push(`u.username == "${formatUsername}"`);
139
- }
140
- if (formatEmail) {
141
- filters.push(`u.email == "${formatEmail}"`);
142
- }
143
- if (formatPhone) {
144
- filters.push(`u.phone == ${formatPhone}`);
145
- }
146
- const checkQuery = `FOR u IN users
147
- FILTER ${filters.join(" || ")}
1
+ import{createHash as v,createPassword as P,parseArangoId as M,parseChar as Q,parseEmail as H,parseNum as J,parsePassword as L,parsePhone as Y,parseUsername as S}from"@nlabs/utils";import{aql as U}from"arangojs";import{DateTime as X}from"luxon";import z from"stripe";import{parseUser as D}from"../adapters/userAdapter";import{Config as K}from"../config";import{sendEmail as Z}from"./email";import{sendSms as ee}from"./sms";import{ErrorTypes as d}from"../types/error.types";import{logError as I,logException as q}from"../utils/analyticsUtils";import{getDocId as j,getLimit as se,selectReactionCountByType as re,useDb as g}from"../utils/arangodbUtils";import{detectLanguage as te}from"../utils/languageDetection";import{getSession as _,setSession as ne}from"../utils/sessionUtils";const E="users",oe="2025-05-28.basil";var ae=(t=>(t[t.DEACTIVATED=0]="DEACTIVATED",t[t.ACTIVE=1]="ACTIVE",t[t.PREMIUM=2]="PREMIUM",t[t.CONTENT_ADMIN=3]="CONTENT_ADMIN",t[t.ADMIN=4]="ADMIN",t))(ae||{});const B=(e,o,s,i=15)=>{const n=X.local(),t=n.plus({minutes:i}),u=Math.floor(n.toSeconds()),r=Math.floor(t.toSeconds()),a=ne({exp:r,iat:u,userAccess:s,userId:e,username:o});return{expires:t.toMillis(),issued:n.toMillis(),token:a,userId:e,username:o}},$=(e=[])=>e.reduce((o,s)=>s.includes("Count")?re("users","u",s,o):o,{objects:[],queries:[]}),b=(e={})=>{const{from:o=0,to:s=30}=e,i=se(o,s);return{...e,limit:i}},Ue=async(e,o)=>{const s="addUser",{databaseName:i,languageContext:n}=e,{email:t,password:u,phone:r,username:a,userId:c,_key:l,_id:p,...m}=D(o),T=L(u);if(!!!T||!(!!a||!!r||!!t))return q({action:s,category:E,params:{username:a},value:d.INVALID_ARGUMENTS},e);const A=v(`${a||r||t}${T}`,null),N=P(T,A),h=[];a&&h.push(`u.username == "${a}"`),t&&h.push(`u.email == "${t}"`),r&&h.push(`u.phone == ${r}`);const w=`FOR u IN users
2
+ FILTER ${h.join(" || ")}
148
3
  LIMIT 1
149
- RETURN u`;
150
- try {
151
- const existingUsers = await database.query(checkQuery).then((cursor) => cursor.all());
152
- if (existingUsers.length) {
153
- return (0, import_analyticsUtils.logException)({
154
- action,
155
- category: eventCategory,
156
- params: {
157
- email: formatEmail,
158
- phone: formatPhone,
159
- username: formatUsername
160
- },
161
- value: import_error.ErrorTypes.EXISTING_ITEM
162
- }, context);
163
- }
164
- } catch (error) {
165
- return (0, import_analyticsUtils.logError)({
166
- action,
167
- category: eventCategory,
168
- params: { username },
169
- value: import_error.ErrorTypes.DATABASE_ERROR
170
- }, error, context);
171
- }
172
- const verifiedEmailCode = Math.floor(1e5 + Math.random() * 9e5);
173
- const verifiedPhoneCode = Math.floor(1e5 + Math.random() * 9e5);
174
- const insert = {
175
- _key: (0, import_utils.createHash)(formatUsername, null),
176
- added: Date.now(),
177
- email: formatEmail,
178
- modified: Date.now(),
179
- password: encryptedPassword,
180
- phone: formatPhone,
181
- salt,
182
- userAccess: 1,
183
- username: formatUsername,
184
- verifiedEmail: false,
185
- verifiedEmailCode,
186
- verifiedPhone: false,
187
- verifiedPhoneCode
188
- };
189
- const insertQuery = import_arangojs.aql`INSERT ${insert} IN users RETURN NEW`;
190
- return await database.query(insertQuery).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
191
- action,
192
- category: eventCategory,
193
- params: { username },
194
- value: import_error.ErrorTypes.DATABASE_ERROR
195
- }, error, context));
196
- };
197
- const updateUser = async (context, user) => {
198
- const action = "updateUser";
199
- const { database, session } = context;
200
- const { _key, _id, id, tags = [], userId, ...updated } = (0, import_userAdapter.parseUser)(user);
201
- if (!(0, import_session.isAdminUser)(session) && session?.userId !== userId) {
202
- return (0, import_analyticsUtils.logException)({
203
- action,
204
- category: eventCategory,
205
- params: { session },
206
- value: import_error.ErrorTypes.INVALID_SESSION
207
- }, context);
208
- }
209
- const userQuery = import_arangojs.aql`LET u = DOCUMENT(${id})
210
- UPDATE u WITH ${updated} IN users
211
- RETURN NEW`;
212
- try {
213
- const updatedUser = await database.query(userQuery).then((cursor) => cursor.next());
214
- const tagCollection = database.collection("isTagged");
215
- await Promise.all(tags.map(({ id: tagDocId, name }) => {
216
- const tagQuery = import_arangojs.aql`FOR it IN isTagged
217
- FILTER it._from == ${tagDocId} && it._to == ${id} && it.name == ${name}
4
+ RETURN u`;try{if((await g(i).query(w).then(W=>W.all())).length)return q({action:s,category:E,params:{email:t,phone:r,username:a},value:d.EXISTING_ITEM},e)}catch(C){return I({action:s,category:E,params:{email:t,phone:r,username:a},value:d.DATABASE_ERROR},C,e)}const x=r?.replace(/\D/g,"").substring(0,3),O=te({...n,phoneCountryCode:x,userPreference:m.locale||"en"}),F=Math.floor(1e5+Math.random()*9e5),k=Math.floor(1e5+Math.random()*9e5),G={...m,_key:v(a,null),added:Date.now(),email:t,locale:O,modified:Date.now(),password:N,phone:r,salt:A,userAccess:1,username:a,verifiedEmail:!1,verifiedEmailCode:F,verifiedPhone:!1,verifiedSmsCode:k},V=U`INSERT ${G} IN users RETURN NEW`;return await g(i).query(V).then(C=>C.next()).catch(C=>I({action:s,category:E,params:{username:a},value:d.DATABASE_ERROR},C,e))},Ne=async(e,o)=>{const s="updateUser",{databaseName:i}=e,n=D(o),{_key:t,_id:u,tags:r=[],...a}=n,{id:c}=n,l=U`LET u = DOCUMENT(${c})
5
+ UPDATE u WITH ${a} IN users
6
+ RETURN NEW`;try{const p=g(i),m=await p.query(l).then(y=>y.next()),T=p.collection("isTagged");return await Promise.all(r.map(({id:y,name:f})=>{const R=U`FOR it IN isTagged
7
+ FILTER it._from == ${y} && it._to == ${c} && it.name == ${f}
218
8
  LIMIT 1
219
- RETURN it`;
220
- return database.query(tagQuery).then((cursor) => cursor.next()).then((tagEdge) => {
221
- if (!!tagEdge) {
222
- return tagEdge;
223
- }
224
- const edge = {
225
- _from: tagDocId,
226
- _key: (0, import_utils.createHash)(`isTagged-${tagDocId}-${id}`),
227
- _to: id,
228
- added: Date.now(),
229
- name
230
- };
231
- return tagCollection.save(edge, { returnNew: true }).then(() => edge);
232
- });
233
- }));
234
- return updatedUser;
235
- } catch (error) {
236
- return (0, import_analyticsUtils.logError)({
237
- action,
238
- category: eventCategory,
239
- params: { user },
240
- value: import_error.ErrorTypes.DATABASE_ERROR
241
- }, error, context);
242
- }
243
- };
244
- const forgotPassword = async (context, { email, phone, username }) => {
245
- const action = "forgotPassword";
246
- const { app, database } = context;
247
- const aqlQuery = import_arangojs.aql`FOR u IN users
248
- FILTER u.email == ${email} || u.phone == ${phone} || u.username == ${username}
9
+ RETURN it`;return p.query(R).then(A=>A.next()).then(A=>{if(A)return A;const N={_from:y,_key:v(`isTagged-${y}-${c}`),_to:c,added:Date.now(),name:f};return T.save(N,{returnNew:!0}).then(()=>N)})})),m}catch(p){return I({action:s,category:E,params:{user:o},value:d.DATABASE_ERROR},p,e)}},he=async(e,{email:o,phone:s,username:i})=>{const n="forgotPassword",{databaseName:t}=e,u=U`FOR u IN users
10
+ FILTER u.email == ${o} || u.phone == ${s} || u.username == ${i}
249
11
  LIMIT 1
250
- RETURN u`;
251
- try {
252
- return await database.query(aqlQuery).then(async (cursor) => {
253
- const user = cursor.next();
254
- if (user) {
255
- const { email: email2, phone: phone2, verifiedEmail, verifiedPhone } = user;
256
- const codeExpires = 1e3 * 60 * 15;
257
- const code = Math.floor(1e5 + Math.random() * 9e5);
258
- const userDocId = (0, import_arangodbUtils.getDocId)("users", user);
259
- let update;
260
- if (email2 && verifiedEmail) {
261
- (0, import_email.sendEmail)({
262
- app,
263
- text: `Your code is ${code}`
264
- });
265
- update = { verifiedEmailCode: code, verifiedEmailExpires: codeExpires };
266
- }
267
- if (phone2 && verifiedPhone) {
268
- (0, import_sms.sendSms)({
269
- app,
270
- text: `Your code is ${code}`
271
- });
272
- update = { verifiedPhoneCode: code, verifiedPhoneExpires: codeExpires };
273
- }
274
- if (update.verifiedEmailCode || update.verifiedPhoneCode) {
275
- const updateQuery = import_arangojs.aql`UPDATE ${userDocId} WITH ${update} IN users`;
276
- await database.query(updateQuery);
277
- return true;
278
- }
279
- return false;
280
- }
281
- return false;
282
- });
283
- } catch (error) {
284
- (0, import_analyticsUtils.logError)({
285
- action,
286
- category: eventCategory,
287
- params: { email, phone, username },
288
- value: import_error.ErrorTypes.DATABASE_ERROR
289
- }, error, context);
290
- return false;
291
- }
292
- };
293
- const resetPassword = async (context, {
294
- code,
295
- password,
296
- type,
297
- username
298
- }) => {
299
- const action = "resetPassword";
300
- const { database } = context;
301
- const formatPassword = (0, import_utils.parsePassword)(password);
302
- const aqlQuery = import_arangojs.aql`FOR u IN users
303
- FILTER u.username == ${username}
304
- LIMIT 1
305
- RETURN u`;
306
- try {
307
- return await database.query(aqlQuery).then(async (cursor) => {
308
- const user = cursor.next();
309
- if (user) {
310
- const {
311
- _id: userDocId,
312
- salt,
313
- verifiedEmailCode,
314
- verifiedEmailExpires,
315
- verifiedPhoneCode,
316
- verifiedPhoneExpires
317
- } = user;
318
- const now = Date.now();
319
- let update;
320
- switch (type) {
321
- case "email":
322
- if (code === verifiedEmailCode && verifiedEmailExpires > now) {
323
- const password2 = (0, import_utils.createPassword)(formatPassword, salt);
324
- update = { password: password2 };
325
- }
326
- break;
327
- case "phone":
328
- if (code === verifiedPhoneCode && verifiedPhoneExpires > now) {
329
- const password2 = (0, import_utils.createPassword)(formatPassword, salt);
330
- update = { password: password2 };
331
- }
332
- break;
333
- default:
334
- return false;
335
- }
336
- if (update) {
337
- const updateQuery = import_arangojs.aql`UPDATE ${userDocId} WITH ${update} IN users`;
338
- await database.query(updateQuery);
339
- return true;
340
- }
341
- }
342
- return false;
343
- });
344
- } catch (error) {
345
- (0, import_analyticsUtils.logError)({
346
- action,
347
- category: eventCategory,
348
- params: { username },
349
- value: import_error.ErrorTypes.DATABASE_ERROR
350
- }, error, context);
351
- return false;
352
- }
353
- };
354
- const confirmCode = async (context, {
355
- code,
356
- type
357
- }) => {
358
- const action = "confirmCode";
359
- const { database, session: { userId: sessionId } } = context;
360
- const userDocId = (0, import_arangodbUtils.getDocId)("users", { userId: sessionId });
361
- const aqlQuery = import_arangojs.aql`LET u = DOCUMENT(${userDocId}) RETURN u`;
362
- try {
363
- return await database.query(aqlQuery).then((cursor) => cursor.next()).then(({ verifiedEmailCode, verifiedPhoneCode }) => {
364
- switch (type) {
365
- case "email":
366
- return code === verifiedEmailCode;
367
- case "phone":
368
- return code === verifiedPhoneCode;
369
- default:
370
- return false;
371
- }
372
- });
373
- } catch (error) {
374
- (0, import_analyticsUtils.logError)({
375
- action,
376
- category: eventCategory,
377
- params: { code, type },
378
- value: import_error.ErrorTypes.DATABASE_ERROR
379
- }, error, context);
380
- return false;
381
- }
382
- };
383
- const deleteUser = (context, user) => {
384
- const action = "deleteUser";
385
- const { database } = context;
386
- const { userId } = (0, import_userAdapter.parseUser)(user);
387
- const aqlQuery = import_arangojs.aql`FOR u IN users
388
- FILTER u._key == ${userId}
12
+ RETURN u`;try{return await g(t).query(u).then(async r=>{const a=r.next();if(a){const{email:c,phone:l,verifiedEmail:p,verifiedPhone:m}=a,T=1e3*60*15,y=Math.floor(1e5+Math.random()*9e5),f=j("users",a);let R;if(c&&p&&(Z({context:e,text:`Your code is ${y}`}),R={verifiedEmailCode:y,verifiedEmailExpires:T}),l&&m&&(ee({context:e,text:`Your code is ${y}`}),R={verifiedSmsCode:y,verifiedPhoneExpires:T}),R.verifiedEmailCode||R.verifiedSmsCode){const A=U`UPDATE ${f} WITH ${R} IN users`;return await g(t).query(A),!0}return!1}return!1})}catch(r){return I({action:n,category:E,params:{email:o,phone:s,username:i},value:d.DATABASE_ERROR},r,e),!1}},Oe=async(e,{code:o,password:s,type:i,username:n})=>{const t="resetPassword",{databaseName:u}=e,r=L(s),a=U`FOR u IN users
13
+ FILTER u.username == ${n}
389
14
  LIMIT 1
15
+ RETURN u`;try{return await g(u).query(a).then(async c=>{const l=c.next();if(l){const{_id:p,salt:m,verifiedEmailCode:T,verifiedEmailExpires:y,verifiedSmsCode:f,verifiedPhoneExpires:R}=l,A=Date.now();let N;switch(i){case"email":o===T&&y>A&&(N={password:P(r,m)});break;case"phone":o===f&&R>A&&(N={password:P(r,m)});break;default:return!1}if(N){const h=U`UPDATE ${p} WITH ${N} IN users`;return await g(u).query(h),!0}}return!1})}catch(c){return I({action:t,category:E,params:{username:n},value:d.DATABASE_ERROR},c,e),!1}},$e=async(e,{code:o,type:s})=>{const i="confirmCode",{databaseName:n,session:{userId:t}}=e,u=j("users",{userId:t}),r=U`LET u = DOCUMENT(${u}) RETURN u`;try{return await g(n).query(r).then(a=>a.next()).then(({verifiedEmailCode:a,verifiedSmsCode:c})=>{switch(s){case"email":return o===a;case"phone":return o===c;default:return!1}})}catch(a){return I({action:i,category:E,params:{code:o,type:s},value:d.DATABASE_ERROR},a,e),!1}},Ce=(e,o)=>{const s="deleteUser",{databaseName:i}=e,{id:n}=D(o),t=U`LET u = DOCUMENT(${n})
390
16
  REMOVE u IN users
391
- RETURN OLD`;
392
- const stripeClient = new import_stripe.default(import_config.Config.get("stripe.token"), { apiVersion: STRIPE_API_VERSION, typescript: true });
393
- return database.query(aqlQuery).then((cursor) => cursor.next()).then((deletedUser) => stripeClient.customers.del(deletedUser?.stripeCustomerId).then(() => stripeClient.accounts.del(deletedUser?.stripeAccountId)).then(() => deletedUser)).catch((error) => (0, import_analyticsUtils.logError)({
394
- action,
395
- category: eventCategory,
396
- params: { userId },
397
- value: import_error.ErrorTypes.DATABASE_ERROR
398
- }, error, context));
399
- };
400
- const deactivateUser = (context, user) => {
401
- const action = "delete";
402
- const { database } = context;
403
- const { userId } = (0, import_userAdapter.parseUser)(user);
404
- const updated = {
405
- userAccess: 0
406
- };
407
- const aqlQuery = import_arangojs.aql`UPDATE ${userId} WITH ${updated} IN users LIMIT 1 RETURN NEW`;
408
- return database.query(aqlQuery).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
409
- action,
410
- category: eventCategory,
411
- params: { userId },
412
- value: import_error.ErrorTypes.DATABASE_ERROR
413
- }, error, context));
414
- };
415
- const getDisplayName = (user) => {
416
- const { first, last, name = "", username = "" } = user;
417
- const fullname = [first, last].join(" ").trim();
418
- if (name) {
419
- return name;
420
- } else if (fullname !== "") {
421
- return fullname;
422
- } else if (username) {
423
- return username;
424
- }
425
- return "Unknown";
426
- };
427
- const getSessionUser = (context) => {
428
- const action = "getSessionUser";
429
- console.log("getSessionUser", { action, context });
430
- const { database, fields, session: { userId: sessionId, username } } = context;
431
- const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
432
- const formatSessionId = (0, import_utils.parseArangoId)(`users/${sessionId}`);
433
- const aqlQuery = `LET u = DOCUMENT("${formatSessionId}")
434
- ${selectQueries.join("\n")}
435
- RETURN MERGE(u, {${selectObjects.join(", ")}})`;
436
- return database.query(aqlQuery).then((cursor) => cursor.next()).catch((error) => {
437
- (0, import_analyticsUtils.logError)({
438
- action,
439
- category: eventCategory,
440
- params: { userId: sessionId, username },
441
- value: import_error.ErrorTypes.DATABASE_ERROR
442
- }, error, context);
443
- return null;
444
- });
445
- };
446
- const getUser = (context, user) => {
447
- const action = "getUser";
448
- const { id, userId, username } = (0, import_userAdapter.parseUser)(user);
449
- const { database, fields } = context;
450
- const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
451
- let aqlQuery;
452
- console.log({ id, userId, username });
453
- if (id) {
454
- aqlQuery = `LET u = DOCUMENT("${id}")
455
- ${selectQueries.join("\n")}
17
+ RETURN OLD`,u=new z(K.get("stripe.token"),{apiVersion:oe,typescript:!0});return g(i).query(t).then(r=>r.next()).then(r=>u.customers.del(r?.stripeCustomerId).then(()=>u.accounts.del(r?.stripeAccountId)).then(()=>r)).catch(r=>I({action:s,category:E,params:{id:n},value:d.DATABASE_ERROR},r,e))},Se=(e,o)=>{const s="delete",{databaseName:i}=e,{id:n}=D(o),u=U`UPDATE ${n} WITH ${{userAccess:0}} IN users LIMIT 1 RETURN NEW`;return g(i).query(u).then(r=>r.next()).catch(r=>I({action:s,category:E,params:{id:n},value:d.DATABASE_ERROR},r,e))},qe=e=>{const{first:o,last:s,name:i="",username:n=""}=e,t=[o,s].join(" ").trim();return i||(t!==""?t:n||"Unknown")},De=e=>{const o="getSessionUser",{databaseName:s,fields:i,session:{userId:n,username:t}}=e,{objects:u,queries:r}=$(i),c=`LET u = DOCUMENT("${M(`users/${n}`)}")
18
+ ${r.join(`
19
+ `)}
20
+ RETURN MERGE(u, {${u.join(", ")}})`;return g(s).query(c).then(l=>l.next()).catch(l=>(I({action:o,category:E,params:{userId:n,username:t},value:d.DATABASE_ERROR},l,e),null))},be=(e,o)=>{const s="getUser",{email:i,id:n,phone:t,userId:u,username:r}=D(o),{databaseName:a,fields:c}=e,{objects:l,queries:p}=$(c);let m;return n?m=`LET u = DOCUMENT("${n}")
21
+ ${p.join(`
22
+ `)}
456
23
  FILTER u.userAccess > 0
457
- RETURN MERGE(u, {${selectObjects.join(", ")}})`;
458
- } else if (username) {
459
- aqlQuery = `FOR u IN users
460
- FILTER u.username == "${username}"
461
- ${selectQueries.join("\n")}
462
- RETURN MERGE(u, {${selectObjects.join(", ")}})`;
463
- }
464
- return database.query(aqlQuery).then((cursor) => cursor.next()).then((user2) => user2).catch((error) => (0, import_analyticsUtils.logError)({
465
- action,
466
- category: eventCategory,
467
- params: { id, userId, username },
468
- value: import_error.ErrorTypes.DATABASE_ERROR
469
- }, error, context));
470
- };
471
- const getUsers = (context, options) => {
472
- const action = "getUserList";
473
- const { database, fields } = context;
474
- const { limit, username } = parseUserOptions(options);
475
- const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
476
- const filterBy = ["u.userAccess > 0"];
477
- if (username) {
478
- filterBy.push(`CONTAINS(u.username, "${(0, import_utils.parseUsername)(username)}")`);
479
- }
480
- const aqlQuery = `FOR u IN users
481
- FILTER ${filterBy.join(" && ")}
482
- ${selectQueries.join("\n")}
483
- ${limit.aql}
24
+ RETURN MERGE(u, {${l.join(", ")}})`:r?m=`FOR u IN users
25
+ FILTER u.username == "${r}"
26
+ ${p.join(`
27
+ `)}
28
+ RETURN MERGE(u, {${l.join(", ")}})`:i?m=`FOR u IN users
29
+ FILTER u.email == "${i}"
30
+ ${p.join(`
31
+ `)}
32
+ RETURN MERGE(u, {${l.join(", ")}})`:t&&(m=`FOR u IN users
33
+ FILTER u.phone == "${t}"
34
+ ${p.join(`
35
+ `)}
36
+ RETURN MERGE(u, {${l.join(", ")}})`),g(a).query(m).then(T=>T.next()).then(T=>T).catch(T=>I({action:s,category:E,params:{id:n,userId:u,username:r},value:d.DATABASE_ERROR},T,e))},we=(e,o)=>{const s="getUserList",{databaseName:i,fields:n}=e,{limit:t,username:u}=b(o),{objects:r,queries:a}=$(n),c=["u.userAccess > 0"];u&&c.push(`CONTAINS(u.username, "${S(u)}")`);const l=`FOR u IN users
37
+ FILTER ${c.join(" && ")}
38
+ ${a.join(`
39
+ `)}
40
+ ${t.aql}
484
41
  SORT u.username
485
- RETURN MERGE(u, {${selectObjects.join(", ")}})`;
486
- return database.query(aqlQuery).then((cursor) => cursor.all()).catch((error) => {
487
- (0, import_analyticsUtils.logError)({
488
- action,
489
- category: eventCategory,
490
- value: import_error.ErrorTypes.DATABASE_ERROR
491
- }, error, context);
492
- return [];
493
- });
494
- };
495
- const getUsersByReactions = (context, { reactions = [], username }, options) => {
496
- const action = "getUsersByReactions";
497
- const { database, fields, session: { userId: sessionId } } = context;
498
- const formatReactions = reactions.map((reactionName) => (0, import_utils.parseChar)(reactionName, 32).toLowerCase());
499
- const { limit } = parseUserOptions(options);
500
- const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
501
- const formatSessionId = `users/${sessionId}`;
502
- const formatUsername = (0, import_utils.parseUsername)(username);
503
- const filterBy = [
504
- "u.userAccess > 0",
505
- `POSITION(${JSON.stringify(formatReactions)}, LOWER(r.name))`
506
- ];
507
- if (username) {
508
- filterBy.push(`CONTAINS(u.username, "${formatUsername}")`);
509
- }
510
- const aqlQuery = `FOR u, r IN OUTBOUND "${formatSessionId}" hasReaction
42
+ RETURN MERGE(u, {${r.join(", ")}})`;return g(i).query(l).then(p=>p.all()).catch(p=>(I({action:s,category:E,value:d.DATABASE_ERROR},p,e),[]))},Pe=(e,{reactions:o=[],username:s},i)=>{const n="getUsersByReactions",{databaseName:t,fields:u,session:{userId:r}}=e,a=o.map(R=>Q(R,32).toLowerCase()),{limit:c}=b(i),{objects:l,queries:p}=$(u),m=`users/${r}`,T=S(s),y=["u.userAccess > 0",`POSITION(${JSON.stringify(a)}, LOWER(r.name))`];s&&y.push(`CONTAINS(u.username, "${T}")`);const f=`FOR u, r IN OUTBOUND "${m}" hasReaction
511
43
  OPTIONS {vertexCollections: "users"}
512
- ${selectQueries.join("\n")}
513
- FILTER ${filterBy.join(" && ")}
514
- ${limit.aql}
515
- RETURN MERGE(u, {${selectObjects.join(", ")}})`;
516
- return database.query(aqlQuery).then((cursor) => cursor.all()).catch((error) => {
517
- (0, import_analyticsUtils.logError)({
518
- action,
519
- category: eventCategory,
520
- value: import_error.ErrorTypes.DATABASE_ERROR
521
- }, error, context);
522
- return [];
523
- });
524
- };
525
- const getUsersByTags = (context, { tags, username }, options) => {
526
- const action = "getUsersByTags";
527
- const { database, fields, session: { userId: sessionId } } = context;
528
- const formatTags = tags?.reduce((list, tagName) => {
529
- if (tagName) {
530
- list.push((0, import_utils.parseChar)(tagName, 32).toLowerCase());
531
- }
532
- return list;
533
- }, []);
534
- const { limit } = parseUserOptions(options);
535
- const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
536
- const formatUsername = (0, import_utils.parseUsername)(username);
537
- const filterBy = [
538
- `u._key != "${sessionId}"`,
539
- "u.userAccess > 0"
540
- ];
541
- if (username) {
542
- filterBy.push(`CONTAINS(u.username, "${formatUsername}")`);
543
- }
544
- const aqlQuery = `FOR t IN tags
545
- FILTER POSITION(${JSON.stringify(formatTags)}, LOWER(t.name))
44
+ ${p.join(`
45
+ `)}
46
+ FILTER ${y.join(" && ")}
47
+ ${c.aql}
48
+ RETURN MERGE(u, {${l.join(", ")}})`;return g(t).query(f).then(R=>R.all()).catch(R=>(I({action:n,category:E,value:d.DATABASE_ERROR},R,e),[]))},xe=(e,{tags:o,username:s},i)=>{const n="getUsersByTags",{databaseName:t,fields:u,session:{userId:r}}=e,a=o?.reduce((f,R)=>(R&&f.push(Q(R,32).toLowerCase()),f),[]),{limit:c}=b(i),{objects:l,queries:p}=$(u),m=S(s),T=[`u._key != "${r}"`,"u.userAccess > 0"];s&&T.push(`CONTAINS(u.username, "${m}")`);const y=`FOR t IN tags
49
+ FILTER POSITION(${JSON.stringify(a)}, LOWER(t.name))
546
50
  FOR u, it IN OUTBOUND t isTagged
547
51
  OPTIONS {bfs: true, uniqueVertices: "global", vertexCollections: "users"}
548
- ${selectQueries.join("\n")}
549
- FILTER ${filterBy.join(" && ")}
550
- ${limit.aql}
551
- RETURN DISTINCT MERGE(u, {${selectObjects.join(", ")}})`;
552
- return database.query(aqlQuery).then((cursor) => cursor.all()).catch((error) => {
553
- (0, import_analyticsUtils.logError)({
554
- action,
555
- category: eventCategory,
556
- value: import_error.ErrorTypes.DATABASE_ERROR
557
- }, error, context);
558
- return [];
559
- });
560
- };
561
- const getUsersByLatest = (context, { username }, options) => {
562
- const action = "getUsersByLatest";
563
- const { database, fields, session: { userId } } = context;
564
- const { limit } = parseUserOptions(options);
565
- const filter = [
566
- "u._id != session._id",
567
- "u.userAccess > 0"
568
- ];
569
- const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
570
- if (username) {
571
- filter.push(`CONTAINS(u.username, "${(0, import_utils.parseUsername)(username)}")`);
572
- }
573
- const aqlQuery = `FOR u IN users
574
- LET session = DOCUMENT("users/${userId}")
575
- FILTER ${filter.join(" && ")}
576
- ${selectQueries.join("\n")}
52
+ ${p.join(`
53
+ `)}
54
+ FILTER ${T.join(" && ")}
55
+ ${c.aql}
56
+ RETURN DISTINCT MERGE(u, {${l.join(", ")}})`;return g(t).query(y).then(f=>f.all()).catch(f=>(I({action:n,category:E,value:d.DATABASE_ERROR},f,e),[]))},ve=(e,{username:o},s)=>{const i="getUsersByLatest",{databaseName:n,fields:t,session:{userId:u}}=e,{limit:r}=b(s),a=["u._id != session._id","u.userAccess > 0"],{objects:c,queries:l}=$(t);o&&a.push(`CONTAINS(u.username, "${S(o)}")`);const p=`FOR u IN users
57
+ LET session = DOCUMENT("users/${u}")
58
+ FILTER ${a.join(" && ")}
59
+ ${l.join(`
60
+ `)}
577
61
  LET distance = DISTANCE(u.latitude || 0, u.longitude || 0, session.latitude || 0, session.longitude || 0)
578
- ${limit.aql}
62
+ ${r.aql}
579
63
  SORT distance ASC, u.added DESC
580
- RETURN MERGE(u, {${selectObjects.join(", ")}})`;
581
- return database.query(aqlQuery).then((cursor) => cursor.all()).catch((error) => {
582
- (0, import_analyticsUtils.logError)({
583
- action,
584
- category: eventCategory,
585
- value: import_error.ErrorTypes.DATABASE_ERROR
586
- }, error, context);
587
- return [];
588
- });
589
- };
590
- const getUsersByConnection = (context, { userId }, options) => {
591
- const action = "getUsersByConnection";
592
- const { database, fields } = context;
593
- const { limit, username } = parseUserOptions(options);
594
- const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
595
- const formatUserId = (0, import_utils.parseArangoId)(`users/${userId}`);
596
- const filterBy = [
597
- "u.userAccess > 0"
598
- ];
599
- if (username) {
600
- filterBy.push(`CONTAINS(u.username, "${(0, import_utils.parseUsername)(username)}")`);
601
- }
602
- const aqlQuery = `FOR cu IN users
603
- LET session = DOCUMENT("${formatUserId}")
64
+ RETURN MERGE(u, {${c.join(", ")}})`;return g(n).query(p).then(m=>m.all()).catch(m=>(I({action:i,category:E,value:d.DATABASE_ERROR},m,e),[]))},Le=(e,{userId:o},s)=>{const i="getUsersByConnection",{databaseName:n,fields:t}=e,{limit:u,username:r}=b(s),{objects:a,queries:c}=$(t),l=M(`users/${o}`),p=["u.userAccess > 0"];r&&p.push(`CONTAINS(u.username, "${S(r)}")`);const m=`FOR cu IN users
65
+ LET session = DOCUMENT("${l}")
604
66
  FOR u, connection IN OUTBOUND cu hasConnection
605
67
  OPTIONS {bfs: true, uniqueVertices: "global", vertexCollections: "users"}
606
- ${selectQueries.join("\n")}
607
- FILTER ${filterBy.join(" && ")}
608
- ${limit.aql}
609
- RETURN DISTINCT MERGE(u, {${selectObjects.join(", ")}})`;
610
- return database.query(aqlQuery).then((cursor) => cursor.all()).catch((error) => {
611
- (0, import_analyticsUtils.logError)({
612
- action,
613
- category: eventCategory,
614
- value: import_error.ErrorTypes.DATABASE_ERROR
615
- }, error, context);
616
- return [];
617
- });
618
- };
619
- const refreshSession = ({ expires, token }) => {
620
- try {
621
- const { userId, username, userAccess } = (0, import_session.getSession)(token);
622
- return createToken(userId, username, userAccess, expires);
623
- } catch (error) {
624
- throw error;
625
- }
626
- };
627
- const signIn = async (context, args) => {
628
- const action = "signIn";
629
- const { database } = context;
630
- const { email, expires, password, phone, username } = args;
631
- const formatEmail = (0, import_utils.parseEmail)(email);
632
- const formatUsername = (0, import_utils.parseUsername)(username);
633
- const formatPassword = (0, import_utils.parsePassword)(password);
634
- const formatPhone = (0, import_utils.parsePhone)(phone);
635
- const formatExpires = (0, import_utils.parseNum)(expires) || 15;
636
- if (!formatUsername && !formatEmail && !formatPhone || !formatPassword) {
637
- (0, import_analyticsUtils.logException)({
638
- action,
639
- category: eventCategory,
640
- params: { username },
641
- value: import_error.ErrorTypes.INVALID_ARGUMENTS
642
- }, context);
643
- return null;
644
- }
645
- const filters = [];
646
- if (formatEmail) {
647
- filters.push(`u.email == "${formatEmail}"`);
648
- }
649
- if (formatPhone) {
650
- filters.push(`u.phone == ${formatPhone}`);
651
- }
652
- if (formatUsername) {
653
- filters.push(`u.username == "${formatUsername}"`);
654
- }
655
- const checkQuery = `FOR u IN users
656
- FILTER ${filters.join(" || ")}
68
+ ${c.join(`
69
+ `)}
70
+ FILTER ${p.join(" && ")}
71
+ ${u.aql}
72
+ RETURN DISTINCT MERGE(u, {${a.join(", ")}})`;return g(n).query(m).then(T=>T.all()).catch(T=>(I({action:i,category:E,value:d.DATABASE_ERROR},T,e),[]))},je=({expires:e,token:o})=>{try{const{userId:s,username:i,userAccess:n}=_(o);return B(s,i,n,e)}catch(s){throw s}},Me=async(e,o)=>{const s="signIn",{databaseName:i}=e,{email:n,expires:t,password:u,phone:r,username:a}=o,c=H(n),l=S(a),p=L(u),m=Y(r),T=J(t)||15;if(!l&&!c&&!m||!p)return q({action:s,category:E,params:{username:a},value:d.INVALID_ARGUMENTS},e),null;const y=[];c&&y.push(`u.email == "${c}"`),m&&y.push(`u.phone == ${m}`),l&&y.push(`u.username == "${l}"`);const f=`FOR u IN users
73
+ FILTER ${y.join(" || ")}
657
74
  LIMIT 1
658
- RETURN u`;
659
- let checkUser;
660
- try {
661
- checkUser = await database.query(checkQuery).then((cursor) => cursor.next());
662
- } catch (error) {
663
- (0, import_analyticsUtils.logError)({
664
- action,
665
- category: eventCategory,
666
- params: { username: formatUsername },
667
- value: import_error.ErrorTypes.DATABASE_ERROR
668
- }, error, context);
669
- return null;
670
- }
671
- if (!checkUser) {
672
- (0, import_analyticsUtils.logException)({
673
- action,
674
- category: eventCategory,
675
- params: { username },
676
- value: import_error.ErrorTypes.INVALID_AUTHENTICATION
677
- }, context);
678
- return null;
679
- }
680
- const { _key: userId, password: validPassword, salt, userAccess } = checkUser;
681
- const authPassword = (0, import_utils.createPassword)(formatPassword, salt);
682
- if (validPassword !== authPassword) {
683
- (0, import_analyticsUtils.logException)({
684
- action,
685
- category: eventCategory,
686
- params: { userAccess, userId, username },
687
- value: import_error.ErrorTypes.INVALID_AUTHENTICATION
688
- }, context);
689
- return null;
690
- }
691
- try {
692
- console.log({ formatExpires, userAccess, userId, username });
693
- const token = createToken(userId, username, userAccess, formatExpires);
694
- console.log({ token });
695
- return token;
696
- } catch (error) {
697
- (0, import_analyticsUtils.logError)({
698
- action,
699
- category: eventCategory,
700
- value: import_error.ErrorTypes.DATABASE_ERROR
701
- }, error, context);
702
- return null;
703
- }
704
- };
705
- const signOut = async (context) => {
706
- const action = "signOut";
707
- const { database, session: { userId: sessionId, username } } = context;
708
- const userDocId = `users/${sessionId}`;
709
- const update = {
710
- lastOnline: Date.now(),
711
- sessionId: null
712
- };
713
- const sessionQuery = import_arangojs.aql`LET u = DOCUMENT(${userDocId})
714
- UPDATE u WITH ${update} IN users
75
+ RETURN u`;let R;try{R=await g(i).query(f).then(O=>O.next())}catch(O){return I({action:s,category:E,params:{username:l},value:d.DATABASE_ERROR},O,e),null}if(!R)return q({action:s,category:E,params:{username:a},value:d.INVALID_AUTHENTICATION},e),null;const{_key:A,password:N,salt:h,userAccess:w}=R,x=P(p,h);if(N!==x)return q({action:s,category:E,params:{userAccess:w,userId:A,username:a},value:d.INVALID_AUTHENTICATION},e),null;try{return B(A,a,w,T)}catch(O){return I({action:s,category:E,params:{userId:A,username:a},value:d.DATABASE_ERROR},O,e),null}},Qe=async e=>{const o="signOut",{databaseName:s,session:{userId:i,username:n}}=e,t=`users/${i}`,u={lastOnline:Date.now(),sessionId:null},r=U`LET u = DOCUMENT(${t})
76
+ UPDATE u WITH ${u} IN users
715
77
  LIMIT 1
716
- RETURN NEW`;
717
- try {
718
- await database.query(sessionQuery).then((cursor) => cursor.next());
719
- } catch (error) {
720
- await (0, import_analyticsUtils.logError)({
721
- action,
722
- category: eventCategory,
723
- params: { userId: sessionId, username },
724
- value: import_error.ErrorTypes.DATABASE_ERROR
725
- }, error, context);
726
- }
727
- return true;
728
- };
729
- const getActiveUserCount = (context) => {
730
- const action = "getActiveUserCount";
731
- const { database } = context;
732
- const countQuery = import_arangojs.aql`LET docs = (
78
+ RETURN NEW`;try{await g(s).query(r).then(a=>a.next())}catch(a){await I({action:o,category:E,params:{userId:i,username:n},value:d.DATABASE_ERROR},a,e)}return!0},_e=e=>{const o="getActiveUserCount",{databaseName:s}=e,i=U`LET docs = (
733
79
  FOR u IN users
734
80
  FILTER u.active == true
735
81
  RETURN u
736
82
  )
737
- RETURN LENGTH(docs)`;
738
- return database.query(countQuery).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
739
- action,
740
- category: eventCategory,
741
- value: import_error.ErrorTypes.DATABASE_ERROR
742
- }, error, context));
743
- };
744
- const getUserByToken = (context, token) => {
745
- const action = "getUserByToken";
746
- const { database } = context;
747
- const { userId } = (0, import_session.getSession)(token);
748
- const userDocId = (0, import_arangodbUtils.getDocId)("users", { userId });
749
- const aqlQuery = import_arangojs.aql`LET u = DOCUMENT("${userDocId}") RETURN u`;
750
- return database.query(aqlQuery).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
751
- action,
752
- category: eventCategory,
753
- params: { userId },
754
- value: import_error.ErrorTypes.DATABASE_ERROR
755
- }, error, context));
756
- };
757
- // Annotate the CommonJS export names for ESM import in node:
758
- 0 && (module.exports = {
759
- UserAccess,
760
- addUser,
761
- confirmCode,
762
- createToken,
763
- deactivateUser,
764
- deleteUser,
765
- forgotPassword,
766
- getActiveUserCount,
767
- getDisplayName,
768
- getSessionUser,
769
- getUser,
770
- getUserByToken,
771
- getUserOptional,
772
- getUsers,
773
- getUsersByConnection,
774
- getUsersByLatest,
775
- getUsersByReactions,
776
- getUsersByTags,
777
- parseUserOptions,
778
- refreshSession,
779
- resetPassword,
780
- signIn,
781
- signOut,
782
- updateUser
783
- });
784
- //# 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  parseArangoId,\n  parseChar,\n  parseEmail,\n  parseNum,\n  parsePassword,\n  parsePhone,\n  parseUsername\n} from '@nlabs/utils';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {DateTime} from 'luxon';\nimport Stripe from 'stripe';\n\nimport {parseUser} from '../adapters/userAdapter';\nimport {Config} from '../config';\nimport {sendEmail} from './email';\nimport {sendSms} from './sms';\nimport {ErrorTypes, SessionError} from '../types/error.types';\nimport {logError, logException} from '../utils/analyticsUtils';\nimport {getDocId, getLimit, selectReactionCountByType} from '../utils/arangodbUtils';\nimport {getSession, isAdminUser, SessionToken, setSession} from '../utils/session';\n\nimport type {ApiContext} from '../types/auth.types';\nimport type {UserType} from '../types/users.types';\nimport type {EdgeCollection} from 'arangojs/collections';\n\n\nconst eventCategory = 'users';\nconst STRIPE_API_VERSION = '2025-05-28.basil';\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  expiresInMinutes: number = 15\n): SessionToken => {\n  const now: DateTime = DateTime.local();\n  const sessionExpires: DateTime = now.plus({minutes: expiresInMinutes});\n  const iat: number = Math.floor(now.toSeconds());\n  const exp: number = Math.floor(sessionExpires.toSeconds());\n  const token = setSession({\n    exp,\n    iat,\n    userAccess,\n    userId,\n    username\n  });\n\n  return {\n    expires: sessionExpires.toMillis(),\n    issued: now.toMillis(),\n    token,\n    userId,\n    username\n  };\n};\n\ninterface SelectAccumulator {\n  objects: string[];\n  queries: string[];\n}\n\nexport const getUserOptional = (fields: string[] = []): SelectAccumulator =>\n  fields.reduce((selects: SelectAccumulator, 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  const limit = getLimit(from, to);\n\n  return {\n    ...options,\n    limit\n  };\n};\n\nexport const addUser = async (context: ApiContext, user: UserType): Promise<UserType> => {\n  const action = 'addUser';\n  const {database} = context;\n  const {email, password, phone, username} = parseUser(user);\n  const formatUsername: string = parseUsername(username);\n  const formatEmail: string = parseEmail(email);\n  const formatPhone: string = parsePhone(phone);\n  const formatPassword: string = parsePassword(password);\n  const hasPassword = !!formatPassword;\n  const hasUsername = !!formatUsername || !!formatPhone || !!formatEmail;\n\n  if(!hasPassword || !hasUsername) {\n    return logException({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.INVALID_ARGUMENTS\n    }, context);\n  }\n\n  const hashId = formatUsername || formatPhone || formatEmail;\n  const salt: string = createHash(`${hashId}${formatPassword}`, null);\n  const encryptedPassword = createPassword(formatPassword, salt);\n  const filters: string[] = [];\n\n  if(formatUsername) {\n    filters.push(`u.username == \"${formatUsername}\"`);\n  }\n\n  if(formatEmail) {\n    filters.push(`u.email == \"${formatEmail}\"`);\n  }\n\n  if(formatPhone) {\n    filters.push(`u.phone == ${formatPhone}`);\n  }\n\n  const checkQuery: string = `FOR u IN users\n    FILTER ${filters.join(' || ')}\n    LIMIT 1\n    RETURN u`;\n\n  try {\n    const existingUsers = await database.query(checkQuery).then((cursor) => cursor.all());\n\n    if(existingUsers.length) {\n      return logException({\n        action,\n        category: eventCategory,\n        params: {\n          email: formatEmail,\n          phone: formatPhone,\n          username: formatUsername\n        },\n        value: ErrorTypes.EXISTING_ITEM\n      }, context);\n    }\n  } catch(error) {\n    return logError({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, 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  const insert: UserType = {\n    _key: createHash(formatUsername, null),\n    added: Date.now(),\n    email: formatEmail,\n    modified: Date.now(),\n    password: encryptedPassword,\n    phone: formatPhone,\n    salt,\n    userAccess: 1,\n    username: formatUsername,\n    verifiedEmail: false,\n    verifiedEmailCode,\n    verifiedPhone: false,\n    verifiedPhoneCode\n  };\n\n  const insertQuery: AqlQuery = aql`INSERT ${insert} IN users RETURN NEW`;\n\n  return await database.query(insertQuery)\n    .then((cursor) => cursor.next())\n    .catch((error) => logError({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const updateUser = async (context: ApiContext, user: UserType): Promise<UserType> => {\n  const action = 'updateUser';\n  const {database, session} = context;\n  const {_key, _id, id, tags = [], userId, ...updated} = parseUser(user);\n\n  if(!isAdminUser(session) && (session?.userId !== userId)) {\n    return logException({\n      action,\n      category: eventCategory,\n      params: {session},\n      value: ErrorTypes.INVALID_SESSION\n    }, context);\n  }\n\n  const userQuery: AqlQuery = aql`LET u = DOCUMENT(${id})\n    UPDATE u WITH ${updated} IN users\n    RETURN NEW`;\n\n  try {\n    const updatedUser = await database.query(userQuery).then((cursor) => cursor.next());\n    const tagCollection: EdgeCollection = database.collection('isTagged');\n\n    await Promise.all(tags.map(({id: tagDocId, name}) => {\n      const tagQuery: AqlQuery = aql`FOR it IN isTagged\n        FILTER it._from == ${tagDocId} && it._to == ${id} && it.name == ${name}\n        LIMIT 1\n        RETURN it`;\n\n      return database.query(tagQuery)\n        .then((cursor) => cursor.next())\n        .then((tagEdge) => {\n          if(!!tagEdge) {\n            return tagEdge;\n          }\n\n          const edge = {\n            _from: tagDocId,\n            _key: createHash(`isTagged-${tagDocId}-${id}`),\n            _to: id,\n            added: Date.now(),\n            name\n          };\n\n          return tagCollection.save(edge, {returnNew: true}).then(() => edge);\n        });\n    }));\n\n    return updatedUser;\n  } catch(error) {\n    return logError({\n      action,\n      category: eventCategory,\n      params: {user},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n  }\n};\n\nexport const forgotPassword = async (context: ApiContext, {email, phone, username}): Promise<boolean> => {\n  const action = 'forgotPassword';\n  const {app, database} = context;\n  const aqlQuery: AqlQuery = aql`FOR u IN users\n    FILTER u.email == ${email} || u.phone == ${phone} || u.username == ${username}\n    LIMIT 1\n    RETURN u`;\n\n  try {\n    return await database.query(aqlQuery)\n      .then(async (cursor) => {\n        const user = cursor.next();\n\n        if(user) {\n          const {email, phone, verifiedEmail, verifiedPhone} = user as UserType;\n          const codeExpires = 1000 * 60 * 15; // 15 minutes\n          const code = Math.floor(100000 + (Math.random() * 900000));\n          const userDocId = getDocId('users', user);\n          let update;\n\n          if(email && verifiedEmail) {\n            sendEmail({\n              app,\n              text: `Your code is ${code}`\n            });\n            update = {verifiedEmailCode: code, verifiedEmailExpires: codeExpires};\n          }\n\n          if(phone && verifiedPhone) {\n            sendSms({\n              app,\n              text: `Your code is ${code}`\n            });\n            update = {verifiedPhoneCode: code, verifiedPhoneExpires: codeExpires};\n          }\n\n          if(update.verifiedEmailCode || update.verifiedPhoneCode) {\n            const updateQuery: AqlQuery = aql`UPDATE ${userDocId} WITH ${update} IN users`;\n\n            await database.query(updateQuery);\n\n            return true;\n          }\n\n          return false;\n        }\n\n        return false;\n      });\n  } catch(error) {\n    logError({\n      action,\n      category: eventCategory,\n      params: {email, phone, username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n\n    return false;\n  }\n};\n\nexport const resetPassword = async (\n  context: ApiContext,\n  {\n    code,\n    password,\n    type,\n    username\n  }: {\n    code: number,\n    password: string,\n    type: 'phone' | 'email',\n    username: string\n  }\n): Promise<boolean> => {\n  const action = 'resetPassword';\n  const {database} = context;\n  const formatPassword: string = parsePassword(password);\n  const aqlQuery: AqlQuery = aql`FOR u IN users\n    FILTER u.username == ${username}\n    LIMIT 1\n    RETURN u`;\n\n  try {\n    return await database.query(aqlQuery)\n      .then(async (cursor) => {\n        const user = cursor.next();\n\n        if(user) {\n          const {\n            _id: userDocId,\n            salt,\n            verifiedEmailCode,\n            verifiedEmailExpires,\n            verifiedPhoneCode,\n            verifiedPhoneExpires\n          } = user as UserType;\n          const now = Date.now();\n          let update;\n\n          switch(type) {\n            case 'email':\n              if(code === verifiedEmailCode && verifiedEmailExpires > now) {\n                const password: string = createPassword(formatPassword, salt);\n                update = {password};\n              }\n              break;\n            case 'phone':\n              if(code === verifiedPhoneCode && verifiedPhoneExpires > now) {\n                const password: string = createPassword(formatPassword, salt);\n                update = {password};\n              }\n              break;\n            default:\n              return false;\n          }\n\n          if(update) {\n            const updateQuery: AqlQuery = aql`UPDATE ${userDocId} WITH ${update} IN users`;\n\n            await database.query(updateQuery);\n\n            return true;\n          }\n        }\n\n        return false;\n      });\n  } catch(error) {\n    logError({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n\n    return false;\n  }\n};\n\nexport const confirmCode = async (\n  context: ApiContext,\n  {\n    code,\n    type\n  }: {\n    code: number,\n    type: 'phone' | 'email'\n  }\n): Promise<boolean> => {\n  const action = 'confirmCode';\n  const {database, session: {userId: sessionId}} = context;\n  const userDocId = getDocId('users', {userId: sessionId});\n  const aqlQuery: AqlQuery = aql`LET u = DOCUMENT(${userDocId}) RETURN u`;\n\n  try {\n    return await database.query(aqlQuery)\n      .then((cursor) => cursor.next())\n      .then(({verifiedEmailCode, verifiedPhoneCode}: UserType) => {\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) {\n    logError({\n      action,\n      category: eventCategory,\n      params: {code, type},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n\n    return false;\n  }\n};\n\nexport const deleteUser = (context: ApiContext, user: UserType): Promise<UserType> => {\n  const action = 'deleteUser';\n  const {database} = context;\n  const {userId} = parseUser(user);\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  const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion: STRIPE_API_VERSION, typescript: true});\n\n  return database.query(aqlQuery)\n    .then((cursor) => cursor.next())\n    .then((deletedUser) => stripeClient.customers.del(deletedUser?.stripeCustomerId)\n      .then(() => stripeClient.accounts.del(deletedUser?.stripeAccountId))\n      .then(() => deletedUser))\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      params: {userId},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error,context));\n};\n\nexport const deactivateUser = (context: ApiContext, user: UserType): Promise<UserType> => {\n  const action = 'delete';\n  const {database} = context;\n  const {userId} = parseUser(user);\n  const updated: UserType = {\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) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      params: {userId},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error,context));\n};\n\nexport const getDisplayName = (user: UserType): string => {\n  const {first, last, name = '', username = ''} = user;\n  const fullname = ([first, last]).join(' ').trim();\n\n  if(name) {\n    return name;\n  } else if(fullname !== '') {\n    return fullname;\n  } else if(username) {\n    return username;\n  }\n\n  return 'Unknown';\n};\n\nexport const getSessionUser = (context: ApiContext): Promise<UserType> => {\n  const action = 'getSessionUser';\n  console.log('getSessionUser', {action, context});\n  const {database, fields, session: {userId: sessionId, username}} = context;\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n  const formatSessionId = parseArangoId(`users/${sessionId}`);\n\n  const aqlQuery: string = `LET u = DOCUMENT(\"${formatSessionId}\")\n  ${selectQueries.join('\\n')}\n  RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQuery)\n    .then((cursor) => cursor.next() as unknown as UserType)\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        params: {userId: sessionId, username},\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return null;\n    });\n};\n\nexport const getUser = (context: ApiContext, user: UserType): Promise<UserType> => {\n  const action = 'getUser';\n  const {id, userId, username} = parseUser(user);\n  const {database, fields} = context;\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n  let aqlQuery: string;\n\n  console.log({id, userId, username});\n  if(id) {\n    aqlQuery = `LET u = DOCUMENT(\"${id}\")\n    ${selectQueries.join('\\n')}\n    FILTER u.userAccess > 0\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n  } else if(username) {\n    aqlQuery = `FOR u IN users\n    FILTER u.username == \"${username}\"\n    ${selectQueries.join('\\n')}\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n  }\n\n  return database.query(aqlQuery)\n    .then((cursor) => cursor.next())\n    .then((user) => user)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      params: {id, userId, username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getUsers = (context: ApiContext, options?: UserOptions): Promise<UserType[]> => {\n  const action = '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(username) {\n    filterBy.push(`CONTAINS(u.username, \"${parseUsername(username)}\")`);\n  }\n\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) => cursor.all() as unknown as UserType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as UserType[];\n    });\n};\n\nexport const getUsersByReactions = (\n  context: ApiContext,\n  {reactions = [], username},\n  options?: UserOptions\n): Promise<UserType[]> => {\n  const action = '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  const formatSessionId: string = `users/${sessionId}`;\n  const formatUsername: string = parseUsername(username);\n  const filterBy: string[] = [\n    'u.userAccess > 0',\n    `POSITION(${JSON.stringify(formatReactions)}, LOWER(r.name))`\n  ];\n\n  if(username) {\n    filterBy.push(`CONTAINS(u.username, \"${formatUsername}\")`);\n  }\n\n  const aqlQuery: string = `FOR u, r IN OUTBOUND \"${formatSessionId}\" hasReaction\n    OPTIONS {vertexCollections: \"users\"}\n    ${selectQueries.join('\\n')}\n    FILTER ${filterBy.join(' && ')}\n    ${limit.aql}\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQuery)\n    .then((cursor) => cursor.all() as unknown as UserType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as UserType[];\n    });\n};\n\nexport const getUsersByTags = (\n  context: ApiContext,\n  {tags, username},\n  options?: UserOptions\n): Promise<UserType[]> => {\n  const action = 'getUsersByTags';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const formatTags: string[] =  tags?.reduce((list: string[], tagName: string) => {\n    if(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  const formatUsername: string = parseUsername(username);\n  const filterBy: string[] = [\n    `u._key != \"${sessionId}\"`,\n    'u.userAccess > 0'\n  ];\n\n  if(username) {\n    filterBy.push(`CONTAINS(u.username, \"${formatUsername}\")`);\n  }\n\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    FILTER ${filterBy.join(' && ')}\n    ${limit.aql}\n    RETURN DISTINCT MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQuery)\n    .then((cursor) => cursor.all() as unknown as UserType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as UserType[];\n    });\n};\n\nexport const getUsersByLatest = (context: ApiContext, {username}, options?: UserOptions): Promise<UserType[]> => {\n  const action = 'getUsersByLatest';\n  const {database, fields, session: {userId}} = context;\n  const {limit} = parseUserOptions(options);\n  const filter = [\n    'u._id != session._id',\n    'u.userAccess > 0'\n  ];\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  if(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) => cursor.all() as unknown as UserType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as UserType[];\n    });\n};\n\nexport const getUsersByConnection = (\n  context: ApiContext,\n  {userId}: UserType,\n  options?: UserOptions\n): Promise<UserType[]> => {\n  const action = 'getUsersByConnection';\n  const {database, fields} = context;\n  const {limit, username} = parseUserOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n  const formatUserId: string = parseArangoId(`users/${userId}`);\n  const filterBy: string[] = [\n    'u.userAccess > 0'\n  ];\n\n  if(username) {\n    filterBy.push(`CONTAINS(u.username, \"${parseUsername(username)}\")`);\n  }\n\n  const aqlQuery: string = `FOR cu IN users\n    LET session = DOCUMENT(\"${formatUserId}\")\n    FOR u, connection IN OUTBOUND cu hasConnection\n    OPTIONS {bfs: true, uniqueVertices: \"global\", vertexCollections: \"users\"}\n    ${selectQueries.join('\\n')}\n    FILTER ${filterBy.join(' && ')}\n    ${limit.aql}\n    RETURN DISTINCT MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQuery)\n    .then((cursor) => cursor.all() as unknown as UserType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as UserType[];\n    });\n};\n\nexport const refreshSession = ({expires, token}): SessionToken | SessionError => {\n  try {\n    const {userId, username, userAccess} = getSession(token);\n    return createToken(userId, username, userAccess, expires);\n  } catch(error) {\n    throw error; // Re-throw the error from getSession\n  }\n};\n\nexport const signIn = async (context: ApiContext, args): Promise<SessionToken> => {\n  const action = 'signIn';\n  const {database} = context;\n  const {email, expires, password, phone, username} = args;\n  const formatEmail: string = parseEmail(email);\n  const formatUsername: string = parseUsername(username);\n  const formatPassword: string = parsePassword(password);\n  const formatPhone: string = parsePhone(phone);\n  const formatExpires: number = parseNum(expires) || 15;\n\n  if((!formatUsername && !formatEmail && !formatPhone) || !formatPassword) {\n    logException({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.INVALID_ARGUMENTS\n    }, context);\n    return null;\n  }\n\n  const filters: string[] = [];\n\n  if(formatEmail) {\n    filters.push(`u.email == \"${formatEmail}\"`);\n  }\n\n  if(formatPhone) {\n    filters.push(`u.phone == ${formatPhone}`);\n  }\n\n  if(formatUsername) {\n    filters.push(`u.username == \"${formatUsername}\"`);\n  }\n\n  const checkQuery: string = `FOR u IN users\n    FILTER ${filters.join(' || ')}\n    LIMIT 1\n    RETURN u`;\n\n  let checkUser: UserType;\n\n  try {\n    checkUser = await database.query(checkQuery).then((cursor) => cursor.next());\n  } catch(error) {\n    logError({\n      action,\n      category: eventCategory,\n      params: {username: formatUsername},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n\n    return null;\n  }\n\n  if(!checkUser) {\n    logException({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.INVALID_AUTHENTICATION\n    }, context);\n\n    return null;\n  }\n\n  const {_key: userId, password: validPassword, salt, userAccess} = checkUser;\n  const authPassword: string = createPassword(formatPassword, salt);\n\n  if(validPassword !== authPassword) {\n    logException({\n      action,\n      category: eventCategory,\n      params: {userAccess, userId, username},\n      value: ErrorTypes.INVALID_AUTHENTICATION\n    }, context);\n\n    return null;\n  }\n\n  try {\n    console.log({formatExpires, userAccess, userId, username});\n    const token = createToken(userId, username, userAccess, formatExpires);\n    console.log({token});\n\n    return token;\n  } catch(error) {\n    logError({\n      action,\n      category: eventCategory,\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n\n    return null;\n  }\n};\n\nexport const signOut = async (context: ApiContext): Promise<boolean> => {\n  const action = '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).then((cursor) => cursor.next());\n  } catch(error) {\n    await logError({\n      action,\n      category: eventCategory,\n      params: {userId: sessionId, username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n  }\n\n  return true;\n};\n\nexport const getActiveUserCount = (context: ApiContext): Promise<number> => {\n  const action = '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      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getUserByToken = (context: ApiContext, token: string): Promise<UserType> => {\n  const action = 'getUserByToken';\n  const {database} = context;\n  const {userId} = getSession(token);\n  const userDocId = getDocId('users', {userId});\n  const aqlQuery: AqlQuery = aql`LET u = DOCUMENT(\"${userDocId}\") RETURN u`;\n\n  return database.query(aqlQuery)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      params: {userId},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};"],
  "mappings": "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",
  "names": ["UserAccess", "email", "phone", "password", "Stripe", "user"]
}

83
+ RETURN LENGTH(docs)`;return g(s).query(i).then(n=>n.next()).catch(n=>I({action:o,category:E,value:d.DATABASE_ERROR},n,e))},Be=(e,o)=>{const s="getUserByToken",{databaseName:i}=e,{userId:n}=_(o),t=j("users",{userId:n}),u=U`LET u = DOCUMENT("${t}") RETURN u`;return g(i).query(u).then(r=>r.next()).catch(r=>I({action:s,category:E,params:{userId:n},value:d.DATABASE_ERROR},r,e))};export{ae as UserAccess,Ue as addUser,$e as confirmCode,B as createToken,Se as deactivateUser,Ce as deleteUser,he as forgotPassword,_e as getActiveUserCount,qe as getDisplayName,De as getSessionUser,be as getUser,Be as getUserByToken,$ as getUserOptional,we as getUsers,Le as getUsersByConnection,ve as getUsersByLatest,Pe as getUsersByReactions,xe as getUsersByTags,b as parseUserOptions,je as refreshSession,Oe as resetPassword,Me as signIn,Qe as signOut,Ne as updateUser};
84
+ //# 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  parseArangoId,\n  parseChar,\n  parseEmail,\n  parseNum,\n  parsePassword,\n  parsePhone,\n  parseUsername\n} from '@nlabs/utils';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {DateTime} from 'luxon';\nimport Stripe from 'stripe';\n\nimport {parseUser} from '../adapters/userAdapter';\nimport {Config} from '../config';\nimport {sendEmail} from './email';\nimport {sendSms} from './sms';\nimport {ErrorTypes, SessionError} from '../types/error.types';\nimport {logError, logException} from '../utils/analyticsUtils';\nimport {getDocId, getLimit, selectReactionCountByType, useDb} from '../utils/arangodbUtils';\nimport {detectLanguage} from '../utils/languageDetection';\nimport {getSession, SessionToken, setSession} from '../utils/sessionUtils';\n\nimport type {ApiContext} from '../types/auth.types';\nimport type {UserInput, UserType} from '../types/users.types';\nimport type {EdgeCollection} from 'arangojs/collections';\n\n\nconst eventCategory = 'users';\nconst STRIPE_API_VERSION = '2025-05-28.basil';\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  expiresInMinutes: number = 15\n): SessionToken => {\n  const now: DateTime = DateTime.local();\n  const sessionExpires: DateTime = now.plus({minutes: expiresInMinutes});\n  const iat: number = Math.floor(now.toSeconds());\n  const exp: number = Math.floor(sessionExpires.toSeconds());\n  const token = setSession({\n    exp,\n    iat,\n    userAccess,\n    userId,\n    username\n  });\n\n  return {\n    expires: sessionExpires.toMillis(),\n    issued: now.toMillis(),\n    token,\n    userId,\n    username\n  };\n};\n\ninterface SelectAccumulator {\n  objects: string[];\n  queries: string[];\n}\n\nexport const getUserOptional = (fields: string[] = []): SelectAccumulator =>\n  fields.reduce((selects: SelectAccumulator, 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  const limit = getLimit(from, to);\n\n  return {\n    ...options,\n    limit\n  };\n};\n\nexport const addUser = async (context: ApiContext, user: UserInput): Promise<UserType> => {\n  const action = 'addUser';\n  const {databaseName, languageContext} = context;\n  const {email, password, phone, username, userId, _key, _id, ...insertUser} = parseUser(user);\n  const formatPassword: string = parsePassword(password);\n  const hasPassword = !!formatPassword;\n  const hasUsername = !!username || !!phone || !!email;\n\n  if(!hasPassword || !hasUsername) {\n    return logException({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.INVALID_ARGUMENTS\n    }, context);\n  }\n\n  const hashId = username || phone || email;\n  const salt: string = createHash(`${hashId}${formatPassword}`, null);\n  const encryptedPassword = createPassword(formatPassword, salt);\n  const filters: string[] = [];\n\n  if(username) {\n    filters.push(`u.username == \"${username}\"`);\n  }\n\n  if(email) {\n    filters.push(`u.email == \"${email}\"`);\n  }\n\n  if(phone) {\n    filters.push(`u.phone == ${phone}`);\n  }\n\n  const checkQuery: string = `FOR u IN users\n    FILTER ${filters.join(' || ')}\n    LIMIT 1\n    RETURN u`;\n\n  try {\n    const existingUsers = await useDb(databaseName).query(checkQuery).then((cursor) => cursor.all());\n\n    if(existingUsers.length) {\n      return logException({\n        action,\n        category: eventCategory,\n        params: {\n          email,\n          phone,\n          username\n        },\n        value: ErrorTypes.EXISTING_ITEM\n      }, context);\n    }\n  } catch(error) {\n    return logError({\n      action,\n      category: eventCategory,\n      params: {email, phone, username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n  }\n\n  const phoneCountryCode = phone?.replace(/\\D/g, '').substring(0, 3);\n  const locale = detectLanguage({\n    ...languageContext,\n    phoneCountryCode,\n    userPreference: insertUser.locale || 'en'\n  });\n  const verifiedEmailCode: number = Math.floor(100000 + (Math.random() * 900000));\n  const verifiedSmsCode: number = Math.floor(100000 + (Math.random() * 900000));\n\n  const insert: UserType = {\n    ...insertUser,\n    _key: createHash(username, null),\n    added: Date.now(),\n    email,\n    locale,\n    modified: Date.now(),\n    password: encryptedPassword,\n    phone,\n    salt,\n    userAccess: 1,\n    username,\n    verifiedEmail: false,\n    verifiedEmailCode,\n    verifiedPhone: false,\n    verifiedSmsCode\n  };\n\n  const insertQuery: AqlQuery = aql`INSERT ${insert} IN users RETURN NEW`;\n\n  return await useDb(databaseName).query(insertQuery)\n    .then((cursor) => cursor.next())\n    .catch((error) => logError({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const updateUser = async (context: ApiContext, user: UserType): Promise<UserType> => {\n  const action = 'updateUser';\n  const {databaseName} = context;\n  const updatedUser = parseUser(user);\n  const {_key, _id, tags = [], ...update} = updatedUser;\n  const {id} = updatedUser;\n\n  const userQuery: AqlQuery = aql`LET u = DOCUMENT(${id})\n    UPDATE u WITH ${update} IN users\n    RETURN NEW`;\n  try {\n    const database = useDb(databaseName);\n    const updatedUser = await database.query(userQuery).then((cursor) => cursor.next());\n    const tagCollection: EdgeCollection = database.collection('isTagged');\n\n    await Promise.all(tags.map(({id: tagDocId, name}) => {\n      const tagQuery: AqlQuery = aql`FOR it IN isTagged\n        FILTER it._from == ${tagDocId} && it._to == ${id} && it.name == ${name}\n        LIMIT 1\n        RETURN it`;\n\n      return database.query(tagQuery)\n        .then((cursor) => cursor.next())\n        .then((tagEdge) => {\n          console.log({tagEdge});\n          if(!!tagEdge) {\n            return tagEdge;\n          }\n\n          const edge = {\n            _from: tagDocId,\n            _key: createHash(`isTagged-${tagDocId}-${id}`),\n            _to: id,\n            added: Date.now(),\n            name\n          };\n\n          return tagCollection.save(edge, {returnNew: true}).then(() => edge);\n        });\n    }));\n\n    return updatedUser;\n  } catch(error) {\n    return logError({\n      action,\n      category: eventCategory,\n      params: {user},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n  }\n};\n\nexport const forgotPassword = async (context: ApiContext, {email, phone, username}): Promise<boolean> => {\n  const action = 'forgotPassword';\n  const {databaseName} = context;\n  const aqlQuery: AqlQuery = aql`FOR u IN users\n    FILTER u.email == ${email} || u.phone == ${phone} || u.username == ${username}\n    LIMIT 1\n    RETURN u`;\n\n  try {\n    return await useDb(databaseName).query(aqlQuery)\n      .then(async (cursor) => {\n        const user = cursor.next();\n\n        if(user) {\n          const {email, phone, verifiedEmail, verifiedPhone} = user as UserType;\n          const codeExpires = 1000 * 60 * 15; // 15 minutes\n          const code = Math.floor(100000 + (Math.random() * 900000));\n          const userDocId = getDocId('users', user);\n          let update;\n\n          if(email && verifiedEmail) {\n            sendEmail({\n              context,\n              text: `Your code is ${code}`\n            });\n            update = {verifiedEmailCode: code, verifiedEmailExpires: codeExpires};\n          }\n\n          if(phone && verifiedPhone) {\n            sendSms({\n              context,\n              text: `Your code is ${code}`\n            });\n            update = {verifiedSmsCode: code, verifiedPhoneExpires: codeExpires};\n          }\n\n          if(update.verifiedEmailCode || update.verifiedSmsCode) {\n            const updateQuery: AqlQuery = aql`UPDATE ${userDocId} WITH ${update} IN users`;\n\n            await useDb(databaseName).query(updateQuery);\n\n            return true;\n          }\n\n          return false;\n        }\n\n        return false;\n      });\n  } catch(error) {\n    logError({\n      action,\n      category: eventCategory,\n      params: {email, phone, username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n\n    return false;\n  }\n};\n\nexport const resetPassword = async (\n  context: ApiContext,\n  {\n    code,\n    password,\n    type,\n    username\n  }: {\n    code: number,\n    password: string,\n    type: 'phone' | 'email',\n    username: string\n  }\n): Promise<boolean> => {\n  const action = 'resetPassword';\n  const {databaseName} = context;\n  const formatPassword: string = parsePassword(password);\n  const aqlQuery: AqlQuery = aql`FOR u IN users\n    FILTER u.username == ${username}\n    LIMIT 1\n    RETURN u`;\n\n  try {\n    return await useDb(databaseName).query(aqlQuery)\n      .then(async (cursor) => {\n        const user = cursor.next();\n\n        if(user) {\n          const {\n            _id: userDocId,\n            salt,\n            verifiedEmailCode,\n            verifiedEmailExpires,\n            verifiedSmsCode,\n            verifiedPhoneExpires\n          } = user as UserType;\n          const now = Date.now();\n          let update;\n\n          switch(type) {\n            case 'email':\n              if(code === verifiedEmailCode && verifiedEmailExpires > now) {\n                const password: string = createPassword(formatPassword, salt);\n                update = {password};\n              }\n              break;\n            case 'phone':\n              if(code === verifiedSmsCode && verifiedPhoneExpires > now) {\n                const password: string = createPassword(formatPassword, salt);\n                update = {password};\n              }\n              break;\n            default:\n              return false;\n          }\n\n          if(update) {\n            const updateQuery: AqlQuery = aql`UPDATE ${userDocId} WITH ${update} IN users`;\n\n            await useDb(databaseName).query(updateQuery);\n\n            return true;\n          }\n        }\n\n        return false;\n      });\n  } catch(error) {\n    logError({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n\n    return false;\n  }\n};\n\nexport const confirmCode = async (\n  context: ApiContext,\n  {\n    code,\n    type\n  }: {\n    code: number,\n    type: 'phone' | 'email'\n  }\n): Promise<boolean> => {\n  const action = 'confirmCode';\n  const {databaseName, session: {userId: sessionId}} = context;\n  const userDocId = getDocId('users', {userId: sessionId});\n  const aqlQuery: AqlQuery = aql`LET u = DOCUMENT(${userDocId}) RETURN u`;\n\n  try {\n    return await useDb(databaseName).query(aqlQuery)\n      .then((cursor) => cursor.next())\n      .then(({verifiedEmailCode, verifiedSmsCode}: UserType) => {\n        switch(type) {\n          case 'email':\n            return code === verifiedEmailCode;\n          case 'phone':\n            return code === verifiedSmsCode;\n          default:\n            return false;\n        }\n      });\n  } catch(error) {\n    logError({\n      action,\n      category: eventCategory,\n      params: {code, type},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n\n    return false;\n  }\n};\n\nexport const deleteUser = (context: ApiContext, user: UserType): Promise<UserType> => {\n  const action = 'deleteUser';\n  const {databaseName} = context;\n  const {id} = parseUser(user);\n\n  const aqlQuery: AqlQuery = aql`LET u = DOCUMENT(${id})\n    REMOVE u IN users\n    RETURN OLD`;\n\n  const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion: STRIPE_API_VERSION, typescript: true});\n\n  return useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.next())\n    .then((deletedUser) => stripeClient.customers.del(deletedUser?.stripeCustomerId)\n      .then(() => stripeClient.accounts.del(deletedUser?.stripeAccountId))\n      .then(() => deletedUser))\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      params: {id},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error,context));\n};\n\nexport const deactivateUser = (context: ApiContext, user: UserType): Promise<UserType> => {\n  const action = 'delete';\n  const {databaseName} = context;\n  const {id} = parseUser(user);\n  const updated: UserType = {\n    userAccess: 0\n  };\n  const aqlQuery: AqlQuery = aql`UPDATE ${id} WITH ${updated} IN users LIMIT 1 RETURN NEW`;\n\n  return useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      params: {id},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error,context));\n};\n\nexport const getDisplayName = (user: UserType): string => {\n  const {first, last, name = '', username = ''} = user;\n  const fullname = ([first, last]).join(' ').trim();\n\n  if(name) {\n    return name;\n  } else if(fullname !== '') {\n    return fullname;\n  } else if(username) {\n    return username;\n  }\n\n  return 'Unknown';\n};\n\nexport const getSessionUser = (context: ApiContext): Promise<UserType> => {\n  const action = 'getSessionUser';\n  console.log('getSessionUser', {action, context});\n  const {databaseName, fields, session: {userId: sessionId, username}} = context;\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n  const formatSessionId = parseArangoId(`users/${sessionId}`);\n\n  const aqlQuery: string = `LET u = DOCUMENT(\"${formatSessionId}\")\n  ${selectQueries.join('\\n')}\n  RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.next() as unknown as UserType)\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        params: {userId: sessionId, username},\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return null;\n    });\n};\n\nexport const getUser = (context: ApiContext, user: UserType): Promise<UserType> => {\n  const action = 'getUser';\n  const {email, id, phone, userId, username} = parseUser(user);\n  const {databaseName, fields} = context;\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n  let aqlQuery: string;\n\n  if(id) {\n    aqlQuery = `LET u = DOCUMENT(\"${id}\")\n    ${selectQueries.join('\\n')}\n    FILTER u.userAccess > 0\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n  } else if(username) {\n    aqlQuery = `FOR u IN users\n    FILTER u.username == \"${username}\"\n    ${selectQueries.join('\\n')}\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n  } else if(email) {\n    aqlQuery = `FOR u IN users\n    FILTER u.email == \"${email}\"\n    ${selectQueries.join('\\n')}\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n  } else if(phone) {\n    aqlQuery = `FOR u IN users\n    FILTER u.phone == \"${phone}\"\n    ${selectQueries.join('\\n')}\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n  }\n\n  return useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.next())\n    .then((user) => user)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      params: {id, userId, username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getUsers = (context: ApiContext, options?: UserOptions): Promise<UserType[]> => {\n  const action = 'getUserList';\n  const {databaseName, 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(username) {\n    filterBy.push(`CONTAINS(u.username, \"${parseUsername(username)}\")`);\n  }\n\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 useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.all() as unknown as UserType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as UserType[];\n    });\n};\n\nexport const getUsersByReactions = (\n  context: ApiContext,\n  {reactions = [], username}: {reactions?: string[], username?: string},\n  options?: UserOptions\n): Promise<UserType[]> => {\n  const action = 'getUsersByReactions';\n  const {databaseName, 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  const formatSessionId: string = `users/${sessionId}`;\n  const formatUsername: string = parseUsername(username);\n  const filterBy: string[] = [\n    'u.userAccess > 0',\n    `POSITION(${JSON.stringify(formatReactions)}, LOWER(r.name))`\n  ];\n\n  if(username) {\n    filterBy.push(`CONTAINS(u.username, \"${formatUsername}\")`);\n  }\n\n  const aqlQuery: string = `FOR u, r IN OUTBOUND \"${formatSessionId}\" hasReaction\n    OPTIONS {vertexCollections: \"users\"}\n    ${selectQueries.join('\\n')}\n    FILTER ${filterBy.join(' && ')}\n    ${limit.aql}\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.all() as unknown as UserType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as UserType[];\n    });\n};\n\nexport const getUsersByTags = (\n  context: ApiContext,\n  {tags, username},\n  options?: UserOptions\n): Promise<UserType[]> => {\n  const action = 'getUsersByTags';\n  const {databaseName, fields, session: {userId: sessionId}} = context;\n  const formatTags: string[] =  tags?.reduce((list: string[], tagName: string) => {\n    if(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  const formatUsername: string = parseUsername(username);\n  const filterBy: string[] = [\n    `u._key != \"${sessionId}\"`,\n    'u.userAccess > 0'\n  ];\n\n  if(username) {\n    filterBy.push(`CONTAINS(u.username, \"${formatUsername}\")`);\n  }\n\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    FILTER ${filterBy.join(' && ')}\n    ${limit.aql}\n    RETURN DISTINCT MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.all() as unknown as UserType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as UserType[];\n    });\n};\n\nexport const getUsersByLatest = (context: ApiContext, {username}, options?: UserOptions): Promise<UserType[]> => {\n  const action = 'getUsersByLatest';\n  const {databaseName, fields, session: {userId}} = context;\n  const {limit} = parseUserOptions(options);\n  const filter = [\n    'u._id != session._id',\n    'u.userAccess > 0'\n  ];\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  if(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 useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.all() as unknown as UserType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as UserType[];\n    });\n};\n\nexport const getUsersByConnection = (\n  context: ApiContext,\n  {userId}: UserType,\n  options?: UserOptions\n): Promise<UserType[]> => {\n  const action = 'getUsersByConnection';\n  const {databaseName, fields} = context;\n  const {limit, username} = parseUserOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n  const formatUserId: string = parseArangoId(`users/${userId}`);\n  const filterBy: string[] = [\n    'u.userAccess > 0'\n  ];\n\n  if(username) {\n    filterBy.push(`CONTAINS(u.username, \"${parseUsername(username)}\")`);\n  }\n\n  const aqlQuery: string = `FOR cu IN users\n    LET session = DOCUMENT(\"${formatUserId}\")\n    FOR u, connection IN OUTBOUND cu hasConnection\n    OPTIONS {bfs: true, uniqueVertices: \"global\", vertexCollections: \"users\"}\n    ${selectQueries.join('\\n')}\n    FILTER ${filterBy.join(' && ')}\n    ${limit.aql}\n    RETURN DISTINCT MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.all() as unknown as UserType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as UserType[];\n    });\n};\n\nexport const refreshSession = ({expires, token}): SessionToken | SessionError => {\n  try {\n    const {userId, username, userAccess} = getSession(token);\n    return createToken(userId, username, userAccess, expires);\n  } catch(error) {\n    throw error; // Re-throw the error from getSession\n  }\n};\n\nexport const signIn = async (context: ApiContext, args): Promise<SessionToken> => {\n  const action = 'signIn';\n  const {databaseName} = context;\n  const {email, expires, password, phone, username} = args;\n  const formatEmail: string = parseEmail(email);\n  const formatUsername: string = parseUsername(username);\n  const formatPassword: string = parsePassword(password);\n  const formatPhone: string = parsePhone(phone);\n  const formatExpires: number = parseNum(expires) || 15;\n\n  if((!formatUsername && !formatEmail && !formatPhone) || !formatPassword) {\n    logException({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.INVALID_ARGUMENTS\n    }, context);\n    return null;\n  }\n\n  const filters: string[] = [];\n\n  if(formatEmail) {\n    filters.push(`u.email == \"${formatEmail}\"`);\n  }\n\n  if(formatPhone) {\n    filters.push(`u.phone == ${formatPhone}`);\n  }\n\n  if(formatUsername) {\n    filters.push(`u.username == \"${formatUsername}\"`);\n  }\n\n  const checkQuery: string = `FOR u IN users\n    FILTER ${filters.join(' || ')}\n    LIMIT 1\n    RETURN u`;\n\n  let checkUser: UserType;\n\n  try {\n    checkUser = await useDb(databaseName).query(checkQuery).then((cursor) => cursor.next());\n  } catch(error) {\n    logError({\n      action,\n      category: eventCategory,\n      params: {username: formatUsername},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n\n    return null;\n  }\n\n  if(!checkUser) {\n    logException({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.INVALID_AUTHENTICATION\n    }, context);\n\n    return null;\n  }\n\n  const {_key: userId, password: validPassword, salt, userAccess} = checkUser;\n  const authPassword: string = createPassword(formatPassword, salt);\n\n  if(validPassword !== authPassword) {\n    logException({\n      action,\n      category: eventCategory,\n      params: {userAccess, userId, username},\n      value: ErrorTypes.INVALID_AUTHENTICATION\n    }, context);\n\n    return null;\n  }\n\n  try {\n    console.log({formatExpires, userAccess, userId, username});\n    const token = createToken(userId, username, userAccess, formatExpires);\n    console.log({token});\n\n    return token;\n  } catch(error) {\n    logError({\n      action,\n      category: eventCategory,\n      params: {userId, username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n\n    return null;\n  }\n};\n\nexport const signOut = async (context: ApiContext): Promise<boolean> => {\n  const action = 'signOut';\n  const {databaseName, 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 useDb(databaseName).query(sessionQuery).then((cursor) => cursor.next());\n  } catch(error) {\n    await logError({\n      action,\n      category: eventCategory,\n      params: {userId: sessionId, username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n  }\n\n  return true;\n};\n\nexport const getActiveUserCount = (context: ApiContext): Promise<number> => {\n  const action = 'getActiveUserCount';\n  const {databaseName} = 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 useDb(databaseName).query(countQuery)\n    .then((cursor) => cursor.next())\n    .catch((error) => logError({\n      action,\n      category: eventCategory,\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getUserByToken = (context: ApiContext, token: string): Promise<UserType> => {\n  const action = 'getUserByToken';\n  const {databaseName} = context;\n  const {userId} = getSession(token);\n  const userDocId = getDocId('users', {userId});\n  const aqlQuery: AqlQuery = aql`LET u = DOCUMENT(\"${userDocId}\") RETURN u`;\n\n  return useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      params: {userId},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};"],
  "mappings": "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",
  "names": ["createHash", "createPassword", "parseArangoId", "parseChar", "parseEmail", "parseNum", "parsePassword", "parsePhone", "parseUsername", "aql", "DateTime", "Stripe", "parseUser", "Config", "sendEmail", "sendSms", "ErrorTypes", "logError", "logException", "getDocId", "getLimit", "selectReactionCountByType", "useDb", "detectLanguage", "getSession", "setSession", "eventCategory", "STRIPE_API_VERSION", "UserAccess", "createToken", "userId", "username", "userAccess", "expiresInMinutes", "now", "sessionExpires", "iat", "exp", "token", "getUserOptional", "fields", "selects", "field", "parseUserOptions", "options", "from", "to", "limit", "addUser", "context", "user", "action", "databaseName", "languageContext", "email", "password", "phone", "_key", "_id", "insertUser", "formatPassword", "salt", "encryptedPassword", "filters", "checkQuery", "cursor", "error", "phoneCountryCode", "locale", "verifiedEmailCode", "verifiedSmsCode", "insert", "insertQuery", "updateUser", "updatedUser", "tags", "update", "id", "userQuery", "database", "tagCollection", "tagDocId", "name", "tagQuery", "tagEdge", "edge", "forgotPassword", "aqlQuery", "verifiedEmail", "verifiedPhone", "codeExpires", "code", "userDocId", "updateQuery", "resetPassword", "type", "verifiedEmailExpires", "verifiedPhoneExpires", "confirmCode", "sessionId", "deleteUser", "stripeClient", "deletedUser", "deactivateUser", "getDisplayName", "first", "last", "fullname", "getSessionUser", "selectObjects", "selectQueries", "getUser", "getUsers", "filterBy", "getUsersByReactions", "reactions", "formatReactions", "reactionName", "formatSessionId", "formatUsername", "getUsersByTags", "formatTags", "list", "tagName", "getUsersByLatest", "filter", "getUsersByConnection", "formatUserId", "refreshSession", "expires", "signIn", "args", "formatEmail", "formatPhone", "formatExpires", "checkUser", "validPassword", "authPassword", "signOut", "sessionQuery", "getActiveUserCount", "countQuery", "getUserByToken"]
}
