@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.
- package/CLAUDE.md +2 -2
- package/README.md +1 -1
- package/manifiestos/modulos.json +2 -1
- package/package.json +92 -92
- package/plugin.json +371 -371
- package/scripts/auditar-claudemd.js +44 -0
- package/scripts/lib/detector-autoduplicacion-intra-archivo.js +234 -0
- package/scripts/lib/reglas-globales-conocidas.json +73 -5
|
@@ -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.
|
|
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
|
|
52
|
-
"\\bnunca\\s+incluir\\s+[\"']?Co-Authored-By
|
|
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": {
|