@nlabs/reaktor 0.1.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 (154) hide show
  1. package/.vscode/extensions.json +15 -0
  2. package/.vscode/settings.json +82 -0
  3. package/README.md +211 -0
  4. package/index.d.ts +1 -0
  5. package/index.js +5 -0
  6. package/lex.config.js +4 -0
  7. package/lib/config.d.ts +21 -0
  8. package/lib/config.js +127 -0
  9. package/lib/data/conversations.d.ts +6 -0
  10. package/lib/data/conversations.js +201 -0
  11. package/lib/data/dynamodb.d.ts +8 -0
  12. package/lib/data/dynamodb.js +139 -0
  13. package/lib/data/email.d.ts +7 -0
  14. package/lib/data/email.js +164 -0
  15. package/lib/data/files.d.ts +16 -0
  16. package/lib/data/files.js +407 -0
  17. package/lib/data/groups.d.ts +13 -0
  18. package/lib/data/groups.js +354 -0
  19. package/lib/data/images.d.ts +12 -0
  20. package/lib/data/images.js +668 -0
  21. package/lib/data/index.d.ts +19 -0
  22. package/lib/data/index.js +24 -0
  23. package/lib/data/ios.d.ts +6 -0
  24. package/lib/data/ios.js +302 -0
  25. package/lib/data/locations.d.ts +3 -0
  26. package/lib/data/locations.js +132 -0
  27. package/lib/data/messages.d.ts +9 -0
  28. package/lib/data/messages.js +248 -0
  29. package/lib/data/notifications.d.ts +5 -0
  30. package/lib/data/notifications.js +42 -0
  31. package/lib/data/payments.d.ts +11 -0
  32. package/lib/data/payments.js +748 -0
  33. package/lib/data/posts.d.ts +14 -0
  34. package/lib/data/posts.js +458 -0
  35. package/lib/data/reactions.d.ts +6 -0
  36. package/lib/data/reactions.js +218 -0
  37. package/lib/data/s3.d.ts +6 -0
  38. package/lib/data/s3.js +103 -0
  39. package/lib/data/search.d.ts +3 -0
  40. package/lib/data/search.js +98 -0
  41. package/lib/data/sms.d.ts +3 -0
  42. package/lib/data/sms.js +59 -0
  43. package/lib/data/subscription.d.ts +7 -0
  44. package/lib/data/subscription.js +284 -0
  45. package/lib/data/tags.d.ts +14 -0
  46. package/lib/data/tags.js +304 -0
  47. package/lib/data/users.d.ts +12 -0
  48. package/lib/data/users.js +312 -0
  49. package/lib/index.d.ts +3 -0
  50. package/lib/index.js +8 -0
  51. package/lib/types/apps.d.ts +44 -0
  52. package/lib/types/apps.js +2 -0
  53. package/lib/types/arangodb.d.ts +17 -0
  54. package/lib/types/arangodb.js +2 -0
  55. package/lib/types/auth.d.ts +9 -0
  56. package/lib/types/auth.js +2 -0
  57. package/lib/types/conversations.d.ts +6 -0
  58. package/lib/types/conversations.js +2 -0
  59. package/lib/types/email.d.ts +12 -0
  60. package/lib/types/email.js +2 -0
  61. package/lib/types/files.d.ts +28 -0
  62. package/lib/types/files.js +2 -0
  63. package/lib/types/google.d.ts +27 -0
  64. package/lib/types/google.js +2 -0
  65. package/lib/types/groups.d.ts +22 -0
  66. package/lib/types/groups.js +2 -0
  67. package/lib/types/images.d.ts +25 -0
  68. package/lib/types/images.js +2 -0
  69. package/lib/types/index.d.ts +17 -0
  70. package/lib/types/index.js +22 -0
  71. package/lib/types/locations.d.ts +21 -0
  72. package/lib/types/locations.js +2 -0
  73. package/lib/types/messages.d.ts +12 -0
  74. package/lib/types/messages.js +2 -0
  75. package/lib/types/notifications.d.ts +19 -0
  76. package/lib/types/notifications.js +2 -0
  77. package/lib/types/payments.d.ts +119 -0
  78. package/lib/types/payments.js +2 -0
  79. package/lib/types/posts.d.ts +20 -0
  80. package/lib/types/posts.js +2 -0
  81. package/lib/types/reactions.d.ts +4 -0
  82. package/lib/types/reactions.js +2 -0
  83. package/lib/types/tags.d.ts +10 -0
  84. package/lib/types/tags.js +2 -0
  85. package/lib/types/users.d.ts +78 -0
  86. package/lib/types/users.js +2 -0
  87. package/lib/utils/analytics.d.ts +3 -0
  88. package/lib/utils/analytics.js +47 -0
  89. package/lib/utils/arangodb.d.ts +9 -0
  90. package/lib/utils/arangodb.js +98 -0
  91. package/lib/utils/auth.d.ts +2 -0
  92. package/lib/utils/auth.js +43 -0
  93. package/lib/utils/index.d.ts +5 -0
  94. package/lib/utils/index.js +10 -0
  95. package/lib/utils/objects.d.ts +3 -0
  96. package/lib/utils/objects.js +34 -0
  97. package/lib/utils/redis.d.ts +1 -0
  98. package/lib/utils/redis.js +15 -0
  99. package/package.json +75 -0
  100. package/src/config.ts +121 -0
  101. package/src/data/conversations.ts +183 -0
  102. package/src/data/dynamodb.ts +157 -0
  103. package/src/data/email.ts +164 -0
  104. package/src/data/files.ts +352 -0
  105. package/src/data/groups.ts +308 -0
  106. package/src/data/images.ts +606 -0
  107. package/src/data/index.ts +23 -0
  108. package/src/data/ios.ts +249 -0
  109. package/src/data/locations.ts +114 -0
  110. package/src/data/messages.ts +237 -0
  111. package/src/data/notifications.ts +48 -0
  112. package/src/data/payments.ts +675 -0
  113. package/src/data/posts.ts +508 -0
  114. package/src/data/reactions.ts +186 -0
  115. package/src/data/s3.ts +117 -0
  116. package/src/data/search.ts +74 -0
  117. package/src/data/sms.ts +60 -0
  118. package/src/data/subscription.ts +228 -0
  119. package/src/data/tags.ts +230 -0
  120. package/src/data/users.ts +256 -0
  121. package/src/index.ts +7 -0
  122. package/src/types/apps.ts +57 -0
  123. package/src/types/arangodb.ts +23 -0
  124. package/src/types/auth.ts +19 -0
  125. package/src/types/conversations.ts +11 -0
  126. package/src/types/email.ts +17 -0
  127. package/src/types/files.ts +33 -0
  128. package/src/types/google.ts +37 -0
  129. package/src/types/groups.ts +28 -0
  130. package/src/types/images.ts +33 -0
  131. package/src/types/index.ts +21 -0
  132. package/src/types/locations.ts +25 -0
  133. package/src/types/messages.ts +16 -0
  134. package/src/types/notifications.ts +26 -0
  135. package/src/types/payments.ts +134 -0
  136. package/src/types/posts.ts +25 -0
  137. package/src/types/reactions.ts +8 -0
  138. package/src/types/tags.ts +14 -0
  139. package/src/types/users.ts +89 -0
  140. package/src/utils/analytics.ts +41 -0
  141. package/src/utils/arangodb.ts +100 -0
  142. package/src/utils/auth.ts +28 -0
  143. package/src/utils/index.ts +9 -0
  144. package/src/utils/objects.ts +34 -0
  145. package/src/utils/redis.ts +17 -0
  146. package/templates/email/layout.html +279 -0
  147. package/templates/email/passwordForgot.html +15 -0
  148. package/templates/email/passwordRecovery.html +12 -0
  149. package/templates/email/verifyEmail.html +15 -0
  150. package/templates/sms/passwordForgot.txt +1 -0
  151. package/templates/sms/passwordRecovery.txt +1 -0
  152. package/templates/sms/verifyEmail.txt +1 -0
  153. package/templates/sms/verifyPhone.txt +1 -0
  154. package/tsconfig.json +45 -0
