@nlabs/reaktor 0.8.1 → 0.10.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (445) hide show
  1. package/.prettierrc.js +4 -0
  2. package/README.md +10 -1
  3. package/coverage/actions/groups.ts.html +1039 -0
  4. package/coverage/actions/images.ts.html +2500 -0
  5. package/coverage/actions/index.html +131 -0
  6. package/coverage/actions/tags.ts.html +1000 -0
  7. package/coverage/adapters/arangoAdapter.ts.html +151 -0
  8. package/coverage/adapters/index.html +146 -0
  9. package/coverage/adapters/reaktorAdapter.ts.html +127 -0
  10. package/coverage/adapters/tagAdapter.ts.html +160 -0
  11. package/coverage/base.css +224 -0
  12. package/coverage/block-navigation.js +87 -0
  13. package/coverage/clover.xml +6 -0
  14. package/coverage/coverage-final.json +1 -0
  15. package/coverage/favicon.png +0 -0
  16. package/coverage/index.html +221 -0
  17. package/coverage/lcov-report/base.css +224 -0
  18. package/coverage/lcov-report/block-navigation.js +87 -0
  19. package/coverage/lcov-report/favicon.png +0 -0
  20. package/coverage/lcov-report/index.html +101 -0
  21. package/coverage/lcov-report/prettify.css +1 -0
  22. package/coverage/lcov-report/prettify.js +2 -0
  23. package/coverage/lcov-report/sort-arrow-sprite.png +0 -0
  24. package/coverage/lcov-report/sorter.js +196 -0
  25. package/coverage/lcov.info +0 -0
  26. package/coverage/mocks/file.ts.html +118 -0
  27. package/coverage/mocks/group.ts.html +145 -0
  28. package/coverage/mocks/image.ts.html +136 -0
  29. package/coverage/mocks/index.html +146 -0
  30. package/coverage/mocks/post.ts.html +169 -0
  31. package/coverage/mocks/tag.ts.html +121 -0
  32. package/coverage/mocks/user.ts.html +268 -0
  33. package/coverage/prettify.css +1 -0
  34. package/coverage/prettify.js +2 -0
  35. package/coverage/sort-arrow-sprite.png +0 -0
  36. package/coverage/sorter.js +196 -0
  37. package/coverage/types/error.ts.html +145 -0
  38. package/coverage/types/index.html +116 -0
  39. package/coverage/utils/adapterUtils.ts.html +151 -0
  40. package/coverage/utils/analyticsUtils.ts.html +292 -0
  41. package/coverage/utils/arangodbUtils.ts.html +463 -0
  42. package/coverage/utils/index.html +146 -0
  43. package/dist/actions/apps.js +242 -0
  44. package/dist/actions/connections.js +90 -0
  45. package/dist/actions/conversations.js +350 -0
  46. package/dist/actions/dynamodb.js +150 -0
  47. package/dist/actions/email.js +152 -0
  48. package/dist/actions/files.js +283 -0
  49. package/dist/actions/groups.js +292 -0
  50. package/dist/actions/images.js +735 -0
  51. package/dist/actions/index.js +66 -0
  52. package/dist/actions/ios.js +164 -0
  53. package/dist/actions/locations.js +122 -0
  54. package/dist/actions/messages.js +208 -0
  55. package/dist/actions/notifications.js +59 -0
  56. package/dist/actions/payments.js +497 -0
  57. package/dist/actions/personas.js +110 -0
  58. package/dist/actions/posts.js +595 -0
  59. package/dist/actions/reactions.js +322 -0
  60. package/dist/actions/s3.js +133 -0
  61. package/dist/actions/search.js +90 -0
  62. package/dist/actions/sms.js +108 -0
  63. package/dist/actions/statistics.js +62 -0
  64. package/dist/actions/subscription.js +220 -0
  65. package/dist/actions/tags.js +292 -0
  66. package/dist/actions/users.js +784 -0
  67. package/dist/actions/websockets.js +174 -0
  68. package/dist/adapters/arangoAdapter.js +46 -0
  69. package/dist/adapters/fileAdapter.js +76 -0
  70. package/dist/adapters/imageAdapter.js +40 -0
  71. package/dist/adapters/messageAdapter.js +49 -0
  72. package/dist/adapters/postAdapter.js +70 -0
  73. package/dist/adapters/reaktorAdapter.js +44 -0
  74. package/dist/adapters/tagAdapter.js +50 -0
  75. package/dist/adapters/userAdapter.js +115 -0
  76. package/dist/config.js +125 -0
  77. package/dist/index.js +66 -0
  78. package/dist/lambdas/actions/websockets.js +132 -0
  79. package/dist/lambdas/authorizer.js +67 -0
  80. package/dist/lambdas/connection.js +91 -0
  81. package/dist/lambdas/utils/message.js +42 -0
  82. package/dist/lambdas/utils/websocket.js +105 -0
  83. package/dist/mocks/conversation.js +35 -0
  84. package/dist/mocks/file.js +38 -0
  85. package/dist/mocks/group.js +47 -0
  86. package/dist/mocks/image.js +44 -0
  87. package/dist/mocks/nlabs.png +0 -0
  88. package/dist/mocks/post.js +55 -0
  89. package/dist/mocks/tag.js +37 -0
  90. package/dist/mocks/user.js +88 -0
  91. package/dist/mutations/index.js +26 -0
  92. package/dist/mutations/locations.js +44 -0
  93. package/dist/mutations/messages.js +86 -0
  94. package/dist/mutations/personas.js +100 -0
  95. package/dist/mutations/posts.js +53 -0
  96. package/dist/mutations/reactions.js +51 -0
  97. package/dist/mutations/statistics.js +39 -0
  98. package/dist/mutations/subscriptions.js +56 -0
  99. package/dist/mutations/tags.js +120 -0
  100. package/dist/mutations/users.js +116 -0
  101. package/dist/objectTypes/app.js +173 -0
  102. package/dist/objectTypes/bankAccount.js +76 -0
  103. package/dist/objectTypes/connection.js +48 -0
  104. package/dist/objectTypes/conversation.js +77 -0
  105. package/dist/objectTypes/creditCard.js +86 -0
  106. package/dist/objectTypes/document.js +46 -0
  107. package/dist/objectTypes/error.js +46 -0
  108. package/dist/objectTypes/external.js +74 -0
  109. package/dist/objectTypes/file.js +100 -0
  110. package/dist/objectTypes/filter.js +43 -0
  111. package/dist/objectTypes/group.js +123 -0
  112. package/dist/objectTypes/iapSubscription.js +40 -0
  113. package/dist/objectTypes/image.js +129 -0
  114. package/dist/objectTypes/index.js +68 -0
  115. package/dist/objectTypes/location.js +109 -0
  116. package/dist/objectTypes/message.js +96 -0
  117. package/dist/objectTypes/passcode.js +42 -0
  118. package/dist/objectTypes/persona.js +87 -0
  119. package/dist/objectTypes/plan.js +95 -0
  120. package/dist/objectTypes/post.js +125 -0
  121. package/dist/objectTypes/reaction.js +61 -0
  122. package/dist/objectTypes/relation.js +49 -0
  123. package/dist/objectTypes/search.js +72 -0
  124. package/dist/objectTypes/statistics.js +39 -0
  125. package/dist/objectTypes/subscription.js +117 -0
  126. package/dist/objectTypes/tag.js +65 -0
  127. package/dist/objectTypes/user.js +144 -0
  128. package/dist/queries/index.js +33 -0
  129. package/dist/queries/locations.js +45 -0
  130. package/dist/queries/messages.js +52 -0
  131. package/dist/queries/posts.js +154 -0
  132. package/dist/queries/reactions.js +56 -0
  133. package/dist/queries/statistics.js +39 -0
  134. package/dist/queries/subscriptions.js +44 -0
  135. package/dist/queries/tags.js +75 -0
  136. package/dist/queries/users.js +64 -0
  137. package/dist/templates/email/layout.js +302 -0
  138. package/dist/templates/email/passwordForgot.js +38 -0
  139. package/dist/templates/email/passwordRecovery.js +35 -0
  140. package/dist/templates/email/verifyEmail.js +38 -0
  141. package/dist/templates/email/welcome.js +38 -0
  142. package/dist/templates/sms/passwordForgot.js +24 -0
  143. package/dist/templates/sms/passwordRecovery.js +24 -0
  144. package/dist/templates/sms/verifyEmail.js +24 -0
  145. package/dist/templates/sms/verifyPhone.js +24 -0
  146. package/dist/templates/sms/welcome.js +24 -0
  147. package/dist/types/apps.js +32 -0
  148. package/dist/types/arangodb.js +16 -0
  149. package/{lib → dist}/types/auth.js +1 -1
  150. package/{lib → dist}/types/connections.js +1 -1
  151. package/{lib → dist}/types/conversations.js +1 -1
  152. package/{lib → dist}/types/email.js +1 -1
  153. package/dist/types/error.js +44 -0
  154. package/{lib → dist}/types/files.js +1 -1
  155. package/dist/types/google.js +16 -0
  156. package/{lib → dist}/types/groups.js +1 -1
  157. package/dist/types/images.js +16 -0
  158. package/dist/types/index.js +60 -0
  159. package/{lib → dist}/types/locations.js +1 -1
  160. package/dist/types/messages.js +16 -0
  161. package/{lib → dist}/types/notifications.js +1 -1
  162. package/dist/types/payments.js +16 -0
  163. package/dist/types/personas.js +16 -0
  164. package/dist/types/posts.js +16 -0
  165. package/{lib → dist}/types/statistics.js +1 -1
  166. package/{lib → dist}/types/tags.js +1 -1
  167. package/dist/types/users.js +16 -0
  168. package/dist/types/websockets.js +16 -0
  169. package/dist/utils/adapterUtils.js +45 -0
  170. package/dist/utils/analyticsUtils.js +72 -0
  171. package/dist/utils/arangodbUtils.js +165 -0
  172. package/dist/utils/auth.js +57 -0
  173. package/dist/utils/index.js +30 -0
  174. package/dist/utils/session.js +60 -0
  175. package/jest.setup.js +0 -0
  176. package/jpg:- +0 -0
  177. package/lex.config.cjs +13 -0
  178. package/lib/actions/apps.d.ts +25 -0
  179. package/lib/actions/apps.js +242 -0
  180. package/lib/actions/connections.d.ts +4 -0
  181. package/lib/actions/connections.js +90 -0
  182. package/lib/actions/conversations.d.ts +12 -12
  183. package/lib/actions/conversations.js +147 -131
  184. package/lib/actions/dynamodb.d.ts +8 -8
  185. package/lib/actions/dynamodb.js +35 -32
  186. package/lib/actions/email.d.ts +3 -5
  187. package/lib/actions/email.js +33 -63
  188. package/lib/actions/files.d.ts +17 -14
  189. package/lib/actions/files.js +184 -202
  190. package/lib/actions/groups.d.ts +4 -4
  191. package/lib/actions/groups.js +47 -45
  192. package/lib/actions/images.d.ts +17 -13
  193. package/lib/actions/images.js +325 -264
  194. package/lib/actions/index.d.ts +3 -0
  195. package/lib/actions/index.js +7 -1
  196. package/lib/actions/ios.js +11 -10
  197. package/lib/actions/locations.d.ts +5 -2
  198. package/lib/actions/locations.js +41 -37
  199. package/lib/actions/messages.d.ts +4 -3
  200. package/lib/actions/messages.js +35 -32
  201. package/lib/actions/notifications.d.ts +2 -2
  202. package/lib/actions/notifications.js +1 -1
  203. package/lib/actions/payments.d.ts +2 -2
  204. package/lib/actions/payments.js +87 -83
  205. package/lib/actions/personas.d.ts +3 -0
  206. package/lib/actions/personas.js +110 -0
  207. package/lib/actions/posts.d.ts +11 -10
  208. package/lib/actions/posts.js +186 -152
  209. package/lib/actions/reactions.d.ts +5 -5
  210. package/lib/actions/reactions.js +30 -28
  211. package/lib/actions/s3.d.ts +7 -7
  212. package/lib/actions/s3.js +37 -32
  213. package/lib/actions/search.d.ts +3 -3
  214. package/lib/actions/search.js +13 -11
  215. package/lib/actions/sms.d.ts +9 -3
  216. package/lib/actions/sms.js +60 -34
  217. package/lib/actions/statistics.d.ts +3 -2
  218. package/lib/actions/statistics.js +21 -18
  219. package/lib/actions/subscription.d.ts +2 -2
  220. package/lib/actions/subscription.js +32 -39
  221. package/lib/actions/tags.d.ts +23 -20
  222. package/lib/actions/tags.js +161 -205
  223. package/lib/actions/users.d.ts +50 -22
  224. package/lib/actions/users.js +441 -217
  225. package/lib/actions/websockets.d.ts +19 -5
  226. package/lib/actions/websockets.js +89 -61
  227. package/lib/adapters/arangoAdapter.d.ts +2 -0
  228. package/lib/adapters/arangoAdapter.js +46 -0
  229. package/lib/adapters/fileAdapter.d.ts +3 -0
  230. package/lib/adapters/fileAdapter.js +76 -0
  231. package/lib/adapters/imageAdapter.d.ts +2 -0
  232. package/lib/adapters/imageAdapter.js +40 -0
  233. package/lib/adapters/messageAdapter.d.ts +2 -0
  234. package/lib/adapters/messageAdapter.js +49 -0
  235. package/lib/adapters/postAdapter.d.ts +2 -0
  236. package/lib/adapters/postAdapter.js +70 -0
  237. package/lib/adapters/reaktorAdapter.d.ts +6 -0
  238. package/lib/adapters/reaktorAdapter.js +44 -0
  239. package/lib/adapters/tagAdapter.d.ts +2 -0
  240. package/lib/adapters/tagAdapter.js +50 -0
  241. package/lib/adapters/userAdapter.d.ts +2 -0
  242. package/lib/adapters/userAdapter.js +115 -0
  243. package/lib/config.js +16 -17
  244. package/lib/index.d.ts +7 -0
  245. package/lib/index.js +44 -8
  246. package/lib/lambdas/actions/websockets.d.ts +7 -6
  247. package/lib/lambdas/actions/websockets.js +15 -11
  248. package/lib/lambdas/authorizer.js +4 -4
  249. package/lib/lambdas/connection.js +20 -20
  250. package/lib/lambdas/utils/message.js +1 -1
  251. package/lib/lambdas/utils/websocket.js +8 -7
  252. package/lib/mocks/conversation.d.ts +8 -0
  253. package/lib/mocks/conversation.js +35 -0
  254. package/lib/mocks/file.d.ts +11 -0
  255. package/lib/mocks/file.js +38 -0
  256. package/lib/mocks/group.d.ts +17 -0
  257. package/lib/mocks/group.js +47 -0
  258. package/lib/mocks/image.d.ts +3 -0
  259. package/lib/mocks/image.js +44 -0
  260. package/lib/mocks/nlabs.png +0 -0
  261. package/lib/mocks/post.d.ts +38 -0
  262. package/lib/mocks/post.js +55 -0
  263. package/lib/mocks/tag.d.ts +2 -0
  264. package/lib/mocks/tag.js +37 -0
  265. package/lib/mocks/user.d.ts +4 -0
  266. package/lib/mocks/user.js +88 -0
  267. package/lib/mutations/index.d.ts +3 -0
  268. package/lib/mutations/index.js +26 -0
  269. package/lib/mutations/locations.d.ts +2 -0
  270. package/lib/mutations/locations.js +44 -0
  271. package/lib/mutations/messages.d.ts +2 -0
  272. package/lib/mutations/messages.js +86 -0
  273. package/lib/mutations/personas.d.ts +2 -0
  274. package/lib/mutations/personas.js +100 -0
  275. package/lib/mutations/posts.d.ts +2 -0
  276. package/lib/mutations/posts.js +53 -0
  277. package/lib/mutations/reactions.d.ts +2 -0
  278. package/lib/mutations/reactions.js +51 -0
  279. package/lib/mutations/statistics.d.ts +2 -0
  280. package/lib/mutations/statistics.js +39 -0
  281. package/lib/mutations/subscriptions.d.ts +2 -0
  282. package/lib/mutations/subscriptions.js +56 -0
  283. package/lib/mutations/tags.d.ts +2 -0
  284. package/lib/mutations/tags.js +120 -0
  285. package/lib/mutations/users.d.ts +1 -0
  286. package/lib/mutations/users.js +116 -0
  287. package/lib/objectTypes/app.d.ts +3 -0
  288. package/lib/objectTypes/app.js +173 -0
  289. package/lib/objectTypes/bankAccount.d.ts +1 -0
  290. package/lib/objectTypes/bankAccount.js +76 -0
  291. package/lib/objectTypes/connection.d.ts +1 -0
  292. package/lib/objectTypes/connection.js +48 -0
  293. package/lib/objectTypes/conversation.d.ts +2 -0
  294. package/lib/objectTypes/conversation.js +77 -0
  295. package/lib/objectTypes/creditCard.d.ts +1 -0
  296. package/lib/objectTypes/creditCard.js +86 -0
  297. package/lib/objectTypes/document.d.ts +1 -0
  298. package/lib/objectTypes/document.js +46 -0
  299. package/lib/objectTypes/error.d.ts +1 -0
  300. package/lib/objectTypes/error.js +46 -0
  301. package/lib/objectTypes/external.d.ts +1 -0
  302. package/lib/objectTypes/external.js +74 -0
  303. package/lib/objectTypes/file.d.ts +2 -0
  304. package/lib/objectTypes/file.js +100 -0
  305. package/lib/objectTypes/filter.d.ts +1 -0
  306. package/lib/objectTypes/filter.js +43 -0
  307. package/lib/objectTypes/group.d.ts +3 -0
  308. package/lib/objectTypes/group.js +123 -0
  309. package/lib/objectTypes/iapSubscription.d.ts +1 -0
  310. package/lib/objectTypes/iapSubscription.js +40 -0
  311. package/lib/objectTypes/image.d.ts +2 -0
  312. package/lib/objectTypes/image.js +129 -0
  313. package/lib/objectTypes/index.d.ts +24 -0
  314. package/lib/objectTypes/index.js +68 -0
  315. package/lib/objectTypes/location.d.ts +2 -0
  316. package/lib/objectTypes/location.js +109 -0
  317. package/lib/objectTypes/message.d.ts +2 -0
  318. package/lib/objectTypes/message.js +96 -0
  319. package/lib/objectTypes/passcode.d.ts +1 -0
  320. package/lib/objectTypes/passcode.js +42 -0
  321. package/lib/objectTypes/persona.d.ts +3 -0
  322. package/lib/objectTypes/persona.js +87 -0
  323. package/lib/objectTypes/plan.d.ts +2 -0
  324. package/lib/objectTypes/plan.js +95 -0
  325. package/lib/objectTypes/post.d.ts +2 -0
  326. package/lib/objectTypes/post.js +125 -0
  327. package/lib/objectTypes/reaction.d.ts +2 -0
  328. package/lib/objectTypes/reaction.js +61 -0
  329. package/lib/objectTypes/relation.d.ts +1 -0
  330. package/lib/objectTypes/relation.js +49 -0
  331. package/lib/objectTypes/search.d.ts +1 -0
  332. package/lib/objectTypes/search.js +72 -0
  333. package/lib/objectTypes/statistics.d.ts +1 -0
  334. package/lib/objectTypes/statistics.js +39 -0
  335. package/lib/objectTypes/subscription.d.ts +2 -0
  336. package/lib/objectTypes/subscription.js +117 -0
  337. package/lib/objectTypes/tag.d.ts +2 -0
  338. package/lib/objectTypes/tag.js +65 -0
  339. package/lib/objectTypes/user.d.ts +4 -0
  340. package/lib/objectTypes/user.js +144 -0
  341. package/lib/queries/index.d.ts +3 -0
  342. package/lib/queries/index.js +33 -0
  343. package/lib/queries/locations.d.ts +2 -0
  344. package/lib/queries/locations.js +45 -0
  345. package/lib/queries/messages.d.ts +2 -0
  346. package/lib/queries/messages.js +52 -0
  347. package/lib/queries/posts.d.ts +2 -0
  348. package/lib/queries/posts.js +154 -0
  349. package/lib/queries/reactions.d.ts +2 -0
  350. package/lib/queries/reactions.js +56 -0
  351. package/lib/queries/statistics.d.ts +2 -0
  352. package/lib/queries/statistics.js +39 -0
  353. package/lib/queries/subscriptions.d.ts +2 -0
  354. package/lib/queries/subscriptions.js +44 -0
  355. package/lib/queries/tags.d.ts +2 -0
  356. package/lib/queries/tags.js +75 -0
  357. package/lib/queries/users.d.ts +1 -0
  358. package/lib/queries/users.js +64 -0
  359. package/lib/types/{apps.d.ts → apps.types.d.ts} +19 -17
  360. package/lib/types/apps.types.js +32 -0
  361. package/lib/types/arangodb.types.d.ts +34 -0
  362. package/lib/types/arangodb.types.js +16 -0
  363. package/lib/types/auth.types.d.ts +9 -0
  364. package/lib/types/auth.types.js +16 -0
  365. package/lib/types/connections.types.d.ts +5 -0
  366. package/lib/types/connections.types.js +16 -0
  367. package/lib/types/conversations.types.d.ts +27 -0
  368. package/lib/types/conversations.types.js +16 -0
  369. package/lib/types/email.types.d.ts +13 -0
  370. package/lib/types/email.types.js +16 -0
  371. package/lib/types/error.types.d.ts +20 -0
  372. package/lib/types/error.types.js +44 -0
  373. package/lib/types/{files.d.ts → files.types.d.ts} +9 -12
  374. package/lib/types/files.types.js +16 -0
  375. package/lib/types/google.types.d.ts +29 -0
  376. package/lib/types/google.types.js +16 -0
  377. package/lib/types/{groups.d.ts → groups.types.d.ts} +6 -10
  378. package/lib/types/groups.types.js +16 -0
  379. package/lib/types/images.types.d.ts +52 -0
  380. package/lib/types/images.types.js +16 -0
  381. package/lib/types/index.d.ts +20 -18
  382. package/lib/types/index.js +41 -37
  383. package/lib/types/{locations.d.ts → locations.types.d.ts} +4 -6
  384. package/lib/types/locations.types.js +16 -0
  385. package/lib/types/messages.types.d.ts +16 -0
  386. package/lib/types/messages.types.js +16 -0
  387. package/lib/types/notifications.types.d.ts +19 -0
  388. package/lib/types/notifications.types.js +16 -0
  389. package/lib/types/{payments.d.ts → payments.types.d.ts} +13 -19
  390. package/lib/types/payments.types.js +16 -0
  391. package/lib/types/personas.types.d.ts +32 -0
  392. package/lib/types/personas.types.js +16 -0
  393. package/lib/types/posts.types.d.ts +28 -0
  394. package/lib/types/posts.types.js +16 -0
  395. package/lib/types/statistics.types.d.ts +3 -0
  396. package/lib/types/statistics.types.js +16 -0
  397. package/lib/types/tags.types.d.ts +15 -0
  398. package/lib/types/tags.types.js +16 -0
  399. package/lib/types/{users.d.ts → users.types.d.ts} +21 -22
  400. package/lib/types/users.types.js +16 -0
  401. package/lib/types/{websocket.d.ts → websockets.types.d.ts} +7 -3
  402. package/lib/types/websockets.types.js +16 -0
  403. package/lib/utils/adapterUtils.d.ts +1 -0
  404. package/lib/utils/adapterUtils.js +45 -0
  405. package/lib/utils/analyticsUtils.d.ts +21 -0
  406. package/lib/utils/analyticsUtils.js +72 -0
  407. package/lib/utils/arangodbUtils.d.ts +66 -0
  408. package/lib/utils/arangodbUtils.js +165 -0
  409. package/lib/utils/auth.d.ts +19 -3
  410. package/lib/utils/auth.js +20 -30
  411. package/lib/utils/index.d.ts +3 -4
  412. package/lib/utils/index.js +7 -9
  413. package/lib/utils/session.d.ts +10 -10
  414. package/lib/utils/session.js +15 -3
  415. package/lib/utils/stripeUtils.d.ts +3 -0
  416. package/lib/utils/{graphql.js → stripeUtils.js} +12 -15
  417. package/package.json +38 -30
  418. package/lib/types/apps.js +0 -16
  419. package/lib/types/arangodb.d.ts +0 -17
  420. package/lib/types/arangodb.js +0 -16
  421. package/lib/types/auth.d.ts +0 -7
  422. package/lib/types/connections.d.ts +0 -8
  423. package/lib/types/conversations.d.ts +0 -16
  424. package/lib/types/email.d.ts +0 -12
  425. package/lib/types/google.d.ts +0 -27
  426. package/lib/types/google.js +0 -16
  427. package/lib/types/images.d.ts +0 -42
  428. package/lib/types/images.js +0 -16
  429. package/lib/types/messages.d.ts +0 -27
  430. package/lib/types/messages.js +0 -16
  431. package/lib/types/notifications.d.ts +0 -19
  432. package/lib/types/payments.js +0 -16
  433. package/lib/types/posts.d.ts +0 -47
  434. package/lib/types/posts.js +0 -16
  435. package/lib/types/statistics.d.ts +0 -3
  436. package/lib/types/tags.d.ts +0 -15
  437. package/lib/types/users.js +0 -16
  438. package/lib/types/websocket.js +0 -16
  439. package/lib/utils/analytics.d.ts +0 -14
  440. package/lib/utils/analytics.js +0 -88
  441. package/lib/utils/arangodb.d.ts +0 -9
  442. package/lib/utils/arangodb.js +0 -118
  443. package/lib/utils/graphql.d.ts +0 -1
  444. package/lib/utils/objects.d.ts +0 -3
  445. package/lib/utils/objects.js +0 -59
