@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,{"version":3,"names":["execSync","commandLineArgs","fs","path","EnabledDatasets","loadAssembleeData","CodeTypeOrgane","TypeMandat","optionsDefinitions","alias","help","name","type","String","Boolean","multiple","defaultOption","options","commitAndPush","repositoryDir","commit","remotes","error","cwd","env","process","encoding","stdio","childProcess","stderr","test","stdout","console","output","remote","retrievePhotosSenateurs","dataDir","acteurByUid","ActeursEtOrganes","legislature","senateurs","Object","values","filter","acteur","mandats","mandatSenateur","mandat","xsiType","MandatParlementaireType","typeOrgane","Senat","undefined","adresses","url","adresse","typeLibelle","valElec","sort","a","b","uid","length","localeCompare","gitGroupUrl","clone","trim","replace","photosDirName","photosDir","join","ensureDirSync","fetch","missingPhotoFilePath","senateur","ident","etatCivil","senateurStem","stemFromSenateur","photoFilename","photoFilePath","photoTempFilename","photoTempFilePath","urlPhoto","silent","log","prenom","nom","renameSync","existsSync","warn","copyFileSync","photoByIdSenateur","rowsFilenames","senateurIndex","rowIndex","row","slice","photosFilenames","columnIndex","entries","push","chemin","cheminMosaique","hauteur","largeur","xMosaique","yMosaique","rowFilename","unlinkSync","jsonFilePath","writeFileSync","JSON","stringify","match","then","exit","catch"],"sources":["../../src/scripts/retrieve_senateurs_photos.ts"],"sourcesContent":["import { ChildProcess, execSync } from \"child_process\"\nimport commandLineArgs from \"command-line-args\"\nimport fs from \"fs-extra\"\n// import fetch from \"node-fetch\"\nimport path from \"path\"\n// import stream from \"stream\"\n// import util from \"util\"\n\nimport { EnabledDatasets } from \"../datasets\"\nimport { loadAssembleeData } from \"../loaders\"\nimport {\n  Acteur,\n  CodeTypeOrgane,\n  Mandat,\n  Photo,\n  TypeMandat,\n} from \"../types/acteurs_et_organes\"\n\nconst optionsDefinitions = [\n  {\n    alias: \"C\",\n    help: \"clone repositories from given group (or organization) git URL\",\n    name: \"clone\",\n    type: String,\n  },\n  {\n    alias: \"c\",\n    help: \"commit photos\",\n    name: \"commit\",\n    type: Boolean,\n  },\n  {\n    alias: \"f\",\n    help: \"fetch sénateurs' pictures instead of retrieving them from files\",\n    name: \"fetch\",\n    type: Boolean,\n  },\n  {\n    alias: \"r\",\n    help: \"push commit to given remote\",\n    multiple: true,\n    name: \"remote\",\n    type: String,\n  },\n  {\n    alias: \"s\",\n    help: \"don't log anything\",\n    name: \"silent\",\n    type: Boolean,\n  },\n  {\n    defaultOption: true,\n    help: \"directory containing Assemblée open data files\",\n    name: \"dataDir\",\n    type: String,\n  },\n]\nconst options = commandLineArgs(optionsDefinitions)\n// const pipeline = util.promisify(stream.pipeline)\n\nfunction commitAndPush(\n  repositoryDir: string,\n  commit: boolean,\n  remotes?: string[],\n): boolean {\n  let error = false\n  if (commit) {\n    execSync(\"git add .\", {\n      cwd: repositoryDir,\n      env: process.env,\n      encoding: \"utf-8\",\n      stdio: [\"ignore\", \"ignore\", \"pipe\"],\n    })\n    try {\n      execSync('git commit -m \"Nouvelle moisson\"', {\n        cwd: repositoryDir,\n        env: process.env,\n        encoding: \"utf-8\",\n      })\n    } catch (childProcess) {\n      if (\n        (childProcess as ChildProcess).stderr === null ||\n        !/nothing to commit/.test((childProcess as any).stdout)\n      ) {\n        console.error((childProcess as any).output)\n        throw childProcess\n      }\n    }\n    for (const remote of remotes || []) {\n      try {\n        execSync(`git push ${remote} master`, {\n          cwd: repositoryDir,\n          env: process.env,\n          encoding: \"utf-8\",\n          stdio: [\"ignore\", \"ignore\", \"pipe\"],\n        })\n      } catch (childProcess) {\n        // Don't stop when push fails.\n        console.error((childProcess as any).output)\n        error = true\n      }\n    }\n  }\n  return error\n}\n\nasync function retrievePhotosSenateurs(): Promise<boolean> {\n  const dataDir = options.dataDir\n  const { acteurByUid } = loadAssembleeData(\n    dataDir,\n    EnabledDatasets.ActeursEtOrganes,\n    options.legislature,\n  )\n  const senateurs = Object.values(acteurByUid)\n    .filter((acteur) => {\n      const { mandats } = acteur\n      if (!mandats) {\n        return false\n      }\n      const mandatSenateur = mandats.filter(\n        (mandat: Mandat) =>\n          mandat.xsiType === TypeMandat.MandatParlementaireType &&\n          // mandat.legislature === options.legislature &&\n          mandat.typeOrgane === CodeTypeOrgane.Senat,\n      )[0]\n      if (mandatSenateur === undefined) {\n        return false\n      }\n      const { adresses } = acteur\n      if (adresses === undefined) {\n        return false\n      }\n      let url = null\n      for (const adresse of adresses) {\n        if (\n          adresse.xsiType === \"AdresseSiteWeb_Type\" &&\n          adresse.typeLibelle === \"Url sénateur\"\n        ) {\n          url = adresse.valElec\n        }\n      }\n      return Boolean(url)\n    })\n    .sort((a, b) =>\n      a.uid.length === b.uid.length\n        ? a.uid.localeCompare(b.uid)\n        : a.uid.length - b.uid.length,\n    )\n\n  const gitGroupUrl = options.clone\n    ? options.clone.trim().replace(/\\/+$/, \"\")\n    : undefined\n  const photosDirName = \"photos_senateurs\"\n  if (gitGroupUrl !== undefined) {\n    execSync(`git clone ${gitGroupUrl}/${photosDirName}.git`, {\n      cwd: dataDir,\n      env: process.env,\n      encoding: \"utf-8\",\n      stdio: [\"ignore\", \"ignore\", \"pipe\"],\n    })\n  }\n  const photosDir = path.join(dataDir, photosDirName)\n\n  // Download photos.\n  fs.ensureDirSync(photosDir)\n  if (options.fetch) {\n    const missingPhotoFilePath = \"images/transparent_155x225.jpg\"\n    for (const senateur of senateurs) {\n      const ident = senateur.etatCivil.ident\n      const senateurStem = stemFromSenateur(senateur)\n      const photoFilename = `${senateurStem}.jpg`\n      const photoFilePath = path.join(photosDir, photoFilename)\n      const photoTempFilename = `${senateurStem}_temp.jpg`\n      const photoTempFilePath = path.join(photosDir, photoTempFilename)\n      const urlPhoto = `https://www.senat.fr/senimg/${photoFilename}`\n      if (!options.silent) {\n        console.log(\n          `Loading photo ${urlPhoto} for ${ident.prenom} ${ident.nom}…`,\n        )\n      }\n      // for (let retries = 0; retries < 3; retries++) {\n      //   const response = await fetch(urlPhoto)\n      //   if (response.ok) {\n      //     await pipeline(response.body, fs.createWriteStream(photoTempFilePath))\n      //     fs.renameSync(photoTempFilePath, photoFilePath)\n      //     break\n      //   }\n      //   if (retries >= 2) {\n      //     console.warn(`Fetch failed: ${urlPhoto} (${ident.prenom} ${ident.nom})`)\n      //     console.warn(response.status, response.statusText)\n      //     console.warn(await response.text())\n      //     if (fs.existsSync(photoFilePath)) {\n      //       console.warn(\"  => Reusing existing image\")\n      //     } else {\n      //       console.warn(\"  => Using blank image\")\n      //       fs.copyFileSync(missingPhotoFilePath, photoFilePath)\n      //     }\n      //     break\n      //   }\n      // }\n      try {\n        execSync(`wget --quiet -O ${photoTempFilename} ${urlPhoto}`, {\n          cwd: photosDir,\n          env: process.env,\n          encoding: \"utf-8\",\n          // stdio: [\"ignore\", \"ignore\", \"pipe\"],\n        })\n        fs.renameSync(photoTempFilePath, photoFilePath)\n      } catch (error) {\n        console.error(error)\n        if (fs.existsSync(photoFilePath)) {\n          console.warn(\"  => Reusing existing image\")\n        } else {\n          console.warn(\"  => Using blank image\")\n          fs.copyFileSync(missingPhotoFilePath, photoFilePath)\n        }\n      }\n    }\n  }\n\n  // Resize photos to 155x225, because some haven't exactly this size.\n  for (const senateur of senateurs) {\n    const ident = senateur.etatCivil.ident\n    const senateurStem = stemFromSenateur(senateur)\n    if (!options.silent) {\n      console.log(\n        `Resizing photo ${senateurStem} for ${ident.prenom} ${ident.nom}…`,\n      )\n    }\n    execSync(\n      `gm convert -resize 155x225! ${senateurStem}.jpg ${senateurStem}_155x225.jpg`,\n      {\n        cwd: photosDir,\n      },\n    )\n  }\n\n  // Create a mosaic of photos.\n  const photoByIdSenateur: { [uid: string]: Photo } = {}\n  const rowsFilenames: string[] = []\n  for (\n    let senateurIndex = 0, rowIndex = 0;\n    senateurIndex < senateurs.length;\n    senateurIndex += 25, rowIndex++\n  ) {\n    const row = senateurs.slice(senateurIndex, senateurIndex + 25)\n    const photosFilenames: string[] = []\n    for (const [columnIndex, senateur] of row.entries()) {\n      const senateurStem = stemFromSenateur(senateur)\n      const photoFilename = `${senateurStem}_155x225.jpg`\n      photosFilenames.push(photoFilename)\n      photoByIdSenateur[senateur.uid] = {\n        chemin: `photos_senateurs/${photoFilename}`,\n        cheminMosaique: `photos_senateurs/senateurs.jpg`,\n        hauteur: 225,\n        largeur: 155,\n        xMosaique: columnIndex * 155,\n        yMosaique: rowIndex * 225,\n      }\n    }\n    const rowFilename = `row-${rowIndex}.jpg`\n    execSync(`gm convert ${photosFilenames.join(\" \")} +append ${rowFilename}`, {\n      cwd: photosDir,\n    })\n    rowsFilenames.push(rowFilename)\n  }\n  execSync(`gm convert ${rowsFilenames.join(\" \")} -append senateurs.jpg`, {\n    cwd: photosDir,\n  })\n  for (const rowFilename of rowsFilenames) {\n    fs.unlinkSync(path.join(photosDir, rowFilename))\n  }\n\n  const jsonFilePath = path.join(photosDir, \"senateurs.json\")\n  fs.writeFileSync(jsonFilePath, JSON.stringify(photoByIdSenateur, null, 2))\n\n  return commitAndPush(photosDir, options.commit, options.remote)\n}\n\nfunction stemFromSenateur(senateur: Acteur) {\n  const { adresses } = senateur\n  let url = null\n  for (const adresse of adresses!) {\n    if (\n      adresse.xsiType === \"AdresseSiteWeb_Type\" &&\n      adresse.typeLibelle === \"Url sénateur\"\n    ) {\n      url = adresse.valElec\n    }\n  }\n  const match = url!.match(/^https:\\/\\/www\\.senat\\.fr\\/senateur\\/(.*)\\.html$/)\n  const senateurStem = match![1]\n  return senateurStem\n}\n\nretrievePhotosSenateurs()\n  .then((error) => process.exit(error ? 1 : 0))\n  .catch((error) => {\n    console.log(error)\n    process.exit(2)\n  })\n"],"mappings":"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"}