UNPKG

@tricoteuses/assemblee

Version:

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

184 lines (183 loc) 25.3 kB
function _defineProperty(e, r, t) { return (r = _toPropertyKey(r)) in e ? Object.defineProperty(e, r, { value: t, enumerable: !0, configurable: !0, writable: !0 }) : e[r] = t, e; } function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == typeof i ? i : i + ""; } function _toPrimitive(t, r) { if ("object" != typeof t || !t) return t; var e = t[Symbol.toPrimitive]; if (void 0 !== e) { var i = e.call(t, r || "default"); if ("object" != typeof i) return i; throw new TypeError("@@toPrimitive must return a primitive value."); } return ("string" === r ? String : Number)(t); } import Ajv from "ajv"; import commandLineArgs from "command-line-args"; import commandLineUsage from "command-line-usage"; import { load, getFiles } from "../file_systems.mjs"; import * as git from "../git.mjs"; import temp from "temp"; import fs from "fs-extra"; import path from "path"; import { validDataset, getDatasets, getSchemas, datasetDirectorySchema } from "../../src/datasets.mjs"; // Automatically track and cleanup files at exit temp.track(); function parseArgs(argv) { const optionsDefinitions = [{ description: "pathname to the directory containing the dataset", defaultOption: true, name: "data", type: String }, { description: "URL or path to the tricoteuses-assemblee git repository", defaultValue: "https://git.en-root.org/tricoteuses/tricoteuses-assemblee.git", name: "repository", type: String }, { description: `dataset (possible values must include exactly one of ${getDatasets()})`, name: "dataset", type: String }, { description: `schema (possible values are ${getSchemas()})`, name: "schema", type: String }, { description: "ignore schemaVersion from JSON files and use the current directory instead of the --repository", name: "dev", type: Boolean }, { description: "increase verbosity", name: "verbose", type: Boolean }, { name: "help", description: "Print this usage guide." }]; const sections = [{ header: "Validate JSON files", content: "Validate JSON files" }, { header: "Options", optionList: optionsDefinitions }]; const options = commandLineArgs(optionsDefinitions, { argv: argv }); if (options.dataset) { if (!validDataset(options.dataset)) { console.error(`--dataset ${options.dataset}`); options.help = true; } } if (options.schema) { if (!getSchemas().includes(options.schema)) { console.error(`--schema ${options.schema}`); options.help = true; } } if ("help" in options) { const usage = commandLineUsage(sections); console.warn(usage); return null; } return options; } class Validate { constructor(options) { _defineProperty(this, "options", void 0); _defineProperty(this, "version2schema", void 0); _defineProperty(this, "tmpDir", void 0); this.options = options; this.version2schema = {}; this.tmpDir = temp.mkdirSync("validation"); } getSchemaDir(tag) { if (this.options.dev == true) { if (this.options.verbose) console.log("--dev: use src/schemas"); return "src/schemas"; } const repositoryDir = path.join(this.tmpDir, "tricoteuses-assemblee"); const tagDir = path.join(this.tmpDir, tag); if (!fs.existsSync(tagDir)) { if (!fs.existsSync(repositoryDir)) { git.run(this.tmpDir, `clone --quiet ${this.options.repository} ${repositoryDir}`); } git.run(repositoryDir, `checkout tags/${tag} -- src/schemas > /dev/null 2>&1`); fs.renameSync(path.join(repositoryDir, "src/schemas"), tagDir); } return tagDir; } getValidator(version) { if (!(version in this.version2schema)) { const ajv = new Ajv(); const dir = this.getSchemaDir(`schema-${version}`); let references = []; for (const reference of getFiles([`${dir}/${this.options.schema}/*.json`])) references.push(reference); for (const file of getFiles(references)) { if (this.options.verbose) console.log(`reference: ${file}`); ajv.addSchema(load(file)); } const schema = this.options.schema; const Schema = schema.charAt(0).toUpperCase() + schema.slice(1); const s = `${dir}/${schema}/${Schema}.json`; if (this.options.verbose) console.log(`schema ${s}`); this.version2schema[version] = ajv.compile(load(s)); } return this.version2schema[version]; } getVersion(content) { if (content.schemaVersion != undefined) return content.schemaVersion; return `${this.options.schema}-1.0`; } getSchemaURL(version) { if (this.options.dev == true) return "src/schemas";else { if (fs.existsSync(this.options.repository)) { return `${this.options.repository}/src/schemas at tag schema-${version}`; } else { const repository = this.options.repository.replace(/\.git$/, ""); return `${repository}/tree/schema-${version}/src/schemas`; } } } validate(content, filename) { const tag = this.getVersion(content); const validator = this.getValidator(tag); const valid = validator(content); const schemaURL = this.getSchemaURL(tag); if (!valid) { console.error(`${schemaURL} finds ${filename} is invalid`); console.error(validator.errors); return 1; } else { if (this.options.verbose) console.log(`${schemaURL} finds ${filename} is valid`); return 0; } } async process() { let status = 0; for (const file of getFiles([this.options.data])) { const content = load(file); status |= this.validate(content, file); } return status; } } async function main(argv) { const options = parseArgs(argv); if (options === null) return 1; let directorySchema = []; if (options.schema) { directorySchema = [[options.data, options.schema]]; } else if (options.dataset) { directorySchema = datasetDirectorySchema(options.dataset); } else { console.error("either --schema or --dataset is required"); return 1; } for (const d of directorySchema) { options.schema = d[1]; options.data = d[0]; if (options.verbose) console.log(`working on files ${options.data} with schema ${options.schema}`); const validate = new Validate(options); const status = await validate.process(); if (status != 0) return status; } return 0; } /* istanbul ignore if */ if (process.argv[1].endsWith("validate_json.ts")) main(process.argv).then(exitCode => { process.exit(exitCode); }).catch(error => { console.error(error); process.exit(1); }); //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Ajv","commandLineArgs","commandLineUsage","load","getFiles","git","temp","fs","path","validDataset","getDatasets","getSchemas","datasetDirectorySchema","track","parseArgs","argv","optionsDefinitions","description","defaultOption","name","type","String","defaultValue","Boolean","sections","header","content","optionList","options","dataset","console","error","help","schema","includes","usage","warn","Validate","constructor","_defineProperty","version2schema","tmpDir","mkdirSync","getSchemaDir","tag","dev","verbose","log","repositoryDir","join","tagDir","existsSync","run","repository","renameSync","getValidator","version","ajv","dir","references","reference","push","file","addSchema","Schema","charAt","toUpperCase","slice","s","compile","getVersion","schemaVersion","undefined","getSchemaURL","replace","validate","filename","validator","valid","schemaURL","errors","process","status","data","main","directorySchema","d","endsWith","then","exitCode","exit","catch"],"sources":["../../src/scripts/validate_json.ts"],"sourcesContent":["import Ajv from \"ajv\"\nimport commandLineArgs from \"command-line-args\"\nimport commandLineUsage from \"command-line-usage\"\nimport { load, getFiles } from \"../file_systems\"\nimport * as git from \"../git\"\nimport temp from \"temp\"\nimport fs from \"fs-extra\"\nimport path from \"path\"\nimport {\n  validDataset,\n  getDatasets,\n  getSchemas,\n  datasetDirectorySchema,\n} from \"../../src/datasets\"\n\n// Automatically track and cleanup files at exit\ntemp.track()\n\nfunction parseArgs(argv: string[]): any {\n  const optionsDefinitions = [\n    {\n      description: \"pathname to the directory containing the dataset\",\n      defaultOption: true,\n      name: \"data\",\n      type: String,\n    },\n    {\n      description: \"URL or path to the tricoteuses-assemblee git repository\",\n      defaultValue:\n        \"https://git.en-root.org/tricoteuses/tricoteuses-assemblee.git\",\n      name: \"repository\",\n      type: String,\n    },\n    {\n      description: `dataset (possible values must include exactly one of ${getDatasets()})`,\n      name: \"dataset\",\n      type: String,\n    },\n    {\n      description: `schema (possible values are ${getSchemas()})`,\n      name: \"schema\",\n      type: String,\n    },\n    {\n      description:\n        \"ignore schemaVersion from JSON files and use the current directory instead of the --repository\",\n      name: \"dev\",\n      type: Boolean,\n    },\n    {\n      description: \"increase verbosity\",\n      name: \"verbose\",\n      type: Boolean,\n    },\n    {\n      name: \"help\",\n      description: \"Print this usage guide.\",\n    },\n  ]\n  const sections = [\n    {\n      header: \"Validate JSON files\",\n      content: \"Validate JSON files\",\n    },\n    {\n      header: \"Options\",\n      optionList: optionsDefinitions,\n    },\n  ]\n  const options = commandLineArgs(optionsDefinitions, {\n    argv: argv,\n  })\n\n  if (options.dataset) {\n    if (!validDataset(options.dataset)) {\n      console.error(`--dataset ${options.dataset}`)\n      options.help = true\n    }\n  }\n\n  if (options.schema) {\n    if (!getSchemas().includes(options.schema)) {\n      console.error(`--schema ${options.schema}`)\n      options.help = true\n    }\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 Validate {\n  options: any\n  version2schema: any\n  tmpDir: string\n\n  constructor(options: any) {\n    this.options = options\n    this.version2schema = {}\n    this.tmpDir = temp.mkdirSync(\"validation\")\n  }\n\n  getSchemaDir(tag: string) {\n    if (this.options.dev == true) {\n      if (this.options.verbose) console.log(\"--dev: use src/schemas\")\n      return \"src/schemas\"\n    }\n    const repositoryDir = path.join(this.tmpDir, \"tricoteuses-assemblee\")\n    const tagDir = path.join(this.tmpDir, tag)\n    if (!fs.existsSync(tagDir)) {\n      if (!fs.existsSync(repositoryDir)) {\n        git.run(\n          this.tmpDir,\n          `clone --quiet ${this.options.repository} ${repositoryDir}`,\n        )\n      }\n      git.run(\n        repositoryDir,\n        `checkout tags/${tag} -- src/schemas > /dev/null 2>&1`,\n      )\n      fs.renameSync(path.join(repositoryDir, \"src/schemas\"), tagDir)\n    }\n    return tagDir\n  }\n\n  getValidator(version: string): any {\n    if (!(version in this.version2schema)) {\n      const ajv = new Ajv()\n      const dir = this.getSchemaDir(`schema-${version}`)\n      let references = []\n      for (const reference of getFiles([\n        `${dir}/${this.options.schema}/*.json`,\n      ]))\n        references.push(reference)\n      for (const file of getFiles(references)) {\n        if (this.options.verbose) console.log(`reference: ${file}`)\n        ajv.addSchema(load(file))\n      }\n      const schema = this.options.schema\n      const Schema = schema.charAt(0).toUpperCase() + schema.slice(1)\n      const s = `${dir}/${schema}/${Schema}.json`\n      if (this.options.verbose) console.log(`schema ${s}`)\n      this.version2schema[version] = ajv.compile(load(s))\n    }\n    return this.version2schema[version]\n  }\n\n  getVersion(content: any): string {\n    if (content.schemaVersion != undefined) return content.schemaVersion\n    return `${this.options.schema}-1.0`\n  }\n\n  getSchemaURL(version: string): any {\n    if (this.options.dev == true) return \"src/schemas\"\n    else {\n      if (fs.existsSync(this.options.repository)) {\n        return `${this.options.repository}/src/schemas at tag schema-${version}`\n      } else {\n        const repository = this.options.repository.replace(/\\.git$/, \"\")\n        return `${repository}/tree/schema-${version}/src/schemas`\n      }\n    }\n  }\n\n  validate(content: any, filename: string): number {\n    const tag = this.getVersion(content)\n    const validator = this.getValidator(tag)\n    const valid = validator(content)\n    const schemaURL = this.getSchemaURL(tag)\n    if (!valid) {\n      console.error(`${schemaURL} finds ${filename} is invalid`)\n      console.error(validator.errors)\n      return 1\n    } else {\n      if (this.options.verbose)\n        console.log(`${schemaURL} finds ${filename} is valid`)\n      return 0\n    }\n  }\n\n  async process(): Promise<number> {\n    let status = 0\n    for (const file of getFiles([this.options.data])) {\n      const content = load(file)\n      status |= this.validate(content, file)\n    }\n    return status\n  }\n}\n\nasync function main(argv: any): Promise<number> {\n  const options = parseArgs(argv)\n  if (options === null) return 1\n  let directorySchema = []\n  if (options.schema) {\n    directorySchema = [[options.data, options.schema]]\n  } else if (options.dataset) {\n    directorySchema = datasetDirectorySchema(options.dataset)\n  } else {\n    console.error(\"either --schema or --dataset is required\")\n    return 1\n  }\n  for (const d of directorySchema) {\n    options.schema = d[1]\n    options.data = d[0]\n    if (options.verbose)\n      console.log(\n        `working on files ${options.data} with schema ${options.schema}`,\n      )\n    const validate = new Validate(options)\n    const status = await validate.process()\n    if (status != 0) return status\n  }\n  return 0\n}\n\n/* istanbul ignore if */\nif (process.argv[1].endsWith(\"validate_json.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":";;;AAAA,OAAOA,GAAG,MAAM,KAAK;AACrB,OAAOC,eAAe,MAAM,mBAAmB;AAC/C,OAAOC,gBAAgB,MAAM,oBAAoB;AAAA,SACxCC,IAAI,EAAEC,QAAQ;AAAA,OAChB,KAAKC,GAAG;AACf,OAAOC,IAAI,MAAM,MAAM;AACvB,OAAOC,EAAE,MAAM,UAAU;AACzB,OAAOC,IAAI,MAAM,MAAM;AAAA,SAErBC,YAAY,EACZC,WAAW,EACXC,UAAU,EACVC,sBAAsB,kCAGxB;AACAN,IAAI,CAACO,KAAK,CAAC,CAAC;AAEZ,SAASC,SAASA,CAACC,IAAc,EAAO;EACtC,MAAMC,kBAAkB,GAAG,CACzB;IACEC,WAAW,EAAE,kDAAkD;IAC/DC,aAAa,EAAE,IAAI;IACnBC,IAAI,EAAE,MAAM;IACZC,IAAI,EAAEC;EACR,CAAC,EACD;IACEJ,WAAW,EAAE,yDAAyD;IACtEK,YAAY,EACV,+DAA+D;IACjEH,IAAI,EAAE,YAAY;IAClBC,IAAI,EAAEC;EACR,CAAC,EACD;IACEJ,WAAW,EAAE,wDAAwDP,WAAW,CAAC,CAAC,GAAG;IACrFS,IAAI,EAAE,SAAS;IACfC,IAAI,EAAEC;EACR,CAAC,EACD;IACEJ,WAAW,EAAE,+BAA+BN,UAAU,CAAC,CAAC,GAAG;IAC3DQ,IAAI,EAAE,QAAQ;IACdC,IAAI,EAAEC;EACR,CAAC,EACD;IACEJ,WAAW,EACT,gGAAgG;IAClGE,IAAI,EAAE,KAAK;IACXC,IAAI,EAAEG;EACR,CAAC,EACD;IACEN,WAAW,EAAE,oBAAoB;IACjCE,IAAI,EAAE,SAAS;IACfC,IAAI,EAAEG;EACR,CAAC,EACD;IACEJ,IAAI,EAAE,MAAM;IACZF,WAAW,EAAE;EACf,CAAC,CACF;EACD,MAAMO,QAAQ,GAAG,CACf;IACEC,MAAM,EAAE,qBAAqB;IAC7BC,OAAO,EAAE;EACX,CAAC,EACD;IACED,MAAM,EAAE,SAAS;IACjBE,UAAU,EAAEX;EACd,CAAC,CACF;EACD,MAAMY,OAAO,GAAG3B,eAAe,CAACe,kBAAkB,EAAE;IAClDD,IAAI,EAAEA;EACR,CAAC,CAAC;EAEF,IAAIa,OAAO,CAACC,OAAO,EAAE;IACnB,IAAI,CAACpB,YAAY,CAACmB,OAAO,CAACC,OAAO,CAAC,EAAE;MAClCC,OAAO,CAACC,KAAK,CAAC,aAAaH,OAAO,CAACC,OAAO,EAAE,CAAC;MAC7CD,OAAO,CAACI,IAAI,GAAG,IAAI;IACrB;EACF;EAEA,IAAIJ,OAAO,CAACK,MAAM,EAAE;IAClB,IAAI,CAACtB,UAAU,CAAC,CAAC,CAACuB,QAAQ,CAACN,OAAO,CAACK,MAAM,CAAC,EAAE;MAC1CH,OAAO,CAACC,KAAK,CAAC,YAAYH,OAAO,CAACK,MAAM,EAAE,CAAC;MAC3CL,OAAO,CAACI,IAAI,GAAG,IAAI;IACrB;EACF;EAEA,IAAI,MAAM,IAAIJ,OAAO,EAAE;IACrB,MAAMO,KAAK,GAAGjC,gBAAgB,CAACsB,QAAQ,CAAC;IACxCM,OAAO,CAACM,IAAI,CAACD,KAAK,CAAC;IACnB,OAAO,IAAI;EACb;EACA,OAAOP,OAAO;AAChB;AAEA,MAAMS,QAAQ,CAAC;EAKbC,WAAWA,CAACV,OAAY,EAAE;IAAAW,eAAA;IAAAA,eAAA;IAAAA,eAAA;IACxB,IAAI,CAACX,OAAO,GAAGA,OAAO;IACtB,IAAI,CAACY,cAAc,GAAG,CAAC,CAAC;IACxB,IAAI,CAACC,MAAM,GAAGnC,IAAI,CAACoC,SAAS,CAAC,YAAY,CAAC;EAC5C;EAEAC,YAAYA,CAACC,GAAW,EAAE;IACxB,IAAI,IAAI,CAAChB,OAAO,CAACiB,GAAG,IAAI,IAAI,EAAE;MAC5B,IAAI,IAAI,CAACjB,OAAO,CAACkB,OAAO,EAAEhB,OAAO,CAACiB,GAAG,CAAC,wBAAwB,CAAC;MAC/D,OAAO,aAAa;IACtB;IACA,MAAMC,aAAa,GAAGxC,IAAI,CAACyC,IAAI,CAAC,IAAI,CAACR,MAAM,EAAE,uBAAuB,CAAC;IACrE,MAAMS,MAAM,GAAG1C,IAAI,CAACyC,IAAI,CAAC,IAAI,CAACR,MAAM,EAAEG,GAAG,CAAC;IAC1C,IAAI,CAACrC,EAAE,CAAC4C,UAAU,CAACD,MAAM,CAAC,EAAE;MAC1B,IAAI,CAAC3C,EAAE,CAAC4C,UAAU,CAACH,aAAa,CAAC,EAAE;QACjC3C,GAAG,CAAC+C,GAAG,CACL,IAAI,CAACX,MAAM,EACX,iBAAiB,IAAI,CAACb,OAAO,CAACyB,UAAU,IAAIL,aAAa,EAC3D,CAAC;MACH;MACA3C,GAAG,CAAC+C,GAAG,CACLJ,aAAa,EACb,iBAAiBJ,GAAG,kCACtB,CAAC;MACDrC,EAAE,CAAC+C,UAAU,CAAC9C,IAAI,CAACyC,IAAI,CAACD,aAAa,EAAE,aAAa,CAAC,EAAEE,MAAM,CAAC;IAChE;IACA,OAAOA,MAAM;EACf;EAEAK,YAAYA,CAACC,OAAe,EAAO;IACjC,IAAI,EAAEA,OAAO,IAAI,IAAI,CAAChB,cAAc,CAAC,EAAE;MACrC,MAAMiB,GAAG,GAAG,IAAIzD,GAAG,CAAC,CAAC;MACrB,MAAM0D,GAAG,GAAG,IAAI,CAACf,YAAY,CAAC,UAAUa,OAAO,EAAE,CAAC;MAClD,IAAIG,UAAU,GAAG,EAAE;MACnB,KAAK,MAAMC,SAAS,IAAIxD,QAAQ,CAAC,CAC/B,GAAGsD,GAAG,IAAI,IAAI,CAAC9B,OAAO,CAACK,MAAM,SAAS,CACvC,CAAC,EACA0B,UAAU,CAACE,IAAI,CAACD,SAAS,CAAC;MAC5B,KAAK,MAAME,IAAI,IAAI1D,QAAQ,CAACuD,UAAU,CAAC,EAAE;QACvC,IAAI,IAAI,CAAC/B,OAAO,CAACkB,OAAO,EAAEhB,OAAO,CAACiB,GAAG,CAAC,cAAce,IAAI,EAAE,CAAC;QAC3DL,GAAG,CAACM,SAAS,CAAC5D,IAAI,CAAC2D,IAAI,CAAC,CAAC;MAC3B;MACA,MAAM7B,MAAM,GAAG,IAAI,CAACL,OAAO,CAACK,MAAM;MAClC,MAAM+B,MAAM,GAAG/B,MAAM,CAACgC,MAAM,CAAC,CAAC,CAAC,CAACC,WAAW,CAAC,CAAC,GAAGjC,MAAM,CAACkC,KAAK,CAAC,CAAC,CAAC;MAC/D,MAAMC,CAAC,GAAG,GAAGV,GAAG,IAAIzB,MAAM,IAAI+B,MAAM,OAAO;MAC3C,IAAI,IAAI,CAACpC,OAAO,CAACkB,OAAO,EAAEhB,OAAO,CAACiB,GAAG,CAAC,UAAUqB,CAAC,EAAE,CAAC;MACpD,IAAI,CAAC5B,cAAc,CAACgB,OAAO,CAAC,GAAGC,GAAG,CAACY,OAAO,CAAClE,IAAI,CAACiE,CAAC,CAAC,CAAC;IACrD;IACA,OAAO,IAAI,CAAC5B,cAAc,CAACgB,OAAO,CAAC;EACrC;EAEAc,UAAUA,CAAC5C,OAAY,EAAU;IAC/B,IAAIA,OAAO,CAAC6C,aAAa,IAAIC,SAAS,EAAE,OAAO9C,OAAO,CAAC6C,aAAa;IACpE,OAAO,GAAG,IAAI,CAAC3C,OAAO,CAACK,MAAM,MAAM;EACrC;EAEAwC,YAAYA,CAACjB,OAAe,EAAO;IACjC,IAAI,IAAI,CAAC5B,OAAO,CAACiB,GAAG,IAAI,IAAI,EAAE,OAAO,aAAa,MAC7C;MACH,IAAItC,EAAE,CAAC4C,UAAU,CAAC,IAAI,CAACvB,OAAO,CAACyB,UAAU,CAAC,EAAE;QAC1C,OAAO,GAAG,IAAI,CAACzB,OAAO,CAACyB,UAAU,8BAA8BG,OAAO,EAAE;MAC1E,CAAC,MAAM;QACL,MAAMH,UAAU,GAAG,IAAI,CAACzB,OAAO,CAACyB,UAAU,CAACqB,OAAO,CAAC,QAAQ,EAAE,EAAE,CAAC;QAChE,OAAO,GAAGrB,UAAU,gBAAgBG,OAAO,cAAc;MAC3D;IACF;EACF;EAEAmB,QAAQA,CAACjD,OAAY,EAAEkD,QAAgB,EAAU;IAC/C,MAAMhC,GAAG,GAAG,IAAI,CAAC0B,UAAU,CAAC5C,OAAO,CAAC;IACpC,MAAMmD,SAAS,GAAG,IAAI,CAACtB,YAAY,CAACX,GAAG,CAAC;IACxC,MAAMkC,KAAK,GAAGD,SAAS,CAACnD,OAAO,CAAC;IAChC,MAAMqD,SAAS,GAAG,IAAI,CAACN,YAAY,CAAC7B,GAAG,CAAC;IACxC,IAAI,CAACkC,KAAK,EAAE;MACVhD,OAAO,CAACC,KAAK,CAAC,GAAGgD,SAAS,UAAUH,QAAQ,aAAa,CAAC;MAC1D9C,OAAO,CAACC,KAAK,CAAC8C,SAAS,CAACG,MAAM,CAAC;MAC/B,OAAO,CAAC;IACV,CAAC,MAAM;MACL,IAAI,IAAI,CAACpD,OAAO,CAACkB,OAAO,EACtBhB,OAAO,CAACiB,GAAG,CAAC,GAAGgC,SAAS,UAAUH,QAAQ,WAAW,CAAC;MACxD,OAAO,CAAC;IACV;EACF;EAEA,MAAMK,OAAOA,CAAA,EAAoB;IAC/B,IAAIC,MAAM,GAAG,CAAC;IACd,KAAK,MAAMpB,IAAI,IAAI1D,QAAQ,CAAC,CAAC,IAAI,CAACwB,OAAO,CAACuD,IAAI,CAAC,CAAC,EAAE;MAChD,MAAMzD,OAAO,GAAGvB,IAAI,CAAC2D,IAAI,CAAC;MAC1BoB,MAAM,IAAI,IAAI,CAACP,QAAQ,CAACjD,OAAO,EAAEoC,IAAI,CAAC;IACxC;IACA,OAAOoB,MAAM;EACf;AACF;AAEA,eAAeE,IAAIA,CAACrE,IAAS,EAAmB;EAC9C,MAAMa,OAAO,GAAGd,SAAS,CAACC,IAAI,CAAC;EAC/B,IAAIa,OAAO,KAAK,IAAI,EAAE,OAAO,CAAC;EAC9B,IAAIyD,eAAe,GAAG,EAAE;EACxB,IAAIzD,OAAO,CAACK,MAAM,EAAE;IAClBoD,eAAe,GAAG,CAAC,CAACzD,OAAO,CAACuD,IAAI,EAAEvD,OAAO,CAACK,MAAM,CAAC,CAAC;EACpD,CAAC,MAAM,IAAIL,OAAO,CAACC,OAAO,EAAE;IAC1BwD,eAAe,GAAGzE,sBAAsB,CAACgB,OAAO,CAACC,OAAO,CAAC;EAC3D,CAAC,MAAM;IACLC,OAAO,CAACC,KAAK,CAAC,0CAA0C,CAAC;IACzD,OAAO,CAAC;EACV;EACA,KAAK,MAAMuD,CAAC,IAAID,eAAe,EAAE;IAC/BzD,OAAO,CAACK,MAAM,GAAGqD,CAAC,CAAC,CAAC,CAAC;IACrB1D,OAAO,CAACuD,IAAI,GAAGG,CAAC,CAAC,CAAC,CAAC;IACnB,IAAI1D,OAAO,CAACkB,OAAO,EACjBhB,OAAO,CAACiB,GAAG,CACT,oBAAoBnB,OAAO,CAACuD,IAAI,gBAAgBvD,OAAO,CAACK,MAAM,EAChE,CAAC;IACH,MAAM0C,QAAQ,GAAG,IAAItC,QAAQ,CAACT,OAAO,CAAC;IACtC,MAAMsD,MAAM,GAAG,MAAMP,QAAQ,CAACM,OAAO,CAAC,CAAC;IACvC,IAAIC,MAAM,IAAI,CAAC,EAAE,OAAOA,MAAM;EAChC;EACA,OAAO,CAAC;AACV;;AAEA;AACA,IAAID,OAAO,CAAClE,IAAI,CAAC,CAAC,CAAC,CAACwE,QAAQ,CAAC,kBAAkB,CAAC,EAC9CH,IAAI,CAACH,OAAO,CAAClE,IAAI,CAAC,CACfyE,IAAI,CAAEC,QAAQ,IAAK;EAClBR,OAAO,CAACS,IAAI,CAACD,QAAQ,CAAC;AACxB,CAAC,CAAC,CACDE,KAAK,CAAE5D,KAAK,IAAK;EAChBD,OAAO,CAACC,KAAK,CAACA,KAAK,CAAC;EACpBkD,OAAO,CAACS,IAAI,CAAC,CAAC,CAAC;AACjB,CAAC,CAAC","ignoreList":[]}