@@ -29,101 +29,110 @@ var tags_exports = {};
29
29
  __export(tags_exports, {
30
30
  addTag: () => addTag,
31
31
  addTagToItem: () => addTagToItem,
32
- createTag: () => createTag,
33
- createTagEdge: () => createTagEdge,
34
32
  deleteTag: () => deleteTag,
33
+ deleteTagFromEdge: () => deleteTagFromEdge,
35
34
  deleteTagFromItem: () => deleteTagFromItem,
36
35
  extractTags: () => extractTags,
37
36
  getTag: () => getTag,
38
37
  getTags: () => getTags,
39
38
  getTagsByItem: () => getTagsByItem,
40
- getTagsByOwner: () => getTagsByOwner,
41
- linkTags: () => linkTags,
42
- updateTag: () => updateTag
39
+ getTagsByName: () => getTagsByName,
40
+ updateTag: () => updateTag,
41
+ updateTagsInItem: () => updateTagsInItem
43
42
  });
44
43
  module.exports = __toCommonJS(tags_exports);
45
44
  var import_utils = require("@nlabs/utils");
46
45
  var import_arangojs = require("arangojs");
47
- var import_words = __toESM(require("lodash/words"));
48
- var import_utils2 = require("../utils");
46
+ var import_words = __toESM(require("lodash/words"), 1);
47
+ var import_tagAdapter = require("../adapters/tagAdapter");
48
+ var import_error = require("../types/error.types");
49
+ var import_analyticsUtils = require("../utils/analyticsUtils");
50
+ var import_arangodbUtils = require("../utils/arangodbUtils");
49
51
  const eventCategory = "tags";
