@nlabs/reaktor 0.10.0 → 0.10.2

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 (448) 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/actions/index.html +21 -36
  6. package/coverage/actions/notifications.ts.html +223 -0
  7. package/coverage/actions/posts.ts.html +2356 -0
  8. package/coverage/adapters/arangoAdapter.ts.html +170 -20
  9. package/coverage/adapters/fileAdapter.ts.html +445 -0
  10. package/coverage/adapters/index.html +72 -42
  11. package/coverage/adapters/postAdapter.ts.html +436 -0
  12. package/coverage/adapters/reaktorAdapter.ts.html +201 -18
  13. package/coverage/adapters/tagAdapter.ts.html +274 -25
  14. package/coverage/adapters/userAdapter.ts.html +829 -0
  15. package/coverage/analyticsUtils.ts.html +286 -0
  16. package/coverage/config.ts.html +766 -0
  17. package/coverage/index.html +77 -77
  18. package/coverage/mocks/group.ts.html +5 -5
  19. package/coverage/mocks/image.ts.html +18 -12
  20. package/coverage/mocks/index.html +37 -7
  21. package/coverage/mocks/post.ts.html +8 -8
  22. package/coverage/mocks/tag.ts.html +2 -2
  23. package/coverage/mocks/user.ts.html +18 -15
  24. package/coverage/testUtils.ts.html +1309 -0
  25. package/coverage/translationQueue.ts.html +592 -0
  26. package/coverage/types/error.types.ts.html +148 -0
  27. package/coverage/types/index.html +2 -2
  28. package/coverage/utils/adapterUtils.ts.html +39 -27
  29. package/coverage/utils/analyticsUtils.ts.html +35 -41
  30. package/coverage/utils/authUtils.ts.html +328 -0
  31. package/coverage/utils/dbI18n.ts.html +280 -0
  32. package/coverage/utils/googleTranslate.ts.html +385 -0
  33. package/coverage/utils/index.html +29 -44
  34. package/coverage/utils/localeUtils.ts.html +193 -0
  35. package/coverage/utils/sessionUtils.ts.html +211 -0
  36. package/coverage/utils/testUtils.ts.html +1309 -0
  37. package/index.js +1 -1
  38. package/lex.config.mjs +34 -0
  39. package/lib/actions/apps.d.ts +4 -4
  40. package/lib/actions/apps.js +17 -239
  41. package/lib/actions/connections.d.ts +2 -1
  42. package/lib/actions/connections.js +6 -89
  43. package/lib/actions/content.d.ts +13 -0
  44. package/lib/actions/content.js +17 -0
  45. package/lib/actions/conversations.d.ts +4 -2
  46. package/lib/actions/conversations.js +19 -336
  47. package/lib/actions/dynamodb.d.ts +4 -0
  48. package/lib/actions/dynamodb.js +2 -150
  49. package/lib/actions/email.d.ts +5 -2
  50. package/lib/actions/email.js +2 -152
  51. package/lib/actions/files.d.ts +3 -3
  52. package/lib/actions/files.js +5 -283
  53. package/lib/actions/groups.d.ts +4 -5
  54. package/lib/actions/groups.js +19 -259
  55. package/lib/actions/images.d.ts +8 -1
  56. package/lib/actions/images.js +31 -700
  57. package/lib/actions/index.d.ts +27 -23
  58. package/lib/actions/index.js +2 -66
  59. package/lib/actions/ios.d.ts +3 -3
  60. package/lib/actions/ios.js +9 -162
  61. package/lib/actions/locations.d.ts +3 -3
  62. package/lib/actions/locations.js +7 -122
  63. package/lib/actions/messages.d.ts +3 -4
  64. package/lib/actions/messages.js +21 -196
  65. package/lib/actions/notifications.d.ts +5 -1
  66. package/lib/actions/notifications.js +2 -59
  67. package/lib/actions/payments.d.ts +3 -1
  68. package/lib/actions/payments.js +11 -464
  69. package/lib/actions/posts.d.ts +7 -4
  70. package/lib/actions/posts.js +75 -527
  71. package/lib/actions/profiles.d.ts +4 -0
  72. package/lib/actions/profiles.js +8 -0
  73. package/lib/actions/reactions.d.ts +3 -1
  74. package/lib/actions/reactions.js +25 -312
  75. package/lib/actions/s3.d.ts +4 -0
  76. package/lib/actions/s3.js +2 -133
  77. package/lib/actions/search.d.ts +2 -2
  78. package/lib/actions/search.js +5 -90
  79. package/lib/actions/sms.d.ts +2 -9
  80. package/lib/actions/sms.js +2 -108
  81. package/lib/actions/statistics.d.ts +2 -1
  82. package/lib/actions/statistics.js +6 -60
  83. package/lib/actions/{subscription.d.ts → subscriptions.d.ts} +2 -2
  84. package/lib/actions/subscriptions.js +12 -0
  85. package/lib/actions/tags.d.ts +3 -3
  86. package/lib/actions/tags.js +19 -287
  87. package/lib/actions/users.d.ts +13 -12
  88. package/lib/actions/users.js +66 -763
  89. package/lib/actions/websockets.d.ts +2 -2
  90. package/lib/actions/websockets.js +14 -158
  91. package/lib/adapters/arangoAdapter.d.ts +7 -1
  92. package/lib/adapters/arangoAdapter.js +2 -46
  93. package/lib/adapters/contentAdapter.d.ts +8 -0
  94. package/lib/adapters/contentAdapter.js +2 -0
  95. package/lib/adapters/fileAdapter.d.ts +8 -2
  96. package/lib/adapters/fileAdapter.js +2 -76
  97. package/lib/adapters/imageAdapter.d.ts +7 -1
  98. package/lib/adapters/imageAdapter.js +2 -40
  99. package/lib/adapters/index.d.ts +9 -0
  100. package/lib/adapters/index.js +2 -0
  101. package/lib/adapters/messageAdapter.d.ts +7 -1
  102. package/lib/adapters/messageAdapter.js +2 -49
  103. package/lib/adapters/postAdapter.d.ts +7 -1
  104. package/lib/adapters/postAdapter.js +2 -70
  105. package/lib/adapters/reaktorAdapter.d.ts +16 -6
  106. package/lib/adapters/reaktorAdapter.js +2 -44
  107. package/lib/adapters/tagAdapter.d.ts +8 -1
  108. package/lib/adapters/tagAdapter.js +2 -50
  109. package/lib/adapters/userAdapter.d.ts +12 -1
  110. package/lib/adapters/userAdapter.js +2 -115
  111. package/lib/config.d.ts +43 -6
  112. package/lib/config.js +2 -125
  113. package/lib/handlers/graphqlHandler.d.ts +4 -0
  114. package/lib/handlers/graphqlHandler.js +2 -0
  115. package/lib/index.d.ts +17 -10
  116. package/lib/index.js +2 -66
  117. package/lib/lambdas/actions/websockets.d.ts +3 -3
  118. package/lib/lambdas/actions/websockets.js +14 -116
  119. package/lib/lambdas/authorizer.js +2 -67
  120. package/lib/lambdas/connection.js +2 -91
  121. package/lib/lambdas/utils/message.js +2 -42
  122. package/lib/lambdas/utils/websocket.js +2 -105
  123. package/lib/mocks/conversation.js +2 -35
  124. package/lib/mocks/file.js +2 -38
  125. package/lib/mocks/group.js +2 -47
  126. package/lib/mocks/image.d.ts +1 -1
  127. package/lib/mocks/image.js +2 -44
  128. package/lib/mocks/post.js +2 -55
  129. package/lib/mocks/tag.d.ts +1 -1
  130. package/lib/mocks/tag.js +2 -37
  131. package/lib/mocks/user.d.ts +1 -1
  132. package/lib/mocks/user.js +2 -88
  133. package/lib/mutations/content.d.ts +2 -0
  134. package/lib/mutations/content.js +2 -0
  135. package/lib/mutations/index.d.ts +11 -3
  136. package/lib/mutations/index.js +2 -26
  137. package/lib/mutations/locations.d.ts +1 -1
  138. package/lib/mutations/locations.js +2 -44
  139. package/lib/mutations/messages.d.ts +1 -1
  140. package/lib/mutations/messages.js +2 -86
  141. package/lib/mutations/posts.d.ts +1 -1
  142. package/lib/mutations/posts.js +2 -53
  143. package/lib/mutations/profiles.d.ts +2 -0
  144. package/lib/mutations/profiles.js +2 -0
  145. package/lib/mutations/reactions.d.ts +1 -1
  146. package/lib/mutations/reactions.js +2 -51
  147. package/lib/mutations/statistics.d.ts +1 -1
  148. package/lib/mutations/statistics.js +2 -39
  149. package/lib/mutations/subscriptions.d.ts +1 -1
  150. package/lib/mutations/subscriptions.js +2 -56
  151. package/lib/mutations/tags.d.ts +1 -1
  152. package/lib/mutations/tags.js +2 -120
  153. package/lib/mutations/users.js +2 -116
  154. package/lib/objectTypes/app.js +2 -173
  155. package/lib/objectTypes/bankAccount.js +2 -76
  156. package/lib/objectTypes/connection.js +2 -48
  157. package/lib/objectTypes/content.d.ts +2 -0
  158. package/lib/objectTypes/content.js +2 -0
  159. package/lib/objectTypes/conversation.js +2 -77
  160. package/lib/objectTypes/creditCard.js +2 -86
  161. package/lib/objectTypes/document.js +2 -46
  162. package/lib/objectTypes/error.js +2 -46
  163. package/lib/objectTypes/external.js +2 -74
  164. package/lib/objectTypes/file.js +2 -100
  165. package/lib/objectTypes/filter.js +2 -43
  166. package/lib/objectTypes/group.js +2 -123
  167. package/lib/objectTypes/iapSubscription.js +2 -40
  168. package/lib/objectTypes/image.d.ts +4 -0
  169. package/lib/objectTypes/image.js +2 -129
  170. package/lib/objectTypes/index.d.ts +26 -24
  171. package/lib/objectTypes/index.js +2 -68
  172. package/lib/objectTypes/location.js +2 -109
  173. package/lib/objectTypes/message.js +2 -96
  174. package/lib/objectTypes/passcode.js +2 -42
  175. package/lib/objectTypes/plan.js +2 -95
  176. package/lib/objectTypes/post.js +2 -125
  177. package/lib/objectTypes/profile.d.ts +3 -0
  178. package/lib/objectTypes/profile.js +2 -0
  179. package/lib/objectTypes/reaction.js +2 -61
  180. package/lib/objectTypes/relation.js +2 -49
  181. package/lib/objectTypes/search.js +2 -72
  182. package/lib/objectTypes/statistics.js +2 -39
  183. package/lib/objectTypes/subscription.js +2 -117
  184. package/lib/objectTypes/tag.js +2 -65
  185. package/lib/objectTypes/user.js +2 -144
  186. package/lib/queries/content.d.ts +2 -0
  187. package/lib/queries/content.js +2 -0
  188. package/lib/queries/index.d.ts +10 -3
  189. package/lib/queries/index.js +2 -33
  190. package/lib/queries/locations.d.ts +1 -1
  191. package/lib/queries/locations.js +2 -45
  192. package/lib/queries/messages.d.ts +1 -1
  193. package/lib/queries/messages.js +2 -52
  194. package/lib/queries/posts.d.ts +1 -1
  195. package/lib/queries/posts.js +2 -154
  196. package/lib/queries/reactions.d.ts +1 -1
  197. package/lib/queries/reactions.js +2 -56
  198. package/lib/queries/statistics.d.ts +1 -1
  199. package/lib/queries/statistics.js +2 -39
  200. package/lib/queries/subscriptions.d.ts +1 -1
  201. package/lib/queries/subscriptions.js +2 -44
  202. package/lib/queries/tags.d.ts +1 -1
  203. package/lib/queries/tags.js +2 -75
  204. package/lib/queries/users.js +2 -64
  205. package/lib/templates/email/layout.js +3 -25
  206. package/lib/templates/email/passwordForgot.js +3 -25
  207. package/lib/templates/email/passwordRecovery.js +3 -25
  208. package/lib/templates/email/verifyEmail.js +3 -25
  209. package/lib/templates/email/welcome.js +3 -25
  210. package/lib/templates/sms/passwordForgot.js +2 -24
  211. package/lib/templates/sms/passwordRecovery.js +2 -24
  212. package/lib/templates/sms/verifyEmail.js +2 -24
  213. package/lib/templates/sms/verifyPhone.js +2 -24
  214. package/lib/templates/sms/welcome.js +2 -24
  215. package/lib/types/apps.types.d.ts +6 -2
  216. package/lib/types/apps.types.js +2 -32
  217. package/lib/types/arangodb.types.d.ts +8 -4
  218. package/lib/types/arangodb.types.js +1 -16
  219. package/lib/types/auth.types.d.ts +11 -5
  220. package/lib/types/auth.types.js +1 -16
  221. package/lib/types/connections.types.d.ts +5 -1
  222. package/lib/types/connections.types.js +1 -16
  223. package/lib/types/content.types.d.ts +31 -0
  224. package/lib/types/content.types.js +1 -0
  225. package/lib/types/conversations.types.d.ts +6 -2
  226. package/lib/types/conversations.types.js +1 -16
  227. package/lib/types/email.types.d.ts +9 -5
  228. package/lib/types/email.types.js +1 -16
  229. package/lib/types/error.types.d.ts +1 -0
  230. package/lib/types/error.types.js +2 -44
  231. package/lib/types/files.types.d.ts +5 -1
  232. package/lib/types/files.types.js +1 -16
  233. package/lib/types/google.types.d.ts +4 -0
  234. package/lib/types/google.types.js +1 -16
  235. package/lib/types/groups.types.d.ts +5 -1
  236. package/lib/types/groups.types.js +1 -16
  237. package/lib/types/images.types.d.ts +6 -2
  238. package/lib/types/images.types.js +1 -16
  239. package/lib/types/index.d.ts +26 -20
  240. package/lib/types/index.js +2 -60
  241. package/lib/types/locations.types.d.ts +5 -1
  242. package/lib/types/locations.types.js +1 -16
  243. package/lib/types/messages.types.d.ts +7 -3
  244. package/lib/types/messages.types.js +1 -16
  245. package/lib/types/notifications.types.d.ts +9 -5
  246. package/lib/types/notifications.types.js +1 -16
  247. package/lib/types/payments.types.d.ts +5 -1
  248. package/lib/types/payments.types.js +1 -16
  249. package/lib/types/posts.types.d.ts +7 -3
  250. package/lib/types/posts.types.js +1 -16
  251. package/lib/types/{personas.types.d.ts → profiles.types.d.ts} +3 -15
  252. package/lib/types/profiles.types.js +1 -0
  253. package/lib/types/statistics.types.js +1 -16
  254. package/lib/types/tags.types.d.ts +5 -1
  255. package/lib/types/tags.types.js +1 -16
  256. package/lib/types/users.types.d.ts +41 -10
  257. package/lib/types/users.types.js +1 -16
  258. package/lib/types/websockets.types.d.ts +5 -1
  259. package/lib/types/websockets.types.js +1 -16
  260. package/lib/utils/adapterUtils.d.ts +4 -0
  261. package/lib/utils/adapterUtils.js +2 -45
  262. package/lib/utils/analyticsUtils.d.ts +5 -1
  263. package/lib/utils/analyticsUtils.js +2 -72
  264. package/lib/utils/arangodbUtils.d.ts +6 -2
  265. package/lib/utils/arangodbUtils.js +5 -163
  266. package/lib/utils/{auth.d.ts → authUtils.d.ts} +3 -1
  267. package/lib/utils/authUtils.js +2 -0
  268. package/lib/utils/contextUtils.d.ts +3 -0
  269. package/lib/utils/contextUtils.js +2 -0
  270. package/lib/utils/dbI18n.d.ts +10 -0
  271. package/lib/utils/dbI18n.example.d.ts +20 -0
  272. package/lib/utils/dbI18n.example.js +6 -0
  273. package/lib/utils/dbI18n.js +2 -0
  274. package/lib/utils/googleTranslate.d.ts +6 -0
  275. package/lib/utils/googleTranslate.js +2 -0
  276. package/lib/utils/graphqlUtils.d.ts +10 -0
  277. package/lib/utils/graphqlUtils.js +2 -0
  278. package/lib/utils/index.d.ts +18 -5
  279. package/lib/utils/index.js +2 -30
  280. package/lib/utils/languageDetection.d.ts +8 -0
  281. package/lib/utils/languageDetection.js +2 -0
  282. package/lib/utils/localeUtils.d.ts +11 -0
  283. package/lib/utils/localeUtils.example.d.ts +5 -0
  284. package/lib/utils/localeUtils.example.js +2 -0
  285. package/lib/utils/localeUtils.js +2 -0
  286. package/lib/utils/middlewareUtils.d.ts +2 -0
  287. package/lib/utils/middlewareUtils.js +2 -0
  288. package/lib/utils/{session.d.ts → sessionUtils.d.ts} +1 -1
  289. package/lib/utils/sessionUtils.js +2 -0
  290. package/lib/utils/stripeUtils.d.ts +7 -3
  291. package/lib/utils/stripeUtils.js +2 -43
  292. package/lib/utils/templateUtils.d.ts +3 -0
  293. package/lib/utils/templateUtils.js +2 -0
  294. package/lib/utils/testUtils.d.ts +95 -0
  295. package/lib/utils/testUtils.js +2 -0
  296. package/lib/utils/translationQueue.d.ts +29 -0
  297. package/lib/utils/translationQueue.example.d.ts +33 -0
  298. package/lib/utils/translationQueue.example.js +2 -0
  299. package/lib/utils/translationQueue.js +2 -0
  300. package/package.json +32 -28
  301. package/tsconfig.build.json +21 -0
  302. package/tsconfig.lint.json +33 -0
  303. package/tsconfig.test.json +31 -0
  304. package/.prettierrc.js +0 -4
  305. package/dist/actions/apps.js +0 -242
  306. package/dist/actions/connections.js +0 -90
  307. package/dist/actions/conversations.js +0 -350
  308. package/dist/actions/dynamodb.js +0 -150
  309. package/dist/actions/email.js +0 -152
  310. package/dist/actions/files.js +0 -283
  311. package/dist/actions/groups.js +0 -292
  312. package/dist/actions/images.js +0 -735
  313. package/dist/actions/index.js +0 -66
  314. package/dist/actions/ios.js +0 -164
  315. package/dist/actions/locations.js +0 -122
  316. package/dist/actions/messages.js +0 -208
  317. package/dist/actions/notifications.js +0 -59
  318. package/dist/actions/payments.js +0 -497
  319. package/dist/actions/personas.js +0 -110
  320. package/dist/actions/posts.js +0 -595
  321. package/dist/actions/reactions.js +0 -322
  322. package/dist/actions/s3.js +0 -133
  323. package/dist/actions/search.js +0 -90
  324. package/dist/actions/sms.js +0 -108
  325. package/dist/actions/statistics.js +0 -62
  326. package/dist/actions/subscription.js +0 -220
  327. package/dist/actions/tags.js +0 -292
  328. package/dist/actions/users.js +0 -784
  329. package/dist/actions/websockets.js +0 -174
  330. package/dist/adapters/arangoAdapter.js +0 -46
  331. package/dist/adapters/fileAdapter.js +0 -76
  332. package/dist/adapters/imageAdapter.js +0 -40
  333. package/dist/adapters/messageAdapter.js +0 -49
  334. package/dist/adapters/postAdapter.js +0 -70
  335. package/dist/adapters/reaktorAdapter.js +0 -44
  336. package/dist/adapters/tagAdapter.js +0 -50
  337. package/dist/adapters/userAdapter.js +0 -115
  338. package/dist/config.js +0 -125
  339. package/dist/index.js +0 -66
  340. package/dist/lambdas/actions/websockets.js +0 -132
  341. package/dist/lambdas/authorizer.js +0 -67
  342. package/dist/lambdas/connection.js +0 -91
  343. package/dist/lambdas/utils/message.js +0 -42
  344. package/dist/lambdas/utils/websocket.js +0 -105
  345. package/dist/mocks/conversation.js +0 -35
  346. package/dist/mocks/file.js +0 -38
  347. package/dist/mocks/group.js +0 -47
  348. package/dist/mocks/image.js +0 -44
  349. package/dist/mocks/nlabs.png +0 -0
  350. package/dist/mocks/post.js +0 -55
  351. package/dist/mocks/tag.js +0 -37
  352. package/dist/mocks/user.js +0 -88
  353. package/dist/mutations/index.js +0 -26
  354. package/dist/mutations/locations.js +0 -44
  355. package/dist/mutations/messages.js +0 -86
  356. package/dist/mutations/personas.js +0 -100
  357. package/dist/mutations/posts.js +0 -53
  358. package/dist/mutations/reactions.js +0 -51
  359. package/dist/mutations/statistics.js +0 -39
  360. package/dist/mutations/subscriptions.js +0 -56
  361. package/dist/mutations/tags.js +0 -120
  362. package/dist/mutations/users.js +0 -116
  363. package/dist/objectTypes/app.js +0 -173
  364. package/dist/objectTypes/bankAccount.js +0 -76
  365. package/dist/objectTypes/connection.js +0 -48
  366. package/dist/objectTypes/conversation.js +0 -77
  367. package/dist/objectTypes/creditCard.js +0 -86
  368. package/dist/objectTypes/document.js +0 -46
  369. package/dist/objectTypes/error.js +0 -46
  370. package/dist/objectTypes/external.js +0 -74
  371. package/dist/objectTypes/file.js +0 -100
  372. package/dist/objectTypes/filter.js +0 -43
  373. package/dist/objectTypes/group.js +0 -123
  374. package/dist/objectTypes/iapSubscription.js +0 -40
  375. package/dist/objectTypes/image.js +0 -129
  376. package/dist/objectTypes/index.js +0 -68
  377. package/dist/objectTypes/location.js +0 -109
  378. package/dist/objectTypes/message.js +0 -96
  379. package/dist/objectTypes/passcode.js +0 -42
  380. package/dist/objectTypes/persona.js +0 -87
  381. package/dist/objectTypes/plan.js +0 -95
  382. package/dist/objectTypes/post.js +0 -125
  383. package/dist/objectTypes/reaction.js +0 -61
  384. package/dist/objectTypes/relation.js +0 -49
  385. package/dist/objectTypes/search.js +0 -72
  386. package/dist/objectTypes/statistics.js +0 -39
  387. package/dist/objectTypes/subscription.js +0 -117
  388. package/dist/objectTypes/tag.js +0 -65
  389. package/dist/objectTypes/user.js +0 -144
  390. package/dist/queries/index.js +0 -33
  391. package/dist/queries/locations.js +0 -45
  392. package/dist/queries/messages.js +0 -52
  393. package/dist/queries/posts.js +0 -154
  394. package/dist/queries/reactions.js +0 -56
  395. package/dist/queries/statistics.js +0 -39
  396. package/dist/queries/subscriptions.js +0 -44
  397. package/dist/queries/tags.js +0 -75
  398. package/dist/queries/users.js +0 -64
  399. package/dist/templates/email/layout.js +0 -302
  400. package/dist/templates/email/passwordForgot.js +0 -38
  401. package/dist/templates/email/passwordRecovery.js +0 -35
  402. package/dist/templates/email/verifyEmail.js +0 -38
  403. package/dist/templates/email/welcome.js +0 -38
  404. package/dist/templates/sms/passwordForgot.js +0 -24
  405. package/dist/templates/sms/passwordRecovery.js +0 -24
  406. package/dist/templates/sms/verifyEmail.js +0 -24
  407. package/dist/templates/sms/verifyPhone.js +0 -24
  408. package/dist/templates/sms/welcome.js +0 -24
  409. package/dist/types/apps.js +0 -32
  410. package/dist/types/arangodb.js +0 -16
  411. package/dist/types/auth.js +0 -16
  412. package/dist/types/connections.js +0 -16
  413. package/dist/types/conversations.js +0 -16
  414. package/dist/types/email.js +0 -16
  415. package/dist/types/error.js +0 -44
  416. package/dist/types/files.js +0 -16
  417. package/dist/types/google.js +0 -16
  418. package/dist/types/groups.js +0 -16
  419. package/dist/types/images.js +0 -16
  420. package/dist/types/index.js +0 -60
  421. package/dist/types/locations.js +0 -16
  422. package/dist/types/messages.js +0 -16
  423. package/dist/types/notifications.js +0 -16
  424. package/dist/types/payments.js +0 -16
  425. package/dist/types/personas.js +0 -16
  426. package/dist/types/posts.js +0 -16
  427. package/dist/types/statistics.js +0 -16
  428. package/dist/types/tags.js +0 -16
  429. package/dist/types/users.js +0 -16
  430. package/dist/types/websockets.js +0 -16
  431. package/dist/utils/adapterUtils.js +0 -45
  432. package/dist/utils/analyticsUtils.js +0 -72
  433. package/dist/utils/arangodbUtils.js +0 -165
  434. package/dist/utils/auth.js +0 -57
  435. package/dist/utils/index.js +0 -30
  436. package/dist/utils/session.js +0 -60
  437. package/jest.setup.js +0 -0
  438. package/lex.config.cjs +0 -13
  439. package/lib/actions/personas.d.ts +0 -3
  440. package/lib/actions/personas.js +0 -110
  441. package/lib/actions/subscription.js +0 -208
  442. package/lib/mutations/personas.d.ts +0 -2
  443. package/lib/mutations/personas.js +0 -100
  444. package/lib/objectTypes/persona.d.ts +0 -3
  445. package/lib/objectTypes/persona.js +0 -87
  446. package/lib/types/personas.types.js +0 -16
  447. package/lib/utils/auth.js +0 -57
  448. package/lib/utils/session.js +0 -60
