@nlabs/reaktor 0.10.2 → 0.10.6

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 (472) hide show
  1. package/coverage/block-navigation.js +1 -1
  2. package/coverage/index.html +23 -23
  3. package/coverage/sorter.js +21 -7
  4. package/lib/actions/apps.d.ts +1 -0
  5. package/lib/actions/apps.d.ts.map +1 -0
  6. package/lib/actions/apps.js +218 -17
  7. package/lib/actions/connections.d.ts +1 -0
  8. package/lib/actions/connections.d.ts.map +1 -0
  9. package/lib/actions/connections.js +61 -6
  10. package/lib/actions/content.d.ts +1 -0
  11. package/lib/actions/content.d.ts.map +1 -0
  12. package/lib/actions/content.js +232 -9
  13. package/lib/actions/conversations.d.ts +1 -0
  14. package/lib/actions/conversations.d.ts.map +1 -0
  15. package/lib/actions/conversations.js +321 -19
  16. package/lib/actions/dynamodb.d.ts +1 -0
  17. package/lib/actions/dynamodb.d.ts.map +1 -0
  18. package/lib/actions/dynamodb.js +119 -2
  19. package/lib/actions/email.d.ts +1 -0
  20. package/lib/actions/email.d.ts.map +1 -0
  21. package/lib/actions/email.js +56 -2
  22. package/lib/actions/files.d.ts +1 -0
  23. package/lib/actions/files.d.ts.map +1 -0
  24. package/lib/actions/files.js +226 -5
  25. package/lib/actions/groups.d.ts +1 -0
  26. package/lib/actions/groups.d.ts.map +1 -0
  27. package/lib/actions/groups.js +234 -19
  28. package/lib/actions/images.d.ts +1 -0
  29. package/lib/actions/images.d.ts.map +1 -0
  30. package/lib/actions/images.js +688 -31
  31. package/lib/actions/index.d.ts +2 -0
  32. package/lib/actions/index.d.ts.map +1 -0
  33. package/lib/actions/index.js +29 -2
  34. package/lib/actions/ios.d.ts +1 -0
  35. package/lib/actions/ios.d.ts.map +1 -0
  36. package/lib/actions/ios.js +228 -9
  37. package/lib/actions/locations.d.ts +1 -0
  38. package/lib/actions/locations.d.ts.map +1 -0
  39. package/lib/actions/locations.js +111 -7
  40. package/lib/actions/messages.d.ts +1 -0
  41. package/lib/actions/messages.d.ts.map +1 -0
  42. package/lib/actions/messages.js +155 -21
  43. package/lib/actions/notifications.d.ts +1 -0
  44. package/lib/actions/notifications.d.ts.map +1 -0
  45. package/lib/actions/notifications.js +37 -2
  46. package/lib/actions/payments.d.ts +1 -0
  47. package/lib/actions/payments.d.ts.map +1 -0
  48. package/lib/actions/payments.js +428 -11
  49. package/lib/actions/posts.d.ts +1 -0
  50. package/lib/actions/posts.d.ts.map +1 -0
  51. package/lib/actions/posts.js +540 -75
  52. package/lib/actions/profiles.d.ts +1 -0
  53. package/lib/actions/profiles.d.ts.map +1 -0
  54. package/lib/actions/profiles.js +58 -8
  55. package/lib/actions/reactions.d.ts +1 -0
  56. package/lib/actions/reactions.d.ts.map +1 -0
  57. package/lib/actions/reactions.js +276 -25
  58. package/lib/actions/s3.d.ts +1 -0
  59. package/lib/actions/s3.d.ts.map +1 -0
  60. package/lib/actions/s3.js +102 -2
  61. package/lib/actions/search.d.ts +1 -0
  62. package/lib/actions/search.d.ts.map +1 -0
  63. package/lib/actions/search.js +81 -5
  64. package/lib/actions/sms.d.ts +1 -0
  65. package/lib/actions/sms.d.ts.map +1 -0
  66. package/lib/actions/sms.js +52 -2
  67. package/lib/actions/statistics.d.ts +1 -0
  68. package/lib/actions/statistics.d.ts.map +1 -0
  69. package/lib/actions/statistics.js +41 -6
  70. package/lib/actions/subscriptions.d.ts +1 -0
  71. package/lib/actions/subscriptions.d.ts.map +1 -0
  72. package/lib/actions/subscriptions.js +196 -6
  73. package/lib/actions/tags.d.ts +1 -0
  74. package/lib/actions/tags.d.ts.map +1 -0
  75. package/lib/actions/tags.js +258 -19
  76. package/lib/actions/users.d.ts +1 -0
  77. package/lib/actions/users.d.ts.map +1 -0
  78. package/lib/actions/users.js +781 -66
  79. package/lib/actions/videos.d.ts +26 -0
  80. package/lib/actions/videos.d.ts.map +1 -0
  81. package/lib/actions/videos.js +423 -0
  82. package/lib/actions/websockets.d.ts +1 -0
  83. package/lib/actions/websockets.d.ts.map +1 -0
  84. package/lib/actions/websockets.js +148 -14
  85. package/lib/adapters/arangoAdapter.d.ts +1 -0
  86. package/lib/adapters/arangoAdapter.d.ts.map +1 -0
  87. package/lib/adapters/arangoAdapter.js +70 -2
  88. package/lib/adapters/contentAdapter.d.ts +1 -0
  89. package/lib/adapters/contentAdapter.d.ts.map +1 -0
  90. package/lib/adapters/contentAdapter.js +108 -2
  91. package/lib/adapters/fileAdapter.d.ts +1 -0
  92. package/lib/adapters/fileAdapter.d.ts.map +1 -0
  93. package/lib/adapters/fileAdapter.js +119 -2
  94. package/lib/adapters/imageAdapter.d.ts +1 -0
  95. package/lib/adapters/imageAdapter.d.ts.map +1 -0
  96. package/lib/adapters/imageAdapter.js +112 -2
  97. package/lib/adapters/index.d.ts +1 -0
  98. package/lib/adapters/index.d.ts.map +1 -0
  99. package/lib/adapters/index.js +11 -2
  100. package/lib/adapters/messageAdapter.d.ts +1 -0
  101. package/lib/adapters/messageAdapter.d.ts.map +1 -0
  102. package/lib/adapters/messageAdapter.js +82 -2
  103. package/lib/adapters/postAdapter.d.ts +1 -0
  104. package/lib/adapters/postAdapter.d.ts.map +1 -0
  105. package/lib/adapters/postAdapter.js +117 -2
  106. package/lib/adapters/reaktorAdapter.d.ts +1 -0
  107. package/lib/adapters/reaktorAdapter.d.ts.map +1 -0
  108. package/lib/adapters/reaktorAdapter.js +59 -2
  109. package/lib/adapters/tagAdapter.d.ts +1 -0
  110. package/lib/adapters/tagAdapter.d.ts.map +1 -0
  111. package/lib/adapters/tagAdapter.js +99 -2
  112. package/lib/adapters/userAdapter.d.ts +1 -0
  113. package/lib/adapters/userAdapter.d.ts.map +1 -0
  114. package/lib/adapters/userAdapter.js +264 -2
  115. package/lib/config.d.ts +1 -0
  116. package/lib/config.d.ts.map +1 -0
  117. package/lib/config.js +161 -2
  118. package/lib/handlers/graphqlHandler.d.ts +1 -0
  119. package/lib/handlers/graphqlHandler.d.ts.map +1 -0
  120. package/lib/handlers/graphqlHandler.js +118 -2
  121. package/lib/index.d.ts +1 -0
  122. package/lib/index.d.ts.map +1 -0
  123. package/lib/index.js +18 -2
  124. package/lib/lambdas/actions/websockets.d.ts +1 -0
  125. package/lib/lambdas/actions/websockets.d.ts.map +1 -0
  126. package/lib/lambdas/actions/websockets.js +89 -14
  127. package/lib/lambdas/authorizer.d.ts +1 -0
  128. package/lib/lambdas/authorizer.d.ts.map +1 -0
  129. package/lib/lambdas/authorizer.js +41 -2
  130. package/lib/lambdas/connection.d.ts +1 -0
  131. package/lib/lambdas/connection.d.ts.map +1 -0
  132. package/lib/lambdas/connection.js +85 -2
  133. package/lib/lambdas/utils/message.d.ts +1 -0
  134. package/lib/lambdas/utils/message.d.ts.map +1 -0
  135. package/lib/lambdas/utils/message.js +20 -2
  136. package/lib/lambdas/utils/websocket.d.ts +1 -0
  137. package/lib/lambdas/utils/websocket.d.ts.map +1 -0
  138. package/lib/lambdas/utils/websocket.js +78 -2
  139. package/lib/mocks/conversation.d.ts +1 -0
  140. package/lib/mocks/conversation.d.ts.map +1 -0
  141. package/lib/mocks/conversation.js +10 -2
  142. package/lib/mocks/file.d.ts +1 -0
  143. package/lib/mocks/file.d.ts.map +1 -0
  144. package/lib/mocks/file.js +13 -2
  145. package/lib/mocks/group.d.ts +1 -0
  146. package/lib/mocks/group.d.ts.map +1 -0
  147. package/lib/mocks/group.js +20 -2
  148. package/lib/mocks/image.d.ts +1 -0
  149. package/lib/mocks/image.d.ts.map +1 -0
  150. package/lib/mocks/image.js +17 -2
  151. package/lib/mocks/post.d.ts +1 -0
  152. package/lib/mocks/post.d.ts.map +1 -0
  153. package/lib/mocks/post.js +28 -2
  154. package/lib/mocks/tag.d.ts +1 -0
  155. package/lib/mocks/tag.d.ts.map +1 -0
  156. package/lib/mocks/tag.js +12 -2
  157. package/lib/mocks/user.d.ts +1 -0
  158. package/lib/mocks/user.d.ts.map +1 -0
  159. package/lib/mocks/user.js +61 -2
  160. package/lib/mocks/video.d.ts +4 -0
  161. package/lib/mocks/video.d.ts.map +1 -0
  162. package/lib/mocks/video.js +17 -0
  163. package/lib/mutations/content.d.ts +1 -0
  164. package/lib/mutations/content.d.ts.map +1 -0
  165. package/lib/mutations/content.js +27 -2
  166. package/lib/mutations/index.d.ts +1 -0
  167. package/lib/mutations/index.d.ts.map +1 -0
  168. package/lib/mutations/index.js +29 -2
  169. package/lib/mutations/locations.d.ts +1 -0
  170. package/lib/mutations/locations.d.ts.map +1 -0
  171. package/lib/mutations/locations.js +22 -2
  172. package/lib/mutations/messages.d.ts +1 -0
  173. package/lib/mutations/messages.d.ts.map +1 -0
  174. package/lib/mutations/messages.js +75 -2
  175. package/lib/mutations/posts.d.ts +1 -0
  176. package/lib/mutations/posts.d.ts.map +1 -0
  177. package/lib/mutations/posts.js +31 -2
  178. package/lib/mutations/profiles.d.ts +1 -0
  179. package/lib/mutations/profiles.d.ts.map +1 -0
  180. package/lib/mutations/profiles.js +78 -2
  181. package/lib/mutations/reactions.d.ts +1 -0
  182. package/lib/mutations/reactions.d.ts.map +1 -0
  183. package/lib/mutations/reactions.js +29 -2
  184. package/lib/mutations/statistics.d.ts +1 -0
  185. package/lib/mutations/statistics.d.ts.map +1 -0
  186. package/lib/mutations/statistics.js +17 -2
  187. package/lib/mutations/subscriptions.d.ts +1 -0
  188. package/lib/mutations/subscriptions.d.ts.map +1 -0
  189. package/lib/mutations/subscriptions.js +38 -2
  190. package/lib/mutations/tags.d.ts +1 -0
  191. package/lib/mutations/tags.d.ts.map +1 -0
  192. package/lib/mutations/tags.js +109 -2
  193. package/lib/mutations/users.d.ts +1 -0
  194. package/lib/mutations/users.d.ts.map +1 -0
  195. package/lib/mutations/users.js +129 -2
  196. package/lib/objectTypes/app.d.ts +1 -0
  197. package/lib/objectTypes/app.d.ts.map +1 -0
  198. package/lib/objectTypes/app.js +147 -2
  199. package/lib/objectTypes/bankAccount.d.ts +1 -0
  200. package/lib/objectTypes/bankAccount.d.ts.map +1 -0
  201. package/lib/objectTypes/bankAccount.js +54 -2
  202. package/lib/objectTypes/connection.d.ts +1 -0
  203. package/lib/objectTypes/connection.d.ts.map +1 -0
  204. package/lib/objectTypes/connection.js +26 -2
  205. package/lib/objectTypes/content.d.ts +1 -0
  206. package/lib/objectTypes/content.d.ts.map +1 -0
  207. package/lib/objectTypes/content.js +79 -2
  208. package/lib/objectTypes/conversation.d.ts +1 -0
  209. package/lib/objectTypes/conversation.d.ts.map +1 -0
  210. package/lib/objectTypes/conversation.js +53 -2
  211. package/lib/objectTypes/creditCard.d.ts +1 -0
  212. package/lib/objectTypes/creditCard.d.ts.map +1 -0
  213. package/lib/objectTypes/creditCard.js +64 -2
  214. package/lib/objectTypes/document.d.ts +1 -0
  215. package/lib/objectTypes/document.d.ts.map +1 -0
  216. package/lib/objectTypes/document.js +21 -2
  217. package/lib/objectTypes/error.d.ts +1 -0
  218. package/lib/objectTypes/error.d.ts.map +1 -0
  219. package/lib/objectTypes/error.js +24 -2
  220. package/lib/objectTypes/external.d.ts +1 -0
  221. package/lib/objectTypes/external.d.ts.map +1 -0
  222. package/lib/objectTypes/external.js +52 -2
  223. package/lib/objectTypes/file.d.ts +1 -0
  224. package/lib/objectTypes/file.d.ts.map +1 -0
  225. package/lib/objectTypes/file.js +76 -2
  226. package/lib/objectTypes/filter.d.ts +1 -0
  227. package/lib/objectTypes/filter.d.ts.map +1 -0
  228. package/lib/objectTypes/filter.js +21 -2
  229. package/lib/objectTypes/group.d.ts +1 -0
  230. package/lib/objectTypes/group.d.ts.map +1 -0
  231. package/lib/objectTypes/group.js +97 -2
  232. package/lib/objectTypes/iapSubscription.d.ts +1 -0
  233. package/lib/objectTypes/iapSubscription.d.ts.map +1 -0
  234. package/lib/objectTypes/iapSubscription.js +18 -2
  235. package/lib/objectTypes/image.d.ts +1 -0
  236. package/lib/objectTypes/image.d.ts.map +1 -0
  237. package/lib/objectTypes/image.js +105 -2
  238. package/lib/objectTypes/index.d.ts +1 -0
  239. package/lib/objectTypes/index.d.ts.map +1 -0
  240. package/lib/objectTypes/index.js +28 -2
  241. package/lib/objectTypes/location.d.ts +1 -0
  242. package/lib/objectTypes/location.d.ts.map +1 -0
  243. package/lib/objectTypes/location.js +85 -2
  244. package/lib/objectTypes/message.d.ts +1 -0
  245. package/lib/objectTypes/message.d.ts.map +1 -0
  246. package/lib/objectTypes/message.js +72 -2
  247. package/lib/objectTypes/passcode.d.ts +1 -0
  248. package/lib/objectTypes/passcode.d.ts.map +1 -0
  249. package/lib/objectTypes/passcode.js +20 -2
  250. package/lib/objectTypes/plan.d.ts +1 -0
  251. package/lib/objectTypes/plan.d.ts.map +1 -0
  252. package/lib/objectTypes/plan.js +71 -2
  253. package/lib/objectTypes/post.d.ts +1 -0
  254. package/lib/objectTypes/post.d.ts.map +1 -0
  255. package/lib/objectTypes/post.js +101 -2
  256. package/lib/objectTypes/profile.d.ts +1 -0
  257. package/lib/objectTypes/profile.d.ts.map +1 -0
  258. package/lib/objectTypes/profile.js +68 -2
  259. package/lib/objectTypes/reaction.d.ts +1 -0
  260. package/lib/objectTypes/reaction.d.ts.map +1 -0
  261. package/lib/objectTypes/reaction.js +37 -2
  262. package/lib/objectTypes/relation.d.ts +1 -0
  263. package/lib/objectTypes/relation.d.ts.map +1 -0
  264. package/lib/objectTypes/relation.js +27 -2
  265. package/lib/objectTypes/search.d.ts +1 -0
  266. package/lib/objectTypes/search.d.ts.map +1 -0
  267. package/lib/objectTypes/search.js +50 -2
  268. package/lib/objectTypes/statistics.d.ts +1 -0
  269. package/lib/objectTypes/statistics.d.ts.map +1 -0
  270. package/lib/objectTypes/statistics.js +17 -2
  271. package/lib/objectTypes/subscription.d.ts +1 -0
  272. package/lib/objectTypes/subscription.d.ts.map +1 -0
  273. package/lib/objectTypes/subscription.js +102 -2
  274. package/lib/objectTypes/tag.d.ts +1 -0
  275. package/lib/objectTypes/tag.d.ts.map +1 -0
  276. package/lib/objectTypes/tag.js +41 -2
  277. package/lib/objectTypes/user.d.ts +1 -0
  278. package/lib/objectTypes/user.d.ts.map +1 -0
  279. package/lib/objectTypes/user.js +111 -2
  280. package/lib/queries/content.d.ts +1 -0
  281. package/lib/queries/content.d.ts.map +1 -0
  282. package/lib/queries/content.js +50 -2
  283. package/lib/queries/index.d.ts +1 -0
  284. package/lib/queries/index.d.ts.map +1 -0
  285. package/lib/queries/index.js +27 -2
  286. package/lib/queries/locations.d.ts +1 -0
  287. package/lib/queries/locations.d.ts.map +1 -0
  288. package/lib/queries/locations.js +23 -2
  289. package/lib/queries/messages.d.ts +1 -0
  290. package/lib/queries/messages.d.ts.map +1 -0
  291. package/lib/queries/messages.js +35 -2
  292. package/lib/queries/posts.d.ts +1 -0
  293. package/lib/queries/posts.d.ts.map +1 -0
  294. package/lib/queries/posts.js +154 -2
  295. package/lib/queries/reactions.d.ts +1 -0
  296. package/lib/queries/reactions.d.ts.map +1 -0
  297. package/lib/queries/reactions.js +34 -2
  298. package/lib/queries/statistics.d.ts +1 -0
  299. package/lib/queries/statistics.d.ts.map +1 -0
  300. package/lib/queries/statistics.js +17 -2
  301. package/lib/queries/subscriptions.d.ts +1 -0
  302. package/lib/queries/subscriptions.d.ts.map +1 -0
  303. package/lib/queries/subscriptions.js +21 -2
  304. package/lib/queries/tags.d.ts +1 -0
  305. package/lib/queries/tags.d.ts.map +1 -0
  306. package/lib/queries/tags.js +56 -2
  307. package/lib/queries/users.d.ts +1 -0
  308. package/lib/queries/users.d.ts.map +1 -0
  309. package/lib/queries/users.js +39 -2
  310. package/lib/templates/email/layout.d.ts +1 -0
  311. package/lib/templates/email/layout.d.ts.map +1 -0
  312. package/lib/templates/email/layout.js +4 -3
  313. package/lib/templates/email/passwordForgot.d.ts +1 -0
  314. package/lib/templates/email/passwordForgot.d.ts.map +1 -0
  315. package/lib/templates/email/passwordForgot.js +4 -3
  316. package/lib/templates/email/passwordRecovery.d.ts +1 -0
  317. package/lib/templates/email/passwordRecovery.d.ts.map +1 -0
  318. package/lib/templates/email/passwordRecovery.js +4 -3
  319. package/lib/templates/email/verifyEmail.d.ts +1 -0
  320. package/lib/templates/email/verifyEmail.d.ts.map +1 -0
  321. package/lib/templates/email/verifyEmail.js +4 -3
  322. package/lib/templates/email/welcome.d.ts +1 -0
  323. package/lib/templates/email/welcome.d.ts.map +1 -0
  324. package/lib/templates/email/welcome.js +4 -3
  325. package/lib/templates/sms/passwordForgot.d.ts +1 -0
  326. package/lib/templates/sms/passwordForgot.d.ts.map +1 -0
  327. package/lib/templates/sms/passwordForgot.js +3 -2
  328. package/lib/templates/sms/passwordRecovery.d.ts +1 -0
  329. package/lib/templates/sms/passwordRecovery.d.ts.map +1 -0
  330. package/lib/templates/sms/passwordRecovery.js +3 -2
  331. package/lib/templates/sms/verifyEmail.d.ts +1 -0
  332. package/lib/templates/sms/verifyEmail.d.ts.map +1 -0
  333. package/lib/templates/sms/verifyEmail.js +3 -2
  334. package/lib/templates/sms/verifyPhone.d.ts +1 -0
  335. package/lib/templates/sms/verifyPhone.d.ts.map +1 -0
  336. package/lib/templates/sms/verifyPhone.js +3 -2
  337. package/lib/templates/sms/welcome.d.ts +1 -0
  338. package/lib/templates/sms/welcome.d.ts.map +1 -0
  339. package/lib/templates/sms/welcome.js +3 -2
  340. package/lib/types/apps.types.d.ts +1 -0
  341. package/lib/types/apps.types.d.ts.map +1 -0
  342. package/lib/types/apps.types.js +10 -2
  343. package/lib/types/arangodb.types.d.ts +2 -1
  344. package/lib/types/arangodb.types.d.ts.map +1 -0
  345. package/lib/types/arangodb.types.js +6 -1
  346. package/lib/types/auth.types.d.ts +1 -0
  347. package/lib/types/auth.types.d.ts.map +1 -0
  348. package/lib/types/auth.types.js +6 -1
  349. package/lib/types/connections.types.d.ts +1 -0
  350. package/lib/types/connections.types.d.ts.map +1 -0
  351. package/lib/types/connections.types.js +6 -1
  352. package/lib/types/content.types.d.ts +1 -0
  353. package/lib/types/content.types.d.ts.map +1 -0
  354. package/lib/types/content.types.js +6 -1
  355. package/lib/types/conversations.types.d.ts +1 -0
  356. package/lib/types/conversations.types.d.ts.map +1 -0
  357. package/lib/types/conversations.types.js +6 -1
  358. package/lib/types/email.types.d.ts +1 -0
  359. package/lib/types/email.types.d.ts.map +1 -0
  360. package/lib/types/email.types.js +6 -1
  361. package/lib/types/error.types.d.ts +1 -0
  362. package/lib/types/error.types.d.ts.map +1 -0
  363. package/lib/types/error.types.js +20 -2
  364. package/lib/types/files.types.d.ts +1 -0
  365. package/lib/types/files.types.d.ts.map +1 -0
  366. package/lib/types/files.types.js +6 -1
  367. package/lib/types/google.types.d.ts +1 -0
  368. package/lib/types/google.types.d.ts.map +1 -0
  369. package/lib/types/google.types.js +6 -1
  370. package/lib/types/groups.types.d.ts +1 -0
  371. package/lib/types/groups.types.d.ts.map +1 -0
  372. package/lib/types/groups.types.js +6 -1
  373. package/lib/types/images.types.d.ts +1 -0
  374. package/lib/types/images.types.d.ts.map +1 -0
  375. package/lib/types/images.types.js +6 -1
  376. package/lib/types/index.d.ts +2 -0
  377. package/lib/types/index.d.ts.map +1 -0
  378. package/lib/types/index.js +28 -2
  379. package/lib/types/locations.types.d.ts +1 -0
  380. package/lib/types/locations.types.d.ts.map +1 -0
  381. package/lib/types/locations.types.js +6 -1
  382. package/lib/types/messages.types.d.ts +1 -0
  383. package/lib/types/messages.types.d.ts.map +1 -0
  384. package/lib/types/messages.types.js +6 -1
  385. package/lib/types/notifications.types.d.ts +1 -0
  386. package/lib/types/notifications.types.d.ts.map +1 -0
  387. package/lib/types/notifications.types.js +6 -1
  388. package/lib/types/payments.types.d.ts +1 -0
  389. package/lib/types/payments.types.d.ts.map +1 -0
  390. package/lib/types/payments.types.js +6 -1
  391. package/lib/types/posts.types.d.ts +1 -0
  392. package/lib/types/posts.types.d.ts.map +1 -0
  393. package/lib/types/posts.types.js +6 -1
  394. package/lib/types/profiles.types.d.ts +1 -0
  395. package/lib/types/profiles.types.d.ts.map +1 -0
  396. package/lib/types/profiles.types.js +3 -1
  397. package/lib/types/statistics.types.d.ts +1 -0
  398. package/lib/types/statistics.types.d.ts.map +1 -0
  399. package/lib/types/statistics.types.js +3 -1
  400. package/lib/types/tags.types.d.ts +1 -0
  401. package/lib/types/tags.types.d.ts.map +1 -0
  402. package/lib/types/tags.types.js +6 -1
  403. package/lib/types/users.types.d.ts +1 -0
  404. package/lib/types/users.types.d.ts.map +1 -0
  405. package/lib/types/users.types.js +6 -1
  406. package/lib/types/videos.types.d.ts +47 -0
  407. package/lib/types/videos.types.d.ts.map +1 -0
  408. package/lib/types/videos.types.js +6 -0
  409. package/lib/types/websockets.types.d.ts +1 -0
  410. package/lib/types/websockets.types.d.ts.map +1 -0
  411. package/lib/types/websockets.types.js +6 -1
  412. package/lib/utils/adapterUtils.d.ts +1 -0
  413. package/lib/utils/adapterUtils.d.ts.map +1 -0
  414. package/lib/utils/adapterUtils.js +23 -2
  415. package/lib/utils/analyticsUtils.d.ts +1 -0
  416. package/lib/utils/analyticsUtils.d.ts.map +1 -0
  417. package/lib/utils/analyticsUtils.js +45 -2
  418. package/lib/utils/arangodbUtils.d.ts +1 -0
  419. package/lib/utils/arangodbUtils.d.ts.map +1 -0
  420. package/lib/utils/arangodbUtils.js +128 -5
  421. package/lib/utils/authUtils.d.ts +1 -0
  422. package/lib/utils/authUtils.d.ts.map +1 -0
  423. package/lib/utils/authUtils.js +54 -2
  424. package/lib/utils/contextUtils.d.ts +1 -0
  425. package/lib/utils/contextUtils.d.ts.map +1 -0
  426. package/lib/utils/contextUtils.js +10 -2
  427. package/lib/utils/dbI18n.d.ts +1 -0
  428. package/lib/utils/dbI18n.d.ts.map +1 -0
  429. package/lib/utils/dbI18n.example.d.ts +1 -0
  430. package/lib/utils/dbI18n.example.d.ts.map +1 -0
  431. package/lib/utils/dbI18n.example.js +92 -5
  432. package/lib/utils/dbI18n.js +28 -2
  433. package/lib/utils/googleTranslate.d.ts +1 -0
  434. package/lib/utils/googleTranslate.d.ts.map +1 -0
  435. package/lib/utils/googleTranslate.js +70 -2
  436. package/lib/utils/graphqlUtils.d.ts +1 -0
  437. package/lib/utils/graphqlUtils.d.ts.map +1 -0
  438. package/lib/utils/graphqlUtils.js +11 -2
  439. package/lib/utils/index.d.ts +1 -0
  440. package/lib/utils/index.d.ts.map +1 -0
  441. package/lib/utils/index.js +19 -2
  442. package/lib/utils/languageDetection.d.ts +1 -0
  443. package/lib/utils/languageDetection.d.ts.map +1 -0
  444. package/lib/utils/languageDetection.js +120 -2
  445. package/lib/utils/localeUtils.d.ts +1 -0
  446. package/lib/utils/localeUtils.d.ts.map +1 -0
  447. package/lib/utils/localeUtils.example.d.ts +1 -0
  448. package/lib/utils/localeUtils.example.d.ts.map +1 -0
  449. package/lib/utils/localeUtils.example.js +124 -2
  450. package/lib/utils/localeUtils.js +71 -2
  451. package/lib/utils/middlewareUtils.d.ts +1 -0
  452. package/lib/utils/middlewareUtils.d.ts.map +1 -0
  453. package/lib/utils/middlewareUtils.js +10 -2
  454. package/lib/utils/sessionUtils.d.ts +1 -0
  455. package/lib/utils/sessionUtils.d.ts.map +1 -0
  456. package/lib/utils/sessionUtils.js +26 -2
  457. package/lib/utils/stripeUtils.d.ts +2 -1
  458. package/lib/utils/stripeUtils.d.ts.map +1 -0
  459. package/lib/utils/stripeUtils.js +12 -2
  460. package/lib/utils/templateUtils.d.ts +1 -0
  461. package/lib/utils/templateUtils.d.ts.map +1 -0
  462. package/lib/utils/templateUtils.js +11 -2
  463. package/lib/utils/testUtils.d.ts +1 -0
  464. package/lib/utils/testUtils.d.ts.map +1 -0
  465. package/lib/utils/testUtils.js +292 -2
  466. package/lib/utils/translationQueue.d.ts +1 -0
  467. package/lib/utils/translationQueue.d.ts.map +1 -0
  468. package/lib/utils/translationQueue.example.d.ts +1 -0
  469. package/lib/utils/translationQueue.example.d.ts.map +1 -0
  470. package/lib/utils/translationQueue.example.js +340 -2
  471. package/lib/utils/translationQueue.js +113 -2
  472. package/package.json +47 -29