50
- const getTags = (context, options) => {
52
+ const getTags = (context, options = {}) => {
51
53
  const action = "getList";
52
54
  const { database } = context;
53
- const { search = "", from = 0, to = 30 } = options;
54
- const limit = (0, import_utils2.getLimit)(from, to);
55
- const filter = search.length ? `FILTER LIKE(t.name, "%${search}%")` : "";
55
+ const { search = "", from = 0, to = 30, userId } = options;
56
+ const limit = (0, import_arangodbUtils.getLimit)(from, to);
57
+ const filters = [];
58
+ if (search) {
59
+ filters.push(`LIKE(t.name, "%${search}%")`);
60
+ }
61
+ if (userId) {
62
+ filters.push(`t.userId == "${userId}"`);
63
+ }
56
64
  const aqlQry = `FOR t IN tags
57
- ${filter}
65
+ FILTER ${filters.join(" || ")}
58
66
  ${limit.aql}
59
67
  SORT t.name
60
68
  RETURN t`;
61
- return database.query(aqlQry).then((cursor) => cursor.all()).then((list = []) => list).catch((error) => (0, import_utils2.logError)({
62
- action,
63
- category: eventCategory,
64
- label: "db_error"
65
- }, error, {}).then(() => null).catch((error2) => Promise.reject(error2)));
69
+ return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => {
70
+ (0, import_analyticsUtils.logError)({
71
+ action,
72
+ category: eventCategory,
73
+ value: import_error.ErrorTypes.DATABASE_ERROR
74
+ }, error, {});
75
+ return [];
76
+ });
66
77
  };