@@ -0,0 +1,668 @@
1
+ function _templateObject4() {
2
+ var data = _taggedTemplateLiteral(["FOR i IN images\n FILTER i._key == ", " && i.userId == ", "\n REMOVE i IN images\n RETURN OLD"]);
3
+
4
+ _templateObject4 = function _templateObject4() {
5
+ return data;
6
+ };
7
+
8
+ return data;
9
+ }
10
+
11
+ function _templateObject3() {
12
+ var data = _taggedTemplateLiteral(["INSERT ", " IN images RETURN NEW"]);
13
+
14
+ _templateObject3 = function _templateObject3() {
15
+ return data;
16
+ };
17
+
18
+ return data;
19
+ }
20
+
21
+ 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; }
22
+
23
+ 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; }
24
+
25
+ function _templateObject2() {
26
+ var data = _taggedTemplateLiteral(["UPDATE ", " WITH ", " IN images RETURN NEW"]);
27
+
28
+ _templateObject2 = function _templateObject2() {
29
+ return data;
30
+ };
31
+
32
+ return data;
33
+ }
34
+
35
+ function _templateObject() {
36
+ var data = _taggedTemplateLiteral(["FOR i IN images\n FILTER i._key==", "\n LIMIT 1\n RETURN i"]);
37
+
38
+ _templateObject = function _templateObject() {
39
+ return data;
40
+ };
41
+
42
+ return data;
43
+ }
44
+
45
+ function _taggedTemplateLiteral(strings, raw) { if (!raw) { raw = strings.slice(0); } return Object.freeze(Object.defineProperties(strings, { raw: { value: Object.freeze(raw) } })); }
46
+
47
+ import { get as httpGet } from '@nlabs/rip-hunter';
48
+ import { createHash, parseChar, parseId, parseNum } from '@nlabs/utils';
49
+ import { aql } from 'arangojs';
50
+ import * as gm from 'gm';
51
+ import cloneDeep from 'lodash/cloneDeep';
52
+ import isEmpty from 'lodash/isEmpty';
53
+ import { DateTime } from 'luxon';
54
+ import { Config } from '../config';
55
+ import { defaultObject, getLimit, logError, logException, lowerCaseKeys, useDb } from '../utils';
56
+ import { decodeBase64 } from './files';
57
+ import { getGroupDetails, isGrouped } from './groups';
58
+ import { s3DeleteList, s3Put } from './s3';
59
+ /**
60
+ * Copyright (c) 2019-Present, Nitrogen Labs, Inc.
61
+ * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.
62
+ */
63
+
64
+ var eventCategory = 'images';
65
+ export var getImageListByUser = function getImageListByUser(context, userId, from, to) {
66
+ var action = 'getListByUser';
67
+ var database = context.database;
68
+ var formatUserId = parseId(userId);
69
+ var limit = getLimit(from, to);
70
+ var aqlQry = "FOR i IN images\n FILTER i.userId == \"".concat(formatUserId, "\"\n LET user = (\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: FIRST(user)})");
71
+ return useDb(database).query(aqlQry).then(function (cursor) {
72
+ return cursor.all();
73
+ }).catch(function (error) {
74
+ return logError({
75
+ action: action,
76
+ category: eventCategory,
77
+ label: 'db_error'
78
+ }, error, context).then(function () {
79
+ return null;
80
+ });
81
+ });
82
+ };
83
+ export var getImageListByGroup = function getImageListByGroup(context, params) {
84
+ var action = 'getListByGroup';
85
+ var database = context.database,
86
+ sessionId = context.userId;
87
+ var _params$filters = params.filters,
88
+ filters = _params$filters === void 0 ? [] : _params$filters,
89
+ groupId = params.groupId,
90
+ from = params.from,
91
+ to = params.to;
92
+ var formatGroupId = parseId(groupId);
93
+ var limit = getLimit(from, to);
94
+ filters.map(function (filter) {
95
+ var conditional = filter.conditional,
96
+ name = filter.name,
97
+ value = filter.value;
98
+ var formatCond = conditional;
99
+
100
+ if (conditional !== '>=' && conditional !== '<=' && conditional !== '>' && conditional !== '<') {
101
+ formatCond = '==';
102
+ }
103
+
104
+ switch (name) {
105
+ case 'added':
106
+ return "p.added ".concat(formatCond, " ").concat(parseNum(value));
107
+
108
+ default:
109
+ return '';
110
+ }
111
+ });
112
+ return getGroupDetails(context, formatGroupId).then(function (group) {
113
+ if (group.privacy === 'public') {
114
+ filters.push("p.groupId == \"".concat(groupId, "\""));
115
+ var filterStr = filters.join(' && ');
116
+ 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");
117
+ return useDb(database).query(aqlQry).then(function (cursor) {
118
+ return cursor.all();
119
+ }).catch(function (error) {
120
+ return logError({
121
+ action: action,
122
+ category: eventCategory,
123
+ label: 'db_error'
124
+ }, error, context).then(function () {
125
+ return null;
126
+ });
127
+ });
128
+ }
129
+
130
+ return isGrouped(database, sessionId, groupId).then(function (grouped) {
131
+ if (grouped.isValid) {
132
+ filters.push("p.groupId == \"".concat(grouped.groupId, "\""));
133
+ var filterList = filters.join(' && ');
134
+
135
+ 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");
136
+
137
+ return useDb(database).query(_aqlQry).then(function (cursor) {
138
+ return cursor.all();
139
+ }).catch(function (error) {
140
+ return logError({
141
+ action: action,
142
+ category: eventCategory,
143
+ label: 'db_error'
144
+ }, error, context).then(function () {
145
+ return null;
146
+ });
147
+ });
148
+ }
149
+
150
+ return [];
151
+ });
152
+ });
153
+ };
154
+ export var getImage = function getImage(context, id) {
155
+ var action = 'getItem';
156
+ var database = context.database;
157
+ var formatId = parseId(id);
158
+ var aqlQry = aql(_templateObject(), formatId);
159
+ return useDb(database).query(aqlQry).then(function (cursor) {
160
+ return cursor.next();
161
+ }).then(function () {
162
+ var image = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
163
+ return image;
164
+ }).catch(function (error) {
165
+ return logError({
166
+ action: action,
167
+ category: eventCategory,
168
+ label: 'db_error'
169
+ }, error, context).then(function () {
170
+ return null;
171
+ });
172
+ });
173
+ };
174
+ export var updateImage = function updateImage(context, item) {
175
+ var action = 'update';
176
+ var database = context.database,
177
+ sessionId = context.userId; // Item props
178
+
179
+ var _item$base = item.base64,
180
+ base64 = _item$base === void 0 ? '' : _item$base,
181
+ _item$description = item.description,
182
+ description = _item$description === void 0 ? '' : _item$description,
183
+ _item$fileType = item.fileType,
184
+ fileType = _item$fileType === void 0 ? '' : _item$fileType,
185
+ id = item.id,
186
+ itemId = item.itemId,
187
+ itemType = item.itemType,
188
+ _item$url = item.url,
189
+ url = _item$url === void 0 ? '' : _item$url; // Save Base64 data
190
+
191
+ var photoId = id || createHash("image-".concat(sessionId));
192
+ var now = Date.now();
193
+ var contentType = fileType;
194
+ var formatItemId = parseId(itemId);
195
+ var formatItemType = parseChar(itemType).toLowerCase();
196
+
197
+ if (base64 !== '') {
198
+ var decodedBase64;
199
+
200
+ try {
201
+ decodedBase64 = decodeBase64(base64);
202
+ } catch (error) {
203
+ return logError({
204
+ action: action,
205
+ category: eventCategory,
206
+ label: 'decode_base64'
207
+ }, error, context).then(function () {
208
+ return null;
209
+ });
210
+ }
211
+
212
+ var _decodedBase = decodedBase64,
213
+ data = _decodedBase.data,
214
+ type = _decodedBase.type;
215
+ var imgParams = {
216
+ buffer: data,
217
+ description: description,
218
+ fileType: type,
219
+ id: photoId,
220
+ userId: sessionId
221
+ };
222
+ return addImage(context, imgParams).then(function (image) {
223
+ if (formatItemId && formatItemType) {
224
+ var imageEdge = {
225
+ imgId: photoId,
226
+ itemId: formatItemId,
227
+ itemType: formatItemType
228
+ };
229
+ return addImageEdge(context, imageEdge).then(function () {
230
+ return image;
231
+ });
232
+ }
233
+
234
+ return image;
235
+ }).catch(function (error) {
236
+ return logError({
237
+ action: action,
238
+ category: eventCategory,
239
+ label: 'image_save_error'
240
+ }, error, context).then(function () {
241
+ return null;
242
+ });
243
+ });
244
+ } else if (url !== '') {
245
+ // Download image from the web
246
+ return httpGet(url).then(function (res) {
247
+ if (res.status !== 200) {
248
+ return logException({
249
+ action: action,
250
+ category: eventCategory,
251
+ label: 'fetch_image_url',
252
+ value: res.statusText
253
+ }, context).then(function () {
254
+ return null;
255
+ });
256
+ }
257
+
258
+ contentType = res.headers.get('content-type');
259
+ return res;
260
+ }).then(function (res) {
261
+ return res.buffer();
262
+ }).then(function (buffer) {
263
+ var imgParams = {
264
+ buffer: buffer,
265
+ description: description,
266
+ id: photoId,
267
+ type: contentType,
268
+ userId: sessionId
269
+ };
270
+ return addImage(context, imgParams).then(function (image) {
271
+ if (formatItemId && formatItemType) {
272
+ var imageEdge = {
273
+ imgId: id,
274
+ itemId: formatItemId,
275
+ itemType: formatItemType
276
+ };
277
+ return addImageEdge(context, imageEdge).then(function () {
278
+ return image;
279
+ });
280
+ }
281
+
282
+ return image;
283
+ });
284
+ }).catch(function (error) {
285
+ return logError({
286
+ action: action,
287
+ category: eventCategory,
288
+ label: 'fetch_error'
289
+ }, error, context).then(function () {
290
+ return null;
291
+ });
292
+ });
293
+ } else if (id !== '') {
294
+ // Update metadata
295
+ var update = {
296
+ description: description,
297
+ modified: now
298
+ };
299
+ var aqlQry = aql(_templateObject2(), id, update);
300
+ return useDb(database).query(aqlQry).then(function (cursor) {
301
+ return cursor.next();
302
+ }).catch(function (error) {
303
+ return logError({
304
+ action: action,
305
+ category: eventCategory,
306
+ label: 'db_error'
307
+ }, error, context).then(function () {
308
+ return null;
309
+ });
310
+ });
311
+ }
312
+
313
+ return null;
314
+ };
315
+ export var addImage = function addImage(context, image) {
316
+ var action = 'addImage';
317
+ var database = context.database;
318
+ var userId = image.userId,
319
+ id = image.id,
320
+ description = image.description,
321
+ buffer = image.buffer,
322
+ fileType = image.fileType;
323
+ var now = Date.now();
324
+ return resizeSaveImage(userId, id, buffer, fileType).then(function (resizedImage) {
325
+ var insert = _objectSpread({}, resizedImage, {
326
+ _key: id,
327
+ added: now,
328
+ description: description,
329
+ fileType: fileType,
330
+ modified: now,
331
+ userId: userId
332
+ });
333
+
334
+ var aqlQry = aql(_templateObject3(), insert);
335
+ return useDb(database).query(aqlQry).then(function (cursor) {
336
+ return cursor.next();
337
+ }).then(defaultObject).catch(function (error) {
338
+ return logError({
339
+ action: action,
340
+ category: eventCategory,
341
+ isInternal: true,
342
+ label: 'db_error'
343
+ }, error, context).then(function () {
344
+ return null;
345
+ });
346
+ });
347
+ }).catch(function (error) {
348
+ return logError({
349
+ action: action,
350
+ category: eventCategory,
351
+ isInternal: true,
352
+ label: 'image_resize'
353
+ }, error, context).then(function () {
354
+ return null;
355
+ });
356
+ });
357
+ };
358
+ export var addImageEdge = function addImageEdge(context, imageEdge) {
359
+ var action = 'addImageEdge';
360
+ var database = context.database,
361
+ sessionId = context.userId;
362
+ var imgId = imageEdge.imgId,
363
+ itemId = imageEdge.itemId,
364
+ itemType = imageEdge.itemType;
365
+ var now = Date.now();
366
+ var edgeCollection = useDb(database).edgeCollection('hasImage');
367
+ var edgeId = createHash("hasImage-".concat(imgId, "-").concat(itemId, "-").concat(sessionId));
368
+ var edge = {
369
+ _key: edgeId,
370
+ added: now
371
+ };
372
+ var formatItemType = parseChar(itemType).toLowerCase();
373
+ var formatItemId = parseId(itemId);
374
+ var itemDocId;
375
+ var formatImgId = parseId(imgId);
376
+ var imageDocId = "images/".concat(formatImgId);
377
+
378
+ switch (formatItemType) {
379
+ case 'posts':
380
+ itemDocId = "posts/".concat(formatItemId);
381
+ break;
382
+
383
+ default:
384
+ itemDocId = '';
385
+ break;
386
+ }
387
+
388
+ if (itemDocId) {
389
+ return edgeCollection.save(edge, imageDocId, itemDocId).then(function (fileEdge) {
390
+ return edgeCollection.edge(fileEdge);
391
+ }).catch(function (error) {
392
+ return logError({
393
+ action: action,
394
+ category: eventCategory,
395
+ isInternal: true,
396
+ label: 'db_error'
397
+ }, error, context).then(function () {
398
+ return null;
399
+ });
400
+ });
401
+ }
402
+
403
+ return Promise.resolve({});
404
+ };
405
+ export var deleteImage = function deleteImage(context, imgId) {
406
+ var action = 'delete';
407
+ var database = context.database,
408
+ sessionId = context.userId;
409
+ var formatImgId = parseId(imgId);
410
+ var aqlQry = aql(_templateObject4(), formatImgId, sessionId);
411
+ return useDb(database).query(aqlQry).then(function (cursor) {
412
+ return cursor.next();
413
+ }).then(function () {
414
+ var image = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
415
+
416
+ if (!isEmpty(image)) {
417
+ var imageKey = image._key;
418
+ var params = {
419
+ Bucket: null,
420
+ Delete: {
421
+ Objects: [{
422
+ Key: "users/".concat(sessionId, "/images/").concat(imageKey, ".jpg")
423
+ }, {
424
+ Key: "users/".concat(sessionId, "/thumbs/").concat(imageKey, ".jpg")
425
+ }],
426
+ Quiet: true
427
+ }
428
+ };
429
+ return s3DeleteList(params).then(function () {
430
+ return image;
431
+ });
432
+ }
433
+
434
+ return {};
435
+ }).catch(function (error) {
436
+ return logError({
437
+ action: action,
438
+ category: eventCategory,
439
+ label: 'db_error'
440
+ }, error, context).then(function () {
441
+ return null;
442
+ });
443
+ });
444
+ }; // Images
445
+
446
+ export var getPathUserImages = function getPathUserImages(userID, photoId, type) {
447
+ var dir = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : 'images';
448
+ var filename = photoId;
449
+
450
+ switch (type) {
451
+ case 'image/png':
452
+ filename = "".concat(photoId, ".png");
453
+ break;
454
+
455
+ default:
456
+ filename = "".concat(photoId, ".jpg");
457
+ break;
458
+ }
459
+
460
+ return "users/".concat(userID, "/").concat(dir, "/").concat(filename);
461
+ };
462
+ export var getUrlImages = function getUrlImages(data) {
463
+ var appId = data.appId,
464
+ imgId = data.imgId,
465
+ _data$directory = data.directory,
466
+ directory = _data$directory === void 0 ? 'images' : _data$directory,
467
+ _data$imgType = data.imgType,
468
+ imgType = _data$imgType === void 0 ? 'profile' : _data$imgType,
469
+ isThumb = data.isThumb,
470
+ type = data.type,
471
+ typeId = data.typeId;
472
+ var host = Config.get('environment') === 'production' ? "https://box.".concat(Config.get('app.url')) : "https://s3.amazonaws.com/dev.".concat(Config.get('app.url'));
473
+ var suffix = isThumb ? '-th' : '';
474
+
475
+ if (imgId) {
476
+ switch (type) {
477
+ case 'app':
478
+ // https://box.reaktor.io/myApp/app/images/123.jpg
479
+ return "".concat(host, "/").concat(appId, "/").concat(type, "/").concat(directory, "/").concat(imgId).concat(suffix, ".jpg");
480
+
481
+ case 'users':
482
+ // https://box.reaktor.io/myApp/users/demoUser/images/123.jpg
483
+ return "".concat(host, "/").concat(appId, "/").concat(type, "/").concat(typeId, "/").concat(directory, "/").concat(imgId).concat(suffix, ".jpg");
484
+ }
485
+
486
+ if (imgType === 'profile') {
487
+ return "".concat(host, "/").concat(appId, "/defaults/").concat(type, "_bk").concat(suffix, ".jpg");
488
+ }
489
+
490
+ return "".concat(host, "/").concat(appId, "/defaults/").concat(type, "_wh").concat(suffix, ".jpg");
491
+ }
492
+
493
+ return '';
494
+ };
495
+ export var resizeSaveImage = function resizeSaveImage(userId, photoId, buffer) {
496
+ var type = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : 'image/jpeg';
497
+ var action = 'resizeSaveImage';
498
+ var imgW = Config.get('image.imgSize');
499
+ var imgQ = Config.get('image.imgQuality');
500
+ var photo = {};
501
+ var format = type.split('/')[1];
502
+ return new Promise(function (resolve) {
503
+ gm(buffer, 'img').setFormat(format).quality(imgQ).autoOrient().resize(imgW, imgW, '>').identify({
504
+ bufferStream: true
505
+ }, function (error) {
506
+ var val = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {};
507
+
508
+ if (error) {
509
+ logError({
510
+ action: action,
511
+ category: eventCategory,
512
+ label: 'image_save',
513
+ value: 'gm_image_identify'
514
+ }, error, {
515
+ id: userId
516
+ }).catch(function (error) {
517
+ throw error;
518
+ });
519
+ } else {
520
+ var formatVals = lowerCaseKeys(val);
521
+ var cameraMake = formatVals.make,
522
+ cameraModel = formatVals.model,
523
+ taken = formatVals.datetimeoriginal;
524
+ photo = _objectSpread({}, cloneDeep(photo), {
525
+ make: cameraMake,
526
+ model: cameraModel,
527
+ taken: DateTime.fromMillis(taken).millisecond
528
+ }); // If no background color, get the mean color value
529
+
530
+ var stats = formatVals['channel statistics'];
531
+
532
+ if (stats) {
533
+ var red = stats.red,
534
+ green = stats.green,
535
+ blue = stats.blue,
536
+ mean = stats.mean;
537
+
538
+ if (red) {
539
+ mean = red['standard deviation'] || red.mean;
540
+ red = mean ? +mean.split(' ')[0].substring(0, 3) : 0;
541
+ } else {
542
+ red = 0;
543
+ }
544
+
545
+ if (green) {
546
+ mean = green['standard deviation'] || green.mean;
547
+ green = mean ? +mean.split(' ')[0].substring(0, 3) : 0;
548
+ } else {
549
+ green = 0;
550
+ }
551
+
552
+ if (blue) {
553
+ mean = blue['standard deviation'] || blue.mean;
554
+ blue = mean ? +mean.split(' ')[0].substring(0, 3) : 0;
555
+ } else {
556
+ blue = 0;
557
+ }
558
+
559
+ var rgb = blue | green << 8 | red << 16;
560
+ var color = rgb.toString(16);
561
+ photo.color = color === '0' ? '000000' : color;
562
+ }
563
+ }
564
+ }).stream(function (error, stdout) {
565
+ if (error) {
566
+ logError({
567
+ action: action,
568
+ category: eventCategory,
569
+ isInternal: true,
570
+ label: 'image_save',
571
+ value: 'gm_image_steam'
572
+ }, error, {
573
+ id: userId
574
+ }).then(function () {
575
+ return null;
576
+ });
577
+ } else {
578
+ var imageBuffer = new Buffer('');
579
+ stdout.on('data', function (data) {
580
+ imageBuffer = Buffer.concat([imageBuffer, data]);
581
+ });
582
+ stdout.on('end', function () {
583
+ // Get file size
584
+ photo.fileSize = imageBuffer.length; // Upload data
585
+
586
+ var imageObj = {
587
+ ACL: 'public-read',
588
+ Body: imageBuffer,
589
+ Bucket: null,
590
+ ContentType: type,
591
+ Key: getPathUserImages(userId, photoId, type, 'images')
592
+ };
593
+ s3Put(imageObj).then(function () {
594
+ var thmW = Config.get('image.thmSize');
595
+ var thmQ = Config.get('image.thmQuality'); // Upload thumbnail
596
+
597
+ gm(imageBuffer, 'img').setFormat('jpeg').size(function (thumbError, resizeVal) {
598
+ if (!thumbError) {
599
+ // Get updated resize values
600
+ var height = resizeVal.height,
601
+ width = resizeVal.width;
602
+ photo = _objectSpread({}, cloneDeep(photo), {
603
+ height: height,
604
+ width: width
605
+ });
606
+ }
607
+ }).gravity('Center').resize(thmW, thmW, '^').extent(thmW, thmW).quality(thmQ).stream(function (streamError, thumbStdout) {
608
+ if (streamError) {
609
+ logError({
610
+ action: action,
611
+ category: eventCategory,
612
+ label: 'image_save',
613
+ value: 'gm_thumbnail_steam'
614
+ }, streamError, {
615
+ id: userId
616
+ }).catch(function (error) {
617
+ throw error;
618
+ });
619
+ } else {
620
+ var thumbBuffer = new Buffer('');
621
+ thumbStdout.on('data', function (data) {
622
+ thumbBuffer = Buffer.concat([thumbBuffer, data]);
623
+ });
624
+ thumbStdout.on('end', function () {
625
+ // Upload data
626
+ var thumbObj = {
627
+ ACL: 'public-read',
628
+ Body: thumbBuffer,
629
+ Bucket: null,
630
+ ContentType: type,
631
+ Key: getPathUserImages(userId, photoId, type, 'thumbs')
632
+ };
633
+ s3Put(thumbObj).then(function () {
634
+ resolve(photo);
635
+ }).catch(function () {
636
+ return logError({
637
+ action: action,
638
+ category: eventCategory,
639
+ label: 'image_save',
640
+ value: 's3_put_image'
641
+ }, streamError, {
642
+ id: userId
643
+ }).then(function () {
644
+ return null;
645
+ });
646
+ });
647
+ });
648
+ }
649
+ });
650
+ }).catch(function () {
651
+ return logError({
652
+ action: action,
653
+ category: eventCategory,
654
+ isInternal: true,
655
+ label: 'image_save',
656
+ value: 's3_image_save'
657
+ }, error, {
658
+ id: userId
659
+ }).then(function () {
660
+ return null;
661
+ });
662
+ });
663
+ });
664
+ }
665
+ });
666
+ });
667
+ };
668
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kYXRhL2ltYWdlcy50cyJdLCJuYW1lcyI6WyJnZXQiLCJodHRwR2V0IiwiY3JlYXRlSGFzaCIsInBhcnNlQ2hhciIsInBhcnNlSWQiLCJwYXJzZU51bSIsImFxbCIsImdtIiwiY2xvbmVEZWVwIiwiaXNFbXB0eSIsIkRhdGVUaW1lIiwiQ29uZmlnIiwiZGVmYXVsdE9iamVjdCIsImdldExpbWl0IiwibG9nRXJyb3IiLCJsb2dFeGNlcHRpb24iLCJsb3dlckNhc2VLZXlzIiwidXNlRGIiLCJkZWNvZGVCYXNlNjQiLCJnZXRHcm91cERldGFpbHMiLCJpc0dyb3VwZWQiLCJzM0RlbGV0ZUxpc3QiLCJzM1B1dCIsImV2ZW50Q2F0ZWdvcnkiLCJnZXRJbWFnZUxpc3RCeVVzZXIiLCJjb250ZXh0IiwidXNlcklkIiwiZnJvbSIsInRvIiwiYWN0aW9uIiwiZGF0YWJhc2UiLCJmb3JtYXRVc2VySWQiLCJsaW1pdCIsImFxbFFyeSIsInF1ZXJ5IiwidGhlbiIsImN1cnNvciIsImFsbCIsImNhdGNoIiwiZXJyb3IiLCJjYXRlZ29yeSIsImxhYmVsIiwiZ2V0SW1hZ2VMaXN0QnlHcm91cCIsInBhcmFtcyIsInNlc3Npb25JZCIsImZpbHRlcnMiLCJncm91cElkIiwiZm9ybWF0R3JvdXBJZCIsIm1hcCIsImZpbHRlciIsImNvbmRpdGlvbmFsIiwibmFtZSIsInZhbHVlIiwiZm9ybWF0Q29uZCIsImdyb3VwIiwicHJpdmFjeSIsInB1c2giLCJmaWx0ZXJTdHIiLCJqb2luIiwiZ3JvdXBlZCIsImlzVmFsaWQiLCJmaWx0ZXJMaXN0IiwiZ2V0SW1hZ2UiLCJpZCIsImZvcm1hdElkIiwibmV4dCIsImltYWdlIiwidXBkYXRlSW1hZ2UiLCJpdGVtIiwiYmFzZTY0IiwiZGVzY3JpcHRpb24iLCJmaWxlVHlwZSIsIml0ZW1JZCIsIml0ZW1UeXBlIiwidXJsIiwicGhvdG9JZCIsIm5vdyIsIkRhdGUiLCJjb250ZW50VHlwZSIsImZvcm1hdEl0ZW1JZCIsImZvcm1hdEl0ZW1UeXBlIiwidG9Mb3dlckNhc2UiLCJkZWNvZGVkQmFzZTY0IiwiZGF0YSIsInR5cGUiLCJpbWdQYXJhbXMiLCJidWZmZXIiLCJhZGRJbWFnZSIsImltYWdlRWRnZSIsImltZ0lkIiwiYWRkSW1hZ2VFZGdlIiwicmVzIiwic3RhdHVzIiwic3RhdHVzVGV4dCIsImhlYWRlcnMiLCJ1cGRhdGUiLCJtb2RpZmllZCIsInJlc2l6ZVNhdmVJbWFnZSIsInJlc2l6ZWRJbWFnZSIsImluc2VydCIsIl9rZXkiLCJhZGRlZCIsImlzSW50ZXJuYWwiLCJlZGdlQ29sbGVjdGlvbiIsImVkZ2VJZCIsImVkZ2UiLCJpdGVtRG9jSWQiLCJmb3JtYXRJbWdJZCIsImltYWdlRG9jSWQiLCJzYXZlIiwiZmlsZUVkZ2UiLCJQcm9taXNlIiwicmVzb2x2ZSIsImRlbGV0ZUltYWdlIiwiaW1hZ2VLZXkiLCJCdWNrZXQiLCJEZWxldGUiLCJPYmplY3RzIiwiS2V5IiwiUXVpZXQiLCJnZXRQYXRoVXNlckltYWdlcyIsInVzZXJJRCIsImRpciIsImZpbGVuYW1lIiwiZ2V0VXJsSW1hZ2VzIiwiYXBwSWQiLCJkaXJlY3RvcnkiLCJpbWdUeXBlIiwiaXNUaHVtYiIsInR5cGVJZCIsImhvc3QiLCJzdWZmaXgiLCJpbWdXIiwiaW1nUSIsInBob3RvIiwiZm9ybWF0Iiwic3BsaXQiLCJzZXRGb3JtYXQiLCJxdWFsaXR5IiwiYXV0b09yaWVudCIsInJlc2l6ZSIsImlkZW50aWZ5IiwiYnVmZmVyU3RyZWFtIiwidmFsIiwiZm9ybWF0VmFscyIsImNhbWVyYU1ha2UiLCJtYWtlIiwiY2FtZXJhTW9kZWwiLCJtb2RlbCIsInRha2VuIiwiZGF0ZXRpbWVvcmlnaW5hbCIsImZyb21NaWxsaXMiLCJtaWxsaXNlY29uZCIsInN0YXRzIiwicmVkIiwiZ3JlZW4iLCJibHVlIiwibWVhbiIsInN1YnN0cmluZyIsInJnYiIsImNvbG9yIiwidG9TdHJpbmciLCJzdHJlYW0iLCJzdGRvdXQiLCJpbWFnZUJ1ZmZlciIsIkJ1ZmZlciIsIm9uIiwiY29uY2F0IiwiZmlsZVNpemUiLCJsZW5ndGgiLCJpbWFnZU9iaiIsIkFDTCIsIkJvZHkiLCJDb250ZW50VHlwZSIsInRobVciLCJ0aG1RIiwic2l6ZSIsInRodW1iRXJyb3IiLCJyZXNpemVWYWwiLCJoZWlnaHQiLCJ3aWR0aCIsImdyYXZpdHkiLCJleHRlbnQiLCJzdHJlYW1FcnJvciIsInRodW1iU3Rkb3V0IiwidGh1bWJCdWZmZXIiLCJ0aHVtYk9iaiJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLFNBQVFBLEdBQUcsSUFBSUMsT0FBZixRQUE2QixtQkFBN0I7QUFDQSxTQUFRQyxVQUFSLEVBQW9CQyxTQUFwQixFQUErQkMsT0FBL0IsRUFBd0NDLFFBQXhDLFFBQXVELGNBQXZEO0FBQ0EsU0FBUUMsR0FBUixRQUFrQixVQUFsQjtBQUlBLE9BQU8sS0FBS0MsRUFBWixNQUFvQixJQUFwQjtBQUNBLE9BQU9DLFNBQVAsTUFBc0Isa0JBQXRCO0FBQ0EsT0FBT0MsT0FBUCxNQUFvQixnQkFBcEI7QUFDQSxTQUFRQyxRQUFSLFFBQXVCLE9BQXZCO0FBRUEsU0FBUUMsTUFBUixRQUFxQixXQUFyQjtBQWNBLFNBQVFDLGFBQVIsRUFBdUJDLFFBQXZCLEVBQWlDQyxRQUFqQyxFQUEyQ0MsWUFBM0MsRUFBeURDLGFBQXpELEVBQXdFQyxLQUF4RSxRQUFvRixVQUFwRjtBQUNBLFNBQVFDLFlBQVIsUUFBMkIsU0FBM0I7QUFDQSxTQUFRQyxlQUFSLEVBQXlCQyxTQUF6QixRQUF5QyxVQUF6QztBQUNBLFNBQVFDLFlBQVIsRUFBc0JDLEtBQXRCLFFBQWtDLE1BQWxDO0FBRUE7Ozs7O0FBSUEsSUFBTUMsYUFBcUIsR0FBRyxRQUE5QjtBQUVBLE9BQU8sSUFBTUMsa0JBQWtCLEdBQUcsU0FBckJBLGtCQUFxQixDQUFDQyxPQUFELEVBQXNCQyxNQUF0QixFQUFzQ0MsSUFBdEMsRUFBb0RDLEVBQXBELEVBQXlGO0FBQ3pILE1BQU1DLE1BQWMsR0FBRyxl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LGFBQUQsRUFNbEJiLEtBTmtCLEVBTVg7QUFBQ3dCLGNBQUFBLEVBQUUsRUFBRXJDO0FBQUwsYUFOVyxDQUFSLENBTVdTLElBTlgsQ0FNZ0I7QUFBQSxxQkFBTSxJQUFOO0FBQUEsYUFOaEIsQ0FBTjtBQUFBLFdBN0RUO0FBb0VELFNBakZEO0FBa0ZEO0FBQ0YsS0E3Skg7QUE4SkQsR0EvSk0sQ0FBUDtBQWdLRCxDQTFLTSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7Z2V0IGFzIGh0dHBHZXR9IGZyb20gJ0BubGFicy9yaXAtaHVudGVyJztcbmltcG9ydCB7Y3JlYXRlSGFzaCwgcGFyc2VDaGFyLCBwYXJzZUlkLCBwYXJzZU51bX0gZnJvbSAnQG5sYWJzL3V0aWxzJztcbmltcG9ydCB7YXFsfSBmcm9tICdhcmFuZ29qcyc7XG5pbXBvcnQge0FxbFF1ZXJ5fSBmcm9tICdhcmFuZ29qcy9saWIvY2pzL2FxbC1xdWVyeSc7XG5pbXBvcnQge0FycmF5Q3Vyc29yfSBmcm9tICdhcmFuZ29qcy9saWIvY2pzL2N1cnNvcic7XG5pbXBvcnQge0RlbGV0ZU9iamVjdHNSZXF1ZXN0LCBQdXRPYmplY3RSZXF1ZXN0fSBmcm9tICdhd3Mtc2RrL2NsaWVudHMvczMnO1xuaW1wb3J0ICogYXMgZ20gZnJvbSAnZ20nO1xuaW1wb3J0IGNsb25lRGVlcCBmcm9tICdsb2Rhc2gvY2xvbmVEZWVwJztcbmltcG9ydCBpc0VtcHR5IGZyb20gJ2xvZGFzaC9pc0VtcHR5JztcbmltcG9ydCB7RGF0ZVRpbWV9IGZyb20gJ2x1eG9uJztcblxuaW1wb3J0IHtDb25maWd9IGZyb20gJy4uL2NvbmZpZyc7XG5pbXBvcnQge1xuICBBcGlDb250ZXh0LFxuICBBcmFuZ29EQkxpbWl0LFxuICBGaWxlRGVjb2RlZFR5cGUsXG4gIEZpbGVFZGdlVHlwZSxcbiAgRmlsZVR5cGUsXG4gIEdyb3VwVHlwZSxcbiAgR3JvdXBVc2VyVHlwZSxcbiAgSW1hZ2VJZGVudGlmeVR5cGUsXG4gIEltYWdlVHlwZSxcbiAgSW1hZ2VVcmxEYXRhLFxuICBRdWVyeUZpbHRlclxufSBmcm9tICcuLi90eXBlcyc7XG5pbXBvcnQge2RlZmF1bHRPYmplY3QsIGdldExpbWl0LCBsb2dFcnJvciwgbG9nRXhjZXB0aW9uLCBsb3dlckNhc2VLZXlzLCB1c2VEYn0gZnJvbSAnLi4vdXRpbHMnO1xuaW1wb3J0IHtkZWNvZGVCYXNlNjR9IGZyb20gJy4vZmlsZXMnO1xuaW1wb3J0IHtnZXRHcm91cERldGFpbHMsIGlzR3JvdXBlZH0gZnJvbSAnLi9ncm91cHMnO1xuaW1wb3J0IHtzM0RlbGV0ZUxpc3QsIHMzUHV0fSBmcm9tICcuL3MzJztcblxuLyoqXG4gKiBDb3B5cmlnaHQgKGMpIDIwMTktUHJlc2VudCwgTml0cm9nZW4gTGFicywgSW5jLlxuICogQ29weXJpZ2h0cyBsaWNlbnNlZCB1bmRlciB0aGUgTUlUIExpY2Vuc2UuIFNlZSB0aGUgYWNjb21wYW55aW5nIExJQ0VOU0UgZmlsZSBmb3IgdGVybXMuXG4gKi9cbmNvbnN0IGV2ZW50Q2F0ZWdvcnk6IHN0cmluZyA9ICdpbWFnZXMnO1xuXG5leHBvcnQgY29uc3QgZ2V0SW1hZ2VMaXN0QnlVc2VyID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIHVzZXJJZDogc3RyaW5nLCBmcm9tOiBudW1iZXIsIHRvOiBudW1iZXIpOiBQcm9taXNlPEltYWdlVHlwZVtdPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2dldExpc3RCeVVzZXInO1xuICBjb25zdCB7ZGF0YWJhc2V9ID0gY29udGV4dDtcbiAgY29uc3QgZm9ybWF0VXNlcklkOiBzdHJpbmcgPSBwYXJzZUlkKHVzZXJJZCk7XG4gIGNvbnN0IGxpbWl0OiBBcmFuZ29EQkxpbWl0ID0gZ2V0TGltaXQoZnJvbSwgdG8pO1xuICBjb25zdCBhcWxRcnk6IHN0cmluZyA9IGBGT1IgaSBJTiBpbWFnZXNcbiAgICAgIEZJTFRFUiBpLnVzZXJJZCA9PSBcIiR7Zm9ybWF0VXNlcklkfVwiXG4gICAgICBMRVQgdXNlciA9IChcbiAgICAgICAgRk9SIHUgSU4gdXNlcnNcbiAgICAgICAgRklMVEVSIHUuX2tleSA9PSBpLnVzZXJJZFxuICAgICAgICBMSU1JVCAxXG4gICAgICAgIFJFVFVSTiB1XG4gICAgICApXG4gICAgICAke2xpbWl0LmFxbH1cbiAgICAgIFNPUlQgaS5hZGRlZFxuICAgICAgUkVUVVJOIE1FUkdFKGksIHt1c2VyOiBGSVJTVCh1c2VyKX0pYDtcblxuICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLmFsbCgpKVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICBhY3Rpb24sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIGxhYmVsOiAnZGJfZXJyb3InXG4gICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCkpO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldEltYWdlTGlzdEJ5R3JvdXAgPSAoY29udGV4dDogQXBpQ29udGV4dCwgcGFyYW1zKTogUHJvbWlzZTxJbWFnZVR5cGVbXT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRMaXN0QnlHcm91cCc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgdXNlcklkOiBzZXNzaW9uSWR9ID0gY29udGV4dDtcbiAgY29uc3Qge2ZpbHRlcnMgPSBbXSwgZ3JvdXBJZCwgZnJvbSwgdG99ID0gcGFyYW1zO1xuICBjb25zdCBmb3JtYXRHcm91cElkOiBzdHJpbmcgPSBwYXJzZUlkKGdyb3VwSWQpO1xuICBjb25zdCBsaW1pdCA9IGdldExpbWl0KGZyb20sIHRvKTtcblxuICBmaWx0ZXJzXG4gICAgLm1hcCgoZmlsdGVyOiBRdWVyeUZpbHRlcikgPT4ge1xuICAgICAgY29uc3Qge2NvbmRpdGlvbmFsLCBuYW1lLCB2YWx1ZX0gPSBmaWx0ZXI7XG4gICAgICBsZXQgZm9ybWF0Q29uZDogc3RyaW5nID0gY29uZGl0aW9uYWw7XG5cbiAgICAgIGlmKGNvbmRpdGlvbmFsICE9PSAnPj0nICYmIGNvbmRpdGlvbmFsICE9PSAnPD0nICYmIGNvbmRpdGlvbmFsICE9PSAnPicgJiYgY29uZGl0aW9uYWwgIT09ICc8Jykge1xuICAgICAgICBmb3JtYXRDb25kID0gJz09JztcbiAgICAgIH1cblxuICAgICAgc3dpdGNoKG5hbWUpIHtcbiAgICAgICAgY2FzZSAnYWRkZWQnOlxuICAgICAgICAgIHJldHVybiBgcC5hZGRlZCAke2Zvcm1hdENvbmR9ICR7cGFyc2VOdW0odmFsdWUpfWA7XG4gICAgICAgIGRlZmF1bHQ6XG4gICAgICAgICAgcmV0dXJuICcnO1xuICAgICAgfVxuICAgIH0pO1xuXG4gIHJldHVybiBnZXRHcm91cERldGFpbHMoY29udGV4dCwgZm9ybWF0R3JvdXBJZClcbiAgICAudGhlbigoZ3JvdXA6IEdyb3VwVHlwZSkgPT4ge1xuICAgICAgaWYoZ3JvdXAucHJpdmFjeSA9PT0gJ3B1YmxpYycpIHtcbiAgICAgICAgZmlsdGVycy5wdXNoKGBwLmdyb3VwSWQgPT0gXCIke2dyb3VwSWR9XCJgKTtcbiAgICAgICAgY29uc3QgZmlsdGVyU3RyID0gZmlsdGVycy5qb2luKCcgJiYgJyk7XG4gICAgICAgIGNvbnN0IGFxbFFyeTogc3RyaW5nID0gYEZPUiBpIElOXG4gICAgICAgICAgRkxBVFRFTihcbiAgICAgICAgICAgIEZPUiBwIElOIHBvc3RzXG4gICAgICAgICAgICBGSUxURVIgJHtmaWx0ZXJTdHJ9XG4gICAgICAgICAgICBMRVQgaW1hZ2VzID0gKFxuICAgICAgICAgICAgICBGT1IgaSwgZSBJTiBJTkJPVU5EIHAuX2lkIGhhc0ltYWdlXG4gICAgICAgICAgICAgIFJFVFVSTiBpXG4gICAgICAgICAgICApXG4gICAgICAgICAgICBTT1JUIHAuYWRkZWQgREVTQ1xuICAgICAgICAgICAgUkVUVVJOIGltYWdlc1xuICAgICAgICAgIClcbiAgICAgICAgICBTT1JUIGkuYWRkZWQgREVTQ1xuICAgICAgICAgICR7bGltaXQuYXFsfVxuICAgICAgICAgIFJFVFVSTiBpYDtcblxuICAgICAgICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAgICAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLmFsbCgpKVxuICAgICAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgICAgIGxhYmVsOiAnZGJfZXJyb3InXG4gICAgICAgICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCkpO1xuICAgICAgfVxuICAgICAgcmV0dXJuIGlzR3JvdXBlZChkYXRhYmFzZSwgc2Vzc2lvbklkLCBncm91cElkKVxuICAgICAgICAudGhlbigoZ3JvdXBlZDogR3JvdXBVc2VyVHlwZSkgPT4ge1xuICAgICAgICAgIGlmKGdyb3VwZWQuaXNWYWxpZCkge1xuICAgICAgICAgICAgZmlsdGVycy5wdXNoKGBwLmdyb3VwSWQgPT0gXCIke2dyb3VwZWQuZ3JvdXBJZH1cImApO1xuICAgICAgICAgICAgY29uc3QgZmlsdGVyTGlzdDogc3RyaW5nID0gZmlsdGVycy5qb2luKCcgJiYgJyk7XG4gICAgICAgICAgICBjb25zdCBhcWxRcnk6IHN0cmluZyA9IGBGT1IgcCBJTiBwb3N0XG4gICAgICAgICAgICAgICAgRklMVEVSICR7ZmlsdGVyTGlzdH1cbiAgICAgICAgICAgICAgICBGT1IgZiBJTiBwLmZpbGVzXG4gICAgICAgICAgICAgICAgRklMVEVSIGYudHlwZSA9PSBcImltYWdlL2pwZWdcIiB8fCBmLnR5cGUgPT0gXCJpbWFnZS9wbmdcIlxuICAgICAgICAgICAgICAgICR7bGltaXQuYXFsfVxuICAgICAgICAgICAgICAgIFNPUlQgcC5hZGRlZCBERVNDXG4gICAgICAgICAgICAgICAgUkVUVVJOIGZgO1xuXG4gICAgICAgICAgICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAgICAgICAgICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5hbGwoKSlcbiAgICAgICAgICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgICAgICAgbGFiZWw6ICdkYl9lcnJvcidcbiAgICAgICAgICAgICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCkpO1xuICAgICAgICAgIH1cbiAgICAgICAgICByZXR1cm4gW107XG4gICAgICAgIH0pO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldEltYWdlID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGlkOiBzdHJpbmcpOiBQcm9taXNlPEltYWdlVHlwZT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRJdGVtJztcbiAgY29uc3Qge2RhdGFiYXNlfSA9IGNvbnRleHQ7XG4gIGNvbnN0IGZvcm1hdElkOiBzdHJpbmcgPSBwYXJzZUlkKGlkKTtcbiAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBGT1IgaSBJTiBpbWFnZXNcbiAgICBGSUxURVIgaS5fa2V5PT0ke2Zvcm1hdElkfVxuICAgIExJTUlUIDFcbiAgICBSRVRVUk4gaWA7XG5cbiAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgLnRoZW4oKGltYWdlOiBJbWFnZVR5cGUgPSB7fSkgPT4gaW1hZ2UpXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgbGFiZWw6ICdkYl9lcnJvcidcbiAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKSk7XG59O1xuXG5leHBvcnQgY29uc3QgdXBkYXRlSW1hZ2UgPSAoY29udGV4dDogQXBpQ29udGV4dCwgaXRlbTogRmlsZVR5cGUpOiBQcm9taXNlPEltYWdlVHlwZT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICd1cGRhdGUnO1xuICBjb25zdCB7ZGF0YWJhc2UsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG5cbiAgLy8gSXRlbSBwcm9wc1xuICBjb25zdCB7XG4gICAgYmFzZTY0ID0gJycsXG4gICAgZGVzY3JpcHRpb24gPSAnJyxcbiAgICBmaWxlVHlwZSA9ICcnLFxuICAgIGlkLFxuICAgIGl0ZW1JZCxcbiAgICBpdGVtVHlwZSxcbiAgICB1cmwgPSAnJ1xuICB9ID0gaXRlbTtcblxuICAvLyBTYXZlIEJhc2U2NCBkYXRhXG4gIGNvbnN0IHBob3RvSWQ6IHN0cmluZyA9IGlkIHx8IGNyZWF0ZUhhc2goYGltYWdlLSR7c2Vzc2lvbklkfWApO1xuICBjb25zdCBub3c6IG51bWJlciA9IERhdGUubm93KCk7XG4gIGxldCBjb250ZW50VHlwZTogc3RyaW5nID0gZmlsZVR5cGU7XG4gIGNvbnN0IGZvcm1hdEl0ZW1JZDogc3RyaW5nID0gcGFyc2VJZChpdGVtSWQpO1xuICBjb25zdCBmb3JtYXRJdGVtVHlwZTogc3RyaW5nID0gcGFyc2VDaGFyKGl0ZW1UeXBlKS50b0xvd2VyQ2FzZSgpO1xuXG4gIGlmKGJhc2U2NCAhPT0gJycpIHtcbiAgICBsZXQgZGVjb2RlZEJhc2U2NDogRmlsZURlY29kZWRUeXBlO1xuXG4gICAgdHJ5IHtcbiAgICAgIGRlY29kZWRCYXNlNjQgPSBkZWNvZGVCYXNlNjQoYmFzZTY0KTtcbiAgICB9IGNhdGNoKGVycm9yKSB7XG4gICAgICByZXR1cm4gbG9nRXJyb3Ioe1xuICAgICAgICBhY3Rpb24sXG4gICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICBsYWJlbDogJ2RlY29kZV9iYXNlNjQnXG4gICAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKTtcbiAgICB9XG5cbiAgICBjb25zdCB7ZGF0YSwgdHlwZX0gPSBkZWNvZGVkQmFzZTY0O1xuICAgIGNvbnN0IGltZ1BhcmFtczogRmlsZVR5cGUgPSB7XG4gICAgICBidWZmZXI6IGRhdGEsXG4gICAgICBkZXNjcmlwdGlvbixcbiAgICAgIGZpbGVUeXBlOiB0eXBlLFxuICAgICAgaWQ6IHBob3RvSWQsXG4gICAgICB1c2VySWQ6IHNlc3Npb25JZFxuICAgIH07XG5cbiAgICByZXR1cm4gYWRkSW1hZ2UoY29udGV4dCwgaW1nUGFyYW1zKVxuICAgICAgLnRoZW4oKGltYWdlOiBJbWFnZVR5cGUpID0+IHtcbiAgICAgICAgaWYoZm9ybWF0SXRlbUlkICYmIGZvcm1hdEl0ZW1UeXBlKSB7XG4gICAgICAgICAgY29uc3QgaW1hZ2VFZGdlOiBGaWxlRWRnZVR5cGUgPSB7aW1nSWQ6IHBob3RvSWQsIGl0ZW1JZDogZm9ybWF0SXRlbUlkLCBpdGVtVHlwZTogZm9ybWF0SXRlbVR5cGV9O1xuICAgICAgICAgIHJldHVybiBhZGRJbWFnZUVkZ2UoY29udGV4dCwgaW1hZ2VFZGdlKS50aGVuKCgpID0+IGltYWdlKTtcbiAgICAgICAgfVxuICAgICAgICByZXR1cm4gaW1hZ2U7XG4gICAgICB9KVxuICAgICAgLmNhdGNoKChlcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgICBhY3Rpb24sXG4gICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICBsYWJlbDogJ2ltYWdlX3NhdmVfZXJyb3InXG4gICAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKSk7XG4gIH0gZWxzZSBpZih1cmwgIT09ICcnKSB7XG4gICAgLy8gRG93bmxvYWQgaW1hZ2UgZnJvbSB0aGUgd2ViXG4gICAgcmV0dXJuIGh0dHBHZXQodXJsKVxuICAgICAgLnRoZW4oKHJlczogUmVzcG9uc2UpID0+IHtcbiAgICAgICAgaWYocmVzLnN0YXR1cyAhPT0gMjAwKSB7XG4gICAgICAgICAgcmV0dXJuIGxvZ0V4Y2VwdGlvbih7XG4gICAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgICAgIGxhYmVsOiAnZmV0Y2hfaW1hZ2VfdXJsJyxcbiAgICAgICAgICAgIHZhbHVlOiByZXMuc3RhdHVzVGV4dFxuICAgICAgICAgIH0sIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCk7XG4gICAgICAgIH1cblxuICAgICAgICBjb250ZW50VHlwZSA9IHJlcy5oZWFkZXJzLmdldCgnY29udGVudC10eXBlJyk7XG5cbiAgICAgICAgcmV0dXJuIHJlcztcbiAgICAgIH0pXG4gICAgICAudGhlbigocmVzKSA9PiByZXMuYnVmZmVyKCkpXG4gICAgICAudGhlbigoYnVmZmVyOiBCdWZmZXIpID0+IHtcbiAgICAgICAgY29uc3QgaW1nUGFyYW1zID0ge1xuICAgICAgICAgIGJ1ZmZlcixcbiAgICAgICAgICBkZXNjcmlwdGlvbixcbiAgICAgICAgICBpZDogcGhvdG9JZCxcbiAgICAgICAgICB0eXBlOiBjb250ZW50VHlwZSxcbiAgICAgICAgICB1c2VySWQ6IHNlc3Npb25JZFxuICAgICAgICB9O1xuICAgICAgICByZXR1cm4gYWRkSW1hZ2UoY29udGV4dCwgaW1nUGFyYW1zKS50aGVuKChpbWFnZTogSW1hZ2VUeXBlKSA9PiB7XG4gICAgICAgICAgaWYoZm9ybWF0SXRlbUlkICYmIGZvcm1hdEl0ZW1UeXBlKSB7XG4gICAgICAgICAgICBjb25zdCBpbWFnZUVkZ2U6IEZpbGVFZGdlVHlwZSA9IHtpbWdJZDogaWQsIGl0ZW1JZDogZm9ybWF0SXRlbUlkLCBpdGVtVHlwZTogZm9ybWF0SXRlbVR5cGV9O1xuICAgICAgICAgICAgcmV0dXJuIGFkZEltYWdlRWRnZShjb250ZXh0LCBpbWFnZUVkZ2UpLnRoZW4oKCkgPT4gaW1hZ2UpO1xuICAgICAgICAgIH1cblxuICAgICAgICAgIHJldHVybiBpbWFnZTtcbiAgICAgICAgfSk7XG4gICAgICB9KVxuICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgYWN0aW9uLFxuICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgbGFiZWw6ICdmZXRjaF9lcnJvcidcbiAgICAgIH0sIGVycm9yLCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpKTtcbiAgfSBlbHNlIGlmKGlkICE9PSAnJykge1xuICAgIC8vIFVwZGF0ZSBtZXRhZGF0YVxuICAgIGNvbnN0IHVwZGF0ZTogYW55ID0ge1xuICAgICAgZGVzY3JpcHRpb24sXG4gICAgICBtb2RpZmllZDogbm93XG4gICAgfTtcbiAgICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYFVQREFURSAke2lkfSBXSVRIICR7dXBkYXRlfSBJTiBpbWFnZXMgUkVUVVJOIE5FV2A7XG5cbiAgICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgYWN0aW9uLFxuICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgbGFiZWw6ICdkYl9lcnJvcidcbiAgICAgIH0sIGVycm9yLCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpKTtcbiAgfVxuXG4gIHJldHVybiBudWxsO1xufTtcblxuZXhwb3J0IGNvbnN0IGFkZEltYWdlID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGltYWdlOiBJbWFnZVR5cGUpOiBQcm9taXNlPEltYWdlVHlwZT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdhZGRJbWFnZSc7XG4gIGNvbnN0IHtkYXRhYmFzZX0gPSBjb250ZXh0O1xuICBjb25zdCB7dXNlcklkLCBpZCwgZGVzY3JpcHRpb24sIGJ1ZmZlciwgZmlsZVR5cGV9ID0gaW1hZ2U7XG4gIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcblxuICByZXR1cm4gcmVzaXplU2F2ZUltYWdlKHVzZXJJZCwgaWQsIGJ1ZmZlciwgZmlsZVR5cGUpXG4gICAgLnRoZW4oKHJlc2l6ZWRJbWFnZTogYW55KSA9PiB7XG4gICAgICBjb25zdCBpbnNlcnQ6IEltYWdlVHlwZSA9IHtcbiAgICAgICAgLi4ucmVzaXplZEltYWdlLFxuICAgICAgICBfa2V5OiBpZCxcbiAgICAgICAgYWRkZWQ6IG5vdyxcbiAgICAgICAgZGVzY3JpcHRpb24sXG4gICAgICAgIGZpbGVUeXBlLFxuICAgICAgICBtb2RpZmllZDogbm93LFxuICAgICAgICB1c2VySWRcbiAgICAgIH07XG5cbiAgICAgIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgSU5TRVJUICR7aW5zZXJ0fSBJTiBpbWFnZXMgUkVUVVJOIE5FV2A7XG5cbiAgICAgIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgICAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAgICAgLnRoZW4oZGVmYXVsdE9iamVjdClcbiAgICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgaXNJbnRlcm5hbDogdHJ1ZSxcbiAgICAgICAgICBsYWJlbDogJ2RiX2Vycm9yJ1xuICAgICAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKSk7XG4gICAgfSlcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgYWN0aW9uLFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICBpc0ludGVybmFsOiB0cnVlLFxuICAgICAgbGFiZWw6ICdpbWFnZV9yZXNpemUnXG4gICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCkpO1xufTtcblxuZXhwb3J0IGNvbnN0IGFkZEltYWdlRWRnZSA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBpbWFnZUVkZ2U6IEZpbGVFZGdlVHlwZSk6IFByb21pc2U8b2JqZWN0PiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2FkZEltYWdlRWRnZSc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgdXNlcklkOiBzZXNzaW9uSWR9ID0gY29udGV4dDtcbiAgY29uc3Qge2ltZ0lkLCBpdGVtSWQsIGl0ZW1UeXBlfSA9IGltYWdlRWRnZTtcbiAgY29uc3Qgbm93OiBudW1iZXIgPSBEYXRlLm5vdygpO1xuICBjb25zdCBlZGdlQ29sbGVjdGlvbiA9IHVzZURiKGRhdGFiYXNlKS5lZGdlQ29sbGVjdGlvbignaGFzSW1hZ2UnKTtcbiAgY29uc3QgZWRnZUlkOiBzdHJpbmcgPSBjcmVhdGVIYXNoKGBoYXNJbWFnZS0ke2ltZ0lkfS0ke2l0ZW1JZH0tJHtzZXNzaW9uSWR9YCk7XG4gIGNvbnN0IGVkZ2U6IGFueSA9IHtcbiAgICBfa2V5OiBlZGdlSWQsXG4gICAgYWRkZWQ6IG5vd1xuICB9O1xuXG4gIGNvbnN0IGZvcm1hdEl0ZW1UeXBlOiBzdHJpbmcgPSBwYXJzZUNoYXIoaXRlbVR5cGUpLnRvTG93ZXJDYXNlKCk7XG4gIGNvbnN0IGZvcm1hdEl0ZW1JZDogc3RyaW5nID0gcGFyc2VJZChpdGVtSWQpO1xuICBsZXQgaXRlbURvY0lkOiBzdHJpbmc7XG4gIGNvbnN0IGZvcm1hdEltZ0lkOiBzdHJpbmcgPSBwYXJzZUlkKGltZ0lkKTtcbiAgY29uc3QgaW1hZ2VEb2NJZCA9IGBpbWFnZXMvJHtmb3JtYXRJbWdJZH1gO1xuXG4gIHN3aXRjaChmb3JtYXRJdGVtVHlwZSkge1xuICAgIGNhc2UgJ3Bvc3RzJzpcbiAgICAgIGl0ZW1Eb2NJZCA9IGBwb3N0cy8ke2Zvcm1hdEl0ZW1JZH1gO1xuICAgICAgYnJlYWs7XG4gICAgZGVmYXVsdDpcbiAgICAgIGl0ZW1Eb2NJZCA9ICcnO1xuICAgICAgYnJlYWs7XG4gIH1cblxuICBpZihpdGVtRG9jSWQpIHtcbiAgICByZXR1cm4gZWRnZUNvbGxlY3Rpb24uc2F2ZShlZGdlLCBpbWFnZURvY0lkLCBpdGVtRG9jSWQpXG4gICAgICAudGhlbigoZmlsZUVkZ2UpID0+IGVkZ2VDb2xsZWN0aW9uLmVkZ2UoZmlsZUVkZ2UpKVxuICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgYWN0aW9uLFxuICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgaXNJbnRlcm5hbDogdHJ1ZSxcbiAgICAgICAgbGFiZWw6ICdkYl9lcnJvcidcbiAgICAgIH0sIGVycm9yLCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpKTtcbiAgfVxuXG4gIHJldHVybiBQcm9taXNlLnJlc29sdmUoe30pO1xufTtcblxuZXhwb3J0IGNvbnN0IGRlbGV0ZUltYWdlID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGltZ0lkKTogUHJvbWlzZTxJbWFnZVR5cGU+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnZGVsZXRlJztcbiAgY29uc3Qge2RhdGFiYXNlLCB1c2VySWQ6IHNlc3Npb25JZH0gPSBjb250ZXh0O1xuICBjb25zdCBmb3JtYXRJbWdJZCA9IHBhcnNlSWQoaW1nSWQpO1xuXG4gIGNvbnN0IGFxbFFyeSA9IGFxbGBGT1IgaSBJTiBpbWFnZXNcbiAgICBGSUxURVIgaS5fa2V5ID09ICR7Zm9ybWF0SW1nSWR9ICYmIGkudXNlcklkID09ICR7c2Vzc2lvbklkfVxuICAgIFJFTU9WRSBpIElOIGltYWdlc1xuICAgIFJFVFVSTiBPTERgO1xuXG4gIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgIC50aGVuKChpbWFnZTogSW1hZ2VUeXBlID0ge30pID0+IHtcbiAgICAgIGlmKCFpc0VtcHR5KGltYWdlKSkge1xuICAgICAgICBjb25zdCB7X2tleTogaW1hZ2VLZXl9ID0gaW1hZ2U7XG4gICAgICAgIGNvbnN0IHBhcmFtczogRGVsZXRlT2JqZWN0c1JlcXVlc3QgPSB7XG4gICAgICAgICAgQnVja2V0OiBudWxsLFxuICAgICAgICAgIERlbGV0ZToge1xuICAgICAgICAgICAgT2JqZWN0czogW1xuICAgICAgICAgICAgICB7S2V5OiBgdXNlcnMvJHtzZXNzaW9uSWR9L2ltYWdlcy8ke2ltYWdlS2V5fS5qcGdgfSxcbiAgICAgICAgICAgICAge0tleTogYHVzZXJzLyR7c2Vzc2lvbklkfS90aHVtYnMvJHtpbWFnZUtleX0uanBnYH1cbiAgICAgICAgICAgIF0sXG4gICAgICAgICAgICBRdWlldDogdHJ1ZVxuICAgICAgICAgIH1cbiAgICAgICAgfTtcblxuICAgICAgICByZXR1cm4gczNEZWxldGVMaXN0KHBhcmFtcykudGhlbigoKSA9PiBpbWFnZSk7XG4gICAgICB9XG4gICAgICByZXR1cm4ge307XG4gICAgfSlcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgYWN0aW9uLFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICBsYWJlbDogJ2RiX2Vycm9yJ1xuICAgIH0sIGVycm9yLCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpKTtcbn07XG5cbi8vIEltYWdlc1xuZXhwb3J0IGNvbnN0IGdldFBhdGhVc2VySW1hZ2VzID0gKHVzZXJJRDogc3RyaW5nLCBwaG90b0lkOiBzdHJpbmcsIHR5cGU6IHN0cmluZywgZGlyOiBzdHJpbmcgPSAnaW1hZ2VzJyk6IHN0cmluZyA9PiB7XG4gIGxldCBmaWxlbmFtZTogc3RyaW5nID0gcGhvdG9JZDtcblxuICBzd2l0Y2godHlwZSkge1xuICAgIGNhc2UgJ2ltYWdlL3BuZyc6XG4gICAgICBmaWxlbmFtZSA9IGAke3Bob3RvSWR9LnBuZ2A7XG4gICAgICBicmVhaztcbiAgICBkZWZhdWx0OlxuICAgICAgZmlsZW5hbWUgPSBgJHtwaG90b0lkfS5qcGdgO1xuICAgICAgYnJlYWs7XG4gIH1cblxuICByZXR1cm4gYHVzZXJzLyR7dXNlcklEfS8ke2Rpcn0vJHtmaWxlbmFtZX1gO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldFVybEltYWdlcyA9IChkYXRhOiBJbWFnZVVybERhdGEpOiBzdHJpbmcgPT4ge1xuICBjb25zdCB7YXBwSWQsIGltZ0lkLCBkaXJlY3RvcnkgPSAnaW1hZ2VzJywgaW1nVHlwZSA9ICdwcm9maWxlJywgaXNUaHVtYiwgdHlwZSwgdHlwZUlkfSA9IGRhdGE7XG4gIGNvbnN0IGhvc3Q6IHN0cmluZyA9IENvbmZpZy5nZXQoJ2Vudmlyb25tZW50JykgPT09ICdwcm9kdWN0aW9uJ1xuICAgID8gYGh0dHBzOi8vYm94LiR7Q29uZmlnLmdldCgnYXBwLnVybCcpfWBcbiAgICA6IGBodHRwczovL3MzLmFtYXpvbmF3cy5jb20vZGV2LiR7Q29uZmlnLmdldCgnYXBwLnVybCcpfWA7XG4gIGNvbnN0IHN1ZmZpeDogc3RyaW5nID0gaXNUaHVtYiA/ICctdGgnIDogJyc7XG5cbiAgaWYoaW1nSWQpIHtcbiAgICBzd2l0Y2godHlwZSkge1xuICAgICAgY2FzZSAnYXBwJzpcbiAgICAgICAgLy8gaHR0cHM6Ly9ib3gucmVha3Rvci5pby9teUFwcC9hcHAvaW1hZ2VzLzEyMy5qcGdcbiAgICAgICAgcmV0dXJuIGAke2hvc3R9LyR7YXBwSWR9LyR7dHlwZX0vJHtkaXJlY3Rvcnl9LyR7aW1nSWR9JHtzdWZmaXh9LmpwZ2A7XG4gICAgICBjYXNlICd1c2Vycyc6XG4gICAgICAgIC8vIGh0dHBzOi8vYm94LnJlYWt0b3IuaW8vbXlBcHAvdXNlcnMvZGVtb1VzZXIvaW1hZ2VzLzEyMy5qcGdcbiAgICAgICAgcmV0dXJuIGAke2hvc3R9LyR7YXBwSWR9LyR7dHlwZX0vJHt0eXBlSWR9LyR7ZGlyZWN0b3J5fS8ke2ltZ0lkfSR7c3VmZml4fS5qcGdgO1xuICAgIH1cblxuICAgIGlmKGltZ1R5cGUgPT09ICdwcm9maWxlJykge1xuICAgICAgcmV0dXJuIGAke2hvc3R9LyR7YXBwSWR9L2RlZmF1bHRzLyR7dHlwZX1fYmske3N1ZmZpeH0uanBnYDtcbiAgICB9XG5cbiAgICByZXR1cm4gYCR7aG9zdH0vJHthcHBJZH0vZGVmYXVsdHMvJHt0eXBlfV93aCR7c3VmZml4fS5qcGdgO1xuICB9XG5cbiAgcmV0dXJuICcnO1xufTtcblxuZXhwb3J0IGNvbnN0IHJlc2l6ZVNhdmVJbWFnZSA9ICh1c2VySWQ6IHN0cmluZyxcbiAgcGhvdG9JZDogc3RyaW5nLFxuICBidWZmZXI6IEJ1ZmZlcixcbiAgdHlwZTogc3RyaW5nID0gJ2ltYWdlL2pwZWcnKTogUHJvbWlzZTxJbWFnZVR5cGU+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAncmVzaXplU2F2ZUltYWdlJztcbiAgY29uc3QgaW1nVzogbnVtYmVyID0gQ29uZmlnLmdldCgnaW1hZ2UuaW1nU2l6ZScpO1xuICBjb25zdCBpbWdROiBudW1iZXIgPSBDb25maWcuZ2V0KCdpbWFnZS5pbWdRdWFsaXR5Jyk7XG4gIGxldCBwaG90bzogSW1hZ2VUeXBlID0ge307XG4gIGNvbnN0IGZvcm1hdDogc3RyaW5nID0gKHR5cGUuc3BsaXQoJy8nKSlbMV07XG5cbiAgcmV0dXJuIG5ldyBQcm9taXNlKChyZXNvbHZlKSA9PiB7XG4gICAgZ20oYnVmZmVyLCAnaW1nJylcbiAgICAgIC5zZXRGb3JtYXQoZm9ybWF0KVxuICAgICAgLnF1YWxpdHkoaW1nUSlcbiAgICAgIC5hdXRvT3JpZW50KClcbiAgICAgIC5yZXNpemUoaW1nVywgaW1nVywgJz4nKVxuICAgICAgLmlkZW50aWZ5KHtidWZmZXJTdHJlYW06IHRydWV9LCAoZXJyb3I6IEVycm9yLCB2YWw6IEltYWdlSWRlbnRpZnlUeXBlID0ge30pOiBhbnkgPT4ge1xuICAgICAgICBpZihlcnJvcikge1xuICAgICAgICAgIGxvZ0Vycm9yKHtcbiAgICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgICAgbGFiZWw6ICdpbWFnZV9zYXZlJyxcbiAgICAgICAgICAgIHZhbHVlOiAnZ21faW1hZ2VfaWRlbnRpZnknXG4gICAgICAgICAgfSwgZXJyb3IsIHtpZDogdXNlcklkfSkuY2F0Y2goKGVycm9yKSA9PiB7XG4gICAgICAgICAgICB0aHJvdyBlcnJvcjtcbiAgICAgICAgICB9KTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBjb25zdCBmb3JtYXRWYWxzID0gbG93ZXJDYXNlS2V5cyh2YWwpO1xuICAgICAgICAgIGNvbnN0IHttYWtlOiBjYW1lcmFNYWtlLCBtb2RlbDogY2FtZXJhTW9kZWwsIGRhdGV0aW1lb3JpZ2luYWw6IHRha2VufTogSW1hZ2VJZGVudGlmeVR5cGUgPSBmb3JtYXRWYWxzO1xuICAgICAgICAgIHBob3RvID0ge1xuICAgICAgICAgICAgLi4uY2xvbmVEZWVwKHBob3RvKSxcbiAgICAgICAgICAgIG1ha2U6IGNhbWVyYU1ha2UsXG4gICAgICAgICAgICBtb2RlbDogY2FtZXJhTW9kZWwsXG4gICAgICAgICAgICB0YWtlbjogRGF0ZVRpbWUuZnJvbU1pbGxpcyh0YWtlbikubWlsbGlzZWNvbmRcbiAgICAgICAgICB9O1xuXG4gICAgICAgICAgLy8gSWYgbm8gYmFja2dyb3VuZCBjb2xvciwgZ2V0IHRoZSBtZWFuIGNvbG9yIHZhbHVlXG4gICAgICAgICAgY29uc3Qgc3RhdHMgPSBmb3JtYXRWYWxzWydjaGFubmVsIHN0YXRpc3RpY3MnXTtcblxuICAgICAgICAgIGlmKHN0YXRzKSB7XG4gICAgICAgICAgICBsZXQge3JlZCwgZ3JlZW4sIGJsdWUsIG1lYW59ID0gc3RhdHM7XG5cbiAgICAgICAgICAgIGlmKHJlZCkge1xuICAgICAgICAgICAgICBtZWFuID0gcmVkWydzdGFuZGFyZCBkZXZpYXRpb24nXSB8fCByZWQubWVhbjtcbiAgICAgICAgICAgICAgcmVkID0gbWVhbiA/ICsoKG1lYW4uc3BsaXQoJyAnKVswXSkuc3Vic3RyaW5nKDAsIDMpKSA6IDA7XG4gICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICByZWQgPSAwO1xuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICBpZihncmVlbikge1xuICAgICAgICAgICAgICBtZWFuID0gZ3JlZW5bJ3N0YW5kYXJkIGRldmlhdGlvbiddIHx8IGdyZWVuLm1lYW47XG4gICAgICAgICAgICAgIGdyZWVuID0gbWVhbiA/ICsoKG1lYW4uc3BsaXQoJyAnKVswXSkuc3Vic3RyaW5nKDAsIDMpKSA6IDA7XG4gICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICBncmVlbiA9IDA7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIGlmKGJsdWUpIHtcbiAgICAgICAgICAgICAgbWVhbiA9IGJsdWVbJ3N0YW5kYXJkIGRldmlhdGlvbiddIHx8IGJsdWUubWVhbjtcbiAgICAgICAgICAgICAgYmx1ZSA9IG1lYW4gPyArKChtZWFuLnNwbGl0KCcgJylbMF0pLnN1YnN0cmluZygwLCAzKSkgOiAwO1xuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgYmx1ZSA9IDA7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIGNvbnN0IHJnYiA9IGJsdWUgfCAoZ3JlZW4gPDwgOCkgfCAocmVkIDw8IDE2KTtcbiAgICAgICAgICAgIGNvbnN0IGNvbG9yID0gcmdiLnRvU3RyaW5nKDE2KTtcbiAgICAgICAgICAgIHBob3RvLmNvbG9yID0gY29sb3IgPT09ICcwJyA/ICcwMDAwMDAnIDogY29sb3I7XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICB9KVxuICAgICAgLnN0cmVhbSgoZXJyb3I6IEVycm9yLCBzdGRvdXQpOiBhbnkgPT4ge1xuICAgICAgICBpZihlcnJvcikge1xuICAgICAgICAgIGxvZ0Vycm9yKHtcbiAgICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgICAgaXNJbnRlcm5hbDogdHJ1ZSxcbiAgICAgICAgICAgIGxhYmVsOiAnaW1hZ2Vfc2F2ZScsXG4gICAgICAgICAgICB2YWx1ZTogJ2dtX2ltYWdlX3N0ZWFtJ1xuICAgICAgICAgIH0sIGVycm9yLCB7aWQ6IHVzZXJJZH0pLnRoZW4oKCkgPT4gbnVsbCk7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgbGV0IGltYWdlQnVmZmVyOiBCdWZmZXIgPSBuZXcgQnVmZmVyKCcnKTtcblxuICAgICAgICAgIHN0ZG91dC5vbignZGF0YScsIChkYXRhKSA9PiB7XG4gICAgICAgICAgICBpbWFnZUJ1ZmZlciA9IEJ1ZmZlci5jb25jYXQoW2ltYWdlQnVmZmVyLCBkYXRhXSk7XG4gICAgICAgICAgfSk7XG5cbiAgICAgICAgICBzdGRvdXQub24oJ2VuZCcsICgpID0+IHtcbiAgICAgICAgICAgIC8vIEdldCBmaWxlIHNpemVcbiAgICAgICAgICAgIHBob3RvLmZpbGVTaXplID0gaW1hZ2VCdWZmZXIubGVuZ3RoO1xuXG4gICAgICAgICAgICAvLyBVcGxvYWQgZGF0YVxuICAgICAgICAgICAgY29uc3QgaW1hZ2VPYmo6IFB1dE9iamVjdFJlcXVlc3QgPSB7XG4gICAgICAgICAgICAgIEFDTDogJ3B1YmxpYy1yZWFkJyxcbiAgICAgICAgICAgICAgQm9keTogaW1hZ2VCdWZmZXIsXG4gICAgICAgICAgICAgIEJ1Y2tldDogbnVsbCxcbiAgICAgICAgICAgICAgQ29udGVudFR5cGU6IHR5cGUsXG4gICAgICAgICAgICAgIEtleTogZ2V0UGF0aFVzZXJJbWFnZXModXNlcklkLCBwaG90b0lkLCB0eXBlLCAnaW1hZ2VzJylcbiAgICAgICAgICAgIH07XG5cbiAgICAgICAgICAgIHMzUHV0KGltYWdlT2JqKVxuICAgICAgICAgICAgICAudGhlbigoKSA9PiB7XG4gICAgICAgICAgICAgICAgY29uc3QgdGhtVyA9IENvbmZpZy5nZXQoJ2ltYWdlLnRobVNpemUnKTtcbiAgICAgICAgICAgICAgICBjb25zdCB0aG1RID0gQ29uZmlnLmdldCgnaW1hZ2UudGhtUXVhbGl0eScpO1xuXG4gICAgICAgICAgICAgICAgLy8gVXBsb2FkIHRodW1ibmFpbFxuICAgICAgICAgICAgICAgIGdtKGltYWdlQnVmZmVyLCAnaW1nJylcbiAgICAgICAgICAgICAgICAgIC5zZXRGb3JtYXQoJ2pwZWcnKVxuICAgICAgICAgICAgICAgICAgLnNpemUoKHRodW1iRXJyb3I6IEVycm9yLCByZXNpemVWYWwpID0+IHtcbiAgICAgICAgICAgICAgICAgICAgaWYoIXRodW1iRXJyb3IpIHtcbiAgICAgICAgICAgICAgICAgICAgICAvLyBHZXQgdXBkYXRlZCByZXNpemUgdmFsdWVzXG4gICAgICAgICAgICAgICAgICAgICAgY29uc3Qge2hlaWdodCwgd2lkdGh9ID0gcmVzaXplVmFsO1xuICAgICAgICAgICAgICAgICAgICAgIHBob3RvID0gey4uLmNsb25lRGVlcChwaG90byksIGhlaWdodCwgd2lkdGh9O1xuICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICB9KVxuICAgICAgICAgICAgICAgICAgLmdyYXZpdHkoJ0NlbnRlcicpXG4gICAgICAgICAgICAgICAgICAucmVzaXplKHRobVcsIHRobVcsICdeJylcbiAgICAgICAgICAgICAgICAgIC5leHRlbnQodGhtVywgdGhtVylcbiAgICAgICAgICAgICAgICAgIC5xdWFsaXR5KHRobVEpXG4gICAgICAgICAgICAgICAgICAuc3RyZWFtKChzdHJlYW1FcnJvcjogRXJyb3IsIHRodW1iU3Rkb3V0KTogYW55ID0+IHtcbiAgICAgICAgICAgICAgICAgICAgaWYoc3RyZWFtRXJyb3IpIHtcbiAgICAgICAgICAgICAgICAgICAgICBsb2dFcnJvcih7XG4gICAgICAgICAgICAgICAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgICAgICAgICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgICAgICAgICAgICAgICAgIGxhYmVsOiAnaW1hZ2Vfc2F2ZScsXG4gICAgICAgICAgICAgICAgICAgICAgICB2YWx1ZTogJ2dtX3RodW1ibmFpbF9zdGVhbSdcbiAgICAgICAgICAgICAgICAgICAgICB9LCBzdHJlYW1FcnJvciwge2lkOiB1c2VySWR9KVxuICAgICAgICAgICAgICAgICAgICAgICAgLmNhdGNoKChlcnJvcikgPT4ge1xuICAgICAgICAgICAgICAgICAgICAgICAgICB0aHJvdyBlcnJvcjtcbiAgICAgICAgICAgICAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICAgICAgICAgIGxldCB0aHVtYkJ1ZmZlcjogQnVmZmVyID0gbmV3IEJ1ZmZlcignJyk7XG5cbiAgICAgICAgICAgICAgICAgICAgICB0aHVtYlN0ZG91dC5vbignZGF0YScsIChkYXRhKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgICB0aHVtYkJ1ZmZlciA9IEJ1ZmZlci5jb25jYXQoW3RodW1iQnVmZmVyLCBkYXRhXSk7XG4gICAgICAgICAgICAgICAgICAgICAgfSk7XG5cbiAgICAgICAgICAgICAgICAgICAgICB0aHVtYlN0ZG91dC5vbignZW5kJywgKCkgPT4ge1xuICAgICAgICAgICAgICAgICAgICAgICAgLy8gVXBsb2FkIGRhdGFcbiAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IHRodW1iT2JqOiBQdXRPYmplY3RSZXF1ZXN0ID0ge1xuICAgICAgICAgICAgICAgICAgICAgICAgICBBQ0w6ICdwdWJsaWMtcmVhZCcsXG4gICAgICAgICAgICAgICAgICAgICAgICAgIEJvZHk6IHRodW1iQnVmZmVyLFxuICAgICAgICAgICAgICAgICAgICAgICAgICBCdWNrZXQ6IG51bGwsXG4gICAgICAgICAgICAgICAgICAgICAgICAgIENvbnRlbnRUeXBlOiB0eXBlLFxuICAgICAgICAgICAgICAgICAgICAgICAgICBLZXk6IGdldFBhdGhVc2VySW1hZ2VzKHVzZXJJZCwgcGhvdG9JZCwgdHlwZSwgJ3RodW1icycpXG4gICAgICAgICAgICAgICAgICAgICAgICB9O1xuXG4gICAgICAgICAgICAgICAgICAgICAgICBzM1B1dCh0aHVtYk9iailcbiAgICAgICAgICAgICAgICAgICAgICAgICAgLnRoZW4oKCkgPT4ge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJlc29sdmUocGhvdG8pO1xuICAgICAgICAgICAgICAgICAgICAgICAgICB9KVxuICAgICAgICAgICAgICAgICAgICAgICAgICAuY2F0Y2goKCkgPT4gbG9nRXJyb3Ioe1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBsYWJlbDogJ2ltYWdlX3NhdmUnLFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIHZhbHVlOiAnczNfcHV0X2ltYWdlJ1xuICAgICAgICAgICAgICAgICAgICAgICAgICB9LCBzdHJlYW1FcnJvciwge2lkOiB1c2VySWR9KS50aGVuKCgpID0+IG51bGwpKTtcbiAgICAgICAgICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICAgIH0pXG4gICAgICAgICAgICAgIC5jYXRjaCgoKSA9PiBsb2dFcnJvcih7XG4gICAgICAgICAgICAgICAgYWN0aW9uLFxuICAgICAgICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgICAgICAgIGlzSW50ZXJuYWw6IHRydWUsXG4gICAgICAgICAgICAgICAgbGFiZWw6ICdpbWFnZV9zYXZlJyxcbiAgICAgICAgICAgICAgICB2YWx1ZTogJ3MzX2ltYWdlX3NhdmUnXG4gICAgICAgICAgICAgIH0sIGVycm9yLCB7aWQ6IHVzZXJJZH0pLnRoZW4oKCkgPT4gbnVsbCkpO1xuICAgICAgICAgIH0pO1xuICAgICAgICB9XG4gICAgICB9KTtcbiAgfSk7XG59O1xuIl19