@tricoteuses/assemblee 1.1.12 → 1.2.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 (69) hide show
  1. package/README.md +5 -5
  2. package/lib/bugs/agenda-00011.js +4 -5
  3. package/lib/bugs/agenda-00011.mjs +1 -2
  4. package/lib/bugs.js +3 -1
  5. package/lib/cleaners/actes_legislatifs.js +3 -1
  6. package/lib/cleaners/acteurs.js +3 -1
  7. package/lib/cleaners/amendements.js +3 -1
  8. package/lib/cleaners/documents.js +3 -1
  9. package/lib/cleaners/dossiers_legislatifs.js +3 -1
  10. package/lib/cleaners/reunions.js +2 -1
  11. package/lib/cleaners/scrutins.js +3 -1
  12. package/lib/cleaners/xml.js +3 -1
  13. package/lib/dates.js +1 -1
  14. package/lib/dossiers_legislatifs.d.ts +2 -1
  15. package/lib/dossiers_legislatifs.js +45 -2
  16. package/lib/dossiers_legislatifs.mjs +7 -1
  17. package/lib/file_systems.js +4 -1
  18. package/lib/index.d.ts +2 -2
  19. package/lib/index.js +13 -1
  20. package/lib/index.mjs +3 -3
  21. package/lib/inserters.js +3 -1
  22. package/lib/loaders.js +4 -1
  23. package/lib/parsers/documents.js +3 -1
  24. package/lib/parsers/recherche_amendements.js +4 -5
  25. package/lib/parsers/recherche_amendements.mjs +1 -2
  26. package/lib/raw_types/acteurs_et_organes.js +2 -1
  27. package/lib/raw_types/agendas.js +2 -1
  28. package/lib/raw_types/amendements.d.ts +1 -1
  29. package/lib/raw_types/amendements.js +3 -2
  30. package/lib/raw_types/amendements.mjs +2 -2
  31. package/lib/raw_types/debats.js +2 -1
  32. package/lib/raw_types/dossiers_legislatifs.js +2 -1
  33. package/lib/raw_types/questions.js +2 -1
  34. package/lib/raw_types/scrutins.js +2 -1
  35. package/lib/scripts/bugs_helper.js +3 -1
  36. package/lib/scripts/clean_reorganized_data.js +4 -2
  37. package/lib/scripts/clean_reorganized_data.mjs +2 -2
  38. package/lib/scripts/diff_amendements.js +3 -1
  39. package/lib/scripts/document_dossiers_legislatifs.js +2 -1
  40. package/lib/scripts/merge_scrutins.js +4 -1
  41. package/lib/scripts/parse_textes_lois.js +3 -1
  42. package/lib/scripts/raw_types_from_amendements.js +3 -1
  43. package/lib/scripts/reorganize_data.js +4 -2
  44. package/lib/scripts/reorganize_data.mjs +2 -2
  45. package/lib/scripts/retrieve_deputes_photos.js +4 -3
  46. package/lib/scripts/retrieve_deputes_photos.mjs +1 -2
  47. package/lib/scripts/retrieve_documents.js +342 -521
  48. package/lib/scripts/retrieve_documents.mjs +180 -229
  49. package/lib/scripts/retrieve_open_data.js +4 -4
  50. package/lib/scripts/retrieve_open_data.mjs +2 -3
  51. package/lib/scripts/retrieve_pending_amendments.js +3 -1
  52. package/lib/scripts/retrieve_senateurs_photos.js +5 -4
  53. package/lib/scripts/retrieve_senateurs_photos.mjs +2 -3
  54. package/lib/scripts/retrieve_textes_lois.js +7 -4
  55. package/lib/scripts/retrieve_textes_lois.mjs +5 -2
  56. package/lib/scripts/validate_json.js +3 -1
  57. package/lib/types/acteurs_et_organes.js +2 -1
  58. package/lib/types/agendas.js +2 -1
  59. package/lib/types/amendements.d.ts +1 -1
  60. package/lib/types/amendements.js +3 -2
  61. package/lib/types/amendements.mjs +2 -2
  62. package/lib/types/debats.js +2 -1
  63. package/lib/types/dossiers_legislatifs.js +2 -1
  64. package/lib/types/questions.js +2 -1
  65. package/lib/types/scrutins.js +2 -1
  66. package/lib/urls.d.ts +8 -2
  67. package/lib/urls.js +153 -23
  68. package/lib/urls.mjs +97 -22
  69. package/package.json +3 -5
