@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 CHANGED
@@ -1,4 +1,4 @@
1
- # CLAUDE.md — @saulwade/swl-ses v1.7.1
1
+ # CLAUDE.md — @saulwade/swl-ses v1.7.2
2
2
 
3
3
  ## Reglas de máxima prioridad (aplican SIEMPRE, sin excepción)
4
4
 
package/README.md CHANGED
@@ -1,4 +1,4 @@
1
- # swl-ses v1.7.1
1
+ # swl-ses v1.7.2
2
2
 
3
3
  > El paquete anterior `@saulwadeleon/swl-software-engineering-system` está deprecado. Migrar a `@saulwade/swl-ses` (npmjs.org canónico) o `@saul-wade/swl-ses` (mirror en GitHub Packages) — el CLI `swl-ses` no cambia.
4
4
 
@@ -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.1",
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.1 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).",
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.1",
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.1 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).",
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
+ };