@saulwade/swl-ses 1.7.1 → 1.7.2
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/CLAUDE.md +1 -1
- package/README.md +1 -1
- package/manifiestos/modulos.json +2 -1
- package/package.json +2 -2
- package/plugin.json +2 -2
- package/scripts/auditar-claudemd.js +44 -0
- package/scripts/lib/detector-autoduplicacion-intra-archivo.js +234 -0
package/CLAUDE.md
CHANGED
package/README.md
CHANGED
package/manifiestos/modulos.json
CHANGED
|
@@ -1218,11 +1218,12 @@
|
|
|
1218
1218
|
"opt_in": true
|
|
1219
1219
|
},
|
|
1220
1220
|
"claudemd-auditor": {
|
|
1221
|
-
"descripcion": "Auditor de calidad de CLAUDE.md según best practices Anthropic (ADR-0016). Detecta inflación (líneas excesivas, bullets gigantes, secciones canónicas ausentes, sin @references, placeholders). Usado por el comando /swl:claudemd y por el hook claudemd-bloat-detector. Zero-deps. Soporta --json y --strict. Umbrales configurables: SWL_CLAUDEMD_MAX_LINES (default 200), SWL_CLAUDEMD_MAX_BULLET_CHARS (default 1000).",
|
|
1221
|
+
"descripcion": "Auditor de calidad de CLAUDE.md según best practices Anthropic (ADR-0016). Detecta inflación (líneas excesivas, bullets gigantes, secciones canónicas ausentes, sin @references, placeholders). Dimensión 8 (v1.7.0): duplicación inline de reglas globales ~/.claude/rules/ vía catálogo declarativo. Dimensión 9 (v1.7.2): auto-duplicación intra-archivo (2+ secciones H2 citan la misma regla global con prosa). Usado por el comando /swl:claudemd y por el hook claudemd-bloat-detector. Zero-deps. Soporta --json y --strict. Umbrales configurables: SWL_CLAUDEMD_MAX_LINES (default 200), SWL_CLAUDEMD_MAX_BULLET_CHARS (default 1000).",
|
|
1222
1222
|
"tipo": "scripts",
|
|
1223
1223
|
"archivos": [
|
|
1224
1224
|
"scripts/auditar-claudemd.js",
|
|
1225
1225
|
"scripts/lib/detector-reglas-duplicadas.js",
|
|
1226
|
+
"scripts/lib/detector-autoduplicacion-intra-archivo.js",
|
|
1226
1227
|
"scripts/lib/reglas-globales-conocidas.json",
|
|
1227
1228
|
"docs/variables-entorno.md"
|
|
1228
1229
|
],
|
package/package.json
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@saulwade/swl-ses",
|
|
3
|
-
"version": "1.7.
|
|
4
|
-
"description": "Sistema de ingenieria de software auto-evolutivo multi-runtime polyglot con 61 agentes, 177 habilidades, 44 comandos, 71 reglas y 43 hooks. Soporta 11 lenguajes y 7 runtimes: Claude Code, OpenClaude, OpenCode, Gemini CLI, Cursor, Codex CLI (soporte completo); GitHub Copilot (soporte parcial). 100% en espanol (Mexico). Multi-target install (--target CSV / --all-runtimes), autoconfig MCP en Cursor/Codex con --with-mcp, agentes Codex en TOML, hooks Cursor (17 eventos) y Codex (6 eventos). Gateway bidireccional con relay Telegram y auditoria profunda Nemesis con loop evaluator-optimizer opt-in (ADR-0021) y 8 tools ejecutables. v1.7.
|
|
3
|
+
"version": "1.7.2",
|
|
4
|
+
"description": "Sistema de ingenieria de software auto-evolutivo multi-runtime polyglot con 61 agentes, 177 habilidades, 44 comandos, 71 reglas y 43 hooks. Soporta 11 lenguajes y 7 runtimes: Claude Code, OpenClaude, OpenCode, Gemini CLI, Cursor, Codex CLI (soporte completo); GitHub Copilot (soporte parcial). 100% en espanol (Mexico). Multi-target install (--target CSV / --all-runtimes), autoconfig MCP en Cursor/Codex con --with-mcp, agentes Codex en TOML, hooks Cursor (17 eventos) y Codex (6 eventos). Gateway bidireccional con relay Telegram y auditoria profunda Nemesis con loop evaluator-optimizer opt-in (ADR-0021) y 8 tools ejecutables. v1.7.2 introduce dimension 8 del auditor CLAUDE.md (deteccion de duplicacion de reglas globales) + hook PostToolUse claudemd-duplicacion-detector + regla sin-duplicacion-reglas-globales.md + catalogo declarativo reglas-globales-conocidas.json con 6 reglas (idioma, brevedad, git-coauthor, arreglar-al-detectar, debatir, context7).",
|
|
5
5
|
"bin": {
|
|
6
6
|
"swl-ses": "bin/swl-ses.js",
|
|
7
7
|
"swl-telegram-bot": "bin/swl-telegram-bot.js",
|
package/plugin.json
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "swl-ses",
|
|
3
|
-
"version": "1.7.
|
|
4
|
-
"description": "Sistema de ingenieria de software auto-evolutivo multi-runtime polyglot. 61 agentes, 177 habilidades, 44 comandos, 71 reglas y 43 hooks. 62 librerias. 11 lenguajes. Soporta Claude Code, Copilot, OpenCode, Codex y Gemini CLI. Loop evaluator-optimizer en /swl:nemesis (ADR-0021). v1.7.
|
|
3
|
+
"version": "1.7.2",
|
|
4
|
+
"description": "Sistema de ingenieria de software auto-evolutivo multi-runtime polyglot. 61 agentes, 177 habilidades, 44 comandos, 71 reglas y 43 hooks. 62 librerias. 11 lenguajes. Soporta Claude Code, Copilot, OpenCode, Codex y Gemini CLI. Loop evaluator-optimizer en /swl:nemesis (ADR-0021). v1.7.2 introduce dimension 8 del auditor CLAUDE.md (deteccion de duplicacion de reglas globales), hook PostToolUse claudemd-duplicacion-detector, regla sin-duplicacion-reglas-globales y catalogo declarativo de 6 reglas globales conocidas (idioma, brevedad, git-coauthor, arreglar-al-detectar, debatir, context7).",
|
|
5
5
|
"author": "Saul Wade Leon",
|
|
6
6
|
"license": "MIT",
|
|
7
7
|
"repository": "https://github.com/saul-wade/swl-ses",
|
|
@@ -40,6 +40,15 @@ function cargarDetectorReglasDuplicadas() {
|
|
|
40
40
|
}
|
|
41
41
|
}
|
|
42
42
|
|
|
43
|
+
// Dimensión 9 (v1.7.2): auto-duplicación intra-archivo. Lazy igual que dim 8.
|
|
44
|
+
function cargarDetectorAutoduplicacion() {
|
|
45
|
+
try {
|
|
46
|
+
return require('./lib/detector-autoduplicacion-intra-archivo').detectarAutoduplicacionIntraArchivo;
|
|
47
|
+
} catch (_) {
|
|
48
|
+
return null;
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
|
|
43
52
|
// ─── Config ───────────────────────────────────────────────────────────────
|
|
44
53
|
const MAX_LINES = parseInt(process.env.SWL_CLAUDEMD_MAX_LINES, 10) || 200;
|
|
45
54
|
const MAX_BULLET_CHARS =
|
|
@@ -188,6 +197,30 @@ function auditar(rutaClaudeMd) {
|
|
|
188
197
|
}
|
|
189
198
|
}
|
|
190
199
|
|
|
200
|
+
// 8. Dimensión 9 (v1.7.2): auto-duplicación intra-archivo. 2+ secciones H2
|
|
201
|
+
// del mismo CLAUDE.md citando la misma regla global con prosa (no tabla
|
|
202
|
+
// ni @-reference standalone) → boilerplate redundante.
|
|
203
|
+
let resultadoAutoduplicacion = { evaluado: false, autoduplicaciones: [], total_secciones: 0 };
|
|
204
|
+
const detectarAutoduplicacion = cargarDetectorAutoduplicacion();
|
|
205
|
+
if (detectarAutoduplicacion) {
|
|
206
|
+
try {
|
|
207
|
+
resultadoAutoduplicacion = detectarAutoduplicacion(contenido, {
|
|
208
|
+
esUserLevel: !esProjectLevel,
|
|
209
|
+
});
|
|
210
|
+
for (const dup of resultadoAutoduplicacion.autoduplicaciones) {
|
|
211
|
+
hallazgos.push({
|
|
212
|
+
severidad: dup.severidad || 'WARN',
|
|
213
|
+
regla: 'autoduplicacion-intra-archivo',
|
|
214
|
+
mensaje: `${dup.cantidad_secciones} secciones citan la misma regla global \`${dup.regla_global}\`: ${dup.secciones.map(s => `"${s.titulo}" (L${s.linea})`).join(', ')}`,
|
|
215
|
+
sugerencia: dup.remediacion,
|
|
216
|
+
referencia_canonica: dup.referencia_canonica,
|
|
217
|
+
});
|
|
218
|
+
}
|
|
219
|
+
} catch (e) {
|
|
220
|
+
// Falla silenciosa: no bloquea la auditoría.
|
|
221
|
+
}
|
|
222
|
+
}
|
|
223
|
+
|
|
191
224
|
// ─── Veredicto ───────────────────────────────────────────────────────────
|
|
192
225
|
const tieneError = hallazgos.some(h => h.severidad === 'ERROR');
|
|
193
226
|
const tieneWarn = hallazgos.some(h => h.severidad === 'WARN');
|
|
@@ -212,6 +245,12 @@ function auditar(rutaClaudeMd) {
|
|
|
212
245
|
detectadas: resultadoDuplicaciones.duplicaciones.length,
|
|
213
246
|
ids: resultadoDuplicaciones.duplicaciones.map(d => d.id),
|
|
214
247
|
},
|
|
248
|
+
autoduplicacion_intra_archivo: {
|
|
249
|
+
evaluado: resultadoAutoduplicacion.evaluado,
|
|
250
|
+
total_secciones: resultadoAutoduplicacion.total_secciones,
|
|
251
|
+
detectadas: resultadoAutoduplicacion.autoduplicaciones.length,
|
|
252
|
+
ids: resultadoAutoduplicacion.autoduplicaciones.map(d => d.id),
|
|
253
|
+
},
|
|
215
254
|
},
|
|
216
255
|
hallazgos,
|
|
217
256
|
};
|
|
@@ -343,6 +382,10 @@ function imprimirReporte(resultado) {
|
|
|
343
382
|
const d = m.duplicaciones_reglas_globales;
|
|
344
383
|
console.log(` - Duplicaciones de reglas globales: ${d.detectadas}/${d.total_reglas_evaluadas} reglas duplicadas${d.detectadas > 0 ? ' (' + d.ids.join(', ') + ')' : ''}`);
|
|
345
384
|
}
|
|
385
|
+
if (m.autoduplicacion_intra_archivo && m.autoduplicacion_intra_archivo.evaluado) {
|
|
386
|
+
const a = m.autoduplicacion_intra_archivo;
|
|
387
|
+
console.log(` - Auto-duplicación intra-archivo: ${a.detectadas} regla(s) citada(s) en 2+ secciones del mismo CLAUDE.md${a.detectadas > 0 ? ' (' + a.ids.join(', ') + ')' : ''}`);
|
|
388
|
+
}
|
|
346
389
|
console.log('');
|
|
347
390
|
}
|
|
348
391
|
|
|
@@ -400,4 +443,5 @@ module.exports = {
|
|
|
400
443
|
KARPATHY_MIN_LINES,
|
|
401
444
|
// Re-export lazy del detector (puede ser null si lib/ no está disponible):
|
|
402
445
|
cargarDetectorReglasDuplicadas,
|
|
446
|
+
cargarDetectorAutoduplicacion,
|
|
403
447
|
};
|
|
@@ -0,0 +1,234 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* scripts/lib/detector-autoduplicacion-intra-archivo.js
|
|
5
|
+
*
|
|
6
|
+
* Dimensión 9 del auditor CLAUDE.md (v1.7.2). Detecta auto-duplicación
|
|
7
|
+
* INTRA-archivo: cuando 2+ secciones H2 del mismo CLAUDE.md citan el
|
|
8
|
+
* mismo archivo de regla global de ~/.claude/rules/. Complementa la
|
|
9
|
+
* dimensión 8 (detector-reglas-duplicadas.js) que detecta duplicación
|
|
10
|
+
* INTER-archivo (paráfrasis de regla global).
|
|
11
|
+
*
|
|
12
|
+
* Caso de uso típico:
|
|
13
|
+
* ## Reglas obligatorias
|
|
14
|
+
* ...referencia a usar-sistema-swl.md...
|
|
15
|
+
*
|
|
16
|
+
* ## Flujo de trabajo
|
|
17
|
+
* ...referencia a usar-sistema-swl.md + matiz local...
|
|
18
|
+
*
|
|
19
|
+
* Diagnóstico: ambas secciones citan la misma regla. Una probablemente
|
|
20
|
+
* es boilerplate sin contenido propio; consolidar en UNA sola sección
|
|
21
|
+
* con referencia + matiz local (si lo hay).
|
|
22
|
+
*
|
|
23
|
+
* Excepciones (no cuenta como duplicación):
|
|
24
|
+
* - Citas dentro de tabla Markdown (catálogo organizado intencional):
|
|
25
|
+
* `| usar-sistema-swl.md | Siempre — matriz operacional |`
|
|
26
|
+
* - Archivos < 20 LOC.
|
|
27
|
+
* - User-level (~/.claude/CLAUDE.md): preferencias personales aceptadas.
|
|
28
|
+
*
|
|
29
|
+
* Public API:
|
|
30
|
+
* detectarAutoduplicacionIntraArchivo(contenido, opciones) → { autoduplicaciones }
|
|
31
|
+
* parsearSeccionesH2(contenido) → [{titulo, lineaInicio, contenido, fin}]
|
|
32
|
+
*
|
|
33
|
+
* Zero-dependency (solo fs/path).
|
|
34
|
+
*/
|
|
35
|
+
|
|
36
|
+
const fs = require('fs');
|
|
37
|
+
const path = require('path');
|
|
38
|
+
|
|
39
|
+
const RUTA_CATALOGO_DEFAULT = path.join(__dirname, 'reglas-globales-conocidas.json');
|
|
40
|
+
|
|
41
|
+
function cargarCatalogo(rutaJson = RUTA_CATALOGO_DEFAULT) {
|
|
42
|
+
if (!fs.existsSync(rutaJson)) {
|
|
43
|
+
throw new Error(`Catálogo no encontrado: ${rutaJson}`);
|
|
44
|
+
}
|
|
45
|
+
return JSON.parse(fs.readFileSync(rutaJson, 'utf8'));
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
/**
|
|
49
|
+
* Parsea el contenido en secciones H2 (encabezados `## `). Cada sección
|
|
50
|
+
* incluye todo el contenido hasta el siguiente H2 (o EOF), incluyendo
|
|
51
|
+
* sub-secciones H3/H4 anidadas.
|
|
52
|
+
*
|
|
53
|
+
* Sección virtual "preludio" si el archivo tiene contenido antes del
|
|
54
|
+
* primer H2 (típicamente H1 + descripción).
|
|
55
|
+
*
|
|
56
|
+
* @returns {Array} secciones con {titulo, lineaInicio, lineaFin, contenido}
|
|
57
|
+
*/
|
|
58
|
+
function parsearSeccionesH2(contenido) {
|
|
59
|
+
const lineas = contenido.split(/\r?\n/);
|
|
60
|
+
const secciones = [];
|
|
61
|
+
let actual = {
|
|
62
|
+
titulo: '__preludio__',
|
|
63
|
+
lineaInicio: 1,
|
|
64
|
+
inicio: 0,
|
|
65
|
+
fin: null,
|
|
66
|
+
contenido: '',
|
|
67
|
+
};
|
|
68
|
+
|
|
69
|
+
for (let i = 0; i < lineas.length; i++) {
|
|
70
|
+
const linea = lineas[i];
|
|
71
|
+
// H2 exacto (## seguido de espacio, NO ### ni más)
|
|
72
|
+
const matchH2 = linea.match(/^##\s+(?!#)(.+?)\s*$/);
|
|
73
|
+
if (matchH2) {
|
|
74
|
+
// cerrar la sección actual antes de abrir la nueva
|
|
75
|
+
actual.fin = i;
|
|
76
|
+
actual.contenido = lineas.slice(actual.inicio, actual.fin).join('\n');
|
|
77
|
+
if (actual.titulo !== '__preludio__' || actual.contenido.trim().length > 0) {
|
|
78
|
+
secciones.push(actual);
|
|
79
|
+
}
|
|
80
|
+
actual = {
|
|
81
|
+
titulo: matchH2[1].trim(),
|
|
82
|
+
lineaInicio: i + 1,
|
|
83
|
+
inicio: i,
|
|
84
|
+
fin: null,
|
|
85
|
+
contenido: '',
|
|
86
|
+
};
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
// cerrar última sección
|
|
90
|
+
actual.fin = lineas.length;
|
|
91
|
+
actual.contenido = lineas.slice(actual.inicio, actual.fin).join('\n');
|
|
92
|
+
if (actual.titulo !== '__preludio__' || actual.contenido.trim().length > 0) {
|
|
93
|
+
secciones.push(actual);
|
|
94
|
+
}
|
|
95
|
+
return secciones;
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
/**
|
|
99
|
+
* Verifica si TODAS las menciones del archivo de regla en una sección
|
|
100
|
+
* están dentro de patrones excluidos (no son prosa boilerplate). Si todas
|
|
101
|
+
* están excluidas, la sección NO cuenta como mención sustantiva.
|
|
102
|
+
*
|
|
103
|
+
* Patrones excluidos:
|
|
104
|
+
* 1. Mención dentro de tabla Markdown (línea empieza con `|`)
|
|
105
|
+
* → catálogo organizado intencional, no boilerplate
|
|
106
|
+
* 2. Mención como @-reference standalone (línea entera es `@path/X.md`)
|
|
107
|
+
* → @-include que carga el contenido de la regla, no descripción
|
|
108
|
+
*
|
|
109
|
+
* Una sección con todas sus menciones excluidas se considera "catálogo o
|
|
110
|
+
* include" y no entra en el conteo de auto-duplicación intra-archivo.
|
|
111
|
+
*/
|
|
112
|
+
function todasLasMencionesEstanExcluidas(seccionContenido, reglaArchivo) {
|
|
113
|
+
const lineas = seccionContenido.split(/\r?\n/);
|
|
114
|
+
let totalMenciones = 0;
|
|
115
|
+
let mencionesExcluidas = 0;
|
|
116
|
+
const reFilename = new RegExp('\\b' + escapeRegex(reglaArchivo), 'i');
|
|
117
|
+
|
|
118
|
+
for (const linea of lineas) {
|
|
119
|
+
if (!reFilename.test(linea)) continue;
|
|
120
|
+
totalMenciones++;
|
|
121
|
+
const trimmed = linea.trim();
|
|
122
|
+
// Excepción 1: línea de tabla Markdown
|
|
123
|
+
if (/^\|/.test(trimmed)) {
|
|
124
|
+
mencionesExcluidas++;
|
|
125
|
+
continue;
|
|
126
|
+
}
|
|
127
|
+
// Excepción 2: @-reference standalone (línea entera es @path/X.md)
|
|
128
|
+
if (/^@[\w./~\-]+\.md\s*$/.test(trimmed)) {
|
|
129
|
+
mencionesExcluidas++;
|
|
130
|
+
continue;
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
return totalMenciones > 0 && mencionesExcluidas === totalMenciones;
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
/**
|
|
137
|
+
* Detecta auto-duplicación intra-archivo.
|
|
138
|
+
*
|
|
139
|
+
* @param {string} contenido - CLAUDE.md completo
|
|
140
|
+
* @param {object} [opciones]
|
|
141
|
+
* @param {boolean} [opciones.skipUserLevel=true]
|
|
142
|
+
* @param {boolean} [opciones.esUserLevel=false]
|
|
143
|
+
* @param {object} [opciones.catalogo] - catálogo precargado (opcional)
|
|
144
|
+
* @returns {object} { evaluado, autoduplicaciones, total_secciones }
|
|
145
|
+
*/
|
|
146
|
+
function detectarAutoduplicacionIntraArchivo(contenido, opciones = {}) {
|
|
147
|
+
const skipUserLevel = opciones.skipUserLevel !== false;
|
|
148
|
+
const esUserLevel = opciones.esUserLevel === true;
|
|
149
|
+
|
|
150
|
+
if (skipUserLevel && esUserLevel) {
|
|
151
|
+
return {
|
|
152
|
+
evaluado: false,
|
|
153
|
+
razon: 'user-level — no evaluado',
|
|
154
|
+
autoduplicaciones: [],
|
|
155
|
+
total_secciones: 0,
|
|
156
|
+
};
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
const catalogo = opciones.catalogo || cargarCatalogo();
|
|
160
|
+
const minLineas = catalogo.configuracion?.min_lineas_archivo ?? 20;
|
|
161
|
+
const lineas = contenido.split(/\r?\n/);
|
|
162
|
+
|
|
163
|
+
if (lineas.length < minLineas) {
|
|
164
|
+
return {
|
|
165
|
+
evaluado: false,
|
|
166
|
+
razon: `archivo < ${minLineas} LOC — no acumula auto-duplicaciones`,
|
|
167
|
+
autoduplicaciones: [],
|
|
168
|
+
total_secciones: 0,
|
|
169
|
+
};
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
const secciones = parsearSeccionesH2(contenido);
|
|
173
|
+
|
|
174
|
+
if (secciones.length < 2) {
|
|
175
|
+
return {
|
|
176
|
+
evaluado: true,
|
|
177
|
+
autoduplicaciones: [],
|
|
178
|
+
total_secciones: secciones.length,
|
|
179
|
+
};
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
const autoduplicaciones = [];
|
|
183
|
+
|
|
184
|
+
for (const regla of catalogo.reglas) {
|
|
185
|
+
const reFilename = new RegExp('\\b' + escapeRegex(regla.regla_global), 'i');
|
|
186
|
+
const seccionesQueLaCitan = [];
|
|
187
|
+
|
|
188
|
+
for (const sec of secciones) {
|
|
189
|
+
if (!reFilename.test(sec.contenido)) continue;
|
|
190
|
+
// Excluir si TODAS las menciones en esta sección están en tabla
|
|
191
|
+
if (todasLasMencionesEstanExcluidas(sec.contenido, regla.regla_global)) continue;
|
|
192
|
+
seccionesQueLaCitan.push({
|
|
193
|
+
titulo: sec.titulo,
|
|
194
|
+
linea: sec.lineaInicio,
|
|
195
|
+
});
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
if (seccionesQueLaCitan.length >= 2) {
|
|
199
|
+
autoduplicaciones.push({
|
|
200
|
+
id: regla.id,
|
|
201
|
+
regla_global: regla.regla_global,
|
|
202
|
+
referencia_canonica: regla.referencia_canonica,
|
|
203
|
+
secciones: seccionesQueLaCitan,
|
|
204
|
+
cantidad_secciones: seccionesQueLaCitan.length,
|
|
205
|
+
remediacion:
|
|
206
|
+
`${seccionesQueLaCitan.length} secciones del mismo CLAUDE.md citan ` +
|
|
207
|
+
`\`${regla.regla_global}\`: ${seccionesQueLaCitan.map(s => `"${s.titulo}" (L${s.linea})`).join(', ')}. ` +
|
|
208
|
+
`Consolidar en UNA sola sección con referencia canónica + matiz local ` +
|
|
209
|
+
`(si lo hay). Si ninguna sección agrega matiz operacional propio sobre ` +
|
|
210
|
+
`la regla global, eliminar todas menos una.`,
|
|
211
|
+
severidad: catalogo.configuracion?.severidad_default || 'WARN',
|
|
212
|
+
});
|
|
213
|
+
}
|
|
214
|
+
}
|
|
215
|
+
|
|
216
|
+
return {
|
|
217
|
+
evaluado: true,
|
|
218
|
+
autoduplicaciones,
|
|
219
|
+
total_secciones: secciones.length,
|
|
220
|
+
};
|
|
221
|
+
}
|
|
222
|
+
|
|
223
|
+
function escapeRegex(s) {
|
|
224
|
+
return s.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
|
|
225
|
+
}
|
|
226
|
+
|
|
227
|
+
module.exports = {
|
|
228
|
+
cargarCatalogo,
|
|
229
|
+
parsearSeccionesH2,
|
|
230
|
+
detectarAutoduplicacionIntraArchivo,
|
|
231
|
+
// exports privados para tests
|
|
232
|
+
_todasLasMencionesEstanExcluidas: todasLasMencionesEstanExcluidas,
|
|
233
|
+
RUTA_CATALOGO_DEFAULT,
|
|
234
|
+
};
|