@@ -1,246 +1,197 @@
1
- function _defineProperty(obj, key, value) { key = _toPropertyKey(key); if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
2
- function _toPropertyKey(arg) { var key = _toPrimitive(arg, "string"); return typeof key === "symbol" ? key : String(key); }
3
- function _toPrimitive(input, hint) { if (typeof input !== "object" || input === null) return input; var prim = input[Symbol.toPrimitive]; if (prim !== undefined) { var res = prim.call(input, hint || "default"); if (typeof res !== "object") return res; throw new TypeError("@@toPrimitive must return a primitive value."); } return (hint === "string" ? String : Number)(input); }
1
+ import assert from "assert";
2
+ import { execSync } from "child_process";
4
3
  import commandLineArgs from "command-line-args";
5
- import commandLineUsage from "command-line-usage";
6
- import * as git from "../../src/git.mjs";
4
+ import { createHash } from "crypto";
7
5
  import fs from "fs-extra";
8
- let fetch = require("node-fetch");
9
6
  import path from "path";
10
- import { load, getFiles } from "../file_systems.mjs";
11
- import { DocumentUrlFormat, urlFromDocument } from "../urls.mjs";
12
- function parseArgs(argv) {
13
- const optionsDefinitions = [{
14
- alias: "c",
15
- help: "commit documents",
16
- name: "commit",
17
- type: Boolean
18
- }, {
19
- help: "dry run",
20
- name: "dry-run",
21
- type: Boolean
22
- }, {
23
- help: "retry failed documents",
24
- name: "retry-failed",
25
- type: Boolean
26
- }, {
27
- help: "retry all documents",
28
- name: "retry",
29
- type: Boolean
30
- }, {
31
- alias: "l",
32
- defaultValue: "16",
33
- name: "legislature",
34
- type: String
35
- }, {
36
- alias: "r",
37
- help: "push commit to given remote",
38
- name: "remote",
39
- type: String
40
- }, {
41
- alias: "s",
42
- help: "don't log anything",
43
- name: "silent",
44
- type: Boolean
45
- }, {
46
- help: "textes dir",
47
- name: "textes",
48
- type: String
49
- }, {
50
- alias: "v",
51
- help: "verbose logs",
52
- name: "verbose",
53
- type: Boolean
54
- }, {
55
- defaultOption: true,
56
- help: "files or globs to documents",
57
- multiple: true,
58
- name: "data",
59
- type: String
60
- }];
61
- const sections = [{
62
- header: "Retrieve documents",
63
- content: "Retrieve documents"
64
- }, {
65
- header: "Options",
66
- optionList: optionsDefinitions
67
- }];
68
- const options = commandLineArgs(optionsDefinitions, {
69
- argv: argv
70
- });
71
- if ("help" in options) {
72
- const usage = commandLineUsage(sections);
73
- console.warn(usage);
74
- return null;
7
+ import { walkDocumentAndDivisions } from "../dossiers_legislatifs.mjs";
8
+ import * as git from "../git.mjs";
9
+ import { iterLoadAssembleeDocuments, pathFromDocumentUid } from "../loaders.mjs";
10
+ import { DocumentUrlFormat, iterDocumentOrDivisionUrls } from "../urls.mjs";
11
+ const optionsDefinitions = [{
12
+ alias: "c",
13
+ help: "commit documents",
14
+ name: "commit",
15
+ type: Boolean
16
+ }, {
17
+ alias: "f",
18
+ help: "retrieve all documents, even already retrieved ones",
19
+ name: "full",
20
+ type: Boolean
21
+ }, {
22
+ alias: "l",
23
+ defaultValue: "16",
24
+ help: 'legislatures to retrieve, "*" for all',
25
+ name: "legislature",
26
+ type: String
27
+ }, {
28
+ alias: "n",
29
+ help: "try to also retrieve documents that were previously not found",
30
+ name: "not-found",
31
+ type: Boolean
32
+ }, {
33
+ alias: "p",
34
+ help: "pull repositories before proceeding",
35
+ name: "pull",
36
+ type: Boolean
37
+ }, {
38
+ alias: "r",
39
+ help: "push commit to given remote",
40
+ multiple: true,
41
+ name: "remote",
42
+ type: String
43
+ }, {
44
+ alias: "s",
45
+ help: "don't log anything",
46
+ name: "silent",
47
+ type: Boolean
48
+ }, {
49
+ alias: "u",
50
+ help: "UID of first Assemblée's document or division to retrieve",
51
+ name: "uid",
52
+ type: String
53
+ }, {
54
+ alias: "v",
55
+ help: "verbose logs",
56
+ name: "verbose",
57
+ type: Boolean
58
+ }, {
59
+ defaultOption: true,
60
+ help: "directory containing Assemblée open data files",
61
+ name: "dataDir",
62
+ type: String
63
+ }];
64
+ const options = commandLineArgs(optionsDefinitions);
65
+ async function retrieveDocuments() {
66
+ assert(!options.commit || options.uid, 'Options "commit" & "uid" are incompatible');
67
+ const dataDir = options.dataDir;
68
+ const documentsDir = path.join(dataDir, "Documents");
69
+ if (options.pull) {
70
+ execSync(`git reset --hard origin/master`, {
71
+ cwd: documentsDir,
72
+ env: process.env,
73
+ encoding: "utf-8",
74
+ stdio: ["ignore", "ignore", "pipe"]
75
+ });
76
+ execSync(`git pull --rebase`, {
77
+ cwd: documentsDir,
78
+ env: process.env,
79
+ encoding: "utf-8",
80
+ stdio: ["ignore", "ignore", "pipe"]
81
+ });
75
82
  }
76
- return options;
77
- }
78
- class Retrieve {
79
- constructor(options) {
80
- _defineProperty(this, "options", void 0);
81
- this.options = options;
82
- }
83
- getDivisions(division) {
84
- let divisions = [division];
85
- if (division.divisions != undefined) {
86
- for (const d of division.divisions) divisions = divisions.concat(this.getDivisions(d));
87
- }
88
- return divisions;
89
- }
90
- getDivisionsOrDocument(document) {
91
- if (document.divisions != undefined) return [].concat(...document.divisions.map(d => this.getDivisions(d)));else return [document];
92
- }
93
- async commitAndPush(repositoryDir) {
94
- if (this.options.commit == undefined) return;
95
- git.commit(repositoryDir, "Nouvelle moisson");
96
- if (this.options.remote == undefined) return;
97
- git.run(repositoryDir, `push ${this.options.remote} HEAD:master`);
98
- }
99
- getUrls(divisionOrDocument, format) {
100
- const ext = format == "html" ? "raw" : "pdf";
101
- let urls = [`http://www.assemblee-nationale.fr/dyn/docs/${divisionOrDocument.uid}.${ext}`];
102
- if (format == "html") {
103
- urls.push(urlFromDocument(divisionOrDocument, DocumentUrlFormat.RawHtml));
104
- urls.push(urlFromDocument(divisionOrDocument, DocumentUrlFormat.Html));
105
- } else if (format == "pdf") {
106
- urls.push(urlFromDocument(divisionOrDocument, DocumentUrlFormat.Pdf));
107
- }
108
- return urls;
109
- }
110
- static sanityCheckFormat(page, format) {
111
- if (format == "pdf") {
112
- const length = Math.min(page.length, 4);
113
- const head = page.slice(0, length).toString();
114
- if (!head.startsWith("%PDF")) return [Buffer.from(""), `bad pdf ${encodeURIComponent(head)}...`];
115
- } else if (format == "html") {
116
- const length = Math.min(page.length, 50);
117
- const head = page.slice(0, length).toString();
118
- if (!head.includes("DOCTYPE") && !head.includes("<?xml")) {
119
- return [Buffer.from(""), `bad html ${encodeURIComponent(head.slice(0, 4))}...`];
120
- }
121
- }
122
- return [page, "OK"];
123
- }
124
- async needsDownload(filePath, uid) {
125
- if (fs.existsSync(filePath)) {
126
- const isNotEmpty = fs.statSync(filePath).size > 0;
127
- const retryFailed = ("retry-failed" in this.options);
128
- if (!retryFailed || isNotEmpty) {
129
- if (this.options.verbose) {
130
- if (isNotEmpty) {
131
- const urlPath = `${filePath}.url`;
132
- let url;
133
- if (fs.existsSync(urlPath)) url = fs.readFileSync(urlPath).toString();else url = "<unknown url>";
134
- console.log(`${uid}: Cached OK ${filePath} from ${url}`);
135
- } else {
136
- console.log(`${uid}: Cached FAIL ${filePath}`);
137
- }
138
- }
139
- return false;
140
- }
141
- }
142
- return true;
143
- }
144
- async get(url, filePath, format, uid) {
145
- if (this.options.verbose) console.log(`${uid}: fetch(${url})`);
146
- const response = await fetch(url);
147
- let page = await response.buffer();
148
- let status;
149
- if (!response.ok) {
150
- status = response.status;
151
- if (response.status !== 404) {
152
- if (this.options.verbose) {
153
- console.error(`Error while getting page "${url}" (uid: ${uid}):\n\nError:\n${JSON.stringify({
154
- code: response.status,
155
- message: response.statusText
156
- }, null, 2)}`);
157
- }
83
+ fs.ensureDirSync(documentsDir);
84
+ if (options.full && !options.uid) {
85
+ for (const filename of fs.readdirSync(documentsDir)) {
86
+ if (filename[0] === ".") {
87
+ continue;
158
88
  }
159
- fs.writeFileSync(filePath, "");
160
- } else {
161
- ;
162
- [page, status] = Retrieve.sanityCheckFormat(page, format);
163
- fs.writeFileSync(filePath, page);
89
+ fs.removeSync(path.join(documentsDir, filename));
164
90
  }
165
- return status;
166
91
  }
167
- static urlFromUid(uid) {
168
- const url = "https://git.en-root.org/tricoteuses/data.tricoteuses.fr/Dossiers_Legislatifs_XV/-/tree/master/documents";
169
- return `[${uid}](${url}/uid/${uid}.json)`;
170
- }
171
- async download(textesDir, divisionOrDocument, format) {
172
- const uid = divisionOrDocument.uid;
173
- const filePath = `${textesDir}/uid/${uid}.${format}`;
174
- if (!(await this.needsDownload(filePath, uid))) return 0;
175
- if (this.options.verbose) console.log(`${uid}: Checking ${filePath}`);
176
- let success = false;
177
- let info = "";
178
- let url;
179
- for (url of this.getUrls(divisionOrDocument, format)) {
180
- if ("dry-run" in this.options) {
181
- info += ` * ${url}\n`;
92
+ const firstUid = options.uid;
93
+ let skip = Boolean(firstUid);
94
+ for (const {
95
+ document
96
+ } of iterLoadAssembleeDocuments(dataDir, options.legislature)) {
97
+ for (const documentOrDivision of walkDocumentAndDivisions(document)) {
98
+ // Ignore documents from Sénat.
99
+ if (documentOrDivision.uid.substring(4, 6) === "SN") {
182
100
  continue;
183
- } else {
184
- info += ` * ${url} `;
185
101
  }
186
- const status = await this.get(url, filePath, format, uid);
187
- info += `${status}\n`;
188
- if (status == "OK") {
189
- success = true;
190
- break;
102
+ if (skip) {
103
+ if (documentOrDivision.uid === firstUid) {
104
+ skip = false;
105
+ } else {
106
+ continue;
107
+ }
191
108
  }
192
- }
193
- const urlPath = `${filePath}.url`;
194
- if (success) {
195
- fs.writeFileSync(urlPath, url);
196
- } else if (fs.existsSync(urlPath)) {
197
- fs.unlinkSync(urlPath);
198
- }
199
- if (info != "" && !this.options.silent) {
200
- console.log(`* ${Retrieve.urlFromUid(uid)}: Retrieving ${filePath}\n${info.slice(0, -1)}`);
201
- return 1;
202
- } else {
203
- return 0;
204
- }
205
- }
206
- async retrieveDocuments() {
207
- const gitDir = `${this.options.textes}/.git`;
208
- if (!fs.existsSync(gitDir)) {
209
- console.error(`${gitDir} is expected to exist but does not`);
210
- return 1;
211
- }
212
- const textesDir = path.join(this.options.textes, this.options.legislature);
213
- fs.ensureDirSync(`${textesDir}/uid`);
214
- for (const filename of getFiles(this.options.data)) {
215
- const document = load(filename);
216
- if (/^....SN/.test(document.uid) || /^DECL/.test(document.uid)) continue;
217
- for (const divisionOrDocument of this.getDivisionsOrDocument(document)) {
218
- if (/-COMPA$/.test(divisionOrDocument.uid)) continue;
219
- let status = 0;
220
- for (const format of ["html", "pdf"]) {
221
- status |= await this.download(textesDir, divisionOrDocument, format);
109
+ const documentDir = pathFromDocumentUid(documentsDir, documentOrDivision.uid);
110
+ fs.ensureDirSync(documentDir);
111
+ const filenameBySha256 = {};
112
+ const indexPath = path.join(documentDir, "index.json");
113
+ const index = fs.pathExistsSync(indexPath) ? fs.readJsonSync(indexPath) : {};
114
+ for (const {
115
+ format,
116
+ type,
117
+ url
118
+ } of iterDocumentOrDivisionUrls(documentOrDivision)) {
119
+ if (index[format] === undefined) {
120
+ index[format] = [];
121
+ }
122
+ let fileInfos = index[format].find(fileInfos => fileInfos.url === url);
123
+ if (fileInfos === undefined) {
124
+ fileInfos = {};
125
+ index[format].push(fileInfos);
126
+ }
127
+ fileInfos.url = url;
128
+ const filename = `${type}.${format === DocumentUrlFormat.Pdf ? "pdf" : "html"}`;
129
+ if (fileInfos.status === 200 && !options.full) {
130
+ filenameBySha256[fileInfos.sha256] = filename;
131
+ continue;
132
+ }
133
+ if (fileInfos.status === 404 && !options["not-found"]) {
134
+ continue;
222
135
  }
223
- if (status && !this.options.silent) console.log("---");
136
+ if (!options.silent) {
137
+ console.log(`Retrieving document or division ${documentOrDivision.uid} at ${url}…`);
138
+ }
139
+ const response = await fetch(url);
140
+ const filePath = path.join(documentDir, filename);
141
+ if (response.ok) {
142
+ const arrayBuffer = await response.arrayBuffer();
143
+ const buffer = Buffer.from(arrayBuffer);
144
+ if (format === DocumentUrlFormat.Pdf && !buffer.subarray(0, 4).toString().startsWith("%PDF")) {
145
+ // Instead of a PDF, the received data may be an HTML page with a message like
146
+ // "Document non encore publié".
147
+ if (!options.silent) {
148
+ console.warn(` PDF "${url}" not found.`);
149
+ }
150
+ fs.removeSync(filePath);
151
+ delete fileInfos.filename;
152
+ delete fileInfos.sha256;
153
+ } else {
154
+ const sha256 = createHash("sha256").update(buffer).digest("hex");
155
+ const existingFilename = filenameBySha256[sha256];
156
+ if (existingFilename === undefined) {
157
+ fs.createWriteStream(filePath).write(buffer);
158
+ fileInfos.filename = filename;
159
+ filenameBySha256[sha256] = filename;
160
+ } else {
161
+ fileInfos.filename = existingFilename;
162
+ }
163
+ fileInfos.sha256 = sha256;
164
+ }
165
+ } else {
166
+ if (response.status === 404) {
167
+ if (!options.silent) {
168
+ console.warn(` Page "${url}" not found.`);
169
+ }
170
+ } else {
171
+ console.error(` Error:\n${JSON.stringify({
172
+ code: response.status,
173
+ message: response.statusText
174
+ }, null, 2)}`);
175
+ }
176
+ fs.removeSync(filePath);
177
+ delete fileInfos.filename;
178
+ delete fileInfos.sha256;
179
+ }
180
+ fileInfos.status = response.status;
224
181
  }
182
+ fs.writeJsonSync(indexPath, index, {
183
+ encoding: "utf-8",
184
+ spaces: 2
185
+ });
225
186
  }
226
- if (!("dry-run" in this.options)) {
227
- await this.commitAndPush(textesDir);
228
- }
229
- return 0;
230
187
  }
188
+ if (options.commit) {
189
+ return git.commitAndPush(documentsDir, "Nouvelle moisson", options.remote);
190
+ }
191
+ return 0;
231
192
  }
232
- async function main(argv) {
233
- const options = parseArgs(argv);
234
- if (options === null) return 1;
235
- const retrieve = new Retrieve(options);
236
- return retrieve.retrieveDocuments();
237
- }
238
-
239
- /* istanbul ignore if */
240
- if (process.argv[1].endsWith("retrieve_documents.ts")) main(process.argv).then(exitCode => {
241
- process.exit(exitCode);
242
- }).catch(error => {
243
- console.error(error);
193
+ retrieveDocuments().then(exitCode => process.exit(exitCode)).catch(error => {
194
+ console.log(error);
244
195
  process.exit(1);
245
196
  });
246
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["commandLineArgs","commandLineUsage","git","fs","fetch","require","path","load","getFiles","DocumentUrlFormat","urlFromDocument","parseArgs","argv","optionsDefinitions","alias","help","name","type","Boolean","defaultValue","String","defaultOption","multiple","sections","header","content","optionList","options","usage","console","warn","Retrieve","constructor","_defineProperty","getDivisions","division","divisions","undefined","d","concat","getDivisionsOrDocument","document","map","commitAndPush","repositoryDir","commit","remote","run","getUrls","divisionOrDocument","format","ext","urls","uid","push","RawHtml","Html","Pdf","sanityCheckFormat","page","length","Math","min","head","slice","toString","startsWith","Buffer","from","encodeURIComponent","includes","needsDownload","filePath","existsSync","isNotEmpty","statSync","size","retryFailed","verbose","urlPath","url","readFileSync","log","get","response","buffer","status","ok","error","JSON","stringify","code","message","statusText","writeFileSync","urlFromUid","download","textesDir","success","info","unlinkSync","silent","retrieveDocuments","gitDir","textes","join","legislature","ensureDirSync","filename","data","test","main","retrieve","process","endsWith","then","exitCode","exit","catch"],"sources":["../../src/scripts/retrieve_documents.ts"],"sourcesContent":["import commandLineArgs from \"command-line-args\"\nimport commandLineUsage from \"command-line-usage\"\nimport * as git from \"../../src/git\"\nimport fs from \"fs-extra\"\nlet fetch = require(\"node-fetch\")\nimport path from \"path\"\n\nimport { load, getFiles } from \"../file_systems\"\nimport { DocumentUrlFormat, urlFromDocument } from \"../urls\"\n\nfunction parseArgs(argv: string[]): any {\n  const optionsDefinitions = [\n    {\n      alias: \"c\",\n      help: \"commit documents\",\n      name: \"commit\",\n      type: Boolean,\n    },\n    {\n      help: \"dry run\",\n      name: \"dry-run\",\n      type: Boolean,\n    },\n    {\n      help: \"retry failed documents\",\n      name: \"retry-failed\",\n      type: Boolean,\n    },\n    {\n      help: \"retry all documents\",\n      name: \"retry\",\n      type: Boolean,\n    },\n    {\n      alias: \"l\",\n      defaultValue: \"16\",\n      name: \"legislature\",\n      type: String,\n    },\n    {\n      alias: \"r\",\n      help: \"push commit to given remote\",\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      help: \"textes dir\",\n      name: \"textes\",\n      type: String,\n    },\n    {\n      alias: \"v\",\n      help: \"verbose logs\",\n      name: \"verbose\",\n      type: Boolean,\n    },\n    {\n      defaultOption: true,\n      help: \"files or globs to documents\",\n      multiple: true,\n      name: \"data\",\n      type: String,\n    },\n  ]\n  const sections = [\n    {\n      header: \"Retrieve documents\",\n      content: \"Retrieve documents\",\n    },\n    {\n      header: \"Options\",\n      optionList: optionsDefinitions,\n    },\n  ]\n  const options = commandLineArgs(optionsDefinitions, {\n    argv: argv,\n  })\n\n  if (\"help\" in options) {\n    const usage = commandLineUsage(sections)\n    console.warn(usage)\n    return null\n  }\n  return options\n}\n\nclass Retrieve {\n  options: any\n\n  constructor(options: any) {\n    this.options = options\n  }\n\n  getDivisions(division: any) {\n    let divisions = [division]\n    if (division.divisions != undefined) {\n      for (const d of division.divisions)\n        divisions = divisions.concat(this.getDivisions(d))\n    }\n    return divisions\n  }\n\n  getDivisionsOrDocument(document: any) {\n    if (document.divisions != undefined)\n      return [].concat(\n        ...document.divisions.map((d: any) => this.getDivisions(d)),\n      )\n    else return [document]\n  }\n\n  async commitAndPush(repositoryDir: string): Promise<void> {\n    if (this.options.commit == undefined) return\n    git.commit(repositoryDir, \"Nouvelle moisson\")\n\n    if (this.options.remote == undefined) return\n    git.run(repositoryDir, `push ${this.options.remote} HEAD:master`)\n  }\n\n  getUrls(divisionOrDocument: any, format: string) {\n    const ext = format == \"html\" ? \"raw\" : \"pdf\"\n    let urls: any = [\n      `http://www.assemblee-nationale.fr/dyn/docs/${divisionOrDocument.uid}.${ext}`,\n    ]\n    if (format == \"html\") {\n      urls.push(urlFromDocument(divisionOrDocument, DocumentUrlFormat.RawHtml))\n      urls.push(urlFromDocument(divisionOrDocument, DocumentUrlFormat.Html))\n    } else if (format == \"pdf\") {\n      urls.push(urlFromDocument(divisionOrDocument, DocumentUrlFormat.Pdf))\n    }\n    return urls\n  }\n\n  static sanityCheckFormat(page: Buffer, format: string): [Buffer, string] {\n    if (format == \"pdf\") {\n      const length = Math.min(page.length, 4)\n      const head = page.slice(0, length).toString()\n      if (!head.startsWith(\"%PDF\"))\n        return [Buffer.from(\"\"), `bad pdf ${encodeURIComponent(head)}...`]\n    } else if (format == \"html\") {\n      const length = Math.min(page.length, 50)\n      const head = page.slice(0, length).toString()\n      if (!head.includes(\"DOCTYPE\") && !head.includes(\"<?xml\")) {\n        return [\n          Buffer.from(\"\"),\n          `bad html ${encodeURIComponent(head.slice(0, 4))}...`,\n        ]\n      }\n    }\n    return [page, \"OK\"]\n  }\n\n  async needsDownload(filePath: string, uid: string): Promise<boolean> {\n    if (fs.existsSync(filePath)) {\n      const isNotEmpty = fs.statSync(filePath).size > 0\n      const retryFailed = \"retry-failed\" in this.options\n      if (!retryFailed || isNotEmpty) {\n        if (this.options.verbose) {\n          if (isNotEmpty) {\n            const urlPath = `${filePath}.url`\n            let url\n            if (fs.existsSync(urlPath))\n              url = fs.readFileSync(urlPath).toString()\n            else url = \"<unknown url>\"\n            console.log(`${uid}: Cached OK ${filePath} from ${url}`)\n          } else {\n            console.log(`${uid}: Cached FAIL ${filePath}`)\n          }\n        }\n        return false\n      }\n    }\n    return true\n  }\n\n  async get(\n    url: string,\n    filePath: string,\n    format: string,\n    uid: string,\n  ): Promise<string> {\n    if (this.options.verbose) console.log(`${uid}: fetch(${url})`)\n    const response = await fetch(url)\n    let page: any = await response.buffer()\n    let status\n    if (!response.ok) {\n      status = response.status\n      if (response.status !== 404) {\n        if (this.options.verbose) {\n          console.error(\n            `Error while getting page \"${url}\" (uid: ${uid}):\\n\\nError:\\n${JSON.stringify(\n              { code: response.status, message: response.statusText },\n              null,\n              2,\n            )}`,\n          )\n        }\n      }\n      fs.writeFileSync(filePath, \"\")\n    } else {\n      ;[page, status] = Retrieve.sanityCheckFormat(page, format)\n      fs.writeFileSync(filePath, page)\n    }\n    return status\n  }\n\n  static urlFromUid(uid: string): string {\n    const url =\n      \"https://git.en-root.org/tricoteuses/data.tricoteuses.fr/Dossiers_Legislatifs_XV/-/tree/master/documents\"\n    return `[${uid}](${url}/uid/${uid}.json)`\n  }\n\n  async download(\n    textesDir: string,\n    divisionOrDocument: any,\n    format: string,\n  ): Promise<number> {\n    const uid = divisionOrDocument.uid\n    const filePath = `${textesDir}/uid/${uid}.${format}`\n    if (!(await this.needsDownload(filePath, uid))) return 0\n    if (this.options.verbose) console.log(`${uid}: Checking ${filePath}`)\n    let success = false\n    let info = \"\"\n    let url\n\n    for (url of this.getUrls(divisionOrDocument, format)) {\n      if (\"dry-run\" in this.options) {\n        info += `  * ${url}\\n`\n        continue\n      } else {\n        info += `  * ${url} `\n      }\n\n      const status = await this.get(url, filePath, format, uid)\n\n      info += `${status}\\n`\n      if (status == \"OK\") {\n        success = true\n        break\n      }\n    }\n    const urlPath = `${filePath}.url`\n    if (success) {\n      fs.writeFileSync(urlPath, url)\n    } else if (fs.existsSync(urlPath)) {\n      fs.unlinkSync(urlPath)\n    }\n    if (info != \"\" && !this.options.silent) {\n      console.log(\n        `* ${Retrieve.urlFromUid(uid)}: Retrieving ${filePath}\\n${info.slice(\n          0,\n          -1,\n        )}`,\n      )\n      return 1\n    } else {\n      return 0\n    }\n  }\n\n  async retrieveDocuments(): Promise<number> {\n    const gitDir = `${this.options.textes}/.git`\n    if (!fs.existsSync(gitDir)) {\n      console.error(`${gitDir} is expected to exist but does not`)\n      return 1\n    }\n\n    const textesDir = path.join(this.options.textes, this.options.legislature)\n    fs.ensureDirSync(`${textesDir}/uid`)\n\n    for (const filename of getFiles(this.options.data)) {\n      const document = load(filename)\n      if (/^....SN/.test(document.uid) || /^DECL/.test(document.uid)) continue\n      for (const divisionOrDocument of this.getDivisionsOrDocument(document)) {\n        if (/-COMPA$/.test(divisionOrDocument.uid)) continue\n        let status = 0\n        for (const format of [\"html\", \"pdf\"]) {\n          status |= await this.download(textesDir, divisionOrDocument, format)\n        }\n        if (status && !this.options.silent) console.log(\"---\")\n      }\n    }\n\n    if (!(\"dry-run\" in this.options)) {\n      await this.commitAndPush(textesDir)\n    }\n\n    return 0\n  }\n}\n\nasync function main(argv: any): Promise<number> {\n  const options = parseArgs(argv)\n  if (options === null) return 1\n  const retrieve = new Retrieve(options)\n  return retrieve.retrieveDocuments()\n}\n\n/* istanbul ignore if */\nif (process.argv[1].endsWith(\"retrieve_documents.ts\"))\n  main(process.argv)\n    .then((exitCode) => {\n      process.exit(exitCode)\n    })\n    .catch((error) => {\n      console.error(error)\n      process.exit(1)\n    })\n"],"mappings":"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"}
197
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["assert","execSync","commandLineArgs","createHash","fs","path","walkDocumentAndDivisions","git","iterLoadAssembleeDocuments","pathFromDocumentUid","DocumentUrlFormat","iterDocumentOrDivisionUrls","optionsDefinitions","alias","help","name","type","Boolean","defaultValue","String","multiple","defaultOption","options","retrieveDocuments","commit","uid","dataDir","documentsDir","join","pull","cwd","env","process","encoding","stdio","ensureDirSync","full","filename","readdirSync","removeSync","firstUid","skip","document","legislature","documentOrDivision","substring","documentDir","filenameBySha256","indexPath","index","pathExistsSync","readJsonSync","format","url","undefined","fileInfos","find","push","Pdf","status","sha256","silent","console","log","response","fetch","filePath","ok","arrayBuffer","buffer","Buffer","from","subarray","toString","startsWith","warn","update","digest","existingFilename","createWriteStream","write","error","JSON","stringify","code","message","statusText","writeJsonSync","spaces","commitAndPush","remote","then","exitCode","exit","catch"],"sources":["../../src/scripts/retrieve_documents.ts"],"sourcesContent":["import assert from \"assert\"\nimport { execSync } from \"child_process\"\nimport commandLineArgs from \"command-line-args\"\nimport { createHash } from \"crypto\"\nimport fs from \"fs-extra\"\nimport path from \"path\"\n\nimport { walkDocumentAndDivisions } from \"../dossiers_legislatifs\"\nimport * as git from \"../git\"\nimport { iterLoadAssembleeDocuments, pathFromDocumentUid } from \"../loaders\"\nimport { DocumentUrlFormat, iterDocumentOrDivisionUrls } from \"../urls\"\n\ninterface FileInfos {\n  filename?: string\n  sha256?: string\n  url: string\n  status: number\n}\n\nconst optionsDefinitions = [\n  {\n    alias: \"c\",\n    help: \"commit documents\",\n    name: \"commit\",\n    type: Boolean,\n  },\n  {\n    alias: \"f\",\n    help: \"retrieve all documents, even already retrieved ones\",\n    name: \"full\",\n    type: Boolean,\n  },\n  {\n    alias: \"l\",\n    defaultValue: \"16\",\n    help: 'legislatures to retrieve, \"*\" for all',\n    name: \"legislature\",\n    type: String,\n  },\n  {\n    alias: \"n\",\n    help: \"try to also retrieve documents that were previously not found\",\n    name: \"not-found\",\n    type: Boolean,\n  },\n  {\n    alias: \"p\",\n    help: \"pull repositories before proceeding\",\n    name: \"pull\",\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    alias: \"u\",\n    help: \"UID of first Assemblée's document or division to retrieve\",\n    name: \"uid\",\n    type: String,\n  },\n  {\n    alias: \"v\",\n    help: \"verbose logs\",\n    name: \"verbose\",\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\nasync function retrieveDocuments(): Promise<number> {\n  assert(\n    !options.commit || options.uid,\n    'Options \"commit\" & \"uid\" are incompatible',\n  )\n\n  const dataDir = options.dataDir\n  const documentsDir = path.join(dataDir, \"Documents\")\n  if (options.pull) {\n    execSync(`git reset --hard origin/master`, {\n      cwd: documentsDir,\n      env: process.env,\n      encoding: \"utf-8\",\n      stdio: [\"ignore\", \"ignore\", \"pipe\"],\n    })\n    execSync(`git pull --rebase`, {\n      cwd: documentsDir,\n      env: process.env,\n      encoding: \"utf-8\",\n      stdio: [\"ignore\", \"ignore\", \"pipe\"],\n    })\n  }\n\n  fs.ensureDirSync(documentsDir)\n  if (options.full && !options.uid) {\n    for (const filename of fs.readdirSync(documentsDir)) {\n      if (filename[0] === \".\") {\n        continue\n      }\n      fs.removeSync(path.join(documentsDir, filename))\n    }\n  }\n\n  const firstUid = options.uid\n  let skip = Boolean(firstUid)\n  for (const { document } of iterLoadAssembleeDocuments(\n    dataDir,\n    options.legislature,\n  )) {\n    for (const documentOrDivision of walkDocumentAndDivisions(document)) {\n      // Ignore documents from Sénat.\n      if (documentOrDivision.uid.substring(4, 6) === \"SN\") {\n        continue\n      }\n\n      if (skip) {\n        if (documentOrDivision.uid === firstUid) {\n          skip = false\n        } else {\n          continue\n        }\n      }\n\n      const documentDir = pathFromDocumentUid(\n        documentsDir,\n        documentOrDivision.uid,\n      )\n      fs.ensureDirSync(documentDir)\n\n      const filenameBySha256: { [digest: string]: string } = {}\n      const indexPath = path.join(documentDir, \"index.json\")\n      const index = (\n        fs.pathExistsSync(indexPath) ? fs.readJsonSync(indexPath) : {}\n      ) as { [format: string]: FileInfos[] }\n      for (const { format, type, url } of iterDocumentOrDivisionUrls(\n        documentOrDivision,\n      )) {\n        if (index[format] === undefined) {\n          index[format] = []\n        }\n        let fileInfos = index[format].find((fileInfos) => fileInfos.url === url)\n        if (fileInfos === undefined) {\n          fileInfos = {} as FileInfos\n          index[format].push(fileInfos)\n        }\n        fileInfos.url = url\n\n        const filename = `${type}.${\n          format === DocumentUrlFormat.Pdf ? \"pdf\" : \"html\"\n        }`\n        if (fileInfos.status === 200 && !options.full) {\n          filenameBySha256[fileInfos.sha256 as string] = filename\n          continue\n        }\n        if (fileInfos.status === 404 && !options[\"not-found\"]) {\n          continue\n        }\n\n        if (!options.silent) {\n          console.log(\n            `Retrieving document or division ${documentOrDivision.uid} at ${url}…`,\n          )\n        }\n\n        const response = await fetch(url)\n        const filePath = path.join(documentDir, filename)\n        if (response.ok) {\n          const arrayBuffer = await response.arrayBuffer()\n          const buffer = Buffer.from(arrayBuffer)\n          if (\n            format === DocumentUrlFormat.Pdf &&\n            !buffer.subarray(0, 4).toString().startsWith(\"%PDF\")\n          ) {\n            // Instead of a PDF, the received data may be an HTML page with a message like\n            // \"Document non encore publié\".\n            if (!options.silent) {\n              console.warn(`  PDF \"${url}\" not found.`)\n            }\n            fs.removeSync(filePath)\n            delete fileInfos.filename\n            delete fileInfos.sha256\n          } else {\n            const sha256 = createHash(\"sha256\").update(buffer).digest(\"hex\")\n            const existingFilename = filenameBySha256[sha256]\n            if (existingFilename === undefined) {\n              fs.createWriteStream(filePath).write(buffer)\n              fileInfos.filename = filename\n              filenameBySha256[sha256] = filename\n            } else {\n              fileInfos.filename = existingFilename\n            }\n            fileInfos.sha256 = sha256\n          }\n        } else {\n          if (response.status === 404) {\n            if (!options.silent) {\n              console.warn(`  Page \"${url}\" not found.`)\n            }\n          } else {\n            console.error(\n              `  Error:\\n${JSON.stringify(\n                { code: response.status, message: response.statusText },\n                null,\n                2,\n              )}`,\n            )\n          }\n          fs.removeSync(filePath)\n          delete fileInfos.filename\n          delete fileInfos.sha256\n        }\n        fileInfos.status = response.status\n      }\n      fs.writeJsonSync(indexPath, index, { encoding: \"utf-8\", spaces: 2 })\n    }\n  }\n\n  if (options.commit) {\n    return git.commitAndPush(documentsDir, \"Nouvelle moisson\", options.remote)\n  }\n  return 0\n}\n\nretrieveDocuments()\n  .then((exitCode) => process.exit(exitCode))\n  .catch((error) => {\n    console.log(error)\n    process.exit(1)\n  })\n"],"mappings":"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"}