@tricoteuses/assemblee 1.8.0 → 1.9.1

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 (101) hide show
  1. package/lib/bugs/acteur-00010/plugin.test.js +2 -2
  2. package/lib/bugs/acteur-00010.js +3 -3
  3. package/lib/bugs/agenda-00002/plugin.test.js +3 -3
  4. package/lib/bugs/agenda-00002.js +5 -5
  5. package/lib/bugs/agenda-00008/plugin.test.js +2 -2
  6. package/lib/bugs/agenda-00008.js +3 -3
  7. package/lib/bugs/agenda-00011/plugin.test.js +6 -6
  8. package/lib/bugs/agenda-00011.js +8 -8
  9. package/lib/bugs.js +11 -11
  10. package/lib/cleaners/index.d.ts +1 -0
  11. package/lib/cleaners/index.js +8 -1
  12. package/lib/cleaners/index.mjs +2 -1
  13. package/lib/cleaners/scrutins.js +2 -1
  14. package/lib/cleaners/scrutins.mjs +3 -2
  15. package/lib/cleaners/textes_loi.d.ts +1 -0
  16. package/lib/cleaners/textes_loi.js +12 -0
  17. package/lib/cleaners/textes_loi.mjs +5 -0
  18. package/lib/cleaners/xml.d.ts +3 -0
  19. package/lib/cleaners/xml.js +13 -1
  20. package/lib/cleaners/xml.mjs +11 -1
  21. package/lib/datasets.d.ts +9 -2
  22. package/lib/datasets.js +24 -3
  23. package/lib/datasets.mjs +24 -3
  24. package/lib/dates.js +3 -3
  25. package/lib/examples/PIONANR5L15B0020/input.d.ts +2 -0
  26. package/lib/examples/PIONANR5L15B0020/input.js +10 -0
  27. package/lib/examples/PIONANR5L15B0020/input.mjs +193 -0
  28. package/lib/examples/PRJLANR5L16B0914/input.d.ts +2 -0
  29. package/lib/examples/PRJLANR5L16B0914/input.js +10 -0
  30. package/lib/examples/PRJLANR5L16B0914/input.mjs +322 -0
  31. package/lib/examples/PRJLANR5L16B2014/input.d.ts +2 -0
  32. package/lib/examples/PRJLANR5L16B2014/input.js +10 -0
  33. package/lib/examples/PRJLANR5L16B2014/input.mjs +577 -0
  34. package/lib/examples/PRJLANR5L16B2424/input.d.ts +2 -0
  35. package/lib/examples/PRJLANR5L16B2424/input.js +10 -0
  36. package/lib/examples/PRJLANR5L16B2424/input.mjs +322 -0
  37. package/lib/examples/PRJLANR5L16B2462/input.d.ts +2 -0
  38. package/lib/examples/PRJLANR5L16B2462/input.js +10 -0
  39. package/lib/examples/PRJLANR5L16B2462/input.mjs +1308 -0
  40. package/lib/loaders.d.ts +5 -0
  41. package/lib/loaders.js +308 -262
  42. package/lib/loaders.mjs +16 -3
  43. package/lib/logger.d.ts +16 -0
  44. package/lib/logger.js +35 -0
  45. package/lib/logger.mjs +26 -0
  46. package/lib/parsers/documents.js +2 -2
  47. package/lib/parsers/textes_lois.d.ts +16 -1
  48. package/lib/parsers/textes_lois.js +733 -28
  49. package/lib/parsers/textes_lois.mjs +587 -22
  50. package/lib/raw_types/acteurs_et_organes.js +1 -1
  51. package/lib/raw_types/agendas.js +1 -1
  52. package/lib/raw_types/amendements.js +1 -1
  53. package/lib/raw_types/debats.js +1 -1
  54. package/lib/raw_types/dossiers_legislatifs.js +1 -1
  55. package/lib/raw_types/questions.js +1 -1
  56. package/lib/raw_types/scrutins.d.ts +18 -13
  57. package/lib/raw_types/scrutins.js +28 -14
  58. package/lib/raw_types/scrutins.mjs +28 -14
  59. package/lib/raw_types/textes_loi.d.ts +93 -0
  60. package/lib/raw_types/textes_loi.js +429 -0
  61. package/lib/raw_types/textes_loi.mjs +378 -0
  62. package/lib/scripts/bugs_helper.js +9 -9
  63. package/lib/scripts/document_dossiers_legislatifs.js +1 -1
  64. package/lib/scripts/process_open_dataset.d.ts +17 -0
  65. package/lib/scripts/process_open_dataset.js +302 -0
  66. package/lib/scripts/process_open_dataset.mjs +136 -0
  67. package/lib/scripts/process_textes_loi_dataset.d.ts +7 -0
  68. package/lib/scripts/process_textes_loi_dataset.js +374 -0
  69. package/lib/scripts/process_textes_loi_dataset.mjs +237 -0
  70. package/lib/scripts/raw_types_from_amendements.js +2 -2
  71. package/lib/scripts/reorganize_data.js +1 -1
  72. package/lib/scripts/reorganize_data.mjs +1 -1
  73. package/lib/scripts/retrieve_deputes_photos.js +3 -3
  74. package/lib/scripts/retrieve_documents.js +4 -4
  75. package/lib/scripts/retrieve_open_data.js +41 -279
  76. package/lib/scripts/retrieve_open_data.mjs +13 -126
  77. package/lib/scripts/retrieve_pending_amendments.js +2 -2
  78. package/lib/scripts/retrieve_senateurs_photos.js +2 -2
  79. package/lib/scripts/retrieve_textes_lois.js +2 -2
  80. package/lib/scripts/shared/cli_helpers.d.ts +5 -0
  81. package/lib/scripts/shared/cli_helpers.js +7 -2
  82. package/lib/scripts/shared/cli_helpers.mjs +6 -1
  83. package/lib/scripts/test_iter_load.js +25 -10
  84. package/lib/scripts/test_iter_load.mjs +19 -11
  85. package/lib/scripts/validate_json.js +3 -3
  86. package/lib/types/acteurs_et_organes.js +1 -1
  87. package/lib/types/agendas.js +1 -1
  88. package/lib/types/amendements.js +1 -1
  89. package/lib/types/debats.js +1 -1
  90. package/lib/types/dossiers_legislatifs.js +1 -1
  91. package/lib/types/questions.js +1 -1
  92. package/lib/types/scrutins.d.ts +6 -3
  93. package/lib/types/scrutins.js +13 -7
  94. package/lib/types/scrutins.mjs +13 -7
  95. package/lib/types/textes_loi.d.ts +94 -0
  96. package/lib/types/textes_loi.js +429 -0
  97. package/lib/types/textes_loi.mjs +378 -0
  98. package/package.json +13 -12
  99. package/lib/scripts/retrieve_textes_lois_llm.d.ts +0 -1
  100. package/lib/scripts/retrieve_textes_lois_llm.js +0 -188
  101. package/lib/scripts/retrieve_textes_lois_llm.mjs +0 -85
@@ -1,43 +1,608 @@
1
1
  import { ChatOpenAI } from "@langchain/openai";
2
2
  import { ChatPromptTemplate } from "@langchain/core/prompts";
3
3
  import Ajv from "ajv";
