@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.
- package/lib/bugs/acteur-00010/plugin.test.js +2 -2
- package/lib/bugs/acteur-00010.js +3 -3
- package/lib/bugs/agenda-00002/plugin.test.js +3 -3
- package/lib/bugs/agenda-00002.js +5 -5
- package/lib/bugs/agenda-00008/plugin.test.js +2 -2
- package/lib/bugs/agenda-00008.js +3 -3
- package/lib/bugs/agenda-00011/plugin.test.js +6 -6
- package/lib/bugs/agenda-00011.js +8 -8
- package/lib/bugs.js +11 -11
- package/lib/cleaners/index.d.ts +1 -0
- package/lib/cleaners/index.js +8 -1
- package/lib/cleaners/index.mjs +2 -1
- package/lib/cleaners/scrutins.js +2 -1
- package/lib/cleaners/scrutins.mjs +3 -2
- package/lib/cleaners/textes_loi.d.ts +1 -0
- package/lib/cleaners/textes_loi.js +12 -0
- package/lib/cleaners/textes_loi.mjs +5 -0
- package/lib/cleaners/xml.d.ts +3 -0
- package/lib/cleaners/xml.js +13 -1
- package/lib/cleaners/xml.mjs +11 -1
- package/lib/datasets.d.ts +9 -2
- package/lib/datasets.js +24 -3
- package/lib/datasets.mjs +24 -3
- package/lib/dates.js +3 -3
- package/lib/examples/PIONANR5L15B0020/input.d.ts +2 -0
- package/lib/examples/PIONANR5L15B0020/input.js +10 -0
- package/lib/examples/PIONANR5L15B0020/input.mjs +193 -0
- package/lib/examples/PRJLANR5L16B0914/input.d.ts +2 -0
- package/lib/examples/PRJLANR5L16B0914/input.js +10 -0
- package/lib/examples/PRJLANR5L16B0914/input.mjs +322 -0
- package/lib/examples/PRJLANR5L16B2014/input.d.ts +2 -0
- package/lib/examples/PRJLANR5L16B2014/input.js +10 -0
- package/lib/examples/PRJLANR5L16B2014/input.mjs +577 -0
- package/lib/examples/PRJLANR5L16B2424/input.d.ts +2 -0
- package/lib/examples/PRJLANR5L16B2424/input.js +10 -0
- package/lib/examples/PRJLANR5L16B2424/input.mjs +322 -0
- package/lib/examples/PRJLANR5L16B2462/input.d.ts +2 -0
- package/lib/examples/PRJLANR5L16B2462/input.js +10 -0
- package/lib/examples/PRJLANR5L16B2462/input.mjs +1308 -0
- package/lib/loaders.d.ts +5 -0
- package/lib/loaders.js +308 -262
- package/lib/loaders.mjs +16 -3
- package/lib/logger.d.ts +16 -0
- package/lib/logger.js +35 -0
- package/lib/logger.mjs +26 -0
- package/lib/parsers/documents.js +2 -2
- package/lib/parsers/textes_lois.d.ts +16 -1
- package/lib/parsers/textes_lois.js +733 -28
- package/lib/parsers/textes_lois.mjs +587 -22
- package/lib/raw_types/acteurs_et_organes.js +1 -1
- package/lib/raw_types/agendas.js +1 -1
- package/lib/raw_types/amendements.js +1 -1
- package/lib/raw_types/debats.js +1 -1
- package/lib/raw_types/dossiers_legislatifs.js +1 -1
- package/lib/raw_types/questions.js +1 -1
- package/lib/raw_types/scrutins.d.ts +18 -13
- package/lib/raw_types/scrutins.js +28 -14
- package/lib/raw_types/scrutins.mjs +28 -14
- package/lib/raw_types/textes_loi.d.ts +93 -0
- package/lib/raw_types/textes_loi.js +429 -0
- package/lib/raw_types/textes_loi.mjs +378 -0
- package/lib/scripts/bugs_helper.js +9 -9
- package/lib/scripts/document_dossiers_legislatifs.js +1 -1
- package/lib/scripts/process_open_dataset.d.ts +17 -0
- package/lib/scripts/process_open_dataset.js +302 -0
- package/lib/scripts/process_open_dataset.mjs +136 -0
- package/lib/scripts/process_textes_loi_dataset.d.ts +7 -0
- package/lib/scripts/process_textes_loi_dataset.js +374 -0
- package/lib/scripts/process_textes_loi_dataset.mjs +237 -0
- package/lib/scripts/raw_types_from_amendements.js +2 -2
- package/lib/scripts/reorganize_data.js +1 -1
- package/lib/scripts/reorganize_data.mjs +1 -1
- package/lib/scripts/retrieve_deputes_photos.js +3 -3
- package/lib/scripts/retrieve_documents.js +4 -4
- package/lib/scripts/retrieve_open_data.js +41 -279
- package/lib/scripts/retrieve_open_data.mjs +13 -126
- package/lib/scripts/retrieve_pending_amendments.js +2 -2
- package/lib/scripts/retrieve_senateurs_photos.js +2 -2
- package/lib/scripts/retrieve_textes_lois.js +2 -2
- package/lib/scripts/shared/cli_helpers.d.ts +5 -0
- package/lib/scripts/shared/cli_helpers.js +7 -2
- package/lib/scripts/shared/cli_helpers.mjs +6 -1
- package/lib/scripts/test_iter_load.js +25 -10
- package/lib/scripts/test_iter_load.mjs +19 -11
- package/lib/scripts/validate_json.js +3 -3
- package/lib/types/acteurs_et_organes.js +1 -1
- package/lib/types/agendas.js +1 -1
- package/lib/types/amendements.js +1 -1
- package/lib/types/debats.js +1 -1
- package/lib/types/dossiers_legislatifs.js +1 -1
- package/lib/types/questions.js +1 -1
- package/lib/types/scrutins.d.ts +6 -3
- package/lib/types/scrutins.js +13 -7
- package/lib/types/scrutins.mjs +13 -7
- package/lib/types/textes_loi.d.ts +94 -0
- package/lib/types/textes_loi.js +429 -0
- package/lib/types/textes_loi.mjs +378 -0
- package/package.json +13 -12
- package/lib/scripts/retrieve_textes_lois_llm.d.ts +0 -1
- package/lib/scripts/retrieve_textes_lois_llm.js +0 -188
- 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
|
-
|
|
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
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
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
|
-
|
|
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
|
-
|
|
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
|
-
|
|
24
|
-
|
|
25
|
-
|
|
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
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
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(
|
|
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
|
-
|
|
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":[]}
|