@nlabs/reaktor 0.5.4 → 0.8.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 (135) hide show
  1. package/README.md +7 -7
  2. package/lib/actions/conversations.js +68 -69
  3. package/lib/actions/dynamodb.js +22 -33
  4. package/lib/actions/email.js +32 -30
  5. package/lib/actions/files.js +39 -58
  6. package/lib/actions/groups.js +33 -28
  7. package/lib/actions/images.js +71 -82
  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 +31 -45
  12. package/lib/actions/notifications.js +13 -20
  13. package/lib/actions/payments.js +40 -52
  14. package/lib/actions/posts.js +68 -48
  15. package/lib/actions/reactions.js +29 -52
  16. package/lib/actions/s3.js +21 -18
  17. package/lib/actions/search.js +26 -40
  18. package/lib/actions/sms.js +24 -21
  19. package/lib/actions/statistics.js +14 -21
  20. package/lib/actions/subscription.js +27 -24
  21. package/lib/actions/tags.js +49 -56
  22. package/lib/actions/users.js +92 -106
  23. package/lib/actions/websockets.js +28 -41
  24. package/lib/config.js +21 -18
  25. package/lib/index.js +24 -22
  26. package/lib/lambdas/actions/websockets.js +21 -45
  27. package/lib/lambdas/authorizer.js +16 -24
  28. package/lib/lambdas/connection.js +18 -43
  29. package/lib/lambdas/utils/message.js +20 -17
  30. package/lib/lambdas/utils/websocket.js +33 -25
  31. package/lib/templates/email/layout.js +16 -9
  32. package/lib/templates/email/passwordForgot.js +16 -9
  33. package/lib/templates/email/passwordRecovery.js +16 -9
  34. package/lib/templates/email/verifyEmail.js +16 -9
  35. package/lib/templates/email/welcome.js +16 -9
  36. package/lib/templates/sms/passwordForgot.js +16 -9
  37. package/lib/templates/sms/passwordRecovery.js +16 -9
  38. package/lib/templates/sms/verifyEmail.js +16 -9
  39. package/lib/templates/sms/verifyPhone.js +16 -9
  40. package/lib/templates/sms/welcome.js +16 -9
  41. package/lib/types/apps.js +14 -5
  42. package/lib/types/arangodb.js +14 -5
  43. package/lib/types/auth.js +14 -5
  44. package/lib/types/connections.js +14 -5
  45. package/lib/types/conversations.js +14 -5
  46. package/lib/types/email.js +14 -5
  47. package/lib/types/files.js +14 -5
  48. package/lib/types/google.js +14 -5
  49. package/lib/types/groups.js +14 -5
  50. package/lib/types/images.js +14 -5
  51. package/lib/types/index.js +50 -35
  52. package/lib/types/locations.js +14 -5
  53. package/lib/types/messages.js +14 -5
  54. package/lib/types/notifications.js +14 -5
  55. package/lib/types/payments.js +14 -5
  56. package/lib/types/posts.js +14 -5
  57. package/lib/types/statistics.js +14 -5
  58. package/lib/types/tags.js +14 -5
  59. package/lib/types/users.js +14 -5
  60. package/lib/types/websocket.js +14 -5
  61. package/lib/utils/analytics.js +24 -44
  62. package/lib/utils/arangodb.js +15 -22
  63. package/lib/utils/auth.js +22 -36
  64. package/lib/utils/graphql.js +20 -17
  65. package/lib/utils/index.js +26 -23
  66. package/lib/utils/objects.js +22 -19
  67. package/lib/utils/session.js +20 -17
  68. package/package.json +31 -33
  69. package/.eslintrc +0 -18
  70. package/lib/actions/conversations.d.ts +0 -14
  71. package/lib/actions/dynamodb.d.ts +0 -8
  72. package/lib/actions/email.d.ts +0 -7
  73. package/lib/actions/files.d.ts +0 -16
  74. package/lib/actions/groups.d.ts +0 -14
  75. package/lib/actions/images.d.ts +0 -22
  76. package/lib/actions/index.d.ts +0 -20
  77. package/lib/actions/ios.d.ts +0 -7
  78. package/lib/actions/locations.d.ts +0 -3
  79. package/lib/actions/messages.d.ts +0 -13
  80. package/lib/actions/notifications.d.ts +0 -5
  81. package/lib/actions/payments.d.ts +0 -10
  82. package/lib/actions/posts.d.ts +0 -19
  83. package/lib/actions/reactions.d.ts +0 -30
  84. package/lib/actions/s3.d.ts +0 -7
  85. package/lib/actions/search.d.ts +0 -3
  86. package/lib/actions/sms.d.ts +0 -3
  87. package/lib/actions/statistics.d.ts +0 -2
  88. package/lib/actions/subscription.d.ts +0 -7
  89. package/lib/actions/tags.d.ts +0 -26
  90. package/lib/actions/users.d.ts +0 -44
  91. package/lib/actions/websockets.d.ts +0 -6
  92. package/lib/config.d.ts +0 -20
  93. package/lib/index.d.ts +0 -5
  94. package/lib/lambdas/actions/websockets.d.ts +0 -6
  95. package/lib/lambdas/authorizer.d.ts +0 -20
  96. package/lib/lambdas/connection.d.ts +0 -12
  97. package/lib/lambdas/utils/message.d.ts +0 -1
  98. package/lib/lambdas/utils/websocket.d.ts +0 -7
  99. package/lib/templates/email/layout.d.ts +0 -2
  100. package/lib/templates/email/passwordForgot.d.ts +0 -2
  101. package/lib/templates/email/passwordRecovery.d.ts +0 -2
  102. package/lib/templates/email/verifyEmail.d.ts +0 -2
  103. package/lib/templates/email/welcome.d.ts +0 -2
  104. package/lib/templates/sms/passwordForgot.d.ts +0 -2
  105. package/lib/templates/sms/passwordRecovery.d.ts +0 -2
  106. package/lib/templates/sms/verifyEmail.d.ts +0 -2
  107. package/lib/templates/sms/verifyPhone.d.ts +0 -2
  108. package/lib/templates/sms/welcome.d.ts +0 -2
  109. package/lib/types/apps.d.ts +0 -44
  110. package/lib/types/arangodb.d.ts +0 -17
  111. package/lib/types/auth.d.ts +0 -7
  112. package/lib/types/connections.d.ts +0 -8
  113. package/lib/types/conversations.d.ts +0 -11
  114. package/lib/types/email.d.ts +0 -12
  115. package/lib/types/files.d.ts +0 -26
  116. package/lib/types/google.d.ts +0 -27
  117. package/lib/types/groups.d.ts +0 -22
  118. package/lib/types/images.d.ts +0 -42
  119. package/lib/types/index.d.ts +0 -18
  120. package/lib/types/locations.d.ts +0 -20
  121. package/lib/types/messages.d.ts +0 -27
  122. package/lib/types/notifications.d.ts +0 -19
  123. package/lib/types/payments.d.ts +0 -115
  124. package/lib/types/posts.d.ts +0 -47
  125. package/lib/types/statistics.d.ts +0 -3
  126. package/lib/types/tags.d.ts +0 -15
  127. package/lib/types/users.d.ts +0 -80
  128. package/lib/types/websocket.d.ts +0 -14
  129. package/lib/utils/analytics.d.ts +0 -14
  130. package/lib/utils/arangodb.d.ts +0 -9
  131. package/lib/utils/auth.d.ts +0 -5
  132. package/lib/utils/graphql.d.ts +0 -1
  133. package/lib/utils/index.d.ts +0 -6
  134. package/lib/utils/objects.d.ts +0 -3
  135. package/lib/utils/session.d.ts +0 -18
@@ -4,48 +4,50 @@ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
4
4
  var __getOwnPropNames = Object.getOwnPropertyNames;
5
5
  var __getProtoOf = Object.getPrototypeOf;
6
6
  var __hasOwnProp = Object.prototype.hasOwnProperty;
7
- var __markAsModule = (target) => __defProp(target, "__esModule", { value: true });
8
- var __require = typeof require !== "undefined" ? require : (x) => {
9
- throw new Error('Dynamic require of "' + x + '" is not supported');
10
- };
11
7
  var __export = (target, all) => {
12
- __markAsModule(target);
13
8
  for (var name in all)
14
9
  __defProp(target, name, { get: all[name], enumerable: true });
15
10
  };
16
- var __reExport = (target, module2, desc) => {
17
- if (module2 && typeof module2 === "object" || typeof module2 === "function") {
18
- for (let key of __getOwnPropNames(module2))
19
- if (!__hasOwnProp.call(target, key) && key !== "default")
20
- __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 });
21
16
  }
22
- return target;
23
- };
24
- var __toModule = (module2) => {
25
- 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);
17
+ return to;
26
18
  };
27
- __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 email_exports = {};
29
+ __export(email_exports, {
28
30
  appTemplate: () => appTemplate,
29
31
  sendEmail: () => sendEmail,
30
32
  sendTemplate: () => sendTemplate,
31
33
  sesSend: () => sesSend
32
34
  });
