@nlabs/reaktor 0.8.0 → 0.9.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (198) hide show
  1. package/.eslintrc +10 -0
  2. package/.prettierrc.js +4 -0
  3. package/README.md +1 -1
  4. package/coverage/actions/groups.ts.html +1039 -0
  5. package/coverage/actions/images.ts.html +2500 -0
  6. package/coverage/actions/index.html +131 -0
  7. package/coverage/actions/tags.ts.html +1000 -0
  8. package/coverage/adapters/arangoAdapter.ts.html +151 -0
  9. package/coverage/adapters/index.html +146 -0
  10. package/coverage/adapters/reaktorAdapter.ts.html +127 -0
  11. package/coverage/adapters/tagAdapter.ts.html +160 -0
  12. package/coverage/base.css +224 -0
  13. package/coverage/block-navigation.js +87 -0
  14. package/coverage/clover.xml +6 -0
  15. package/coverage/coverage-final.json +1 -0
  16. package/coverage/favicon.png +0 -0
  17. package/coverage/index.html +176 -0
  18. package/coverage/lcov-report/base.css +224 -0
  19. package/coverage/lcov-report/block-navigation.js +87 -0
  20. package/coverage/lcov-report/favicon.png +0 -0
  21. package/coverage/lcov-report/index.html +101 -0
  22. package/coverage/lcov-report/prettify.css +1 -0
  23. package/coverage/lcov-report/prettify.js +2 -0
  24. package/coverage/lcov-report/sort-arrow-sprite.png +0 -0
  25. package/coverage/lcov-report/sorter.js +196 -0
  26. package/coverage/lcov.info +0 -0
  27. package/coverage/mocks/file.ts.html +118 -0
  28. package/coverage/mocks/group.ts.html +145 -0
  29. package/coverage/mocks/image.ts.html +136 -0
  30. package/coverage/mocks/index.html +146 -0
  31. package/coverage/mocks/post.ts.html +169 -0
  32. package/coverage/mocks/tag.ts.html +121 -0
  33. package/coverage/mocks/user.ts.html +268 -0
  34. package/coverage/prettify.css +1 -0
  35. package/coverage/prettify.js +2 -0
  36. package/coverage/sort-arrow-sprite.png +0 -0
  37. package/coverage/sorter.js +196 -0
  38. package/coverage/types/error.ts.html +145 -0
  39. package/coverage/types/index.html +116 -0
  40. package/coverage/utils/adapterUtils.ts.html +151 -0
  41. package/coverage/utils/analyticsUtils.ts.html +292 -0
  42. package/coverage/utils/arangodbUtils.ts.html +463 -0
  43. package/coverage/utils/index.html +146 -0
  44. package/jest.setup.js +0 -0
  45. package/jpg:- +0 -0
  46. package/lib/actions/apps.d.ts +25 -0
  47. package/lib/actions/apps.js +252 -0
  48. package/lib/actions/conversations.d.ts +14 -0
  49. package/lib/actions/conversations.js +131 -126
  50. package/lib/actions/dynamodb.d.ts +8 -0
  51. package/lib/actions/dynamodb.js +35 -32
  52. package/lib/actions/email.d.ts +5 -0
  53. package/lib/actions/email.js +23 -53
  54. package/lib/actions/files.d.ts +19 -0
  55. package/lib/actions/files.js +188 -202
  56. package/lib/actions/groups.d.ts +14 -0
  57. package/lib/actions/groups.js +38 -36
  58. package/lib/actions/images.d.ts +26 -0
  59. package/lib/actions/images.js +236 -229
  60. package/lib/actions/index.d.ts +21 -0
  61. package/lib/actions/index.js +3 -1
  62. package/lib/actions/ios.d.ts +7 -0
  63. package/lib/actions/ios.js +11 -10
  64. package/lib/actions/locations.d.ts +5 -0
  65. package/lib/actions/locations.js +29 -37
  66. package/lib/actions/messages.d.ts +13 -0
  67. package/lib/actions/messages.js +10 -10
  68. package/lib/actions/notifications.d.ts +5 -0
  69. package/lib/actions/notifications.js +1 -1
  70. package/lib/actions/payments.d.ts +10 -0
  71. package/lib/actions/payments.js +27 -26
  72. package/lib/actions/posts.d.ts +19 -0
  73. package/lib/actions/posts.js +176 -156
  74. package/lib/actions/reactions.d.ts +30 -0
  75. package/lib/actions/reactions.js +30 -28
  76. package/lib/actions/s3.d.ts +7 -0
  77. package/lib/actions/s3.js +37 -32
  78. package/lib/actions/search.d.ts +3 -0
  79. package/lib/actions/search.js +11 -9
  80. package/lib/actions/sms.d.ts +3 -0
  81. package/lib/actions/sms.js +58 -34
  82. package/lib/actions/statistics.d.ts +3 -0
  83. package/lib/actions/statistics.js +21 -18
  84. package/lib/actions/subscription.d.ts +7 -0
  85. package/lib/actions/subscription.js +24 -21
  86. package/lib/actions/tags.d.ts +29 -0
  87. package/lib/actions/tags.js +129 -198
  88. package/lib/actions/users.d.ts +47 -0
  89. package/lib/actions/users.js +188 -194
  90. package/lib/actions/websockets.d.ts +19 -0
  91. package/lib/actions/websockets.js +60 -34
  92. package/lib/adapters/arangoAdapter.d.ts +2 -0
  93. package/lib/adapters/arangoAdapter.js +46 -0
  94. package/lib/adapters/fileAdapter.d.ts +3 -0
  95. package/lib/adapters/fileAdapter.js +76 -0
  96. package/lib/adapters/postAdapter.d.ts +2 -0
  97. package/lib/adapters/postAdapter.js +70 -0
  98. package/lib/adapters/reaktorAdapter.d.ts +6 -0
  99. package/lib/adapters/reaktorAdapter.js +44 -0
  100. package/lib/adapters/tagAdapter.d.ts +2 -0
  101. package/lib/adapters/tagAdapter.js +50 -0
  102. package/lib/adapters/userAdapter.d.ts +2 -0
  103. package/lib/adapters/userAdapter.js +110 -0
  104. package/lib/config.d.ts +20 -0
  105. package/lib/config.js +14 -15
  106. package/lib/index.d.ts +5 -0
  107. package/lib/lambdas/actions/websockets.d.ts +6 -0
  108. package/lib/lambdas/actions/websockets.js +7 -7
  109. package/lib/lambdas/authorizer.d.ts +20 -0
  110. package/lib/lambdas/authorizer.js +1 -1
  111. package/lib/lambdas/connection.d.ts +12 -0
  112. package/lib/lambdas/connection.js +5 -4
  113. package/lib/lambdas/utils/message.d.ts +1 -0
  114. package/lib/lambdas/utils/websocket.d.ts +7 -0
  115. package/lib/lambdas/utils/websocket.js +8 -6
  116. package/lib/mocks/conversation.d.ts +8 -0
  117. package/lib/mocks/conversation.js +35 -0
  118. package/lib/mocks/file.d.ts +11 -0
  119. package/lib/mocks/file.js +38 -0
  120. package/lib/mocks/group.d.ts +17 -0
  121. package/lib/mocks/group.js +47 -0
  122. package/lib/mocks/image.d.ts +3 -0
  123. package/lib/mocks/image.js +43 -0
  124. package/lib/mocks/nlabs.png +0 -0
  125. package/lib/mocks/post.d.ts +38 -0
  126. package/lib/mocks/post.js +55 -0
  127. package/lib/mocks/tag.d.ts +2 -0
  128. package/lib/mocks/tag.js +37 -0
  129. package/lib/mocks/user.d.ts +4 -0
  130. package/lib/mocks/user.js +88 -0
  131. package/lib/templates/email/layout.d.ts +2 -0
  132. package/lib/templates/email/passwordForgot.d.ts +2 -0
  133. package/lib/templates/email/passwordRecovery.d.ts +2 -0
  134. package/lib/templates/email/verifyEmail.d.ts +2 -0
  135. package/lib/templates/email/welcome.d.ts +2 -0
  136. package/lib/templates/sms/passwordForgot.d.ts +2 -0
  137. package/lib/templates/sms/passwordRecovery.d.ts +2 -0
  138. package/lib/templates/sms/verifyEmail.d.ts +2 -0
  139. package/lib/templates/sms/verifyPhone.d.ts +2 -0
  140. package/lib/templates/sms/welcome.d.ts +2 -0
  141. package/lib/types/apps.d.ts +46 -0
  142. package/lib/types/apps.js +17 -1
  143. package/lib/types/arangodb.d.ts +30 -0
  144. package/lib/types/arangodb.js +1 -1
  145. package/lib/types/auth.d.ts +7 -0
  146. package/lib/types/auth.js +1 -1
  147. package/lib/types/connections.d.ts +7 -0
  148. package/lib/types/connections.js +1 -1
  149. package/lib/types/conversations.d.ts +29 -0
  150. package/lib/types/conversations.js +1 -1
  151. package/lib/types/email.d.ts +13 -0
  152. package/lib/types/email.js +1 -1
  153. package/lib/types/error.d.ts +20 -0
  154. package/lib/types/error.js +44 -0
  155. package/lib/types/files.d.ts +26 -0
  156. package/lib/types/files.js +1 -1
  157. package/lib/types/google.d.ts +29 -0
  158. package/lib/types/google.js +1 -1
  159. package/lib/types/groups.d.ts +21 -0
  160. package/lib/types/groups.js +1 -1
  161. package/lib/types/images.d.ts +51 -0
  162. package/lib/types/images.js +1 -1
  163. package/lib/types/index.d.ts +18 -0
  164. package/lib/types/locations.d.ts +20 -0
  165. package/lib/types/locations.js +1 -1
  166. package/lib/types/messages.d.ts +16 -0
  167. package/lib/types/messages.js +1 -1
  168. package/lib/types/notifications.d.ts +17 -0
  169. package/lib/types/notifications.js +1 -1
  170. package/lib/types/payments.d.ts +112 -0
  171. package/lib/types/payments.js +1 -1
  172. package/lib/types/posts.d.ts +31 -0
  173. package/lib/types/posts.js +1 -1
  174. package/lib/types/statistics.d.ts +3 -0
  175. package/lib/types/statistics.js +1 -1
  176. package/lib/types/tags.d.ts +10 -0
  177. package/lib/types/tags.js +1 -1
  178. package/lib/types/users.d.ts +76 -0
  179. package/lib/types/users.js +1 -1
  180. package/lib/types/websocket.d.ts +13 -0
  181. package/lib/types/websocket.js +1 -1
  182. package/lib/utils/adapterUtils.d.ts +1 -0
  183. package/lib/utils/adapterUtils.js +45 -0
  184. package/lib/utils/analyticsUtils.d.ts +21 -0
  185. package/lib/utils/analyticsUtils.js +72 -0
  186. package/lib/utils/arangodbUtils.d.ts +65 -0
  187. package/lib/utils/arangodbUtils.js +144 -0
  188. package/lib/utils/auth.d.ts +20 -0
  189. package/lib/utils/auth.js +13 -30
  190. package/lib/utils/index.d.ts +5 -0
  191. package/lib/utils/index.js +7 -9
  192. package/lib/utils/session.d.ts +16 -0
  193. package/lib/utils/session.js +11 -2
  194. package/package.json +12 -5
  195. package/lib/utils/analytics.js +0 -88
  196. package/lib/utils/arangodb.js +0 -118
  197. package/lib/utils/graphql.js +0 -46
  198. package/lib/utils/objects.js +0 -59
@@ -48,15 +48,16 @@ module.exports = __toCommonJS(images_exports);
48
48
  var import_rip_hunter = require("@nlabs/rip-hunter");
49
49
  var import_utils = require("@nlabs/utils");
50
50
  var import_arangojs = require("arangojs");
51
- var import_file_type = __toESM(require("file-type"));
52
51
  var import_gm = __toESM(require("gm"));
53
- var import_cloneDeep = __toESM(require("lodash/cloneDeep"));
54
52
  var import_luxon = require("luxon");
55
- var import_config = require("../config");
56
- var import_utils2 = require("../utils");
57
53
  var import_groups = require("./groups");
58
- var import_s32 = require("./s3");
54
+ var import_s3 = require("./s3");
55
+ var import_config = require("../config");
56
+ var import_analyticsUtils = require("../utils/analyticsUtils");
57
+ var import_error = require("../types/error");
58
+ var import_arangodbUtils = require("../utils/arangodbUtils");
59
59
  const eventCategory = "images";
