@nlabs/reaktor 0.5.4 → 0.6.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 (73) hide show
  1. package/README.md +7 -7
  2. package/lib/actions/conversations.js +36 -47
  3. package/lib/actions/dynamodb.js +22 -33
  4. package/lib/actions/email.js +32 -29
  5. package/lib/actions/files.js +45 -53
  6. package/lib/actions/groups.js +33 -28
  7. package/lib/actions/images.js +53 -63
  8. package/lib/actions/index.js +54 -37
  9. package/lib/actions/ios.js +21 -40
  10. package/lib/actions/locations.js +25 -22
  11. package/lib/actions/messages.js +28 -41
  12. package/lib/actions/notifications.d.ts +1 -1
  13. package/lib/actions/notifications.js +13 -20
  14. package/lib/actions/payments.js +37 -48
  15. package/lib/actions/posts.js +19 -42
  16. package/lib/actions/reactions.d.ts +1 -1
  17. package/lib/actions/reactions.js +17 -40
  18. package/lib/actions/s3.js +21 -18
  19. package/lib/actions/search.js +26 -40
  20. package/lib/actions/sms.js +24 -21
  21. package/lib/actions/statistics.js +14 -21
  22. package/lib/actions/subscription.js +27 -24
  23. package/lib/actions/tags.js +47 -54
  24. package/lib/actions/users.js +81 -94
  25. package/lib/actions/websockets.js +26 -39
  26. package/lib/config.js +21 -18
  27. package/lib/index.js +24 -22
  28. package/lib/lambdas/actions/websockets.js +27 -40
  29. package/lib/lambdas/authorizer.js +21 -18
  30. package/lib/lambdas/connection.js +26 -40
  31. package/lib/lambdas/utils/message.js +20 -17
  32. package/lib/lambdas/utils/websocket.js +33 -25
  33. package/lib/templates/email/layout.js +16 -9
  34. package/lib/templates/email/passwordForgot.js +16 -9
  35. package/lib/templates/email/passwordRecovery.js +16 -9
  36. package/lib/templates/email/verifyEmail.js +16 -9
  37. package/lib/templates/email/welcome.js +16 -9
  38. package/lib/templates/sms/passwordForgot.js +16 -9
  39. package/lib/templates/sms/passwordRecovery.js +16 -9
  40. package/lib/templates/sms/verifyEmail.js +16 -9
  41. package/lib/templates/sms/verifyPhone.js +16 -9
  42. package/lib/templates/sms/welcome.js +16 -9
  43. package/lib/types/apps.d.ts +1 -1
  44. package/lib/types/apps.js +14 -5
  45. package/lib/types/arangodb.js +14 -5
  46. package/lib/types/auth.js +14 -5
  47. package/lib/types/connections.js +14 -5
  48. package/lib/types/conversations.js +14 -5
  49. package/lib/types/email.js +14 -5
  50. package/lib/types/files.js +14 -5
  51. package/lib/types/google.js +14 -5
  52. package/lib/types/groups.js +14 -5
  53. package/lib/types/images.js +14 -5
  54. package/lib/types/index.js +50 -35
  55. package/lib/types/locations.js +14 -5
  56. package/lib/types/messages.js +14 -5
  57. package/lib/types/notifications.js +14 -5
  58. package/lib/types/payments.d.ts +1 -1
  59. package/lib/types/payments.js +14 -5
  60. package/lib/types/posts.js +14 -5
  61. package/lib/types/statistics.js +14 -5
  62. package/lib/types/tags.js +14 -5
  63. package/lib/types/users.js +14 -5
  64. package/lib/types/websocket.js +14 -5
  65. package/lib/utils/analytics.js +24 -44
  66. package/lib/utils/arangodb.js +14 -21
  67. package/lib/utils/auth.js +22 -36
  68. package/lib/utils/graphql.js +20 -17
  69. package/lib/utils/index.js +26 -23
  70. package/lib/utils/objects.js +22 -19
  71. package/lib/utils/session.js +20 -17
  72. package/package.json +30 -32
  73. package/.eslintrc +0 -18
@@ -1,47 +1,32 @@
1
1
  var __create = Object.create;
2
2
  var __defProp = Object.defineProperty;
3
- var __defProps = Object.defineProperties;
4
3
  var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
5
- var __getOwnPropDescs = Object.getOwnPropertyDescriptors;
6
4
  var __getOwnPropNames = Object.getOwnPropertyNames;
7
- var __getOwnPropSymbols = Object.getOwnPropertySymbols;
8
5
  var __getProtoOf = Object.getPrototypeOf;
9
6
  var __hasOwnProp = Object.prototype.hasOwnProperty;
10
- var __propIsEnum = Object.prototype.propertyIsEnumerable;
11
- var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
12
- var __spreadValues = (a, b) => {
13
- for (var prop in b || (b = {}))
14
- if (__hasOwnProp.call(b, prop))
15
- __defNormalProp(a, prop, b[prop]);
16
- if (__getOwnPropSymbols)
17
- for (var prop of __getOwnPropSymbols(b)) {
18
- if (__propIsEnum.call(b, prop))
19
- __defNormalProp(a, prop, b[prop]);
20
- }
21
- return a;
22
- };
23
- var __spreadProps = (a, b) => __defProps(a, __getOwnPropDescs(b));
24
- var __markAsModule = (target) => __defProp(target, "__esModule", { value: true });
25
- var __require = typeof require !== "undefined" ? require : (x) => {
26
- throw new Error('Dynamic require of "' + x + '" is not supported');
27
- };
28
7
  var __export = (target, all) => {
29
- __markAsModule(target);
30
8
  for (var name in all)
31
9
  __defProp(target, name, { get: all[name], enumerable: true });
32
10
  };
