@nlabs/reaktor 0.1.11 → 0.1.12

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 (48) hide show
  1. package/package.json +14 -14
  2. package/lib/config.js +0 -139
  3. package/lib/data/conversations.js +0 -228
  4. package/lib/data/dynamodb.js +0 -172
  5. package/lib/data/email.js +0 -194
  6. package/lib/data/files.js +0 -463
  7. package/lib/data/groups.js +0 -401
  8. package/lib/data/images.js +0 -841
  9. package/lib/data/index.js +0 -234
  10. package/lib/data/ios.js +0 -327
  11. package/lib/data/locations.js +0 -148
  12. package/lib/data/messages.js +0 -281
  13. package/lib/data/notifications.js +0 -59
  14. package/lib/data/payments.js +0 -798
  15. package/lib/data/posts.js +0 -637
  16. package/lib/data/reactions.js +0 -243
  17. package/lib/data/s3.js +0 -133
  18. package/lib/data/search.js +0 -111
  19. package/lib/data/sms.js +0 -79
  20. package/lib/data/subscription.js +0 -311
  21. package/lib/data/tags.js +0 -343
  22. package/lib/data/users.js +0 -415
  23. package/lib/index.js +0 -42
  24. package/lib/types/apps.js +0 -2
  25. package/lib/types/arangodb.js +0 -2
  26. package/lib/types/auth.js +0 -2
  27. package/lib/types/conversations.js +0 -2
  28. package/lib/types/email.js +0 -2
  29. package/lib/types/files.js +0 -2
  30. package/lib/types/google.js +0 -2
  31. package/lib/types/groups.js +0 -2
  32. package/lib/types/images.js +0 -2
  33. package/lib/types/index.js +0 -210
  34. package/lib/types/locations.js +0 -2
  35. package/lib/types/messages.js +0 -2
  36. package/lib/types/notifications.js +0 -2
  37. package/lib/types/payments.js +0 -2
  38. package/lib/types/posts.js +0 -2
  39. package/lib/types/reactions.js +0 -2
  40. package/lib/types/tags.js +0 -2
  41. package/lib/types/users.js +0 -2
  42. package/lib/utils/analytics.js +0 -59
  43. package/lib/utils/arangodb.js +0 -131
  44. package/lib/utils/auth.js +0 -104
  45. package/lib/utils/graphql.js +0 -19
  46. package/lib/utils/index.js +0 -78
  47. package/lib/utils/objects.js +0 -54
  48. package/lib/utils/redis.js +0 -28
