@nlabs/reaktor 0.2.2 → 0.3.0

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