@nlabs/reaktor 0.8.0 → 0.9.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 (198) hide show
  1. package/.eslintrc +10 -0
  2. package/.prettierrc.js +4 -0
  3. package/README.md +1 -1
  4. package/coverage/actions/groups.ts.html +1039 -0
  5. package/coverage/actions/images.ts.html +2500 -0
  6. package/coverage/actions/index.html +131 -0
  7. package/coverage/actions/tags.ts.html +1000 -0
  8. package/coverage/adapters/arangoAdapter.ts.html +151 -0
  9. package/coverage/adapters/index.html +146 -0
  10. package/coverage/adapters/reaktorAdapter.ts.html +127 -0
  11. package/coverage/adapters/tagAdapter.ts.html +160 -0
  12. package/coverage/base.css +224 -0
  13. package/coverage/block-navigation.js +87 -0
  14. package/coverage/clover.xml +6 -0
  15. package/coverage/coverage-final.json +1 -0
  16. package/coverage/favicon.png +0 -0
  17. package/coverage/index.html +176 -0
  18. package/coverage/lcov-report/base.css +224 -0
  19. package/coverage/lcov-report/block-navigation.js +87 -0
  20. package/coverage/lcov-report/favicon.png +0 -0
  21. package/coverage/lcov-report/index.html +101 -0
  22. package/coverage/lcov-report/prettify.css +1 -0
  23. package/coverage/lcov-report/prettify.js +2 -0
  24. package/coverage/lcov-report/sort-arrow-sprite.png +0 -0
  25. package/coverage/lcov-report/sorter.js +196 -0
  26. package/coverage/lcov.info +0 -0
  27. package/coverage/mocks/file.ts.html +118 -0
  28. package/coverage/mocks/group.ts.html +145 -0
  29. package/coverage/mocks/image.ts.html +136 -0
  30. package/coverage/mocks/index.html +146 -0
  31. package/coverage/mocks/post.ts.html +169 -0
  32. package/coverage/mocks/tag.ts.html +121 -0
  33. package/coverage/mocks/user.ts.html +268 -0
  34. package/coverage/prettify.css +1 -0
  35. package/coverage/prettify.js +2 -0
  36. package/coverage/sort-arrow-sprite.png +0 -0
  37. package/coverage/sorter.js +196 -0
  38. package/coverage/types/error.ts.html +145 -0
  39. package/coverage/types/index.html +116 -0
  40. package/coverage/utils/adapterUtils.ts.html +151 -0
  41. package/coverage/utils/analyticsUtils.ts.html +292 -0
  42. package/coverage/utils/arangodbUtils.ts.html +463 -0
  43. package/coverage/utils/index.html +146 -0
  44. package/jest.setup.js +0 -0
  45. package/jpg:- +0 -0
  46. package/lib/actions/apps.d.ts +25 -0
  47. package/lib/actions/apps.js +252 -0
  48. package/lib/actions/conversations.d.ts +14 -0
  49. package/lib/actions/conversations.js +131 -126
  50. package/lib/actions/dynamodb.d.ts +8 -0
  51. package/lib/actions/dynamodb.js +35 -32
  52. package/lib/actions/email.d.ts +5 -0
  53. package/lib/actions/email.js +23 -53
  54. package/lib/actions/files.d.ts +19 -0
  55. package/lib/actions/files.js +188 -202
  56. package/lib/actions/groups.d.ts +14 -0
  57. package/lib/actions/groups.js +38 -36
  58. package/lib/actions/images.d.ts +26 -0
  59. package/lib/actions/images.js +236 -229
  60. package/lib/actions/index.d.ts +21 -0
  61. package/lib/actions/index.js +3 -1
  62. package/lib/actions/ios.d.ts +7 -0
  63. package/lib/actions/ios.js +11 -10
  64. package/lib/actions/locations.d.ts +5 -0
  65. package/lib/actions/locations.js +29 -37
  66. package/lib/actions/messages.d.ts +13 -0
  67. package/lib/actions/messages.js +10 -10
  68. package/lib/actions/notifications.d.ts +5 -0
  69. package/lib/actions/notifications.js +1 -1
  70. package/lib/actions/payments.d.ts +10 -0
  71. package/lib/actions/payments.js +27 -26
  72. package/lib/actions/posts.d.ts +19 -0
  73. package/lib/actions/posts.js +176 -156
  74. package/lib/actions/reactions.d.ts +30 -0
  75. package/lib/actions/reactions.js +30 -28
  76. package/lib/actions/s3.d.ts +7 -0
  77. package/lib/actions/s3.js +37 -32
  78. package/lib/actions/search.d.ts +3 -0
  79. package/lib/actions/search.js +11 -9
  80. package/lib/actions/sms.d.ts +3 -0
  81. package/lib/actions/sms.js +58 -34
  82. package/lib/actions/statistics.d.ts +3 -0
  83. package/lib/actions/statistics.js +21 -18
  84. package/lib/actions/subscription.d.ts +7 -0
  85. package/lib/actions/subscription.js +24 -21
  86. package/lib/actions/tags.d.ts +29 -0
  87. package/lib/actions/tags.js +129 -198
  88. package/lib/actions/users.d.ts +47 -0
  89. package/lib/actions/users.js +188 -194
  90. package/lib/actions/websockets.d.ts +19 -0
  91. package/lib/actions/websockets.js +60 -34
  92. package/lib/adapters/arangoAdapter.d.ts +2 -0
  93. package/lib/adapters/arangoAdapter.js +46 -0
  94. package/lib/adapters/fileAdapter.d.ts +3 -0
  95. package/lib/adapters/fileAdapter.js +76 -0
  96. package/lib/adapters/postAdapter.d.ts +2 -0
  97. package/lib/adapters/postAdapter.js +70 -0
  98. package/lib/adapters/reaktorAdapter.d.ts +6 -0
  99. package/lib/adapters/reaktorAdapter.js +44 -0
  100. package/lib/adapters/tagAdapter.d.ts +2 -0
  101. package/lib/adapters/tagAdapter.js +50 -0
  102. package/lib/adapters/userAdapter.d.ts +2 -0
  103. package/lib/adapters/userAdapter.js +110 -0
  104. package/lib/config.d.ts +20 -0
  105. package/lib/config.js +14 -15
  106. package/lib/index.d.ts +5 -0
  107. package/lib/lambdas/actions/websockets.d.ts +6 -0
  108. package/lib/lambdas/actions/websockets.js +7 -7
  109. package/lib/lambdas/authorizer.d.ts +20 -0
  110. package/lib/lambdas/authorizer.js +1 -1
  111. package/lib/lambdas/connection.d.ts +12 -0
  112. package/lib/lambdas/connection.js +5 -4
  113. package/lib/lambdas/utils/message.d.ts +1 -0
  114. package/lib/lambdas/utils/websocket.d.ts +7 -0
  115. package/lib/lambdas/utils/websocket.js +8 -6
  116. package/lib/mocks/conversation.d.ts +8 -0
  117. package/lib/mocks/conversation.js +35 -0
  118. package/lib/mocks/file.d.ts +11 -0
  119. package/lib/mocks/file.js +38 -0
  120. package/lib/mocks/group.d.ts +17 -0
  121. package/lib/mocks/group.js +47 -0
  122. package/lib/mocks/image.d.ts +3 -0
  123. package/lib/mocks/image.js +43 -0
  124. package/lib/mocks/nlabs.png +0 -0
  125. package/lib/mocks/post.d.ts +38 -0
  126. package/lib/mocks/post.js +55 -0
  127. package/lib/mocks/tag.d.ts +2 -0
  128. package/lib/mocks/tag.js +37 -0
  129. package/lib/mocks/user.d.ts +4 -0
  130. package/lib/mocks/user.js +88 -0
  131. package/lib/templates/email/layout.d.ts +2 -0
  132. package/lib/templates/email/passwordForgot.d.ts +2 -0
  133. package/lib/templates/email/passwordRecovery.d.ts +2 -0
  134. package/lib/templates/email/verifyEmail.d.ts +2 -0
  135. package/lib/templates/email/welcome.d.ts +2 -0
  136. package/lib/templates/sms/passwordForgot.d.ts +2 -0
  137. package/lib/templates/sms/passwordRecovery.d.ts +2 -0
  138. package/lib/templates/sms/verifyEmail.d.ts +2 -0
  139. package/lib/templates/sms/verifyPhone.d.ts +2 -0
  140. package/lib/templates/sms/welcome.d.ts +2 -0
  141. package/lib/types/apps.d.ts +46 -0
  142. package/lib/types/apps.js +17 -1
  143. package/lib/types/arangodb.d.ts +30 -0
  144. package/lib/types/arangodb.js +1 -1
  145. package/lib/types/auth.d.ts +7 -0
  146. package/lib/types/auth.js +1 -1
  147. package/lib/types/connections.d.ts +7 -0
  148. package/lib/types/connections.js +1 -1
  149. package/lib/types/conversations.d.ts +29 -0
  150. package/lib/types/conversations.js +1 -1
  151. package/lib/types/email.d.ts +13 -0
  152. package/lib/types/email.js +1 -1
  153. package/lib/types/error.d.ts +20 -0
  154. package/lib/types/error.js +44 -0
  155. package/lib/types/files.d.ts +26 -0
  156. package/lib/types/files.js +1 -1
  157. package/lib/types/google.d.ts +29 -0
  158. package/lib/types/google.js +1 -1
  159. package/lib/types/groups.d.ts +21 -0
  160. package/lib/types/groups.js +1 -1
  161. package/lib/types/images.d.ts +51 -0
  162. package/lib/types/images.js +1 -1
  163. package/lib/types/index.d.ts +18 -0
  164. package/lib/types/locations.d.ts +20 -0
  165. package/lib/types/locations.js +1 -1
  166. package/lib/types/messages.d.ts +16 -0
  167. package/lib/types/messages.js +1 -1
  168. package/lib/types/notifications.d.ts +17 -0
  169. package/lib/types/notifications.js +1 -1
  170. package/lib/types/payments.d.ts +112 -0
  171. package/lib/types/payments.js +1 -1
  172. package/lib/types/posts.d.ts +31 -0
  173. package/lib/types/posts.js +1 -1
  174. package/lib/types/statistics.d.ts +3 -0
  175. package/lib/types/statistics.js +1 -1
  176. package/lib/types/tags.d.ts +10 -0
  177. package/lib/types/tags.js +1 -1
  178. package/lib/types/users.d.ts +76 -0
  179. package/lib/types/users.js +1 -1
  180. package/lib/types/websocket.d.ts +13 -0
  181. package/lib/types/websocket.js +1 -1
  182. package/lib/utils/adapterUtils.d.ts +1 -0
  183. package/lib/utils/adapterUtils.js +45 -0
  184. package/lib/utils/analyticsUtils.d.ts +21 -0
  185. package/lib/utils/analyticsUtils.js +72 -0
  186. package/lib/utils/arangodbUtils.d.ts +65 -0
  187. package/lib/utils/arangodbUtils.js +144 -0
  188. package/lib/utils/auth.d.ts +20 -0
  189. package/lib/utils/auth.js +13 -30
  190. package/lib/utils/index.d.ts +5 -0
  191. package/lib/utils/index.js +7 -9
  192. package/lib/utils/session.d.ts +16 -0
  193. package/lib/utils/session.js +11 -2
  194. package/package.json +12 -5
  195. package/lib/utils/analytics.js +0 -88
  196. package/lib/utils/arangodb.js +0 -118
  197. package/lib/utils/graphql.js +0 -46
  198. package/lib/utils/objects.js +0 -59
