UNPKG

@tricoteuses/assemblee

Version:

Retrieve, clean up & handle French Assemblée nationale's open data

242 lines (239 loc) 35.9 kB
import assert from "assert"; import commandLineArgs from "command-line-args"; import { createHash } from "crypto"; import { differenceInDays } from "date-fns"; import fs from "fs-extra"; import path from "path"; import { walkDocumentAndDivisions } from "../dossiers_legislatifs.mjs"; import * as git from "../git.mjs"; import { iterLoadAssembleeDocuments, pathFromDocumentUid } from "../loaders.mjs"; import { parseTexte } from "../parsers/index.mjs"; import { DocumentUrlFormat, iterDocumentOrDivisionUrls } from "../urls.mjs"; import { commitOption, dataDirDefaultOption, legislatureOption, pullOption, remoteOption, silentOption, verboseOption, documentFormatOption } from "./shared/cli_helpers.mjs"; function parseArgs(argv) { const optionsDefinitions = [commitOption, legislatureOption, remoteOption, silentOption, verboseOption, dataDirDefaultOption, pullOption, { alias: "f", help: "retrieve all documents, even already retrieved ones", name: "full", type: Boolean }, { alias: "n", help: "try to also retrieve documents that were previously not found", name: "not-found", type: Boolean }, documentFormatOption, { alias: "T", help: "type of documents to retrieve (for example: PION)", multiple: true, name: "document-type", type: String }]; const options = commandLineArgs(optionsDefinitions, { argv: argv }); return options; } const today = new Date(); async function fetchWithRetry(url, retries = 3, backoff = 300) { for (let attempt = 0; attempt < retries; attempt++) { try { return await fetch(url); } catch (error) { if (attempt === retries - 1) { throw error; } console.warn(`Fetch attempt ${attempt + 1} for ${url} failed. Retrying in ${backoff}ms...`); await new Promise(resolve => setTimeout(resolve, backoff)); backoff *= 2; } } throw new Error(`Failed to fetch ${url} after ${retries} attempts`); } export async function downloadAndParse(document, documentsDir, options) { if (document.uid.substring(4, 6) === "SN") { return; } if (["RAPP", "RINF"].includes(document.uid.substring(0, 4))) { return; } if (options.legislature?.toString() !== document.legislature) { return; } if (options.fetchDocuments) { const dataDir = options.dataDir; const documentsFilesDir = path.join(dataDir, "Documents"); const documentDir = pathFromDocumentUid(`${documentsDir}/documents`, document.uid); const documentFileDir = pathFromDocumentUid(documentsFilesDir, document.uid); fs.ensureDirSync(documentsFilesDir); await processDocumentOrDivision(document, documentsFilesDir, options); if (options.parseDocuments) { const documentPath = `${documentDir}.json`; const htmlPath = path.join(documentFileDir, `dyn-opendata.html`); if (fs.existsSync(htmlPath)) { const html = fs.readFileSync(htmlPath, { encoding: "utf-8" }); try { const parsedDocument = parseTexte("https://www.assemblee-nationale.fr", html); if (parsedDocument && fs.existsSync(documentPath)) { const json = fs.readFileSync(documentPath, { encoding: "utf-8" }); const documentJson = JSON.parse(json); documentJson.subdivisions = parsedDocument?.subdivisions; fs.writeFileSync(documentPath, JSON.stringify(documentJson, null, 2)); } } catch (e) { console.log(`Unable to parse document ${document.uid}`); return; } } } } } async function retrieveDocuments(options) { assert(!options.commit || !options.uid, 'Options "commit" & "uid" are incompatible'); const dataDir = options.dataDir; const documentsDir = path.join(dataDir, "Documents"); if (options.pull) { git.resetAndPull(documentsDir); } fs.ensureDirSync(documentsDir); if (options.full && !options.uid) { for (const filename of fs.readdirSync(documentsDir)) { if (filename[0] === ".") { continue; } fs.removeSync(path.join(documentsDir, filename)); } } const firstUid = options.uid; let skip = Boolean(firstUid); for (const { document } of iterLoadAssembleeDocuments(dataDir, options.legislature)) { for (const documentOrDivision of walkDocumentAndDivisions(document)) { // Ignore documents from Sénat. if (documentOrDivision.uid.substring(4, 6) === "SN") { continue; } if (skip) { if (documentOrDivision.uid === firstUid) { skip = false; } else { continue; } } await processDocumentOrDivision(documentOrDivision, documentsDir, options); } } if (options.commit) { return git.commitAndPush(documentsDir, "Nouvelle moisson", options.remote); } return 0; } async function processDocumentOrDivision(documentOrDivision, documentsDir, options) { const documentDir = pathFromDocumentUid(documentsDir, documentOrDivision.uid); fs.ensureDirSync(documentDir); const filenameBySha256 = {}; const indexPath = path.join(documentDir, "index.json"); const index = fs.pathExistsSync(indexPath) ? fs.readJsonSync(indexPath) : {}; for (const { format, type, url } of iterDocumentOrDivisionUrls(documentOrDivision)) { const filename = `${type}.${format === DocumentUrlFormat.Pdf ? "pdf" : "html"}`; // Filter by file extension if option is passed const fileExtension = url.split(".").pop(); if (options.format && fileExtension && options.format !== fileExtension) { continue; } if (options.legislature && options.legislature.toString() !== documentOrDivision.legislature) { continue; } // Filter by document type if option is passed const documentType = documentOrDivision.classification?.type?.code; if (options["document-type"] !== undefined && !options["document-type"].includes(documentType)) { continue; } let formatFilesInfos = index[format] ?? (index[format] = []); let fileInfos = formatFilesInfos.find(file => file.url === url) ?? {}; if (!formatFilesInfos.includes(fileInfos)) { formatFilesInfos.push(fileInfos); } fileInfos.url = url; if (fileInfos.status === 200 && !options.full) { filenameBySha256[fileInfos.sha256] = filename; continue; } if (fileInfos.status === 404 && !options["not-found"] && differenceInDays(today, documentOrDivision.cycleDeVie.chrono.dateCreation ?? documentOrDivision.cycleDeVie.chrono.dateDepot) > 10) { continue; } if (!options.silent) { console.log(`Retrieving document or division ${documentOrDivision.uid} at ${url}…`); } const response = await fetchWithRetry(url); const filePath = path.join(documentDir, filename); if (response.ok) { const arrayBuffer = await response.arrayBuffer(); const buffer = Buffer.from(arrayBuffer); if (format === DocumentUrlFormat.Pdf && !buffer.subarray(0, 4).toString().startsWith("%PDF")) { // Instead of a PDF, the received data may be an HTML page with a message like // "Document non encore publié". if (!options.silent) { console.warn(` PDF "${url}" not found.`); } fs.removeSync(filePath); delete fileInfos.filename; delete fileInfos.sha256; fileInfos.status = 404; } else { const sha256 = createHash("sha256").update(buffer).digest("hex"); const existingFilename = filenameBySha256[sha256]; if (existingFilename === undefined) { fs.writeFileSync(filePath, buffer, { encoding: "utf8" }); fileInfos.filename = filename; filenameBySha256[sha256] = filename; } else { fileInfos.filename = existingFilename; } fileInfos.sha256 = sha256; fileInfos.status = response.status; } } else { if (response.status === 404) { if (!options.silent) { console.warn(` Page "${url}" not found.`); } } else { console.error(` Error:\n${JSON.stringify({ code: response.status, message: response.statusText }, null, 2)}`); } fs.removeSync(filePath); delete fileInfos.filename; delete fileInfos.sha256; fileInfos.status = response.status; } } fs.writeJsonSync(indexPath, index, { encoding: "utf-8", spaces: 2 }); } function main(argv) { const options = parseArgs(argv); return retrieveDocuments(options); } /* istanbul ignore if */ if (process.argv[1].endsWith("retrieve_documents.ts")) { main(process.argv).then(exitCode => process.exit(exitCode)).catch(error => { console.log(error); process.exit(1); }); } //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["assert","commandLineArgs","createHash","differenceInDays","fs","path","walkDocumentAndDivisions","git","iterLoadAssembleeDocuments","pathFromDocumentUid","parseTexte","DocumentUrlFormat","iterDocumentOrDivisionUrls","commitOption","dataDirDefaultOption","legislatureOption","pullOption","remoteOption","silentOption","verboseOption","documentFormatOption","parseArgs","argv","optionsDefinitions","alias","help","name","type","Boolean","multiple","String","options","today","Date","fetchWithRetry","url","retries","backoff","attempt","fetch","error","console","warn","Promise","resolve","setTimeout","Error","downloadAndParse","document","documentsDir","uid","substring","includes","legislature","toString","fetchDocuments","dataDir","documentsFilesDir","join","documentDir","documentFileDir","ensureDirSync","processDocumentOrDivision","parseDocuments","documentPath","htmlPath","existsSync","html","readFileSync","encoding","parsedDocument","json","documentJson","JSON","parse","subdivisions","writeFileSync","stringify","e","log","retrieveDocuments","commit","pull","resetAndPull","full","filename","readdirSync","removeSync","firstUid","skip","documentOrDivision","commitAndPush","remote","filenameBySha256","indexPath","index","pathExistsSync","readJsonSync","format","Pdf","fileExtension","split","pop","documentType","classification","code","undefined","formatFilesInfos","fileInfos","find","file","push","status","sha256","cycleDeVie","chrono","dateCreation","dateDepot","silent","response","filePath","ok","arrayBuffer","buffer","Buffer","from","subarray","startsWith","update","digest","existingFilename","message","statusText","writeJsonSync","spaces","main","process","endsWith","then","exitCode","exit","catch"],"sources":["../../src/scripts/retrieve_documents.ts"],"sourcesContent":["import assert from \"assert\"\nimport commandLineArgs from \"command-line-args\"\nimport { createHash } from \"crypto\"\nimport { differenceInDays } from \"date-fns\"\nimport fs from \"fs-extra\"\nimport path from \"path\"\n\nimport {\n  DocumentFileInfos,\n  DocumentFilesIndex,\n  walkDocumentAndDivisions,\n} from \"../dossiers_legislatifs\"\nimport * as git from \"../git\"\nimport { iterLoadAssembleeDocuments, pathFromDocumentUid } from \"../loaders\"\nimport { parseTexte } from \"../parsers\"\nimport { DocumentUrlFormat, iterDocumentOrDivisionUrls } from \"../urls\"\nimport {\n  commitOption,\n  dataDirDefaultOption,\n  legislatureOption,\n  pullOption,\n  remoteOption,\n  silentOption,\n  verboseOption,\n  documentFormatOption,\n} from \"./shared/cli_helpers\"\n\nfunction parseArgs(argv: string[]): any {\n  const optionsDefinitions = [\n    commitOption,\n    legislatureOption,\n    remoteOption,\n    silentOption,\n    verboseOption,\n    dataDirDefaultOption,\n    pullOption,\n    {\n      alias: \"f\",\n      help: \"retrieve all documents, even already retrieved ones\",\n      name: \"full\",\n      type: Boolean,\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    documentFormatOption,\n    {\n      alias: \"T\",\n      help: \"type of documents to retrieve (for example: PION)\",\n      multiple: true,\n      name: \"document-type\",\n      type: String,\n    },\n  ]\n  const options = commandLineArgs(optionsDefinitions, {\n    argv: argv,\n  })\n  return options\n}\n\nconst today = new Date()\n\nasync function fetchWithRetry(\n  url: string,\n  retries: number = 3,\n  backoff: number = 300,\n): Promise<Response> {\n  for (let attempt = 0; attempt < retries; attempt++) {\n    try {\n      return await fetch(url)\n    } catch (error) {\n      if (attempt === retries - 1) {\n        throw error\n      }\n      console.warn(\n        `Fetch attempt ${attempt + 1} for ${url} failed. Retrying in ${backoff}ms...`,\n      )\n      await new Promise((resolve) => setTimeout(resolve, backoff))\n      backoff *= 2\n    }\n  }\n  throw new Error(`Failed to fetch ${url} after ${retries} attempts`)\n}\n\nexport async function downloadAndParse(\n  document: any,\n  documentsDir: string,\n  options: any,\n): Promise<void> {\n  if (document.uid.substring(4, 6) === \"SN\") {\n    return\n  }\n\n  if ([\"RAPP\", \"RINF\"].includes(document.uid.substring(0, 4))) {\n    return\n  }\n\n  if (options.legislature?.toString() !== document.legislature) {\n    return\n  }\n\n  if (options.fetchDocuments) {\n    const dataDir = options.dataDir\n    const documentsFilesDir = path.join(dataDir, \"Documents\")\n    const documentDir = pathFromDocumentUid(\n      `${documentsDir}/documents`,\n      document.uid,\n    )\n    const documentFileDir = pathFromDocumentUid(documentsFilesDir, document.uid)\n\n    fs.ensureDirSync(documentsFilesDir)\n    await processDocumentOrDivision(document, documentsFilesDir, options)\n\n    if (options.parseDocuments) {\n      const documentPath = `${documentDir}.json`\n      const htmlPath = path.join(documentFileDir, `dyn-opendata.html`)\n\n      if (fs.existsSync(htmlPath)) {\n        const html = fs.readFileSync(htmlPath, { encoding: \"utf-8\" })\n        try {\n          const parsedDocument = parseTexte(\n            \"https://www.assemblee-nationale.fr\",\n            html,\n          )\n          if (parsedDocument && fs.existsSync(documentPath)) {\n            const json = fs.readFileSync(documentPath, { encoding: \"utf-8\" })\n            const documentJson = JSON.parse(json)\n            documentJson.subdivisions = parsedDocument?.subdivisions\n            fs.writeFileSync(\n              documentPath,\n              JSON.stringify(documentJson, null, 2),\n            )\n          }\n        } catch (e) {\n          console.log(`Unable to parse document ${document.uid}`)\n          return\n        }\n      }\n    }\n  }\n}\n\nasync function retrieveDocuments(options: any): 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    git.resetAndPull(documentsDir)\n  }\n  fs.ensureDirSync(documentsDir)\n\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      await processDocumentOrDivision(documentOrDivision, documentsDir, options)\n    }\n  }\n\n  if (options.commit) {\n    return git.commitAndPush(documentsDir, \"Nouvelle moisson\", options.remote)\n  }\n  return 0\n}\n\nasync function processDocumentOrDivision(\n  documentOrDivision: any,\n  documentsDir: string,\n  options: any,\n): Promise<void> {\n  const documentDir = pathFromDocumentUid(documentsDir, documentOrDivision.uid)\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 DocumentFilesIndex\n\n  for (const { format, type, url } of iterDocumentOrDivisionUrls(\n    documentOrDivision,\n  )) {\n    const filename = `${type}.${format === DocumentUrlFormat.Pdf ? \"pdf\" : \"html\"}`\n\n    // Filter by file extension if option is passed\n    const fileExtension = url.split(\".\").pop()\n    if (options.format && fileExtension && options.format !== fileExtension) {\n      continue\n    }\n\n    if (\n      options.legislature &&\n      options.legislature.toString() !== documentOrDivision.legislature\n    ) {\n      continue\n    }\n\n    // Filter by document type if option is passed\n    const documentType = documentOrDivision.classification?.type?.code\n    if (\n      options[\"document-type\"] !== undefined &&\n      !options[\"document-type\"].includes(documentType)\n    ) {\n      continue\n    }\n\n    let formatFilesInfos = index[format] ?? (index[format] = [])\n    let fileInfos =\n      formatFilesInfos.find((file) => file.url === url) ??\n      ({} as DocumentFileInfos)\n    if (!formatFilesInfos.includes(fileInfos)) {\n      formatFilesInfos.push(fileInfos)\n    }\n    fileInfos.url = url\n\n    if (fileInfos.status === 200 && !options.full) {\n      filenameBySha256[fileInfos.sha256 as string] = filename\n      continue\n    }\n    if (\n      fileInfos.status === 404 &&\n      !options[\"not-found\"] &&\n      differenceInDays(\n        today,\n        documentOrDivision.cycleDeVie.chrono.dateCreation ??\n          (documentOrDivision.cycleDeVie.chrono.dateDepot as Date | string),\n      ) > 10\n    ) {\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 fetchWithRetry(url)\n    const filePath = path.join(documentDir, filename)\n\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        fileInfos.status = 404\n      } else {\n        const sha256 = createHash(\"sha256\").update(buffer).digest(\"hex\")\n        const existingFilename = filenameBySha256[sha256]\n        if (existingFilename === undefined) {\n          fs.writeFileSync(filePath, buffer, {\n            encoding: \"utf8\",\n          })\n          fileInfos.filename = filename\n          filenameBySha256[sha256] = filename\n        } else {\n          fileInfos.filename = existingFilename\n        }\n        fileInfos.sha256 = sha256\n        fileInfos.status = response.status\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      fileInfos.status = response.status\n    }\n  }\n\n  fs.writeJsonSync(indexPath, index, { encoding: \"utf-8\", spaces: 2 })\n}\n\nfunction main(argv: any): Promise<number> {\n  const options = parseArgs(argv)\n  return retrieveDocuments(options)\n}\n\n/* istanbul ignore if */\nif (process.argv[1].endsWith(\"retrieve_documents.ts\")) {\n  main(process.argv)\n    .then((exitCode) => process.exit(exitCode))\n    .catch((error) => {\n      console.log(error)\n      process.exit(1)\n    })\n}\n"],"mappings":"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","ignoreList":[]}