33
- var import_aws_sdk = __toModule(require("aws-sdk"));
34
- var import_isEmpty = __toModule(require("lodash/isEmpty"));
35
- var import_config = __toModule(require("../config"));
36
- var import_layout = __toModule(require("../templates/email/layout"));
37
- var import_passwordForgot = __toModule(require("../templates/email/passwordForgot"));
38
- var import_passwordRecovery = __toModule(require("../templates/email/passwordRecovery"));
39
- var import_verifyEmail = __toModule(require("../templates/email/verifyEmail"));
40
- var import_welcome = __toModule(require("../templates/email/welcome"));
41
- var import_passwordForgot2 = __toModule(require("../templates/sms/passwordForgot"));
42
- var import_passwordRecovery2 = __toModule(require("../templates/sms/passwordRecovery"));
43
- var import_verifyEmail2 = __toModule(require("../templates/sms/verifyEmail"));
44
- var import_welcome2 = __toModule(require("../templates/sms/welcome"));
45
- var import_s32 = __toModule(require("./s3"));
35
+ module.exports = __toCommonJS(email_exports);
36
+ var import_aws_sdk = __toESM(require("aws-sdk"));
37
+ var import_config = require("../config");
38
+ var import_layout = __toESM(require("../templates/email/layout"));
39
+ var import_passwordForgot = __toESM(require("../templates/email/passwordForgot"));
40
+ var import_passwordRecovery = __toESM(require("../templates/email/passwordRecovery"));
41
+ var import_verifyEmail = __toESM(require("../templates/email/verifyEmail"));
42
+ var import_welcome = __toESM(require("../templates/email/welcome"));
43
+ var import_passwordForgot2 = __toESM(require("../templates/sms/passwordForgot"));
44
+ var import_passwordRecovery2 = __toESM(require("../templates/sms/passwordRecovery"));
45
+ var import_verifyEmail2 = __toESM(require("../templates/sms/verifyEmail"));
46
+ var import_welcome2 = __toESM(require("../templates/sms/welcome"));
47
+ var import_s32 = require("./s3");
46
48
  const appTemplate = (app, message = "", vars = {}) => {
47
49
  let updatedMessage = "";
48
- if (!(0, import_isEmpty.default)(app)) {
50
+ if (app) {
49
51
  updatedMessage = message.replace(/\[appName\]/g, app.name).replace(/\[appUrl\]/g, app.url).replace(/\[appEmail\]/g, app.email);
50
52
  }
51
53
  Object.keys(vars).forEach((key) => {
@@ -177,4 +179,4 @@ const sendTemplate = (templateName, emailParams) => {
177
179
  sendTemplate,
178
180
  sesSend
179
181
  });
180
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/email.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 aws, {SES} from 'aws-sdk';\nimport {GetObjectRequest} from 'aws-sdk/clients/s3';\nimport {SendEmailRequest, SendEmailResponse} from 'aws-sdk/clients/ses';\nimport isEmpty from 'lodash/isEmpty';\n\nimport {Config} from '../config';\nimport layoutTemplate from '../templates/email/layout';\nimport forgotEmailTemplate from '../templates/email/passwordForgot';\nimport recoveryEmailTemplate from '../templates/email/passwordRecovery';\nimport verifyEmailTemplate from '../templates/email/verifyEmail';\nimport welcomeEmailTemplate from '../templates/email/welcome';\nimport forgotSmsTemplate from '../templates/sms/passwordForgot';\nimport recoverySmsTemplate from '../templates/sms/passwordRecovery';\nimport verifySmsTemplate from '../templates/sms/verifyEmail';\nimport welcomeSmsTemplate from '../templates/sms/welcome';\nimport {AppType} from '../types/apps';\nimport {EmailType} from '../types/email';\nimport {s3Get} from './s3';\n\n// const eventCategory: string = 'email';\n\nexport const appTemplate = (app: AppType, message: string = '', vars = {}): string => {\n  let updatedMessage: string = '';\n\n  if(!isEmpty(app)) {\n    updatedMessage = message.replace(/\\[appName\\]/g, app.name)\n      .replace(/\\[appUrl\\]/g, app.url)\n      .replace(/\\[appEmail\\]/g, app.email);\n  }\n\n  Object.keys(vars).forEach((key: string) => {\n    if(vars[key]) {\n      updatedMessage = message.replace(new RegExp(`\\\\[${key}\\\\]`, 'g'), vars[key]);\n    }\n  });\n\n  return updatedMessage;\n};\n\n// AWS::SES\nexport const sesSend = (params: SendEmailRequest): Promise<SendEmailResponse> => new Promise((resolve, reject) => {\n  aws.config.update(Config.get('aws'));\n  const ses: SES = new SES();\n\n  ses.sendEmail(params, (error: Error, response: SendEmailResponse) => {\n    if(error) {\n      return reject(error);\n    }\n\n    return resolve(response);\n  });\n});\n\nexport const sendEmail = (params: EmailType): Promise<boolean> => {\n  const {\n    app = {},\n    html,\n    subject,\n    subTitle = '',\n    text,\n    title = '',\n    user = {}\n  } = params;\n\n  const to = [user.email];\n  let promise;\n\n  const formatSubject: string = appTemplate(app, subject);\n  const formatSubTitle: string = appTemplate(app, subTitle);\n  const formatTitle: string = appTemplate(app, title);\n  const formatText: string = appTemplate(app, text);\n  const formatHtml: string = appTemplate(app, html);\n  const {_key: hasCustomEmail, urlFacebook, urlTwitter} = app;\n\n  const parseTemplate = (body: string) => {\n    const templateBody: string = appTemplate(app, body || '');\n\n    // Links\n    let links: string = '';\n    const spacer = '<td width=\"5\"><img width=\"1\" height=\"1\" border=\"0\" title=\"\" alt=\"\" ' +\n      'src=\"https://box.reaktor.io/images/email/spacer.gif\"/></td>';\n\n    if(urlFacebook) {\n      links = `<td width=\"30\"><a href=\"${urlFacebook}\"><img width=\"30\" height=\"30\" border=\"0\" ` +\n        'title=\"Facebook\" alt=\"Facebook\" src=\"https://box.reaktor.io/images/email/icon-facebook.png\"/></a></td>';\n    }\n\n    if(urlTwitter) {\n      if(urlFacebook) {\n        links += spacer;\n      }\n\n      links += `<td width=\"30\"><a href=\"${urlTwitter}\"><img width=\"30\" height=\"30\" border=\"0\" ` +\n        'title=\"Twitter\" alt=\"Twitter\" src=\"https://box.reaktor.io/images/email/icon-twitter.png\"/></a></td>';\n    }\n\n    if(!links.length) {\n      links += spacer;\n    }\n\n    return templateBody.replace(/\\[title\\]/g, formatTitle)\n      .replace(/\\[subTitle\\]/g, formatSubTitle)\n      .replace(/\\[subject\\]/g, formatSubject)\n      .replace(/\\[message\\]/g, formatHtml)\n      .replace(/\\[links\\]/g, links);\n  };\n\n  if(hasCustomEmail) {\n    const s3Params: GetObjectRequest = {Bucket: null, Key: 'apps/templates/email.html'};\n    promise = s3Get(s3Params).then((results) => parseTemplate(results.Body.toString()));\n  } else {\n    promise = Promise.resolve(parseTemplate(layoutTemplate));\n  }\n\n  return promise.then((parsedHtml: string) => {\n    const emailParams: SendEmailRequest = {\n      Destination: {ToAddresses: to},\n      Message: {\n        Body: {\n          Html: {Data: parsedHtml},\n          Text: {Data: formatText}\n        },\n        Subject: {Data: formatSubject}\n      },\n      Source: `\"${app.supportName}\" <noreply@${Config.get('app.url')}>`\n    };\n\n    return sesSend(emailParams);\n  });\n};\n\nexport const sendTemplate = (templateName: string, emailParams: EmailType): Promise<boolean> => {\n  const {app, user, subject, subTitle, title, vars}: EmailType = emailParams;\n  let html: string;\n  let text: string;\n\n  try {\n    switch(templateName) {\n      case 'passwordForgot': {\n        html = forgotEmailTemplate;\n        text = forgotSmsTemplate;\n        break;\n      }\n      case 'passwordRecovery': {\n        html = recoveryEmailTemplate;\n        text = recoverySmsTemplate;\n        break;\n      }\n      case 'verifyRecovery': {\n        html = verifyEmailTemplate;\n        text = verifySmsTemplate;\n        break;\n      }\n      case 'verifyEmail': {\n        html = verifyEmailTemplate;\n        text = verifySmsTemplate;\n        break;\n      }\n      case 'welcome': {\n        html = welcomeEmailTemplate;\n        text = welcomeSmsTemplate;\n        break;\n      }\n    }\n\n    const templateHtml: string = appTemplate(app, html, vars);\n    const templateText: string = appTemplate(app, text, vars);\n\n    const params = {\n      app,\n      html: templateHtml,\n      subTitle,\n      subject,\n      text: templateText,\n      title,\n      user\n    };\n\n    return sendEmail(params)\n      .then(() => true)\n      .catch(() => false);\n  } catch(error) {\n    return Promise.reject(error);\n  }\n};\n"],
  "mappings": "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",
  "names": []
}

182
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/email.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 aws, {SES} from 'aws-sdk';\nimport {GetObjectRequest} from 'aws-sdk/clients/s3';\nimport {SendEmailRequest, SendEmailResponse} from 'aws-sdk/clients/ses';\n\nimport {Config} from '../config';\nimport layoutTemplate from '../templates/email/layout';\nimport forgotEmailTemplate from '../templates/email/passwordForgot';\nimport recoveryEmailTemplate from '../templates/email/passwordRecovery';\nimport verifyEmailTemplate from '../templates/email/verifyEmail';\nimport welcomeEmailTemplate from '../templates/email/welcome';\nimport forgotSmsTemplate from '../templates/sms/passwordForgot';\nimport recoverySmsTemplate from '../templates/sms/passwordRecovery';\nimport verifySmsTemplate from '../templates/sms/verifyEmail';\nimport welcomeSmsTemplate from '../templates/sms/welcome';\nimport {AppType} from '../types/apps';\nimport {EmailType} from '../types/email';\nimport {s3Get} from './s3';\n\n// const eventCategory: string = 'email';\n\nexport const appTemplate = (app: AppType, message: string = '', vars = {}): string => {\n  let updatedMessage: string = '';\n\n  if(app) {\n    updatedMessage = message.replace(/\\[appName\\]/g, app.name)\n      .replace(/\\[appUrl\\]/g, app.url)\n      .replace(/\\[appEmail\\]/g, app.email);\n  }\n\n  Object.keys(vars).forEach((key: string) => {\n    if(vars[key]) {\n      updatedMessage = message.replace(new RegExp(`\\\\[${key}\\\\]`, 'g'), vars[key]);\n    }\n  });\n\n  return updatedMessage;\n};\n\n// AWS::SES\nexport const sesSend = (params: SendEmailRequest): Promise<SendEmailResponse> => new Promise((resolve, reject) => {\n  aws.config.update(Config.get('aws'));\n  const ses: SES = new SES();\n\n  ses.sendEmail(params, (error: Error, response: SendEmailResponse) => {\n    if(error) {\n      return reject(error);\n    }\n\n    return resolve(response);\n  });\n});\n\nexport const sendEmail = (params: EmailType): Promise<boolean> => {\n  const {\n    app = {},\n    html,\n    subject,\n    subTitle = '',\n    text,\n    title = '',\n    user = {}\n  } = params;\n\n  const to = [user.email];\n  let promise;\n\n  const formatSubject: string = appTemplate(app, subject);\n  const formatSubTitle: string = appTemplate(app, subTitle);\n  const formatTitle: string = appTemplate(app, title);\n  const formatText: string = appTemplate(app, text);\n  const formatHtml: string = appTemplate(app, html);\n  const {_key: hasCustomEmail, urlFacebook, urlTwitter} = app;\n\n  const parseTemplate = (body: string) => {\n    const templateBody: string = appTemplate(app, body || '');\n\n    // Links\n    let links: string = '';\n    const spacer = '<td width=\"5\"><img width=\"1\" height=\"1\" border=\"0\" title=\"\" alt=\"\" ' +\n      'src=\"https://box.reaktor.io/images/email/spacer.gif\"/></td>';\n\n    if(urlFacebook) {\n      links = `<td width=\"30\"><a href=\"${urlFacebook}\"><img width=\"30\" height=\"30\" border=\"0\" ` +\n        'title=\"Facebook\" alt=\"Facebook\" src=\"https://box.reaktor.io/images/email/icon-facebook.png\"/></a></td>';\n    }\n\n    if(urlTwitter) {\n      if(urlFacebook) {\n        links += spacer;\n      }\n\n      links += `<td width=\"30\"><a href=\"${urlTwitter}\"><img width=\"30\" height=\"30\" border=\"0\" ` +\n        'title=\"Twitter\" alt=\"Twitter\" src=\"https://box.reaktor.io/images/email/icon-twitter.png\"/></a></td>';\n    }\n\n    if(!links.length) {\n      links += spacer;\n    }\n\n    return templateBody.replace(/\\[title\\]/g, formatTitle)\n      .replace(/\\[subTitle\\]/g, formatSubTitle)\n      .replace(/\\[subject\\]/g, formatSubject)\n      .replace(/\\[message\\]/g, formatHtml)\n      .replace(/\\[links\\]/g, links);\n  };\n\n  if(hasCustomEmail) {\n    const s3Params: GetObjectRequest = {Bucket: null, Key: 'apps/templates/email.html'};\n    promise = s3Get(s3Params).then((results) => parseTemplate(results.Body.toString()));\n  } else {\n    promise = Promise.resolve(parseTemplate(layoutTemplate));\n  }\n\n  return promise.then((parsedHtml: string) => {\n    const emailParams: SendEmailRequest = {\n      Destination: {ToAddresses: to},\n      Message: {\n        Body: {\n          Html: {Data: parsedHtml},\n          Text: {Data: formatText}\n        },\n        Subject: {Data: formatSubject}\n      },\n      Source: `\"${app.supportName}\" <noreply@${Config.get('app.url')}>`\n    };\n\n    return sesSend(emailParams);\n  });\n};\n\nexport const sendTemplate = (templateName: string, emailParams: EmailType): Promise<boolean> => {\n  const {app, user, subject, subTitle, title, vars}: EmailType = emailParams;\n  let html: string;\n  let text: string;\n\n  try {\n    switch(templateName) {\n      case 'passwordForgot': {\n        html = forgotEmailTemplate;\n        text = forgotSmsTemplate;\n        break;\n      }\n      case 'passwordRecovery': {\n        html = recoveryEmailTemplate;\n        text = recoverySmsTemplate;\n        break;\n      }\n      case 'verifyRecovery': {\n        html = verifyEmailTemplate;\n        text = verifySmsTemplate;\n        break;\n      }\n      case 'verifyEmail': {\n        html = verifyEmailTemplate;\n        text = verifySmsTemplate;\n        break;\n      }\n      case 'welcome': {\n        html = welcomeEmailTemplate;\n        text = welcomeSmsTemplate;\n        break;\n      }\n    }\n\n    const templateHtml: string = appTemplate(app, html, vars);\n    const templateText: string = appTemplate(app, text, vars);\n\n    const params = {\n      app,\n      html: templateHtml,\n      subTitle,\n      subject,\n      text: templateText,\n      title,\n      user\n    };\n\n    return sendEmail(params)\n      .then(() => true)\n      .catch(() => false);\n  } catch(error) {\n    return Promise.reject(error);\n  }\n};\n"],
  "mappings": "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",
  "names": ["import_passwordForgot", "import_passwordRecovery", "import_verifyEmail", "import_welcome", "import_s3", "aws", "layoutTemplate", "forgotEmailTemplate", "forgotSmsTemplate", "recoveryEmailTemplate", "recoverySmsTemplate", "verifyEmailTemplate", "verifySmsTemplate", "welcomeEmailTemplate", "welcomeSmsTemplate"]
}

@@ -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 files_exports = {};
19
+ __export(files_exports, {
45
20
  addFile: () => addFile,
46
21
  createFile: () => createFile,
47
22
  decodeBase64: () => decodeBase64,
@@ -55,15 +30,15 @@ __export(exports, {
55
30
  linkFiles: () => linkFiles,
56
31
  updateFiles: () => updateFiles
57
32
  });
58
- var import_rip_hunter = __toModule(require("@nlabs/rip-hunter"));
59
- var import_utils = __toModule(require("@nlabs/utils"));
60
- var import_arangojs = __toModule(require("arangojs"));
61
- var import_googleapis = __toModule(require("googleapis"));
62
- var import_isEmpty = __toModule(require("lodash/isEmpty"));
63
- var import_config = __toModule(require("../config"));
64
- var import_analytics = __toModule(require("../utils/analytics"));
65
- var import_images = __toModule(require("./images"));
66
- var import_posts = __toModule(require("./posts"));
33
+ module.exports = __toCommonJS(files_exports);
34
+ var import_rip_hunter = require("@nlabs/rip-hunter");
35
+ var import_utils = require("@nlabs/utils");
36
+ var import_arangojs = require("arangojs");
37
+ var import_googleapis = require("googleapis");
38
+ var import_config = require("../config");
39
+ var import_analytics = require("../utils/analytics");
40
+ var import_images = require("./images");
41
+ var import_posts = require("./posts");
67
42
  const youtube = import_googleapis.google.youtube({ auth: import_config.Config.get("google.key"), version: "v3" });
68
43
  const addFile = (context, item = {}) => {
69
44
  const { database, session: { userId: sessionId, userAccess } } = context;
@@ -75,7 +50,7 @@ const addFile = (context, item = {}) => {
75
50
  url
76
51
  } = item;
77
52
  const formatFileId = fileId ? (0, import_utils.parseId)(fileId) : (0, import_utils.createHash)(`file-${sessionId}`);
78
- const isUrl = (0, import_isEmpty.default)(url);
53
+ const isUrl = !!url;
79
54
  let formatName = (0, import_utils.parseString)(name, 160);
80
55
  let formatType = (0, import_utils.parseString)(fileType, 16);
81
56
  if (formatName === "" && isUrl) {
@@ -123,7 +98,8 @@ const addFile = (context, item = {}) => {
123
98
  const now = Date.now();
124
99
  if (isImage) {
125
100
  return (0, import_images.resizeSaveImage)(context, formatFileId, buf, uploadType).then((resizedImage) => {
126
- const insert2 = __spreadProps(__spreadValues({}, resizedImage), {
101
+ const insert2 = {
102
+ ...resizedImage,
127
103
  _key: formatFileId,
128
104
  added: now,
129
105
  description: formatDesc,
@@ -131,7 +107,7 @@ const addFile = (context, item = {}) => {
131
107
  modified: now,
132
108
  name: formatName,
133
109
  userId: sessionId
134
- });
110
+ };
135
111
  return saveToDb(insert2);
136
112
  }).catch((error) => {
137
113
  throw error;
@@ -160,7 +136,7 @@ const addFile = (context, item = {}) => {
160
136
  };
161
137
  const getGiphyTrends = (context, limit = 30) => {
162
138
  const gifUrl = `http://api.giphy.com/v1/gifs/trending?api_key=${import_config.Config.get("giphy.key")}&limit=${limit}`;
163
- return (0, import_rip_hunter.get)(gifUrl).then((res) => res.json()).then((json) => json.data.map((gifImage = { id: null, images: null }) => {
139
+ return (0, import_rip_hunter.get)(gifUrl).then((res) => res.json()).then((json) => json?.data?.map((gifImage = { id: null, images: null }) => {
164
140
  const {
165
141
  id,
166
142
  images: {
@@ -179,7 +155,7 @@ const getGiphyTrends = (context, limit = 30) => {
179
155
  const getGiphySearch = (context, query, limit = 30) => {
180
156
  const formatQuery = encodeURI(query);
181
157
  const gifUrl = `http://api.giphy.com/v1/gifs/search?q=${formatQuery}&api_key=${import_config.Config.get("giphy.key")}&limit=${limit}`;
182
- return fetch(gifUrl).then((res) => res.json()).then((json) => json.data.map((gifImage = { id: null, images: null }) => {
158
+ return fetch(gifUrl).then((res) => res.json()).then((json) => json?.data?.map((gifImage = { id: null, images: null }) => {
183
159
  const {
184
160
  id,
185
161
  images: {
@@ -220,6 +196,7 @@ const getYouTubeSearch = (context, query, limit = 30) => new Promise((resolve, r
220
196
  youtube.search.list({
221
197
  maxResults: limit,
222
198
  part: ["snippet"],
199
+ // eslint-disable-next-line
223
200
  q: query,
224
201
  regionCode: "US"
225
202
  }, (error, data) => {
@@ -262,18 +239,22 @@ const createFile = (db, file) => {
262
239
  throw error;
263
240
  });
264
241
  };
265
- const linkFiles = (db, files, postId) => Promise.all(files.map((file) => createFile(db, file).then((file2) => (0, import_posts.createPostEdge)(db, file2, postId))));
242
+ const linkFiles = (db, files, postId) => Promise.all(
243
+ files.map((file) => createFile(db, file).then((file2) => (0, import_posts.createPostEdge)(db, file2, postId)))
244
+ );
266
245
  const updateFiles = (db, postId, files) => {
267
246
  const edgeCollection = db.collection("isPosted");
268
- return edgeCollection.inEdges(postId).then((edges) => {
247
+ return edgeCollection.inEdges(postId, {}).then((edges) => {
269
248
  if (edges.length) {
270
- return Promise.all(edges.map((edge) => {
271
- const { _key: edgeKey } = edge;
272
- const aqlQry = import_arangojs.aql`REMOVE {_key:${edgeKey}} IN isPosted`;
273
- return db.query(aqlQry).catch((error) => {
274
- throw error;
275
- });
276
- })).then(() => {
249
+ return Promise.all(
250
+ edges.map((edge) => {
251
+ const { _key: edgeKey } = edge;
252
+ const aqlQry = import_arangojs.aql`REMOVE {_key:${edgeKey}} IN isPosted`;
253
+ return db.query(aqlQry).catch((error) => {
254
+ throw error;
255
+ });
256
+ })
257
+ ).then(() => {
277
258
  if (files.length) {
278
259
  return linkFiles(db, files, postId).then(() => files);
279
260
  }
@@ -317,4 +298,4 @@ const decodeBase64 = (dataString) => {
317
298
  linkFiles,
318
299
  updateFiles
319
300
  });
320
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/files.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 {get as httpGet} from '@nlabs/rip-hunter';\nimport {createHash, parseId, parseString} from '@nlabs/utils';\nimport {aql, Database} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {EdgeCollection} from 'arangojs/collection';\nimport {ArrayCursor} from 'arangojs/cursor';\nimport {google} from 'googleapis';\nimport isEmpty from 'lodash/isEmpty';\n\nimport {Config} from '../config';\nimport {ApiContext} from '../types/auth';\nimport {FileType} from '../types/files';\nimport {UserError} from '../utils/analytics';\nimport {resizeSaveImage} from './images';\nimport {createPostEdge} from './posts';\n\nconst youtube = google.youtube({auth: Config.get('google.key'), version: 'v3'});\n\n// const eventCategory: string = 'files';\n\n// Upload file\nexport const addFile = (context: ApiContext, item: FileType = {}): Promise<FileType> => {\n  const {database, session: {userId: sessionId, userAccess}} = context;\n  const {\n    description,\n    fileId,\n    fileType,\n    name,\n    url\n  } = item;\n\n  // Id\n  const formatFileId: string = fileId ? parseId(fileId) : createHash(`file-${sessionId}`);\n\n  // Name\n  const isUrl: boolean = isEmpty(url);\n\n  // If no name, get it from url path\n  let formatName: string = parseString(name, 160);\n  let formatType: string = parseString(fileType, 16);\n\n  if(formatName === '' && isUrl) {\n    formatName = url.substring(url.lastIndexOf('/') + 1);\n  }\n\n  if(formatType === '') {\n    const nameArr: string[] = formatName.split('.');\n    const ext: string = nameArr[nameArr.length - 1];\n\n    switch(ext) {\n      case 'jpeg':\n      case 'jpg':\n        formatType = 'image/jpeg';\n        break;\n      case 'png':\n        formatType = 'image/png';\n        break;\n      case 'zip':\n        formatType = 'application/zip';\n        break;\n      default:\n        break;\n    }\n  }\n\n  let isImage: boolean;\n\n  switch(formatType) {\n    case 'image/jpeg':\n    case 'image/png':\n      isImage = true;\n      break;\n    default:\n      isImage = false;\n      break;\n  }\n\n  // Description\n  const formatDesc: string = parseString(description, 500);\n\n  // Only allow file uploads to premium users\n  if(!isImage && userAccess !== 2) {\n    throw new UserError('account_restriction');\n  }\n\n  const saveToDb = (insert: FileType) => {\n    const aqlQry: AqlQuery = aql`INSERT ${insert} IN files RETURN NEW`;\n\n    return database.query(aqlQry)\n      .then((cursor: ArrayCursor) => cursor.next())\n      .then((file = {}) => file)\n      .catch((error: Error) => {\n        throw error;\n      });\n  };\n\n  const uploadFile = (buf: Buffer, uploadType: string) => {\n    const now: number = Date.now();\n\n    // If image, resize and create a thumbnail\n    if(isImage) {\n      return resizeSaveImage(context, formatFileId, buf, uploadType)\n        .then((resizedImage: FileType) => {\n          const insert: FileType = {\n            ...resizedImage,\n            _key: formatFileId,\n            added: now,\n            description: formatDesc,\n            fileType: formatType,\n            modified: now,\n            name: formatName,\n            userId: sessionId\n          };\n\n          return saveToDb(insert);\n        })\n        .catch((error: Error) => {\n          throw error;\n        });\n    }\n    const insert: FileType = {\n      _key: formatFileId,\n      added: now,\n      description: formatDesc,\n      fileType: formatType,\n      modified: now,\n      name: formatName,\n      userId: sessionId\n    };\n\n    return saveToDb(insert);\n  };\n\n  // If file is a url path, download the file and save\n  if(isUrl) {\n    return httpGet(url)\n      .then((body) => uploadFile(Buffer.from(body, 'binary'), formatType))\n      .catch(() => {\n        throw new UserError('file_request');\n      });\n  } else if(item.base64 !== '') {\n    const buffer: Buffer = Buffer.from(item.base64);\n    return uploadFile(buffer, formatType);\n  }\n  throw new Error('file_required');\n};\n\n// Giphy\nexport const getGiphyTrends = (context: ApiContext, limit: number = 30): Promise<any[]> => {\n  const gifUrl: string = `http://api.giphy.com/v1/gifs/trending?api_key=${Config.get('giphy.key')}&limit=${limit}`;\n\n  return httpGet(gifUrl)\n    .then((res: Response) => res.json())\n    .then((json) => json.data.map((gifImage = {id: null, images: null}) => {\n      const {\n        id,\n        images: {\n          original: {url = ''} = {},\n          fixed_height_small: {url: thumb = ''} = {}\n        } = {}\n      } = gifImage;\n\n      return {\n        id,\n        thumb,\n        type: 'giphy',\n        url\n      };\n    }));\n};\n\nexport const getGiphySearch = (context: ApiContext, query: string, limit: number = 30): Promise<any[]> => {\n  const formatQuery: string = encodeURI(query);\n  const gifUrl: string = `http://api.giphy.com/v1/gifs/search?q=${formatQuery}&api_key=${Config.get('giphy.key')}&limit=${limit}`;\n\n  return fetch(gifUrl)\n    .then((res: Response) => res.json())\n    .then((json) => json.data.map((gifImage = {id: null, images: null}) => {\n      const {\n        id,\n        images: {\n          original: {url = ''} = {},\n          fixed_height_small: {url: thumb = ''} = {}\n        } = {}\n      } = gifImage;\n\n      return {\n        id,\n        thumb,\n        type: 'giphy',\n        url\n      };\n    }));\n};\n\nexport const getYouTubeTrends = (context: ApiContext, limit: number = 30): Promise<any[]> =>\n  new Promise((resolve, reject) => {\n    youtube.videos.list({\n      chart: 'mostPopular',\n      maxResults: limit,\n      part: ['snippet'],\n      regionCode: 'US'\n    }, (error: Error, data: any) => {\n      if(error) {\n        console.error(error);\n        reject(new Error(error[0].message));\n      } else if(data) {\n        const list = data.items.map((item) => ({\n          id: item.id,\n          thumb: item.snippet.thumbnails.high.url,\n          type: 'youtube',\n          url: `http://www.youtube.com/embed/${item.id}`\n        }));\n\n        resolve(list);\n      }\n    });\n  });\n\n\nexport const getYouTubeSearch = (context: ApiContext, query: string, limit: number = 30): Promise<any[]> =>\n  new Promise((resolve, reject) => {\n    youtube.search.list({\n      maxResults: limit,\n      part: ['snippet'],\n      // eslint-disable-next-line\n      q: query,\n      regionCode: 'US'\n    }, (error: Error, data: any) => {\n      if(error) {\n        console.error(error);\n        reject(new Error(error[0].message));\n      } else if(data) {\n        const {items} = data;\n        const list = items.map((item) => ({\n          id: item.id,\n          thumb: item.snippet.thumbnails.high.url,\n          type: 'youtube',\n          url: `http://www.youtube.com/embed/${item.id}`\n        }));\n\n        resolve(list);\n      }\n    });\n  });\n\n// Files\nexport const getPathUserFiles = (userId: string, filename: string): string => `users/${userId}/files/${filename}`;\n\nexport const getUrlUserFiles = (userId: string, filename: string, dir: string = 'files', type: string = 'profile'): string => {\n  if(filename) {\n    return `https://box.${Config.get('app.url')}/users/${userId}/${dir}/${filename}`;\n  }\n\n  if(type === 'profile') {\n    return `https://box.${Config.get('app.url')}/defaults/user_bk.jpg`;\n  }\n\n  return `https://box.${Config.get('app.url')}/defaults/user_wh.jpg`;\n};\n\nexport const createFile = (db: Database, file: FileType): Promise<FileType> => {\n  const {fileId} = file;\n  const insert: any = {\n    _key: fileId,\n    added: Date.now()\n  };\n\n  const aqlQry: AqlQuery = aql`UPSERT {_key: ${fileId}}\n    INSERT ${insert}\n    UPDATE {}\n    IN files RETURN NEW`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((updatedFile: FileType = {}) => updatedFile)\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const linkFiles = (db: Database, files: FileType[], postId: string): Promise<any> =>\n  Promise.all(\n    files.map((file: FileType) => createFile(db, file)\n      .then((file: FileType) => createPostEdge(db, file, postId)))\n  );\n\nexport const updateFiles = (db: Database, postId: string, files: FileType[]): Promise<any> => {\n  const edgeCollection: EdgeCollection = db.collection('isPosted');\n\n  return edgeCollection.inEdges(postId)\n    .then((edges: any) => {\n      if(edges.length) {\n        // Remove linked edges\n        return Promise.all(\n          edges.map((edge) => {\n            const {_key: edgeKey} = edge;\n            const aqlQry: AqlQuery = aql`REMOVE {_key:${edgeKey}} IN isPosted`;\n\n            return db.query(aqlQry).catch((error: Error) => {\n              throw error;\n            });\n          }))\n          .then(() => {\n            if(files.length) {\n              // Link files\n              return linkFiles(db, files, postId).then(() => files);\n            }\n            return files;\n          });\n      } else if(files.length) {\n        // Link files\n        return linkFiles(db, files, postId).then(() => files);\n      }\n      return files;\n    })\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const encodeBase64 = (buffer: Buffer): string => Buffer.from(buffer).toString('base64');\n\nexport const decodeBase64 = (dataString: string): object => {\n  const getData = (str: string) => str.match(/^data:([A-Za-z-+\\/]+);base64,(.+)$/) || [];\n  // const getData = (base64: string) => base64.substr(base64.indexOf(',') + 1);\n  let matches = getData(dataString);\n\n  if(matches.length !== 3) {\n    // If invalid make sure we don't need to decode\n    matches = getData(decodeURIComponent(dataString));\n\n    // Check it again.\n    if(matches.length !== 3) {\n      throw Error('Invalid input string');\n    }\n  }\n\n  return {\n    data: Buffer.from(matches[2], 'base64'),\n    type: matches[1]\n  };\n};\n"],
  "mappings": "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",
  "names": []
}

301
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/files.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 {get as httpGet} from '@nlabs/rip-hunter';\nimport {createHash, parseId, parseString} from '@nlabs/utils';\nimport {aql, Database} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {EdgeCollection} from 'arangojs/collection';\nimport {ArrayCursor} from 'arangojs/cursor';\nimport {google} from 'googleapis';\n\nimport {Config} from '../config';\nimport {ApiContext} from '../types/auth';\nimport {FileType} from '../types/files';\nimport {UserError} from '../utils/analytics';\nimport {resizeSaveImage} from './images';\nimport {createPostEdge} from './posts';\n\nconst youtube = google.youtube({auth: Config.get('google.key'), version: 'v3'});\n\n// const eventCategory: string = 'files';\n\n// Upload file\nexport const addFile = (context: ApiContext, item: FileType = {}): Promise<FileType> => {\n  const {database, session: {userId: sessionId, userAccess}} = context;\n  const {\n    description,\n    fileId,\n    fileType,\n    name,\n    url\n  } = item;\n\n  // Id\n  const formatFileId: string = fileId ? parseId(fileId) : createHash(`file-${sessionId}`);\n\n  // Name\n  const isUrl: boolean = !!url;\n\n  // If no name, get it from url path\n  let formatName: string = parseString(name, 160);\n  let formatType: string = parseString(fileType, 16);\n\n  if(formatName === '' && isUrl) {\n    formatName = url.substring(url.lastIndexOf('/') + 1);\n  }\n\n  if(formatType === '') {\n    const nameArr: string[] = formatName.split('.');\n    const ext: string = nameArr[nameArr.length - 1];\n\n    switch(ext) {\n      case 'jpeg':\n      case 'jpg':\n        formatType = 'image/jpeg';\n        break;\n      case 'png':\n        formatType = 'image/png';\n        break;\n      case 'zip':\n        formatType = 'application/zip';\n        break;\n      default:\n        break;\n    }\n  }\n\n  let isImage: boolean;\n\n  switch(formatType) {\n    case 'image/jpeg':\n    case 'image/png':\n      isImage = true;\n      break;\n    default:\n      isImage = false;\n      break;\n  }\n\n  // Description\n  const formatDesc: string = parseString(description, 500);\n\n  // Only allow file uploads to premium users\n  if(!isImage && userAccess !== 2) {\n    throw new UserError('account_restriction');\n  }\n\n  const saveToDb = (insert: FileType) => {\n    const aqlQry: AqlQuery = aql`INSERT ${insert} IN files RETURN NEW`;\n\n    return database.query(aqlQry)\n      .then((cursor: ArrayCursor) => cursor.next())\n      .then((file = {}) => file)\n      .catch((error: Error) => {\n        throw error;\n      });\n  };\n\n  const uploadFile = (buf: Buffer, uploadType: string) => {\n    const now: number = Date.now();\n\n    // If image, resize and create a thumbnail\n    if(isImage) {\n      return resizeSaveImage(context, formatFileId, buf, uploadType)\n        .then((resizedImage: FileType) => {\n          const insert: FileType = {\n            ...resizedImage,\n            _key: formatFileId,\n            added: now,\n            description: formatDesc,\n            fileType: formatType,\n            modified: now,\n            name: formatName,\n            userId: sessionId\n          };\n\n          return saveToDb(insert);\n        })\n        .catch((error: Error) => {\n          throw error;\n        });\n    }\n    const insert: FileType = {\n      _key: formatFileId,\n      added: now,\n      description: formatDesc,\n      fileType: formatType,\n      modified: now,\n      name: formatName,\n      userId: sessionId\n    };\n\n    return saveToDb(insert);\n  };\n\n  // If file is a url path, download the file and save\n  if(isUrl) {\n    return httpGet(url)\n      .then((body) => uploadFile(Buffer.from(body, 'binary'), formatType))\n      .catch(() => {\n        throw new UserError('file_request');\n      });\n  } else if(item.base64 !== '') {\n    const buffer: Buffer = Buffer.from(item.base64);\n    return uploadFile(buffer, formatType);\n  }\n  throw new Error('file_required');\n};\n\n// Giphy\nexport const getGiphyTrends = (context: ApiContext, limit: number = 30): Promise<any[]> => {\n  const gifUrl = `http://api.giphy.com/v1/gifs/trending?api_key=${Config.get('giphy.key')}&limit=${limit}`;\n\n  return httpGet(gifUrl)\n    .then((res: Response) => res.json())\n    .then((json: any) => json?.data?.map((gifImage = {id: null, images: null}) => {\n      const {\n        id,\n        images: {\n          original: {url = ''} = {},\n          fixed_height_small: {url: thumb = ''} = {}\n        } = {}\n      } = gifImage;\n\n      return {\n        id,\n        thumb,\n        type: 'giphy',\n        url\n      };\n    }));\n};\n\nexport const getGiphySearch = (context: ApiContext, query: string, limit: number = 30): Promise<any[]> => {\n  const formatQuery = encodeURI(query);\n  const gifUrl = `http://api.giphy.com/v1/gifs/search?q=${formatQuery}&api_key=${Config.get('giphy.key')}&limit=${limit}`;\n\n  return fetch(gifUrl)\n    .then((res: Response) => res.json())\n    .then((json: any) => json?.data?.map((gifImage = {id: null, images: null}) => {\n      const {\n        id,\n        images: {\n          original: {url = ''} = {},\n          fixed_height_small: {url: thumb = ''} = {}\n        } = {}\n      } = gifImage;\n\n      return {\n        id,\n        thumb,\n        type: 'giphy',\n        url\n      };\n    }));\n};\n\nexport const getYouTubeTrends = (context: ApiContext, limit: number = 30): Promise<any[]> =>\n  new Promise((resolve, reject) => {\n    youtube.videos.list({\n      chart: 'mostPopular',\n      maxResults: limit,\n      part: ['snippet'],\n      regionCode: 'US'\n    }, (error: Error, data: any) => {\n      if(error) {\n        console.error(error);\n        reject(new Error(error[0].message));\n      } else if(data) {\n        const list = data.items.map((item) => ({\n          id: item.id,\n          thumb: item.snippet.thumbnails.high.url,\n          type: 'youtube',\n          url: `http://www.youtube.com/embed/${item.id}`\n        }));\n\n        resolve(list);\n      }\n    });\n  });\n\n\nexport const getYouTubeSearch = (context: ApiContext, query: string, limit: number = 30): Promise<any[]> =>\n  new Promise((resolve, reject) => {\n    youtube.search.list({\n      maxResults: limit,\n      part: ['snippet'],\n      // eslint-disable-next-line\n      q: query,\n      regionCode: 'US'\n    }, (error: Error, data: any) => {\n      if(error) {\n        console.error(error);\n        reject(new Error(error[0].message));\n      } else if(data) {\n        const {items} = data;\n        const list = items.map((item) => ({\n          id: item.id,\n          thumb: item.snippet.thumbnails.high.url,\n          type: 'youtube',\n          url: `http://www.youtube.com/embed/${item.id}`\n        }));\n\n        resolve(list);\n      }\n    });\n  });\n\n// Files\nexport const getPathUserFiles = (userId: string, filename: string): string => `users/${userId}/files/${filename}`;\n\nexport const getUrlUserFiles = (userId: string, filename: string, dir: string = 'files', type: string = 'profile'): string => {\n  if(filename) {\n    return `https://box.${Config.get('app.url')}/users/${userId}/${dir}/${filename}`;\n  }\n\n  if(type === 'profile') {\n    return `https://box.${Config.get('app.url')}/defaults/user_bk.jpg`;\n  }\n\n  return `https://box.${Config.get('app.url')}/defaults/user_wh.jpg`;\n};\n\nexport const createFile = (db: Database, file: FileType): Promise<FileType> => {\n  const {fileId} = file;\n  const insert: any = {\n    _key: fileId,\n    added: Date.now()\n  };\n\n  const aqlQry: AqlQuery = aql`UPSERT {_key: ${fileId}}\n    INSERT ${insert}\n    UPDATE {}\n    IN files RETURN NEW`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((updatedFile: FileType = {}) => updatedFile)\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const linkFiles = (db: Database, files: FileType[], postId: string): Promise<any> =>\n  Promise.all(\n    files.map((file: FileType) => createFile(db, file)\n      .then((file: FileType) => createPostEdge(db, file, postId)))\n  );\n\nexport const updateFiles = (db: Database, postId: string, files: FileType[]): Promise<any> => {\n  const edgeCollection: EdgeCollection = db.collection('isPosted');\n\n  return edgeCollection.inEdges(postId, {})\n    .then((edges: any) => {\n      if(edges.length) {\n        // Remove linked edges\n        return Promise.all(\n          edges.map((edge) => {\n            const {_key: edgeKey} = edge;\n            const aqlQry: AqlQuery = aql`REMOVE {_key:${edgeKey}} IN isPosted`;\n\n            return db.query(aqlQry).catch((error: Error) => {\n              throw error;\n            });\n          }))\n          .then(() => {\n            if(files.length) {\n              // Link files\n              return linkFiles(db, files, postId).then(() => files);\n            }\n            return files;\n          });\n      } else if(files.length) {\n        // Link files\n        return linkFiles(db, files, postId).then(() => files);\n      }\n      return files;\n    })\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const encodeBase64 = (buffer: Buffer): string => Buffer.from(buffer).toString('base64');\n\nexport const decodeBase64 = (dataString: string): object => {\n  const getData = (str: string) => str.match(/^data:([A-Za-z-+\\/]+);base64,(.+)$/) || [];\n  // const getData = (base64: string) => base64.substr(base64.indexOf(',') + 1);\n  let matches = getData(dataString);\n\n  if(matches.length !== 3) {\n    // If invalid make sure we don't need to decode\n    matches = getData(decodeURIComponent(dataString));\n\n    // Check it again.\n    if(matches.length !== 3) {\n      throw Error('Invalid input string');\n    }\n  }\n\n  return {\n    data: Buffer.from(matches[2], 'base64'),\n    type: matches[1]\n  };\n};\n"],
  "mappings": "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",
  "names": ["insert", "httpGet", "file"]
}

@@ -4,27 +4,29 @@ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
4
4
  var __getOwnPropNames = Object.getOwnPropertyNames;
5
5
  var __getProtoOf = Object.getPrototypeOf;
6
6
  var __hasOwnProp = Object.prototype.hasOwnProperty;
7
- var __markAsModule = (target) => __defProp(target, "__esModule", { value: true });
8
- var __require = typeof require !== "undefined" ? require : (x) => {
9
- throw new Error('Dynamic require of "' + x + '" is not supported');
10
- };
11
7
  var __export = (target, all) => {
12
- __markAsModule(target);
13
8
  for (var name in all)
14
9
  __defProp(target, name, { get: all[name], enumerable: true });
15
10
  };
16
- var __reExport = (target, module2, desc) => {
17
- if (module2 && typeof module2 === "object" || typeof module2 === "function") {
18
- for (let key of __getOwnPropNames(module2))
19
- if (!__hasOwnProp.call(target, key) && key !== "default")
20
- __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 });
21
16
  }
22
- return target;
23
- };
24
- var __toModule = (module2) => {
25
- 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);
17
+ return to;
26
18
  };
27
- __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 groups_exports = {};
29
+ __export(groups_exports, {
28
30
  addGroup: () => addGroup,
29
31
  createGroupEdge: () => createGroupEdge,
30
32
  deleteGroup: () => deleteGroup,
@@ -37,12 +39,13 @@ __export(exports, {
37
39
  isGrouped: () => isGrouped,
38
40
  updateGroup: () => updateGroup
39
41
  });
40
- var import_utils = __toModule(require("@nlabs/utils"));
41
- var import_arangojs = __toModule(require("arangojs"));
42
- var import_flatten = __toModule(require("lodash/flatten"));
43
- var import_uniqBy = __toModule(require("lodash/uniqBy"));
44
- var import_utils2 = __toModule(require("../utils"));
45
- var import_tags = __toModule(require("./tags"));
42
+ module.exports = __toCommonJS(groups_exports);
43
+ var import_utils = require("@nlabs/utils");
44
+ var import_arangojs = require("arangojs");
45
+ var import_flatten = __toESM(require("lodash/flatten"));
46
+ var import_uniqBy = __toESM(require("lodash/uniqBy"));
47
+ var import_utils2 = require("../utils");
48
+ var import_tags = require("./tags");
46
49
  const eventCategory = "groups";
47
50
  const getGroupList = (context, from, to) => {
48
51
  const action = "getListByApp";
@@ -80,10 +83,11 @@ const getGroupListByUser = (context, from, to) => {
80
83
  const getGroupListByTags = (context, tags, from, to) => {
81
84
  const action = "getListByTags";
82
85
  const { database } = context;
83
- return Promise.all((tags || []).map((tagName) => {
84
- const formatTagName = (0, import_utils.parseId)(tagName);
85
- const limit = (0, import_utils2.getLimit)(from, to);
86
- const aqlQry = `FOR p, e IN OUTBOUND "${`tags/${formatTagName}`}" isTagged
86
+ return Promise.all(
87
+ (tags || []).map((tagName) => {
88
+ const formatTagName = (0, import_utils.parseId)(tagName);
89
+ const limit = (0, import_utils2.getLimit)(from, to);
90
+ const aqlQry = `FOR p, e IN OUTBOUND "${`tags/${formatTagName}`}" isTagged
87
91
  FOR u IN users
88
92
  LET likes = (
89
93
  FOR post, like IN INBOUND p._id likes
@@ -99,8 +103,9 @@ const getGroupListByTags = (context, tags, from, to) => {
99
103
  ${limit.aql}
100
104
  SORT p.added
101
105
  RETURN DISTINCT MERGE(p, {user:u, likes:LENGTH(likes), dislikes:LENGTH(dislikes)})`;
102
- return database.query(aqlQry).then((cursor) => cursor.all());
103
- })).then((results) => (0, import_uniqBy.default)((0, import_flatten.default)(results), "_key")).catch((error) => (0, import_utils2.logError)({
106
+ return database.query(aqlQry).then((cursor) => cursor.all());
107
+ })
108
+ ).then((results) => (0, import_uniqBy.default)((0, import_flatten.default)(results), "_key")).catch((error) => (0, import_utils2.logError)({
104
109
  action,
105
110
  category: eventCategory,
106
111
  label: "db_error"
@@ -282,4 +287,4 @@ const isGrouped = (database, userId, groupId) => {
282
287
  isGrouped,
283
288
  updateGroup
284
289
  });
285
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/groups.ts"],
  "sourcesContent": ["/**\n * Copyright (c) 2019-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n */\nimport {createHash, parseArangoId, parseChar, parseId} from '@nlabs/utils';\nimport {aql, Database} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {EdgeCollection} from 'arangojs/collection';\nimport {ArrayCursor} from 'arangojs/cursor';\nimport flatten from 'lodash/flatten';\nimport uniqBy from 'lodash/uniqBy';\n\nimport {ArangoDBLimit} from '../types/arangodb';\nimport {ApiContext} from '../types/auth';\nimport {GroupEdgeType, GroupType, GroupUser} from '../types/groups';\nimport {getLimit, logError} from '../utils';\nimport {extractTags} from './tags';\n\nconst eventCategory: string = 'groups';\n\nexport const getGroupList = (context: ApiContext, from: number, to: number): Promise<GroupType[]> => {\n  const action: string = 'getListByApp';\n  const {database} = context;\n  const limit: ArangoDBLimit = getLimit(from, to);\n  const aqlQry: string = `FOR g in groups\n      LET users = (\n        FOR u, l IN OUTBOUND g._id isGrouped\n        RETURN MERGE (u, {type:l.type})\n      )\n      ${limit.aql}\n      SORT g.added\n      RETURN DISTINCT MERGE(g, {users:users})`;\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(() => null));\n};\n\nexport const getGroupListByUser = (context: ApiContext, from: number, to: number): Promise<GroupType[]> => {\n  // const action: string = 'getListByUser';\n  const {database, session: {userId: sessionId}} = context;\n  const limit: ArangoDBLimit = getLimit(from, to);\n  const aqlQry: string = `FOR g, e IN INBOUND \"${`users/${sessionId}`}\" isGrouped\n      LET users = (\n        FOR u, l IN OUTBOUND g._id isGrouped\n        RETURN MERGE (u, {type:l.type})\n      )\n      ${limit.aql}\n      SORT g.added\n      RETURN DISTINCT MERGE(g, {users:users})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const getGroupListByTags = (\n  context: ApiContext,\n  tags: string[],\n  from: number,\n  to: number\n): Promise<GroupType[]> => {\n  const action: string = 'getListByTags';\n  const {database} = context;\n\n  return Promise.all(\n    (tags || []).map((tagName: string) => {\n      const formatTagName: string = parseId(tagName);\n      const limit: ArangoDBLimit = getLimit(from, to);\n      const aqlQry: string = `FOR p, e IN OUTBOUND \"${`tags/${formatTagName}`}\" isTagged\n          FOR u IN users\n          LET likes = (\n            FOR post, like IN INBOUND p._id likes\n            FILTER like.value == 'like'\n            RETURN like\n          )\n          LET dislikes = (\n            FOR post, like IN INBOUND p._id likes\n            FILTER like.value == 'dislike'\n            RETURN like\n          )\n          FILTER e.type == 'post' && p.userId == u._key\n          ${limit.aql}\n          SORT p.added\n          RETURN DISTINCT MERGE(p, {user:u, likes:LENGTH(likes), dislikes:LENGTH(dislikes)})`;\n\n      return database.query(aqlQry).then((cursor: ArrayCursor) => cursor.all());\n    })\n  )\n    .then((results) => uniqBy(flatten(results), '_key'))\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getGroup = (context: ApiContext, itemId: string): Promise<GroupType> => {\n  const action: string = 'getItem';\n  const {database, session: {userId: sessionId}} = context;\n  const formatItemId: string = parseId(itemId);\n\n  const aqlQry: AqlQuery = aql`FOR g, e IN INBOUND ${`users/${sessionId}`} isGrouped\n      FILTER g._key == ${formatItemId}\n      LET users = (\n        FOR u, l IN OUTBOUND g._id isGrouped\n        RETURN MERGE (u, {type:l.type})\n      )\n      LIMIT 1\n      RETURN DISTINCT MERGE(g, {users:users})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((group: GroupType = {}) => group)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getGroupDetails = (context: ApiContext, groupId: string): Promise<GroupType> => {\n  const action: string = 'getDetails';\n  const {database} = context;\n  const formatGroupId: string = parseId(groupId);\n  const aqlQry: AqlQuery = aql`FOR g IN groups\n      FILTER g._key == ${formatGroupId}\n      LIMIT 1\n      RETURN g`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((group: GroupType = {}) => group)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const createGroupEdge = (\n  database: Database,\n  userId: string,\n  groupId: string,\n): Promise<GroupEdgeType> => {\n  const action: string = 'createGroupEdge';\n  const formatUserId: string = parseArangoId(userId);\n  const formatGroupId: string = parseArangoId(groupId);\n\n  const now: number = Date.now();\n  const edgeCollection: EdgeCollection = database.collection('isGrouped');\n  const edgeId: string = createHash(`group-${formatUserId}-${formatGroupId}`);\n\n  const edge: any = {\n    _from: formatUserId,\n    _key: edgeId,\n    _to: formatGroupId,\n    added: now,\n    type: 'groups'\n  };\n\n  return edgeCollection.save(edge, {returnNew: true})\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, {id: formatUserId}).then(() => null));\n};\n\nexport const addGroup = (context: ApiContext, item: GroupType = {}): Promise<GroupType> => {\n  const action: string = 'add';\n  const {database, session: {userId: sessionId}} = context;\n  const now: number = Date.now();\n  const {\n    description = '',\n    name = 'Untitled',\n    type\n  }: GroupType = item;\n  const id: string = createHash(`group-${sessionId}`);\n  const formatDesc: string = description.substr(0, 640);\n  const formatType: string = parseChar(type, 16) || 'private';\n  const insert: GroupType = {\n    _key: id,\n    added: now,\n    description: formatDesc,\n    modified: now,\n    name,\n    type: formatType\n  };\n  const db: Database = database;\n  const aqlQry: AqlQuery = aql`INSERT ${insert} IN groups RETURN NEW`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((group: GroupType = {}) => {\n      // Update linked tags\n      const {_id: groupDocId}: GroupType = group;\n      return extractTags(db, groupDocId, formatDesc)\n        .then(() => createGroupEdge(database, sessionId, groupDocId).then(() => group));\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const updateGroup = (context: ApiContext, item: GroupType = {}): Promise<GroupType> => {\n  const action: string = 'update';\n  const {database} = context;\n  const {\n    description = '',\n    id,\n    name = 'Untitled'\n  }: GroupType = item;\n  const itemId: string = parseId(id);\n  const now: number = Date.now();\n  const formatDesc: string = description.substr(0, 640);\n  const update: any = {\n    description: formatDesc,\n    modified: now,\n    name\n  };\n  const db: Database = database;\n  const aqlQry: AqlQuery = aql`UPDATE ${itemId} WITH ${update} IN groups RETURN NEW`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((group: GroupType = {}) => {\n      const {_id: groupDocId} = group;\n\n      // Update linked tags\n      return extractTags(db, groupDocId, description).then(() => group);\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const deleteGroup = (context: ApiContext, itemId: string): Promise<GroupType> => {\n  const action: string = 'delete';\n  const {database} = context;\n  const formatItemId: string = parseId(itemId);\n  const aqlQry: AqlQuery = aql`FOR g IN groups\n      FILTER g._key == ${formatItemId}\n      REMOVE g IN groups\n      RETURN OLD`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((group: GroupType = {}) => group)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getGroupsByReaction = (context: ApiContext, reaction: string): Promise<GroupType[]> => {\n  const action: string = 'getGroupsByReaction';\n  const {database, session: {userId: sessionId}} = context;\n  const formatReaction: string = parseChar(reaction, 32);\n  const userDocId: string = `users/${sessionId}`;\n\n  // Query\n  const aqlQry: AqlQuery = aql`FOR u, r IN OUTBOUND ${userDocId} hasReaction\n      FILTER r.value == ${formatReaction}\n      COLLECT reactionName = r.value INTO reactionItems\n      RETURN {value: reactionName, count: LENGTH(reactionItems[*].r.value)}`;\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(() => null));\n};\n\nexport const isGrouped = (database: Database, userId: string, groupId: string): Promise<GroupUser> => {\n  const action: string = 'isGrouped';\n  const formatUserId: string = parseId(userId);\n  const formatGroupId: string = parseId(groupId);\n  const aqlQry: AqlQuery = aql`FOR g IN groups\n    FILTER g._key == ${formatGroupId}\n    FOR u IN INBOUND g._id isGrouped\n    FILTER u._key == ${formatUserId}\n    LIMIT 1\n    RETURN u`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .then((results = []) => {\n      const isValid = !!results.length;\n\n      return {\n        groupId,\n        isValid,\n        userId\n      };\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, {id: userId}).then(() => null));\n};\n"],
  "mappings": "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",
  "names": []
}

290
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/groups.ts"],
  "sourcesContent": ["/**\n * Copyright (c) 2019-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n */\nimport {createHash, parseArangoId, parseChar, parseId} from '@nlabs/utils';\nimport {aql, Database} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {EdgeCollection} from 'arangojs/collection';\nimport {ArrayCursor} from 'arangojs/cursor';\nimport flatten from 'lodash/flatten';\nimport uniqBy from 'lodash/uniqBy';\n\nimport {ArangoDBLimit} from '../types/arangodb';\nimport {ApiContext} from '../types/auth';\nimport {GroupEdgeType, GroupType, GroupUser} from '../types/groups';\nimport {getLimit, logError} from '../utils';\nimport {extractTags} from './tags';\n\nconst eventCategory: string = 'groups';\n\nexport const getGroupList = (context: ApiContext, from: number, to: number): Promise<GroupType[]> => {\n  const action: string = 'getListByApp';\n  const {database} = context;\n  const limit: ArangoDBLimit = getLimit(from, to);\n  const aqlQry: string = `FOR g in groups\n      LET users = (\n        FOR u, l IN OUTBOUND g._id isGrouped\n        RETURN MERGE (u, {type:l.type})\n      )\n      ${limit.aql}\n      SORT g.added\n      RETURN DISTINCT MERGE(g, {users:users})`;\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(() => null));\n};\n\nexport const getGroupListByUser = (context: ApiContext, from: number, to: number): Promise<GroupType[]> => {\n  // const action: string = 'getListByUser';\n  const {database, session: {userId: sessionId}} = context;\n  const limit: ArangoDBLimit = getLimit(from, to);\n  const aqlQry: string = `FOR g, e IN INBOUND \"${`users/${sessionId}`}\" isGrouped\n      LET users = (\n        FOR u, l IN OUTBOUND g._id isGrouped\n        RETURN MERGE (u, {type:l.type})\n      )\n      ${limit.aql}\n      SORT g.added\n      RETURN DISTINCT MERGE(g, {users:users})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const getGroupListByTags = (\n  context: ApiContext,\n  tags: string[],\n  from: number,\n  to: number\n): Promise<GroupType[]> => {\n  const action: string = 'getListByTags';\n  const {database} = context;\n\n  return Promise.all(\n    (tags || []).map((tagName: string) => {\n      const formatTagName: string = parseId(tagName);\n      const limit: ArangoDBLimit = getLimit(from, to);\n      const aqlQry: string = `FOR p, e IN OUTBOUND \"${`tags/${formatTagName}`}\" isTagged\n          FOR u IN users\n          LET likes = (\n            FOR post, like IN INBOUND p._id likes\n            FILTER like.value == 'like'\n            RETURN like\n          )\n          LET dislikes = (\n            FOR post, like IN INBOUND p._id likes\n            FILTER like.value == 'dislike'\n            RETURN like\n          )\n          FILTER e.type == 'post' && p.userId == u._key\n          ${limit.aql}\n          SORT p.added\n          RETURN DISTINCT MERGE(p, {user:u, likes:LENGTH(likes), dislikes:LENGTH(dislikes)})`;\n\n      return database.query(aqlQry).then((cursor: ArrayCursor) => cursor.all());\n    })\n  )\n    .then((results) => uniqBy(flatten(results), '_key'))\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getGroup = (context: ApiContext, itemId: string): Promise<GroupType> => {\n  const action: string = 'getItem';\n  const {database, session: {userId: sessionId}} = context;\n  const formatItemId: string = parseId(itemId);\n\n  const aqlQry: AqlQuery = aql`FOR g, e IN INBOUND ${`users/${sessionId}`} isGrouped\n      FILTER g._key == ${formatItemId}\n      LET users = (\n        FOR u, l IN OUTBOUND g._id isGrouped\n        RETURN MERGE (u, {type:l.type})\n      )\n      LIMIT 1\n      RETURN DISTINCT MERGE(g, {users:users})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((group: GroupType = {}) => group)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getGroupDetails = (context: ApiContext, groupId: string): Promise<GroupType> => {\n  const action: string = 'getDetails';\n  const {database} = context;\n  const formatGroupId: string = parseId(groupId);\n  const aqlQry: AqlQuery = aql`FOR g IN groups\n      FILTER g._key == ${formatGroupId}\n      LIMIT 1\n      RETURN g`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((group: GroupType = {}) => group)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const createGroupEdge = (\n  database: Database,\n  userId: string,\n  groupId: string,\n): Promise<GroupEdgeType> => {\n  const action: string = 'createGroupEdge';\n  const formatUserId: string = parseArangoId(userId);\n  const formatGroupId: string = parseArangoId(groupId);\n\n  const now: number = Date.now();\n  const edgeCollection: EdgeCollection = database.collection('isGrouped');\n  const edgeId: string = createHash(`group-${formatUserId}-${formatGroupId}`);\n\n  const edge: any = {\n    _from: formatUserId,\n    _key: edgeId,\n    _to: formatGroupId,\n    added: now,\n    type: 'groups'\n  };\n\n  return edgeCollection.save(edge, {returnNew: true})\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, {id: formatUserId}).then(() => null));\n};\n\nexport const addGroup = (context: ApiContext, item: GroupType = {}): Promise<GroupType> => {\n  const action: string = 'add';\n  const {database, session: {userId: sessionId}} = context;\n  const now: number = Date.now();\n  const {\n    description = '',\n    name = 'Untitled',\n    type\n  }: GroupType = item;\n  const id: string = createHash(`group-${sessionId}`);\n  const formatDesc: string = description.substr(0, 640);\n  const formatType: string = parseChar(type, 16) || 'private';\n  const insert: GroupType = {\n    _key: id,\n    added: now,\n    description: formatDesc,\n    modified: now,\n    name,\n    type: formatType\n  };\n  const db: Database = database;\n  const aqlQry: AqlQuery = aql`INSERT ${insert} IN groups RETURN NEW`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((group: GroupType = {}) => {\n      // Update linked tags\n      const {_id: groupDocId}: GroupType = group;\n      return extractTags(db, groupDocId, formatDesc)\n        .then(() => createGroupEdge(database, sessionId, groupDocId).then(() => group));\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const updateGroup = (context: ApiContext, item: GroupType = {}): Promise<GroupType> => {\n  const action: string = 'update';\n  const {database} = context;\n  const {\n    description = '',\n    id,\n    name = 'Untitled'\n  }: GroupType = item;\n  const itemId: string = parseId(id);\n  const now: number = Date.now();\n  const formatDesc: string = description.substr(0, 640);\n  const update: any = {\n    description: formatDesc,\n    modified: now,\n    name\n  };\n  const db: Database = database;\n  const aqlQry: AqlQuery = aql`UPDATE ${itemId} WITH ${update} IN groups RETURN NEW`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((group: GroupType = {}) => {\n      const {_id: groupDocId} = group;\n\n      // Update linked tags\n      return extractTags(db, groupDocId, description).then(() => group);\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const deleteGroup = (context: ApiContext, itemId: string): Promise<GroupType> => {\n  const action: string = 'delete';\n  const {database} = context;\n  const formatItemId: string = parseId(itemId);\n  const aqlQry: AqlQuery = aql`FOR g IN groups\n      FILTER g._key == ${formatItemId}\n      REMOVE g IN groups\n      RETURN OLD`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((group: GroupType = {}) => group)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getGroupsByReaction = (context: ApiContext, reaction: string): Promise<GroupType[]> => {\n  const action: string = 'getGroupsByReaction';\n  const {database, session: {userId: sessionId}} = context;\n  const formatReaction: string = parseChar(reaction, 32);\n  const userDocId: string = `users/${sessionId}`;\n\n  // Query\n  const aqlQry: AqlQuery = aql`FOR u, r IN OUTBOUND ${userDocId} hasReaction\n      FILTER r.value == ${formatReaction}\n      COLLECT reactionName = r.value INTO reactionItems\n      RETURN {value: reactionName, count: LENGTH(reactionItems[*].r.value)}`;\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(() => null));\n};\n\nexport const isGrouped = (database: Database, userId: string, groupId: string): Promise<GroupUser> => {\n  const action: string = 'isGrouped';\n  const formatUserId: string = parseId(userId);\n  const formatGroupId: string = parseId(groupId);\n  const aqlQry: AqlQuery = aql`FOR g IN groups\n    FILTER g._key == ${formatGroupId}\n    FOR u IN INBOUND g._id isGrouped\n    FILTER u._key == ${formatUserId}\n    LIMIT 1\n    RETURN u`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .then((results = []) => {\n      const isValid = !!results.length;\n\n      return {\n        groupId,\n        isValid,\n        userId\n      };\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, {id: userId}).then(() => null));\n};\n"],
  "mappings": "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",
  "names": ["import_utils", "uniqBy", "flatten"]
}
