@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,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kYXRhL2ltYWdlcy50cyJdLCJuYW1lcyI6WyJldmVudENhdGVnb3J5IiwicGFyc2VJbWFnZU9wdGlvbnMiLCJvcHRpb25zIiwiZnJvbSIsInRvIiwidHlwZSIsImxpbWl0IiwiZ2V0SW1hZ2VPcHRpb25hbCIsImZpZWxkcyIsInJlZHVjZSIsInNlbGVjdHMiLCJmaWVsZCIsInF1ZXJpZXMiLCJwdXNoIiwib2JqZWN0cyIsImdldEltYWdlTGlzdEJ5VXNlciIsImNvbnRleHQiLCJ1c2VySWQiLCJhY3Rpb24iLCJkYXRhYmFzZSIsImZvcm1hdFVzZXJJZCIsImFxbFFyeSIsImFxbCIsInF1ZXJ5IiwidGhlbiIsImN1cnNvciIsImFsbCIsImNhdGNoIiwiZXJyb3IiLCJjYXRlZ29yeSIsImxhYmVsIiwiZ2V0SW1hZ2VMaXN0QnlMaW5rIiwiaXRlbUlkIiwiaXRlbVR5cGUiLCJzZXNzaW9uSWQiLCJmb3JtYXRJdGVtSWQiLCJmb3JtYXRJdGVtVHlwZSIsImNvbGxlY3Rpb25OYW1lIiwidmVyaWZ5IiwiUHJvbWlzZSIsInJlc29sdmUiLCJ2ZXJpZnlCeVR5cGUiLCJ2ZXJpZnlMaW5rIiwidmVyaWZ5RXJyb3IiLCJFcnJvciIsInNlbGVjdE9iamVjdHMiLCJzZWxlY3RRdWVyaWVzIiwiam9pbiIsImdldEltYWdlTGlzdEJ5R3JvdXAiLCJwYXJhbXMiLCJmaWx0ZXJzIiwiZ3JvdXBJZCIsImZvcm1hdEdyb3VwSWQiLCJtYXAiLCJmaWx0ZXIiLCJjb25kaXRpb25hbCIsIm5hbWUiLCJ2YWx1ZSIsImZvcm1hdENvbmQiLCJncm91cCIsInByaXZhY3kiLCJmaWx0ZXJTdHIiLCJncm91cGVkIiwiaXNWYWxpZCIsImZpbHRlckxpc3QiLCJnZXRJbWFnZSIsImlkIiwiZm9ybWF0SWQiLCJuZXh0IiwiaW1hZ2UiLCJnZXRQYXRoVXNlckltYWdlcyIsInN1YiIsImltZ0lkIiwiZGlyIiwiZmlsZW5hbWUiLCJnZXRVcmxJbWFnZXMiLCJkYXRhIiwiZGlyZWN0b3J5IiwiaW1nVHlwZSIsImlzVGh1bWIiLCJ0eXBlSWQiLCJob3N0IiwiQ29uZmlnIiwiZ2V0Iiwic3VmZml4IiwicmVzaXplU2F2ZUltYWdlIiwiYnVmZmVyIiwiczNPcHRpb25zIiwiaW1nVyIsImltZ1EiLCJwaG90byIsImZvcm1hdCIsInNwbGl0IiwiZ20iLCJzZXRGb3JtYXQiLCJxdWFsaXR5IiwiYXV0b09yaWVudCIsInJlc2l6ZSIsImlkZW50aWZ5IiwiYnVmZmVyU3RyZWFtIiwidmFsIiwiZm9ybWF0VmFscyIsImNhbWVyYU1ha2UiLCJtYWtlIiwiY2FtZXJhTW9kZWwiLCJtb2RlbCIsInRha2VuIiwiZGF0ZXRpbWVvcmlnaW5hbCIsIkRhdGVUaW1lIiwiZnJvbU1pbGxpcyIsIm1pbGxpc2Vjb25kIiwic3RhdHMiLCJyZWQiLCJncmVlbiIsImJsdWUiLCJtZWFuIiwic3Vic3RyaW5nIiwicmdiIiwiY29sb3IiLCJ0b1N0cmluZyIsInN0cmVhbSIsInN0ZG91dCIsImlzSW50ZXJuYWwiLCJpbWFnZUJ1ZmZlciIsIkJ1ZmZlciIsIm9uIiwiY29uY2F0IiwiZmlsZVNpemUiLCJsZW5ndGgiLCJpbWFnZU9iaiIsIkFDTCIsIkJvZHkiLCJCdWNrZXQiLCJDb250ZW50VHlwZSIsIktleSIsInRobVciLCJ0aG1RIiwic2l6ZSIsInRodW1iRXJyb3IiLCJyZXNpemVWYWwiLCJoZWlnaHQiLCJ3aWR0aCIsImdyYXZpdHkiLCJleHRlbnQiLCJzdHJlYW1FcnJvciIsInRodW1iU3Rkb3V0IiwidGh1bWJCdWZmZXIiLCJ0aHVtYk9iaiIsImFkZEltYWdlIiwiZGVzY3JpcHRpb24iLCJmaWxlVHlwZSIsIm5vdyIsIkRhdGUiLCJyZXNpemVkSW1hZ2UiLCJpbnNlcnQiLCJfa2V5IiwiYWRkZWQiLCJtb2RpZmllZCIsImRlZmF1bHRPYmplY3QiLCJhZGRJbWFnZUVkZ2UiLCJpbWFnZUVkZ2UiLCJlZGdlQ29sbGVjdGlvbiIsImVkZ2VJZCIsImVkZ2UiLCJ0b0xvd2VyQ2FzZSIsIml0ZW1Eb2NJZCIsImZvcm1hdEltZ0lkIiwiZmlsZURvY0lkIiwic2F2ZSIsImZpbGVFZGdlIiwidXBkYXRlSW1hZ2UiLCJpdGVtIiwiYmFzZTY0IiwidXJsIiwiY3VzdG9tUGFyYW1zIiwiZm9ybWF0QmFzZTY0Iiwic3Vic3RyIiwiaW5kZXhPZiIsInVwZGF0ZWRGaWxlVHlwZSIsImZpbGVNaW1lIiwibWltZSIsImltZ1BhcmFtcyIsImNvbnRlbnRUeXBlIiwicmVzIiwic3RhdHVzIiwic3RhdHVzVGV4dCIsImhlYWRlcnMiLCJ1cGRhdGUiLCJkZWxldGVJbWFnZSIsImltYWdlS2V5IiwiRGVsZXRlIiwiT2JqZWN0cyIsIlF1aWV0Il0sIm1hcHBpbmdzIjoiOzs7Ozs7O0FBQUE7O0FBQ0E7O0FBQ0E7O0FBSUE7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBQ0E7O0FBRUE7O0FBZUE7O0FBQ0E7O0FBQ0E7O0FBQ0E7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFFQTs7OztBQUlBLElBQU1BLGFBQXFCLEdBQUcsUUFBOUI7O0FBRU8sSUFBTUMsaUJBQWlCLEdBQUcsU0FBcEJBLGlCQUFvQixHQUFnQztBQUFBLE1BQS9CQyxPQUErQix1RUFBUCxFQUFPO0FBQUEsc0JBSzNEQSxPQUwyRCxDQUU3REMsSUFGNkQ7QUFBQSxNQUU3REEsSUFGNkQsOEJBRXRELENBRnNEO0FBQUEsb0JBSzNERCxPQUwyRCxDQUc3REUsRUFINkQ7QUFBQSxNQUc3REEsRUFINkQsNEJBR3hELEVBSHdEO0FBQUEsc0JBSzNERixPQUwyRCxDQUk3REcsSUFKNkQ7QUFBQSxNQUk3REEsSUFKNkQsOEJBSXRELFNBSnNEO0FBUS9ELFNBQU87QUFDTEMsSUFBQUEsS0FBSyxFQUFFLHNCQUFTSCxJQUFULEVBQWVDLEVBQWYsQ0FERjtBQUVMQyxJQUFBQSxJQUFJLEVBQUUsc0JBQVVBLElBQVYsRUFBZ0IsRUFBaEI7QUFGRCxHQUFQO0FBSUQsQ0FaTTs7OztBQWNBLElBQU1FLGdCQUFnQixHQUFHLFNBQW5CQSxnQkFBbUIsQ0FB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LEdBQUcsa0JBQVdwRyxTQUFYLHFCQUErQjJKLFFBQS9CO0FBQUosV0FGTyxDQURIO0FBS05HLFVBQUFBLEtBQUssRUFBRTtBQUxEO0FBRjJCLE9BQXJDO0FBV0EsYUFBTyxxQkFBYS9JLE1BQWIsRUFBcUJ6QixJQUFyQixDQUEwQjtBQUFBLGVBQU02QyxLQUFOO0FBQUEsT0FBMUIsQ0FBUDtBQUNEOztBQUNELFdBQU8sRUFBUDtBQUNELEdBbkJJLEVBb0JKMUMsS0FwQkksQ0FvQkUsVUFBQ0MsS0FBRDtBQUFBLFdBQWtCLHNCQUFTO0FBQ2hDVixNQUFBQSxNQUFNLEVBQU5BLE1BRGdDO0FBRWhDVyxNQUFBQSxRQUFRLEVBQUU3QixhQUZzQjtBQUdoQzhCLE1BQUFBLEtBQUssRUFBRTtBQUh5QixLQUFULEVBSXRCRixLQUpzQixFQUlmWixPQUplLEVBSU5RLElBSk0sQ0FJRDtBQUFBLGFBQU0sSUFBTjtBQUFBLEtBSkMsQ0FBbEI7QUFBQSxHQXBCRixDQUFQO0FBeUJELENBbkNNIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtnZXQgYXMgaHR0cEdldH0gZnJvbSAnQG5sYWJzL3JpcC1odW50ZXInO1xuaW1wb3J0IHtjcmVhdGVIYXNoLCBwYXJzZUNoYXIsIHBhcnNlSWQsIHBhcnNlTnVtfSBmcm9tICdAbmxhYnMvdXRpbHMnO1xuaW1wb3J0IHthcWx9IGZyb20gJ2FyYW5nb2pzJztcbmltcG9ydCB7QXFsUXVlcnl9IGZyb20gJ2FyYW5nb2pzL2xpYi9janMvYXFsLXF1ZXJ5JztcbmltcG9ydCB7QXJyYXlDdXJzb3J9IGZyb20gJ2FyYW5nb2pzL2xpYi9janMvY3Vyc29yJztcbmltcG9ydCB7RGVsZXRlT2JqZWN0c1JlcXVlc3QsIFB1dE9iamVjdFJlcXVlc3R9IGZyb20gJ2F3cy1zZGsvY2xpZW50cy9zMyc7XG5pbXBvcnQgZ2V0RmlsZVR5cGUsIHtGaWxlVHlwZVJlc3VsdH0gZnJvbSAnZmlsZS10eXBlJztcbmltcG9ydCAqIGFzIGdtIGZyb20gJ2dtJztcbmltcG9ydCBjbG9uZURlZXAgZnJvbSAnbG9kYXNoL2Nsb25lRGVlcCc7XG5pbXBvcnQgaXNFbXB0eSBmcm9tICdsb2Rhc2gvaXNFbXB0eSc7XG5pbXBvcnQge0RhdGVUaW1lfSBmcm9tICdsdXhvbic7XG5cbmltcG9ydCB7Q29uZmlnfSBmcm9tICcuLi9jb25maWcnO1xuaW1wb3J0IHtcbiAgQXBpQ29udGV4dCxcbiAgQXJhbmdvREJMaW1pdCxcbiAgR3JvdXBUeXBlLFxuICBHcm91cFVzZXJUeXBlLFxuICBJbWFnZUVkZ2VUeXBlLFxuICBJbWFnZUlkZW50aWZ5VHlwZSxcbiAgSW1hZ2VMaW5rVHlwZSxcbiAgSW1hZ2VPcHRpb25zLFxuICBJbWFnZVR5cGUsXG4gIEltYWdlVXJsRGF0YSxcbiAgUXVlcnlGaWx0ZXIsXG4gIFVzZXJSZWxhdGlvblR5cGVcbn0gZnJvbSAnLi4vdHlwZXMnO1xuaW1wb3J0IHtkZWZhdWx0T2JqZWN0LCBnZXRMaW1pdCwgbG9nRXJyb3IsIGxvZ0V4Y2VwdGlvbiwgbG93ZXJDYXNlS2V5cywgdXNlRGJ9IGZyb20gJy4uL3V0aWxzJztcbmltcG9ydCB7Z2V0R3JvdXBEZXRhaWxzLCBpc0dyb3VwZWR9IGZyb20gJy4vZ3JvdXBzJztcbmltcG9ydCB7czNEZWxldGVMaXN0LCBzM1B1dH0gZnJvbSAnLi9zMyc7XG5pbXBvcnQge2dldFJlbGF0aW9uc2hpcH0gZnJvbSAnLi91c2Vycyc7XG5cbi8qKlxuICogQ29weXJpZ2h0IChjKSAyMDE5LVByZXNlbnQsIE5pdHJvZ2VuIExhYnMsIEluYy5cbiAqIENvcHlyaWdodHMgbGljZW5zZWQgdW5kZXIgdGhlIE1JVCBMaWNlbnNlLiBTZWUgdGhlIGFjY29tcGFueWluZyBMSUNFTlNFIGZpbGUgZm9yIHRlcm1zLlxuICovXG5jb25zdCBldmVudENhdGVnb3J5OiBzdHJpbmcgPSAnaW1hZ2VzJztcblxuZXhwb3J0IGNvbnN0IHBhcnNlSW1hZ2VPcHRpb25zID0gKG9wdGlvbnM6IEltYWdlT3B0aW9ucyA9IHt9KSA9PiB7XG4gIGNvbnN0IHtcbiAgICBmcm9tID0gMCxcbiAgICB0byA9IDMwLFxuICAgIHR5cGUgPSAnZGVmYXVsdCdcbiAgfSA9IG9wdGlvbnM7XG5cblxuICByZXR1cm4ge1xuICAgIGxpbWl0OiBnZXRMaW1pdChmcm9tLCB0byksXG4gICAgdHlwZTogcGFyc2VDaGFyKHR5cGUsIDMyKVxuICB9O1xufTtcblxuZXhwb3J0IGNvbnN0IGdldEltYWdlT3B0aW9uYWwgPSAoZmllbGRzOiBzdHJpbmdbXSkgPT5cbiAgZmllbGRzLnJlZHVjZSgoc2VsZWN0czogYW55LCBmaWVsZDogc3RyaW5nKSA9PiB7XG4gICAgc3dpdGNoKGZpZWxkKSB7XG4gICAgICBjYXNlICdyZWFjdGlvbnMnOiB7XG4gICAgICAgIHNlbGVjdHMucXVlcmllcy5wdXNoKGBMRVQgcmVhY3Rpb25zID0gKFxuICAgICAgICAgIEZPUiBpbWFnZSwgciBJTiBJTkJPVU5EIGkuX2lkIHJlYWN0aW9uc1xuICAgICAgICAgIENPTExFQ1QgcmVhY3Rpb25OYW1lID0gci52YWx1ZSBJTlRPIHJlYWN0aW9uSXRlbXNcbiAgICAgICAgICBSRVRVUk4ge25hbWU6IHJlYWN0aW9uTmFtZSwgY291bnQ6IExFTkdUSChyZWFjdGlvbkl0ZW1zWypdLnIudmFsdWUpfVxuICAgICAgICApYCk7XG4gICAgICAgIHNlbGVjdHMub2JqZWN0cy5wdXNoKCdyZWFjdGlvbnM6cmVhY3Rpb25zJyk7XG4gICAgICAgIHJldHVybiBzZWxlY3RzO1xuICAgICAgfVxuICAgICAgY2FzZSAndGFncyc6IHtcbiAgICAgICAgc2VsZWN0cy5xdWVyaWVzLnB1c2goYExFVCB0YWdzID0gKFxuICAgICAgICAgIEZPUiB0LCBwbCBJTiBJTkJPVU5EIGkuX2lkIGlzVGFnZ2VkXG4gICAgICAgICAgUkVUVVJOIHRcbiAgICAgICAgKWApO1xuICAgICAgICBzZWxlY3RzLm9iamVjdHMucHVzaCgndGFnczp0YWdzJyk7XG4gICAgICAgIHJldHVybiBzZWxlY3RzO1xuICAgICAgfVxuICAgICAgY2FzZSAndXNlcic6IHtcbiAgICAgICAgc2VsZWN0cy5xdWVyaWVzLnB1c2goYExFVCB1c2VyID0gRklSU1QoXG4gICAgICAgICAgRk9SIHUgSU4gdXNlcnNcbiAgICAgICAgICBGSUxURVIgcC51c2VySWQgPT0gdS5fa2V5XG4gICAgICAgICAgTElNSVQgMVxuICAgICAgICAgIFJFVFVSTiB1XG4gICAgICAgIClgKTtcbiAgICAgICAgc2VsZWN0cy5vYmplY3RzLnB1c2goJ3VzZXI6dXNlcicpO1xuICAgICAgICByZXR1cm4gc2VsZWN0cztcbiAgICAgIH1cbiAgICAgIGRlZmF1bHQ6IHtcbiAgICAgICAgcmV0dXJuIHNlbGVjdHM7XG4gICAgICB9XG4gICAgfVxuICB9LCB7b2JqZWN0czogW10sIHF1ZXJpZXM6IFtdfSk7XG5cbmV4cG9ydCBjb25zdCBnZXRJbWFnZUxpc3RCeVVzZXIgPSAoXG4gIGNvbnRleHQ6IEFwaUNvbnRleHQsXG4gIHVzZXJJZDogc3RyaW5nLFxuICBmcm9tOiBudW1iZXIsXG4gIHRvOiBudW1iZXJcbik6IFByb21pc2U8SW1hZ2VUeXBlW10+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnZ2V0TGlzdEJ5VXNlcic7XG4gIGNvbnN0IHtkYXRhYmFzZX0gPSBjb250ZXh0O1xuICBjb25zdCBmb3JtYXRVc2VySWQ6IHN0cmluZyA9IHBhcnNlSWQodXNlcklkKTtcbiAgY29uc3QgbGltaXQ6IEFyYW5nb0RCTGltaXQgPSBnZXRMaW1pdChmcm9tLCB0byk7XG4gIGNvbnN0IGFxbFFyeTogc3RyaW5nID0gYEZPUiBpIElOIGltYWdlc1xuICAgICAgRklMVEVSIGkudXNlcklkID09IFwiJHtmb3JtYXRVc2VySWR9XCJcbiAgICAgIExFVCB1c2VyID0gRklSU1QoXG4gICAgICAgIEZPUiB1IElOIHVzZXJzXG4gICAgICAgIEZJTFRFUiB1Ll9rZXkgPT0gaS51c2VySWRcbiAgICAgICAgTElNSVQgMVxuICAgICAgICBSRVRVUk4gdVxuICAgICAgKVxuICAgICAgJHtsaW1pdC5hcWx9XG4gICAgICBTT1JUIGkuYWRkZWRcbiAgICAgIFJFVFVSTiBNRVJHRShpLCB7dXNlcjp1c2VyfSlgO1xuXG4gIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IuYWxsKCkpXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgbGFiZWw6ICdkYl9lcnJvcidcbiAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKSk7XG59O1xuXG5leHBvcnQgY29uc3QgZ2V0SW1hZ2VMaXN0QnlMaW5rID0gKFxuICBjb250ZXh0OiBBcGlDb250ZXh0LFxuICBpdGVtSWQ6IHN0cmluZyxcbiAgaXRlbVR5cGU6IEltYWdlTGlua1R5cGUsXG4gIG9wdGlvbnM6IEltYWdlT3B0aW9uc1xuKTogUHJvbWlzZTxJbWFnZVR5cGVbXT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRMaXN0QnlVc2VyJztcbiAgY29uc3Qge2RhdGFiYXNlLCBmaWVsZHMsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG4gIGNvbnN0IGZvcm1hdEl0ZW1JZDogc3RyaW5nID0gcGFyc2VJZChpdGVtSWQpO1xuICBjb25zdCBmb3JtYXRJdGVtVHlwZTogc3RyaW5nID0gcGFyc2VDaGFyKGl0ZW1UeXBlLCAxNik7XG4gIGxldCBjb2xsZWN0aW9uTmFtZTogc3RyaW5nO1xuICBsZXQgdmVyaWZ5OiBQcm9taXNlPFVzZXJSZWxhdGlvblR5cGU+ID0gUHJvbWlzZS5yZXNvbHZlKHt9KTtcbiAgbGV0IHZlcmlmeUJ5VHlwZTogUHJvbWlzZTxVc2VyUmVsYXRpb25UeXBlPjtcblxuICBzd2l0Y2goZm9ybWF0SXRlbVR5cGUpIHtcbiAgICBjYXNlICdncm91cCc6XG4gICAgICBjb2xsZWN0aW9uTmFtZSA9ICdncm91cHMnO1xuICAgIGNhc2UgJ21lc3NhZ2UnOlxuICAgICAgY29sbGVjdGlvbk5hbWUgPSAnbWVzc2FnZXMnO1xuICAgIGNhc2UgJ2V2ZW50JzpcbiAgICBjYXNlICdwb3N0JzpcbiAgICAgIGNvbGxlY3Rpb25OYW1lID0gJ3Bvc3RzJztcbiAgICBkZWZhdWx0OlxuICAgICAgY29sbGVjdGlvbk5hbWUgPSAndXNlcnMnO1xuICAgICAgdmVyaWZ5QnlUeXBlID0gZ2V0UmVsYXRpb25zaGlwKGNvbnRleHQsIGZvcm1hdEl0ZW1JZCk7XG4gIH1cblxuICAvLyBNYWtlIHN1cmUgd2UgdmVyaWZ5IHRoZSByZWxhdGlvbnNoaXAgYmVmb3JlIHdlIGV4cG9zZSB0aGVpciBpbWFnZXNcbiAgaWYoc2Vzc2lvbklkICE9PSBmb3JtYXRJdGVtSWQpIHtcbiAgICB2ZXJpZnkgPSB2ZXJpZnlCeVR5cGU7XG4gIH1cblxuICByZXR1cm4gdmVyaWZ5LnRoZW4oKHZlcmlmeUxpbmspID0+IHtcbiAgICBpZighdmVyaWZ5TGluaykge1xuICAgICAgY29uc3QgdmVyaWZ5RXJyb3IgPSBuZXcgRXJyb3IoJ1VuYXV0aG9yaXplZC4nKTtcbiAgICAgIHJldHVybiBsb2dFcnJvcih7XG4gICAgICAgIGFjdGlvbixcbiAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgIGxhYmVsOiAnZGJfZXJyb3InXG4gICAgICB9LCB2ZXJpZnlFcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKTtcbiAgICB9XG5cbiAgICBjb25zdCB7bGltaXR9ID0gcGFyc2VJbWFnZU9wdGlvbnMob3B0aW9ucyk7XG4gICAgY29uc3Qge29iamVjdHM6IHNlbGVjdE9iamVjdHMsIHF1ZXJpZXM6IHNlbGVjdFF1ZXJpZXN9ID0gZ2V0SW1hZ2VPcHRpb25hbChmaWVsZHMpO1xuXG4gICAgY29uc3QgYXFsUXJ5OiBzdHJpbmcgPSBgRk9SIGksIGwgSU4gMS4uMSBJTkJPVU5EIFwiJHtjb2xsZWN0aW9uTmFtZX0vJHtmb3JtYXRJdGVtSWR9XCIgaGFzSW1hZ2VcbiAgICAke3NlbGVjdFF1ZXJpZXMuam9pbignXFxuJyl9XG4gICAgRklMVEVSIGwudHlwZSA9PSBcIiR7Zm9ybWF0SXRlbVR5cGV9XCJcbiAgICBTT1JUIGkuYWRkZWRcbiAgICAke2xpbWl0LmFxbH1cbiAgICBSRVRVUk4gTUVSR0UoaSwgeyR7c2VsZWN0T2JqZWN0cy5qb2luKCcsICcpfX0pYDtcblxuICAgIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5hbGwoKSlcbiAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgIGFjdGlvbixcbiAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgIGxhYmVsOiAnZGJfZXJyb3InXG4gICAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKSk7XG4gIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldEltYWdlTGlzdEJ5R3JvdXAgPSAoY29udGV4dDogQXBpQ29udGV4dCwgcGFyYW1zKTogUHJvbWlzZTxJbWFnZVR5cGVbXT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRMaXN0QnlHcm91cCc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgdXNlcklkOiBzZXNzaW9uSWR9ID0gY29udGV4dDtcbiAgY29uc3Qge2ZpbHRlcnMgPSBbXSwgZ3JvdXBJZCwgZnJvbSwgdG99ID0gcGFyYW1zO1xuICBjb25zdCBmb3JtYXRHcm91cElkOiBzdHJpbmcgPSBwYXJzZUlkKGdyb3VwSWQpO1xuICBjb25zdCBsaW1pdCA9IGdldExpbWl0KGZyb20sIHRvKTtcblxuICBmaWx0ZXJzXG4gICAgLm1hcCgoZmlsdGVyOiBRdWVyeUZpbHRlcikgPT4ge1xuICAgICAgY29uc3Qge2NvbmRpdGlvbmFsLCBuYW1lLCB2YWx1ZX0gPSBmaWx0ZXI7XG4gICAgICBsZXQgZm9ybWF0Q29uZDogc3RyaW5nID0gY29uZGl0aW9uYWw7XG5cbiAgICAgIGlmKGNvbmRpdGlvbmFsICE9PSAnPj0nICYmIGNvbmRpdGlvbmFsICE9PSAnPD0nICYmIGNvbmRpdGlvbmFsICE9PSAnPicgJiYgY29uZGl0aW9uYWwgIT09ICc8Jykge1xuICAgICAgICBmb3JtYXRDb25kID0gJz09JztcbiAgICAgIH1cblxuICAgICAgc3dpdGNoKG5hbWUpIHtcbiAgICAgICAgY2FzZSAnYWRkZWQnOlxuICAgICAgICAgIHJldHVybiBgcC5hZGRlZCAke2Zvcm1hdENvbmR9ICR7cGFyc2VOdW0odmFsdWUpfWA7XG4gICAgICAgIGRlZmF1bHQ6XG4gICAgICAgICAgcmV0dXJuICcnO1xuICAgICAgfVxuICAgIH0pO1xuXG4gIHJldHVybiBnZXRHcm91cERldGFpbHMoY29udGV4dCwgZm9ybWF0R3JvdXBJZClcbiAgICAudGhlbigoZ3JvdXA6IEdyb3VwVHlwZSkgPT4ge1xuICAgICAgaWYoZ3JvdXAucHJpdmFjeSA9PT0gJ3B1YmxpYycpIHtcbiAgICAgICAgZmlsdGVycy5wdXNoKGBwLmdyb3VwSWQgPT0gXCIke2dyb3VwSWR9XCJgKTtcbiAgICAgICAgY29uc3QgZmlsdGVyU3RyID0gZmlsdGVycy5qb2luKCcgJiYgJyk7XG4gICAgICAgIGNvbnN0IGFxbFFyeTogc3RyaW5nID0gYEZPUiBpIElOXG4gICAgICAgICAgRkxBVFRFTihcbiAgICAgICAgICAgIEZPUiBwIElOIHBvc3RzXG4gICAgICAgICAgICBGSUxURVIgJHtmaWx0ZXJTdHJ9XG4gICAgICAgICAgICBMRVQgaW1hZ2VzID0gKFxuICAgICAgICAgICAgICBGT1IgaSwgZSBJTiBJTkJPVU5EIHAuX2lkIGhhc0ltYWdlXG4gICAgICAgICAgICAgIFJFVFVSTiBpXG4gICAgICAgICAgICApXG4gICAgICAgICAgICBTT1JUIHAuYWRkZWQgREVTQ1xuICAgICAgICAgICAgUkVUVVJOIGltYWdlc1xuICAgICAgICAgIClcbiAgICAgICAgICBTT1JUIGkuYWRkZWQgREVTQ1xuICAgICAgICAgICR7bGltaXQuYXFsfVxuICAgICAgICAgIFJFVFVSTiBpYDtcblxuICAgICAgICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAgICAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLmFsbCgpKVxuICAgICAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgICAgIGxhYmVsOiAnZGJfZXJyb3InXG4gICAgICAgICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCkpO1xuICAgICAgfVxuICAgICAgcmV0dXJuIGlzR3JvdXBlZChkYXRhYmFzZSwgc2Vzc2lvbklkLCBncm91cElkKVxuICAgICAgICAudGhlbigoZ3JvdXBlZDogR3JvdXBVc2VyVHlwZSkgPT4ge1xuICAgICAgICAgIGlmKGdyb3VwZWQuaXNWYWxpZCkge1xuICAgICAgICAgICAgZmlsdGVycy5wdXNoKGBwLmdyb3VwSWQgPT0gXCIke2dyb3VwZWQuZ3JvdXBJZH1cImApO1xuICAgICAgICAgICAgY29uc3QgZmlsdGVyTGlzdDogc3RyaW5nID0gZmlsdGVycy5qb2luKCcgJiYgJyk7XG4gICAgICAgICAgICBjb25zdCBhcWxRcnk6IHN0cmluZyA9IGBGT1IgcCBJTiBwb3N0XG4gICAgICAgICAgICAgICAgRklMVEVSICR7ZmlsdGVyTGlzdH1cbiAgICAgICAgICAgICAgICBGT1IgZiBJTiBwLmZpbGVzXG4gICAgICAgICAgICAgICAgRklMVEVSIGYudHlwZSA9PSBcImltYWdlL2pwZWdcIiB8fCBmLnR5cGUgPT0gXCJpbWFnZS9wbmdcIlxuICAgICAgICAgICAgICAgICR7bGltaXQuYXFsfVxuICAgICAgICAgICAgICAgIFNPUlQgcC5hZGRlZCBERVNDXG4gICAgICAgICAgICAgICAgUkVUVVJOIGZgO1xuXG4gICAgICAgICAgICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAgICAgICAgICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5hbGwoKSlcbiAgICAgICAgICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgICAgICAgbGFiZWw6ICdkYl9lcnJvcidcbiAgICAgICAgICAgICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCkpO1xuICAgICAgICAgIH1cbiAgICAgICAgICByZXR1cm4gW107XG4gICAgICAgIH0pO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldEltYWdlID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGlkOiBzdHJpbmcpOiBQcm9taXNlPEltYWdlVHlwZT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRJdGVtJztcbiAgY29uc3Qge2RhdGFiYXNlfSA9IGNvbnRleHQ7XG4gIGNvbnN0IGZvcm1hdElkOiBzdHJpbmcgPSBwYXJzZUlkKGlkKTtcbiAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBGT1IgaSBJTiBpbWFnZXNcbiAgICBGSUxURVIgaS5fa2V5PT0ke2Zvcm1hdElkfVxuICAgIExJTUlUIDFcbiAgICBSRVRVUk4gaWA7XG5cbiAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgLnRoZW4oKGltYWdlOiBJbWFnZVR5cGUgPSB7fSkgPT4gaW1hZ2UpXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgbGFiZWw6ICdkYl9lcnJvcidcbiAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKSk7XG59O1xuXG5leHBvcnQgY29uc3QgZ2V0UGF0aFVzZXJJbWFnZXMgPSAoc3ViOiBzdHJpbmcsIGltZ0lkOiBzdHJpbmcsIHR5cGU6IHN0cmluZywgZGlyOiBzdHJpbmcgPSAnaW1hZ2VzJyk6IHN0cmluZyA9PiB7XG4gIGxldCBmaWxlbmFtZTogc3RyaW5nID0gaW1nSWQ7XG5cbiAgc3dpdGNoKHR5cGUpIHtcbiAgICBjYXNlICdpbWFnZS9wbmcnOlxuICAgICAgZmlsZW5hbWUgPSBgJHtpbWdJZH0ucG5nYDtcbiAgICAgIGJyZWFrO1xuICAgIGRlZmF1bHQ6XG4gICAgICBmaWxlbmFtZSA9IGAke2ltZ0lkfS5qcGdgO1xuICAgICAgYnJlYWs7XG4gIH1cblxuICByZXR1cm4gYHVzZXJzLyR7c3VifS8ke2Rpcn0vJHtmaWxlbmFtZX1gO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldFVybEltYWdlcyA9IChkYXRhOiBJbWFnZVVybERhdGEpOiBzdHJpbmcgPT4ge1xuICBjb25zdCB7aW1nSWQsIGRpcmVjdG9yeSA9ICdpbWFnZXMnLCBpbWdUeXBlID0gJ3Byb2ZpbGUnLCBpc1RodW1iLCB0eXBlLCB0eXBlSWR9ID0gZGF0YTtcbiAgY29uc3QgaG9zdDogc3RyaW5nID0gQ29uZmlnLmdldCgnZW52aXJvbm1lbnQnKSA9PT0gJ3Byb2QnXG4gICAgPyBgaHR0cHM6Ly9ib3guJHtDb25maWcuZ2V0KCdhcHAudXJsJyl9YFxuICAgIDogYGh0dHBzOi8vczMuYW1hem9uYXdzLmNvbS9kZXYuJHtDb25maWcuZ2V0KCdhcHAudXJsJyl9YDtcbiAgY29uc3Qgc3VmZml4OiBzdHJpbmcgPSBpc1RodW1iID8gJy10aCcgOiAnJztcblxuICBpZihpbWdJZCkge1xuICAgIHN3aXRjaCh0eXBlKSB7XG4gICAgICBjYXNlICdhcHAnOlxuICAgICAgICAvLyBodHRwczovL2JveC5yZWFrdG9yLmlvL215QXBwL2FwcC9pbWFnZXMvMTIzLmpwZ1xuICAgICAgICByZXR1cm4gYCR7aG9zdH0vJHt0eXBlfS8ke2RpcmVjdG9yeX0vJHtpbWdJZH0ke3N1ZmZpeH0uanBnYDtcbiAgICAgIGNhc2UgJ3VzZXJzJzpcbiAgICAgICAgLy8gaHR0cHM6Ly9ib3gucmVha3Rvci5pby9teUFwcC91c2Vycy9kZW1vVXNlci9pbWFnZXMvMTIzLmpwZ1xuICAgICAgICByZXR1cm4gYCR7aG9zdH0vJHt0eXBlfS8ke3R5cGVJZH0vJHtkaXJlY3Rvcnl9LyR7aW1nSWR9JHtzdWZmaXh9LmpwZ2A7XG4gICAgfVxuXG4gICAgaWYoaW1nVHlwZSA9PT0gJ3Byb2ZpbGUnKSB7XG4gICAgICByZXR1cm4gYCR7aG9zdH0vZGVmYXVsdHMvJHt0eXBlfV9iayR7c3VmZml4fS5qcGdgO1xuICAgIH1cblxuICAgIHJldHVybiBgJHtob3N0fS9kZWZhdWx0cy8ke3R5cGV9X3doJHtzdWZmaXh9LmpwZ2A7XG4gIH1cblxuICByZXR1cm4gJyc7XG59O1xuXG5leHBvcnQgY29uc3QgcmVzaXplU2F2ZUltYWdlID0gKFxuICBjb250ZXh0OiBBcGlDb250ZXh0LFxuICBpbWdJZDogc3RyaW5nLFxuICBidWZmZXI6IEJ1ZmZlcixcbiAgdHlwZTogc3RyaW5nID0gJ2ltYWdlL2pwZWcnLCBzM09wdGlvbnM/OiBQdXRPYmplY3RSZXF1ZXN0KTogUHJvbWlzZTxJbWFnZVR5cGU+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAncmVzaXplU2F2ZUltYWdlJztcbiAgY29uc3Qge3N1Yn0gPSBjb250ZXh0O1xuICBjb25zdCBpbWdXOiBudW1iZXIgPSBDb25maWcuZ2V0KCdpbWFnZS5pbWdTaXplJyk7XG4gIGNvbnN0IGltZ1E6IG51bWJlciA9IENvbmZpZy5nZXQoJ2ltYWdlLmltZ1F1YWxpdHknKTtcbiAgbGV0IHBob3RvOiBJbWFnZVR5cGUgPSB7fTtcbiAgY29uc3QgZm9ybWF0OiBzdHJpbmcgPSAodHlwZS5zcGxpdCgnLycpKVsxXTtcblxuICByZXR1cm4gbmV3IFByb21pc2UoKHJlc29sdmUpID0+IHtcbiAgICBnbShidWZmZXIsICdpbWcnKVxuICAgICAgLnNldEZvcm1hdChmb3JtYXQpXG4gICAgICAucXVhbGl0eShpbWdRKVxuICAgICAgLmF1dG9PcmllbnQoKVxuICAgICAgLnJlc2l6ZShpbWdXLCBpbWdXLCAnPicpXG4gICAgICAuaWRlbnRpZnkoe2J1ZmZlclN0cmVhbTogdHJ1ZX0sIChlcnJvcjogRXJyb3IsIHZhbDogSW1hZ2VJZGVudGlmeVR5cGUgPSB7fSk6IGFueSA9PiB7XG4gICAgICAgIGlmKGVycm9yKSB7XG4gICAgICAgICAgbG9nRXJyb3Ioe1xuICAgICAgICAgICAgYWN0aW9uLFxuICAgICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgICBsYWJlbDogJ2ltYWdlX3NhdmUnLFxuICAgICAgICAgICAgdmFsdWU6ICdnbV9pbWFnZV9pZGVudGlmeSdcbiAgICAgICAgICB9LCBlcnJvciwgY29udGV4dCkuY2F0Y2goKGVycm9yKSA9PiB7XG4gICAgICAgICAgICB0aHJvdyBlcnJvcjtcbiAgICAgICAgICB9KTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBjb25zdCBmb3JtYXRWYWxzID0gbG93ZXJDYXNlS2V5cyh2YWwpO1xuICAgICAgICAgIGNvbnN0IHttYWtlOiBjYW1lcmFNYWtlLCBtb2RlbDogY2FtZXJhTW9kZWwsIGRhdGV0aW1lb3JpZ2luYWw6IHRha2VufTogSW1hZ2VJZGVudGlmeVR5cGUgPSBmb3JtYXRWYWxzO1xuICAgICAgICAgIHBob3RvID0ge1xuICAgICAgICAgICAgLi4uY2xvbmVEZWVwKHBob3RvKSxcbiAgICAgICAgICAgIG1ha2U6IGNhbWVyYU1ha2UsXG4gICAgICAgICAgICBtb2RlbDogY2FtZXJhTW9kZWwsXG4gICAgICAgICAgICB0YWtlbjogdGFrZW4gPyBEYXRlVGltZS5mcm9tTWlsbGlzKHRha2VuKS5taWxsaXNlY29uZCA6IG51bGxcbiAgICAgICAgICB9O1xuXG4gICAgICAgICAgLy8gSWYgbm8gYmFja2dyb3VuZCBjb2xvciwgZ2V0IHRoZSBtZWFuIGNvbG9yIHZhbHVlXG4gICAgICAgICAgY29uc3Qgc3RhdHMgPSBmb3JtYXRWYWxzWydjaGFubmVsIHN0YXRpc3RpY3MnXTtcblxuICAgICAgICAgIGlmKHN0YXRzKSB7XG4gICAgICAgICAgICBsZXQge3JlZCwgZ3JlZW4sIGJsdWUsIG1lYW59ID0gc3RhdHM7XG5cbiAgICAgICAgICAgIGlmKHJlZCkge1xuICAgICAgICAgICAgICBtZWFuID0gcmVkWydzdGFuZGFyZCBkZXZpYXRpb24nXSB8fCByZWQubWVhbjtcbiAgICAgICAgICAgICAgcmVkID0gbWVhbiA/ICsoKG1lYW4uc3BsaXQoJyAnKVswXSkuc3Vic3RyaW5nKDAsIDMpKSA6IDA7XG4gICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICByZWQgPSAwO1xuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICBpZihncmVlbikge1xuICAgICAgICAgICAgICBtZWFuID0gZ3JlZW5bJ3N0YW5kYXJkIGRldmlhdGlvbiddIHx8IGdyZWVuLm1lYW47XG4gICAgICAgICAgICAgIGdyZWVuID0gbWVhbiA/ICsoKG1lYW4uc3BsaXQoJyAnKVswXSkuc3Vic3RyaW5nKDAsIDMpKSA6IDA7XG4gICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICBncmVlbiA9IDA7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIGlmKGJsdWUpIHtcbiAgICAgICAgICAgICAgbWVhbiA9IGJsdWVbJ3N0YW5kYXJkIGRldmlhdGlvbiddIHx8IGJsdWUubWVhbjtcbiAgICAgICAgICAgICAgYmx1ZSA9IG1lYW4gPyArKChtZWFuLnNwbGl0KCcgJylbMF0pLnN1YnN0cmluZygwLCAzKSkgOiAwO1xuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgYmx1ZSA9IDA7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIGNvbnN0IHJnYiA9IGJsdWUgfCAoZ3JlZW4gPDwgOCkgfCAocmVkIDw8IDE2KTtcbiAgICAgICAgICAgIGNvbnN0IGNvbG9yID0gcmdiLnRvU3RyaW5nKDE2KTtcbiAgICAgICAgICAgIHBob3RvLmNvbG9yID0gY29sb3IgPT09ICcwJyA/ICcwMDAwMDAnIDogY29sb3I7XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICB9KVxuICAgICAgLnN0cmVhbSgoZXJyb3I6IEVycm9yLCBzdGRvdXQpOiBhbnkgPT4ge1xuICAgICAgICBpZihlcnJvcikge1xuICAgICAgICAgIGxvZ0Vycm9yKHtcbiAgICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgICAgaXNJbnRlcm5hbDogdHJ1ZSxcbiAgICAgICAgICAgIGxhYmVsOiAnaW1hZ2Vfc2F2ZScsXG4gICAgICAgICAgICB2YWx1ZTogJ2dtX2ltYWdlX3N0ZWFtJ1xuICAgICAgICAgIH0sIGVycm9yLCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIGxldCBpbWFnZUJ1ZmZlcjogQnVmZmVyID0gbmV3IEJ1ZmZlcignJyk7XG5cbiAgICAgICAgICBzdGRvdXQub24oJ2RhdGEnLCAoZGF0YSkgPT4ge1xuICAgICAgICAgICAgaW1hZ2VCdWZmZXIgPSBCdWZmZXIuY29uY2F0KFtpbWFnZUJ1ZmZlciwgZGF0YV0pO1xuICAgICAgICAgIH0pO1xuXG4gICAgICAgICAgc3Rkb3V0Lm9uKCdlbmQnLCAoKSA9PiB7XG4gICAgICAgICAgICAvLyBHZXQgZmlsZSBzaXplXG4gICAgICAgICAgICBwaG90by5maWxlU2l6ZSA9IGltYWdlQnVmZmVyLmxlbmd0aDtcblxuICAgICAgICAgICAgLy8gVXBsb2FkIGRhdGFcbiAgICAgICAgICAgIGNvbnN0IGltYWdlT2JqOiBQdXRPYmplY3RSZXF1ZXN0ID0ge1xuICAgICAgICAgICAgICBBQ0w6ICdhdXRoZW50aWNhdGVkLXJlYWQnLFxuICAgICAgICAgICAgICBCb2R5OiBpbWFnZUJ1ZmZlcixcbiAgICAgICAgICAgICAgQnVja2V0OiBudWxsLFxuICAgICAgICAgICAgICBDb250ZW50VHlwZTogdHlwZSxcbiAgICAgICAgICAgICAgS2V5OiBnZXRQYXRoVXNlckltYWdlcyhzdWIsIGltZ0lkLCB0eXBlLCAnaW1hZ2VzJyksXG4gICAgICAgICAgICAgIC4uLnMzT3B0aW9ucyB8fCB7fVxuICAgICAgICAgICAgfTtcblxuICAgICAgICAgICAgczNQdXQoaW1hZ2VPYmopXG4gICAgICAgICAgICAgIC50aGVuKCgpID0+IHtcbiAgICAgICAgICAgICAgICBjb25zdCB0aG1XID0gQ29uZmlnLmdldCgnaW1hZ2UudGhtU2l6ZScpO1xuICAgICAgICAgICAgICAgIGNvbnN0IHRobVEgPSBDb25maWcuZ2V0KCdpbWFnZS50aG1RdWFsaXR5Jyk7XG5cbiAgICAgICAgICAgICAgICAvLyBVcGxvYWQgdGh1bWJuYWlsXG4gICAgICAgICAgICAgICAgZ20oaW1hZ2VCdWZmZXIsICdpbWcnKVxuICAgICAgICAgICAgICAgICAgLnNldEZvcm1hdCgnanBlZycpXG4gICAgICAgICAgICAgICAgICAuc2l6ZSgodGh1bWJFcnJvcjogRXJyb3IsIHJlc2l6ZVZhbCkgPT4ge1xuICAgICAgICAgICAgICAgICAgICBpZighdGh1bWJFcnJvcikge1xuICAgICAgICAgICAgICAgICAgICAgIC8vIEdldCB1cGRhdGVkIHJlc2l6ZSB2YWx1ZXNcbiAgICAgICAgICAgICAgICAgICAgICBjb25zdCB7aGVpZ2h0LCB3aWR0aH0gPSByZXNpemVWYWw7XG4gICAgICAgICAgICAgICAgICAgICAgcGhvdG8gPSB7Li4uY2xvbmVEZWVwKHBob3RvKSwgaGVpZ2h0LCB3aWR0aH07XG4gICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgIH0pXG4gICAgICAgICAgICAgICAgICAuZ3Jhdml0eSgnQ2VudGVyJylcbiAgICAgICAgICAgICAgICAgIC5yZXNpemUodGhtVywgdGhtVywgJ14nKVxuICAgICAgICAgICAgICAgICAgLmV4dGVudCh0aG1XLCB0aG1XKVxuICAgICAgICAgICAgICAgICAgLnF1YWxpdHkodGhtUSlcbiAgICAgICAgICAgICAgICAgIC5zdHJlYW0oKHN0cmVhbUVycm9yOiBFcnJvciwgdGh1bWJTdGRvdXQpOiBhbnkgPT4ge1xuICAgICAgICAgICAgICAgICAgICBpZihzdHJlYW1FcnJvcikge1xuICAgICAgICAgICAgICAgICAgICAgIGxvZ0Vycm9yKHtcbiAgICAgICAgICAgICAgICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICAgICAgICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgICAgICAgICAgICAgICAgbGFiZWw6ICdpbWFnZV9zYXZlJyxcbiAgICAgICAgICAgICAgICAgICAgICAgIHZhbHVlOiAnZ21fdGh1bWJuYWlsX3N0ZWFtJ1xuICAgICAgICAgICAgICAgICAgICAgIH0sIHN0cmVhbUVycm9yLCBjb250ZXh0KVxuICAgICAgICAgICAgICAgICAgICAgICAgLmNhdGNoKChlcnJvcikgPT4ge1xuICAgICAgICAgICAgICAgICAgICAgICAgICB0aHJvdyBlcnJvcjtcbiAgICAgICAgICAgICAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICAgICAgICAgIGxldCB0aHVtYkJ1ZmZlcjogQnVmZmVyID0gbmV3IEJ1ZmZlcignJyk7XG5cbiAgICAgICAgICAgICAgICAgICAgICB0aHVtYlN0ZG91dC5vbignZGF0YScsIChkYXRhKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgICB0aHVtYkJ1ZmZlciA9IEJ1ZmZlci5jb25jYXQoW3RodW1iQnVmZmVyLCBkYXRhXSk7XG4gICAgICAgICAgICAgICAgICAgICAgfSk7XG5cbiAgICAgICAgICAgICAgICAgICAgICB0aHVtYlN0ZG91dC5vbignZW5kJywgKCkgPT4ge1xuICAgICAgICAgICAgICAgICAgICAgICAgLy8gVXBsb2FkIGRhdGFcbiAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IHRodW1iT2JqOiBQdXRPYmplY3RSZXF1ZXN0ID0ge1xuICAgICAgICAgICAgICAgICAgICAgICAgICBBQ0w6ICdhdXRoZW50aWNhdGVkLXJlYWQnLFxuICAgICAgICAgICAgICAgICAgICAgICAgICBCb2R5OiB0aHVtYkJ1ZmZlcixcbiAgICAgICAgICAgICAgICAgICAgICAgICAgQnVja2V0OiBudWxsLFxuICAgICAgICAgICAgICAgICAgICAgICAgICBDb250ZW50VHlwZTogdHlwZSxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgS2V5OiBnZXRQYXRoVXNlckltYWdlcyhzdWIsIGltZ0lkLCB0eXBlLCAndGh1bWJzJyksXG4gICAgICAgICAgICAgICAgICAgICAgICAgIC4uLnMzT3B0aW9ucyB8fCB7fVxuICAgICAgICAgICAgICAgICAgICAgICAgfTtcblxuICAgICAgICAgICAgICAgICAgICAgICAgczNQdXQodGh1bWJPYmopXG4gICAgICAgICAgICAgICAgICAgICAgICAgIC50aGVuKCgpID0+IHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICByZXNvbHZlKHBob3RvKTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgfSlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgLmNhdGNoKCgpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgbGFiZWw6ICdpbWFnZV9zYXZlJyxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB2YWx1ZTogJ3MzX3B1dF9pbWFnZSdcbiAgICAgICAgICAgICAgICAgICAgICAgICAgfSwgc3RyZWFtRXJyb3IsIGNvbnRleHQpXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgLmNhdGNoKChlcnJvcikgPT4ge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhyb3cgZXJyb3I7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgfSkpO1xuICAgICAgICAgICAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgICAgfSlcbiAgICAgICAgICAgICAgLmNhdGNoKCgpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgICAgICAgaXNJbnRlcm5hbDogdHJ1ZSxcbiAgICAgICAgICAgICAgICBsYWJlbDogJ2ltYWdlX3NhdmUnLFxuICAgICAgICAgICAgICAgIHZhbHVlOiAnczNfaW1hZ2Vfc2F2ZSdcbiAgICAgICAgICAgICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCkpO1xuICAgICAgICAgIH0pO1xuICAgICAgICB9XG4gICAgICB9KTtcbiAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgYWRkSW1hZ2UgPSAoY29udGV4dDogQXBpQ29udGV4dCwgaW1hZ2U6IEltYWdlVHlwZSwgczNPcHRpb25zOiBQdXRPYmplY3RSZXF1ZXN0KTogUHJvbWlzZTxJbWFnZVR5cGU+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnYWRkSW1hZ2UnO1xuICBjb25zdCB7ZGF0YWJhc2UsIHN1YiwgdXNlcklkOiBzZXNzaW9uSWR9ID0gY29udGV4dDtcbiAgY29uc3Qge2ltZ0lkLCBkZXNjcmlwdGlvbiwgYnVmZmVyLCBmaWxlVHlwZX0gPSBpbWFnZTtcbiAgY29uc3Qgbm93OiBudW1iZXIgPSBEYXRlLm5vdygpO1xuXG4gIHJldHVybiByZXNpemVTYXZlSW1hZ2UoY29udGV4dCwgaW1nSWQsIGJ1ZmZlciwgZmlsZVR5cGUsIHMzT3B0aW9ucylcbiAgICAudGhlbigocmVzaXplZEltYWdlOiBhbnkpID0+IHtcbiAgICAgIGNvbnN0IGluc2VydDogSW1hZ2VUeXBlID0ge1xuICAgICAgICAuLi5yZXNpemVkSW1hZ2UsXG4gICAgICAgIF9rZXk6IGltZ0lkLFxuICAgICAgICBhZGRlZDogbm93LFxuICAgICAgICBkZXNjcmlwdGlvbixcbiAgICAgICAgZmlsZVR5cGUsXG4gICAgICAgIG1vZGlmaWVkOiBub3csXG4gICAgICAgIHN1YixcbiAgICAgICAgdXNlcklkOiBzZXNzaW9uSWRcbiAgICAgIH07XG5cbiAgICAgIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgSU5TRVJUICR7aW5zZXJ0fSBJTiBpbWFnZXMgUkVUVVJOIE5FV2A7XG5cbiAgICAgIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgICAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAgICAgLnRoZW4oZGVmYXVsdE9iamVjdClcbiAgICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgaXNJbnRlcm5hbDogdHJ1ZSxcbiAgICAgICAgICBsYWJlbDogJ2RiX2Vycm9yJ1xuICAgICAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKSk7XG4gICAgfSlcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgYWN0aW9uLFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICBpc0ludGVybmFsOiB0cnVlLFxuICAgICAgbGFiZWw6ICdpbWFnZV9yZXNpemUnXG4gICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCkpO1xufTtcblxuZXhwb3J0IGNvbnN0IGFkZEltYWdlRWRnZSA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBpbWFnZUVkZ2U6IEltYWdlRWRnZVR5cGUpOiBQcm9taXNlPG9iamVjdD4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdhZGRJbWFnZUVkZ2UnO1xuICBjb25zdCB7ZGF0YWJhc2UsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG4gIGNvbnN0IHtpbWdJZCwgaXRlbUlkLCBpdGVtVHlwZX0gPSBpbWFnZUVkZ2U7XG4gIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcbiAgY29uc3QgZWRnZUNvbGxlY3Rpb24gPSB1c2VEYihkYXRhYmFzZSkuZWRnZUNvbGxlY3Rpb24oJ2hhc0ltYWdlJyk7XG4gIGNvbnN0IGVkZ2VJZDogc3RyaW5nID0gY3JlYXRlSGFzaChgaGFzSW1hZ2UtJHtpbWdJZH0tJHtpdGVtSWR9LSR7c2Vzc2lvbklkfWApO1xuICBjb25zdCBlZGdlOiBhbnkgPSB7XG4gICAgX2tleTogZWRnZUlkLFxuICAgIGFkZGVkOiBub3csXG4gICAgdHlwZTogaXRlbVR5cGVcbiAgfTtcblxuICBjb25zdCBmb3JtYXRJdGVtVHlwZTogc3RyaW5nID0gcGFyc2VDaGFyKGl0ZW1UeXBlKS50b0xvd2VyQ2FzZSgpO1xuICBjb25zdCBmb3JtYXRJdGVtSWQ6IHN0cmluZyA9IHBhcnNlSWQoaXRlbUlkKTtcbiAgbGV0IGl0ZW1Eb2NJZDogc3RyaW5nO1xuICBjb25zdCBmb3JtYXRJbWdJZDogc3RyaW5nID0gcGFyc2VJZChpbWdJZCk7XG4gIGNvbnN0IGZpbGVEb2NJZDogc3RyaW5nID0gYGltYWdlcy8ke2Zvcm1hdEltZ0lkfWA7XG5cbiAgc3dpdGNoKGZvcm1hdEl0ZW1UeXBlKSB7XG4gICAgY2FzZSAncG9zdHMnOlxuICAgICAgaXRlbURvY0lkID0gYHBvc3RzLyR7Zm9ybWF0SXRlbUlkfWA7XG4gICAgICBicmVhaztcbiAgICBjYXNlICdwcm9maWxlJzpcbiAgICAgIGl0ZW1Eb2NJZCA9IGB1c2Vycy8ke2Zvcm1hdEl0ZW1JZH1gO1xuICAgICAgYnJlYWs7XG4gICAgZGVmYXVsdDpcbiAgICAgIGl0ZW1Eb2NJZCA9ICcnO1xuICAgICAgYnJlYWs7XG4gIH1cblxuICBpZihpdGVtRG9jSWQpIHtcbiAgICByZXR1cm4gZWRnZUNvbGxlY3Rpb24uc2F2ZShlZGdlLCBmaWxlRG9jSWQsIGl0ZW1Eb2NJZClcbiAgICAgIC50aGVuKChmaWxlRWRnZSkgPT4gZWRnZUNvbGxlY3Rpb24uZWRnZShmaWxlRWRnZSkpXG4gICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgICBhY3Rpb24sXG4gICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICBpc0ludGVybmFsOiB0cnVlLFxuICAgICAgICBsYWJlbDogJ2RiX2Vycm9yJ1xuICAgICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCkpO1xuICB9XG5cbiAgcmV0dXJuIFByb21pc2UucmVzb2x2ZSh7fSk7XG59O1xuXG5leHBvcnQgY29uc3QgdXBkYXRlSW1hZ2UgPSAoY29udGV4dDogQXBpQ29udGV4dCwgaXRlbTogSW1hZ2VUeXBlLCBzM09wdGlvbnM6IFB1dE9iamVjdFJlcXVlc3QpOiBQcm9taXNlPEltYWdlVHlwZT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICd1cGRhdGUnO1xuICBjb25zdCB7ZGF0YWJhc2UsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG5cbiAgLy8gSXRlbSBwcm9wc1xuICBjb25zdCB7XG4gICAgYmFzZTY0ID0gJycsXG4gICAgZGVzY3JpcHRpb24gPSAnJyxcbiAgICBpbWdJZCxcbiAgICBpdGVtSWQsXG4gICAgaXRlbVR5cGUsXG4gICAgZmlsZVR5cGUsXG4gICAgdXJsID0gJydcbiAgfSA9IGl0ZW07XG5cbiAgLy8gU2F2ZSBCYXNlNjQgZGF0YVxuICBjb25zdCBub3c6IG51bWJlciA9IERhdGUubm93KCk7XG4gIGNvbnN0IGZvcm1hdEltZ0lkOiBzdHJpbmcgPSBpbWdJZCB8fCBjcmVhdGVIYXNoKGBpbWFnZS0ke3Nlc3Npb25JZH1gKTtcbiAgY29uc3QgZm9ybWF0SXRlbUlkOiBzdHJpbmcgPSBwYXJzZUlkKGl0ZW1JZCk7XG4gIGNvbnN0IGZvcm1hdEl0ZW1UeXBlOiBzdHJpbmcgPSBwYXJzZUNoYXIoaXRlbVR5cGUsIDE2KS50b0xvd2VyQ2FzZSgpO1xuICBjb25zdCBjdXN0b21QYXJhbXM6IEltYWdlVHlwZSA9IHtcbiAgICBkZXNjcmlwdGlvbixcbiAgICBpbWdJZDogZm9ybWF0SW1nSWQsXG4gICAgdXNlcklkOiBzZXNzaW9uSWRcbiAgfTtcblxuICBpZihiYXNlNjQgIT09ICcnKSB7XG4gICAgLy8gUGFyc2UgQmFzZTY0IGRhdGFcbiAgICBjb25zdCBmb3JtYXRCYXNlNjQ6IHN0cmluZyA9IGJhc2U2NC5zdWJzdHIoYmFzZTY0LmluZGV4T2YoJywnKSArIDEpO1xuICAgIGNvbnN0IGJ1ZmZlciA9IG5ldyBCdWZmZXIoZm9ybWF0QmFzZTY0LCAnYmFzZTY0Jyk7XG4gICAgbGV0IHVwZGF0ZWRGaWxlVHlwZSA9IGZpbGVUeXBlO1xuXG4gICAgaWYoIWZpbGVUeXBlKSB7XG4gICAgICBjb25zdCBmaWxlTWltZTogRmlsZVR5cGVSZXN1bHQgPSBnZXRGaWxlVHlwZShidWZmZXIpO1xuICAgICAgY29uc3Qge21pbWV9ID0gZmlsZU1pbWU7XG4gICAgICB1cGRhdGVkRmlsZVR5cGUgPSBtaW1lO1xuICAgIH1cblxuICAgIGNvbnN0IGltZ1BhcmFtczogSW1hZ2VUeXBlID0ge1xuICAgICAgLi4uY3VzdG9tUGFyYW1zLFxuICAgICAgYnVmZmVyLFxuICAgICAgZmlsZVR5cGU6IHVwZGF0ZWRGaWxlVHlwZVxuICAgIH07XG5cbiAgICByZXR1cm4gYWRkSW1hZ2UoY29udGV4dCwgaW1nUGFyYW1zLCBzM09wdGlvbnMpXG4gICAgICAudGhlbigoaW1hZ2U6IEltYWdlVHlwZSkgPT4ge1xuICAgICAgICBpZihmb3JtYXRJdGVtSWQgJiYgZm9ybWF0SXRlbVR5cGUpIHtcbiAgICAgICAgICBjb25zdCBpbWFnZUVkZ2U6IEltYWdlRWRnZVR5cGUgPSB7XG4gICAgICAgICAgICBpbWdJZDogZm9ybWF0SW1nSWQsXG4gICAgICAgICAgICBpdGVtSWQ6IGZvcm1hdEl0ZW1JZCxcbiAgICAgICAgICAgIGl0ZW1UeXBlOiBmb3JtYXRJdGVtVHlwZVxuICAgICAgICAgIH07XG4gICAgICAgICAgcmV0dXJuIGFkZEltYWdlRWRnZShjb250ZXh0LCBpbWFnZUVkZ2UpLnRoZW4oKCkgPT4gaW1hZ2UpO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiBpbWFnZTtcbiAgICAgIH0pXG4gICAgICAuY2F0Y2goKGVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgIGFjdGlvbixcbiAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgIGxhYmVsOiAnaW1hZ2Vfc2F2ZV9lcnJvcidcbiAgICAgIH0sIGVycm9yLCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpKTtcbiAgfSBlbHNlIGlmKHVybCAhPT0gJycpIHtcbiAgICAvLyBEb3dubG9hZCBpbWFnZSBmcm9tIHRoZSB3ZWJcbiAgICBsZXQgY29udGVudFR5cGU6IHN0cmluZztcblxuICAgIHJldHVybiBodHRwR2V0KHVybClcbiAgICAgIC50aGVuKChyZXM6IFJlc3BvbnNlKSA9PiB7XG4gICAgICAgIGlmKHJlcy5zdGF0dXMgIT09IDIwMCkge1xuICAgICAgICAgIHJldHVybiBsb2dFeGNlcHRpb24oe1xuICAgICAgICAgICAgYWN0aW9uLFxuICAgICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgICBsYWJlbDogJ2ZldGNoX2ltYWdlX3VybCcsXG4gICAgICAgICAgICB2YWx1ZTogcmVzLnN0YXR1c1RleHRcbiAgICAgICAgICB9LCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpO1xuICAgICAgICB9XG5cbiAgICAgICAgY29udGVudFR5cGUgPSByZXMuaGVhZGVycy5nZXQoJ2NvbnRlbnQtdHlwZScpO1xuXG4gICAgICAgIHJldHVybiByZXM7XG4gICAgICB9KVxuICAgICAgLnRoZW4oKHJlcykgPT4gcmVzLmJ1ZmZlcigpKVxuICAgICAgLnRoZW4oKGJ1ZmZlcjogQnVmZmVyKSA9PiB7XG4gICAgICAgIGNvbnN0IGltZ1BhcmFtczogSW1hZ2VUeXBlID0ge1xuICAgICAgICAgIC4uLmN1c3RvbVBhcmFtcyxcbiAgICAgICAgICBidWZmZXIsXG4gICAgICAgICAgZmlsZVR5cGU6IGNvbnRlbnRUeXBlXG4gICAgICAgIH07XG5cbiAgICAgICAgcmV0dXJuIGFkZEltYWdlKGNvbnRleHQsIGltZ1BhcmFtcywgczNPcHRpb25zKS50aGVuKChpbWFnZTogSW1hZ2VUeXBlKSA9PiB7XG4gICAgICAgICAgaWYoZm9ybWF0SXRlbUlkICYmIGZvcm1hdEl0ZW1UeXBlKSB7XG4gICAgICAgICAgICBjb25zdCBpbWFnZUVkZ2U6IEltYWdlRWRnZVR5cGUgPSB7aW1nSWQ6IGZvcm1hdEltZ0lkLCBpdGVtSWQ6IGZvcm1hdEl0ZW1JZCwgaXRlbVR5cGU6IGZvcm1hdEl0ZW1UeXBlfTtcbiAgICAgICAgICAgIHJldHVybiBhZGRJbWFnZUVkZ2UoY29udGV4dCwgaW1hZ2VFZGdlKS50aGVuKCgpID0+IGltYWdlKTtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICByZXR1cm4gaW1hZ2U7XG4gICAgICAgIH0pO1xuICAgICAgfSlcbiAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgIGFjdGlvbixcbiAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgIGxhYmVsOiAnZmV0Y2hfZXJyb3InXG4gICAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKSk7XG4gIH0gZWxzZSBpZihpbWdJZCAhPT0gJycpIHtcbiAgICAvLyBVcGRhdGUgbWV0YWRhdGFcbiAgICBjb25zdCB1cGRhdGU6IGFueSA9IHtcbiAgICAgIGRlc2NyaXB0aW9uLFxuICAgICAgbW9kaWZpZWQ6IG5vd1xuICAgIH07XG4gICAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBVUERBVEUgJHtmb3JtYXRJbWdJZH0gV0lUSCAke3VwZGF0ZX0gSU4gaW1hZ2VzIFJFVFVSTiBORVdgO1xuXG4gICAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgIGFjdGlvbixcbiAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgIGxhYmVsOiAnZGJfZXJyb3InXG4gICAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKSk7XG4gIH1cblxuICByZXR1cm4gbnVsbDtcbn07XG5cbmV4cG9ydCBjb25zdCBkZWxldGVJbWFnZSA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBpbWdJZDogc3RyaW5nKTogUHJvbWlzZTxJbWFnZVR5cGU+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnZGVsZXRlJztcbiAgY29uc3Qge2RhdGFiYXNlLCB1c2VySWQ6IHNlc3Npb25JZH0gPSBjb250ZXh0O1xuICBjb25zdCBmb3JtYXRJbWdJZCA9IHBhcnNlSWQoaW1nSWQpO1xuXG4gIGNvbnN0IGFxbFFyeSA9IGFxbGBGT1IgaSBJTiBpbWFnZXNcbiAgICBGSUxURVIgaS5fa2V5ID09ICR7Zm9ybWF0SW1nSWR9ICYmIGkudXNlcklkID09ICR7c2Vzc2lvbklkfVxuICAgIFJFTU9WRSBpIElOIGltYWdlc1xuICAgIFJFVFVSTiBPTERgO1xuXG4gIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgIC50aGVuKChpbWFnZTogSW1hZ2VUeXBlID0ge30pID0+IHtcbiAgICAgIGlmKCFpc0VtcHR5KGltYWdlKSkge1xuICAgICAgICBjb25zdCB7X2tleTogaW1hZ2VLZXl9ID0gaW1hZ2U7XG4gICAgICAgIGNvbnN0IHBhcmFtczogRGVsZXRlT2JqZWN0c1JlcXVlc3QgPSB7XG4gICAgICAgICAgQnVja2V0OiBudWxsLFxuICAgICAgICAgIERlbGV0ZToge1xuICAgICAgICAgICAgT2JqZWN0czogW1xuICAgICAgICAgICAgICB7S2V5OiBgdXNlcnMvJHtzZXNzaW9uSWR9L2ltYWdlcy8ke2ltYWdlS2V5fS5qcGdgfSxcbiAgICAgICAgICAgICAge0tleTogYHVzZXJzLyR7c2Vzc2lvbklkfS90aHVtYnMvJHtpbWFnZUtleX0uanBnYH1cbiAgICAgICAgICAgIF0sXG4gICAgICAgICAgICBRdWlldDogdHJ1ZVxuICAgICAgICAgIH1cbiAgICAgICAgfTtcblxuICAgICAgICByZXR1cm4gczNEZWxldGVMaXN0KHBhcmFtcykudGhlbigoKSA9PiBpbWFnZSk7XG4gICAgICB9XG4gICAgICByZXR1cm4ge307XG4gICAgfSlcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgYWN0aW9uLFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICBsYWJlbDogJ2RiX2Vycm9yJ1xuICAgIH0sIGVycm9yLCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpKTtcbn07XG4iXX0=