67
- const getTagsByItem = (context, itemId, args) => {
68
- const action = "getList";
69
- const { from = 0, to = 30 } = args;
78
+ const getTagsByItem = (context, itemDocId, options = {}) => {
79
+ const action = "getTagsByItem";
80
+ const { from = 0, to = 30 } = options;
70
81
  const { database } = context;
71
- const limit = (0, import_utils2.getLimit)(from, to);
72
- const formatItemId = (0, import_utils.parseArangoId)(itemId);
82
+ const limit = (0, import_arangodbUtils.getLimit)(from, to);
83
+ const formatItemId = (0, import_utils.parseArangoId)(itemDocId);
73
84
  const tagQuery = `FOR t, it IN INBOUND "${formatItemId}" isTagged
74
85
  ${limit.aql}
75
86
  SORT it.added
76
87
  RETURN t`;
77
- return database.query(tagQuery).then((cursor) => cursor.all() || []).catch((error) => (0, import_utils2.logError)({
78
- action,
79
- category: eventCategory,
80
- label: "db_error"
81
- }, error, {}).then(() => null).catch((error2) => Promise.reject(error2)));
82
- };
83
- const getTagsByOwner = (context, args) => {
84
- const action = "getList";
85
- const { from = 0, id, to = 30 } = args;
86
- const { database } = context;
87
- const limit = (0, import_utils2.getLimit)(from, to);
88
- const aqlQry = `FOR t IN tags
89
- FILTER t.userId == "${id}"
90
- ${limit.aql}
91
- SORT t.added
92
- RETURN t`;
93
- return database.query(aqlQry).then((cursor) => cursor.all()).then((list = []) => list).catch((error) => (0, import_utils2.logError)({
94
- action,
95
- category: eventCategory,
96
- label: "db_error"
97
- }, error, {}).then(() => null).catch((error2) => Promise.reject(error2)));
88
+ return database.query(tagQuery).then((cursor) => cursor.all()).catch((error) => {
89
+ (0, import_analyticsUtils.logError)({
90
+ action,
91
+ category: eventCategory,
92
+ value: import_error.ErrorTypes.DATABASE_ERROR
93
+ }, error, {});
94
+ return [];
95
+ });
98
96
  };
99
97
  const getTag = (context, tagId) => {
100
- const action = "getItem";
98
+ const action = "getTag";
101
99
  const { database } = context;
102
- const formatId = JSON.stringify((0, import_utils.parseId)(tagId));
100
+ const formatId = (0, import_utils.parseId)(tagId);
103
101
  const aqlQry = `FOR t IN tags
104
- FILTER t._key == ${formatId}
102
+ FILTER t._key == "${formatId}"
105
103
  LIMIT 1
106
104
  RETURN t`;
107
- return database.query(aqlQry).then((cursor) => cursor.next()).then((tag = {}) => tag).catch((error) => (0, import_utils2.logError)({
105
+ return database.query(aqlQry).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
108
106
  action,
109
107
  category: eventCategory,
110
- label: "db_error"
111
- }, error, {}).then(() => null).catch((error2) => Promise.reject(error2)));
108
+ value: import_error.ErrorTypes.DATABASE_ERROR
109
+ }, error, {}));
112
110
  };
113
- const addTag = (context, { tag }) => {
114
- const action = "addTag";
115
- const { database, session } = context;
116
- const { userAccess } = session;
117
- if (userAccess < 3) {
118
- return (0, import_utils2.logException)({
111
+ const getTagsByName = (context, tagNames) => {
112
+ const action = "getTagsByName";
113
+ const { database } = context;
114
+ const formatTagNames = tagNames.map((tagName) => (0, import_utils.parseVarChar)(tagName, 32)).filter((tagName) => !!tagName);
115
+ if (!formatTagNames.length) {
116
+ return Promise.resolve([]);
117
+ }
118
+ const aqlQry = import_arangojs.aql`FOR t IN tags
119
+ FILTER POSITION(${JSON.stringify(formatTagNames)}, t.name)
120
+ RETURN t`;
121
+ return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => {
122
+ (0, import_analyticsUtils.logError)({
119
123
  action,
120
124
  category: eventCategory,
121
- label: "db_error"
122
- }, session);
123
- }
125
+ value: import_error.ErrorTypes.DATABASE_ERROR
126
+ }, error, {});
127
+ return [];
128
+ });
129
+ };
130
+ const addTag = (context, tag) => {
131
+ const action = "addTag";
132
+ const { database } = context;
124
133
  const now = Date.now();
125
- const { description, name } = tag;
126
- const formatId = (0, import_utils.createHash)(`tag-${name.toLowercase()}`, null);
134
+ const { description, name } = (0, import_tagAdapter.parseTag)(tag);
135
+ const formatId = (0, import_utils.createHash)(`tag-${name.toLowerCase()}`, null);
127
136
  const insert = {
128
137
  _key: formatId,
129
138
  added: now,
@@ -132,205 +141,152 @@ const addTag = (context, { tag }) => {
132
141
  name: (0, import_utils.parseVarChar)(name, 32)
133
142
  };
134
143
  const aqlQry = import_arangojs.aql`INSERT ${insert} IN tags RETURN NEW`;
135
- return database.query(aqlQry).then((cursor) => cursor.next()).then((tag2 = {}) => tag2).catch((error) => (0, import_utils2.logError)({
144
+ return database.query(aqlQry).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
136
145
  action,
137
146
  category: eventCategory,
138
- label: "db_error"
139
- }, error, {}).then(() => null).catch((error2) => Promise.reject(error2)));
147
+ value: import_error.ErrorTypes.DATABASE_ERROR
148
+ }, error, {}));
140
149
  };