@@ -1,784 +1,87 @@
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{parseNum as Y}from"@nlabs/utils/parsers/numbers";import{createHash as L,createPassword as b,parseArangoId as x,parseChar as j,parseEmail as _,parsePassword as M,parsePhone as Q,parseUsername as w}from"@nlabs/utils/parsers/strings";import{aql as U}from"arangojs";import{DateTime as X}from"luxon";import z from"stripe";import{parseUser as D}from"../adapters/userAdapter.js";import{Config as K}from"../config.js";import{ErrorTypes as T}from"../types/error.types.js";import{logError as g,logException as q}from"../utils/analyticsUtils.js";import{getDocId as Z,getLimit as ee,selectReactionCountByType as se,useDb as f}from"../utils/arangodbUtils.js";import{detectLanguage as re}from"../utils/languageDetection.js";import{getSession as B,setSession as te}from"../utils/sessionUtils.js";import{sendEmail as oe}from"./email.js";import{sendSms as ne}from"./sms.js";const E="users",ae="2025-07-30.basil";var ie=(n=>(n[n.DEACTIVATED=0]="DEACTIVATED",n[n.ACTIVE=1]="ACTIVE",n[n.PREMIUM=2]="PREMIUM",n[n.CONTENT_ADMIN=3]="CONTENT_ADMIN",n[n.ADMIN=4]="ADMIN",n))(ie||{});const F=(e,o="",s=0,a=15)=>{const t=X.local(),n=t.plus({minutes:a}),c=Math.floor(t.toSeconds()),r=Math.floor(n.toSeconds()),l=te({exp:r,iat:c,userAccess:s,userId:e,username:o});return{expires:n.toMillis(),issued:t.toMillis(),token:l,userId:e,username:o}},C=(e=[])=>e.reduce((o,s)=>s.includes("Count")?se("users","u",s,o):o,{objects:[],queries:[]}),v=(e={})=>{const{from:o=0,to:s=30}=e,a=ee(o,s);return{...e,limit:a}},Ue=async(e,o)=>{const s="addUser",{databaseName:a,languageContext:t}=e,{confirm:n,...c}=o,{email:r,password:l,phone:u,username:i,userId:p,_key:m,_id:d,...y}=D(c);if(!!!l||!(!!i||!!u||!!r))return q({action:s,category:E,params:{username:i},value:T.INVALID_ARGUMENTS},e),null;const h=L(`${i||u||r}${l}`,""),O=b(l,h),$=[];i&&$.push(`u.username == "${i}"`),r&&$.push(`u.email == "${r}"`),u&&$.push(`u.phone == ${u}`);const P=`FOR u IN users
2
+ FILTER ${$.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 f(a).query(P).then(J=>J.all())).length)throw q({action:s,category:E,params:{email:r,phone:u,username:i},value:T.EXISTING_ITEM},e)}catch(S){throw g({action:s,category:E,params:{email:r,phone:u,username:i},value:T.DATABASE_ERROR},S,e)}const N=u?.replace(/\D/g,"").substring(0,3),k=re({...t,phoneCountryCode:N||"",userPreference:y.locale||"en"}),G=Math.floor(1e5+Math.random()*9e5),V=Math.floor(1e5+Math.random()*9e5),W={...y,_key:L(i),added:Date.now(),email:r,locale:k,modified:Date.now(),password:O,phone:u,salt:h,userAccess:1,username:i,verifiedEmail:!1,verifiedEmailCode:G,verifiedPhone:!1,verifiedSmsCode:V},H=U`INSERT ${W} IN users RETURN NEW`;return await f(a).query(H).then(S=>S.next()).catch(S=>{throw g({action:s,category:E,params:{username:i},value:T.DATABASE_ERROR},S,e)})},he=async(e,o)=>{const s="updateUser",{databaseName:a}=e,t=D(o),{_key:n,_id:c,tags:r=[],...l}=t,{id:u}=t,i=U`LET u = DOCUMENT(${u})
5
+ UPDATE u WITH ${l} IN users
6
+ RETURN NEW`;try{const p=f(a),m=await p.query(i).then(y=>y.next()),d=p.collection("isTagged");return await Promise.all(r.map(({id:y,name:I})=>{const R=U`FOR it IN isTagged
7
+ FILTER it._from == ${y} && it._to == ${u} && it.name == ${I}
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 h={_from:y,_key:L(`isTagged-${y}-${u}`),_to:u,added:Date.now(),name:I};return d.save(h,{returnNew:!0}).then(()=>h)})})),m}catch(p){throw g({action:s,category:E,params:{user:o},value:T.DATABASE_ERROR},p,e)}},Ne=async(e,{email:o,phone:s,username:a})=>{const t="forgotPassword",{databaseName:n}=e,c=U`FOR u IN users
10
+ FILTER u.email == ${o} || u.phone == ${s} || u.username == ${a}
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}
12
+ RETURN u`;try{return await f(n).query(c).then(async r=>{const l=r.next();if(l){const{email:u,phone:i,verifiedEmail:p,verifiedPhone:m}=l,d=1e3*60*15,y=Math.floor(1e5+Math.random()*9e5),I=Z("users",l);let R;if(u&&p&&(oe({context:e,text:`Your code is ${y}`}),R={verifiedEmailCode:y,verifiedEmailExpires:d}),i&&m&&(ne({context:e,text:`Your code is ${y}`}),R={verifiedPhoneExpires:d,verifiedSmsCode:y}),R.verifiedEmailCode||R.verifiedSmsCode){const A=U`UPDATE ${I} WITH ${R} IN users`;return await f(n).query(A),!0}return!1}return!1})}catch(r){return g({action:t,category:E,params:{email:o,phone:s,username:a},value:T.DATABASE_ERROR},r,e),!1}},Oe=async(e,{code:o,password:s,type:a,username:t})=>{const n="resetPassword",{databaseName:c}=e,r=M(s),l=U`FOR u IN users
13
+ FILTER u.username == ${t}
304
14
  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}
15
+ RETURN u`;try{return await f(c).query(l).then(async u=>{const i=u.next();if(i){const{_id:p,salt:m,verifiedEmailCode:d,verifiedEmailExpires:y=0,verifiedSmsCode:I,verifiedPhoneExpires:R=0}=i,A=Date.now();let h;switch(a){case"email":o===d&&y>A&&(h={password:b(r,m)});break;case"phone":o===I&&R>A&&(h={password:b(r,m)});break;default:return!1}if(h){const O=U`UPDATE ${p} WITH ${h} IN users`;return await f(c).query(O),!0}}return!1})}catch(u){return g({action:n,category:E,params:{username:t},value:T.DATABASE_ERROR},u,e),!1}},$e=async(e,{code:o,type:s,value:a})=>{const t="confirmCode",{databaseName:n}=e,c=s==="sms",r=c?Q(a):_(a),u=`FOR u IN users
16
+ ${c?`FILTER u.phone == "${r}" && u.verifiedSmsCode == ${o}`:`FILTER u.email == "${r}" && u.verifiedEmailCode == ${o}`}
389
17
  LIMIT 1
18
+ RETURN u`;try{return await f(n).query(u).then(i=>i.next()).then(i=>{if(i){const p=c?{verifiedPhone:!0,verifiedSmsCode:0}:{verifiedEmail:!0,verifiedEmailCode:0},m=U`UPDATE ${i._key} WITH ${p} IN users`;return f(n).query(m).then(()=>!0).catch(d=>(g({action:t,category:E,params:{code:o,type:s,value:r},value:T.DATABASE_ERROR},d,e),!1))}return!1})}catch(i){return g({action:t,category:E,params:{code:o,type:s,value:r},value:T.DATABASE_ERROR},i,e),!1}},Ce=(e,o)=>{const s="deleteUser",{databaseName:a}=e,{id:t}=D(o),n=U`LET u = DOCUMENT(${t})
390
19
  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")}
20
+ RETURN OLD`,c=new z(K.get("stripe.token"),{apiVersion:ae,typescript:!0});return f(a).query(n).then(r=>r.next()).then(r=>c.customers.del(r?.stripeCustomerId).then(()=>c.accounts.del(r?.stripeAccountId)).then(()=>r)).catch(r=>g({action:s,category:E,params:{id:t},value:T.DATABASE_ERROR},r,e))},Se=(e,o)=>{const s="delete",{databaseName:a}=e,{id:t}=D(o),c=U`UPDATE ${t} WITH ${{userAccess:0}} IN users LIMIT 1 RETURN NEW`;return f(a).query(c).then(r=>r.next()).catch(r=>{throw g({action:s,category:E,params:{id:t},value:T.DATABASE_ERROR},r,e)})},we=e=>{const{first:o,last:s,name:a="",username:t=""}=e,n=[o,s].join(" ").trim();return a||(n!==""?n:t||"Unknown")},qe=e=>{const o="getSessionUser",{databaseName:s,fields:a,session:{userId:t,username:n}={}}=e,{objects:c,queries:r}=C(a),u=`LET u = DOCUMENT("${x(`users/${t}`)}")
21
+ ${r.join(`
22
+ `)}
23
+ RETURN MERGE(u, {${c.join(", ")}})`;return f(s).query(u).then(i=>i.next()).catch(i=>{throw g({action:o,category:E,params:{userId:t,username:n},value:T.DATABASE_ERROR},i,e)})},De=(e,o)=>{const s="getUser",{email:a,id:t,phone:n,userId:c,username:r}=D(o),{databaseName:l,fields:u}=e,{objects:i,queries:p}=C(u);let m="";return t?m=`LET u = DOCUMENT("${t}")
24
+ ${p.join(`
25
+ `)}
456
26
  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}
27
+ RETURN MERGE(u, {${i.join(", ")}})`:r?m=`FOR u IN users
28
+ FILTER u.username == "${r}"
29
+ ${p.join(`
30
+ `)}
31
+ RETURN MERGE(u, {${i.join(", ")}})`:a?m=`FOR u IN users
32
+ FILTER u.email == "${a}"
33
+ ${p.join(`
34
+ `)}
35
+ RETURN MERGE(u, {${i.join(", ")}})`:n&&(m=`FOR u IN users
36
+ FILTER u.phone == "${n}"
37
+ ${p.join(`
38
+ `)}
39
+ RETURN MERGE(u, {${i.join(", ")}})`),f(l).query(m).then(d=>d.next()).then(d=>d).catch(d=>g({action:s,category:E,params:{id:t,userId:c,username:r},value:T.DATABASE_ERROR},d,e))},ve=(e,o)=>{const s="getUserList",{databaseName:a,fields:t}=e,{limit:n,username:c}=v(o),{objects:r,queries:l}=C(t),u=["u.userAccess > 0"];c&&u.push(`CONTAINS(u.username, "${w(c)}")`);const i=`FOR u IN users
40
+ FILTER ${u.join(" && ")}
41
+ ${l.join(`
42
+ `)}
43
+ ${n.aql}
484
44
  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
45
+ RETURN MERGE(u, {${r.join(", ")}})`;return f(a).query(i).then(p=>p.all()).catch(p=>(g({action:s,category:E,value:T.DATABASE_ERROR},p,e),[]))},be=(e,{reactions:o=[],username:s},a)=>{const t="getUsersByReactions",{databaseName:n,fields:c,session:{userId:r}={}}=e,l=o.map(R=>j(R,32).toLowerCase()),{limit:u}=v(a),{objects:i,queries:p}=C(c),m=`users/${r}`,d=w(s),y=["u.userAccess > 0",`POSITION(${JSON.stringify(l)}, LOWER(r.name))`];s&&y.push(`CONTAINS(u.username, "${d}")`);const I=`FOR u, r IN OUTBOUND "${m}" hasReaction
511
46
  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))
47
+ ${p.join(`
48
+ `)}
49
+ FILTER ${y.join(" && ")}
50
+ ${u.aql}
51
+ RETURN MERGE(u, {${i.join(", ")}})`;return f(n).query(I).then(R=>R.all()).catch(R=>(g({action:t,category:E,value:T.DATABASE_ERROR},R,e),[]))},Pe=(e,{tags:o,username:s},a)=>{const t="getUsersByTags",{databaseName:n,fields:c,session:{userId:r}={}}=e,l=o?.reduce((I,R)=>(R&&I.push(j(R,32).toLowerCase()),I),[]),{limit:u}=v(a),{objects:i,queries:p}=C(c),m=w(s),d=[`u._key != "${r}"`,"u.userAccess > 0"];s&&d.push(`CONTAINS(u.username, "${m}")`);const y=`FOR t IN tags
52
+ FILTER POSITION(${JSON.stringify(l)}, LOWER(t.name))
546
53
  FOR u, it IN OUTBOUND t isTagged
547
54
  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")}
55
+ ${p.join(`
56
+ `)}
57
+ FILTER ${d.join(" && ")}
58
+ ${u.aql}
59
+ RETURN DISTINCT MERGE(u, {${i.join(", ")}})`;return f(n).query(y).then(I=>I.all()).catch(I=>(g({action:t,category:E,value:T.DATABASE_ERROR},I,e),[]))},Le=(e,{username:o},s)=>{const a="getUsersByLatest",{databaseName:t,fields:n,session:{userId:c}={}}=e,{limit:r}=v(s),l=["u._id != session._id","u.userAccess > 0"],{objects:u,queries:i}=C(n);o&&l.push(`CONTAINS(u.username, "${w(o)}")`);const p=`FOR u IN users
60
+ LET session = DOCUMENT("users/${c}")
61
+ FILTER ${l.join(" && ")}
62
+ ${i.join(`
63
+ `)}
577
64
  LET distance = DISTANCE(u.latitude || 0, u.longitude || 0, session.latitude || 0, session.longitude || 0)
578
- ${limit.aql}
65
+ ${r.aql}
579
66
  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}")
67
+ RETURN MERGE(u, {${u.join(", ")}})`;return f(t).query(p).then(m=>m.all()).catch(m=>(g({action:a,category:E,value:T.DATABASE_ERROR},m,e),[]))},xe=(e,{userId:o},s)=>{const a="getUsersByConnection",{databaseName:t,fields:n}=e,{limit:c,username:r}=v(s),{objects:l,queries:u}=C(n),i=x(`users/${o}`),p=["u.userAccess > 0"];r&&p.push(`CONTAINS(u.username, "${w(r)}")`);const m=`FOR cu IN users
68
+ LET session = DOCUMENT("${i}")
604
69
  FOR u, connection IN OUTBOUND cu hasConnection
605
70
  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(" || ")}
71
+ ${u.join(`
72
+ `)}
73
+ FILTER ${p.join(" && ")}
74
+ ${c.aql}
75
+ RETURN DISTINCT MERGE(u, {${l.join(", ")}})`;return f(t).query(m).then(d=>d.all()).catch(d=>(g({action:a,category:E,value:T.DATABASE_ERROR},d,e),[]))},je=({expires:e,token:o})=>{try{const{userId:s,username:a,userAccess:t}=B(o);return F(s,a,t,e)}catch(s){throw s}},_e=async(e,o)=>{const s="signIn",{databaseName:a}=e,{email:t,expires:n,password:c,phone:r,username:l}=o,u=_(t),i=w(l),p=M(c),m=Q(r),d=Y(n)||15;if(!i&&!u&&!m||!p)throw q({action:s,category:E,params:{username:l},value:T.INVALID_ARGUMENTS},e);const y=[];u&&y.push(`u.email == "${u}"`),m&&y.push(`u.phone == ${m}`),i&&y.push(`u.username == "${i}"`);const I=`FOR u IN users
76
+ FILTER ${y.join(" || ")}
657
77
  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
78
+ RETURN u`;let R;try{R=await f(a).query(I).then(N=>N.next())}catch(N){throw g({action:s,category:E,params:{username:i},value:T.DATABASE_ERROR},N,e)}if(!R)throw q({action:s,category:E,params:{username:l},value:T.INVALID_AUTHENTICATION},e);const{_key:A,password:h,salt:O,userAccess:$}=R,P=b(p,O);if(h!==P)throw q({action:s,category:E,params:{userAccess:$,userId:A,username:l},value:T.INVALID_AUTHENTICATION},e);try{return F(A||"",l||"",$,d)}catch(N){throw g({action:s,category:E,params:{userId:A,username:l},value:T.DATABASE_ERROR},N,e)}},Me=async e=>{const o="signOut",{databaseName:s,session:{userId:a,username:t}={}}=e,n=`users/${a}`,c={lastOnline:Date.now(),sessionId:null},r=U`LET u = DOCUMENT(${n})
79
+ UPDATE u WITH ${c} IN users
715
80
  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 = (
81
+ RETURN NEW`;try{await f(s).query(r).then(l=>l.next())}catch(l){return g({action:o,category:E,params:{userId:a,username:t},value:T.DATABASE_ERROR},l,e),!1}return!0},Qe=e=>{const o="getActiveUserCount",{databaseName:s}=e,a=U`LET docs = (
733
82
  FOR u IN users
734
83
  FILTER u.active == true
735
84
  RETURN u
736
85
  )
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"]
}

86
+ RETURN LENGTH(docs)`;return f(s).query(a).then(t=>t.next()).catch(t=>{throw g({action:o,category:E,value:T.DATABASE_ERROR},t,e)})},Be=(e,o)=>{const s="getUserByToken",{databaseName:a}=e,{userId:t}=B(o),n=x(`users/${t}`),c=U`LET u = DOCUMENT("${n}") RETURN u`;return f(a).query(c).then(r=>r.next()).catch(r=>{throw g({action:s,category:E,params:{userId:t},value:T.DATABASE_ERROR},r,e)})};export{ie as UserAccess,Ue as addUser,$e as confirmCode,F as createToken,Se as deactivateUser,Ce as deleteUser,Ne as forgotPassword,Qe as getActiveUserCount,we as getDisplayName,qe as getSessionUser,De as getUser,Be as getUserByToken,C as getUserOptional,ve as getUsers,xe as getUsersByConnection,Le as getUsersByLatest,be as getUsersByReactions,Pe as getUsersByTags,v as parseUserOptions,je as refreshSession,Oe as resetPassword,_e as signIn,Me as signOut,he as updateUser};
87
+ //# 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 {parseNum} from '@nlabs/utils/parsers/numbers';\nimport {\n  createHash,\n  createPassword,\n  parseArangoId,\n  parseChar,\n  parseEmail,\n  parsePassword,\n  parsePhone,\n  parseUsername\n} from '@nlabs/utils/parsers/strings';\nimport {aql} from 'arangojs';\nimport {DateTime} from 'luxon';\nimport Stripe from 'stripe';\n\nimport {parseUser} from '../adapters/userAdapter.js';\nimport {Config} from '../config.js';\nimport {ErrorTypes, type SessionError} from '../types/error.types.js';\nimport {logError, logException} from '../utils/analyticsUtils.js';\nimport {getDocId, getLimit, selectReactionCountByType, useDb} from '../utils/arangodbUtils.js';\nimport {detectLanguage} from '../utils/languageDetection.js';\nimport {getSession, setSession, type SessionToken} from '../utils/sessionUtils.js';\nimport {sendEmail} from './email.js';\nimport {sendSms} from './sms.js';\n\nimport type {AqlQuery} from 'arangojs/aql';\nimport type {EdgeCollection} from 'arangojs/collections';\nimport type {ApiContext} from '../types/auth.types.js';\nimport type {UserInput, UserType} from '../types/users.types.js';\n\nconst eventCategory = 'users';\nconst STRIPE_API_VERSION = '2025-07-30.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 = 0,\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 | null> => {\n  const action = 'addUser';\n  const {databaseName, languageContext} = context;\n  const {confirm: _confirm, ...newUser} = user;\n  const {email, password, phone, username, userId, _key, _id, ...insertUser} = parseUser(newUser);\n  const hasPassword = !!password;\n  const hasUsername = !!username || !!phone || !!email;\n\n  if(!hasPassword || !hasUsername) {\n    logException({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.INVALID_ARGUMENTS\n    }, context);\n\n    return null;\n  }\n\n  const hashId = username || phone || email;\n  const salt: string = createHash(`${hashId}${password}`, '');\n  const encryptedPassword = createPassword(password, 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      throw 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    throw 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: 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),\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) => {\n      throw logError({\n        action,\n        category: eventCategory,\n        params: {username},\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n    });\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          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    throw 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 = {verifiedPhoneExpires: codeExpires, verifiedSmsCode: code};\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 = 0,\n            verifiedSmsCode,\n            verifiedPhoneExpires = 0\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    value\n  }: {\n    code: number;\n    type: 'sms' | 'email';\n    value: string;\n  }\n): Promise<boolean> => {\n  const action = 'confirmCode';\n  const {databaseName} = context;\n  const isSms = type === 'sms';\n  const formattedValue = isSms ? parsePhone(value) : parseEmail(value);\n  const filterByType = isSms\n    ? `FILTER u.phone == \"${formattedValue}\" && u.verifiedSmsCode == ${code}`\n    : `FILTER u.email == \"${formattedValue}\" && u.verifiedEmailCode == ${code}`;\n  const aqlQuery: string = `FOR u IN users\n    ${filterByType}\n    LIMIT 1\n    RETURN u`;\n\n  try {\n    return await useDb(databaseName).query(aqlQuery)\n      .then((cursor) => cursor.next())\n      .then((user) => {\n        if(user) {\n          const updatedUser = isSms\n            ? {verifiedPhone: true, verifiedSmsCode: 0}\n            : {verifiedEmail: true, verifiedEmailCode: 0};\n          const aqlQuery: AqlQuery = aql`UPDATE ${user._key} WITH ${updatedUser} IN users`;\n\n          return useDb(databaseName).query(aqlQuery)\n            .then(() => true)\n            .catch((error: Error) => {\n              logError({\n                action,\n                category: eventCategory,\n                params: {code, type, value: formattedValue},\n                value: ErrorTypes.DATABASE_ERROR\n              }, error, context);\n\n              return false;\n            });\n        }\n\n        return false;\n      });\n  } catch(error) {\n    logError({\n      action,\n      category: eventCategory,\n      params: {code, type, value: formattedValue},\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) => {\n      throw logError({\n      action,\n      category: eventCategory,\n      params: {id},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n  });\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  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      throw logError({\n        action,\n        category: eventCategory,\n        params: {userId: sessionId, username},\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\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)\n    .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 | null> => {\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    throw logException({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.INVALID_ARGUMENTS\n    }, context);\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    throw logError({\n      action,\n      category: eventCategory,\n      params: {username: formatUsername},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n  }\n\n  if(!checkUser) {\n    throw logException({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.INVALID_AUTHENTICATION\n    }, context);\n  }\n\n  const {_key: userId, password: validPassword, salt, userAccess} = checkUser as UserType;\n  const authPassword: string = createPassword(formatPassword, salt);\n\n  console.log({\n    createPassword,\n    authPassword,\n    validPassword,\n    formatPassword,\n    salt\n  });\n\n  if(validPassword !== authPassword) {\n    throw logException({\n      action,\n      category: eventCategory,\n      params: {userAccess, userId, username},\n      value: ErrorTypes.INVALID_AUTHENTICATION\n    }, context);\n  }\n\n  try {\n    const token = createToken(userId || '', username || '', userAccess, formatExpires);\n\n    return token;\n  } catch(error) {\n    throw logError({\n      action,\n      category: eventCategory,\n      params: {userId, username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\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    logError({\n      action,\n      category: eventCategory,\n      params: {userId: sessionId, username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n\n    return false;\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) => {\n      throw logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n    });\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 = parseArangoId(`users/${userId}`);\n  const aqlQuery: AqlQuery = aql`LET u = DOCUMENT(\"${userDocId}\") RETURN u`;\n\n  return useDb(databaseName)\n    .query(aqlQuery)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => {\n      throw logError({\n        action,\n        category: eventCategory,\n        params: {userId},\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n    });\n};"],
  "mappings": "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",
  "names": ["parseNum", "createHash", "createPassword", "parseArangoId", "parseChar", "parseEmail", "parsePassword", "parsePhone", "parseUsername", "aql", "DateTime", "Stripe", "parseUser", "Config", "ErrorTypes", "logError", "logException", "getDocId", "getLimit", "selectReactionCountByType", "useDb", "detectLanguage", "getSession", "setSession", "sendEmail", "sendSms", "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", "_confirm", "newUser", "email", "password", "phone", "_key", "_id", "insertUser", "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", "formatPassword", "verifiedEmailExpires", "verifiedPhoneExpires", "confirmCode", "value", "isSms", "formattedValue", "deleteUser", "stripeClient", "deletedUser", "deactivateUser", "getDisplayName", "first", "last", "fullname", "getSessionUser", "sessionId", "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"]
}