33
- var __reExport = (target, module2, desc) => {
34
- if (module2 && typeof module2 === "object" || typeof module2 === "function") {
35
- for (let key of __getOwnPropNames(module2))
36
- if (!__hasOwnProp.call(target, key) && key !== "default")
37
- __defProp(target, key, { get: () => module2[key], enumerable: !(desc = __getOwnPropDesc(module2, key)) || desc.enumerable });
11
+ var __copyProps = (to, from, except, desc) => {
12
+ if (from && typeof from === "object" || typeof from === "function") {
13
+ for (let key of __getOwnPropNames(from))
14
+ if (!__hasOwnProp.call(to, key) && key !== except)
15
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
38
16
  }
39
- return target;
17
+ return to;
40
18
  };
41
- var __toModule = (module2) => {
42
- return __reExport(__markAsModule(__defProp(module2 != null ? __create(__getProtoOf(module2)) : {}, "default", module2 && module2.__esModule && "default" in module2 ? { get: () => module2.default, enumerable: true } : { value: module2, enumerable: true })), module2);
43
- };
44
- __export(exports, {
19
+ var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
20
+ // If the importer is in node compatibility mode or this is not an ESM
21
+ // file that has been converted to a CommonJS file using a Babel-
22
+ // compatible transform (i.e. "__esModule" has not been set), then set
23
+ // "default" to the CommonJS "module.exports" for node compatibility.
24
+ isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
25
+ mod
26
+ ));
27
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
28
+ var payments_exports = {};
29
+ __export(payments_exports, {
45
30
  addBankAccount: () => addBankAccount,
46
31
  addCreditCard: () => addCreditCard,
47
32
  addCustomerAccount: () => addCustomerAccount,
@@ -52,13 +37,14 @@ __export(exports, {
52
37
  getCreditCards: () => getCreditCards,
53
38
  updateCreditCard: () => updateCreditCard
54
39
  });
55
- var import_utils = __toModule(require("@nlabs/utils"));
56
- var import_arangojs = __toModule(require("arangojs"));
57
- var import_isEmpty = __toModule(require("lodash/isEmpty"));
58
- var import_stripe = __toModule(require("stripe"));
59
- var import_config = __toModule(require("../config"));
60
- var import_analytics = __toModule(require("../utils/analytics"));
61
- var import_users = __toModule(require("./users"));
40
+ module.exports = __toCommonJS(payments_exports);
41
+ var import_utils = require("@nlabs/utils");
42
+ var import_arangojs = require("arangojs");
43
+ var import_isEmpty = __toESM(require("lodash/isEmpty"));
44
+ var import_stripe = __toESM(require("stripe"));
45
+ var import_config = require("../config");
46
+ var import_analytics = require("../utils/analytics");
47
+ var import_users = require("./users");
62
48
  const eventCategory = "payments";
63
49
  const apiVersion = "2020-03-02";
64
50
  const addCustomerAccount = (context) => {
@@ -222,7 +208,8 @@ const addCreditCard = (context, card) => {
222
208
  return stripeClient.customers.createSource(stripeAccountId, { source }).then((newSource) => {
223
209
  const { brand, cvc_check: cvcCheck, last4 } = newSource;
224
210
  const now = Date.now();
225
- const insert = __spreadProps(__spreadValues({}, paymentCard), {
211
+ const insert = {
212
+ ...paymentCard,
226
213
  _key: (0, import_utils.createHash)(`user-payment-${sessionId}`),
227
214
  accountNumber: last4,
228
215
  added: now,
@@ -232,7 +219,7 @@ const addCreditCard = (context, card) => {
232
219
  expYear,
233
220
  modified: now,
234
221
  userId: sessionId
235
- });
222
+ };
236
223
  const insertAqlQry = import_arangojs.aql`INSERT ${insert} IN creditCards RETURN NEW`;
237
224
  return database.query(insertAqlQry).then((cursor) => cursor.next()).then((newCard = {}) => {
238
225
  if (!(0, import_isEmpty.default)(newCard)) {
@@ -383,13 +370,15 @@ const deleteCreditCard = (context, cardId) => {
383
370
  const { card, user } = result;
384
371
  const { _key: cardKey } = card;
385
372
  const edgeCollection = database.collection("hasPayment");
386
- return edgeCollection.outEdges(cardKey).then((edges) => {
373
+ return edgeCollection.outEdges(cardKey, {}).then((edges) => {
387
374
  if (edges.length) {
388
- return Promise.all(edges.map((edge) => {
389
- const { _key: edgeKey } = edge;
390
- const removeAqlQry = import_arangojs.aql`REMOVE {_key:${edgeKey}} IN hasPayment`;
391
- return database.query(removeAqlQry);
392
- })).then(() => {
375
+ return Promise.all(
376
+ edges.map((edge) => {
377
+ const { _key: edgeKey } = edge;
378
+ const removeAqlQry = import_arangojs.aql`REMOVE {_key:${edgeKey}} IN hasPayment`;
379
+ return database.query(removeAqlQry);
380
+ })
381
+ ).then(() => {
393
382
  const stripeClient = new import_stripe.default(import_config.Config.get("stripe.token"), { apiVersion, typescript: true });
394
383
  return stripeClient.customers.deleteSource(user.stripeCustomerId, card.stripeId).then(() => true).catch((error) => {
395
384
  console.log("payments::deleteCard::error", error);
@@ -490,4 +479,4 @@ const createPaymentHold = (context, payment) => {
490
479
  getCreditCards,
491
480
  updateCreditCard
492
481
  });
493
- //# 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 {createHash, parseChar, parseId, parseNum, parseString, parseVarChar} from '@nlabs/utils';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {EdgeCollection} from 'arangojs/collection';\nimport {ArrayCursor} from 'arangojs/cursor';\nimport isEmpty from 'lodash/isEmpty';\nimport Stripe from 'stripe';\n\nimport {Config} from '../config';\nimport {ApiContext, PaymentBankAccount, PaymentCardType, PaymentCharge, PaymentTransfer, User} from '../types';\nimport {logError, UserError} from '../utils/analytics';\nimport {getUser} from './users';\n\nconst eventCategory: string = 'payments';\nconst apiVersion: any = '2020-03-02';\n\nexport const addCustomerAccount = (context: ApiContext): Promise<boolean> => {\n  const action: string = 'addCustomerAccount';\n  const {database, session: {userId: sessionId, username}} = context;\n\n  // Stripe\n  const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion, typescript: true});\n\n  return stripeClient.customers\n    .create({\n      metadata: {\n        userId: sessionId,\n        username\n      }\n    })\n    .then((customer) => {\n      // Create session\n      const now: number = Date.now();\n      const update: User = {\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 database.query(aqlQry)\n        .then((cursor: ArrayCursor) => cursor.next())\n        .then((updatedUser: User = {}) => !isEmpty(updatedUser))\n        .catch((error: Error) => logError({\n          action,\n          category: eventCategory,\n          label: 'db_error'\n        }, error, context).then(() => null));\n    });\n};\n\nexport const addBankAccount = (context: ApiContext, bankAccount: PaymentBankAccount): Promise<boolean> => {\n  const action: string = 'addPaymentAccountBank';\n  const {database, 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, sessionId)\n    .then((user: User) => {\n      const {stripeAccountId}: User = user;\n\n      // Stripe\n      const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion, typescript: true});\n      const source: any = {\n        account_holder_name: formatFullName,\n        account_holder_type: 'individual',\n        account_number: formatAccount,\n        country: 'US',\n        currency: 'USD',\n        object: 'bank_account',\n        routing_number: formatRouting\n      };\n\n      return stripeClient.customers\n        .createSource(stripeAccountId, {source})\n        .then((account: any) => {\n          const {id: bankId, last4: bankAccount, routing_number: bankRouting} = account;\n          const update: any = {\n            bankAccount,\n            bankFullName: formatFullName,\n            bankId,\n            bankRouting,\n            modified: Date.now()\n          };\n\n          const aqlQry: AqlQuery = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;\n\n          return database.query(aqlQry)\n            .then((cursor: ArrayCursor) => cursor.next())\n            .then((updatedUser: User = {}) => 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<User> => {\n  const action: string = 'addCreditCard';\n  const {database, session: {userId: sessionId}} = context;\n\n  return getUser(context, sessionId)\n    .then((user: User) => {\n      // User\n      const {stripeAccountId}: User = 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      // Stripe\n      const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion, typescript: true});\n      const source: any = {\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,\n        exp_month: formatExpMonth,\n        exp_year: formatExpYear,\n        name: fullName,\n        number: formatNumber,\n        object: 'card'\n      };\n\n      return stripeClient.customers\n        .createSource(stripeAccountId, {source})\n        .then((newSource: any) => {\n          const {brand, cvc_check: cvcCheck, last4} = newSource;\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 database.query(insertAqlQry)\n            .then((cursor: ArrayCursor) => cursor.next())\n            .then((newCard: PaymentCardType = {}) => {\n              if(!isEmpty(newCard)) {\n                // Add linked edge\n                const {_id: cardId, _key: cardKey} = card;\n                const edgeCollection: EdgeCollection = database.collection('hasPayment');\n                const edgeId = createHash(`payment-${cardKey}`);\n                const edge: any = {\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 {database, 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: any = 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 database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((results = {card: {}, user: {}}) => {\n      const updatedCard: PaymentCardType = results.card;\n      const {user} = results;\n\n      if(isEmpty(updatedCard)) {\n        throw new UserError('not_found');\n      }\n\n      const {stripeCustomerId} = user;\n      const {stripeId} = card;\n\n      // Stripe\n      const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion, typescript: true});\n      const update: any = {\n        address_city: formatCity,\n        address_country: formatCountry,\n        address_line1: formatStreet1,\n        address_state: formatState,\n        address_zip: formatZip,\n        exp_month: formatExpMonth,\n        exp_year: formatExpYear,\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: string = 'getCreditCards';\n  const {database, session: {userId: sessionId}} = context;\n  const aqlQry: AqlQuery = aql`FOR c IN creditCards\n    FILTER c.userId == ${sessionId}\n    RETURN c`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .then((list: PaymentCardType[] = []) => list)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const deleteCreditCard = (context: ApiContext, cardId: string): Promise<boolean> => {\n  const {database, 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 database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((result = {card: {}, user: {}}) => {\n      if(isEmpty(result)) {\n        return false;\n      }\n\n      const {card, user} = result;\n      const {_key: cardKey} = card;\n\n      // Remove linked edges\n      const edgeCollection = database.collection('hasPayment');\n\n      return edgeCollection.outEdges(cardKey)\n        .then((edges: any) => {\n          if(edges.length) {\n            return Promise.all(\n              edges.map((edge) => {\n                const {_key: edgeKey} = edge;\n                const removeAqlQry: AqlQuery = aql`REMOVE {_key:${edgeKey}} IN hasPayment`;\n                return database.query(removeAqlQry);\n              }))\n              .then(() => {\n                // Stripe\n                const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion, typescript: true});\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\n          return false;\n        });\n    });\n};\n\nexport const deleteBankAccount = (context: ApiContext, bankId: string): Promise<boolean> => {\n  const {database, session: {userId: sessionId}} = context;\n\n  // Clean db\n  const update: User = {\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 database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((user: User = {}) => {\n      const {stripeAccountId} = user;\n\n      // Stripe\n      const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion, typescript: true});\n\n      return stripeClient.customers\n        .deleteSource(stripeAccountId, bankId)\n        .then((response: any = {deleted: false}) => response.deleted)\n        .catch(() => Promise.resolve(false));\n    });\n};\n\nexport const createPaymentTransfer = (context: ApiContext, transfer: PaymentTransfer): Promise<PaymentTransfer> => {\n  const {database, 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, sessionId)\n    .then((user: User) => {\n      const {stripeAccountId} = user;\n\n      // Stripe\n      const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion, typescript: true});\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 database.query(aqlQry)\n            .then((cursor: ArrayCursor) => cursor.next())\n            .then((newTransfer: PaymentTransfer) => newTransfer);\n        });\n    });\n};\n\nexport const createPaymentHold = (context: ApiContext, payment: PaymentCharge): Promise<PaymentCharge> => {\n  const {database, session: {userId: sessionId}} = context;\n  const {amount, capture, cardId, currency, description} = payment;\n  const formatCurrency = parseChar(currency, 3, 'USD').toUpperCase();\n\n  const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion, typescript: true});\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 database.query(aqlQry)\n        .then((cursor: ArrayCursor) => 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"],
  "mappings": "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",
  "names": []
}

482
+ //# 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 {createHash, parseChar, parseId, parseNum, parseString, parseVarChar} from '@nlabs/utils';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {EdgeCollection} from 'arangojs/collection';\nimport {ArrayCursor} from 'arangojs/cursor';\nimport isEmpty from 'lodash/isEmpty';\nimport Stripe from 'stripe';\n\nimport {Config} from '../config';\nimport {ApiContext, PaymentBankAccount, PaymentCardType, PaymentCharge, PaymentTransfer, User} from '../types';\nimport {logError, UserError} from '../utils/analytics';\nimport {getUser} from './users';\n\nconst eventCategory: string = 'payments';\nconst apiVersion: any = '2020-03-02';\n\nexport const addCustomerAccount = (context: ApiContext): Promise<boolean> => {\n  const action: string = 'addCustomerAccount';\n  const {database, session: {userId: sessionId, username}} = context;\n\n  // Stripe\n  const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion, typescript: true});\n\n  return stripeClient.customers\n    .create({\n      metadata: {\n        userId: sessionId,\n        username\n      }\n    })\n    .then((customer) => {\n      // Create session\n      const now: number = Date.now();\n      const update: User = {\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 database.query(aqlQry)\n        .then((cursor: ArrayCursor) => cursor.next())\n        .then((updatedUser: User = {}) => !isEmpty(updatedUser))\n        .catch((error: Error) => logError({\n          action,\n          category: eventCategory,\n          label: 'db_error'\n        }, error, context).then(() => null));\n    });\n};\n\nexport const addBankAccount = (context: ApiContext, bankAccount: PaymentBankAccount): Promise<boolean> => {\n  const action: string = 'addPaymentAccountBank';\n  const {database, 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, sessionId)\n    .then((user: User) => {\n      const {stripeAccountId}: User = user;\n\n      // Stripe\n      const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion, typescript: true});\n      const source: any = {\n        account_holder_name: formatFullName,\n        account_holder_type: 'individual',\n        account_number: formatAccount,\n        country: 'US',\n        currency: 'USD',\n        object: 'bank_account',\n        routing_number: formatRouting\n      };\n\n      return stripeClient.customers\n        .createSource(stripeAccountId, {source})\n        .then((account: any) => {\n          const {id: bankId, last4: bankAccount, routing_number: bankRouting} = account;\n          const update: any = {\n            bankAccount,\n            bankFullName: formatFullName,\n            bankId,\n            bankRouting,\n            modified: Date.now()\n          };\n\n          const aqlQry: AqlQuery = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;\n\n          return database.query(aqlQry)\n            .then((cursor: ArrayCursor) => cursor.next())\n            .then((updatedUser: User = {}) => 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<User> => {\n  const action: string = 'addCreditCard';\n  const {database, session: {userId: sessionId}} = context;\n\n  return getUser(context, sessionId)\n    .then((user: User) => {\n      // User\n      const {stripeAccountId}: User = 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      // Stripe\n      const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion, typescript: true});\n      const source: any = {\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,\n        exp_month: formatExpMonth,\n        exp_year: formatExpYear,\n        name: fullName,\n        number: formatNumber,\n        object: 'card'\n      };\n\n      return stripeClient.customers\n        .createSource(stripeAccountId, {source})\n        .then((newSource: any) => {\n          const {brand, cvc_check: cvcCheck, last4} = newSource;\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 database.query(insertAqlQry)\n            .then((cursor: ArrayCursor) => cursor.next())\n            .then((newCard: PaymentCardType = {}) => {\n              if(!isEmpty(newCard)) {\n                // Add linked edge\n                const {_id: cardId, _key: cardKey} = card;\n                const edgeCollection: EdgeCollection = database.collection('hasPayment');\n                const edgeId = createHash(`payment-${cardKey}`);\n                const edge: any = {\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 {database, 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: any = 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 database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((results = {card: {}, user: {}}) => {\n      const updatedCard: PaymentCardType = results.card;\n      const {user} = results;\n\n      if(isEmpty(updatedCard)) {\n        throw new UserError('not_found');\n      }\n\n      const {stripeCustomerId} = user;\n      const {stripeId} = card;\n\n      // Stripe\n      const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion, typescript: true});\n      const update: any = {\n        address_city: formatCity,\n        address_country: formatCountry,\n        address_line1: formatStreet1,\n        address_state: formatState,\n        address_zip: formatZip,\n        exp_month: formatExpMonth,\n        exp_year: formatExpYear,\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: string = 'getCreditCards';\n  const {database, session: {userId: sessionId}} = context;\n  const aqlQry: AqlQuery = aql`FOR c IN creditCards\n    FILTER c.userId == ${sessionId}\n    RETURN c`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .then((list: PaymentCardType[] = []) => list)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const deleteCreditCard = (context: ApiContext, cardId: string): Promise<boolean> => {\n  const {database, 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 database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((result = {card: {}, user: {}}) => {\n      if(isEmpty(result)) {\n        return false;\n      }\n\n      const {card, user} = result;\n      const {_key: cardKey} = card;\n\n      // Remove linked edges\n      const edgeCollection = database.collection('hasPayment');\n\n      return edgeCollection.outEdges(cardKey, {})\n        .then((edges: any) => {\n          if(edges.length) {\n            return Promise.all(\n              edges.map((edge) => {\n                const {_key: edgeKey} = edge;\n                const removeAqlQry: AqlQuery = aql`REMOVE {_key:${edgeKey}} IN hasPayment`;\n                return database.query(removeAqlQry);\n              }))\n              .then(() => {\n                // Stripe\n                const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion, typescript: true});\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\n          return false;\n        });\n    });\n};\n\nexport const deleteBankAccount = (context: ApiContext, bankId: string): Promise<boolean> => {\n  const {database, session: {userId: sessionId}} = context;\n\n  // Clean db\n  const update: User = {\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 database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((user: User = {}) => {\n      const {stripeAccountId} = user;\n\n      // Stripe\n      const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion, typescript: true});\n\n      return stripeClient.customers\n        .deleteSource(stripeAccountId, bankId)\n        .then((response: any = {deleted: false}) => response.deleted)\n        .catch(() => Promise.resolve(false));\n    });\n};\n\nexport const createPaymentTransfer = (context: ApiContext, transfer: PaymentTransfer): Promise<PaymentTransfer> => {\n  const {database, 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, sessionId)\n    .then((user: User) => {\n      const {stripeAccountId} = user;\n\n      // Stripe\n      const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion, typescript: true});\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 database.query(aqlQry)\n            .then((cursor: ArrayCursor) => cursor.next())\n            .then((newTransfer: PaymentTransfer) => newTransfer);\n        });\n    });\n};\n\nexport const createPaymentHold = (context: ApiContext, payment: PaymentCharge): Promise<PaymentCharge> => {\n  const {database, session: {userId: sessionId}} = context;\n  const {amount, capture, cardId, currency, description} = payment;\n  const formatCurrency = parseChar(currency, 3, 'USD').toUpperCase();\n\n  const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion, typescript: true});\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 database.query(aqlQry)\n        .then((cursor: ArrayCursor) => 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"],
  "mappings": "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",
  "names": ["Stripe", "isEmpty", "bankAccount", "update"]
}

@@ -1,47 +1,22 @@
1
- var __create = Object.create;
2
1
  var __defProp = Object.defineProperty;
3
- var __defProps = Object.defineProperties;
4
2
  var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
5
- var __getOwnPropDescs = Object.getOwnPropertyDescriptors;
6
3
  var __getOwnPropNames = Object.getOwnPropertyNames;
7
- var __getOwnPropSymbols = Object.getOwnPropertySymbols;
8
- var __getProtoOf = Object.getPrototypeOf;
9
4
  var __hasOwnProp = Object.prototype.hasOwnProperty;
10
- var __propIsEnum = Object.prototype.propertyIsEnumerable;
11
- var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
12
- var __spreadValues = (a, b) => {
13
- for (var prop in b || (b = {}))
14
- if (__hasOwnProp.call(b, prop))
15
- __defNormalProp(a, prop, b[prop]);
16
- if (__getOwnPropSymbols)
17
- for (var prop of __getOwnPropSymbols(b)) {
18
- if (__propIsEnum.call(b, prop))
19
- __defNormalProp(a, prop, b[prop]);
20
- }
21
- return a;
22
- };
23
- var __spreadProps = (a, b) => __defProps(a, __getOwnPropDescs(b));
24
- var __markAsModule = (target) => __defProp(target, "__esModule", { value: true });
25
- var __require = typeof require !== "undefined" ? require : (x) => {
26
- throw new Error('Dynamic require of "' + x + '" is not supported');
27
- };
28
5
  var __export = (target, all) => {
29
- __markAsModule(target);
30
6
  for (var name in all)
31
7
  __defProp(target, name, { get: all[name], enumerable: true });
32
8
  };
33
- var __reExport = (target, module2, desc) => {
34
- if (module2 && typeof module2 === "object" || typeof module2 === "function") {
35
- for (let key of __getOwnPropNames(module2))
36
- if (!__hasOwnProp.call(target, key) && key !== "default")
37
- __defProp(target, key, { get: () => module2[key], enumerable: !(desc = __getOwnPropDesc(module2, key)) || desc.enumerable });
9
+ var __copyProps = (to, from, except, desc) => {
10
+ if (from && typeof from === "object" || typeof from === "function") {
11
+ for (let key of __getOwnPropNames(from))
12
+ if (!__hasOwnProp.call(to, key) && key !== except)
13
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
38
14
  }
39
- return target;
40
- };
41
- var __toModule = (module2) => {
42
- return __reExport(__markAsModule(__defProp(module2 != null ? __create(__getProtoOf(module2)) : {}, "default", module2 && module2.__esModule && "default" in module2 ? { get: () => module2.default, enumerable: true } : { value: module2, enumerable: true })), module2);
15
+ return to;
43
16
  };
44
- __export(exports, {
17
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
18
+ var posts_exports = {};
19
+ __export(posts_exports, {
45
20
  addPost: () => addPost,
46
21
  cleanPosts: () => cleanPosts,
47
22
  createPostEdge: () => createPostEdge,
@@ -55,11 +30,12 @@ __export(exports, {
55
30
  parsePostOptions: () => parsePostOptions,
56
31
  updatePost: () => updatePost
57
32
  });
58
- var import_utils = __toModule(require("@nlabs/utils"));
59
- var import_arangojs = __toModule(require("arangojs"));
60
- var import_utils2 = __toModule(require("../utils"));
61
- var import_files = __toModule(require("./files"));
62
- var import_tags = __toModule(require("./tags"));
33
+ module.exports = __toCommonJS(posts_exports);
34
+ var import_utils = require("@nlabs/utils");
35
+ var import_arangojs = require("arangojs");
36
+ var import_utils2 = require("../utils");
37
+ var import_files = require("./files");
38
+ var import_tags = require("./tags");
63
39
  const MAX_CONTENT_LENGTH = 1e5;
64
40
  const eventCategory = "posts";
65
41
  const parsePostOptions = (options = {}) => {
@@ -427,13 +403,14 @@ const updatePost = async (context, post) => {
427
403
  let formatId = (0, import_utils.parseId)(postId);
428
404
  formatId = formatId === "" ? (0, import_utils.createHash)(`post-${sessionId}`) : formatId;
429
405
  const formatGroupId = (0, import_utils.parseId)(groupId);
430
- const insert = __spreadProps(__spreadValues({}, update), {
406
+ const insert = {
407
+ ...update,
431
408
  _key: formatId,
432
409
  added: now,
433
410
  groupId: formatGroupId,
434
411
  privacy,
435
412
  userId: sessionId
436
- });
413
+ };
437
414
  const db = database;
438
415
  const aqlQry = import_arangojs.aql`UPSERT {_key: ${formatId}, userId: ${sessionId}}
439
416
  INSERT ${insert}
@@ -538,4 +515,4 @@ const createPostEdge = (db, file, postId) => {
538
515
  parsePostOptions,
539
516
  updatePost
540
517
  });
541
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/posts.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, parseChar, parseId, parseNum, parseString, parseVarChar} from '@nlabs/utils';\nimport {aql, Database} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {EdgeCollection} from 'arangojs/collection';\nimport {ArrayCursor} from 'arangojs/cursor';\n\nimport {ApiContext, FileType, PostInputType, PostOptions, PostType, TagType} from '../types';\nimport {getLimit, logError} from '../utils';\nimport {updateFiles} from './files';\nimport {extractTags, linkTags} from './tags';\n\nconst MAX_CONTENT_LENGTH: number = 100000;\nconst eventCategory: string = 'posts';\n\nexport const parsePostOptions = (options: PostOptions = {}) => {\n  const {\n    from = 0,\n    latitude = 0,\n    longitude = 0,\n    to = 30,\n    type = 'post'\n  } = options;\n\n  return {\n    latitude: parseNum(latitude, 32),\n    limit: getLimit(from, to),\n    longitude: parseNum(longitude, 32),\n    type: parseChar(type, 32)\n  };\n};\n\nexport const getPostOptional = (fields: string[], sessionId: string) =>\n  fields.reduce((selects: any, field: string) => {\n    switch(field) {\n      case 'hasRsvp': {\n        selects.queries.push(`LET hasRsvp = TO_BOOL(FIRST(\n          FOR post, r IN INBOUND p._id hasReactions\n          FILTER r.name == \"rsvp\" && r.type == \"posts\" && r._from == \"users/${sessionId}\"\n          COLLECT WITH COUNT INTO count\n          RETURN count\n        ))`);\n        selects.objects.push('hasRsvp:hasRsvp');\n        return selects;\n      }\n      case 'isSaved': {\n        selects.queries.push(`LET isSaved = TO_BOOL(FIRST(\n          FOR post, r IN INBOUND p._id hasReactions\n          FILTER r.name == \"pin\" && r.type == \"posts\" && r._from == \"users/${sessionId}\"\n          COLLECT WITH COUNT INTO count\n          RETURN count\n        ))`);\n        selects.objects.push('isSaved:isSaved');\n        return selects;\n      }\n      case 'reactions': {\n        selects.queries.push(`LET reactions = (\n          FOR post, r IN INBOUND p._id hasReactions\n          COLLECT reactionName = r.value INTO reactionItems\n          RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}\n        )`);\n        selects.objects.push('reactions:reactions');\n        return selects;\n      }\n      case 'rsvpCount': {\n        selects.queries.push(`LET rsvpCount = FIRST(\n          FOR post, r IN INBOUND p._id hasReactions\n          FILTER r.name == \"rsvp\" && r.type == \"posts\"\n          COLLECT WITH COUNT INTO count\n          RETURN count\n        )`);\n        selects.objects.push('rsvpCount:rsvpCount');\n        return selects;\n      }\n      case 'viewCount': {\n        selects.queries.push(`LET viewCount = FIRST(\n          FOR post, r IN INBOUND p._id hasReactions\n          FILTER r.name == \"view\" && r.type == \"posts\"\n          COLLECT WITH COUNT INTO count\n          RETURN count\n        )`);\n        selects.objects.push('viewCount:viewCount');\n        return selects;\n      }\n      default: {\n        return selects;\n      }\n    }\n  }, {objects: [], queries: []});\n\nexport const getPost = (context: ApiContext, itemId: string, options: PostOptions): Promise<Partial<PostType>> => {\n  const action: string = 'getPost';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const formatItemId: string = parseId(itemId);\n  const {type} = parsePostOptions(options);\n  const db = database;\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  const aqlQry: AqlQuery = aql`FOR p IN posts\n    FILTER p._key == ${formatItemId} && p.type == ${type}\n    LIMIT 1\n    RETURN p`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((post: PostType = {}) => {\n      const {\n        _id: postDocId,\n        groupId,\n        privacy = 'default'\n      }: PostType = post;\n\n      // Query based on privacy level\n      let privacyAqlQry: string;\n\n      if(groupId && privacy === 'group') {\n        privacyAqlQry = `LET p = DOCUMENT(\"${postDocId}\")\n          ${selectQueries.join('\\n')}\n          FOR group IN groups\n          FILTER group._key == p.groupId\n          FOR u, e IN OUTBOUND group._id isGrouped\n          FILTER u._key == \"${sessionId}\"\n          LIMIT 1\n          RETURN MERGE(p, {${selectObjects.join(', ')}})`;\n      } else if(privacy === 'public') {\n        privacyAqlQry = `LET p = DOCUMENT(\"${postDocId}\")\n          ${selectQueries.join('\\n')}\n          LIMIT 1\n          RETURN MERGE(p, {${selectObjects.join(', ')}})`;\n      }\n\n      if(privacyAqlQry) {\n        return db.query(privacyAqlQry)\n          .then((cursor: ArrayCursor) => cursor.next() || {})\n          .catch((error: Error) => logError({\n            action,\n            category: eventCategory,\n            label: 'db_error'\n          }, error, context).then(() => ({})));\n      }\n\n      return {};\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => ({})));\n};\n\n// export const getPostList = (context: ApiContext, options?: PostOptions): Promise<PostType[]> => {\n//   // const action: string = 'getListByApp';\n//   const {database, fields, session: {userId: sessionId}} = context;\n//   console.log('getPostList::context', context);\n//   const {limit, type} = parsePostOptions(options);\n//   const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n//   const aqlQry: string = `FOR p IN posts\n//     FILTER p.type == \"${type}\" && p.privacy == \"public\" && p.parent == null\n//     ${selectQueries.join('\\n')}\n//     ${limit.aql}\n//     SORT p.added\n//     RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n//   return database.query(aqlQry)\n//     .then((cursor: ArrayCursor) => cursor.all())\n//     .catch((error: Error) => {\n//       console.log('getPostList::error', error);\n//       throw error;\n//     });\n// };\n\n// export const getPostsByArea = (\n//   context: ApiContext,\n//   latitude: number,\n//   longitude: number,\n//   options?: PostOptions\n// ): Promise<PostType[]> => {\n//   // const action: string = 'getListByUser';\n//   const {database, fields, session: {userId: sessionId}} = context;\n//   const {limit, type} = parsePostOptions(options);\n//   const formatLatitude: number = parseNum(latitude);\n//   const formatLongitude: number = parseNum(longitude);\n//   const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n//   selectQueries.push(`LET distance = DISTANCE(\n//     ${formatLatitude},\n//     ${formatLongitude},\n//     NOT_NULL(p.latitude, 0),\n//     NOT_NULL(p.longitude, 0))\n//   `);\n//   selectObjects.push('distance:distance');\n\n//   const aqlQry: string = `FOR p IN posts\n//     ${selectQueries.join('\\n')}\n//     FILTER p.type == \"${type}\" && p.privacy == \"public\" && p.parentId == null\n//     ${limit.aql}\n//     SORT distance, p.added\n//     RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n//   return database.query(aqlQry)\n//     .then((cursor: ArrayCursor) => cursor.all())\n//     .catch((error: Error) => {\n//       throw error;\n//     });\n// };\n\n// export const getPostsByGroup = (\n//   context: ApiContext,\n//   groupId: string,\n//   options?: PostOptions\n// ): Promise<PostType[]> => {\n//   // const action: string = 'getListByGroup';\n//   const {database, fields, session: {userId: sessionId}} = context;\n//   const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n\n//   // Group id\n//   const formatGroupId: string = parseId(groupId);\n//   const db = database;\n//   const aqlQry: string = `FOR u, g IN INBOUND ${formatGroupId} hasGroup\n//       FILTER u._key == ${sessionId}\n//       RETURN g`;\n\n//   return db.query(aqlQry)\n//     .then((cursor: ArrayCursor) => cursor.all())\n//     .then((groups: GroupType[] = []) => {\n//       if(groups.length) {\n//         const {limit, type} = parsePostOptions(options);\n//         const postAqlQry: string = `FOR p IN posts\n//           FILTER p.type == \"${type}\" && p.groupId == \"${formatGroupId}\" && p.parent == null\n//           ${selectQueries.join('\\n')}\n//           ${limit.aql}\n//           SORT p.added\n//           RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n//         return db.query(postAqlQry)\n//           .then((cursor: ArrayCursor) => cursor.all())\n//           .catch((error: Error) => {\n//             throw error;\n//           });\n//       }\n\n//       return [];\n//     })\n//     .catch((error: Error) => {\n//       throw error;\n//     });\n// };\n\n// export const getPostsByLatest = (context: ApiContext, options?: PostOptions): Promise<PostType[]> => {\n//   // const action: string = 'getListByLatest';\n//   console.log('getPostsByLatest::options', options);\n//   const {database, fields, session: {userId: sessionId}} = context;\n//   const {limit, type} = parsePostOptions(options);\n//   const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n//   const aqlQry: string = `FOR p IN posts\n//     FILTER p.type == \"${type}\" && p.privacy == \"public\" && p.parent == null\n//     ${selectQueries.join('\\n')}\n//     ${limit.aql}\n//     SORT p.added DESC\n//     RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n//   console.log('getPostsByLatest::aqlQry', aqlQry);\n//   return database.query(aqlQry)\n//     .then((cursor: ArrayCursor) => cursor.all())\n//     .catch((error: Error) => {\n//       throw error;\n//     });\n// };\n\nexport const getPostsByReactions = (\n  context: ApiContext,\n  reactions: string[] = [],\n  options?: PostOptions\n): Promise<PostType[]> => {\n  const action: string = 'getPostsByReactions';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const {latitude, limit, longitude, type} = parsePostOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  const formatSessionId: string = `users/${sessionId}`;\n  const formatReactions: string = JSON.stringify(reactions.map((reaction) => parseChar(reaction, 32).toLowerCase()));\n  const sortBy: string[] = [];\n  const filters: string[] = [`p.type == \"${type}\"`, 'p.privacy == \"public\"'];\n  const formatLatitude: number = parseNum(latitude);\n  const formatLongitude: number = parseNum(longitude);\n\n  if(formatLatitude && formatLongitude) {\n    selectQueries.push(`LET distance = DISTANCE(\n      ${formatLatitude},\n      ${formatLongitude},\n      NOT_NULL(p.latitude, 0),\n      NOT_NULL(p.longitude, 0))\n    `);\n    selectObjects.push('distance:distance');\n    sortBy.push('distance');\n  }\n\n  if(reactions.length) {\n    sortBy.push('matchedTags DESC');\n    selectQueries.push(`LET matchedReactions = LENGTH(\n      FOR mr IN reactions\n      FILTER mr.matched == true\n      RETURN mr\n    )`);\n    selectObjects.push('matchedReactions:matchedReactions');\n    filters.push('matchedReactions > 0');\n  }\n\n  sortBy.push('p.added DESC');\n  selectObjects.push('reactions:reactions');\n\n  // Get data from database\n  const aqlQry: string = `FOR p, r IN OUTBOUND \"${formatSessionId}\" hasReactions\n    LET reactions = (\n      FOR reaction, hr IN 1..1 INBOUND p isTagged\n      LET matched = LENGTH(${formatReactions}) > 0 && POSITION(${formatReactions}, reaction.name)\n      SORT reaction.name\n      RETURN MERGE(reaction, {matched:matched})\n    )\n    ${selectQueries.join('\\n')}\n    FILTER ${filters.join(' && ')}\n    ${limit.aql}\n    RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => []));\n};\n\nexport const getPostsByTags = (\n  context: ApiContext,\n  tags: string[] = [],\n  options?: PostOptions\n): Promise<PostType[]> => {\n  // const action: string = 'getListByTags';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const {latitude, limit, longitude, type} = parsePostOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  const formatTagNames: string = JSON.stringify(tags.map((tag) => parseChar(tag, 32).toLowerCase()));\n  const sortBy: string[] = [];\n  const filters: string[] = [`p.type == \"${type}\"`, 'p.privacy == \"public\"'];\n  const formatLatitude: number = parseNum(latitude);\n  const formatLongitude: number = parseNum(longitude);\n\n  if(formatLatitude && formatLongitude) {\n    selectQueries.push(`LET distance = DISTANCE(\n      ${formatLatitude},\n      ${formatLongitude},\n      NOT_NULL(p.latitude, 0),\n      NOT_NULL(p.longitude, 0))\n    `);\n    selectObjects.push('distance:distance');\n    sortBy.push('distance');\n  }\n\n  if(tags.length) {\n    sortBy.push('matchedTags DESC');\n    selectQueries.push(`LET matchedTags = LENGTH(\n      FOR t IN tags\n      FILTER t.matched == true\n      RETURN t\n    )`);\n    selectObjects.push('matchedTags:matchedTags');\n    filters.push('matchedTags > 0');\n  }\n\n  sortBy.push('p.added DESC');\n  selectObjects.push('tags:tags');\n\n  const aqlQry: string = `FOR p IN posts\n    LET tags = (\n      FOR tag, it IN 1..1 INBOUND p isTagged\n      LET matched = LENGTH(${formatTagNames}) > 0 && POSITION(${formatTagNames}, tag.name)\n      SORT tag.name\n      RETURN MERGE(tag, {matched:matched})\n    )\n    ${selectQueries.join('\\n')}\n    FILTER ${filters.join(' && ')}\n    ${limit.aql}\n    SORT ${sortBy.join(', ')}\n    RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n  console.log({aqlQry, options});\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch(() => []);\n};\n\nexport const getPostsByUser = (context: ApiContext, userId: string, options?: PostOptions): Promise<PostType[]> => {\n  // const action: string = 'getListByUser';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const {limit, type} = parsePostOptions(options);\n  const formatUserId: string = parseId(userId);\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  const aqlQry: string = `FOR p IN posts\n    FILTER p.userId == \"${formatUserId}\" && p.type == \"${type}\" && p.privacy == \"public\" && p.parent == null\n    ${selectQueries.join('\\n')}\n    ${limit.aql}\n    SORT p.added\n    RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const getPostComments = (context: ApiContext, itemId: string, options?: PostOptions): Promise<PostType[]> => {\n  // const action: string = 'getComments';\n  const {database, session: {userId: sessionId}} = context;\n  const {limit, type} = parsePostOptions(options);\n  const formatItemId: string = parseId(itemId);\n\n  // Get the parent post to get restrictions\n  const db = database;\n  const aqlQry: AqlQuery = aql`FOR p IN posts\n    FILTER p.type == ${type} && p._key == ${formatItemId}\n    LIMIT 1\n    RETURN p`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((post: PostType = {}) => {\n      const {\n        _key,\n        groupId,\n        privacy = 'public'\n      }: PostType = post;\n\n      // Query based on privacy level\n      let privacyAqlQry: string;\n\n      if(groupId && privacy === 'group') {\n        privacyAqlQry = `FOR p IN posts\n          FOR user IN users\n          FILTER p.parent == \"${_key}\" && user._key == p.userId\n          LET reactions = (\n            FOR post, r IN INBOUND p._id reactions\n            COLLECT reactionName = r.value INTO reactionItems\n            RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}\n          )\n          FOR group IN groups\n          FILTER group._key == p.groupId\n          FOR u, e IN OUTBOUND group._id isGrouped\n          FILTER u._key == \"${sessionId}\"\n          SORT p.added\n          ${limit.aql}\n          RETURN MERGE(p, {user: user, reactions: reactions})`;\n      } else if(privacy === 'public') {\n        privacyAqlQry = `FOR p IN posts\n          FOR user IN users\n          FILTER p.parent == \"${_key}\" && user._key == p.userId\n          LET reactions = (\n            FOR post, r IN INBOUND p._id reactions\n            COLLECT reactionName = r.value INTO reactionItems\n            RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}\n          )\n          SORT p.added\n          ${limit.aql}\n          RETURN MERGE(p, {user: user, reactions: reactions})`;\n      }\n\n      if(privacyAqlQry) {\n        return db.query(privacyAqlQry)\n          .then((cursor: ArrayCursor) => cursor.all())\n          .catch((error: Error) => {\n            throw error;\n          });\n      }\n\n      return [];\n    })\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const addPost = async (context: ApiContext, post: PostInputType): Promise<PostType> => {\n  // const action: string = 'add';\n  const {database, session: {userId: sessionId}} = context;\n\n  const {\n    content = '',\n    endDate,\n    groupId = '',\n    location,\n    latitude,\n    longitude,\n    name = '',\n    parentId = null,\n    privacy = 'public',\n    tags = [],\n    startDate,\n    type = 'default'\n  } = post;\n\n  const now: number = Date.now();\n\n  const insert: PostType = {\n    _key: createHash(`post-${sessionId}`),\n    added: now,\n    content: parseString(content, MAX_CONTENT_LENGTH),\n    endDate: endDate ? parseNum(endDate, 13) : undefined,\n    groupId: groupId ? parseId(groupId) : undefined,\n    latitude: latitude !== undefined ? parseNum(latitude) : undefined,\n    location: location ? parseString(location, 160) : undefined,\n    longitude: longitude !== undefined ? parseNum(longitude) : undefined,\n    modified: now,\n    name: parseString(name, 160),\n    parentId: parentId ? parseId(parentId) : undefined,\n    privacy: privacy ? parseVarChar(privacy, 16) : undefined,\n    startDate: startDate ? parseNum(startDate, 13) : undefined,\n    type: parseChar(type, 32),\n    userId: sessionId\n  };\n\n  const db: Database = database;\n  const aqlQry: AqlQuery = aql`INSERT ${insert} IN posts RETURN NEW`;\n\n  try {\n    const savedPost: PostType = await db.query(aqlQry)\n      .then((cursor: ArrayCursor) => cursor.next() || {});\n\n    const {_id: postDocId} = savedPost;\n\n    console.log({tags});\n    if(tags && tags.length) {\n      const tagNames: string[] = tags.map(({name}) => parseChar(name, 32));\n      console.log({tagNames, postDocId});\n      await linkTags(db, tagNames, postDocId);\n    } else {\n      // Update linked tags within posts\n      const tagList: TagType[] = await extractTags(db, postDocId, insert.content);\n      savedPost.tags = tagList;\n    }\n\n    return savedPost;\n  } catch(error) {\n    throw error;\n  }\n};\n\nexport const updatePost = async (context: ApiContext, post: PostInputType): Promise<PostType> => {\n  // const action: string = 'update';\n  const {database, session: {userId: sessionId}} = context;\n  const now: number = Date.now();\n  const {\n    content,\n    endDate,\n    groupId,\n    name,\n    parentId,\n    postId,\n    privacy,\n    startDate,\n    tags = [],\n    type\n  } = post;\n\n  const update: PostType = {\n    content: content ? parseString(content, MAX_CONTENT_LENGTH) : undefined,\n    endDate: endDate ? parseNum(endDate, 13) : undefined,\n    modified: now,\n    name: name ? parseString(name, 160) : undefined,\n    parentId: parentId ? parseString(parentId, 160) : undefined,\n    privacy: privacy ? parseVarChar(privacy, 16) : undefined,\n    startDate: startDate ? parseNum(startDate, 13) : undefined,\n    type: type !== undefined ? parseChar(type, 16) : undefined\n  };\n\n  let formatId: string = parseId(postId);\n  formatId = formatId === '' ? createHash(`post-${sessionId}`) : formatId;\n  const formatGroupId: string = parseId(groupId);\n  const insert: any = {\n    ...update,\n    _key: formatId,\n    added: now,\n    groupId: formatGroupId,\n    privacy,\n    userId: sessionId\n  };\n  const db: Database = database;\n  const aqlQry: AqlQuery = aql`UPSERT {_key: ${formatId}, userId: ${sessionId}}\n    INSERT ${insert}\n    UPDATE ${update}\n    IN posts RETURN NEW`;\n\n  try {\n    const updatedPost: PostType = await db.query(aqlQry)\n      .then((cursor: ArrayCursor) => cursor.next() || {});\n\n    const {_id: updatedPostId} = updatedPost;\n\n    if(tags && tags.length) {\n      const tagNames: string[] = tags.map(({name}) => parseChar(name, 32));\n      const tagQuery: AqlQuery = aql`FOR t IN tags\n        FILTER POSITION(${JSON.stringify(tagNames)}, t.name)\n        RETURN t`;\n\n      const existingTags = await db.query(tagQuery).then((cursor: ArrayCursor) => cursor.all());\n      const tagIds: string[] = existingTags.map(({_id: tagId}) => tagId);\n      await linkTags(db, tagIds, updatedPostId);\n    } else {\n      // Update linked tags\n      const tagList = await extractTags(db, updatedPostId, update.content || '') || [];\n      updatedPost.tags = tagList;\n    }\n\n    // Update linked files\n    const files: FileType[] = updatedPost.files || [];\n\n    if(files.length) {\n      const fileList = await updateFiles(db, formatId, files) || [];\n      updatedPost.files = fileList;\n    } else {\n      updatedPost.files = [];\n    }\n\n    return updatedPost;\n  } catch(error) {\n    throw error;\n  }\n};\n\nexport const deletePost = (context: ApiContext, postId: string): Promise<PostType> => {\n  // const action: string = 'delete';\n  const {database, session: {userId: sessionId}} = context;\n  const formatItemId: string = parseId(postId);\n  const db: Database = database;\n  const aqlQry = aql`FOR p IN posts\n      FILTER p._key == ${formatItemId} && p.userId == ${sessionId}\n      LIMIT 1\n      REMOVE p IN posts\n      RETURN OLD`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((post: PostType = {}) => {\n      if(post) {\n        // Remove tag links\n        const edgeAqlQry: AqlQuery = aql`FOR t IN isTagged\n            FILTER t._to == ${formatItemId}\n            REMOVE t IN isTagged`;\n\n        return db.query(edgeAqlQry)\n          .then(() => {\n            // Remove attached files\n            const fileAqlQry: AqlQuery = aql`FOR f IN hasFile\n                FILTER f._to == ${formatItemId}\n                REMOVE f IN hasFile`;\n\n            return db.query(fileAqlQry)\n              .then(() => post)\n              .catch((error: Error) => {\n                throw error;\n              });\n          })\n          .catch((error: Error) => {\n            throw error;\n          });\n      }\n      return {};\n    })\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const cleanPosts = (database: Database): Promise<number> => {\n  // Remove all messages that are over 60 days and not saved\n  const aqlQry: AqlQuery = aql`FOR p IN posts\n      FILTER p.added < DATE_TIMESTAMP(DATE_SUBTRACT(DATE_NOW(), 60, 'day')) && p.type == \"default\"\n      REMOVE p IN posts\n      RETURN OLD`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .then((results: PostType[] = []) => results.length)\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const createPostEdge = (db: Database, file: FileType, postId: string): Promise<FileType> => {\n  const edgeCollection: EdgeCollection = db.collection('isPosted');\n  const {fileId} = file;\n  const formatFileId: string = parseId(fileId);\n  const edgeId: string = createHash(`file-${postId}-${formatFileId}`);\n  const formatPostId: string = parseId(postId);\n  const fileType: string = parseChar(file.fileType, 16);\n\n  const edge: any = {\n    _from: `posts/${formatPostId}`,\n    _key: edgeId,\n    _to: `files/${formatFileId}`,\n    added: Date.now(),\n    type: fileType\n  };\n\n  return edgeCollection.save(edge, {returnNew: true})\n    .then(() => file)\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n"],
  "mappings": "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",
  "names": []
}

518
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/posts.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, parseChar, parseId, parseNum, parseString, parseVarChar} from '@nlabs/utils';\nimport {aql, Database} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {EdgeCollection} from 'arangojs/collection';\nimport {ArrayCursor} from 'arangojs/cursor';\n\nimport {ApiContext, FileType, PostInputType, PostOptions, PostType, TagType} from '../types';\nimport {getLimit, logError} from '../utils';\nimport {updateFiles} from './files';\nimport {extractTags, linkTags} from './tags';\n\nconst MAX_CONTENT_LENGTH: number = 100000;\nconst eventCategory: string = 'posts';\n\nexport const parsePostOptions = (options: PostOptions = {}) => {\n  const {\n    from = 0,\n    latitude = 0,\n    longitude = 0,\n    to = 30,\n    type = 'post'\n  } = options;\n\n  return {\n    latitude: parseNum(latitude, 32),\n    limit: getLimit(from, to),\n    longitude: parseNum(longitude, 32),\n    type: parseChar(type, 32)\n  };\n};\n\nexport const getPostOptional = (fields: string[], sessionId: string) =>\n  fields.reduce((selects: any, field: string) => {\n    switch(field) {\n      case 'hasRsvp': {\n        selects.queries.push(`LET hasRsvp = TO_BOOL(FIRST(\n          FOR post, r IN INBOUND p._id hasReactions\n          FILTER r.name == \"rsvp\" && r.type == \"posts\" && r._from == \"users/${sessionId}\"\n          COLLECT WITH COUNT INTO count\n          RETURN count\n        ))`);\n        selects.objects.push('hasRsvp:hasRsvp');\n        return selects;\n      }\n      case 'isSaved': {\n        selects.queries.push(`LET isSaved = TO_BOOL(FIRST(\n          FOR post, r IN INBOUND p._id hasReactions\n          FILTER r.name == \"pin\" && r.type == \"posts\" && r._from == \"users/${sessionId}\"\n          COLLECT WITH COUNT INTO count\n          RETURN count\n        ))`);\n        selects.objects.push('isSaved:isSaved');\n        return selects;\n      }\n      case 'reactions': {\n        selects.queries.push(`LET reactions = (\n          FOR post, r IN INBOUND p._id hasReactions\n          COLLECT reactionName = r.value INTO reactionItems\n          RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}\n        )`);\n        selects.objects.push('reactions:reactions');\n        return selects;\n      }\n      case 'rsvpCount': {\n        selects.queries.push(`LET rsvpCount = FIRST(\n          FOR post, r IN INBOUND p._id hasReactions\n          FILTER r.name == \"rsvp\" && r.type == \"posts\"\n          COLLECT WITH COUNT INTO count\n          RETURN count\n        )`);\n        selects.objects.push('rsvpCount:rsvpCount');\n        return selects;\n      }\n      case 'viewCount': {\n        selects.queries.push(`LET viewCount = FIRST(\n          FOR post, r IN INBOUND p._id hasReactions\n          FILTER r.name == \"view\" && r.type == \"posts\"\n          COLLECT WITH COUNT INTO count\n          RETURN count\n        )`);\n        selects.objects.push('viewCount:viewCount');\n        return selects;\n      }\n      default: {\n        return selects;\n      }\n    }\n  }, {objects: [], queries: []});\n\nexport const getPost = (context: ApiContext, itemId: string, options: PostOptions): Promise<Partial<PostType>> => {\n  const action: string = 'getPost';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const formatItemId: string = parseId(itemId);\n  const {type} = parsePostOptions(options);\n  const db = database;\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  const aqlQry: AqlQuery = aql`FOR p IN posts\n    FILTER p._key == ${formatItemId} && p.type == ${type}\n    LIMIT 1\n    RETURN p`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((post: PostType = {}) => {\n      const {\n        _id: postDocId,\n        groupId,\n        privacy = 'default'\n      }: PostType = post;\n\n      // Query based on privacy level\n      let privacyAqlQry: string;\n\n      if(groupId && privacy === 'group') {\n        privacyAqlQry = `LET p = DOCUMENT(\"${postDocId}\")\n          ${selectQueries.join('\\n')}\n          FOR group IN groups\n          FILTER group._key == p.groupId\n          FOR u, e IN OUTBOUND group._id isGrouped\n          FILTER u._key == \"${sessionId}\"\n          LIMIT 1\n          RETURN MERGE(p, {${selectObjects.join(', ')}})`;\n      } else if(privacy === 'public') {\n        privacyAqlQry = `LET p = DOCUMENT(\"${postDocId}\")\n          ${selectQueries.join('\\n')}\n          LIMIT 1\n          RETURN MERGE(p, {${selectObjects.join(', ')}})`;\n      }\n\n      if(privacyAqlQry) {\n        return db.query(privacyAqlQry)\n          .then((cursor: ArrayCursor) => cursor.next() || {})\n          .catch((error: Error) => logError({\n            action,\n            category: eventCategory,\n            label: 'db_error'\n          }, error, context).then(() => ({})));\n      }\n\n      return {};\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => ({})));\n};\n\n// export const getPostList = (context: ApiContext, options?: PostOptions): Promise<PostType[]> => {\n//   // const action: string = 'getListByApp';\n//   const {database, fields, session: {userId: sessionId}} = context;\n//   console.log('getPostList::context', context);\n//   const {limit, type} = parsePostOptions(options);\n//   const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n//   const aqlQry: string = `FOR p IN posts\n//     FILTER p.type == \"${type}\" && p.privacy == \"public\" && p.parent == null\n//     ${selectQueries.join('\\n')}\n//     ${limit.aql}\n//     SORT p.added\n//     RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n//   return database.query(aqlQry)\n//     .then((cursor: ArrayCursor) => cursor.all())\n//     .catch((error: Error) => {\n//       console.log('getPostList::error', error);\n//       throw error;\n//     });\n// };\n\n// export const getPostsByArea = (\n//   context: ApiContext,\n//   latitude: number,\n//   longitude: number,\n//   options?: PostOptions\n// ): Promise<PostType[]> => {\n//   // const action: string = 'getListByUser';\n//   const {database, fields, session: {userId: sessionId}} = context;\n//   const {limit, type} = parsePostOptions(options);\n//   const formatLatitude: number = parseNum(latitude);\n//   const formatLongitude: number = parseNum(longitude);\n//   const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n//   selectQueries.push(`LET distance = DISTANCE(\n//     ${formatLatitude},\n//     ${formatLongitude},\n//     NOT_NULL(p.latitude, 0),\n//     NOT_NULL(p.longitude, 0))\n//   `);\n//   selectObjects.push('distance:distance');\n\n//   const aqlQry: string = `FOR p IN posts\n//     ${selectQueries.join('\\n')}\n//     FILTER p.type == \"${type}\" && p.privacy == \"public\" && p.parentId == null\n//     ${limit.aql}\n//     SORT distance, p.added\n//     RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n//   return database.query(aqlQry)\n//     .then((cursor: ArrayCursor) => cursor.all())\n//     .catch((error: Error) => {\n//       throw error;\n//     });\n// };\n\n// export const getPostsByGroup = (\n//   context: ApiContext,\n//   groupId: string,\n//   options?: PostOptions\n// ): Promise<PostType[]> => {\n//   // const action: string = 'getListByGroup';\n//   const {database, fields, session: {userId: sessionId}} = context;\n//   const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n\n//   // Group id\n//   const formatGroupId: string = parseId(groupId);\n//   const db = database;\n//   const aqlQry: string = `FOR u, g IN INBOUND ${formatGroupId} hasGroup\n//       FILTER u._key == ${sessionId}\n//       RETURN g`;\n\n//   return db.query(aqlQry)\n//     .then((cursor: ArrayCursor) => cursor.all())\n//     .then((groups: GroupType[] = []) => {\n//       if(groups.length) {\n//         const {limit, type} = parsePostOptions(options);\n//         const postAqlQry: string = `FOR p IN posts\n//           FILTER p.type == \"${type}\" && p.groupId == \"${formatGroupId}\" && p.parent == null\n//           ${selectQueries.join('\\n')}\n//           ${limit.aql}\n//           SORT p.added\n//           RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n//         return db.query(postAqlQry)\n//           .then((cursor: ArrayCursor) => cursor.all())\n//           .catch((error: Error) => {\n//             throw error;\n//           });\n//       }\n\n//       return [];\n//     })\n//     .catch((error: Error) => {\n//       throw error;\n//     });\n// };\n\n// export const getPostsByLatest = (context: ApiContext, options?: PostOptions): Promise<PostType[]> => {\n//   // const action: string = 'getListByLatest';\n//   console.log('getPostsByLatest::options', options);\n//   const {database, fields, session: {userId: sessionId}} = context;\n//   const {limit, type} = parsePostOptions(options);\n//   const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n//   const aqlQry: string = `FOR p IN posts\n//     FILTER p.type == \"${type}\" && p.privacy == \"public\" && p.parent == null\n//     ${selectQueries.join('\\n')}\n//     ${limit.aql}\n//     SORT p.added DESC\n//     RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n//   console.log('getPostsByLatest::aqlQry', aqlQry);\n//   return database.query(aqlQry)\n//     .then((cursor: ArrayCursor) => cursor.all())\n//     .catch((error: Error) => {\n//       throw error;\n//     });\n// };\n\nexport const getPostsByReactions = (\n  context: ApiContext,\n  reactions: string[] = [],\n  options?: PostOptions\n): Promise<PostType[]> => {\n  const action: string = 'getPostsByReactions';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const {latitude, limit, longitude, type} = parsePostOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  const formatSessionId: string = `users/${sessionId}`;\n  const formatReactions: string = JSON.stringify(reactions.map((reaction) => parseChar(reaction, 32).toLowerCase()));\n  const sortBy: string[] = [];\n  const filters: string[] = [`p.type == \"${type}\"`, 'p.privacy == \"public\"'];\n  const formatLatitude: number = parseNum(latitude);\n  const formatLongitude: number = parseNum(longitude);\n\n  if(formatLatitude && formatLongitude) {\n    selectQueries.push(`LET distance = DISTANCE(\n      ${formatLatitude},\n      ${formatLongitude},\n      NOT_NULL(p.latitude, 0),\n      NOT_NULL(p.longitude, 0))\n    `);\n    selectObjects.push('distance:distance');\n    sortBy.push('distance');\n  }\n\n  if(reactions.length) {\n    sortBy.push('matchedTags DESC');\n    selectQueries.push(`LET matchedReactions = LENGTH(\n      FOR mr IN reactions\n      FILTER mr.matched == true\n      RETURN mr\n    )`);\n    selectObjects.push('matchedReactions:matchedReactions');\n    filters.push('matchedReactions > 0');\n  }\n\n  sortBy.push('p.added DESC');\n  selectObjects.push('reactions:reactions');\n\n  // Get data from database\n  const aqlQry: string = `FOR p, r IN OUTBOUND \"${formatSessionId}\" hasReactions\n    LET reactions = (\n      FOR reaction, hr IN 1..1 INBOUND p isTagged\n      LET matched = LENGTH(${formatReactions}) > 0 && POSITION(${formatReactions}, reaction.name)\n      SORT reaction.name\n      RETURN MERGE(reaction, {matched:matched})\n    )\n    ${selectQueries.join('\\n')}\n    FILTER ${filters.join(' && ')}\n    ${limit.aql}\n    RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => []));\n};\n\nexport const getPostsByTags = (\n  context: ApiContext,\n  tags: string[] = [],\n  options?: PostOptions\n): Promise<PostType[]> => {\n  // const action: string = 'getListByTags';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const {latitude, limit, longitude, type} = parsePostOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  const formatTagNames: string = JSON.stringify(tags.map((tag) => parseChar(tag, 32).toLowerCase()));\n  const sortBy: string[] = [];\n  const filters: string[] = [`p.type == \"${type}\"`, 'p.privacy == \"public\"'];\n  const formatLatitude: number = parseNum(latitude);\n  const formatLongitude: number = parseNum(longitude);\n\n  if(formatLatitude && formatLongitude) {\n    selectQueries.push(`LET distance = DISTANCE(\n      ${formatLatitude},\n      ${formatLongitude},\n      NOT_NULL(p.latitude, 0),\n      NOT_NULL(p.longitude, 0))\n    `);\n    selectObjects.push('distance:distance');\n    sortBy.push('distance');\n  }\n\n  if(tags.length) {\n    sortBy.push('matchedTags DESC');\n    selectQueries.push(`LET matchedTags = LENGTH(\n      FOR t IN tags\n      FILTER t.matched == true\n      RETURN t\n    )`);\n    selectObjects.push('matchedTags:matchedTags');\n    filters.push('matchedTags > 0');\n  }\n\n  sortBy.push('p.added DESC');\n  selectObjects.push('tags:tags');\n\n  const aqlQry: string = `FOR p IN posts\n    LET tags = (\n      FOR tag, it IN 1..1 INBOUND p isTagged\n      LET matched = LENGTH(${formatTagNames}) > 0 && POSITION(${formatTagNames}, tag.name)\n      SORT tag.name\n      RETURN MERGE(tag, {matched:matched})\n    )\n    ${selectQueries.join('\\n')}\n    FILTER ${filters.join(' && ')}\n    ${limit.aql}\n    SORT ${sortBy.join(', ')}\n    RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n  console.log({aqlQry, options});\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch(() => []);\n};\n\nexport const getPostsByUser = (context: ApiContext, userId: string, options?: PostOptions): Promise<PostType[]> => {\n  // const action: string = 'getListByUser';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const {limit, type} = parsePostOptions(options);\n  const formatUserId: string = parseId(userId);\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  const aqlQry: string = `FOR p IN posts\n    FILTER p.userId == \"${formatUserId}\" && p.type == \"${type}\" && p.privacy == \"public\" && p.parent == null\n    ${selectQueries.join('\\n')}\n    ${limit.aql}\n    SORT p.added\n    RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const getPostComments = (context: ApiContext, itemId: string, options?: PostOptions): Promise<PostType[]> => {\n  // const action: string = 'getComments';\n  const {database, session: {userId: sessionId}} = context;\n  const {limit, type} = parsePostOptions(options);\n  const formatItemId: string = parseId(itemId);\n\n  // Get the parent post to get restrictions\n  const db = database;\n  const aqlQry: AqlQuery = aql`FOR p IN posts\n    FILTER p.type == ${type} && p._key == ${formatItemId}\n    LIMIT 1\n    RETURN p`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((post: PostType = {}) => {\n      const {\n        _key,\n        groupId,\n        privacy = 'public'\n      }: PostType = post;\n\n      // Query based on privacy level\n      let privacyAqlQry: string;\n\n      if(groupId && privacy === 'group') {\n        privacyAqlQry = `FOR p IN posts\n          FOR user IN users\n          FILTER p.parent == \"${_key}\" && user._key == p.userId\n          LET reactions = (\n            FOR post, r IN INBOUND p._id reactions\n            COLLECT reactionName = r.value INTO reactionItems\n            RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}\n          )\n          FOR group IN groups\n          FILTER group._key == p.groupId\n          FOR u, e IN OUTBOUND group._id isGrouped\n          FILTER u._key == \"${sessionId}\"\n          SORT p.added\n          ${limit.aql}\n          RETURN MERGE(p, {user: user, reactions: reactions})`;\n      } else if(privacy === 'public') {\n        privacyAqlQry = `FOR p IN posts\n          FOR user IN users\n          FILTER p.parent == \"${_key}\" && user._key == p.userId\n          LET reactions = (\n            FOR post, r IN INBOUND p._id reactions\n            COLLECT reactionName = r.value INTO reactionItems\n            RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}\n          )\n          SORT p.added\n          ${limit.aql}\n          RETURN MERGE(p, {user: user, reactions: reactions})`;\n      }\n\n      if(privacyAqlQry) {\n        return db.query(privacyAqlQry)\n          .then((cursor: ArrayCursor) => cursor.all())\n          .catch((error: Error) => {\n            throw error;\n          });\n      }\n\n      return [];\n    })\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const addPost = async (context: ApiContext, post: PostInputType): Promise<PostType> => {\n  // const action: string = 'add';\n  const {database, session: {userId: sessionId}} = context;\n\n  const {\n    content = '',\n    endDate,\n    groupId = '',\n    location,\n    latitude,\n    longitude,\n    name = '',\n    parentId = null,\n    privacy = 'public',\n    tags = [],\n    startDate,\n    type = 'default'\n  } = post;\n\n  const now: number = Date.now();\n\n  const insert: PostType = {\n    _key: createHash(`post-${sessionId}`),\n    added: now,\n    content: parseString(content, MAX_CONTENT_LENGTH),\n    endDate: endDate ? parseNum(endDate, 13) : undefined,\n    groupId: groupId ? parseId(groupId) : undefined,\n    latitude: latitude !== undefined ? parseNum(latitude) : undefined,\n    location: location ? parseString(location, 160) : undefined,\n    longitude: longitude !== undefined ? parseNum(longitude) : undefined,\n    modified: now,\n    name: parseString(name, 160),\n    parentId: parentId ? parseId(parentId) : undefined,\n    privacy: privacy ? parseVarChar(privacy, 16) : undefined,\n    startDate: startDate ? parseNum(startDate, 13) : undefined,\n    type: parseChar(type, 32),\n    userId: sessionId\n  };\n\n  const db: Database = database;\n  const aqlQry: AqlQuery = aql`INSERT ${insert} IN posts RETURN NEW`;\n\n  try {\n    const savedPost: PostType = await db.query(aqlQry)\n      .then((cursor: ArrayCursor) => cursor.next() || {});\n\n    const {_id: postDocId} = savedPost;\n\n    console.log({tags});\n    if(tags && tags.length) {\n      const tagNames: string[] = tags.map(({name}) => parseChar(name, 32));\n      console.log({tagNames, postDocId});\n      await linkTags(db, tagNames, postDocId);\n    } else {\n      // Update linked tags within posts\n      const tagList: TagType[] = await extractTags(db, postDocId, insert.content);\n      savedPost.tags = tagList;\n    }\n\n    return savedPost;\n  } catch(error) {\n    throw error;\n  }\n};\n\nexport const updatePost = async (context: ApiContext, post: PostInputType): Promise<PostType> => {\n  // const action: string = 'update';\n  const {database, session: {userId: sessionId}} = context;\n  const now: number = Date.now();\n  const {\n    content,\n    endDate,\n    groupId,\n    name,\n    parentId,\n    postId,\n    privacy,\n    startDate,\n    tags = [],\n    type\n  } = post;\n\n  const update: PostType = {\n    content: content ? parseString(content, MAX_CONTENT_LENGTH) : undefined,\n    endDate: endDate ? parseNum(endDate, 13) : undefined,\n    modified: now,\n    name: name ? parseString(name, 160) : undefined,\n    parentId: parentId ? parseString(parentId, 160) : undefined,\n    privacy: privacy ? parseVarChar(privacy, 16) : undefined,\n    startDate: startDate ? parseNum(startDate, 13) : undefined,\n    type: type !== undefined ? parseChar(type, 16) : undefined\n  };\n\n  let formatId: string = parseId(postId);\n  formatId = formatId === '' ? createHash(`post-${sessionId}`) : formatId;\n  const formatGroupId: string = parseId(groupId);\n  const insert: any = {\n    ...update,\n    _key: formatId,\n    added: now,\n    groupId: formatGroupId,\n    privacy,\n    userId: sessionId\n  };\n  const db: Database = database;\n  const aqlQry: AqlQuery = aql`UPSERT {_key: ${formatId}, userId: ${sessionId}}\n    INSERT ${insert}\n    UPDATE ${update}\n    IN posts RETURN NEW`;\n\n  try {\n    const updatedPost: PostType = await db.query(aqlQry)\n      .then((cursor: ArrayCursor) => cursor.next() || {});\n\n    const {_id: updatedPostId} = updatedPost;\n\n    if(tags && tags.length) {\n      const tagNames: string[] = tags.map(({name}) => parseChar(name, 32));\n      const tagQuery: AqlQuery = aql`FOR t IN tags\n        FILTER POSITION(${JSON.stringify(tagNames)}, t.name)\n        RETURN t`;\n\n      const existingTags = await db.query(tagQuery).then((cursor: ArrayCursor) => cursor.all());\n      const tagIds: string[] = existingTags.map(({_id: tagId}) => tagId);\n      await linkTags(db, tagIds, updatedPostId);\n    } else {\n      // Update linked tags\n      const tagList = await extractTags(db, updatedPostId, update.content || '') || [];\n      updatedPost.tags = tagList;\n    }\n\n    // Update linked files\n    const files: FileType[] = updatedPost.files || [];\n\n    if(files.length) {\n      const fileList = await updateFiles(db, formatId, files) || [];\n      updatedPost.files = fileList;\n    } else {\n      updatedPost.files = [];\n    }\n\n    return updatedPost;\n  } catch(error) {\n    throw error;\n  }\n};\n\nexport const deletePost = (context: ApiContext, postId: string): Promise<PostType> => {\n  // const action: string = 'delete';\n  const {database, session: {userId: sessionId}} = context;\n  const formatItemId: string = parseId(postId);\n  const db: Database = database;\n  const aqlQry = aql`FOR p IN posts\n      FILTER p._key == ${formatItemId} && p.userId == ${sessionId}\n      LIMIT 1\n      REMOVE p IN posts\n      RETURN OLD`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((post: PostType = {}) => {\n      if(post) {\n        // Remove tag links\n        const edgeAqlQry: AqlQuery = aql`FOR t IN isTagged\n            FILTER t._to == ${formatItemId}\n            REMOVE t IN isTagged`;\n\n        return db.query(edgeAqlQry)\n          .then(() => {\n            // Remove attached files\n            const fileAqlQry: AqlQuery = aql`FOR f IN hasFile\n                FILTER f._to == ${formatItemId}\n                REMOVE f IN hasFile`;\n\n            return db.query(fileAqlQry)\n              .then(() => post)\n              .catch((error: Error) => {\n                throw error;\n              });\n          })\n          .catch((error: Error) => {\n            throw error;\n          });\n      }\n      return {};\n    })\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const cleanPosts = (database: Database): Promise<number> => {\n  // Remove all messages that are over 60 days and not saved\n  const aqlQry: AqlQuery = aql`FOR p IN posts\n      FILTER p.added < DATE_TIMESTAMP(DATE_SUBTRACT(DATE_NOW(), 60, 'day')) && p.type == \"default\"\n      REMOVE p IN posts\n      RETURN OLD`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .then((results: PostType[] = []) => results.length)\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const createPostEdge = (db: Database, file: FileType, postId: string): Promise<FileType> => {\n  const edgeCollection: EdgeCollection = db.collection('isPosted');\n  const {fileId} = file;\n  const formatFileId: string = parseId(fileId);\n  const edgeId: string = createHash(`file-${postId}-${formatFileId}`);\n  const formatPostId: string = parseId(postId);\n  const fileType: string = parseChar(file.fileType, 16);\n\n  const edge: any = {\n    _from: `posts/${formatPostId}`,\n    _key: edgeId,\n    _to: `files/${formatFileId}`,\n    added: Date.now(),\n    type: fileType\n  };\n\n  return edgeCollection.save(edge, {returnNew: true})\n    .then(() => file)\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n"],
  "mappings": ";;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAIA,mBAAkF;AAClF,sBAA4B;AAM5B,IAAAA,gBAAiC;AACjC,mBAA0B;AAC1B,kBAAoC;AAEpC,MAAM,qBAA6B;AACnC,MAAM,gBAAwB;AAEvB,MAAM,mBAAmB,CAAC,UAAuB,CAAC,MAAM;AAC7D,QAAM;AAAA,IACJ,OAAO;AAAA,IACP,WAAW;AAAA,IACX,YAAY;AAAA,IACZ,KAAK;AAAA,IACL,OAAO;AAAA,EACT,IAAI;AAEJ,SAAO;AAAA,IACL,cAAU,uBAAS,UAAU,EAAE;AAAA,IAC/B,WAAO,wBAAS,MAAM,EAAE;AAAA,IACxB,eAAW,uBAAS,WAAW,EAAE;AAAA,IACjC,UAAM,wBAAU,MAAM,EAAE;AAAA,EAC1B;AACF;AAEO,MAAM,kBAAkB,CAAC,QAAkB,cAChD,OAAO,OAAO,CAAC,SAAc,UAAkB;AAC7C,UAAO,OAAO;AAAA,IACZ,KAAK,WAAW;AACd,cAAQ,QAAQ,KAAK;AAAA;AAAA,8EAEiD,SAAS;AAAA;AAAA;AAAA,WAG5E;AACH,cAAQ,QAAQ,KAAK,iBAAiB;AACtC,aAAO;AAAA,IACT;AAAA,IACA,KAAK,WAAW;AACd,cAAQ,QAAQ,KAAK;AAAA;AAAA,6EAEgD,SAAS;AAAA;AAAA;AAAA,WAG3E;AACH,cAAQ,QAAQ,KAAK,iBAAiB;AACtC,aAAO;AAAA,IACT;AAAA,IACA,KAAK,aAAa;AAChB,cAAQ,QAAQ,KAAK;AAAA;AAAA;AAAA;AAAA,UAInB;AACF,cAAQ,QAAQ,KAAK,qBAAqB;AAC1C,aAAO;AAAA,IACT;AAAA,IACA,KAAK,aAAa;AAChB,cAAQ,QAAQ,KAAK;AAAA;AAAA;AAAA;AAAA;AAAA,UAKnB;AACF,cAAQ,QAAQ,KAAK,qBAAqB;AAC1C,aAAO;AAAA,IACT;AAAA,IACA,KAAK,aAAa;AAChB,cAAQ,QAAQ,KAAK;AAAA;AAAA;AAAA;AAAA;AAAA,UAKnB;AACF,cAAQ,QAAQ,KAAK,qBAAqB;AAC1C,aAAO;AAAA,IACT;AAAA,IACA,SAAS;AACP,aAAO;AAAA,IACT;AAAA,EACF;AACF,GAAG,EAAC,SAAS,CAAC,GAAG,SAAS,CAAC,EAAC,CAAC;AAExB,MAAM,UAAU,CAAC,SAAqB,QAAgB,YAAqD;AAChH,QAAM,SAAiB;AACvB,QAAM,EAAC,UAAU,QAAQ,SAAS,EAAC,QAAQ,UAAS,EAAC,IAAI;AACzD,QAAM,mBAAuB,sBAAQ,MAAM;AAC3C,QAAM,EAAC,KAAI,IAAI,iBAAiB,OAAO;AACvC,QAAM,KAAK;AACX,QAAM,EAAC,SAAS,eAAe,SAAS,cAAa,IAAI,gBAAgB,QAAQ,SAAS;AAC1F,QAAM,SAAmB;AAAA,uBACJ,YAAY,iBAAiB,IAAI;AAAA;AAAA;AAItD,SAAO,GAAG,MAAM,MAAM,EACnB,KAAK,CAAC,WAAwB,OAAO,KAAK,CAAC,EAC3C,KAAK,CAAC,OAAiB,CAAC,MAAM;AAC7B,UAAM;AAAA,MACJ,KAAK;AAAA,MACL;AAAA,MACA,UAAU;AAAA,IACZ,IAAc;AAGd,QAAI;AAEJ,QAAG,WAAW,YAAY,SAAS;AACjC,sBAAgB,qBAAqB,SAAS;AAAA,YAC1C,cAAc,KAAK,IAAI,CAAC;AAAA;AAAA;AAAA;AAAA,8BAIN,SAAS;AAAA;AAAA,6BAEV,cAAc,KAAK,IAAI,CAAC;AAAA,IAC/C,WAAU,YAAY,UAAU;AAC9B,sBAAgB,qBAAqB,SAAS;AAAA,YAC1C,cAAc,KAAK,IAAI,CAAC;AAAA;AAAA,6BAEP,cAAc,KAAK,IAAI,CAAC;AAAA,IAC/C;AAEA,QAAG,eAAe;AAChB,aAAO,GAAG,MAAM,aAAa,EAC1B,KAAK,CAAC,WAAwB,OAAO,KAAK,KAAK,CAAC,CAAC,EACjD,MAAM,CAAC,cAAiB,wBAAS;AAAA,QAChC;AAAA,QACA,UAAU;AAAA,QACV,OAAO;AAAA,MACT,GAAG,OAAO,OAAO,EAAE,KAAK,OAAO,CAAC,EAAE,CAAC;AAAA,IACvC;AAEA,WAAO,CAAC;AAAA,EACV,CAAC,EACA,MAAM,CAAC,cAAiB,wBAAS;AAAA,IAChC;AAAA,IACA,UAAU;AAAA,IACV,OAAO;AAAA,EACT,GAAG,OAAO,OAAO,EAAE,KAAK,OAAO,CAAC,EAAE,CAAC;AACvC;AAwHO,MAAM,sBAAsB,CACjC,SACA,YAAsB,CAAC,GACvB,YACwB;AACxB,QAAM,SAAiB;AACvB,QAAM,EAAC,UAAU,QAAQ,SAAS,EAAC,QAAQ,UAAS,EAAC,IAAI;AACzD,QAAM,EAAC,UAAU,OAAO,WAAW,KAAI,IAAI,iBAAiB,OAAO;AACnE,QAAM,EAAC,SAAS,eAAe,SAAS,cAAa,IAAI,gBAAgB,QAAQ,SAAS;AAC1F,QAAM,kBAA0B,SAAS,SAAS;AAClD,QAAM,kBAA0B,KAAK,UAAU,UAAU,IAAI,CAAC,iBAAa,wBAAU,UAAU,EAAE,EAAE,YAAY,CAAC,CAAC;AACjH,QAAM,SAAmB,CAAC;AAC1B,QAAM,UAAoB,CAAC,cAAc,IAAI,KAAK,uBAAuB;AACzE,QAAM,qBAAyB,uBAAS,QAAQ;AAChD,QAAM,sBAA0B,uBAAS,SAAS;AAElD,MAAG,kBAAkB,iBAAiB;AACpC,kBAAc,KAAK;AAAA,QACf,cAAc;AAAA,QACd,eAAe;AAAA;AAAA;AAAA,KAGlB;AACD,kBAAc,KAAK,mBAAmB;AACtC,WAAO,KAAK,UAAU;AAAA,EACxB;AAEA,MAAG,UAAU,QAAQ;AACnB,WAAO,KAAK,kBAAkB;AAC9B,kBAAc,KAAK;AAAA;AAAA;AAAA;AAAA,MAIjB;AACF,kBAAc,KAAK,mCAAmC;AACtD,YAAQ,KAAK,sBAAsB;AAAA,EACrC;AAEA,SAAO,KAAK,cAAc;AAC1B,gBAAc,KAAK,qBAAqB;AAGxC,QAAM,SAAiB,yBAAyB,eAAe;AAAA;AAAA;AAAA,6BAGpC,eAAe,qBAAqB,eAAe;AAAA;AAAA;AAAA;AAAA,MAI1E,cAAc,KAAK,IAAI,CAAC;AAAA,aACjB,QAAQ,KAAK,MAAM,CAAC;AAAA,MAC3B,MAAM,GAAG;AAAA,gCACiB,cAAc,KAAK,IAAI,CAAC;AAEtD,SAAO,SAAS,MAAM,MAAM,EACzB,KAAK,CAAC,WAAwB,OAAO,IAAI,CAAC,EAC1C,MAAM,CAAC,cAAiB,wBAAS;AAAA,IAChC;AAAA,IACA,UAAU;AAAA,IACV,OAAO;AAAA,EACT,GAAG,OAAO,OAAO,EAAE,KAAK,MAAM,CAAC,CAAC,CAAC;AACrC;AAEO,MAAM,iBAAiB,CAC5B,SACA,OAAiB,CAAC,GAClB,YACwB;AAExB,QAAM,EAAC,UAAU,QAAQ,SAAS,EAAC,QAAQ,UAAS,EAAC,IAAI;AACzD,QAAM,EAAC,UAAU,OAAO,WAAW,KAAI,IAAI,iBAAiB,OAAO;AACnE,QAAM,EAAC,SAAS,eAAe,SAAS,cAAa,IAAI,gBAAgB,QAAQ,SAAS;AAC1F,QAAM,iBAAyB,KAAK,UAAU,KAAK,IAAI,CAAC,YAAQ,wBAAU,KAAK,EAAE,EAAE,YAAY,CAAC,CAAC;AACjG,QAAM,SAAmB,CAAC;AAC1B,QAAM,UAAoB,CAAC,cAAc,IAAI,KAAK,uBAAuB;AACzE,QAAM,qBAAyB,uBAAS,QAAQ;AAChD,QAAM,sBAA0B,uBAAS,SAAS;AAElD,MAAG,kBAAkB,iBAAiB;AACpC,kBAAc,KAAK;AAAA,QACf,cAAc;AAAA,QACd,eAAe;AAAA;AAAA;AAAA,KAGlB;AACD,kBAAc,KAAK,mBAAmB;AACtC,WAAO,KAAK,UAAU;AAAA,EACxB;AAEA,MAAG,KAAK,QAAQ;AACd,WAAO,KAAK,kBAAkB;AAC9B,kBAAc,KAAK;AAAA;AAAA;AAAA;AAAA,MAIjB;AACF,kBAAc,KAAK,yBAAyB;AAC5C,YAAQ,KAAK,iBAAiB;AAAA,EAChC;AAEA,SAAO,KAAK,cAAc;AAC1B,gBAAc,KAAK,WAAW;AAE9B,QAAM,SAAiB;AAAA;AAAA;AAAA,6BAGI,cAAc,qBAAqB,cAAc;AAAA;AAAA;AAAA;AAAA,MAIxE,cAAc,KAAK,IAAI,CAAC;AAAA,aACjB,QAAQ,KAAK,MAAM,CAAC;AAAA,MAC3B,MAAM,GAAG;AAAA,WACJ,OAAO,KAAK,IAAI,CAAC;AAAA,gCACI,cAAc,KAAK,IAAI,CAAC;AAEtD,UAAQ,IAAI,EAAC,QAAQ,QAAO,CAAC;AAC7B,SAAO,SAAS,MAAM,MAAM,EACzB,KAAK,CAAC,WAAwB,OAAO,IAAI,CAAC,EAC1C,MAAM,MAAM,CAAC,CAAC;AACnB;AAEO,MAAM,iBAAiB,CAAC,SAAqB,QAAgB,YAA+C;AAEjH,QAAM,EAAC,UAAU,QAAQ,SAAS,EAAC,QAAQ,UAAS,EAAC,IAAI;AACzD,QAAM,EAAC,OAAO,KAAI,IAAI,iBAAiB,OAAO;AAC9C,QAAM,mBAAuB,sBAAQ,MAAM;AAC3C,QAAM,EAAC,SAAS,eAAe,SAAS,cAAa,IAAI,gBAAgB,QAAQ,SAAS;AAC1F,QAAM,SAAiB;AAAA,0BACC,YAAY,mBAAmB,IAAI;AAAA,MACvD,cAAc,KAAK,IAAI,CAAC;AAAA,MACxB,MAAM,GAAG;AAAA;AAAA,gCAEiB,cAAc,KAAK,IAAI,CAAC;AAEtD,SAAO,SAAS,MAAM,MAAM,EACzB,KAAK,CAAC,WAAwB,OAAO,IAAI,CAAC,EAC1C,MAAM,CAAC,UAAiB;AACvB,UAAM;AAAA,EACR,CAAC;AACL;AAEO,MAAM,kBAAkB,CAAC,SAAqB,QAAgB,YAA+C;AAElH,QAAM,EAAC,UAAU,SAAS,EAAC,QAAQ,UAAS,EAAC,IAAI;AACjD,QAAM,EAAC,OAAO,KAAI,IAAI,iBAAiB,OAAO;AAC9C,QAAM,mBAAuB,sBAAQ,MAAM;AAG3C,QAAM,KAAK;AACX,QAAM,SAAmB;AAAA,uBACJ,IAAI,iBAAiB,YAAY;AAAA;AAAA;AAItD,SAAO,GAAG,MAAM,MAAM,EACnB,KAAK,CAAC,WAAwB,OAAO,KAAK,CAAC,EAC3C,KAAK,CAAC,OAAiB,CAAC,MAAM;AAC7B,UAAM;AAAA,MACJ;AAAA,MACA;AAAA,MACA,UAAU;AAAA,IACZ,IAAc;AAGd,QAAI;AAEJ,QAAG,WAAW,YAAY,SAAS;AACjC,sBAAgB;AAAA;AAAA,gCAEQ,IAAI;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,8BASN,SAAS;AAAA;AAAA,YAE3B,MAAM,GAAG;AAAA;AAAA,IAEf,WAAU,YAAY,UAAU;AAC9B,sBAAgB;AAAA;AAAA,gCAEQ,IAAI;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,YAOxB,MAAM,GAAG;AAAA;AAAA,IAEf;AAEA,QAAG,eAAe;AAChB,aAAO,GAAG,MAAM,aAAa,EAC1B,KAAK,CAAC,WAAwB,OAAO,IAAI,CAAC,EAC1C,MAAM,CAAC,UAAiB;AACvB,cAAM;AAAA,MACR,CAAC;AAAA,IACL;AAEA,WAAO,CAAC;AAAA,EACV,CAAC,EACA,MAAM,CAAC,UAAiB;AACvB,UAAM;AAAA,EACR,CAAC;AACL;AAEO,MAAM,UAAU,OAAO,SAAqB,SAA2C;AAE5F,QAAM,EAAC,UAAU,SAAS,EAAC,QAAQ,UAAS,EAAC,IAAI;AAEjD,QAAM;AAAA,IACJ,UAAU;AAAA,IACV;AAAA,IACA,UAAU;AAAA,IACV;AAAA,IACA;AAAA,IACA;AAAA,IACA,OAAO;AAAA,IACP,WAAW;AAAA,IACX,UAAU;AAAA,IACV,OAAO,CAAC;AAAA,IACR;AAAA,IACA,OAAO;AAAA,EACT,IAAI;AAEJ,QAAM,MAAc,KAAK,IAAI;AAE7B,QAAM,SAAmB;AAAA,IACvB,UAAM,yBAAW,QAAQ,SAAS,EAAE;AAAA,IACpC,OAAO;AAAA,IACP,aAAS,0BAAY,SAAS,kBAAkB;AAAA,IAChD,SAAS,cAAU,uBAAS,SAAS,EAAE,IAAI;AAAA,IAC3C,SAAS,cAAU,sBAAQ,OAAO,IAAI;AAAA,IACtC,UAAU,aAAa,aAAY,uBAAS,QAAQ,IAAI;AAAA,IACxD,UAAU,eAAW,0BAAY,UAAU,GAAG,IAAI;AAAA,IAClD,WAAW,cAAc,aAAY,uBAAS,SAAS,IAAI;AAAA,IAC3D,UAAU;AAAA,IACV,UAAM,0BAAY,MAAM,GAAG;AAAA,IAC3B,UAAU,eAAW,sBAAQ,QAAQ,IAAI;AAAA,IACzC,SAAS,cAAU,2BAAa,SAAS,EAAE,IAAI;AAAA,IAC/C,WAAW,gBAAY,uBAAS,WAAW,EAAE,IAAI;AAAA,IACjD,UAAM,wBAAU,MAAM,EAAE;AAAA,IACxB,QAAQ;AAAA,EACV;AAEA,QAAM,KAAe;AACrB,QAAM,SAAmB,6BAAa,MAAM;AAE5C,MAAI;AACF,UAAM,YAAsB,MAAM,GAAG,MAAM,MAAM,EAC9C,KAAK,CAAC,WAAwB,OAAO,KAAK,KAAK,CAAC,CAAC;AAEpD,UAAM,EAAC,KAAK,UAAS,IAAI;AAEzB,YAAQ,IAAI,EAAC,KAAI,CAAC;AAClB,QAAG,QAAQ,KAAK,QAAQ;AACtB,YAAM,WAAqB,KAAK,IAAI,CAAC,EAAC,MAAAC,MAAI,UAAM,wBAAUA,OAAM,EAAE,CAAC;AACnE,cAAQ,IAAI,EAAC,UAAU,UAAS,CAAC;AACjC,gBAAM,sBAAS,IAAI,UAAU,SAAS;AAAA,IACxC,OAAO;AAEL,YAAM,UAAqB,UAAM,yBAAY,IAAI,WAAW,OAAO,OAAO;AAC1E,gBAAU,OAAO;AAAA,IACnB;AAEA,WAAO;AAAA,EACT,SAAQ,OAAO;AACb,UAAM;AAAA,EACR;AACF;AAEO,MAAM,aAAa,OAAO,SAAqB,SAA2C;AAE/F,QAAM,EAAC,UAAU,SAAS,EAAC,QAAQ,UAAS,EAAC,IAAI;AACjD,QAAM,MAAc,KAAK,IAAI;AAC7B,QAAM;AAAA,IACJ;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA,OAAO,CAAC;AAAA,IACR;AAAA,EACF,IAAI;AAEJ,QAAM,SAAmB;AAAA,IACvB,SAAS,cAAU,0BAAY,SAAS,kBAAkB,IAAI;AAAA,IAC9D,SAAS,cAAU,uBAAS,SAAS,EAAE,IAAI;AAAA,IAC3C,UAAU;AAAA,IACV,MAAM,WAAO,0BAAY,MAAM,GAAG,IAAI;AAAA,IACtC,UAAU,eAAW,0BAAY,UAAU,GAAG,IAAI;AAAA,IAClD,SAAS,cAAU,2BAAa,SAAS,EAAE,IAAI;AAAA,IAC/C,WAAW,gBAAY,uBAAS,WAAW,EAAE,IAAI;AAAA,IACjD,MAAM,SAAS,aAAY,wBAAU,MAAM,EAAE,IAAI;AAAA,EACnD;AAEA,MAAI,eAAmB,sBAAQ,MAAM;AACrC,aAAW,aAAa,SAAK,yBAAW,QAAQ,SAAS,EAAE,IAAI;AAC/D,QAAM,oBAAwB,sBAAQ,OAAO;AAC7C,QAAM,SAAc;AAAA,IAClB,GAAG;AAAA,IACH,MAAM;AAAA,IACN,OAAO;AAAA,IACP,SAAS;AAAA,IACT;AAAA,IACA,QAAQ;AAAA,EACV;AACA,QAAM,KAAe;AACrB,QAAM,SAAmB,oCAAoB,QAAQ,aAAa,SAAS;AAAA,aAChE,MAAM;AAAA,aACN,MAAM;AAAA;AAGjB,MAAI;AACF,UAAM,cAAwB,MAAM,GAAG,MAAM,MAAM,EAChD,KAAK,CAAC,WAAwB,OAAO,KAAK,KAAK,CAAC,CAAC;AAEpD,UAAM,EAAC,KAAK,cAAa,IAAI;AAE7B,QAAG,QAAQ,KAAK,QAAQ;AACtB,YAAM,WAAqB,KAAK,IAAI,CAAC,EAAC,MAAAA,MAAI,UAAM,wBAAUA,OAAM,EAAE,CAAC;AACnE,YAAM,WAAqB;AAAA,0BACP,KAAK,UAAU,QAAQ,CAAC;AAAA;AAG5C,YAAM,eAAe,MAAM,GAAG,MAAM,QAAQ,EAAE,KAAK,CAAC,WAAwB,OAAO,IAAI,CAAC;AACxF,YAAM,SAAmB,aAAa,IAAI,CAAC,EAAC,KAAK,MAAK,MAAM,KAAK;AACjE,gBAAM,sBAAS,IAAI,QAAQ,aAAa;AAAA,IAC1C,OAAO;AAEL,YAAM,UAAU,UAAM,yBAAY,IAAI,eAAe,OAAO,WAAW,EAAE,KAAK,CAAC;AAC/E,kBAAY,OAAO;AAAA,IACrB;AAGA,UAAM,QAAoB,YAAY,SAAS,CAAC;AAEhD,QAAG,MAAM,QAAQ;AACf,YAAM,WAAW,UAAM,0BAAY,IAAI,UAAU,KAAK,KAAK,CAAC;AAC5D,kBAAY,QAAQ;AAAA,IACtB,OAAO;AACL,kBAAY,QAAQ,CAAC;AAAA,IACvB;AAEA,WAAO;AAAA,EACT,SAAQ,OAAO;AACb,UAAM;AAAA,EACR;AACF;AAEO,MAAM,aAAa,CAAC,SAAqB,WAAsC;AAEpF,QAAM,EAAC,UAAU,SAAS,EAAC,QAAQ,UAAS,EAAC,IAAI;AACjD,QAAM,mBAAuB,sBAAQ,MAAM;AAC3C,QAAM,KAAe;AACrB,QAAM,SAAS;AAAA,yBACQ,YAAY,mBAAmB,SAAS;AAAA;AAAA;AAAA;AAK/D,SAAO,GAAG,MAAM,MAAM,EACnB,KAAK,CAAC,WAAwB,OAAO,KAAK,CAAC,EAC3C,KAAK,CAAC,OAAiB,CAAC,MAAM;AAC7B,QAAG,MAAM;AAEP,YAAM,aAAuB;AAAA,8BACP,YAAY;AAAA;AAGlC,aAAO,GAAG,MAAM,UAAU,EACvB,KAAK,MAAM;AAEV,cAAM,aAAuB;AAAA,kCACP,YAAY;AAAA;AAGlC,eAAO,GAAG,MAAM,UAAU,EACvB,KAAK,MAAM,IAAI,EACf,MAAM,CAAC,UAAiB;AACvB,gBAAM;AAAA,QACR,CAAC;AAAA,MACL,CAAC,EACA,MAAM,CAAC,UAAiB;AACvB,cAAM;AAAA,MACR,CAAC;AAAA,IACL;AACA,WAAO,CAAC;AAAA,EACV,CAAC,EACA,MAAM,CAAC,UAAiB;AACvB,UAAM;AAAA,EACR,CAAC;AACL;AAEO,MAAM,aAAa,CAAC,aAAwC;AAEjE,QAAM,SAAmB;AAAA;AAAA;AAAA;AAKzB,SAAO,SAAS,MAAM,MAAM,EACzB,KAAK,CAAC,WAAwB,OAAO,IAAI,CAAC,EAC1C,KAAK,CAAC,UAAsB,CAAC,MAAM,QAAQ,MAAM,EACjD,MAAM,CAAC,UAAiB;AACvB,UAAM;AAAA,EACR,CAAC;AACL;AAEO,MAAM,iBAAiB,CAAC,IAAc,MAAgB,WAAsC;AACjG,QAAM,iBAAiC,GAAG,WAAW,UAAU;AAC/D,QAAM,EAAC,OAAM,IAAI;AACjB,QAAM,mBAAuB,sBAAQ,MAAM;AAC3C,QAAM,aAAiB,yBAAW,QAAQ,MAAM,IAAI,YAAY,EAAE;AAClE,QAAM,mBAAuB,sBAAQ,MAAM;AAC3C,QAAM,eAAmB,wBAAU,KAAK,UAAU,EAAE;AAEpD,QAAM,OAAY;AAAA,IAChB,OAAO,SAAS,YAAY;AAAA,IAC5B,MAAM;AAAA,IACN,KAAK,SAAS,YAAY;AAAA,IAC1B,OAAO,KAAK,IAAI;AAAA,IAChB,MAAM;AAAA,EACR;AAEA,SAAO,eAAe,KAAK,MAAM,EAAC,WAAW,KAAI,CAAC,EAC/C,KAAK,MAAM,IAAI,EACf,MAAM,CAAC,UAAiB;AACvB,UAAM;AAAA,EACR,CAAC;AACL;",
  "names": ["import_utils", "name"]
}

@@ -1,5 +1,5 @@
1
1
  import { ApiContext, AppCollectionTypes, User, UserReactionQuery } from '../types';
2
- export declare type ReactionDirection = 'INBOUND' | 'OUTBOUND';
2
+ export type ReactionDirection = 'INBOUND' | 'OUTBOUND';
3
3
  export interface Reaction {
4
4
  readonly _from?: string;
5
5
  readonly _key?: string;