@nlabs/reaktor 0.3.0 → 0.4.1

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 (128) hide show
  1. package/lib/actions/conversations.d.ts +14 -0
  2. package/lib/actions/conversations.js +333 -0
  3. package/lib/actions/dynamodb.js +155 -0
  4. package/lib/actions/email.js +177 -0
  5. package/lib/actions/files.js +319 -0
  6. package/lib/{data → actions}/groups.d.ts +4 -3
  7. package/lib/actions/groups.js +282 -0
  8. package/lib/actions/images.d.ts +22 -0
  9. package/lib/actions/images.js +682 -0
  10. package/lib/actions/index.js +40 -0
  11. package/lib/{data → actions}/ios.d.ts +2 -1
  12. package/lib/actions/ios.js +179 -0
  13. package/lib/actions/locations.js +112 -0
  14. package/lib/actions/messages.d.ts +13 -0
  15. package/lib/actions/messages.js +216 -0
  16. package/lib/actions/notifications.js +63 -0
  17. package/lib/{data → actions}/payments.d.ts +2 -2
  18. package/lib/actions/payments.js +491 -0
  19. package/lib/actions/posts.d.ts +19 -0
  20. package/lib/actions/posts.js +538 -0
  21. package/lib/actions/reactions.d.ts +30 -0
  22. package/lib/actions/reactions.js +340 -0
  23. package/lib/{data → actions}/s3.d.ts +1 -1
  24. package/lib/actions/s3.js +122 -0
  25. package/lib/{data → actions}/search.d.ts +2 -2
  26. package/lib/actions/search.js +99 -0
  27. package/lib/actions/sms.js +76 -0
  28. package/lib/actions/statistics.d.ts +2 -0
  29. package/lib/actions/statistics.js +63 -0
  30. package/lib/actions/subscription.js +209 -0
  31. package/lib/actions/tags.d.ts +26 -0
  32. package/lib/actions/tags.js +340 -0
  33. package/lib/actions/users.d.ts +44 -0
  34. package/lib/actions/users.js +571 -0
  35. package/lib/{data → actions}/websockets.d.ts +1 -1
  36. package/lib/actions/websockets.js +156 -0
  37. package/lib/config.d.ts +2 -3
  38. package/lib/config.js +120 -0
  39. package/lib/index.d.ts +1 -1
  40. package/lib/index.js +23 -0
  41. package/lib/templates/email/layout.d.ts +2 -0
  42. package/lib/templates/email/layout.js +292 -0
  43. package/lib/templates/email/passwordForgot.d.ts +2 -0
  44. package/lib/templates/email/passwordForgot.js +28 -0
  45. package/lib/templates/email/passwordRecovery.d.ts +2 -0
  46. package/lib/templates/email/passwordRecovery.js +25 -0
  47. package/lib/templates/email/verifyEmail.d.ts +2 -0
  48. package/lib/templates/email/verifyEmail.js +28 -0
  49. package/lib/templates/email/welcome.d.ts +2 -0
  50. package/lib/templates/email/welcome.js +28 -0
  51. package/lib/templates/sms/passwordForgot.d.ts +2 -0
  52. package/lib/templates/sms/passwordForgot.js +14 -0
  53. package/lib/templates/sms/passwordRecovery.d.ts +2 -0
  54. package/lib/templates/sms/passwordRecovery.js +14 -0
  55. package/lib/templates/sms/verifyEmail.d.ts +2 -0
  56. package/lib/templates/sms/verifyEmail.js +14 -0
  57. package/lib/templates/sms/verifyPhone.d.ts +2 -0
  58. package/lib/templates/sms/verifyPhone.js +14 -0
  59. package/lib/templates/sms/welcome.d.ts +2 -0
  60. package/lib/templates/sms/welcome.js +14 -0
  61. package/lib/types/apps.d.ts +2 -2
  62. package/lib/types/apps.js +4 -0
  63. package/lib/types/arangodb.js +4 -0
  64. package/lib/types/auth.d.ts +4 -8
  65. package/lib/types/auth.js +4 -0
  66. package/lib/types/conversations.d.ts +5 -3
  67. package/lib/types/conversations.js +4 -0
  68. package/lib/types/email.d.ts +2 -2
  69. package/lib/types/email.js +4 -0
  70. package/lib/types/files.js +4 -0
  71. package/lib/types/google.js +4 -0
  72. package/lib/types/groups.d.ts +2 -1
  73. package/lib/types/groups.js +4 -0
  74. package/lib/types/images.d.ts +8 -5
  75. package/lib/types/images.js +4 -0
  76. package/lib/types/index.d.ts +1 -1
  77. package/lib/types/index.js +37 -0
  78. package/lib/types/locations.js +4 -0
  79. package/lib/types/messages.d.ts +12 -2
  80. package/lib/types/messages.js +4 -0
  81. package/lib/types/notifications.d.ts +2 -2
  82. package/lib/types/notifications.js +4 -0
  83. package/lib/types/payments.js +4 -0
  84. package/lib/types/posts.d.ts +18 -1
  85. package/lib/types/posts.js +4 -0
  86. package/lib/types/statistics.d.ts +3 -0
  87. package/lib/types/statistics.js +4 -0
  88. package/lib/types/tags.d.ts +6 -0
  89. package/lib/types/tags.js +4 -0
  90. package/lib/types/users.d.ts +15 -10
  91. package/lib/types/users.js +4 -0
  92. package/lib/utils/analytics.d.ts +7 -0
  93. package/lib/utils/analytics.js +107 -0
  94. package/lib/utils/arangodb.d.ts +1 -1
  95. package/lib/utils/arangodb.js +122 -0
  96. package/lib/utils/auth.js +78 -0
  97. package/lib/utils/graphql.js +40 -0
  98. package/lib/utils/index.d.ts +1 -1
  99. package/lib/utils/index.js +26 -0
  100. package/lib/utils/objects.js +53 -0
  101. package/lib/utils/session.d.ts +18 -0
  102. package/lib/utils/session.js +42 -0
  103. package/package.json +35 -33
  104. package/lib/data/conversations.d.ts +0 -8
  105. package/lib/data/images.d.ts +0 -21
  106. package/lib/data/messages.d.ts +0 -9
  107. package/lib/data/posts.d.ts +0 -23
  108. package/lib/data/reactions.d.ts +0 -14
  109. package/lib/data/tags.d.ts +0 -14
  110. package/lib/data/users.d.ts +0 -17
  111. package/lib/types/reactions.d.ts +0 -15
  112. package/lib/utils/redis.d.ts +0 -1
  113. package/templates/email/layout.html +0 -279
  114. package/templates/email/passwordForgot.html +0 -15
  115. package/templates/email/passwordRecovery.html +0 -12
  116. package/templates/email/verifyEmail.html +0 -15
  117. package/templates/sms/passwordForgot.txt +0 -1
  118. package/templates/sms/passwordRecovery.txt +0 -1
  119. package/templates/sms/verifyEmail.txt +0 -1
  120. package/templates/sms/verifyPhone.txt +0 -1
  121. /package/lib/{data → actions}/dynamodb.d.ts +0 -0
  122. /package/lib/{data → actions}/email.d.ts +0 -0
  123. /package/lib/{data → actions}/files.d.ts +0 -0
  124. /package/lib/{data → actions}/index.d.ts +0 -0
  125. /package/lib/{data → actions}/locations.d.ts +0 -0
  126. /package/lib/{data → actions}/notifications.d.ts +0 -0
  127. /package/lib/{data → actions}/sms.d.ts +0 -0
  128. /package/lib/{data → actions}/subscription.d.ts +0 -0
