@saulwade/swl-ses 1.7.1 → 1.7.3
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 +2 -2
- package/README.md +2 -2
- package/habilidades/calidad-anti-patrones-universales/SKILL.md +368 -0
- package/habilidades/estilo-sin-ai-isms/SKILL.md +52 -1
- package/manifiestos/modulos.json +1351 -1349
- package/manifiestos/skills-lock.json +13 -6
- package/package.json +2 -2
- package/plugin.json +3 -2
- package/scripts/auditar-claudemd.js +44 -0
- package/scripts/lib/detector-autoduplicacion-intra-archivo.js +234 -0
- package/scripts/lib/pr-analyzer.js +399 -0
|
@@ -1,8 +1,8 @@
|
|
|
1
1
|
{
|
|
2
2
|
"lockfileVersion": 1,
|
|
3
|
-
"generatedAt": "2026-05-
|
|
4
|
-
"skillsCount":
|
|
5
|
-
"lockHash": "sha256:
|
|
3
|
+
"generatedAt": "2026-05-24T23:17:38.255Z",
|
|
4
|
+
"skillsCount": 178,
|
|
5
|
+
"lockHash": "sha256:1c37732adb57b8daf54b93978d2559419f65e73a7e85cc83af449408c9f7ec44",
|
|
6
6
|
"skills": [
|
|
7
7
|
{
|
|
8
8
|
"nombre": "accesibilidad-a11y",
|
|
@@ -242,6 +242,13 @@
|
|
|
242
242
|
"bytes": 13490,
|
|
243
243
|
"version": "\"1.0.0\""
|
|
244
244
|
},
|
|
245
|
+
{
|
|
246
|
+
"nombre": "calidad-anti-patrones-universales",
|
|
247
|
+
"path": "habilidades/calidad-anti-patrones-universales/SKILL.md",
|
|
248
|
+
"hash": "sha256:5047fc360759b8d894b3e5c20a344cae46562c8131267c5a59f9d8f8839ef178",
|
|
249
|
+
"bytes": 11840,
|
|
250
|
+
"version": "\"1.0.0\""
|
|
251
|
+
},
|
|
245
252
|
{
|
|
246
253
|
"nombre": "changelog-generator",
|
|
247
254
|
"path": "habilidades/changelog-generator/SKILL.md",
|
|
@@ -455,9 +462,9 @@
|
|
|
455
462
|
{
|
|
456
463
|
"nombre": "estilo-sin-ai-isms",
|
|
457
464
|
"path": "habilidades/estilo-sin-ai-isms/SKILL.md",
|
|
458
|
-
"hash": "sha256:
|
|
459
|
-
"bytes":
|
|
460
|
-
"version": "\"1.1.
|
|
465
|
+
"hash": "sha256:f3dc9dfafa65029fb70ebf9d6887832f71415853293c3d1e82716ce5fc447884",
|
|
466
|
+
"bytes": 39938,
|
|
467
|
+
"version": "\"1.1.1\""
|
|
461
468
|
},
|
|
462
469
|
{
|
|
463
470
|
"nombre": "estructura-proyecto-claude",
|
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,
|
|
3
|
+
"version": "1.7.3",
|
|
4
|
+
"description": "Sistema de ingenieria de software auto-evolutivo multi-runtime polyglot con 61 agentes, 178 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 agrega skill calidad-anti-patrones-universales (catalogo de 10 anti-patrones nombrados) + utility scripts/lib/pr-analyzer.js (analizador zero-deps de PRs) + 3 sub-secciones nuevas en estilo-sin-ai-isms (agencia falsa, narrador-distancia, aperturas Wh-).",
|
|
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,
|
|
3
|
+
"version": "1.7.3",
|
|
4
|
+
"description": "Sistema de ingenieria de software auto-evolutivo multi-runtime polyglot. 61 agentes, 178 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 agrega skill calidad-anti-patrones-universales (catalogo de 10 anti-patrones nombrados) + utility scripts/lib/pr-analyzer.js (analizador zero-deps de PRs) + 3 sub-secciones nuevas en estilo-sin-ai-isms (agencia falsa, narrador-distancia, aperturas Wh-).",
|
|
5
5
|
"author": "Saul Wade Leon",
|
|
6
6
|
"license": "MIT",
|
|
7
7
|
"repository": "https://github.com/saul-wade/swl-ses",
|
|
@@ -40,6 +40,7 @@
|
|
|
40
40
|
"habilidades/build-errors-rust",
|
|
41
41
|
"habilidades/build-errors-swift",
|
|
42
42
|
"habilidades/build-errors-typescript",
|
|
43
|
+
"habilidades/calidad-anti-patrones-universales",
|
|
43
44
|
"habilidades/changelog-generator",
|
|
44
45
|
"habilidades/checklist-calidad",
|
|
45
46
|
"habilidades/checklist-seguridad",
|
|
@@ -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
|
+
};
|