60
+ import_gm.default.subClass({ imageMagick: "7+" });
60
61
  const parseImageOptions = (options = {}) => {
61
62
  const {
62
63
  from = 0,
@@ -64,13 +65,13 @@ const parseImageOptions = (options = {}) => {
64
65
  type = "default"
65
66
  } = options;
66
67
  return {
67
- limit: (0, import_utils2.getLimit)(from, to),
68
+ limit: (0, import_arangodbUtils.getLimit)(from, to),
68
69
  type: (0, import_utils.parseChar)(type, 32)
69
70
  };
70
71
  };
71
72
  const getImageOptional = (fields = []) => fields.reduce((selects, field) => {
72
73
  if (field.includes("Count")) {
73
- return (0, import_utils2.selectReactionCountByType)("images", "i", field, selects);
74
+ return (0, import_arangodbUtils.selectReactionCountByType)("images", "i", field, selects);
74
75
  }
75
76
  switch (field) {
76
77
  case "reactions": {
@@ -90,14 +91,14 @@ const getImageOptional = (fields = []) => fields.reduce((selects, field) => {
90
91
  selects.objects.push("tags:tags");
91
92
  return selects;
92
93
  }
93
- case "user": {
94
- selects.queries.push(`LET user = FIRST(
94
+ case "users": {
95
+ selects.queries.push(`LET users = FIRST(
95
96
  FOR u IN users
96
97
  FILTER i.userId == u._key
97
98
  LIMIT 1
98
99
  RETURN u
99
100
  )`);
100
- selects.objects.push("user:user");
101
+ selects.objects.push("users:users");
101
102
  return selects;
102
103
  }
103
104
  default: {
@@ -105,11 +106,11 @@ const getImageOptional = (fields = []) => fields.reduce((selects, field) => {
105
106
  }
106
107
  }
107
108
  }, { objects: [], queries: [] });
108
- const getImagesByUser = (context, userId, from, to) => {
109
+ const getImagesByUser = (context, userId, from = 0, to = 30) => {
109
110
  const action = "getImagesByUser";
110
111
  const { database } = context;
111
112
  const formatUserId = (0, import_utils.parseId)(userId);
112
- const limit = (0, import_utils2.getLimit)(from, to);
113
+ const limit = (0, import_arangodbUtils.getLimit)(from, to);
113
114
  const aqlQry = `FOR i IN images
114
115
  FILTER i.userId == "${formatUserId}"
115
116
  LET user = FIRST(
@@ -121,11 +122,12 @@ const getImagesByUser = (context, userId, from, to) => {
121
122
  ${limit.aql}
122
123
  SORT i.added
123
124
  RETURN MERGE(i, {user:user})`;
124
- return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_utils2.logError)({
125
+ return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_analyticsUtils.logError)({
125
126
  action,
126
127
  category: eventCategory,
127
- label: "db_error"
128
- }, error, context).then(() => null));
128
+ value: import_error.ErrorTypes.DATABASE_ERROR,
129
+ params: { from, to, userId }
130
+ }, error, context));
129
131
  };
130
132
  const getImageCountByItem = (context, itemId) => {
131
133
  const action = "getImageCountByItem";
@@ -134,11 +136,12 @@ const getImageCountByItem = (context, itemId) => {
134
136
  const aqlQry = import_arangojs.aql`FOR i IN hasImage
135
137
  FILTER i._to == ${formatItemId}
136
138
  RETURN {count: COUNT(i)}`;
137
- return database.query(aqlQry).then((cursor) => cursor.next()).then(({ count } = { count: 0 }) => count).catch((error) => (0, import_utils2.logError)({
139
+ return database.query(aqlQry).then((cursor) => cursor.next()).then(({ count }) => count).catch((error) => (0, import_analyticsUtils.logError)({
138
140
  action,
139
141
  category: eventCategory,
140
- label: "db_error"
141
- }, error, context).then(() => 0));
142
+ value: import_error.ErrorTypes.DATABASE_ERROR,
143
+ params: { itemId }
144
+ }, error, context));
142
145
  };
143
146
  const getImagesByItem = async (context, itemId, options = {}) => {
144
147
  const action = "getImagesByItem";
@@ -152,18 +155,19 @@ const getImagesByItem = async (context, itemId, options = {}) => {
152
155
  SORT i.added
153
156
  ${limit.aql}
154
157
  RETURN MERGE(i, {${selectObjects.join(", ")}})`;
155
- return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_utils2.logError)({
158
+ return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_analyticsUtils.logError)({
156
159
  action,
157
160
  category: eventCategory,
158
- label: "db_error"
159
- }, error, context).then(() => null));
161
+ value: import_error.ErrorTypes.DATABASE_ERROR,
162
+ params: { itemId, options }
163
+ }, error, context));
160
164
  };
161
165
  const getImagesByGroup = (context, params) => {
162
166
  const action = "getImagesByGroup";
163
167
  const { database, session: { userId: sessionId } } = context;
164
- const { filters = [], groupId, from, to } = params;
168
+ const { filters, groupId, from, to } = params;
165
169
  const formatGroupId = (0, import_utils.parseId)(groupId);
166
- const limit = (0, import_utils2.getLimit)(from, to);
170
+ const limit = (0, import_arangodbUtils.getLimit)(from, to);
167
171
  filters.map((filter) => {
168
172
  const { conditional, name, value } = filter;
169
173
  let formatCond = conditional;
@@ -179,7 +183,7 @@ const getImagesByGroup = (context, params) => {
179
183
  });
180
184
  return (0, import_groups.getGroupDetails)(context, formatGroupId).then((group) => {
181
185
  if (group.privacy === "public") {
182
- filters.push(`p.groupId == "${groupId}"`);
186
+ filters.push(`p.groupId == "${formatGroupId}"`);
183
187
  const filterStr = filters.join(" && ");
184
188
  const aqlQry = `FOR i IN
185
189
  FLATTEN(
@@ -195,11 +199,12 @@ const getImagesByGroup = (context, params) => {
195
199
  SORT i.added DESC
196
200
  ${limit.aql}
197
201
  RETURN i`;
198
- return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_utils2.logError)({
202
+ return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_analyticsUtils.logError)({
199
203
  action,
200
204
  category: eventCategory,
201
- label: "db_error"
202
- }, error, context).then(() => null));
205
+ value: import_error.ErrorTypes.DATABASE_ERROR,
206
+ params
207
+ }, error, context));
203
208
  }
204
209
  return (0, import_groups.isGrouped)(database, sessionId, groupId).then((grouped) => {
205
210
  if (grouped.isValid) {
@@ -212,11 +217,12 @@ const getImagesByGroup = (context, params) => {
212
217
  ${limit.aql}
213
218
  SORT p.added DESC
214
219
  RETURN f`;
215
- return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_utils2.logError)({
220
+ return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_analyticsUtils.logError)({
216
221
  action,
217
222
  category: eventCategory,
218
- label: "db_error"
219
- }, error, context).then(() => null));
223
+ value: import_error.ErrorTypes.DATABASE_ERROR,
224
+ params
225
+ }, error, context));
220
226
  }
221
227
  return [];
222
228
  });
@@ -239,25 +245,27 @@ const getImagesByReactions = (context, reactions = [], options) => {
239
245
  FILTER ${filterBy.join(" && ")}
240
246
  ${limit.aql}
241
247
  RETURN MERGE(i, {${selectObjects.join(", ")}})`;
242
- return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_utils2.logError)({
248
+ return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_analyticsUtils.logError)({
243
249
  action,
244
250
  category: eventCategory,
245
- label: "db_error"
251
+ value: import_error.ErrorTypes.DATABASE_ERROR,
252
+ params: { reactions, options }
246
253
  }, error, context).then(() => []));
247
254
  };
248
- const getImage = (context, id) => {
255
+ const getImage = (context, imageId) => {
249
256
  const action = "getItem";
250
257
  const { database } = context;
251
- const formatId = (0, import_utils.parseId)(id);
258
+ const formatImageId = (0, import_utils.parseId)(imageId);
252
259
  const aqlQry = import_arangojs.aql`FOR i IN images
253
- FILTER i._key==${formatId}
260
+ FILTER i._key==${formatImageId}
254
261
  LIMIT 1
255
262
  RETURN i`;
256
- return database.query(aqlQry).then((cursor) => cursor.next()).then((image = {}) => image).catch((error) => (0, import_utils2.logError)({
263
+ return database.query(aqlQry).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
257
264
  action,
258
265
  category: eventCategory,
259
- label: "db_error"
260
- }, error, context).then(() => null));
266
+ value: import_error.ErrorTypes.DATABASE_ERROR,
267
+ params: { imageId }
268
+ }, error, context));
261
269
  };
262
270
  const getPathUserImages = (userId, imageId, type, dir = "images") => {
263
271
  let filename = imageId;
@@ -289,149 +297,161 @@ const getAppImageUrl = (data) => {
289
297
  }
290
298
  return "";
291
299
  };
292
- const getImageUrl = (data) => {
300
+ const getImageUrl = async (data) => {
293
301
  const { bucket, imageId, isThumb = false, type, typeId } = data;
294
302
  const imgDir = isThumb ? "thumbs" : "images";
295
303
  if (imageId) {
296
304
  const imageKey = `${type}/${typeId}/${imgDir}/${imageId}.jpg`;
297
- return (0, import_s32.s3GetSignedUrl)({ Bucket: bucket, Key: imageKey, Expires: 900 });
305
+ return await (0, import_s3.s3GetSignedUrl)({ Bucket: bucket, Key: imageKey, Expires: 900 });
298
306
  }
299
307
  return "";
300
308
  };
301
- const resizeSaveImage = (context, imageId, buffer, type = "image/jpeg", s3Options) => {
309
+ const resizeSaveImage = async (context, imageId, buffer, fileType = "image/jpeg", s3Options) => {
302
310
  const action = "resizeSaveImage";
303
311
  const { session: { userId: sessionId } } = context;
304
312
  const imgW = import_config.Config.get("image.imgSize");
305
313
  const imgQ = import_config.Config.get("image.imgQuality");
306
- let photo = {};
307
- const format = type.split("/")[1];
308
- console.log({ buffer, format, imgW, imgQ });
309
- return new Promise((resolve) => {
310
- (0, import_gm.default)(buffer, "img").setFormat(format).quality(imgQ).autoOrient().resize(imgW, imgW, ">").identify({ bufferStream: true }, (error, val = {}) => {
314
+ const format = fileType.split("/")[1];
315
+ const imageOptimizedBuffer = await new Promise((resolve, reject) => {
316
+ (0, import_gm.default)(buffer, "img").setFormat(format).quality(imgQ).autoOrient().resize(imgW, imgW, ">").stream((streamError, stdout) => {
317
+ if (streamError) {
318
+ reject(
319
+ (0, import_analyticsUtils.logError)({
320
+ action,
321
+ category: eventCategory,
322
+ params: { imageId, fileType },
323
+ value: import_error.ErrorTypes.IMAGE_SAVE
324
+ }, streamError, context)
325
+ );
326
+ return;
327
+ }
328
+ let imageBuffer = Buffer.from("");
329
+ stdout.on("data", (data) => {
330
+ imageBuffer = Buffer.concat([imageBuffer, data]);
331
+ });
332
+ stdout.on("end", () => resolve(imageBuffer));
333
+ });
334
+ });
335
+ try {
336
+ const imageObj = {
337
+ ACL: "authenticated-read",
338
+ Body: imageOptimizedBuffer,
339
+ Bucket: null,
340
+ ContentType: fileType,
341
+ ...s3Options || {},
342
+ Key: getPathUserImages(sessionId, imageId, fileType, "images")
343
+ };
344
+ await (0, import_s3.s3Put)(imageObj);
345
+ } catch (error) {
346
+ return (0, import_analyticsUtils.logError)({
347
+ action,
348
+ category: eventCategory,
349
+ params: { imageId, fileType },
350
+ value: import_error.ErrorTypes.IMAGE_SAVE
351
+ }, error, context);
352
+ }
353
+ const imageIdentity = await new Promise((resolve) => {
354
+ (0, import_gm.default)(imageOptimizedBuffer, "img").identify({ bufferStream: true }, (error, imageData = {}) => {
311
355
  if (error) {
312
- (0, import_utils2.logError)({
356
+ return (0, import_analyticsUtils.logError)({
313
357
  action,
314
358
  category: eventCategory,
315
- label: "image_save",
316
- value: "gm_image_identify"
317
- }, error, context).catch((error2) => {
318
- throw error2;
319
- });
320
- } else {
321
- const formatVals = (0, import_utils2.lowerCaseKeys)(val);
322
- const { make: cameraMake, model: cameraModel, datetimeoriginal: taken } = formatVals;
323
- photo = {
324
- ...(0, import_cloneDeep.default)(photo),
325
- make: cameraMake,
326
- model: cameraModel,
327
- taken: taken ? import_luxon.DateTime.fromMillis(taken).millisecond : null
328
- };
329
- const stats = formatVals["channel statistics"];
330
- if (stats) {
331
- let { red, green, blue, mean } = stats;
332
- if (red) {
333
- mean = red["standard deviation"] || red.mean;
334
- red = mean ? +mean.split(" ")[0].substring(0, 3) : 0;
335
- } else {
336
- red = 0;
337
- }
338
- if (green) {
339
- mean = green["standard deviation"] || green.mean;
340
- green = mean ? +mean.split(" ")[0].substring(0, 3) : 0;
341
- } else {
342
- green = 0;
343
- }
344
- if (blue) {
345
- mean = blue["standard deviation"] || blue.mean;
346
- blue = mean ? +mean.split(" ")[0].substring(0, 3) : 0;
347
- } else {
348
- blue = 0;
349
- }
350
- const rgb = blue | green << 8 | red << 16;
351
- const color = rgb.toString(16);
352
- photo.color = color === "0" ? "000000" : color;
359
+ value: import_error.ErrorTypes.IMAGE_SAVE
360
+ }, error, context);
361
+ }
362
+ const formatValues = (0, import_utils.lowerCaseKeys)(imageData);
363
+ const { make: cameraMake, model: cameraModel, datetimeoriginal: taken } = formatValues;
364
+ const stats = formatValues["channel statistics"];
365
+ let color;
366
+ if (stats) {
367
+ let { red, green, blue, mean } = stats;
368
+ if (red) {
369
+ mean = red["standard deviation"] || red.mean;
370
+ red = mean ? +mean.split(" ")[0].substring(0, 3) : 0;
371
+ } else {
372
+ red = 0;
353
373
  }
374
+ if (green) {
375
+ mean = green["standard deviation"] || green.mean;
376
+ green = mean ? +mean.split(" ")[0].substring(0, 3) : 0;
377
+ } else {
378
+ green = 0;
379
+ }
380
+ if (blue) {
381
+ mean = blue["standard deviation"] || blue.mean;
382
+ blue = mean ? +mean.split(" ")[0].substring(0, 3) : 0;
383
+ } else {
384
+ blue = 0;
385
+ }
386
+ const rgb = blue | green << 8 | red << 16;
387
+ const rgbColor = rgb.toString(16);
388
+ color = rgbColor === "0" ? "000000" : rgbColor;
354
389
  }
355
- }).stream((error, stdout) => {
356
- if (error) {
357
- (0, import_utils2.logError)({
390
+ const {
391
+ "JPEG-Quality": quality,
392
+ Orientation: orientationData,
393
+ Resolution: resolutionData,
394
+ size
395
+ } = imageData;
396
+ let resolution;
397
+ if (resolutionData) {
398
+ const [resolutionNumber, resolutionUnit] = resolutionData.split(" ");
399
+ const resolutionValue = resolutionNumber.split("x")[0];
400
+ const resolutionUnitValue = resolutionUnit === "pixels/inch" ? "ppi" : "";
401
+ resolution = `${resolutionValue} ${resolutionUnitValue}`;
402
+ }
403
+ return resolve({
404
+ color,
405
+ fileSize: imageOptimizedBuffer.length,
406
+ height: size?.height || 0,
407
+ make: cameraMake,
408
+ model: cameraModel,
409
+ orientation: orientationData === "Unknown" ? void 0 : orientationData?.toLowerCase(),
410
+ quality: quality ? +quality : void 0,
411
+ resolution,
412
+ taken: taken ? import_luxon.DateTime.fromMillis(taken).millisecond : void 0,
413
+ width: size?.width || 0
414
+ });
415
+ });
416
+ });
417
+ const thumbOptimizedBuffer = await new Promise((resolve, reject) => {
418
+ const thmW = import_config.Config.get("image.thmSize");
419
+ const thmQ = import_config.Config.get("image.thmQuality");
420
+ (0, import_gm.default)(imageOptimizedBuffer, "img").setFormat(format).gravity("Center").resize(thmW, thmW, "^").extent(thmW, thmW).quality(thmQ).stream((streamError, thumbStdout) => {
421
+ if (streamError) {
422
+ (0, import_analyticsUtils.logError)({
358
423
  action,
359
424
  category: eventCategory,
360
- isInternal: true,
361
- label: "image_save",
362
- value: "gm_image_stream"
363
- }, error, context).then(() => null);
364
- } else {
365
- let imageBuffer = Buffer.from("");
366
- stdout.on("data", (data) => {
367
- imageBuffer = Buffer.concat([imageBuffer, data]);
368
- });
369
- stdout.on("end", () => {
370
- photo.fileSize = imageBuffer.length;
371
- const imageObj = {
372
- ACL: "authenticated-read",
373
- Body: imageBuffer,
374
- Bucket: null,
375
- ContentType: type,
376
- ...s3Options || {},
377
- Key: getPathUserImages(sessionId, imageId, type, "images")
378
- };
379
- (0, import_s32.s3Put)(imageObj).then(() => {
380
- const thmW = import_config.Config.get("image.thmSize");
381
- const thmQ = import_config.Config.get("image.thmQuality");
382
- (0, import_gm.default)(imageBuffer, "img").setFormat("jpeg").size((thumbError, resizeVal) => {
383
- if (!thumbError) {
384
- const { height, width } = resizeVal;
385
- photo = { ...(0, import_cloneDeep.default)(photo), height, width };
386
- }
387
- }).gravity("Center").resize(thmW, thmW, "^").extent(thmW, thmW).quality(thmQ).stream((streamError, thumbStdout) => {
388
- if (streamError) {
389
- (0, import_utils2.logError)({
390
- action,
391
- category: eventCategory,
392
- label: "image_save",
393
- value: "gm_thumbnail_steam"
394
- }, streamError, context).catch((error2) => {
395
- throw error2;
396
- });
397
- } else {
398
- let thumbBuffer = Buffer.from("");
399
- thumbStdout.on("data", (data) => {
400
- thumbBuffer = Buffer.concat([thumbBuffer, data]);
401
- });
402
- thumbStdout.on("end", () => {
403
- const thumbObj = {
404
- ACL: "authenticated-read",
405
- Body: thumbBuffer,
406
- Bucket: null,
407
- ContentType: type,
408
- ...s3Options || {},
409
- Key: getPathUserImages(sessionId, imageId, type, "thumbs")
410
- };
411
- (0, import_s32.s3Put)(thumbObj).then(() => {
412
- resolve(photo);
413
- }).catch((s3PutError) => (0, import_utils2.logError)({
414
- action,
415
- category: eventCategory,
416
- label: "image_save",
417
- value: "s3_put_image"
418
- }, s3PutError, context).catch((error2) => {
419
- throw error2;
420
- }));
421
- });
422
- }
423
- });
424
- }).catch((s3Error) => (0, import_utils2.logError)({
425
- action,
426
- category: eventCategory,
427
- isInternal: true,
428
- label: "image_save",
429
- value: "s3_image_save"
430
- }, s3Error, context).then(() => null));
431
- });
425
+ params: { imageId, fileType },
426
+ value: import_error.ErrorTypes.IMAGE_SAVE
427
+ }, streamError, context);
428
+ return;
432
429
  }
430
+ let thumbBuffer = Buffer.from("");
431
+ thumbStdout.on("data", (data) => {
432
+ thumbBuffer = Buffer.concat([thumbBuffer, data]);
433
+ });
434
+ thumbStdout.on("end", () => resolve(thumbBuffer));
433
435
  });
434
436
  });
437
+ try {
438
+ const thumbObj = {
439
+ ACL: "authenticated-read",
440
+ Body: thumbOptimizedBuffer,
441
+ Bucket: null,
442
+ ContentType: fileType,
443
+ ...s3Options || {},
444
+ Key: getPathUserImages(sessionId, imageId, fileType, "thumbs")
445
+ };
446
+ await (0, import_s3.s3Put)(thumbObj);
447
+ } catch (error) {
448
+ return (0, import_analyticsUtils.logError)({
449
+ action,
450
+ category: eventCategory,
451
+ value: import_error.ErrorTypes.IMAGE_SAVE
452
+ }, error, context);
453
+ }
454
+ return imageIdentity;
435
455
  };
436
456
  const addImage = (context, image, s3Options) => {
437
457
  const action = "addImage";
@@ -449,18 +469,17 @@ const addImage = (context, image, s3Options) => {
449
469
  userId: sessionId
450
470
  };
451
471
  const aqlQry = import_arangojs.aql`INSERT ${insert} IN images RETURN NEW`;
452
- return database.query(aqlQry).then((cursor) => cursor.next()).then(import_utils2.defaultObject).catch((error) => (0, import_utils2.logError)({
472
+ return database.query(aqlQry).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
453
473
  action,
454
474
  category: eventCategory,
455
- isInternal: true,
456
- label: "db_error"
457
- }, error, context).then(() => null));
458
- }).catch((error) => (0, import_utils2.logError)({
475
+ params: { image, s3Options },
476
+ value: import_error.ErrorTypes.DATABASE_ERROR
477
+ }, error, context));
478
+ }).catch((error) => (0, import_analyticsUtils.logError)({
459
479
  action,
460
480
  category: eventCategory,
461
- isInternal: true,
462
- label: "image_resize"
463
- }, error, context).then(() => null));
481
+ value: import_error.ErrorTypes.IMAGE_RESIZE
482
+ }, error, context));
464
483
  };
465
484
  const addImageEdge = async (context, imageEdge) => {
466
485
  const action = "addImageEdge";
@@ -471,24 +490,9 @@ const addImageEdge = async (context, imageEdge) => {
471
490
  const edgeId = (0, import_utils.createHash)(`hasImage-${imageId}-${itemId}-${sessionId}`);
472
491
  const formatItemType = (0, import_utils.parseChar)(itemType).toLowerCase();
473
492
  const formatItemId = (0, import_utils.parseId)(itemId);
474
- let itemDocId;
493
+ const itemDocId = (0, import_arangodbUtils.getDocId)(formatItemType, { _key: formatItemId });
475
494
  const formatImageId = (0, import_utils.parseId)(imageId);
476
495
  const fileDocId = `images/${formatImageId}`;
477
- switch (formatItemType) {
478
- case "groups":
479
- itemDocId = `groups/${formatItemId}`;
480
- break;
481
- case "posts":
482
- itemDocId = `posts/${formatItemId}`;
483
- break;
484
- case "users":
485
- case "profile":
486
- itemDocId = `users/${formatItemId}`;
487
- break;
488
- default:
489
- itemDocId = "";
490
- break;
491
- }
492
496
  const edge = {
493
497
  _from: itemDocId,
494
498
  _key: edgeId,
@@ -497,11 +501,9 @@ const addImageEdge = async (context, imageEdge) => {
497
501
  type: itemType
498
502
  };
499
503
  if (itemDocId) {
500
- return edgeCollection.save(edge, { returnNew: true }).then((fileEdge) => edgeCollection.document(fileEdge)).catch((error) => (0, import_utils2.logError)({
504
+ return edgeCollection.save(edge, { returnNew: true }).then((fileEdge) => edgeCollection.document(fileEdge)).catch((error) => (0, import_analyticsUtils.logError)({
501
505
  action,
502
506
  category: eventCategory,
503
- isInternal: true,
504
- label: "db_error",
505
507
  params: {
506
508
  edge,
507
509
  fileDocId,
@@ -509,10 +511,11 @@ const addImageEdge = async (context, imageEdge) => {
509
511
  itemDocId,
510
512
  itemId,
511
513
  itemType
512
- }
513
- }, error, context).then(() => null));
514
+ },
515
+ value: import_error.ErrorTypes.DATABASE_ERROR
516
+ }, error, context));
514
517
  }
515
- return {};
518
+ return null;
516
519
  };
517
520
  const updateImage = async (context, image, s3Options) => {
518
521
  const action = "updateImage";
@@ -544,8 +547,8 @@ const updateImage = async (context, image, s3Options) => {
544
547
  }
545
548
  let updatedFileType = fileType;
546
549
  if (!fileType) {
547
- const fileMime = await import_file_type.default.fileTypeFromBuffer(buffer);
548
- const { mime } = fileMime;
550
+ const { fileTypeFromBuffer } = await import("file-type");
551
+ const { mime } = await fileTypeFromBuffer(buffer);
549
552
  updatedFileType = mime;
550
553
  }
551
554
  const imgParams = {
@@ -563,21 +566,20 @@ const updateImage = async (context, image, s3Options) => {
563
566
  return addImageEdge(context, imageEdge).then(() => image2);
564
567
  }
565
568
  return image2;
566
- }).catch((error) => (0, import_utils2.logError)({
569
+ }).catch((error) => (0, import_analyticsUtils.logError)({
567
570
  action,
568
571
  category: eventCategory,
569
- label: "image_save_error"
570
- }, error, context).then(() => null));
572
+ value: import_error.ErrorTypes.IMAGE_SAVE
573
+ }, error, context));
571
574
  } else if (url !== "") {
572
575
  let contentType;
573
576
  return (0, import_rip_hunter.get)(url).then((res) => {
574
577
  if (res.status !== 200) {
575
- return (0, import_utils2.logException)({
578
+ return (0, import_analyticsUtils.logException)({
576
579
  action,
577
580
  category: eventCategory,
578
- label: "fetch_image_url",
579
- value: res.statusText
580
- }, context).then(() => null);
581
+ value: import_error.ErrorTypes.IMAGE_REQUEST
582
+ }, context);
581
583
  }
582
584
  contentType = res.headers.get("content-type");
583
585
  return res;
@@ -594,27 +596,27 @@ const updateImage = async (context, image, s3Options) => {
594
596
  }
595
597
  return image2;
596
598
  });
597
- }).catch((error) => (0, import_utils2.logError)({
599
+ }).catch((error) => (0, import_analyticsUtils.logError)({
598
600
  action,
599
601
  category: eventCategory,
600
- label: "fetch_error"
601
- }, error, context).then(() => null));
602
+ value: "fetch_error"
603
+ }, error, context));
602
604
  } else if (imageId !== "") {
603
605
  const update = {
604
606
  description,
605
607
  modified: now
606
608
  };
607
609
  const aqlQry = import_arangojs.aql`UPDATE {_key: ${formatImageId}} WITH ${update} IN images RETURN NEW`;
608
- return database.query(aqlQry).then((cursor) => cursor.next()).catch((error) => (0, import_utils2.logError)({
610
+ return database.query(aqlQry).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
609
611
  action,
610
612
  category: eventCategory,
611
- label: "db_error",
612
613
  params: {
613
614
  aqlQry,
614
615
  formatImageId,
615
616
  update
616
- }
617
- }, error, context).then(() => null));
617
+ },
618
+ value: import_error.ErrorTypes.DATABASE_ERROR
619
+ }, error, context));
618
620
  }
619
621
  return null;
620
622
  };
@@ -622,35 +624,40 @@ const deleteImage = async (context, imageId) => {
622
624
  const action = "delete";
623
625
  const { database, session: { userId: sessionId } } = context;
624
626
  const formatImageId = (0, import_utils.parseId)(imageId);
625
- const removeEdgeQuery = import_arangojs.aql`FOR hi IN hasImage
627
+ try {
628
+ const removeEdgeQuery = import_arangojs.aql`FOR hi IN hasImage
626
629
  FILTER hi._from == ${formatImageId}
627
630
  REMOVE hi IN hasImage
628
631
  RETURN OLD`;
629
- await database.query(removeEdgeQuery);
630
- const aqlQuery = import_arangojs.aql`FOR i IN images
632
+ await database.query(removeEdgeQuery);
633
+ const aqlQuery = import_arangojs.aql`FOR i IN images
631
634
  FILTER i._key == ${formatImageId} && i.userId == ${sessionId}
632
635
  REMOVE i IN images
633
636
  RETURN OLD`;
634
- const image = await database.query(aqlQuery).then((cursor) => cursor.next()).catch((error) => (0, import_utils2.logError)({
635
- action,
636
- category: eventCategory,
637
- label: "db_error"
638
- }, error, context).then(() => null));
639
- if (image) {
640
- const { _key: imageKey } = image;
641
- const params = {
642
- Bucket: null,
643
- Delete: {
644
- Objects: [
645
- { Key: `users/${sessionId}/images/${imageKey}.jpg` },
646
- { Key: `users/${sessionId}/thumbs/${imageKey}.jpg` }
647
- ],
648
- Quiet: true
649
- }
650
- };
651
- return (0, import_s32.s3DeleteList)(params).then(() => image);
637
+ const image = await database.query(aqlQuery).then((cursor) => cursor.next());
638
+ if (image) {
639
+ const { _key: imageKey } = image;
640
+ const params = {
641
+ Bucket: null,
642
+ Delete: {
643
+ Objects: [
644
+ { Key: `users/${sessionId}/images/${imageKey}.jpg` },
645
+ { Key: `users/${sessionId}/thumbs/${imageKey}.jpg` }
646
+ ],
647
+ Quiet: true
648
+ }
649
+ };
650
+ return (0, import_s3.s3DeleteList)(params).then(() => image);
651
+ }
652
+ return null;
653
+ } catch (error) {
654
+ return (0, import_analyticsUtils.logError)({
655
+ action,
656
+ category: eventCategory,
657
+ params: { imageId },
658
+ value: import_error.ErrorTypes.DATABASE_ERROR
659
+ }, error, context);
652
660
  }
653
- return {};
654
661
  };
655
662
  // Annotate the CommonJS export names for ESM import in node:
