@tricoteuses/assemblee 1.0.0-next.2 → 1.0.2

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 (135) hide show
  1. package/README.md +2 -0
  2. package/lib/api.js +23 -7
  3. package/lib/api.mjs +41 -0
  4. package/lib/bugs/acteur-00010/plugin.test.js +139 -33
  5. package/lib/bugs/acteur-00010/plugin.test.mjs +33 -0
  6. package/lib/bugs/acteur-00010.js +166 -24
  7. package/lib/bugs/acteur-00010.mjs +31 -0
  8. package/lib/bugs/agenda-00002/plugin.test.js +178 -52
  9. package/lib/bugs/agenda-00002/plugin.test.mjs +52 -0
  10. package/lib/bugs/agenda-00002.js +507 -143
  11. package/lib/bugs/agenda-00002.mjs +197 -0
  12. package/lib/bugs/agenda-00008/plugin.test.js +143 -31
  13. package/lib/bugs/agenda-00008/plugin.test.mjs +31 -0
  14. package/lib/bugs/agenda-00008.js +180 -25
  15. package/lib/bugs/agenda-00008.mjs +30 -0
  16. package/lib/bugs/agenda-00011/plugin.test.js +371 -156
  17. package/lib/bugs/agenda-00011/plugin.test.mjs +156 -0
  18. package/lib/bugs/agenda-00011.js +416 -117
  19. package/lib/bugs/agenda-00011.mjs +150 -0
  20. package/lib/bugs.js +591 -54
  21. package/lib/bugs.mjs +93 -0
  22. package/lib/cleaners/actes_legislatifs.js +149 -61
  23. package/lib/cleaners/actes_legislatifs.mjs +311 -0
  24. package/lib/cleaners/acteurs.js +255 -123
  25. package/lib/cleaners/acteurs.mjs +211 -0
  26. package/lib/cleaners/amendements.js +242 -117
  27. package/lib/cleaners/amendements.mjs +292 -0
  28. package/lib/cleaners/documents.js +147 -62
  29. package/lib/cleaners/documents.mjs +162 -0
  30. package/lib/cleaners/dossiers_legislatifs.js +132 -45
  31. package/lib/cleaners/dossiers_legislatifs.mjs +123 -0
  32. package/lib/cleaners/index.js +64 -8
  33. package/lib/cleaners/index.mjs +8 -0
  34. package/lib/cleaners/organes.js +54 -21
  35. package/lib/cleaners/organes.mjs +64 -0
  36. package/lib/cleaners/reunions.js +237 -127
  37. package/lib/cleaners/reunions.mjs +318 -0
  38. package/lib/cleaners/scrutins.js +261 -99
  39. package/lib/cleaners/scrutins.mjs +254 -0
  40. package/lib/cleaners/xml.js +81 -16
  41. package/lib/cleaners/xml.mjs +42 -0
  42. package/lib/datasets.js +284 -153
  43. package/lib/datasets.mjs +596 -0
  44. package/lib/dates.js +93 -34
  45. package/lib/dates.mjs +98 -0
  46. package/lib/dossiers_legislatifs.js +107 -33
  47. package/lib/dossiers_legislatifs.mjs +215 -0
  48. package/lib/file_systems.js +203 -35
  49. package/lib/file_systems.mjs +64 -0
  50. package/lib/git.js +139 -41
  51. package/lib/git.mjs +114 -0
  52. package/lib/index.js +310 -16
  53. package/lib/index.mjs +16 -0
  54. package/lib/inserters.js +355 -131
  55. package/lib/inserters.mjs +413 -0
  56. package/lib/loaders.js +900 -440
  57. package/lib/loaders.mjs +678 -0
  58. package/lib/organes.js +15 -5
  59. package/lib/organes.mjs +18 -0
  60. package/lib/parsers/documents.js +241 -89
  61. package/lib/parsers/documents.mjs +433 -0
  62. package/lib/parsers/index.js +36 -3
  63. package/lib/parsers/index.mjs +3 -0
  64. package/lib/parsers/recherche_amendements.js +616 -173
  65. package/lib/parsers/recherche_amendements.mjs +189 -0
  66. package/lib/raw_types/acteurs_et_organes.d.ts +36 -148
  67. package/lib/raw_types/acteurs_et_organes.js +228 -506
  68. package/lib/raw_types/acteurs_et_organes.mjs +1240 -0
  69. package/lib/raw_types/agendas.js +165 -75
  70. package/lib/raw_types/agendas.mjs +944 -0
  71. package/lib/raw_types/amendements.js +165 -77
  72. package/lib/raw_types/amendements.mjs +1040 -0
  73. package/lib/raw_types/dossiers_legislatifs.d.ts +1 -43
  74. package/lib/raw_types/dossiers_legislatifs.js +181 -143
  75. package/lib/raw_types/dossiers_legislatifs.mjs +1792 -0
  76. package/lib/raw_types/scrutins.js +144 -61
  77. package/lib/raw_types/scrutins.mjs +645 -0
  78. package/lib/scripts/bugs_helper.js +657 -155
  79. package/lib/scripts/bugs_helper.mjs +286 -0
  80. package/lib/scripts/clean_data.js +380 -145
  81. package/lib/scripts/clean_data.mjs +190 -0
  82. package/lib/scripts/clean_reorganized_data.js +627 -331
  83. package/lib/scripts/clean_reorganized_data.mjs +520 -0
  84. package/lib/scripts/diff_amendements.js +159 -73
  85. package/lib/scripts/diff_amendements.mjs +234 -0
  86. package/lib/scripts/document_dossiers_legislatifs.js +570 -288
  87. package/lib/scripts/document_dossiers_legislatifs.mjs +494 -0
  88. package/lib/scripts/get_today_reunions.js +37 -20
  89. package/lib/scripts/get_today_reunions.mjs +41 -0
  90. package/lib/scripts/merge_scrutins.js +121 -39
  91. package/lib/scripts/merge_scrutins.mjs +69 -0
  92. package/lib/scripts/parse_textes_lois.js +179 -125
  93. package/lib/scripts/parse_textes_lois.mjs +215 -0
  94. package/lib/scripts/raw_types_from_amendements.js +276 -77
  95. package/lib/scripts/raw_types_from_amendements.mjs +118 -0
  96. package/lib/scripts/reorganize_data.js +867 -446
  97. package/lib/scripts/reorganize_data.mjs +712 -0
  98. package/lib/scripts/retrieve_deputes_photos.js +378 -157
  99. package/lib/scripts/retrieve_deputes_photos.mjs +230 -0
  100. package/lib/scripts/retrieve_documents.js +648 -172
  101. package/lib/scripts/retrieve_documents.mjs +292 -0
  102. package/lib/scripts/retrieve_open_data.js +286 -85
  103. package/lib/scripts/retrieve_open_data.mjs +125 -0
  104. package/lib/scripts/retrieve_pending_amendments.js +302 -69
  105. package/lib/scripts/retrieve_pending_amendments.mjs +130 -0
  106. package/lib/scripts/retrieve_senateurs_photos.js +374 -198
  107. package/lib/scripts/retrieve_senateurs_photos.mjs +277 -0
  108. package/lib/scripts/retrieve_textes_lois.js +343 -89
  109. package/lib/scripts/retrieve_textes_lois.mjs +168 -0
  110. package/lib/scripts/test_load.js +30 -11
  111. package/lib/scripts/test_load.mjs +26 -0
  112. package/lib/scripts/test_load_big_files.js +30 -11
  113. package/lib/scripts/test_load_big_files.mjs +26 -0
  114. package/lib/scripts/validate_json.js +370 -120
  115. package/lib/scripts/validate_json.mjs +216 -0
  116. package/lib/shared_types/codes_actes.js +50 -15
  117. package/lib/shared_types/codes_actes.mjs +228 -0
  118. package/lib/strings.js +16 -3
  119. package/lib/strings.mjs +7 -0
  120. package/lib/types/acteurs_et_organes.js +156 -71
  121. package/lib/types/acteurs_et_organes.mjs +918 -0
  122. package/lib/types/agendas.js +154 -67
  123. package/lib/types/agendas.mjs +623 -0
  124. package/lib/types/amendements.js +166 -77
  125. package/lib/types/amendements.mjs +885 -0
  126. package/lib/types/dossiers_legislatifs.d.ts +1 -42
  127. package/lib/types/dossiers_legislatifs.js +182 -143
  128. package/lib/types/dossiers_legislatifs.mjs +1120 -0
  129. package/lib/types/legislatures.js +12 -3
  130. package/lib/types/legislatures.mjs +9 -0
  131. package/lib/types/scrutins.js +144 -61
  132. package/lib/types/scrutins.mjs +528 -0
  133. package/lib/urls.js +44 -25
  134. package/lib/urls.mjs +201 -0
  135. package/package.json +33 -12
