@nlabs/reaktor 0.4.0 → 0.4.1

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 (151) hide show
  1. package/lib/actions/conversations.d.ts +14 -0
  2. package/lib/actions/conversations.js +333 -0
  3. package/lib/actions/dynamodb.js +155 -0
  4. package/lib/actions/email.js +177 -0
  5. package/lib/actions/files.js +319 -0
  6. package/lib/{data → actions}/groups.d.ts +4 -3
  7. package/lib/actions/groups.js +282 -0
  8. package/lib/actions/images.d.ts +22 -0
  9. package/lib/actions/images.js +682 -0
  10. package/lib/actions/index.js +40 -0
  11. package/lib/{data → actions}/ios.d.ts +2 -1
  12. package/lib/actions/ios.js +179 -0
  13. package/lib/actions/locations.js +112 -0
  14. package/lib/actions/messages.d.ts +13 -0
  15. package/lib/actions/messages.js +216 -0
  16. package/lib/{data → actions}/notifications.d.ts +2 -2
  17. package/lib/actions/notifications.js +63 -0
  18. package/lib/{data → actions}/payments.d.ts +2 -2
  19. package/lib/actions/payments.js +491 -0
  20. package/lib/actions/posts.d.ts +19 -0
  21. package/lib/actions/posts.js +538 -0
  22. package/lib/actions/reactions.d.ts +30 -0
  23. package/lib/actions/reactions.js +340 -0
  24. package/lib/{data → actions}/s3.d.ts +1 -1
  25. package/lib/actions/s3.js +122 -0
  26. package/lib/{data → actions}/search.d.ts +2 -2
  27. package/lib/actions/search.js +99 -0
  28. package/lib/actions/sms.js +76 -0
  29. package/lib/actions/statistics.d.ts +2 -0
  30. package/lib/actions/statistics.js +63 -0
  31. package/lib/actions/subscription.js +209 -0
  32. package/lib/actions/tags.d.ts +26 -0
  33. package/lib/actions/tags.js +340 -0
  34. package/lib/actions/users.d.ts +44 -0
  35. package/lib/actions/users.js +571 -0
  36. package/lib/{data → actions}/websockets.d.ts +1 -1
  37. package/lib/actions/websockets.js +156 -0
  38. package/lib/config.d.ts +2 -3
  39. package/lib/config.js +116 -149
  40. package/lib/index.d.ts +1 -1
  41. package/lib/index.js +23 -45
  42. package/lib/templates/email/layout.d.ts +2 -0
  43. package/lib/templates/email/layout.js +292 -0
  44. package/lib/templates/email/passwordForgot.d.ts +2 -0
  45. package/lib/templates/email/passwordForgot.js +28 -0
  46. package/lib/templates/email/passwordRecovery.d.ts +2 -0
  47. package/lib/templates/email/passwordRecovery.js +25 -0
  48. package/lib/templates/email/verifyEmail.d.ts +2 -0
  49. package/lib/templates/email/verifyEmail.js +28 -0
  50. package/lib/templates/email/welcome.d.ts +2 -0
  51. package/lib/templates/email/welcome.js +28 -0
  52. package/lib/templates/sms/passwordForgot.d.ts +2 -0
  53. package/lib/templates/sms/passwordForgot.js +14 -0
  54. package/lib/templates/sms/passwordRecovery.d.ts +2 -0
  55. package/lib/templates/sms/passwordRecovery.js +14 -0
  56. package/lib/templates/sms/verifyEmail.d.ts +2 -0
  57. package/lib/templates/sms/verifyEmail.js +14 -0
  58. package/lib/templates/sms/verifyPhone.d.ts +2 -0
  59. package/lib/templates/sms/verifyPhone.js +14 -0
  60. package/lib/templates/sms/welcome.d.ts +2 -0
  61. package/lib/templates/sms/welcome.js +14 -0
  62. package/lib/types/apps.d.ts +2 -2
  63. package/lib/types/apps.js +4 -2
  64. package/lib/types/arangodb.js +4 -2
  65. package/lib/types/auth.d.ts +4 -8
  66. package/lib/types/auth.js +4 -2
  67. package/lib/types/conversations.d.ts +3 -3
  68. package/lib/types/conversations.js +4 -2
  69. package/lib/types/email.d.ts +2 -2
  70. package/lib/types/email.js +4 -2
  71. package/lib/types/files.js +4 -2
  72. package/lib/types/google.js +4 -2
  73. package/lib/types/groups.d.ts +2 -1
  74. package/lib/types/groups.js +4 -2
  75. package/lib/types/images.d.ts +8 -5
  76. package/lib/types/images.js +4 -2
  77. package/lib/types/index.d.ts +1 -1
  78. package/lib/types/index.js +37 -227
  79. package/lib/types/locations.js +4 -2
  80. package/lib/types/messages.d.ts +12 -2
  81. package/lib/types/messages.js +4 -2
  82. package/lib/types/notifications.d.ts +2 -2
  83. package/lib/types/notifications.js +4 -2
  84. package/lib/types/payments.js +4 -2
  85. package/lib/types/posts.d.ts +18 -1
  86. package/lib/types/posts.js +4 -2
  87. package/lib/types/statistics.d.ts +3 -0
  88. package/lib/types/statistics.js +4 -0
  89. package/lib/types/tags.d.ts +6 -0
  90. package/lib/types/tags.js +4 -2
  91. package/lib/types/users.d.ts +15 -11
  92. package/lib/types/users.js +4 -2
  93. package/lib/utils/analytics.d.ts +7 -0
  94. package/lib/utils/analytics.js +101 -77
  95. package/lib/utils/arangodb.d.ts +1 -1
  96. package/lib/utils/arangodb.js +93 -114
  97. package/lib/utils/auth.js +58 -55
  98. package/lib/utils/graphql.js +38 -19
  99. package/lib/utils/index.d.ts +1 -1
  100. package/lib/utils/index.js +26 -84
  101. package/lib/utils/objects.js +44 -53
  102. package/lib/utils/session.d.ts +18 -0
  103. package/lib/utils/session.js +42 -0
  104. package/package.json +32 -30
  105. package/lib/data/conversations.d.ts +0 -8
  106. package/lib/data/conversations.js +0 -311
  107. package/lib/data/dynamodb.js +0 -206
  108. package/lib/data/email.js +0 -222
  109. package/lib/data/files.js +0 -525
  110. package/lib/data/groups.js +0 -435
  111. package/lib/data/images.d.ts +0 -22
  112. package/lib/data/images.js +0 -1051
  113. package/lib/data/index.js +0 -266
  114. package/lib/data/ios.js +0 -355
  115. package/lib/data/locations.js +0 -172
  116. package/lib/data/messages.d.ts +0 -9
  117. package/lib/data/messages.js +0 -299
  118. package/lib/data/notifications.js +0 -59
  119. package/lib/data/payments.js +0 -771
  120. package/lib/data/posts.d.ts +0 -23
  121. package/lib/data/posts.js +0 -766
  122. package/lib/data/reactions.d.ts +0 -14
  123. package/lib/data/reactions.js +0 -529
  124. package/lib/data/s3.js +0 -155
  125. package/lib/data/search.js +0 -155
  126. package/lib/data/sms.js +0 -83
  127. package/lib/data/subscription.js +0 -337
  128. package/lib/data/tags.d.ts +0 -14
  129. package/lib/data/tags.js +0 -397
  130. package/lib/data/users.d.ts +0 -20
  131. package/lib/data/users.js +0 -470
  132. package/lib/data/websockets.js +0 -250
  133. package/lib/types/reactions.d.ts +0 -17
  134. package/lib/types/reactions.js +0 -2
  135. package/lib/utils/redis.d.ts +0 -1
  136. package/lib/utils/redis.js +0 -36
  137. package/templates/email/layout.html +0 -279
  138. package/templates/email/passwordForgot.html +0 -15
  139. package/templates/email/passwordRecovery.html +0 -12
  140. package/templates/email/verifyEmail.html +0 -15
  141. package/templates/sms/passwordForgot.txt +0 -1
  142. package/templates/sms/passwordRecovery.txt +0 -1
  143. package/templates/sms/verifyEmail.txt +0 -1
  144. package/templates/sms/verifyPhone.txt +0 -1
  145. /package/lib/{data → actions}/dynamodb.d.ts +0 -0
  146. /package/lib/{data → actions}/email.d.ts +0 -0
  147. /package/lib/{data → actions}/files.d.ts +0 -0
  148. /package/lib/{data → actions}/index.d.ts +0 -0
  149. /package/lib/{data → actions}/locations.d.ts +0 -0
  150. /package/lib/{data → actions}/sms.d.ts +0 -0
  151. /package/lib/{data → actions}/subscription.d.ts +0 -0