@@ -1,841 +0,0 @@
1
- "use strict";
2
-
3
- Object.defineProperty(exports, "__esModule", {
4
- value: true
5
- });
6
- exports.deleteImage = exports.updateImage = exports.addImageEdge = exports.addImage = exports.resizeSaveImage = exports.getUrlImages = exports.getPathUserImages = exports.getImage = exports.getImageListByGroup = exports.getImageListByLink = exports.getImageListByUser = exports.getImageOptional = exports.parseImageOptions = void 0;
7
-
8
- var _ripHunter = require("@nlabs/rip-hunter");
9
-
10
- var _utils = require("@nlabs/utils");
11
-
12
- var _arangojs = require("arangojs");
13
-
14
- var _fileType = _interopRequireDefault(require("file-type"));
15
-
16
- var gm = _interopRequireWildcard(require("gm"));
17
-
18
- var _cloneDeep = _interopRequireDefault(require("lodash/cloneDeep"));
19
-
20
- var _isEmpty = _interopRequireDefault(require("lodash/isEmpty"));
21
-
22
- var _luxon = require("luxon");
23
-
24
- var _config = require("../config");
25
-
26
- var _utils2 = require("../utils");
27
-
28
- var _groups = require("./groups");
29
-
30
- var _s = require("./s3");
31
-
32
- var _users = require("./users");
33
-
34
- function _interopRequireWildcard(obj) { if (obj && obj.__esModule) { return obj; } else { var newObj = {}; if (obj != null) { for (var key in obj) { if (Object.prototype.hasOwnProperty.call(obj, key)) { var desc = Object.defineProperty && Object.getOwnPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : {}; if (desc.get || desc.set) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } } newObj.default = obj; return newObj; } }
35
-
36
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
37
-
38
- function _templateObject4() {
39
- var data = _taggedTemplateLiteral(["FOR i IN images\n FILTER i._key == ", " && i.userId == ", "\n REMOVE i IN images\n RETURN OLD"]);
40
-
41
- _templateObject4 = function _templateObject4() {
42
- return data;
43
- };
44
-
45
- return data;
46
- }
47
-
48
- function _templateObject3() {
49
- var data = _taggedTemplateLiteral(["UPDATE ", " WITH ", " IN images RETURN NEW"]);
50
-
51
- _templateObject3 = function _templateObject3() {
52
- return data;
53
- };
54
-
55
- return data;
56
- }
57
-
58
- function _templateObject2() {
59
- var data = _taggedTemplateLiteral(["INSERT ", " IN images RETURN NEW"]);
60
-
61
- _templateObject2 = function _templateObject2() {
62
- return data;
63
- };
64
-
65
- return data;
66
- }
67
-
68
- function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; var ownKeys = Object.keys(source); if (typeof Object.getOwnPropertySymbols === 'function') { ownKeys = ownKeys.concat(Object.getOwnPropertySymbols(source).filter(function (sym) { return Object.getOwnPropertyDescriptor(source, sym).enumerable; })); } ownKeys.forEach(function (key) { _defineProperty(target, key, source[key]); }); } return target; }
69
-
70
- function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
71
-
72
- function _templateObject() {
73
- var data = _taggedTemplateLiteral(["FOR i IN images\n FILTER i._key==", "\n LIMIT 1\n RETURN i"]);
74
-
75
- _templateObject = function _templateObject() {
76
- return data;
77
- };
78
-
79
- return data;
80
- }
81
-
82
- function _taggedTemplateLiteral(strings, raw) { if (!raw) { raw = strings.slice(0); } return Object.freeze(Object.defineProperties(strings, { raw: { value: Object.freeze(raw) } })); }
83
-
84
- /**
85
- * Copyright (c) 2019-Present, Nitrogen Labs, Inc.
86
- * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.
87
- */
88
- var eventCategory = 'images';
89
-
90
- var parseImageOptions = function parseImageOptions() {
91
- var options = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
92
- var _options$from = options.from,
93
- from = _options$from === void 0 ? 0 : _options$from,
94
- _options$to = options.to,
95
- to = _options$to === void 0 ? 30 : _options$to,
96
- _options$type = options.type,
97
- type = _options$type === void 0 ? 'default' : _options$type;
98
- return {
99
- limit: (0, _utils2.getLimit)(from, to),
100
- type: (0, _utils.parseChar)(type, 32)
101
- };
102
- };
103
-
104
- exports.parseImageOptions = parseImageOptions;
105
-
106
- var getImageOptional = function getImageOptional(fields) {
107
- return fields.reduce(function (selects, field) {
108
- switch (field) {
109
- case 'reactions':
110
- {
111
- 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 )");
112
- selects.objects.push('reactions:reactions');
113
- return selects;
114
- }
115
-
116
- case 'tags':
117
- {
118
- selects.queries.push("LET tags = (\n FOR t, pl IN INBOUND i._id isTagged\n RETURN t\n )");
119
- selects.objects.push('tags:tags');
120
- return selects;
121
- }
122
-
123
- case 'user':
124
- {
125
- selects.queries.push("LET user = FIRST(\n FOR u IN users\n FILTER p.userId == u._key\n LIMIT 1\n RETURN u\n )");
126
- selects.objects.push('user:user');
127
- return selects;
128
- }
129
-
130
- default:
131
- {
132
- return selects;
133
- }
134
- }
135
- }, {
136
- objects: [],
137
- queries: []
138
- });
139
- };
140
-
141
- exports.getImageOptional = getImageOptional;
142
-
143
- var getImageListByUser = function getImageListByUser(context, userId, from, to) {
144
- var action = 'getListByUser';
145
- var database = context.database;
146
- var formatUserId = (0, _utils.parseId)(userId);
147
- var limit = (0, _utils2.getLimit)(from, to);
148
- var aqlQry = "FOR i IN images\n FILTER i.userId == \"".concat(formatUserId, "\"\n LET user = FIRST(\n FOR u IN users\n FILTER u._key == i.userId\n LIMIT 1\n RETURN u\n )\n ").concat(limit.aql, "\n SORT i.added\n RETURN MERGE(i, {user:user})");
149
- return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
150
- return cursor.all();
151
- }).catch(function (error) {
152
- return (0, _utils2.logError)({
153
- action: action,
154
- category: eventCategory,
155
- label: 'db_error'
156
- }, error, context).then(function () {
157
- return null;
158
- });
159
- });
160
- };
161
-
162
- exports.getImageListByUser = getImageListByUser;
163
-
164
- var getImageListByLink = function getImageListByLink(context, itemId, itemType, options) {
165
- var action = 'getListByUser';
166
- var database = context.database,
167
- fields = context.fields,
168
- sessionId = context.userId;
169
- var formatItemId = (0, _utils.parseId)(itemId);
170
- var formatItemType = (0, _utils.parseChar)(itemType, 16);
171
- var collectionName;
172
- var verify = Promise.resolve({});
173
- var verifyByType;
174
-
175
- switch (formatItemType) {
176
- case 'group':
177
- collectionName = 'groups';
178
-
179
- case 'message':
180
- collectionName = 'messages';
181
-
182
- case 'event':
183
- case 'post':
184
- collectionName = 'posts';
185
-
186
- default:
187
- collectionName = 'users';
188
- verifyByType = (0, _users.getRelationship)(context, formatItemId);
189
- } // Make sure we verify the relationship before we expose their images
190
-
191
-
192
- if (sessionId !== formatItemId) {
193
- verify = verifyByType;
194
- }
195
-
196
- return verify.then(function (verifyLink) {
197
- if (!verifyLink) {
198
- var verifyError = new Error('Unauthorized.');
199
- return (0, _utils2.logError)({
200
- action: action,
201
- category: eventCategory,
202
- label: 'db_error'
203
- }, verifyError, context).then(function () {
204
- return null;
205
- });
206
- }
207
-
208
- var _parseImageOptions = parseImageOptions(options),
209
- limit = _parseImageOptions.limit;
210
-
211
- var _getImageOptional = getImageOptional(fields),
212
- selectObjects = _getImageOptional.objects,
213
- selectQueries = _getImageOptional.queries;
214
-
215
- var aqlQry = "FOR i, l IN 1..1 INBOUND \"".concat(collectionName, "/").concat(formatItemId, "\" hasImage\n ").concat(selectQueries.join('\n'), "\n FILTER l.type == \"").concat(formatItemType, "\"\n SORT i.added\n ").concat(limit.aql, "\n RETURN MERGE(i, {").concat(selectObjects.join(', '), "})");
216
- return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
217
- return cursor.all();
218
- }).catch(function (error) {
219
- return (0, _utils2.logError)({
220
- action: action,
221
- category: eventCategory,
222
- label: 'db_error'
223
- }, error, context).then(function () {
224
- return null;
225
- });
226
- });
227
- });
228
- };
229
-
230
- exports.getImageListByLink = getImageListByLink;
231
-
232
- var getImageListByGroup = function getImageListByGroup(context, params) {
233
- var action = 'getListByGroup';
234
- var database = context.database,
235
- sessionId = context.userId;
236
- var _params$filters = params.filters,
237
- filters = _params$filters === void 0 ? [] : _params$filters,
238
- groupId = params.groupId,
239
- from = params.from,
240
- to = params.to;
241
- var formatGroupId = (0, _utils.parseId)(groupId);
242
- var limit = (0, _utils2.getLimit)(from, to);
243
- filters.map(function (filter) {
244
- var conditional = filter.conditional,
245
- name = filter.name,
246
- value = filter.value;
247
- var formatCond = conditional;
248
-
249
- if (conditional !== '>=' && conditional !== '<=' && conditional !== '>' && conditional !== '<') {
250
- formatCond = '==';
251
- }
252
-
253
- switch (name) {
254
- case 'added':
255
- return "p.added ".concat(formatCond, " ").concat((0, _utils.parseNum)(value));
256
-
257
- default:
258
- return '';
259
- }
260
- });
261
- return (0, _groups.getGroupDetails)(context, formatGroupId).then(function (group) {
262
- if (group.privacy === 'public') {
263
- filters.push("p.groupId == \"".concat(groupId, "\""));
264
- var filterStr = filters.join(' && ');
265
- var aqlQry = "FOR i IN\n FLATTEN(\n FOR p IN posts\n FILTER ".concat(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 ").concat(limit.aql, "\n RETURN i");
266
- return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
267
- return cursor.all();
268
- }).catch(function (error) {
269
- return (0, _utils2.logError)({
270
- action: action,
271
- category: eventCategory,
272
- label: 'db_error'
273
- }, error, context).then(function () {
274
- return null;
275
- });
276
- });
277
- }
278
-
279
- return (0, _groups.isGrouped)(database, sessionId, groupId).then(function (grouped) {
280
- if (grouped.isValid) {
281
- filters.push("p.groupId == \"".concat(grouped.groupId, "\""));
282
- var filterList = filters.join(' && ');
283
-
284
- var _aqlQry = "FOR p IN post\n FILTER ".concat(filterList, "\n FOR f IN p.files\n FILTER f.type == \"image/jpeg\" || f.type == \"image/png\"\n ").concat(limit.aql, "\n SORT p.added DESC\n RETURN f");
285
-
286
- return (0, _utils2.useDb)(database).query(_aqlQry).then(function (cursor) {
287
- return cursor.all();
288
- }).catch(function (error) {
289
- return (0, _utils2.logError)({
290
- action: action,
291
- category: eventCategory,
292
- label: 'db_error'
293
- }, error, context).then(function () {
294
- return null;
295
- });
296
- });
297
- }
298
-
299
- return [];
300
- });
301
- });
302
- };
303
-
304
- exports.getImageListByGroup = getImageListByGroup;
305
-
306
- var getImage = function getImage(context, id) {
307
- var action = 'getItem';
308
- var database = context.database;
309
- var formatId = (0, _utils.parseId)(id);
310
- var aqlQry = (0, _arangojs.aql)(_templateObject(), formatId);
311
- return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
312
- return cursor.next();
313
- }).then(function () {
314
- var image = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
315
- return image;
316
- }).catch(function (error) {
317
- return (0, _utils2.logError)({
318
- action: action,
319
- category: eventCategory,
320
- label: 'db_error'
321
- }, error, context).then(function () {
322
- return null;
323
- });
324
- });
325
- };
326
-
327
- exports.getImage = getImage;
328
-
329
- var getPathUserImages = function getPathUserImages(sub, imgId, type) {
330
- var dir = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : 'images';
331
- var filename = imgId;
332
-
333
- switch (type) {
334
- case 'image/png':
335
- filename = "".concat(imgId, ".png");
336
- break;
337
-
338
- default:
339
- filename = "".concat(imgId, ".jpg");
340
- break;
341
- }
342
-
343
- return "users/".concat(sub, "/").concat(dir, "/").concat(filename);
344
- };
345
-
346
- exports.getPathUserImages = getPathUserImages;
347
-
348
- var getUrlImages = function getUrlImages(data) {
349
- var imgId = data.imgId,
350
- _data$directory = data.directory,
351
- directory = _data$directory === void 0 ? 'images' : _data$directory,
352
- _data$imgType = data.imgType,
353
- imgType = _data$imgType === void 0 ? 'profile' : _data$imgType,
354
- isThumb = data.isThumb,
355
- type = data.type,
356
- typeId = data.typeId;
357
- var host = _config.Config.get('environment') === 'prod' ? "https://box.".concat(_config.Config.get('app.url')) : "https://s3.amazonaws.com/dev.".concat(_config.Config.get('app.url'));
358
- var suffix = isThumb ? '-th' : '';
359
-
360
- if (imgId) {
361
- switch (type) {
362
- case 'app':
363
- // https://box.reaktor.io/myApp/app/images/123.jpg
364
- return "".concat(host, "/").concat(type, "/").concat(directory, "/").concat(imgId).concat(suffix, ".jpg");
365
-
366
- case 'users':
367
- // https://box.reaktor.io/myApp/users/demoUser/images/123.jpg
368
- return "".concat(host, "/").concat(type, "/").concat(typeId, "/").concat(directory, "/").concat(imgId).concat(suffix, ".jpg");
369
- }
370
-
371
- if (imgType === 'profile') {
372
- return "".concat(host, "/defaults/").concat(type, "_bk").concat(suffix, ".jpg");
373
- }
374
-
375
- return "".concat(host, "/defaults/").concat(type, "_wh").concat(suffix, ".jpg");
376
- }
377
-
378
- return '';
379
- };
380
-
381
- exports.getUrlImages = getUrlImages;
382
-
383
- var resizeSaveImage = function resizeSaveImage(context, imgId, buffer) {
384
- var type = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : 'image/jpeg';
385
- var s3Options = arguments.length > 4 ? arguments[4] : undefined;
386
- var action = 'resizeSaveImage';
387
- var sub = context.sub;
388
-
389
- var imgW = _config.Config.get('image.imgSize');
390
-
391
- var imgQ = _config.Config.get('image.imgQuality');
392
-
393
- var photo = {};
394
- var format = type.split('/')[1];
395
- return new Promise(function (resolve) {
396
- gm(buffer, 'img').setFormat(format).quality(imgQ).autoOrient().resize(imgW, imgW, '>').identify({
397
- bufferStream: true
398
- }, function (error) {
399
- var val = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {};
400
-
401
- if (error) {
402
- (0, _utils2.logError)({
403
- action: action,
404
- category: eventCategory,
405
- label: 'image_save',
406
- value: 'gm_image_identify'
407
- }, error, context).catch(function (error) {
408
- throw error;
409
- });
410
- } else {
411
- var formatVals = (0, _utils2.lowerCaseKeys)(val);
412
- var cameraMake = formatVals.make,
413
- cameraModel = formatVals.model,
414
- taken = formatVals.datetimeoriginal;
415
- photo = _objectSpread({}, (0, _cloneDeep.default)(photo), {
416
- make: cameraMake,
417
- model: cameraModel,
418
- taken: taken ? _luxon.DateTime.fromMillis(taken).millisecond : null
419
- }); // If no background color, get the mean color value
420
-
421
- var stats = formatVals['channel statistics'];
422
-
423
- if (stats) {
424
- var red = stats.red,
425
- green = stats.green,
426
- blue = stats.blue,
427
- mean = stats.mean;
428
-
429
- if (red) {
430
- mean = red['standard deviation'] || red.mean;
431
- red = mean ? +mean.split(' ')[0].substring(0, 3) : 0;
432
- } else {
433
- red = 0;
434
- }
435
-
436
- if (green) {
437
- mean = green['standard deviation'] || green.mean;
438
- green = mean ? +mean.split(' ')[0].substring(0, 3) : 0;
439
- } else {
440
- green = 0;
441
- }
442
-
443
- if (blue) {
444
- mean = blue['standard deviation'] || blue.mean;
445
- blue = mean ? +mean.split(' ')[0].substring(0, 3) : 0;
446
- } else {
447
- blue = 0;
448
- }
449
-
450
- var rgb = blue | green << 8 | red << 16;
451
- var color = rgb.toString(16);
452
- photo.color = color === '0' ? '000000' : color;
453
- }
454
- }
455
- }).stream(function (error, stdout) {
456
- if (error) {
457
- (0, _utils2.logError)({
458
- action: action,
459
- category: eventCategory,
460
- isInternal: true,
461
- label: 'image_save',
462
- value: 'gm_image_steam'
463
- }, error, context).then(function () {
464
- return null;
465
- });
466
- } else {
467
- var imageBuffer = new Buffer('');
468
- stdout.on('data', function (data) {
469
- imageBuffer = Buffer.concat([imageBuffer, data]);
470
- });
471
- stdout.on('end', function () {
472
- // Get file size
473
- photo.fileSize = imageBuffer.length; // Upload data
474
-
475
- var imageObj = _objectSpread({
476
- ACL: 'authenticated-read',
477
- Body: imageBuffer,
478
- Bucket: null,
479
- ContentType: type,
480
- Key: getPathUserImages(sub, imgId, type, 'images')
481
- }, s3Options || {});
482
-
483
- (0, _s.s3Put)(imageObj).then(function () {
484
- var thmW = _config.Config.get('image.thmSize');
485
-
486
- var thmQ = _config.Config.get('image.thmQuality'); // Upload thumbnail
487
-
488
-
489
- gm(imageBuffer, 'img').setFormat('jpeg').size(function (thumbError, resizeVal) {
490
- if (!thumbError) {
491
- // Get updated resize values
492
- var height = resizeVal.height,
493
- width = resizeVal.width;
494
- photo = _objectSpread({}, (0, _cloneDeep.default)(photo), {
495
- height: height,
496
- width: width
497
- });
498
- }
499
- }).gravity('Center').resize(thmW, thmW, '^').extent(thmW, thmW).quality(thmQ).stream(function (streamError, thumbStdout) {
500
- if (streamError) {
501
- (0, _utils2.logError)({
502
- action: action,
503
- category: eventCategory,
504
- label: 'image_save',
505
- value: 'gm_thumbnail_steam'
506
- }, streamError, context).catch(function (error) {
507
- throw error;
508
- });
509
- } else {
510
- var thumbBuffer = new Buffer('');
511
- thumbStdout.on('data', function (data) {
512
- thumbBuffer = Buffer.concat([thumbBuffer, data]);
513
- });
514
- thumbStdout.on('end', function () {
515
- // Upload data
516
- var thumbObj = _objectSpread({
517
- ACL: 'authenticated-read',
518
- Body: thumbBuffer,
519
- Bucket: null,
520
- ContentType: type,
521
- Key: getPathUserImages(sub, imgId, type, 'thumbs')
522
- }, s3Options || {});
523
-
524
- (0, _s.s3Put)(thumbObj).then(function () {
525
- resolve(photo);
526
- }).catch(function () {
527
- return (0, _utils2.logError)({
528
- action: action,
529
- category: eventCategory,
530
- label: 'image_save',
531
- value: 's3_put_image'
532
- }, streamError, context).catch(function (error) {
533
- throw error;
534
- });
535
- });
536
- });
537
- }
538
- });
539
- }).catch(function () {
540
- return (0, _utils2.logError)({
541
- action: action,
542
- category: eventCategory,
543
- isInternal: true,
544
- label: 'image_save',
545
- value: 's3_image_save'
546
- }, error, context).then(function () {
547
- return null;
548
- });
549
- });
550
- });
551
- }
552
- });
553
- });
554
- };
555
-
556
- exports.resizeSaveImage = resizeSaveImage;
557
-
558
- var addImage = function addImage(context, image, s3Options) {
559
- var action = 'addImage';
560
- var database = context.database,
561
- sub = context.sub,
562
- sessionId = context.userId;
563
- var imgId = image.imgId,
564
- description = image.description,
565
- buffer = image.buffer,
566
- fileType = image.fileType;
567
- var now = Date.now();
568
- return resizeSaveImage(context, imgId, buffer, fileType, s3Options).then(function (resizedImage) {
569
- var insert = _objectSpread({}, resizedImage, {
570
- _key: imgId,
571
- added: now,
572
- description: description,
573
- fileType: fileType,
574
- modified: now,
575
- sub: sub,
576
- userId: sessionId
577
- });
578
-
579
- var aqlQry = (0, _arangojs.aql)(_templateObject2(), insert);
580
- return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
581
- return cursor.next();
582
- }).then(_utils2.defaultObject).catch(function (error) {
583
- return (0, _utils2.logError)({
584
- action: action,
585
- category: eventCategory,
586
- isInternal: true,
587
- label: 'db_error'
588
- }, error, context).then(function () {
589
- return null;
590
- });
591
- });
592
- }).catch(function (error) {
593
- return (0, _utils2.logError)({
594
- action: action,
595
- category: eventCategory,
596
- isInternal: true,
597
- label: 'image_resize'
598
- }, error, context).then(function () {
599
- return null;
600
- });
601
- });
602
- };
603
-
604
- exports.addImage = addImage;
605
-
606
- var addImageEdge = function addImageEdge(context, imageEdge) {
607
- var action = 'addImageEdge';
608
- var database = context.database,
609
- sessionId = context.userId;
610
- var imgId = imageEdge.imgId,
611
- itemId = imageEdge.itemId,
612
- itemType = imageEdge.itemType;
613
- var now = Date.now();
614
- var edgeCollection = (0, _utils2.useDb)(database).edgeCollection('hasImage');
615
- var edgeId = (0, _utils.createHash)("hasImage-".concat(imgId, "-").concat(itemId, "-").concat(sessionId));
616
- var edge = {
617
- _key: edgeId,
618
- added: now,
619
- type: itemType
620
- };
621
- var formatItemType = (0, _utils.parseChar)(itemType).toLowerCase();
622
- var formatItemId = (0, _utils.parseId)(itemId);
623
- var itemDocId;
624
- var formatImgId = (0, _utils.parseId)(imgId);
625
- var fileDocId = "images/".concat(formatImgId);
626
-
627
- switch (formatItemType) {
628
- case 'posts':
629
- itemDocId = "posts/".concat(formatItemId);
630
- break;
631
-
632
- case 'profile':
633
- itemDocId = "users/".concat(formatItemId);
634
- break;
635
-
636
- default:
637
- itemDocId = '';
638
- break;
639
- }
640
-
641
- if (itemDocId) {
642
- return edgeCollection.save(edge, fileDocId, itemDocId).then(function (fileEdge) {
643
- return edgeCollection.edge(fileEdge);
644
- }).catch(function (error) {
645
- return (0, _utils2.logError)({
646
- action: action,
647
- category: eventCategory,
648
- isInternal: true,
649
- label: 'db_error'
650
- }, error, context).then(function () {
651
- return null;
652
- });
653
- });
654
- }
655
-
656
- return Promise.resolve({});
657
- };
658
-
659
- exports.addImageEdge = addImageEdge;
660
-
661
- var updateImage = function updateImage(context, item, s3Options) {
662
- var action = 'update';
663
- var database = context.database,
664
- sessionId = context.userId; // Item props
665
-
666
- var _item$base = item.base64,
667
- base64 = _item$base === void 0 ? '' : _item$base,
668
- _item$description = item.description,
669
- description = _item$description === void 0 ? '' : _item$description,
670
- imgId = item.imgId,
671
- itemId = item.itemId,
672
- itemType = item.itemType,
673
- fileType = item.fileType,
674
- _item$url = item.url,
675
- url = _item$url === void 0 ? '' : _item$url; // Save Base64 data
676
-
677
- var now = Date.now();
678
- var formatImgId = imgId || (0, _utils.createHash)("image-".concat(sessionId));
679
- var formatItemId = (0, _utils.parseId)(itemId);
680
- var formatItemType = (0, _utils.parseChar)(itemType, 16).toLowerCase();
681
- var customParams = {
682
- description: description,
683
- imgId: formatImgId,
684
- userId: sessionId
685
- };
686
-
687
- if (base64 !== '') {
688
- // Parse Base64 data
689
- var formatBase64 = base64.substr(base64.indexOf(',') + 1);
690
- var buffer = new Buffer(formatBase64, 'base64');
691
- var updatedFileType = fileType;
692
-
693
- if (!fileType) {
694
- var fileMime = (0, _fileType.default)(buffer);
695
- var mime = fileMime.mime;
696
- updatedFileType = mime;
697
- }
698
-
699
- var imgParams = _objectSpread({}, customParams, {
700
- buffer: buffer,
701
- fileType: updatedFileType
702
- });
703
-
704
- return addImage(context, imgParams, s3Options).then(function (image) {
705
- if (formatItemId && formatItemType) {
706
- var imageEdge = {
707
- imgId: formatImgId,
708
- itemId: formatItemId,
709
- itemType: formatItemType
710
- };
711
- return addImageEdge(context, imageEdge).then(function () {
712
- return image;
713
- });
714
- }
715
-
716
- return image;
717
- }).catch(function (error) {
718
- return (0, _utils2.logError)({
719
- action: action,
720
- category: eventCategory,
721
- label: 'image_save_error'
722
- }, error, context).then(function () {
723
- return null;
724
- });
725
- });
726
- } else if (url !== '') {
727
- // Download image from the web
728
- var contentType;
729
- return (0, _ripHunter.get)(url).then(function (res) {
730
- if (res.status !== 200) {
731
- return (0, _utils2.logException)({
732
- action: action,
733
- category: eventCategory,
734
- label: 'fetch_image_url',
735
- value: res.statusText
736
- }, context).then(function () {
737
- return null;
738
- });
739
- }
740
-
741
- contentType = res.headers.get('content-type');
742
- return res;
743
- }).then(function (res) {
744
- return res.buffer();
745
- }).then(function (buffer) {
746
- var imgParams = _objectSpread({}, customParams, {
747
- buffer: buffer,
748
- fileType: contentType
749
- });
750
-
751
- return addImage(context, imgParams, s3Options).then(function (image) {
752
- if (formatItemId && formatItemType) {
753
- var imageEdge = {
754
- imgId: formatImgId,
755
- itemId: formatItemId,
756
- itemType: formatItemType
757
- };
758
- return addImageEdge(context, imageEdge).then(function () {
759
- return image;
760
- });
761
- }
762
-
763
- return image;
764
- });
765
- }).catch(function (error) {
766
- return (0, _utils2.logError)({
767
- action: action,
768
- category: eventCategory,
769
- label: 'fetch_error'
770
- }, error, context).then(function () {
771
- return null;
772
- });
773
- });
774
- } else if (imgId !== '') {
775
- // Update metadata
776
- var update = {
777
- description: description,
778
- modified: now
779
- };
780
- var aqlQry = (0, _arangojs.aql)(_templateObject3(), formatImgId, update);
781
- return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
782
- return cursor.next();
783
- }).catch(function (error) {
784
- return (0, _utils2.logError)({
785
- action: action,
786
- category: eventCategory,
787
- label: 'db_error'
788
- }, error, context).then(function () {
789
- return null;
790
- });
791
- });
792
- }
793
-
794
- return null;
795
- };
796
-
797
- exports.updateImage = updateImage;
798
-
799
- var deleteImage = function deleteImage(context, imgId) {
800
- var action = 'delete';
801
- var database = context.database,
802
- sessionId = context.userId;
803
- var formatImgId = (0, _utils.parseId)(imgId);
804
- var aqlQry = (0, _arangojs.aql)(_templateObject4(), formatImgId, sessionId);
805
- return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
806
- return cursor.next();
807
- }).then(function () {
808
- var image = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
809
-
810
- if (!(0, _isEmpty.default)(image)) {
811
- var imageKey = image._key;
812
- var params = {
813
- Bucket: null,
814
- Delete: {
815
- Objects: [{
816
- Key: "users/".concat(sessionId, "/images/").concat(imageKey, ".jpg")
817
- }, {
818
- Key: "users/".concat(sessionId, "/thumbs/").concat(imageKey, ".jpg")
819
- }],
820
- Quiet: true
821
- }
822
- };
823
- return (0, _s.s3DeleteList)(params).then(function () {
824
- return image;
825
- });
826
- }
827
-
828
- return {};
829
- }).catch(function (error) {
830
- return (0, _utils2.logError)({
831
- action: action,
832
- category: eventCategory,
833
- label: 'db_error'
834
- }, error, context).then(function () {
835
- return null;
836
- });
837
- });
838
- };
839
-
840
- exports.deleteImage = deleteImage;
841
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/data/images.ts"],"names":["eventCategory","parseImageOptions","options","from","to","type","limit","getImageOptional","fields","reduce","selects","field","queries","push","objects","getImageListByUser","context","userId","action","database","formatUserId","aqlQry","aql","query","then","cursor","all","catch","error","category","label","getImageListByLink","itemId","itemType","sessionId","formatItemId","formatItemType","collectionName","verify","Promise","resolve","verifyByType","verifyLink","verifyError","Error","selectObjects","selectQueries","join","getImageListByGroup","params","filters","groupId","formatGroupId","map","filter","conditional","name","value","formatCond","group","privacy","filterStr","grouped","isValid","filterList","getImage","id","formatId","next","image","getPathUserImages","sub","imgId","dir","filename","getUrlImages","data","directory","imgType","isThumb","typeId","host","Config","get","suffix","resizeSaveImage","buffer","s3Options","imgW","imgQ","photo","format","split","gm","setFormat","quality","autoOrient","resize","identify","bufferStream","val","formatVals","cameraMake","make","cameraModel","model","taken","datetimeoriginal","DateTime","fromMillis","millisecond","stats","red","green","blue","mean","substring","rgb","color","toString","stream","stdout","isInternal","imageBuffer","Buffer","on","concat","fileSize","length","imageObj","ACL","Body","Bucket","ContentType","Key","thmW","thmQ","size","thumbError","resizeVal","height","width","gravity","extent","streamError","thumbStdout","thumbBuffer","thumbObj","addImage","description","fileType","now","Date","resizedImage","insert","_key","added","modified","defaultObject","addImageEdge","imageEdge","edgeCollection","edgeId","edge","toLowerCase","itemDocId","formatImgId","fileDocId","save","fileEdge","updateImage","item","base64","url","customParams","formatBase64","substr","indexOf","updatedFileType","fileMime","mime","imgParams","contentType","res","status","statusText","headers","update","deleteImage","imageKey","Delete","Objects","Quiet"],"mappings":"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","sourcesContent":["import {get as httpGet} from '@nlabs/rip-hunter';\nimport {createHash, parseChar, parseId, parseNum} from '@nlabs/utils';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/lib/cjs/aql-query';\nimport {ArrayCursor} from 'arangojs/lib/cjs/cursor';\nimport {DeleteObjectsRequest, PutObjectRequest} from 'aws-sdk/clients/s3';\nimport getFileType, {FileTypeResult} from 'file-type';\nimport * as 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  GroupUserType,\n  ImageEdgeType,\n  ImageIdentifyType,\n  ImageLinkType,\n  ImageOptions,\n  ImageType,\n  ImageUrlData,\n  QueryFilter,\n  UserRelationType\n} from '../types';\nimport {defaultObject, getLimit, logError, logException, lowerCaseKeys, useDb} from '../utils';\nimport {getGroupDetails, isGrouped} from './groups';\nimport {s3DeleteList, s3Put} from './s3';\nimport {getRelationship} from './users';\n\n/**\n * Copyright (c) 2019-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\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\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    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 p.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 getImageListByUser = (\n  context: ApiContext,\n  userId: string,\n  from: number,\n  to: number\n): Promise<ImageType[]> => {\n  const action: string = 'getListByUser';\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 useDb(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 getImageListByLink = (\n  context: ApiContext,\n  itemId: string,\n  itemType: ImageLinkType,\n  options: ImageOptions\n): Promise<ImageType[]> => {\n  const action: string = 'getListByUser';\n  const {database, fields, userId: sessionId} = context;\n  const formatItemId: string = parseId(itemId);\n  const formatItemType: string = parseChar(itemType, 16);\n  let collectionName: string;\n  let verify: Promise<UserRelationType> = Promise.resolve({});\n  let verifyByType: Promise<UserRelationType>;\n\n  switch(formatItemType) {\n    case 'group':\n      collectionName = 'groups';\n    case 'message':\n      collectionName = 'messages';\n    case 'event':\n    case 'post':\n      collectionName = 'posts';\n    default:\n      collectionName = 'users';\n      verifyByType = getRelationship(context, formatItemId);\n  }\n\n  // Make sure we verify the relationship before we expose their images\n  if(sessionId !== formatItemId) {\n    verify = verifyByType;\n  }\n\n  return verify.then((verifyLink) => {\n    if(!verifyLink) {\n      const verifyError = new Error('Unauthorized.');\n      return logError({\n        action,\n        category: eventCategory,\n        label: 'db_error'\n      }, verifyError, context).then(() => null);\n    }\n\n    const {limit} = parseImageOptions(options);\n    const {objects: selectObjects, queries: selectQueries} = getImageOptional(fields);\n\n    const aqlQry: string = `FOR i, l IN 1..1 INBOUND \"${collectionName}/${formatItemId}\" hasImage\n    ${selectQueries.join('\\n')}\n    FILTER l.type == \"${formatItemType}\"\n    SORT i.added\n    ${limit.aql}\n    RETURN MERGE(i, {${selectObjects.join(', ')}})`;\n\n    return useDb(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};\n\nexport const getImageListByGroup = (context: ApiContext, params): Promise<ImageType[]> => {\n  const action: string = 'getListByGroup';\n  const {database, 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 useDb(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: GroupUserType) => {\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 useDb(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 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 useDb(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 = (sub: string, imgId: string, type: string, dir: string = 'images'): string => {\n  let filename: string = imgId;\n\n  switch(type) {\n    case 'image/png':\n      filename = `${imgId}.png`;\n      break;\n    default:\n      filename = `${imgId}.jpg`;\n      break;\n  }\n\n  return `users/${sub}/${dir}/${filename}`;\n};\n\nexport const getUrlImages = (data: ImageUrlData): string => {\n  const {imgId, directory = 'images', imgType = '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(imgId) {\n    switch(type) {\n      case 'app':\n        // https://box.reaktor.io/myApp/app/images/123.jpg\n        return `${host}/${type}/${directory}/${imgId}${suffix}.jpg`;\n      case 'users':\n        // https://box.reaktor.io/myApp/users/demoUser/images/123.jpg\n        return `${host}/${type}/${typeId}/${directory}/${imgId}${suffix}.jpg`;\n    }\n\n    if(imgType === '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 resizeSaveImage = (\n  context: ApiContext,\n  imgId: string,\n  buffer: Buffer,\n  type: string = 'image/jpeg', s3Options?: PutObjectRequest): Promise<ImageType> => {\n  const action: string = 'resizeSaveImage';\n  const {sub} = 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  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_steam'\n          }, error, context).then(() => null);\n        } else {\n          let imageBuffer: Buffer = new Buffer('');\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              Key: getPathUserImages(sub, imgId, type, 'images'),\n              ...s3Options || {}\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 = new Buffer('');\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                          Key: getPathUserImages(sub, imgId, type, 'thumbs'),\n                          ...s3Options || {}\n                        };\n\n                        s3Put(thumbObj)\n                          .then(() => {\n                            resolve(photo);\n                          })\n                          .catch(() => logError({\n                            action,\n                            category: eventCategory,\n                            label: 'image_save',\n                            value: 's3_put_image'\n                          }, streamError, context)\n                            .catch((error) => {\n                              throw error;\n                            }));\n                      });\n                    }\n                  });\n              })\n              .catch(() => logError({\n                action,\n                category: eventCategory,\n                isInternal: true,\n                label: 'image_save',\n                value: 's3_image_save'\n              }, error, 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, sub, userId: sessionId} = context;\n  const {imgId, description, buffer, fileType} = image;\n  const now: number = Date.now();\n\n  return resizeSaveImage(context, imgId, buffer, fileType, s3Options)\n    .then((resizedImage: any) => {\n      const insert: ImageType = {\n        ...resizedImage,\n        _key: imgId,\n        added: now,\n        description,\n        fileType,\n        modified: now,\n        sub,\n        userId: sessionId\n      };\n\n      const aqlQry: AqlQuery = aql`INSERT ${insert} IN images RETURN NEW`;\n\n      return useDb(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 = (context: ApiContext, imageEdge: ImageEdgeType): Promise<object> => {\n  const action: string = 'addImageEdge';\n  const {database, userId: sessionId} = context;\n  const {imgId, itemId, itemType} = imageEdge;\n  const now: number = Date.now();\n  const edgeCollection = useDb(database).edgeCollection('hasImage');\n  const edgeId: string = createHash(`hasImage-${imgId}-${itemId}-${sessionId}`);\n  const edge: any = {\n    _key: edgeId,\n    added: now,\n    type: itemType\n  };\n\n  const formatItemType: string = parseChar(itemType).toLowerCase();\n  const formatItemId: string = parseId(itemId);\n  let itemDocId: string;\n  const formatImgId: string = parseId(imgId);\n  const fileDocId: string = `images/${formatImgId}`;\n\n  switch(formatItemType) {\n    case 'posts':\n      itemDocId = `posts/${formatItemId}`;\n      break;\n    case 'profile':\n      itemDocId = `users/${formatItemId}`;\n      break;\n    default:\n      itemDocId = '';\n      break;\n  }\n\n  if(itemDocId) {\n    return edgeCollection.save(edge, fileDocId, itemDocId)\n      .then((fileEdge) => edgeCollection.edge(fileEdge))\n      .catch((error: Error) => logError({\n        action,\n        category: eventCategory,\n        isInternal: true,\n        label: 'db_error'\n      }, error, context).then(() => null));\n  }\n\n  return Promise.resolve({});\n};\n\nexport const updateImage = (context: ApiContext, item: ImageType, s3Options: PutObjectRequest): Promise<ImageType> => {\n  const action: string = 'update';\n  const {database, userId: sessionId} = context;\n\n  // Item props\n  const {\n    base64 = '',\n    description = '',\n    imgId,\n    itemId,\n    itemType,\n    fileType,\n    url = ''\n  } = item;\n\n  // Save Base64 data\n  const now: number = Date.now();\n  const formatImgId: string = imgId || createHash(`image-${sessionId}`);\n  const formatItemId: string = parseId(itemId);\n  const formatItemType: string = parseChar(itemType, 16).toLowerCase();\n  const customParams: ImageType = {\n    description,\n    imgId: formatImgId,\n    userId: sessionId\n  };\n\n  if(base64 !== '') {\n    // Parse Base64 data\n    const formatBase64: string = base64.substr(base64.indexOf(',') + 1);\n    const buffer = new Buffer(formatBase64, 'base64');\n    let updatedFileType = fileType;\n\n    if(!fileType) {\n      const fileMime: FileTypeResult = getFileType(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            imgId: formatImgId,\n            itemId: formatItemId,\n            itemType: formatItemType\n          };\n          return addImageEdge(context, imageEdge).then(() => image);\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 = {imgId: formatImgId, 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(imgId !== '') {\n    // Update metadata\n    const update: any = {\n      description,\n      modified: now\n    };\n    const aqlQry: AqlQuery = aql`UPDATE ${formatImgId} WITH ${update} IN images RETURN NEW`;\n\n    return useDb(database).query(aqlQry)\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\n  return null;\n};\n\nexport const deleteImage = (context: ApiContext, imgId: string): Promise<ImageType> => {\n  const action: string = 'delete';\n  const {database, userId: sessionId} = context;\n  const formatImgId = parseId(imgId);\n\n  const aqlQry = aql`FOR i IN images\n    FILTER i._key == ${formatImgId} && i.userId == ${sessionId}\n    REMOVE i IN images\n    RETURN OLD`;\n\n  return useDb(database).query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((image: ImageType = {}) => {\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      return {};\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n"]}