@nlabs/reaktor 0.1.2 → 0.1.3

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 (144) hide show
  1. package/.DS_Store +0 -0
  2. package/lib/config.d.ts +21 -0
  3. package/lib/config.js +130 -0
  4. package/lib/data/conversations.d.ts +6 -0
  5. package/lib/data/conversations.js +197 -0
  6. package/lib/data/dynamodb.d.ts +8 -0
  7. package/lib/data/dynamodb.js +139 -0
  8. package/lib/data/email.d.ts +7 -0
  9. package/lib/data/email.js +163 -0
  10. package/lib/data/files.d.ts +16 -0
  11. package/lib/data/files.js +406 -0
  12. package/lib/data/groups.d.ts +13 -0
  13. package/lib/data/groups.js +354 -0
  14. package/lib/data/images.d.ts +12 -0
  15. package/lib/data/images.js +667 -0
  16. package/{src/data/index.ts → lib/data/index.d.ts} +1 -5
  17. package/lib/data/index.js +24 -0
  18. package/lib/data/ios.d.ts +6 -0
  19. package/lib/data/ios.js +302 -0
  20. package/lib/data/locations.d.ts +3 -0
  21. package/lib/data/locations.js +132 -0
  22. package/lib/data/messages.d.ts +9 -0
  23. package/lib/data/messages.js +248 -0
  24. package/lib/data/notifications.d.ts +5 -0
  25. package/lib/data/notifications.js +42 -0
  26. package/lib/data/payments.d.ts +11 -0
  27. package/lib/data/payments.js +748 -0
  28. package/lib/data/posts.d.ts +22 -0
  29. package/lib/data/posts.js +578 -0
  30. package/lib/data/reactions.d.ts +6 -0
  31. package/lib/data/reactions.js +218 -0
  32. package/lib/data/s3.d.ts +6 -0
  33. package/lib/data/s3.js +103 -0
  34. package/lib/data/search.d.ts +3 -0
  35. package/lib/data/search.js +98 -0
  36. package/lib/data/sms.d.ts +3 -0
  37. package/lib/data/sms.js +59 -0
  38. package/lib/data/subscription.d.ts +7 -0
  39. package/lib/data/subscription.js +284 -0
  40. package/lib/data/tags.d.ts +14 -0
  41. package/lib/data/tags.js +304 -0
  42. package/lib/data/users.d.ts +12 -0
  43. package/lib/data/users.js +310 -0
  44. package/lib/index.d.ts +3 -0
  45. package/lib/index.js +8 -0
  46. package/lib/types/apps.d.ts +43 -0
  47. package/lib/types/apps.js +2 -0
  48. package/lib/types/arangodb.d.ts +17 -0
  49. package/lib/types/arangodb.js +2 -0
  50. package/lib/types/auth.d.ts +10 -0
  51. package/lib/types/auth.js +2 -0
  52. package/lib/types/conversations.d.ts +6 -0
  53. package/lib/types/conversations.js +2 -0
  54. package/lib/types/email.d.ts +12 -0
  55. package/lib/types/email.js +2 -0
  56. package/lib/types/files.d.ts +26 -0
  57. package/lib/types/files.js +2 -0
  58. package/lib/types/google.d.ts +27 -0
  59. package/lib/types/google.js +2 -0
  60. package/lib/types/groups.d.ts +21 -0
  61. package/lib/types/groups.js +2 -0
  62. package/lib/types/images.d.ts +24 -0
  63. package/lib/types/images.js +2 -0
  64. package/{src/types/index.ts → lib/types/index.d.ts} +0 -4
  65. package/lib/types/index.js +22 -0
  66. package/lib/types/locations.d.ts +20 -0
  67. package/lib/types/locations.js +2 -0
  68. package/lib/types/messages.d.ts +12 -0
  69. package/lib/types/messages.js +2 -0
  70. package/lib/types/notifications.d.ts +19 -0
  71. package/lib/types/notifications.js +2 -0
  72. package/lib/types/payments.d.ts +114 -0
  73. package/lib/types/payments.js +2 -0
  74. package/lib/types/posts.d.ts +28 -0
  75. package/lib/types/posts.js +2 -0
  76. package/lib/types/reactions.d.ts +4 -0
  77. package/lib/types/reactions.js +2 -0
  78. package/lib/types/tags.d.ts +9 -0
  79. package/lib/types/tags.js +2 -0
  80. package/lib/types/users.d.ts +78 -0
  81. package/lib/types/users.js +2 -0
  82. package/lib/utils/analytics.d.ts +3 -0
  83. package/lib/utils/analytics.js +47 -0
  84. package/lib/utils/arangodb.d.ts +9 -0
  85. package/lib/utils/arangodb.js +98 -0
  86. package/lib/utils/auth.d.ts +7 -0
  87. package/lib/utils/auth.js +80 -0
  88. package/lib/utils/graphql.d.ts +1 -0
  89. package/lib/utils/graphql.js +7 -0
  90. package/{src/utils/index.ts → lib/utils/index.d.ts} +0 -4
  91. package/lib/utils/index.js +11 -0
  92. package/lib/utils/objects.d.ts +3 -0
  93. package/lib/utils/objects.js +34 -0
  94. package/lib/utils/redis.d.ts +1 -0
  95. package/lib/utils/redis.js +15 -0
  96. package/package.json +5 -5
  97. package/.vscode/extensions.json +0 -15
  98. package/.vscode/settings.json +0 -82
  99. package/lex.config.js +0 -4
  100. package/src/config.ts +0 -127
  101. package/src/data/conversations.ts +0 -181
  102. package/src/data/dynamodb.ts +0 -157
  103. package/src/data/email.ts +0 -163
  104. package/src/data/files.ts +0 -352
  105. package/src/data/groups.ts +0 -308
  106. package/src/data/images.ts +0 -606
  107. package/src/data/ios.ts +0 -249
  108. package/src/data/locations.ts +0 -114
  109. package/src/data/messages.ts +0 -237
  110. package/src/data/notifications.ts +0 -48
  111. package/src/data/payments.ts +0 -675
  112. package/src/data/posts.ts +0 -580
  113. package/src/data/reactions.ts +0 -186
  114. package/src/data/s3.ts +0 -117
  115. package/src/data/search.ts +0 -74
  116. package/src/data/sms.ts +0 -60
  117. package/src/data/subscription.ts +0 -228
  118. package/src/data/tags.ts +0 -230
  119. package/src/data/users.ts +0 -254
  120. package/src/index.ts +0 -7
  121. package/src/types/apps.ts +0 -56
  122. package/src/types/arangodb.ts +0 -23
  123. package/src/types/auth.ts +0 -20
  124. package/src/types/conversations.ts +0 -11
  125. package/src/types/email.ts +0 -17
  126. package/src/types/files.ts +0 -31
  127. package/src/types/google.ts +0 -37
  128. package/src/types/groups.ts +0 -27
  129. package/src/types/images.ts +0 -32
  130. package/src/types/locations.ts +0 -24
  131. package/src/types/messages.ts +0 -16
  132. package/src/types/notifications.ts +0 -26
  133. package/src/types/payments.ts +0 -129
  134. package/src/types/posts.ts +0 -34
  135. package/src/types/reactions.ts +0 -8
  136. package/src/types/tags.ts +0 -13
  137. package/src/types/users.ts +0 -89
  138. package/src/utils/analytics.ts +0 -41
  139. package/src/utils/arangodb.ts +0 -100
  140. package/src/utils/auth.ts +0 -61
  141. package/src/utils/graphql.ts +0 -7
  142. package/src/utils/objects.ts +0 -34
  143. package/src/utils/redis.ts +0 -17
  144. package/tsconfig.json +0 -45