@@ -1,32 +1,449 @@
1
- import{parseNum as A}from"@nlabs/utils/parsers/numbers";import{createHash as Y,parseChar as x,parseId as K,parseString as B,parseVarChar as _}from"@nlabs/utils/parsers/strings";import{aql as T}from"arangojs";import{ErrorTypes as V}from"../types/error.types.js";import{logError as L,UserError as h}from"../utils/analyticsUtils.js";import{useDb as I}from"../utils/arangodbUtils.js";import{getStripeClient as N}from"../utils/stripeUtils.js";import{getUser as H}from"./users.js";const M="payments",pe=r=>{const t="addCustomerAccount",{databaseName:c,session:{userId:n,username:s}={}}=r;return N().customers.create({metadata:{userId:n,username:s}}).then(o=>{const d={modified:Date.now(),stripeCustomerId:o.id},m=T`UPDATE ${n} WITH ${d} IN users LIMIT 1 RETURN NEW`;return I(c).query(m).then(y=>y.next()).then(y=>!!y).catch(y=>L({action:t,category:M,label:V.DATABASE_ERROR},y,r).then(()=>null))})},le=(r,t)=>{const c="addPaymentAccountBank",{databaseName:n,session:{userId:s}={}}=r,{accountNumber:a,fullName:o,routing:u}=t,d=B(a,32);if(d==="")throw new h("required_account_number");const m=_(o,128);if(m==="")throw new h("required_full_name");const y=B(u,32);if(y==="")throw new h("required_routing_number");return H(r,{userId:s}).then(p=>{const{stripeAccountId:l}=p,f=N();return f.tokens.create({bank_account:{account_holder_name:m,account_holder_type:"individual",account_number:d,country:"US",currency:"USD",routing_number:y}}).then(e=>f.customers.createSource(l,{source:e.id})).then(e=>{const i=Date.now(),E={bankAccount:t,bankFullName:m,bankId:e.id,bankRouting:y,modified:i},b=T`UPDATE ${s} WITH ${E} IN users LIMIT 1 RETURN NEW`;return I(n).query(b).then(g=>g.next()).then(g=>g)}).catch(e=>e.message==="A bank account with that routing number and account number already exists for this customer."?L({action:c,category:M,label:"bank_account_exists"},e,r).then(()=>null):L({action:c,category:M,label:"payment_error"},e,r).then(()=>null))})},fe=(r,t)=>{const c="addCreditCard",{databaseName:n,session:{userId:s}}=r;return H(r,{userId:s}).then(a=>{const{stripeAccountId:o}=a,{accountNumber:u,city:d,country:m,cvc:y,expMonth:p,expYear:l,fullName:f,street1:e,street2:i,state:E,zip:b}=t,g=A(u,16);if(!g)throw new h("required_credit_card_number");const P=A(p,2);if(!P)throw new h("required_credit_card_exp_month");const q=A(l,2);if(!q)throw new h("required_credit_card_exp_year");const U=A(y,3),C={},v=_(d,32);v&&(C.city=v);const S=x(m,2);S&&(C.country=S);const R=_(f,32);R&&(C.fullName=R);const k=_(e,32);k&&(C.street1=k);const F=_(i,32);F&&(C.street2=F);const Q=x(E,2);Q&&(C.state=Q);const $=_(b,10);$&&(C.zip=$);const W=N();return W.tokens.create({card:{address_city:v,address_country:S,address_line1:k,address_line2:F,address_state:Q,address_zip:$,cvc:U.toString(),exp_month:P.toString(),exp_year:q.toString(),name:f,number:g.toString()}}).then(D=>W.customers.createSource(o,{source:D.id})).then(D=>{const O=D,Z=O.brand||"",j=O.cvc_check||"",G=O.last4||"",z=Date.now(),J={...C,_key:Y(`user-payment-${s}`),accountNumber:G,added:z,brand:Z,cvcCheck:j,expMonth:p,expYear:l,modified:z,userId:s},X=T`INSERT ${J} IN creditCards RETURN NEW`;return I(n).query(X).then(w=>w.next()).then(w=>{if(w){const{_id:ee,_key:re}=t,te=I(n).collection("hasPayment"),ne=Y(`payment-${re}`),se={_from:`users/${s}`,_key:ne,_to:ee};return te.save(se,{returnNew:!0}).then(()=>t)}return w}).catch(w=>L({action:c,category:M,label:"payment_error"},w,r).then(()=>null))})})},Ce=(r,t)=>{const{databaseName:c,session:{userId:n}}=r,{city:s,country:a,expMonth:o,expYear:u,fullName:d,id:m,street1:y,state:p,zip:l}=t,f=K(m);if(f)throw new h("required_credit_card_id");const e={},i=A(o,2),E=A(u,2),b=_(s,32),g=x(a,2),P=_(d,32),q=B(y,32),U=x(p,2),C=_(l,10);i&&(e.expMonth=i),E&&(e.expYear=E),b&&(e.city=b),g&&(e.country=g),P&&(e.fullName=P),q&&(e.street1=q),U&&(e.state=U),C&&(e.zip=C);const S=T`
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, parseChar, parseId, parseString, parseVarChar } from '@nlabs/utils/parsers/strings';
6
+ import { aql } from 'arangojs';
7
+ import { ErrorTypes } from '../types/error.types.js';
8
+ import { logError, UserError } from '../utils/analyticsUtils.js';
9
+ import { useDb } from '../utils/arangodbUtils.js';
10
+ import { getStripeClient } from '../utils/stripeUtils.js';
11
+ import { getUser } from './users.js';
12
+ const eventCategory = 'payments';
13
+ export const addCustomerAccount = (context)=>{
14
+ const action = 'addCustomerAccount';
15
+ const { databaseName, session: { userId: sessionId, username } = {} } = context;
16
+ const stripeClient = getStripeClient();
17
+ return stripeClient.customers.create({
18
+ metadata: {
19
+ userId: sessionId,
20
+ username
21
+ }
22
+ }).then((customer)=>{
23
+ // Create session
24
+ const now = Date.now();
25
+ const update = {
26
+ modified: now,
27
+ stripeCustomerId: customer.id
28
+ };
29
+ const aqlQry = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;
30
+ return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.next()).then((updatedUser)=>!!updatedUser).catch((error)=>logError({
31
+ action,
32
+ category: eventCategory,
33
+ label: ErrorTypes.DATABASE_ERROR
34
+ }, error, context).then(()=>null));
35
+ });
36
+ };
37
+ export const addBankAccount = (context, bankAccount)=>{
38
+ const action = 'addPaymentAccountBank';
39
+ const { databaseName, session: { userId: sessionId } = {} } = context;
40
+ // Params
41
+ const { accountNumber, fullName, routing } = bankAccount;
42
+ const formatAccount = parseString(accountNumber, 32);
43
+ if (formatAccount === '') {
44
+ throw new UserError('required_account_number');
45
+ }
46
+ const formatFullName = parseVarChar(fullName, 128);
47
+ if (formatFullName === '') {
48
+ throw new UserError('required_full_name');
49
+ }
50
+ const formatRouting = parseString(routing, 32);
51
+ if (formatRouting === '') {
52
+ throw new UserError('required_routing_number');
53
+ }
54
+ return getUser(context, {
55
+ userId: sessionId
56
+ }).then((user)=>{
57
+ const { stripeAccountId } = user;
58
+ const stripeClient = getStripeClient();
59
+ // Create a token first, then use the token as the source
60
+ return stripeClient.tokens.create({
61
+ bank_account: {
62
+ account_holder_name: formatFullName,
63
+ account_holder_type: 'individual',
64
+ account_number: formatAccount,
65
+ country: 'US',
66
+ currency: 'USD',
67
+ routing_number: formatRouting
68
+ }
69
+ }).then((token)=>stripeClient.customers.createSource(stripeAccountId, {
70
+ source: token.id
71
+ })).then((account)=>{
72
+ // Use type assertion for card properties
73
+ // const cardSource = account as unknown as Card;
74
+ // const brand = cardSource.brand || '';
75
+ // const cvcCheck = cardSource.cvc_check || '';
76
+ // const last4 = cardSource.last4 || '';
77
+ // Create session
78
+ const now = Date.now();
79
+ const update = {
80
+ bankAccount,
81
+ bankFullName: formatFullName,
82
+ bankId: account.id,
83
+ bankRouting: formatRouting,
84
+ modified: now
85
+ };
86
+ const aqlQry = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;
87
+ return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.next()).then((updatedUser)=>updatedUser);
88
+ }).catch((error)=>{
89
+ const msg = error.message;
90
+ if (msg === 'A bank account with that routing number and account number ' + 'already exists for this customer.') {
91
+ return logError({
92
+ action,
93
+ category: eventCategory,
94
+ label: 'bank_account_exists'
95
+ }, error, context).then(()=>null);
96
+ }
97
+ return logError({
98
+ action,
99
+ category: eventCategory,
100
+ label: 'payment_error'
101
+ }, error, context).then(()=>null);
102
+ });
103
+ });
104
+ };
105
+ export const addCreditCard = (context, card)=>{
106
+ const action = 'addCreditCard';
107
+ const { databaseName, session: { userId: sessionId } } = context;
108
+ return getUser(context, {
109
+ userId: sessionId
110
+ }).then((user)=>{
111
+ // User
112
+ const { stripeAccountId } = user;
113
+ // Card
114
+ const { accountNumber, city, country, cvc, expMonth, expYear, fullName, street1, street2, state, zip } = card;
115
+ const formatNumber = parseNum(accountNumber, 16);
116
+ if (!formatNumber) {
117
+ throw new UserError('required_credit_card_number');
118
+ }
119
+ const formatExpMonth = parseNum(expMonth, 2);
120
+ if (!formatExpMonth) {
121
+ throw new UserError('required_credit_card_exp_month');
122
+ }
123
+ const formatExpYear = parseNum(expYear, 2);
124
+ if (!formatExpYear) {
125
+ throw new UserError('required_credit_card_exp_year');
126
+ }
127
+ const formatCvc = parseNum(cvc, 3);
128
+ // Address
129
+ const paymentCard = {};
130
+ const formatCity = parseVarChar(city, 32);
131
+ if (formatCity) {
132
+ paymentCard.city = formatCity;
133
+ }
134
+ const formatCountry = parseChar(country, 2);
135
+ if (formatCountry) {
136
+ paymentCard.country = formatCountry;
137
+ }
138
+ const formatFullName = parseVarChar(fullName, 32);
139
+ if (formatFullName) {
140
+ paymentCard.fullName = formatFullName;
141
+ }
142
+ const formatStreet1 = parseVarChar(street1, 32);
143
+ if (formatStreet1) {
144
+ paymentCard.street1 = formatStreet1;
145
+ }
146
+ const formatStreet2 = parseVarChar(street2, 32);
147
+ if (formatStreet2) {
148
+ paymentCard.street2 = formatStreet2;
149
+ }
150
+ const formatState = parseChar(state, 2);
151
+ if (formatState) {
152
+ paymentCard.state = formatState;
153
+ }
154
+ const formatZip = parseVarChar(zip, 10);
155
+ if (formatZip) {
156
+ paymentCard.zip = formatZip;
157
+ }
158
+ const stripeClient = getStripeClient();
159
+ // Create a token first, then use the token as the source
160
+ return stripeClient.tokens.create({
161
+ card: {
162
+ address_city: formatCity,
163
+ address_country: formatCountry,
164
+ address_line1: formatStreet1,
165
+ address_line2: formatStreet2,
166
+ address_state: formatState,
167
+ address_zip: formatZip,
168
+ cvc: formatCvc.toString(),
169
+ exp_month: formatExpMonth.toString(),
170
+ exp_year: formatExpYear.toString(),
171
+ name: fullName,
172
+ number: formatNumber.toString()
173
+ }
174
+ }).then((token)=>stripeClient.customers.createSource(stripeAccountId, {
175
+ source: token.id
176
+ })).then((newSource)=>{
177
+ // Use type assertion for card properties
178
+ const cardSource = newSource;
179
+ const brand = cardSource.brand || '';
180
+ const cvcCheck = cardSource.cvc_check || '';
181
+ const last4 = cardSource.last4 || '';
182
+ // Create session
183
+ const now = Date.now();
184
+ const insert = {
185
+ ...paymentCard,
186
+ _key: createHash(`user-payment-${sessionId}`),
187
+ accountNumber: last4,
188
+ added: now,
189
+ brand,
190
+ cvcCheck,
191
+ expMonth,
192
+ expYear,
193
+ modified: now,
194
+ userId: sessionId
195
+ };
196
+ const insertAqlQry = aql`INSERT ${insert} IN creditCards RETURN NEW`;
197
+ return useDb(databaseName).query(insertAqlQry).then((cursor)=>cursor.next()).then((newCard)=>{
198
+ if (newCard) {
199
+ // Add linked edge
200
+ const { _id: cardId, _key: cardKey } = card;
201
+ const edgeCollection = useDb(databaseName).collection('hasPayment');
202
+ const edgeId = createHash(`payment-${cardKey}`);
203
+ const edge = {
204
+ _from: `users/${sessionId}`,
205
+ _key: edgeId,
206
+ _to: cardId
207
+ };
208
+ return edgeCollection.save(edge, {
209
+ returnNew: true
210
+ }).then(()=>card);
211
+ }
212
+ return newCard;
213
+ }).catch((error)=>logError({
214
+ action,
215
+ category: eventCategory,
216
+ label: 'payment_error'
217
+ }, error, context).then(()=>null));
218
+ });
219
+ });
220
+ };
221
+ export const updateCreditCard = (context, card)=>{
222
+ const { databaseName, session: { userId: sessionId } } = context;
223
+ const { city, country, expMonth, expYear, fullName, id, street1, state, zip } = card;
224
+ const formatId = parseId(id);
225
+ if (formatId) {
226
+ throw new UserError('required_credit_card_id');
227
+ }
228
+ const paymentCard = {};
229
+ const formatExpMonth = parseNum(expMonth, 2);
230
+ const formatExpYear = parseNum(expYear, 2);
231
+ const formatCity = parseVarChar(city, 32);
232
+ const formatCountry = parseChar(country, 2);
233
+ const formatFullName = parseVarChar(fullName, 32);
234
+ const formatStreet1 = parseString(street1, 32);
235
+ const formatState = parseChar(state, 2);
236
+ const formatZip = parseVarChar(zip, 10);
237
+ if (formatExpMonth) {
238
+ paymentCard.expMonth = formatExpMonth;
239
+ }
240
+ if (formatExpYear) {
241
+ paymentCard.expYear = formatExpYear;
242
+ }
243
+ if (formatCity) {
244
+ paymentCard.city = formatCity;
245
+ }
246
+ if (formatCountry) {
247
+ paymentCard.country = formatCountry;
248
+ }
249
+ if (formatFullName) {
250
+ paymentCard.fullName = formatFullName;
251
+ }
252
+ if (formatStreet1) {
253
+ paymentCard.street1 = formatStreet1;
254
+ }
255
+ if (formatState) {
256
+ paymentCard.state = formatState;
257
+ }
258
+ if (formatZip) {
259
+ paymentCard.zip = formatZip;
260
+ }
261
+ const update = paymentCard;
262
+ const aqlQry = aql`
2
263
  LET updatedCard = FIRST(
3
264
  FOR c IN creditCards
4
- FILTER c._key == ${f} && c.userId == ${n}
5
- UPDATE c WITH ${e} IN creditCards
265
+ FILTER c._key == ${formatId} && c.userId == ${sessionId}
266
+ UPDATE c WITH ${update} IN creditCards
6
267
  LIMIT 1
7
268
  RETURN NEW
8
269
  )
9
270
  LET user = FIRST(
10
271
  FOR u IN users
11
- FILTER u._key == ${n}
272
+ FILTER u._key == ${sessionId}
12
273
  LIMIT 1
13
274
  RETURN u
14
275
  )
15
- RETURN {user: user, card: updatedCard}`;return I(c).query(S).then(R=>R.next()).then((R={card:{},user:{}})=>{const k=R.card,{user:F}=R;if(!k)throw new h("not_found");const{stripeCustomerId:Q}=F,{stripeId:$}=t,W=N(),D={address_city:b,address_country:g,address_line1:q,address_state:U,address_zip:C,exp_month:i.toString(),exp_year:E.toString(),name:P};return W.customers.updateSource(Q,$,D).then(()=>t).catch(O=>{throw new h("payment_error")})})},Ie=r=>{const t="getCreditCards",{databaseName:c,session:{userId:n}}=r,s=T`FOR c IN creditCards
16
- FILTER c.userId == ${n}
17
- RETURN c`;return I(c).query(s).then(a=>a.all()).then((a=[])=>a).catch(a=>L({action:t,category:M,label:V.DATABASE_ERROR},a,r).then(()=>null))},he=(r,t)=>{const{databaseName:c,session:{userId:n}}=r,s=K(t),a=T`
276
+ RETURN {user: user, card: updatedCard}`;
277
+ return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.next()).then((results = {
278
+ card: {},
279
+ user: {}
280
+ })=>{
281
+ const updatedCard = results.card;
282
+ const { user } = results;
283
+ if (!updatedCard) {
284
+ throw new UserError('not_found');
285
+ }
286
+ const { stripeCustomerId } = user;
287
+ const { stripeId } = card;
288
+ const stripeClient = getStripeClient();
289
+ const update = {
290
+ address_city: formatCity,
291
+ address_country: formatCountry,
292
+ address_line1: formatStreet1,
293
+ address_state: formatState,
294
+ address_zip: formatZip,
295
+ exp_month: formatExpMonth.toString(),
296
+ exp_year: formatExpYear.toString(),
297
+ name: formatFullName
298
+ };
299
+ return stripeClient.customers.updateSource(stripeCustomerId, stripeId, update).then(()=>card).catch((error)=>{
300
+ console.log('payments::updateCard::error', error);
301
+ throw new UserError('payment_error');
302
+ });
303
+ });
304
+ };
305
+ export const getCreditCards = (context)=>{
306
+ const action = 'getCreditCards';
307
+ const { databaseName, session: { userId: sessionId } } = context;
308
+ const aqlQry = aql`FOR c IN creditCards
309
+ FILTER c.userId == ${sessionId}
310
+ RETURN c`;
311
+ return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.all()).then((list = [])=>list).catch((error)=>logError({
312
+ action,
313
+ category: eventCategory,
314
+ label: ErrorTypes.DATABASE_ERROR
315
+ }, error, context).then(()=>null));
316
+ };
317
+ export const deleteCreditCard = (context, cardId)=>{
318
+ const { databaseName, session: { userId: sessionId } } = context;
319
+ const formatCardId = parseId(cardId);
320
+ const aqlQry = aql`
18
321
  LET card = FIRST(
19
322
  FOR c IN creditCards
20
- FILTER c._key == ${s} && c.userId == ${n}
323
+ FILTER c._key == ${formatCardId} && c.userId == ${sessionId}
21
324
  LIMIT 1
22
325
  REMOVE c IN creditCards
23
326
  RETURN OLD
24
327
  )
25
328
  LET user = FIRST(
26
329
  FOR u IN users
27
- FILTER u._key == ${n}
330
+ FILTER u._key == ${sessionId}
28
331
  LIMIT 1
29
332
  RETURN u
30
333
  )
31
- RETURN {user: user, card: card}`;return I(c).query(a).then(o=>o.next()).then((o={card:{},user:{}})=>{if(!o)return!1;const{card:u,user:d}=o,{_key:m}=u;return I(c).collection("hasPayment").outEdges(m,{}).then(async p=>{const l=Array.isArray(p)?p:[];return l.length?(await Promise.all(l.map(f=>{const{_key:e}=f,i=T`REMOVE {_key:${e}} IN hasPayment`;return I(c).query(i)})).then(()=>N().customers.deleteSource(d.stripeCustomerId,u.stripeId).then(()=>!0).catch(e=>{throw new h("payment_error")})),!0):!1})})},ge=(r,t)=>{const{databaseName:c,session:{userId:n}}=r,s={bankAccount:"",bankFullName:"",bankId:"",bankRouting:"",modified:Date.now()},a=T`UPDATE ${n} WITH ${s} IN users LIMIT 1 RETURN NEW`;return I(c).query(a).then(o=>o.next()).then(o=>{const{stripeAccountId:u}=o;return N().customers.deleteSource(u,t).then(()=>!0).catch(()=>Promise.resolve(!1))})},Te=(r,t)=>{const{databaseName:c,session:{userId:n}}=r,{amount:s,currency:a}=t,o=A(s),u=x(a,3,"USD").toUpperCase();return H(r,{userId:n}).then(d=>{const{stripeAccountId:m}=d;return N().transfers.create({amount:o,currency:u,destination:m}).then(p=>{const l=Date.now(),e=T`INSERT ${{added:l,amount:o,currency:u,modified:l,userId:n}} IN transfers RETURN NEW`;return I(c).query(e).then(i=>i.next()).then(i=>i)})})},_e=(r,t)=>{const{databaseName:c,session:{userId:n}}=r,{amount:s,capture:a,cardId:o,currency:u,description:d}=t,m=x(u,3,"USD").toUpperCase();return N().charges.create({amount:s,capture:a,currency:m,description:d,source:o}).then(p=>{const l=Date.now(),f={added:l,amount:s,capture:a,cardId:o,chargeFailCode:p.failure_code,chargeFailMsg:p.failure_message,chargeId:p.id,chargeStatus:p.status,currency:m,description:d,modified:l,userId:n},e=T`INSERT ${f} IN payments RETURN NEW`;return I(c).query(e).then(i=>i.next()).then(i=>i)}).catch(p=>{throw new h("payment_error")})};export{le as addBankAccount,fe as addCreditCard,pe as addCustomerAccount,_e as createPaymentHold,Te as createPaymentTransfer,ge as deleteBankAccount,he as deleteCreditCard,Ie as getCreditCards,Ce as updateCreditCard};
32
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/payments.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 {createHash, parseChar, parseId, parseString, parseVarChar} from '@nlabs/utils/parsers/strings';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport stripe from 'stripe';\n\nimport {ApiContext} from '../types/auth.types.js';\nimport {ErrorTypes} from '../types/error.types.js';\nimport {PaymentBankAccount, PaymentCardType, PaymentCharge, PaymentTransfer} from '../types/payments.types.js';\nimport {UserType} from '../types/users.types.js';\nimport {logError, UserError} from '../utils/analyticsUtils.js';\nimport {useDb} from '../utils/arangodbUtils.js';\nimport {getStripeClient} from '../utils/stripeUtils.js';\nimport {getUser} from './users.js';\n\nimport type {EdgeCollection} from 'arangojs/collections';\n\nconst eventCategory = 'payments';\n\nexport const addCustomerAccount = (context: ApiContext): Promise<boolean> => {\n  const action = 'addCustomerAccount';\n  const {databaseName, session: {userId: sessionId, username} = {}} = context;\n\n  const stripeClient = getStripeClient();\n\n  return stripeClient.customers\n    .create({\n      metadata: {\n        userId: sessionId,\n        username\n      }\n    } as any)\n    .then((customer) => {\n      // Create session\n      const now: number = Date.now();\n      const update: UserType = {\n        modified: now,\n        stripeCustomerId: customer.id\n      };\n\n      const aqlQry: AqlQuery = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;\n\n      return useDb(databaseName).query(aqlQry)\n        .then((cursor) => cursor.next())\n        .then((updatedUser: UserType) => !!updatedUser)\n        .catch((error: Error) => logError({\n          action,\n          category: eventCategory,\n          label: ErrorTypes.DATABASE_ERROR\n        }, error, context).then(() => null));\n    });\n};\n\nexport const addBankAccount = (context: ApiContext, bankAccount: PaymentBankAccount): Promise<boolean> => {\n  const action = 'addPaymentAccountBank';\n  const {databaseName, session: {userId: sessionId} = {}} = context;\n\n  // Params\n  const {\n    accountNumber,\n    fullName,\n    routing\n  } = bankAccount;\n\n  const formatAccount: string = parseString(accountNumber, 32);\n\n  if(formatAccount === '') {\n    throw new UserError('required_account_number');\n  }\n\n  const formatFullName: string = parseVarChar(fullName, 128);\n\n  if(formatFullName === '') {\n    throw new UserError('required_full_name');\n  }\n\n  const formatRouting: string = parseString(routing, 32);\n\n  if(formatRouting === '') {\n    throw new UserError('required_routing_number');\n  }\n\n  return getUser(context, {userId: sessionId})\n    .then((user: UserType) => {\n      const {stripeAccountId} = user;\n      const stripeClient = getStripeClient();\n\n      // Create a token first, then use the token as the source\n      return stripeClient.tokens.create({\n        bank_account: {\n          account_holder_name: formatFullName,\n          account_holder_type: 'individual',\n          account_number: formatAccount,\n          country: 'US',\n          currency: 'USD',\n          routing_number: formatRouting\n        }\n      })\n        .then((token) => stripeClient.customers.createSource(\n          stripeAccountId,\n          {source: token.id}\n        ))\n        .then((account) => {\n          // Use type assertion for card properties\n          // const cardSource = account as unknown as Card;\n          // const brand = cardSource.brand || '';\n          // const cvcCheck = cardSource.cvc_check || '';\n          // const last4 = cardSource.last4 || '';\n\n          // Create session\n          const now: number = Date.now();\n          const update = {\n            bankAccount,\n            bankFullName: formatFullName,\n            bankId: account.id,\n            bankRouting: formatRouting,\n            modified: now\n          };\n\n          const aqlQry: AqlQuery = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;\n\n          return useDb(databaseName).query(aqlQry)\n            .then((cursor) => cursor.next())\n            .then((updatedUser: UserType) => updatedUser);\n        })\n        .catch((error: Error) => {\n          const msg = error.message;\n\n          if(msg === 'A bank account with that routing number and account number ' +\n            'already exists for this customer.') {\n            return logError({\n              action,\n              category: eventCategory,\n              label: 'bank_account_exists'\n            }, error, context).then(() => null);\n          }\n          return logError({\n            action,\n            category: eventCategory,\n            label: 'payment_error'\n          }, error, context).then(() => null);\n        });\n    });\n};\n\nexport const addCreditCard = (context: ApiContext, card: PaymentCardType): Promise<UserType> => {\n  const action = 'addCreditCard';\n  const {databaseName, session: {userId: sessionId}} = context;\n\n  return getUser(context, {userId: sessionId})\n    .then((user: UserType) => {\n      // User\n      const {stripeAccountId} = user;\n\n      // Card\n      const {\n        accountNumber,\n        city,\n        country,\n        cvc,\n        expMonth,\n        expYear,\n        fullName,\n        street1,\n        street2,\n        state,\n        zip\n      }: PaymentCardType = card;\n\n      const formatNumber: number = parseNum(accountNumber, 16);\n\n      if(!formatNumber) {\n        throw new UserError('required_credit_card_number');\n      }\n\n      const formatExpMonth: number = parseNum(expMonth, 2);\n\n      if(!formatExpMonth) {\n        throw new UserError('required_credit_card_exp_month');\n      }\n\n      const formatExpYear: number = parseNum(expYear, 2);\n\n      if(!formatExpYear) {\n        throw new UserError('required_credit_card_exp_year');\n      }\n\n      const formatCvc: number = parseNum(cvc, 3);\n\n      // Address\n      const paymentCard: PaymentCardType = {};\n      const formatCity: string = parseVarChar(city, 32);\n\n      if(formatCity) {\n        paymentCard.city = formatCity;\n      }\n\n      const formatCountry: string = parseChar(country, 2);\n\n      if(formatCountry) {\n        paymentCard.country = formatCountry;\n      }\n\n      const formatFullName: string = parseVarChar(fullName, 32);\n\n      if(formatFullName) {\n        paymentCard.fullName = formatFullName;\n      }\n\n      const formatStreet1: string = parseVarChar(street1, 32);\n\n      if(formatStreet1) {\n        paymentCard.street1 = formatStreet1;\n      }\n\n      const formatStreet2: string = parseVarChar(street2, 32);\n\n      if(formatStreet2) {\n        paymentCard.street2 = formatStreet2;\n      }\n\n      const formatState: string = parseChar(state, 2);\n\n      if(formatState) {\n        paymentCard.state = formatState;\n      }\n\n      const formatZip: string = parseVarChar(zip, 10);\n\n      if(formatZip) {\n        paymentCard.zip = formatZip;\n      }\n\n      const stripeClient = getStripeClient();\n\n      // Create a token first, then use the token as the source\n      return stripeClient.tokens.create({\n        card: {\n          address_city: formatCity,\n          address_country: formatCountry,\n          address_line1: formatStreet1,\n          address_line2: formatStreet2,\n          address_state: formatState,\n          address_zip: formatZip,\n          cvc: formatCvc.toString(),\n          exp_month: formatExpMonth.toString(),\n          exp_year: formatExpYear.toString(),\n          name: fullName,\n          number: formatNumber.toString()\n        }\n      })\n        .then((token) => stripeClient.customers.createSource(\n          stripeAccountId,\n          {source: token.id}\n        ))\n        .then((newSource) => {\n          // Use type assertion for card properties\n          const cardSource = newSource as unknown as Card;\n          const brand = cardSource.brand || '';\n          const cvcCheck = cardSource.cvc_check || '';\n          const last4 = cardSource.last4 || '';\n\n          // Create session\n          const now: number = Date.now();\n          const insert = {\n            ...paymentCard,\n            _key: createHash(`user-payment-${sessionId}`),\n            accountNumber: last4,\n            added: now,\n            brand,\n            cvcCheck,\n            expMonth,\n            expYear,\n            modified: now,\n            userId: sessionId\n          };\n          const insertAqlQry: AqlQuery = aql`INSERT ${insert} IN creditCards RETURN NEW`;\n\n          return useDb(databaseName).query(insertAqlQry)\n            .then((cursor) => cursor.next())\n            .then((newCard: PaymentCardType) => {\n              if(newCard) {\n                // Add linked edge\n                const {_id: cardId, _key: cardKey} = card;\n                const edgeCollection: EdgeCollection = useDb(databaseName).collection('hasPayment');\n                const edgeId = createHash(`payment-${cardKey}`);\n                const edge = {\n                  _from: `users/${sessionId}`,\n                  _key: edgeId,\n                  _to: cardId\n                };\n\n                return edgeCollection.save(edge, {returnNew: true}).then(() => card);\n              }\n\n              return newCard;\n            })\n            .catch((error: Error) => logError({\n              action,\n              category: eventCategory,\n              label: 'payment_error'\n            }, error, context).then(() => null));\n        });\n    });\n};\n\nexport const updateCreditCard = (context: ApiContext, card: PaymentCardType): Promise<PaymentCardType> => {\n  const {databaseName, session: {userId: sessionId}} = context;\n\n  const {\n    city,\n    country,\n    expMonth,\n    expYear,\n    fullName,\n    id,\n    street1,\n    state,\n    zip\n  }: PaymentCardType = card;\n\n  const formatId: string = parseId(id);\n\n  if(formatId) {\n    throw new UserError('required_credit_card_id');\n  }\n\n  const paymentCard: PaymentCardType = {};\n  const formatExpMonth: number = parseNum(expMonth, 2);\n  const formatExpYear: number = parseNum(expYear, 2);\n  const formatCity: string = parseVarChar(city, 32);\n  const formatCountry: string = parseChar(country, 2);\n  const formatFullName: string = parseVarChar(fullName, 32);\n  const formatStreet1: string = parseString(street1, 32);\n  const formatState: string = parseChar(state, 2);\n  const formatZip: string = parseVarChar(zip, 10);\n\n  if(formatExpMonth) {\n    paymentCard.expMonth = formatExpMonth;\n  }\n\n  if(formatExpYear) {\n    paymentCard.expYear = formatExpYear;\n  }\n\n  if(formatCity) {\n    paymentCard.city = formatCity;\n  }\n\n  if(formatCountry) {\n    paymentCard.country = formatCountry;\n  }\n\n  if(formatFullName) {\n    paymentCard.fullName = formatFullName;\n  }\n\n  if(formatStreet1) {\n    paymentCard.street1 = formatStreet1;\n  }\n\n  if(formatState) {\n    paymentCard.state = formatState;\n  }\n\n  if(formatZip) {\n    paymentCard.zip = formatZip;\n  }\n\n  const update = paymentCard;\n  const aqlQry: AqlQuery = aql`\n      LET updatedCard = FIRST(\n        FOR c IN creditCards\n        FILTER c._key == ${formatId} && c.userId == ${sessionId}\n        UPDATE c WITH ${update} IN creditCards\n        LIMIT 1\n        RETURN NEW\n      )\n      LET user = FIRST(\n        FOR u IN users\n        FILTER u._key == ${sessionId}\n        LIMIT 1\n        RETURN u\n      )\n      RETURN {user: user, card: updatedCard}`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.next())\n    .then((results = {card: {}, user: {}}) => {\n      const updatedCard: PaymentCardType = results.card;\n      const {user} = results;\n\n      if(!updatedCard) {\n        throw new UserError('not_found');\n      }\n\n      const {stripeCustomerId} = user;\n      const {stripeId} = card;\n      const stripeClient = getStripeClient();\n      const update: stripe.CustomerUpdateSourceParams = {\n        address_city: formatCity,\n        address_country: formatCountry,\n        address_line1: formatStreet1,\n        address_state: formatState,\n        address_zip: formatZip,\n        exp_month: formatExpMonth.toString(),\n        exp_year: formatExpYear.toString(),\n        name: formatFullName\n      };\n\n      return stripeClient.customers\n        .updateSource(stripeCustomerId, stripeId, update)\n        .then(() => card)\n        .catch((error: Error) => {\n          console.log('payments::updateCard::error', error);\n          throw new UserError('payment_error');\n        });\n    });\n};\n\nexport const getCreditCards = (context: ApiContext): Promise<PaymentCardType[]> => {\n  const action = 'getCreditCards';\n  const {databaseName, session: {userId: sessionId}} = context;\n  const aqlQry: AqlQuery = aql`FOR c IN creditCards\n    FILTER c.userId == ${sessionId}\n    RETURN c`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.all())\n    .then((list: PaymentCardType[] = []) => list)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context).then(() => null));\n};\n\nexport const deleteCreditCard = (context: ApiContext, cardId: string): Promise<boolean> => {\n  const {databaseName, session: {userId: sessionId}} = context;\n  const formatCardId: string = parseId(cardId);\n  const aqlQry: AqlQuery = aql`\n    LET card = FIRST(\n      FOR c IN creditCards\n      FILTER c._key == ${formatCardId} && c.userId == ${sessionId}\n      LIMIT 1\n      REMOVE c IN creditCards\n      RETURN OLD\n    )\n    LET user = FIRST(\n      FOR u IN users\n      FILTER u._key == ${sessionId}\n      LIMIT 1\n      RETURN u\n    )\n    RETURN {user: user, card: card}`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.next())\n    .then((result = {card: {}, user: {}}) => {\n      if(!result) {\n        return false;\n      }\n\n      const {card, user} = result;\n      const {_key: cardKey} = card;\n\n      // Remove linked edges\n      const edgeCollection = useDb(databaseName).collection('hasPayment');\n\n      return edgeCollection.outEdges(cardKey, {})\n        .then(async (response) => {\n          // Extract edges from the response\n          const edges = Array.isArray(response) ? response : [];\n\n          if(edges.length) {\n            await Promise.all(\n              edges.map((edge) => {\n                const {_key: edgeKey} = edge;\n                const removeAqlQry: AqlQuery = aql`REMOVE {_key:${edgeKey}} IN hasPayment`;\n                return useDb(databaseName).query(removeAqlQry);\n              }))\n              .then(() => {\n                // Stripe\n                const stripeClient = getStripeClient();\n\n                return stripeClient.customers\n                  .deleteSource(user.stripeCustomerId, card.stripeId)\n                  .then(() => true)\n                  .catch((error: Error) => {\n                    console.log('payments::deleteCard::error', error);\n                    throw new UserError('payment_error');\n                  });\n              });\n\n            return true;\n          }\n\n          return false;\n        });\n    });\n};\n\nexport const deleteBankAccount = (context: ApiContext, bankId: string): Promise<boolean> => {\n  const {databaseName, session: {userId: sessionId}} = context;\n\n  // Clean db\n  const update: UserType = {\n    bankAccount: '',\n    bankFullName: '',\n    bankId: '',\n    bankRouting: '',\n    modified: Date.now()\n  };\n  const aqlQry: AqlQuery = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.next())\n    .then((user: UserType) => {\n      const {stripeAccountId} = user;\n      const stripeClient = getStripeClient();\n\n      return stripeClient.customers\n        .deleteSource(stripeAccountId, bankId)\n        .then(() => true)\n        .catch(() => Promise.resolve(false));\n    });\n};\n\nexport const createPaymentTransfer = (context: ApiContext, transfer: PaymentTransfer): Promise<PaymentTransfer> => {\n  const {databaseName, session: {userId: sessionId}} = context;\n  const {amount, currency} = transfer;\n  const formatAmount: number = parseNum(amount);\n  const formatCurrency: string = parseChar(currency, 3, 'USD').toUpperCase();\n\n  return getUser(context, {userId: sessionId})\n    .then((user: UserType) => {\n      const {stripeAccountId} = user;\n      const stripeClient = getStripeClient();\n\n      return stripeClient.transfers\n        .create({\n          amount: formatAmount,\n          currency: formatCurrency,\n          destination: stripeAccountId\n        })\n        .then((stripeTransfer) => {\n          console.log(stripeTransfer);\n          const now: number = Date.now();\n          const insert: PaymentTransfer = {\n            added: now,\n            amount: formatAmount,\n            currency: formatCurrency,\n            modified: now,\n            userId: sessionId\n          };\n          const aqlQry: AqlQuery = aql`INSERT ${insert} IN transfers RETURN NEW`;\n\n          return useDb(databaseName).query(aqlQry)\n            .then((cursor) => cursor.next())\n            .then((newTransfer: PaymentTransfer) => newTransfer);\n        });\n    });\n};\n\nexport const createPaymentHold = (context: ApiContext, payment: PaymentCharge): Promise<PaymentCharge> => {\n  const {databaseName, session: {userId: sessionId}} = context;\n  const {amount, capture, cardId, currency, description} = payment;\n  const formatCurrency = parseChar(currency, 3, 'USD').toUpperCase();\n  const stripeClient = getStripeClient();\n\n  return stripeClient.charges\n    .create({\n      amount,\n      capture,\n      currency: formatCurrency,\n      description,\n      source: cardId\n    })\n    .then((stripeCharge) => {\n      const now: number = Date.now();\n      const insert: PaymentCharge = {\n        added: now,\n        amount,\n        capture,\n        cardId,\n        chargeFailCode: stripeCharge.failure_code,\n        chargeFailMsg: stripeCharge.failure_message,\n        chargeId: stripeCharge.id,\n        chargeStatus: stripeCharge.status,\n        currency: formatCurrency,\n        description,\n        modified: now,\n        userId: sessionId\n      };\n      const aqlQry: AqlQuery = aql`INSERT ${insert} IN payments RETURN NEW`;\n\n      return useDb(databaseName).query(aqlQry)\n        .then((cursor) => cursor.next())\n        .then((newPayment: PaymentCharge) => newPayment);\n    })\n    .catch((error: Error) => {\n      console.log('payments::createHold::error', error);\n      throw new UserError('payment_error');\n    });\n};\n\ninterface Card {\n  id: string;\n  brand?: string;\n  cvc_check?: string;\n  last4?: string;\n}\n"],
  "mappings": "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",
  "names": ["parseNum", "createHash", "parseChar", "parseId", "parseString", "parseVarChar", "aql", "ErrorTypes", "logError", "UserError", "useDb", "getStripeClient", "getUser", "eventCategory", "addCustomerAccount", "context", "action", "databaseName", "sessionId", "username", "customer", "update", "aqlQry", "cursor", "updatedUser", "error", "addBankAccount", "bankAccount", "accountNumber", "fullName", "routing", "formatAccount", "formatFullName", "formatRouting", "user", "stripeAccountId", "stripeClient", "token", "account", "now", "addCreditCard", "card", "city", "country", "cvc", "expMonth", "expYear", "street1", "street2", "state", "zip", "formatNumber", "formatExpMonth", "formatExpYear", "formatCvc", "paymentCard", "formatCity", "formatCountry", "formatStreet1", "formatStreet2", "formatState", "formatZip", "newSource", "cardSource", "brand", "cvcCheck", "last4", "insert", "insertAqlQry", "newCard", "cardId", "cardKey", "edgeCollection", "edgeId", "edge", "updateCreditCard", "id", "formatId", "results", "updatedCard", "stripeCustomerId", "stripeId", "getCreditCards", "list", "deleteCreditCard", "formatCardId", "result", "response", "edges", "edgeKey", "removeAqlQry", "deleteBankAccount", "bankId", "createPaymentTransfer", "transfer", "amount", "currency", "formatAmount", "formatCurrency", "stripeTransfer", "newTransfer", "createPaymentHold", "payment", "capture", "description", "stripeCharge", "newPayment"]
}

334
+ RETURN {user: user, card: card}`;
335
+ return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.next()).then((result = {
336
+ card: {},
337
+ user: {}
338
+ })=>{
339
+ if (!result) {
340
+ return false;
341
+ }
342
+ const { card, user } = result;
343
+ const { _key: cardKey } = card;
344
+ // Remove linked edges
345
+ const edgeCollection = useDb(databaseName).collection('hasPayment');
346
+ return edgeCollection.outEdges(cardKey, {}).then(async (response)=>{
347
+ // Extract edges from the response
348
+ const edges = Array.isArray(response) ? response : [];
349
+ if (edges.length) {
350
+ await Promise.all(edges.map((edge)=>{
351
+ const { _key: edgeKey } = edge;
352
+ const removeAqlQry = aql`REMOVE {_key:${edgeKey}} IN hasPayment`;
353
+ return useDb(databaseName).query(removeAqlQry);
354
+ })).then(()=>{
355
+ // Stripe
356
+ const stripeClient = getStripeClient();
357
+ return stripeClient.customers.deleteSource(user.stripeCustomerId, card.stripeId).then(()=>true).catch((error)=>{
358
+ console.log('payments::deleteCard::error', error);
359
+ throw new UserError('payment_error');
360
+ });
361
+ });
362
+ return true;
363
+ }
364
+ return false;
365
+ });
366
+ });
367
+ };
368
+ export const deleteBankAccount = (context, bankId)=>{
369
+ const { databaseName, session: { userId: sessionId } } = context;
370
+ // Clean db
371
+ const update = {
372
+ bankAccount: '',
373
+ bankFullName: '',
374
+ bankId: '',
375
+ bankRouting: '',
376
+ modified: Date.now()
377
+ };
378
+ const aqlQry = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;
379
+ return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.next()).then((user)=>{
380
+ const { stripeAccountId } = user;
381
+ const stripeClient = getStripeClient();
382
+ return stripeClient.customers.deleteSource(stripeAccountId, bankId).then(()=>true).catch(()=>Promise.resolve(false));
383
+ });
384
+ };
385
+ export const createPaymentTransfer = (context, transfer)=>{
386
+ const { databaseName, session: { userId: sessionId } } = context;
387
+ const { amount, currency } = transfer;
388
+ const formatAmount = parseNum(amount);
389
+ const formatCurrency = parseChar(currency, 3, 'USD').toUpperCase();
390
+ return getUser(context, {
391
+ userId: sessionId
392
+ }).then((user)=>{
393
+ const { stripeAccountId } = user;
394
+ const stripeClient = getStripeClient();
395
+ return stripeClient.transfers.create({
396
+ amount: formatAmount,
397
+ currency: formatCurrency,
398
+ destination: stripeAccountId
399
+ }).then((stripeTransfer)=>{
400
+ console.log(stripeTransfer);
401
+ const now = Date.now();
402
+ const insert = {
403
+ added: now,
404
+ amount: formatAmount,
405
+ currency: formatCurrency,
406
+ modified: now,
407
+ userId: sessionId
408
+ };
409
+ const aqlQry = aql`INSERT ${insert} IN transfers RETURN NEW`;
410
+ return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.next()).then((newTransfer)=>newTransfer);
411
+ });
412
+ });
413
+ };
414
+ export const createPaymentHold = (context, payment)=>{
415
+ const { databaseName, session: { userId: sessionId } } = context;
416
+ const { amount, capture, cardId, currency, description } = payment;
417
+ const formatCurrency = parseChar(currency, 3, 'USD').toUpperCase();
418
+ const stripeClient = getStripeClient();
419
+ return stripeClient.charges.create({
420
+ amount,
421
+ capture,
422
+ currency: formatCurrency,
423
+ description,
424
+ source: cardId
425
+ }).then((stripeCharge)=>{
426
+ const now = Date.now();
427
+ const insert = {
428
+ added: now,
429
+ amount,
430
+ capture,
431
+ cardId,
432
+ chargeFailCode: stripeCharge.failure_code,
433
+ chargeFailMsg: stripeCharge.failure_message,
434
+ chargeId: stripeCharge.id,
435
+ chargeStatus: stripeCharge.status,
436
+ currency: formatCurrency,
437
+ description,
438
+ modified: now,
439
+ userId: sessionId
440
+ };
441
+ const aqlQry = aql`INSERT ${insert} IN payments RETURN NEW`;
442
+ return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.next()).then((newPayment)=>newPayment);
443
+ }).catch((error)=>{
444
+ console.log('payments::createHold::error', error);
445
+ throw new UserError('payment_error');
446
+ });
447
+ };
448
+
449
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["/Users/nitrog7/Development/reaktor/src/actions/payments.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 {createHash, parseChar, parseId, parseString, parseVarChar} from '@nlabs/utils/parsers/strings';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport stripe from 'stripe';\n\nimport {ApiContext} from '../types/auth.types.js';\nimport {ErrorTypes} from '../types/error.types.js';\nimport {PaymentBankAccount, PaymentCardType, PaymentCharge, PaymentTransfer} from '../types/payments.types.js';\nimport {UserType} from '../types/users.types.js';\nimport {logError, UserError} from '../utils/analyticsUtils.js';\nimport {useDb} from '../utils/arangodbUtils.js';\nimport {getStripeClient} from '../utils/stripeUtils.js';\nimport {getUser} from './users.js';\n\nimport type {EdgeCollection} from 'arangojs/collections';\n\nconst eventCategory = 'payments';\n\nexport const addCustomerAccount = (context: ApiContext): Promise<boolean> => {\n  const action = 'addCustomerAccount';\n  const {databaseName, session: {userId: sessionId, username} = {}} = context;\n\n  const stripeClient = getStripeClient();\n\n  return stripeClient.customers\n    .create({\n      metadata: {\n        userId: sessionId,\n        username\n      }\n    } as any)\n    .then((customer) => {\n      // Create session\n      const now: number = Date.now();\n      const update: UserType = {\n        modified: now,\n        stripeCustomerId: customer.id\n      };\n\n      const aqlQry: AqlQuery = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;\n\n      return useDb(databaseName).query(aqlQry)\n        .then((cursor) => cursor.next())\n        .then((updatedUser: UserType) => !!updatedUser)\n        .catch((error: Error) => logError({\n          action,\n          category: eventCategory,\n          label: ErrorTypes.DATABASE_ERROR\n        }, error, context).then(() => null));\n    });\n};\n\nexport const addBankAccount = (context: ApiContext, bankAccount: PaymentBankAccount): Promise<boolean> => {\n  const action = 'addPaymentAccountBank';\n  const {databaseName, session: {userId: sessionId} = {}} = context;\n\n  // Params\n  const {\n    accountNumber,\n    fullName,\n    routing\n  } = bankAccount;\n\n  const formatAccount: string = parseString(accountNumber, 32);\n\n  if(formatAccount === '') {\n    throw new UserError('required_account_number');\n  }\n\n  const formatFullName: string = parseVarChar(fullName, 128);\n\n  if(formatFullName === '') {\n    throw new UserError('required_full_name');\n  }\n\n  const formatRouting: string = parseString(routing, 32);\n\n  if(formatRouting === '') {\n    throw new UserError('required_routing_number');\n  }\n\n  return getUser(context, {userId: sessionId})\n    .then((user: UserType) => {\n      const {stripeAccountId} = user;\n      const stripeClient = getStripeClient();\n\n      // Create a token first, then use the token as the source\n      return stripeClient.tokens.create({\n        bank_account: {\n          account_holder_name: formatFullName,\n          account_holder_type: 'individual',\n          account_number: formatAccount,\n          country: 'US',\n          currency: 'USD',\n          routing_number: formatRouting\n        }\n      })\n        .then((token) => stripeClient.customers.createSource(\n          stripeAccountId,\n          {source: token.id}\n        ))\n        .then((account) => {\n          // Use type assertion for card properties\n          // const cardSource = account as unknown as Card;\n          // const brand = cardSource.brand || '';\n          // const cvcCheck = cardSource.cvc_check || '';\n          // const last4 = cardSource.last4 || '';\n\n          // Create session\n          const now: number = Date.now();\n          const update = {\n            bankAccount,\n            bankFullName: formatFullName,\n            bankId: account.id,\n            bankRouting: formatRouting,\n            modified: now\n          };\n\n          const aqlQry: AqlQuery = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;\n\n          return useDb(databaseName).query(aqlQry)\n            .then((cursor) => cursor.next())\n            .then((updatedUser: UserType) => updatedUser);\n        })\n        .catch((error: Error) => {\n          const msg = error.message;\n\n          if(msg === 'A bank account with that routing number and account number ' +\n            'already exists for this customer.') {\n            return logError({\n              action,\n              category: eventCategory,\n              label: 'bank_account_exists'\n            }, error, context).then(() => null);\n          }\n          return logError({\n            action,\n            category: eventCategory,\n            label: 'payment_error'\n          }, error, context).then(() => null);\n        });\n    });\n};\n\nexport const addCreditCard = (context: ApiContext, card: PaymentCardType): Promise<UserType> => {\n  const action = 'addCreditCard';\n  const {databaseName, session: {userId: sessionId}} = context;\n\n  return getUser(context, {userId: sessionId})\n    .then((user: UserType) => {\n      // User\n      const {stripeAccountId} = user;\n\n      // Card\n      const {\n        accountNumber,\n        city,\n        country,\n        cvc,\n        expMonth,\n        expYear,\n        fullName,\n        street1,\n        street2,\n        state,\n        zip\n      }: PaymentCardType = card;\n\n      const formatNumber: number = parseNum(accountNumber, 16);\n\n      if(!formatNumber) {\n        throw new UserError('required_credit_card_number');\n      }\n\n      const formatExpMonth: number = parseNum(expMonth, 2);\n\n      if(!formatExpMonth) {\n        throw new UserError('required_credit_card_exp_month');\n      }\n\n      const formatExpYear: number = parseNum(expYear, 2);\n\n      if(!formatExpYear) {\n        throw new UserError('required_credit_card_exp_year');\n      }\n\n      const formatCvc: number = parseNum(cvc, 3);\n\n      // Address\n      const paymentCard: PaymentCardType = {};\n      const formatCity: string = parseVarChar(city, 32);\n\n      if(formatCity) {\n        paymentCard.city = formatCity;\n      }\n\n      const formatCountry: string = parseChar(country, 2);\n\n      if(formatCountry) {\n        paymentCard.country = formatCountry;\n      }\n\n      const formatFullName: string = parseVarChar(fullName, 32);\n\n      if(formatFullName) {\n        paymentCard.fullName = formatFullName;\n      }\n\n      const formatStreet1: string = parseVarChar(street1, 32);\n\n      if(formatStreet1) {\n        paymentCard.street1 = formatStreet1;\n      }\n\n      const formatStreet2: string = parseVarChar(street2, 32);\n\n      if(formatStreet2) {\n        paymentCard.street2 = formatStreet2;\n      }\n\n      const formatState: string = parseChar(state, 2);\n\n      if(formatState) {\n        paymentCard.state = formatState;\n      }\n\n      const formatZip: string = parseVarChar(zip, 10);\n\n      if(formatZip) {\n        paymentCard.zip = formatZip;\n      }\n\n      const stripeClient = getStripeClient();\n\n      // Create a token first, then use the token as the source\n      return stripeClient.tokens.create({\n        card: {\n          address_city: formatCity,\n          address_country: formatCountry,\n          address_line1: formatStreet1,\n          address_line2: formatStreet2,\n          address_state: formatState,\n          address_zip: formatZip,\n          cvc: formatCvc.toString(),\n          exp_month: formatExpMonth.toString(),\n          exp_year: formatExpYear.toString(),\n          name: fullName,\n          number: formatNumber.toString()\n        }\n      })\n        .then((token) => stripeClient.customers.createSource(\n          stripeAccountId,\n          {source: token.id}\n        ))\n        .then((newSource) => {\n          // Use type assertion for card properties\n          const cardSource = newSource as unknown as Card;\n          const brand = cardSource.brand || '';\n          const cvcCheck = cardSource.cvc_check || '';\n          const last4 = cardSource.last4 || '';\n\n          // Create session\n          const now: number = Date.now();\n          const insert = {\n            ...paymentCard,\n            _key: createHash(`user-payment-${sessionId}`),\n            accountNumber: last4,\n            added: now,\n            brand,\n            cvcCheck,\n            expMonth,\n            expYear,\n            modified: now,\n            userId: sessionId\n          };\n          const insertAqlQry: AqlQuery = aql`INSERT ${insert} IN creditCards RETURN NEW`;\n\n          return useDb(databaseName).query(insertAqlQry)\n            .then((cursor) => cursor.next())\n            .then((newCard: PaymentCardType) => {\n              if(newCard) {\n                // Add linked edge\n                const {_id: cardId, _key: cardKey} = card;\n                const edgeCollection: EdgeCollection = useDb(databaseName).collection('hasPayment');\n                const edgeId = createHash(`payment-${cardKey}`);\n                const edge = {\n                  _from: `users/${sessionId}`,\n                  _key: edgeId,\n                  _to: cardId\n                };\n\n                return edgeCollection.save(edge, {returnNew: true}).then(() => card);\n              }\n\n              return newCard;\n            })\n            .catch((error: Error) => logError({\n              action,\n              category: eventCategory,\n              label: 'payment_error'\n            }, error, context).then(() => null));\n        });\n    });\n};\n\nexport const updateCreditCard = (context: ApiContext, card: PaymentCardType): Promise<PaymentCardType> => {\n  const {databaseName, session: {userId: sessionId}} = context;\n\n  const {\n    city,\n    country,\n    expMonth,\n    expYear,\n    fullName,\n    id,\n    street1,\n    state,\n    zip\n  }: PaymentCardType = card;\n\n  const formatId: string = parseId(id);\n\n  if(formatId) {\n    throw new UserError('required_credit_card_id');\n  }\n\n  const paymentCard: PaymentCardType = {};\n  const formatExpMonth: number = parseNum(expMonth, 2);\n  const formatExpYear: number = parseNum(expYear, 2);\n  const formatCity: string = parseVarChar(city, 32);\n  const formatCountry: string = parseChar(country, 2);\n  const formatFullName: string = parseVarChar(fullName, 32);\n  const formatStreet1: string = parseString(street1, 32);\n  const formatState: string = parseChar(state, 2);\n  const formatZip: string = parseVarChar(zip, 10);\n\n  if(formatExpMonth) {\n    paymentCard.expMonth = formatExpMonth;\n  }\n\n  if(formatExpYear) {\n    paymentCard.expYear = formatExpYear;\n  }\n\n  if(formatCity) {\n    paymentCard.city = formatCity;\n  }\n\n  if(formatCountry) {\n    paymentCard.country = formatCountry;\n  }\n\n  if(formatFullName) {\n    paymentCard.fullName = formatFullName;\n  }\n\n  if(formatStreet1) {\n    paymentCard.street1 = formatStreet1;\n  }\n\n  if(formatState) {\n    paymentCard.state = formatState;\n  }\n\n  if(formatZip) {\n    paymentCard.zip = formatZip;\n  }\n\n  const update = paymentCard;\n  const aqlQry: AqlQuery = aql`\n      LET updatedCard = FIRST(\n        FOR c IN creditCards\n        FILTER c._key == ${formatId} && c.userId == ${sessionId}\n        UPDATE c WITH ${update} IN creditCards\n        LIMIT 1\n        RETURN NEW\n      )\n      LET user = FIRST(\n        FOR u IN users\n        FILTER u._key == ${sessionId}\n        LIMIT 1\n        RETURN u\n      )\n      RETURN {user: user, card: updatedCard}`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.next())\n    .then((results = {card: {}, user: {}}) => {\n      const updatedCard: PaymentCardType = results.card;\n      const {user} = results;\n\n      if(!updatedCard) {\n        throw new UserError('not_found');\n      }\n\n      const {stripeCustomerId} = user;\n      const {stripeId} = card;\n      const stripeClient = getStripeClient();\n      const update: stripe.CustomerUpdateSourceParams = {\n        address_city: formatCity,\n        address_country: formatCountry,\n        address_line1: formatStreet1,\n        address_state: formatState,\n        address_zip: formatZip,\n        exp_month: formatExpMonth.toString(),\n        exp_year: formatExpYear.toString(),\n        name: formatFullName\n      };\n\n      return stripeClient.customers\n        .updateSource(stripeCustomerId, stripeId, update)\n        .then(() => card)\n        .catch((error: Error) => {\n          console.log('payments::updateCard::error', error);\n          throw new UserError('payment_error');\n        });\n    });\n};\n\nexport const getCreditCards = (context: ApiContext): Promise<PaymentCardType[]> => {\n  const action = 'getCreditCards';\n  const {databaseName, session: {userId: sessionId}} = context;\n  const aqlQry: AqlQuery = aql`FOR c IN creditCards\n    FILTER c.userId == ${sessionId}\n    RETURN c`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.all())\n    .then((list: PaymentCardType[] = []) => list)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context).then(() => null));\n};\n\nexport const deleteCreditCard = (context: ApiContext, cardId: string): Promise<boolean> => {\n  const {databaseName, session: {userId: sessionId}} = context;\n  const formatCardId: string = parseId(cardId);\n  const aqlQry: AqlQuery = aql`\n    LET card = FIRST(\n      FOR c IN creditCards\n      FILTER c._key == ${formatCardId} && c.userId == ${sessionId}\n      LIMIT 1\n      REMOVE c IN creditCards\n      RETURN OLD\n    )\n    LET user = FIRST(\n      FOR u IN users\n      FILTER u._key == ${sessionId}\n      LIMIT 1\n      RETURN u\n    )\n    RETURN {user: user, card: card}`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.next())\n    .then((result = {card: {}, user: {}}) => {\n      if(!result) {\n        return false;\n      }\n\n      const {card, user} = result;\n      const {_key: cardKey} = card;\n\n      // Remove linked edges\n      const edgeCollection = useDb(databaseName).collection('hasPayment');\n\n      return edgeCollection.outEdges(cardKey, {})\n        .then(async (response) => {\n          // Extract edges from the response\n          const edges = Array.isArray(response) ? response : [];\n\n          if(edges.length) {\n            await Promise.all(\n              edges.map((edge) => {\n                const {_key: edgeKey} = edge;\n                const removeAqlQry: AqlQuery = aql`REMOVE {_key:${edgeKey}} IN hasPayment`;\n                return useDb(databaseName).query(removeAqlQry);\n              }))\n              .then(() => {\n                // Stripe\n                const stripeClient = getStripeClient();\n\n                return stripeClient.customers\n                  .deleteSource(user.stripeCustomerId, card.stripeId)\n                  .then(() => true)\n                  .catch((error: Error) => {\n                    console.log('payments::deleteCard::error', error);\n                    throw new UserError('payment_error');\n                  });\n              });\n\n            return true;\n          }\n\n          return false;\n        });\n    });\n};\n\nexport const deleteBankAccount = (context: ApiContext, bankId: string): Promise<boolean> => {\n  const {databaseName, session: {userId: sessionId}} = context;\n\n  // Clean db\n  const update: UserType = {\n    bankAccount: '',\n    bankFullName: '',\n    bankId: '',\n    bankRouting: '',\n    modified: Date.now()\n  };\n  const aqlQry: AqlQuery = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.next())\n    .then((user: UserType) => {\n      const {stripeAccountId} = user;\n      const stripeClient = getStripeClient();\n\n      return stripeClient.customers\n        .deleteSource(stripeAccountId, bankId)\n        .then(() => true)\n        .catch(() => Promise.resolve(false));\n    });\n};\n\nexport const createPaymentTransfer = (context: ApiContext, transfer: PaymentTransfer): Promise<PaymentTransfer> => {\n  const {databaseName, session: {userId: sessionId}} = context;\n  const {amount, currency} = transfer;\n  const formatAmount: number = parseNum(amount);\n  const formatCurrency: string = parseChar(currency, 3, 'USD').toUpperCase();\n\n  return getUser(context, {userId: sessionId})\n    .then((user: UserType) => {\n      const {stripeAccountId} = user;\n      const stripeClient = getStripeClient();\n\n      return stripeClient.transfers\n        .create({\n          amount: formatAmount,\n          currency: formatCurrency,\n          destination: stripeAccountId\n        })\n        .then((stripeTransfer) => {\n          console.log(stripeTransfer);\n          const now: number = Date.now();\n          const insert: PaymentTransfer = {\n            added: now,\n            amount: formatAmount,\n            currency: formatCurrency,\n            modified: now,\n            userId: sessionId\n          };\n          const aqlQry: AqlQuery = aql`INSERT ${insert} IN transfers RETURN NEW`;\n\n          return useDb(databaseName).query(aqlQry)\n            .then((cursor) => cursor.next())\n            .then((newTransfer: PaymentTransfer) => newTransfer);\n        });\n    });\n};\n\nexport const createPaymentHold = (context: ApiContext, payment: PaymentCharge): Promise<PaymentCharge> => {\n  const {databaseName, session: {userId: sessionId}} = context;\n  const {amount, capture, cardId, currency, description} = payment;\n  const formatCurrency = parseChar(currency, 3, 'USD').toUpperCase();\n  const stripeClient = getStripeClient();\n\n  return stripeClient.charges\n    .create({\n      amount,\n      capture,\n      currency: formatCurrency,\n      description,\n      source: cardId\n    })\n    .then((stripeCharge) => {\n      const now: number = Date.now();\n      const insert: PaymentCharge = {\n        added: now,\n        amount,\n        capture,\n        cardId,\n        chargeFailCode: stripeCharge.failure_code,\n        chargeFailMsg: stripeCharge.failure_message,\n        chargeId: stripeCharge.id,\n        chargeStatus: stripeCharge.status,\n        currency: formatCurrency,\n        description,\n        modified: now,\n        userId: sessionId\n      };\n      const aqlQry: AqlQuery = aql`INSERT ${insert} IN payments RETURN NEW`;\n\n      return useDb(databaseName).query(aqlQry)\n        .then((cursor) => cursor.next())\n        .then((newPayment: PaymentCharge) => newPayment);\n    })\n    .catch((error: Error) => {\n      console.log('payments::createHold::error', error);\n      throw new UserError('payment_error');\n    });\n};\n\ninterface Card {\n  id: string;\n  brand?: string;\n  cvc_check?: string;\n  last4?: string;\n}\n"],"names":["parseNum","createHash","parseChar","parseId","parseString","parseVarChar","aql","ErrorTypes","logError","UserError","useDb","getStripeClient","getUser","eventCategory","addCustomerAccount","context","action","databaseName","session","userId","sessionId","username","stripeClient","customers","create","metadata","then","customer","now","Date","update","modified","stripeCustomerId","id","aqlQry","query","cursor","next","updatedUser","catch","error","category","label","DATABASE_ERROR","addBankAccount","bankAccount","accountNumber","fullName","routing","formatAccount","formatFullName","formatRouting","user","stripeAccountId","tokens","bank_account","account_holder_name","account_holder_type","account_number","country","currency","routing_number","token","createSource","source","account","bankFullName","bankId","bankRouting","msg","message","addCreditCard","card","city","cvc","expMonth","expYear","street1","street2","state","zip","formatNumber","formatExpMonth","formatExpYear","formatCvc","paymentCard","formatCity","formatCountry","formatStreet1","formatStreet2","formatState","formatZip","address_city","address_country","address_line1","address_line2","address_state","address_zip","toString","exp_month","exp_year","name","number","newSource","cardSource","brand","cvcCheck","cvc_check","last4","insert","_key","added","insertAqlQry","newCard","_id","cardId","cardKey","edgeCollection","collection","edgeId","edge","_from","_to","save","returnNew","updateCreditCard","formatId","results","updatedCard","stripeId","updateSource","console","log","getCreditCards","all","list","deleteCreditCard","formatCardId","result","outEdges","response","edges","Array","isArray","length","Promise","map","edgeKey","removeAqlQry","deleteSource","deleteBankAccount","resolve","createPaymentTransfer","transfer","amount","formatAmount","formatCurrency","toUpperCase","transfers","destination","stripeTransfer","newTransfer","createPaymentHold","payment","capture","description","charges","stripeCharge","chargeFailCode","failure_code","chargeFailMsg","failure_message","chargeId","chargeStatus","status","newPayment"],"mappings":"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"}