UNPKG

@tricoteuses/assemblee

Version:

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

158 lines (155 loc) 27.1 kB
import { execSync } from "child_process"; import commandLineArgs from "command-line-args"; import fs from "fs-extra"; import path from "path"; import stream from "stream"; import util from "util"; import * as git from "../git.mjs"; import { iterLoadAssembleeActeurs } from "../loaders.mjs"; import { CodeTypeOrgane, TypeMandat } from "../types/acteurs_et_organes.mjs"; import { cloneOption, commitOption, remoteOption, silentOption, dataDirDefaultOption, pullOption, legislatureOption, fetchOption } from "./shared/cli_helpers.mjs"; const optionsDefinitions = [cloneOption, commitOption, remoteOption, silentOption, dataDirDefaultOption, pullOption, fetchOption, legislatureOption, { alias: "S", defaultValue: false, name: "square", type: Boolean }, { alias: "w", defaultValue: 150, name: "width", type: Number }, { alias: "h", defaultValue: 192, name: "height", type: Number }]; const options = commandLineArgs(optionsDefinitions); export function cloneAndUpdateGitRepository(photosDirName, options) { const gitGroupUrl = options.clone ? options.clone.trim().replace(/\/+$/, "") : undefined; const photosDir = path.join(options.dataDir, photosDirName); git.clone(gitGroupUrl, photosDirName, options.dataDir); if (options.pull) { git.resetAndPull(photosDir); } } export function resizePhoto(acteur, width, height, photosDir, photoFilename, options) { const ident = acteur.etatCivil.ident; if (!options.silent) { console.log(`Resizing photo ${photoFilename} for ${ident.prenom} ${ident.nom}…`); } execSync(`gm convert -resize ${width}x${height}! ${photoFilename}.jpg ${photoFilename}_${width}x${height}.jpg`, { cwd: photosDir }); } export async function downloadPhoto(urlPhoto, photoTempFilePath, photoFilePath, missingPhotoFilePath, retries = 3) { const pipeline = util.promisify(stream.pipeline); for (let attempt = 1; attempt <= retries; attempt++) { try { const response = await fetch(urlPhoto); if (response.ok) { await pipeline(response.body, fs.createWriteStream(photoTempFilePath)); fs.renameSync(photoTempFilePath, photoFilePath); return; } console.warn(`Fetch failed: ${urlPhoto} (Attempt ${attempt}) - Status: ${response.status} ${response.statusText}`); } catch (error) { console.log(error); console.warn(`Attempt ${attempt} failed for ${urlPhoto}`, error); } } if (fs.existsSync(photoFilePath)) { console.warn(`Reusing existing image: ${photoFilePath}`); } else { console.warn(`Using blank image: ${missingPhotoFilePath}`); fs.copyFileSync(missingPhotoFilePath, photoFilePath); } } export function createMosaic(acteurUids, photosDir, photosWidth, photosHeight, mosaicFilename) { const photoByIdActeur = {}; const rowsFilenames = []; if (acteurUids.length === 0) return; for (let acteurIndex = 0, rowIndex = 0; acteurIndex < acteurUids.length; acteurIndex += 25, rowIndex++) { const row = acteurUids.slice(acteurIndex, acteurIndex + 25); const photosFilenames = row.map((acteurUid, columnIndex) => { const photoFilename = `${acteurUid}_${photosWidth}x${photosHeight}.jpg`; photoByIdActeur[acteurUid] = { chemin: `${photosDir}/${photoFilename}`, cheminMosaique: `${photosDir}/${mosaicFilename}.jpg`, largeur: photosWidth, hauteur: photosHeight, xMosaique: columnIndex * photosWidth, yMosaique: rowIndex * photosHeight }; return photoFilename; }); const rowFilename = `row-${rowIndex}.jpg`; execSync(`gm convert ${photosFilenames.join(" ")} +append ${rowFilename}`, { cwd: photosDir }); rowsFilenames.push(rowFilename); } // Combine all rows into one image execSync(`gm convert ${rowsFilenames.join(" ")} -append ${mosaicFilename}.jpg`, { cwd: photosDir }); // Cleanup individual row images rowsFilenames.forEach(rowFilename => { fs.unlinkSync(path.join(photosDir, rowFilename)); }); const jsonFilePath = path.join(photosDir, `${mosaicFilename}.json`); fs.writeFileSync(jsonFilePath, JSON.stringify(photoByIdActeur, null, 2)); } async function retrievePhotosDeputes() { const dataDir = options.dataDir; const photosPrefix = "photos_deputes"; const photosDirName = `${photosPrefix}_${options.legislature}`; const photosDir = path.join(dataDir, photosDirName); const photosWidth = options.width; const photosHeight = options.height; const missingPhotoFilePath = `images/transparent_${photosWidth}x${photosHeight}.jpg`; cloneAndUpdateGitRepository(photosDirName, options); fs.ensureDirSync(photosDir); const acteurUids = []; for (const { acteur } of iterLoadAssembleeActeurs(dataDir, options.legislature, { noValidation: true })) { const { mandats } = acteur; if (!mandats) continue; const mandatDepute = mandats.find(mandat => mandat.xsiType === TypeMandat.MandatParlementaireType && mandat.legislature === options.legislature?.toString() && mandat.typeOrgane === CodeTypeOrgane.Assemblee); if (!mandatDepute) continue; const numeroDepute = acteur.uid.substring(2); // Download photos. if (options.fetch) { const ident = acteur.etatCivil.ident; const photoFilename = `${numeroDepute}.jpg`; const photoFilePath = path.join(photosDir, photoFilename); const photoTempFilename = `${numeroDepute}_temp.jpg`; const photoTempFilePath = path.join(photosDir, photoTempFilename); let urlPhoto = `http://www2.assemblee-nationale.fr/static/tribun/${options.legislature}/photos/${photoFilename}`; if (options.square) { urlPhoto = `https://www2.assemblee-nationale.fr/static/tribun/${options.legislature}/photos/carre/${photoFilename}`; } if (!options.silent) { console.log(`Loading photo ${urlPhoto} for ${ident.prenom} ${ident.nom}…`); } await downloadPhoto(urlPhoto, photoTempFilePath, photoFilePath, missingPhotoFilePath); resizePhoto(acteur, photosWidth, photosHeight, photosDir, numeroDepute, options); } acteurUids.push(numeroDepute); } createMosaic(acteurUids, photosDir, photosWidth, photosHeight, "deputes"); if (options.commit) { return git.commitAndPush(photosDir, "Nouvelle moisson", options.remote); } return 0; } retrievePhotosDeputes().then(exitCode => process.exit(exitCode)).catch(error => { console.log(error); process.exit(1); }); //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["execSync","commandLineArgs","fs","path","stream","util","git","iterLoadAssembleeActeurs","CodeTypeOrgane","TypeMandat","cloneOption","commitOption","remoteOption","silentOption","dataDirDefaultOption","pullOption","legislatureOption","fetchOption","optionsDefinitions","alias","defaultValue","name","type","Boolean","Number","options","cloneAndUpdateGitRepository","photosDirName","gitGroupUrl","clone","trim","replace","undefined","photosDir","join","dataDir","pull","resetAndPull","resizePhoto","acteur","width","height","photoFilename","ident","etatCivil","silent","console","log","prenom","nom","cwd","downloadPhoto","urlPhoto","photoTempFilePath","photoFilePath","missingPhotoFilePath","retries","pipeline","promisify","attempt","response","fetch","ok","body","createWriteStream","renameSync","warn","status","statusText","error","existsSync","copyFileSync","createMosaic","acteurUids","photosWidth","photosHeight","mosaicFilename","photoByIdActeur","rowsFilenames","length","acteurIndex","rowIndex","row","slice","photosFilenames","map","acteurUid","columnIndex","chemin","cheminMosaique","largeur","hauteur","xMosaique","yMosaique","rowFilename","push","forEach","unlinkSync","jsonFilePath","writeFileSync","JSON","stringify","retrievePhotosDeputes","photosPrefix","legislature","ensureDirSync","noValidation","mandats","mandatDepute","find","mandat","xsiType","MandatParlementaireType","toString","typeOrgane","Assemblee","numeroDepute","uid","substring","photoTempFilename","square","commit","commitAndPush","remote","then","exitCode","process","exit","catch"],"sources":["../../src/scripts/retrieve_deputes_photos.ts"],"sourcesContent":["import { execSync } from \"child_process\"\nimport commandLineArgs from \"command-line-args\"\nimport fs from \"fs-extra\"\nimport path from \"path\"\nimport stream from \"stream\"\nimport util from \"util\"\nimport * as git from \"../git\"\n\nimport { iterLoadAssembleeActeurs } from \"../loaders\"\nimport {\n  Acteur,\n  CodeTypeOrgane,\n  Mandat,\n  Photo,\n  TypeMandat,\n} from \"../types/acteurs_et_organes\"\nimport {\n  cloneOption,\n  commitOption,\n  remoteOption,\n  silentOption,\n  dataDirDefaultOption,\n  pullOption,\n  legislatureOption,\n  fetchOption,\n} from \"./shared/cli_helpers\"\n\nconst optionsDefinitions = [\n  cloneOption,\n  commitOption,\n  remoteOption,\n  silentOption,\n  dataDirDefaultOption,\n  pullOption,\n  fetchOption,\n  legislatureOption,\n  {\n    alias: \"S\",\n    defaultValue: false,\n    name: \"square\",\n    type: Boolean,\n  },\n  {\n    alias: \"w\",\n    defaultValue: 150,\n    name: \"width\",\n    type: Number,\n  },\n  {\n    alias: \"h\",\n    defaultValue: 192,\n    name: \"height\",\n    type: Number,\n  },\n]\nconst options = commandLineArgs(optionsDefinitions)\n\nexport function cloneAndUpdateGitRepository(\n  photosDirName: string,\n  options: any,\n): void {\n  const gitGroupUrl = options.clone\n    ? options.clone.trim().replace(/\\/+$/, \"\")\n    : undefined\n  const photosDir = path.join(options.dataDir, photosDirName)\n  git.clone(gitGroupUrl, photosDirName, options.dataDir)\n\n  if (options.pull) {\n    git.resetAndPull(photosDir)\n  }\n}\n\nexport function resizePhoto(\n  acteur: Acteur,\n  width: number,\n  height: number,\n  photosDir: string,\n  photoFilename: string,\n  options: any,\n): void {\n  const ident = acteur.etatCivil.ident\n  if (!options.silent) {\n    console.log(\n      `Resizing photo ${photoFilename} for ${ident.prenom} ${ident.nom}…`,\n    )\n  }\n  execSync(\n    `gm convert -resize ${width}x${height}! ${photoFilename}.jpg ${photoFilename}_${width}x${height}.jpg`,\n    {\n      cwd: photosDir,\n    },\n  )\n}\n\nexport async function downloadPhoto(\n  urlPhoto: string,\n  photoTempFilePath: string,\n  photoFilePath: string,\n  missingPhotoFilePath: string,\n  retries: number = 3,\n): Promise<void> {\n  const pipeline = util.promisify(stream.pipeline)\n\n  for (let attempt = 1; attempt <= retries; attempt++) {\n    try {\n      const response = await fetch(urlPhoto)\n\n      if (response.ok) {\n        await pipeline(\n          response.body as unknown as NodeJS.ReadableStream,\n          fs.createWriteStream(photoTempFilePath),\n        )\n        fs.renameSync(photoTempFilePath, photoFilePath)\n        return\n      }\n\n      console.warn(\n        `Fetch failed: ${urlPhoto} (Attempt ${attempt}) - Status: ${response.status} ${response.statusText}`,\n      )\n    } catch (error) {\n      console.log(error)\n      console.warn(`Attempt ${attempt} failed for ${urlPhoto}`, error)\n    }\n  }\n\n  if (fs.existsSync(photoFilePath)) {\n    console.warn(`Reusing existing image: ${photoFilePath}`)\n  } else {\n    console.warn(`Using blank image: ${missingPhotoFilePath}`)\n    fs.copyFileSync(missingPhotoFilePath, photoFilePath)\n  }\n}\n\nexport function createMosaic(\n  acteurUids: string[],\n  photosDir: string,\n  photosWidth: number,\n  photosHeight: number,\n  mosaicFilename: string,\n): void {\n  const photoByIdActeur: { [uid: string]: Photo } = {}\n  const rowsFilenames: string[] = []\n\n  if (acteurUids.length === 0) return\n\n  for (\n    let acteurIndex = 0, rowIndex = 0;\n    acteurIndex < acteurUids.length;\n    acteurIndex += 25, rowIndex++\n  ) {\n    const row = acteurUids.slice(acteurIndex, acteurIndex + 25)\n    const photosFilenames: string[] = row.map((acteurUid, columnIndex) => {\n      const photoFilename = `${acteurUid}_${photosWidth}x${photosHeight}.jpg`\n      photoByIdActeur[acteurUid] = {\n        chemin: `${photosDir}/${photoFilename}`,\n        cheminMosaique: `${photosDir}/${mosaicFilename}.jpg`,\n        largeur: photosWidth,\n        hauteur: photosHeight,\n        xMosaique: columnIndex * photosWidth,\n        yMosaique: rowIndex * photosHeight,\n      }\n      return photoFilename\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\n  // Combine all rows into one image\n  execSync(\n    `gm convert ${rowsFilenames.join(\" \")} -append ${mosaicFilename}.jpg`,\n    { cwd: photosDir },\n  )\n\n  // Cleanup individual row images\n  rowsFilenames.forEach((rowFilename) => {\n    fs.unlinkSync(path.join(photosDir, rowFilename))\n  })\n\n  const jsonFilePath = path.join(photosDir, `${mosaicFilename}.json`)\n  fs.writeFileSync(jsonFilePath, JSON.stringify(photoByIdActeur, null, 2))\n}\n\nasync function retrievePhotosDeputes(): Promise<number> {\n  const dataDir = options.dataDir\n  const photosPrefix = \"photos_deputes\"\n  const photosDirName = `${photosPrefix}_${options.legislature}`\n  const photosDir = path.join(dataDir, photosDirName)\n  const photosWidth = options.width\n  const photosHeight = options.height\n  const missingPhotoFilePath = `images/transparent_${photosWidth}x${photosHeight}.jpg`\n\n  cloneAndUpdateGitRepository(photosDirName, options)\n  fs.ensureDirSync(photosDir)\n\n  const acteurUids = []\n  for (const { acteur } of iterLoadAssembleeActeurs(\n    dataDir,\n    options.legislature,\n    { noValidation: true },\n  )) {\n    const { mandats } = acteur\n    if (!mandats) continue\n\n    const mandatDepute = mandats.find(\n      (mandat: Mandat) =>\n        mandat.xsiType === TypeMandat.MandatParlementaireType &&\n        mandat.legislature === options.legislature?.toString() &&\n        mandat.typeOrgane === CodeTypeOrgane.Assemblee,\n    )\n    if (!mandatDepute) continue\n\n    const numeroDepute = acteur.uid.substring(2)\n\n    // Download photos.\n    if (options.fetch) {\n      const ident = acteur.etatCivil.ident\n      const photoFilename = `${numeroDepute}.jpg`\n      const photoFilePath = path.join(photosDir, photoFilename)\n      const photoTempFilename = `${numeroDepute}_temp.jpg`\n      const photoTempFilePath = path.join(photosDir, photoTempFilename)\n      let urlPhoto = `http://www2.assemblee-nationale.fr/static/tribun/${options.legislature}/photos/${photoFilename}`\n      if (options.square) {\n        urlPhoto = `https://www2.assemblee-nationale.fr/static/tribun/${options.legislature}/photos/carre/${photoFilename}`\n      }\n\n      if (!options.silent) {\n        console.log(\n          `Loading photo ${urlPhoto} for ${ident.prenom} ${ident.nom}…`,\n        )\n      }\n\n      await downloadPhoto(\n        urlPhoto,\n        photoTempFilePath,\n        photoFilePath,\n        missingPhotoFilePath,\n      )\n      resizePhoto(\n        acteur,\n        photosWidth,\n        photosHeight,\n        photosDir,\n        numeroDepute,\n        options,\n      )\n    }\n\n    acteurUids.push(numeroDepute)\n  }\n\n  createMosaic(acteurUids, photosDir, photosWidth, photosHeight, \"deputes\")\n\n  if (options.commit) {\n    return git.commitAndPush(photosDir, \"Nouvelle moisson\", options.remote)\n  }\n  return 0\n}\n\nretrievePhotosDeputes()\n  .then((exitCode) => process.exit(exitCode))\n  .catch((error) => {\n    console.log(error)\n    process.exit(1)\n  })\n"],"mappings":"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","ignoreList":[]}