@nlabs/reaktor 0.5.4 → 0.8.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +7 -7
- package/lib/actions/conversations.js +68 -69
- package/lib/actions/dynamodb.js +22 -33
- package/lib/actions/email.js +32 -30
- package/lib/actions/files.js +39 -58
- package/lib/actions/groups.js +33 -28
- package/lib/actions/images.js +71 -82
- package/lib/actions/index.js +54 -37
- package/lib/actions/ios.js +21 -40
- package/lib/actions/locations.js +25 -22
- package/lib/actions/messages.js +31 -45
- package/lib/actions/notifications.js +13 -20
- package/lib/actions/payments.js +40 -52
- package/lib/actions/posts.js +68 -48
- package/lib/actions/reactions.js +29 -52
- package/lib/actions/s3.js +21 -18
- package/lib/actions/search.js +26 -40
- package/lib/actions/sms.js +24 -21
- package/lib/actions/statistics.js +14 -21
- package/lib/actions/subscription.js +27 -24
- package/lib/actions/tags.js +49 -56
- package/lib/actions/users.js +92 -106
- package/lib/actions/websockets.js +28 -41
- package/lib/config.js +21 -18
- package/lib/index.js +24 -22
- package/lib/lambdas/actions/websockets.js +21 -45
- package/lib/lambdas/authorizer.js +16 -24
- package/lib/lambdas/connection.js +18 -43
- package/lib/lambdas/utils/message.js +20 -17
- package/lib/lambdas/utils/websocket.js +33 -25
- package/lib/templates/email/layout.js +16 -9
- package/lib/templates/email/passwordForgot.js +16 -9
- package/lib/templates/email/passwordRecovery.js +16 -9
- package/lib/templates/email/verifyEmail.js +16 -9
- package/lib/templates/email/welcome.js +16 -9
- package/lib/templates/sms/passwordForgot.js +16 -9
- package/lib/templates/sms/passwordRecovery.js +16 -9
- package/lib/templates/sms/verifyEmail.js +16 -9
- package/lib/templates/sms/verifyPhone.js +16 -9
- package/lib/templates/sms/welcome.js +16 -9
- package/lib/types/apps.js +14 -5
- package/lib/types/arangodb.js +14 -5
- package/lib/types/auth.js +14 -5
- package/lib/types/connections.js +14 -5
- package/lib/types/conversations.js +14 -5
- package/lib/types/email.js +14 -5
- package/lib/types/files.js +14 -5
- package/lib/types/google.js +14 -5
- package/lib/types/groups.js +14 -5
- package/lib/types/images.js +14 -5
- package/lib/types/index.js +50 -35
- package/lib/types/locations.js +14 -5
- package/lib/types/messages.js +14 -5
- package/lib/types/notifications.js +14 -5
- package/lib/types/payments.js +14 -5
- package/lib/types/posts.js +14 -5
- package/lib/types/statistics.js +14 -5
- package/lib/types/tags.js +14 -5
- package/lib/types/users.js +14 -5
- package/lib/types/websocket.js +14 -5
- package/lib/utils/analytics.js +24 -44
- package/lib/utils/arangodb.js +15 -22
- package/lib/utils/auth.js +22 -36
- package/lib/utils/graphql.js +20 -17
- package/lib/utils/index.js +26 -23
- package/lib/utils/objects.js +22 -19
- package/lib/utils/session.js +20 -17
- package/package.json +31 -33
- package/.eslintrc +0 -18
- package/lib/actions/conversations.d.ts +0 -14
- package/lib/actions/dynamodb.d.ts +0 -8
- package/lib/actions/email.d.ts +0 -7
- package/lib/actions/files.d.ts +0 -16
- package/lib/actions/groups.d.ts +0 -14
- package/lib/actions/images.d.ts +0 -22
- package/lib/actions/index.d.ts +0 -20
- package/lib/actions/ios.d.ts +0 -7
- package/lib/actions/locations.d.ts +0 -3
- package/lib/actions/messages.d.ts +0 -13
- package/lib/actions/notifications.d.ts +0 -5
- package/lib/actions/payments.d.ts +0 -10
- package/lib/actions/posts.d.ts +0 -19
- package/lib/actions/reactions.d.ts +0 -30
- package/lib/actions/s3.d.ts +0 -7
- package/lib/actions/search.d.ts +0 -3
- package/lib/actions/sms.d.ts +0 -3
- package/lib/actions/statistics.d.ts +0 -2
- package/lib/actions/subscription.d.ts +0 -7
- package/lib/actions/tags.d.ts +0 -26
- package/lib/actions/users.d.ts +0 -44
- package/lib/actions/websockets.d.ts +0 -6
- package/lib/config.d.ts +0 -20
- package/lib/index.d.ts +0 -5
- package/lib/lambdas/actions/websockets.d.ts +0 -6
- package/lib/lambdas/authorizer.d.ts +0 -20
- package/lib/lambdas/connection.d.ts +0 -12
- package/lib/lambdas/utils/message.d.ts +0 -1
- package/lib/lambdas/utils/websocket.d.ts +0 -7
- package/lib/templates/email/layout.d.ts +0 -2
- package/lib/templates/email/passwordForgot.d.ts +0 -2
- package/lib/templates/email/passwordRecovery.d.ts +0 -2
- package/lib/templates/email/verifyEmail.d.ts +0 -2
- package/lib/templates/email/welcome.d.ts +0 -2
- package/lib/templates/sms/passwordForgot.d.ts +0 -2
- package/lib/templates/sms/passwordRecovery.d.ts +0 -2
- package/lib/templates/sms/verifyEmail.d.ts +0 -2
- package/lib/templates/sms/verifyPhone.d.ts +0 -2
- package/lib/templates/sms/welcome.d.ts +0 -2
- package/lib/types/apps.d.ts +0 -44
- package/lib/types/arangodb.d.ts +0 -17
- package/lib/types/auth.d.ts +0 -7
- package/lib/types/connections.d.ts +0 -8
- package/lib/types/conversations.d.ts +0 -11
- package/lib/types/email.d.ts +0 -12
- package/lib/types/files.d.ts +0 -26
- package/lib/types/google.d.ts +0 -27
- package/lib/types/groups.d.ts +0 -22
- package/lib/types/images.d.ts +0 -42
- package/lib/types/index.d.ts +0 -18
- package/lib/types/locations.d.ts +0 -20
- package/lib/types/messages.d.ts +0 -27
- package/lib/types/notifications.d.ts +0 -19
- package/lib/types/payments.d.ts +0 -115
- package/lib/types/posts.d.ts +0 -47
- package/lib/types/statistics.d.ts +0 -3
- package/lib/types/tags.d.ts +0 -15
- package/lib/types/users.d.ts +0 -80
- package/lib/types/websocket.d.ts +0 -14
- package/lib/utils/analytics.d.ts +0 -14
- package/lib/utils/arangodb.d.ts +0 -9
- package/lib/utils/auth.d.ts +0 -5
- package/lib/utils/graphql.d.ts +0 -1
- package/lib/utils/index.d.ts +0 -6
- package/lib/utils/objects.d.ts +0 -3
- package/lib/utils/session.d.ts +0 -18
package/lib/actions/images.js
CHANGED
|
@@ -1,47 +1,32 @@
|
|
|
1
1
|
var __create = Object.create;
|
|
2
2
|
var __defProp = Object.defineProperty;
|
|
3
|
-
var __defProps = Object.defineProperties;
|
|
4
3
|
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
5
|
-
var __getOwnPropDescs = Object.getOwnPropertyDescriptors;
|
|
6
4
|
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
7
|
-
var __getOwnPropSymbols = Object.getOwnPropertySymbols;
|
|
8
5
|
var __getProtoOf = Object.getPrototypeOf;
|
|
9
6
|
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
10
|
-
var __propIsEnum = Object.prototype.propertyIsEnumerable;
|
|
11
|
-
var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
|
|
12
|
-
var __spreadValues = (a, b) => {
|
|
13
|
-
for (var prop in b || (b = {}))
|
|
14
|
-
if (__hasOwnProp.call(b, prop))
|
|
15
|
-
__defNormalProp(a, prop, b[prop]);
|
|
16
|
-
if (__getOwnPropSymbols)
|
|
17
|
-
for (var prop of __getOwnPropSymbols(b)) {
|
|
18
|
-
if (__propIsEnum.call(b, prop))
|
|
19
|
-
__defNormalProp(a, prop, b[prop]);
|
|
20
|
-
}
|
|
21
|
-
return a;
|
|
22
|
-
};
|
|
23
|
-
var __spreadProps = (a, b) => __defProps(a, __getOwnPropDescs(b));
|
|
24
|
-
var __markAsModule = (target) => __defProp(target, "__esModule", { value: true });
|
|
25
|
-
var __require = typeof require !== "undefined" ? require : (x) => {
|
|
26
|
-
throw new Error('Dynamic require of "' + x + '" is not supported');
|
|
27
|
-
};
|
|
28
7
|
var __export = (target, all) => {
|
|
29
|
-
__markAsModule(target);
|
|
30
8
|
for (var name in all)
|
|
31
9
|
__defProp(target, name, { get: all[name], enumerable: true });
|
|
32
10
|
};
|
|
33
|
-
var
|
|
34
|
-
if (
|
|
35
|
-
for (let key of __getOwnPropNames(
|
|
36
|
-
if (!__hasOwnProp.call(
|
|
37
|
-
__defProp(
|
|
11
|
+
var __copyProps = (to, from, except, desc) => {
|
|
12
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
|
13
|
+
for (let key of __getOwnPropNames(from))
|
|
14
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
15
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
38
16
|
}
|
|
39
|
-
return
|
|
17
|
+
return to;
|
|
40
18
|
};
|
|
41
|
-
var
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
19
|
+
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
|
|
20
|
+
// If the importer is in node compatibility mode or this is not an ESM
|
|
21
|
+
// file that has been converted to a CommonJS file using a Babel-
|
|
22
|
+
// compatible transform (i.e. "__esModule" has not been set), then set
|
|
23
|
+
// "default" to the CommonJS "module.exports" for node compatibility.
|
|
24
|
+
isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
|
|
25
|
+
mod
|
|
26
|
+
));
|
|
27
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
28
|
+
var images_exports = {};
|
|
29
|
+
__export(images_exports, {
|
|
45
30
|
addImage: () => addImage,
|
|
46
31
|
addImageEdge: () => addImageEdge,
|
|
47
32
|
deleteImage: () => deleteImage,
|
|
@@ -49,28 +34,28 @@ __export(exports, {
|
|
|
49
34
|
getImage: () => getImage,
|
|
50
35
|
getImageCountByItem: () => getImageCountByItem,
|
|
51
36
|
getImageOptional: () => getImageOptional,
|
|
37
|
+
getImageUrl: () => getImageUrl,
|
|
52
38
|
getImagesByGroup: () => getImagesByGroup,
|
|
53
39
|
getImagesByItem: () => getImagesByItem,
|
|
54
40
|
getImagesByReactions: () => getImagesByReactions,
|
|
55
41
|
getImagesByUser: () => getImagesByUser,
|
|
56
42
|
getPathUserImages: () => getPathUserImages,
|
|
57
|
-
getUserImageUrl: () => getUserImageUrl,
|
|
58
43
|
parseImageOptions: () => parseImageOptions,
|
|
59
44
|
resizeSaveImage: () => resizeSaveImage,
|
|
60
45
|
updateImage: () => updateImage
|
|
61
46
|
});
|
|
62
|
-
|
|
63
|
-
var
|
|
64
|
-
var
|
|
65
|
-
var
|
|
66
|
-
var
|
|
67
|
-
var
|
|
68
|
-
var
|
|
69
|
-
var import_luxon =
|
|
70
|
-
var import_config =
|
|
71
|
-
var import_utils2 =
|
|
72
|
-
var import_groups =
|
|
73
|
-
var import_s32 =
|
|
47
|
+
module.exports = __toCommonJS(images_exports);
|
|
48
|
+
var import_rip_hunter = require("@nlabs/rip-hunter");
|
|
49
|
+
var import_utils = require("@nlabs/utils");
|
|
50
|
+
var import_arangojs = require("arangojs");
|
|
51
|
+
var import_file_type = __toESM(require("file-type"));
|
|
52
|
+
var import_gm = __toESM(require("gm"));
|
|
53
|
+
var import_cloneDeep = __toESM(require("lodash/cloneDeep"));
|
|
54
|
+
var import_luxon = require("luxon");
|
|
55
|
+
var import_config = require("../config");
|
|
56
|
+
var import_utils2 = require("../utils");
|
|
57
|
+
var import_groups = require("./groups");
|
|
58
|
+
var import_s32 = require("./s3");
|
|
74
59
|
const eventCategory = "images";
|
|
75
60
|
const parseImageOptions = (options = {}) => {
|
|
76
61
|
const {
|
|
@@ -146,7 +131,7 @@ const getImageCountByItem = (context, itemId) => {
|
|
|
146
131
|
const action = "getImageCountByItem";
|
|
147
132
|
const { database } = context;
|
|
148
133
|
const formatItemId = (0, import_utils.parseArangoId)(itemId);
|
|
149
|
-
const aqlQry = import_arangojs.aql`FOR i IN
|
|
134
|
+
const aqlQry = import_arangojs.aql`FOR i IN hasImage
|
|
150
135
|
FILTER i._to == ${formatItemId}
|
|
151
136
|
RETURN {count: COUNT(i)}`;
|
|
152
137
|
return database.query(aqlQry).then((cursor) => cursor.next()).then(({ count } = { count: 0 }) => count).catch((error) => (0, import_utils2.logError)({
|
|
@@ -161,7 +146,7 @@ const getImagesByItem = async (context, itemId, options = {}) => {
|
|
|
161
146
|
const formatItemId = (0, import_utils.parseArangoId)(itemId);
|
|
162
147
|
const { limit } = parseImageOptions(options);
|
|
163
148
|
const { objects: selectObjects, queries: selectQueries } = getImageOptional(fields);
|
|
164
|
-
const aqlQry = `FOR i, l IN 1..1 OUTBOUND "${formatItemId}"
|
|
149
|
+
const aqlQry = `FOR i, l IN 1..1 OUTBOUND "${formatItemId}" hasImage
|
|
165
150
|
FILTER NOT IS_NULL(i)
|
|
166
151
|
${selectQueries.join("\n")}
|
|
167
152
|
SORT i.added
|
|
@@ -201,7 +186,7 @@ const getImagesByGroup = (context, params) => {
|
|
|
201
186
|
FOR p IN posts
|
|
202
187
|
FILTER ${filterStr}
|
|
203
188
|
LET images = (
|
|
204
|
-
FOR i, e IN INBOUND p._id
|
|
189
|
+
FOR i, e IN INBOUND p._id hasImage
|
|
205
190
|
RETURN i
|
|
206
191
|
)
|
|
207
192
|
SORT p.added DESC
|
|
@@ -248,7 +233,7 @@ const getImagesByReactions = (context, reactions = [], options) => {
|
|
|
248
233
|
'r.type == "images"',
|
|
249
234
|
`POSITION(${JSON.stringify(formatReactions)}, LOWER(r.name))`
|
|
250
235
|
];
|
|
251
|
-
const aqlQry = `FOR i, r IN OUTBOUND "${formatSessionId}"
|
|
236
|
+
const aqlQry = `FOR i, r IN OUTBOUND "${formatSessionId}" hasReaction
|
|
252
237
|
OPTIONS {vertexCollections: "images"}
|
|
253
238
|
${selectQueries.join("\n")}
|
|
254
239
|
FILTER ${filterBy.join(" && ")}
|
|
@@ -304,11 +289,11 @@ const getAppImageUrl = (data) => {
|
|
|
304
289
|
}
|
|
305
290
|
return "";
|
|
306
291
|
};
|
|
307
|
-
const
|
|
308
|
-
const { bucket, imageId, isThumb = false,
|
|
292
|
+
const getImageUrl = (data) => {
|
|
293
|
+
const { bucket, imageId, isThumb = false, type, typeId } = data;
|
|
309
294
|
const imgDir = isThumb ? "thumbs" : "images";
|
|
310
295
|
if (imageId) {
|
|
311
|
-
const imageKey =
|
|
296
|
+
const imageKey = `${type}/${typeId}/${imgDir}/${imageId}.jpg`;
|
|
312
297
|
return (0, import_s32.s3GetSignedUrl)({ Bucket: bucket, Key: imageKey, Expires: 900 });
|
|
313
298
|
}
|
|
314
299
|
return "";
|
|
@@ -335,11 +320,12 @@ const resizeSaveImage = (context, imageId, buffer, type = "image/jpeg", s3Option
|
|
|
335
320
|
} else {
|
|
336
321
|
const formatVals = (0, import_utils2.lowerCaseKeys)(val);
|
|
337
322
|
const { make: cameraMake, model: cameraModel, datetimeoriginal: taken } = formatVals;
|
|
338
|
-
photo =
|
|
323
|
+
photo = {
|
|
324
|
+
...(0, import_cloneDeep.default)(photo),
|
|
339
325
|
make: cameraMake,
|
|
340
326
|
model: cameraModel,
|
|
341
327
|
taken: taken ? import_luxon.DateTime.fromMillis(taken).millisecond : null
|
|
342
|
-
}
|
|
328
|
+
};
|
|
343
329
|
const stats = formatVals["channel statistics"];
|
|
344
330
|
if (stats) {
|
|
345
331
|
let { red, green, blue, mean } = stats;
|
|
@@ -382,21 +368,21 @@ const resizeSaveImage = (context, imageId, buffer, type = "image/jpeg", s3Option
|
|
|
382
368
|
});
|
|
383
369
|
stdout.on("end", () => {
|
|
384
370
|
photo.fileSize = imageBuffer.length;
|
|
385
|
-
const imageObj =
|
|
371
|
+
const imageObj = {
|
|
386
372
|
ACL: "authenticated-read",
|
|
387
373
|
Body: imageBuffer,
|
|
388
374
|
Bucket: null,
|
|
389
|
-
ContentType: type
|
|
390
|
-
|
|
375
|
+
ContentType: type,
|
|
376
|
+
...s3Options || {},
|
|
391
377
|
Key: getPathUserImages(sessionId, imageId, type, "images")
|
|
392
|
-
}
|
|
378
|
+
};
|
|
393
379
|
(0, import_s32.s3Put)(imageObj).then(() => {
|
|
394
380
|
const thmW = import_config.Config.get("image.thmSize");
|
|
395
381
|
const thmQ = import_config.Config.get("image.thmQuality");
|
|
396
382
|
(0, import_gm.default)(imageBuffer, "img").setFormat("jpeg").size((thumbError, resizeVal) => {
|
|
397
383
|
if (!thumbError) {
|
|
398
384
|
const { height, width } = resizeVal;
|
|
399
|
-
photo =
|
|
385
|
+
photo = { ...(0, import_cloneDeep.default)(photo), height, width };
|
|
400
386
|
}
|
|
401
387
|
}).gravity("Center").resize(thmW, thmW, "^").extent(thmW, thmW).quality(thmQ).stream((streamError, thumbStdout) => {
|
|
402
388
|
if (streamError) {
|
|
@@ -414,14 +400,14 @@ const resizeSaveImage = (context, imageId, buffer, type = "image/jpeg", s3Option
|
|
|
414
400
|
thumbBuffer = Buffer.concat([thumbBuffer, data]);
|
|
415
401
|
});
|
|
416
402
|
thumbStdout.on("end", () => {
|
|
417
|
-
const thumbObj =
|
|
403
|
+
const thumbObj = {
|
|
418
404
|
ACL: "authenticated-read",
|
|
419
405
|
Body: thumbBuffer,
|
|
420
406
|
Bucket: null,
|
|
421
|
-
ContentType: type
|
|
422
|
-
|
|
407
|
+
ContentType: type,
|
|
408
|
+
...s3Options || {},
|
|
423
409
|
Key: getPathUserImages(sessionId, imageId, type, "thumbs")
|
|
424
|
-
}
|
|
410
|
+
};
|
|
425
411
|
(0, import_s32.s3Put)(thumbObj).then(() => {
|
|
426
412
|
resolve(photo);
|
|
427
413
|
}).catch((s3PutError) => (0, import_utils2.logError)({
|
|
@@ -453,14 +439,15 @@ const addImage = (context, image, s3Options) => {
|
|
|
453
439
|
const { imageId, description, buffer, fileType } = image;
|
|
454
440
|
const now = Date.now();
|
|
455
441
|
return resizeSaveImage(context, imageId, buffer, fileType, s3Options).then((resizedImage) => {
|
|
456
|
-
const insert =
|
|
442
|
+
const insert = {
|
|
443
|
+
...resizedImage,
|
|
457
444
|
_key: imageId,
|
|
458
445
|
added: now,
|
|
459
446
|
description,
|
|
460
447
|
fileType,
|
|
461
448
|
modified: now,
|
|
462
449
|
userId: sessionId
|
|
463
|
-
}
|
|
450
|
+
};
|
|
464
451
|
const aqlQry = import_arangojs.aql`INSERT ${insert} IN images RETURN NEW`;
|
|
465
452
|
return database.query(aqlQry).then((cursor) => cursor.next()).then(import_utils2.defaultObject).catch((error) => (0, import_utils2.logError)({
|
|
466
453
|
action,
|
|
@@ -480,13 +467,13 @@ const addImageEdge = async (context, imageEdge) => {
|
|
|
480
467
|
const { database, session: { userId: sessionId } } = context;
|
|
481
468
|
const { imageId, itemId, itemType } = imageEdge;
|
|
482
469
|
const now = Date.now();
|
|
483
|
-
const edgeCollection = database.collection("
|
|
484
|
-
const edgeId = (0, import_utils.createHash)(`
|
|
470
|
+
const edgeCollection = database.collection("hasImage");
|
|
471
|
+
const edgeId = (0, import_utils.createHash)(`hasImage-${imageId}-${itemId}-${sessionId}`);
|
|
485
472
|
const formatItemType = (0, import_utils.parseChar)(itemType).toLowerCase();
|
|
486
473
|
const formatItemId = (0, import_utils.parseId)(itemId);
|
|
487
474
|
let itemDocId;
|
|
488
|
-
const
|
|
489
|
-
const fileDocId = `images/${
|
|
475
|
+
const formatImageId = (0, import_utils.parseId)(imageId);
|
|
476
|
+
const fileDocId = `images/${formatImageId}`;
|
|
490
477
|
switch (formatItemType) {
|
|
491
478
|
case "groups":
|
|
492
479
|
itemDocId = `groups/${formatItemId}`;
|
|
@@ -509,7 +496,7 @@ const addImageEdge = async (context, imageEdge) => {
|
|
|
509
496
|
added: now,
|
|
510
497
|
type: itemType
|
|
511
498
|
};
|
|
512
|
-
if (
|
|
499
|
+
if (itemDocId) {
|
|
513
500
|
return edgeCollection.save(edge, { returnNew: true }).then((fileEdge) => edgeCollection.document(fileEdge)).catch((error) => (0, import_utils2.logError)({
|
|
514
501
|
action,
|
|
515
502
|
category: eventCategory,
|
|
@@ -549,22 +536,23 @@ const updateImage = async (context, image, s3Options) => {
|
|
|
549
536
|
imageId: formatImageId,
|
|
550
537
|
userId: sessionId
|
|
551
538
|
};
|
|
552
|
-
if (
|
|
539
|
+
if (base64 || imageBuffer) {
|
|
553
540
|
let buffer = imageBuffer;
|
|
554
|
-
if (
|
|
541
|
+
if (base64) {
|
|
555
542
|
const formatBase64 = base64.substr(base64.indexOf(",") + 1);
|
|
556
543
|
buffer = Buffer.from(formatBase64, "base64");
|
|
557
544
|
}
|
|
558
545
|
let updatedFileType = fileType;
|
|
559
546
|
if (!fileType) {
|
|
560
|
-
const fileMime = await import_file_type.default.
|
|
547
|
+
const fileMime = await import_file_type.default.fileTypeFromBuffer(buffer);
|
|
561
548
|
const { mime } = fileMime;
|
|
562
549
|
updatedFileType = mime;
|
|
563
550
|
}
|
|
564
|
-
const imgParams =
|
|
551
|
+
const imgParams = {
|
|
552
|
+
...customParams,
|
|
565
553
|
buffer,
|
|
566
554
|
fileType: updatedFileType
|
|
567
|
-
}
|
|
555
|
+
};
|
|
568
556
|
return addImage(context, imgParams, s3Options).then((image2) => {
|
|
569
557
|
if (formatItemId && formatItemType) {
|
|
570
558
|
const imageEdge = {
|
|
@@ -594,10 +582,11 @@ const updateImage = async (context, image, s3Options) => {
|
|
|
594
582
|
contentType = res.headers.get("content-type");
|
|
595
583
|
return res;
|
|
596
584
|
}).then((res) => res.buffer()).then((buffer) => {
|
|
597
|
-
const imgParams =
|
|
585
|
+
const imgParams = {
|
|
586
|
+
...customParams,
|
|
598
587
|
buffer,
|
|
599
588
|
fileType: contentType
|
|
600
|
-
}
|
|
589
|
+
};
|
|
601
590
|
return addImage(context, imgParams, s3Options).then((image2) => {
|
|
602
591
|
if (formatItemId && formatItemType) {
|
|
603
592
|
const imageEdge = { imageId: formatImageId, itemId: formatItemId, itemType: formatItemType };
|
|
@@ -633,9 +622,9 @@ const deleteImage = async (context, imageId) => {
|
|
|
633
622
|
const action = "delete";
|
|
634
623
|
const { database, session: { userId: sessionId } } = context;
|
|
635
624
|
const formatImageId = (0, import_utils.parseId)(imageId);
|
|
636
|
-
const removeEdgeQuery = import_arangojs.aql`FOR hi IN
|
|
625
|
+
const removeEdgeQuery = import_arangojs.aql`FOR hi IN hasImage
|
|
637
626
|
FILTER hi._from == ${formatImageId}
|
|
638
|
-
REMOVE hi IN
|
|
627
|
+
REMOVE hi IN hasImage
|
|
639
628
|
RETURN OLD`;
|
|
640
629
|
await database.query(removeEdgeQuery);
|
|
641
630
|
const aqlQuery = import_arangojs.aql`FOR i IN images
|
|
@@ -647,7 +636,7 @@ const deleteImage = async (context, imageId) => {
|
|
|
647
636
|
category: eventCategory,
|
|
648
637
|
label: "db_error"
|
|
649
638
|
}, error, context).then(() => null));
|
|
650
|
-
if (
|
|
639
|
+
if (image) {
|
|
651
640
|
const { _key: imageKey } = image;
|
|
652
641
|
const params = {
|
|
653
642
|
Bucket: null,
|
|
@@ -672,14 +661,14 @@ const deleteImage = async (context, imageId) => {
|
|
|
672
661
|
getImage,
|
|
673
662
|
getImageCountByItem,
|
|
674
663
|
getImageOptional,
|
|
664
|
+
getImageUrl,
|
|
675
665
|
getImagesByGroup,
|
|
676
666
|
getImagesByItem,
|
|
677
667
|
getImagesByReactions,
|
|
678
668
|
getImagesByUser,
|
|
679
669
|
getPathUserImages,
|
|
680
|
-
getUserImageUrl,
|
|
681
670
|
parseImageOptions,
|
|
682
671
|
resizeSaveImage,
|
|
683
672
|
updateImage
|
|
684
673
|
});
|
|
685
|
-
//# 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": []
}

|
|
674
|
+
//# 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 {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 hasImage\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}\" hasImage\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 hasImage\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}\" hasReaction\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 getImageUrl = (data: ImageUrlData): string => {\n  const {bucket, imageId, isThumb = false, type, typeId} = data;\n  const imgDir: string = isThumb ? 'thumbs' : 'images';\n\n  if(imageId) {\n    const imageKey: string = `${type}/${typeId}/${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('hasImage');\n  const edgeId: string = createHash(`hasImage-${imageId}-${itemId}-${sessionId}`);\n  const formatItemType: string = parseChar(itemType).toLowerCase();\n  const formatItemId: string = parseId(itemId);\n  let itemDocId: string;\n  const formatImageId: string = parseId(imageId);\n  const fileDocId: string = `images/${formatImageId}`;\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(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(base64 || imageBuffer) {\n    let buffer: Buffer = imageBuffer;\n\n    // Parse Base64 data\n    if(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.fileTypeFromBuffer(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 hasImage\n    FILTER hi._from == ${formatImageId}\n    REMOVE hi IN hasImage\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(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": ["import_utils", "import_s3", "gm", "error", "cloneDeep", "FileType", "image", "httpGet"]
}

|