@@ -17,18 +17,22 @@ var __copyProps = (to, from, except, desc) => {
17
17
  var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
18
18
  var files_exports = {};
19
19
  __export(files_exports, {
20
+ GIPHY_URL: () => GIPHY_URL,
20
21
  addFile: () => addFile,
21
- createFile: () => createFile,
22
22
  decodeBase64: () => decodeBase64,
23
23
  encodeBase64: () => encodeBase64,
24
24
  getGiphySearch: () => getGiphySearch,
25
25
  getGiphyTrends: () => getGiphyTrends,
26
26
  getPathUserFiles: () => getPathUserFiles,
27
- getUrlUserFiles: () => getUrlUserFiles,
27
+ getYouTube: () => getYouTube,
28
28
  getYouTubeSearch: () => getYouTubeSearch,
29
29
  getYouTubeTrends: () => getYouTubeTrends,
30
+ insertFile: () => insertFile,
30
31
  linkFiles: () => linkFiles,
31
- updateFiles: () => updateFiles
32
+ parseGiphyItem: () => parseGiphyItem,
33
+ updateFiles: () => updateFiles,
34
+ uploadFile: () => uploadFile,
35
+ upsertFile: () => upsertFile
32
36
  });
33
37
  module.exports = __toCommonJS(files_exports);
34
38
  var import_rip_hunter = require("@nlabs/rip-hunter");
@@ -36,246 +40,224 @@ var import_utils = require("@nlabs/utils");
36
40
  var import_arangojs = require("arangojs");
37
41
  var import_googleapis = require("googleapis");
38
42
  var import_config = require("../config");
39
- var import_analytics = require("../utils/analytics");
43
+ var import_analyticsUtils = require("../utils/analyticsUtils");
40
44
  var import_images = require("./images");
41
45
  var import_posts = require("./posts");
42
- const youtube = import_googleapis.google.youtube({ auth: import_config.Config.get("google.key"), version: "v3" });
43
- const addFile = (context, item = {}) => {
44
- const { database, session: { userId: sessionId, userAccess } } = context;
46
+ var import_error = require("../types/error");
47
+ var import_fileAdapter = require("../adapters/fileAdapter");
48
+ const eventCategory = "files";
49
+ const GIPHY_URL = "http://api.giphy.com/v1/gifs";
50
+ const getYouTube = () => import_googleapis.google.youtube({ auth: import_config.Config.get("google.key"), version: "v3" });
51
+ const insertFile = (context, file) => {
52
+ const action = "insertFile";
53
+ const { database } = context;
54
+ const formattedFile = (0, import_fileAdapter.parseFile)(file);
55
+ const aqlQry = import_arangojs.aql`INSERT ${formattedFile} IN files RETURN NEW`;
56
+ return database.query(aqlQry).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
57
+ action,
58
+ category: eventCategory,
59
+ label: import_error.ErrorTypes.DATABASE_ERROR
60
+ }, error, context));
61
+ };
62
+ const upsertFile = (context, file) => {
63
+ const action = "upsertFile";
64
+ const { database } = context;
65
+ const formattedFile = (0, import_fileAdapter.parseFile)(file);
66
+ const { fileId } = formattedFile;
67
+ const insert = {
68
+ ...formattedFile,
69
+ _key: fileId,
70
+ added: Date.now(),
71
+ modified: Date.now()
72
+ };
73
+ const update = {
74
+ ...formattedFile,
75
+ _key: fileId,
76
+ modified: Date.now()
77
+ };
78
+ const aqlQry = import_arangojs.aql`UPSERT {_key: ${fileId}}
79
+ INSERT ${insert}
80
+ UPDATE ${update}
81
+ IN files RETURN NEW`;
82
+ return database.query(aqlQry).then((cursor) => cursor.next()).then((updatedFile) => updatedFile).catch((error) => (0, import_analyticsUtils.logError)({
83
+ action,
84
+ category: eventCategory,
85
+ label: import_error.ErrorTypes.DATABASE_ERROR
86
+ }, error, context));
87
+ };
88
+ const uploadFile = (context, file, buffer) => {
89
+ const { session: { userId: sessionId } } = context;
90
+ const formattedFile = (0, import_fileAdapter.parseFile)(file);
45
91
  const {
46
92
  description,
47
93
  fileId,
48
94
  fileType,
49
95
  name,
50
96
  url
51
- } = item;
52
- const formatFileId = fileId ? (0, import_utils.parseId)(fileId) : (0, import_utils.createHash)(`file-${sessionId}`);
53
- const isUrl = !!url;
54
- let formatName = (0, import_utils.parseString)(name, 160);
55
- let formatType = (0, import_utils.parseString)(fileType, 16);
56
- if (formatName === "" && isUrl) {
57
- formatName = url.substring(url.lastIndexOf("/") + 1);
58
- }
59
- if (formatType === "") {
60
- const nameArr = formatName.split(".");
61
- const ext = nameArr[nameArr.length - 1];
62
- switch (ext) {
63
- case "jpeg":
64
- case "jpg":
65
- formatType = "image/jpeg";
66
- break;
67
- case "png":
68
- formatType = "image/png";
69
- break;
70
- case "zip":
71
- formatType = "application/zip";
72
- break;
73
- default:
74
- break;
75
- }
97
+ } = formattedFile;
98
+ const now = Date.now();
99
+ const formatFileId = fileId || (0, import_utils.createHash)(`file-${sessionId}`);
100
+ let updatedName = name;
101
+ if (!updatedName && !!url) {
102
+ updatedName = url.substring(url.lastIndexOf("/") + 1);
76
103
  }
77
- let isImage;
78
- switch (formatType) {
79
- case "image/jpeg":
80
- case "image/png":
81
- isImage = true;
82
- break;
83
- default:
84
- isImage = false;
85
- break;
86
- }
87
- const formatDesc = (0, import_utils.parseString)(description, 500);
88
- if (!isImage && userAccess !== 2) {
89
- throw new import_analytics.UserError("account_restriction");
90
- }
91
- const saveToDb = (insert) => {
92
- const aqlQry = import_arangojs.aql`INSERT ${insert} IN files RETURN NEW`;
93
- return database.query(aqlQry).then((cursor) => cursor.next()).then((file = {}) => file).catch((error) => {
94
- throw error;
95
- });
96
- };
97
- const uploadFile = (buf, uploadType) => {
98
- const now = Date.now();
99
- if (isImage) {
100
- return (0, import_images.resizeSaveImage)(context, formatFileId, buf, uploadType).then((resizedImage) => {
101
- const insert2 = {
102
- ...resizedImage,
103
- _key: formatFileId,
104
- added: now,
105
- description: formatDesc,
106
- fileType: formatType,
107
- modified: now,
108
- name: formatName,
109
- userId: sessionId
110
- };
111
- return saveToDb(insert2);
112
- }).catch((error) => {
113
- throw error;
114
- });
115
- }
116
- const insert = {
104
+ if (fileType === "image/jpeg" || fileType === "image/png") {
105
+ return (0, import_images.resizeSaveImage)(context, formatFileId, buffer, fileType).then((resizedImage) => insertFile(context, {
106
+ ...resizedImage,
117
107
  _key: formatFileId,
118
108
  added: now,
119
- description: formatDesc,
120
- fileType: formatType,
121
109
  modified: now,
122
- name: formatName,
110
+ name: updatedName,
123
111
  userId: sessionId
124
- };
125
- return saveToDb(insert);
126
- };
127
- if (isUrl) {
128
- return (0, import_rip_hunter.get)(url).then((body) => uploadFile(Buffer.from(body, "binary"), formatType)).catch(() => {
129
- throw new import_analytics.UserError("file_request");
130
- });
131
- } else if (item.base64 !== "") {
132
- const buffer = Buffer.from(item.base64);
133
- return uploadFile(buffer, formatType);
112
+ }));
134
113
  }
135
- throw new Error("file_required");
114
+ return insertFile(context, {
115
+ ...formattedFile,
116
+ added: now,
117
+ modified: now,
118
+ userId: sessionId
119
+ });
136
120
  };
137
- const getGiphyTrends = (context, limit = 30) => {
138
- const gifUrl = `http://api.giphy.com/v1/gifs/trending?api_key=${import_config.Config.get("giphy.key")}&limit=${limit}`;
139
- return (0, import_rip_hunter.get)(gifUrl).then((res) => res.json()).then((json) => json?.data?.map((gifImage = { id: null, images: null }) => {
140
- const {
141
- id,
142
- images: {
143
- original: { url = "" } = {},
144
- fixed_height_small: { url: thumb = "" } = {}
145
- } = {}
146
- } = gifImage;
147
- return {
148
- id,
149
- thumb,
150
- type: "giphy",
151
- url
152
- };
153
- }));
121
+ const addFile = (context, file) => {
122
+ const action = "addFile";
123
+ const { base64, url } = file;
124
+ if (!!url) {
125
+ return (0, import_rip_hunter.get)(url).then((body) => uploadFile(context, file, Buffer.from(body, "binary"))).catch((error) => (0, import_analyticsUtils.logError)({
126
+ action,
127
+ category: eventCategory,
128
+ label: import_error.ErrorTypes.FILE_REQUEST
129
+ }, error, context));
130
+ } else if (!!base64) {
131
+ const buffer = Buffer.from(base64);
132
+ return uploadFile(context, file, buffer);
133
+ }
134
+ return (0, import_analyticsUtils.logException)({
135
+ action,
136
+ category: eventCategory,
137
+ value: import_error.ErrorTypes.FILE_REQUIRED
138
+ }, {});
154
139
  };
155
- const getGiphySearch = (context, query, limit = 30) => {
140
+ const parseGiphyItem = (gifImage) => {
141
+ const {
142
+ id,
143
+ images: {
144
+ original: { url = "" } = {},
145
+ fixed_height_small: { url: thumb = "" } = {}
146
+ } = {}
147
+ } = gifImage;
148
+ if (!id || !url) {
149
+ return null;
150
+ }
151
+ return {
152
+ id,
153
+ thumb,
154
+ type: "giphy",
155
+ url
156
+ };
157
+ };
158
+ const getGiphyTrends = (limit = 30) => {
159
+ const gifUrl = `${GIPHY_URL}/trending?api_key=${import_config.Config.get("giphy.key")}&limit=${limit}`;
160
+ return (0, import_rip_hunter.get)(gifUrl).then((res) => res.json()).then((json) => json?.data?.map(parseGiphyItem).filter((item) => !!item));
161
+ };
162
+ const getGiphySearch = (query, limit = 30) => {
156
163
  const formatQuery = encodeURI(query);
157
- const gifUrl = `http://api.giphy.com/v1/gifs/search?q=${formatQuery}&api_key=${import_config.Config.get("giphy.key")}&limit=${limit}`;
158
- return fetch(gifUrl).then((res) => res.json()).then((json) => json?.data?.map((gifImage = { id: null, images: null }) => {
159
- const {
160
- id,
161
- images: {
162
- original: { url = "" } = {},
163
- fixed_height_small: { url: thumb = "" } = {}
164
- } = {}
165
- } = gifImage;
166
- return {
167
- id,
168
- thumb,
169
- type: "giphy",
170
- url
171
- };
172
- }));
164
+ const gifUrl = `${GIPHY_URL}/search?q=${formatQuery}&api_key=${import_config.Config.get("giphy.key")}&limit=${limit}`;
165
+ return (0, import_rip_hunter.get)(gifUrl).then((res) => res.json()).then((json) => json?.data?.map(parseGiphyItem).filter((item) => !!item));
173
166
  };
174
- const getYouTubeTrends = (context, limit = 30) => new Promise((resolve, reject) => {
175
- youtube.videos.list({
176
- chart: "mostPopular",
177
- maxResults: limit,
178
- part: ["snippet"],
179
- regionCode: "US"
180
- }, (error, data) => {
181
- if (error) {
182
- console.error(error);
183
- reject(new Error(error[0].message));
184
- } else if (data) {
185
- const list = data.items.map((item) => ({
186
- id: item.id,
187
- thumb: item.snippet.thumbnails.high.url,
188
- type: "youtube",
189
- url: `http://www.youtube.com/embed/${item.id}`
190
- }));
191
- resolve(list);
192
- }
193
- });
194
- });
195
- const getYouTubeSearch = (context, query, limit = 30) => new Promise((resolve, reject) => {
196
- youtube.search.list({
197
- maxResults: limit,
198
- part: ["snippet"],
199
- // eslint-disable-next-line
200
- q: query,
201
- regionCode: "US"
202
- }, (error, data) => {
203
- if (error) {
204
- console.error(error);
205
- reject(new Error(error[0].message));
206
- } else if (data) {
207
- const { items } = data;
208
- const list = items.map((item) => ({
209
- id: item.id,
210
- thumb: item.snippet.thumbnails.high.url,
211
- type: "youtube",
212
- url: `http://www.youtube.com/embed/${item.id}`
213
- }));
214
- resolve(list);
215
- }
167
+ const getYouTubeTrends = (limit = 30) => {
168
+ const action = "getYouTubeTrends";
169
+ const youtube = getYouTube();
170
+ return new Promise((resolve, reject) => {
171
+ youtube.videos.list({
172
+ chart: "mostPopular",
173
+ maxResults: limit,
174
+ part: ["snippet"],
175
+ regionCode: "US"
176
+ }, (error, data) => {
177
+ if (error) {
178
+ console.error(error);
179
+ reject(new Error(import_error.ErrorTypes.YOUTUBE_ERROR));
180
+ } else if (data) {
181
+ const list = data.items.map((item) => ({
182
+ id: item.id,
183
+ thumb: item.snippet.thumbnails.high.url,
184
+ type: "youtube",
185
+ url: `http://www.youtube.com/embed/${item.id}`
186
+ }));
187
+ resolve(list);
188
+ }
189
+ });
216
190
  });
217
- });
218
- const getPathUserFiles = (userId, filename) => `users/${userId}/files/${filename}`;
219
- const getUrlUserFiles = (userId, filename, dir = "files", type = "profile") => {
220
- if (filename) {
221
- return `https://box.${import_config.Config.get("app.url")}/users/${userId}/${dir}/${filename}`;
222
- }
223
- if (type === "profile") {
224
- return `https://box.${import_config.Config.get("app.url")}/defaults/user_bk.jpg`;
225
- }
226
- return `https://box.${import_config.Config.get("app.url")}/defaults/user_wh.jpg`;
227
191
  };
228
- const createFile = (db, file) => {
229
- const { fileId } = file;
230
- const insert = {
231
- _key: fileId,
232
- added: Date.now()
233
- };
234
- const aqlQry = import_arangojs.aql`UPSERT {_key: ${fileId}}
235
- INSERT ${insert}
236
- UPDATE {}
237
- IN files RETURN NEW`;
238
- return db.query(aqlQry).then((cursor) => cursor.next()).then((updatedFile = {}) => updatedFile).catch((error) => {
239
- throw error;
192
+ const getYouTubeSearch = (query, limit = 30) => {
193
+ const action = "getYouTubeSearch";
194
+ const youtube = getYouTube();
195
+ return new Promise((resolve, reject) => {
196
+ youtube.search.list({
197
+ maxResults: limit,
198
+ part: ["snippet"],
199
+ q: query,
200
+ regionCode: "US"
201
+ }, (error, data) => {
202
+ if (error) {
203
+ console.error(error);
204
+ reject(new Error(import_error.ErrorTypes.YOUTUBE_ERROR));
205
+ } else if (data) {
206
+ const { items } = data;
207
+ const list = items.map((item) => ({
208
+ id: item.id,
209
+ thumb: item.snippet.thumbnails.high.url,
210
+ type: "youtube",
211
+ url: `http://www.youtube.com/embed/${item.id}`
212
+ }));
213
+ resolve(list);
214
+ }
215
+ });
240
216
  });
241
217
  };
242
- const linkFiles = (db, files, postId) => Promise.all(
243
- files.map((file) => createFile(db, file).then((file2) => (0, import_posts.createPostEdge)(db, file2, postId)))
218
+ const getPathUserFiles = (userId, filename) => `users/${userId}/files/${filename}`;
219
+ const linkFiles = (context, postDocId, files) => Promise.all(
220
+ files.map((file) => upsertFile(context, file).then((file2) => (0, import_posts.createPostEdge)(context, postDocId, file2.id, "isPosted", { type: file2.fileType })))
244
221
  );
245
- const updateFiles = (db, postId, files) => {
246
- const edgeCollection = db.collection("isPosted");
247
- return edgeCollection.inEdges(postId, {}).then((edges) => {
222
+ const updateFiles = (context, postDocId, files) => {
223
+ const action = "updateFiles";
224
+ const { database } = context;
225
+ const edgeCollection = database.collection("isPosted");
226
+ return edgeCollection.inEdges(postDocId, {}).then((edges) => {
248
227
  if (edges.length) {
249
228
  return Promise.all(
250
229
  edges.map((edge) => {
251
230
  const { _key: edgeKey } = edge;
252
231
  const aqlQry = import_arangojs.aql`REMOVE {_key:${edgeKey}} IN isPosted`;
253
- return db.query(aqlQry).catch((error) => {
232
+ return database.query(aqlQry).catch((error) => {
254
233
  throw error;
255
234
  });
256
235
  })
257
236
  ).then(() => {
258
237
  if (files.length) {
259
- return linkFiles(db, files, postId).then(() => files);
238
+ return linkFiles(context, postDocId, files).then(() => files);
260
239
  }
261
- return files;
240
+ return [];
262
241
  });
263
242
  } else if (files.length) {
264
- return linkFiles(db, files, postId).then(() => files);
243
+ return linkFiles(context, postDocId, files).then(() => files);
265
244
  }
266
245
  return files;
267
- }).catch((error) => {
268
- throw error;
269
- });
246
+ }).catch((error) => (0, import_analyticsUtils.logError)({
247
+ action,
248
+ category: eventCategory,
249
+ label: import_error.ErrorTypes.DATABASE_ERROR
250
+ }, error, context));
270
251
  };
271
252
  const encodeBase64 = (buffer) => Buffer.from(buffer).toString("base64");
272
253
  const decodeBase64 = (dataString) => {
273
- const getData = (str) => str.match(/^data:([A-Za-z-+\/]+);base64,(.+)$/) || [];
254
+ const action = "decodeBase64";
255
+ const getData = (str) => str.match(/^data:([A-Za-z-+/]+);base64,(.+)$/) || [];
274
256
  let matches = getData(dataString);
275
257
  if (matches.length !== 3) {
276
258
  matches = getData(decodeURIComponent(dataString));
277
259
  if (matches.length !== 3) {
278
- throw Error("Invalid input string");
260
+ throw Error(import_error.ErrorTypes.INVALID_INPUT);
279
261
  }
280
262
  }
281
263
  return {
@@ -285,17 +267,21 @@ const decodeBase64 = (dataString) => {
285
267
  };
286
268
  // Annotate the CommonJS export names for ESM import in node:
287
269
  0 && (module.exports = {
270
+ GIPHY_URL,
288
271
  addFile,
289
- createFile,
290
272
  decodeBase64,
291
273
  encodeBase64,
292
274
  getGiphySearch,
293
275
  getGiphyTrends,
294
276
  getPathUserFiles,
295
- getUrlUserFiles,
277
+ getYouTube,
296
278
  getYouTubeSearch,
297
279
  getYouTubeTrends,
280
+ insertFile,
298
281
  linkFiles,
299
- updateFiles
282
+ parseGiphyItem,
283
+ updateFiles,
284
+ uploadFile,
285
+ upsertFile
300
286
  });
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"]
}

287
+ //# 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} from '@nlabs/utils';\nimport {aql} 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 {logError, logException} from '../utils/analyticsUtils';\nimport {resizeSaveImage} from './images';\nimport {createPostEdge} from './posts';\nimport {ErrorTypes} from '../types/error';\nimport {ImageType} from '../types';\nimport {parseFile} from '../adapters/fileAdapter';\n\nconst eventCategory: string = 'files';\n\nexport const GIPHY_URL: string = 'http://api.giphy.com/v1/gifs';\n\nexport const getYouTube = () => google.youtube({auth: Config.get('google.key'), version: 'v3'});\n\nexport const insertFile = (context: ApiContext, file: FileType) => {\n  const action = 'insertFile';\n  const {database} = context;\n  const formattedFile = parseFile(file);\n  const aqlQry: AqlQuery = aql`INSERT ${formattedFile} IN files RETURN NEW`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const upsertFile = (context: ApiContext, file: FileType): Promise<FileType> => {\n  const action = 'upsertFile';\n  const {database} = context;\n  const formattedFile = parseFile(file);\n  const {fileId} = formattedFile;\n  const insert: any = {\n    ...formattedFile,\n    _key: fileId,\n    added: Date.now(),\n    modified: Date.now()\n  };\n  const update: any = {\n    ...formattedFile,\n    _key: fileId,\n    modified: Date.now()\n  };\n\n  const aqlQry: AqlQuery = aql`UPSERT {_key: ${fileId}}\n    INSERT ${insert}\n    UPDATE ${update}\n    IN files RETURN NEW`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((updatedFile: FileType) => updatedFile)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const uploadFile = (context, file: FileType, buffer: Buffer) => {\n  const {session: {userId: sessionId}} = context;\n  const formattedFile = parseFile(file);\n  const {\n    description,\n    fileId,\n    fileType,\n    name,\n    url\n  } = formattedFile;\n  const now = Date.now();\n  const formatFileId: string = fileId || createHash(`file-${sessionId}`);\n  let updatedName = name;\n\n  if(!updatedName && !!url) {\n    updatedName = url.substring(url.lastIndexOf('/') + 1);\n  }\n\n  if(fileType === 'image/jpeg' || fileType === 'image/png') {\n    return resizeSaveImage(context, formatFileId, buffer, fileType)\n      .then((resizedImage: FileType) => insertFile(context, {\n        ...resizedImage,\n        _key: formatFileId,\n        added: now,\n        modified: now,\n        name: updatedName,\n        userId: sessionId\n      }));\n  }\n\n  return insertFile(context, {\n    ...formattedFile,\n    added: now,\n    modified: now,\n    userId: sessionId\n  });\n};\n\nexport const addFile = (context: ApiContext, file: FileType): Promise<FileType> => {\n  const action = 'addFile';\n  const {base64, url} = file;\n\n  if(!!url) {\n    return httpGet(url)\n      .then((body) => uploadFile(context, file, Buffer.from(body, 'binary')))\n      .catch((error: Error) => logError({\n        action,\n        category: eventCategory,\n        label: ErrorTypes.FILE_REQUEST\n      }, error, context));\n  } else if(!!base64) {\n    const buffer: Buffer = Buffer.from(base64);\n    return uploadFile(context, file, buffer);\n  }\n\n  return logException({\n    action,\n    category: eventCategory,\n    value: ErrorTypes.FILE_REQUIRED\n  }, {});\n};\n\nexport const parseGiphyItem = (gifImage: any): ImageType => {\n  const {\n    id,\n    images: {\n      original: {url = ''} = {},\n      fixed_height_small: {url: thumb = ''} = {}\n    } = {}\n  } = gifImage;\n\n  if(!id || !url) {\n    return null;\n  }\n\n  return {\n    id,\n    thumb,\n    type: 'giphy',\n    url\n  };\n};\n\nexport const getGiphyTrends = (limit: number = 30): Promise<ImageType[]> => {\n  const gifUrl = `${GIPHY_URL}/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(parseGiphyItem).filter((item) => !!item));\n};\n\nexport const getGiphySearch = (query: string, limit: number = 30): Promise<ImageType[]> => {\n  const formatQuery = encodeURI(query);\n  const gifUrl = `${GIPHY_URL}/search?q=${formatQuery}&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(parseGiphyItem).filter((item) => !!item));\n};\n\nexport const getYouTubeTrends = (limit: number = 30): Promise<any[]> => {\n  const action = 'getYouTubeTrends';\n  const youtube = getYouTube();\n\n  return 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(ErrorTypes.YOUTUBE_ERROR));\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 = (query: string, limit: number = 30): Promise<any[]> => {\n  const action = 'getYouTubeSearch';\n  const youtube = getYouTube();\n\n  return new Promise((resolve, reject) => {\n    youtube.search.list({\n      maxResults: limit,\n      part: ['snippet'],\n      q: query,\n      regionCode: 'US'\n    }, (error: Error, data: any) => {\n      if(error) {\n        console.error(error);\n        reject(new Error(ErrorTypes.YOUTUBE_ERROR));\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\nexport const getPathUserFiles = (userId: string, filename: string): string => `users/${userId}/files/${filename}`;\n\nexport const linkFiles = (context: ApiContext, postDocId: string, files: FileType[]): Promise<any> =>\n  Promise.all(\n    files.map((file: FileType) => upsertFile(context, file)\n      .then((file: FileType) => createPostEdge(context, postDocId, file.id, 'isPosted', {type: file.fileType})))\n  );\n\nexport const updateFiles = (context: ApiContext, postDocId: string, files: FileType[]): Promise<any> => {\n  const action = 'updateFiles';\n  const {database} = context;\n  const edgeCollection: EdgeCollection = database.collection('isPosted');\n\n  return edgeCollection.inEdges(postDocId, {})\n    .then((edges: any) => {\n      if(edges.length) {\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 database.query(aqlQry)\n              .catch((error: Error) => {\n                throw error;\n              });\n          }))\n          .then(() => {\n            if(files.length) {\n              return linkFiles(context, postDocId, files).then(() => files);\n            }\n\n            return [];\n          });\n      } else if(files.length) {\n        return linkFiles(context, postDocId, files)\n          .then(() => files);\n      }\n\n      return files;\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const encodeBase64 = (buffer: Buffer): string => Buffer.from(buffer).toString('base64');\n\nexport const decodeBase64 = (dataString: string): any => {\n  const action = 'decodeBase64';\n  const getData = (str: string) => str.match(/^data:([A-Za-z-+/]+);base64,(.+)$/) || [];\n  let matches = getData(dataString);\n\n  if(matches.length !== 3) {\n    matches = getData(decodeURIComponent(dataString));\n\n    if(matches.length !== 3) {\n      throw Error(ErrorTypes.INVALID_INPUT);\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": ["httpGet", "file"]
}

@@ -0,0 +1,14 @@
1
+ import { Database } from 'arangojs';
2
+ import { ApiContext } from '../types/auth';
3
+ import { GroupEdgeType, GroupType, GroupUser } from '../types/groups';
4
+ export declare const getGroupList: (context: ApiContext, from: number, to: number) => Promise<GroupType[]>;
5
+ export declare const getGroupListByUser: (context: ApiContext, from: number, to: number) => Promise<GroupType[]>;
6
+ export declare const getGroupListByTags: (context: ApiContext, tags: string[], from: number, to: number) => Promise<GroupType[]>;
7
+ export declare const getGroup: (context: ApiContext, itemId: string) => Promise<GroupType>;
8
+ export declare const getGroupDetails: (context: ApiContext, groupId: string) => Promise<GroupType>;
9
+ export declare const createGroupEdge: (database: Database, userId: string, groupId: string) => Promise<GroupEdgeType>;
10
+ export declare const addGroup: (context: ApiContext, item: GroupType) => Promise<GroupType>;
11
+ export declare const updateGroup: (context: ApiContext, item: GroupType) => Promise<GroupType>;
12
+ export declare const deleteGroup: (context: ApiContext, itemId: string) => Promise<GroupType>;
13
+ export declare const getGroupsByReaction: (context: ApiContext, reaction: string) => Promise<GroupType[]>;
14
+ export declare const isGrouped: (database: Database, userId: string, groupId: string) => Promise<GroupUser>;