@@ -0,0 +1,682 @@
1
+ var __create = Object.create;
2
+ var __defProp = Object.defineProperty;
3
+ var __defProps = Object.defineProperties;
4
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
5
+ var __getOwnPropDescs = Object.getOwnPropertyDescriptors;
6
+ var __getOwnPropNames = Object.getOwnPropertyNames;
7
+ var __getOwnPropSymbols = Object.getOwnPropertySymbols;
8
+ var __getProtoOf = Object.getPrototypeOf;
9
+ 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 __export = (target, all) => {
26
+ __markAsModule(target);
27
+ for (var name in all)
28
+ __defProp(target, name, { get: all[name], enumerable: true });
29
+ };
30
+ var __reExport = (target, module2, desc) => {
31
+ if (module2 && typeof module2 === "object" || typeof module2 === "function") {
32
+ for (let key of __getOwnPropNames(module2))
33
+ if (!__hasOwnProp.call(target, key) && key !== "default")
34
+ __defProp(target, key, { get: () => module2[key], enumerable: !(desc = __getOwnPropDesc(module2, key)) || desc.enumerable });
35
+ }
36
+ return target;
37
+ };
38
+ var __toModule = (module2) => {
39
+ 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);
40
+ };
41
+ __export(exports, {
42
+ addImage: () => addImage,
43
+ addImageEdge: () => addImageEdge,
44
+ deleteImage: () => deleteImage,
45
+ getAppImageUrl: () => getAppImageUrl,
46
+ getImage: () => getImage,
47
+ getImageCountByItem: () => getImageCountByItem,
48
+ getImageOptional: () => getImageOptional,
49
+ getImagesByGroup: () => getImagesByGroup,
50
+ getImagesByItem: () => getImagesByItem,
51
+ getImagesByReactions: () => getImagesByReactions,
52
+ getImagesByUser: () => getImagesByUser,
53
+ getPathUserImages: () => getPathUserImages,
54
+ getUserImageUrl: () => getUserImageUrl,
55
+ parseImageOptions: () => parseImageOptions,
56
+ resizeSaveImage: () => resizeSaveImage,
57
+ updateImage: () => updateImage
58
+ });
59
+ var import_rip_hunter = __toModule(require("@nlabs/rip-hunter"));
60
+ var import_utils = __toModule(require("@nlabs/utils"));
61
+ var import_arangojs = __toModule(require("arangojs"));
62
+ var import_file_type = __toModule(require("file-type"));
63
+ var import_gm = __toModule(require("gm"));
64
+ var import_cloneDeep = __toModule(require("lodash/cloneDeep"));
65
+ var import_isEmpty = __toModule(require("lodash/isEmpty"));
66
+ var import_luxon = __toModule(require("luxon"));
67
+ var import_config = __toModule(require("../config"));
68
+ var import_utils2 = __toModule(require("../utils"));
69
+ var import_groups = __toModule(require("./groups"));
70
+ var import_s32 = __toModule(require("./s3"));
71
+ const eventCategory = "images";
72
+ const parseImageOptions = (options = {}) => {
73
+ const {
74
+ from = 0,
75
+ to = 30,
76
+ type = "default"
77
+ } = options;
78
+ return {
79
+ limit: (0, import_utils2.getLimit)(from, to),
80
+ type: (0, import_utils.parseChar)(type, 32)
81
+ };
82
+ };
83
+ const getImageOptional = (fields = []) => fields.reduce((selects, field) => {
84
+ if (field.includes("Count")) {
85
+ return (0, import_utils2.selectReactionCountByType)("images", "i", field, selects);
86
+ }
87
+ switch (field) {
88
+ case "reactions": {
89
+ selects.queries.push(`LET reactions = (
90
+ FOR image, r IN INBOUND i._id reactions
91
+ COLLECT reactionName = r.value INTO reactionItems
92
+ RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}
93
+ )`);
94
+ selects.objects.push("reactions:reactions");
95
+ return selects;
96
+ }
97
+ case "tags": {
98
+ selects.queries.push(`LET tags = (
99
+ FOR t, pl IN INBOUND i._id isTagged
100
+ RETURN t
101
+ )`);
102
+ selects.objects.push("tags:tags");
103
+ return selects;
104
+ }
105
+ case "user": {
106
+ selects.queries.push(`LET user = FIRST(
107
+ FOR u IN users
108
+ FILTER i.userId == u._key
109
+ LIMIT 1
110
+ RETURN u
111
+ )`);
112
+ selects.objects.push("user:user");
113
+ return selects;
114
+ }
115
+ default: {
116
+ return selects;
117
+ }
118
+ }
119
+ }, { objects: [], queries: [] });
120
+ const getImagesByUser = (context, userId, from, to) => {
121
+ const action = "getImagesByUser";
122
+ const { database } = context;
123
+ const formatUserId = (0, import_utils.parseId)(userId);
124
+ const limit = (0, import_utils2.getLimit)(from, to);
125
+ const aqlQry = `FOR i IN images
126
+ FILTER i.userId == "${formatUserId}"
127
+ LET user = FIRST(
128
+ FOR u IN users
129
+ FILTER u._key == i.userId
130
+ LIMIT 1
131
+ RETURN u
132
+ )
133
+ ${limit.aql}
134
+ SORT i.added
135
+ RETURN MERGE(i, {user:user})`;
136
+ return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_utils2.logError)({
137
+ action,
138
+ category: eventCategory,
139
+ label: "db_error"
140
+ }, error, context).then(() => null));
141
+ };
142
+ const getImageCountByItem = (context, itemId) => {
143
+ const action = "getImageCountByItem";
144
+ const { database } = context;
145
+ const formatItemId = (0, import_utils.parseArangoId)(itemId);
146
+ const aqlQry = import_arangojs.aql`FOR i IN hasImages
147
+ FILTER i._to == ${formatItemId}
148
+ RETURN {count: COUNT(i)}`;
149
+ return database.query(aqlQry).then((cursor) => cursor.next()).then(({ count } = { count: 0 }) => count).catch((error) => (0, import_utils2.logError)({
150
+ action,
151
+ category: eventCategory,
152
+ label: "db_error"
153
+ }, error, context).then(() => 0));
154
+ };
155
+ const getImagesByItem = async (context, itemId, options = {}) => {
156
+ const action = "getImagesByItem";
157
+ const { database, fields = [] } = context;
158
+ const formatItemId = (0, import_utils.parseArangoId)(itemId);
159
+ const { limit } = parseImageOptions(options);
160
+ const { objects: selectObjects, queries: selectQueries } = getImageOptional(fields);
161
+ const aqlQry = `FOR i, l IN 1..1 OUTBOUND "${formatItemId}" hasImages
162
+ FILTER NOT IS_NULL(i)
163
+ ${selectQueries.join("\n")}
164
+ SORT i.added
165
+ ${limit.aql}
166
+ RETURN MERGE(i, {${selectObjects.join(", ")}})`;
167
+ return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_utils2.logError)({
168
+ action,
169
+ category: eventCategory,
170
+ label: "db_error"
171
+ }, error, context).then(() => null));
172
+ };
173
+ const getImagesByGroup = (context, params) => {
174
+ const action = "getImagesByGroup";
175
+ const { database, session: { userId: sessionId } } = context;
176
+ const { filters = [], groupId, from, to } = params;
177
+ const formatGroupId = (0, import_utils.parseId)(groupId);
178
+ const limit = (0, import_utils2.getLimit)(from, to);
179
+ filters.map((filter) => {
180
+ const { conditional, name, value } = filter;
181
+ let formatCond = conditional;
182
+ if (conditional !== ">=" && conditional !== "<=" && conditional !== ">" && conditional !== "<") {
183
+ formatCond = "==";
184
+ }
185
+ switch (name) {
186
+ case "added":
187
+ return `p.added ${formatCond} ${(0, import_utils.parseNum)(value)}`;
188
+ default:
189
+ return "";
190
+ }
191
+ });
192
+ return (0, import_groups.getGroupDetails)(context, formatGroupId).then((group) => {
193
+ if (group.privacy === "public") {
194
+ filters.push(`p.groupId == "${groupId}"`);
195
+ const filterStr = filters.join(" && ");
196
+ const aqlQry = `FOR i IN
197
+ FLATTEN(
198
+ FOR p IN posts
199
+ FILTER ${filterStr}
200
+ LET images = (
201
+ FOR i, e IN INBOUND p._id hasImages
202
+ RETURN i
203
+ )
204
+ SORT p.added DESC
205
+ RETURN images
206
+ )
207
+ SORT i.added DESC
208
+ ${limit.aql}
209
+ RETURN i`;
210
+ return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_utils2.logError)({
211
+ action,
212
+ category: eventCategory,
213
+ label: "db_error"
214
+ }, error, context).then(() => null));
215
+ }
216
+ return (0, import_groups.isGrouped)(database, sessionId, groupId).then((grouped) => {
217
+ if (grouped.isValid) {
218
+ filters.push(`p.groupId == "${grouped.groupId}"`);
219
+ const filterList = filters.join(" && ");
220
+ const aqlQry = `FOR p IN post
221
+ FILTER ${filterList}
222
+ FOR f IN p.files
223
+ FILTER f.type == "image/jpeg" || f.type == "image/png"
224
+ ${limit.aql}
225
+ SORT p.added DESC
226
+ RETURN f`;
227
+ return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_utils2.logError)({
228
+ action,
229
+ category: eventCategory,
230
+ label: "db_error"
231
+ }, error, context).then(() => null));
232
+ }
233
+ return [];
234
+ });
235
+ });
236
+ };
237
+ const getImagesByReactions = (context, reactions = [], options) => {
238
+ const action = "getUsersByImage";
239
+ const { database, fields = [], session: { userId: sessionId } } = context;
240
+ const { limit } = parseImageOptions(options);
241
+ const { objects: selectObjects, queries: selectQueries } = getImageOptional(fields);
242
+ const formatSessionId = `users/${sessionId}`;
243
+ const formatReactions = reactions.map((reactionName) => (0, import_utils.parseChar)(reactionName, 32));
244
+ const filterBy = [
245
+ 'r.type == "images"',
246
+ `POSITION(${JSON.stringify(formatReactions)}, LOWER(r.name))`
247
+ ];
248
+ const aqlQry = `FOR i, r IN OUTBOUND "${formatSessionId}" hasReactions
249
+ OPTIONS {vertexCollections: "images"}
250
+ ${selectQueries.join("\n")}
251
+ FILTER ${filterBy.join(" && ")}
252
+ ${limit.aql}
253
+ RETURN MERGE(i, {${selectObjects.join(", ")}})`;
254
+ return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_utils2.logError)({
255
+ action,
256
+ category: eventCategory,
257
+ label: "db_error"
258
+ }, error, context).then(() => []));
259
+ };
260
+ const getImage = (context, id) => {
261
+ const action = "getItem";
262
+ const { database } = context;
263
+ const formatId = (0, import_utils.parseId)(id);
264
+ const aqlQry = import_arangojs.aql`FOR i IN images
265
+ FILTER i._key==${formatId}
266
+ LIMIT 1
267
+ RETURN i`;
268
+ return database.query(aqlQry).then((cursor) => cursor.next()).then((image = {}) => image).catch((error) => (0, import_utils2.logError)({
269
+ action,
270
+ category: eventCategory,
271
+ label: "db_error"
272
+ }, error, context).then(() => null));
273
+ };
274
+ const getPathUserImages = (userId, imageId, type, dir = "images") => {
275
+ let filename = imageId;
276
+ switch (type) {
277
+ case "image/png":
278
+ filename = `${imageId}.png`;
279
+ break;
280
+ default:
281
+ filename = `${imageId}.jpg`;
282
+ break;
283
+ }
284
+ return `users/${userId}/${dir}/${filename}`;
285
+ };
286
+ const getAppImageUrl = (data) => {
287
+ const { imageId, directory = "images", imageType = "profile", isThumb, type, typeId } = data;
288
+ const host = import_config.Config.get("environment") === "prod" ? `https://box.${import_config.Config.get("app.url")}` : `https://s3.amazonaws.com/dev.${import_config.Config.get("app.url")}`;
289
+ const suffix = isThumb ? "-th" : "";
290
+ if (imageId) {
291
+ switch (type) {
292
+ case "apps":
293
+ return `${host}/${type}/${directory}/${imageId}${suffix}.jpg`;
294
+ case "users":
295
+ return `${host}/${type}/${typeId}/${directory}/${imageId}${suffix}.jpg`;
296
+ }
297
+ if (imageType === "profile") {
298
+ return `${host}/defaults/${type}_bk${suffix}.jpg`;
299
+ }
300
+ return `${host}/defaults/${type}_wh${suffix}.jpg`;
301
+ }
302
+ return "";
303
+ };
304
+ const getUserImageUrl = (data) => {
305
+ const { bucket, imageId, isThumb = false, userId } = data;
306
+ const imgDir = isThumb ? "thumbs" : "images";
307
+ if (imageId) {
308
+ const imageKey = `users/${userId}/${imgDir}/${imageId}.jpg`;
309
+ return (0, import_s32.s3GetSignedUrl)({ Bucket: bucket, Key: imageKey, Expires: 900 });
310
+ }
311
+ return "";
312
+ };
313
+ const resizeSaveImage = (context, imageId, buffer, type = "image/jpeg", s3Options) => {
314
+ const action = "resizeSaveImage";
315
+ const { session: { userId: sessionId } } = context;
316
+ const imgW = import_config.Config.get("image.imgSize");
317
+ const imgQ = import_config.Config.get("image.imgQuality");
318
+ let photo = {};
319
+ const format = type.split("/")[1];
320
+ console.log({ buffer, format, imgW, imgQ });
321
+ return new Promise((resolve) => {
322
+ (0, import_gm.default)(buffer, "img").setFormat(format).quality(imgQ).autoOrient().resize(imgW, imgW, ">").identify({ bufferStream: true }, (error, val = {}) => {
323
+ if (error) {
324
+ (0, import_utils2.logError)({
325
+ action,
326
+ category: eventCategory,
327
+ label: "image_save",
328
+ value: "gm_image_identify"
329
+ }, error, context).catch((error2) => {
330
+ throw error2;
331
+ });
332
+ } else {
333
+ const formatVals = (0, import_utils2.lowerCaseKeys)(val);
334
+ const { make: cameraMake, model: cameraModel, datetimeoriginal: taken } = formatVals;
335
+ photo = __spreadProps(__spreadValues({}, (0, import_cloneDeep.default)(photo)), {
336
+ make: cameraMake,
337
+ model: cameraModel,
338
+ taken: taken ? import_luxon.DateTime.fromMillis(taken).millisecond : null
339
+ });
340
+ const stats = formatVals["channel statistics"];
341
+ if (stats) {
342
+ let { red, green, blue, mean } = stats;
343
+ if (red) {
344
+ mean = red["standard deviation"] || red.mean;
345
+ red = mean ? +mean.split(" ")[0].substring(0, 3) : 0;
346
+ } else {
347
+ red = 0;
348
+ }
349
+ if (green) {
350
+ mean = green["standard deviation"] || green.mean;
351
+ green = mean ? +mean.split(" ")[0].substring(0, 3) : 0;
352
+ } else {
353
+ green = 0;
354
+ }
355
+ if (blue) {
356
+ mean = blue["standard deviation"] || blue.mean;
357
+ blue = mean ? +mean.split(" ")[0].substring(0, 3) : 0;
358
+ } else {
359
+ blue = 0;
360
+ }
361
+ const rgb = blue | green << 8 | red << 16;
362
+ const color = rgb.toString(16);
363
+ photo.color = color === "0" ? "000000" : color;
364
+ }
365
+ }
366
+ }).stream((error, stdout) => {
367
+ if (error) {
368
+ (0, import_utils2.logError)({
369
+ action,
370
+ category: eventCategory,
371
+ isInternal: true,
372
+ label: "image_save",
373
+ value: "gm_image_stream"
374
+ }, error, context).then(() => null);
375
+ } else {
376
+ let imageBuffer = Buffer.from("");
377
+ stdout.on("data", (data) => {
378
+ imageBuffer = Buffer.concat([imageBuffer, data]);
379
+ });
380
+ stdout.on("end", () => {
381
+ photo.fileSize = imageBuffer.length;
382
+ const imageObj = __spreadProps(__spreadValues({
383
+ ACL: "authenticated-read",
384
+ Body: imageBuffer,
385
+ Bucket: null,
386
+ ContentType: type
387
+ }, s3Options || {}), {
388
+ Key: getPathUserImages(sessionId, imageId, type, "images")
389
+ });
390
+ (0, import_s32.s3Put)(imageObj).then(() => {
391
+ const thmW = import_config.Config.get("image.thmSize");
392
+ const thmQ = import_config.Config.get("image.thmQuality");
393
+ (0, import_gm.default)(imageBuffer, "img").setFormat("jpeg").size((thumbError, resizeVal) => {
394
+ if (!thumbError) {
395
+ const { height, width } = resizeVal;
396
+ photo = __spreadProps(__spreadValues({}, (0, import_cloneDeep.default)(photo)), { height, width });
397
+ }
398
+ }).gravity("Center").resize(thmW, thmW, "^").extent(thmW, thmW).quality(thmQ).stream((streamError, thumbStdout) => {
399
+ if (streamError) {
400
+ (0, import_utils2.logError)({
401
+ action,
402
+ category: eventCategory,
403
+ label: "image_save",
404
+ value: "gm_thumbnail_steam"
405
+ }, streamError, context).catch((error2) => {
406
+ throw error2;
407
+ });
408
+ } else {
409
+ let thumbBuffer = Buffer.from("");
410
+ thumbStdout.on("data", (data) => {
411
+ thumbBuffer = Buffer.concat([thumbBuffer, data]);
412
+ });
413
+ thumbStdout.on("end", () => {
414
+ const thumbObj = __spreadProps(__spreadValues({
415
+ ACL: "authenticated-read",
416
+ Body: thumbBuffer,
417
+ Bucket: null,
418
+ ContentType: type
419
+ }, s3Options || {}), {
420
+ Key: getPathUserImages(sessionId, imageId, type, "thumbs")
421
+ });
422
+ (0, import_s32.s3Put)(thumbObj).then(() => {
423
+ resolve(photo);
424
+ }).catch((s3PutError) => (0, import_utils2.logError)({
425
+ action,
426
+ category: eventCategory,
427
+ label: "image_save",
428
+ value: "s3_put_image"
429
+ }, s3PutError, context).catch((error2) => {
430
+ throw error2;
431
+ }));
432
+ });
433
+ }
434
+ });
435
+ }).catch((s3Error) => (0, import_utils2.logError)({
436
+ action,
437
+ category: eventCategory,
438
+ isInternal: true,
439
+ label: "image_save",
440
+ value: "s3_image_save"
441
+ }, s3Error, context).then(() => null));
442
+ });
443
+ }
444
+ });
445
+ });
446
+ };
447
+ const addImage = (context, image, s3Options) => {
448
+ const action = "addImage";
449
+ const { database, session: { userId: sessionId } } = context;
450
+ const { imageId, description, buffer, fileType } = image;
451
+ const now = Date.now();
452
+ return resizeSaveImage(context, imageId, buffer, fileType, s3Options).then((resizedImage) => {
453
+ const insert = __spreadProps(__spreadValues({}, resizedImage), {
454
+ _key: imageId,
455
+ added: now,
456
+ description,
457
+ fileType,
458
+ modified: now,
459
+ userId: sessionId
460
+ });
461
+ const aqlQry = import_arangojs.aql`INSERT ${insert} IN images RETURN NEW`;
462
+ return database.query(aqlQry).then((cursor) => cursor.next()).then(import_utils2.defaultObject).catch((error) => (0, import_utils2.logError)({
463
+ action,
464
+ category: eventCategory,
465
+ isInternal: true,
466
+ label: "db_error"
467
+ }, error, context).then(() => null));
468
+ }).catch((error) => (0, import_utils2.logError)({
469
+ action,
470
+ category: eventCategory,
471
+ isInternal: true,
472
+ label: "image_resize"
473
+ }, error, context).then(() => null));
474
+ };
475
+ const addImageEdge = async (context, imageEdge) => {
476
+ const action = "addImageEdge";
477
+ const { database, session: { userId: sessionId } } = context;
478
+ const { imageId, itemId, itemType } = imageEdge;
479
+ const now = Date.now();
480
+ const edgeCollection = database.collection("hasImages");
481
+ const edgeId = (0, import_utils.createHash)(`hasImages-${imageId}-${itemId}-${sessionId}`);
482
+ const formatItemType = (0, import_utils.parseChar)(itemType).toLowerCase();
483
+ const formatItemId = (0, import_utils.parseId)(itemId);
484
+ let itemDocId;
485
+ const formatImgId = (0, import_utils.parseId)(imageId);
486
+ const fileDocId = `images/${formatImgId}`;
487
+ switch (formatItemType) {
488
+ case "groups":
489
+ itemDocId = `groups/${formatItemId}`;
490
+ break;
491
+ case "posts":
492
+ itemDocId = `posts/${formatItemId}`;
493
+ break;
494
+ case "users":
495
+ case "profile":
496
+ itemDocId = `users/${formatItemId}`;
497
+ break;
498
+ default:
499
+ itemDocId = "";
500
+ break;
501
+ }
502
+ const edge = {
503
+ _from: itemDocId,
504
+ _key: edgeId,
505
+ _to: fileDocId,
506
+ added: now,
507
+ type: itemType
508
+ };
509
+ if (!(0, import_isEmpty.default)(itemDocId)) {
510
+ return edgeCollection.save(edge, { returnNew: true }).then((fileEdge) => edgeCollection.document(fileEdge)).catch((error) => (0, import_utils2.logError)({
511
+ action,
512
+ category: eventCategory,
513
+ isInternal: true,
514
+ label: "db_error",
515
+ params: {
516
+ edge,
517
+ fileDocId,
518
+ imageId,
519
+ itemDocId,
520
+ itemId,
521
+ itemType
522
+ }
523
+ }, error, context).then(() => null));
524
+ }
525
+ return {};
526
+ };
527
+ const updateImage = async (context, image, s3Options) => {
528
+ const action = "updateImage";
529
+ const { database, session: { userId: sessionId } } = context;
530
+ const {
531
+ base64 = "",
532
+ buffer: imageBuffer,
533
+ description = "",
534
+ imageId,
535
+ itemId,
536
+ itemType,
537
+ fileType,
538
+ url = ""
539
+ } = image;
540
+ const now = Date.now();
541
+ const formatImageId = imageId || (0, import_utils.createHash)(`image-${sessionId}`);
542
+ const formatItemId = (0, import_utils.parseId)(itemId);
543
+ const formatItemType = (0, import_utils.parseChar)(itemType, 16).toLowerCase();
544
+ const customParams = {
545
+ description,
546
+ imageId: formatImageId,
547
+ userId: sessionId
548
+ };
549
+ if (!(0, import_isEmpty.default)(base64) || imageBuffer) {
550
+ let buffer = imageBuffer;
551
+ if (!(0, import_isEmpty.default)(base64)) {
552
+ const formatBase64 = base64.substr(base64.indexOf(",") + 1);
553
+ buffer = Buffer.from(formatBase64, "base64");
554
+ }
555
+ let updatedFileType = fileType;
556
+ if (!fileType) {
557
+ const fileMime = await import_file_type.default.fromBuffer(buffer);
558
+ const { mime } = fileMime;
559
+ updatedFileType = mime;
560
+ }
561
+ const imgParams = __spreadProps(__spreadValues({}, customParams), {
562
+ buffer,
563
+ fileType: updatedFileType
564
+ });
565
+ return addImage(context, imgParams, s3Options).then((image2) => {
566
+ if (formatItemId && formatItemType) {
567
+ const imageEdge = {
568
+ imageId: formatImageId,
569
+ itemId: formatItemId,
570
+ itemType: formatItemType
571
+ };
572
+ return addImageEdge(context, imageEdge).then(() => image2);
573
+ }
574
+ return image2;
575
+ }).catch((error) => (0, import_utils2.logError)({
576
+ action,
577
+ category: eventCategory,
578
+ label: "image_save_error"
579
+ }, error, context).then(() => null));
580
+ } else if (url !== "") {
581
+ let contentType;
582
+ return (0, import_rip_hunter.get)(url).then((res) => {
583
+ if (res.status !== 200) {
584
+ return (0, import_utils2.logException)({
585
+ action,
586
+ category: eventCategory,
587
+ label: "fetch_image_url",
588
+ value: res.statusText
589
+ }, context).then(() => null);
590
+ }
591
+ contentType = res.headers.get("content-type");
592
+ return res;
593
+ }).then((res) => res.buffer()).then((buffer) => {
594
+ const imgParams = __spreadProps(__spreadValues({}, customParams), {
595
+ buffer,
596
+ fileType: contentType
597
+ });
598
+ return addImage(context, imgParams, s3Options).then((image2) => {
599
+ if (formatItemId && formatItemType) {
600
+ const imageEdge = { imageId: formatImageId, itemId: formatItemId, itemType: formatItemType };
601
+ return addImageEdge(context, imageEdge).then(() => image2);
602
+ }
603
+ return image2;
604
+ });
605
+ }).catch((error) => (0, import_utils2.logError)({
606
+ action,
607
+ category: eventCategory,
608
+ label: "fetch_error"
609
+ }, error, context).then(() => null));
610
+ } else if (imageId !== "") {
611
+ const update = {
612
+ description,
613
+ modified: now
614
+ };
615
+ const aqlQry = import_arangojs.aql`UPDATE {_key: ${formatImageId}} WITH ${update} IN images RETURN NEW`;
616
+ return database.query(aqlQry).then((cursor) => cursor.next()).catch((error) => (0, import_utils2.logError)({
617
+ action,
618
+ category: eventCategory,
619
+ label: "db_error",
620
+ params: {
621
+ aqlQry,
622
+ formatImageId,
623
+ update
624
+ }
625
+ }, error, context).then(() => null));
626
+ }
627
+ return null;
628
+ };
629
+ const deleteImage = async (context, imageId) => {
630
+ const action = "delete";
631
+ const { database, session: { userId: sessionId } } = context;
632
+ const formatImageId = (0, import_utils.parseId)(imageId);
633
+ const removeEdgeQuery = import_arangojs.aql`FOR hi IN hasImages
634
+ FILTER hi._from == ${formatImageId}
635
+ REMOVE hi IN hasImages
636
+ RETURN OLD`;
637
+ await database.query(removeEdgeQuery);
638
+ const aqlQuery = import_arangojs.aql`FOR i IN images
639
+ FILTER i._key == ${formatImageId} && i.userId == ${sessionId}
640
+ REMOVE i IN images
641
+ RETURN OLD`;
642
+ const image = await database.query(aqlQuery).then((cursor) => cursor.next()).catch((error) => (0, import_utils2.logError)({
643
+ action,
644
+ category: eventCategory,
645
+ label: "db_error"
646
+ }, error, context).then(() => null));
647
+ if (!(0, import_isEmpty.default)(image)) {
648
+ const { _key: imageKey } = image;
649
+ const params = {
650
+ Bucket: null,
651
+ Delete: {
652
+ Objects: [
653
+ { Key: `users/${sessionId}/images/${imageKey}.jpg` },
654
+ { Key: `users/${sessionId}/thumbs/${imageKey}.jpg` }
655
+ ],
656
+ Quiet: true
657
+ }
658
+ };
659
+ return (0, import_s32.s3DeleteList)(params).then(() => image);
660
+ }
661
+ return {};
662
+ };
663
+ // Annotate the CommonJS export names for ESM import in node:
664
+ 0 && (module.exports = {
665
+ addImage,
666
+ addImageEdge,
667
+ deleteImage,
668
+ getAppImageUrl,
669
+ getImage,
670
+ getImageCountByItem,
671
+ getImageOptional,
672
+ getImagesByGroup,
673
+ getImagesByItem,
674
+ getImagesByReactions,
675
+ getImagesByUser,
676
+ getPathUserImages,
677
+ getUserImageUrl,
678
+ parseImageOptions,
679
+ resizeSaveImage,
680
+ updateImage
681
+ });
682
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/images.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, parseArangoId, parseChar, parseId, parseNum} from '@nlabs/utils';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {EdgeCollection} from 'arangojs/collection';\nimport {ArrayCursor} from 'arangojs/cursor';\nimport {DeleteObjectsRequest, PutObjectRequest} from 'aws-sdk/clients/s3';\nimport FileType, {FileTypeResult} from 'file-type';\nimport gm from 'gm';\nimport cloneDeep from 'lodash/cloneDeep';\nimport isEmpty from 'lodash/isEmpty';\nimport {DateTime} from 'luxon';\n\nimport {Config} from '../config';\nimport {\n  ApiContext,\n  ArangoDBLimit,\n  GroupType,\n  GroupUser,\n  ImageEdgeType,\n  ImageIdentifyType,\n  ImageOptions,\n  ImageType,\n  ImageUrlData,\n  QueryFilter\n} from '../types';\nimport {defaultObject, getLimit, logError, logException, lowerCaseKeys, selectReactionCountByType} from '../utils';\nimport {getGroupDetails, isGrouped} from './groups';\nimport {s3DeleteList, s3GetSignedUrl, s3Put} from './s3';\n\nconst eventCategory: string = 'images';\n\nexport const parseImageOptions = (options: ImageOptions = {}) => {\n  const {\n    from = 0,\n    to = 30,\n    type = 'default'\n  } = options;\n\n  return {\n    limit: getLimit(from, to),\n    type: parseChar(type, 32)\n  };\n};\n\nexport const getImageOptional = (fields: string[] = []) =>\n  fields.reduce((selects: any, field: string) => {\n    if(field.includes('Count')) {\n      return selectReactionCountByType('images', 'i', field, selects);\n    }\n\n    switch(field) {\n      case 'reactions': {\n        selects.queries.push(`LET reactions = (\n          FOR image, r IN INBOUND i._id reactions\n          COLLECT reactionName = r.value INTO reactionItems\n          RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}\n        )`);\n        selects.objects.push('reactions:reactions');\n        return selects;\n      }\n      case 'tags': {\n        selects.queries.push(`LET tags = (\n          FOR t, pl IN INBOUND i._id isTagged\n          RETURN t\n        )`);\n        selects.objects.push('tags:tags');\n        return selects;\n      }\n      case 'user': {\n        selects.queries.push(`LET user = FIRST(\n          FOR u IN users\n          FILTER i.userId == u._key\n          LIMIT 1\n          RETURN u\n        )`);\n        selects.objects.push('user:user');\n        return selects;\n      }\n      default: {\n        return selects;\n      }\n    }\n  }, {objects: [], queries: []});\n\nexport const getImagesByUser = (\n  context: ApiContext,\n  userId: string,\n  from: number,\n  to: number\n): Promise<ImageType[]> => {\n  const action: string = 'getImagesByUser';\n  const {database} = context;\n  const formatUserId: string = parseId(userId);\n  const limit: ArangoDBLimit = getLimit(from, to);\n  const aqlQry: string = `FOR i IN images\n      FILTER i.userId == \"${formatUserId}\"\n      LET user = FIRST(\n        FOR u IN users\n        FILTER u._key == i.userId\n        LIMIT 1\n        RETURN u\n      )\n      ${limit.aql}\n      SORT i.added\n      RETURN MERGE(i, {user:user})`;\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 getImageCountByItem = (context: ApiContext, itemId: string): Promise<number> => {\n  const action: string = 'getImageCountByItem';\n  const {database} = context;\n  const formatItemId: string = parseArangoId(itemId);\n  const aqlQry: AqlQuery = aql`FOR i IN hasImages\n      FILTER i._to == ${formatItemId}\n      RETURN {count: COUNT(i)}`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then(({count} = {count: 0}) => count)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => 0));\n};\n\nexport const getImagesByItem = async (\n  context: ApiContext,\n  itemId: string,\n  options: ImageOptions = {}\n): Promise<ImageType[]> => {\n  const action: string = 'getImagesByItem';\n  const {database, fields = []} = context;\n  const formatItemId: string = parseArangoId(itemId);\n  const {limit} = parseImageOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getImageOptional(fields);\n  const aqlQry: string = `FOR i, l IN 1..1 OUTBOUND \"${formatItemId}\" hasImages\n    FILTER NOT IS_NULL(i)\n    ${selectQueries.join('\\n')}\n    SORT i.added\n    ${limit.aql}\n    RETURN MERGE(i, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getImagesByGroup = (context: ApiContext, params): Promise<ImageType[]> => {\n  const action: string = 'getImagesByGroup';\n  const {database, session: {userId: sessionId}} = context;\n  const {filters = [], groupId, from, to} = params;\n  const formatGroupId: string = parseId(groupId);\n  const limit = getLimit(from, to);\n\n  filters\n    .map((filter: QueryFilter) => {\n      const {conditional, name, value} = filter;\n      let formatCond: string = conditional;\n\n      if(conditional !== '>=' && conditional !== '<=' && conditional !== '>' && conditional !== '<') {\n        formatCond = '==';\n      }\n\n      switch(name) {\n        case 'added':\n          return `p.added ${formatCond} ${parseNum(value)}`;\n        default:\n          return '';\n      }\n    });\n\n  return getGroupDetails(context, formatGroupId)\n    .then((group: GroupType) => {\n      if(group.privacy === 'public') {\n        filters.push(`p.groupId == \"${groupId}\"`);\n        const filterStr = filters.join(' && ');\n        const aqlQry: string = `FOR i IN\n          FLATTEN(\n            FOR p IN posts\n            FILTER ${filterStr}\n            LET images = (\n              FOR i, e IN INBOUND p._id hasImages\n              RETURN i\n            )\n            SORT p.added DESC\n            RETURN images\n          )\n          SORT i.added DESC\n          ${limit.aql}\n          RETURN i`;\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      return isGrouped(database, sessionId, groupId)\n        .then((grouped: GroupUser) => {\n          if(grouped.isValid) {\n            filters.push(`p.groupId == \"${grouped.groupId}\"`);\n            const filterList: string = filters.join(' && ');\n            const aqlQry: string = `FOR p IN post\n                FILTER ${filterList}\n                FOR f IN p.files\n                FILTER f.type == \"image/jpeg\" || f.type == \"image/png\"\n                ${limit.aql}\n                SORT p.added DESC\n                RETURN f`;\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          return [];\n        });\n    });\n};\n\nexport const getImagesByReactions = (\n  context: ApiContext,\n  reactions: string[] = [],\n  options?: ImageOptions\n): Promise<ImageType[]> => {\n  const action: string = 'getUsersByImage';\n  const {database, fields = [], session: {userId: sessionId}} = context;\n  const {limit} = parseImageOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getImageOptional(fields);\n\n  const formatSessionId: string = `users/${sessionId}`;\n  const formatReactions: string[] = reactions.map((reactionName) => parseChar(reactionName, 32));\n  const filterBy: string[] = [\n    'r.type == \"images\"',\n    `POSITION(${JSON.stringify(formatReactions)}, LOWER(r.name))`];\n\n  // Get data from database\n  const aqlQry: string = `FOR i, r IN OUTBOUND \"${formatSessionId}\" hasReactions\n    OPTIONS {vertexCollections: \"images\"}\n    ${selectQueries.join('\\n')}\n    FILTER ${filterBy.join(' && ')}\n    ${limit.aql}\n    RETURN MERGE(i, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => []));\n};\n\nexport const getImage = (context: ApiContext, id: string): Promise<ImageType> => {\n  const action: string = 'getItem';\n  const {database} = context;\n  const formatId: string = parseId(id);\n  const aqlQry: AqlQuery = aql`FOR i IN images\n    FILTER i._key==${formatId}\n    LIMIT 1\n    RETURN i`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((image: ImageType = {}) => image)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getPathUserImages = (userId: string, imageId: string, type: string, dir: string = 'images'): string => {\n  let filename: string = imageId;\n\n  switch(type) {\n    case 'image/png':\n      filename = `${imageId}.png`;\n      break;\n    default:\n      filename = `${imageId}.jpg`;\n      break;\n  }\n\n  return `users/${userId}/${dir}/${filename}`;\n};\n\nexport const getAppImageUrl = (data: ImageUrlData): string => {\n  const {imageId, directory = 'images', imageType = 'profile', isThumb, type, typeId} = data;\n  const host: string = Config.get('environment') === 'prod'\n    ? `https://box.${Config.get('app.url')}`\n    : `https://s3.amazonaws.com/dev.${Config.get('app.url')}`;\n  const suffix: string = isThumb ? '-th' : '';\n\n  if(imageId) {\n    switch(type) {\n      case 'apps':\n        // https://box.reaktor.io/myApp/app/images/123.jpg\n        return `${host}/${type}/${directory}/${imageId}${suffix}.jpg`;\n      case 'users':\n        // https://box.reaktor.io/myApp/users/demoUser/images/123.jpg\n        return `${host}/${type}/${typeId}/${directory}/${imageId}${suffix}.jpg`;\n    }\n\n    if(imageType === 'profile') {\n      return `${host}/defaults/${type}_bk${suffix}.jpg`;\n    }\n\n    return `${host}/defaults/${type}_wh${suffix}.jpg`;\n  }\n\n  return '';\n};\n\nexport const getUserImageUrl = (data: ImageUrlData): string => {\n  const {bucket, imageId, isThumb = false, userId} = data;\n  const imgDir: string = isThumb ? 'thumbs' : 'images';\n\n  if(imageId) {\n    const imageKey: string = `users/${userId}/${imgDir}/${imageId}.jpg`;\n    return s3GetSignedUrl({Bucket: bucket, Key: imageKey, Expires: 900});\n  }\n\n  return '';\n};\n\nexport const resizeSaveImage = (\n  context: ApiContext,\n  imageId: string,\n  buffer: Buffer,\n  type: string = 'image/jpeg', s3Options?: PutObjectRequest): Promise<ImageType> => {\n  const action: string = 'resizeSaveImage';\n  const {session: {userId: sessionId}} = context;\n  const imgW: number = Config.get('image.imgSize');\n  const imgQ: number = Config.get('image.imgQuality');\n  let photo: ImageType = {};\n  const format: string = (type.split('/'))[1];\n\n  console.log({buffer, format, imgW, imgQ});\n  return new Promise((resolve) => {\n    gm(buffer, 'img')\n      .setFormat(format)\n      .quality(imgQ)\n      .autoOrient()\n      .resize(imgW, imgW, '>')\n      .identify({bufferStream: true}, (error: Error, val: ImageIdentifyType = {}): any => {\n        if(error) {\n          logError({\n            action,\n            category: eventCategory,\n            label: 'image_save',\n            value: 'gm_image_identify'\n          }, error, context).catch((error) => {\n            throw error;\n          });\n        } else {\n          const formatVals = lowerCaseKeys(val);\n          const {make: cameraMake, model: cameraModel, datetimeoriginal: taken}: ImageIdentifyType = formatVals;\n          photo = {\n            ...cloneDeep(photo),\n            make: cameraMake,\n            model: cameraModel,\n            taken: taken ? DateTime.fromMillis(taken).millisecond : null\n          };\n\n          // If no background color, get the mean color value\n          const stats = formatVals['channel statistics'];\n\n          if(stats) {\n            let {red, green, blue, mean} = stats;\n\n            if(red) {\n              mean = red['standard deviation'] || red.mean;\n              red = mean ? +((mean.split(' ')[0]).substring(0, 3)) : 0;\n            } else {\n              red = 0;\n            }\n\n            if(green) {\n              mean = green['standard deviation'] || green.mean;\n              green = mean ? +((mean.split(' ')[0]).substring(0, 3)) : 0;\n            } else {\n              green = 0;\n            }\n\n            if(blue) {\n              mean = blue['standard deviation'] || blue.mean;\n              blue = mean ? +((mean.split(' ')[0]).substring(0, 3)) : 0;\n            } else {\n              blue = 0;\n            }\n\n            const rgb = blue | (green << 8) | (red << 16);\n            const color = rgb.toString(16);\n            photo.color = color === '0' ? '000000' : color;\n          }\n        }\n      })\n      .stream((error: Error, stdout): any => {\n        if(error) {\n          logError({\n            action,\n            category: eventCategory,\n            isInternal: true,\n            label: 'image_save',\n            value: 'gm_image_stream'\n          }, error, context).then(() => null);\n        } else {\n          let imageBuffer: Buffer = Buffer.from('');\n\n          stdout.on('data', (data) => {\n            imageBuffer = Buffer.concat([imageBuffer, data]);\n          });\n\n          stdout.on('end', () => {\n            // Get file size\n            photo.fileSize = imageBuffer.length;\n\n            // Upload data\n            const imageObj: PutObjectRequest = {\n              ACL: 'authenticated-read',\n              Body: imageBuffer,\n              Bucket: null,\n              ContentType: type,\n              ...s3Options || {},\n              Key: getPathUserImages(sessionId, imageId, type, 'images')\n            };\n\n            s3Put(imageObj)\n              .then(() => {\n                const thmW = Config.get('image.thmSize');\n                const thmQ = Config.get('image.thmQuality');\n\n                // Upload thumbnail\n                gm(imageBuffer, 'img')\n                  .setFormat('jpeg')\n                  .size((thumbError: Error, resizeVal) => {\n                    if(!thumbError) {\n                      // Get updated resize values\n                      const {height, width} = resizeVal;\n                      photo = {...cloneDeep(photo), height, width};\n                    }\n                  })\n                  .gravity('Center')\n                  .resize(thmW, thmW, '^')\n                  .extent(thmW, thmW)\n                  .quality(thmQ)\n                  .stream((streamError: Error, thumbStdout): any => {\n                    if(streamError) {\n                      logError({\n                        action,\n                        category: eventCategory,\n                        label: 'image_save',\n                        value: 'gm_thumbnail_steam'\n                      }, streamError, context)\n                        .catch((error) => {\n                          throw error;\n                        });\n                    } else {\n                      let thumbBuffer: Buffer = Buffer.from('');\n\n                      thumbStdout.on('data', (data) => {\n                        thumbBuffer = Buffer.concat([thumbBuffer, data]);\n                      });\n\n                      thumbStdout.on('end', () => {\n                        // Upload data\n                        const thumbObj: PutObjectRequest = {\n                          ACL: 'authenticated-read',\n                          Body: thumbBuffer,\n                          Bucket: null,\n                          ContentType: type,\n                          ...s3Options || {},\n                          Key: getPathUserImages(sessionId, imageId, type, 'thumbs')\n                        };\n\n                        s3Put(thumbObj)\n                          .then(() => {\n                            resolve(photo);\n                          })\n                          .catch((s3PutError) => logError({\n                            action,\n                            category: eventCategory,\n                            label: 'image_save',\n                            value: 's3_put_image'\n                          }, s3PutError, context)\n                            .catch((error) => {\n                              throw error;\n                            }));\n                      });\n                    }\n                  });\n              })\n              .catch((s3Error) => logError({\n                action,\n                category: eventCategory,\n                isInternal: true,\n                label: 'image_save',\n                value: 's3_image_save'\n              }, s3Error, context).then(() => null));\n          });\n        }\n      });\n  });\n};\n\nexport const addImage = (context: ApiContext, image: ImageType, s3Options?: PutObjectRequest): Promise<ImageType> => {\n  const action: string = 'addImage';\n  const {database, session: {userId: sessionId}} = context;\n  const {imageId, description, buffer, fileType} = image;\n  const now: number = Date.now();\n\n  return resizeSaveImage(context, imageId, buffer, fileType, s3Options)\n    .then((resizedImage: any) => {\n      const insert: ImageType = {\n        ...resizedImage,\n        _key: imageId,\n        added: now,\n        description,\n        fileType,\n        modified: now,\n        userId: sessionId\n      };\n\n      const aqlQry: AqlQuery = aql`INSERT ${insert} IN images RETURN NEW`;\n\n      return database.query(aqlQry)\n        .then((cursor: ArrayCursor) => cursor.next())\n        .then(defaultObject)\n        .catch((error: Error) => logError({\n          action,\n          category: eventCategory,\n          isInternal: true,\n          label: 'db_error'\n        }, error, context).then(() => null));\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      isInternal: true,\n      label: 'image_resize'\n    }, error, context).then(() => null));\n};\n\nexport const addImageEdge = async (context: ApiContext, imageEdge: ImageEdgeType): Promise<object> => {\n  const action: string = 'addImageEdge';\n  const {database, session: {userId: sessionId}} = context;\n  const {imageId, itemId, itemType} = imageEdge;\n  const now: number = Date.now();\n  const edgeCollection: EdgeCollection = database.collection('hasImages');\n  const edgeId: string = createHash(`hasImages-${imageId}-${itemId}-${sessionId}`);\n  const formatItemType: string = parseChar(itemType).toLowerCase();\n  const formatItemId: string = parseId(itemId);\n  let itemDocId: string;\n  const formatImgId: string = parseId(imageId);\n  const fileDocId: string = `images/${formatImgId}`;\n\n  switch(formatItemType) {\n    case 'groups':\n      itemDocId = `groups/${formatItemId}`;\n      break;\n    case 'posts':\n      itemDocId = `posts/${formatItemId}`;\n      break;\n    case 'users':\n    case 'profile':\n      itemDocId = `users/${formatItemId}`;\n      break;\n    default:\n      itemDocId = '';\n      break;\n  }\n  const edge: any = {\n    _from: itemDocId,\n    _key: edgeId,\n    _to: fileDocId,\n    added: now,\n    type: itemType\n  };\n\n  if(!isEmpty(itemDocId)) {\n    return edgeCollection.save(edge, {returnNew: true})\n      .then((fileEdge) => edgeCollection.document(fileEdge))\n      .catch((error: Error) => logError({\n        action,\n        category: eventCategory,\n        isInternal: true,\n        label: 'db_error',\n        params: {\n          edge,\n          fileDocId,\n          imageId,\n          itemDocId,\n          itemId,\n          itemType\n        }\n      }, error, context).then(() => null));\n  }\n\n  return {};\n};\n\nexport const updateImage = async (\n  context: ApiContext,\n  image: ImageType,\n  s3Options?: PutObjectRequest\n): Promise<ImageType> => {\n  const action: string = 'updateImage';\n  const {database, session: {userId: sessionId}} = context;\n\n  // Item props\n  const {\n    base64 = '',\n    buffer: imageBuffer,\n    description = '',\n    imageId,\n    itemId,\n    itemType,\n    fileType,\n    url = ''\n  } = image;\n\n  // Save Base64 data\n  const now: number = Date.now();\n  const formatImageId: string = imageId || createHash(`image-${sessionId}`);\n  const formatItemId: string = parseId(itemId);\n  const formatItemType: string = parseChar(itemType, 16).toLowerCase();\n  const customParams: ImageType = {\n    description,\n    imageId: formatImageId,\n    userId: sessionId\n  };\n\n  if(!isEmpty(base64) || imageBuffer) {\n    let buffer: Buffer = imageBuffer;\n\n    // Parse Base64 data\n    if(!isEmpty(base64)) {\n      const formatBase64: string = base64.substr(base64.indexOf(',') + 1);\n      buffer = Buffer.from(formatBase64, 'base64');\n    }\n\n    let updatedFileType = fileType;\n\n    if(!fileType) {\n      const fileMime: FileTypeResult = await FileType.fromBuffer(buffer);\n      const {mime} = fileMime;\n      updatedFileType = mime;\n    }\n\n    const imgParams: ImageType = {\n      ...customParams,\n      buffer,\n      fileType: updatedFileType\n    };\n\n    return addImage(context, imgParams, s3Options)\n      .then((image: ImageType) => {\n        if(formatItemId && formatItemType) {\n          const imageEdge: ImageEdgeType = {\n            imageId: formatImageId,\n            itemId: formatItemId,\n            itemType: formatItemType\n          };\n          return addImageEdge(context, imageEdge).then(() => image);\n        }\n\n        return image;\n      })\n      .catch((error) => logError({\n        action,\n        category: eventCategory,\n        label: 'image_save_error'\n      }, error, context).then(() => null));\n  } else if(url !== '') {\n    // Download image from the web\n    let contentType: string;\n\n    return httpGet(url)\n      .then((res: Response) => {\n        if(res.status !== 200) {\n          return logException({\n            action,\n            category: eventCategory,\n            label: 'fetch_image_url',\n            value: res.statusText\n          }, context).then(() => null);\n        }\n\n        contentType = res.headers.get('content-type');\n\n        return res;\n      })\n      .then((res) => res.buffer())\n      .then((buffer: Buffer) => {\n        const imgParams: ImageType = {\n          ...customParams,\n          buffer,\n          fileType: contentType\n        };\n\n        return addImage(context, imgParams, s3Options).then((image: ImageType) => {\n          if(formatItemId && formatItemType) {\n            const imageEdge: ImageEdgeType = {imageId: formatImageId, itemId: formatItemId, itemType: formatItemType};\n            return addImageEdge(context, imageEdge).then(() => image);\n          }\n\n          return image;\n        });\n      })\n      .catch((error: Error) => logError({\n        action,\n        category: eventCategory,\n        label: 'fetch_error'\n      }, error, context).then(() => null));\n  } else if(imageId !== '') {\n    // Update metadata\n    const update: any = {\n      description,\n      modified: now\n    };\n    const aqlQry: AqlQuery = aql`UPDATE {_key: ${formatImageId}} WITH ${update} IN images 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: 'db_error',\n        params: {\n          aqlQry,\n          formatImageId,\n          update\n        }\n      }, error, context).then(() => null));\n  }\n\n  return null;\n};\n\nexport const deleteImage = async (context: ApiContext, imageId: string): Promise<ImageType> => {\n  const action: string = 'delete';\n  const {database, session: {userId: sessionId}} = context;\n  const formatImageId = parseId(imageId);\n\n  const removeEdgeQuery = aql`FOR hi IN hasImages\n    FILTER hi._from == ${formatImageId}\n    REMOVE hi IN hasImages\n    RETURN OLD`;\n\n  await database.query(removeEdgeQuery);\n\n  const aqlQuery = aql`FOR i IN images\n    FILTER i._key == ${formatImageId} && i.userId == ${sessionId}\n    REMOVE i IN images\n    RETURN OLD`;\n\n  const image = await database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n\n  if(!isEmpty(image)) {\n    const {_key: imageKey} = image;\n    const params: DeleteObjectsRequest = {\n      Bucket: null,\n      Delete: {\n        Objects: [\n          {Key: `users/${sessionId}/images/${imageKey}.jpg`},\n          {Key: `users/${sessionId}/thumbs/${imageKey}.jpg`}\n        ],\n        Quiet: true\n      }\n    };\n\n    return s3DeleteList(params).then(() => image);\n  }\n\n  return {};\n};\n"],
  "mappings": "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",
  "names": []
}