141
- const addTagToItem = async (context, { itemId, tagId }) => {
150
+ const addTagToItem = async (context, { itemDocId, tagBy, tagDocId }) => {
142
151
  const action = "addTagToItem";
143
- const { database } = context;
152
+ const { database, session: { userId: sessionId } } = context;
144
153
  const added = Date.now();
145
- const formatTagId = (0, import_utils.parseArangoId)(tagId);
146
- const formatItemId = (0, import_utils.parseArangoId)(itemId);
147
- const edgeId = (0, import_utils.createHash)(`isTagged-${formatTagId}-${formatItemId}`, null);
154
+ const formatTagDocId = (0, import_utils.parseArangoId)(tagDocId);
155
+ const formatItemDocId = (0, import_utils.parseArangoId)(itemDocId);
156
+ const edgeId = (0, import_utils.createHash)(`isTagged-${formatTagDocId}-${formatItemDocId}`, null);
148
157
  const edgeCollection = database.collection("isTagged");
149
- const edge = { _from: formatTagId, _key: edgeId, _to: formatItemId, added };
150
- console.log({ edge });
151
- await edgeCollection.save(edge, { overwriteMode: "ignore", silent: true }).catch((error) => (0, import_utils2.logError)({
158
+ const type = formatItemDocId.split("/")[0];
159
+ const formatTagBy = (0, import_utils.parseId)(tagBy) || sessionId;
160
+ const edge = { _from: formatTagDocId, _key: edgeId, _to: formatItemDocId, added, tagBy: formatTagBy, type };
161
+ await edgeCollection.save(edge, { overwriteMode: "ignore", returnNew: true, silent: true }).catch((error) => (0, import_analyticsUtils.logError)({
152
162
  action,
153
163
  category: eventCategory,
154
- label: "db_error",
155
164
  params: {
156
165
  edge
157
- }
158
- }, error, {}).then(() => null).catch((error2) => Promise.reject(error2)));
159
- const tagQuery = import_arangojs.aql`LET t = DOCUMENT(${formatTagId}) RETURN t`;
160
- return database.query(tagQuery).then((cursor) => cursor.next()).catch((error) => (0, import_utils2.logError)({
166
+ },
167
+ value: import_error.ErrorTypes.DATABASE_ERROR
168
+ }, error, {}));
169
+ const tagQuery = import_arangojs.aql`LET t = DOCUMENT(${formatTagDocId}) RETURN t`;
170
+ return database.query(tagQuery).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
161
171
  action,
162
172
  category: eventCategory,
163
- label: "db_error",
164
173
  params: {
165
174
  edge
166
- }
167
- }, error, {}).then(() => null).catch((error2) => Promise.reject(error2)));
175
+ },
176
+ value: import_error.ErrorTypes.DATABASE_ERROR
177
+ }, error, {}));
168
178
  };
169
- const updateTag = (context, item = {}) => {
170
- const action = "update";
171
- const { database, session } = context;
172
- const { userAccess } = session;
173
- if (userAccess < 3) {
174
- return (0, import_utils2.logException)({
175
- action,
176
- category: eventCategory,
177
- label: "db_error"
178
- }, session);
179
- }
179
+ const updateTag = (context, tag) => {
180
+ const action = "updateTag";
181
+ const { database, session: { userId: sessionId } } = context;
180
182
  const now = Date.now();
181
- const { description, id, name } = item;
183
+ const { description, name, tagId } = (0, import_tagAdapter.parseTag)(tag);
182
184
  const formatName = (0, import_utils.parseVarChar)(name, 32);
183
185
  const update = {
184
186
  description: (0, import_utils.parseVarChar)(description, 64),
185
187
  modified: now,
186
188
  name: formatName
187
189
  };
188
- const formatId = (0, import_utils.parseId)(id);
189
- const tagId = !formatId ? (0, import_utils.createHash)(`tag-${formatName.toLowerCase()}`, null) : formatId;
190
+ const newTagKey = tagId || (0, import_utils.createHash)(`tag-${formatName.toLowerCase()}`, null);
190
191
  const insert = {
191
192
  ...update,
192
- _key: tagId,
193
- added: now
193
+ _id: `tags/${newTagKey}`,
194
+ _key: newTagKey,
195
+ added: now,
196
+ tagBy: sessionId
194
197
  };
195
- const aqlQry = import_arangojs.aql`UPSERT {_key: ${tagId}}
198
+ const aqlQry = import_arangojs.aql`UPSERT {name: ${formatName}}
196
199
  INSERT ${insert}
197
200
  UPDATE ${update}
198
201
  IN tags RETURN NEW`;
199
- return database.query(aqlQry).then((cursor) => cursor.next()).then((tag = {}) => tag).catch((error) => (0, import_utils2.logError)({
202
+ return database.query(aqlQry).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
200
203
  action,
201
204
  category: eventCategory,
202
- label: "db_error"
203
- }, error, {}).then(() => null).catch((error2) => Promise.reject(error2)));
205
+ value: import_error.ErrorTypes.DATABASE_ERROR
206
+ }, error, {}));
204
207
  };
205
- const deleteTag = (context, { tagId }) => {
208
+ const deleteTag = async (context, tag) => {
206
209
  const action = "deleteTag";
207
- const { database, session } = context;
208
- const { userAccess } = session;
209
- if (userAccess < 3) {
210
- return (0, import_utils2.logException)({
211
- action,
212
- category: eventCategory,
213
- label: "db_error"
214
- }, session);
215
- }
216
- const formatTagId = (0, import_utils.parseId)(tagId);
217
- const aqlQry = import_arangojs.aql`FOR t IN tags
218
- FILTER t._key == ${formatTagId}
210
+ const { database } = context;
211
+ const formatItemId = (0, import_arangodbUtils.getDocId)("tags", tag);
212
+ const tagQuery = `FOR it FROM isTagged
213
+ FILTER it._from == "${formatItemId}" || it._to == "${formatItemId}"
214
+ REMOVE it IN isTagged
215
+ RETURN OLD`;
216
+ await database.query(tagQuery).then((cursor) => cursor.all()).catch((error) => (0, import_analyticsUtils.logError)({
217
+ action,
218
+ category: eventCategory,
219
+ value: import_error.ErrorTypes.DATABASE_ERROR
220
+ }, error, {}));
221
+ const aqlQry = `LET t = DOCUMENT(${formatItemId})
219
222
  REMOVE t IN tags
220
223
  RETURN OLD`;
221
- return database.query(aqlQry).then((cursor) => cursor.next()).then((tag = {}) => tag).catch((error) => (0, import_utils2.logError)({
224
+ return database.query(aqlQry).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
222
225
  action,
223
226
  category: eventCategory,
224
- label: "db_error"
225
- }, error, {}).then(() => null).catch((error2) => Promise.reject(error2)));
227
+ value: import_error.ErrorTypes.DATABASE_ERROR
228
+ }, error, {}));
226
229
  };
227
- const deleteTagFromItem = async (context, { itemId, tagId }) => {
228
- const action = "deleteTagFromItem";
230
+ const deleteTagFromEdge = async (context, { tagDocId, edgeDocId }) => {
231
+ const action = "deleteTagFromEdge";
229
232
  const { database } = context;
230
- const formatTagId = (0, import_utils.parseArangoId)(tagId);
231
- const formatItemId = (0, import_utils.parseArangoId)(itemId);
232
- const edgeId = (0, import_utils.createHash)(`isTagged-${formatTagId}-${formatItemId}`, null);
233
+ const formatTagDocId = (0, import_utils.parseArangoId)(tagDocId);
234
+ const formatEdgeDocId = (0, import_utils.parseArangoId)(edgeDocId);
233
235
  const edgeCollection = database.collection("isTagged");
234
- const edge = { _key: edgeId };
235
- await edgeCollection.remove(edge, { silent: true }).catch((error) => (0, import_utils2.logError)({
236
+ const edge = { _id: formatEdgeDocId };
237
+ await edgeCollection.remove(edge, { silent: true }).catch((error) => (0, import_analyticsUtils.logError)({
236
238
  action,
237
239
  category: eventCategory,
238
- label: "db_error"
239
- }, error, {}).then(() => null).catch((error2) => Promise.reject(error2)));
240
- const tagQuery = import_arangojs.aql`LET t = DOCUMENT(${tagId}) RETURN t`;
241
- return database.query(tagQuery).then((cursor) => cursor.next()).catch((error) => (0, import_utils2.logError)({
240
+ value: import_error.ErrorTypes.DATABASE_ERROR
241
+ }, error, {}));
242
+ const tagQuery = import_arangojs.aql`LET t = DOCUMENT(${formatTagDocId}) RETURN t`;
243
+ return database.query(tagQuery).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
242
244
  action,
243
245
  category: eventCategory,
244
- label: "db_error"
245
- }, error, {}).then(() => null).catch((error2) => Promise.reject(error2)));
246
+ value: import_error.ErrorTypes.DATABASE_ERROR
247
+ }, error, {}));
246
248
  };
