@nlabs/reaktor 0.9.0 → 0.10.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (357) hide show
  1. package/README.md +9 -0
  2. package/coverage/index.html +92 -47
  3. package/dist/actions/apps.js +242 -0
  4. package/dist/actions/connections.js +90 -0
  5. package/dist/actions/conversations.js +350 -0
  6. package/dist/actions/dynamodb.js +150 -0
  7. package/dist/actions/email.js +152 -0
  8. package/dist/actions/files.js +283 -0
  9. package/dist/actions/groups.js +292 -0
  10. package/dist/actions/images.js +735 -0
  11. package/dist/actions/index.js +66 -0
  12. package/dist/actions/ios.js +164 -0
  13. package/dist/actions/locations.js +122 -0
  14. package/dist/actions/messages.js +208 -0
  15. package/dist/actions/notifications.js +59 -0
  16. package/dist/actions/payments.js +497 -0
  17. package/dist/actions/personas.js +110 -0
  18. package/dist/actions/posts.js +595 -0
  19. package/dist/actions/reactions.js +322 -0
  20. package/dist/actions/s3.js +133 -0
  21. package/dist/actions/search.js +90 -0
  22. package/dist/actions/sms.js +108 -0
  23. package/dist/actions/statistics.js +62 -0
  24. package/dist/actions/subscription.js +220 -0
  25. package/dist/actions/tags.js +292 -0
  26. package/dist/actions/users.js +784 -0
  27. package/dist/actions/websockets.js +174 -0
  28. package/dist/adapters/arangoAdapter.js +46 -0
  29. package/dist/adapters/fileAdapter.js +76 -0
  30. package/dist/adapters/imageAdapter.js +40 -0
  31. package/dist/adapters/messageAdapter.js +49 -0
  32. package/dist/adapters/postAdapter.js +70 -0
  33. package/dist/adapters/reaktorAdapter.js +44 -0
  34. package/dist/adapters/tagAdapter.js +50 -0
  35. package/dist/adapters/userAdapter.js +115 -0
  36. package/dist/config.js +125 -0
  37. package/dist/index.js +66 -0
  38. package/dist/lambdas/actions/websockets.js +132 -0
  39. package/dist/lambdas/authorizer.js +67 -0
  40. package/dist/lambdas/connection.js +91 -0
  41. package/dist/lambdas/utils/message.js +42 -0
  42. package/dist/lambdas/utils/websocket.js +105 -0
  43. package/dist/mocks/conversation.js +35 -0
  44. package/dist/mocks/file.js +38 -0
  45. package/dist/mocks/group.js +47 -0
  46. package/dist/mocks/image.js +44 -0
  47. package/dist/mocks/nlabs.png +0 -0
  48. package/dist/mocks/post.js +55 -0
  49. package/dist/mocks/tag.js +37 -0
  50. package/dist/mocks/user.js +88 -0
  51. package/dist/mutations/index.js +26 -0
  52. package/dist/mutations/locations.js +44 -0
  53. package/dist/mutations/messages.js +86 -0
  54. package/dist/mutations/personas.js +100 -0
  55. package/dist/mutations/posts.js +53 -0
  56. package/dist/mutations/reactions.js +51 -0
  57. package/dist/mutations/statistics.js +39 -0
  58. package/dist/mutations/subscriptions.js +56 -0
  59. package/dist/mutations/tags.js +120 -0
  60. package/dist/mutations/users.js +116 -0
  61. package/dist/objectTypes/app.js +173 -0
  62. package/dist/objectTypes/bankAccount.js +76 -0
  63. package/dist/objectTypes/connection.js +48 -0
  64. package/dist/objectTypes/conversation.js +77 -0
  65. package/dist/objectTypes/creditCard.js +86 -0
  66. package/dist/objectTypes/document.js +46 -0
  67. package/dist/objectTypes/error.js +46 -0
  68. package/dist/objectTypes/external.js +74 -0
  69. package/dist/objectTypes/file.js +100 -0
  70. package/dist/objectTypes/filter.js +43 -0
  71. package/dist/objectTypes/group.js +123 -0
  72. package/dist/objectTypes/iapSubscription.js +40 -0
  73. package/dist/objectTypes/image.js +129 -0
  74. package/dist/objectTypes/index.js +68 -0
  75. package/dist/objectTypes/location.js +109 -0
  76. package/dist/objectTypes/message.js +96 -0
  77. package/dist/objectTypes/passcode.js +42 -0
  78. package/dist/objectTypes/persona.js +87 -0
  79. package/dist/objectTypes/plan.js +95 -0
  80. package/dist/objectTypes/post.js +125 -0
  81. package/dist/objectTypes/reaction.js +61 -0
  82. package/dist/objectTypes/relation.js +49 -0
  83. package/dist/objectTypes/search.js +72 -0
  84. package/dist/objectTypes/statistics.js +39 -0
  85. package/dist/objectTypes/subscription.js +117 -0
  86. package/dist/objectTypes/tag.js +65 -0
  87. package/dist/objectTypes/user.js +144 -0
  88. package/dist/queries/index.js +33 -0
  89. package/dist/queries/locations.js +45 -0
  90. package/dist/queries/messages.js +52 -0
  91. package/dist/queries/posts.js +154 -0
  92. package/dist/queries/reactions.js +56 -0
  93. package/dist/queries/statistics.js +39 -0
  94. package/dist/queries/subscriptions.js +44 -0
  95. package/dist/queries/tags.js +75 -0
  96. package/dist/queries/users.js +64 -0
  97. package/dist/templates/email/layout.js +302 -0
  98. package/dist/templates/email/passwordForgot.js +38 -0
  99. package/dist/templates/email/passwordRecovery.js +35 -0
  100. package/dist/templates/email/verifyEmail.js +38 -0
  101. package/dist/templates/email/welcome.js +38 -0
  102. package/dist/templates/sms/passwordForgot.js +24 -0
  103. package/dist/templates/sms/passwordRecovery.js +24 -0
  104. package/dist/templates/sms/verifyEmail.js +24 -0
  105. package/dist/templates/sms/verifyPhone.js +24 -0
  106. package/dist/templates/sms/welcome.js +24 -0
  107. package/dist/types/apps.js +32 -0
  108. package/{lib → dist}/types/arangodb.js +1 -1
  109. package/{lib → dist}/types/auth.js +1 -1
  110. package/{lib → dist}/types/connections.js +1 -1
  111. package/dist/types/conversations.js +16 -0
  112. package/{lib → dist}/types/email.js +1 -1
  113. package/dist/types/files.js +16 -0
  114. package/dist/types/google.js +16 -0
  115. package/{lib → dist}/types/groups.js +1 -1
  116. package/dist/types/images.js +16 -0
  117. package/dist/types/index.js +60 -0
  118. package/{lib → dist}/types/locations.js +1 -1
  119. package/{lib → dist}/types/messages.js +1 -1
  120. package/{lib → dist}/types/notifications.js +1 -1
  121. package/dist/types/payments.js +16 -0
  122. package/dist/types/personas.js +16 -0
  123. package/dist/types/posts.js +16 -0
  124. package/{lib → dist}/types/tags.js +1 -1
  125. package/dist/types/users.js +16 -0
  126. package/dist/types/websockets.js +16 -0
  127. package/dist/utils/adapterUtils.js +45 -0
  128. package/dist/utils/analyticsUtils.js +72 -0
  129. package/dist/utils/arangodbUtils.js +165 -0
  130. package/dist/utils/auth.js +57 -0
  131. package/dist/utils/index.js +30 -0
  132. package/dist/utils/session.js +60 -0
  133. package/lex.config.cjs +13 -0
  134. package/lib/actions/apps.d.ts +3 -3
  135. package/lib/actions/apps.js +38 -48
  136. package/lib/actions/connections.d.ts +4 -0
  137. package/lib/actions/connections.js +90 -0
  138. package/lib/actions/conversations.d.ts +1 -1
  139. package/lib/actions/conversations.js +32 -21
  140. package/lib/actions/email.d.ts +1 -1
  141. package/lib/actions/email.js +11 -11
  142. package/lib/actions/files.d.ts +2 -2
  143. package/lib/actions/files.js +4 -8
  144. package/lib/actions/groups.d.ts +2 -2
  145. package/lib/actions/groups.js +12 -12
  146. package/lib/actions/images.d.ts +5 -5
  147. package/lib/actions/images.js +120 -66
  148. package/lib/actions/index.d.ts +2 -0
  149. package/lib/actions/index.js +5 -1
  150. package/lib/actions/ios.js +2 -2
  151. package/lib/actions/locations.d.ts +4 -3
  152. package/lib/actions/locations.js +16 -4
  153. package/lib/actions/messages.d.ts +4 -3
  154. package/lib/actions/messages.js +26 -23
  155. package/lib/actions/notifications.d.ts +1 -1
  156. package/lib/actions/notifications.js +1 -1
  157. package/lib/actions/payments.js +63 -60
  158. package/lib/actions/personas.d.ts +3 -0
  159. package/lib/actions/personas.js +110 -0
  160. package/lib/actions/posts.d.ts +5 -2
  161. package/lib/actions/posts.js +55 -41
  162. package/lib/actions/reactions.js +2 -2
  163. package/lib/actions/search.d.ts +2 -2
  164. package/lib/actions/search.js +5 -5
  165. package/lib/actions/sms.d.ts +9 -3
  166. package/lib/actions/sms.js +9 -7
  167. package/lib/actions/statistics.d.ts +1 -1
  168. package/lib/actions/statistics.js +2 -2
  169. package/lib/actions/subscription.d.ts +2 -2
  170. package/lib/actions/subscription.js +12 -22
  171. package/lib/actions/tags.d.ts +8 -3
  172. package/lib/actions/tags.js +46 -21
  173. package/lib/actions/users.d.ts +38 -13
  174. package/lib/actions/users.js +291 -61
  175. package/lib/actions/websockets.d.ts +6 -5
  176. package/lib/actions/websockets.js +37 -35
  177. package/lib/adapters/arangoAdapter.d.ts +1 -1
  178. package/lib/adapters/arangoAdapter.js +1 -1
  179. package/lib/adapters/imageAdapter.d.ts +2 -0
  180. package/lib/adapters/imageAdapter.js +40 -0
  181. package/lib/adapters/messageAdapter.d.ts +2 -0
  182. package/lib/adapters/messageAdapter.js +49 -0
  183. package/lib/adapters/postAdapter.js +4 -4
  184. package/lib/adapters/tagAdapter.js +2 -2
  185. package/lib/adapters/userAdapter.js +10 -5
  186. package/lib/config.js +2 -2
  187. package/lib/index.d.ts +7 -0
  188. package/lib/index.js +44 -8
  189. package/lib/lambdas/actions/websockets.d.ts +7 -6
  190. package/lib/lambdas/actions/websockets.js +9 -5
  191. package/lib/lambdas/authorizer.js +4 -4
  192. package/lib/lambdas/connection.js +16 -17
  193. package/lib/lambdas/utils/message.js +1 -1
  194. package/lib/lambdas/utils/websocket.js +1 -1
  195. package/lib/mocks/image.js +3 -2
  196. package/lib/mocks/user.js +3 -3
  197. package/lib/mutations/index.d.ts +3 -0
  198. package/lib/mutations/index.js +26 -0
  199. package/lib/mutations/locations.d.ts +2 -0
  200. package/lib/mutations/locations.js +44 -0
  201. package/lib/mutations/messages.d.ts +2 -0
  202. package/lib/mutations/messages.js +86 -0
  203. package/lib/mutations/personas.d.ts +2 -0
  204. package/lib/mutations/personas.js +100 -0
  205. package/lib/mutations/posts.d.ts +2 -0
  206. package/lib/mutations/posts.js +53 -0
  207. package/lib/mutations/reactions.d.ts +2 -0
  208. package/lib/mutations/reactions.js +51 -0
  209. package/lib/mutations/statistics.d.ts +2 -0
  210. package/lib/mutations/statistics.js +39 -0
  211. package/lib/mutations/subscriptions.d.ts +2 -0
  212. package/lib/mutations/subscriptions.js +56 -0
  213. package/lib/mutations/tags.d.ts +2 -0
  214. package/lib/mutations/tags.js +120 -0
  215. package/lib/mutations/users.d.ts +1 -0
  216. package/lib/mutations/users.js +116 -0
  217. package/lib/objectTypes/app.d.ts +3 -0
  218. package/lib/objectTypes/app.js +173 -0
  219. package/lib/objectTypes/bankAccount.d.ts +1 -0
  220. package/lib/objectTypes/bankAccount.js +76 -0
  221. package/lib/objectTypes/connection.d.ts +1 -0
  222. package/lib/objectTypes/connection.js +48 -0
  223. package/lib/objectTypes/conversation.d.ts +2 -0
  224. package/lib/objectTypes/conversation.js +77 -0
  225. package/lib/objectTypes/creditCard.d.ts +1 -0
  226. package/lib/objectTypes/creditCard.js +86 -0
  227. package/lib/objectTypes/document.d.ts +1 -0
  228. package/lib/objectTypes/document.js +46 -0
  229. package/lib/objectTypes/error.d.ts +1 -0
  230. package/lib/objectTypes/error.js +46 -0
  231. package/lib/objectTypes/external.d.ts +1 -0
  232. package/lib/objectTypes/external.js +74 -0
  233. package/lib/objectTypes/file.d.ts +2 -0
  234. package/lib/objectTypes/file.js +100 -0
  235. package/lib/objectTypes/filter.d.ts +1 -0
  236. package/lib/objectTypes/filter.js +43 -0
  237. package/lib/objectTypes/group.d.ts +3 -0
  238. package/lib/objectTypes/group.js +123 -0
  239. package/lib/objectTypes/iapSubscription.d.ts +1 -0
  240. package/lib/objectTypes/iapSubscription.js +40 -0
  241. package/lib/objectTypes/image.d.ts +2 -0
  242. package/lib/objectTypes/image.js +129 -0
  243. package/lib/objectTypes/index.d.ts +24 -0
  244. package/lib/objectTypes/index.js +68 -0
  245. package/lib/objectTypes/location.d.ts +2 -0
  246. package/lib/objectTypes/location.js +109 -0
  247. package/lib/objectTypes/message.d.ts +2 -0
  248. package/lib/objectTypes/message.js +96 -0
  249. package/lib/objectTypes/passcode.d.ts +1 -0
  250. package/lib/objectTypes/passcode.js +42 -0
  251. package/lib/objectTypes/persona.d.ts +3 -0
  252. package/lib/objectTypes/persona.js +87 -0
  253. package/lib/objectTypes/plan.d.ts +2 -0
  254. package/lib/objectTypes/plan.js +95 -0
  255. package/lib/objectTypes/post.d.ts +2 -0
  256. package/lib/objectTypes/post.js +125 -0
  257. package/lib/objectTypes/reaction.d.ts +2 -0
  258. package/lib/objectTypes/reaction.js +61 -0
  259. package/lib/objectTypes/relation.d.ts +1 -0
  260. package/lib/objectTypes/relation.js +49 -0
  261. package/lib/objectTypes/search.d.ts +1 -0
  262. package/lib/objectTypes/search.js +72 -0
  263. package/lib/objectTypes/statistics.d.ts +1 -0
  264. package/lib/objectTypes/statistics.js +39 -0
  265. package/lib/objectTypes/subscription.d.ts +2 -0
  266. package/lib/objectTypes/subscription.js +117 -0
  267. package/lib/objectTypes/tag.d.ts +2 -0
  268. package/lib/objectTypes/tag.js +65 -0
  269. package/lib/objectTypes/user.d.ts +4 -0
  270. package/lib/objectTypes/user.js +144 -0
  271. package/lib/queries/index.d.ts +3 -0
  272. package/lib/queries/index.js +33 -0
  273. package/lib/queries/locations.d.ts +2 -0
  274. package/lib/queries/locations.js +45 -0
  275. package/lib/queries/messages.d.ts +2 -0
  276. package/lib/queries/messages.js +52 -0
  277. package/lib/queries/posts.d.ts +2 -0
  278. package/lib/queries/posts.js +154 -0
  279. package/lib/queries/reactions.d.ts +2 -0
  280. package/lib/queries/reactions.js +56 -0
  281. package/lib/queries/statistics.d.ts +2 -0
  282. package/lib/queries/statistics.js +39 -0
  283. package/lib/queries/subscriptions.d.ts +2 -0
  284. package/lib/queries/subscriptions.js +44 -0
  285. package/lib/queries/tags.d.ts +2 -0
  286. package/lib/queries/tags.js +75 -0
  287. package/lib/queries/users.d.ts +1 -0
  288. package/lib/queries/users.js +64 -0
  289. package/lib/types/{apps.d.ts → apps.types.d.ts} +2 -2
  290. package/lib/types/apps.types.js +32 -0
  291. package/lib/types/{arangodb.d.ts → arangodb.types.d.ts} +4 -0
  292. package/lib/types/arangodb.types.js +16 -0
  293. package/lib/types/auth.types.d.ts +9 -0
  294. package/lib/types/auth.types.js +16 -0
  295. package/lib/types/{connections.d.ts → connections.types.d.ts} +1 -3
  296. package/lib/types/connections.types.js +16 -0
  297. package/lib/types/{conversations.d.ts → conversations.types.d.ts} +2 -4
  298. package/lib/types/conversations.types.js +16 -0
  299. package/lib/types/{email.d.ts → email.types.d.ts} +3 -3
  300. package/lib/types/email.types.js +16 -0
  301. package/lib/types/error.types.js +44 -0
  302. package/lib/types/{files.d.ts → files.types.d.ts} +1 -4
  303. package/lib/types/files.types.js +16 -0
  304. package/lib/types/google.types.js +16 -0
  305. package/lib/types/{groups.d.ts → groups.types.d.ts} +1 -4
  306. package/lib/types/groups.types.js +16 -0
  307. package/lib/types/{images.d.ts → images.types.d.ts} +9 -8
  308. package/lib/types/images.types.js +16 -0
  309. package/lib/types/index.d.ts +20 -18
  310. package/lib/types/index.js +41 -37
  311. package/lib/types/{locations.d.ts → locations.types.d.ts} +1 -3
  312. package/lib/types/locations.types.js +16 -0
  313. package/lib/types/{messages.d.ts → messages.types.d.ts} +5 -5
  314. package/lib/types/messages.types.js +16 -0
  315. package/lib/types/{notifications.d.ts → notifications.types.d.ts} +4 -2
  316. package/lib/types/notifications.types.js +16 -0
  317. package/lib/types/{payments.d.ts → payments.types.d.ts} +1 -4
  318. package/lib/types/payments.types.js +16 -0
  319. package/lib/types/personas.types.d.ts +32 -0
  320. package/lib/types/personas.types.js +16 -0
  321. package/lib/types/{posts.d.ts → posts.types.d.ts} +4 -7
  322. package/lib/types/posts.types.js +16 -0
  323. package/lib/types/statistics.types.js +16 -0
  324. package/lib/types/{tags.d.ts → tags.types.d.ts} +6 -1
  325. package/lib/types/tags.types.js +16 -0
  326. package/lib/types/{users.d.ts → users.types.d.ts} +12 -9
  327. package/lib/types/users.types.js +16 -0
  328. package/lib/types/{websocket.d.ts → websockets.types.d.ts} +6 -1
  329. package/lib/types/websockets.types.js +16 -0
  330. package/lib/utils/analyticsUtils.d.ts +3 -3
  331. package/lib/utils/analyticsUtils.js +3 -3
  332. package/lib/utils/arangodbUtils.d.ts +2 -1
  333. package/lib/utils/arangodbUtils.js +22 -1
  334. package/lib/utils/auth.d.ts +2 -1
  335. package/lib/utils/auth.js +8 -1
  336. package/lib/utils/index.js +1 -1
  337. package/lib/utils/session.d.ts +3 -1
  338. package/lib/utils/session.js +10 -7
  339. package/lib/utils/stripeUtils.d.ts +3 -0
  340. package/lib/utils/stripeUtils.js +43 -0
  341. package/package.json +35 -34
  342. package/.eslintrc +0 -10
  343. package/lib/types/apps.js +0 -32
  344. package/lib/types/auth.d.ts +0 -7
  345. package/lib/types/conversations.js +0 -16
  346. package/lib/types/files.js +0 -16
  347. package/lib/types/google.js +0 -16
  348. package/lib/types/images.js +0 -16
  349. package/lib/types/payments.js +0 -16
  350. package/lib/types/posts.js +0 -16
  351. package/lib/types/users.js +0 -16
  352. package/lib/types/websocket.js +0 -16
  353. /package/{lib → dist}/types/error.js +0 -0
  354. /package/{lib → dist}/types/statistics.js +0 -0
  355. /package/lib/types/{error.d.ts → error.types.d.ts} +0 -0
  356. /package/lib/types/{google.d.ts → google.types.d.ts} +0 -0
  357. /package/lib/types/{statistics.d.ts → statistics.types.d.ts} +0 -0
@@ -0,0 +1,784 @@
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");
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-02-24.acacia";
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(" || ")}
148
+ 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}
218
+ 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}
249
+ 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}
389
+ LIMIT 1
390
+ 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")}
456
+ 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}
484
+ 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
511
+ 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))
546
+ FOR u, it IN OUTBOUND t isTagged
547
+ 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")}
577
+ LET distance = DISTANCE(u.latitude || 0, u.longitude || 0, session.latitude || 0, session.longitude || 0)
578
+ ${limit.aql}
579
+ 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}")
604
+ FOR u, connection IN OUTBOUND cu hasConnection
605
+ 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(" || ")}
657
+ 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
715
+ 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 = (
733
+ FOR u IN users
734
+ FILTER u.active == true
735
+ RETURN u
736
+ )
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';\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';\nimport type {UserType} from '../types/users';\nimport type {EdgeCollection} from 'arangojs/collections';\n\n\nconst eventCategory = 'users';\nconst STRIPE_API_VERSION = '2025-02-24.acacia';\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"]
}