package/lib/data/posts.js DELETED
@@ -1,766 +0,0 @@
1
- "use strict";
2
-
3
- require("core-js/modules/es.symbol");
4
-
5
- require("core-js/modules/es.array.concat");
6
-
7
- require("core-js/modules/es.array.filter");
8
-
9
- require("core-js/modules/es.array.for-each");
10
-
11
- require("core-js/modules/es.array.iterator");
12
-
13
- require("core-js/modules/es.array.join");
14
-
15
- require("core-js/modules/es.array.map");
16
-
17
- require("core-js/modules/es.array.reduce");
18
-
19
- require("core-js/modules/es.array.slice");
20
-
21
- require("core-js/modules/es.date.now");
22
-
23
- require("core-js/modules/es.date.to-string");
24
-
25
- require("core-js/modules/es.function.name");
26
-
27
- require("core-js/modules/es.object.define-properties");
28
-
29
- require("core-js/modules/es.object.define-property");
30
-
31
- require("core-js/modules/es.object.freeze");
32
-
33
- require("core-js/modules/es.object.get-own-property-descriptor");
34
-
35
- require("core-js/modules/es.object.get-own-property-descriptors");
36
-
37
- require("core-js/modules/es.object.keys");
38
-
39
- require("core-js/modules/es.object.to-string");
40
-
41
- require("core-js/modules/es.promise");
42
-
43
- require("core-js/modules/es.string.iterator");
44
-
45
- require("core-js/modules/web.dom-collections.for-each");
46
-
47
- require("core-js/modules/web.dom-collections.iterator");
48
-
49
- Object.defineProperty(exports, "__esModule", {
50
- value: true
51
- });
52
- exports.createPostEdge = exports.cleanPosts = exports.deletePost = exports.updatePost = exports.addPost = exports.getPostComments = exports.getPostsByUser = exports.getPostsByTags = exports.getPostsByReaction = exports.getPostsByLatest = exports.getPostsByGroup = exports.getPostsByArea = exports.getPostList = exports.getPost = exports.getPostOptional = exports.parsePostOptions = void 0;
53
-
54
- var _utils = require("@nlabs/utils");
55
-
56
- var _arangojs = require("arangojs");
57
-
58
- var _flatten = _interopRequireDefault(require("lodash/flatten"));
59
-
60
- var _uniqBy = _interopRequireDefault(require("lodash/uniqBy"));
61
-
62
- var _utils2 = require("../utils");
63
-
64
- var _files = require("./files");
65
-
66
- var _tags = require("./tags");
67
-
68
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { "default": obj }; }
69
-
70
- function _templateObject8() {
71
- 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"]);
72
-
73
- _templateObject8 = function _templateObject8() {
74
- return data;
75
- };
76
-
77
- return data;
78
- }
79
-
80
- function _templateObject7() {
81
- var data = _taggedTemplateLiteral(["FOR f IN hasFile\n FILTER f._to == ", "\n REMOVE f IN hasFile"]);
82
-
83
- _templateObject7 = function _templateObject7() {
84
- return data;
85
- };
86
-
87
- return data;
88
- }
89
-
90
- function _templateObject6() {
91
- var data = _taggedTemplateLiteral(["FOR t IN isTagged\n FILTER t._to == ", "\n REMOVE t IN isTagged"]);
92
-
93
- _templateObject6 = function _templateObject6() {
94
- return data;
95
- };
96
-
97
- return data;
98
- }
99
-
100
- function _templateObject5() {
101
- var data = _taggedTemplateLiteral(["FOR p IN posts\n FILTER p._key == ", " && p.userId == ", "\n LIMIT 1\n REMOVE p IN posts\n RETURN OLD"]);
102
-
103
- _templateObject5 = function _templateObject5() {
104
- return data;
105
- };
106
-
107
- return data;
108
- }
109
-
110
- function _templateObject4() {
111
- var data = _taggedTemplateLiteral(["UPSERT {_key: ", ", userId: ", "}\n INSERT ", "\n UPDATE ", "\n IN posts RETURN NEW"]);
112
-
113
- _templateObject4 = function _templateObject4() {
114
- return data;
115
- };
116
-
117
- return data;
118
- }
119
-
120
- function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; }
121
-
122
- function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; if (i % 2) { ownKeys(Object(source), true).forEach(function (key) { _defineProperty(target, key, source[key]); }); } else if (Object.getOwnPropertyDescriptors) { Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)); } else { ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } } return target; }
123
-
124
- 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; }
125
-
126
- function _templateObject3() {
127
- var data = _taggedTemplateLiteral(["INSERT ", " IN posts RETURN NEW"]);
128
-
129
- _templateObject3 = function _templateObject3() {
130
- return data;
131
- };
132
-
133
- return data;
134
- }
135
-
136
- function _templateObject2() {
137
- var data = _taggedTemplateLiteral(["FOR p IN posts\n FILTER p.type == ", " && p._key == ", "\n LIMIT 1\n RETURN p"]);
138
-
139
- _templateObject2 = function _templateObject2() {
140
- return data;
141
- };
142
-
143
- return data;
144
- }
145
-
146
- function _templateObject() {
147
- var data = _taggedTemplateLiteral(["FOR p IN posts\n FILTER p._key == ", "\n LIMIT 1\n RETURN p"]);
148
-
149
- _templateObject = function _templateObject() {
150
- return data;
151
- };
152
-
153
- return data;
154
- }
155
-
156
- function _taggedTemplateLiteral(strings, raw) { if (!raw) { raw = strings.slice(0); } return Object.freeze(Object.defineProperties(strings, { raw: { value: Object.freeze(raw) } })); }
157
-
158
- var MAX_CONTENT_LENGTH = 100000;
159
- var eventCategory = 'posts';
160
-
161
- var parsePostOptions = function parsePostOptions() {
162
- var options = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
163
- var _options$from = options.from,
164
- from = _options$from === void 0 ? 0 : _options$from,
165
- _options$latitude = options.latitude,
166
- latitude = _options$latitude === void 0 ? 0 : _options$latitude,
167
- _options$longitude = options.longitude,
168
- longitude = _options$longitude === void 0 ? 0 : _options$longitude,
169
- _options$to = options.to,
170
- to = _options$to === void 0 ? 30 : _options$to,
171
- _options$type = options.type,
172
- type = _options$type === void 0 ? 'default' : _options$type;
173
- return {
174
- latitude: (0, _utils.parseNum)(latitude, 32),
175
- limit: (0, _utils2.getLimit)(from, to),
176
- longitude: (0, _utils.parseNum)(longitude, 32),
177
- type: (0, _utils.parseChar)(type, 32)
178
- };
179
- };
180
-
181
- exports.parsePostOptions = parsePostOptions;
182
-
183
- var getPostOptional = function getPostOptional(fields, sessionId) {
184
- return fields.reduce(function (selects, field) {
185
- switch (field) {
186
- case 'hasRsvp':
187
- {
188
- selects.queries.push("LET hasRsvp = TO_BOOL(FIRST(\n FOR post, r IN INBOUND p._id hasReactions\n FILTER r.name == \"rsvp\" && r.type == \"posts\" && r._from == \"users/".concat(sessionId, "\"\n COLLECT WITH COUNT INTO count\n RETURN count\n ))"));
189
- selects.objects.push('hasRsvp:hasRsvp');
190
- return selects;
191
- }
192
-
193
- case 'isSaved':
194
- {
195
- selects.queries.push("LET isSaved = TO_BOOL(FIRST(\n FOR post, r IN INBOUND p._id hasReactions\n FILTER r.name == \"pin\" && r.type == \"posts\" && r._from == \"users/".concat(sessionId, "\"\n COLLECT WITH COUNT INTO count\n RETURN count\n ))"));
196
- selects.objects.push('isSaved:isSaved');
197
- return selects;
198
- }
199
-
200
- case 'reactions':
201
- {
202
- selects.queries.push("LET reactions = (\n FOR post, r IN INBOUND p._id hasReactions\n COLLECT reactionName = r.value INTO reactionItems\n RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}\n )");
203
- selects.objects.push('reactions:reactions');
204
- return selects;
205
- }
206
-
207
- case 'rsvpCount':
208
- {
209
- selects.queries.push("LET rsvpCount = FIRST(\n FOR post, r IN INBOUND p._id hasReactions\n FILTER r.name == \"rsvp\" && r.type == \"posts\"\n COLLECT WITH COUNT INTO count\n RETURN count\n )");
210
- selects.objects.push('rsvpCount:rsvpCount');
211
- return selects;
212
- }
213
-
214
- case 'tags':
215
- {
216
- selects.queries.push("LET tags = (\n FOR t, pl IN INBOUND p._id isTagged\n RETURN t\n )");
217
- selects.objects.push('tags:tags');
218
- return selects;
219
- }
220
-
221
- case 'user':
222
- {
223
- selects.queries.push("LET user = FIRST(\n FOR u IN users\n FILTER p.userId == u._key\n LIMIT 1\n RETURN u\n )");
224
- selects.objects.push('user:user');
225
- return selects;
226
- }
227
-
228
- case 'viewCount':
229
- {
230
- selects.queries.push("LET viewCount = FIRST(\n FOR post, r IN INBOUND p._id hasReactions\n FILTER r.name == \"view\" && r.type == \"posts\"\n COLLECT WITH COUNT INTO count\n RETURN count\n )");
231
- selects.objects.push('viewCount:viewCount');
232
- return selects;
233
- }
234
-
235
- default:
236
- {
237
- return selects;
238
- }
239
- }
240
- }, {
241
- objects: [],
242
- queries: []
243
- });
244
- };
245
-
246
- exports.getPostOptional = getPostOptional;
247
-
248
- var getPost = function getPost(context, itemId) {
249
- // const action: string = 'getItem';
250
- var database = context.database,
251
- fields = context.fields,
252
- sessionId = context.userId;
253
- var formatItemId = (0, _utils.parseId)(itemId);
254
- var db = (0, _utils2.useDb)(database);
255
-
256
- var _getPostOptional = getPostOptional(fields, sessionId),
257
- selectObjects = _getPostOptional.objects,
258
- selectQueries = _getPostOptional.queries;
259
-
260
- var aqlQry = (0, _arangojs.aql)(_templateObject(), formatItemId);
261
- return db.query(aqlQry).then(function (cursor) {
262
- return cursor.next();
263
- }).then(function () {
264
- var post = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
265
- var _key = post._key,
266
- groupId = post.groupId,
267
- _post$privacy = post.privacy,
268
- privacy = _post$privacy === void 0 ? 'default' : _post$privacy; // Query based on privacy level
269
-
270
- var privacyAqlQry;
271
-
272
- if (groupId && privacy === 'group') {
273
- 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(', '), "})");
274
- } else if (privacy === 'public') {
275
- 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(', '), "})");
276
- }
277
-
278
- if (privacyAqlQry) {
279
- return db.query(privacyAqlQry).then(function (cursor) {
280
- return cursor.next();
281
- }).then(function () {
282
- var filteredPost = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
283
- return filteredPost;
284
- })["catch"](function (error) {
285
- throw error;
286
- });
287
- }
288
-
289
- return {};
290
- })["catch"](function (error) {
291
- throw error;
292
- });
293
- };
294
-
295
- exports.getPost = getPost;
296
-
297
- var getPostList = function getPostList(context, options) {
298
- // const action: string = 'getListByApp';
299
- var database = context.database,
300
- fields = context.fields,
301
- sessionId = context.userId;
302
- console.log('getPostList::context', context);
303
-
304
- var _parsePostOptions = parsePostOptions(options),
305
- limit = _parsePostOptions.limit,
306
- type = _parsePostOptions.type;
307
-
308
- var _getPostOptional2 = getPostOptional(fields, sessionId),
309
- selectObjects = _getPostOptional2.objects,
310
- selectQueries = _getPostOptional2.queries;
311
-
312
- 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(', '), "})");
313
- return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
314
- return cursor.all();
315
- })["catch"](function (error) {
316
- console.log('getPostList::error', error);
317
- throw error;
318
- });
319
- };
320
-
321
- exports.getPostList = getPostList;
322
-
323
- var getPostsByArea = function getPostsByArea(context, latitude, longitude, options) {
324
- // const action: string = 'getListByUser';
325
- var database = context.database,
326
- fields = context.fields,
327
- sessionId = context.userId;
328
-
329
- var _parsePostOptions2 = parsePostOptions(options),
330
- limit = _parsePostOptions2.limit,
331
- type = _parsePostOptions2.type;
332
-
333
- var formatLatitude = (0, _utils.parseNum)(latitude);
334
- var formatLongitude = (0, _utils.parseNum)(longitude);
335
-
336
- var _getPostOptional3 = getPostOptional(fields, sessionId),
337
- selectObjects = _getPostOptional3.objects,
338
- selectQueries = _getPostOptional3.queries;
339
-
340
- selectQueries.push("LET distance = DISTANCE(\n ".concat(formatLatitude, ",\n ").concat(formatLongitude, ",\n NOT_NULL(p.latitude, 0),\n NOT_NULL(p.longitude, 0))\n "));
341
- selectObjects.push('distance:distance');
342
- 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(', '), "})");
343
- return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
344
- return cursor.all();
345
- })["catch"](function (error) {
346
- throw error;
347
- });
348
- };
349
-
350
- exports.getPostsByArea = getPostsByArea;
351
-
352
- var getPostsByGroup = function getPostsByGroup(context, groupId, options) {
353
- // const action: string = 'getListByGroup';
354
- var database = context.database,
355
- fields = context.fields,
356
- sessionId = context.userId;
357
-
358
- var _getPostOptional4 = getPostOptional(fields, sessionId),
359
- selectObjects = _getPostOptional4.objects,
360
- selectQueries = _getPostOptional4.queries; // Group id
361
-
362
-
363
- var formatGroupId = (0, _utils.parseId)(groupId);
364
- var db = (0, _utils2.useDb)(database);
365
- var aqlQry = "FOR u, g IN INBOUND ".concat(formatGroupId, " hasGroup\n FILTER u._key == ").concat(sessionId, "\n RETURN g");
366
- return db.query(aqlQry).then(function (cursor) {
367
- return cursor.all();
368
- }).then(function () {
369
- var groups = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [];
370
-
371
- if (groups.length) {
372
- var _parsePostOptions3 = parsePostOptions(options),
373
- limit = _parsePostOptions3.limit,
374
- type = _parsePostOptions3.type;
375
-
376
- 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(', '), "})");
377
- return db.query(postAqlQry).then(function (cursor) {
378
- return cursor.all();
379
- })["catch"](function (error) {
380
- throw error;
381
- });
382
- }
383
-
384
- return [];
385
- })["catch"](function (error) {
386
- throw error;
387
- });
388
- };
389
-
390
- exports.getPostsByGroup = getPostsByGroup;
391
-
392
- var getPostsByLatest = function getPostsByLatest(context, options) {
393
- // const action: string = 'getListByLatest';
394
- console.log('getPostsByLatest::options', options);
395
- var database = context.database,
396
- fields = context.fields,
397
- sessionId = context.userId;
398
-
399
- var _parsePostOptions4 = parsePostOptions(options),
400
- limit = _parsePostOptions4.limit,
401
- type = _parsePostOptions4.type;
402
-
403
- var _getPostOptional5 = getPostOptional(fields, sessionId),
404
- selectObjects = _getPostOptional5.objects,
405
- selectQueries = _getPostOptional5.queries;
406
-
407
- 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 DESC\n RETURN DISTINCT MERGE(p, {").concat(selectObjects.join(', '), "})");
408
- console.log('getPostsByLatest::aqlQry', aqlQry);
409
- return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
410
- return cursor.all();
411
- })["catch"](function (error) {
412
- throw error;
413
- });
414
- };
415
-
416
- exports.getPostsByLatest = getPostsByLatest;
417
-
418
- var getPostsByReaction = function getPostsByReaction(context, reactionName, options) {
419
- var action = 'getPostsByReaction';
420
- var database = context.database,
421
- fields = context.fields,
422
- sessionId = context.userId;
423
-
424
- var _parsePostOptions5 = parsePostOptions(options),
425
- limit = _parsePostOptions5.limit;
426
-
427
- var _getPostOptional6 = getPostOptional(fields, sessionId),
428
- selectObjects = _getPostOptional6.objects,
429
- selectQueries = _getPostOptional6.queries;
430
-
431
- var formatSessionId = "users/".concat(sessionId);
432
- var formatReactionName = (0, _utils.parseChar)(reactionName, 32); // Get data from database
433
-
434
- var aqlQry = "FOR p, r IN OUTBOUND \"".concat(formatSessionId, "\" hasReactions\n ").concat(selectQueries.join('\n'), "\n FILTER r.type == \"posts\" && r.name == \"").concat(formatReactionName, "\"\n ").concat(limit.aql, "\n RETURN MERGE(p, {").concat(selectObjects.join(', '), "})");
435
- return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
436
- return cursor.all();
437
- })["catch"](function (error) {
438
- return (0, _utils2.logError)({
439
- action: action,
440
- category: eventCategory,
441
- label: 'db_error'
442
- }, error, context).then(function () {
443
- return [];
444
- });
445
- });
446
- };
447
-
448
- exports.getPostsByReaction = getPostsByReaction;
449
-
450
- var getPostsByTags = function getPostsByTags(context, tagNames, options) {
451
- // const action: string = 'getListByTags';
452
- var database = context.database,
453
- fields = context.fields,
454
- sessionId = context.userId;
455
-
456
- var _parsePostOptions6 = parsePostOptions(options),
457
- latitude = _parsePostOptions6.latitude,
458
- longitude = _parsePostOptions6.longitude,
459
- limit = _parsePostOptions6.limit,
460
- type = _parsePostOptions6.type;
461
-
462
- var _getPostOptional7 = getPostOptional(fields, sessionId),
463
- selectObjects = _getPostOptional7.objects,
464
- selectQueries = _getPostOptional7.queries;
465
-
466
- var sortBy = [];
467
-
468
- if (latitude !== undefined && longitude !== undefined) {
469
- var formatLatitude = (0, _utils.parseNum)(latitude);
470
- var formatLongitude = (0, _utils.parseNum)(longitude);
471
- selectQueries.push("LET distance = DISTANCE(\n ".concat(formatLatitude, ",\n ").concat(formatLongitude, ",\n NOT_NULL(p.latitude, 0),\n NOT_NULL(p.longitude, 0))\n "));
472
- selectObjects.push('distance:distance');
473
- sortBy.push('distance');
474
- }
475
-
476
- sortBy.push('p.added');
477
- return Promise.all(tagNames.map(function (tagName) {
478
- var aqlQry = "FOR targetTag IN tags\n FILTER LOWER(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(', '), "})");
479
- return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
480
- return cursor.all();
481
- })["catch"](function () {
482
- return [];
483
- });
484
- })).then(function (results) {
485
- return (0, _uniqBy["default"])((0, _flatten["default"])(results), '_key');
486
- })["catch"](function (error) {
487
- throw error;
488
- });
489
- };
490
-
491
- exports.getPostsByTags = getPostsByTags;
492
-
493
- var getPostsByUser = function getPostsByUser(context, userId, options) {
494
- // const action: string = 'getListByUser';
495
- var database = context.database,
496
- fields = context.fields,
497
- sessionId = context.userId;
498
-
499
- var _parsePostOptions7 = parsePostOptions(options),
500
- limit = _parsePostOptions7.limit,
501
- type = _parsePostOptions7.type;
502
-
503
- var formatUserId = (0, _utils.parseId)(userId);
504
-
505
- var _getPostOptional8 = getPostOptional(fields, sessionId),
506
- selectObjects = _getPostOptional8.objects,
507
- selectQueries = _getPostOptional8.queries;
508
-
509
- 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(', '), "})");
510
- return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
511
- return cursor.all();
512
- })["catch"](function (error) {
513
- throw error;
514
- });
515
- };
516
-
517
- exports.getPostsByUser = getPostsByUser;
518
-
519
- var getPostComments = function getPostComments(context, itemId, options) {
520
- // const action: string = 'getComments';
521
- var database = context.database,
522
- sessionId = context.userId;
523
-
524
- var _parsePostOptions8 = parsePostOptions(options),
525
- limit = _parsePostOptions8.limit,
526
- type = _parsePostOptions8.type;
527
-
528
- var formatItemId = (0, _utils.parseId)(itemId); // Get the parent post to get restrictions
529
-
530
- var db = (0, _utils2.useDb)(database);
531
- var aqlQry = (0, _arangojs.aql)(_templateObject2(), type, formatItemId);
532
- return db.query(aqlQry).then(function (cursor) {
533
- return cursor.next();
534
- }).then(function () {
535
- var post = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
536
- var _key = post._key,
537
- groupId = post.groupId,
538
- _post$privacy2 = post.privacy,
539
- privacy = _post$privacy2 === void 0 ? 'public' : _post$privacy2; // Query based on privacy level
540
-
541
- var privacyAqlQry;
542
-
543
- if (groupId && privacy === 'group') {
544
- 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})");
545
- } else if (privacy === 'public') {
546
- 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})");
547
- }
548
-
549
- if (privacyAqlQry) {
550
- return db.query(privacyAqlQry).then(function (cursor) {
551
- return cursor.all();
552
- })["catch"](function (error) {
553
- throw error;
554
- });
555
- }
556
-
557
- return [];
558
- })["catch"](function (error) {
559
- throw error;
560
- });
561
- };
562
-
563
- exports.getPostComments = getPostComments;
564
-
565
- var addPost = function addPost(context, item) {
566
- // const action: string = 'add';
567
- var database = context.database,
568
- sessionId = context.userId;
569
- var _item$content = item.content,
570
- content = _item$content === void 0 ? '' : _item$content,
571
- endDate = item.endDate,
572
- _item$groupId = item.groupId,
573
- groupId = _item$groupId === void 0 ? '' : _item$groupId,
574
- location = item.location,
575
- latitude = item.latitude,
576
- longitude = item.longitude,
577
- _item$name = item.name,
578
- name = _item$name === void 0 ? '' : _item$name,
579
- _item$parentId = item.parentId,
580
- parentId = _item$parentId === void 0 ? null : _item$parentId,
581
- _item$privacy = item.privacy,
582
- privacy = _item$privacy === void 0 ? 'public' : _item$privacy,
583
- startDate = item.startDate,
584
- _item$type = item.type,
585
- type = _item$type === void 0 ? 'default' : _item$type;
586
- var now = Date.now();
587
- var insert = {
588
- _key: (0, _utils.createHash)("post-".concat(sessionId)),
589
- added: now,
590
- content: (0, _utils.parseString)(content, MAX_CONTENT_LENGTH),
591
- endDate: endDate ? (0, _utils.parseNum)(endDate, 13) : undefined,
592
- groupId: groupId ? (0, _utils.parseId)(groupId) : undefined,
593
- latitude: latitude !== undefined ? (0, _utils.parseNum)(latitude) : undefined,
594
- location: location ? (0, _utils.parseString)(location, 160) : undefined,
595
- longitude: longitude !== undefined ? (0, _utils.parseNum)(longitude) : undefined,
596
- modified: now,
597
- name: (0, _utils.parseString)(name, 160),
598
- parentId: parentId ? (0, _utils.parseId)(parentId) : undefined,
599
- privacy: privacy ? (0, _utils.parseVarChar)(privacy, 16) : undefined,
600
- startDate: startDate ? (0, _utils.parseNum)(startDate, 13) : undefined,
601
- type: (0, _utils.parseChar)(type, 32),
602
- userId: sessionId
603
- };
604
- var db = (0, _utils2.useDb)(database);
605
- var aqlQry = (0, _arangojs.aql)(_templateObject3(), insert);
606
- return db.query(aqlQry).then(function (cursor) {
607
- return cursor.next();
608
- }).then(function () {
609
- var post = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
610
- var postKey = post._key; // Update linked tags within posts
611
-
612
- return (0, _tags.extractTags)(db, 'posts', postKey, insert.content).then(function (tagList) {
613
- post.tags = tagList;
614
- return post;
615
- });
616
- })["catch"](function (error) {
617
- throw error;
618
- });
619
- };
620
-
621
- exports.addPost = addPost;
622
-
623
- var updatePost = function updatePost(context, item) {
624
- // const action: string = 'update';
625
- var database = context.database,
626
- sessionId = context.userId;
627
- var now = Date.now();
628
- var content = item.content,
629
- endDate = item.endDate,
630
- groupId = item.groupId,
631
- name = item.name,
632
- parentId = item.parentId,
633
- postId = item.postId,
634
- privacy = item.privacy,
635
- startDate = item.startDate,
636
- type = item.type;
637
- var update = {
638
- content: content ? (0, _utils.parseString)(content, MAX_CONTENT_LENGTH) : undefined,
639
- endDate: endDate ? (0, _utils.parseNum)(endDate, 13) : undefined,
640
- modified: now,
641
- name: name ? (0, _utils.parseString)(name, 160) : undefined,
642
- parentId: parentId ? (0, _utils.parseString)(parentId, 160) : undefined,
643
- privacy: privacy ? (0, _utils.parseVarChar)(privacy, 16) : undefined,
644
- startDate: startDate ? (0, _utils.parseNum)(startDate, 13) : undefined,
645
- type: type !== undefined ? (0, _utils.parseChar)(type, 16) : undefined
646
- };
647
- var formatId = (0, _utils.parseId)(postId);
648
- formatId = formatId === '' ? (0, _utils.createHash)("post-".concat(sessionId)) : formatId;
649
- var formatGroupId = (0, _utils.parseId)(groupId);
650
-
651
- var insert = _objectSpread(_objectSpread({}, update), {}, {
652
- _key: formatId,
653
- added: now,
654
- groupId: formatGroupId,
655
- privacy: privacy,
656
- userId: sessionId
657
- });
658
-
659
- var db = (0, _utils2.useDb)(database);
660
- var aqlQry = (0, _arangojs.aql)(_templateObject4(), formatId, sessionId, insert, update);
661
- return db.query(aqlQry).then(function (cursor) {
662
- return cursor.next();
663
- }).then(function () {
664
- var updatedPost = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
665
- var updatedPostKey = updatedPost._key; // Update linked tags
666
-
667
- return (0, _tags.extractTags)(db, 'posts', updatedPostKey, update.content || '').then(function () {
668
- var tagList = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [];
669
- updatedPost.tags = tagList; // Update linked files
670
-
671
- var files = updatedPost.files || [];
672
-
673
- if (files.length) {
674
- return (0, _files.updateFiles)(db, formatId, files).then(function () {
675
- var fileList = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [];
676
- updatedPost.files = fileList;
677
- return updatedPost;
678
- });
679
- }
680
-
681
- updatedPost.files = [];
682
- return updatedPost;
683
- });
684
- })["catch"](function (error) {
685
- throw error;
686
- });
687
- };
688
-
689
- exports.updatePost = updatePost;
690
-
691
- var deletePost = function deletePost(context, itemId) {
692
- // const action: string = 'delete';
693
- var database = context.database,
694
- sessionId = context.userId;
695
- var formatItemId = (0, _utils.parseId)(itemId);
696
- var db = (0, _utils2.useDb)(database);
697
- var aqlQry = (0, _arangojs.aql)(_templateObject5(), formatItemId, sessionId);
698
- return db.query(aqlQry).then(function (cursor) {
699
- return cursor.next();
700
- }).then(function () {
701
- var post = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
702
-
703
- if (post) {
704
- // Remove tag links
705
- var edgeAqlQry = (0, _arangojs.aql)(_templateObject6(), formatItemId);
706
- return db.query(edgeAqlQry).then(function () {
707
- // Remove attached files
708
- var fileAqlQry = (0, _arangojs.aql)(_templateObject7(), formatItemId);
709
- return db.query(fileAqlQry).then(function () {
710
- return post;
711
- })["catch"](function (error) {
712
- throw error;
713
- });
714
- })["catch"](function (error) {
715
- throw error;
716
- });
717
- }
718
-
719
- return {};
720
- })["catch"](function (error) {
721
- throw error;
722
- });
723
- };
724
-
725
- exports.deletePost = deletePost;
726
-
727
- var cleanPosts = function cleanPosts(database) {
728
- // Remove all messages that are over 60 days and not saved
729
- var aqlQry = (0, _arangojs.aql)(_templateObject8());
730
- return (0, _utils2.useDb)(database).query(aqlQry).then(function (cursor) {
731
- return cursor.all();
732
- }).then(function () {
733
- var results = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [];
734
- return results.length;
735
- })["catch"](function (error) {
736
- throw error;
737
- });
738
- };
739
-
740
- exports.cleanPosts = cleanPosts;
741
-
742
- var createPostEdge = function createPostEdge(db, file, postId) {
743
- var edgeCollection = db.collection('isPosted');
744
- var fileId = file.fileId;
745
- var formatFileId = (0, _utils.parseId)(fileId);
746
- var edgeId = (0, _utils.createHash)("file-".concat(postId, "-").concat(formatFileId));
747
- var formatPostId = (0, _utils.parseId)(postId);
748
- var fileType = (0, _utils.parseChar)(file.fileType, 16);
749
- var edge = {
750
- _from: "posts/".concat(formatPostId),
751
- _key: edgeId,
752
- _to: "files/".concat(formatFileId),
753
- added: Date.now(),
754
- type: fileType
755
- };
756
- return edgeCollection.save(edge, {
757
- returnNew: true
758
- }).then(function () {
759
- return file;
760
- })["catch"](function (error) {
761
- throw error;
762
- });
763
- };
764
-
765
- exports.createPostEdge = createPostEdge;
766
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kYXRhL3Bvc3RzLnRzIl0sIm5hbWVzIjpbIk1BWF9DT05URU5UX0xFTkdUSCIsImV2ZW50Q2F0ZWdvcnkiLCJwYXJzZVBvc3RPcHRpb25zIiwib3B0aW9ucyIsImZyb20iLCJsYXRpdHVkZSIsImxvbmdpdHVkZSIsInRvIiwidHlwZSIsImxpbWl0IiwiZ2V0UG9zdE9wdGlvbmFsIiwiZmllbGRzIiwic2Vzc2lvbklkIiwicmVkdWNlIiwic2VsZWN0cyIsImZpZWxkIiwicXVlcmllcyIsInB1c2giLCJvYmplY3RzIiwiZ2V0UG9zdCIsImNvbnRleHQiLCJpdGVtSWQiLCJkYXRhYmFzZSIsInVzZXJJZCIsImZvcm1hdEl0ZW1JZCIsImRiIiwic2VsZWN0T2JqZWN0cyIsInNlbGVjdFF1ZXJpZXMiLCJhcWxRcnkiLCJhcWwiLCJxdWVyeSIsInRoZW4iLCJjdXJzb3IiLCJuZXh0IiwicG9zdCIsIl9rZXkiLCJncm91cElkIiwicHJpdmFjeSIsInByaXZhY3lBcWxRcnkiLCJqb2luIiwiZmlsdGVyZWRQb3N0IiwiZXJyb3IiLCJnZXRQb3N0TGlzdCIsImNvbnNvbGUiLCJsb2ciLCJhbGwiLCJnZXRQb3N0c0J5QXJlYSIsImZvcm1hdExhdGl0dWRlIiwiZm9ybWF0TG9uZ2l0dWRlIiwiZ2V0UG9zdHNCeUdyb3VwIiwiZm9ybWF0R3JvdXBJZCIsImdyb3VwcyIsImxlbmd0aCIsInBvc3RBcWxRcnkiLCJnZXRQb3N0c0J5TGF0ZXN0IiwiZ2V0UG9zdHNCeVJlYWN0aW9uIiwicmVhY3Rpb25OYW1lIiwiYWN0aW9uIiwiZm9ybWF0U2Vzc2lvbklkIiwiZm9ybWF0UmVhY3Rpb25OYW1lIiwiY2F0ZWdvcnkiLCJsYWJlbCIsImdldFBvc3RzQnlUYWdzIiwidGFnTmFtZXMiLCJzb3J0QnkiLCJ1bmRlZmluZWQiLCJQcm9taXNlIiwibWFwIiwidGFnTmFtZSIsInJlc3VsdHMiLCJnZXRQb3N0c0J5VXNlciIsImZvcm1hdFVzZXJJZCIsImdldFBvc3RDb21tZW50cyIsImFkZFBvc3QiLCJpdGVtIiwiY29udGVudCIsImVuZERhdGUiLCJsb2NhdGlvbiIsIm5hbWUiLCJwYXJlbnRJZCIsInN0YXJ0RGF0ZSIsIm5vdyIsIkRhdGUiLCJpbnNlcnQiLCJhZGRlZCIsIm1vZGlmaWVkIiwicG9zdEtleSIsInRhZ0xpc3QiLCJ0YWdzIiwidXBkYXRlUG9zdCIsInBvc3RJZCIsInVwZGF0ZSIsImZvcm1hdElkIiwidXBkYXRlZFBvc3QiLCJ1cGRhdGVkUG9zdEtleSIsImZpbGVzIiwiZmlsZUxpc3QiLCJkZWxldGVQb3N0IiwiZWRnZUFxbFFyeSIsImZpbGVBcWxRcnkiLCJjbGVhblBvc3RzIiwiY3JlYXRlUG9zdEVkZ2UiLCJmaWxlIiwiZWRnZUNvbGxlY3Rpb24iLCJjb2xsZWN0aW9uIiwiZmlsZUlkIiwiZm9ybWF0RmlsZUlkIiwiZWRnZUlkIiwiZm9ybWF0UG9zdElkIiwiZmlsZVR5cGUiLCJlZGdlIiwiX2Zyb20iLCJfdG8iLCJzYXZlIiwicmV0dXJuTmV3Il0sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUlBOztBQUNBOztBQUlBOztBQUNBOztBQUdBOztBQUNBOztBQUNBOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUVBLElBQU1BLGtCQUEwQixHQUFHLE1BQW5DO0FBQ0EsSUFBTUMsYUFBcUIsR0FBRyxPQUE5Qjs7QUFFTyxJQUFNQyxnQkFBZ0IsR0FBRyxTQUFuQkEsZ0JBQW1CLEdBQStCO0FBQUEsTUFBOUJDLE9BQThCLHVFQUFQLEVBQU87QUFBQSxzQkFPekRBLE9BUHlELENBRTNEQyxJQUYyRDtBQUFBLE1BRTNEQSxJQUYyRCw4QkFFcEQsQ0FGb0Q7QUFBQSwwQkFPekRELE9BUHlELENBRzNERSxRQUgyRDtBQUFBLE1BRzNEQSxRQUgyRCxrQ0FHaEQsQ0FI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TUFBTUUsTUFBYyxHQUFHLHNDQUFtQmpCLE1BQW5CLGNBQTZCZ0IsWUFBN0IsRUFBdkI7QUFDQSxNQUFNRSxZQUFvQixHQUFHLG9CQUFRbEIsTUFBUixDQUE3QjtBQUNBLE1BQU1tQixRQUFnQixHQUFHLHNCQUFVUCxJQUFJLENBQUNPLFFBQWYsRUFBeUIsRUFBekIsQ0FBekI7QUFFQSxNQUFNQyxJQUFTLEdBQUc7QUFDaEJDLElBQUFBLEtBQUssa0JBQVdILFlBQVgsQ0FEVztBQUVoQnpFLElBQUFBLElBQUksRUFBRXdFLE1BRlU7QUFHaEJLLElBQUFBLEdBQUcsa0JBQVdOLFlBQVgsQ0FIYTtBQUloQnRCLElBQUFBLEtBQUssRUFBRUYsSUFBSSxDQUFDRCxHQUFMLEVBSlM7QUFLaEJ6RSxJQUFBQSxJQUFJLEVBQUVxRztBQUxVLEdBQWxCO0FBUUEsU0FBT04sY0FBYyxDQUFDVSxJQUFmLENBQW9CSCxJQUFwQixFQUEwQjtBQUFDSSxJQUFBQSxTQUFTLEVBQUU7QUFBWixHQUExQixFQUNKbkYsSUFESSxDQUNDO0FBQUEsV0FBTXVFLElBQU47QUFBQSxHQURELFdBRUUsVUFBQzdELEtBQUQsRUFBa0I7QUFDdkIsVUFBTUEsS0FBTjtBQUNELEdBSkksQ0FBUDtBQUtELENBckJNIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBDb3B5cmlnaHQgKGMpIDIwMTktUHJlc2VudCwgTml0cm9nZW4gTGFicywgSW5jLlxuICogQ29weXJpZ2h0cyBsaWNlbnNlZCB1bmRlciB0aGUgTUlUIExpY2Vuc2UuIFNlZSB0aGUgYWNjb21wYW55aW5nIExJQ0VOU0UgZmlsZSBmb3IgdGVybXMuXG4gKi9cbmltcG9ydCB7Y3JlYXRlSGFzaCwgcGFyc2VDaGFyLCBwYXJzZUlkLCBwYXJzZU51bSwgcGFyc2VTdHJpbmcsIHBhcnNlVmFyQ2hhcn0gZnJvbSAnQG5sYWJzL3V0aWxzJztcbmltcG9ydCB7YXFsLCBEYXRhYmFzZX0gZnJvbSAnYXJhbmdvanMnO1xuaW1wb3J0IHtBcWxRdWVyeX0gZnJvbSAnYXJhbmdvanMvYXFsJztcbmltcG9ydCB7RWRnZUNvbGxlY3Rpb259IGZyb20gJ2FyYW5nb2pzL2NvbGxlY3Rpb24nO1xuaW1wb3J0IHtBcnJheUN1cnNvcn0gZnJvbSAnYXJhbmdvanMvY3Vyc29yJztcbmltcG9ydCBmbGF0dGVuIGZyb20gJ2xvZGFzaC9mbGF0dGVuJztcbmltcG9ydCB1bmlxQnkgZnJvbSAnbG9kYXNoL3VuaXFCeSc7XG5cbmltcG9ydCB7QXBpQ29udGV4dCwgRmlsZVR5cGUsIEdyb3VwVHlwZSwgUG9zdE9wdGlvbnMsIFBvc3RUeXBlLCBUYWdUeXBlfSBmcm9tICcuLi90eXBlcyc7XG5pbXBvcnQge2dldExpbWl0LCBsb2dFcnJvciwgdXNlRGJ9IGZyb20gJy4uL3V0aWxzJztcbmltcG9ydCB7dXBkYXRlRmlsZXN9IGZyb20gJy4vZmlsZXMnO1xuaW1wb3J0IHtleHRyYWN0VGFnc30gZnJvbSAnLi90YWdzJztcblxuY29uc3QgTUFYX0NPTlRFTlRfTEVOR1RIOiBudW1iZXIgPSAxMDAwMDA7XG5jb25zdCBldmVudENhdGVnb3J5OiBzdHJpbmcgPSAncG9zdHMnO1xuXG5leHBvcnQgY29uc3QgcGFyc2VQb3N0T3B0aW9ucyA9IChvcHRpb25zOiBQb3N0T3B0aW9ucyA9IHt9KSA9PiB7XG4gIGNvbnN0IHtcbiAgICBmcm9tID0gMCxcbiAgICBsYXRpdHVkZSA9IDAsXG4gICAgbG9uZ2l0dWRlID0gMCxcbiAgICB0byA9IDMwLFxuICAgIHR5cGUgPSAnZGVmYXVsdCdcbiAgfSA9IG9wdGlvbnM7XG5cbiAgcmV0dXJuIHtcbiAgICBsYXRpdHVkZTogcGFyc2VOdW0obGF0aXR1ZGUsIDMyKSxcbiAgICBsaW1pdDogZ2V0TGltaXQoZnJvbSwgdG8pLFxuICAgIGxvbmdpdHVkZTogcGFyc2VOdW0obG9uZ2l0dWRlLCAzMiksXG4gICAgdHlwZTogcGFyc2VDaGFyKHR5cGUsIDMyKVxuICB9O1xufTtcblxuZXhwb3J0IGNvbnN0IGdldFBvc3RPcHRpb25hbCA9IChmaWVsZHM6IHN0cmluZ1tdLCBzZXNzaW9uSWQ6IHN0cmluZykgPT5cbiAgZmllbGRzLnJlZHVjZSgoc2VsZWN0czogYW55LCBmaWVsZDogc3RyaW5nKSA9PiB7XG4gICAgc3dpdGNoKGZpZWxkKSB7XG4gICAgICBjYXNlICdoYXNSc3ZwJzoge1xuICAgICAgICBzZWxlY3RzLnF1ZXJpZXMucHVzaChgTEVUIGhhc1JzdnAgPSBUT19CT09MKEZJUlNUKFxuICAgICAgICAgIEZPUiBwb3N0LCByIElOIElOQk9VTkQgcC5faWQgaGFzUmVhY3Rpb25zXG4gICAgICAgICAgRklMVEVSIHIubmFtZSA9PSBcInJzdnBcIiAmJiByLnR5cGUgPT0gXCJwb3N0c1wiICYmIHIuX2Zyb20gPT0gXCJ1c2Vycy8ke3Nlc3Npb25JZH1cIlxuICAgICAgICAgIENPTExFQ1QgV0lUSCBDT1VOVCBJTlRPIGNvdW50XG4gICAgICAgICAgUkVUVVJOIGNvdW50XG4gICAgICAgICkpYCk7XG4gICAgICAgIHNlbGVjdHMub2JqZWN0cy5wdXNoKCdoYXNSc3ZwOmhhc1JzdnAnKTtcbiAgICAgICAgcmV0dXJuIHNlbGVjdHM7XG4gICAgICB9XG4gICAgICBjYXNlICdpc1NhdmVkJzoge1xuICAgICAgICBzZWxlY3RzLnF1ZXJpZXMucHVzaChgTEVUIGlzU2F2ZWQgPSBUT19CT09MKEZJUlNUKFxuICAgICAgICAgIEZPUiBwb3N0LCByIElOIElOQk9VTkQgcC5faWQgaGFzUmVhY3Rpb25zXG4gICAgICAgICAgRklMVEVSIHIubmFtZSA9PSBcInBpblwiICYmIHIudHlwZSA9PSBcInBvc3RzXCIgJiYgci5fZnJvbSA9PSBcInVzZXJzLyR7c2Vzc2lvbklkfVwiXG4gICAgICAgICAgQ09MTEVDVCBXSVRIIENPVU5UIElOVE8gY291bnRcbiAgICAgICAgICBSRVRVUk4gY291bnRcbiAgICAgICAgKSlgKTtcbiAgICAgICAgc2VsZWN0cy5vYmplY3RzLnB1c2goJ2lzU2F2ZWQ6aXNTYXZlZCcpO1xuICAgICAgICByZXR1cm4gc2VsZWN0cztcbiAgICAgIH1cbiAgICAgIGNhc2UgJ3JlYWN0aW9ucyc6IHtcbiAgICAgICAgc2VsZWN0cy5xdWVyaWVzLnB1c2goYExFVCByZWFjdGlvbnMgPSAoXG4gICAgICAgICAgRk9SIHBvc3QsIHIgSU4gSU5CT1VORCBwLl9pZCBoYXNSZWFjdGlvbnNcbiAgICAgICAgICBDT0xMRUNUIHJlYWN0aW9uTmFtZSA9IHIudmFsdWUgSU5UTyByZWFjdGlvbkl0ZW1zXG4gICAgICAgICAgUkVUVVJOIHtuYW1lOiByZWFjdGlvbk5hbWUsIGNvdW50OiBMRU5HVEgocmVhY3Rpb25JdGVtc1sqXS5yLnZhbHVlKX1cbiAgICAgICAgKWApO1xuICAgICAgICBzZWxlY3RzLm9iamVjdHMucHVzaCgncmVhY3Rpb25zOnJlYWN0aW9ucycpO1xuICAgICAgICByZXR1cm4gc2VsZWN0cztcbiAgICAgIH1cbiAgICAgIGNhc2UgJ3JzdnBDb3VudCc6IHtcbiAgICAgICAgc2VsZWN0cy5xdWVyaWVzLnB1c2goYExFVCByc3ZwQ291bnQgPSBGSVJTVChcbiAgICAgICAgICBGT1IgcG9zdCwgciBJTiBJTkJPVU5EIHAuX2lkIGhhc1JlYWN0aW9uc1xuICAgICAgICAgIEZJTFRFUiByLm5hbWUgPT0gXCJyc3ZwXCIgJiYgci50eXBlID09IFwicG9zdHNcIlxuICAgICAgICAgIENPTExFQ1QgV0lUSCBDT1VOVCBJTlRPIGNvdW50XG4gICAgICAgICAgUkVUVVJOIGNvdW50XG4gICAgICAgIClgKTtcbiAgICAgICAgc2VsZWN0cy5vYmplY3RzLnB1c2goJ3JzdnBDb3VudDpyc3ZwQ291bnQnKTtcbiAgICAgICAgcmV0dXJuIHNlbGVjdHM7XG4gICAgICB9XG4gICAgICBjYXNlICd0YWdzJzoge1xuICAgICAgICBzZWxlY3RzLnF1ZXJpZXMucHVzaChgTEVUIHRhZ3MgPSAoXG4gICAgICAgICAgRk9SIHQsIHBsIElOIElOQk9VTkQgcC5faWQgaXNUYWdnZWRcbiAgICAgICAgICBSRVRVUk4gdFxuICAgICAgICApYCk7XG4gICAgICAgIHNlbGVjdHMub2JqZWN0cy5wdXNoKCd0YWdzOnRhZ3MnKTtcbiAgICAgICAgcmV0dXJuIHNlbGVjdHM7XG4gICAgICB9XG4gICAgICBjYXNlICd1c2VyJzoge1xuICAgICAgICBzZWxlY3RzLnF1ZXJpZXMucHVzaChgTEVUIHVzZXIgPSBGSVJTVChcbiAgICAgICAgICBGT1IgdSBJTiB1c2Vyc1xuICAgICAgICAgIEZJTFRFUiBwLnVzZXJJZCA9PSB1Ll9rZXlcbiAgICAgICAgICBMSU1JVCAxXG4gICAgICAgICAgUkVUVVJOIHVcbiAgICAgICAgKWApO1xuICAgICAgICBzZWxlY3RzLm9iamVjdHMucHVzaCgndXNlcjp1c2VyJyk7XG4gICAgICAgIHJldHVybiBzZWxlY3RzO1xuICAgICAgfVxuICAgICAgY2FzZSAndmlld0NvdW50Jzoge1xuICAgICAgICBzZWxlY3RzLnF1ZXJpZXMucHVzaChgTEVUIHZpZXdDb3VudCA9IEZJUlNUKFxuICAgICAgICAgIEZPUiBwb3N0LCByIElOIElOQk9VTkQgcC5faWQgaGFzUmVhY3Rpb25zXG4gICAgICAgICAgRklMVEVSIHIubmFtZSA9PSBcInZpZXdcIiAmJiByLnR5cGUgPT0gXCJwb3N0c1wiXG4gICAgICAgICAgQ09MTEVDVCBXSVRIIENPVU5UIElOVE8gY291bnRcbiAgICAgICAgICBSRVRVUk4gY291bnRcbiAgICAgICAgKWApO1xuICAgICAgICBzZWxlY3RzLm9iamVjdHMucHVzaCgndmlld0NvdW50OnZpZXdDb3VudCcpO1xuICAgICAgICByZXR1cm4gc2VsZWN0cztcbiAgICAgIH1cbiAgICAgIGRlZmF1bHQ6IHtcbiAgICAgICAgcmV0dXJuIHNlbGVjdHM7XG4gICAgICB9XG4gICAgfVxuICB9LCB7b2JqZWN0czogW10sIHF1ZXJpZXM6IFtdfSk7XG5cbmV4cG9ydCBjb25zdCBnZXRQb3N0ID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGl0ZW1JZDogc3RyaW5nKTogUHJvbWlzZTxQb3N0VHlwZT4gPT4ge1xuICAvLyBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRJdGVtJztcbiAgY29uc3Qge2RhdGFiYXNlLCBmaWVsZHMsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG4gIGNvbnN0IGZvcm1hdEl0ZW1JZDogc3RyaW5nID0gcGFyc2VJZChpdGVtSWQpO1xuICBjb25zdCBkYiA9IHVzZURiKGRhdGFiYXNlKTtcbiAgY29uc3Qge29iamVjdHM6IHNlbGVjdE9iamVjdHMsIHF1ZXJpZXM6IHNlbGVjdFF1ZXJpZXN9ID0gZ2V0UG9zdE9wdGlvbmFsKGZpZWxkcywgc2Vzc2lvbklkKTtcbiAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBGT1IgcCBJTiBwb3N0c1xuICAgIEZJTFRFUiBwLl9rZXkgPT0gJHtmb3JtYXRJdGVtSWR9XG4gICAgTElNSVQgMVxuICAgIFJFVFVSTiBwYDtcblxuICByZXR1cm4gZGIucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgIC50aGVuKChwb3N0OiBQb3N0VHlwZSA9IHt9KSA9PiB7XG4gICAgICBjb25zdCB7XG4gICAgICAgIF9rZXksXG4gICAgICAgIGdyb3VwSWQsXG4gICAgICAgIHByaXZhY3kgPSAnZGVmYXVsdCdcbiAgICAgIH06IFBvc3RUeXBlID0gcG9zdDtcblxuICAgICAgLy8gUXVlcnkgYmFzZWQgb24gcHJpdmFjeSBsZXZlbFxuICAgICAgbGV0IHByaXZhY3lBcWxRcnk6IHN0cmluZztcblxuICAgICAgaWYoZ3JvdXBJZCAmJiBwcml2YWN5ID09PSAnZ3JvdXAnKSB7XG4gICAgICAgIHByaXZhY3lBcWxRcnkgPSBgRk9SIHAgSU4gcG9zdHNcbiAgICAgICAgICBGSUxURVIgcC5fa2V5ID09IFwiJHtfa2V5fVwiXG4gICAgICAgICAgJHtzZWxlY3RRdWVyaWVzLmpvaW4oJ1xcbicpfVxuICAgICAgICAgIEZPUiBncm91cCBJTiBncm91cHNcbiAgICAgICAgICBGSUxURVIgZ3JvdXAuX2tleSA9PSBwLmdyb3VwSWRcbiAgICAgICAgICBGT1IgdSwgZSBJTiBPVVRCT1VORCBncm91cC5faWQgaXNHcm91cGVkXG4gICAgICAgICAgRklMVEVSIHUuX2tleSA9PSAke3Nlc3Npb25JZH1cbiAgICAgICAgICBMSU1JVCAxXG4gICAgICAgICAgUkVUVVJOIE1FUkdFKHAsIHske3NlbGVjdE9iamVjdHMuam9pbignLCAnKX19KWA7XG4gICAgICB9IGVsc2UgaWYocHJpdmFjeSA9PT0gJ3B1YmxpYycpIHtcbiAgICAgICAgcHJpdmFjeUFxbFFyeSA9IGBGT1IgcCBJTiBwb3N0c1xuICAgICAgICAgIEZJTFRFUiBwLl9rZXkgPT0gXCIke19rZXl9XCJcbiAgICAgICAgICAke3NlbGVjdFF1ZXJpZXMuam9pbignXFxuJyl9XG4gICAgICAgICAgTElNSVQgMVxuICAgICAgICAgIFJFVFVSTiBNRVJHRShwLCB7JHtzZWxlY3RPYmplY3RzLmpvaW4oJywgJyl9fSlgO1xuICAgICAgfVxuXG4gICAgICBpZihwcml2YWN5QXFsUXJ5KSB7XG4gICAgICAgIHJldHVybiBkYi5xdWVyeShwcml2YWN5QXFsUXJ5KVxuICAgICAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgICAgICAgIC50aGVuKChmaWx0ZXJlZFBvc3Q6IFBvc3RUeXBlID0ge30pID0+IGZpbHRlcmVkUG9zdClcbiAgICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgICAgICAgdGhyb3cgZXJyb3I7XG4gICAgICAgICAgfSk7XG4gICAgICB9XG5cbiAgICAgIHJldHVybiB7fTtcbiAgICB9KVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICB0aHJvdyBlcnJvcjtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRQb3N0TGlzdCA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBvcHRpb25zPzogUG9zdE9wdGlvbnMpOiBQcm9taXNlPFBvc3RUeXBlW10+ID0+IHtcbiAgLy8gY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnZ2V0TGlzdEJ5QXBwJztcbiAgY29uc3Qge2RhdGFiYXNlLCBmaWVsZHMsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG4gIGNvbnNvbGUubG9nKCdnZXRQb3N0TGlzdDo6Y29udGV4dCcsIGNvbnRleHQpO1xuICBjb25zdCB7bGltaXQsIHR5cGV9ID0gcGFyc2VQb3N0T3B0aW9ucyhvcHRpb25zKTtcbiAgY29uc3Qge29iamVjdHM6IHNlbGVjdE9iamVjdHMsIHF1ZXJpZXM6IHNlbGVjdFF1ZXJpZXN9ID0gZ2V0UG9zdE9wdGlvbmFsKGZpZWxkcywgc2Vzc2lvbklkKTtcbiAgY29uc3QgYXFsUXJ5OiBzdHJpbmcgPSBgRk9SIHAgSU4gcG9zdHNcbiAgICBGSUxURVIgcC50eXBlID09IFwiJHt0eXBlfVwiICYmIHAucHJpdmFjeSA9PSBcInB1YmxpY1wiICYmIHAucGFyZW50ID09IG51bGxcbiAgICAke3NlbGVjdFF1ZXJpZXMuam9pbignXFxuJyl9XG4gICAgJHtsaW1pdC5hcWx9XG4gICAgU09SVCBwLmFkZGVkXG4gICAgUkVUVVJOIERJU1RJTkNUIE1FUkdFKHAsIHske3NlbGVjdE9iamVjdHMuam9pbignLCAnKX19KWA7XG5cbiAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5hbGwoKSlcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgY29uc29sZS5sb2coJ2dldFBvc3RMaXN0OjplcnJvcicsIGVycm9yKTtcbiAgICAgIHRocm93IGVycm9yO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldFBvc3RzQnlBcmVhID0gKFxuICBjb250ZXh0OiBBcGlDb250ZXh0LFxuICBsYXRpdHVkZTogbnVtYmVyLFxuICBsb25naXR1ZGU6IG51bWJlcixcbiAgb3B0aW9ucz86IFBvc3RPcHRpb25zXG4pOiBQcm9taXNlPFBvc3RUeXBlW10+ID0+IHtcbiAgLy8gY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnZ2V0TGlzdEJ5VXNlcic7XG4gIGNvbnN0IHtkYXRhYmFzZSwgZmllbGRzLCB1c2VySWQ6IHNlc3Npb25JZH0gPSBjb250ZXh0O1xuICBjb25zdCB7bGltaXQsIHR5cGV9ID0gcGFyc2VQb3N0T3B0aW9ucyhvcHRpb25zKTtcbiAgY29uc3QgZm9ybWF0TGF0aXR1ZGU6IG51bWJlciA9IHBhcnNlTnVtKGxhdGl0dWRlKTtcbiAgY29uc3QgZm9ybWF0TG9uZ2l0dWRlOiBudW1iZXIgPSBwYXJzZU51bShsb25naXR1ZGUpO1xuICBjb25zdCB7b2JqZWN0czogc2VsZWN0T2JqZWN0cywgcXVlcmllczogc2VsZWN0UXVlcmllc30gPSBnZXRQb3N0T3B0aW9uYWwoZmllbGRzLCBzZXNzaW9uSWQpO1xuICBzZWxlY3RRdWVyaWVzLnB1c2goYExFVCBkaXN0YW5jZSA9IERJU1RBTkNFKFxuICAgICR7Zm9ybWF0TGF0aXR1ZGV9LFxuICAgICR7Zm9ybWF0TG9uZ2l0dWRlfSxcbiAgICBOT1RfTlVMTChwLmxhdGl0dWRlLCAwKSxcbiAgICBOT1RfTlVMTChwLmxvbmdpdHVkZSwgMCkpXG4gIGApO1xuICBzZWxlY3RPYmplY3RzLnB1c2goJ2Rpc3RhbmNlOmRpc3RhbmNlJyk7XG5cbiAgY29uc3QgYXFsUXJ5OiBzdHJpbmcgPSBgRk9SIHAgSU4gcG9zdHNcbiAgICAke3NlbGVjdFF1ZXJpZXMuam9pbignXFxuJyl9XG4gICAgRklMVEVSIHAudHlwZSA9PSBcIiR7dHlwZX1cIiAmJiBwLnByaXZhY3kgPT0gXCJwdWJsaWNcIiAmJiBwLnBhcmVudElkID09IG51bGxcbiAgICAke2xpbWl0LmFxbH1cbiAgICBTT1JUIGRpc3RhbmNlLCBwLmFkZGVkXG4gICAgUkVUVVJOIERJU1RJTkNUIE1FUkdFKHAsIHske3NlbGVjdE9iamVjdHMuam9pbignLCAnKX19KWA7XG5cbiAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5hbGwoKSlcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgdGhyb3cgZXJyb3I7XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgZ2V0UG9zdHNCeUdyb3VwID0gKFxuICBjb250ZXh0OiBBcGlDb250ZXh0LFxuICBncm91cElkOiBzdHJpbmcsXG4gIG9wdGlvbnM/OiBQb3N0T3B0aW9uc1xuKTogUHJvbWlzZTxQb3N0VHlwZVtdPiA9PiB7XG4gIC8vIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2dldExpc3RCeUdyb3VwJztcbiAgY29uc3Qge2RhdGFiYXNlLCBmaWVsZHMsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG4gIGNvbnN0IHtvYmplY3RzOiBzZWxlY3RPYmplY3RzLCBxdWVyaWVzOiBzZWxlY3RRdWVyaWVzfSA9IGdldFBvc3RPcHRpb25hbChmaWVsZHMsIHNlc3Npb25JZCk7XG5cbiAgLy8gR3JvdXAgaWRcbiAgY29uc3QgZm9ybWF0R3JvdXBJZDogc3RyaW5nID0gcGFyc2VJZChncm91cElkKTtcbiAgY29uc3QgZGIgPSB1c2VEYihkYXRhYmFzZSk7XG4gIGNvbnN0IGFxbFFyeTogc3RyaW5nID0gYEZPUiB1LCBnIElOIElOQk9VTkQgJHtmb3JtYXRHcm91cElkfSBoYXNHcm91cFxuICAgICAgRklMVEVSIHUuX2tleSA9PSAke3Nlc3Npb25JZH1cbiAgICAgIFJFVFVSTiBnYDtcblxuICByZXR1cm4gZGIucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IuYWxsKCkpXG4gICAgLnRoZW4oKGdyb3VwczogR3JvdXBUeXBlW10gPSBbXSkgPT4ge1xuICAgICAgaWYoZ3JvdXBzLmxlbmd0aCkge1xuICAgICAgICBjb25zdCB7bGltaXQsIHR5cGV9ID0gcGFyc2VQb3N0T3B0aW9ucyhvcHRpb25zKTtcbiAgICAgICAgY29uc3QgcG9zdEFxbFFyeTogc3RyaW5nID0gYEZPUiBwIElOIHBvc3RzXG4gICAgICAgICAgRklMVEVSIHAudHlwZSA9PSBcIiR7dHlwZX1cIiAmJiBwLmdyb3VwSWQgPT0gXCIke2Zvcm1hdEdyb3VwSWR9XCIgJiYgcC5wYXJlbnQgPT0gbnVsbFxuICAgICAgICAgICR7c2VsZWN0UXVlcmllcy5qb2luKCdcXG4nKX1cbiAgICAgICAgICAke2xpbWl0LmFxbH1cbiAgICAgICAgICBTT1JUIHAuYWRkZWRcbiAgICAgICAgICBSRVRVUk4gRElTVElOQ1QgTUVSR0UocCwgeyR7c2VsZWN0T2JqZWN0cy5qb2luKCcsICcpfX0pYDtcblxuICAgICAgICByZXR1cm4gZGIucXVlcnkocG9zdEFxbFFyeSlcbiAgICAgICAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLmFsbCgpKVxuICAgICAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICAgICAgICB0aHJvdyBlcnJvcjtcbiAgICAgICAgICB9KTtcbiAgICAgIH1cblxuICAgICAgcmV0dXJuIFtdO1xuICAgIH0pXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IHtcbiAgICAgIHRocm93IGVycm9yO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldFBvc3RzQnlMYXRlc3QgPSAoY29udGV4dDogQXBpQ29udGV4dCwgb3B0aW9ucz86IFBvc3RPcHRpb25zKTogUHJvbWlzZTxQb3N0VHlwZVtdPiA9PiB7XG4gIC8vIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2dldExpc3RCeUxhdGVzdCc7XG4gIGNvbnNvbGUubG9nKCdnZXRQb3N0c0J5TGF0ZXN0OjpvcHRpb25zJywgb3B0aW9ucyk7XG4gIGNvbnN0IHtkYXRhYmFzZSwgZmllbGRzLCB1c2VySWQ6IHNlc3Npb25JZH0gPSBjb250ZXh0O1xuICBjb25zdCB7bGltaXQsIHR5cGV9ID0gcGFyc2VQb3N0T3B0aW9ucyhvcHRpb25zKTtcbiAgY29uc3Qge29iamVjdHM6IHNlbGVjdE9iamVjdHMsIHF1ZXJpZXM6IHNlbGVjdFF1ZXJpZXN9ID0gZ2V0UG9zdE9wdGlvbmFsKGZpZWxkcywgc2Vzc2lvbklkKTtcbiAgY29uc3QgYXFsUXJ5OiBzdHJpbmcgPSBgRk9SIHAgSU4gcG9zdHNcbiAgICBGSUxURVIgcC50eXBlID09IFwiJHt0eXBlfVwiICYmIHAucHJpdmFjeSA9PSBcInB1YmxpY1wiICYmIHAucGFyZW50ID09IG51bGxcbiAgICAke3NlbGVjdFF1ZXJpZXMuam9pbignXFxuJyl9XG4gICAgJHtsaW1pdC5hcWx9XG4gICAgU09SVCBwLmFkZGVkIERFU0NcbiAgICBSRVRVUk4gRElTVElOQ1QgTUVSR0UocCwgeyR7c2VsZWN0T2JqZWN0cy5qb2luKCcsICcpfX0pYDtcblxuICBjb25zb2xlLmxvZygnZ2V0UG9zdHNCeUxhdGVzdDo6YXFsUXJ5JywgYXFsUXJ5KTtcbiAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5hbGwoKSlcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgdGhyb3cgZXJyb3I7XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgZ2V0UG9zdHNCeVJlYWN0aW9uID0gKFxuICBjb250ZXh0OiBBcGlDb250ZXh0LFxuICByZWFjdGlvbk5hbWU6IHN0cmluZyxcbiAgb3B0aW9ucz86IFBvc3RPcHRpb25zXG4pOiBQcm9taXNlPFBvc3RUeXBlW10+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnZ2V0UG9zdHNCeVJlYWN0aW9uJztcbiAgY29uc3Qge2RhdGFiYXNlLCBmaWVsZHMsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG4gIGNvbnN0IHtsaW1pdH0gPSBwYXJzZVBvc3RPcHRpb25zKG9wdGlvbnMpO1xuICBjb25zdCB7b2JqZWN0czogc2VsZWN0T2JqZWN0cywgcXVlcmllczogc2VsZWN0UXVlcmllc30gPSBnZXRQb3N0T3B0aW9uYWwoZmllbGRzLCBzZXNzaW9uSWQpO1xuICBjb25zdCBmb3JtYXRTZXNzaW9uSWQ6IHN0cmluZyA9IGB1c2Vycy8ke3Nlc3Npb25JZH1gO1xuICBjb25zdCBmb3JtYXRSZWFjdGlvbk5hbWU6IHN0cmluZyA9IHBhcnNlQ2hhcihyZWFjdGlvbk5hbWUsIDMyKTtcblxuICAvLyBHZXQgZGF0YSBmcm9tIGRhdGFiYXNlXG4gIGNvbnN0IGFxbFFyeTogc3RyaW5nID0gYEZPUiBwLCByIElOIE9VVEJPVU5EIFwiJHtmb3JtYXRTZXNzaW9uSWR9XCIgaGFzUmVhY3Rpb25zXG4gICAgJHtzZWxlY3RRdWVyaWVzLmpvaW4oJ1xcbicpfVxuICAgIEZJTFRFUiByLnR5cGUgPT0gXCJwb3N0c1wiICYmIHIubmFtZSA9PSBcIiR7Zm9ybWF0UmVhY3Rpb25OYW1lfVwiXG4gICAgJHtsaW1pdC5hcWx9XG4gICAgUkVUVVJOIE1FUkdFKHAsIHske3NlbGVjdE9iamVjdHMuam9pbignLCAnKX19KWA7XG5cbiAgcmV0dXJuIHVzZURiKGRhdGFiYXNlKS5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5hbGwoKSlcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgYWN0aW9uLFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICBsYWJlbDogJ2RiX2Vycm9yJ1xuICAgIH0sIGVycm9yLCBjb250ZXh0KS50aGVuKCgpID0+IFtdKSk7XG59O1xuXG5leHBvcnQgY29uc3QgZ2V0UG9zdHNCeVRhZ3MgPSAoXG4gIGNvbnRleHQ6IEFwaUNvbnRleHQsXG4gIHRhZ05hbWVzOiBzdHJpbmdbXSxcbiAgb3B0aW9ucz86IFBvc3RPcHRpb25zXG4pOiBQcm9taXNlPFBvc3RUeXBlW10+ID0+IHtcbiAgLy8gY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnZ2V0TGlzdEJ5VGFncyc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgZmllbGRzLCB1c2VySWQ6IHNlc3Npb25JZH0gPSBjb250ZXh0O1xuICBjb25zdCB7bGF0aXR1ZGUsIGxvbmdpdHVkZSwgbGltaXQsIHR5cGV9ID0gcGFyc2VQb3N0T3B0aW9ucyhvcHRpb25zKTtcbiAgY29uc3Qge29iamVjdHM6IHNlbGVjdE9iamVjdHMsIHF1ZXJpZXM6IHNlbGVjdFF1ZXJpZXN9ID0gZ2V0UG9zdE9wdGlvbmFsKGZpZWxkcywgc2Vzc2lvbklkKTtcbiAgY29uc3Qgc29ydEJ5OiBzdHJpbmdbXSA9IFtdO1xuXG4gIGlmKGxhdGl0dWRlICE9PSB1bmRlZmluZWQgJiYgbG9uZ2l0dWRlICE9PSB1bmRlZmluZWQpIHtcbiAgICBjb25zdCBmb3JtYXRMYXRpdHVkZTogbnVtYmVyID0gcGFyc2VOdW0obGF0aXR1ZGUpO1xuICAgIGNvbnN0IGZvcm1hdExvbmdpdHVkZTogbnVtYmVyID0gcGFyc2VOdW0obG9uZ2l0dWRlKTtcbiAgICBzZWxlY3RRdWVyaWVzLnB1c2goYExFVCBkaXN0YW5jZSA9IERJU1RBTkNFKFxuICAgICAgJHtmb3JtYXRMYXRpdHVkZX0sXG4gICAgICAke2Zvcm1hdExvbmdpdHVkZX0sXG4gICAgICBOT1RfTlVMTChwLmxhdGl0dWRlLCAwKSxcbiAgICAgIE5PVF9OVUxMKHAubG9uZ2l0dWRlLCAwKSlcbiAgICBgKTtcbiAgICBzZWxlY3RPYmplY3RzLnB1c2goJ2Rpc3RhbmNlOmRpc3RhbmNlJyk7XG4gICAgc29ydEJ5LnB1c2goJ2Rpc3RhbmNlJyk7XG4gIH1cblxuICBzb3J0QnkucHVzaCgncC5hZGRlZCcpO1xuXG4gIHJldHVybiBQcm9taXNlLmFsbChcbiAgICB0YWdOYW1lcy5tYXAoKHRhZ05hbWU6IHN0cmluZykgPT4ge1xuICAgICAgY29uc3QgYXFsUXJ5OiBzdHJpbmcgPSBgRk9SIHRhcmdldFRhZyBJTiB0YWdzXG4gICAgICAgIEZJTFRFUiBMT1dFUih0YXJnZXRUYWcubmFtZSkgPT0gXCIke3RhZ05hbWV9XCJcbiAgICAgICAgRk9SIHAsIGUgSU4gT1VUQk9VTkQgdGFyZ2V0VGFnLl9pZCBpc1RhZ2dlZFxuICAgICAgICAke3NlbGVjdFF1ZXJpZXMuam9pbignXFxuJyl9XG4gICAgICAgIEZJTFRFUiBwLnR5cGUgPT0gXCIke3R5cGV9XCIgJiYgcC5wcml2YWN5ID09IFwicHVibGljXCIgJiYgZS50eXBlID09ICdwb3N0cydcbiAgICAgICAgJHtsaW1pdC5hcWx9XG4gICAgICAgIFNPUlQgJHtzb3J0Qnkuam9pbignLCAnKX1cbiAgICAgICAgUkVUVVJOIERJU1RJTkNUIE1FUkdFKHAsIHske3NlbGVjdE9iamVjdHMuam9pbignLCAnKX19KWA7XG5cbiAgICAgIHJldHVybiB1c2VEYihkYXRhYmFzZSkucXVlcnkoYXFsUXJ5KVxuICAgICAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLmFsbCgpKVxuICAgICAgICAuY2F0Y2goKCkgPT4gW10pO1xuICAgIH0pKVxuICAgIC50aGVuKChyZXN1bHRzKSA9PiB1bmlxQnkoZmxhdHRlbihyZXN1bHRzKSwgJ19rZXknKSlcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgdGhyb3cgZXJyb3I7XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgZ2V0UG9zdHNCeVVzZXIgPSAoY29udGV4dDogQXBpQ29udGV4dCwgdXNlcklkOiBzdHJpbmcsIG9wdGlvbnM/OiBQb3N0T3B0aW9ucyk6IFByb21pc2U8UG9zdFR5cGVbXT4gPT4ge1xuICAvLyBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRMaXN0QnlVc2VyJztcbiAgY29uc3Qge2RhdGFiYXNlLCBmaWVsZHMsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG4gIGNvbnN0IHtsaW1pdCwgdHlwZX0gPSBwYXJzZVBvc3RPcHRpb25zKG9wdGlvbnMpO1xuICBjb25zdCBmb3JtYXRVc2VySWQ6IHN0cmluZyA9IHBhcnNlSWQodXNlcklkKTtcbiAgY29uc3Qge29iamVjdHM6IHNlbGVjdE9iamVjdHMsIHF1ZXJpZXM6IHNlbGVjdFF1ZXJpZXN9ID0gZ2V0UG9zdE9wdGlvbmFsKGZpZWxkcywgc2Vzc2lvbklkKTtcbiAgY29uc3QgYXFsUXJ5OiBzdHJpbmcgPSBgRk9SIHAgSU4gcG9zdHNcbiAgICBGSUxURVIgcC51c2VySWQgPT0gXCIke2Zvcm1hdFVzZXJJZH1cIiAmJiBwLnR5cGUgPT0gXCIke3R5cGV9XCIgJiYgcC5wcml2YWN5ID09IFwicHVibGljXCIgJiYgcC5wYXJlbnQgPT0gbnVsbFxuICAgICR7c2VsZWN0UXVlcmllcy5qb2luKCdcXG4nKX1cbiAgICAke2xpbWl0LmFxbH1cbiAgICBTT1JUIHAuYWRkZWRcbiAgICBSRVRVUk4gRElTVElOQ1QgTUVSR0UocCwgeyR7c2VsZWN0T2JqZWN0cy5qb2luKCcsICcpfX0pYDtcblxuICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLmFsbCgpKVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICB0aHJvdyBlcnJvcjtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRQb3N0Q29tbWVudHMgPSAoY29udGV4dDogQXBpQ29udGV4dCwgaXRlbUlkOiBzdHJpbmcsIG9wdGlvbnM/OiBQb3N0T3B0aW9ucyk6IFByb21pc2U8UG9zdFR5cGVbXT4gPT4ge1xuICAvLyBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRDb21tZW50cyc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgdXNlcklkOiBzZXNzaW9uSWR9ID0gY29udGV4dDtcbiAgY29uc3Qge2xpbWl0LCB0eXBlfSA9IHBhcnNlUG9zdE9wdGlvbnMob3B0aW9ucyk7XG4gIGNvbnN0IGZvcm1hdEl0ZW1JZDogc3RyaW5nID0gcGFyc2VJZChpdGVtSWQpO1xuXG4gIC8vIEdldCB0aGUgcGFyZW50IHBvc3QgdG8gZ2V0IHJlc3RyaWN0aW9uc1xuICBjb25zdCBkYiA9IHVzZURiKGRhdGFiYXNlKTtcbiAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBGT1IgcCBJTiBwb3N0c1xuICAgIEZJTFRFUiBwLnR5cGUgPT0gJHt0eXBlfSAmJiBwLl9rZXkgPT0gJHtmb3JtYXRJdGVtSWR9XG4gICAgTElNSVQgMVxuICAgIFJFVFVSTiBwYDtcblxuICByZXR1cm4gZGIucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgIC50aGVuKChwb3N0OiBQb3N0VHlwZSA9IHt9KSA9PiB7XG4gICAgICBjb25zdCB7XG4gICAgICAgIF9rZXksXG4gICAgICAgIGdyb3VwSWQsXG4gICAgICAgIHByaXZhY3kgPSAncHVibGljJ1xuICAgICAgfTogUG9zdFR5cGUgPSBwb3N0O1xuXG4gICAgICAvLyBRdWVyeSBiYXNlZCBvbiBwcml2YWN5IGxldmVsXG4gICAgICBsZXQgcHJpdmFjeUFxbFFyeTogc3RyaW5nO1xuXG4gICAgICBpZihncm91cElkICYmIHByaXZhY3kgPT09ICdncm91cCcpIHtcbiAgICAgICAgcHJpdmFjeUFxbFFyeSA9IGBGT1IgcCBJTiBwb3N0c1xuICAgICAgICAgIEZPUiB1c2VyIElOIHVzZXJzXG4gICAgICAgICAgRklMVEVSIHAucGFyZW50ID09IFwiJHtfa2V5fVwiICYmIHVzZXIuX2tleSA9PSBwLnVzZXJJZFxuICAgICAgICAgIExFVCByZWFjdGlvbnMgPSAoXG4gICAgICAgICAgICBGT1IgcG9zdCwgciBJTiBJTkJPVU5EIHAuX2lkIHJlYWN0aW9uc1xuICAgICAgICAgICAgQ09MTEVDVCByZWFjdGlvbk5hbWUgPSByLnZhbHVlIElOVE8gcmVhY3Rpb25JdGVtc1xuICAgICAgICAgICAgUkVUVVJOIHtuYW1lOiByZWFjdGlvbk5hbWUsIGNvdW50OiBMRU5HVEgocmVhY3Rpb25JdGVtc1sqXS5yLnZhbHVlKX1cbiAgICAgICAgICApXG4gICAgICAgICAgRk9SIGdyb3VwIElOIGdyb3Vwc1xuICAgICAgICAgIEZJTFRFUiBncm91cC5fa2V5ID09IHAuZ3JvdXBJZFxuICAgICAgICAgIEZPUiB1LCBlIElOIE9VVEJPVU5EIGdyb3VwLl9pZCBpc0dyb3VwZWRcbiAgICAgICAgICBGSUxURVIgdS5fa2V5ID09IFwiJHtzZXNzaW9uSWR9XCJcbiAgICAgICAgICBTT1JUIHAuYWRkZWRcbiAgICAgICAgICAke2xpbWl0LmFxbH1cbiAgICAgICAgICBSRVRVUk4gTUVSR0UocCwge3VzZXI6IHVzZXIsIHJlYWN0aW9uczogcmVhY3Rpb25zfSlgO1xuICAgICAgfSBlbHNlIGlmKHByaXZhY3kgPT09ICdwdWJsaWMnKSB7XG4gICAgICAgIHByaXZhY3lBcWxRcnkgPSBgRk9SIHAgSU4gcG9zdHNcbiAgICAgICAgICBGT1IgdXNlciBJTiB1c2Vyc1xuICAgICAgICAgIEZJTFRFUiBwLnBhcmVudCA9PSBcIiR7X2tleX1cIiAmJiB1c2VyLl9rZXkgPT0gcC51c2VySWRcbiAgICAgICAgICBMRVQgcmVhY3Rpb25zID0gKFxuICAgICAgICAgICAgRk9SIHBvc3QsIHIgSU4gSU5CT1VORCBwLl9pZCByZWFjdGlvbnNcbiAgICAgICAgICAgIENPTExFQ1QgcmVhY3Rpb25OYW1lID0gci52YWx1ZSBJTlRPIHJlYWN0aW9uSXRlbXNcbiAgICAgICAgICAgIFJFVFVSTiB7bmFtZTogcmVhY3Rpb25OYW1lLCBjb3VudDogTEVOR1RIKHJlYWN0aW9uSXRlbXNbKl0uci52YWx1ZSl9XG4gICAgICAgICAgKVxuICAgICAgICAgIFNPUlQgcC5hZGRlZFxuICAgICAgICAgICR7bGltaXQuYXFsfVxuICAgICAgICAgIFJFVFVSTiBNRVJHRShwLCB7dXNlcjogdXNlciwgcmVhY3Rpb25zOiByZWFjdGlvbnN9KWA7XG4gICAgICB9XG5cbiAgICAgIGlmKHByaXZhY3lBcWxRcnkpIHtcbiAgICAgICAgcmV0dXJuIGRiLnF1ZXJ5KHByaXZhY3lBcWxRcnkpXG4gICAgICAgICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5hbGwoKSlcbiAgICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgICAgICAgdGhyb3cgZXJyb3I7XG4gICAgICAgICAgfSk7XG4gICAgICB9XG5cbiAgICAgIHJldHVybiBbXTtcbiAgICB9KVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICB0aHJvdyBlcnJvcjtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBhZGRQb3N0ID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGl0ZW06IFBvc3RUeXBlKTogUHJvbWlzZTxQb3N0VHlwZT4gPT4ge1xuICAvLyBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdhZGQnO1xuICBjb25zdCB7ZGF0YWJhc2UsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG5cbiAgY29uc3Qge1xuICAgIGNvbnRlbnQgPSAnJyxcbiAgICBlbmREYXRlLFxuICAgIGdyb3VwSWQgPSAnJyxcbiAgICBsb2NhdGlvbixcbiAgICBsYXRpdHVkZSxcbiAgICBsb25naXR1ZGUsXG4gICAgbmFtZSA9ICcnLFxuICAgIHBhcmVudElkID0gbnVsbCxcbiAgICBwcml2YWN5ID0gJ3B1YmxpYycsXG4gICAgc3RhcnREYXRlLFxuICAgIHR5cGUgPSAnZGVmYXVsdCdcbiAgfTogUG9zdFR5cGUgPSBpdGVtO1xuXG4gIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcblxuICBjb25zdCBpbnNlcnQ6IFBvc3RUeXBlID0ge1xuICAgIF9rZXk6IGNyZWF0ZUhhc2goYHBvc3QtJHtzZXNzaW9uSWR9YCksXG4gICAgYWRkZWQ6IG5vdyxcbiAgICBjb250ZW50OiBwYXJzZVN0cmluZyhjb250ZW50LCBNQVhfQ09OVEVOVF9MRU5HVEgpLFxuICAgIGVuZERhdGU6IGVuZERhdGUgPyBwYXJzZU51bShlbmREYXRlLCAxMykgOiB1bmRlZmluZWQsXG4gICAgZ3JvdXBJZDogZ3JvdXBJZCA/IHBhcnNlSWQoZ3JvdXBJZCkgOiB1bmRlZmluZWQsXG4gICAgbGF0aXR1ZGU6IGxhdGl0dWRlICE9PSB1bmRlZmluZWQgPyBwYXJzZU51bShsYXRpdHVkZSkgOiB1bmRlZmluZWQsXG4gICAgbG9jYXRpb246IGxvY2F0aW9uID8gcGFyc2VTdHJpbmcobG9jYXRpb24sIDE2MCkgOiB1bmRlZmluZWQsXG4gICAgbG9uZ2l0dWRlOiBsb25naXR1ZGUgIT09IHVuZGVmaW5lZCA/IHBhcnNlTnVtKGxvbmdpdHVkZSkgOiB1bmRlZmluZWQsXG4gICAgbW9kaWZpZWQ6IG5vdyxcbiAgICBuYW1lOiBwYXJzZVN0cmluZyhuYW1lLCAxNjApLFxuICAgIHBhcmVudElkOiBwYXJlbnRJZCA/IHBhcnNlSWQocGFyZW50SWQpIDogdW5kZWZpbmVkLFxuICAgIHByaXZhY3k6IHByaXZhY3kgPyBwYXJzZVZhckNoYXIocHJpdmFjeSwgMTYpIDogdW5kZWZpbmVkLFxuICAgIHN0YXJ0RGF0ZTogc3RhcnREYXRlID8gcGFyc2VOdW0oc3RhcnREYXRlLCAxMykgOiB1bmRlZmluZWQsXG4gICAgdHlwZTogcGFyc2VDaGFyKHR5cGUsIDMyKSxcbiAgICB1c2VySWQ6IHNlc3Npb25JZFxuICB9O1xuXG4gIGNvbnN0IGRiOiBEYXRhYmFzZSA9IHVzZURiKGRhdGFiYXNlKTtcbiAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBJTlNFUlQgJHtpbnNlcnR9IElOIHBvc3RzIFJFVFVSTiBORVdgO1xuXG4gIHJldHVybiBkYi5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgLnRoZW4oKHBvc3Q6IFBvc3RUeXBlID0ge30pID0+IHtcbiAgICAgIGNvbnN0IHtfa2V5OiBwb3N0S2V5fSA9IHBvc3Q7XG5cbiAgICAgIC8vIFVwZGF0ZSBsaW5rZWQgdGFncyB3aXRoaW4gcG9zdHNcbiAgICAgIHJldHVybiBleHRyYWN0VGFncyhkYiwgJ3Bvc3RzJywgcG9zdEtleSwgaW5zZXJ0LmNvbnRlbnQpXG4gICAgICAgIC50aGVuKCh0YWdMaXN0OiBUYWdUeXBlW10pID0+IHtcbiAgICAgICAgICBwb3N0LnRhZ3MgPSB0YWdMaXN0O1xuICAgICAgICAgIHJldHVybiBwb3N0O1xuICAgICAgICB9KTtcbiAgICB9KVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICB0aHJvdyBlcnJvcjtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCB1cGRhdGVQb3N0ID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGl0ZW06IFBvc3RUeXBlKTogUHJvbWlzZTxQb3N0VHlwZT4gPT4ge1xuICAvLyBjb25zdCBhY3Rpb246IHN0cmluZyA9ICd1cGRhdGUnO1xuICBjb25zdCB7ZGF0YWJhc2UsIHVzZXJJZDogc2Vzc2lvbklkfSA9IGNvbnRleHQ7XG4gIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcbiAgY29uc3Qge1xuICAgIGNvbnRlbnQsXG4gICAgZW5kRGF0ZSxcbiAgICBncm91cElkLFxuICAgIG5hbWUsXG4gICAgcGFyZW50SWQsXG4gICAgcG9zdElkLFxuICAgIHByaXZhY3ksXG4gICAgc3RhcnREYXRlLFxuICAgIHR5cGVcbiAgfTogUG9zdFR5cGUgPSBpdGVtO1xuXG4gIGNvbnN0IHVwZGF0ZTogUG9zdFR5cGUgPSB7XG4gICAgY29udGVudDogY29udGVudCA/IHBhcnNlU3RyaW5nKGNvbnRlbnQsIE1BWF9DT05URU5UX0xFTkdUSCkgOiB1bmRlZmluZWQsXG4gICAgZW5kRGF0ZTogZW5kRGF0ZSA/IHBhcnNlTnVtKGVuZERhdGUsIDEzKSA6IHVuZGVmaW5lZCxcbiAgICBtb2RpZmllZDogbm93LFxuICAgIG5hbWU6IG5hbWUgPyBwYXJzZVN0cmluZyhuYW1lLCAxNjApIDogdW5kZWZpbmVkLFxuICAgIHBhcmVudElkOiBwYXJlbnRJZCA/IHBhcnNlU3RyaW5nKHBhcmVudElkLCAxNjApIDogdW5kZWZpbmVkLFxuICAgIHByaXZhY3k6IHByaXZhY3kgPyBwYXJzZVZhckNoYXIocHJpdmFjeSwgMTYpIDogdW5kZWZpbmVkLFxuICAgIHN0YXJ0RGF0ZTogc3RhcnREYXRlID8gcGFyc2VOdW0oc3RhcnREYXRlLCAxMykgOiB1bmRlZmluZWQsXG4gICAgdHlwZTogdHlwZSAhPT0gdW5kZWZpbmVkID8gcGFyc2VDaGFyKHR5cGUsIDE2KSA6IHVuZGVmaW5lZFxuICB9O1xuXG4gIGxldCBmb3JtYXRJZDogc3RyaW5nID0gcGFyc2VJZChwb3N0SWQpO1xuICBmb3JtYXRJZCA9IGZvcm1hdElkID09PSAnJyA/IGNyZWF0ZUhhc2goYHBvc3QtJHtzZXNzaW9uSWR9YCkgOiBmb3JtYXRJZDtcbiAgY29uc3QgZm9ybWF0R3JvdXBJZDogc3RyaW5nID0gcGFyc2VJZChncm91cElkKTtcbiAgY29uc3QgaW5zZXJ0OiBhbnkgPSB7XG4gICAgLi4udXBkYXRlLFxuICAgIF9rZXk6IGZvcm1hdElkLFxuICAgIGFkZGVkOiBub3csXG4gICAgZ3JvdXBJZDogZm9ybWF0R3JvdXBJZCxcbiAgICBwcml2YWN5LFxuICAgIHVzZXJJZDogc2Vzc2lvbklkXG4gIH07XG4gIGNvbnN0IGRiOiBEYXRhYmFzZSA9IHVzZURiKGRhdGFiYXNlKTtcbiAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBVUFNFUlQge19rZXk6ICR7Zm9ybWF0SWR9LCB1c2VySWQ6ICR7c2Vzc2lvbklkfX1cbiAgICBJTlNFUlQgJHtpbnNlcnR9XG4gICAgVVBEQVRFICR7dXBkYXRlfVxuICAgIElOIHBvc3RzIFJFVFVSTiBORVdgO1xuXG4gIHJldHVybiBkYi5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgLnRoZW4oKHVwZGF0ZWRQb3N0OiBQb3N0VHlwZSA9IHt9KSA9PiB7XG4gICAgICBjb25zdCB7X2tleTogdXBkYXRlZFBvc3RLZXl9ID0gdXBkYXRlZFBvc3Q7XG5cbiAgICAgIC8vIFVwZGF0ZSBsaW5rZWQgdGFnc1xuICAgICAgcmV0dXJuIGV4dHJhY3RUYWdzKGRiLCAncG9zdHMnLCB1cGRhdGVkUG9zdEtleSwgdXBkYXRlLmNvbnRlbnQgfHwgJycpXG4gICAgICAgIC50aGVuKCh0YWdMaXN0ID0gW10pID0+IHtcbiAgICAgICAgICB1cGRhdGVkUG9zdC50YWdzID0gdGFnTGlzdDtcblxuICAgICAgICAgIC8vIFVwZGF0ZSBsaW5rZWQgZmlsZXNcbiAgICAgICAgICBjb25zdCBmaWxlczogRmlsZVR5cGVbXSA9IHVwZGF0ZWRQb3N0LmZpbGVzIHx8IFtdO1xuXG4gICAgICAgICAgaWYoZmlsZXMubGVuZ3RoKSB7XG4gICAgICAgICAgICByZXR1cm4gdXBkYXRlRmlsZXMoZGIsIGZvcm1hdElkLCBmaWxlcylcbiAgICAgICAgICAgICAgLnRoZW4oKGZpbGVMaXN0ID0gW10pID0+IHtcbiAgICAgICAgICAgICAgICB1cGRhdGVkUG9zdC5maWxlcyA9IGZpbGVMaXN0O1xuICAgICAgICAgICAgICAgIHJldHVybiB1cGRhdGVkUG9zdDtcbiAgICAgICAgICAgICAgfSk7XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgdXBkYXRlZFBvc3QuZmlsZXMgPSBbXTtcbiAgICAgICAgICByZXR1cm4gdXBkYXRlZFBvc3Q7XG4gICAgICAgIH0pO1xuICAgIH0pXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IHtcbiAgICAgIHRocm93IGVycm9yO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGRlbGV0ZVBvc3QgPSAoY29udGV4dDogQXBpQ29udGV4dCwgaXRlbUlkOiBzdHJpbmcpOiBQcm9taXNlPFBvc3RUeXBlPiA9PiB7XG4gIC8vIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2RlbGV0ZSc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgdXNlcklkOiBzZXNzaW9uSWR9ID0gY29udGV4dDtcbiAgY29uc3QgZm9ybWF0SXRlbUlkOiBzdHJpbmcgPSBwYXJzZUlkKGl0ZW1JZCk7XG4gIGNvbnN0IGRiOiBEYXRhYmFzZSA9IHVzZURiKGRhdGFiYXNlKTtcbiAgY29uc3QgYXFsUXJ5ID0gYXFsYEZPUiBwIElOIHBvc3RzXG4gICAgICBGSUxURVIgcC5fa2V5ID09ICR7Zm9ybWF0SXRlbUlkfSAmJiBwLnVzZXJJZCA9PSAke3Nlc3Npb25JZH1cbiAgICAgIExJTUlUIDFcbiAgICAgIFJFTU9WRSBwIElOIHBvc3RzXG4gICAgICBSRVRVUk4gT0xEYDtcblxuICByZXR1cm4gZGIucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgIC50aGVuKChwb3N0OiBQb3N0VHlwZSA9IHt9KSA9PiB7XG4gICAgICBpZihwb3N0KSB7XG4gICAgICAgIC8vIFJlbW92ZSB0YWcgbGlua3NcbiAgICAgICAgY29uc3QgZWRnZUFxbFFyeTogQXFsUXVlcnkgPSBhcWxgRk9SIHQgSU4gaXNUYWdnZWRcbiAgICAgICAgICAgIEZJTFRFUiB0Ll90byA9PSAke2Zvcm1hdEl0ZW1JZH1cbiAgICAgICAgICAgIFJFTU9WRSB0IElOIGlzVGFnZ2VkYDtcblxuICAgICAgICByZXR1cm4gZGIucXVlcnkoZWRnZUFxbFFyeSlcbiAgICAgICAgICAudGhlbigoKSA9PiB7XG4gICAgICAgICAgICAvLyBSZW1vdmUgYXR0YWNoZWQgZmlsZXNcbiAgICAgICAgICAgIGNvbnN0IGZpbGVBcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYEZPUiBmIElOIGhhc0ZpbGVcbiAgICAgICAgICAgICAgICBGSUxURVIgZi5fdG8gPT0gJHtmb3JtYXRJdGVtSWR9XG4gICAgICAgICAgICAgICAgUkVNT1ZFIGYgSU4gaGFzRmlsZWA7XG5cbiAgICAgICAgICAgIHJldHVybiBkYi5xdWVyeShmaWxlQXFsUXJ5KVxuICAgICAgICAgICAgICAudGhlbigoKSA9PiBwb3N0KVxuICAgICAgICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgICAgICAgICAgIHRocm93IGVycm9yO1xuICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICB9KVxuICAgICAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICAgICAgICB0aHJvdyBlcnJvcjtcbiAgICAgICAgICB9KTtcbiAgICAgIH1cbiAgICAgIHJldHVybiB7fTtcbiAgICB9KVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICB0aHJvdyBlcnJvcjtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBjbGVhblBvc3RzID0gKGRhdGFiYXNlOiBzdHJpbmcpOiBQcm9taXNlPG51bWJlcj4gPT4ge1xuICAvLyBSZW1vdmUgYWxsIG1lc3NhZ2VzIHRoYXQgYXJlIG92ZXIgNjAgZGF5cyBhbmQgbm90IHNhdmVkXG4gIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgRk9SIHAgSU4gcG9zdHNcbiAgICAgIEZJTFRFUiBwLmFkZGVkIDwgREFURV9USU1FU1RBTVAoREFURV9TVUJUUkFDVChEQVRFX05PVygpLCA2MCwgJ2RheScpKSAmJiBwLnR5cGUgPT0gXCJkZWZhdWx0XCJcbiAgICAgIFJFTU9WRSBwIElOIHBvc3RzXG4gICAgICBSRVRVUk4gT0xEYDtcblxuICByZXR1cm4gdXNlRGIoZGF0YWJhc2UpLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLmFsbCgpKVxuICAgIC50aGVuKChyZXN1bHRzOiBQb3N0VHlwZVtdID0gW10pID0+IHJlc3VsdHMubGVuZ3RoKVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICB0aHJvdyBlcnJvcjtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBjcmVhdGVQb3N0RWRnZSA9IChkYjogRGF0YWJhc2UsIGZpbGU6IEZpbGVUeXBlLCBwb3N0SWQ6IHN0cmluZyk6IFByb21pc2U8RmlsZVR5cGU+ID0+IHtcbiAgY29uc3QgZWRnZUNvbGxlY3Rpb246IEVkZ2VDb2xsZWN0aW9uID0gZGIuY29sbGVjdGlvbignaXNQb3N0ZWQnKTtcbiAgY29uc3Qge2ZpbGVJZH0gPSBmaWxlO1xuICBjb25zdCBmb3JtYXRGaWxlSWQ6IHN0cmluZyA9IHBhcnNlSWQoZmlsZUlkKTtcbiAgY29uc3QgZWRnZUlkOiBzdHJpbmcgPSBjcmVhdGVIYXNoKGBmaWxlLSR7cG9zdElkfS0ke2Zvcm1hdEZpbGVJZH1gKTtcbiAgY29uc3QgZm9ybWF0UG9zdElkOiBzdHJpbmcgPSBwYXJzZUlkKHBvc3RJZCk7XG4gIGNvbnN0IGZpbGVUeXBlOiBzdHJpbmcgPSBwYXJzZUNoYXIoZmlsZS5maWxlVHlwZSwgMTYpO1xuXG4gIGNvbnN0IGVkZ2U6IGFueSA9IHtcbiAgICBfZnJvbTogYHBvc3RzLyR7Zm9ybWF0UG9zdElkfWAsXG4gICAgX2tleTogZWRnZUlkLFxuICAgIF90bzogYGZpbGVzLyR7Zm9ybWF0RmlsZUlkfWAsXG4gICAgYWRkZWQ6IERhdGUubm93KCksXG4gICAgdHlwZTogZmlsZVR5cGVcbiAgfTtcblxuICByZXR1cm4gZWRnZUNvbGxlY3Rpb24uc2F2ZShlZGdlLCB7cmV0dXJuTmV3OiB0cnVlfSlcbiAgICAudGhlbigoKSA9PiBmaWxlKVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICB0aHJvdyBlcnJvcjtcbiAgICB9KTtcbn07XG4iXX0=