4
- const texteLoiSchema = require("../schemas/texte_loi/TexteLoi.json");
4
+ import { delimiter, runLogFun } from "../logger.mjs";
5
+ const EXPOSE_MOTIFS_SCHEMA = require("../schemas/texte_loi/ExposeMotifs");
6
+ const META_DONNEES_SCHEMA = require("../schemas/texte_loi/MetaDonnees");
7
+ const PROJET_LOI_SCHEMA = require("../schemas/texte_loi/ProjetLoi");
8
+ const ALINEA_SCHEMA = require("../schemas/texte_loi/Alinea");
9
+ const SOMMAIRE_SCHEMA = require("../schemas/texte_loi/Sommaire");
10
+
11
+ /**
12
+ * Object containing input examples and their output schemas.
13
+ * Used to train the LLM.
14
+ * @type {Record<string, {motifs: any, input: any, loi: any, meta: any, sommaire: any}>}
15
+ */
16
+ const EXAMPLES = {
17
+ PRJLANR5L16B0914: {
18
+ motifs: require("../examples/PRJLANR5L16B0914/motifs"),
19
+ input: require("../examples/PRJLANR5L16B0914/input"),
20
+ loi: require("../examples/PRJLANR5L16B0914/loi"),
21
+ meta: require("../examples/PRJLANR5L16B0914/meta"),
22
+ sommaire: require("../examples/PRJLANR5L16B0914/sommaire")
23
+ },
24
+ PRJLANR5L16B2014: {
25
+ motifs: require("../examples/PRJLANR5L16B2014/motifs"),
26
+ input: require("../examples/PRJLANR5L16B2014/input"),
27
+ loi: require("../examples/PRJLANR5L16B2014/loi"),
28
+ meta: require("../examples/PRJLANR5L16B2014/meta"),
29
+ sommaire: require("../examples/PRJLANR5L16B2014/sommaire")
30
+ },
31
+ PRJLANR5L16B2424: {
32
+ alineas: require("../examples/PRJLANR5L16B2424/alineas"),
33
+ motifs: require("../examples/PRJLANR5L16B2424/motifs"),
34
+ input: require("../examples/PRJLANR5L16B2424/input"),
35
+ loi: require("../examples/PRJLANR5L16B2424/loi"),
36
+ meta: require("../examples/PRJLANR5L16B2424/meta"),
37
+ sommaire: require("../examples/PRJLANR5L16B2424/sommaire")
38
+ }
39
+ };
40
+ const FULL_TEXTE_SCHEMA = {
41
+ $schema: "http://json-schema.org/draft-07/schema#",
42
+ title: "Sommaire",
43
+ description: "Récupération d'un texte complet",
44
+ type: "object",
45
+ properties: {
46
+ texte: {
47
+ description: "Texte complet de la section indiquée.",
48
+ type: "string"
49
+ }
50
+ },
51
+ required: ["texte"]
52
+ };
53
+
54
+ /**
55
+ * LLM model.
56
+ */
57
+ let MODEL = null;
58
+ if (process.env.OPENAI_API_KEY) {
59
+ MODEL = new ChatOpenAI({
60
+ temperature: 0,
61
+ topP: 0.0,
62
+ model: "gpt-4o-mini",
63
+ apiKey: process.env.OPENAI_API_KEY,
64
+ maxTokens: -1
65
+ });
66
+ }
67
+
68
+ /**
69
+ * System input template for the LLM.
70
+ */
71
+ const SYSTEM_TEMPLATE = `
72
+ # Instructions :
73
+
74
+ Tu es un expert de la loi et de l'analyse de pages HTML.
75
+ Tu dois analyser un texte de loi afin de séparer et de structurer les éléments qui le composent. Le format d'entrée est HTML.
76
+
77
+ Le texte est divisé en deux parties :
78
+ A) un exposé des motifs qui indique les raisons pour lesquelles ce projet est soumis au Parlement. Il peut contenir 3 éléments : 1) une citation au début d'un document, 2) le texte principal qui développe les arguments de l'auteur à l'appui de la modification législative ou des dispositions nouvelles qu'il propose, 3) des paragraphes qui présentent les articles.
79
+ B) le dispositif qui est la partie normative et rédigée en articles. Il peut comporter des divisions nommées titres, chapitres, sections, sous-sections. Lorsqu'il y a un seul niveau de structure, on retrouvera le chapitre ; s’il y a deux niveaux de structure, on retrouvera des chapitres puis des sections. S’il y a trois niveaux de structure, on retrouvera des titres puis des chapitres puis des sections.
80
+
81
+ Si le texte de loi ne comporte pas de division, le sommaire doit rester vide.
82
+ Toutes les lignes qui débutent par un '«' ne sont pas à prendre en compte dans les divisions
83
+ Ta réponse doit être un JSON structuré avec les éléments du texte de loi en respectant le schéma JSON fourni.
84
+ `;
85
+ let SILENT_LOG = false;
86
+
87
+ /**
88
+ * Parses the given HTML string and returns the parsed textes loi.
89
+ *
90
+ * @param {string} html - The HTML string to parse.
91
+ * @return {Promise<{partial: {sommaire: any, loi: any, motifs: any, meta: any}, completed: {sommaire: any, loi: any, motifs: any, meta: any}}>} - The parsed textes loi, including the partial and completed versions.
92
+ */
93
+ export async function parseTexteLoi(html, options = {
94
+ silent: false
95
+ }) {
96
+ SILENT_LOG = options.silent;
97
+ runLogFun(console.info("Start extracting loi..."), options);
98
+ const meta = await extractMetaDonnees(html);
99
+ const sommaire = await extractSommaire(html);
100
+ const partialMotifs = await extractMotifs(html);
101
+ const partialLoi = await extractProjetLoi(html);
102
+ const exposeMotifs = await populateMotifs(partialMotifs, html);
103
+ const loi = await populateProjetLoi(partialLoi, html);
104
+ return {
105
+ sommaire,
106
+ loi,
107
+ exposeMotifs,
108
+ meta
109
+ };
110
+ }
111
+
112
+ /**
113
+ * Cleans the given HTML string by removing unnecessary tags and content.
114
+ *
115
+ * @param {string} html - The HTML string to be cleaned.
116
+ * @return {string} The cleaned HTML string.
117
+ */
5
118
  function cleanHtml(html) {
6
119
  let cleanedHtml = html.replace(/(<style[\w\W]+style>)/g, "");
7
120
  cleanedHtml = cleanedHtml.replace(/(<head[\w\W]+head>)/g, "");
8
121
  cleanedHtml = cleanedHtml.replace(/<img[^>]*>/g, "");
9
122
  return cleanedHtml;
10
123
  }
