@nlabs/reaktor 0.10.5 → 0.10.7

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