@@ -0,0 +1,578 @@
1
+ function _templateObject8() {
2
+ var data = _taggedTemplateLiteral(["FOR p IN posts\n FILTER p.added < DATE_TIMESTAMP(DATE_SUBTRACT(DATE_NOW(), 60, 'day')) && p.type == \"default\"\n REMOVE p IN posts\n RETURN OLD"]);
3
+
4
+ _templateObject8 = function _templateObject8() {
5
+ return data;
6
+ };
7
+
8
+ return data;
9
+ }
10
+
11
+ function _templateObject7() {
12
+ var data = _taggedTemplateLiteral(["FOR f IN hasFile\n FILTER f._to == ", "\n REMOVE f IN hasFile"]);
13
+
14
+ _templateObject7 = function _templateObject7() {
15
+ return data;
16
+ };
17
+
18
+ return data;
19
+ }
20
+
21
+ function _templateObject6() {
22
+ var data = _taggedTemplateLiteral(["FOR t IN isTagged\n FILTER t._to == ", "\n REMOVE t IN isTagged"]);
23
+
24
+ _templateObject6 = function _templateObject6() {
25
+ return data;
26
+ };
27
+
28
+ return data;
29
+ }
30
+
31
+ function _templateObject5() {
32
+ var data = _taggedTemplateLiteral(["FOR p IN posts\n FILTER p._key == ", " && p.userId == ", "\n LIMIT 1\n REMOVE p IN posts\n RETURN OLD"]);
33
+
34
+ _templateObject5 = function _templateObject5() {
35
+ return data;
36
+ };
37
+
38
+ return data;
39
+ }
40
+
41
+ function _templateObject4() {
42
+ var data = _taggedTemplateLiteral(["UPSERT {_key: ", ", userId: ", "}\n INSERT ", "\n UPDATE ", "\n IN posts RETURN NEW"]);
43
+
44
+ _templateObject4 = function _templateObject4() {
45
+ return data;
46
+ };
47
+
48
+ return data;
49
+ }
50
+
51
+ 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; }
52
+
53
+ 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; }
54
+
55
+ function _templateObject3() {
56
+ var data = _taggedTemplateLiteral(["INSERT ", " IN posts RETURN NEW"]);
57
+
58
+ _templateObject3 = function _templateObject3() {
59
+ return data;
60
+ };
61
+
62
+ return data;
63
+ }
64
+
65
+ function _templateObject2() {
66
+ var data = _taggedTemplateLiteral(["FOR p IN posts\n FILTER p.type == ", " && p._key == ", "\n LIMIT 1\n RETURN p"]);
67
+
68
+ _templateObject2 = function _templateObject2() {
69
+ return data;
70
+ };
71
+
72
+ return data;
73
+ }
74
+
75
+ function _templateObject() {
76
+ var data = _taggedTemplateLiteral(["FOR p IN posts\n FILTER p._key == ", "\n LIMIT 1\n RETURN p"]);
77
+
78
+ _templateObject = function _templateObject() {
79
+ return data;
80
+ };
81
+
82
+ return data;
83
+ }
84
+
85
+ function _taggedTemplateLiteral(strings, raw) { if (!raw) { raw = strings.slice(0); } return Object.freeze(Object.defineProperties(strings, { raw: { value: Object.freeze(raw) } })); }
86
+
87
+ /**
88
+ * Copyright (c) 2019-Present, Nitrogen Labs, Inc.
89
+ * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.
90
+ */
91
+ import { createHash, parseChar, parseId, parseNum, parseString, parseVarChar } from '@nlabs/utils';
92
+ import { aql } from 'arangojs';
93
+ import flatten from 'lodash/flatten';
94
+ import uniqBy from 'lodash/uniqBy';
95
+ import { getLimit, useDb } from '../utils';
96
+ import { updateFiles } from './files';
97
+ import { extractTags } from './tags'; // const eventCategory: string = 'posts';
98
+
99
+ var MAX_CONTENT_LENGTH = 100000;
100
+ export var parsePostOptions = function parsePostOptions() {
101
+ var options = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
102
+ var _options$from = options.from,
103
+ from = _options$from === void 0 ? 0 : _options$from,
104
+ latitude = options.latitude,
105
+ longitude = options.longitude,
106
+ _options$to = options.to,
107
+ to = _options$to === void 0 ? 30 : _options$to,
108
+ _options$type = options.type,
109
+ type = _options$type === void 0 ? 'default' : _options$type;
110
+ return {
111
+ latitude: parseNum(latitude, 32),
112
+ longitude: parseNum(longitude, 32),
113
+ limit: getLimit(from, to),
114
+ type: parseChar(type, 32)
115
+ };
116
+ };
117
+ export var getPostOptional = function getPostOptional(fields) {
118
+ return fields.reduce(function (selects, field) {
119
+ switch (field) {
120
+ case 'reactions':
121
+ {
122
+ selects.queries.push("LET reactions = (\n FOR post, r IN INBOUND p._id reactions\n COLLECT reactionName = r.value INTO reactionItems\n RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}\n )");
123
+ selects.objects.push('reactions:reactions');
124
+ return selects;
125
+ }
126
+
127
+ case 'tags':
128
+ {
129
+ selects.queries.push("LET tags = (\n FOR t, pl IN INBOUND p._id isTagged\n RETURN t\n )");
130
+ selects.objects.push('tags:tags');
131
+ return selects;
132
+ }
133
+
134
+ case 'user':
135
+ {
136
+ selects.queries.push("LET user = FIRST(\n FOR u IN users\n FILTER p.userId == u._key\n LIMIT 1\n RETURN u\n )");
137
+ selects.objects.push('user:user');
138
+ return selects;
139
+ }
140
+
141
+ default:
142
+ {
143
+ return selects;
144
+ }
145
+ }
146
+ }, {
147
+ objects: [],
148
+ queries: []
149
+ });
150
+ };
151
+ export var getPostList = function getPostList(context, options) {
152
+ // const action: string = 'getListByApp';
153
+ var database = context.database,
154
+ fields = context.fields;
155
+
156
+ var _parsePostOptions = parsePostOptions(options),
157
+ limit = _parsePostOptions.limit,
158
+ type = _parsePostOptions.type;
159
+
160
+ var _getPostOptional = getPostOptional(fields),
161
+ selectObjects = _getPostOptional.objects,
162
+ selectQueries = _getPostOptional.queries;
163
+
164
+ var aqlQry = "FOR p IN posts\n FILTER p.type == \"".concat(type, "\" && p.privacy == \"public\" && p.parent == null\n ").concat(selectQueries.join('\n'), "\n ").concat(limit.aql, "\n SORT p.added\n RETURN DISTINCT MERGE(p, {").concat(selectObjects.join(', '), "})");
165
+ return useDb(database).query(aqlQry).then(function (cursor) {
166
+ return cursor.all();
167
+ }).catch(function (error) {
168
+ throw error;
169
+ });
170
+ };
171
+ export var getPostListByGroup = function getPostListByGroup(context, groupId, options) {
172
+ // const action: string = 'getListByGroup';
173
+ var database = context.database,
174
+ fields = context.fields,
175
+ sessionId = context.userId;
176
+
177
+ var _getPostOptional2 = getPostOptional(fields),
178
+ selectObjects = _getPostOptional2.objects,
179
+ selectQueries = _getPostOptional2.queries; // Group id
180
+
181
+
182
+ var formatGroupId = parseId(groupId);
183
+ var db = useDb(database);
184
+ var aqlQry = "FOR u, g IN INBOUND ".concat(formatGroupId, " hasGroup\n FILTER u._key == ").concat(sessionId, "\n RETURN g");
185
+ return db.query(aqlQry).then(function (cursor) {
186
+ return cursor.all();
187
+ }).then(function () {
188
+ var groups = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [];
189
+
190
+ if (groups.length) {
191
+ var _parsePostOptions2 = parsePostOptions(options),
192
+ limit = _parsePostOptions2.limit,
193
+ type = _parsePostOptions2.type;
194
+
195
+ var postAqlQry = "FOR p IN posts\n FILTER p.type == \"".concat(type, "\" && p.groupId == \"").concat(formatGroupId, "\" && p.parent == null\n ").concat(selectQueries.join('\n'), "\n ").concat(limit.aql, "\n SORT p.added\n RETURN DISTINCT MERGE(p, {").concat(selectObjects.join(', '), "})");
196
+ return db.query(postAqlQry).then(function (cursor) {
197
+ return cursor.all();
198
+ }).catch(function (error) {
199
+ throw error;
200
+ });
201
+ }
202
+
203
+ return [];
204
+ }).catch(function (error) {
205
+ throw error;
206
+ });
207
+ };
208
+ export var getPostListByLatest = function getPostListByLatest(context, options) {
209
+ // const action: string = 'getListByLatest';
210
+ var database = context.database,
211
+ fields = context.fields;
212
+
213
+ var _parsePostOptions3 = parsePostOptions(options),
214
+ limit = _parsePostOptions3.limit,
215
+ type = _parsePostOptions3.type;
216
+
217
+ var _getPostOptional3 = getPostOptional(fields),
218
+ selectObjects = _getPostOptional3.objects,
219
+ selectQueries = _getPostOptional3.queries;
220
+
221
+ var aqlQry = "FOR p IN posts\n FILTER p.type == \"".concat(type, "\" && p.privacy == \"public\" && p.parent == null\n ").concat(selectQueries.join('\n'), "\n ").concat(limit.aql, "\n SORT p.added\n RETURN DISTINCT MERGE(p, {").concat(selectObjects.join(', '), "})");
222
+ return useDb(database).query(aqlQry).then(function (cursor) {
223
+ return cursor.all();
224
+ }).catch(function (error) {
225
+ throw error;
226
+ });
227
+ };
228
+ export var getPostListByTags = function getPostListByTags(context, tagNames, options) {
229
+ // const action: string = 'getListByTags';
230
+ var database = context.database,
231
+ fields = context.fields;
232
+
233
+ var _parsePostOptions4 = parsePostOptions(options),
234
+ latitude = _parsePostOptions4.latitude,
235
+ longitude = _parsePostOptions4.longitude,
236
+ limit = _parsePostOptions4.limit,
237
+ type = _parsePostOptions4.type;
238
+
239
+ var _getPostOptional4 = getPostOptional(fields),
240
+ selectObjects = _getPostOptional4.objects,
241
+ selectQueries = _getPostOptional4.queries;
242
+
243
+ var sortBy = [];
244
+
245
+ if (latitude !== undefined && longitude !== undefined) {
246
+ var formatLatitude = parseNum(latitude);
247
+ var formatLongitude = parseNum(longitude);
248
+ selectQueries.push("LET distance = DISTANCE(\n ".concat(formatLatitude, ",\n ").concat(formatLongitude, ",\n NOT_NULL(p.latitude, 0),\n NOT_NULL(p.longitude, 0))\n "));
249
+ selectObjects.push('distance:distance');
250
+ sortBy.push('distance');
251
+ }
252
+
253
+ sortBy.push('p.added');
254
+ return Promise.all(tagNames.map(function (tagName) {
255
+ var aqlQry = "FOR targetTag IN tags\n FILTER targetTag.name == \"".concat(tagName, "\"\n FOR p, e IN OUTBOUND targetTag._id isTagged\n ").concat(selectQueries.join('\n'), "\n FILTER p.type = \"").concat(type, "\" && p.privacy == \"public\" && e.type == 'posts'\n ").concat(limit.aql, "\n SORT ").concat(sortBy.join(', '), "\n RETURN DISTINCT MERGE(p, {").concat(selectObjects.join(', '), "})");
256
+ console.log(aqlQry);
257
+ return useDb(database).query(aqlQry).then(function (cursor) {
258
+ return cursor.all();
259
+ }).catch(function () {
260
+ return [];
261
+ });
262
+ })).then(function (results) {
263
+ return uniqBy(flatten(results), '_key');
264
+ }).catch(function (error) {
265
+ throw error;
266
+ });
267
+ };
268
+ export var getPostListByUser = function getPostListByUser(context, userId, options) {
269
+ // const action: string = 'getListByUser';
270
+ var database = context.database,
271
+ fields = context.fields;
272
+
273
+ var _parsePostOptions5 = parsePostOptions(options),
274
+ limit = _parsePostOptions5.limit,
275
+ type = _parsePostOptions5.type;
276
+
277
+ var formatUserId = parseId(userId);
278
+
279
+ var _getPostOptional5 = getPostOptional(fields),
280
+ selectObjects = _getPostOptional5.objects,
281
+ selectQueries = _getPostOptional5.queries;
282
+
283
+ var aqlQry = "FOR p IN posts\n FILTER p.userId == \"".concat(formatUserId, "\" && p.type == \"").concat(type, "\" && p.privacy == \"public\" && p.parent == null\n ").concat(selectQueries.join('\n'), "\n ").concat(limit.aql, "\n SORT p.added\n RETURN DISTINCT MERGE(p, {").concat(selectObjects.join(', '), "})");
284
+ return useDb(database).query(aqlQry).then(function (cursor) {
285
+ return cursor.all();
286
+ }).catch(function (error) {
287
+ throw error;
288
+ });
289
+ };
290
+ export var getPostListByArea = function getPostListByArea(context, latitude, longitude, options) {
291
+ // const action: string = 'getListByUser';
292
+ var database = context.database,
293
+ fields = context.fields;
294
+
295
+ var _parsePostOptions6 = parsePostOptions(options),
296
+ limit = _parsePostOptions6.limit,
297
+ type = _parsePostOptions6.type;
298
+
299
+ var formatLatitude = parseNum(latitude);
300
+ var formatLongitude = parseNum(longitude);
301
+
302
+ var _getPostOptional6 = getPostOptional(fields),
303
+ selectObjects = _getPostOptional6.objects,
304
+ selectQueries = _getPostOptional6.queries;
305
+
306
+ selectQueries.push("LET distance = DISTANCE(\n ".concat(formatLatitude, ",\n ").concat(formatLongitude, ",\n NOT_NULL(p.latitude, 0),\n NOT_NULL(p.longitude, 0))\n "));
307
+ selectObjects.push('distance:distance');
308
+ var aqlQry = "FOR p IN posts\n ".concat(selectQueries.join('\n'), "\n FILTER p.type == \"").concat(type, "\" && p.privacy == \"public\" && p.parentId == null\n ").concat(limit.aql, "\n SORT distance, p.added\n RETURN DISTINCT MERGE(p, {").concat(selectObjects.join(', '), "})");
309
+ return useDb(database).query(aqlQry).then(function (cursor) {
310
+ return cursor.all();
311
+ }).catch(function (error) {
312
+ throw error;
313
+ });
314
+ };
315
+ export var getPost = function getPost(context, itemId) {
316
+ // const action: string = 'getItem';
317
+ var database = context.database,
318
+ fields = context.fields,
319
+ sessionId = context.userId;
320
+ var formatItemId = parseId(itemId);
321
+ var db = useDb(database);
322
+
323
+ var _getPostOptional7 = getPostOptional(fields),
324
+ selectObjects = _getPostOptional7.objects,
325
+ selectQueries = _getPostOptional7.queries;
326
+
327
+ var aqlQry = aql(_templateObject(), formatItemId);
328
+ return db.query(aqlQry).then(function (cursor) {
329
+ return cursor.next();
330
+ }).then(function () {
331
+ var post = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
332
+ var _key = post._key,
333
+ groupId = post.groupId,
334
+ _post$privacy = post.privacy,
335
+ privacy = _post$privacy === void 0 ? 'default' : _post$privacy; // Query based on privacy level
336
+
337
+ var privacyAqlQry;
338
+
339
+ if (groupId && privacy === 'group') {
340
+ privacyAqlQry = "FOR p IN posts\n FILTER p._key == \"".concat(_key, "\"\n ").concat(selectQueries.join('\n'), "\n FOR group IN groups\n FILTER group._key == p.groupId\n FOR u, e IN OUTBOUND group._id isGrouped\n FILTER u._key == ").concat(sessionId, "\n LIMIT 1\n RETURN MERGE(p, {").concat(selectObjects.join(', '), "})");
341
+ } else if (privacy === 'public') {
342
+ privacyAqlQry = "FOR p IN posts\n FILTER p._key == \"".concat(_key, "\"\n ").concat(selectQueries.join('\n'), "\n LIMIT 1\n RETURN MERGE(p, {").concat(selectObjects.join(', '), "})");
343
+ }
344
+
345
+ if (privacyAqlQry) {
346
+ return db.query(privacyAqlQry).then(function (cursor) {
347
+ return cursor.next();
348
+ }).then(function () {
349
+ var filteredPost = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
350
+ return filteredPost;
351
+ }).catch(function (error) {
352
+ throw error;
353
+ });
354
+ }
355
+
356
+ return {};
357
+ }).catch(function (error) {
358
+ throw error;
359
+ });
360
+ };
361
+ export var getPostComments = function getPostComments(context, itemId, options) {
362
+ // const action: string = 'getComments';
363
+ var database = context.database,
364
+ sessionId = context.userId;
365
+
366
+ var _parsePostOptions7 = parsePostOptions(options),
367
+ limit = _parsePostOptions7.limit,
368
+ type = _parsePostOptions7.type;
369
+
370
+ var formatItemId = parseId(itemId); // Get the parent post to get restrictions
371
+
372
+ var db = useDb(database);
373
+ var aqlQry = aql(_templateObject2(), type, formatItemId);
374
+ return db.query(aqlQry).then(function (cursor) {
375
+ return cursor.next();
376
+ }).then(function () {
377
+ var post = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
378
+ var _key = post._key,
379
+ groupId = post.groupId,
380
+ _post$privacy2 = post.privacy,
381
+ privacy = _post$privacy2 === void 0 ? 'public' : _post$privacy2; // Query based on privacy level
382
+
383
+ var privacyAqlQry;
384
+
385
+ if (groupId && privacy === 'group') {
386
+ privacyAqlQry = "FOR p IN posts\n FOR user IN users\n FILTER p.parent == \"".concat(_key, "\" && user._key == p.userId\n LET reactions = (\n FOR post, r IN INBOUND p._id reactions\n COLLECT reactionName = r.value INTO reactionItems\n RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}\n )\n FOR group IN groups\n FILTER group._key == p.groupId\n FOR u, e IN OUTBOUND group._id isGrouped\n FILTER u._key == \"").concat(sessionId, "\"\n SORT p.added\n ").concat(limit.aql, "\n RETURN MERGE(p, {user: user, reactions: reactions})");
387
+ } else if (privacy === 'public') {
388
+ privacyAqlQry = "FOR p IN posts\n FOR user IN users\n FILTER p.parent == \"".concat(_key, "\" && user._key == p.userId\n LET reactions = (\n FOR post, r IN INBOUND p._id reactions\n COLLECT reactionName = r.value INTO reactionItems\n RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}\n )\n SORT p.added\n ").concat(limit.aql, "\n RETURN MERGE(p, {user: user, reactions: reactions})");
389
+ }
390
+
391
+ if (privacyAqlQry) {
392
+ return db.query(privacyAqlQry).then(function (cursor) {
393
+ return cursor.all();
394
+ }).catch(function (error) {
395
+ throw error;
396
+ });
397
+ }
398
+
399
+ return [];
400
+ }).catch(function (error) {
401
+ throw error;
402
+ });
403
+ };
404
+ export var addPost = function addPost(context, item) {
405
+ // const action: string = 'add';
406
+ var database = context.database,
407
+ sessionId = context.userId;
408
+ var _item$content = item.content,
409
+ content = _item$content === void 0 ? '' : _item$content,
410
+ _item$groupId = item.groupId,
411
+ groupId = _item$groupId === void 0 ? '' : _item$groupId,
412
+ location = item.location,
413
+ latitude = item.latitude,
414
+ longitude = item.longitude,
415
+ _item$name = item.name,
416
+ name = _item$name === void 0 ? '' : _item$name,
417
+ _item$parentId = item.parentId,
418
+ parentId = _item$parentId === void 0 ? null : _item$parentId,
419
+ _item$privacy = item.privacy,
420
+ privacy = _item$privacy === void 0 ? 'public' : _item$privacy,
421
+ _item$type = item.type,
422
+ type = _item$type === void 0 ? 'default' : _item$type;
423
+ var now = Date.now();
424
+ var insert = {
425
+ _key: createHash("post-".concat(sessionId)),
426
+ added: now,
427
+ content: parseString(content, MAX_CONTENT_LENGTH),
428
+ groupId: groupId ? parseId(groupId) : undefined,
429
+ latitude: latitude !== undefined ? parseNum(latitude) : undefined,
430
+ location: location ? parseString(location, 160) : undefined,
431
+ longitude: longitude !== undefined ? parseNum(longitude) : undefined,
432
+ modified: now,
433
+ name: parseString(name, 160),
434
+ parentId: parentId ? parseId(parentId) : undefined,
435
+ privacy: privacy ? parseVarChar(privacy, 16) : undefined,
436
+ type: parseChar(type, 32),
437
+ userId: sessionId
438
+ };
439
+ var db = useDb(database);
440
+ var aqlQry = aql(_templateObject3(), insert);
441
+ return db.query(aqlQry).then(function (cursor) {
442
+ return cursor.next();
443
+ }).then(function () {
444
+ var post = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
445
+ var postKey = post._key; // Update linked tags within posts
446
+
447
+ return extractTags(db, 'posts', postKey, insert.content).then(function (tagList) {
448
+ post.tags = tagList;
449
+ return post;
450
+ });
451
+ }).catch(function (error) {
452
+ throw error;
453
+ });
454
+ };
455
+ export var updatePost = function updatePost(context, item) {
456
+ // const action: string = 'update';
457
+ var database = context.database,
458
+ sessionId = context.userId;
459
+ var now = Date.now();
460
+ var content = item.content,
461
+ groupId = item.groupId,
462
+ name = item.name,
463
+ parentId = item.parentId,
464
+ postId = item.postId,
465
+ privacy = item.privacy,
466
+ type = item.type;
467
+ var update = {
468
+ content: content ? parseString(content, MAX_CONTENT_LENGTH) : undefined,
469
+ modified: now,
470
+ name: name ? parseString(name, 160) : undefined,
471
+ parentId: parentId ? parseString(parentId, 160) : undefined,
472
+ privacy: privacy ? parseVarChar(privacy, 16) : undefined,
473
+ type: type !== undefined ? parseChar(type, 16) : undefined
474
+ };
475
+ var formatId = parseId(postId);
476
+ formatId = formatId === '' ? createHash("post-".concat(sessionId)) : formatId;
477
+ var formatGroupId = parseId(groupId);
478
+
479
+ var insert = _objectSpread({}, update, {
480
+ _key: formatId,
481
+ added: now,
482
+ groupId: formatGroupId,
483
+ privacy: privacy,
484
+ userId: sessionId
485
+ });
486
+
487
+ var db = useDb(database);
488
+ var aqlQry = aql(_templateObject4(), formatId, sessionId, insert, update);
489
+ return db.query(aqlQry).then(function (cursor) {
490
+ return cursor.next();
491
+ }).then(function () {
492
+ var updatedPost = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
493
+ var updatedPostKey = updatedPost._key; // Update linked tags
494
+
495
+ return extractTags(db, 'posts', updatedPostKey, update.content || '').then(function () {
496
+ var tagList = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [];
497
+ updatedPost.tags = tagList; // Update linked files
498
+
499
+ var files = updatedPost.files || [];
500
+
501
+ if (files.length) {
502
+ return updateFiles(db, formatId, files).then(function () {
503
+ var fileList = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [];
504
+ updatedPost.files = fileList;
505
+ return updatedPost;
506
+ });
507
+ }
508
+
509
+ updatedPost.files = [];
510
+ return updatedPost;
511
+ });
512
+ }).catch(function (error) {
513
+ throw error;
514
+ });
515
+ };
516
+ export var deletePost = function deletePost(context, itemId) {
517
+ // const action: string = 'delete';
518
+ var database = context.database,
519
+ sessionId = context.userId;
520
+ var formatItemId = parseId(itemId);
521
+ var db = useDb(database);
522
+ var aqlQry = aql(_templateObject5(), formatItemId, sessionId);
523
+ return db.query(aqlQry).then(function (cursor) {
524
+ return cursor.next();
525
+ }).then(function () {
526
+ var post = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
527
+
528
+ if (post) {
529
+ // Remove tag links
530
+ var edgeAqlQry = aql(_templateObject6(), formatItemId);
531
+ return db.query(edgeAqlQry).then(function () {
532
+ // Remove attached files
533
+ var fileAqlQry = aql(_templateObject7(), formatItemId);
534
+ return db.query(fileAqlQry).then(function () {
535
+ return post;
536
+ }).catch(function (error) {
537
+ throw error;
538
+ });
539
+ }).catch(function (error) {
540
+ throw error;
541
+ });
542
+ }
543
+
544
+ return {};
545
+ }).catch(function (error) {
546
+ throw error;
547
+ });
548
+ };
549
+ export var cleanPosts = function cleanPosts(database) {
550
+ // Remove all messages that are over 60 days and not saved
551
+ var aqlQry = aql(_templateObject8());
552
+ return useDb(database).query(aqlQry).then(function (cursor) {
553
+ return cursor.all();
554
+ }).then(function () {
555
+ var results = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [];
556
+ return results.length;
557
+ }).catch(function (error) {
558
+ throw error;
559
+ });
560
+ };
561
+ export var createPostEdge = function createPostEdge(db, file, postId) {
562
+ var edgeCollection = db.edgeCollection('isPosted');
563
+ var fileId = parseId(file.id);
564
+ var edgeId = createHash("file-".concat(postId, "-").concat(fileId));
565
+ var formatPostId = parseId(postId);
566
+ var fileType = parseChar(file.fileType, 16);
567
+ var edge = {
568
+ _key: edgeId,
569
+ added: Date.now(),
570
+ type: fileType
571
+ };
572
+ return edgeCollection.save(edge, "posts/".concat(formatPostId), "files/".concat(fileId)).then(function () {
573
+ return file;
574
+ }).catch(function (error) {
575
+ throw error;
576
+ });
577
+ };
578
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kYXRhL3Bvc3RzLnRzIl0sIm5hbWVzIjpbImNyZWF0ZUhhc2giLCJwYXJzZUNoYXIiLCJwYXJzZUlkIiwicGFyc2VOdW0iLCJwYXJzZVN0cmluZyIsInBhcnNlVmFyQ2hhciIsImFxbCIsImZsYXR0ZW4iLCJ1bmlxQnkiLCJnZXRMaW1pdCIsInVzZURiIiwidXBkYXRlRmlsZXMiLCJleHRyYWN0VGFncyIsIk1BWF9DT05URU5UX0xFTkdUSCIsInBhcnNlUG9zdE9wdGlvbnMiLCJvcHRpb25zIiwiZnJvbSIsImxhdGl0dWRlIiwibG9uZ2l0dWRlIiwidG8iLCJ0eXBlIiwibGltaXQiLCJnZXRQb3N0T3B0aW9uYWwiLCJmaWVsZHMiLCJyZWR1Y2UiLCJzZWxlY3RzIiwiZmllbGQiLCJxdWVyaWVzIiwicHVzaCIsIm9iamVjdHMiLCJnZXRQb3N0TGlzdCIsImNvbnRleHQiLCJkYXRhYmFzZSIsInNlbGVjdE9iamVjdHMiLCJzZWxlY3RRdWVyaWVzIiwiYXFsUXJ5Iiwiam9pbiIsInF1ZXJ5IiwidGhlbiIsImN1cnNvciIsImFsbCIsImNhdGNoIiwiZXJyb3IiLCJnZXRQb3N0TGlzdEJ5R3JvdXAiLCJncm91cElkIiwic2Vzc2lvbklkIiwidXNlcklkIiwiZm9ybWF0R3JvdXBJZCIsImRiIiwiZ3JvdXBzIiwibGVuZ3RoIiwicG9zdEFxbFFyeSIsImdldFBvc3RMaXN0QnlMYXRlc3QiLCJnZXRQb3N0TGlzdEJ5VGFncyIsInRhZ05hbWVzIiwic29ydEJ5IiwidW5kZWZpbmVkIiwiZm9ybWF0TGF0aXR1ZGUiLCJmb3JtYXRMb25naXR1ZGUiLCJQcm9taXNlIiwibWFwIiwidGFnTmFtZSIsImNvbnNvbGUiLCJsb2ciLCJyZXN1bHRzIiwiZ2V0UG9zdExpc3RCeVVzZXIiLCJmb3JtYXRVc2VySWQiLCJnZXRQb3N0TGlzdEJ5QXJlYSIsImdldFBvc3QiLCJpdGVtSWQiLCJmb3JtYXRJdGVtSWQiLCJuZXh0IiwicG9zdCIsIl9rZXkiLCJwcml2YWN5IiwicHJpdmFjeUFxbFFyeSIsImZpbHRlcmVkUG9zdCIsImdldFBvc3RDb21tZW50cyIsImFkZFBvc3QiLCJpdGVtIiwiY29udGVudCIsImxvY2F0aW9uIiwibmFtZSIsInBhcmVudElkIiwibm93IiwiRGF0ZSIsImluc2VydCIsImFkZGVkIiwibW9kaWZpZWQiLCJwb3N0S2V5IiwidGFnTGlzdCIsInRhZ3MiLCJ1cGRhdGVQb3N0IiwicG9zdElkIiwidXBkYXRlIiwiZm9ybWF0SWQiLCJ1cGRhdGVkUG9zdCIsInVwZGF0ZWRQb3N0S2V5IiwiZmlsZXMiLCJmaWxlTGlzdCIsImRlbGV0ZVBvc3QiLCJlZGdlQXFsUXJ5IiwiZmlsZUFxbFFyeSIsImNsZWFuUG9zdHMiLCJjcmVhdGVQb3N0RWRnZSIsImZpbGUiLCJlZGdlQ29sbGVjdGlvbiIsImZpbGVJZCIsImlkIiwiZWRnZUlkIiwiZm9ybWF0UG9zdElkIiwiZmlsZVR5cGUiLCJlZGdlIiwic2F2ZSJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQTs7OztBQUlBLFNBQVFBLFVBQVIsRUFBb0JDLFNBQXBCLEVBQStCQyxPQUEvQixFQUF3Q0MsUUFBeEMsRUFBa0RDLFdBQWxELEVBQStEQyxZQUEvRCxRQUFrRixjQUFsRjtBQUNBLFNBQVFDLEdBQVIsUUFBNEIsVUFBNUI7QUFHQSxPQUFPQyxPQUFQLE1BQW9CLGdCQUFwQjtBQUNBLE9BQU9DLE1BQVAsTUFBbUIsZUFBbkI7QUFHQSxTQUFRQyxRQUFSLEVBQWtCQyxLQUFsQixRQUE4QixVQUE5QjtBQUNBLFNBQVFDLFdBQVIsUUFBMEIsU0FBMUI7QUFDQSxTQUFRQyxXQUFSLFFBQTBCLFFBQTFCLEMsQ0FFQTs7QUFDQSxJQUFNQyxrQkFBMEIsR0FBRyxNQUFuQztBQUVBLE9BQU8sSUFBTUMsZ0JBQWdCLEdBQUcsU0FBbkJBLGdCQUFtQixHQUErQjtBQUFBLE1BQTlC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QWdCLEdBQUc5RyxTQUFTLENBQUN3RyxJQUFJLENBQUNNLFFBQU4sRUFBZ0IsRUFBaEIsQ0FBbEM7QUFFQSxNQUFNQyxJQUFTLEdBQUc7QUFDaEJ2QyxJQUFBQSxJQUFJLEVBQUVvQyxNQURVO0FBRWhCdEIsSUFBQUEsS0FBSyxFQUFFRixJQUFJLENBQUNELEdBQUwsRUFGUztBQUdoQmhFLElBQUFBLElBQUksRUFBRTJGO0FBSFUsR0FBbEI7QUFNQSxTQUFPTCxjQUFjLENBQUNPLElBQWYsQ0FBb0JELElBQXBCLGtCQUFtQ0YsWUFBbkMsbUJBQTRESCxNQUE1RCxHQUNKckUsSUFESSxDQUNDO0FBQUEsV0FBTW1FLElBQU47QUFBQSxHQURELEVBRUpoRSxLQUZJLENBRUUsVUFBQ0MsS0FBRCxFQUFrQjtBQUN2QixVQUFNQSxLQUFOO0FBQ0QsR0FKSSxDQUFQO0FBS0QsQ0FsQk0iLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIENvcHlyaWdodCAoYykgMjAxOS1QcmVzZW50LCBOaXRyb2dlbiBMYWJzLCBJbmMuXG4gKiBDb3B5cmlnaHRzIGxpY2Vuc2VkIHVuZGVyIHRoZSBNSVQgTGljZW5zZS4gU2VlIHRoZSBhY2NvbXBhbnlpbmcgTElDRU5TRSBmaWxlIGZvciB0ZXJtcy5cbiAqL1xuaW1wb3J0IHtjcmVhdGVIYXNoLCBwYXJzZUNoYXIsIHBhcnNlSWQsIHBhcnNlTnVtLCBwYXJzZVN0cmluZywgcGFyc2VWYXJDaGFyfSBmcm9tICdAbmxhYnMvdXRpbHMnO1xuaW1wb3J0IHthcWwsIERhdGFiYXNlfSBmcm9tICdhcmFuZ29qcyc7XG5pbXBvcnQge0FxbFF1ZXJ5fSBmcm9tICdhcmFuZ29qcy9saWIvY2pzL2FxbC1xdWVyeSc7XG5pbXBvcnQge0FycmF5Q3Vyc29yfSBmcm9tICdhcmFuZ29qcy9saWIvY2pzL2N1cnNvcic7XG5pbXBvcnQgZmxhdHRlbiBmcm9tICdsb2Rhc2gvZmxhdHRlbic7XG5pbXBvcnQgdW5pcUJ5IGZyb20gJ2xvZGFzaC91bmlxQnknO1xuXG5pbXBvcnQge0FwaUNvbnRleHQsIEZpbGVUeXBlLCBHcm91cFR5cGUsIFBvc3RPcHRpb25zLCBQb3N0VHlwZSwgVGFnVHlwZX0gZnJvbSAnLi4vdHlwZXMnO1xuaW1wb3J0IHtnZXRMaW1pdCwgdXNlRGJ9IGZyb20gJy4uL3V0aWxzJztcbmltcG9ydCB7dXBkYXRlRmlsZXN9IGZyb20gJy4vZmlsZXMnO1xuaW1wb3J0IHtleHRyYWN0VGFnc30gZnJvbSAnLi90YWdzJztcblxuLy8gY29uc3QgZXZlbnRDYXRlZ29yeTogc3RyaW5nID0gJ3Bvc3RzJztcbmNvbnN0IE1BWF9DT05URU5UX0xFTkdUSDogbnVtYmVyID0gMTAwMDAwO1xuXG5leHBvcnQgY29uc3QgcGFyc2VQb3N0T3B0aW9ucyA9IChvcHRpb25zOiBQb3N0T3B0aW9ucyA9IHt9KSA9PiB7XG4gIGNvbnN0IHtcbiAgICBmcm9tID0gMCxcbiAgICBsYXRpdHVkZSxcbiAgICBsb25naXR1ZGUsXG4gICAgdG8gPSAzMCxcbiAgICB0eXBlID0gJ2RlZmF1bHQnXG4gIH0gPSBvcHRpb25zO1xuXG5cbiAgcmV0dXJuIHtcbiAgICBsYXRpdHVkZTogcGFyc2VOdW0obGF0aXR1ZGUsIDMyKSxcbiAgICBsb25naXR1ZGU6IHBhcnNlTnVtKGxvbmdpdHVkZSwgMzIpLFxuICAgIGxpbWl0OiBnZXRMaW1pdChmcm9tLCB0byksXG4gICAgdHlwZTogcGFyc2VDaGFyKHR5cGUsIDMyKVxuICB9O1xufTtcblxuZXhwb3J0IGNvbnN0IGdldFBvc3RPcHRpb25hbCA9IChmaWVsZHM6IHN0cmluZ1tdKSA9PlxuICBmaWVsZHMucmVkdWNlKChzZWxlY3RzOiBhbnksIGZpZWxkOiBzdHJpbmcpID0+IHtcbiAgICBzd2l0Y2goZmllbGQpIHtcbiAgICAgIGNhc2UgJ3JlYWN0aW9ucyc6IHtcbiAgICAgICAgc2VsZWN0cy5xdWVyaWVzLnB1c2goYExFVCByZWFjdGlvbnMgPSAoXG4gICAgICAgICAgRk9SIHBvc3QsIHIgSU4gSU5CT1VORCBwLl9pZCByZWFjdGlvbnNcbiAgICAgICAgICBDT0xMRUNUIHJlYWN0aW9uTmFtZSA9IHIudmFsdWUgSU5UTyByZWFjdGlvbkl0ZW1zXG4gICAgICAgICAgUkVUVVJOIHtuYW1lOiByZWFjdGlvbk5hbWUsIGNvdW50OiBMRU5HVEgocmVhY3Rpb25JdGVtc1sqXS5yLnZhbHVlKX1cbiAgICAgICAgKWApO1xuICAgICAgICBzZWxlY3RzLm9iamVjdHMucHVzaCgncmVhY3Rpb25zOnJlYWN0aW9ucycpO1xuICAgICAgICByZXR1cm4gc2VsZWN0cztcbiAgICAgIH1cbiAgICAgIGNhc2UgJ3RhZ3MnOiB7XG4gICAgICAgIHNlbGVjdHMucXVlcmllcy5wdXNoKGBMRVQgdGFncyA9IChcbiAgICAgICAgICBGT1IgdCwgcGwgSU4gSU5CT1VORCBwLl9pZCBpc1RhZ2dlZFxuICAgICAgICAgIFJFVFVSTiB0XG4gICAgICAgIClgKTtcbiAgICAgICAgc2VsZWN0cy5vYmplY3RzLnB1c2goJ3RhZ3M6dGFncycpO1xuICAgICAgICByZXR1cm4gc2VsZWN0cztcbiAgICAgIH1cbiAgICAgIGNhc2UgJ3VzZXInOiB7XG4gICAgICAgIHNlbGVjdHMucXVlcmllcy5wdXNoKGBMRVQgdXNlciA9IEZJUlNUKFxuICAgICAgICAgIEZPUiB1IElOIHVzZXJzXG4gICAgICAgICAgRklMVEVSIHAudXNlcklkID09IHUuX2tleVxuICAgICAgICAgIExJTUlUIDFcbiAgICAgICAgICBSRVRVUk4gdVxuICAgICAgICApYCk7XG4gICAgICAgIHNlbGVjdHMub2JqZWN0cy5wdXNoKCd1c2VyOnVzZXInKTtcbiAgICAgICAgcmV0dXJuIHNlbGVjdHM7XG4gICAgICB9XG4gICAgICBkZWZhdWx0OiB7XG4gICAgICAgIHJldHVybiBzZWxlY3RzO1xuICAgICAgfVxuICAgIH1cbiAgfSwge29iamVjdHM6IFtdLCBxdWVyaWVzOiBbXX0pO1xuXG5leHBvcnQgY29uc3QgZ2V0UG9zdExpc3QgPSAoY29udGV4dDogQXBpQ29udGV4dCwgb3B0aW9ucz86IFBvc3RPcHRpb25zKTogUHJvbWlzZTxQb3N0VHlwZVtdPiA9PiB7XG4gIC8vIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2dldExpc3RCeUFwcCc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgZmllbGRzfSA9IGNvbnRleHQ7XG4gIGNvbnN0IHtsaW1pdCwgdHlwZX0gPSBwYXJzZVBvc3RPcHRpb25zKG9wdGlvbnMpO1xuICBjb25zdCB7b2JqZWN0czogc2VsZWN0T2JqZWN0cywgcXVlcmllczogc2VsZWN0UXVlcmllc30gPSBnZXRQb3N0T3B0aW9uYWwoZmllbGRzKTtcbiAgY29uc3QgYXFsUXJ5OiBzdHJpbmcgPSBgRk9SIHAgSU4gcG9zdHNcbiAgICBGSUxURVIgcC50eXBlID09IFwiJHt0eXBlfVwiICYmIHAucHJpdmFjeSA9PSBcInB1YmxpY1wiICYmIHAucGFyZW50ID09IG51bGxcbiAgICAke3NlbGVjdFF1ZXJpZXMuam9pbignXFxuJyl9XG4gICAgJHtsaW1pdC5hcWx9XG4gICAgU09SVCBwLmFkZGVkXG4gICAgUkVUVVJOIERJU1RJTkNUIE1FUkdFKHAsIHske3NlbGVjdE9iamVjdHMuam9pbignLCAnKX19KWA7XG5cbiAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5hbGwoKSlcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgdGhyb3cgZXJyb3I7XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgZ2V0UG9zdExpc3RCeUdyb3VwID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGdyb3VwSWQ6IHN0cmluZywgb3B0aW9ucz86IFBvc3RPcHRpb25zKTogUHJvbWlzZTxQb3N0VHlwZVtdPiA9PiB7XG4gIC8vIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2dldExpc3RCeUdyb3VwJztcbiAgY29uc3Qge2RhdGFiYXNlLCBmaWVsZHMsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG4gIGNvbnN0IHtvYmplY3RzOiBzZWxlY3RPYmplY3RzLCBxdWVyaWVzOiBzZWxlY3RRdWVyaWVzfSA9IGdldFBvc3RPcHRpb25hbChmaWVsZHMpO1xuXG4gIC8vIEdyb3VwIGlkXG4gIGNvbnN0IGZvcm1hdEdyb3VwSWQ6IHN0cmluZyA9IHBhcnNlSWQoZ3JvdXBJZCk7XG4gIGNvbnN0IGRiID0gdXNlRGIoZGF0YWJhc2UpO1xuICBjb25zdCBhcWxRcnk6IHN0cmluZyA9IGBGT1IgdSwgZyBJTiBJTkJPVU5EICR7Zm9ybWF0R3JvdXBJZH0gaGFzR3JvdXBcbiAgICAgIEZJTFRFUiB1Ll9rZXkgPT0gJHtzZXNzaW9uSWR9XG4gICAgICBSRVRVUk4gZ2A7XG5cbiAgcmV0dXJuIGRiLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLmFsbCgpKVxuICAgIC50aGVuKChncm91cHM6IEdyb3VwVHlwZVtdID0gW10pID0+IHtcbiAgICAgIGlmKGdyb3Vwcy5sZW5ndGgpIHtcbiAgICAgICAgY29uc3Qge2xpbWl0LCB0eXBlfSA9IHBhcnNlUG9zdE9wdGlvbnMob3B0aW9ucyk7XG4gICAgICAgIGNvbnN0IHBvc3RBcWxRcnk6IHN0cmluZyA9IGBGT1IgcCBJTiBwb3N0c1xuICAgICAgICAgIEZJTFRFUiBwLnR5cGUgPT0gXCIke3R5cGV9XCIgJiYgcC5ncm91cElkID09IFwiJHtmb3JtYXRHcm91cElkfVwiICYmIHAucGFyZW50ID09IG51bGxcbiAgICAgICAgICAke3NlbGVjdFF1ZXJpZXMuam9pbignXFxuJyl9XG4gICAgICAgICAgJHtsaW1pdC5hcWx9XG4gICAgICAgICAgU09SVCBwLmFkZGVkXG4gICAgICAgICAgUkVUVVJOIERJU1RJTkNUIE1FUkdFKHAsIHske3NlbGVjdE9iamVjdHMuam9pbignLCAnKX19KWA7XG5cbiAgICAgICAgcmV0dXJuIGRiLnF1ZXJ5KHBvc3RBcWxRcnkpXG4gICAgICAgICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5hbGwoKSlcbiAgICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgICAgICAgdGhyb3cgZXJyb3I7XG4gICAgICAgICAgfSk7XG4gICAgICB9XG5cbiAgICAgIHJldHVybiBbXTtcbiAgICB9KVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICB0aHJvdyBlcnJvcjtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRQb3N0TGlzdEJ5TGF0ZXN0ID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIG9wdGlvbnM/OiBQb3N0T3B0aW9ucyk6IFByb21pc2U8UG9zdFR5cGVbXT4gPT4ge1xuICAvLyBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRMaXN0QnlMYXRlc3QnO1xuICBjb25zdCB7ZGF0YWJhc2UsIGZpZWxkc30gPSBjb250ZXh0O1xuICBjb25zdCB7bGltaXQsIHR5cGV9ID0gcGFyc2VQb3N0T3B0aW9ucyhvcHRpb25zKTtcbiAgY29uc3Qge29iamVjdHM6IHNlbGVjdE9iamVjdHMsIHF1ZXJpZXM6IHNlbGVjdFF1ZXJpZXN9ID0gZ2V0UG9zdE9wdGlvbmFsKGZpZWxkcyk7XG4gIGNvbnN0IGFxbFFyeTogc3RyaW5nID0gYEZPUiBwIElOIHBvc3RzXG4gICAgRklMVEVSIHAudHlwZSA9PSBcIiR7dHlwZX1cIiAmJiBwLnByaXZhY3kgPT0gXCJwdWJsaWNcIiAmJiBwLnBhcmVudCA9PSBudWxsXG4gICAgJHtzZWxlY3RRdWVyaWVzLmpvaW4oJ1xcbicpfVxuICAgICR7bGltaXQuYXFsfVxuICAgIFNPUlQgcC5hZGRlZFxuICAgIFJFVFVSTiBESVNUSU5DVCBNRVJHRShwLCB7JHtzZWxlY3RPYmplY3RzLmpvaW4oJywgJyl9fSlgO1xuXG4gIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IuYWxsKCkpXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IHtcbiAgICAgIHRocm93IGVycm9yO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldFBvc3RMaXN0QnlUYWdzID0gKFxuICBjb250ZXh0OiBBcGlDb250ZXh0LFxuICB0YWdOYW1lczogc3RyaW5nW10sXG4gIG9wdGlvbnM/OiBQb3N0T3B0aW9uc1xuKTogUHJvbWlzZTxQb3N0VHlwZVtdPiA9PiB7XG4gIC8vIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2dldExpc3RCeVRhZ3MnO1xuICBjb25zdCB7ZGF0YWJhc2UsIGZpZWxkc30gPSBjb250ZXh0O1xuICBjb25zdCB7bGF0aXR1ZGUsIGxvbmdpdHVkZSwgbGltaXQsIHR5cGV9ID0gcGFyc2VQb3N0T3B0aW9ucyhvcHRpb25zKTtcbiAgY29uc3Qge29iamVjdHM6IHNlbGVjdE9iamVjdHMsIHF1ZXJpZXM6IHNlbGVjdFF1ZXJpZXN9ID0gZ2V0UG9zdE9wdGlvbmFsKGZpZWxkcyk7XG4gIGNvbnN0IHNvcnRCeTogc3RyaW5nW10gPSBbXTtcblxuICBpZihsYXRpdHVkZSAhPT0gdW5kZWZpbmVkICYmIGxvbmdpdHVkZSAhPT0gdW5kZWZpbmVkKSB7XG4gICAgY29uc3QgZm9ybWF0TGF0aXR1ZGU6IG51bWJlciA9IHBhcnNlTnVtKGxhdGl0dWRlKTtcbiAgICBjb25zdCBmb3JtYXRMb25naXR1ZGU6IG51bWJlciA9IHBhcnNlTnVtKGxvbmdpdHVkZSk7XG4gICAgc2VsZWN0UXVlcmllcy5wdXNoKGBMRVQgZGlzdGFuY2UgPSBESVNUQU5DRShcbiAgICAgICR7Zm9ybWF0TGF0aXR1ZGV9LFxuICAgICAgJHtmb3JtYXRMb25naXR1ZGV9LFxuICAgICAgTk9UX05VTEwocC5sYXRpdHVkZSwgMCksXG4gICAgICBOT1RfTlVMTChwLmxvbmdpdHVkZSwgMCkpXG4gICAgYCk7XG4gICAgc2VsZWN0T2JqZWN0cy5wdXNoKCdkaXN0YW5jZTpkaXN0YW5jZScpO1xuICAgIHNvcnRCeS5wdXNoKCdkaXN0YW5jZScpO1xuICB9XG5cbiAgc29ydEJ5LnB1c2goJ3AuYWRkZWQnKTtcblxuICByZXR1cm4gUHJvbWlzZS5hbGwoXG4gICAgdGFnTmFtZXMubWFwKCh0YWdOYW1lOiBzdHJpbmcpID0+IHtcbiAgICAgIGNvbnN0IGFxbFFyeTogc3RyaW5nID0gYEZPUiB0YXJnZXRUYWcgSU4gdGFnc1xuICAgICAgICBGSUxURVIgdGFyZ2V0VGFnLm5hbWUgPT0gXCIke3RhZ05hbWV9XCJcbiAgICAgICAgRk9SIHAsIGUgSU4gT1VUQk9VTkQgdGFyZ2V0VGFnLl9pZCBpc1RhZ2dlZFxuICAgICAgICAke3NlbGVjdFF1ZXJpZXMuam9pbignXFxuJyl9XG4gICAgICAgIEZJTFRFUiBwLnR5cGUgPSBcIiR7dHlwZX1cIiAmJiBwLnByaXZhY3kgPT0gXCJwdWJsaWNcIiAmJiBlLnR5cGUgPT0gJ3Bvc3RzJ1xuICAgICAgICAke2xpbWl0LmFxbH1cbiAgICAgICAgU09SVCAke3NvcnRCeS5qb2luKCcsICcpfVxuICAgICAgICBSRVRVUk4gRElTVElOQ1QgTUVSR0UocCwgeyR7c2VsZWN0T2JqZWN0cy5qb2luKCcsICcpfX0pYDtcblxuICAgICAgY29uc29sZS5sb2coYXFsUXJ5KTtcbiAgICAgIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgICAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLmFsbCgpKVxuICAgICAgICAuY2F0Y2goKCkgPT4gW10pO1xuICAgIH0pKVxuICAgIC50aGVuKChyZXN1bHRzKSA9PiB1bmlxQnkoZmxhdHRlbihyZXN1bHRzKSwgJ19rZXknKSlcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgdGhyb3cgZXJyb3I7XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgZ2V0UG9zdExpc3RCeVVzZXIgPSAoY29udGV4dDogQXBpQ29udGV4dCwgdXNlcklkOiBzdHJpbmcsIG9wdGlvbnM/OiBQb3N0T3B0aW9ucyk6IFByb21pc2U8UG9zdFR5cGVbXT4gPT4ge1xuICAvLyBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRMaXN0QnlVc2VyJztcbiAgY29uc3Qge2RhdGFiYXNlLCBmaWVsZHN9ID0gY29udGV4dDtcbiAgY29uc3Qge2xpbWl0LCB0eXBlfSA9IHBhcnNlUG9zdE9wdGlvbnMob3B0aW9ucyk7XG4gIGNvbnN0IGZvcm1hdFVzZXJJZDogc3RyaW5nID0gcGFyc2VJZCh1c2VySWQpO1xuICBjb25zdCB7b2JqZWN0czogc2VsZWN0T2JqZWN0cywgcXVlcmllczogc2VsZWN0UXVlcmllc30gPSBnZXRQb3N0T3B0aW9uYWwoZmllbGRzKTtcbiAgY29uc3QgYXFsUXJ5OiBzdHJpbmcgPSBgRk9SIHAgSU4gcG9zdHNcbiAgICBGSUxURVIgcC51c2VySWQgPT0gXCIke2Zvcm1hdFVzZXJJZH1cIiAmJiBwLnR5cGUgPT0gXCIke3R5cGV9XCIgJiYgcC5wcml2YWN5ID09IFwicHVibGljXCIgJiYgcC5wYXJlbnQgPT0gbnVsbFxuICAgICR7c2VsZWN0UXVlcmllcy5qb2luKCdcXG4nKX1cbiAgICAke2xpbWl0LmFxbH1cbiAgICBTT1JUIHAuYWRkZWRcbiAgICBSRVRVUk4gRElTVElOQ1QgTUVSR0UocCwgeyR7c2VsZWN0T2JqZWN0cy5qb2luKCcsICcpfX0pYDtcblxuICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLmFsbCgpKVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICB0aHJvdyBlcnJvcjtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRQb3N0TGlzdEJ5QXJlYSA9IChcbiAgY29udGV4dDogQXBpQ29udGV4dCxcbiAgbGF0aXR1ZGU6IG51bWJlcixcbiAgbG9uZ2l0dWRlOiBudW1iZXIsXG4gIG9wdGlvbnM/OiBQb3N0T3B0aW9uc1xuKTogUHJvbWlzZTxQb3N0VHlwZVtdPiA9PiB7XG4gIC8vIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2dldExpc3RCeVVzZXInO1xuICBjb25zdCB7ZGF0YWJhc2UsIGZpZWxkc30gPSBjb250ZXh0O1xuICBjb25zdCB7bGltaXQsIHR5cGV9ID0gcGFyc2VQb3N0T3B0aW9ucyhvcHRpb25zKTtcbiAgY29uc3QgZm9ybWF0TGF0aXR1ZGU6IG51bWJlciA9IHBhcnNlTnVtKGxhdGl0dWRlKTtcbiAgY29uc3QgZm9ybWF0TG9uZ2l0dWRlOiBudW1iZXIgPSBwYXJzZU51bShsb25naXR1ZGUpO1xuICBjb25zdCB7b2JqZWN0czogc2VsZWN0T2JqZWN0cywgcXVlcmllczogc2VsZWN0UXVlcmllc30gPSBnZXRQb3N0T3B0aW9uYWwoZmllbGRzKTtcbiAgc2VsZWN0UXVlcmllcy5wdXNoKGBMRVQgZGlzdGFuY2UgPSBESVNUQU5DRShcbiAgICAke2Zvcm1hdExhdGl0dWRlfSxcbiAgICAke2Zvcm1hdExvbmdpdHVkZX0sXG4gICAgTk9UX05VTEwocC5sYXRpdHVkZSwgMCksXG4gICAgTk9UX05VTEwocC5sb25naXR1ZGUsIDApKVxuICBgKTtcbiAgc2VsZWN0T2JqZWN0cy5wdXNoKCdkaXN0YW5jZTpkaXN0YW5jZScpO1xuXG4gIGNvbnN0IGFxbFFyeTogc3RyaW5nID0gYEZPUiBwIElOIHBvc3RzXG4gICAgJHtzZWxlY3RRdWVyaWVzLmpvaW4oJ1xcbicpfVxuICAgIEZJTFRFUiBwLnR5cGUgPT0gXCIke3R5cGV9XCIgJiYgcC5wcml2YWN5ID09IFwicHVibGljXCIgJiYgcC5wYXJlbnRJZCA9PSBudWxsXG4gICAgJHtsaW1pdC5hcWx9XG4gICAgU09SVCBkaXN0YW5jZSwgcC5hZGRlZFxuICAgIFJFVFVSTiBESVNUSU5DVCBNRVJHRShwLCB7JHtzZWxlY3RPYmplY3RzLmpvaW4oJywgJyl9fSlgO1xuXG4gIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IuYWxsKCkpXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IHtcbiAgICAgIHRocm93IGVycm9yO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldFBvc3QgPSAoY29udGV4dDogQXBpQ29udGV4dCwgaXRlbUlkOiBzdHJpbmcpOiBQcm9taXNlPFBvc3RUeXBlPiA9PiB7XG4gIC8vIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2dldEl0ZW0nO1xuICBjb25zdCB7ZGF0YWJhc2UsIGZpZWxkcywgdXNlcklkOiBzZXNzaW9uSWR9ID0gY29udGV4dDtcbiAgY29uc3QgZm9ybWF0SXRlbUlkOiBzdHJpbmcgPSBwYXJzZUlkKGl0ZW1JZCk7XG4gIGNvbnN0IGRiID0gdXNlRGIoZGF0YWJhc2UpO1xuICBjb25zdCB7b2JqZWN0czogc2VsZWN0T2JqZWN0cywgcXVlcmllczogc2VsZWN0UXVlcmllc30gPSBnZXRQb3N0T3B0aW9uYWwoZmllbGRzKTtcbiAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBGT1IgcCBJTiBwb3N0c1xuICAgIEZJTFRFUiBwLl9rZXkgPT0gJHtmb3JtYXRJdGVtSWR9XG4gICAgTElNSVQgMVxuICAgIFJFVFVSTiBwYDtcblxuICByZXR1cm4gZGIucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgIC50aGVuKChwb3N0OiBQb3N0VHlwZSA9IHt9KSA9PiB7XG4gICAgICBjb25zdCB7XG4gICAgICAgIF9rZXksXG4gICAgICAgIGdyb3VwSWQsXG4gICAgICAgIHByaXZhY3kgPSAnZGVmYXVsdCdcbiAgICAgIH06IFBvc3RUeXBlID0gcG9zdDtcblxuICAgICAgLy8gUXVlcnkgYmFzZWQgb24gcHJpdmFjeSBsZXZlbFxuICAgICAgbGV0IHByaXZhY3lBcWxRcnk6IHN0cmluZztcblxuICAgICAgaWYoZ3JvdXBJZCAmJiBwcml2YWN5ID09PSAnZ3JvdXAnKSB7XG4gICAgICAgIHByaXZhY3lBcWxRcnkgPSBgRk9SIHAgSU4gcG9zdHNcbiAgICAgICAgICBGSUxURVIgcC5fa2V5ID09IFwiJHtfa2V5fVwiXG4gICAgICAgICAgJHtzZWxlY3RRdWVyaWVzLmpvaW4oJ1xcbicpfVxuICAgICAgICAgIEZPUiBncm91cCBJTiBncm91cHNcbiAgICAgICAgICBGSUxURVIgZ3JvdXAuX2tleSA9PSBwLmdyb3VwSWRcbiAgICAgICAgICBGT1IgdSwgZSBJTiBPVVRCT1VORCBncm91cC5faWQgaXNHcm91cGVkXG4gICAgICAgICAgRklMVEVSIHUuX2tleSA9PSAke3Nlc3Npb25JZH1cbiAgICAgICAgICBMSU1JVCAxXG4gICAgICAgICAgUkVUVVJOIE1FUkdFKHAsIHske3NlbGVjdE9iamVjdHMuam9pbignLCAnKX19KWA7XG4gICAgICB9IGVsc2UgaWYocHJpdmFjeSA9PT0gJ3B1YmxpYycpIHtcbiAgICAgICAgcHJpdmFjeUFxbFFyeSA9IGBGT1IgcCBJTiBwb3N0c1xuICAgICAgICAgIEZJTFRFUiBwLl9rZXkgPT0gXCIke19rZXl9XCJcbiAgICAgICAgICAke3NlbGVjdFF1ZXJpZXMuam9pbignXFxuJyl9XG4gICAgICAgICAgTElNSVQgMVxuICAgICAgICAgIFJFVFVSTiBNRVJHRShwLCB7JHtzZWxlY3RPYmplY3RzLmpvaW4oJywgJyl9fSlgO1xuICAgICAgfVxuXG4gICAgICBpZihwcml2YWN5QXFsUXJ5KSB7XG4gICAgICAgIHJldHVybiBkYi5xdWVyeShwcml2YWN5QXFsUXJ5KVxuICAgICAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgICAgICAgIC50aGVuKChmaWx0ZXJlZFBvc3Q6IFBvc3RUeXBlID0ge30pID0+IGZpbHRlcmVkUG9zdClcbiAgICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgICAgICAgdGhyb3cgZXJyb3I7XG4gICAgICAgICAgfSk7XG4gICAgICB9XG5cbiAgICAgIHJldHVybiB7fTtcbiAgICB9KVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICB0aHJvdyBlcnJvcjtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRQb3N0Q29tbWVudHMgPSAoY29udGV4dDogQXBpQ29udGV4dCwgaXRlbUlkOiBzdHJpbmcsIG9wdGlvbnM/OiBQb3N0T3B0aW9ucyk6IFByb21pc2U8UG9zdFR5cGVbXT4gPT4ge1xuICAvLyBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRDb21tZW50cyc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgdXNlcklkOiBzZXNzaW9uSWR9ID0gY29udGV4dDtcbiAgY29uc3Qge2xpbWl0LCB0eXBlfSA9IHBhcnNlUG9zdE9wdGlvbnMob3B0aW9ucyk7XG4gIGNvbnN0IGZvcm1hdEl0ZW1JZDogc3RyaW5nID0gcGFyc2VJZChpdGVtSWQpO1xuXG4gIC8vIEdldCB0aGUgcGFyZW50IHBvc3QgdG8gZ2V0IHJlc3RyaWN0aW9uc1xuICBjb25zdCBkYiA9IHVzZURiKGRhdGFiYXNlKTtcbiAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBGT1IgcCBJTiBwb3N0c1xuICAgIEZJTFRFUiBwLnR5cGUgPT0gJHt0eXBlfSAmJiBwLl9rZXkgPT0gJHtmb3JtYXRJdGVtSWR9XG4gICAgTElNSVQgMVxuICAgIFJFVFVSTiBwYDtcblxuICByZXR1cm4gZGIucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgIC50aGVuKChwb3N0OiBQb3N0VHlwZSA9IHt9KSA9PiB7XG4gICAgICBjb25zdCB7XG4gICAgICAgIF9rZXksXG4gICAgICAgIGdyb3VwSWQsXG4gICAgICAgIHByaXZhY3kgPSAncHVibGljJ1xuICAgICAgfTogUG9zdFR5cGUgPSBwb3N0O1xuXG4gICAgICAvLyBRdWVyeSBiYXNlZCBvbiBwcml2YWN5IGxldmVsXG4gICAgICBsZXQgcHJpdmFjeUFxbFFyeTogc3RyaW5nO1xuXG4gICAgICBpZihncm91cElkICYmIHByaXZhY3kgPT09ICdncm91cCcpIHtcbiAgICAgICAgcHJpdmFjeUFxbFFyeSA9IGBGT1IgcCBJTiBwb3N0c1xuICAgICAgICAgIEZPUiB1c2VyIElOIHVzZXJzXG4gICAgICAgICAgRklMVEVSIHAucGFyZW50ID09IFwiJHtfa2V5fVwiICYmIHVzZXIuX2tleSA9PSBwLnVzZXJJZFxuICAgICAgICAgIExFVCByZWFjdGlvbnMgPSAoXG4gICAgICAgICAgICBGT1IgcG9zdCwgciBJTiBJTkJPVU5EIHAuX2lkIHJlYWN0aW9uc1xuICAgICAgICAgICAgQ09MTEVDVCByZWFjdGlvbk5hbWUgPSByLnZhbHVlIElOVE8gcmVhY3Rpb25JdGVtc1xuICAgICAgICAgICAgUkVUVVJOIHtuYW1lOiByZWFjdGlvbk5hbWUsIGNvdW50OiBMRU5HVEgocmVhY3Rpb25JdGVtc1sqXS5yLnZhbHVlKX1cbiAgICAgICAgICApXG4gICAgICAgICAgRk9SIGdyb3VwIElOIGdyb3Vwc1xuICAgICAgICAgIEZJTFRFUiBncm91cC5fa2V5ID09IHAuZ3JvdXBJZFxuICAgICAgICAgIEZPUiB1LCBlIElOIE9VVEJPVU5EIGdyb3VwLl9pZCBpc0dyb3VwZWRcbiAgICAgICAgICBGSUxURVIgdS5fa2V5ID09IFwiJHtzZXNzaW9uSWR9XCJcbiAgICAgICAgICBTT1JUIHAuYWRkZWRcbiAgICAgICAgICAke2xpbWl0LmFxbH1cbiAgICAgICAgICBSRVRVUk4gTUVSR0UocCwge3VzZXI6IHVzZXIsIHJlYWN0aW9uczogcmVhY3Rpb25zfSlgO1xuICAgICAgfSBlbHNlIGlmKHByaXZhY3kgPT09ICdwdWJsaWMnKSB7XG4gICAgICAgIHByaXZhY3lBcWxRcnkgPSBgRk9SIHAgSU4gcG9zdHNcbiAgICAgICAgICBGT1IgdXNlciBJTiB1c2Vyc1xuICAgICAgICAgIEZJTFRFUiBwLnBhcmVudCA9PSBcIiR7X2tleX1cIiAmJiB1c2VyLl9rZXkgPT0gcC51c2VySWRcbiAgICAgICAgICBMRVQgcmVhY3Rpb25zID0gKFxuICAgICAgICAgICAgRk9SIHBvc3QsIHIgSU4gSU5CT1VORCBwLl9pZCByZWFjdGlvbnNcbiAgICAgICAgICAgIENPTExFQ1QgcmVhY3Rpb25OYW1lID0gci52YWx1ZSBJTlRPIHJlYWN0aW9uSXRlbXNcbiAgICAgICAgICAgIFJFVFVSTiB7bmFtZTogcmVhY3Rpb25OYW1lLCBjb3VudDogTEVOR1RIKHJlYWN0aW9uSXRlbXNbKl0uci52YWx1ZSl9XG4gICAgICAgICAgKVxuICAgICAgICAgIFNPUlQgcC5hZGRlZFxuICAgICAgICAgICR7bGltaXQuYXFsfVxuICAgICAgICAgIFJFVFVSTiBNRVJHRShwLCB7dXNlcjogdXNlciwgcmVhY3Rpb25zOiByZWFjdGlvbnN9KWA7XG4gICAgICB9XG5cbiAgICAgIGlmKHByaXZhY3lBcWxRcnkpIHtcbiAgICAgICAgcmV0dXJuIGRiLnF1ZXJ5KHByaXZhY3lBcWxRcnkpXG4gICAgICAgICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5hbGwoKSlcbiAgICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgICAgICAgdGhyb3cgZXJyb3I7XG4gICAgICAgICAgfSk7XG4gICAgICB9XG5cbiAgICAgIHJldHVybiBbXTtcbiAgICB9KVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICB0aHJvdyBlcnJvcjtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBhZGRQb3N0ID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGl0ZW06IFBvc3RUeXBlKTogUHJvbWlzZTxQb3N0VHlwZT4gPT4ge1xuICAvLyBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdhZGQnO1xuICBjb25zdCB7ZGF0YWJhc2UsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG5cbiAgY29uc3Qge1xuICAgIGNvbnRlbnQgPSAnJyxcbiAgICBncm91cElkID0gJycsXG4gICAgbG9jYXRpb24sXG4gICAgbGF0aXR1ZGUsXG4gICAgbG9uZ2l0dWRlLFxuICAgIG5hbWUgPSAnJyxcbiAgICBwYXJlbnRJZCA9IG51bGwsXG4gICAgcHJpdmFjeSA9ICdwdWJsaWMnLFxuICAgIHR5cGUgPSAnZGVmYXVsdCdcbiAgfTogUG9zdFR5cGUgPSBpdGVtO1xuXG4gIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcblxuICBjb25zdCBpbnNlcnQ6IFBvc3RUeXBlID0ge1xuICAgIF9rZXk6IGNyZWF0ZUhhc2goYHBvc3QtJHtzZXNzaW9uSWR9YCksXG4gICAgYWRkZWQ6IG5vdyxcbiAgICBjb250ZW50OiBwYXJzZVN0cmluZyhjb250ZW50LCBNQVhfQ09OVEVOVF9MRU5HVEgpLFxuICAgIGdyb3VwSWQ6IGdyb3VwSWQgPyBwYXJzZUlkKGdyb3VwSWQpIDogdW5kZWZpbmVkLFxuICAgIGxhdGl0dWRlOiBsYXRpdHVkZSAhPT0gdW5kZWZpbmVkID8gcGFyc2VOdW0obGF0aXR1ZGUpIDogdW5kZWZpbmVkLFxuICAgIGxvY2F0aW9uOiBsb2NhdGlvbiA/IHBhcnNlU3RyaW5nKGxvY2F0aW9uLCAxNjApIDogdW5kZWZpbmVkLFxuICAgIGxvbmdpdHVkZTogbG9uZ2l0dWRlICE9PSB1bmRlZmluZWQgPyBwYXJzZU51bShsb25naXR1ZGUpIDogdW5kZWZpbmVkLFxuICAgIG1vZGlmaWVkOiBub3csXG4gICAgbmFtZTogcGFyc2VTdHJpbmcobmFtZSwgMTYwKSxcbiAgICBwYXJlbnRJZDogcGFyZW50SWQgPyBwYXJzZUlkKHBhcmVudElkKSA6IHVuZGVmaW5lZCxcbiAgICBwcml2YWN5OiBwcml2YWN5ID8gcGFyc2VWYXJDaGFyKHByaXZhY3ksIDE2KSA6IHVuZGVmaW5lZCxcbiAgICB0eXBlOiBwYXJzZUNoYXIodHlwZSwgMzIpLFxuICAgIHVzZXJJZDogc2Vzc2lvbklkXG4gIH07XG5cbiAgY29uc3QgZGI6IERhdGFiYXNlID0gdXNlRGIoZGF0YWJhc2UpO1xuICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYElOU0VSVCAke2luc2VydH0gSU4gcG9zdHMgUkVUVVJOIE5FV2A7XG5cbiAgcmV0dXJuIGRiLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAudGhlbigocG9zdDogUG9zdFR5cGUgPSB7fSkgPT4ge1xuICAgICAgY29uc3Qge19rZXk6IHBvc3RLZXl9ID0gcG9zdDtcblxuICAgICAgLy8gVXBkYXRlIGxpbmtlZCB0YWdzIHdpdGhpbiBwb3N0c1xuICAgICAgcmV0dXJuIGV4dHJhY3RUYWdzKGRiLCAncG9zdHMnLCBwb3N0S2V5LCBpbnNlcnQuY29udGVudClcbiAgICAgICAgLnRoZW4oKHRhZ0xpc3Q6IFRhZ1R5cGVbXSkgPT4ge1xuICAgICAgICAgIHBvc3QudGFncyA9IHRhZ0xpc3Q7XG4gICAgICAgICAgcmV0dXJuIHBvc3Q7XG4gICAgICAgIH0pO1xuICAgIH0pXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IHtcbiAgICAgIHRocm93IGVycm9yO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IHVwZGF0ZVBvc3QgPSAoY29udGV4dDogQXBpQ29udGV4dCwgaXRlbTogUG9zdFR5cGUpOiBQcm9taXNlPFBvc3RUeXBlPiA9PiB7XG4gIC8vIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ3VwZGF0ZSc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgdXNlcklkOiBzZXNzaW9uSWR9ID0gY29udGV4dDtcbiAgY29uc3Qgbm93OiBudW1iZXIgPSBEYXRlLm5vdygpO1xuICBjb25zdCB7XG4gICAgY29udGVudCxcbiAgICBncm91cElkLFxuICAgIG5hbWUsXG4gICAgcGFyZW50SWQsXG4gICAgcG9zdElkLFxuICAgIHByaXZhY3ksXG4gICAgdHlwZVxuICB9OiBQb3N0VHlwZSA9IGl0ZW07XG5cbiAgY29uc3QgdXBkYXRlOiBQb3N0VHlwZSA9IHtcbiAgICBjb250ZW50OiBjb250ZW50ID8gcGFyc2VTdHJpbmcoY29udGVudCwgTUFYX0NPTlRFTlRfTEVOR1RIKSA6IHVuZGVmaW5lZCxcbiAgICBtb2RpZmllZDogbm93LFxuICAgIG5hbWU6IG5hbWUgPyBwYXJzZVN0cmluZyhuYW1lLCAxNjApIDogdW5kZWZpbmVkLFxuICAgIHBhcmVudElkOiBwYXJlbnRJZCA/IHBhcnNlU3RyaW5nKHBhcmVudElkLCAxNjApIDogdW5kZWZpbmVkLFxuICAgIHByaXZhY3k6IHByaXZhY3kgPyBwYXJzZVZhckNoYXIocHJpdmFjeSwgMTYpIDogdW5kZWZpbmVkLFxuICAgIHR5cGU6IHR5cGUgIT09IHVuZGVmaW5lZCA/IHBhcnNlQ2hhcih0eXBlLCAxNikgOiB1bmRlZmluZWRcbiAgfTtcblxuICBsZXQgZm9ybWF0SWQ6IHN0cmluZyA9IHBhcnNlSWQocG9zdElkKTtcbiAgZm9ybWF0SWQgPSBmb3JtYXRJZCA9PT0gJycgPyBjcmVhdGVIYXNoKGBwb3N0LSR7c2Vzc2lvbklkfWApIDogZm9ybWF0SWQ7XG4gIGNvbnN0IGZvcm1hdEdyb3VwSWQ6IHN0cmluZyA9IHBhcnNlSWQoZ3JvdXBJZCk7XG4gIGNvbnN0IGluc2VydDogYW55ID0ge1xuICAgIC4uLnVwZGF0ZSxcbiAgICBfa2V5OiBmb3JtYXRJZCxcbiAgICBhZGRlZDogbm93LFxuICAgIGdyb3VwSWQ6IGZvcm1hdEdyb3VwSWQsXG4gICAgcHJpdmFjeSxcbiAgICB1c2VySWQ6IHNlc3Npb25JZFxuICB9O1xuICBjb25zdCBkYjogRGF0YWJhc2UgPSB1c2VEYihkYXRhYmFzZSk7XG4gIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgVVBTRVJUIHtfa2V5OiAke2Zvcm1hdElkfSwgdXNlcklkOiAke3Nlc3Npb25JZH19XG4gICAgSU5TRVJUICR7aW5zZXJ0fVxuICAgIFVQREFURSAke3VwZGF0ZX1cbiAgICBJTiBwb3N0cyBSRVRVUk4gTkVXYDtcblxuICByZXR1cm4gZGIucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgIC50aGVuKCh1cGRhdGVkUG9zdDogUG9zdFR5cGUgPSB7fSkgPT4ge1xuICAgICAgY29uc3Qge19rZXk6IHVwZGF0ZWRQb3N0S2V5fSA9IHVwZGF0ZWRQb3N0O1xuXG4gICAgICAvLyBVcGRhdGUgbGlua2VkIHRhZ3NcbiAgICAgIHJldHVybiBleHRyYWN0VGFncyhkYiwgJ3Bvc3RzJywgdXBkYXRlZFBvc3RLZXksIHVwZGF0ZS5jb250ZW50IHx8ICcnKVxuICAgICAgICAudGhlbigodGFnTGlzdCA9IFtdKSA9PiB7XG4gICAgICAgICAgdXBkYXRlZFBvc3QudGFncyA9IHRhZ0xpc3Q7XG5cbiAgICAgICAgICAvLyBVcGRhdGUgbGlua2VkIGZpbGVzXG4gICAgICAgICAgY29uc3QgZmlsZXM6IEZpbGVUeXBlW10gPSB1cGRhdGVkUG9zdC5maWxlcyB8fCBbXTtcblxuICAgICAgICAgIGlmKGZpbGVzLmxlbmd0aCkge1xuICAgICAgICAgICAgcmV0dXJuIHVwZGF0ZUZpbGVzKGRiLCBmb3JtYXRJZCwgZmlsZXMpXG4gICAgICAgICAgICAgIC50aGVuKChmaWxlTGlzdCA9IFtdKSA9PiB7XG4gICAgICAgICAgICAgICAgdXBkYXRlZFBvc3QuZmlsZXMgPSBmaWxlTGlzdDtcbiAgICAgICAgICAgICAgICByZXR1cm4gdXBkYXRlZFBvc3Q7XG4gICAgICAgICAgICAgIH0pO1xuICAgICAgICAgIH1cblxuICAgICAgICAgIHVwZGF0ZWRQb3N0LmZpbGVzID0gW107XG4gICAgICAgICAgcmV0dXJuIHVwZGF0ZWRQb3N0O1xuICAgICAgICB9KTtcbiAgICB9KVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICB0aHJvdyBlcnJvcjtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBkZWxldGVQb3N0ID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGl0ZW1JZDogc3RyaW5nKTogUHJvbWlzZTxQb3N0VHlwZT4gPT4ge1xuICAvLyBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdkZWxldGUnO1xuICBjb25zdCB7ZGF0YWJhc2UsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG4gIGNvbnN0IGZvcm1hdEl0ZW1JZDogc3RyaW5nID0gcGFyc2VJZChpdGVtSWQpO1xuICBjb25zdCBkYjogRGF0YWJhc2UgPSB1c2VEYihkYXRhYmFzZSk7XG4gIGNvbnN0IGFxbFFyeSA9IGFxbGBGT1IgcCBJTiBwb3N0c1xuICAgICAgRklMVEVSIHAuX2tleSA9PSAke2Zvcm1hdEl0ZW1JZH0gJiYgcC51c2VySWQgPT0gJHtzZXNzaW9uSWR9XG4gICAgICBMSU1JVCAxXG4gICAgICBSRU1PVkUgcCBJTiBwb3N0c1xuICAgICAgUkVUVVJOIE9MRGA7XG5cbiAgcmV0dXJuIGRiLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAudGhlbigocG9zdDogUG9zdFR5cGUgPSB7fSkgPT4ge1xuICAgICAgaWYocG9zdCkge1xuICAgICAgICAvLyBSZW1vdmUgdGFnIGxpbmtzXG4gICAgICAgIGNvbnN0IGVkZ2VBcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYEZPUiB0IElOIGlzVGFnZ2VkXG4gICAgICAgICAgICBGSUxURVIgdC5fdG8gPT0gJHtmb3JtYXRJdGVtSWR9XG4gICAgICAgICAgICBSRU1PVkUgdCBJTiBpc1RhZ2dlZGA7XG5cbiAgICAgICAgcmV0dXJuIGRiLnF1ZXJ5KGVkZ2VBcWxRcnkpXG4gICAgICAgICAgLnRoZW4oKCkgPT4ge1xuICAgICAgICAgICAgLy8gUmVtb3ZlIGF0dGFjaGVkIGZpbGVzXG4gICAgICAgICAgICBjb25zdCBmaWxlQXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBGT1IgZiBJTiBoYXNGaWxlXG4gICAgICAgICAgICAgICAgRklMVEVSIGYuX3RvID09ICR7Zm9ybWF0SXRlbUlkfVxuICAgICAgICAgICAgICAgIFJFTU9WRSBmIElOIGhhc0ZpbGVgO1xuXG4gICAgICAgICAgICByZXR1cm4gZGIucXVlcnkoZmlsZUFxbFFyeSlcbiAgICAgICAgICAgICAgLnRoZW4oKCkgPT4gcG9zdClcbiAgICAgICAgICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IHtcbiAgICAgICAgICAgICAgICB0aHJvdyBlcnJvcjtcbiAgICAgICAgICAgICAgfSk7XG4gICAgICAgICAgfSlcbiAgICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgICAgICAgdGhyb3cgZXJyb3I7XG4gICAgICAgICAgfSk7XG4gICAgICB9XG4gICAgICByZXR1cm4ge307XG4gICAgfSlcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgdGhyb3cgZXJyb3I7XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgY2xlYW5Qb3N0cyA9IChkYXRhYmFzZTogc3RyaW5nKTogUHJvbWlzZTxudW1iZXI+ID0+IHtcbiAgLy8gUmVtb3ZlIGFsbCBtZXNzYWdlcyB0aGF0IGFyZSBvdmVyIDYwIGRheXMgYW5kIG5vdCBzYXZlZFxuICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYEZPUiBwIElOIHBvc3RzXG4gICAgICBGSUxURVIgcC5hZGRlZCA8IERBVEVfVElNRVNUQU1QKERBVEVfU1VCVFJBQ1QoREFURV9OT1coKSwgNjAsICdkYXknKSkgJiYgcC50eXBlID09IFwiZGVmYXVsdFwiXG4gICAgICBSRU1PVkUgcCBJTiBwb3N0c1xuICAgICAgUkVUVVJOIE9MRGA7XG5cbiAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5hbGwoKSlcbiAgICAudGhlbigocmVzdWx0czogUG9zdFR5cGVbXSA9IFtdKSA9PiByZXN1bHRzLmxlbmd0aClcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgdGhyb3cgZXJyb3I7XG4gICAgfSk7XG59XG5cbmV4cG9ydCBjb25zdCBjcmVhdGVQb3N0RWRnZSA9IChkYjogRGF0YWJhc2UsIGZpbGU6IEZpbGVUeXBlLCBwb3N0SWQ6IHN0cmluZyk6IFByb21pc2U8RmlsZVR5cGU+ID0+IHtcbiAgY29uc3QgZWRnZUNvbGxlY3Rpb24gPSBkYi5lZGdlQ29sbGVjdGlvbignaXNQb3N0ZWQnKTtcbiAgY29uc3QgZmlsZUlkOiBzdHJpbmcgPSBwYXJzZUlkKGZpbGUuaWQpO1xuICBjb25zdCBlZGdlSWQ6IHN0cmluZyA9IGNyZWF0ZUhhc2goYGZpbGUtJHtwb3N0SWR9LSR7ZmlsZUlkfWApO1xuICBjb25zdCBmb3JtYXRQb3N0SWQ6IHN0cmluZyA9IHBhcnNlSWQocG9zdElkKTtcbiAgY29uc3QgZmlsZVR5cGU6IHN0cmluZyA9IHBhcnNlQ2hhcihmaWxlLmZpbGVUeXBlLCAxNik7XG5cbiAgY29uc3QgZWRnZTogYW55ID0ge1xuICAgIF9rZXk6IGVkZ2VJZCxcbiAgICBhZGRlZDogRGF0ZS5ub3coKSxcbiAgICB0eXBlOiBmaWxlVHlwZVxuICB9O1xuXG4gIHJldHVybiBlZGdlQ29sbGVjdGlvbi5zYXZlKGVkZ2UsIGBwb3N0cy8ke2Zvcm1hdFBvc3RJZH1gLCBgZmlsZXMvJHtmaWxlSWR9YClcbiAgICAudGhlbigoKSA9PiBmaWxlKVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICB0aHJvdyBlcnJvcjtcbiAgICB9KTtcbn07XG4iXX0=