247
- const createTag = (db, tagName) => {
248
- const tagId = (0, import_utils.createHash)(`tag-${tagName.toLowerCase()}`, null);
249
- const insert = {
250
- _key: tagId,
251
- added: Date.now(),
252
- name: tagName
253
- };
254
- const aqlQry = import_arangojs.aql`UPSERT {_key: ${tagId}}
255
- INSERT ${insert}
256
- UPDATE {}
257
- IN tags RETURN NEW`;
258
- return db.query(aqlQry).then((cursor) => cursor.next()).then((tag = {}) => tag).catch((error) => {
259
- throw error;
260
- });
261
- };
262
- const createTagEdge = (db, tag, itemId) => {
263
- const edgeCollection = db.collection("isTagged");
264
- const { _id: tagId } = tag;
265
- const edgeId = (0, import_utils.createHash)(`isTagged-${tagId}-${itemId}`);
266
- const type = itemId.split("/")[0];
267
- const edge = {
268
- _from: tagId,
269
- _key: edgeId,
270
- _to: itemId,
271
- added: Date.now(),
272
- type
273
- };
274
- return edgeCollection.save(edge, { returnNew: true }).then(() => tag);
275
- };
276
- const linkTags = async (db, tagNames, itemId) => {
277
- const edgeCollection = db.collection("isTagged");
278
- const edges = await edgeCollection.inEdges(itemId, {});
279
- console.log({ edges });
280
- if (edges.length) {
281
- await Promise.all(
282
- edges.map((edge) => {
283
- const { _key: edgeKey } = edge;
284
- const aqlQry = import_arangojs.aql`REMOVE {_key:${edgeKey}} IN isTagged`;
285
- return db.query(aqlQry);
286
- })
287
- );
288
- }
289
- return Promise.all(
290
- tagNames.map((tagName) => createTag(db, tagName).then((tag) => createTagEdge(db, tag, itemId)))
291
- );
249
+ const deleteTagFromItem = async (context, { tagId, itemId, itemType }) => {
250
+ const action = "deleteTagFromItem";
251
+ const { database } = context;
252
+ const formatTagDocId = (0, import_utils.parseArangoId)(`tags/${tagId}`);
253
+ const formatItemDocId = (0, import_utils.parseArangoId)(`${itemType}/${itemId}`);
254
+ const aqlQuery = `FOR it FROM isTagged
255
+ FILTER it._from == "${formatTagDocId}" || it._to == "${formatItemDocId}"
256
+ REMOVE it IN isTagged`;
257
+ return database.query(aqlQuery).then(() => true).catch((error) => (0, import_analyticsUtils.logError)({
258
+ action,
259
+ category: eventCategory,
260
+ value: import_error.ErrorTypes.DATABASE_ERROR
261
+ }, error, {}));
292
262
  };
293
- const extractTags = (db, itemId, content) => {
294
- const tags = (0, import_words.default)(content, /#[a-zA-Z\d-]+/g).map((tag) => (0, import_utils.parseId)(tag));
295
- const edgeCollection = db.collection("isTagged");
296
- return edgeCollection.inEdges(itemId, {}).then((edges) => {
297
- if (edges.length) {
298
- return Promise.all(
299
- edges.map((edge) => {
300
- const { _key: edgeKey } = edge;
301
- const aqlQry = import_arangojs.aql`REMOVE {_key:${edgeKey}} IN isTagged`;
302
- return db.query(aqlQry).catch((error) => {
303
- throw error;
304
- });
305
- })
306
- ).then(() => {
307
- if (tags.length) {
308
- return linkTags(db, tags, itemId);
309
- }
310
- return [];
311
- });
312
- } else if (tags.length) {
313
- return linkTags(db, tags, itemId);
314
- }
315
- return [];
316
- }).catch((error) => {
317
- throw error;
318
- });
263
+ const extractTags = (content) => (0, import_words.default)(content, /#[a-zA-Z\d-]+/g).map((tag) => (0, import_utils.parseId)(tag));
264
+ const updateTagsInItem = (context, { itemDocId, tags }) => {
265
+ const action = "updateTagsInItem";
266
+ const { database } = context;
267
+ const edgeCollection = database.collection("isTagged");
268
+ return edgeCollection.inEdges(itemDocId, {}).then((edges) => Promise.all([
269
+ ...edges.filter(({ name: existingTagName }) => !tags.find(({ name: tagName }) => existingTagName === tagName)).map(({ _id: edgeDocId, _from: tagDocId }) => deleteTagFromEdge(context, { edgeDocId, tagDocId })),
270
+ ...tags.filter(({ name: tagName }) => !edges.find(({ name: existingTagName }) => existingTagName === tagName)).map(({ _id: tagDocId, tagBy }) => addTagToItem(context, { itemDocId, tagBy, tagDocId }))
271
+ ]).then(() => tags)).catch((error) => (0, import_analyticsUtils.logError)({
272
+ action,
273
+ category: eventCategory,
274
+ value: import_error.ErrorTypes.DATABASE_ERROR
275
+ }, error, {}));
319
276
  };
320
277
  // Annotate the CommonJS export names for ESM import in node:
321
278
  0 && (module.exports = {
322
279
  addTag,
323
280
  addTagToItem,
324
- createTag,
325
- createTagEdge,
326
281
  deleteTag,
282
+ deleteTagFromEdge,
327
283
  deleteTagFromItem,
328
284
  extractTags,
329
285
  getTag,
330
286
  getTags,
331
287
  getTagsByItem,
332
- getTagsByOwner,
333
- linkTags,
334
- updateTag
288
+ getTagsByName,
289
+ updateTag,
290
+ updateTagsInItem
335
291
  });
336
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/tags.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 {createHash, parseArangoId, parseId, parseVarChar} from '@nlabs/utils';\nimport {aql, Database} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {EdgeCollection} from 'arangojs/collection';\nimport {ArrayCursor} from 'arangojs/cursor';\nimport words from 'lodash/words';\n\nimport {ArangoDBLimit} from '../types/arangodb';\nimport {ApiContext} from '../types/auth';\nimport {TagType} from '../types/tags';\nimport {getLimit, logError, logException} from '../utils';\n\nconst eventCategory: string = 'tags';\nexport interface TagOptions {\n  readonly search?: string;\n  readonly from?: number;\n  readonly to?: number;\n}\n\nexport const getTags = (\n  context: ApiContext,\n  options?: TagOptions\n): Promise<TagType[]> => {\n  const action: string = 'getList';\n  const {database} = context;\n  const {search = '', from = 0, to = 30} = options;\n  const limit: ArangoDBLimit = getLimit(from, to);\n  const filter: string = search.length ? `FILTER LIKE(t.name, \"%${search}%\")` : '';\n  const aqlQry: string = `FOR t IN tags\n    ${filter}\n    ${limit.aql}\n    SORT t.name\n    RETURN t`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .then((list = []) => list)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, {}).then(() => null).catch((error) => Promise.reject(error)));\n};\n\nexport const getTagsByItem = (context: ApiContext, itemId: string,  args): Promise<TagType[]> => {\n  const action: string = 'getList';\n  const {from = 0, to = 30} = args;\n  const {database} = context;\n  const limit: ArangoDBLimit = getLimit(from, to);\n  const formatItemId: string = parseArangoId(itemId);\n\n  const tagQuery: string = `FOR t, it IN INBOUND \"${formatItemId}\" isTagged\n    ${limit.aql}\n    SORT it.added\n    RETURN t`;\n\n  return database.query(tagQuery)\n    .then((cursor: ArrayCursor) => cursor.all() || [])\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, {}).then(() => null).catch((error) => Promise.reject(error)));\n};\n\nexport const getTagsByOwner = (context: ApiContext, args): Promise<TagType[]> => {\n  const action: string = 'getList';\n  const {from = 0, id, to = 30} = args;\n  const {database} = context;\n  const limit: ArangoDBLimit = getLimit(from, to);\n  const aqlQry: string = `FOR t IN tags\n    FILTER t.userId == \"${id}\"\n    ${limit.aql}\n    SORT t.added\n    RETURN t`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .then((list = []) => list)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, {}).then(() => null).catch((error) => Promise.reject(error)));\n};\n\nexport const getTag = (context: ApiContext, tagId: string): Promise<TagType> => {\n  const action: string = 'getItem';\n  const {database} = context;\n  const formatId: string = JSON.stringify(parseId(tagId));\n  const aqlQry: string = `FOR t IN tags\n    FILTER t._key == ${formatId}\n    LIMIT 1\n    RETURN t`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((tag = {}) => tag)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, {}).then(() => null).catch((error) => Promise.reject(error)));\n};\n\nexport const addTag = (context: ApiContext, {tag}): Promise<TagType> => {\n  const action: string = 'addTag';\n  const {database, session} = context;\n  const {userAccess} = session;\n\n  if(userAccess < 3) {\n    return logException({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, session);\n  }\n\n  const now: number = Date.now();\n  const {description, name} = tag;\n  const formatId: string = createHash(`tag-${name.toLowercase()}`, null);\n\n  const insert: any = {\n    _key: formatId,\n    added: now,\n    description: parseVarChar(description, 64),\n    modified: now,\n    name: parseVarChar(name, 32)\n  };\n\n  const aqlQry: AqlQuery = aql`INSERT ${insert} IN tags RETURN NEW`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((tag: TagType = {}) => tag)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, {}).then(() => null).catch((error) => Promise.reject(error)));\n};\n\nexport const addTagToItem = async (context: ApiContext, {itemId, tagId}): Promise<TagType> => {\n  const action: string = 'addTagToItem';\n  const {database} = context;\n  const added: number = Date.now();\n  const formatTagId: string = parseArangoId(tagId);\n  const formatItemId: string = parseArangoId(itemId);\n  const edgeId: string = createHash(`isTagged-${formatTagId}-${formatItemId}`, null);\n  const edgeCollection: EdgeCollection = database.collection('isTagged');\n  const edge: any = {_from: formatTagId, _key: edgeId, _to: formatItemId, added};\n  console.log({edge});\n  await edgeCollection.save(edge, {overwriteMode: 'ignore', silent: true})\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error',\n      params: {\n        edge\n      }\n    }, error, {}).then(() => null).catch((error) => Promise.reject(error)));\n\n\n  const tagQuery: AqlQuery = aql`LET t = DOCUMENT(${formatTagId}) RETURN t`;\n\n  return database.query(tagQuery)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error',\n      params: {\n        edge\n      }\n    }, error, {}).then(() => null).catch((error) => Promise.reject(error)));\n};\n\nexport const updateTag = (context: ApiContext, item: TagType = {}): Promise<TagType> => {\n  const action: string = 'update';\n  const {database, session} = context;\n  const {userAccess} = session;\n\n  if(userAccess < 3) {\n    return logException({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, session);\n  }\n\n  const now: number = Date.now();\n  const {description, id, name} = item;\n  const formatName: string = parseVarChar(name, 32);\n\n  const update: TagType = {\n    description: parseVarChar(description, 64),\n    modified: now,\n    name: formatName\n  };\n\n  const formatId: string = parseId(id);\n  const tagId: string = !formatId ? createHash(`tag-${formatName.toLowerCase()}`, null) : formatId;\n\n  const insert: TagType = {\n    ...update,\n    _key: tagId,\n    added: now\n  };\n  const aqlQry: AqlQuery = aql`UPSERT {_key: ${tagId}}\n    INSERT ${insert}\n    UPDATE ${update}\n    IN tags RETURN NEW`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((tag = {}) => tag)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, {}).then(() => null).catch((error) => Promise.reject(error)));\n};\n\nexport const deleteTag = (context: ApiContext, {tagId}): Promise<TagType> => {\n  const action: string = 'deleteTag';\n  const {database, session} = context;\n  const {userAccess} = session;\n\n  if(userAccess < 3) {\n    return logException({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, session);\n  }\n\n  const formatTagId: string = parseId(tagId);\n  const aqlQry: AqlQuery = aql`FOR t IN tags\n    FILTER t._key == ${formatTagId}\n    REMOVE t IN tags\n    RETURN OLD`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((tag: TagType = {}) => tag)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, {}).then(() => null).catch((error) => Promise.reject(error)));\n};\n\nexport const deleteTagFromItem = async (context: ApiContext, {itemId, tagId}): Promise<TagType> => {\n  const action: string = 'deleteTagFromItem';\n  const {database} = context;\n  const formatTagId: string = parseArangoId(tagId);\n  const formatItemId: string = parseArangoId(itemId);\n  const edgeId: string = createHash(`isTagged-${formatTagId}-${formatItemId}`, null);\n  const edgeCollection: EdgeCollection = database.collection('isTagged');\n  const edge: any = {_key: edgeId};\n\n  await edgeCollection.remove(edge, {silent: true})\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, {}).then(() => null).catch((error) => Promise.reject(error)));\n\n  const tagQuery: AqlQuery = aql`LET t = DOCUMENT(${tagId}) RETURN t`;\n\n  return database.query(tagQuery)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, {}).then(() => null).catch((error) => Promise.reject(error)));\n};\n\nexport const createTag = (db: Database, tagName: string): Promise<TagType> => {\n  const tagId: string = createHash(`tag-${tagName.toLowerCase()}`, null);\n  const insert: TagType = {\n    _key: tagId,\n    added: Date.now(),\n    name: tagName\n  };\n\n  const aqlQry = aql`UPSERT {_key: ${tagId}}\n    INSERT ${insert}\n    UPDATE {}\n    IN tags RETURN NEW`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((tag = {}) => tag)\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const createTagEdge = (db: Database, tag: TagType, itemId: string): Promise<TagType> => {\n  const edgeCollection = db.collection('isTagged');\n  const {_id: tagId} = tag;\n  const edgeId = createHash(`isTagged-${tagId}-${itemId}`);\n  const type: string = itemId.split('/')[0];\n  const edge: any = {\n    _from: tagId,\n    _key: edgeId,\n    _to: itemId,\n    added: Date.now(),\n    type\n  };\n\n  return edgeCollection.save(edge, {returnNew: true}).then(() => tag);\n};\n\nexport const linkTags = async (db: Database, tagNames: string[], itemId: string): Promise<TagType[]> => {\n  const edgeCollection = db.collection('isTagged');\n  const edges: any = await edgeCollection.inEdges(itemId, {});\n\n  console.log({edges});\n  if(edges.length) {\n    await Promise.all(\n      edges.map((edge) => {\n        const {_key: edgeKey} = edge;\n        const aqlQry = aql`REMOVE {_key:${edgeKey}} IN isTagged`;\n        return db.query(aqlQry);\n      })\n    );\n  }\n\n  return Promise.all(tagNames.map((tagName: string) => createTag(db, tagName)\n    .then((tag: TagType) => createTagEdge(db, tag, itemId)))\n  );\n};\n\nexport const extractTags = (db: Database, itemId, content): Promise<TagType[]> => {\n  const tags: string[] = words(content, /#[a-zA-Z\\d-]+/g).map((tag: string) => parseId(tag));\n  const edgeCollection = db.collection('isTagged');\n\n  return edgeCollection.inEdges(itemId, {})\n    .then((edges: any) => {\n      if(edges.length) {\n        // Remove linked edges\n        return Promise.all(\n          edges.map((edge) => {\n            const {_key: edgeKey} = edge;\n            const aqlQry = aql`REMOVE {_key:${edgeKey}} IN isTagged`;\n            return db.query(aqlQry).catch((error: Error) => {\n              throw error;\n            });\n          }))\n          .then(() => {\n            if(tags.length) {\n              // Create tags\n              return linkTags(db, tags, itemId);\n            }\n            return [];\n          });\n      } else if(tags.length) {\n        // Create tags\n        return linkTags(db, tags, itemId);\n      }\n      return [];\n    })\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n"],
  "mappings": "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",
  "names": ["import_utils", "error", "tag", "words"]
}

292
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/tags.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 {createHash, parseArangoId, parseId, parseVarChar} from '@nlabs/utils';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {Edge} from 'arangojs/documents';\nimport words from 'lodash/words';\n\nimport {parseTag} from '../adapters/tagAdapter';\nimport {ErrorTypes} from '../types/error.types';\nimport {logError} from '../utils/analyticsUtils';\nimport {getDocId, getLimit} from '../utils/arangodbUtils';\n\nimport type {ArangoDbLimit} from '../types/arangodb.types';\nimport type {ApiContext} from '../types/auth.types';\nimport type {IsTaggedType, TagType} from '../types/tags.types';\nimport type {EdgeCollection} from 'arangojs/collections';\n\nconst eventCategory: string = 'tags';\nexport interface TagOptions {\n  readonly from?: number;\n  readonly search?: string;\n  readonly to?: number;\n  readonly userId?: string;\n}\n\nexport const getTags = (\n  context: ApiContext,\n  options: TagOptions = {}\n): Promise<TagType[]> => {\n  const action: string = 'getList';\n  const {database} = context;\n  const {search = '', from = 0, to = 30, userId} = options;\n  const limit: ArangoDbLimit = getLimit(from, to);\n\n  const filters: string[] = [];\n\n  if(search) {\n    filters.push(`LIKE(t.name, \"%${search}%\")`);\n  }\n\n  if(userId) {\n    filters.push(`t.userId == \"${userId}\"`);\n  }\n\n  const aqlQry: string = `FOR t IN tags\n    FILTER ${filters.join(' || ')}\n    ${limit.aql}\n    SORT t.name\n    RETURN t`;\n\n  return database.query(aqlQry)\n    .then((cursor) => cursor.all() as unknown as TagType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, {});\n\n      return [] as TagType[];\n    });\n};\n\nexport const getTagsByItem = (\n  context: ApiContext,\n  itemDocId: string,\n  options: TagOptions = {}\n): Promise<TagType[]> => {\n  const action: string = 'getTagsByItem';\n  const {from = 0, to = 30} = options;\n  const {database} = context;\n  const limit: ArangoDbLimit = getLimit(from, to);\n  const formatItemId: string = parseArangoId(itemDocId);\n  const tagQuery: string = `FOR t, it IN INBOUND \"${formatItemId}\" isTagged\n    ${limit.aql}\n    SORT it.added\n    RETURN t`;\n\n  return database.query(tagQuery)\n    .then((cursor) => cursor.all() as unknown as TagType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, {});\n\n      return [] as TagType[];\n    });\n};\n\nexport const getTag = (context: ApiContext, tagId: string): Promise<TagType> => {\n  const action: string = 'getTag';\n  const {database} = context;\n  const formatId: string = parseId(tagId);\n  const aqlQry: string = `FOR t IN tags\n    FILTER t._key == \"${formatId}\"\n    LIMIT 1\n    RETURN t`;\n\n  return database.query(aqlQry)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, {}));\n};\n\nexport const getTagsByName = (context: ApiContext, tagNames: string[]): Promise<TagType[]> => {\n  const action: string = 'getTagsByName';\n  const {database} = context;\n  const formatTagNames = tagNames\n    .map((tagName) => parseVarChar(tagName, 32))\n    .filter((tagName) => !!tagName);\n\n  if(!formatTagNames.length) {\n    return Promise.resolve([]);\n  }\n\n  const aqlQry: AqlQuery = aql`FOR t IN tags\n    FILTER POSITION(${JSON.stringify(formatTagNames)}, t.name)\n    RETURN t`;\n\n  return database.query(aqlQry)\n    .then((cursor) => cursor.all() as unknown as TagType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, {});\n\n      return [] as TagType[];\n    });\n};\n\nexport const addTag = (context: ApiContext, tag: TagType): Promise<TagType> => {\n  const action: string = 'addTag';\n  const {database} = context;\n  const now: number = Date.now();\n  const {description, name} = parseTag(tag);\n  const formatId: string = createHash(`tag-${name.toLowerCase()}`, null);\n  const insert: TagType = {\n    _key: formatId,\n    added: now,\n    description: parseVarChar(description, 64),\n    modified: now,\n    name: parseVarChar(name, 32)\n  };\n  const aqlQry: AqlQuery = aql`INSERT ${insert} IN tags RETURN NEW`;\n\n  return database.query(aqlQry)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, {}));\n};\n\nexport const addTagToItem = async (\n  context: ApiContext,\n  {itemDocId, tagBy, tagDocId}: {itemDocId: string, tagBy?: string, tagDocId: string}\n): Promise<TagType> => {\n  const action: string = 'addTagToItem';\n  const {database, session: {userId: sessionId}} = context;\n  const added: number = Date.now();\n  const formatTagDocId: string = parseArangoId(tagDocId);\n  const formatItemDocId: string = parseArangoId(itemDocId);\n  const edgeId: string = createHash(`isTagged-${formatTagDocId}-${formatItemDocId}`, null);\n  const edgeCollection: EdgeCollection = database.collection('isTagged');\n  const type: string = formatItemDocId.split('/')[0];\n  const formatTagBy: string = parseId(tagBy) || sessionId;\n  const edge: Edge = {_from: formatTagDocId, _key: edgeId, _to: formatItemDocId, added, tagBy: formatTagBy, type};\n\n  await edgeCollection.save(edge, {overwriteMode: 'ignore', returnNew: true, silent: true})\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      params: {\n        edge\n      },\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, {}));\n\n  const tagQuery: AqlQuery = aql`LET t = DOCUMENT(${formatTagDocId}) RETURN t`;\n\n  return database.query(tagQuery)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      params: {\n        edge\n      },\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, {}));\n};\n\nexport const updateTag = (context: ApiContext, tag: TagType): Promise<TagType> => {\n  const action: string = 'updateTag';\n  const {database, session: {userId: sessionId}} = context;\n  const now: number = Date.now();\n  const {description, name, tagId} = parseTag(tag);\n  const formatName: string = parseVarChar(name, 32);\n  const update: TagType = {\n    description: parseVarChar(description, 64),\n    modified: now,\n    name: formatName\n  };\n  const newTagKey: string = tagId || createHash(`tag-${formatName.toLowerCase()}`, null);\n  const insert: TagType = {\n    ...update,\n    _id: `tags/${newTagKey}`,\n    _key: newTagKey,\n    added: now,\n    tagBy: sessionId\n  };\n  const aqlQry: AqlQuery = aql`UPSERT {name: ${formatName}}\n    INSERT ${insert}\n    UPDATE ${update}\n    IN tags RETURN NEW`;\n\n  return database.query(aqlQry)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, {}));\n};\n\nexport const deleteTag = async (context: ApiContext, tag: TagType): Promise<TagType> => {\n  const action: string = 'deleteTag';\n  const {database} = context;\n  const formatItemId = getDocId('tags', tag);\n  const tagQuery: string = `FOR it FROM isTagged\n    FILTER it._from == \"${formatItemId}\" || it._to == \"${formatItemId}\"\n    REMOVE it IN isTagged\n    RETURN OLD`;\n\n  await database.query(tagQuery)\n    .then((cursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, {}));\n\n  const aqlQry = `LET t = DOCUMENT(${formatItemId})\n    REMOVE t IN tags\n    RETURN OLD`;\n\n  return database.query(aqlQry)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, {}));\n};\n\nexport const deleteTagFromEdge = async (context: ApiContext, {tagDocId, edgeDocId}): Promise<TagType> => {\n  const action = 'deleteTagFromEdge';\n  const {database} = context;\n  const formatTagDocId = parseArangoId(tagDocId);\n  const formatEdgeDocId = parseArangoId(edgeDocId);\n  const edgeCollection: EdgeCollection = database.collection('isTagged');\n  const edge = {_id: formatEdgeDocId};\n\n  await edgeCollection.remove(edge, {silent: true})\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, {}));\n\n  const tagQuery: AqlQuery = aql`LET t = DOCUMENT(${formatTagDocId}) RETURN t`;\n\n  return database.query(tagQuery)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, {}));\n};\n\nexport const deleteTagFromItem = async (context: ApiContext, {tagId, itemId, itemType}): Promise<TagType> => {\n  const action = 'deleteTagFromItem';\n  const {database} = context;\n  const formatTagDocId = parseArangoId(`tags/${tagId}`);\n  const formatItemDocId = parseArangoId(`${itemType}/${itemId}`);\n\n  const aqlQuery: string = `FOR it FROM isTagged\n    FILTER it._from == \"${formatTagDocId}\" || it._to == \"${formatItemDocId}\"\n    REMOVE it IN isTagged`;\n\n  return database.query(aqlQuery)\n    .then(() => true)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, {}));\n};\n\nexport const extractTags = (content: string): string[] =>\n  words(content, /#[a-zA-Z\\d-]+/g).map((tag: string) => parseId(tag));\n\nexport const updateTagsInItem = (context: ApiContext, {itemDocId, tags}): Promise<TagType[]> => {\n  const action = 'updateTagsInItem';\n  const {database} = context;\n  const edgeCollection = database.collection('isTagged') as EdgeCollection<TagType>;\n\n  return edgeCollection.inEdges(itemDocId, {})\n    .then((edges) => Promise.all([\n      ...(edges as unknown as IsTaggedType[])\n        .filter(({name: existingTagName}) => !tags.find(({name: tagName}) => existingTagName === tagName))\n        .map(({_id: edgeDocId, _from: tagDocId}) => deleteTagFromEdge(context, {edgeDocId, tagDocId})),\n      ...(tags as unknown as TagType[])\n        .filter(({name: tagName}) => !(edges as unknown as IsTaggedType[])\n          .find(({name: existingTagName}) => existingTagName === tagName))\n        .map(({_id: tagDocId, tagBy}) => addTagToItem(context, {itemDocId, tagBy, tagDocId}))\n    ])\n      .then(() => tags))\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, {}));\n};\n"],
  "mappings": "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",
  "names": ["words"]
}