@@ -0,0 +1,277 @@
1
+ import { execSync } from "child_process";
2
+ import commandLineArgs from "command-line-args";
3
+ import fs from "fs-extra"; // import fetch from "node-fetch"
4
+
5
+ import path from "path"; // import stream from "stream"
6
+ // import util from "util"
7
+
8
+ import { EnabledDatasets } from "../datasets.mjs";
9
+ import { loadAssembleeData } from "../loaders.mjs";
10
+ import { CodeTypeOrgane, TypeMandat } from "../types/acteurs_et_organes.mjs";
11
+ const optionsDefinitions = [{
12
+ alias: "C",
13
+ help: "clone repositories from given group (or organization) git URL",
14
+ name: "clone",
15
+ type: String
16
+ }, {
17
+ alias: "c",
18
+ help: "commit photos",
19
+ name: "commit",
20
+ type: Boolean
21
+ }, {
22
+ alias: "f",
23
+ help: "fetch sénateurs' pictures instead of retrieving them from files",
24
+ name: "fetch",
25
+ type: Boolean
26
+ }, {
27
+ alias: "r",
28
+ help: "push commit to given remote",
29
+ multiple: true,
30
+ name: "remote",
31
+ type: String
32
+ }, {
33
+ alias: "s",
34
+ help: "don't log anything",
35
+ name: "silent",
36
+ type: Boolean
37
+ }, {
38
+ defaultOption: true,
39
+ help: "directory containing Assemblée open data files",
40
+ name: "dataDir",
41
+ type: String
42
+ }];
43
+ const options = commandLineArgs(optionsDefinitions); // const pipeline = util.promisify(stream.pipeline)
44
+
45
+ function commitAndPush(repositoryDir, commit, remotes) {
46
+ let error = false;
47
+
48
+ if (commit) {
49
+ execSync("git add .", {
50
+ cwd: repositoryDir,
51
+ env: process.env,
52
+ encoding: "utf-8",
53
+ stdio: ["ignore", "ignore", "pipe"]
54
+ });
55
+
56
+ try {
57
+ execSync('git commit -m "Nouvelle moisson"', {
58
+ cwd: repositoryDir,
59
+ env: process.env,
60
+ encoding: "utf-8"
61
+ });
62
+ } catch (childProcess) {
63
+ if (childProcess.stderr === null || !/nothing to commit/.test(childProcess.stdout)) {
64
+ console.error(childProcess.output);
65
+ throw childProcess;
66
+ }
67
+ }
68
+
69
+ for (const remote of remotes || []) {
70
+ try {
71
+ execSync(`git push ${remote} master`, {
72
+ cwd: repositoryDir,
73
+ env: process.env,
74
+ encoding: "utf-8",
75
+ stdio: ["ignore", "ignore", "pipe"]
76
+ });
77
+ } catch (childProcess) {
78
+ // Don't stop when push fails.
79
+ console.error(childProcess.output);
80
+ error = true;
81
+ }
82
+ }
83
+ }
84
+
85
+ return error;
86
+ }
87
+
88
+ async function retrievePhotosSenateurs() {
89
+ const dataDir = options.dataDir;
90
+ const {
91
+ acteurByUid
92
+ } = loadAssembleeData(dataDir, EnabledDatasets.ActeursEtOrganes, options.legislature);
93
+ const senateurs = Object.values(acteurByUid).filter(acteur => {
94
+ const {
95
+ mandats
96
+ } = acteur;
97
+
98
+ if (!mandats) {
99
+ return false;
100
+ }
101
+
102
+ const mandatSenateur = mandats.filter(mandat => mandat.xsiType === TypeMandat.MandatParlementaireType && // mandat.legislature === options.legislature &&
103
+ mandat.typeOrgane === CodeTypeOrgane.Senat)[0];
104
+
105
+ if (mandatSenateur === undefined) {
106
+ return false;
107
+ }
108
+
109
+ const {
110
+ adresses
111
+ } = acteur;
112
+
113
+ if (adresses === undefined) {
114
+ return false;
115
+ }
116
+
117
+ let url = null;
118
+
119
+ for (const adresse of adresses) {
120
+ if (adresse.xsiType === "AdresseSiteWeb_Type" && adresse.typeLibelle === "Url sénateur") {
121
+ url = adresse.valElec;
122
+ }
123
+ }
124
+
125
+ return Boolean(url);
126
+ }).sort((a, b) => a.uid.length === b.uid.length ? a.uid.localeCompare(b.uid) : a.uid.length - b.uid.length);
127
+ const gitGroupUrl = options.clone ? options.clone.trim().replace(/\/+$/, "") : undefined;
128
+ const photosDirName = "photos_senateurs";
129
+
130
+ if (gitGroupUrl !== undefined) {
131
+ execSync(`git clone ${gitGroupUrl}/${photosDirName}.git`, {
132
+ cwd: dataDir,
133
+ env: process.env,
134
+ encoding: "utf-8",
135
+ stdio: ["ignore", "ignore", "pipe"]
136
+ });
137
+ }
138
+
139
+ const photosDir = path.join(dataDir, photosDirName); // Download photos.
140
+
141
+ fs.ensureDirSync(photosDir);
142
+
143
+ if (options.fetch) {
144
+ const missingPhotoFilePath = "images/transparent_155x225.jpg";
145
+
146
+ for (const senateur of senateurs) {
147
+ const ident = senateur.etatCivil.ident;
148
+ const senateurStem = stemFromSenateur(senateur);
149
+ const photoFilename = `${senateurStem}.jpg`;
150
+ const photoFilePath = path.join(photosDir, photoFilename);
151
+ const photoTempFilename = `${senateurStem}_temp.jpg`;
152
+ const photoTempFilePath = path.join(photosDir, photoTempFilename);
153
+ const urlPhoto = `https://www.senat.fr/senimg/${photoFilename}`;
154
+
155
+ if (!options.silent) {
156
+ console.log(`Loading photo ${urlPhoto} for ${ident.prenom} ${ident.nom}…`);
157
+ } // for (let retries = 0; retries < 3; retries++) {
158
+ // const response = await fetch(urlPhoto)
159
+ // if (response.ok) {
160
+ // await pipeline(response.body, fs.createWriteStream(photoTempFilePath))
161
+ // fs.renameSync(photoTempFilePath, photoFilePath)
162
+ // break
163
+ // }
164
+ // if (retries >= 2) {
165
+ // console.warn(`Fetch failed: ${urlPhoto} (${ident.prenom} ${ident.nom})`)
166
+ // console.warn(response.status, response.statusText)
167
+ // console.warn(await response.text())
168
+ // if (fs.existsSync(photoFilePath)) {
169
+ // console.warn(" => Reusing existing image")
170
+ // } else {
171
+ // console.warn(" => Using blank image")
172
+ // fs.copyFileSync(missingPhotoFilePath, photoFilePath)
173
+ // }
174
+ // break
175
+ // }
176
+ // }
177
+
178
+
179
+ try {
180
+ execSync(`wget --quiet -O ${photoTempFilename} ${urlPhoto}`, {
181
+ cwd: photosDir,
182
+ env: process.env,
183
+ encoding: "utf-8" // stdio: ["ignore", "ignore", "pipe"],
184
+
185
+ });
186
+ fs.renameSync(photoTempFilePath, photoFilePath);
187
+ } catch (error) {
188
+ console.error(error);
189
+
190
+ if (fs.existsSync(photoFilePath)) {
191
+ console.warn(" => Reusing existing image");
192
+ } else {
193
+ console.warn(" => Using blank image");
194
+ fs.copyFileSync(missingPhotoFilePath, photoFilePath);
195
+ }
196
+ }
197
+ }
198
+ } // Resize photos to 155x225, because some haven't exactly this size.
199
+
200
+
201
+ for (const senateur of senateurs) {
202
+ const ident = senateur.etatCivil.ident;
203
+ const senateurStem = stemFromSenateur(senateur);
204
+
205
+ if (!options.silent) {
206
+ console.log(`Resizing photo ${senateurStem} for ${ident.prenom} ${ident.nom}…`);
207
+ }
208
+
209
+ execSync(`gm convert -resize 155x225! ${senateurStem}.jpg ${senateurStem}_155x225.jpg`, {
210
+ cwd: photosDir
211
+ });
212
+ } // Create a mosaic of photos.
213
+
214
+
215
+ const photoByIdSenateur = {};
216
+ const rowsFilenames = [];
217
+
218
+ for (let senateurIndex = 0, rowIndex = 0; senateurIndex < senateurs.length; senateurIndex += 25, rowIndex++) {
219
+ const row = senateurs.slice(senateurIndex, senateurIndex + 25);
220
+ const photosFilenames = [];
221
+
222
+ for (const [columnIndex, senateur] of row.entries()) {
223
+ const senateurStem = stemFromSenateur(senateur);
224
+ const photoFilename = `${senateurStem}_155x225.jpg`;
225
+ photosFilenames.push(photoFilename);
226
+ photoByIdSenateur[senateur.uid] = {
227
+ chemin: `photos_senateurs/${photoFilename}`,
228
+ cheminMosaique: `photos_senateurs/senateurs.jpg`,
229
+ hauteur: 225,
230
+ largeur: 155,
231
+ xMosaique: columnIndex * 155,
232
+ yMosaique: rowIndex * 225
233
+ };
234
+ }
235
+
236
+ const rowFilename = `row-${rowIndex}.jpg`;
237
+ execSync(`gm convert ${photosFilenames.join(" ")} +append ${rowFilename}`, {
238
+ cwd: photosDir
239
+ });
240
+ rowsFilenames.push(rowFilename);
241
+ }
242
+
243
+ execSync(`gm convert ${rowsFilenames.join(" ")} -append senateurs.jpg`, {
244
+ cwd: photosDir
245
+ });
246
+
247
+ for (const rowFilename of rowsFilenames) {
248
+ fs.unlinkSync(path.join(photosDir, rowFilename));
249
+ }
250
+
251
+ const jsonFilePath = path.join(photosDir, "senateurs.json");
252
+ fs.writeFileSync(jsonFilePath, JSON.stringify(photoByIdSenateur, null, 2));
253
+ return commitAndPush(photosDir, options.commit, options.remote);
254
+ }
255
+
256
+ function stemFromSenateur(senateur) {
257
+ const {
258
+ adresses
259
+ } = senateur;
260
+ let url = null;
261
+
262
+ for (const adresse of adresses) {
263
+ if (adresse.xsiType === "AdresseSiteWeb_Type" && adresse.typeLibelle === "Url sénateur") {
264
+ url = adresse.valElec;
265
+ }
266
+ }
267
+
268
+ const match = url.match(/^https:\/\/www\.senat\.fr\/senateur\/(.*)\.html$/);
269
+ const senateurStem = match[1];
270
+ return senateurStem;
271
+ }
272
+
273
+ retrievePhotosSenateurs().then(error => process.exit(error ? 1 : 0)).catch(error => {
274
+ console.log(error);
275
+ process.exit(2);
276
+ });
277
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJleGVjU3luYyIsImNvbW1hbmRMaW5lQXJncyIsImZzIiwicGF0aCIsIkVuYWJsZWREYXRhc2V0cyIsImxvYWRBc3NlbWJsZWVEYXRhIiwiQ29kZVR5cGVPcmdhbmUiLCJUeXBlTWFuZGF0Iiwib3B0aW9uc0RlZmluaXRpb25zIiwiYWxpYXMiLCJoZWxwIiwibmFtZSIsInR5cGUiLCJTdHJpbmciLCJCb29sZWFuIiwibXVsdGlwbGUiLCJkZWZhdWx0T3B0aW9uIiwib3B0aW9ucyIsImNvbW1pdEFuZFB1c2giLCJyZXBvc2l0b3J5RGlyIiwiY29tbWl0IiwicmVtb3RlcyIsImVycm9yIiwiY3dkIiwiZW52IiwicHJvY2VzcyIsImVuY29kaW5nIiwic3RkaW8iLCJjaGlsZFByb2Nlc3MiLCJzdGRlcnIiLCJ0ZXN0Iiwic3Rkb3V0IiwiY29uc29sZSIsIm91dHB1dCIsInJlbW90ZSIsInJldHJpZXZlUGhvdG9zU2VuYXRldXJzIiwiZGF0YURpciIsImFjdGV1ckJ5VWlkIiwiQWN0ZXVyc0V0T3JnYW5lcyIsImxlZ2lzbGF0dXJlIiwic2VuYXRldXJzIiwiT2JqZWN0IiwidmFsdWVzIiwiZmlsdGVyIiwiYWN0ZXVyIiwibWFuZGF0cyIsIm1hbmRhdFNlbmF0ZXVyIiwibWFuZGF0IiwieHNpVHlwZSIsIk1hbmRhdFBhcmxlbWVudGFpcmVUeXBlIiwidHlwZU9yZ2FuZSIsIlNlbmF0IiwidW5kZWZpbmVkIiwiYWRyZXNzZXMiLCJ1cmwiLCJhZHJlc3NlIiwidHlwZUxpYmVsbGUiLCJ2YWxFbGVjIiwic29ydCIsImEiLCJiIiwidWlkIiwibGVuZ3RoIiwibG9jYWxlQ29tcGFyZSIsImdpdEdyb3VwVXJsIiwiY2xvbmUiLCJ0cmltIiwicmVwbGFjZSIsInBob3Rvc0Rpck5hbWUiLCJwaG90b3NEaXIiLCJqb2luIiwiZW5zdXJlRGlyU3luYyIsImZldGNoIiwibWlzc2luZ1Bob3RvRmlsZVBhdGgiLCJzZW5hdGV1ciIsImlkZW50IiwiZXRhdENpdmlsIiwic2VuYXRldXJTdGVtIiwic3RlbUZyb21TZW5hdGV1ciIsInBob3RvRmlsZW5hbWUiLCJwaG90b0ZpbGVQYXRoIiwicGhvdG9UZW1wRmlsZW5hbWUiLCJwaG90b1RlbXBGaWxlUGF0aCIsInVybFBob3RvIiwic2lsZW50IiwibG9nIiwicHJlbm9tIiwibm9tIiwicmVuYW1lU3luYyIsImV4aXN0c1N5bmMiLCJ3YXJuIiwiY29weUZpbGVTeW5jIiwicGhvdG9CeUlkU2VuYXRldXIiLCJyb3dzRmlsZW5hbWVzIiwic2VuYXRldXJJbmRleCIsInJvd0luZGV4Iiwicm93Iiwic2xpY2UiLCJwaG90b3NGaWxlbmFtZXMiLCJjb2x1bW5JbmRleCIsImVudHJpZXMiLCJwdXNoIiwiY2hlbWluIiwiY2hlbWluTW9zYWlxdWUiLCJoYXV0ZXVyIiwibGFyZ2V1ciIsInhNb3NhaXF1ZSIsInlNb3NhaXF1ZSIsInJvd0ZpbGVuYW1lIiwidW5saW5rU3luYyIsImpzb25GaWxlUGF0aCIsIndyaXRlRmlsZVN5bmMiLCJKU09OIiwic3RyaW5naWZ5IiwibWF0Y2giLCJ0aGVuIiwiZXhpdCIsImNhdGNoIl0sInNvdXJjZXMiOlsiLi4vLi4vc3JjL3NjcmlwdHMvcmV0cmlldmVfc2VuYXRldXJzX3Bob3Rvcy50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDaGlsZFByb2Nlc3MsIGV4ZWNTeW5jIH0gZnJvbSBcImNoaWxkX3Byb2Nlc3NcIlxuaW1wb3J0IGNvbW1hbmRMaW5lQXJncyBmcm9tIFwiY29tbWFuZC1saW5lLWFyZ3NcIlxuaW1wb3J0IGZzIGZyb20gXCJmcy1leHRyYVwiXG4vLyBpbXBvcnQgZmV0Y2ggZnJvbSBcIm5vZGUtZmV0Y2hcIlxuaW1wb3J0IHBhdGggZnJvbSBcInBhdGhcIlxuLy8gaW1wb3J0IHN0cmVhbSBmcm9tIFwic3RyZWFtXCJcbi8vIGltcG9ydCB1dGlsIGZyb20gXCJ1dGlsXCJcblxuaW1wb3J0IHsgRW5hYmxlZERhdGFzZXRzIH0gZnJvbSBcIi4uL2RhdGFzZXRzXCJcbmltcG9ydCB7IGxvYWRBc3NlbWJsZWVEYXRhIH0gZnJvbSBcIi4uL2xvYWRlcnNcIlxuaW1wb3J0IHtcbiAgQWN0ZXVyLFxuICBDb2RlVHlwZU9yZ2FuZSxcbiAgTWFuZGF0LFxuICBQaG90byxcbiAgVHlwZU1hbmRhdCxcbn0gZnJvbSBcIi4uL3R5cGVzL2FjdGV1cnNfZXRfb3JnYW5lc1wiXG5cbmNvbnN0IG9wdGlvbnNEZWZpbml0aW9ucyA9IFtcbiAge1xuICAgIGFsaWFzOiBcIkNcIixcbiAgICBoZWxwOiBcImNsb25lIHJlcG9zaXRvcmllcyBmcm9tIGdpdmVuIGdyb3VwIChvciBvcmdhbml6YXRpb24pIGdpdCBVUkxcIixcbiAgICBuYW1lOiBcImNsb25lXCIsXG4gICAgdHlwZTogU3RyaW5nLFxuICB9LFxuICB7XG4gICAgYWxpYXM6IFwiY1wiLFxuICAgIGhlbHA6IFwiY29tbWl0IHBob3Rvc1wiLFxuICAgIG5hbWU6IFwiY29tbWl0XCIsXG4gICAgdHlwZTogQm9vbGVhbixcbiAgfSxcbiAge1xuICAgIGFsaWFzOiBcImZcIixcbiAgICBoZWxwOiBcImZldGNoIHPDqW5hdGV1cnMnIHBpY3R1cmVzIGluc3RlYWQgb2YgcmV0cmlldmluZyB0aGVtIGZyb20gZmlsZXNcIixcbiAgICBuYW1lOiBcImZldGNoXCIsXG4gICAgdHlwZTogQm9vbGVhbixcbiAgfSxcbiAge1xuICAgIGFsaWFzOiBcInJcIixcbiAgICBoZWxwOiBcInB1c2ggY29tbWl0IHRvIGdpdmVuIHJlbW90ZVwiLFxuICAgIG11bHRpcGxlOiB0cnVlLFxuICAgIG5hbWU6IFwicmVtb3RlXCIsXG4gICAgdHlwZTogU3RyaW5nLFxuICB9LFxuICB7XG4gICAgYWxpYXM6IFwic1wiLFxuICAgIGhlbHA6IFwiZG9uJ3QgbG9nIGFueXRoaW5nXCIsXG4gICAgbmFtZTogXCJzaWxlbnRcIixcbiAgICB0eXBlOiBCb29sZWFuLFxuICB9LFxuICB7XG4gICAgZGVmYXVsdE9wdGlvbjogdHJ1ZSxcbiAgICBoZWxwOiBcImRpcmVjdG9yeSBjb250YWluaW5nIEFzc2VtYmzDqWUgb3BlbiBkYXRhIGZpbGVzXCIsXG4gICAgbmFtZTogXCJkYXRhRGlyXCIsXG4gICAgdHlwZTogU3RyaW5nLFxuICB9LFxuXVxuY29uc3Qgb3B0aW9ucyA9IGNvbW1hbmRMaW5lQXJncyhvcHRpb25zRGVmaW5pdGlvbnMpXG4vLyBjb25zdCBwaXBlbGluZSA9IHV0aWwucHJvbWlzaWZ5KHN0cmVhbS5waXBlbGluZSlcblxuZnVuY3Rpb24gY29tbWl0QW5kUHVzaChcbiAgcmVwb3NpdG9yeURpcjogc3RyaW5nLFxuICBjb21taXQ6IGJvb2xlYW4sXG4gIHJlbW90ZXM/OiBzdHJpbmdbXSxcbik6IGJvb2xlYW4ge1xuICBsZXQgZXJyb3IgPSBmYWxzZVxuICBpZiAoY29tbWl0KSB7XG4gICAgZXhlY1N5bmMoXCJnaXQgYWRkIC5cIiwge1xuICAgICAgY3dkOiByZXBvc2l0b3J5RGlyLFxuICAgICAgZW52OiBwcm9jZXNzLmVudixcbiAgICAgIGVuY29kaW5nOiBcInV0Zi04XCIsXG4gICAgICBzdGRpbzogW1wiaWdub3JlXCIsIFwiaWdub3JlXCIsIFwicGlwZVwiXSxcbiAgICB9KVxuICAgIHRyeSB7XG4gICAgICBleGVjU3luYygnZ2l0IGNvbW1pdCAtbSBcIk5vdXZlbGxlIG1vaXNzb25cIicsIHtcbiAgICAgICAgY3dkOiByZXBvc2l0b3J5RGlyLFxuICAgICAgICBlbnY6IHByb2Nlc3MuZW52LFxuICAgICAgICBlbmNvZGluZzogXCJ1dGYtOFwiLFxuICAgICAgfSlcbiAgICB9IGNhdGNoIChjaGlsZFByb2Nlc3MpIHtcbiAgICAgIGlmIChcbiAgICAgICAgKGNoaWxkUHJvY2VzcyBhcyBDaGlsZFByb2Nlc3MpLnN0ZGVyciA9PT0gbnVsbCB8fFxuICAgICAgICAhL25vdGhpbmcgdG8gY29tbWl0Ly50ZXN0KChjaGlsZFByb2Nlc3MgYXMgYW55KS5zdGRvdXQpXG4gICAgICApIHtcbiAgICAgICAgY29uc29sZS5lcnJvcigoY2hpbGRQcm9jZXNzIGFzIGFueSkub3V0cHV0KVxuICAgICAgICB0aHJvdyBjaGlsZFByb2Nlc3NcbiAgICAgIH1cbiAgICB9XG4gICAgZm9yIChjb25zdCByZW1vdGUgb2YgcmVtb3RlcyB8fCBbXSkge1xuICAgICAgdHJ5IHtcbiAgICAgICAgZXhlY1N5bmMoYGdpdCBwdXNoICR7cmVtb3RlfSBtYXN0ZXJgLCB7XG4gICAgICAgICAgY3dkOiByZXBvc2l0b3J5RGlyLFxuICAgICAgICAgIGVudjogcHJvY2Vzcy5lbnYsXG4gICAgICAgICAgZW5jb2Rpbmc6IFwidXRmLThcIixcbiAgICAgICAgICBzdGRpbzogW1wiaWdub3JlXCIsIFwiaWdub3JlXCIsIFwicGlwZVwiXSxcbiAgICAgICAgfSlcbiAgICAgIH0gY2F0Y2ggKGNoaWxkUHJvY2Vzcykge1xuICAgICAgICAvLyBEb24ndCBzdG9wIHdoZW4gcHVzaCBmYWlscy5cbiAgICAgICAgY29uc29sZS5lcnJvcigoY2hpbGRQcm9jZXNzIGFzIGFueSkub3V0cHV0KVxuICAgICAgICBlcnJvciA9IHRydWVcbiAgICAgIH1cbiAgICB9XG4gIH1cbiAgcmV0dXJuIGVycm9yXG59XG5cbmFzeW5jIGZ1bmN0aW9uIHJldHJpZXZlUGhvdG9zU2VuYXRldXJzKCk6IFByb21pc2U8Ym9vbGVhbj4ge1xuICBjb25zdCBkYXRhRGlyID0gb3B0aW9ucy5kYXRhRGlyXG4gIGNvbnN0IHsgYWN0ZXVyQnlVaWQgfSA9IGxvYWRBc3NlbWJsZWVEYXRhKFxuICAgIGRhdGFEaXIsXG4gICAgRW5hYmxlZERhdGFzZXRzLkFjdGV1cnNFdE9yZ2FuZXMsXG4gICAgb3B0aW9ucy5sZWdpc2xhdHVyZSxcbiAgKVxuICBjb25zdCBzZW5hdGV1cnMgPSBPYmplY3QudmFsdWVzKGFjdGV1ckJ5VWlkKVxuICAgIC5maWx0ZXIoKGFjdGV1cikgPT4ge1xuICAgICAgY29uc3QgeyBtYW5kYXRzIH0gPSBhY3RldXJcbiAgICAgIGlmICghbWFuZGF0cykge1xuICAgICAgICByZXR1cm4gZmFsc2VcbiAgICAgIH1cbiAgICAgIGNvbnN0IG1hbmRhdFNlbmF0ZXVyID0gbWFuZGF0cy5maWx0ZXIoXG4gICAgICAgIChtYW5kYXQ6IE1hbmRhdCkgPT5cbiAgICAgICAgICBtYW5kYXQueHNpVHlwZSA9PT0gVHlwZU1hbmRhdC5NYW5kYXRQYXJsZW1lbnRhaXJlVHlwZSAmJlxuICAgICAgICAgIC8vIG1hbmRhdC5sZWdpc2xhdHVyZSA9PT0gb3B0aW9ucy5sZWdpc2xhdHVyZSAmJlxuICAgICAgICAgIG1hbmRhdC50eXBlT3JnYW5lID09PSBDb2RlVHlwZU9yZ2FuZS5TZW5hdCxcbiAgICAgIClbMF1cbiAgICAgIGlmIChtYW5kYXRTZW5hdGV1ciA9PT0gdW5kZWZpbmVkKSB7XG4gICAgICAgIHJldHVybiBmYWxzZVxuICAgICAgfVxuICAgICAgY29uc3QgeyBhZHJlc3NlcyB9ID0gYWN0ZXVyXG4gICAgICBpZiAoYWRyZXNzZXMgPT09IHVuZGVmaW5lZCkge1xuICAgICAgICByZXR1cm4gZmFsc2VcbiAgICAgIH1cbiAgICAgIGxldCB1cmwgPSBudWxsXG4gICAgICBmb3IgKGNvbnN0IGFkcmVzc2Ugb2YgYWRyZXNzZXMpIHtcbiAgICAgICAgaWYgKFxuICAgICAgICAgIGFkcmVzc2UueHNpVHlwZSA9PT0gXCJBZHJlc3NlU2l0ZVdlYl9UeXBlXCIgJiZcbiAgICAgICAgICBhZHJlc3NlLnR5cGVMaWJlbGxlID09PSBcIlVybCBzw6luYXRldXJcIlxuICAgICAgICApIHtcbiAgICAgICAgICB1cmwgPSBhZHJlc3NlLnZhbEVsZWNcbiAgICAgICAgfVxuICAgICAgfVxuICAgICAgcmV0dXJuIEJvb2xlYW4odXJsKVxuICAgIH0pXG4gICAgLnNvcnQoKGEsIGIpID0+XG4gICAgICBhLnVpZC5sZW5ndGggPT09IGIudWlkLmxlbmd0aFxuICAgICAgICA/IGEudWlkLmxvY2FsZUNvbXBhcmUoYi51aWQpXG4gICAgICAgIDogYS51aWQubGVuZ3RoIC0gYi51aWQubGVuZ3RoLFxuICAgIClcblxuICBjb25zdCBnaXRHcm91cFVybCA9IG9wdGlvbnMuY2xvbmVcbiAgICA/IG9wdGlvbnMuY2xvbmUudHJpbSgpLnJlcGxhY2UoL1xcLyskLywgXCJcIilcbiAgICA6IHVuZGVmaW5lZFxuICBjb25zdCBwaG90b3NEaXJOYW1lID0gXCJwaG90b3Nfc2VuYXRldXJzXCJcbiAgaWYgKGdpdEdyb3VwVXJsICE9PSB1bmRlZmluZWQpIHtcbiAgICBleGVjU3luYyhgZ2l0IGNsb25lICR7Z2l0R3JvdXBVcmx9LyR7cGhvdG9zRGlyTmFtZX0uZ2l0YCwge1xuICAgICAgY3dkOiBkYXRhRGlyLFxuICAgICAgZW52OiBwcm9jZXNzLmVudixcbiAgICAgIGVuY29kaW5nOiBcInV0Zi04XCIsXG4gICAgICBzdGRpbzogW1wiaWdub3JlXCIsIFwiaWdub3JlXCIsIFwicGlwZVwiXSxcbiAgICB9KVxuICB9XG4gIGNvbnN0IHBob3Rvc0RpciA9IHBhdGguam9pbihkYXRhRGlyLCBwaG90b3NEaXJOYW1lKVxuXG4gIC8vIERvd25sb2FkIHBob3Rvcy5cbiAgZnMuZW5zdXJlRGlyU3luYyhwaG90b3NEaXIpXG4gIGlmIChvcHRpb25zLmZldGNoKSB7XG4gICAgY29uc3QgbWlzc2luZ1Bob3RvRmlsZVBhdGggPSBcImltYWdlcy90cmFuc3BhcmVudF8xNTV4MjI1LmpwZ1wiXG4gICAgZm9yIChjb25zdCBzZW5hdGV1ciBvZiBzZW5hdGV1cnMpIHtcbiAgICAgIGNvbnN0IGlkZW50ID0gc2VuYXRldXIuZXRhdENpdmlsLmlkZW50XG4gICAgICBjb25zdCBzZW5hdGV1clN0ZW0gPSBzdGVtRnJvbVNlbmF0ZXVyKHNlbmF0ZXVyKVxuICAgICAgY29uc3QgcGhvdG9GaWxlbmFtZSA9IGAke3NlbmF0ZXVyU3RlbX0uanBnYFxuICAgICAgY29uc3QgcGhvdG9GaWxlUGF0aCA9IHBhdGguam9pbihwaG90b3NEaXIsIHBob3RvRmlsZW5hbWUpXG4gICAgICBjb25zdCBwaG90b1RlbXBGaWxlbmFtZSA9IGAke3NlbmF0ZXVyU3RlbX1fdGVtcC5qcGdgXG4gICAgICBjb25zdCBwaG90b1RlbXBGaWxlUGF0aCA9IHBhdGguam9pbihwaG90b3NEaXIsIHBob3RvVGVtcEZpbGVuYW1lKVxuICAgICAgY29uc3QgdXJsUGhvdG8gPSBgaHR0cHM6Ly93d3cuc2VuYXQuZnIvc2VuaW1nLyR7cGhvdG9GaWxlbmFtZX1gXG4gICAgICBpZiAoIW9wdGlvbnMuc2lsZW50KSB7XG4gICAgICAgIGNvbnNvbGUubG9nKFxuICAgICAgICAgIGBMb2FkaW5nIHBob3RvICR7dXJsUGhvdG99IGZvciAke2lkZW50LnByZW5vbX0gJHtpZGVudC5ub2194oCmYCxcbiAgICAgICAgKVxuICAgICAgfVxuICAgICAgLy8gZm9yIChsZXQgcmV0cmllcyA9IDA7IHJldHJpZXMgPCAzOyByZXRyaWVzKyspIHtcbiAgICAgIC8vICAgY29uc3QgcmVzcG9uc2UgPSBhd2FpdCBmZXRjaCh1cmxQaG90bylcbiAgICAgIC8vICAgaWYgKHJlc3BvbnNlLm9rKSB7XG4gICAgICAvLyAgICAgYXdhaXQgcGlwZWxpbmUocmVzcG9uc2UuYm9keSwgZnMuY3JlYXRlV3JpdGVTdHJlYW0ocGhvdG9UZW1wRmlsZVBhdGgpKVxuICAgICAgLy8gICAgIGZzLnJlbmFtZVN5bmMocGhvdG9UZW1wRmlsZVBhdGgsIHBob3RvRmlsZVBhdGgpXG4gICAgICAvLyAgICAgYnJlYWtcbiAgICAgIC8vICAgfVxuICAgICAgLy8gICBpZiAocmV0cmllcyA+PSAyKSB7XG4gICAgICAvLyAgICAgY29uc29sZS53YXJuKGBGZXRjaCBmYWlsZWQ6ICR7dXJsUGhvdG99ICgke2lkZW50LnByZW5vbX0gJHtpZGVudC5ub219KWApXG4gICAgICAvLyAgICAgY29uc29sZS53YXJuKHJlc3BvbnNlLnN0YXR1cywgcmVzcG9uc2Uuc3RhdHVzVGV4dClcbiAgICAgIC8vICAgICBjb25zb2xlLndhcm4oYXdhaXQgcmVzcG9uc2UudGV4dCgpKVxuICAgICAgLy8gICAgIGlmIChmcy5leGlzdHNTeW5jKHBob3RvRmlsZVBhdGgpKSB7XG4gICAgICAvLyAgICAgICBjb25zb2xlLndhcm4oXCIgID0+IFJldXNpbmcgZXhpc3RpbmcgaW1hZ2VcIilcbiAgICAgIC8vICAgICB9IGVsc2Uge1xuICAgICAgLy8gICAgICAgY29uc29sZS53YXJuKFwiICA9PiBVc2luZyBibGFuayBpbWFnZVwiKVxuICAgICAgLy8gICAgICAgZnMuY29weUZpbGVTeW5jKG1pc3NpbmdQaG90b0ZpbGVQYXRoLCBwaG90b0ZpbGVQYXRoKVxuICAgICAgLy8gICAgIH1cbiAgICAgIC8vICAgICBicmVha1xuICAgICAgLy8gICB9XG4gICAgICAvLyB9XG4gICAgICB0cnkge1xuICAgICAgICBleGVjU3luYyhgd2dldCAtLXF1aWV0IC1PICR7cGhvdG9UZW1wRmlsZW5hbWV9ICR7dXJsUGhvdG99YCwge1xuICAgICAgICAgIGN3ZDogcGhvdG9zRGlyLFxuICAgICAgICAgIGVudjogcHJvY2Vzcy5lbnYsXG4gICAgICAgICAgZW5jb2Rpbmc6IFwidXRmLThcIixcbiAgICAgICAgICAvLyBzdGRpbzogW1wiaWdub3JlXCIsIFwiaWdub3JlXCIsIFwicGlwZVwiXSxcbiAgICAgICAgfSlcbiAgICAgICAgZnMucmVuYW1lU3luYyhwaG90b1RlbXBGaWxlUGF0aCwgcGhvdG9GaWxlUGF0aClcbiAgICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICAgIGNvbnNvbGUuZXJyb3IoZXJyb3IpXG4gICAgICAgIGlmIChmcy5leGlzdHNTeW5jKHBob3RvRmlsZVBhdGgpKSB7XG4gICAgICAgICAgY29uc29sZS53YXJuKFwiICA9PiBSZXVzaW5nIGV4aXN0aW5nIGltYWdlXCIpXG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgY29uc29sZS53YXJuKFwiICA9PiBVc2luZyBibGFuayBpbWFnZVwiKVxuICAgICAgICAgIGZzLmNvcHlGaWxlU3luYyhtaXNzaW5nUGhvdG9GaWxlUGF0aCwgcGhvdG9GaWxlUGF0aClcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cbiAgfVxuXG4gIC8vIFJlc2l6ZSBwaG90b3MgdG8gMTU1eDIyNSwgYmVjYXVzZSBzb21lIGhhdmVuJ3QgZXhhY3RseSB0aGlzIHNpemUuXG4gIGZvciAoY29uc3Qgc2VuYXRldXIgb2Ygc2VuYXRldXJzKSB7XG4gICAgY29uc3QgaWRlbnQgPSBzZW5hdGV1ci5ldGF0Q2l2aWwuaWRlbnRcbiAgICBjb25zdCBzZW5hdGV1clN0ZW0gPSBzdGVtRnJvbVNlbmF0ZXVyKHNlbmF0ZXVyKVxuICAgIGlmICghb3B0aW9ucy5zaWxlbnQpIHtcbiAgICAgIGNvbnNvbGUubG9nKFxuICAgICAgICBgUmVzaXppbmcgcGhvdG8gJHtzZW5hdGV1clN0ZW19IGZvciAke2lkZW50LnByZW5vbX0gJHtpZGVudC5ub2194oCmYCxcbiAgICAgIClcbiAgICB9XG4gICAgZXhlY1N5bmMoXG4gICAgICBgZ20gY29udmVydCAtcmVzaXplIDE1NXgyMjUhICR7c2VuYXRldXJTdGVtfS5qcGcgJHtzZW5hdGV1clN0ZW19XzE1NXgyMjUuanBnYCxcbiAgICAgIHtcbiAgICAgICAgY3dkOiBwaG90b3NEaXIsXG4gICAgICB9LFxuICAgIClcbiAgfVxuXG4gIC8vIENyZWF0ZSBhIG1vc2FpYyBvZiBwaG90b3MuXG4gIGNvbnN0IHBob3RvQnlJZFNlbmF0ZXVyOiB7IFt1aWQ6IHN0cmluZ106IFBob3RvIH0gPSB7fVxuICBjb25zdCByb3dzRmlsZW5hbWVzOiBzdHJpbmdbXSA9IFtdXG4gIGZvciAoXG4gICAgbGV0IHNlbmF0ZXVySW5kZXggPSAwLCByb3dJbmRleCA9IDA7XG4gICAgc2VuYXRldXJJbmRleCA8IHNlbmF0ZXVycy5sZW5ndGg7XG4gICAgc2VuYXRldXJJbmRleCArPSAyNSwgcm93SW5kZXgrK1xuICApIHtcbiAgICBjb25zdCByb3cgPSBzZW5hdGV1cnMuc2xpY2Uoc2VuYXRldXJJbmRleCwgc2VuYXRldXJJbmRleCArIDI1KVxuICAgIGNvbnN0IHBob3Rvc0ZpbGVuYW1lczogc3RyaW5nW10gPSBbXVxuICAgIGZvciAoY29uc3QgW2NvbHVtbkluZGV4LCBzZW5hdGV1cl0gb2Ygcm93LmVudHJpZXMoKSkge1xuICAgICAgY29uc3Qgc2VuYXRldXJTdGVtID0gc3RlbUZyb21TZW5hdGV1cihzZW5hdGV1cilcbiAgICAgIGNvbnN0IHBob3RvRmlsZW5hbWUgPSBgJHtzZW5hdGV1clN0ZW19XzE1NXgyMjUuanBnYFxuICAgICAgcGhvdG9zRmlsZW5hbWVzLnB1c2gocGhvdG9GaWxlbmFtZSlcbiAgICAgIHBob3RvQnlJZFNlbmF0ZXVyW3NlbmF0ZXVyLnVpZF0gPSB7XG4gICAgICAgIGNoZW1pbjogYHBob3Rvc19zZW5hdGV1cnMvJHtwaG90b0ZpbGVuYW1lfWAsXG4gICAgICAgIGNoZW1pbk1vc2FpcXVlOiBgcGhvdG9zX3NlbmF0ZXVycy9zZW5hdGV1cnMuanBnYCxcbiAgICAgICAgaGF1dGV1cjogMjI1LFxuICAgICAgICBsYXJnZXVyOiAxNTUsXG4gICAgICAgIHhNb3NhaXF1ZTogY29sdW1uSW5kZXggKiAxNTUsXG4gICAgICAgIHlNb3NhaXF1ZTogcm93SW5kZXggKiAyMjUsXG4gICAgICB9XG4gICAgfVxuICAgIGNvbnN0IHJvd0ZpbGVuYW1lID0gYHJvdy0ke3Jvd0luZGV4fS5qcGdgXG4gICAgZXhlY1N5bmMoYGdtIGNvbnZlcnQgJHtwaG90b3NGaWxlbmFtZXMuam9pbihcIiBcIil9ICthcHBlbmQgJHtyb3dGaWxlbmFtZX1gLCB7XG4gICAgICBjd2Q6IHBob3Rvc0RpcixcbiAgICB9KVxuICAgIHJvd3NGaWxlbmFtZXMucHVzaChyb3dGaWxlbmFtZSlcbiAgfVxuICBleGVjU3luYyhgZ20gY29udmVydCAke3Jvd3NGaWxlbmFtZXMuam9pbihcIiBcIil9IC1hcHBlbmQgc2VuYXRldXJzLmpwZ2AsIHtcbiAgICBjd2Q6IHBob3Rvc0RpcixcbiAgfSlcbiAgZm9yIChjb25zdCByb3dGaWxlbmFtZSBvZiByb3dzRmlsZW5hbWVzKSB7XG4gICAgZnMudW5saW5rU3luYyhwYXRoLmpvaW4ocGhvdG9zRGlyLCByb3dGaWxlbmFtZSkpXG4gIH1cblxuICBjb25zdCBqc29uRmlsZVBhdGggPSBwYXRoLmpvaW4ocGhvdG9zRGlyLCBcInNlbmF0ZXVycy5qc29uXCIpXG4gIGZzLndyaXRlRmlsZVN5bmMoanNvbkZpbGVQYXRoLCBKU09OLnN0cmluZ2lmeShwaG90b0J5SWRTZW5hdGV1ciwgbnVsbCwgMikpXG5cbiAgcmV0dXJuIGNvbW1pdEFuZFB1c2gocGhvdG9zRGlyLCBvcHRpb25zLmNvbW1pdCwgb3B0aW9ucy5yZW1vdGUpXG59XG5cbmZ1bmN0aW9uIHN0ZW1Gcm9tU2VuYXRldXIoc2VuYXRldXI6IEFjdGV1cikge1xuICBjb25zdCB7IGFkcmVzc2VzIH0gPSBzZW5hdGV1clxuICBsZXQgdXJsID0gbnVsbFxuICBmb3IgKGNvbnN0IGFkcmVzc2Ugb2YgYWRyZXNzZXMhKSB7XG4gICAgaWYgKFxuICAgICAgYWRyZXNzZS54c2lUeXBlID09PSBcIkFkcmVzc2VTaXRlV2ViX1R5cGVcIiAmJlxuICAgICAgYWRyZXNzZS50eXBlTGliZWxsZSA9PT0gXCJVcmwgc8OpbmF0ZXVyXCJcbiAgICApIHtcbiAgICAgIHVybCA9IGFkcmVzc2UudmFsRWxlY1xuICAgIH1cbiAgfVxuICBjb25zdCBtYXRjaCA9IHVybCEubWF0Y2goL15odHRwczpcXC9cXC93d3dcXC5zZW5hdFxcLmZyXFwvc2VuYXRldXJcXC8oLiopXFwuaHRtbCQvKVxuICBjb25zdCBzZW5hdGV1clN0ZW0gPSBtYXRjaCFbMV1cbiAgcmV0dXJuIHNlbmF0ZXVyU3RlbVxufVxuXG5yZXRyaWV2ZVBob3Rvc1NlbmF0ZXVycygpXG4gIC50aGVuKChlcnJvcikgPT4gcHJvY2Vzcy5leGl0KGVycm9yID8gMSA6IDApKVxuICAuY2F0Y2goKGVycm9yKSA9PiB7XG4gICAgY29uc29sZS5sb2coZXJyb3IpXG4gICAgcHJvY2Vzcy5leGl0KDIpXG4gIH0pXG4iXSwibWFwcGluZ3MiOiJBQUFBLFNBQXVCQSxRQUF2QixRQUF1QyxlQUF2QztBQUNBLE9BQU9DLGVBQVAsTUFBNEIsbUJBQTVCO0FBQ0EsT0FBT0MsRUFBUCxNQUFlLFVBQWYsQyxDQUNB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