@saulwade/swl-ses 1.7.0 → 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.
@@ -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
+ };
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "$schema-version": "1.0.0",
3
3
  "description": "Catálogo de reglas globales ~/.claude/rules/ que NO deben duplicarse inline en CLAUDE.md de proyectos. Consumido por scripts/lib/detector-reglas-duplicadas.js y por scripts/auditar-claudemd.js dimensión 'duplicacion-reglas-globales'. Cada entrada lista: nombre canónico, archivo origen, sección canónica para referenciar, patrones de detección (regex), y reemplazo sugerido.",
4
- "version": "1.0.0",
4
+ "version": "1.1.0",
5
5
  "actualizado": "2026-05-22",
6
6
  "reglas": [
7
7
  {
@@ -17,7 +17,7 @@
17
17
  "\\bgram[aá]tica\\s+normativa\\b",
18
18
  "\\bevitar\\s+anglicismos\\b",
19
19
  "^\\s*#+\\s*Language\\s*$",
20
- "\\bdomain\\s+terms\\b.*\\bin\\s+\\*?\\*?Spanish\\*?\\*?",
20
+ "\\bdomain\\s+terms\\b.*\\bin\\s+\\*?\\*?Spanish\\b",
21
21
  "\\bmensajes?\\s+de\\s+commit.*siempre\\s+en\\s+español\\b"
22
22
  ],
23
23
  "min_matches": 2,
@@ -48,9 +48,10 @@
48
48
  "descripcion": "Los commits NO deben incluir Co-Authored-By de Claude/IA. Ya cubierto globalmente.",
49
49
  "patrones": [
50
50
  "\\bSin\\s+co-?autores\\s+en\\s+commits\\b",
51
- "\\bNO\\s+incluir\\s+Co-Authored-By\\b",
52
- "\\bnunca\\s+incluir\\s+[\"']?Co-Authored-By:\\s+Claude",
53
- "\\bsin\\s+atribuci[oó]n\\s+a\\s+Claude\\b"
51
+ "\\bNO\\s+incluir\\s+\\W?Co-?Authored-?By\\b",
52
+ "\\bnunca\\s+incluir\\s+[\"'`]?Co-Authored-By:?\\s+Claude",
53
+ "\\bsin\\s+atribuci[oó]n\\s+a\\s+Claude\\b",
54
+ "\\bcommits?\\s+son\\s+del\\s+desarrollador\\b"
54
55
  ],
55
56
  "min_matches": 1,
56
57
  "scope": "project-level",
@@ -100,6 +101,73 @@
100
101
  "min_matches": 1,
101
102
  "scope": "project-level",
102
103
  "remediacion_sugerida": "Eliminar el bloque local. La regla global aplica."
104
+ },
105
+ {
106
+ "id": "usar-sistema-swl",
107
+ "regla_global": "usar-sistema-swl.md",
108
+ "seccion_canonica": "Uso obligatorio del sistema SWL",
109
+ "referencia_canonica": "@~/.claude/rules/usar-sistema-swl.md",
110
+ "descripcion": "Uso obligatorio del sistema SWL: invocar agentes/skills/comandos especializados en lugar de trabajo directo con Read/Write/Edit. Ya cubierto globalmente.",
111
+ "patrones": [
112
+ "\\bSIEMPRE\\s+usar\\s+el\\s+sistema\\s+SWL\\b",
113
+ "\\busar\\s+el\\s+sistema\\s+SWL\\s+completo\\b",
114
+ "\\bNO\\s+hacer\\s+trabajo\\s+directo\\s+que\\s+un\\s+agente\\s+SWL\\b",
115
+ "\\bagentes?\\s+especializados?\\b.*\\borquestador-swl\\b",
116
+ "\\binvocar\\s+`?orquestador-swl`?\\b.*\\bpunto\\s+de\\s+entrada\\b",
117
+ "\\bFlujo\\s+de\\s+trabajo\\s+obligatorio\\s*[—\\-]\\s*Sistema\\s+SWL\\b"
118
+ ],
119
+ "min_matches": 2,
120
+ "scope": "project-level",
121
+ "remediacion_sugerida": "Eliminar el bloque local de flujo de trabajo SWL. La regla global `~/.claude/rules/usar-sistema-swl.md` cubre el principio y la matriz operacional completa (qué componente usar para cada tipo de tarea, excepciones legítimas, anti-patrones). Si el proyecto tiene matices propios (ej: agentes específicos del dominio, ejemplos de invocaciones validadas, casos edge), conservar SOLO esos matices con frase corta + referencia: \"Ver @~/.claude/rules/usar-sistema-swl.md. En este proyecto, además: <matiz>\"."
122
+ },
123
+ {
124
+ "id": "analisis-previo-tareas-grandes",
125
+ "regla_global": "analisis-previo-tareas-grandes.md",
126
+ "seccion_canonica": "Análisis previo ante tareas grandes",
127
+ "referencia_canonica": "@~/.claude/rules/analisis-previo-tareas-grandes.md",
128
+ "descripcion": "Tareas grandes (>10 archivos, >500 LOC, cross-módulo, replicar/portar todo) requieren tabla comparativa + 3 opciones (mínima/media/completa) + recomendación + confirmación. Ya cubierto globalmente.",
129
+ "patrones": [
130
+ "\\btabla\\s+comparativa\\b.*\\bgap\\b",
131
+ "\\btres\\s+opciones\\s+de\\s+alcance\\b",
132
+ "\\b(mínima|minima)\\s*[\\/\\|]\\s*media\\s*[\\/\\|]\\s*completa\\b",
133
+ "\\breplicar\\s+íntegramente\\b.*\\banálisis\\s+comparativo\\b"
134
+ ],
135
+ "min_matches": 2,
136
+ "scope": "project-level",
137
+ "remediacion_sugerida": "Eliminar el bloque local. La regla global cubre el protocolo de auditoría previa + 3 opciones + recomendación + confirmación. Si el proyecto tiene umbrales propios (ej: >5 archivos en lugar de >10), conservar solo el umbral con referencia."
138
+ },
139
+ {
140
+ "id": "registro-componentes-nuevos",
141
+ "regla_global": "registro-componentes-nuevos.md",
142
+ "seccion_canonica": "Registro obligatorio de componentes nuevos en manifiestos",
143
+ "referencia_canonica": "@~/.claude/rules/registro-componentes-nuevos.md",
144
+ "descripcion": "Todo componente nuevo (agente/skill/comando/hook/regla) DEBE registrarse en manifiestos + plugin.json + INVENTARIO en el mismo commit. Ya cubierto globalmente.",
145
+ "patrones": [
146
+ "\\btodo\\s+componente\\s+nuevo\\b.*\\bregistr(?:ar(?:se)?|o|ado)\\b",
147
+ "\\bregistro\\s+obligatorio\\s+(de\\s+)?componentes?\\s+nuevos\\b",
148
+ "\\bmanifiestos?\\s*\\/\\s*modulos\\.json\\b.*\\bplugin\\.json\\b",
149
+ "\\bplugin\\.json\\b.*\\bmanifiestos?\\s*\\/\\s*modulos\\.json\\b",
150
+ "\\bregistro\\s+obligatorio\\s+en\\s+manifiestos\\b",
151
+ "\\ben\\s+el\\s+mismo\\s+commit\\b.*\\bmanifiestos?\\b"
152
+ ],
153
+ "min_matches": 2,
154
+ "scope": "project-level",
155
+ "remediacion_sugerida": "Eliminar el bloque local. La regla global `~/.claude/rules/registro-componentes-nuevos.md` cubre la tabla de propagación obligatoria (qué archivos tocar por tipo de componente) y los gates de CI que la enforzan. Si el proyecto tiene componentes propios además de los del sistema, conservar SOLO esos matices."
156
+ },
157
+ {
158
+ "id": "consultar-vault-primero",
159
+ "regla_global": "consultar-vault-primero.md",
160
+ "seccion_canonica": "Consultar el vault Obsidian antes de leer múltiples archivos",
161
+ "referencia_canonica": "@~/.claude/rules/consultar-vault-primero.md",
162
+ "descripcion": "Antes de leer 3+ archivos del codebase para construir contexto general, consultar el vault Obsidian con obsidian_simple_search. Ya cubierto globalmente.",
163
+ "patrones": [
164
+ "\\bconsultar\\s+el\\s+vault\\s+Obsidian\\s+primero\\b",
165
+ "\\bantes\\s+de\\s+leer\\s+3\\+?\\s+archivos\\b.*\\bvault\\b",
166
+ "\\bobsidian_simple_search\\b.*\\bantes\\b"
167
+ ],
168
+ "min_matches": 1,
169
+ "scope": "project-level",
170
+ "remediacion_sugerida": "Eliminar el bloque local. La regla global aplica a cualquier proyecto con vault Obsidian disponible."
103
171
  }
104
172
  ],
105
173
  "configuracion": {