@saulwade/swl-ses 1.4.0 → 1.4.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 +4 -3
- package/README.md +15 -14
- package/agentes/nemesis-auditor-swl.md +161 -0
- package/bin/swl-mcp-server.js +187 -187
- package/comandos/swl/.evolved.json +22 -22
- package/comandos/swl/contribuir.md +233 -233
- package/comandos/swl/nemesis.md +122 -0
- package/comandos/swl/salud.md +34 -0
- package/comandos/swl/verificar.md +45 -0
- package/gateway/lib/event-channel.js +191 -191
- package/habilidades/backend-production-resilience/SKILL.md +288 -288
- package/habilidades/benchmark-memoria/SKILL.md +186 -186
- package/habilidades/diagrama-arquitectura/assets/template.html +276 -276
- package/habilidades/doubt-driven-review/SKILL.md +171 -171
- package/habilidades/doubt-driven-review/recursos/EXAMPLES.md +130 -130
- package/habilidades/eval-framework/SKILL.md +212 -212
- package/habilidades/feynman-auditor-swl/SKILL.md +123 -0
- package/habilidades/feynman-auditor-swl/recursos/preguntas-language-agnostic.md +108 -0
- package/habilidades/harness-claude-code/SKILL.md +299 -299
- package/habilidades/infra-github-actions/SKILL.md +166 -166
- package/habilidades/legacy-code-rescue/SKILL.md +267 -267
- package/habilidades/manejo-errores/.evolved.json +8 -8
- package/habilidades/meta-skills-estandar/recursos/convencion-examples.md +93 -93
- package/habilidades/meta-skills-estandar/recursos/skills-as-agents.md +163 -163
- package/habilidades/patrones-python/SKILL.md +229 -229
- package/habilidades/patrones-python/recursos/patrones-avanzados.md +469 -469
- package/habilidades/planear-fase/SKILL.md +319 -319
- package/habilidades/release-semver/.evolved.json +8 -8
- package/habilidades/state-inconsistency-auditor-swl/SKILL.md +166 -0
- package/habilidades/state-inconsistency-auditor-swl/recursos/coupled-state-patterns.md +147 -0
- package/habilidades/testing-python/SKILL.md +340 -340
- package/habilidades/web-fetcher-routing/SKILL.md +75 -0
- package/hooks/claudemd-bloat-detector.js +161 -161
- package/hooks/lib/agent-routing.js +107 -107
- package/hooks/lib/auto-consolidator.js +335 -335
- package/hooks/lib/error-classifier.js +308 -308
- package/hooks/lib/merkle-audit.js +96 -96
- package/hooks/lib/provenance-tracker.js +191 -191
- package/hooks/lib/rate-limit-tracker.js +253 -253
- package/hooks/lib/resource-quota.js +122 -122
- package/hooks/lib/retry-jitter.js +165 -165
- package/hooks/lib/security-net.js +201 -0
- package/hooks/lib/skill-auditor.js +588 -588
- package/hooks/lib/sync-status.js +228 -228
- package/hooks/lib/taint-tracker.js +107 -107
- package/hooks/lib/text-similarity.js +241 -241
- package/hooks/lib/toon-compressor.js +245 -245
- package/hooks/registro-turnos.js +209 -209
- package/hooks/sugerir-regenerar-inventario.js +170 -170
- package/hooks/validar-formato-post-subagente.js +140 -140
- package/hooks/validar-memoria-hook.js +218 -218
- package/instintos/prompt-appendices.yaml +57 -57
- package/manifiestos/agent-output-schemas.json +57 -57
- package/manifiestos/modulos.json +41 -6
- package/manifiestos/perfiles.json +2 -1
- package/manifiestos/skills-lock.json +30 -9
- package/package.json +2 -2
- package/plantillas/auditor-veto-template.md +105 -105
- package/plantillas/github-workflows/README.md +47 -47
- package/plantillas/github-workflows/release-please.yml +44 -44
- package/plantillas/github-workflows/swl-ci.yml +107 -107
- package/plantillas/github-workflows/swl-security.yml +51 -51
- package/plugin.json +10 -2
- package/reglas/analisis-previo-tareas-grandes.md +172 -172
- package/reglas/arreglar-al-detectar.md +147 -147
- package/reglas/fragmentos-compartidos.md +152 -152
- package/reglas/harness-claude-code.md +213 -213
- package/reglas/usar-context7.md +226 -226
- package/schemas/diary-entry.schema.json +80 -80
- package/scripts/audit-tools/audit-history.js +330 -0
- package/scripts/audit-tools/bundle-tracker.js +290 -0
- package/scripts/audit-tools/canary-monitor.js +352 -0
- package/scripts/audit-tools/code-profiler.js +605 -0
- package/scripts/audit-tools/dep-doctor.js +320 -0
- package/scripts/audit-tools/env-validator.js +206 -0
- package/scripts/audit-tools/lib/fs-walk.js +48 -0
- package/scripts/audit-tools/lib/output.js +23 -0
- package/scripts/audit-tools/migration-checker.js +392 -0
- package/scripts/audit-tools/pentest-scanner.js +1436 -0
- package/scripts/benchmark-memoria.js +167 -167
- package/scripts/configurar-branch-protection.js +418 -418
- package/scripts/detectar-aprendizajes-duplicados.js +151 -151
- package/scripts/field-report.js +199 -199
- package/scripts/generar-checklists-consolidados.js +273 -273
- package/scripts/generar-inventario.js +420 -420
- package/scripts/generar-matriz-lenguajes.js +271 -271
- package/scripts/lib/artefactos-python.js +43 -43
- package/scripts/lib/benchmark-metrics.js +160 -160
- package/scripts/lib/budget-enforcer.js +252 -252
- package/scripts/lib/configurar-ci.js +380 -380
- package/scripts/lib/contadores-inventario.js +217 -217
- package/scripts/lib/detectar-stack-detallado.js +307 -307
- package/scripts/lib/diary-entry.js +234 -234
- package/scripts/lib/eval-metrics-store.js +218 -218
- package/scripts/lib/eval-quality.js +171 -171
- package/scripts/lib/eval-schemas.js +144 -144
- package/scripts/lib/eval-self-correct.js +106 -106
- package/scripts/lib/eval-validator.js +185 -185
- package/scripts/lib/jaccard-similarity.js +98 -98
- package/scripts/lib/longmemeval-runner.js +125 -125
- package/scripts/lib/manifiestos.js +42 -1
- package/scripts/lib/npm-version.js +261 -261
- package/scripts/lib/paquetes-conocidos.js +50 -50
- package/scripts/lib/prompt-builder.js +264 -264
- package/scripts/lib/rrf-fusion.js +175 -175
- package/scripts/lib/scoring-instintos.js +277 -277
- package/scripts/lib/semantic-search.js +252 -252
- package/scripts/limpiar-artefactos-python.js +131 -131
- package/scripts/mcp-server/README.md +128 -128
- package/scripts/mcp-server/handlers.js +206 -206
- package/scripts/migrar-csv-a-array.js +168 -168
- package/scripts/migrar-fase-dominio.js +201 -201
- package/scripts/publicar.js +511 -511
- package/scripts/run-eval.js +141 -141
- package/scripts/validar-manifest.js +231 -195
- package/scripts/validar-userland-vacio.js +110 -110
|
@@ -1,96 +1,96 @@
|
|
|
1
|
-
'use strict';
|
|
2
|
-
|
|
3
|
-
/**
|
|
4
|
-
* Merkle Audit — Trail de auditoria con hash-chain criptografica.
|
|
5
|
-
*
|
|
6
|
-
* Adoptado de OpenFang (SECURITY.md: Merkle Hash-Chain Audit Trail).
|
|
7
|
-
* Cada entrada incluye el hash de la entrada anterior, formando una cadena
|
|
8
|
-
* inmutable. Cualquier edicion del archivo se detecta por rotura de la cadena.
|
|
9
|
-
*
|
|
10
|
-
* A diferencia de audit-trail.js (JSONL append-only sin verificacion),
|
|
11
|
-
* merkle-audit garantiza integridad criptografica.
|
|
12
|
-
*
|
|
13
|
-
* Zero dependencias externas.
|
|
14
|
-
*
|
|
15
|
-
* @module hooks/lib/merkle-audit
|
|
16
|
-
*/
|
|
17
|
-
|
|
18
|
-
const crypto = require('crypto');
|
|
19
|
-
const fs = require('fs');
|
|
20
|
-
|
|
21
|
-
function _sha256(data) {
|
|
22
|
-
return crypto.createHash('sha256').update(data).digest('hex');
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
function crearEntrada(datos, hashAnterior) {
|
|
26
|
-
const payload = JSON.stringify({
|
|
27
|
-
ts: new Date().toISOString(),
|
|
28
|
-
tool: datos.tool || '',
|
|
29
|
-
action: datos.action || '',
|
|
30
|
-
target: datos.target || '',
|
|
31
|
-
agent: datos.agent || '',
|
|
32
|
-
result: datos.result || 'ok',
|
|
33
|
-
prev: hashAnterior,
|
|
34
|
-
});
|
|
35
|
-
const hash = _sha256(payload);
|
|
36
|
-
return { payload, hash };
|
|
37
|
-
}
|
|
38
|
-
|
|
39
|
-
function appendAudit(rutaArchivo, datos) {
|
|
40
|
-
let hashAnterior = '0'.repeat(64);
|
|
41
|
-
|
|
42
|
-
if (fs.existsSync(rutaArchivo)) {
|
|
43
|
-
try {
|
|
44
|
-
const contenido = fs.readFileSync(rutaArchivo, 'utf8').trim();
|
|
45
|
-
const lineas = contenido.split('\n').filter(Boolean);
|
|
46
|
-
if (lineas.length > 0) {
|
|
47
|
-
const ultima = JSON.parse(lineas[lineas.length - 1]);
|
|
48
|
-
hashAnterior = ultima.hash || hashAnterior;
|
|
49
|
-
}
|
|
50
|
-
} catch { /* archivo corrupto, empezar nueva cadena */ }
|
|
51
|
-
}
|
|
52
|
-
|
|
53
|
-
const { payload, hash } = crearEntrada(datos, hashAnterior);
|
|
54
|
-
const linea = JSON.stringify({ ...JSON.parse(payload), hash }) + '\n';
|
|
55
|
-
fs.appendFileSync(rutaArchivo, linea, 'utf8');
|
|
56
|
-
return hash;
|
|
57
|
-
}
|
|
58
|
-
|
|
59
|
-
function verificarCadena(rutaArchivo) {
|
|
60
|
-
if (!fs.existsSync(rutaArchivo)) return { valido: true, entradas: 0 };
|
|
61
|
-
|
|
62
|
-
const contenido = fs.readFileSync(rutaArchivo, 'utf8').trim();
|
|
63
|
-
const lineas = contenido.split('\n').filter(Boolean);
|
|
64
|
-
let hashEsperado = '0'.repeat(64);
|
|
65
|
-
let roturaEn = -1;
|
|
66
|
-
|
|
67
|
-
for (let i = 0; i < lineas.length; i++) {
|
|
68
|
-
try {
|
|
69
|
-
const entrada = JSON.parse(lineas[i]);
|
|
70
|
-
if (entrada.prev !== hashEsperado) {
|
|
71
|
-
roturaEn = i;
|
|
72
|
-
break;
|
|
73
|
-
}
|
|
74
|
-
const { hash: storedHash, ...resto } = entrada;
|
|
75
|
-
const payloadRecalc = JSON.stringify(resto);
|
|
76
|
-
const hashRecalc = _sha256(payloadRecalc);
|
|
77
|
-
if (hashRecalc !== storedHash) {
|
|
78
|
-
roturaEn = i;
|
|
79
|
-
break;
|
|
80
|
-
}
|
|
81
|
-
hashEsperado = storedHash;
|
|
82
|
-
} catch {
|
|
83
|
-
roturaEn = i;
|
|
84
|
-
break;
|
|
85
|
-
}
|
|
86
|
-
}
|
|
87
|
-
|
|
88
|
-
return {
|
|
89
|
-
valido: roturaEn === -1,
|
|
90
|
-
entradas: lineas.length,
|
|
91
|
-
roturaEn: roturaEn === -1 ? null : roturaEn,
|
|
92
|
-
ultimoHash: hashEsperado,
|
|
93
|
-
};
|
|
94
|
-
}
|
|
95
|
-
|
|
96
|
-
module.exports = { appendAudit, verificarCadena, crearEntrada };
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Merkle Audit — Trail de auditoria con hash-chain criptografica.
|
|
5
|
+
*
|
|
6
|
+
* Adoptado de OpenFang (SECURITY.md: Merkle Hash-Chain Audit Trail).
|
|
7
|
+
* Cada entrada incluye el hash de la entrada anterior, formando una cadena
|
|
8
|
+
* inmutable. Cualquier edicion del archivo se detecta por rotura de la cadena.
|
|
9
|
+
*
|
|
10
|
+
* A diferencia de audit-trail.js (JSONL append-only sin verificacion),
|
|
11
|
+
* merkle-audit garantiza integridad criptografica.
|
|
12
|
+
*
|
|
13
|
+
* Zero dependencias externas.
|
|
14
|
+
*
|
|
15
|
+
* @module hooks/lib/merkle-audit
|
|
16
|
+
*/
|
|
17
|
+
|
|
18
|
+
const crypto = require('crypto');
|
|
19
|
+
const fs = require('fs');
|
|
20
|
+
|
|
21
|
+
function _sha256(data) {
|
|
22
|
+
return crypto.createHash('sha256').update(data).digest('hex');
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
function crearEntrada(datos, hashAnterior) {
|
|
26
|
+
const payload = JSON.stringify({
|
|
27
|
+
ts: new Date().toISOString(),
|
|
28
|
+
tool: datos.tool || '',
|
|
29
|
+
action: datos.action || '',
|
|
30
|
+
target: datos.target || '',
|
|
31
|
+
agent: datos.agent || '',
|
|
32
|
+
result: datos.result || 'ok',
|
|
33
|
+
prev: hashAnterior,
|
|
34
|
+
});
|
|
35
|
+
const hash = _sha256(payload);
|
|
36
|
+
return { payload, hash };
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
function appendAudit(rutaArchivo, datos) {
|
|
40
|
+
let hashAnterior = '0'.repeat(64);
|
|
41
|
+
|
|
42
|
+
if (fs.existsSync(rutaArchivo)) {
|
|
43
|
+
try {
|
|
44
|
+
const contenido = fs.readFileSync(rutaArchivo, 'utf8').trim();
|
|
45
|
+
const lineas = contenido.split('\n').filter(Boolean);
|
|
46
|
+
if (lineas.length > 0) {
|
|
47
|
+
const ultima = JSON.parse(lineas[lineas.length - 1]);
|
|
48
|
+
hashAnterior = ultima.hash || hashAnterior;
|
|
49
|
+
}
|
|
50
|
+
} catch { /* archivo corrupto, empezar nueva cadena */ }
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
const { payload, hash } = crearEntrada(datos, hashAnterior);
|
|
54
|
+
const linea = JSON.stringify({ ...JSON.parse(payload), hash }) + '\n';
|
|
55
|
+
fs.appendFileSync(rutaArchivo, linea, 'utf8');
|
|
56
|
+
return hash;
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
function verificarCadena(rutaArchivo) {
|
|
60
|
+
if (!fs.existsSync(rutaArchivo)) return { valido: true, entradas: 0 };
|
|
61
|
+
|
|
62
|
+
const contenido = fs.readFileSync(rutaArchivo, 'utf8').trim();
|
|
63
|
+
const lineas = contenido.split('\n').filter(Boolean);
|
|
64
|
+
let hashEsperado = '0'.repeat(64);
|
|
65
|
+
let roturaEn = -1;
|
|
66
|
+
|
|
67
|
+
for (let i = 0; i < lineas.length; i++) {
|
|
68
|
+
try {
|
|
69
|
+
const entrada = JSON.parse(lineas[i]);
|
|
70
|
+
if (entrada.prev !== hashEsperado) {
|
|
71
|
+
roturaEn = i;
|
|
72
|
+
break;
|
|
73
|
+
}
|
|
74
|
+
const { hash: storedHash, ...resto } = entrada;
|
|
75
|
+
const payloadRecalc = JSON.stringify(resto);
|
|
76
|
+
const hashRecalc = _sha256(payloadRecalc);
|
|
77
|
+
if (hashRecalc !== storedHash) {
|
|
78
|
+
roturaEn = i;
|
|
79
|
+
break;
|
|
80
|
+
}
|
|
81
|
+
hashEsperado = storedHash;
|
|
82
|
+
} catch {
|
|
83
|
+
roturaEn = i;
|
|
84
|
+
break;
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
return {
|
|
89
|
+
valido: roturaEn === -1,
|
|
90
|
+
entradas: lineas.length,
|
|
91
|
+
roturaEn: roturaEn === -1 ? null : roturaEn,
|
|
92
|
+
ultimoHash: hashEsperado,
|
|
93
|
+
};
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
module.exports = { appendAudit, verificarCadena, crearEntrada };
|
|
@@ -1,191 +1,191 @@
|
|
|
1
|
-
'use strict';
|
|
2
|
-
|
|
3
|
-
/**
|
|
4
|
-
* Provenance Tracker — Rastreo de proveniencia de skills y componentes.
|
|
5
|
-
*
|
|
6
|
-
* Registra de dónde viene cada skill/agente instalado: si es del core SWL,
|
|
7
|
-
* de un repo externo, de auto-evolución, o de la comunidad. Genera y lee
|
|
8
|
-
* archivos meta.json por componente.
|
|
9
|
-
*
|
|
10
|
-
* Patrón adoptado de skillshare (internal/install/meta.go).
|
|
11
|
-
*
|
|
12
|
-
* Zero dependencias externas.
|
|
13
|
-
*
|
|
14
|
-
* Uso:
|
|
15
|
-
* const { registrarProveniencia, leerProveniencia, listarProveniencias } = require('./lib/provenance-tracker');
|
|
16
|
-
*
|
|
17
|
-
* registrarProveniencia('/ruta/skill', {
|
|
18
|
-
* origin: 'core',
|
|
19
|
-
* repoUrl: 'https://github.com/saulwadeleon/swl-ses',
|
|
20
|
-
* version: '5.6.3',
|
|
21
|
-
* });
|
|
22
|
-
*
|
|
23
|
-
* const meta = leerProveniencia('/ruta/skill');
|
|
24
|
-
* // { origin: 'core', repoUrl: '...', version: '5.6.3', installedAt: '...' }
|
|
25
|
-
*
|
|
26
|
-
* @module hooks/lib/provenance-tracker
|
|
27
|
-
*/
|
|
28
|
-
|
|
29
|
-
const fs = require('fs');
|
|
30
|
-
const path = require('path');
|
|
31
|
-
|
|
32
|
-
// Escritura atómica obligatoria para meta.json (regla CLAUDE.md).
|
|
33
|
-
let atomicWriteJSON;
|
|
34
|
-
try {
|
|
35
|
-
({ atomicWriteJSON } = require('./atomic-write'));
|
|
36
|
-
} catch {
|
|
37
|
-
atomicWriteJSON = (p, o) => fs.writeFileSync(p, JSON.stringify(o, null, 2), 'utf8');
|
|
38
|
-
}
|
|
39
|
-
|
|
40
|
-
// ---------------------------------------------------------------------------
|
|
41
|
-
// Constantes
|
|
42
|
-
// ---------------------------------------------------------------------------
|
|
43
|
-
|
|
44
|
-
const META_FILENAME = 'meta.json';
|
|
45
|
-
|
|
46
|
-
/**
|
|
47
|
-
* Orígenes válidos de un componente.
|
|
48
|
-
* @readonly
|
|
49
|
-
*/
|
|
50
|
-
const ORIGENES = Object.freeze({
|
|
51
|
-
CORE: 'core', // Del paquete swl-ses
|
|
52
|
-
AUTO_EVOLUTION: 'auto-evolution', // Generado por /swl:evolucionar
|
|
53
|
-
IMPORTED: 'imported', // Importado de repo externo
|
|
54
|
-
COMMUNITY: 'community', // De skills.sh o similares
|
|
55
|
-
USERLAND: 'userland', // Del usuario (_userland/)
|
|
56
|
-
ANTHROPIC: 'anthropic', // De anthropics/skills
|
|
57
|
-
});
|
|
58
|
-
|
|
59
|
-
// ---------------------------------------------------------------------------
|
|
60
|
-
// API pública
|
|
61
|
-
// ---------------------------------------------------------------------------
|
|
62
|
-
|
|
63
|
-
/**
|
|
64
|
-
* Registra la proveniencia de un componente instalado.
|
|
65
|
-
*
|
|
66
|
-
* Crea o actualiza meta.json en el directorio del componente.
|
|
67
|
-
*
|
|
68
|
-
* @param {string} componentDir - Ruta al directorio del componente
|
|
69
|
-
* @param {object} datos
|
|
70
|
-
* @param {string} datos.origin - Origen (ver ORIGENES)
|
|
71
|
-
* @param {string} [datos.repoUrl] - URL del repo de origen
|
|
72
|
-
* @param {string} [datos.commit] - Commit hash del origen
|
|
73
|
-
* @param {string} [datos.version] - Versión del sistema al instalar
|
|
74
|
-
* @param {string} [datos.author] - Autor o equipo
|
|
75
|
-
* @param {string} [datos.note] - Nota libre
|
|
76
|
-
*/
|
|
77
|
-
function registrarProveniencia(componentDir, datos) {
|
|
78
|
-
const metaPath = path.join(componentDir, META_FILENAME);
|
|
79
|
-
|
|
80
|
-
const meta = {
|
|
81
|
-
origin: datos.origin || ORIGENES.CORE,
|
|
82
|
-
repoUrl: datos.repoUrl || null,
|
|
83
|
-
commit: datos.commit || null,
|
|
84
|
-
version: datos.version || null,
|
|
85
|
-
author: datos.author || null,
|
|
86
|
-
note: datos.note || null,
|
|
87
|
-
installedAt: new Date().toISOString(),
|
|
88
|
-
installedBy: `swl-ses v${datos.version || 'unknown'}`,
|
|
89
|
-
};
|
|
90
|
-
|
|
91
|
-
// Si ya existe, preservar datos anteriores y agregar historial
|
|
92
|
-
if (fs.existsSync(metaPath)) {
|
|
93
|
-
try {
|
|
94
|
-
const existing = JSON.parse(fs.readFileSync(metaPath, 'utf8'));
|
|
95
|
-
// Mover datos anteriores a historial
|
|
96
|
-
if (!existing.history) existing.history = [];
|
|
97
|
-
const { history, ...prevData } = existing;
|
|
98
|
-
existing.history.push({ ...prevData, replacedAt: meta.installedAt });
|
|
99
|
-
|
|
100
|
-
// Actualizar con nuevos datos
|
|
101
|
-
Object.assign(existing, meta);
|
|
102
|
-
existing.history = existing.history.slice(-5); // Máximo 5 entradas de historial
|
|
103
|
-
|
|
104
|
-
atomicWriteJSON(metaPath, existing);
|
|
105
|
-
} catch {
|
|
106
|
-
// Error leyendo existente — sobrescribir
|
|
107
|
-
atomicWriteJSON(metaPath, meta);
|
|
108
|
-
}
|
|
109
|
-
} else {
|
|
110
|
-
atomicWriteJSON(metaPath, meta);
|
|
111
|
-
}
|
|
112
|
-
}
|
|
113
|
-
|
|
114
|
-
/**
|
|
115
|
-
* Lee la proveniencia de un componente.
|
|
116
|
-
*
|
|
117
|
-
* @param {string} componentDir - Ruta al directorio del componente
|
|
118
|
-
* @returns {object|null} Datos de proveniencia o null si no existe
|
|
119
|
-
*/
|
|
120
|
-
function leerProveniencia(componentDir) {
|
|
121
|
-
const metaPath = path.join(componentDir, META_FILENAME);
|
|
122
|
-
if (!fs.existsSync(metaPath)) return null;
|
|
123
|
-
|
|
124
|
-
try {
|
|
125
|
-
return JSON.parse(fs.readFileSync(metaPath, 'utf8'));
|
|
126
|
-
} catch {
|
|
127
|
-
return null;
|
|
128
|
-
}
|
|
129
|
-
}
|
|
130
|
-
|
|
131
|
-
/**
|
|
132
|
-
* Lista la proveniencia de todos los componentes en un directorio.
|
|
133
|
-
*
|
|
134
|
-
* @param {string} parentDir - Directorio padre (ej: habilidades/)
|
|
135
|
-
* @returns {Array<{ name: string, origin: string, version: string, installedAt: string }>}
|
|
136
|
-
*/
|
|
137
|
-
function listarProveniencias(parentDir) {
|
|
138
|
-
if (!fs.existsSync(parentDir)) return [];
|
|
139
|
-
|
|
140
|
-
const resultado = [];
|
|
141
|
-
const entries = fs.readdirSync(parentDir, { withFileTypes: true });
|
|
142
|
-
|
|
143
|
-
for (const entry of entries) {
|
|
144
|
-
if (!entry.isDirectory()) continue;
|
|
145
|
-
const meta = leerProveniencia(path.join(parentDir, entry.name));
|
|
146
|
-
resultado.push({
|
|
147
|
-
name: entry.name,
|
|
148
|
-
origin: meta ? meta.origin : 'unknown',
|
|
149
|
-
version: meta ? meta.version : null,
|
|
150
|
-
installedAt: meta ? meta.installedAt : null,
|
|
151
|
-
repoUrl: meta ? meta.repoUrl : null,
|
|
152
|
-
});
|
|
153
|
-
}
|
|
154
|
-
|
|
155
|
-
return resultado;
|
|
156
|
-
}
|
|
157
|
-
|
|
158
|
-
/**
|
|
159
|
-
* Genera un resumen de proveniencia por origen.
|
|
160
|
-
*
|
|
161
|
-
* @param {string} parentDir - Directorio padre
|
|
162
|
-
* @returns {{ total: number, byOrigin: Object<string, number>, unknown: number }}
|
|
163
|
-
*/
|
|
164
|
-
function resumenProveniencia(parentDir) {
|
|
165
|
-
const lista = listarProveniencias(parentDir);
|
|
166
|
-
const byOrigin = {};
|
|
167
|
-
let unknown = 0;
|
|
168
|
-
|
|
169
|
-
for (const item of lista) {
|
|
170
|
-
if (item.origin === 'unknown') {
|
|
171
|
-
unknown++;
|
|
172
|
-
} else {
|
|
173
|
-
byOrigin[item.origin] = (byOrigin[item.origin] || 0) + 1;
|
|
174
|
-
}
|
|
175
|
-
}
|
|
176
|
-
|
|
177
|
-
return { total: lista.length, byOrigin, unknown };
|
|
178
|
-
}
|
|
179
|
-
|
|
180
|
-
// ---------------------------------------------------------------------------
|
|
181
|
-
// Exports
|
|
182
|
-
// ---------------------------------------------------------------------------
|
|
183
|
-
|
|
184
|
-
module.exports = {
|
|
185
|
-
registrarProveniencia,
|
|
186
|
-
leerProveniencia,
|
|
187
|
-
listarProveniencias,
|
|
188
|
-
resumenProveniencia,
|
|
189
|
-
ORIGENES,
|
|
190
|
-
META_FILENAME,
|
|
191
|
-
};
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Provenance Tracker — Rastreo de proveniencia de skills y componentes.
|
|
5
|
+
*
|
|
6
|
+
* Registra de dónde viene cada skill/agente instalado: si es del core SWL,
|
|
7
|
+
* de un repo externo, de auto-evolución, o de la comunidad. Genera y lee
|
|
8
|
+
* archivos meta.json por componente.
|
|
9
|
+
*
|
|
10
|
+
* Patrón adoptado de skillshare (internal/install/meta.go).
|
|
11
|
+
*
|
|
12
|
+
* Zero dependencias externas.
|
|
13
|
+
*
|
|
14
|
+
* Uso:
|
|
15
|
+
* const { registrarProveniencia, leerProveniencia, listarProveniencias } = require('./lib/provenance-tracker');
|
|
16
|
+
*
|
|
17
|
+
* registrarProveniencia('/ruta/skill', {
|
|
18
|
+
* origin: 'core',
|
|
19
|
+
* repoUrl: 'https://github.com/saulwadeleon/swl-ses',
|
|
20
|
+
* version: '5.6.3',
|
|
21
|
+
* });
|
|
22
|
+
*
|
|
23
|
+
* const meta = leerProveniencia('/ruta/skill');
|
|
24
|
+
* // { origin: 'core', repoUrl: '...', version: '5.6.3', installedAt: '...' }
|
|
25
|
+
*
|
|
26
|
+
* @module hooks/lib/provenance-tracker
|
|
27
|
+
*/
|
|
28
|
+
|
|
29
|
+
const fs = require('fs');
|
|
30
|
+
const path = require('path');
|
|
31
|
+
|
|
32
|
+
// Escritura atómica obligatoria para meta.json (regla CLAUDE.md).
|
|
33
|
+
let atomicWriteJSON;
|
|
34
|
+
try {
|
|
35
|
+
({ atomicWriteJSON } = require('./atomic-write'));
|
|
36
|
+
} catch {
|
|
37
|
+
atomicWriteJSON = (p, o) => fs.writeFileSync(p, JSON.stringify(o, null, 2), 'utf8');
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
// ---------------------------------------------------------------------------
|
|
41
|
+
// Constantes
|
|
42
|
+
// ---------------------------------------------------------------------------
|
|
43
|
+
|
|
44
|
+
const META_FILENAME = 'meta.json';
|
|
45
|
+
|
|
46
|
+
/**
|
|
47
|
+
* Orígenes válidos de un componente.
|
|
48
|
+
* @readonly
|
|
49
|
+
*/
|
|
50
|
+
const ORIGENES = Object.freeze({
|
|
51
|
+
CORE: 'core', // Del paquete swl-ses
|
|
52
|
+
AUTO_EVOLUTION: 'auto-evolution', // Generado por /swl:evolucionar
|
|
53
|
+
IMPORTED: 'imported', // Importado de repo externo
|
|
54
|
+
COMMUNITY: 'community', // De skills.sh o similares
|
|
55
|
+
USERLAND: 'userland', // Del usuario (_userland/)
|
|
56
|
+
ANTHROPIC: 'anthropic', // De anthropics/skills
|
|
57
|
+
});
|
|
58
|
+
|
|
59
|
+
// ---------------------------------------------------------------------------
|
|
60
|
+
// API pública
|
|
61
|
+
// ---------------------------------------------------------------------------
|
|
62
|
+
|
|
63
|
+
/**
|
|
64
|
+
* Registra la proveniencia de un componente instalado.
|
|
65
|
+
*
|
|
66
|
+
* Crea o actualiza meta.json en el directorio del componente.
|
|
67
|
+
*
|
|
68
|
+
* @param {string} componentDir - Ruta al directorio del componente
|
|
69
|
+
* @param {object} datos
|
|
70
|
+
* @param {string} datos.origin - Origen (ver ORIGENES)
|
|
71
|
+
* @param {string} [datos.repoUrl] - URL del repo de origen
|
|
72
|
+
* @param {string} [datos.commit] - Commit hash del origen
|
|
73
|
+
* @param {string} [datos.version] - Versión del sistema al instalar
|
|
74
|
+
* @param {string} [datos.author] - Autor o equipo
|
|
75
|
+
* @param {string} [datos.note] - Nota libre
|
|
76
|
+
*/
|
|
77
|
+
function registrarProveniencia(componentDir, datos) {
|
|
78
|
+
const metaPath = path.join(componentDir, META_FILENAME);
|
|
79
|
+
|
|
80
|
+
const meta = {
|
|
81
|
+
origin: datos.origin || ORIGENES.CORE,
|
|
82
|
+
repoUrl: datos.repoUrl || null,
|
|
83
|
+
commit: datos.commit || null,
|
|
84
|
+
version: datos.version || null,
|
|
85
|
+
author: datos.author || null,
|
|
86
|
+
note: datos.note || null,
|
|
87
|
+
installedAt: new Date().toISOString(),
|
|
88
|
+
installedBy: `swl-ses v${datos.version || 'unknown'}`,
|
|
89
|
+
};
|
|
90
|
+
|
|
91
|
+
// Si ya existe, preservar datos anteriores y agregar historial
|
|
92
|
+
if (fs.existsSync(metaPath)) {
|
|
93
|
+
try {
|
|
94
|
+
const existing = JSON.parse(fs.readFileSync(metaPath, 'utf8'));
|
|
95
|
+
// Mover datos anteriores a historial
|
|
96
|
+
if (!existing.history) existing.history = [];
|
|
97
|
+
const { history, ...prevData } = existing;
|
|
98
|
+
existing.history.push({ ...prevData, replacedAt: meta.installedAt });
|
|
99
|
+
|
|
100
|
+
// Actualizar con nuevos datos
|
|
101
|
+
Object.assign(existing, meta);
|
|
102
|
+
existing.history = existing.history.slice(-5); // Máximo 5 entradas de historial
|
|
103
|
+
|
|
104
|
+
atomicWriteJSON(metaPath, existing);
|
|
105
|
+
} catch {
|
|
106
|
+
// Error leyendo existente — sobrescribir
|
|
107
|
+
atomicWriteJSON(metaPath, meta);
|
|
108
|
+
}
|
|
109
|
+
} else {
|
|
110
|
+
atomicWriteJSON(metaPath, meta);
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
/**
|
|
115
|
+
* Lee la proveniencia de un componente.
|
|
116
|
+
*
|
|
117
|
+
* @param {string} componentDir - Ruta al directorio del componente
|
|
118
|
+
* @returns {object|null} Datos de proveniencia o null si no existe
|
|
119
|
+
*/
|
|
120
|
+
function leerProveniencia(componentDir) {
|
|
121
|
+
const metaPath = path.join(componentDir, META_FILENAME);
|
|
122
|
+
if (!fs.existsSync(metaPath)) return null;
|
|
123
|
+
|
|
124
|
+
try {
|
|
125
|
+
return JSON.parse(fs.readFileSync(metaPath, 'utf8'));
|
|
126
|
+
} catch {
|
|
127
|
+
return null;
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
/**
|
|
132
|
+
* Lista la proveniencia de todos los componentes en un directorio.
|
|
133
|
+
*
|
|
134
|
+
* @param {string} parentDir - Directorio padre (ej: habilidades/)
|
|
135
|
+
* @returns {Array<{ name: string, origin: string, version: string, installedAt: string }>}
|
|
136
|
+
*/
|
|
137
|
+
function listarProveniencias(parentDir) {
|
|
138
|
+
if (!fs.existsSync(parentDir)) return [];
|
|
139
|
+
|
|
140
|
+
const resultado = [];
|
|
141
|
+
const entries = fs.readdirSync(parentDir, { withFileTypes: true });
|
|
142
|
+
|
|
143
|
+
for (const entry of entries) {
|
|
144
|
+
if (!entry.isDirectory()) continue;
|
|
145
|
+
const meta = leerProveniencia(path.join(parentDir, entry.name));
|
|
146
|
+
resultado.push({
|
|
147
|
+
name: entry.name,
|
|
148
|
+
origin: meta ? meta.origin : 'unknown',
|
|
149
|
+
version: meta ? meta.version : null,
|
|
150
|
+
installedAt: meta ? meta.installedAt : null,
|
|
151
|
+
repoUrl: meta ? meta.repoUrl : null,
|
|
152
|
+
});
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
return resultado;
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
/**
|
|
159
|
+
* Genera un resumen de proveniencia por origen.
|
|
160
|
+
*
|
|
161
|
+
* @param {string} parentDir - Directorio padre
|
|
162
|
+
* @returns {{ total: number, byOrigin: Object<string, number>, unknown: number }}
|
|
163
|
+
*/
|
|
164
|
+
function resumenProveniencia(parentDir) {
|
|
165
|
+
const lista = listarProveniencias(parentDir);
|
|
166
|
+
const byOrigin = {};
|
|
167
|
+
let unknown = 0;
|
|
168
|
+
|
|
169
|
+
for (const item of lista) {
|
|
170
|
+
if (item.origin === 'unknown') {
|
|
171
|
+
unknown++;
|
|
172
|
+
} else {
|
|
173
|
+
byOrigin[item.origin] = (byOrigin[item.origin] || 0) + 1;
|
|
174
|
+
}
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
return { total: lista.length, byOrigin, unknown };
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
// ---------------------------------------------------------------------------
|
|
181
|
+
// Exports
|
|
182
|
+
// ---------------------------------------------------------------------------
|
|
183
|
+
|
|
184
|
+
module.exports = {
|
|
185
|
+
registrarProveniencia,
|
|
186
|
+
leerProveniencia,
|
|
187
|
+
listarProveniencias,
|
|
188
|
+
resumenProveniencia,
|
|
189
|
+
ORIGENES,
|
|
190
|
+
META_FILENAME,
|
|
191
|
+
};
|