11
- export async function parseTexteLoi(html) {
12
- const cleanedHtml = cleanHtml(html);
13
- const model = new ChatOpenAI({
14
- temperature: 0,
15
- model: "gpt-4o",
16
- apiKey: process.env.OPENAI_API_KEY
124
+
125
+ /**
126
+ * Removes all HTML tags from the given HTML string and returns the cleaned text.
127
+ *
128
+ * @param {string} html - The HTML string from which to remove tags.
129
+ * @return {string} The cleaned text without any HTML tags.
130
+ */
131
+ function removeHtmlTags(html) {
132
+ // Regular expression to match HTML tags
133
+ const htmlTagRegex = /<[^>]*>/g;
134
+
135
+ // Replace all HTML tags with an empty string
136
+ const textWithoutTags = html.replace(htmlTagRegex, "");
137
+ return textWithoutTags;
138
+ }
139
+
140
+ /**
141
+ * Populates the motifs object with the extracted text from the given HTML.
142
+ *
143
+ * @param {Record<string, any>} motifs - The motifs object to be populated.
144
+ * @param {string} html - The HTML string from which to extract the text.
145
+ * @return {Promise<Record<string, any>>} - The populated motifs object.
146
+ */
147
+ async function populateMotifs(motifs, html) {
148
+ runLogFun(console.info("Populating expose motifs..."));
149
+ let cleanedHtml = cleanHtml(html);
150
+ if (motifs.debut === motifs.fin || !motifs.fin) {
151
+ motifs.texte = motifs.debut;
152
+ } else {
153
+ const systemTemplate = `
154
+ # Instructions :
155
+
156
+ Tu es un expert de la loi et de l'analyse de pages HTML.
157
+ Tu dois analyser un texte de loi afin d'en extraire certaines parties.
158
+ Le format d'entrée est du HTML. Le format de sortie est du texte.
159
+ Récupère l'intégralité du texte commençant par :
160
+
161
+ """
162
+ {debut}
163
+ """
164
+
165
+ Et finissant par :
166
+
167
+ """
168
+ {fin}
169
+ """
170
+ `;
171
+ const prompt = ChatPromptTemplate.fromMessages([["system", systemTemplate], ["human", "{input}"]]);
172
+ const structuredModel = MODEL.withStructuredOutput(FULL_TEXTE_SCHEMA);
173
+ const chain = prompt.pipe(structuredModel);
174
+ const articlePromptResult = await chain.invoke({
175
+ input: cleanedHtml,
176
+ debut: motifs.debut,
177
+ fin: motifs.fin
178
+ });
179
+ motifs.texte = articlePromptResult.texte;
180
+ }
181
+ runLogFun(console.info("Opération de peuplement de l'exposé des motifs terminée ✅"), {
182
+ silent: SILENT_LOG
17
183
  });
18
- const structuredModel = model.withStructuredOutput(texteLoiSchema);
184
+ delimiter(SILENT_LOG);
185
+ delete motifs.debut;
186
+ delete motifs.fin;
187
+ return motifs;
188
+ }
189
+
190
+ /**
191
+ * Asynchronously populates a projet loi object with the full text of each article and alinea in the given HTML.
192
+ *
193
+ * @param {Record<string, any>} partial - The partial projet loi object to be populated.
194
+ * @param {string} html - The HTML containing the text of the articles and alineas.
195
+ * @return {Promise<Record<string, any>>} A Promise that resolves to the fully populated projet loi object.
196
+ */
197
+ async function populateProjetLoi(partial, html) {
198
+ runLogFun(console.group("Peuplement du projet loi..."), {
199
+ silent: SILENT_LOG
200
+ });
201
+ let cleanedHtml = cleanHtml(html);
202
+ const articles = partial.articles;
203
+ const fullArticles = [];
19
204
  const systemTemplate = `
20
- Tu es un expert de la loi et de l'analyse de pages HTML.
21
- Tu dois analyser un texte de loi afin de séparer et structurer les éléments qui le composent. Le format d'entrée est HTML.
205
+ # Instructions :
22
206
 
23
- Le texte est divisé en deux parties :
24
- A) un exposé des motifs qui indique les raisons pour lesquelles ce projet est soumis au Parlement. Il peut contenir 3 éléments : 1) une citation au début d'un document, 2) le texte principal qui développe les arguments de l'auteur à l'appui de la modification législative ou des dispositions nouvelles qu'il propose, 3) des paragraphes qui présentent les articles.
25
- B) le dispositif qui est la partie normative et rédigée en articles. Il peut comporter des divisions nommées titres, chapitres, sections, sous-sections. Lorsqu'il y a un seul niveau de structure, on retrouvera le chapitre ; s’il y a deux niveaux de structure, on retrouvera des chapitres puis des sections. S’il y a trois niveaux de structure, on retrouvera des titres puis des chapitres puis des sections.
207
+ Tu es un expert de la loi et de l'analyse de pages HTML.
208
+ Tu dois analyser un texte de loi afin d'en extraire certaines parties.
209
+ Le format d'entrée est du HTML. Le format de sortie est du texte.
210
+ Récupère l'intégralité du texte commençant par :
26
211
 
27
- Si le texte de loi ne comporte pas de division, le sommaire doit rester vide.
28
- Toutes les lignes qui débutent par un '«' ne sont pas à prendre en compte dans les divisions
29
- Lorsque les textes ont été modifiés, il n'y pas d'exposé des motifs.
30
- Ta réponse doit être un JSON structuré avec les éléments du texte de loi en respectant le schéma JSON fourni.
31
- `;
212
+ """
213
+ {debut}
214
+ """
215
+
216
+ Et finissant par :
217
+
218
+ """
219
+ {fin}
220
+ """
221
+ `;
32
222
  const prompt = ChatPromptTemplate.fromMessages([["system", systemTemplate], ["human", "{input}"]]);
223
+ const structuredModel = MODEL.withStructuredOutput(FULL_TEXTE_SCHEMA);
224
+ const chain = prompt.pipe(structuredModel);
225
+ for (const article of articles) {
226
+ if (article.debut === article.fin || !article.fin) {
227
+ article.texte = article.debut;
228
+ } else {
229
+ const articlePromptResult = await chain.invoke({
230
+ input: cleanedHtml,
231
+ debut: article.debut,
232
+ fin: article.fin
233
+ });
234
+ article.texte = articlePromptResult.texte;
235
+ }
236
+ for (const alinea of article.alineas) {
237
+ if (alinea.debut === alinea.fin || !alinea.fin) {
238
+ alinea.texte = alinea.debut;
239
+ continue;
240
+ }
241
+ const alineaPromptResult = await chain.invoke({
242
+ input: article.texte,
243
+ debut: alinea.debut,
244
+ fin: alinea.fin
245
+ });
246
+ delete alinea.debut;
247
+ delete alinea.fin;
248
+ alinea.texte = alineaPromptResult.texte;
249
+ }
250
+ runLogFun(console.info("Opération de peuplement du contenu du projet de loi terminée ✅"), {
251
+ silent: SILENT_LOG
252
+ });
253
+ delete article.debut;
254
+ delete article.fin;
255
+ fullArticles.push(article);
256
+ }
257
+ const projetLoi = {
258
+ ...partial,
259
+ articles: fullArticles
260
+ };
261
+ runLogFun(console.groupEnd(), {
262
+ silent: SILENT_LOG
263
+ });
264
+ delimiter();
265
+ return projetLoi;
266
+ }
267
+
268
+ /**
269
+ * Asynchronously extracts the "expose motifs" from the given HTML.
270
+ *
271
+ * @param {string} html - The HTML string from which to extract the "expose motifs".
272
+ * @return {Promise<Record<string, any>>} - A Promise that resolves to the extracted "expose motifs" object.
273
+ * The object contains the following properties:
274
+ * - `debut`: The beginning of the "expose motifs".
275
+ * - `fin`: The end of the "expose motifs".
276
+ * @throws {Error} - If the extracted "expose motifs" do not validate against the EXPOSE_MOTIFS_SCHEMA.
277
+ */
278
+ async function extractMotifs(html) {
279
+ runLogFun(console.group("Parsing expose motifs..."), {
280
+ silent: SILENT_LOG
281
+ });
282
+ let cleanedHtml = cleanHtml(html);
283
+ cleanedHtml = removeHtmlTags(cleanedHtml);
284
+ const structuredModel = MODEL.withStructuredOutput(EXPOSE_MOTIFS_SCHEMA);
285
+ const motifsSystemTemplate = `
286
+ ${SYSTEM_TEMPLATE}
287
+
288
+ # Examples:
289
+
290
+ ## Exemple 1 User Input :
291
+ """
292
+ {example1}
293
+ """
294
+
295
+ ## Exemple 1 JSON Output :
296
+ """
297
+ {output1}
298
+ """
299
+
300
+ ## Exemple 2 User Input :
301
+ """
302
+ {example2}
303
+ """
304
+
305
+ ## Exemple 2 JSON Output :
306
+ """
307
+ {output2}
308
+ """
309
+ `;
310
+ const prompt = ChatPromptTemplate.fromMessages([["system", motifsSystemTemplate], ["human", "{input}"]]);
311
+ const chain = prompt.pipe(structuredModel);
312
+ const motifsPromptResult = await chain.invoke({
313
+ input: cleanedHtml,
314
+ example1: EXAMPLES.PRJLANR5L16B0914.input,
315
+ example2: EXAMPLES.PRJLANR5L16B2424.input,
316
+ output1: EXAMPLES.PRJLANR5L16B0914.motifs,
317
+ output2: EXAMPLES.PRJLANR5L16B2424.motifs
318
+ });
319
+
320
+ // console.log(JSON.stringify(motifsPromptResult))
321
+ runLogFun(console.table([["Début", motifsPromptResult.debut.substring(0, 30)], ["Fin", motifsPromptResult.fin.substring(0, 30)]]), {
322
+ silent: SILENT_LOG
323
+ });
324
+ runLogFun(console.groupEnd(), {
325
+ silent: SILENT_LOG
326
+ });
327
+ delimiter();
328
+ const ajv = new Ajv();
329
+ if (ajv.validate(EXPOSE_MOTIFS_SCHEMA, motifsPromptResult)) {
330
+ runLogFun(console.info("Exposé des motifs OK 👌"), {
331
+ silent: SILENT_LOG
332
+ });
333
+ return motifsPromptResult;
334
+ }
335
+ runLogFun(console.warn("Exposé des motifs KO ❌"), {
336
+ silent: SILENT_LOG
337
+ });
338
+ return {};
339
+ }
340
+
341
+ /**
342
+ * Asynchronously extracts the metadata from the given HTML.
343
+ *
344
+ * @param {string} html - The HTML string from which to extract the metadata.
345
+ * @return {Promise<Record<string, any>>} - A Promise that resolves to the extracted metadata object.
346
+ * The object contains the following properties:
347
+ * - `documentRef`: The reference of the document.
348
+ * - `titre`: The title of the document.
349
+ * - `numeroTexte`: The number of the text.
350
+ * - `procedureAcceleree`: The accelerated procedure.
351
+ * @throws {Error} - If the extracted metadata do not validate against the META_DONNEES_SCHEMA.
352
+ */
353
+ async function extractMetaDonnees(html) {
354
+ delimiter();
355
+ let cleanedHtml = cleanHtml(html);
356
+ cleanedHtml = removeHtmlTags(cleanedHtml);
357
+ const structuredModel = MODEL.withStructuredOutput(META_DONNEES_SCHEMA);
358
+ const metaDonneesSystemTemplate = `
359
+ ${SYSTEM_TEMPLATE}
360
+ `;
361
+ const prompt = ChatPromptTemplate.fromMessages([["system", metaDonneesSystemTemplate], ["human", "{input}"]]);
33
362
  const chain = prompt.pipe(structuredModel);
34
363
  const result = await chain.invoke({
35
- input: cleanedHtml
364
+ input: cleanedHtml,
365
+ example1: EXAMPLES.PRJLANR5L16B0914.input,
366
+ example2: EXAMPLES.PRJLANR5L16B2424.input,
367
+ output1: EXAMPLES.PRJLANR5L16B0914.meta,
368
+ output2: EXAMPLES.PRJLANR5L16B2424.meta
36
369
  });
370
+ console.group("Meta-données");
371
+ console.table([["Référence", result.documentRef], ["Titre", result.titre], ["Numéro du texte", result.numeroTexte], ["Procédure accéléré", result.procedureAcceleree]]);
372
+ console.groupEnd();
373
+ delimiter();
37
374
  const ajv = new Ajv();
38
- if (ajv.validate(texteLoiSchema, result)) {
375
+ if (ajv.validate(META_DONNEES_SCHEMA, result)) {
376
+ runLogFun(console.warn("Meta-données OK 👌"), {
377
+ silent: SILENT_LOG
378
+ });
39
379
  return result;
40
380
  }
41
- return false;
381
+ runLogFun(console.warn("Meta-données non-conforme ❌"), {
382
+ silent: SILENT_LOG
383
+ });
384
+ return {};
385
+ }
386
+
387
+ /**
388
+ * Asynchronously extracts the "projet de loi" from the given HTML.
389
+ *
390
+ * @param {string} html - The HTML string from which to extract the "projet de loi".
391
+ * @return {Promise<Record<string, any>>} - A Promise that resolves to the extracted "projet de loi" object.
392
+ * The object contains the following properties:
393
+ * - `articles`: An array of article objects, each containing the following properties:
394
+ * - `numeroArticle`: The number of the article.
395
+ * - `cardinal`: The cardinal of the article.
396
+ * - `titre`: The title of the article.
397
+ * - `debut`: The beginning of the article.
398
+ * - `fin`: The end of the article.
399
+ * - `alineas`: An array of alineas.
400
+ * @throws {Error} - If the extracted "projet de loi" does not validate against the PROJET_LOI_SCHEMA.
401
+ */
402
+ async function extractProjetLoi(html) {
403
+ runLogFun(console.group("Parsing loi..."), {
404
+ silent: SILENT_LOG
405
+ });
406
+ let cleanedHtml = cleanHtml(html);
407
+ const structuredModel = MODEL.withStructuredOutput(PROJET_LOI_SCHEMA);
408
+ const projetLoiSystemTemplate = `
409
+ ${SYSTEM_TEMPLATE}
410
+
411
+ # Examples:
412
+
413
+ ## Exemple 1 User Input :
414
+ """
415
+ {example1}
416
+ """
417
+
418
+ ## Exemple 1 JSON Output :
419
+ """
420
+ {output1}
421
+ """
422
+
423
+ ## Exemple 2 User Input :
424
+ """
425
+ {example2}
426
+ """
427
+
428
+ ## Exemple 2 JSON Output :
429
+ """
430
+ {output2}
431
+ """
432
+ `;
433
+ const prompt = ChatPromptTemplate.fromMessages([["system", projetLoiSystemTemplate], ["human", "{input}"]]);
434
+ const chain = prompt.pipe(structuredModel);
435
+ const articlesPromptResult = await chain.invoke({
436
+ input: cleanedHtml,
437
+ example1: EXAMPLES.PRJLANR5L16B0914.input,
438
+ example2: EXAMPLES.PRJLANR5L16B2424.input,
439
+ output1: EXAMPLES.PRJLANR5L16B0914.loi,
440
+ output2: EXAMPLES.PRJLANR5L16B2424.loi
441
+ });
442
+ runLogFun(console.log("Found ", articlesPromptResult.articles.length, " articles."), {
443
+ silent: SILENT_LOG
444
+ });
445
+ for (const article of articlesPromptResult.articles) {
446
+ article.alineas = await extractAlineas(html, article);
447
+ }
448
+ runLogFun(console.table(articlesPromptResult.articles.map(article => [article.numeroArticle, article.titre, article.debut.substring(0, 10), article.fin.substring(0, 10), `Nombre d'alineas : ${article.alineas.length}`])), {
449
+ silent: SILENT_LOG
450
+ });
451
+ runLogFun(console.groupEnd(), {
452
+ silent: SILENT_LOG
453
+ });
454
+ const ajv = new Ajv();
455
+ if (ajv.validate(PROJET_LOI_SCHEMA, articlesPromptResult)) {
456
+ runLogFun(console.info("Projet de loi OK 👌"), {
457
+ silent: SILENT_LOG
458
+ });
459
+ delimiter();
460
+ return articlesPromptResult;
461
+ }
462
+ runLogFun(console.warn("Projet de loi KO ❌"), {
463
+ silent: SILENT_LOG
464
+ });
465
+ delimiter();
466
+ return {};
467
+ }
468
+ async function extractAlineas(html, article) {
469
+ runLogFun(console.group("Parsing alineas..."), {
470
+ silent: SILENT_LOG
471
+ });
472
+ let cleanedHtml = cleanHtml(html);
473
+ const structuredModel = MODEL.withStructuredOutput(ALINEA_SCHEMA);
474
+ const alineasSystemTemplate = `
475
+ ${SYSTEM_TEMPLATE}
476
+
477
+ Extrais tous les alineas de la portion de texte commençant par « {debut} » et finissant par « {fin} ».
478
+ Un alinéa contiennent toujours une balise « <img ».
479
+
480
+ # Examples:
481
+
482
+ ## Exemple 1 User Input :
483
+ """
484
+ {example1}
485
+ """
486
+
487
+ ## Exemple 1 JSON Output :
488
+ """
489
+ {output1}
490
+ """
491
+ `;
492
+ const prompt = ChatPromptTemplate.fromMessages([["system", alineasSystemTemplate], ["human", "{input}"]]);
493
+ const chain = prompt.pipe(structuredModel);
494
+ const alineasPromptResult = await chain.invoke({
495
+ input: cleanedHtml,
496
+ debut: article.debut,
497
+ fin: article.fin,
498
+ example1: EXAMPLES.PRJLANR5L16B2424.input,
499
+ output1: EXAMPLES.PRJLANR5L16B2424.alineas
500
+ });
501
+ const ajv = new Ajv();
502
+ if (ajv.validate(ALINEA_SCHEMA, alineasPromptResult)) {
503
+ runLogFun(console.info("Found ", alineasPromptResult.alineas.length, " alineas for Article ", article.numeroArticle, article.titre), {
504
+ silent: SILENT_LOG
505
+ });
506
+ runLogFun(console.groupEnd(), {
507
+ silent: SILENT_LOG
508
+ });
509
+ delimiter();
510
+ return alineasPromptResult.alineas;
511
+ }
512
+ runLogFun(console.warn("Alineas KO"), {
513
+ silent: SILENT_LOG
514
+ });
515
+ delimiter();
516
+ return {};
517
+ }
518
+
519
+ /**
520
+ * Asynchronously extracts the "sommaire" (outline) from the given HTML.
521
+ *
522
+ * @param {string} html - The HTML string from which to extract the "sommaire".
523
+ * @return {Promise<Record<string, any>>} - A Promise that resolves to the extracted "sommaire" object.
524
+ * The object contains the following properties:
525
+ * - `sommaire`: An array of main division objects, each containing the following properties:
526
+ * - `numeroDivision`: The number of the main division.
527
+ * - `titre`: The title of the main division.
528
+ * - `typeDivision`: The type of content in the main division.
529
+ * - `sequence`: The sequence of the main division.
530
+ * - `parent`: The parent of the main division.
531
+ * @throws {Error} - If the extracted "sommaire" does not validate against the SOMMAIRE_SCHEMA.
532
+ */
533
+ async function extractSommaire(html) {
534
+ let cleanedHtml = cleanHtml(html);
535
+ cleanedHtml = removeHtmlTags(cleanedHtml);
536
+ const structuredModel = MODEL.withStructuredOutput(SOMMAIRE_SCHEMA);
537
+ const sommaireSystemTemplate = `
538
+ ${SYSTEM_TEMPLATE}
539
+
540
+ # Examples:
541
+
542
+ ## Exemple 1 User Input :
543
+ """
544
+ {example1}
545
+ """
546
+
547
+ ## Exemple 1 JSON Output :
548
+ """
549
+ {output1}
550
+ """
551
+
552
+ ## Exemple 2 User Input :
553
+ """
554
+ {example2}
555
+ """
556
+
557
+ ## Exemple 2 JSON Output :
558
+ """
559
+ {output2}
560
+ """
561
+ `;
562
+ const prompt = ChatPromptTemplate.fromMessages([["system", sommaireSystemTemplate], ["human", "{input}"]]);
563
+ const chain = prompt.pipe(structuredModel);
564
+ const sommairePromptResult = await chain.invoke({
565
+ input: cleanedHtml,
566
+ example1: EXAMPLES.PRJLANR5L16B2014.input,
567
+ output1: EXAMPLES.PRJLANR5L16B2014.sommaire,
568
+ example2: EXAMPLES.PRJLANR5L16B0914.input,
569
+ output2: EXAMPLES.PRJLANR5L16B0914.sommaire
570
+ });
571
+ if (!sommairePromptResult.sommaire) {
572
+ runLogFun(console.warn("Aucun sommaire n'a été trouvé ❓"), {
573
+ silent: SILENT_LOG
574
+ });
575
+ delimiter();
576
+ return {
577
+ sommaire: []
578
+ };
579
+ }
580
+ runLogFun(console.group("Sommaire"), {
581
+ silent: SILENT_LOG
582
+ });
583
+ runLogFun(console.log("Found ", sommairePromptResult.sommaire.length, " main divisions (level 1)."), {
584
+ silent: SILENT_LOG
585
+ });
586
+ runLogFun(console.table(sommairePromptResult.sommaire.map(item => [item.sequence, item.titre, item.typeDivision, item.parent])), {
587
+ silent: SILENT_LOG
588
+ });
589
+ runLogFun(console.groupEnd(), {
590
+ silent: SILENT_LOG
591
+ });
592
+ const ajv = new Ajv();
593
+ if (ajv.validate(SOMMAIRE_SCHEMA, sommairePromptResult)) {
594
+ runLogFun(console.info("Sommaire OK 👌"), {
595
+ silent: SILENT_LOG
596
+ });
597
+ delimiter();
598
+ return sommairePromptResult.sommaire;
599
+ }
600
+ runLogFun(console.warn("La construction du sommaire est KO ❌"), {
601
+ silent: SILENT_LOG
602
+ });
603
+ delimiter();
604
+ return {
605
+ sommaire: []
606
+ };
42
607
  }
43
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJDaGF0T3BlbkFJIiwiQ2hhdFByb21wdFRlbXBsYXRlIiwiQWp2IiwidGV4dGVMb2lTY2hlbWEiLCJyZXF1aXJlIiwiY2xlYW5IdG1sIiwiaHRtbCIsImNsZWFuZWRIdG1sIiwicmVwbGFjZSIsInBhcnNlVGV4dGVMb2kiLCJtb2RlbCIsInRlbXBlcmF0dXJlIiwiYXBpS2V5IiwicHJvY2VzcyIsImVudiIsIk9QRU5BSV9BUElfS0VZIiwic3RydWN0dXJlZE1vZGVsIiwid2l0aFN0cnVjdHVyZWRPdXRwdXQiLCJzeXN0ZW1UZW1wbGF0ZSIsInByb21wdCIsImZyb21NZXNzYWdlcyIsImNoYWluIiwicGlwZSIsInJlc3VsdCIsImludm9rZSIsImlucHV0IiwiYWp2IiwidmFsaWRhdGUiXSwic291cmNlcyI6WyIuLi8uLi9zcmMvcGFyc2Vycy90ZXh0ZXNfbG9pcy50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDaGF0T3BlbkFJIH0gZnJvbSBcIkBsYW5nY2hhaW4vb3BlbmFpXCJcbmltcG9ydCB7IENoYXRQcm9tcHRUZW1wbGF0ZSB9IGZyb20gXCJAbGFuZ2NoYWluL2NvcmUvcHJvbXB0c1wiXG5pbXBvcnQgQWp2IGZyb20gXCJhanZcIlxuY29uc3QgdGV4dGVMb2lTY2hlbWEgPSByZXF1aXJlKFwiLi4vc2NoZW1hcy90ZXh0ZV9sb2kvVGV4dGVMb2kuanNvblwiKVxuXG5mdW5jdGlvbiBjbGVhbkh0bWwoaHRtbDogc3RyaW5nKSB7XG4gIGxldCBjbGVhbmVkSHRtbCA9IGh0bWwucmVwbGFjZSgvKDxzdHlsZVtcXHdcXFddK3N0eWxlPikvZywgXCJcIilcbiAgY2xlYW5lZEh0bWwgPSBjbGVhbmVkSHRtbC5yZXBsYWNlKC8oPGhlYWRbXFx3XFxXXStoZWFkPikvZywgXCJcIilcbiAgY2xlYW5lZEh0bWwgPSBjbGVhbmVkSHRtbC5yZXBsYWNlKC88aW1nW14+XSo+L2csIFwiXCIpXG5cbiAgcmV0dXJuIGNsZWFuZWRIdG1sXG59XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBwYXJzZVRleHRlTG9pKGh0bWw6IHN0cmluZykge1xuICBjb25zdCBjbGVhbmVkSHRtbCA9IGNsZWFuSHRtbChodG1sKVxuXG4gIGNvbnN0IG1vZGVsID0gbmV3IENoYXRPcGVuQUkoe1xuICAgIHRlbXBlcmF0dXJlOiAwLFxuICAgIG1vZGVsOiBcImdwdC00b1wiLFxuICAgIGFwaUtleTogcHJvY2Vzcy5lbnYuT1BFTkFJX0FQSV9LRVksXG4gIH0pXG5cbiAgY29uc3Qgc3RydWN0dXJlZE1vZGVsID0gbW9kZWwud2l0aFN0cnVjdHVyZWRPdXRwdXQodGV4dGVMb2lTY2hlbWEpXG5cbiAgY29uc3Qgc3lzdGVtVGVtcGxhdGUgPSBgXG4gICAgICAgIFR1IGVzIHVuIGV4cGVydCBkZSBsYSBsb2kgZXQgZGUgbCdhbmFseXNlIGRlIHBhZ2VzIEhUTUwuXG4gICAgICAgIFR1IGRvaXMgYW5hbHlzZXIgdW4gdGV4dGUgZGUgbG9pIGFmaW4gZGUgc8OpcGFyZXIgZXQgc3RydWN0dXJlciBsZXMgw6lsw6ltZW50cyBxdWkgbGUgY29tcG9zZW50LiBMZSBmb3JtYXQgZCdlbnRyw6llIGVzdCBIVE1MLlxuXG4gICAgICAgIExlIHRleHRlIGVzdCBkaXZpc8OpIGVuIGRldXggcGFydGllcyA6XG4gICAgICAgIEEpIHVuIGV4cG9zw6kgZGVzIG1vdGlmcyBxdWkgaW5kaXF1ZSBsZXMgcmFpc29ucyBwb3VyIGxlc3F1ZWxsZXMgY2UgcHJvamV0IGVzdCBzb3VtaXMgYXUgUGFybGVtZW50LiBJbCBwZXV0IGNvbnRlbmlyIDMgw6lsw6ltZW50cyA6IDEpIHVuZSBjaXRhdGlvbiBhdSBkw6lidXQgZCd1biBkb2N1bWVudCwgMikgbGUgdGV4dGUgcHJpbmNpcGFsIHF1aSBkw6l2ZWxvcHBlIGxlcyBhcmd1bWVudHMgZGUgbCdhdXRldXIgw6AgbCdhcHB1aSBkZSBsYSBtb2RpZmljYXRpb24gbMOpZ2lzbGF0aXZlIG91IGRlcyBkaXNwb3NpdGlvbnMgbm91dmVsbGVzIHF1J2lsIHByb3Bvc2UsIDMpIGRlcyBwYXJhZ3JhcGhlcyBxdWkgcHLDqXNlbnRlbnQgbGVzIGFydGljbGVzLlxuICAgICAgICBCKSBsZSBkaXNwb3NpdGlmIHF1aSBlc3QgbGEgcGFydGllIG5vcm1hdGl2ZSBldCByw6lkaWfDqWUgZW4gYXJ0aWNsZXMuIElsIHBldXQgY29tcG9ydGVyIGRlcyBkaXZpc2lvbnMgbm9tbcOpZXMgdGl0cmVzLCBjaGFwaXRyZXMsIHNlY3Rpb25zLCBzb3VzLXNlY3Rpb25zLiBMb3JzcXUnaWwgeSBhIHVuIHNldWwgbml2ZWF1IGRlIHN0cnVjdHVyZSwgb24gcmV0cm91dmVyYSBsZSBjaGFwaXRyZSA7IHPigJlpbCB5IGEgZGV1eCBuaXZlYXV4IGRlIHN0cnVjdHVyZSwgb24gcmV0cm91dmVyYSBkZXMgY2hhcGl0cmVzIHB1aXMgZGVzIHNlY3Rpb25zLiBT4oCZaWwgeSBhIHRyb2lzIG5pdmVhdXggZGUgc3RydWN0dXJlLCBvbiByZXRyb3V2ZXJhIGRlcyB0aXRyZXMgcHVpcyBkZXMgY2hhcGl0cmVzIHB1aXMgZGVzIHNlY3Rpb25zLlxuXG4gICAgICAgIFNpIGxlIHRleHRlIGRlIGxvaSBuZSBjb21wb3J0ZSBwYXMgZGUgZGl2aXNpb24sIGxlIHNvbW1haXJlIGRvaXQgcmVzdGVyIHZpZGUuXG4gICAgICAgIFRvdXRlcyBsZXMgbGlnbmVzIHF1aSBkw6lidXRlbnQgcGFyIHVuICfCqycgbmUgc29udCBwYXMgw6AgcHJlbmRyZSBlbiBjb21wdGUgZGFucyBsZXMgZGl2aXNpb25zXG4gICAgICAgIExvcnNxdWUgbGVzIHRleHRlcyBvbnQgw6l0w6kgbW9kaWZpw6lzLCBpbCBuJ3kgcGFzIGQnZXhwb3PDqSBkZXMgbW90aWZzLlxuICAgICAgICBUYSByw6lwb25zZSBkb2l0IMOqdHJlIHVuIEpTT04gc3RydWN0dXLDqSBhdmVjIGxlcyDDqWzDqW1lbnRzIGR1IHRleHRlIGRlIGxvaSBlbiByZXNwZWN0YW50IGxlIHNjaMOpbWEgSlNPTiBmb3VybmkuXG4gICAgYFxuXG4gIGNvbnN0IHByb21wdCA9IENoYXRQcm9tcHRUZW1wbGF0ZS5mcm9tTWVzc2FnZXMoW1xuICAgIFtcInN5c3RlbVwiLCBzeXN0ZW1UZW1wbGF0ZV0sXG4gICAgW1wiaHVtYW5cIiwgXCJ7aW5wdXR9XCJdLFxuICBdKVxuXG4gIGNvbnN0IGNoYWluID0gcHJvbXB0LnBpcGUoc3RydWN0dXJlZE1vZGVsKVxuICBjb25zdCByZXN1bHQgPSBhd2FpdCBjaGFpbi5pbnZva2UoeyBpbnB1dDogY2xlYW5lZEh0bWwgfSlcblxuICBjb25zdCBhanYgPSBuZXcgQWp2KClcbiAgaWYgKGFqdi52YWxpZGF0ZSh0ZXh0ZUxvaVNjaGVtYSwgcmVzdWx0KSkge1xuICAgIHJldHVybiByZXN1bHRcbiAgfVxuICByZXR1cm4gZmFsc2Vcbn1cbiJdLCJtYXBwaW5ncyI6IkFBQUEsU0FBU0EsVUFBVSxRQUFRLG1CQUFtQjtBQUM5QyxTQUFTQyxrQkFBa0IsUUFBUSx5QkFBeUI7QUFDNUQsT0FBT0MsR0FBRyxNQUFNLEtBQUs7QUFDckIsTUFBTUMsY0FBYyxHQUFHQyxPQUFPLENBQUMsb0NBQW9DLENBQUM7QUFFcEUsU0FBU0MsU0FBU0EsQ0FBQ0MsSUFBWSxFQUFFO0VBQy9CLElBQUlDLFdBQVcsR0FBR0QsSUFBSSxDQUFDRSxPQUFPLENBQUMsd0JBQXdCLEVBQUUsRUFBRSxDQUFDO0VBQzVERCxXQUFXLEdBQUdBLFdBQVcsQ0FBQ0MsT0FBTyxDQUFDLHNCQUFzQixFQUFFLEVBQUUsQ0FBQztFQUM3REQsV0FBVyxHQUFHQSxXQUFXLENBQUNDLE9BQU8sQ0FBQyxhQUFhLEVBQUUsRUFBRSxDQUFDO0VBRXBELE9BQU9ELFdBQVc7QUFDcEI7QUFFQSxPQUFPLGVBQWVFLGFBQWFBLENBQUNILElBQVksRUFBRTtFQUNoRCxNQUFNQyxXQUFXLEdBQUdGLFNBQVMsQ0FBQ0MsSUFBSSxDQUFDO0VBRW5DLE1BQU1JLEtBQUssR0FBRyxJQUFJVixVQUFVLENBQUM7SUFDM0JXLFdBQVcsRUFBRSxDQUFDO0lBQ2RELEtBQUssRUFBRSxRQUFRO0lBQ2ZFLE1BQU0sRUFBRUMsT0FBTyxDQUFDQyxHQUFHLENBQUNDO0VBQ3RCLENBQUMsQ0FBQztFQUVGLE1BQU1DLGVBQWUsR0FBR04sS0FBSyxDQUFDTyxvQkFBb0IsQ0FBQ2QsY0FBYyxDQUFDO0VBRWxFLE1BQU1lLGNBQWMsR0FBRztBQUN6QjtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsS0FBSztFQUVILE1BQU1DLE1BQU0sR0FBR2xCLGtCQUFrQixDQUFDbUIsWUFBWSxDQUFDLENBQzdDLENBQUMsUUFBUSxFQUFFRixjQUFjLENBQUMsRUFDMUIsQ0FBQyxPQUFPLEVBQUUsU0FBUyxDQUFDLENBQ3JCLENBQUM7RUFFRixNQUFNRyxLQUFLLEdBQUdGLE1BQU0sQ0FBQ0csSUFBSSxDQUFDTixlQUFlLENBQUM7RUFDMUMsTUFBTU8sTUFBTSxHQUFHLE1BQU1GLEtBQUssQ0FBQ0csTUFBTSxDQUFDO0lBQUVDLEtBQUssRUFBRWxCO0VBQVksQ0FBQyxDQUFDO0VBRXpELE1BQU1tQixHQUFHLEdBQUcsSUFBSXhCLEdBQUcsQ0FBQyxDQUFDO0VBQ3JCLElBQUl3QixHQUFHLENBQUNDLFFBQVEsQ0FBQ3hCLGNBQWMsRUFBRW9CLE1BQU0sQ0FBQyxFQUFFO0lBQ3hDLE9BQU9BLE1BQU07RUFDZjtFQUNBLE9BQU8sS0FBSztBQUNkIiwiaWdub3JlTGlzdCI6W119
608
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["ChatOpenAI","ChatPromptTemplate","Ajv","delimiter","runLogFun","EXPOSE_MOTIFS_SCHEMA","require","META_DONNEES_SCHEMA","PROJET_LOI_SCHEMA","ALINEA_SCHEMA","SOMMAIRE_SCHEMA","EXAMPLES","PRJLANR5L16B0914","motifs","input","loi","meta","sommaire","PRJLANR5L16B2014","PRJLANR5L16B2424","alineas","FULL_TEXTE_SCHEMA","$schema","title","description","type","properties","texte","required","MODEL","process","env","OPENAI_API_KEY","temperature","topP","model","apiKey","maxTokens","SYSTEM_TEMPLATE","SILENT_LOG","parseTexteLoi","html","options","silent","console","info","extractMetaDonnees","extractSommaire","partialMotifs","extractMotifs","partialLoi","extractProjetLoi","exposeMotifs","populateMotifs","populateProjetLoi","cleanHtml","cleanedHtml","replace","removeHtmlTags","htmlTagRegex","textWithoutTags","debut","fin","systemTemplate","prompt","fromMessages","structuredModel","withStructuredOutput","chain","pipe","articlePromptResult","invoke","partial","group","articles","fullArticles","article","alinea","alineaPromptResult","push","projetLoi","groupEnd","motifsSystemTemplate","motifsPromptResult","example1","example2","output1","output2","table","substring","ajv","validate","warn","metaDonneesSystemTemplate","result","documentRef","titre","numeroTexte","procedureAcceleree","projetLoiSystemTemplate","articlesPromptResult","log","length","extractAlineas","map","numeroArticle","alineasSystemTemplate","alineasPromptResult","sommaireSystemTemplate","sommairePromptResult","item","sequence","typeDivision","parent"],"sources":["../../src/parsers/textes_lois.ts"],"sourcesContent":["import { ChatOpenAI } from \"@langchain/openai\"\nimport { ChatPromptTemplate } from \"@langchain/core/prompts\"\nimport Ajv from \"ajv\"\n\nimport { delimiter, runLogFun } from \"../logger\"\n\nconst EXPOSE_MOTIFS_SCHEMA = require(\"../schemas/texte_loi/ExposeMotifs\")\nconst META_DONNEES_SCHEMA = require(\"../schemas/texte_loi/MetaDonnees\")\nconst PROJET_LOI_SCHEMA = require(\"../schemas/texte_loi/ProjetLoi\")\nconst ALINEA_SCHEMA = require(\"../schemas/texte_loi/Alinea\")\nconst SOMMAIRE_SCHEMA = require(\"../schemas/texte_loi/Sommaire\")\n\n/**\n * Object containing input examples and their output schemas.\n * Used to train the LLM.\n * @type {Record<string, {motifs: any, input: any, loi: any, meta: any, sommaire: any}>}\n */\nconst EXAMPLES = {\n  PRJLANR5L16B0914: {\n    motifs: require(\"../examples/PRJLANR5L16B0914/motifs\"),\n    input: require(\"../examples/PRJLANR5L16B0914/input\"),\n    loi: require(\"../examples/PRJLANR5L16B0914/loi\"),\n    meta: require(\"../examples/PRJLANR5L16B0914/meta\"),\n    sommaire: require(\"../examples/PRJLANR5L16B0914/sommaire\"),\n  },\n  PRJLANR5L16B2014: {\n    motifs: require(\"../examples/PRJLANR5L16B2014/motifs\"),\n    input: require(\"../examples/PRJLANR5L16B2014/input\"),\n    loi: require(\"../examples/PRJLANR5L16B2014/loi\"),\n    meta: require(\"../examples/PRJLANR5L16B2014/meta\"),\n    sommaire: require(\"../examples/PRJLANR5L16B2014/sommaire\"),\n  },\n  PRJLANR5L16B2424: {\n    alineas: require(\"../examples/PRJLANR5L16B2424/alineas\"),\n    motifs: require(\"../examples/PRJLANR5L16B2424/motifs\"),\n    input: require(\"../examples/PRJLANR5L16B2424/input\"),\n    loi: require(\"../examples/PRJLANR5L16B2424/loi\"),\n    meta: require(\"../examples/PRJLANR5L16B2424/meta\"),\n    sommaire: require(\"../examples/PRJLANR5L16B2424/sommaire\"),\n  },\n}\n\nconst FULL_TEXTE_SCHEMA = {\n  $schema: \"http://json-schema.org/draft-07/schema#\",\n  title: \"Sommaire\",\n  description: \"Récupération d'un texte complet\",\n  type: \"object\",\n  properties: {\n    texte: {\n      description: \"Texte complet de la section indiquée.\",\n      type: \"string\",\n    },\n  },\n  required: [\"texte\"],\n}\n\n/**\n * LLM model.\n */\nlet MODEL: any = null\n\nif (process.env.OPENAI_API_KEY) {\n  MODEL = new ChatOpenAI({\n    temperature: 0,\n    topP: 0.0,\n    model: \"gpt-4o-mini\",\n    apiKey: process.env.OPENAI_API_KEY,\n    maxTokens: -1,\n  })\n}\n\n/**\n * System input template for the LLM.\n */\nconst SYSTEM_TEMPLATE: string = `\n  # Instructions :\n  \n  Tu es un expert de la loi et de l'analyse de pages HTML.\n  Tu dois analyser un texte de loi afin de séparer et de structurer les éléments qui le composent. Le format d'entrée est HTML.\n\n  Le texte est divisé en deux parties :\n  A) un exposé des motifs qui indique les raisons pour lesquelles ce projet est soumis au Parlement. Il peut contenir 3 éléments : 1) une citation au début d'un document, 2) le texte principal qui développe les arguments de l'auteur à l'appui de la modification législative ou des dispositions nouvelles qu'il propose, 3) des paragraphes qui présentent les articles.\n  B) le dispositif qui est la partie normative et rédigée en articles. Il peut comporter des divisions nommées titres, chapitres, sections, sous-sections. Lorsqu'il y a un seul niveau de structure, on retrouvera le chapitre ; s’il y a deux niveaux de structure, on retrouvera des chapitres puis des sections. S’il y a trois niveaux de structure, on retrouvera des titres puis des chapitres puis des sections.\n\n  Si le texte de loi ne comporte pas de division, le sommaire doit rester vide.\n  Toutes les lignes qui débutent par un '«' ne sont pas à prendre en compte dans les divisions\n  Ta réponse doit être un JSON structuré avec les éléments du texte de loi en respectant le schéma JSON fourni.\n`\n\nlet SILENT_LOG = false\n\n/**\n * Parses the given HTML string and returns the parsed textes loi.\n *\n * @param {string} html - The HTML string to parse.\n * @return {Promise<{partial: {sommaire: any, loi: any, motifs: any, meta: any}, completed: {sommaire: any, loi: any, motifs: any, meta: any}}>} - The parsed textes loi, including the partial and completed versions.\n */\nexport async function parseTexteLoi(html: string, options = { silent: false }) {\n  SILENT_LOG = options.silent\n\n  runLogFun(console.info(\"Start extracting loi...\"), options)\n  const meta = await extractMetaDonnees(html)\n  const sommaire = await extractSommaire(html)\n  const partialMotifs: any = await extractMotifs(html)\n  const partialLoi: any = await extractProjetLoi(html)\n  const exposeMotifs = await populateMotifs(partialMotifs, html)\n  const loi = await populateProjetLoi(partialLoi, html)\n\n  return {\n    sommaire,\n    loi,\n    exposeMotifs,\n    meta,\n  }\n}\n\n/**\n * Cleans the given HTML string by removing unnecessary tags and content.\n *\n * @param {string} html - The HTML string to be cleaned.\n * @return {string} The cleaned HTML string.\n */\nfunction cleanHtml(html: string) {\n  let cleanedHtml = html.replace(/(<style[\\w\\W]+style>)/g, \"\")\n  cleanedHtml = cleanedHtml.replace(/(<head[\\w\\W]+head>)/g, \"\")\n  cleanedHtml = cleanedHtml.replace(/<img[^>]*>/g, \"\")\n\n  return cleanedHtml\n}\n\n/**\n * Removes all HTML tags from the given HTML string and returns the cleaned text.\n *\n * @param {string} html - The HTML string from which to remove tags.\n * @return {string} The cleaned text without any HTML tags.\n */\nfunction removeHtmlTags(html: string): string {\n  // Regular expression to match HTML tags\n  const htmlTagRegex = /<[^>]*>/g\n\n  // Replace all HTML tags with an empty string\n  const textWithoutTags = html.replace(htmlTagRegex, \"\")\n  return textWithoutTags\n}\n\n/**\n * Populates the motifs object with the extracted text from the given HTML.\n *\n * @param {Record<string, any>} motifs - The motifs object to be populated.\n * @param {string} html - The HTML string from which to extract the text.\n * @return {Promise<Record<string, any>>} - The populated motifs object.\n */\nasync function populateMotifs(motifs: Record<string, any>, html: string) {\n  runLogFun(console.info(\"Populating expose motifs...\"))\n  let cleanedHtml = cleanHtml(html)\n\n  if (motifs.debut === motifs.fin || !motifs.fin) {\n    motifs.texte = motifs.debut\n  } else {\n    const systemTemplate = `\n      # Instructions :\n\n      Tu es un expert de la loi et de l'analyse de pages HTML.\n      Tu dois analyser un texte de loi afin d'en extraire certaines parties. \n      Le format d'entrée est du HTML. Le format de sortie est du texte.\n      Récupère l'intégralité du texte commençant par :\n\n      \"\"\"\n      {debut}\n      \"\"\"\n\n      Et finissant par :\n\n      \"\"\"\n      {fin}\n      \"\"\"\n    `\n\n    const prompt = ChatPromptTemplate.fromMessages([\n      [\"system\", systemTemplate],\n      [\"human\", \"{input}\"],\n    ])\n\n    const structuredModel = MODEL.withStructuredOutput(FULL_TEXTE_SCHEMA)\n    const chain = prompt.pipe(structuredModel)\n\n    const articlePromptResult: any = await chain.invoke({\n      input: cleanedHtml,\n      debut: motifs.debut,\n      fin: motifs.fin,\n    })\n\n    motifs.texte = articlePromptResult.texte\n  }\n\n  runLogFun(\n    console.info(\"Opération de peuplement de l'exposé des motifs terminée ✅\"),\n    { silent: SILENT_LOG },\n  )\n  delimiter(SILENT_LOG)\n\n  delete motifs.debut\n  delete motifs.fin\n  return motifs\n}\n\n/**\n * Asynchronously populates a projet loi object with the full text of each article and alinea in the given HTML.\n *\n * @param {Record<string, any>} partial - The partial projet loi object to be populated.\n * @param {string} html - The HTML containing the text of the articles and alineas.\n * @return {Promise<Record<string, any>>} A Promise that resolves to the fully populated projet loi object.\n */\nasync function populateProjetLoi(partial: Record<string, any>, html: string) {\n  runLogFun(console.group(\"Peuplement du projet loi...\"), {\n    silent: SILENT_LOG,\n  })\n  let cleanedHtml = cleanHtml(html)\n\n  const articles: Record<string, any>[] = partial.articles\n  const fullArticles: Record<string, any>[] = []\n\n  const systemTemplate = `\n    # Instructions :\n\n    Tu es un expert de la loi et de l'analyse de pages HTML.\n    Tu dois analyser un texte de loi afin d'en extraire certaines parties. \n    Le format d'entrée est du HTML. Le format de sortie est du texte.\n    Récupère l'intégralité du texte commençant par :\n\n    \"\"\"\n    {debut}\n    \"\"\"\n\n    Et finissant par :\n\n    \"\"\"\n    {fin}\n    \"\"\"\n  `\n\n  const prompt = ChatPromptTemplate.fromMessages([\n    [\"system\", systemTemplate],\n    [\"human\", \"{input}\"],\n  ])\n\n  const structuredModel = MODEL.withStructuredOutput(FULL_TEXTE_SCHEMA)\n  const chain = prompt.pipe(structuredModel)\n\n  for (const article of articles) {\n    if (article.debut === article.fin || !article.fin) {\n      article.texte = article.debut\n    } else {\n      const articlePromptResult: any = await chain.invoke({\n        input: cleanedHtml,\n        debut: article.debut,\n        fin: article.fin,\n      })\n\n      article.texte = articlePromptResult.texte\n    }\n\n    for (const alinea of article.alineas) {\n      if (alinea.debut === alinea.fin || !alinea.fin) {\n        alinea.texte = alinea.debut\n        continue\n      }\n\n      const alineaPromptResult: any = await chain.invoke({\n        input: article.texte,\n        debut: alinea.debut,\n        fin: alinea.fin,\n      })\n\n      delete alinea.debut\n      delete alinea.fin\n      alinea.texte = alineaPromptResult.texte\n    }\n\n    runLogFun(\n      console.info(\n        \"Opération de peuplement du contenu du projet de loi terminée ✅\",\n      ),\n      { silent: SILENT_LOG },\n    )\n    delete article.debut\n    delete article.fin\n    fullArticles.push(article)\n  }\n\n  const projetLoi = {\n    ...partial,\n    articles: fullArticles,\n  }\n\n  runLogFun(console.groupEnd(), { silent: SILENT_LOG })\n  delimiter()\n  return projetLoi\n}\n\n/**\n * Asynchronously extracts the \"expose motifs\" from the given HTML.\n *\n * @param {string} html - The HTML string from which to extract the \"expose motifs\".\n * @return {Promise<Record<string, any>>} - A Promise that resolves to the extracted \"expose motifs\" object.\n * The object contains the following properties:\n * - `debut`: The beginning of the \"expose motifs\".\n * - `fin`: The end of the \"expose motifs\".\n * @throws {Error} - If the extracted \"expose motifs\" do not validate against the EXPOSE_MOTIFS_SCHEMA.\n */\nasync function extractMotifs(html: string) {\n  runLogFun(console.group(\"Parsing expose motifs...\"), { silent: SILENT_LOG })\n  let cleanedHtml = cleanHtml(html)\n  cleanedHtml = removeHtmlTags(cleanedHtml)\n\n  const structuredModel = MODEL.withStructuredOutput(EXPOSE_MOTIFS_SCHEMA)\n\n  const motifsSystemTemplate = `\n    ${SYSTEM_TEMPLATE}\n\n    # Examples: \n\n    ## Exemple 1 User Input :\n    \"\"\"\n    {example1}\n    \"\"\"\n    \n    ## Exemple 1 JSON Output :\n    \"\"\"\n    {output1}\n    \"\"\"\n\n    ## Exemple 2 User Input :\n    \"\"\"\n    {example2}\n    \"\"\"\n    \n    ## Exemple 2 JSON Output :\n    \"\"\"\n    {output2}\n    \"\"\"\n    `\n\n  const prompt = ChatPromptTemplate.fromMessages([\n    [\"system\", motifsSystemTemplate],\n    [\"human\", \"{input}\"],\n  ])\n\n  const chain = prompt.pipe(structuredModel)\n  const motifsPromptResult: any = await chain.invoke({\n    input: cleanedHtml,\n    example1: EXAMPLES.PRJLANR5L16B0914.input,\n    example2: EXAMPLES.PRJLANR5L16B2424.input,\n    output1: EXAMPLES.PRJLANR5L16B0914.motifs,\n    output2: EXAMPLES.PRJLANR5L16B2424.motifs,\n  })\n\n  // console.log(JSON.stringify(motifsPromptResult))\n  runLogFun(\n    console.table([\n      [\"Début\", motifsPromptResult.debut.substring(0, 30)],\n      [\"Fin\", motifsPromptResult.fin.substring(0, 30)],\n    ]),\n    { silent: SILENT_LOG },\n  )\n\n  runLogFun(console.groupEnd(), { silent: SILENT_LOG })\n  delimiter()\n\n  const ajv = new Ajv()\n  if (ajv.validate(EXPOSE_MOTIFS_SCHEMA, motifsPromptResult)) {\n    runLogFun(console.info(\"Exposé des motifs OK 👌\"), { silent: SILENT_LOG })\n    return motifsPromptResult\n  }\n\n  runLogFun(console.warn(\"Exposé des motifs KO ❌\"), { silent: SILENT_LOG })\n  return {}\n}\n\n/**\n * Asynchronously extracts the metadata from the given HTML.\n *\n * @param {string} html - The HTML string from which to extract the metadata.\n * @return {Promise<Record<string, any>>} - A Promise that resolves to the extracted metadata object.\n * The object contains the following properties:\n * - `documentRef`: The reference of the document.\n * - `titre`: The title of the document.\n * - `numeroTexte`: The number of the text.\n * - `procedureAcceleree`: The accelerated procedure.\n * @throws {Error} - If the extracted metadata do not validate against the META_DONNEES_SCHEMA.\n */\nasync function extractMetaDonnees(html: string) {\n  delimiter()\n  let cleanedHtml = cleanHtml(html)\n  cleanedHtml = removeHtmlTags(cleanedHtml)\n\n  const structuredModel = MODEL.withStructuredOutput(META_DONNEES_SCHEMA)\n\n  const metaDonneesSystemTemplate = `\n    ${SYSTEM_TEMPLATE}\n    `\n\n  const prompt = ChatPromptTemplate.fromMessages([\n    [\"system\", metaDonneesSystemTemplate],\n    [\"human\", \"{input}\"],\n  ])\n\n  const chain = prompt.pipe(structuredModel)\n  const result: any = await chain.invoke({\n    input: cleanedHtml,\n    example1: EXAMPLES.PRJLANR5L16B0914.input,\n    example2: EXAMPLES.PRJLANR5L16B2424.input,\n    output1: EXAMPLES.PRJLANR5L16B0914.meta,\n    output2: EXAMPLES.PRJLANR5L16B2424.meta,\n  })\n\n  console.group(\"Meta-données\")\n  console.table([\n    [\"Référence\", result.documentRef],\n    [\"Titre\", result.titre],\n    [\"Numéro du texte\", result.numeroTexte],\n    [\"Procédure accéléré\", result.procedureAcceleree],\n  ])\n  console.groupEnd()\n  delimiter()\n\n  const ajv = new Ajv()\n  if (ajv.validate(META_DONNEES_SCHEMA, result)) {\n    runLogFun(console.warn(\"Meta-données OK 👌\"), { silent: SILENT_LOG })\n    return result\n  }\n\n  runLogFun(console.warn(\"Meta-données non-conforme ❌\"), {\n    silent: SILENT_LOG,\n  })\n  return {}\n}\n\n/**\n * Asynchronously extracts the \"projet de loi\" from the given HTML.\n *\n * @param {string} html - The HTML string from which to extract the \"projet de loi\".\n * @return {Promise<Record<string, any>>} - A Promise that resolves to the extracted \"projet de loi\" object.\n * The object contains the following properties:\n * - `articles`: An array of article objects, each containing the following properties:\n *   - `numeroArticle`: The number of the article.\n *   - `cardinal`: The cardinal of the article.\n *   - `titre`: The title of the article.\n *   - `debut`: The beginning of the article.\n *   - `fin`: The end of the article.\n *   - `alineas`: An array of alineas.\n * @throws {Error} - If the extracted \"projet de loi\" does not validate against the PROJET_LOI_SCHEMA.\n */\nasync function extractProjetLoi(html: string) {\n  runLogFun(console.group(\"Parsing loi...\"), { silent: SILENT_LOG })\n  let cleanedHtml = cleanHtml(html)\n  const structuredModel = MODEL.withStructuredOutput(PROJET_LOI_SCHEMA)\n\n  const projetLoiSystemTemplate = `\n    ${SYSTEM_TEMPLATE}\n\n    # Examples: \n\n    ## Exemple 1 User Input :\n    \"\"\"\n    {example1}\n    \"\"\"\n    \n    ## Exemple 1 JSON Output :\n    \"\"\"\n    {output1}\n    \"\"\"\n\n    ## Exemple 2 User Input :\n    \"\"\"\n    {example2}\n    \"\"\"\n    \n    ## Exemple 2 JSON Output :\n    \"\"\"\n    {output2}\n    \"\"\"\n    `\n\n  const prompt = ChatPromptTemplate.fromMessages([\n    [\"system\", projetLoiSystemTemplate],\n    [\"human\", \"{input}\"],\n  ])\n\n  const chain = prompt.pipe(structuredModel)\n  const articlesPromptResult: any = await chain.invoke({\n    input: cleanedHtml,\n    example1: EXAMPLES.PRJLANR5L16B0914.input,\n    example2: EXAMPLES.PRJLANR5L16B2424.input,\n    output1: EXAMPLES.PRJLANR5L16B0914.loi,\n    output2: EXAMPLES.PRJLANR5L16B2424.loi,\n  })\n\n  runLogFun(\n    console.log(\"Found \", articlesPromptResult.articles.length, \" articles.\"),\n    { silent: SILENT_LOG },\n  )\n\n  for (const article of articlesPromptResult.articles) {\n    article.alineas = await extractAlineas(html, article)\n  }\n\n  runLogFun(\n    console.table(\n      articlesPromptResult.articles.map((article: any) => [\n        article.numeroArticle,\n        article.titre,\n        article.debut.substring(0, 10),\n        article.fin.substring(0, 10),\n        `Nombre d'alineas : ${article.alineas.length}`,\n      ]),\n    ),\n    { silent: SILENT_LOG },\n  )\n  runLogFun(console.groupEnd(), { silent: SILENT_LOG })\n\n  const ajv = new Ajv()\n  if (ajv.validate(PROJET_LOI_SCHEMA, articlesPromptResult)) {\n    runLogFun(console.info(\"Projet de loi OK 👌\"), { silent: SILENT_LOG })\n    delimiter()\n    return articlesPromptResult\n  }\n\n  runLogFun(console.warn(\"Projet de loi KO ❌\"), { silent: SILENT_LOG })\n  delimiter()\n  return {}\n}\n\nasync function extractAlineas(html: string, article: any) {\n  runLogFun(console.group(\"Parsing alineas...\"), { silent: SILENT_LOG })\n  let cleanedHtml = cleanHtml(html)\n\n  const structuredModel = MODEL.withStructuredOutput(ALINEA_SCHEMA)\n\n  const alineasSystemTemplate = `\n    ${SYSTEM_TEMPLATE}\n\n    Extrais tous les alineas de la portion de texte commençant par « {debut} » et finissant par « {fin} ».\n    Un alinéa contiennent toujours une balise « <img ».\n\n    # Examples: \n\n    ## Exemple 1 User Input :\n    \"\"\"\n    {example1}\n    \"\"\"\n    \n    ## Exemple 1 JSON Output :\n    \"\"\"\n    {output1}\n    \"\"\"\n    `\n\n  const prompt = ChatPromptTemplate.fromMessages([\n    [\"system\", alineasSystemTemplate],\n    [\"human\", \"{input}\"],\n  ])\n\n  const chain = prompt.pipe(structuredModel)\n  const alineasPromptResult: any = await chain.invoke({\n    input: cleanedHtml,\n    debut: article.debut,\n    fin: article.fin,\n    example1: EXAMPLES.PRJLANR5L16B2424.input,\n    output1: EXAMPLES.PRJLANR5L16B2424.alineas,\n  })\n\n  const ajv = new Ajv()\n  if (ajv.validate(ALINEA_SCHEMA, alineasPromptResult as any)) {\n    runLogFun(\n      console.info(\n        \"Found \",\n        alineasPromptResult.alineas.length,\n        \" alineas for Article \",\n        article.numeroArticle,\n        article.titre,\n      ),\n      { silent: SILENT_LOG },\n    )\n    runLogFun(console.groupEnd(), { silent: SILENT_LOG })\n    delimiter()\n    return alineasPromptResult.alineas\n  }\n\n  runLogFun(console.warn(\"Alineas KO\"), { silent: SILENT_LOG })\n  delimiter()\n  return {}\n}\n\n/**\n * Asynchronously extracts the \"sommaire\" (outline) from the given HTML.\n *\n * @param {string} html - The HTML string from which to extract the \"sommaire\".\n * @return {Promise<Record<string, any>>} - A Promise that resolves to the extracted \"sommaire\" object.\n * The object contains the following properties:\n * - `sommaire`: An array of main division objects, each containing the following properties:\n *   - `numeroDivision`: The number of the main division.\n *   - `titre`: The title of the main division.\n *   - `typeDivision`: The type of content in the main division.\n *   - `sequence`: The sequence of the main division.\n *   - `parent`: The parent of the main division.\n * @throws {Error} - If the extracted \"sommaire\" does not validate against the SOMMAIRE_SCHEMA.\n */\nasync function extractSommaire(html: string) {\n  let cleanedHtml = cleanHtml(html)\n  cleanedHtml = removeHtmlTags(cleanedHtml)\n\n  const structuredModel = MODEL.withStructuredOutput(SOMMAIRE_SCHEMA)\n\n  const sommaireSystemTemplate = `\n    ${SYSTEM_TEMPLATE}\n\n    # Examples: \n\n    ## Exemple 1 User Input :\n    \"\"\"\n    {example1}\n    \"\"\"\n    \n    ## Exemple 1 JSON Output :\n    \"\"\"\n    {output1}\n    \"\"\"\n\n    ## Exemple 2 User Input :\n    \"\"\"\n    {example2}\n    \"\"\"\n    \n    ## Exemple 2 JSON Output :\n    \"\"\"\n    {output2}\n    \"\"\"\n    `\n\n  const prompt = ChatPromptTemplate.fromMessages([\n    [\"system\", sommaireSystemTemplate],\n    [\"human\", \"{input}\"],\n  ])\n\n  const chain = prompt.pipe(structuredModel)\n  const sommairePromptResult: any = await chain.invoke({\n    input: cleanedHtml,\n    example1: EXAMPLES.PRJLANR5L16B2014.input,\n    output1: EXAMPLES.PRJLANR5L16B2014.sommaire,\n    example2: EXAMPLES.PRJLANR5L16B0914.input,\n    output2: EXAMPLES.PRJLANR5L16B0914.sommaire,\n  })\n\n  if (!sommairePromptResult.sommaire) {\n    runLogFun(console.warn(\"Aucun sommaire n'a été trouvé ❓\"), {\n      silent: SILENT_LOG,\n    })\n    delimiter()\n    return { sommaire: [] }\n  }\n\n  runLogFun(console.group(\"Sommaire\"), { silent: SILENT_LOG })\n  runLogFun(\n    console.log(\n      \"Found \",\n      sommairePromptResult.sommaire.length,\n      \" main divisions (level 1).\",\n    ),\n    { silent: SILENT_LOG },\n  )\n  runLogFun(\n    console.table(\n      sommairePromptResult.sommaire.map((item: any) => [\n        item.sequence,\n        item.titre,\n        item.typeDivision,\n        item.parent,\n      ]),\n    ),\n    { silent: SILENT_LOG },\n  )\n  runLogFun(console.groupEnd(), { silent: SILENT_LOG })\n\n  const ajv = new Ajv()\n  if (ajv.validate(SOMMAIRE_SCHEMA, sommairePromptResult as any)) {\n    runLogFun(console.info(\"Sommaire OK 👌\"), { silent: SILENT_LOG })\n    delimiter()\n    return sommairePromptResult.sommaire\n  }\n\n  runLogFun(console.warn(\"La construction du sommaire est KO ❌\"), {\n    silent: SILENT_LOG,\n  })\n  delimiter()\n  return { sommaire: [] }\n}\n"],"mappings":"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","ignoreList":[]}