656
663
  0 && (module.exports = {
@@ -671,4 +678,4 @@ const deleteImage = async (context, imageId) => {
671
678
  resizeSaveImage,
672
679
  updateImage
673
680
  });
674
- //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vc3JjL2FjdGlvbnMvaW1hZ2VzLnRzIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyIvKipcbiAqIENvcHlyaWdodCAoYykgMjAxOS1QcmVzZW50LCBOaXRyb2dlbiBMYWJzLCBJbmMuXG4gKiBDb3B5cmlnaHRzIGxpY2Vuc2VkIHVuZGVyIHRoZSBNSVQgTGljZW5zZS4gU2VlIHRoZSBhY2NvbXBhbnlpbmcgTElDRU5TRSBmaWxlIGZvciB0ZXJtcy5cbiAqL1xuaW1wb3J0IHtnZXQgYXMgaHR0cEdldH0gZnJvbSAnQG5sYWJzL3JpcC1odW50ZXInO1xuaW1wb3J0IHtjcmVhdGVIYXNoLCBwYXJzZUFyYW5nb0lkLCBwYXJzZUNoYXIsIHBhcnNlSWQsIHBhcnNlTnVtfSBmcm9tICdAbmxhYnMvdXRpbHMnO1xuaW1wb3J0IHthcWx9IGZyb20gJ2FyYW5nb2pzJztcbmltcG9ydCB7QXFsUXVlcnl9IGZyb20gJ2FyYW5nb2pzL2FxbCc7XG5pbXBvcnQge0VkZ2VDb2xsZWN0aW9ufSBmcm9tICdhcmFuZ29qcy9jb2xsZWN0aW9uJztcbmltcG9ydCB7QXJyYXlDdXJzb3J9IGZyb20gJ2FyYW5nb2pzL2N1cnNvcic7XG5pbXBvcnQge0RlbGV0ZU9iamVjdHNSZXF1ZXN0LCBQdXRPYmplY3RSZXF1ZXN0fSBmcm9tICdhd3Mtc2RrL2NsaWVudHMvczMnO1xuaW1wb3J0IEZpbGVUeXBlLCB7RmlsZVR5cGVSZXN1bHR9IGZyb20gJ2ZpbGUtdHlwZSc7XG5pbXBvcnQgZ20gZnJvbSAnZ20nO1xuaW1wb3J0IGNsb25lRGVlcCBmcm9tICdsb2Rhc2gvY2xvbmVEZWVwJztcbmltcG9ydCB7RGF0ZVRpbWV9IGZyb20gJ2x1eG9uJztcblxuaW1wb3J0IHtDb25maWd9IGZyb20gJy4uL2NvbmZpZyc7XG5pbXBvcnQge1xuICBBcGlDb250ZXh0LFxuICBBcmFuZ29EQkxpbWl0LFxuICBHcm91cFR5cGUsXG4gIEdyb3VwVXNlcixcbiAgSW1hZ2VFZGdlVHlwZSxcbiAgSW1hZ2VJZGVudGlmeVR5cGUsXG4gIEltYWdlT3B0aW9ucyxcbiAgSW1hZ2VUeXBlLFxuICBJbWFnZVVybERhdGEsXG4gIFF1ZXJ5RmlsdGVyXG59IGZyb20gJy4uL3R5cGVzJztcbmltcG9ydCB7ZGVmYXVsdE9iamVjdCwgZ2V0TGltaXQsIGxvZ0Vycm9yLCBsb2dFeGNlcHRpb24sIGxvd2VyQ2FzZUtleXMsIHNlbGVjdFJlYWN0aW9uQ291bnRCeVR5cGV9IGZyb20gJy4uL3V0aWxzJztcbmltcG9ydCB7Z2V0R3JvdXBEZXRhaWxzLCBpc0dyb3VwZWR9IGZyb20gJy4vZ3JvdXBzJztcbmltcG9ydCB7czNEZWxldGVMaXN0LCBzM0dldFNpZ25lZFVybCwgczNQdXR9IGZyb20gJy4vczMnO1xuXG5jb25zdCBldmVudENhdGVnb3J5OiBzdHJpbmcgPSAnaW1hZ2VzJztcblxuZXhwb3J0IGNvbnN0IHBhcnNlSW1hZ2VPcHRpb25zID0gKG9wdGlvbnM6IEltYWdlT3B0aW9ucyA9IHt9KSA9PiB7XG4gIGNvbnN0IHtcbiAgICBmcm9tID0gMCxcbiAgICB0byA9IDMwLFxuICAgIHR5cGUgPSAnZGVmYXVsdCdcbiAgfSA9IG9wdGlvbnM7XG5cbiAgcmV0dXJuIHtcbiAgICBsaW1pdDogZ2V0TGltaXQoZnJvbSwgdG8pLFxuICAgIHR5cGU6IHBhcnNlQ2hhcih0eXBlLCAzMilcbiAgfTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRJbWFnZU9wdGlvbmFsID0gKGZpZWxkczogc3RyaW5nW10gPSBbXSkgPT5cbiAgZmllbGRzLnJlZHVjZSgoc2VsZWN0czogYW55LCBmaWVsZDogc3RyaW5nKSA9PiB7XG4gICAgaWYoZmllbGQuaW5jbHVkZXMoJ0NvdW50JykpIHtcbiAgICAgIHJldHVybiBzZWxlY3RSZWFjdGlvbkNvdW50QnlUeXBlKCdpbWFnZXMnLCAnaScsIGZpZWxkLCBzZWxlY3RzKTtcbiAgICB9XG5cbiAgICBzd2l0Y2goZmllbGQpIHtcbiAgICAgIGNhc2UgJ3JlYWN0aW9ucyc6IHtcbiAgICAgICAgc2VsZWN0cy5xdWVyaWVzLnB1c2goYExFVCByZWFjdGlvbnMgPSAoXG4gICAgICAgICAgRk9SIGltYWdlLCByIElOIElOQk9VTkQgaS5faWQgcmVhY3Rpb25zXG4gICAgICAgICAgQ09MTEVDVCByZWFjdGlvbk5hbWUgPSByLnZhbHVlIElOVE8gcmVhY3Rpb25JdGVtc1xuICAgICAgICAgIFJFVFVSTiB7bmFtZTogcmVhY3Rpb25OYW1lLCBjb3VudDogTEVOR1RIKHJlYWN0aW9uSXRlbXNbKl0uci52YWx1ZSl9XG4gICAgICAgIClgKTtcbiAgICAgICAgc2VsZWN0cy5vYmplY3RzLnB1c2goJ3JlYWN0aW9uczpyZWFjdGlvbnMnKTtcbiAgICAgICAgcmV0dXJuIHNlbGVjdHM7XG4gICAgICB9XG4gICAgICBjYXNlICd0YWdzJzoge1xuICAgICAgICBzZWxlY3RzLnF1ZXJpZXMucHVzaChgTEVUIHRhZ3MgPSAoXG4gICAgICAgICAgRk9SIHQsIHBsIElOIElOQk9VTkQgaS5faWQgaXNUYWdnZWRcbiAgICAgICAgICBSRVRVUk4gdFxuICAgICAgICApYCk7XG4gICAgICAgIHNlbGVjdHMub2JqZWN0cy5wdXNoKCd0YWdzOnRhZ3MnKTtcbiAgICAgICAgcmV0dXJuIHNlbGVjdHM7XG4gICAgICB9XG4gICAgICBjYXNlICd1c2VyJzoge1xuICAgICAgICBzZWxlY3RzLnF1ZXJpZXMucHVzaChgTEVUIHVzZXIgPSBGSVJTVChcbiAgICAgICAgICBGT1IgdSBJTiB1c2Vyc1xuICAgICAgICAgIEZJTFRFUiBpLnVzZXJJZCA9PSB1Ll9rZXlcbiAgICAgICAgICBMSU1JVCAxXG4gICAgICAgICAgUkVUVVJOIHVcbiAgICAgICAgKWApO1xuICAgICAgICBzZWxlY3RzLm9iamVjdHMucHVzaCgndXNlcjp1c2VyJyk7XG4gICAgICAgIHJldHVybiBzZWxlY3RzO1xuICAgICAgfVxuICAgICAgZGVmYXVsdDoge1xuICAgICAgICByZXR1cm4gc2VsZWN0cztcbiAgICAgIH1cbiAgICB9XG4gIH0sIHtvYmplY3RzOiBbXSwgcXVlcmllczogW119KTtcblxuZXhwb3J0IGNvbnN0IGdldEltYWdlc0J5VXNlciA9IChcbiAgY29udGV4dDogQXBpQ29udGV4dCxcbiAgdXNlcklkOiBzdHJpbmcsXG4gIGZyb206IG51bWJlcixcbiAgdG86IG51bWJlclxuKTogUHJvbWlzZTxJbWFnZVR5cGVbXT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRJbWFnZXNCeVVzZXInO1xuICBjb25zdCB7ZGF0YWJhc2V9ID0gY29udGV4dDtcbiAgY29uc3QgZm9ybWF0VXNlcklkOiBzdHJpbmcgPSBwYXJzZUlkKHVzZXJJZCk7XG4gIGNvbnN0IGxpbWl0OiBBcmFuZ29EQkxpbWl0ID0gZ2V0TGltaXQoZnJvbSwgdG8pO1xuICBjb25zdCBhcWxRcnk6IHN0cmluZyA9IGBGT1IgaSBJTiBpbWFnZXNcbiAgICAgIEZJTFRFUiBpLnVzZXJJZCA9PSBcIiR7Zm9ybWF0VXNlcklkfVwiXG4gICAgICBMRVQgdXNlciA9IEZJUlNUKFxuICAgICAgICBGT1IgdSBJTiB1c2Vyc1xuICAgICAgICBGSUxURVIgdS5fa2V5ID09IGkudXNlcklkXG4gICAgICAgIExJTUlUIDFcbiAgICAgICAgUkVUVVJOIHVcbiAgICAgIClcbiAgICAgICR7bGltaXQuYXFsfVxuICAgICAgU09SVCBpLmFkZGVkXG4gICAgICBSRVRVUk4gTUVSR0UoaSwge3VzZXI6dXNlcn0pYDtcblxuICByZXR1cm4gZGF0YWJhc2UucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IuYWxsKCkpXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgbGFiZWw6ICdkYl9lcnJvcidcbiAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKSk7XG59O1xuXG5leHBvcnQgY29uc3QgZ2V0SW1hZ2VDb3VudEJ5SXRlbSA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBpdGVtSWQ6IHN0cmluZyk6IFByb21pc2U8bnVtYmVyPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2dldEltYWdlQ291bnRCeUl0ZW0nO1xuICBjb25zdCB7ZGF0YWJhc2V9ID0gY29udGV4dDtcbiAgY29uc3QgZm9ybWF0SXRlbUlkOiBzdHJpbmcgPSBwYXJzZUFyYW5nb0lkKGl0ZW1JZCk7XG4gIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgRk9SIGkgSU4gaGFzSW1hZ2VcbiAgICAgIEZJTFRFUiBpLl90byA9PSAke2Zvcm1hdEl0ZW1JZH1cbiAgICAgIFJFVFVSTiB7Y291bnQ6IENPVU5UKGkpfWA7XG5cbiAgcmV0dXJuIGRhdGFiYXNlLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAudGhlbigoe2NvdW50fSA9IHtjb3VudDogMH0pID0+IGNvdW50KVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICBhY3Rpb24sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIGxhYmVsOiAnZGJfZXJyb3InXG4gICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gMCkpO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldEltYWdlc0J5SXRlbSA9IGFzeW5jIChcbiAgY29udGV4dDogQXBpQ29udGV4dCxcbiAgaXRlbUlkOiBzdHJpbmcsXG4gIG9wdGlvbnM6IEltYWdlT3B0aW9ucyA9IHt9XG4pOiBQcm9taXNlPEltYWdlVHlwZVtdPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2dldEltYWdlc0J5SXRlbSc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgZmllbGRzID0gW119ID0gY29udGV4dDtcbiAgY29uc3QgZm9ybWF0SXRlbUlkOiBzdHJpbmcgPSBwYXJzZUFyYW5nb0lkKGl0ZW1JZCk7XG4gIGNvbnN0IHtsaW1pdH0gPSBwYXJzZUltYWdlT3B0aW9ucyhvcHRpb25zKTtcbiAgY29uc3Qge29iamVjdHM6IHNlbGVjdE9iamVjdHMsIHF1ZXJpZXM6IHNlbGVjdFF1ZXJpZXN9ID0gZ2V0SW1hZ2VPcHRpb25hbChmaWVsZHMpO1xuICBjb25zdCBhcWxRcnk6IHN0cmluZyA9IGBGT1IgaSwgbCBJTiAxLi4xIE9VVEJPVU5EIFwiJHtmb3JtYXRJdGVtSWR9XCIgaGFzSW1hZ2VcbiAgICBGSUxURVIgTk9UIElTX05VTEwoaSlcbiAgICAke3NlbGVjdFF1ZXJpZXMuam9pbignXFxuJyl9XG4gICAgU09SVCBpLmFkZGVkXG4gICAgJHtsaW1pdC5hcWx9XG4gICAgUkVUVVJOIE1FUkdFKGksIHske3NlbGVjdE9iamVjdHMuam9pbignLCAnKX19KWA7XG5cbiAgcmV0dXJuIGRhdGFiYXNlLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLmFsbCgpKVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICBhY3Rpb24sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIGxhYmVsOiAnZGJfZXJyb3InXG4gICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCkpO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldEltYWdlc0J5R3JvdXAgPSAoY29udGV4dDogQXBpQ29udGV4dCwgcGFyYW1zKTogUHJvbWlzZTxJbWFnZVR5cGVbXT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRJbWFnZXNCeUdyb3VwJztcbiAgY29uc3Qge2RhdGFiYXNlLCBzZXNzaW9uOiB7dXNlcklkOiBzZXNzaW9uSWR9fSA9IGNvbnRleHQ7XG4gIGNvbnN0IHtmaWx0ZXJzID0gW10sIGdyb3VwSWQsIGZyb20sIHRvfSA9IHBhcmFtcztcbiAgY29uc3QgZm9ybWF0R3JvdXBJZDogc3RyaW5nID0gcGFyc2VJZChncm91cElkKTtcbiAgY29uc3QgbGltaXQgPSBnZXRMaW1pdChmcm9tLCB0byk7XG5cbiAgZmlsdGVyc1xuICAgIC5tYXAoKGZpbHRlcjogUXVlcnlGaWx0ZXIpID0+IHtcbiAgICAgIGNvbnN0IHtjb25kaXRpb25hbCwgbmFtZSwgdmFsdWV9ID0gZmlsdGVyO1xuICAgICAgbGV0IGZvcm1hdENvbmQ6IHN0cmluZyA9IGNvbmRpdGlvbmFsO1xuXG4gICAgICBpZihjb25kaXRpb25hbCAhPT0gJz49JyAmJiBjb25kaXRpb25hbCAhPT0gJzw9JyAmJiBjb25kaXRpb25hbCAhPT0gJz4nICYmIGNvbmRpdGlvbmFsICE9PSAnPCcpIHtcbiAgICAgICAgZm9ybWF0Q29uZCA9ICc9PSc7XG4gICAgICB9XG5cbiAgICAgIHN3aXRjaChuYW1lKSB7XG4gICAgICAgIGNhc2UgJ2FkZGVkJzpcbiAgICAgICAgICByZXR1cm4gYHAuYWRkZWQgJHtmb3JtYXRDb25kfSAke3BhcnNlTnVtKHZhbHVlKX1gO1xuICAgICAgICBkZWZhdWx0OlxuICAgICAgICAgIHJldHVybiAnJztcbiAgICAgIH1cbiAgICB9KTtcblxuICByZXR1cm4gZ2V0R3JvdXBEZXRhaWxzKGNvbnRleHQsIGZvcm1hdEdyb3VwSWQpXG4gICAgLnRoZW4oKGdyb3VwOiBHcm91cFR5cGUpID0+IHtcbiAgICAgIGlmKGdyb3VwLnByaXZhY3kgPT09ICdwdWJsaWMnKSB7XG4gICAgICAgIGZpbHRlcnMucHVzaChgcC5ncm91cElkID09IFwiJHtncm91cElkfVwiYCk7XG4gICAgICAgIGNvbnN0IGZpbHRlclN0ciA9IGZpbHRlcnMuam9pbignICYmICcpO1xuICAgICAgICBjb25zdCBhcWxRcnk6IHN0cmluZyA9IGBGT1IgaSBJTlxuICAgICAgICAgIEZMQVRURU4oXG4gICAgICAgICAgICBGT1IgcCBJTiBwb3N0c1xuICAgICAgICAgICAgRklMVEVSICR7ZmlsdGVyU3RyfVxuICAgICAgICAgICAgTEVUIGltYWdlcyA9IChcbiAgICAgICAgICAgICAgRk9SIGksIGUgSU4gSU5CT1VORCBwLl9pZCBoYXNJbWFnZVxuICAgICAgICAgICAgICBSRVRVUk4gaVxuICAgICAgICAgICAgKVxuICAgICAgICAgICAgU09SVCBwLmFkZGVkIERFU0NcbiAgICAgICAgICAgIFJFVFVSTiBpbWFnZXNcbiAgICAgICAgICApXG4gICAgICAgICAgU09SVCBpLmFkZGVkIERFU0NcbiAgICAgICAgICAke2xpbWl0LmFxbH1cbiAgICAgICAgICBSRVRVUk4gaWA7XG5cbiAgICAgICAgcmV0dXJuIGRhdGFiYXNlLnF1ZXJ5KGFxbFFyeSlcbiAgICAgICAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLmFsbCgpKVxuICAgICAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgICAgIGxhYmVsOiAnZGJfZXJyb3InXG4gICAgICAgICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCkpO1xuICAgICAgfVxuICAgICAgcmV0dXJuIGlzR3JvdXBlZChkYXRhYmFzZSwgc2Vzc2lvbklkLCBncm91cElkKVxuICAgICAgICAudGhlbigoZ3JvdXBlZDogR3JvdXBVc2VyKSA9PiB7XG4gICAgICAgICAgaWYoZ3JvdXBlZC5pc1ZhbGlkKSB7XG4gICAgICAgICAgICBmaWx0ZXJzLnB1c2goYHAuZ3JvdXBJZCA9PSBcIiR7Z3JvdXBlZC5ncm91cElkfVwiYCk7XG4gICAgICAgICAgICBjb25zdCBmaWx0ZXJMaXN0OiBzdHJpbmcgPSBmaWx0ZXJzLmpvaW4oJyAmJiAnKTtcbiAgICAgICAgICAgIGNvbnN0IGFxbFFyeTogc3RyaW5nID0gYEZPUiBwIElOIHBvc3RcbiAgICAgICAgICAgICAgICBGSUxURVIgJHtmaWx0ZXJMaXN0fVxuICAgICAgICAgICAgICAgIEZPUiBmIElOIHAuZmlsZXNcbiAgICAgICAgICAgICAgICBGSUxURVIgZi50eXBlID09IFwiaW1hZ2UvanBlZ1wiIHx8IGYudHlwZSA9PSBcImltYWdlL3BuZ1wiXG4gICAgICAgICAgICAgICAgJHtsaW1pdC5hcWx9XG4gICAgICAgICAgICAgICAgU09SVCBwLmFkZGVkIERFU0NcbiAgICAgICAgICAgICAgICBSRVRVUk4gZmA7XG5cbiAgICAgICAgICAgIHJldHVybiBkYXRhYmFzZS5xdWVyeShhcWxRcnkpXG4gICAgICAgICAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IuYWxsKCkpXG4gICAgICAgICAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgICAgICAgICAgYWN0aW9uLFxuICAgICAgICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgICAgICAgIGxhYmVsOiAnZGJfZXJyb3InXG4gICAgICAgICAgICAgIH0sIGVycm9yLCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpKTtcbiAgICAgICAgICB9XG4gICAgICAgICAgcmV0dXJuIFtdO1xuICAgICAgICB9KTtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRJbWFnZXNCeVJlYWN0aW9ucyA9IChcbiAgY29udGV4dDogQXBpQ29udGV4dCxcbiAgcmVhY3Rpb25zOiBzdHJpbmdbXSA9IFtdLFxuICBvcHRpb25zPzogSW1hZ2VPcHRpb25zXG4pOiBQcm9taXNlPEltYWdlVHlwZVtdPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2dldFVzZXJzQnlJbWFnZSc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgZmllbGRzID0gW10sIHNlc3Npb246IHt1c2VySWQ6IHNlc3Npb25JZH19ID0gY29udGV4dDtcbiAgY29uc3Qge2xpbWl0fSA9IHBhcnNlSW1hZ2VPcHRpb25zKG9wdGlvbnMpO1xuICBjb25zdCB7b2JqZWN0czogc2VsZWN0T2JqZWN0cywgcXVlcmllczogc2VsZWN0UXVlcmllc30gPSBnZXRJbWFnZU9wdGlvbmFsKGZpZWxkcyk7XG5cbiAgY29uc3QgZm9ybWF0U2Vzc2lvbklkOiBzdHJpbmcgPSBgdXNlcnMvJHtzZXNzaW9uSWR9YDtcbiAgY29uc3QgZm9ybWF0UmVhY3Rpb25zOiBzdHJpbmdbXSA9IHJlYWN0aW9ucy5tYXAoKHJlYWN0aW9uTmFtZSkgPT4gcGFyc2VDaGFyKHJlYWN0aW9uTmFtZSwgMzIpKTtcbiAgY29uc3QgZmlsdGVyQnk6IHN0cmluZ1tdID0gW1xuICAgICdyLnR5cGUgPT0gXCJpbWFnZXNcIicsXG4gICAgYFBPU0lUSU9OKCR7SlNPTi5zdHJpbmdpZnkoZm9ybWF0UmVhY3Rpb25zKX0sIExPV0VSKHIubmFtZSkpYF07XG5cbiAgLy8gR2V0IGRhdGEgZnJvbSBkYXRhYmFzZVxuICBjb25zdCBhcWxRcnk6IHN0cmluZyA9IGBGT1IgaSwgciBJTiBPVVRCT1VORCBcIiR7Zm9ybWF0U2Vzc2lvbklkfVwiIGhhc1JlYWN0aW9uXG4gICAgT1BUSU9OUyB7dmVydGV4Q29sbGVjdGlvbnM6IFwiaW1hZ2VzXCJ9XG4gICAgJHtzZWxlY3RRdWVyaWVzLmpvaW4oJ1xcbicpfVxuICAgIEZJTFRFUiAke2ZpbHRlckJ5LmpvaW4oJyAmJiAnKX1cbiAgICAke2xpbWl0LmFxbH1cbiAgICBSRVRVUk4gTUVSR0UoaSwgeyR7c2VsZWN0T2JqZWN0cy5qb2luKCcsICcpfX0pYDtcblxuICByZXR1cm4gZGF0YWJhc2UucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IuYWxsKCkpXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgbGFiZWw6ICdkYl9lcnJvcidcbiAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBbXSkpO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldEltYWdlID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGlkOiBzdHJpbmcpOiBQcm9taXNlPEltYWdlVHlwZT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRJdGVtJztcbiAgY29uc3Qge2RhdGFiYXNlfSA9IGNvbnRleHQ7XG4gIGNvbnN0IGZvcm1hdElkOiBzdHJpbmcgPSBwYXJzZUlkKGlkKTtcbiAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBGT1IgaSBJTiBpbWFnZXNcbiAgICBGSUxURVIgaS5fa2V5PT0ke2Zvcm1hdElkfVxuICAgIExJTUlUIDFcbiAgICBSRVRVUk4gaWA7XG5cbiAgcmV0dXJuIGRhdGFiYXNlLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAudGhlbigoaW1hZ2U6IEltYWdlVHlwZSA9IHt9KSA9PiBpbWFnZSlcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgYWN0aW9uLFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICBsYWJlbDogJ2RiX2Vycm9yJ1xuICAgIH0sIGVycm9yLCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpKTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRQYXRoVXNlckltYWdlcyA9ICh1c2VySWQ6IHN0cmluZywgaW1hZ2VJZDogc3RyaW5nLCB0eXBlOiBzdHJpbmcsIGRpcjogc3RyaW5nID0gJ2ltYWdlcycpOiBzdHJpbmcgPT4ge1xuICBsZXQgZmlsZW5hbWU6IHN0cmluZyA9IGltYWdlSWQ7XG5cbiAgc3dpdGNoKHR5cGUpIHtcbiAgICBjYXNlICdpbWFnZS9wbmcnOlxuICAgICAgZmlsZW5hbWUgPSBgJHtpbWFnZUlkfS5wbmdgO1xuICAgICAgYnJlYWs7XG4gICAgZGVmYXVsdDpcbiAgICAgIGZpbGVuYW1lID0gYCR7aW1hZ2VJZH0uanBnYDtcbiAgICAgIGJyZWFrO1xuICB9XG5cbiAgcmV0dXJuIGB1c2Vycy8ke3VzZXJJZH0vJHtkaXJ9LyR7ZmlsZW5hbWV9YDtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRBcHBJbWFnZVVybCA9IChkYXRhOiBJbWFnZVVybERhdGEpOiBzdHJpbmcgPT4ge1xuICBjb25zdCB7aW1hZ2VJZCwgZGlyZWN0b3J5ID0gJ2ltYWdlcycsIGltYWdlVHlwZSA9ICdwcm9maWxlJywgaXNUaHVtYiwgdHlwZSwgdHlwZUlkfSA9IGRhdGE7XG4gIGNvbnN0IGhvc3Q6IHN0cmluZyA9IENvbmZpZy5nZXQoJ2Vudmlyb25tZW50JykgPT09ICdwcm9kJ1xuICAgID8gYGh0dHBzOi8vYm94LiR7Q29uZmlnLmdldCgnYXBwLnVybCcpfWBcbiAgICA6IGBodHRwczovL3MzLmFtYXpvbmF3cy5jb20vZGV2LiR7Q29uZmlnLmdldCgnYXBwLnVybCcpfWA7XG4gIGNvbnN0IHN1ZmZpeDogc3RyaW5nID0gaXNUaHVtYiA/ICctdGgnIDogJyc7XG5cbiAgaWYoaW1hZ2VJZCkge1xuICAgIHN3aXRjaCh0eXBlKSB7XG4gICAgICBjYXNlICdhcHBzJzpcbiAgICAgICAgLy8gaHR0cHM6Ly9ib3gucmVha3Rvci5pby9teUFwcC9hcHAvaW1hZ2VzLzEyMy5qcGdcbiAgICAgICAgcmV0dXJuIGAke2hvc3R9LyR7dHlwZX0vJHtkaXJlY3Rvcnl9LyR7aW1hZ2VJZH0ke3N1ZmZpeH0uanBnYDtcbiAgICAgIGNhc2UgJ3VzZXJzJzpcbiAgICAgICAgLy8gaHR0cHM6Ly9ib3gucmVha3Rvci5pby9teUFwcC91c2Vycy9kZW1vVXNlci9pbWFnZXMvMTIzLmpwZ1xuICAgICAgICByZXR1cm4gYCR7aG9zdH0vJHt0eXBlfS8ke3R5cGVJZH0vJHtkaXJlY3Rvcnl9LyR7aW1hZ2VJZH0ke3N1ZmZpeH0uanBnYDtcbiAgICB9XG5cbiAgICBpZihpbWFnZVR5cGUgPT09ICdwcm9maWxlJykge1xuICAgICAgcmV0dXJuIGAke2hvc3R9L2RlZmF1bHRzLyR7dHlwZX1fYmske3N1ZmZpeH0uanBnYDtcbiAgICB9XG5cbiAgICByZXR1cm4gYCR7aG9zdH0vZGVmYXVsdHMvJHt0eXBlfV93aCR7c3VmZml4fS5qcGdgO1xuICB9XG5cbiAgcmV0dXJuICcnO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldEltYWdlVXJsID0gKGRhdGE6IEltYWdlVXJsRGF0YSk6IHN0cmluZyA9PiB7XG4gIGNvbnN0IHtidWNrZXQsIGltYWdlSWQsIGlzVGh1bWIgPSBmYWxzZSwgdHlwZSwgdHlwZUlkfSA9IGRhdGE7XG4gIGNvbnN0IGltZ0Rpcjogc3RyaW5nID0gaXNUaHVtYiA/ICd0aHVtYnMnIDogJ2ltYWdlcyc7XG5cbiAgaWYoaW1hZ2VJZCkge1xuICAgIGNvbnN0IGltYWdlS2V5OiBzdHJpbmcgPSBgJHt0eXBlfS8ke3R5cGVJZH0vJHtpbWdEaXJ9LyR7aW1hZ2VJZH0uanBnYDtcbiAgICByZXR1cm4gczNHZXRTaWduZWRVcmwoe0J1Y2tldDogYnVja2V0LCBLZXk6IGltYWdlS2V5LCBFeHBpcmVzOiA5MDB9KTtcbiAgfVxuXG4gIHJldHVybiAnJztcbn07XG5cbmV4cG9ydCBjb25zdCByZXNpemVTYXZlSW1hZ2UgPSAoXG4gIGNvbnRleHQ6IEFwaUNvbnRleHQsXG4gIGltYWdlSWQ6IHN0cmluZyxcbiAgYnVmZmVyOiBCdWZmZXIsXG4gIHR5cGU6IHN0cmluZyA9ICdpbWFnZS9qcGVnJywgczNPcHRpb25zPzogUHV0T2JqZWN0UmVxdWVzdCk6IFByb21pc2U8SW1hZ2VUeXBlPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ3Jlc2l6ZVNhdmVJbWFnZSc7XG4gIGNvbnN0IHtzZXNzaW9uOiB7dXNlcklkOiBzZXNzaW9uSWR9fSA9IGNvbnRleHQ7XG4gIGNvbnN0IGltZ1c6IG51bWJlciA9IENvbmZpZy5nZXQoJ2ltYWdlLmltZ1NpemUnKTtcbiAgY29uc3QgaW1nUTogbnVtYmVyID0gQ29uZmlnLmdldCgnaW1hZ2UuaW1nUXVhbGl0eScpO1xuICBsZXQgcGhvdG86IEltYWdlVHlwZSA9IHt9O1xuICBjb25zdCBmb3JtYXQ6IHN0cmluZyA9ICh0eXBlLnNwbGl0KCcvJykpWzFdO1xuXG4gIGNvbnNvbGUubG9nKHtidWZmZXIsIGZvcm1hdCwgaW1nVywgaW1nUX0pO1xuICByZXR1cm4gbmV3IFByb21pc2UoKHJlc29sdmUpID0+IHtcbiAgICBnbShidWZmZXIsICdpbWcnKVxuICAgICAgLnNldEZvcm1hdChmb3JtYXQpXG4gICAgICAucXVhbGl0eShpbWdRKVxuICAgICAgLmF1dG9PcmllbnQoKVxuICAgICAgLnJlc2l6ZShpbWdXLCBpbWdXLCAnPicpXG4gICAgICAuaWRlbnRpZnkoe2J1ZmZlclN0cmVhbTogdHJ1ZX0sIChlcnJvcjogRXJyb3IsIHZhbDogSW1hZ2VJZGVudGlmeVR5cGUgPSB7fSk6IGFueSA9PiB7XG4gICAgICAgIGlmKGVycm9yKSB7XG4gICAgICAgICAgbG9nRXJyb3Ioe1xuICAgICAgICAgICAgYWN0aW9uLFxuICAgICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgICBsYWJlbDogJ2ltYWdlX3NhdmUnLFxuICAgICAgICAgICAgdmFsdWU6ICdnbV9pbWFnZV9pZGVudGlmeSdcbiAgICAgICAgICB9LCBlcnJvciwgY29udGV4dCkuY2F0Y2goKGVycm9yKSA9PiB7XG4gICAgICAgICAgICB0aHJvdyBlcnJvcjtcbiAgICAgICAgICB9KTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBjb25zdCBmb3JtYXRWYWxzID0gbG93ZXJDYXNlS2V5cyh2YWwpO1xuICAgICAgICAgIGNvbnN0IHttYWtlOiBjYW1lcmFNYWtlLCBtb2RlbDogY2FtZXJhTW9kZWwsIGRhdGV0aW1lb3JpZ2luYWw6IHRha2VufTogSW1hZ2VJZGVudGlmeVR5cGUgPSBmb3JtYXRWYWxzO1xuICAgICAgICAgIHBob3RvID0ge1xuICAgICAgICAgICAgLi4uY2xvbmVEZWVwKHBob3RvKSxcbiAgICAgICAgICAgIG1ha2U6IGNhbWVyYU1ha2UsXG4gICAgICAgICAgICBtb2RlbDogY2FtZXJhTW9kZWwsXG4gICAgICAgICAgICB0YWtlbjogdGFrZW4gPyBEYXRlVGltZS5mcm9tTWlsbGlzKHRha2VuKS5taWxsaXNlY29uZCA6IG51bGxcbiAgICAgICAgICB9O1xuXG4gICAgICAgICAgLy8gSWYgbm8gYmFja2dyb3VuZCBjb2xvciwgZ2V0IHRoZSBtZWFuIGNvbG9yIHZhbHVlXG4gICAgICAgICAgY29uc3Qgc3RhdHMgPSBmb3JtYXRWYWxzWydjaGFubmVsIHN0YXRpc3RpY3MnXTtcblxuICAgICAgICAgIGlmKHN0YXRzKSB7XG4gICAgICAgICAgICBsZXQge3JlZCwgZ3JlZW4sIGJsdWUsIG1lYW59ID0gc3RhdHM7XG5cbiAgICAgICAgICAgIGlmKHJlZCkge1xuICAgICAgICAgICAgICBtZWFuID0gcmVkWydzdGFuZGFyZCBkZXZpYXRpb24nXSB8fCByZWQubWVhbjtcbiAgICAgICAgICAgICAgcmVkID0gbWVhbiA/ICsoKG1lYW4uc3BsaXQoJyAnKVswXSkuc3Vic3RyaW5nKDAsIDMpKSA6IDA7XG4gICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICByZWQgPSAwO1xuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICBpZihncmVlbikge1xuICAgICAgICAgICAgICBtZWFuID0gZ3JlZW5bJ3N0YW5kYXJkIGRldmlhdGlvbiddIHx8IGdyZWVuLm1lYW47XG4gICAgICAgICAgICAgIGdyZWVuID0gbWVhbiA/ICsoKG1lYW4uc3BsaXQoJyAnKVswXSkuc3Vic3RyaW5nKDAsIDMpKSA6IDA7XG4gICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICBncmVlbiA9IDA7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIGlmKGJsdWUpIHtcbiAgICAgICAgICAgICAgbWVhbiA9IGJsdWVbJ3N0YW5kYXJkIGRldmlhdGlvbiddIHx8IGJsdWUubWVhbjtcbiAgICAgICAgICAgICAgYmx1ZSA9IG1lYW4gPyArKChtZWFuLnNwbGl0KCcgJylbMF0pLnN1YnN0cmluZygwLCAzKSkgOiAwO1xuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgYmx1ZSA9IDA7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIGNvbnN0IHJnYiA9IGJsdWUgfCAoZ3JlZW4gPDwgOCkgfCAocmVkIDw8IDE2KTtcbiAgICAgICAgICAgIGNvbnN0IGNvbG9yID0gcmdiLnRvU3RyaW5nKDE2KTtcbiAgICAgICAgICAgIHBob3RvLmNvbG9yID0gY29sb3IgPT09ICcwJyA/ICcwMDAwMDAnIDogY29sb3I7XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICB9KVxuICAgICAgLnN0cmVhbSgoZXJyb3I6IEVycm9yLCBzdGRvdXQpOiBhbnkgPT4ge1xuICAgICAgICBpZihlcnJvcikge1xuICAgICAgICAgIGxvZ0Vycm9yKHtcbiAgICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgICAgaXNJbnRlcm5hbDogdHJ1ZSxcbiAgICAgICAgICAgIGxhYmVsOiAnaW1hZ2Vfc2F2ZScsXG4gICAgICAgICAgICB2YWx1ZTogJ2dtX2ltYWdlX3N0cmVhbSdcbiAgICAgICAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBsZXQgaW1hZ2VCdWZmZXI6IEJ1ZmZlciA9IEJ1ZmZlci5mcm9tKCcnKTtcblxuICAgICAgICAgIHN0ZG91dC5vbignZGF0YScsIChkYXRhKSA9PiB7XG4gICAgICAgICAgICBpbWFnZUJ1ZmZlciA9IEJ1ZmZlci5jb25jYXQoW2ltYWdlQnVmZmVyLCBkYXRhXSk7XG4gICAgICAgICAgfSk7XG5cbiAgICAgICAgICBzdGRvdXQub24oJ2VuZCcsICgpID0+IHtcbiAgICAgICAgICAgIC8vIEdldCBmaWxlIHNpemVcbiAgICAgICAgICAgIHBob3RvLmZpbGVTaXplID0gaW1hZ2VCdWZmZXIubGVuZ3RoO1xuXG4gICAgICAgICAgICAvLyBVcGxvYWQgZGF0YVxuICAgICAgICAgICAgY29uc3QgaW1hZ2VPYmo6IFB1dE9iamVjdFJlcXVlc3QgPSB7XG4gICAgICAgICAgICAgIEFDTDogJ2F1dGhlbnRpY2F0ZWQtcmVhZCcsXG4gICAgICAgICAgICAgIEJvZHk6IGltYWdlQnVmZmVyLFxuICAgICAgICAgICAgICBCdWNrZXQ6IG51bGwsXG4gICAgICAgICAgICAgIENvbnRlbnRUeXBlOiB0eXBlLFxuICAgICAgICAgICAgICAuLi5zM09wdGlvbnMgfHwge30sXG4gICAgICAgICAgICAgIEtleTogZ2V0UGF0aFVzZXJJbWFnZXMoc2Vzc2lvbklkLCBpbWFnZUlkLCB0eXBlLCAnaW1hZ2VzJylcbiAgICAgICAgICAgIH07XG5cbiAgICAgICAgICAgIHMzUHV0KGltYWdlT2JqKVxuICAgICAgICAgICAgICAudGhlbigoKSA9PiB7XG4gICAgICAgICAgICAgICAgY29uc3QgdGhtVyA9IENvbmZpZy5nZXQoJ2ltYWdlLnRobVNpemUnKTtcbiAgICAgICAgICAgICAgICBjb25zdCB0aG1RID0gQ29uZmlnLmdldCgnaW1hZ2UudGhtUXVhbGl0eScpO1xuXG4gICAgICAgICAgICAgICAgLy8gVXBsb2FkIHRodW1ibmFpbFxuICAgICAgICAgICAgICAgIGdtKGltYWdlQnVmZmVyLCAnaW1nJylcbiAgICAgICAgICAgICAgICAgIC5zZXRGb3JtYXQoJ2pwZWcnKVxuICAgICAgICAgICAgICAgICAgLnNpemUoKHRodW1iRXJyb3I6IEVycm9yLCByZXNpemVWYWwpID0+IHtcbiAgICAgICAgICAgICAgICAgICAgaWYoIXRodW1iRXJyb3IpIHtcbiAgICAgICAgICAgICAgICAgICAgICAvLyBHZXQgdXBkYXRlZCByZXNpemUgdmFsdWVzXG4gICAgICAgICAgICAgICAgICAgICAgY29uc3Qge2hlaWdodCwgd2lkdGh9ID0gcmVzaXplVmFsO1xuICAgICAgICAgICAgICAgICAgICAgIHBob3RvID0gey4uLmNsb25lRGVlcChwaG90byksIGhlaWdodCwgd2lkdGh9O1xuICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICB9KVxuICAgICAgICAgICAgICAgICAgLmdyYXZpdHkoJ0NlbnRlcicpXG4gICAgICAgICAgICAgICAgICAucmVzaXplKHRobVcsIHRobVcsICdeJylcbiAgICAgICAgICAgICAgICAgIC5leHRlbnQodGhtVywgdGhtVylcbiAgICAgICAgICAgICAgICAgIC5xdWFsaXR5KHRobVEpXG4gICAgICAgICAgICAgICAgICAuc3RyZWFtKChzdHJlYW1FcnJvcjogRXJyb3IsIHRodW1iU3Rkb3V0KTogYW55ID0+IHtcbiAgICAgICAgICAgICAgICAgICAgaWYoc3RyZWFtRXJyb3IpIHtcbiAgICAgICAgICAgICAgICAgICAgICBsb2dFcnJvcih7XG4gICAgICAgICAgICAgICAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgICAgICAgICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgICAgICAgICAgICAgICAgIGxhYmVsOiAnaW1hZ2Vfc2F2ZScsXG4gICAgICAgICAgICAgICAgICAgICAgICB2YWx1ZTogJ2dtX3RodW1ibmFpbF9zdGVhbSdcbiAgICAgICAgICAgICAgICAgICAgICB9LCBzdHJlYW1FcnJvciwgY29udGV4dClcbiAgICAgICAgICAgICAgICAgICAgICAgIC5jYXRjaCgoZXJyb3IpID0+IHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgdGhyb3cgZXJyb3I7XG4gICAgICAgICAgICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICAgICAgICBsZXQgdGh1bWJCdWZmZXI6IEJ1ZmZlciA9IEJ1ZmZlci5mcm9tKCcnKTtcblxuICAgICAgICAgICAgICAgICAgICAgIHRodW1iU3Rkb3V0Lm9uKCdkYXRhJywgKGRhdGEpID0+IHtcbiAgICAgICAgICAgICAgICAgICAgICAgIHRodW1iQnVmZmVyID0gQnVmZmVyLmNvbmNhdChbdGh1bWJCdWZmZXIsIGRhdGFdKTtcbiAgICAgICAgICAgICAgICAgICAgICB9KTtcblxuICAgICAgICAgICAgICAgICAgICAgIHRodW1iU3Rkb3V0Lm9uKCdlbmQnLCAoKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgICAvLyBVcGxvYWQgZGF0YVxuICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgdGh1bWJPYmo6IFB1dE9iamVjdFJlcXVlc3QgPSB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgIEFDTDogJ2F1dGhlbnRpY2F0ZWQtcmVhZCcsXG4gICAgICAgICAgICAgICAgICAgICAgICAgIEJvZHk6IHRodW1iQnVmZmVyLFxuICAgICAgICAgICAgICAgICAgICAgICAgICBCdWNrZXQ6IG51bGwsXG4gICAgICAgICAgICAgICAgICAgICAgICAgIENvbnRlbnRUeXBlOiB0eXBlLFxuICAgICAgICAgICAgICAgICAgICAgICAgICAuLi5zM09wdGlvbnMgfHwge30sXG4gICAgICAgICAgICAgICAgICAgICAgICAgIEtleTogZ2V0UGF0aFVzZXJJbWFnZXMoc2Vzc2lvbklkLCBpbWFnZUlkLCB0eXBlLCAndGh1bWJzJylcbiAgICAgICAgICAgICAgICAgICAgICAgIH07XG5cbiAgICAgICAgICAgICAgICAgICAgICAgIHMzUHV0KHRodW1iT2JqKVxuICAgICAgICAgICAgICAgICAgICAgICAgICAudGhlbigoKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgcmVzb2x2ZShwaG90byk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgIH0pXG4gICAgICAgICAgICAgICAgICAgICAgICAgIC5jYXRjaCgoczNQdXRFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBsYWJlbDogJ2ltYWdlX3NhdmUnLFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIHZhbHVlOiAnczNfcHV0X2ltYWdlJ1xuICAgICAgICAgICAgICAgICAgICAgICAgICB9LCBzM1B1dEVycm9yLCBjb250ZXh0KVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIC5jYXRjaCgoZXJyb3IpID0+IHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRocm93IGVycm9yO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0pKTtcbiAgICAgICAgICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICAgIH0pXG4gICAgICAgICAgICAgIC5jYXRjaCgoczNFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgICAgICAgICBpc0ludGVybmFsOiB0cnVlLFxuICAgICAgICAgICAgICAgIGxhYmVsOiAnaW1hZ2Vfc2F2ZScsXG4gICAgICAgICAgICAgICAgdmFsdWU6ICdzM19pbWFnZV9zYXZlJ1xuICAgICAgICAgICAgICB9LCBzM0Vycm9yLCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpKTtcbiAgICAgICAgICB9KTtcbiAgICAgICAgfVxuICAgICAgfSk7XG4gIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGFkZEltYWdlID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGltYWdlOiBJbWFnZVR5cGUsIHMzT3B0aW9ucz86IFB1dE9iamVjdFJlcXVlc3QpOiBQcm9taXNlPEltYWdlVHlwZT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdhZGRJbWFnZSc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgc2Vzc2lvbjoge3VzZXJJZDogc2Vzc2lvbklkfX0gPSBjb250ZXh0O1xuICBjb25zdCB7aW1hZ2VJZCwgZGVzY3JpcHRpb24sIGJ1ZmZlciwgZmlsZVR5cGV9ID0gaW1hZ2U7XG4gIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcblxuICByZXR1cm4gcmVzaXplU2F2ZUltYWdlKGNvbnRleHQsIGltYWdlSWQsIGJ1ZmZlciwgZmlsZVR5cGUsIHMzT3B0aW9ucylcbiAgICAudGhlbigocmVzaXplZEltYWdlOiBhbnkpID0+IHtcbiAgICAgIGNvbnN0IGluc2VydDogSW1hZ2VUeXBlID0ge1xuICAgICAgICAuLi5yZXNpemVkSW1hZ2UsXG4gICAgICAgIF9rZXk6IGltYWdlSWQsXG4gICAgICAgIGFkZGVkOiBub3csXG4gICAgICAgIGRlc2NyaXB0aW9uLFxuICAgICAgICBmaWxlVHlwZSxcbiAgICAgICAgbW9kaWZpZWQ6IG5vdyxcbiAgICAgICAgdXNlcklkOiBzZXNzaW9uSWRcbiAgICAgIH07XG5cbiAgICAgIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgSU5TRVJUICR7aW5zZXJ0fSBJTiBpbWFnZXMgUkVUVVJOIE5FV2A7XG5cbiAgICAgIHJldHVybiBkYXRhYmFzZS5xdWVyeShhcWxRcnkpXG4gICAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgICAgICAudGhlbihkZWZhdWx0T2JqZWN0KVxuICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgICBpc0ludGVybmFsOiB0cnVlLFxuICAgICAgICAgIGxhYmVsOiAnZGJfZXJyb3InXG4gICAgICAgIH0sIGVycm9yLCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpKTtcbiAgICB9KVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICBhY3Rpb24sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIGlzSW50ZXJuYWw6IHRydWUsXG4gICAgICBsYWJlbDogJ2ltYWdlX3Jlc2l6ZSdcbiAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKSk7XG59O1xuXG5leHBvcnQgY29uc3QgYWRkSW1hZ2VFZGdlID0gYXN5bmMgKGNvbnRleHQ6IEFwaUNvbnRleHQsIGltYWdlRWRnZTogSW1hZ2VFZGdlVHlwZSk6IFByb21pc2U8b2JqZWN0PiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2FkZEltYWdlRWRnZSc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgc2Vzc2lvbjoge3VzZXJJZDogc2Vzc2lvbklkfX0gPSBjb250ZXh0O1xuICBjb25zdCB7aW1hZ2VJZCwgaXRlbUlkLCBpdGVtVHlwZX0gPSBpbWFnZUVkZ2U7XG4gIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcbiAgY29uc3QgZWRnZUNvbGxlY3Rpb246IEVkZ2VDb2xsZWN0aW9uID0gZGF0YWJhc2UuY29sbGVjdGlvbignaGFzSW1hZ2UnKTtcbiAgY29uc3QgZWRnZUlkOiBzdHJpbmcgPSBjcmVhdGVIYXNoKGBoYXNJbWFnZS0ke2ltYWdlSWR9LSR7aXRlbUlkfS0ke3Nlc3Npb25JZH1gKTtcbiAgY29uc3QgZm9ybWF0SXRlbVR5cGU6IHN0cmluZyA9IHBhcnNlQ2hhcihpdGVtVHlwZSkudG9Mb3dlckNhc2UoKTtcbiAgY29uc3QgZm9ybWF0SXRlbUlkOiBzdHJpbmcgPSBwYXJzZUlkKGl0ZW1JZCk7XG4gIGxldCBpdGVtRG9jSWQ6IHN0cmluZztcbiAgY29uc3QgZm9ybWF0SW1hZ2VJZDogc3RyaW5nID0gcGFyc2VJZChpbWFnZUlkKTtcbiAgY29uc3QgZmlsZURvY0lkOiBzdHJpbmcgPSBgaW1hZ2VzLyR7Zm9ybWF0SW1hZ2VJZH1gO1xuXG4gIHN3aXRjaChmb3JtYXRJdGVtVHlwZSkge1xuICAgIGNhc2UgJ2dyb3Vwcyc6XG4gICAgICBpdGVtRG9jSWQgPSBgZ3JvdXBzLyR7Zm9ybWF0SXRlbUlkfWA7XG4gICAgICBicmVhaztcbiAgICBjYXNlICdwb3N0cyc6XG4gICAgICBpdGVtRG9jSWQgPSBgcG9zdHMvJHtmb3JtYXRJdGVtSWR9YDtcbiAgICAgIGJyZWFrO1xuICAgIGNhc2UgJ3VzZXJzJzpcbiAgICBjYXNlICdwcm9maWxlJzpcbiAgICAgIGl0ZW1Eb2NJZCA9IGB1c2Vycy8ke2Zvcm1hdEl0ZW1JZH1gO1xuICAgICAgYnJlYWs7XG4gICAgZGVmYXVsdDpcbiAgICAgIGl0ZW1Eb2NJZCA9ICcnO1xuICAgICAgYnJlYWs7XG4gIH1cbiAgY29uc3QgZWRnZTogYW55ID0ge1xuICAgIF9mcm9tOiBpdGVtRG9jSWQsXG4gICAgX2tleTogZWRnZUlkLFxuICAgIF90bzogZmlsZURvY0lkLFxuICAgIGFkZGVkOiBub3csXG4gICAgdHlwZTogaXRlbVR5cGVcbiAgfTtcblxuICBpZihpdGVtRG9jSWQpIHtcbiAgICByZXR1cm4gZWRnZUNvbGxlY3Rpb24uc2F2ZShlZGdlLCB7cmV0dXJuTmV3OiB0cnVlfSlcbiAgICAgIC50aGVuKChmaWxlRWRnZSkgPT4gZWRnZUNvbGxlY3Rpb24uZG9jdW1lbnQoZmlsZUVkZ2UpKVxuICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgYWN0aW9uLFxuICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgaXNJbnRlcm5hbDogdHJ1ZSxcbiAgICAgICAgbGFiZWw6ICdkYl9lcnJvcicsXG4gICAgICAgIHBhcmFtczoge1xuICAgICAgICAgIGVkZ2UsXG4gICAgICAgICAgZmlsZURvY0lkLFxuICAgICAgICAgIGltYWdlSWQsXG4gICAgICAgICAgaXRlbURvY0lkLFxuICAgICAgICAgIGl0ZW1JZCxcbiAgICAgICAgICBpdGVtVHlwZVxuICAgICAgICB9XG4gICAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKSk7XG4gIH1cblxuICByZXR1cm4ge307XG59O1xuXG5leHBvcnQgY29uc3QgdXBkYXRlSW1hZ2UgPSBhc3luYyAoXG4gIGNvbnRleHQ6IEFwaUNvbnRleHQsXG4gIGltYWdlOiBJbWFnZVR5cGUsXG4gIHMzT3B0aW9ucz86IFB1dE9iamVjdFJlcXVlc3Rcbik6IFByb21pc2U8SW1hZ2VUeXBlPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ3VwZGF0ZUltYWdlJztcbiAgY29uc3Qge2RhdGFiYXNlLCBzZXNzaW9uOiB7dXNlcklkOiBzZXNzaW9uSWR9fSA9IGNvbnRleHQ7XG5cbiAgLy8gSXRlbSBwcm9wc1xuICBjb25zdCB7XG4gICAgYmFzZTY0ID0gJycsXG4gICAgYnVmZmVyOiBpbWFnZUJ1ZmZlcixcbiAgICBkZXNjcmlwdGlvbiA9ICcnLFxuICAgIGltYWdlSWQsXG4gICAgaXRlbUlkLFxuICAgIGl0ZW1UeXBlLFxuICAgIGZpbGVUeXBlLFxuICAgIHVybCA9ICcnXG4gIH0gPSBpbWFnZTtcblxuICAvLyBTYXZlIEJhc2U2NCBkYXRhXG4gIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcbiAgY29uc3QgZm9ybWF0SW1hZ2VJZDogc3RyaW5nID0gaW1hZ2VJZCB8fCBjcmVhdGVIYXNoKGBpbWFnZS0ke3Nlc3Npb25JZH1gKTtcbiAgY29uc3QgZm9ybWF0SXRlbUlkOiBzdHJpbmcgPSBwYXJzZUlkKGl0ZW1JZCk7XG4gIGNvbnN0IGZvcm1hdEl0ZW1UeXBlOiBzdHJpbmcgPSBwYXJzZUNoYXIoaXRlbVR5cGUsIDE2KS50b0xvd2VyQ2FzZSgpO1xuICBjb25zdCBjdXN0b21QYXJhbXM6IEltYWdlVHlwZSA9IHtcbiAgICBkZXNjcmlwdGlvbixcbiAgICBpbWFnZUlkOiBmb3JtYXRJbWFnZUlkLFxuICAgIHVzZXJJZDogc2Vzc2lvbklkXG4gIH07XG5cbiAgaWYoYmFzZTY0IHx8IGltYWdlQnVmZmVyKSB7XG4gICAgbGV0IGJ1ZmZlcjogQnVmZmVyID0gaW1hZ2VCdWZmZXI7XG5cbiAgICAvLyBQYXJzZSBCYXNlNjQgZGF0YVxuICAgIGlmKGJhc2U2NCkge1xuICAgICAgY29uc3QgZm9ybWF0QmFzZTY0OiBzdHJpbmcgPSBiYXNlNjQuc3Vic3RyKGJhc2U2NC5pbmRleE9mKCcsJykgKyAxKTtcbiAgICAgIGJ1ZmZlciA9IEJ1ZmZlci5mcm9tKGZvcm1hdEJhc2U2NCwgJ2Jhc2U2NCcpO1xuICAgIH1cblxuICAgIGxldCB1cGRhdGVkRmlsZVR5cGUgPSBmaWxlVHlwZTtcblxuICAgIGlmKCFmaWxlVHlwZSkge1xuICAgICAgY29uc3QgZmlsZU1pbWU6IEZpbGVUeXBlUmVzdWx0ID0gYXdhaXQgRmlsZVR5cGUuZmlsZVR5cGVGcm9tQnVmZmVyKGJ1ZmZlcik7XG4gICAgICBjb25zdCB7bWltZX0gPSBmaWxlTWltZTtcbiAgICAgIHVwZGF0ZWRGaWxlVHlwZSA9IG1pbWU7XG4gICAgfVxuXG4gICAgY29uc3QgaW1nUGFyYW1zOiBJbWFnZVR5cGUgPSB7XG4gICAgICAuLi5jdXN0b21QYXJhbXMsXG4gICAgICBidWZmZXIsXG4gICAgICBmaWxlVHlwZTogdXBkYXRlZEZpbGVUeXBlXG4gICAgfTtcblxuICAgIHJldHVybiBhZGRJbWFnZShjb250ZXh0LCBpbWdQYXJhbXMsIHMzT3B0aW9ucylcbiAgICAgIC50aGVuKChpbWFnZTogSW1hZ2VUeXBlKSA9PiB7XG4gICAgICAgIGlmKGZvcm1hdEl0ZW1JZCAmJiBmb3JtYXRJdGVtVHlwZSkge1xuICAgICAgICAgIGNvbnN0IGltYWdlRWRnZTogSW1hZ2VFZGdlVHlwZSA9IHtcbiAgICAgICAgICAgIGltYWdlSWQ6IGZvcm1hdEltYWdlSWQsXG4gICAgICAgICAgICBpdGVtSWQ6IGZvcm1hdEl0ZW1JZCxcbiAgICAgICAgICAgIGl0ZW1UeXBlOiBmb3JtYXRJdGVtVHlwZVxuICAgICAgICAgIH07XG4gICAgICAgICAgcmV0dXJuIGFkZEltYWdlRWRnZShjb250ZXh0LCBpbWFnZUVkZ2UpLnRoZW4oKCkgPT4gaW1hZ2UpO1xuICAgICAgICB9XG5cbiAgICAgICAgcmV0dXJuIGltYWdlO1xuICAgICAgfSlcbiAgICAgIC5jYXRjaCgoZXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgYWN0aW9uLFxuICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgbGFiZWw6ICdpbWFnZV9zYXZlX2Vycm9yJ1xuICAgICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCkpO1xuICB9IGVsc2UgaWYodXJsICE9PSAnJykge1xuICAgIC8vIERvd25sb2FkIGltYWdlIGZyb20gdGhlIHdlYlxuICAgIGxldCBjb250ZW50VHlwZTogc3RyaW5nO1xuXG4gICAgcmV0dXJuIGh0dHBHZXQodXJsKVxuICAgICAgLnRoZW4oKHJlczogUmVzcG9uc2UpID0+IHtcbiAgICAgICAgaWYocmVzLnN0YXR1cyAhPT0gMjAwKSB7XG4gICAgICAgICAgcmV0dXJuIGxvZ0V4Y2VwdGlvbih7XG4gICAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgICAgIGxhYmVsOiAnZmV0Y2hfaW1hZ2VfdXJsJyxcbiAgICAgICAgICAgIHZhbHVlOiByZXMuc3RhdHVzVGV4dFxuICAgICAgICAgIH0sIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCk7XG4gICAgICAgIH1cblxuICAgICAgICBjb250ZW50VHlwZSA9IHJlcy5oZWFkZXJzLmdldCgnY29udGVudC10eXBlJyk7XG5cbiAgICAgICAgcmV0dXJuIHJlcztcbiAgICAgIH0pXG4gICAgICAudGhlbigocmVzKSA9PiByZXMuYnVmZmVyKCkpXG4gICAgICAudGhlbigoYnVmZmVyOiBCdWZmZXIpID0+IHtcbiAgICAgICAgY29uc3QgaW1nUGFyYW1zOiBJbWFnZVR5cGUgPSB7XG4gICAgICAgICAgLi4uY3VzdG9tUGFyYW1zLFxuICAgICAgICAgIGJ1ZmZlcixcbiAgICAgICAgICBmaWxlVHlwZTogY29udGVudFR5cGVcbiAgICAgICAgfTtcblxuICAgICAgICByZXR1cm4gYWRkSW1hZ2UoY29udGV4dCwgaW1nUGFyYW1zLCBzM09wdGlvbnMpLnRoZW4oKGltYWdlOiBJbWFnZVR5cGUpID0+IHtcbiAgICAgICAgICBpZihmb3JtYXRJdGVtSWQgJiYgZm9ybWF0SXRlbVR5cGUpIHtcbiAgICAgICAgICAgIGNvbnN0IGltYWdlRWRnZTogSW1hZ2VFZGdlVHlwZSA9IHtpbWFnZUlkOiBmb3JtYXRJbWFnZUlkLCBpdGVtSWQ6IGZvcm1hdEl0ZW1JZCwgaXRlbVR5cGU6IGZvcm1hdEl0ZW1UeXBlfTtcbiAgICAgICAgICAgIHJldHVybiBhZGRJbWFnZUVkZ2UoY29udGV4dCwgaW1hZ2VFZGdlKS50aGVuKCgpID0+IGltYWdlKTtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICByZXR1cm4gaW1hZ2U7XG4gICAgICAgIH0pO1xuICAgICAgfSlcbiAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgIGFjdGlvbixcbiAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgIGxhYmVsOiAnZmV0Y2hfZXJyb3InXG4gICAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKSk7XG4gIH0gZWxzZSBpZihpbWFnZUlkICE9PSAnJykge1xuICAgIC8vIFVwZGF0ZSBtZXRhZGF0YVxuICAgIGNvbnN0IHVwZGF0ZTogYW55ID0ge1xuICAgICAgZGVzY3JpcHRpb24sXG4gICAgICBtb2RpZmllZDogbm93XG4gICAgfTtcbiAgICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYFVQREFURSB7X2tleTogJHtmb3JtYXRJbWFnZUlkfX0gV0lUSCAke3VwZGF0ZX0gSU4gaW1hZ2VzIFJFVFVSTiBORVdgO1xuXG4gICAgcmV0dXJuIGRhdGFiYXNlLnF1ZXJ5KGFxbFFyeSlcbiAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgYWN0aW9uLFxuICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgbGFiZWw6ICdkYl9lcnJvcicsXG4gICAgICAgIHBhcmFtczoge1xuICAgICAgICAgIGFxbFFyeSxcbiAgICAgICAgICBmb3JtYXRJbWFnZUlkLFxuICAgICAgICAgIHVwZGF0ZVxuICAgICAgICB9XG4gICAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKSk7XG4gIH1cblxuICByZXR1cm4gbnVsbDtcbn07XG5cbmV4cG9ydCBjb25zdCBkZWxldGVJbWFnZSA9IGFzeW5jIChjb250ZXh0OiBBcGlDb250ZXh0LCBpbWFnZUlkOiBzdHJpbmcpOiBQcm9taXNlPEltYWdlVHlwZT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdkZWxldGUnO1xuICBjb25zdCB7ZGF0YWJhc2UsIHNlc3Npb246IHt1c2VySWQ6IHNlc3Npb25JZH19ID0gY29udGV4dDtcbiAgY29uc3QgZm9ybWF0SW1hZ2VJZCA9IHBhcnNlSWQoaW1hZ2VJZCk7XG5cbiAgY29uc3QgcmVtb3ZlRWRnZVF1ZXJ5ID0gYXFsYEZPUiBoaSBJTiBoYXNJbWFnZVxuICAgIEZJTFRFUiBoaS5fZnJvbSA9PSAke2Zvcm1hdEltYWdlSWR9XG4gICAgUkVNT1ZFIGhpIElOIGhhc0ltYWdlXG4gICAgUkVUVVJOIE9MRGA7XG5cbiAgYXdhaXQgZGF0YWJhc2UucXVlcnkocmVtb3ZlRWRnZVF1ZXJ5KTtcblxuICBjb25zdCBhcWxRdWVyeSA9IGFxbGBGT1IgaSBJTiBpbWFnZXNcbiAgICBGSUxURVIgaS5fa2V5ID09ICR7Zm9ybWF0SW1hZ2VJZH0gJiYgaS51c2VySWQgPT0gJHtzZXNzaW9uSWR9XG4gICAgUkVNT1ZFIGkgSU4gaW1hZ2VzXG4gICAgUkVUVVJOIE9MRGA7XG5cbiAgY29uc3QgaW1hZ2UgPSBhd2FpdCBkYXRhYmFzZS5xdWVyeShhcWxRdWVyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgYWN0aW9uLFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICBsYWJlbDogJ2RiX2Vycm9yJ1xuICAgIH0sIGVycm9yLCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpKTtcblxuICBpZihpbWFnZSkge1xuICAgIGNvbnN0IHtfa2V5OiBpbWFnZUtleX0gPSBpbWFnZTtcbiAgICBjb25zdCBwYXJhbXM6IERlbGV0ZU9iamVjdHNSZXF1ZXN0ID0ge1xuICAgICAgQnVja2V0OiBudWxsLFxuICAgICAgRGVsZXRlOiB7XG4gICAgICAgIE9iamVjdHM6IFtcbiAgICAgICAgICB7S2V5OiBgdXNlcnMvJHtzZXNzaW9uSWR9L2ltYWdlcy8ke2ltYWdlS2V5fS5qcGdgfSxcbiAgICAgICAgICB7S2V5OiBgdXNlcnMvJHtzZXNzaW9uSWR9L3RodW1icy8ke2ltYWdlS2V5fS5qcGdgfVxuICAgICAgICBdLFxuICAgICAgICBRdWlldDogdHJ1ZVxuICAgICAgfVxuICAgIH07XG5cbiAgICByZXR1cm4gczNEZWxldGVMaXN0KHBhcmFtcykudGhlbigoKSA9PiBpbWFnZSk7XG4gIH1cblxuICByZXR1cm4ge307XG59O1xuIl0sCiAgIm1hcHBpbmdzIjogIjs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUlBLHdCQUE2QjtBQUM3QixtQkFBc0U7QUFDdEUsc0JBQWtCO0FBS2xCLHVCQUF1QztBQUN2QyxnQkFBZTtBQUNmLHVCQUFzQjtBQUN0QixtQkFBdUI7QUFFdkIsb0JBQXFCO0FBYXJCLElBQUFBLGdCQUF3RztBQUN4RyxvQkFBeUM7QUFDekMsSUFBQUMsYUFBa0Q7QUFFbEQsTUFBTSxnQkFBd0I7QUFFdkIsTUFBTSxvQkFB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SyxDQUFDLFdBQXdCLE9BQU8sS0FBSyxDQUFDLEVBQzNDLE1BQU0sQ0FBQyxjQUFpQix3QkFBUztBQUFBLElBQ2hDO0FBQUEsSUFDQSxVQUFVO0FBQUEsSUFDVixPQUFPO0FBQUEsRUFDVCxHQUFHLE9BQU8sT0FBTyxFQUFFLEtBQUssTUFBTSxJQUFJLENBQUM7QUFFckMsTUFBRyxPQUFPO0FBQ1IsVUFBTSxFQUFDLE1BQU0sU0FBUSxJQUFJO0FBQ3pCLFVBQU0sU0FBK0I7QUFBQSxNQUNuQyxRQUFRO0FBQUEsTUFDUixRQUFRO0FBQUEsUUFDTixTQUFTO0FBQUEsVUFDUCxFQUFDLEtBQUssU0FBUyxTQUFTLFdBQVcsUUFBUSxPQUFNO0FBQUEsVUFDakQsRUFBQyxLQUFLLFNBQVMsU0FBUyxXQUFXLFFBQVEsT0FBTTtBQUFBLFFBQ25EO0FBQUEsUUFDQSxPQUFPO0FBQUEsTUFDVDtBQUFBLElBQ0Y7QUFFQSxlQUFPLHlCQUFhLE1BQU0sRUFBRSxLQUFLLE1BQU0sS0FBSztBQUFBLEVBQzlDO0FBRUEsU0FBTyxDQUFDO0FBQ1Y7IiwKICAibmFtZXMiOiBbImltcG9ydF91dGlscyIsICJpbXBvcnRfczMiLCAiZ20iLCAiZXJyb3IiLCAiY2xvbmVEZWVwIiwgIkZpbGVUeXBlIiwgImltYWdlIiwgImh0dHBHZXQiXQp9Cg==
681
+ //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vc3JjL2FjdGlvbnMvaW1hZ2VzLnRzIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyIvKipcbiAqIENvcHlyaWdodCAoYykgMjAxOS1QcmVzZW50LCBOaXRyb2dlbiBMYWJzLCBJbmMuXG4gKiBDb3B5cmlnaHRzIGxpY2Vuc2VkIHVuZGVyIHRoZSBNSVQgTGljZW5zZS4gU2VlIHRoZSBhY2NvbXBhbnlpbmcgTElDRU5TRSBmaWxlIGZvciB0ZXJtcy5cbiAqL1xuaW1wb3J0IHtEZWxldGVPYmplY3RzQ29tbWFuZElucHV0LCBQdXRPYmplY3RDb21tYW5kSW5wdXR9IGZyb20gJ0Bhd3Mtc2RrL2NsaWVudC1zMyc7XG5pbXBvcnQge2dldCBhcyBodHRwR2V0fSBmcm9tICdAbmxhYnMvcmlwLWh1bnRlcic7XG5pbXBvcnQge2NyZWF0ZUhhc2gsIGxvd2VyQ2FzZUtleXMsIHBhcnNlQXJhbmdvSWQsIHBhcnNlQ2hhciwgcGFyc2VJZCwgcGFyc2VOdW19IGZyb20gJ0BubGFicy91dGlscyc7XG5pbXBvcnQge2FxbH0gZnJvbSAnYXJhbmdvanMnO1xuaW1wb3J0IHtBcWxRdWVyeX0gZnJvbSAnYXJhbmdvanMvYXFsJztcbmltcG9ydCB7RWRnZUNvbGxlY3Rpb259IGZyb20gJ2FyYW5nb2pzL2NvbGxlY3Rpb24nO1xuaW1wb3J0IHtBcnJheUN1cnNvcn0gZnJvbSAnYXJhbmdvanMvY3Vyc29yJztcbmltcG9ydCB7TWltZVR5cGV9IGZyb20gJ2ZpbGUtdHlwZSc7XG5pbXBvcnQgZ20gZnJvbSAnZ20nO1xuaW1wb3J0IHtEYXRlVGltZX0gZnJvbSAnbHV4b24nO1xuXG5pbXBvcnQge2dldEdyb3VwRGV0YWlscywgaXNHcm91cGVkfSBmcm9tICcuL2dyb3Vwcyc7XG5pbXBvcnQge3MzRGVsZXRlTGlzdCwgczNHZXRTaWduZWRVcmwsIHMzUHV0fSBmcm9tICcuL3MzJztcbmltcG9ydCB7Q29uZmlnfSBmcm9tICcuLi9jb25maWcnO1xuaW1wb3J0IHR5cGUge1xuICBBcGlDb250ZXh0LFxuICBBcmFuZ29EYkNvbGxlY3Rpb24sXG4gIEFyYW5nb0RiTGltaXQsXG4gIEdyb3VwVHlwZSxcbiAgR3JvdXBVc2VyLFxuICBJbWFnZUVkZ2VUeXBlLFxuICBJbWFnZUlkZW50aWZ5VHlwZSxcbiAgSW1hZ2VPcHRpb25zLFxuICBJbWFnZVR5cGUsXG4gIEltYWdlVXJsRGF0YSxcbiAgUXVlcnlGaWx0ZXJcbn0gZnJvbSAnLi4vdHlwZXMnO1xuaW1wb3J0IHtsb2dFcnJvciwgbG9nRXhjZXB0aW9ufSBmcm9tICcuLi91dGlscy9hbmFseXRpY3NVdGlscyc7XG5pbXBvcnQge0Vycm9yVHlwZXN9IGZyb20gJy4uL3R5cGVzL2Vycm9yJztcbmltcG9ydCB7Z2V0RG9jSWQsIGdldExpbWl0LCBzZWxlY3RSZWFjdGlvbkNvdW50QnlUeXBlfSBmcm9tICcuLi91dGlscy9hcmFuZ29kYlV0aWxzJztcblxuY29uc3QgZXZlbnRDYXRlZ29yeTogc3RyaW5nID0gJ2ltYWdlcyc7XG5nbS5zdWJDbGFzcyh7aW1hZ2VNYWdpY2s6ICc3Kyd9KTtcblxuZXhwb3J0IGNvbnN0IHBhcnNlSW1hZ2VPcHRpb25zID0gKG9wdGlvbnM6IEltYWdlT3B0aW9ucyA9IHt9KSA9PiB7XG4gIGNvbnN0IHtcbiAgICBmcm9tID0gMCxcbiAgICB0byA9IDMwLFxuICAgIHR5cGUgPSAnZGVmYXVsdCdcbiAgfSA9IG9wdGlvbnM7XG5cbiAgcmV0dXJuIHtcbiAgICBsaW1pdDogZ2V0TGltaXQoZnJvbSwgdG8pIGFzIEFyYW5nb0RiTGltaXQsXG4gICAgdHlwZTogcGFyc2VDaGFyKHR5cGUsIDMyKVxuICB9O1xufTtcblxuZXhwb3J0IGNvbnN0IGdldEltYWdlT3B0aW9uYWwgPSAoZmllbGRzOiBzdHJpbmdbXSA9IFtdKSA9PlxuICBmaWVsZHMucmVkdWNlKChzZWxlY3RzOiBhbnksIGZpZWxkOiBzdHJpbmcpID0+IHtcbiAgICBpZihmaWVsZC5pbmNsdWRlcygnQ291bnQnKSkge1xuICAgICAgcmV0dXJuIHNlbGVjdFJlYWN0aW9uQ291bnRCeVR5cGUoJ2ltYWdlcycsICdpJywgZmllbGQsIHNlbGVjdHMpO1xuICAgIH1cblxuICAgIHN3aXRjaChmaWVsZCkge1xuICAgICAgY2FzZSAncmVhY3Rpb25zJzoge1xuICAgICAgICBzZWxlY3RzLnF1ZXJpZXMucHVzaChgTEVUIHJlYWN0aW9ucyA9IChcbiAgICAgICAgICBGT1IgaW1hZ2UsIHIgSU4gSU5CT1VORCBpLl9pZCByZWFjdGlvbnNcbiAgICAgICAgICBDT0xMRUNUIHJlYWN0aW9uTmFtZSA9IHIudmFsdWUgSU5UTyByZWFjdGlvbkl0ZW1zXG4gICAgICAgICAgUkVUVVJOIHtuYW1lOiByZWFjdGlvbk5hbWUsIGNvdW50OiBMRU5HVEgocmVhY3Rpb25JdGVtc1sqXS5yLnZhbHVlKX1cbiAgICAgICAgKWApO1xuICAgICAgICBzZWxlY3RzLm9iamVjdHMucHVzaCgncmVhY3Rpb25zOnJlYWN0aW9ucycpO1xuICAgICAgICByZXR1cm4gc2VsZWN0cztcbiAgICAgIH1cbiAgICAgIGNhc2UgJ3RhZ3MnOiB7XG4gICAgICAgIHNlbGVjdHMucXVlcmllcy5wdXNoKGBMRVQgdGFncyA9IChcbiAgICAgICAgICBGT1IgdCwgcGwgSU4gSU5CT1VORCBpLl9pZCBpc1RhZ2dlZFxuICAgICAgICAgIFJFVFVSTiB0XG4gICAgICAgIClgKTtcbiAgICAgICAgc2VsZWN0cy5vYmplY3RzLnB1c2goJ3RhZ3M6dGFncycpO1xuICAgICAgICByZXR1cm4gc2VsZWN0cztcbiAgICAgIH1cbiAgICAgIGNhc2UgJ3VzZXJzJzoge1xuICAgICAgICBzZWxlY3RzLnF1ZXJpZXMucHVzaChgTEVUIHVzZXJzID0gRklSU1QoXG4gICAgICAgICAgRk9SIHUgSU4gdXNlcnNcbiAgICAgICAgICBGSUxURVIgaS51c2VySWQgPT0gdS5fa2V5XG4gICAgICAgICAgTElNSVQgMVxuICAgICAgICAgIFJFVFVSTiB1XG4gICAgICAgIClgKTtcbiAgICAgICAgc2VsZWN0cy5vYmplY3RzLnB1c2goJ3VzZXJzOnVzZXJzJyk7XG4gICAgICAgIHJldHVybiBzZWxlY3RzO1xuICAgICAgfVxuICAgICAgZGVmYXVsdDoge1xuICAgICAgICByZXR1cm4gc2VsZWN0cztcbiAgICAgIH1cbiAgICB9XG4gIH0sIHtvYmplY3RzOiBbXSwgcXVlcmllczogW119KTtcblxuZXhwb3J0IGNvbnN0IGdldEltYWdlc0J5VXNlciA9IChcbiAgY29udGV4dDogQXBpQ29udGV4dCxcbiAgdXNlcklkOiBzdHJpbmcsXG4gIGZyb206IG51bWJlciA9IDAsXG4gIHRvOiBudW1iZXIgPSAzMFxuKTogUHJvbWlzZTxJbWFnZVR5cGVbXT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRJbWFnZXNCeVVzZXInO1xuICBjb25zdCB7ZGF0YWJhc2V9ID0gY29udGV4dDtcbiAgY29uc3QgZm9ybWF0VXNlcklkOiBzdHJpbmcgPSBwYXJzZUlkKHVzZXJJZCk7XG4gIGNvbnN0IGxpbWl0OiBBcmFuZ29EYkxpbWl0ID0gZ2V0TGltaXQoZnJvbSwgdG8pO1xuICBjb25zdCBhcWxRcnk6IHN0cmluZyA9IGBGT1IgaSBJTiBpbWFnZXNcbiAgICAgIEZJTFRFUiBpLnVzZXJJZCA9PSBcIiR7Zm9ybWF0VXNlcklkfVwiXG4gICAgICBMRVQgdXNlciA9IEZJUlNUKFxuICAgICAgICBGT1IgdSBJTiB1c2Vyc1xuICAgICAgICBGSUxURVIgdS5fa2V5ID09IGkudXNlcklkXG4gICAgICAgIExJTUlUIDFcbiAgICAgICAgUkVUVVJOIHVcbiAgICAgIClcbiAgICAgICR7bGltaXQuYXFsfVxuICAgICAgU09SVCBpLmFkZGVkXG4gICAgICBSRVRVUk4gTUVSR0UoaSwge3VzZXI6dXNlcn0pYDtcblxuICByZXR1cm4gZGF0YWJhc2UucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IuYWxsKCkpXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgdmFsdWU6IEVycm9yVHlwZXMuREFUQUJBU0VfRVJST1IsXG4gICAgICBwYXJhbXM6IHtmcm9tLCB0bywgdXNlcklkfVxuICAgIH0sIGVycm9yLCBjb250ZXh0KSk7XG59O1xuXG5leHBvcnQgY29uc3QgZ2V0SW1hZ2VDb3VudEJ5SXRlbSA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBpdGVtSWQ6IHN0cmluZyk6IFByb21pc2U8bnVtYmVyPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2dldEltYWdlQ291bnRCeUl0ZW0nO1xuICBjb25zdCB7ZGF0YWJhc2V9ID0gY29udGV4dDtcbiAgY29uc3QgZm9ybWF0SXRlbUlkOiBzdHJpbmcgPSBwYXJzZUFyYW5nb0lkKGl0ZW1JZCk7XG4gIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgRk9SIGkgSU4gaGFzSW1hZ2VcbiAgICAgIEZJTFRFUiBpLl90byA9PSAke2Zvcm1hdEl0ZW1JZH1cbiAgICAgIFJFVFVSTiB7Y291bnQ6IENPVU5UKGkpfWA7XG5cbiAgcmV0dXJuIGRhdGFiYXNlLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAudGhlbigoe2NvdW50fSkgPT4gY291bnQpXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgdmFsdWU6IEVycm9yVHlwZXMuREFUQUJBU0VfRVJST1IsXG4gICAgICBwYXJhbXM6IHtpdGVtSWR9XG4gICAgfSwgZXJyb3IsIGNvbnRleHQpKTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRJbWFnZXNCeUl0ZW0gPSBhc3luYyAoXG4gIGNvbnRleHQ6IEFwaUNvbnRleHQsXG4gIGl0ZW1JZDogc3RyaW5nLFxuICBvcHRpb25zOiBJbWFnZU9wdGlvbnMgPSB7fVxuKTogUHJvbWlzZTxJbWFnZVR5cGVbXT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRJbWFnZXNCeUl0ZW0nO1xuICBjb25zdCB7ZGF0YWJhc2UsIGZpZWxkcyA9IFtdfSA9IGNvbnRleHQ7XG4gIGNvbnN0IGZvcm1hdEl0ZW1JZDogc3RyaW5nID0gcGFyc2VBcmFuZ29JZChpdGVtSWQpO1xuICBjb25zdCB7bGltaXR9ID0gcGFyc2VJbWFnZU9wdGlvbnMob3B0aW9ucyk7XG4gIGNvbnN0IHtvYmplY3RzOiBzZWxlY3RPYmplY3RzLCBxdWVyaWVzOiBzZWxlY3RRdWVyaWVzfSA9IGdldEltYWdlT3B0aW9uYWwoZmllbGRzKTtcbiAgY29uc3QgYXFsUXJ5OiBzdHJpbmcgPSBgRk9SIGksIGwgSU4gMS4uMSBPVVRCT1VORCBcIiR7Zm9ybWF0SXRlbUlkfVwiIGhhc0ltYWdlXG4gICAgRklMVEVSIE5PVCBJU19OVUxMKGkpXG4gICAgJHtzZWxlY3RRdWVyaWVzLmpvaW4oJ1xcbicpfVxuICAgIFNPUlQgaS5hZGRlZFxuICAgICR7bGltaXQuYXFsfVxuICAgIFJFVFVSTiBNRVJHRShpLCB7JHtzZWxlY3RPYmplY3RzLmpvaW4oJywgJyl9fSlgO1xuXG4gIHJldHVybiBkYXRhYmFzZS5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5hbGwoKSlcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgYWN0aW9uLFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICB2YWx1ZTogRXJyb3JUeXBlcy5EQVRBQkFTRV9FUlJPUixcbiAgICAgIHBhcmFtczoge2l0ZW1JZCwgb3B0aW9uc31cbiAgICB9LCBlcnJvciwgY29udGV4dCkpO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldEltYWdlc0J5R3JvdXAgPSAoXG4gIGNvbnRleHQ6IEFwaUNvbnRleHQsXG4gIHBhcmFtczoge2ZpbHRlcnM6IGFueVtdLCBncm91cElkOiBzdHJpbmcsIGZyb206IG51bWJlciwgdG86IG51bWJlcn1cbik6IFByb21pc2U8SW1hZ2VUeXBlW10+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnZ2V0SW1hZ2VzQnlHcm91cCc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgc2Vzc2lvbjoge3VzZXJJZDogc2Vzc2lvbklkfX0gPSBjb250ZXh0O1xuICBjb25zdCB7ZmlsdGVycywgZ3JvdXBJZCwgZnJvbSwgdG99ID0gcGFyYW1zO1xuICBjb25zdCBmb3JtYXRHcm91cElkOiBzdHJpbmcgPSBwYXJzZUlkKGdyb3VwSWQpO1xuICBjb25zdCBsaW1pdDogQXJhbmdvRGJMaW1pdCA9IGdldExpbWl0KGZyb20sIHRvKTtcblxuICBmaWx0ZXJzXG4gICAgLm1hcCgoZmlsdGVyOiBRdWVyeUZpbHRlcikgPT4ge1xuICAgICAgY29uc3Qge2NvbmRpdGlvbmFsLCBuYW1lLCB2YWx1ZX0gPSBmaWx0ZXI7XG4gICAgICBsZXQgZm9ybWF0Q29uZDogc3RyaW5nID0gY29uZGl0aW9uYWw7XG5cbiAgICAgIGlmKGNvbmRpdGlvbmFsICE9PSAnPj0nICYmIGNvbmRpdGlvbmFsICE9PSAnPD0nICYmIGNvbmRpdGlvbmFsICE9PSAnPicgJiYgY29uZGl0aW9uYWwgIT09ICc8Jykge1xuICAgICAgICBmb3JtYXRDb25kID0gJz09JztcbiAgICAgIH1cblxuICAgICAgc3dpdGNoKG5hbWUpIHtcbiAgICAgICAgY2FzZSAnYWRkZWQnOlxuICAgICAgICAgIHJldHVybiBgcC5hZGRlZCAke2Zvcm1hdENvbmR9ICR7cGFyc2VOdW0odmFsdWUpfWA7XG4gICAgICAgIGRlZmF1bHQ6XG4gICAgICAgICAgcmV0dXJuICcnO1xuICAgICAgfVxuICAgIH0pO1xuXG4gIHJldHVybiBnZXRHcm91cERldGFpbHMoY29udGV4dCwgZm9ybWF0R3JvdXBJZClcbiAgICAudGhlbigoZ3JvdXA6IEdyb3VwVHlwZSkgPT4ge1xuICAgICAgaWYoZ3JvdXAucHJpdmFjeSA9PT0gJ3B1YmxpYycpIHtcbiAgICAgICAgZmlsdGVycy5wdXNoKGBwLmdyb3VwSWQgPT0gXCIke2Zvcm1hdEdyb3VwSWR9XCJgKTtcbiAgICAgICAgY29uc3QgZmlsdGVyU3RyID0gZmlsdGVycy5qb2luKCcgJiYgJyk7XG4gICAgICAgIGNvbnN0IGFxbFFyeTogc3RyaW5nID0gYEZPUiBpIElOXG4gICAgICAgICAgRkxBVFRFTihcbiAgICAgICAgICAgIEZPUiBwIElOIHBvc3RzXG4gICAgICAgICAgICBGSUxURVIgJHtmaWx0ZXJTdHJ9XG4gICAgICAgICAgICBMRVQgaW1hZ2VzID0gKFxuICAgICAgICAgICAgICBGT1IgaSwgZSBJTiBJTkJPVU5EIHAuX2lkIGhhc0ltYWdlXG4gICAgICAgICAgICAgIFJFVFVSTiBpXG4gICAgICAgICAgICApXG4gICAgICAgICAgICBTT1JUIHAuYWRkZWQgREVTQ1xuICAgICAgICAgICAgUkVUVVJOIGltYWdlc1xuICAgICAgICAgIClcbiAgICAgICAgICBTT1JUIGkuYWRkZWQgREVTQ1xuICAgICAgICAgICR7bGltaXQuYXFsfVxuICAgICAgICAgIFJFVFVSTiBpYDtcblxuICAgICAgICByZXR1cm4gZGF0YWJhc2UucXVlcnkoYXFsUXJ5KVxuICAgICAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IuYWxsKCkpXG4gICAgICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgICAgdmFsdWU6IEVycm9yVHlwZXMuREFUQUJBU0VfRVJST1IsXG4gICAgICAgICAgICBwYXJhbXNcbiAgICAgICAgICB9LCBlcnJvciwgY29udGV4dCkpO1xuICAgICAgfVxuICAgICAgcmV0dXJuIGlzR3JvdXBlZChkYXRhYmFzZSwgc2Vzc2lvbklkLCBncm91cElkKVxuICAgICAgICAudGhlbigoZ3JvdXBlZDogR3JvdXBVc2VyKSA9PiB7XG4gICAgICAgICAgaWYoZ3JvdXBlZC5pc1ZhbGlkKSB7XG4gICAgICAgICAgICBmaWx0ZXJzLnB1c2goYHAuZ3JvdXBJZCA9PSBcIiR7Z3JvdXBlZC5ncm91cElkfVwiYCk7XG4gICAgICAgICAgICBjb25zdCBmaWx0ZXJMaXN0OiBzdHJpbmcgPSBmaWx0ZXJzLmpvaW4oJyAmJiAnKTtcbiAgICAgICAgICAgIGNvbnN0IGFxbFFyeTogc3RyaW5nID0gYEZPUiBwIElOIHBvc3RcbiAgICAgICAgICAgICAgICBGSUxURVIgJHtmaWx0ZXJMaXN0fVxuICAgICAgICAgICAgICAgIEZPUiBmIElOIHAuZmlsZXNcbiAgICAgICAgICAgICAgICBGSUxURVIgZi50eXBlID09IFwiaW1hZ2UvanBlZ1wiIHx8IGYudHlwZSA9PSBcImltYWdlL3BuZ1wiXG4gICAgICAgICAgICAgICAgJHtsaW1pdC5hcWx9XG4gICAgICAgICAgICAgICAgU09SVCBwLmFkZGVkIERFU0NcbiAgICAgICAgICAgICAgICBSRVRVUk4gZmA7XG5cbiAgICAgICAgICAgIHJldHVybiBkYXRhYmFzZS5xdWVyeShhcWxRcnkpXG4gICAgICAgICAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IuYWxsKCkpXG4gICAgICAgICAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgICAgICAgICAgYWN0aW9uLFxuICAgICAgICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgICAgICAgIHZhbHVlOiBFcnJvclR5cGVzLkRBVEFCQVNFX0VSUk9SLFxuICAgICAgICAgICAgICAgIHBhcmFtc1xuICAgICAgICAgICAgICB9LCBlcnJvciwgY29udGV4dCkpO1xuICAgICAgICAgIH1cbiAgICAgICAgICByZXR1cm4gW107XG4gICAgICAgIH0pO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldEltYWdlc0J5UmVhY3Rpb25zID0gKFxuICBjb250ZXh0OiBBcGlDb250ZXh0LFxuICByZWFjdGlvbnM6IHN0cmluZ1tdID0gW10sXG4gIG9wdGlvbnM/OiBJbWFnZU9wdGlvbnNcbik6IFByb21pc2U8SW1hZ2VUeXBlW10+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnZ2V0VXNlcnNCeUltYWdlJztcbiAgY29uc3Qge2RhdGFiYXNlLCBmaWVsZHMgPSBbXSwgc2Vzc2lvbjoge3VzZXJJZDogc2Vzc2lvbklkfX0gPSBjb250ZXh0O1xuICBjb25zdCB7bGltaXR9ID0gcGFyc2VJbWFnZU9wdGlvbnMob3B0aW9ucyk7XG4gIGNvbnN0IHtvYmplY3RzOiBzZWxlY3RPYmplY3RzLCBxdWVyaWVzOiBzZWxlY3RRdWVyaWVzfSA9IGdldEltYWdlT3B0aW9uYWwoZmllbGRzKTtcbiAgY29uc3QgZm9ybWF0U2Vzc2lvbklkOiBzdHJpbmcgPSBgdXNlcnMvJHtzZXNzaW9uSWR9YDtcbiAgY29uc3QgZm9ybWF0UmVhY3Rpb25zOiBzdHJpbmdbXSA9IHJlYWN0aW9ucy5tYXAoKHJlYWN0aW9uTmFtZSkgPT4gcGFyc2VDaGFyKHJlYWN0aW9uTmFtZSwgMzIpKTtcbiAgY29uc3QgZmlsdGVyQnk6IHN0cmluZ1tdID0gW1xuICAgICdyLnR5cGUgPT0gXCJpbWFnZXNcIicsXG4gICAgYFBPU0lUSU9OKCR7SlNPTi5zdHJpbmdpZnkoZm9ybWF0UmVhY3Rpb25zKX0sIExPV0VSKHIubmFtZSkpYF07XG5cbiAgLy8gR2V0IGRhdGEgZnJvbSBkYXRhYmFzZVxuICBjb25zdCBhcWxRcnk6IHN0cmluZyA9IGBGT1IgaSwgciBJTiBPVVRCT1VORCBcIiR7Zm9ybWF0U2Vzc2lvbklkfVwiIGhhc1JlYWN0aW9uXG4gICAgT1BUSU9OUyB7dmVydGV4Q29sbGVjdGlvbnM6IFwiaW1hZ2VzXCJ9XG4gICAgJHtzZWxlY3RRdWVyaWVzLmpvaW4oJ1xcbicpfVxuICAgIEZJTFRFUiAke2ZpbHRlckJ5LmpvaW4oJyAmJiAnKX1cbiAgICAke2xpbWl0LmFxbH1cbiAgICBSRVRVUk4gTUVSR0UoaSwgeyR7c2VsZWN0T2JqZWN0cy5qb2luKCcsICcpfX0pYDtcblxuICByZXR1cm4gZGF0YWJhc2UucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IuYWxsKCkpXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgdmFsdWU6IEVycm9yVHlwZXMuREFUQUJBU0VfRVJST1IsXG4gICAgICBwYXJhbXM6IHtyZWFjdGlvbnMsIG9wdGlvbnN9XG4gICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gW10pKTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRJbWFnZSA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBpbWFnZUlkOiBzdHJpbmcpOiBQcm9taXNlPEltYWdlVHlwZT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRJdGVtJztcbiAgY29uc3Qge2RhdGFiYXNlfSA9IGNvbnRleHQ7XG4gIGNvbnN0IGZvcm1hdEltYWdlSWQ6IHN0cmluZyA9IHBhcnNlSWQoaW1hZ2VJZCk7XG4gIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgRk9SIGkgSU4gaW1hZ2VzXG4gICAgRklMVEVSIGkuX2tleT09JHtmb3JtYXRJbWFnZUlkfVxuICAgIExJTUlUIDFcbiAgICBSRVRVUk4gaWA7XG5cbiAgcmV0dXJuIGRhdGFiYXNlLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgYWN0aW9uLFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICB2YWx1ZTogRXJyb3JUeXBlcy5EQVRBQkFTRV9FUlJPUixcbiAgICAgIHBhcmFtczoge2ltYWdlSWR9XG4gICAgfSwgZXJyb3IsIGNvbnRleHQpKTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRQYXRoVXNlckltYWdlcyA9ICh1c2VySWQ6IHN0cmluZywgaW1hZ2VJZDogc3RyaW5nLCB0eXBlOiBzdHJpbmcsIGRpcjogc3RyaW5nID0gJ2ltYWdlcycpOiBzdHJpbmcgPT4ge1xuICBsZXQgZmlsZW5hbWU6IHN0cmluZyA9IGltYWdlSWQ7XG5cbiAgc3dpdGNoKHR5cGUpIHtcbiAgICBjYXNlICdpbWFnZS9wbmcnOlxuICAgICAgZmlsZW5hbWUgPSBgJHtpbWFnZUlkfS5wbmdgO1xuICAgICAgYnJlYWs7XG4gICAgZGVmYXVsdDpcbiAgICAgIGZpbGVuYW1lID0gYCR7aW1hZ2VJZH0uanBnYDtcbiAgICAgIGJyZWFrO1xuICB9XG5cbiAgcmV0dXJuIGB1c2Vycy8ke3VzZXJJZH0vJHtkaXJ9LyR7ZmlsZW5hbWV9YDtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRBcHBJbWFnZVVybCA9IChkYXRhOiBJbWFnZVVybERhdGEpOiBzdHJpbmcgPT4ge1xuICBjb25zdCB7aW1hZ2VJZCwgZGlyZWN0b3J5ID0gJ2ltYWdlcycsIGltYWdlVHlwZSA9ICdwcm9maWxlJywgaXNUaHVtYiwgdHlwZSwgdHlwZUlkfSA9IGRhdGE7XG4gIGNvbnN0IGhvc3Q6IHN0cmluZyA9IENvbmZpZy5nZXQoJ2Vudmlyb25tZW50JykgPT09ICdwcm9kJ1xuICAgID8gYGh0dHBzOi8vYm94LiR7Q29uZmlnLmdldCgnYXBwLnVybCcpfWBcbiAgICA6IGBodHRwczovL3MzLmFtYXpvbmF3cy5jb20vZGV2LiR7Q29uZmlnLmdldCgnYXBwLnVybCcpfWA7XG4gIGNvbnN0IHN1ZmZpeDogc3RyaW5nID0gaXNUaHVtYiA/ICctdGgnIDogJyc7XG5cbiAgaWYoaW1hZ2VJZCkge1xuICAgIHN3aXRjaCh0eXBlKSB7XG4gICAgICBjYXNlICdhcHBzJzpcbiAgICAgICAgLy8gaHR0cHM6Ly9ib3gucmVha3Rvci5pby9teUFwcC9hcHAvaW1hZ2VzLzEyMy5qcGdcbiAgICAgICAgcmV0dXJuIGAke2hvc3R9LyR7dHlwZX0vJHtkaXJlY3Rvcnl9LyR7aW1hZ2VJZH0ke3N1ZmZpeH0uanBnYDtcbiAgICAgIGNhc2UgJ3VzZXJzJzpcbiAgICAgICAgLy8gaHR0cHM6Ly9ib3gucmVha3Rvci5pby9teUFwcC91c2Vycy9kZW1vVXNlci9pbWFnZXMvMTIzLmpwZ1xuICAgICAgICByZXR1cm4gYCR7aG9zdH0vJHt0eXBlfS8ke3R5cGVJZH0vJHtkaXJlY3Rvcnl9LyR7aW1hZ2VJZH0ke3N1ZmZpeH0uanBnYDtcbiAgICB9XG5cbiAgICBpZihpbWFnZVR5cGUgPT09ICdwcm9maWxlJykge1xuICAgICAgcmV0dXJuIGAke2hvc3R9L2RlZmF1bHRzLyR7dHlwZX1fYmske3N1ZmZpeH0uanBnYDtcbiAgICB9XG5cbiAgICByZXR1cm4gYCR7aG9zdH0vZGVmYXVsdHMvJHt0eXBlfV93aCR7c3VmZml4fS5qcGdgO1xuICB9XG5cbiAgcmV0dXJuICcnO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldEltYWdlVXJsID0gYXN5bmMgKGRhdGE6IEltYWdlVXJsRGF0YSk6IFByb21pc2U8c3RyaW5nPiA9PiB7XG4gIGNvbnN0IHtidWNrZXQsIGltYWdlSWQsIGlzVGh1bWIgPSBmYWxzZSwgdHlwZSwgdHlwZUlkfSA9IGRhdGE7XG4gIGNvbnN0IGltZ0Rpcjogc3RyaW5nID0gaXNUaHVtYiA/ICd0aHVtYnMnIDogJ2ltYWdlcyc7XG5cbiAgaWYoaW1hZ2VJZCkge1xuICAgIGNvbnN0IGltYWdlS2V5OiBzdHJpbmcgPSBgJHt0eXBlfS8ke3R5cGVJZH0vJHtpbWdEaXJ9LyR7aW1hZ2VJZH0uanBnYDtcbiAgICByZXR1cm4gYXdhaXQgczNHZXRTaWduZWRVcmwoe0J1Y2tldDogYnVja2V0LCBLZXk6IGltYWdlS2V5LCBFeHBpcmVzOiA5MDB9KTtcbiAgfVxuXG4gIHJldHVybiAnJztcbn07XG5cbmV4cG9ydCBjb25zdCByZXNpemVTYXZlSW1hZ2UgPSBhc3luYyAoXG4gIGNvbnRleHQ6IEFwaUNvbnRleHQsXG4gIGltYWdlSWQ6IHN0cmluZyxcbiAgYnVmZmVyOiBCdWZmZXIsXG4gIGZpbGVUeXBlOiBzdHJpbmcgPSAnaW1hZ2UvanBlZycsIHMzT3B0aW9ucz86IFB1dE9iamVjdENvbW1hbmRJbnB1dFxuKTogUHJvbWlzZTxJbWFnZVR5cGU+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAncmVzaXplU2F2ZUltYWdlJztcbiAgY29uc3Qge3Nlc3Npb246IHt1c2VySWQ6IHNlc3Npb25JZH19ID0gY29udGV4dDtcbiAgY29uc3QgaW1nVzogbnVtYmVyID0gQ29uZmlnLmdldCgnaW1hZ2UuaW1nU2l6ZScpO1xuICBjb25zdCBpbWdROiBudW1iZXIgPSBDb25maWcuZ2V0KCdpbWFnZS5pbWdRdWFsaXR5Jyk7XG4gIGNvbnN0IGZvcm1hdDogc3RyaW5nID0gKGZpbGVUeXBlLnNwbGl0KCcvJykpWzFdO1xuXG4gIGNvbnN0IGltYWdlT3B0aW1pemVkQnVmZmVyOiBCdWZmZXIgPSBhd2FpdCBuZXcgUHJvbWlzZSgocmVzb2x2ZSwgcmVqZWN0KSA9PiB7XG4gICAgZ20oYnVmZmVyLCAnaW1nJylcbiAgICAgIC5zZXRGb3JtYXQoZm9ybWF0KVxuICAgICAgLnF1YWxpdHkoaW1nUSlcbiAgICAgIC5hdXRvT3JpZW50KClcbiAgICAgIC5yZXNpemUoaW1nVywgaW1nVywgJz4nKVxuICAgICAgLnN0cmVhbSgoc3RyZWFtRXJyb3I6IEVycm9yLCBzdGRvdXQpOiB2b2lkID0+IHtcbiAgICAgICAgaWYoc3RyZWFtRXJyb3IpIHtcbiAgICAgICAgICByZWplY3QoXG4gICAgICAgICAgICBsb2dFcnJvcih7XG4gICAgICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgICAgIHBhcmFtczoge2ltYWdlSWQsIGZpbGVUeXBlfSxcbiAgICAgICAgICAgICAgdmFsdWU6IEVycm9yVHlwZXMuSU1BR0VfU0FWRVxuICAgICAgICAgICAgfSwgc3RyZWFtRXJyb3IsIGNvbnRleHQpXG4gICAgICAgICAgKTtcbiAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cblxuICAgICAgICBsZXQgaW1hZ2VCdWZmZXI6IEJ1ZmZlciA9IEJ1ZmZlci5mcm9tKCcnKTtcblxuICAgICAgICBzdGRvdXQub24oJ2RhdGEnLCAoZGF0YSkgPT4ge1xuICAgICAgICAgIGltYWdlQnVmZmVyID0gQnVmZmVyLmNvbmNhdChbaW1hZ2VCdWZmZXIsIGRhdGFdKTtcbiAgICAgICAgfSk7XG5cbiAgICAgICAgc3Rkb3V0Lm9uKCdlbmQnLCAoKSA9PiByZXNvbHZlKGltYWdlQnVmZmVyKSk7XG4gICAgICB9KTtcbiAgfSk7XG5cbiAgdHJ5IHtcbiAgICBjb25zdCBpbWFnZU9iajogUHV0T2JqZWN0Q29tbWFuZElucHV0ID0ge1xuICAgICAgQUNMOiAnYXV0aGVudGljYXRlZC1yZWFkJyxcbiAgICAgIEJvZHk6IGltYWdlT3B0aW1pemVkQnVmZmVyLFxuICAgICAgQnVja2V0OiBudWxsLFxuICAgICAgQ29udGVudFR5cGU6IGZpbGVUeXBlLFxuICAgICAgLi4uKHMzT3B0aW9ucyB8fCB7fSksXG4gICAgICBLZXk6IGdldFBhdGhVc2VySW1hZ2VzKHNlc3Npb25JZCwgaW1hZ2VJZCwgZmlsZVR5cGUsICdpbWFnZXMnKVxuICAgIH07XG4gICAgYXdhaXQgczNQdXQoaW1hZ2VPYmopO1xuICB9IGNhdGNoKGVycm9yKSB7XG4gICAgcmV0dXJuIGxvZ0Vycm9yKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgcGFyYW1zOiB7aW1hZ2VJZCwgZmlsZVR5cGV9LFxuICAgICAgdmFsdWU6IEVycm9yVHlwZXMuSU1BR0VfU0FWRVxuICAgIH0sIGVycm9yLCBjb250ZXh0KTtcbiAgfVxuXG4gIGNvbnN0IGltYWdlSWRlbnRpdHkgPSBhd2FpdCBuZXcgUHJvbWlzZSgocmVzb2x2ZSkgPT4ge1xuICAgIGdtKGltYWdlT3B0aW1pemVkQnVmZmVyLCAnaW1nJylcbiAgICAgIC5pZGVudGlmeSh7YnVmZmVyU3RyZWFtOiB0cnVlfSwgKGVycm9yOiBFcnJvciwgaW1hZ2VEYXRhOiBJbWFnZUlkZW50aWZ5VHlwZSA9IHt9KTogYW55ID0+IHtcbiAgICAgICAgaWYoZXJyb3IpIHtcbiAgICAgICAgICByZXR1cm4gbG9nRXJyb3Ioe1xuICAgICAgICAgICAgYWN0aW9uLFxuICAgICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgICB2YWx1ZTogRXJyb3JUeXBlcy5JTUFHRV9TQVZFXG4gICAgICAgICAgfSwgZXJyb3IsIGNvbnRleHQpO1xuICAgICAgICB9XG5cbiAgICAgICAgY29uc3QgZm9ybWF0VmFsdWVzID0gbG93ZXJDYXNlS2V5cyhpbWFnZURhdGEpO1xuICAgICAgICBjb25zdCB7bWFrZTogY2FtZXJhTWFrZSwgbW9kZWw6IGNhbWVyYU1vZGVsLCBkYXRldGltZW9yaWdpbmFsOiB0YWtlbn06IEltYWdlSWRlbnRpZnlUeXBlID0gZm9ybWF0VmFsdWVzO1xuICAgICAgICBjb25zdCBzdGF0cyA9IGZvcm1hdFZhbHVlc1snY2hhbm5lbCBzdGF0aXN0aWNzJ107XG4gICAgICAgIGxldCBjb2xvcjogc3RyaW5nO1xuXG4gICAgICAgIGlmKHN0YXRzKSB7XG4gICAgICAgICAgbGV0IHtyZWQsIGdyZWVuLCBibHVlLCBtZWFufSA9IHN0YXRzO1xuXG4gICAgICAgICAgaWYocmVkKSB7XG4gICAgICAgICAgICBtZWFuID0gcmVkWydzdGFuZGFyZCBkZXZpYXRpb24nXSB8fCByZWQubWVhbjtcbiAgICAgICAgICAgIHJlZCA9IG1lYW4gPyArKChtZWFuLnNwbGl0KCcgJylbMF0pLnN1YnN0cmluZygwLCAzKSkgOiAwO1xuICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICByZWQgPSAwO1xuICAgICAgICAgIH1cblxuICAgICAgICAgIGlmKGdyZWVuKSB7XG4gICAgICAgICAgICBtZWFuID0gZ3JlZW5bJ3N0YW5kYXJkIGRldmlhdGlvbiddIHx8IGdyZWVuLm1lYW47XG4gICAgICAgICAgICBncmVlbiA9IG1lYW4gPyArKChtZWFuLnNwbGl0KCcgJylbMF0pLnN1YnN0cmluZygwLCAzKSkgOiAwO1xuICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICBncmVlbiA9IDA7XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgaWYoYmx1ZSkge1xuICAgICAgICAgICAgbWVhbiA9IGJsdWVbJ3N0YW5kYXJkIGRldmlhdGlvbiddIHx8IGJsdWUubWVhbjtcbiAgICAgICAgICAgIGJsdWUgPSBtZWFuID8gKygobWVhbi5zcGxpdCgnICcpWzBdKS5zdWJzdHJpbmcoMCwgMykpIDogMDtcbiAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgYmx1ZSA9IDA7XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgY29uc3QgcmdiID0gYmx1ZSB8IChncmVlbiA8PCA4KSB8IChyZWQgPDwgMTYpO1xuICAgICAgICAgIGNvbnN0IHJnYkNvbG9yID0gcmdiLnRvU3RyaW5nKDE2KTtcbiAgICAgICAgICBjb2xvciA9IHJnYkNvbG9yID09PSAnMCcgPyAnMDAwMDAwJyA6IHJnYkNvbG9yO1xuICAgICAgICB9XG5cbiAgICAgICAgY29uc3Qge1xuICAgICAgICAgICdKUEVHLVF1YWxpdHknOiBxdWFsaXR5LFxuICAgICAgICAgIE9yaWVudGF0aW9uOiBvcmllbnRhdGlvbkRhdGEsXG4gICAgICAgICAgUmVzb2x1dGlvbjogcmVzb2x1dGlvbkRhdGEsXG4gICAgICAgICAgc2l6ZVxuICAgICAgICB9ID0gaW1hZ2VEYXRhO1xuICAgICAgICBsZXQgcmVzb2x1dGlvbjtcblxuICAgICAgICBpZihyZXNvbHV0aW9uRGF0YSkge1xuICAgICAgICAgIGNvbnN0IFtyZXNvbHV0aW9uTnVtYmVyLCByZXNvbHV0aW9uVW5pdF0gPSByZXNvbHV0aW9uRGF0YS5zcGxpdCgnICcpO1xuICAgICAgICAgIGNvbnN0IHJlc29sdXRpb25WYWx1ZSA9IHJlc29sdXRpb25OdW1iZXIuc3BsaXQoJ3gnKVswXTtcbiAgICAgICAgICBjb25zdCByZXNvbHV0aW9uVW5pdFZhbHVlID0gcmVzb2x1dGlvblVuaXQgPT09ICdwaXhlbHMvaW5jaCcgPyAncHBpJyA6ICcnO1xuICAgICAgICAgIHJlc29sdXRpb24gPSBgJHtyZXNvbHV0aW9uVmFsdWV9ICR7cmVzb2x1dGlvblVuaXRWYWx1ZX1gO1xuICAgICAgICB9XG5cbiAgICAgICAgcmV0dXJuIHJlc29sdmUoe1xuICAgICAgICAgIGNvbG9yLFxuICAgICAgICAgIGZpbGVTaXplOiBpbWFnZU9wdGltaXplZEJ1ZmZlci5sZW5ndGgsXG4gICAgICAgICAgaGVpZ2h0OiBzaXplPy5oZWlnaHQgfHwgMCxcbiAgICAgICAgICBtYWtlOiBjYW1lcmFNYWtlLFxuICAgICAgICAgIG1vZGVsOiBjYW1lcmFNb2RlbCxcbiAgICAgICAgICBvcmllbnRhdGlvbjogb3JpZW50YXRpb25EYXRhID09PSAnVW5rbm93bicgPyB1bmRlZmluZWQgOiBvcmllbnRhdGlvbkRhdGE/LnRvTG93ZXJDYXNlKCksXG4gICAgICAgICAgcXVhbGl0eTogcXVhbGl0eSA/ICtxdWFsaXR5IDogdW5kZWZpbmVkLFxuICAgICAgICAgIHJlc29sdXRpb24sXG4gICAgICAgICAgdGFrZW46IHRha2VuID8gRGF0ZVRpbWUuZnJvbU1pbGxpcyh0YWtlbikubWlsbGlzZWNvbmQgOiB1bmRlZmluZWQsXG4gICAgICAgICAgd2lkdGg6IHNpemU/LndpZHRoIHx8IDBcbiAgICAgICAgfSk7XG4gICAgICB9KTtcbiAgfSk7XG5cbiAgY29uc3QgdGh1bWJPcHRpbWl6ZWRCdWZmZXI6IEJ1ZmZlciA9IGF3YWl0IG5ldyBQcm9taXNlKChyZXNvbHZlLCByZWplY3QpID0+IHtcbiAgICBjb25zdCB0aG1XID0gQ29uZmlnLmdldCgnaW1hZ2UudGhtU2l6ZScpO1xuICAgIGNvbnN0IHRobVEgPSBDb25maWcuZ2V0KCdpbWFnZS50aG1RdWFsaXR5Jyk7XG5cbiAgICBnbShpbWFnZU9wdGltaXplZEJ1ZmZlciwgJ2ltZycpXG4gICAgICAuc2V0Rm9ybWF0KGZvcm1hdClcbiAgICAgIC5ncmF2aXR5KCdDZW50ZXInKVxuICAgICAgLnJlc2l6ZSh0aG1XLCB0aG1XLCAnXicpXG4gICAgICAuZXh0ZW50KHRobVcsIHRobVcpXG4gICAgICAucXVhbGl0eSh0aG1RKVxuICAgICAgLnN0cmVhbSgoc3RyZWFtRXJyb3I6IEVycm9yLCB0aHVtYlN0ZG91dCk6IHZvaWQgPT4ge1xuICAgICAgICBpZihzdHJlYW1FcnJvcikge1xuICAgICAgICAgIGxvZ0Vycm9yKHtcbiAgICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgICAgcGFyYW1zOiB7aW1hZ2VJZCwgZmlsZVR5cGV9LFxuICAgICAgICAgICAgdmFsdWU6IEVycm9yVHlwZXMuSU1BR0VfU0FWRVxuICAgICAgICAgIH0sIHN0cmVhbUVycm9yLCBjb250ZXh0KTtcbiAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cblxuICAgICAgICBsZXQgdGh1bWJCdWZmZXI6IEJ1ZmZlciA9IEJ1ZmZlci5mcm9tKCcnKTtcblxuICAgICAgICB0aHVtYlN0ZG91dC5vbignZGF0YScsIChkYXRhKSA9PiB7XG4gICAgICAgICAgdGh1bWJCdWZmZXIgPSBCdWZmZXIuY29uY2F0KFt0aHVtYkJ1ZmZlciwgZGF0YV0pO1xuICAgICAgICB9KTtcblxuICAgICAgICB0aHVtYlN0ZG91dC5vbignZW5kJywgKCkgPT4gcmVzb2x2ZSh0aHVtYkJ1ZmZlcikpO1xuICAgICAgfSk7XG4gIH0pO1xuXG4gIHRyeSB7XG4gICAgY29uc3QgdGh1bWJPYmo6IFB1dE9iamVjdENvbW1hbmRJbnB1dCA9IHtcbiAgICAgIEFDTDogJ2F1dGhlbnRpY2F0ZWQtcmVhZCcsXG4gICAgICBCb2R5OiB0aHVtYk9wdGltaXplZEJ1ZmZlcixcbiAgICAgIEJ1Y2tldDogbnVsbCxcbiAgICAgIENvbnRlbnRUeXBlOiBmaWxlVHlwZSxcbiAgICAgIC4uLihzM09wdGlvbnMgfHwge30pLFxuICAgICAgS2V5OiBnZXRQYXRoVXNlckltYWdlcyhzZXNzaW9uSWQsIGltYWdlSWQsIGZpbGVUeXBlLCAndGh1bWJzJylcbiAgICB9O1xuICAgIGF3YWl0IHMzUHV0KHRodW1iT2JqKTtcbiAgfSBjYXRjaChlcnJvcikge1xuICAgIHJldHVybiBsb2dFcnJvcih7XG4gICAgICBhY3Rpb24sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIHZhbHVlOiBFcnJvclR5cGVzLklNQUdFX1NBVkVcbiAgICB9LCBlcnJvciwgY29udGV4dCk7XG4gIH1cblxuICByZXR1cm4gaW1hZ2VJZGVudGl0eTtcbn07XG5cbmV4cG9ydCBjb25zdCBhZGRJbWFnZSA9IChcbiAgY29udGV4dDogQXBpQ29udGV4dCxcbiAgaW1hZ2U6IEltYWdlVHlwZSxcbiAgczNPcHRpb25zPzogUHV0T2JqZWN0Q29tbWFuZElucHV0XG4pOiBQcm9taXNlPEltYWdlVHlwZT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdhZGRJbWFnZSc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgc2Vzc2lvbjoge3VzZXJJZDogc2Vzc2lvbklkfX0gPSBjb250ZXh0O1xuICBjb25zdCB7aW1hZ2VJZCwgZGVzY3JpcHRpb24sIGJ1ZmZlciwgZmlsZVR5cGV9ID0gaW1hZ2U7XG4gIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcblxuICByZXR1cm4gcmVzaXplU2F2ZUltYWdlKGNvbnRleHQsIGltYWdlSWQsIGJ1ZmZlciwgZmlsZVR5cGUsIHMzT3B0aW9ucylcbiAgICAudGhlbigocmVzaXplZEltYWdlOiBhbnkpID0+IHtcbiAgICAgIGNvbnN0IGluc2VydDogSW1hZ2VUeXBlID0ge1xuICAgICAgICAuLi5yZXNpemVkSW1hZ2UsXG4gICAgICAgIF9rZXk6IGltYWdlSWQsXG4gICAgICAgIGFkZGVkOiBub3csXG4gICAgICAgIGRlc2NyaXB0aW9uLFxuICAgICAgICBmaWxlVHlwZSxcbiAgICAgICAgbW9kaWZpZWQ6IG5vdyxcbiAgICAgICAgdXNlcklkOiBzZXNzaW9uSWRcbiAgICAgIH07XG5cbiAgICAgIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgSU5TRVJUICR7aW5zZXJ0fSBJTiBpbWFnZXMgUkVUVVJOIE5FV2A7XG5cbiAgICAgIHJldHVybiBkYXRhYmFzZS5xdWVyeShhcWxRcnkpXG4gICAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgICBwYXJhbXM6IHtpbWFnZSwgczNPcHRpb25zfSxcbiAgICAgICAgICB2YWx1ZTogRXJyb3JUeXBlcy5EQVRBQkFTRV9FUlJPUlxuICAgICAgICB9LCBlcnJvciwgY29udGV4dCkpO1xuICAgIH0pXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgdmFsdWU6IEVycm9yVHlwZXMuSU1BR0VfUkVTSVpFXG4gICAgfSwgZXJyb3IsIGNvbnRleHQpKTtcbn07XG5cbmV4cG9ydCBjb25zdCBhZGRJbWFnZUVkZ2UgPSBhc3luYyAoY29udGV4dDogQXBpQ29udGV4dCwgaW1hZ2VFZGdlOiBJbWFnZUVkZ2VUeXBlKTogUHJvbWlzZTxvYmplY3Q+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnYWRkSW1hZ2VFZGdlJztcbiAgY29uc3Qge2RhdGFiYXNlLCBzZXNzaW9uOiB7dXNlcklkOiBzZXNzaW9uSWR9fSA9IGNvbnRleHQ7XG4gIGNvbnN0IHtpbWFnZUlkLCBpdGVtSWQsIGl0ZW1UeXBlfSA9IGltYWdlRWRnZTtcbiAgY29uc3Qgbm93OiBudW1iZXIgPSBEYXRlLm5vdygpO1xuICBjb25zdCBlZGdlQ29sbGVjdGlvbjogRWRnZUNvbGxlY3Rpb24gPSBkYXRhYmFzZS5jb2xsZWN0aW9uKCdoYXNJbWFnZScpO1xuICBjb25zdCBlZGdlSWQ6IHN0cmluZyA9IGNyZWF0ZUhhc2goYGhhc0ltYWdlLSR7aW1hZ2VJZH0tJHtpdGVtSWR9LSR7c2Vzc2lvbklkfWApO1xuICBjb25zdCBmb3JtYXRJdGVtVHlwZTogQXJhbmdvRGJDb2xsZWN0aW9uID0gcGFyc2VDaGFyKGl0ZW1UeXBlKS50b0xvd2VyQ2FzZSgpIGFzIEFyYW5nb0RiQ29sbGVjdGlvbjtcbiAgY29uc3QgZm9ybWF0SXRlbUlkOiBzdHJpbmcgPSBwYXJzZUlkKGl0ZW1JZCk7XG4gIGNvbnN0IGl0ZW1Eb2NJZDogc3RyaW5nID0gZ2V0RG9jSWQoZm9ybWF0SXRlbVR5cGUsIHtfa2V5OiBmb3JtYXRJdGVtSWR9KTtcbiAgY29uc3QgZm9ybWF0SW1hZ2VJZDogc3RyaW5nID0gcGFyc2VJZChpbWFnZUlkKTtcbiAgY29uc3QgZmlsZURvY0lkOiBzdHJpbmcgPSBgaW1hZ2VzLyR7Zm9ybWF0SW1hZ2VJZH1gO1xuXG4gIGNvbnN0IGVkZ2U6IGFueSA9IHtcbiAgICBfZnJvbTogaXRlbURvY0lkLFxuICAgIF9rZXk6IGVkZ2VJZCxcbiAgICBfdG86IGZpbGVEb2NJZCxcbiAgICBhZGRlZDogbm93LFxuICAgIHR5cGU6IGl0ZW1UeXBlXG4gIH07XG5cbiAgaWYoaXRlbURvY0lkKSB7XG4gICAgcmV0dXJuIGVkZ2VDb2xsZWN0aW9uLnNhdmUoZWRnZSwge3JldHVybk5ldzogdHJ1ZX0pXG4gICAgICAudGhlbigoZmlsZUVkZ2UpID0+IGVkZ2VDb2xsZWN0aW9uLmRvY3VtZW50KGZpbGVFZGdlKSlcbiAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgIGFjdGlvbixcbiAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgIHBhcmFtczoge1xuICAgICAgICAgIGVkZ2UsXG4gICAgICAgICAgZmlsZURvY0lkLFxuICAgICAgICAgIGltYWdlSWQsXG4gICAgICAgICAgaXRlbURvY0lkLFxuICAgICAgICAgIGl0ZW1JZCxcbiAgICAgICAgICBpdGVtVHlwZVxuICAgICAgICB9LFxuICAgICAgICB2YWx1ZTogRXJyb3JUeXBlcy5EQVRBQkFTRV9FUlJPUlxuICAgICAgfSwgZXJyb3IsIGNvbnRleHQpKTtcbiAgfVxuXG4gIHJldHVybiBudWxsO1xufTtcblxuZXhwb3J0IGNvbnN0IHVwZGF0ZUltYWdlID0gYXN5bmMgKFxuICBjb250ZXh0OiBBcGlDb250ZXh0LFxuICBpbWFnZTogSW1hZ2VUeXBlLFxuICBzM09wdGlvbnM/OiBQdXRPYmplY3RDb21tYW5kSW5wdXRcbik6IFByb21pc2U8SW1hZ2VUeXBlPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ3VwZGF0ZUltYWdlJztcbiAgY29uc3Qge2RhdGFiYXNlLCBzZXNzaW9uOiB7dXNlcklkOiBzZXNzaW9uSWR9fSA9IGNvbnRleHQ7XG4gIGNvbnN0IHtcbiAgICBiYXNlNjQgPSAnJyxcbiAgICBidWZmZXI6IGltYWdlQnVmZmVyLFxuICAgIGRlc2NyaXB0aW9uID0gJycsXG4gICAgaW1hZ2VJZCxcbiAgICBpdGVtSWQsXG4gICAgaXRlbVR5cGUsXG4gICAgZmlsZVR5cGUsXG4gICAgdXJsID0gJydcbiAgfSA9IGltYWdlO1xuXG4gIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcbiAgY29uc3QgZm9ybWF0SW1hZ2VJZDogc3RyaW5nID0gaW1hZ2VJZCB8fCBjcmVhdGVIYXNoKGBpbWFnZS0ke3Nlc3Npb25JZH1gKTtcbiAgY29uc3QgZm9ybWF0SXRlbUlkOiBzdHJpbmcgPSBwYXJzZUlkKGl0ZW1JZCk7XG4gIGNvbnN0IGZvcm1hdEl0ZW1UeXBlOiBBcmFuZ29EYkNvbGxlY3Rpb24gPSBwYXJzZUNoYXIoaXRlbVR5cGUsIDE2KS50b0xvd2VyQ2FzZSgpIGFzIEFyYW5nb0RiQ29sbGVjdGlvbjtcbiAgY29uc3QgY3VzdG9tUGFyYW1zOiBJbWFnZVR5cGUgPSB7XG4gICAgZGVzY3JpcHRpb24sXG4gICAgaW1hZ2VJZDogZm9ybWF0SW1hZ2VJZCxcbiAgICB1c2VySWQ6IHNlc3Npb25JZFxuICB9O1xuXG4gIGlmKGJhc2U2NCB8fCBpbWFnZUJ1ZmZlcikge1xuICAgIGxldCBidWZmZXI6IEJ1ZmZlciA9IGltYWdlQnVmZmVyO1xuXG4gICAgaWYoYmFzZTY0KSB7XG4gICAgICBjb25zdCBmb3JtYXRCYXNlNjQ6IHN0cmluZyA9IGJhc2U2NC5zdWJzdHIoYmFzZTY0LmluZGV4T2YoJywnKSArIDEpO1xuICAgICAgYnVmZmVyID0gQnVmZmVyLmZyb20oZm9ybWF0QmFzZTY0LCAnYmFzZTY0Jyk7XG4gICAgfVxuXG4gICAgbGV0IHVwZGF0ZWRGaWxlVHlwZSA9IGZpbGVUeXBlO1xuXG4gICAgaWYoIWZpbGVUeXBlKSB7XG4gICAgICBjb25zdCB7ZmlsZVR5cGVGcm9tQnVmZmVyfSA9IGF3YWl0IGltcG9ydCgnZmlsZS10eXBlJyk7XG4gICAgICBjb25zdCB7bWltZX0gPSBhd2FpdCBmaWxlVHlwZUZyb21CdWZmZXIoYnVmZmVyKTtcbiAgICAgIHVwZGF0ZWRGaWxlVHlwZSA9IG1pbWUgYXMgYW55O1xuICAgIH1cblxuICAgIGNvbnN0IGltZ1BhcmFtczogSW1hZ2VUeXBlID0ge1xuICAgICAgLi4uY3VzdG9tUGFyYW1zLFxuICAgICAgYnVmZmVyLFxuICAgICAgZmlsZVR5cGU6IHVwZGF0ZWRGaWxlVHlwZVxuICAgIH07XG5cbiAgICByZXR1cm4gYWRkSW1hZ2UoY29udGV4dCwgaW1nUGFyYW1zLCBzM09wdGlvbnMpXG4gICAgICAudGhlbigoaW1hZ2U6IEltYWdlVHlwZSkgPT4ge1xuICAgICAgICBpZihmb3JtYXRJdGVtSWQgJiYgZm9ybWF0SXRlbVR5cGUpIHtcbiAgICAgICAgICBjb25zdCBpbWFnZUVkZ2U6IEltYWdlRWRnZVR5cGUgPSB7XG4gICAgICAgICAgICBpbWFnZUlkOiBmb3JtYXRJbWFnZUlkLFxuICAgICAgICAgICAgaXRlbUlkOiBmb3JtYXRJdGVtSWQsXG4gICAgICAgICAgICBpdGVtVHlwZTogZm9ybWF0SXRlbVR5cGVcbiAgICAgICAgICB9O1xuICAgICAgICAgIHJldHVybiBhZGRJbWFnZUVkZ2UoY29udGV4dCwgaW1hZ2VFZGdlKS50aGVuKCgpID0+IGltYWdlKTtcbiAgICAgICAgfVxuXG4gICAgICAgIHJldHVybiBpbWFnZTtcbiAgICAgIH0pXG4gICAgICAuY2F0Y2goKGVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgIGFjdGlvbixcbiAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgIHZhbHVlOiBFcnJvclR5cGVzLklNQUdFX1NBVkVcbiAgICAgIH0sIGVycm9yLCBjb250ZXh0KSk7XG4gIH0gZWxzZSBpZih1cmwgIT09ICcnKSB7XG4gICAgbGV0IGNvbnRlbnRUeXBlOiBNaW1lVHlwZTtcblxuICAgIHJldHVybiBodHRwR2V0KHVybClcbiAgICAgIC50aGVuKChyZXM6IFJlc3BvbnNlKSA9PiB7XG4gICAgICAgIGlmKHJlcy5zdGF0dXMgIT09IDIwMCkge1xuICAgICAgICAgIHJldHVybiBsb2dFeGNlcHRpb24oe1xuICAgICAgICAgICAgYWN0aW9uLFxuICAgICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgICB2YWx1ZTogRXJyb3JUeXBlcy5JTUFHRV9SRVFVRVNUXG4gICAgICAgICAgfSwgY29udGV4dCk7XG4gICAgICAgIH1cblxuICAgICAgICBjb250ZW50VHlwZSA9IHJlcy5oZWFkZXJzLmdldCgnY29udGVudC10eXBlJykgYXMgTWltZVR5cGU7XG5cbiAgICAgICAgcmV0dXJuIHJlcztcbiAgICAgIH0pXG4gICAgICAudGhlbigocmVzKSA9PiByZXMuYnVmZmVyKCkpXG4gICAgICAudGhlbigoYnVmZmVyOiBCdWZmZXIpID0+IHtcbiAgICAgICAgY29uc3QgaW1nUGFyYW1zOiBJbWFnZVR5cGUgPSB7XG4gICAgICAgICAgLi4uY3VzdG9tUGFyYW1zLFxuICAgICAgICAgIGJ1ZmZlcixcbiAgICAgICAgICBmaWxlVHlwZTogY29udGVudFR5cGVcbiAgICAgICAgfTtcblxuICAgICAgICByZXR1cm4gYWRkSW1hZ2UoY29udGV4dCwgaW1nUGFyYW1zLCBzM09wdGlvbnMpXG4gICAgICAgICAgLnRoZW4oKGltYWdlOiBJbWFnZVR5cGUpID0+IHtcbiAgICAgICAgICAgIGlmKGZvcm1hdEl0ZW1JZCAmJiBmb3JtYXRJdGVtVHlwZSkge1xuICAgICAgICAgICAgICBjb25zdCBpbWFnZUVkZ2U6IEltYWdlRWRnZVR5cGUgPSB7aW1hZ2VJZDogZm9ybWF0SW1hZ2VJZCwgaXRlbUlkOiBmb3JtYXRJdGVtSWQsIGl0ZW1UeXBlOiBmb3JtYXRJdGVtVHlwZX07XG4gICAgICAgICAgICAgIHJldHVybiBhZGRJbWFnZUVkZ2UoY29udGV4dCwgaW1hZ2VFZGdlKS50aGVuKCgpID0+IGltYWdlKTtcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgcmV0dXJuIGltYWdlO1xuICAgICAgICAgIH0pO1xuICAgICAgfSlcbiAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgIGFjdGlvbixcbiAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgIHZhbHVlOiAnZmV0Y2hfZXJyb3InXG4gICAgICB9LCBlcnJvciwgY29udGV4dCkpO1xuICB9IGVsc2UgaWYoaW1hZ2VJZCAhPT0gJycpIHtcbiAgICBjb25zdCB1cGRhdGU6IGFueSA9IHtcbiAgICAgIGRlc2NyaXB0aW9uLFxuICAgICAgbW9kaWZpZWQ6IG5vd1xuICAgIH07XG4gICAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBVUERBVEUge19rZXk6ICR7Zm9ybWF0SW1hZ2VJZH19IFdJVEggJHt1cGRhdGV9IElOIGltYWdlcyBSRVRVUk4gTkVXYDtcblxuICAgIHJldHVybiBkYXRhYmFzZS5xdWVyeShhcWxRcnkpXG4gICAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgIGFjdGlvbixcbiAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgIHBhcmFtczoge1xuICAgICAgICAgIGFxbFFyeSxcbiAgICAgICAgICBmb3JtYXRJbWFnZUlkLFxuICAgICAgICAgIHVwZGF0ZVxuICAgICAgICB9LFxuICAgICAgICB2YWx1ZTogRXJyb3JUeXBlcy5EQVRBQkFTRV9FUlJPUlxuICAgICAgfSwgZXJyb3IsIGNvbnRleHQpKTtcbiAgfVxuXG4gIHJldHVybiBudWxsO1xufTtcblxuZXhwb3J0IGNvbnN0IGRlbGV0ZUltYWdlID0gYXN5bmMgKGNvbnRleHQ6IEFwaUNvbnRleHQsIGltYWdlSWQ6IHN0cmluZyk6IFByb21pc2U8SW1hZ2VUeXBlPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2RlbGV0ZSc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgc2Vzc2lvbjoge3VzZXJJZDogc2Vzc2lvbklkfX0gPSBjb250ZXh0O1xuICBjb25zdCBmb3JtYXRJbWFnZUlkID0gcGFyc2VJZChpbWFnZUlkKTtcblxuICB0cnkge1xuICAgIGNvbnN0IHJlbW92ZUVkZ2VRdWVyeSA9IGFxbGBGT1IgaGkgSU4gaGFzSW1hZ2VcbiAgICBGSUxURVIgaGkuX2Zyb20gPT0gJHtmb3JtYXRJbWFnZUlkfVxuICAgIFJFTU9WRSBoaSBJTiBoYXNJbWFnZVxuICAgIFJFVFVSTiBPTERgO1xuXG4gICAgYXdhaXQgZGF0YWJhc2UucXVlcnkocmVtb3ZlRWRnZVF1ZXJ5KTtcblxuICAgIGNvbnN0IGFxbFF1ZXJ5ID0gYXFsYEZPUiBpIElOIGltYWdlc1xuICAgIEZJTFRFUiBpLl9rZXkgPT0gJHtmb3JtYXRJbWFnZUlkfSAmJiBpLnVzZXJJZCA9PSAke3Nlc3Npb25JZH1cbiAgICBSRU1PVkUgaSBJTiBpbWFnZXNcbiAgICBSRVRVUk4gT0xEYDtcblxuICAgIGNvbnN0IGltYWdlID0gYXdhaXQgZGF0YWJhc2UucXVlcnkoYXFsUXVlcnkpLnRoZW4oKGN1cnNvcjogQXJyYXlDdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpO1xuXG4gICAgaWYoaW1hZ2UpIHtcbiAgICAgIGNvbnN0IHtfa2V5OiBpbWFnZUtleX0gPSBpbWFnZTtcbiAgICAgIGNvbnN0IHBhcmFtczogRGVsZXRlT2JqZWN0c0NvbW1hbmRJbnB1dCA9IHtcbiAgICAgICAgQnVja2V0OiBudWxsLFxuICAgICAgICBEZWxldGU6IHtcbiAgICAgICAgICBPYmplY3RzOiBbXG4gICAgICAgICAgICB7S2V5OiBgdXNlcnMvJHtzZXNzaW9uSWR9L2ltYWdlcy8ke2ltYWdlS2V5fS5qcGdgfSxcbiAgICAgICAgICAgIHtLZXk6IGB1c2Vycy8ke3Nlc3Npb25JZH0vdGh1bWJzLyR7aW1hZ2VLZXl9LmpwZ2B9XG4gICAgICAgICAgXSxcbiAgICAgICAgICBRdWlldDogdHJ1ZVxuICAgICAgICB9XG4gICAgICB9O1xuXG4gICAgICByZXR1cm4gczNEZWxldGVMaXN0KHBhcmFtcykudGhlbigoKSA9PiBpbWFnZSk7XG4gICAgfVxuXG4gICAgcmV0dXJuIG51bGw7XG4gIH0gY2F0Y2goZXJyb3IpIHtcbiAgICByZXR1cm4gbG9nRXJyb3Ioe1xuICAgICAgYWN0aW9uLFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICBwYXJhbXM6IHtpbWFnZUlkfSxcbiAgICAgIHZhbHVlOiBFcnJvclR5cGVzLkRBVEFCQVNFX0VSUk9SXG4gICAgfSwgZXJyb3IsIGNvbnRleHQpO1xuICB9XG59O1xuIl0sCiAgIm1hcHBpbmdzIjogIjs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FB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