@saulwade/swl-ses 2.1.0 → 2.2.0
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 +1 -1
- package/README.md +1 -1
- package/bin/swl-ses.js +63 -0
- package/comandos/swl/adoptar-proyecto.md +258 -255
- package/comandos/swl/aprender.md +828 -840
- package/comandos/swl/aprobar-plan.md +23 -35
- package/comandos/swl/autoresearch.md +12 -14
- package/comandos/swl/briefing.md +5 -8
- package/comandos/swl/checkpoint.md +10 -15
- package/comandos/swl/claudemd.md +239 -234
- package/comandos/swl/configurar-ci.md +20 -19
- package/comandos/swl/cron.md +10 -12
- package/comandos/swl/ejecutar-fase.md +10 -3
- package/comandos/swl/evolucionar.md +6 -11
- package/comandos/swl/inbox.md +10 -10
- package/comandos/swl/modelo.md +7 -9
- package/comandos/swl/notificaciones.md +19 -116
- package/comandos/swl/nuevo-proyecto.md +205 -205
- package/comandos/swl/status.md +333 -348
- package/comandos/swl/verificar.md +817 -813
- package/habilidades/swl-claudemd/SKILL.md +10 -6
- package/hooks/lib/propose-step.js +1 -0
- package/llms.txt +1 -1
- package/manifiestos/skills-lock.json +5 -5
- package/package.json +1 -1
- package/plugin.json +1 -1
- package/scripts/auditar-claudemd.js +38 -0
- package/scripts/cli/aprobar-plan.js +73 -0
- package/scripts/cli/briefing.js +23 -0
- package/scripts/cli/ciclo-evolucion.js +26 -0
- package/scripts/cli/configurar-ci.js +40 -0
- package/scripts/cli/derivar-feature-list.js +25 -0
- package/scripts/cli/detectar-host.js +27 -0
- package/scripts/cli/diary-entry.js +69 -0
- package/scripts/cli/execution-state.js +18 -0
- package/scripts/cli/gateway-notify.js +41 -0
- package/scripts/cli/liberar-fase.js +42 -0
- package/scripts/cli/loop-telemetry.js +125 -0
- package/scripts/cli/mark-evolved.js +56 -0
- package/scripts/cli/metricas-dora.js +26 -0
- package/scripts/cli/near-duplicate.js +55 -0
- package/scripts/cli/notificaciones.js +123 -0
- package/scripts/cli/propose-step.js +29 -0
- package/scripts/cli/schedule-parse.js +19 -0
- package/scripts/cli/sugerir-modelo.js +20 -0
- package/scripts/cli/verificar-plan.js +36 -0
- package/scripts/cli/verificar-trazabilidad.js +35 -0
- package/scripts/derivar-feature-list.js +1 -0
- package/scripts/lib/auditar-invocaciones-comandos.js +104 -0
- package/scripts/lib/resolver-plan-fase.js +37 -0
- package/scripts/validar.js +13 -0
- package/scripts/verificar-trazabilidad.js +1 -1
|
@@ -0,0 +1,125 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Wrapper CLI `swl-ses loop-telemetry <op>` — telemetría de loops iterativos.
|
|
5
|
+
* Reemplaza los inline `require('./hooks/lib/loop-telemetry')` relativos al
|
|
6
|
+
* proyecto (rotos downstream; ver docs/invocacion-cli-cross-scope.md).
|
|
7
|
+
*
|
|
8
|
+
* Ops:
|
|
9
|
+
* iniciar --tipo=X [--direccion=higher_is_better] [--config='{json}'] → imprime dir
|
|
10
|
+
* registrar --dir=D --iteracion=N --metrica=M [--delta=D] [--estado=keep]
|
|
11
|
+
* [--descripcion="..."]
|
|
12
|
+
* handoff --dir=D --source=X --status=Y [--findings='[json]'] [--config='{json}']
|
|
13
|
+
* listar (default) — resumen de las últimas 10 corridas
|
|
14
|
+
*/
|
|
15
|
+
|
|
16
|
+
const fs = require('node:fs');
|
|
17
|
+
const path = require('node:path');
|
|
18
|
+
const lt = require('../../hooks/lib/loop-telemetry');
|
|
19
|
+
|
|
20
|
+
// Parseo no degradante: si el flag está presente pero su JSON es inválido,
|
|
21
|
+
// reporta y devuelve {ok:false} en vez de silenciar a fallback (anti-fallback
|
|
22
|
+
// silencioso). El fallback solo aplica cuando el flag está ausente.
|
|
23
|
+
function parseJSON(v, fallback, flagName) {
|
|
24
|
+
if (v === undefined) return { ok: true, val: fallback };
|
|
25
|
+
try {
|
|
26
|
+
return { ok: true, val: JSON.parse(v) };
|
|
27
|
+
} catch (e) {
|
|
28
|
+
console.error(`[loop-telemetry] --${flagName} JSON inválido: ${e.message}`);
|
|
29
|
+
return { ok: false };
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
function requireDir(opciones) {
|
|
34
|
+
if (!opciones.dir) {
|
|
35
|
+
console.error('Error: falta --dir=<ruta>');
|
|
36
|
+
process.exitCode = 1;
|
|
37
|
+
return false;
|
|
38
|
+
}
|
|
39
|
+
return true;
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
function loopTelemetry(opciones = {}) {
|
|
43
|
+
const op = (opciones._args && opciones._args[0]) || opciones.op || 'listar';
|
|
44
|
+
|
|
45
|
+
if (op === 'iniciar') {
|
|
46
|
+
const config = parseJSON(opciones.config, {}, 'config');
|
|
47
|
+
if (!config.ok) {
|
|
48
|
+
process.exitCode = 1;
|
|
49
|
+
return;
|
|
50
|
+
}
|
|
51
|
+
const r = lt.iniciarCorrida({
|
|
52
|
+
tipo: opciones.tipo,
|
|
53
|
+
direccion: opciones.direccion || 'higher_is_better',
|
|
54
|
+
config: config.val,
|
|
55
|
+
});
|
|
56
|
+
console.log(r.dir);
|
|
57
|
+
return;
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
if (op === 'registrar') {
|
|
61
|
+
if (!requireDir(opciones)) return;
|
|
62
|
+
const fila = {
|
|
63
|
+
iteracion: Number(opciones.iteracion),
|
|
64
|
+
metrica: Number(opciones.metrica),
|
|
65
|
+
estado: opciones.estado || 'keep',
|
|
66
|
+
descripcion: opciones.descripcion || '',
|
|
67
|
+
};
|
|
68
|
+
if (opciones.delta !== undefined) fila.delta = Number(opciones.delta);
|
|
69
|
+
lt.registrarIteracion(opciones.dir, fila);
|
|
70
|
+
console.log('[loop-telemetry] iteración registrada en ' + opciones.dir);
|
|
71
|
+
return;
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
if (op === 'handoff') {
|
|
75
|
+
if (!requireDir(opciones)) return;
|
|
76
|
+
const findings = parseJSON(opciones.findings, [], 'findings');
|
|
77
|
+
const config = parseJSON(opciones.config, {}, 'config');
|
|
78
|
+
if (!findings.ok || !config.ok) {
|
|
79
|
+
process.exitCode = 1;
|
|
80
|
+
return;
|
|
81
|
+
}
|
|
82
|
+
lt.escribirHandoff(opciones.dir, {
|
|
83
|
+
source: opciones.source,
|
|
84
|
+
status: opciones.status,
|
|
85
|
+
findings: findings.val,
|
|
86
|
+
config: config.val,
|
|
87
|
+
});
|
|
88
|
+
console.log('[loop-telemetry] handoff escrito en ' + opciones.dir);
|
|
89
|
+
return;
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
// listar (default)
|
|
93
|
+
const base = path.join(process.cwd(), lt.DIR_LOOPS);
|
|
94
|
+
let dirs = [];
|
|
95
|
+
try {
|
|
96
|
+
dirs = fs
|
|
97
|
+
.readdirSync(base)
|
|
98
|
+
.map((d) => path.join(base, d))
|
|
99
|
+
.filter((d) => fs.statSync(d).isDirectory());
|
|
100
|
+
} catch {
|
|
101
|
+
/* sin directorio de loops */
|
|
102
|
+
}
|
|
103
|
+
if (dirs.length === 0) {
|
|
104
|
+
console.log('(sin corridas de loops registradas)');
|
|
105
|
+
return;
|
|
106
|
+
}
|
|
107
|
+
for (const dir of dirs.sort().slice(-10)) {
|
|
108
|
+
try {
|
|
109
|
+
const t = lt.analizarTrayectoria(dir);
|
|
110
|
+
const h = lt.leerHandoff(dir);
|
|
111
|
+
console.log(
|
|
112
|
+
path.basename(dir) +
|
|
113
|
+
' | iters: ' + t.totalIteraciones +
|
|
114
|
+
' | keep/revert: ' + t.keeps + '/' + t.reverts +
|
|
115
|
+
' | métrica: ' + t.metricaInicial + ' → ' + t.metricaFinal +
|
|
116
|
+
' | plateau: ' + (t.plateau ? 'SÍ' : 'no') +
|
|
117
|
+
' | status: ' + (h ? h.status : 'en curso')
|
|
118
|
+
);
|
|
119
|
+
} catch (e) {
|
|
120
|
+
console.log(path.basename(dir) + ' | (ilegible: ' + e.message + ')');
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
module.exports = loopTelemetry;
|
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Wrapper CLI `swl-ses mark-evolved` — marca un componente SWL como evolucionado
|
|
5
|
+
* (frontmatter evolved: true + metadatos). Reemplaza el inline
|
|
6
|
+
* `require('./hooks/lib/evolution-tracker')` relativo al proyecto (roto
|
|
7
|
+
* downstream; ver docs/invocacion-cli-cross-scope.md).
|
|
8
|
+
*
|
|
9
|
+
* Uso: swl-ses mark-evolved <ruta> [--from=VER] [--by=aprender]
|
|
10
|
+
* [--note="..."] [--rounds=N] [--score="A% → B%"]
|
|
11
|
+
*/
|
|
12
|
+
|
|
13
|
+
const path = require('node:path');
|
|
14
|
+
const { markAsEvolved } = require('../../hooks/lib/evolution-tracker');
|
|
15
|
+
|
|
16
|
+
function markEvolved(opciones = {}) {
|
|
17
|
+
const ruta = (opciones._args && opciones._args[0]) || opciones.ruta;
|
|
18
|
+
if (!ruta) {
|
|
19
|
+
console.error('Error: falta <ruta> del archivo a marcar');
|
|
20
|
+
process.exitCode = 1;
|
|
21
|
+
return;
|
|
22
|
+
}
|
|
23
|
+
// Guard anti path-traversal: la ruta debe resolver dentro del proyecto actual
|
|
24
|
+
// y ser un componente del sistema (.md/.json). Evita sobrescribir archivos
|
|
25
|
+
// arbitrarios con frontmatter si el flag se sintetiza desde input no confiable.
|
|
26
|
+
const cwd = process.cwd();
|
|
27
|
+
const resuelta = path.resolve(cwd, ruta);
|
|
28
|
+
if (resuelta !== cwd && !resuelta.startsWith(cwd + path.sep)) {
|
|
29
|
+
console.error('Error: la ruta debe estar dentro del proyecto actual');
|
|
30
|
+
process.exitCode = 1;
|
|
31
|
+
return;
|
|
32
|
+
}
|
|
33
|
+
if (!/\.(md|json)$/i.test(resuelta)) {
|
|
34
|
+
console.error('Error: solo se pueden marcar componentes .md o .json');
|
|
35
|
+
process.exitCode = 1;
|
|
36
|
+
return;
|
|
37
|
+
}
|
|
38
|
+
let from = opciones.from;
|
|
39
|
+
if (!from) {
|
|
40
|
+
try {
|
|
41
|
+
from = require(path.resolve(process.cwd(), 'package.json')).version;
|
|
42
|
+
} catch {
|
|
43
|
+
from = undefined;
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
const meta = { from, by: opciones.by || 'aprender' };
|
|
47
|
+
if (opciones.note) meta.note = opciones.note;
|
|
48
|
+
if (opciones.rounds) meta.rounds = Number(opciones.rounds);
|
|
49
|
+
if (opciones.score) meta.score = opciones.score;
|
|
50
|
+
|
|
51
|
+
const r = markAsEvolved(ruta, meta);
|
|
52
|
+
console.log(r.marked ? 'Marcado como evolucionado' : 'Error: ' + r.error);
|
|
53
|
+
if (!r.marked) process.exitCode = 1;
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
module.exports = markEvolved;
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Wrapper CLI `swl-ses metricas-dora` — calcula métricas DORA del repo actual.
|
|
5
|
+
* Reemplaza `node scripts/lib/metricas-dora.js` relativo al proyecto (roto
|
|
6
|
+
* downstream; ver docs/invocacion-cli-cross-scope.md).
|
|
7
|
+
*
|
|
8
|
+
* Uso: swl-ses metricas-dora [--json] [--dias=N]
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
const { calcularDora, persistir } = require('../lib/metricas-dora');
|
|
12
|
+
|
|
13
|
+
function metricasDora(opciones = {}) {
|
|
14
|
+
const dias = Number(opciones.dias);
|
|
15
|
+
const ventanaDias = Math.max(1, Math.min(365, Number.isFinite(dias) && dias > 0 ? dias : 30));
|
|
16
|
+
const baseDir = process.cwd();
|
|
17
|
+
const dora = calcularDora(baseDir, { ventanaDias });
|
|
18
|
+
try {
|
|
19
|
+
persistir(baseDir, dora);
|
|
20
|
+
} catch {
|
|
21
|
+
/* persistencia best-effort: no romper el informe si el disco falla */
|
|
22
|
+
}
|
|
23
|
+
console.log(JSON.stringify(dora, null, 2));
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
module.exports = metricasDora;
|
|
@@ -0,0 +1,55 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Wrapper CLI `swl-ses near-duplicate` — detecta si un aprendizaje nuevo es
|
|
5
|
+
* duplicado cercano de los existentes en APRENDIZAJES.md. Reemplaza el inline
|
|
6
|
+
* `require('./hooks/lib/fingerprint-id')` relativo al proyecto (roto downstream;
|
|
7
|
+
* ver docs/invocacion-cli-cross-scope.md).
|
|
8
|
+
*
|
|
9
|
+
* Uso: swl-ses near-duplicate --texto="..." [--archivo=ruta] [--threshold=0.6]
|
|
10
|
+
* echo "texto..." | swl-ses near-duplicate [--archivo=ruta]
|
|
11
|
+
*/
|
|
12
|
+
|
|
13
|
+
const fs = require('node:fs');
|
|
14
|
+
const path = require('node:path');
|
|
15
|
+
const { isNearDuplicate } = require('../../hooks/lib/fingerprint-id');
|
|
16
|
+
|
|
17
|
+
function nearDuplicate(opciones = {}) {
|
|
18
|
+
const archivo = opciones.archivo || '.planning/APRENDIZAJES.md';
|
|
19
|
+
let nuevo = opciones.texto;
|
|
20
|
+
if (!nuevo && !process.stdin.isTTY) {
|
|
21
|
+
try {
|
|
22
|
+
nuevo = fs.readFileSync(0, 'utf8');
|
|
23
|
+
} catch {
|
|
24
|
+
nuevo = '';
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
if (!nuevo) {
|
|
28
|
+
console.error('Error: falta --texto="..." o entrada por stdin');
|
|
29
|
+
process.exitCode = 1;
|
|
30
|
+
return;
|
|
31
|
+
}
|
|
32
|
+
// Guard anti path-traversal: --archivo debe resolver dentro del proyecto.
|
|
33
|
+
const cwd = process.cwd();
|
|
34
|
+
const resuelta = path.resolve(cwd, archivo);
|
|
35
|
+
if (resuelta !== cwd && !resuelta.startsWith(cwd + path.sep)) {
|
|
36
|
+
console.error('Error: --archivo debe estar dentro del proyecto actual');
|
|
37
|
+
process.exitCode = 1;
|
|
38
|
+
return;
|
|
39
|
+
}
|
|
40
|
+
let existentes = [];
|
|
41
|
+
try {
|
|
42
|
+
existentes = fs.readFileSync(resuelta, 'utf8').split(/^## /m).filter(Boolean);
|
|
43
|
+
} catch (e) {
|
|
44
|
+
// Sin archivo de aprendizajes legible: avisar (no degradar en silencio) y
|
|
45
|
+
// tratar todo como único.
|
|
46
|
+
if (e.code !== 'ENOENT') console.error('[near-duplicate] aviso: ' + e.code);
|
|
47
|
+
}
|
|
48
|
+
const t = Number(opciones.threshold);
|
|
49
|
+
const threshold = Number.isFinite(t) && t > 0 ? t : 0.6;
|
|
50
|
+
|
|
51
|
+
const r = isNearDuplicate(nuevo, existentes, threshold);
|
|
52
|
+
console.log(r.nearDuplicate ? 'DUPLICADO (sim=' + r.similarity + ')' : 'ÚNICO');
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
module.exports = nearDuplicate;
|
|
@@ -0,0 +1,123 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Wrapper CLI `swl-ses notificaciones <op>` — gestiona las notificaciones
|
|
5
|
+
* Telegram y su bot daemon. Reemplaza los inline
|
|
6
|
+
* `require('./scripts/lib/notificaciones-telegram')` relativos al proyecto
|
|
7
|
+
* (rotos downstream; ver docs/invocacion-cli-cross-scope.md). Encapsula también
|
|
8
|
+
* el formato de salida que antes vivía en el comando.
|
|
9
|
+
*
|
|
10
|
+
* Ops:
|
|
11
|
+
* init (requiere TTY) bot-start
|
|
12
|
+
* status bot-stop
|
|
13
|
+
* disable [--purge] bot-status
|
|
14
|
+
* repair bot-restart
|
|
15
|
+
* bot-enable-autostart
|
|
16
|
+
* bot-disable-autostart
|
|
17
|
+
*/
|
|
18
|
+
|
|
19
|
+
const nt = require('../lib/notificaciones-telegram');
|
|
20
|
+
|
|
21
|
+
function imprimirStatus() {
|
|
22
|
+
const s = nt.status();
|
|
23
|
+
const botS = nt.botStatus();
|
|
24
|
+
const autostartLabel =
|
|
25
|
+
process.platform === 'win32'
|
|
26
|
+
? 'Scheduled Task'
|
|
27
|
+
: process.platform === 'linux'
|
|
28
|
+
? 'systemd --user'
|
|
29
|
+
: 'LaunchAgent';
|
|
30
|
+
const mudos = s.proyectosSilenciados.length
|
|
31
|
+
? s.proyectosSilenciados.join(', ')
|
|
32
|
+
: 'ninguno';
|
|
33
|
+
console.log('');
|
|
34
|
+
console.log('Notificaciones Telegram — estado actual');
|
|
35
|
+
console.log(' .env: ', s.envExiste ? 'existe' : 'no existe');
|
|
36
|
+
console.log(' token: ', s.tokenConfigurado ? 'configurado' : 'no configurado');
|
|
37
|
+
console.log(
|
|
38
|
+
' chat_id: ',
|
|
39
|
+
s.chatIdConfigurado ? 'configurado (' + s.chatIdParcial + ')' : 'no configurado'
|
|
40
|
+
);
|
|
41
|
+
console.log(' proyectos mudos: ', mudos);
|
|
42
|
+
console.log(' bot daemon: ', botS.activo ? 'activo (PID ' + botS.pid + ')' : 'detenido');
|
|
43
|
+
console.log(' autostart: usar /swl:notificaciones bot enable-autostart (' + autostartLabel + ')');
|
|
44
|
+
console.log('');
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
function imprimirBotStatus() {
|
|
48
|
+
const r = nt.botStatus();
|
|
49
|
+
console.log('');
|
|
50
|
+
console.log('Bot daemon — estado actual');
|
|
51
|
+
console.log(' activo:', r.activo ? 'sí' : 'no');
|
|
52
|
+
console.log(' pid: ', r.pid !== null ? r.pid : 'n/a');
|
|
53
|
+
console.log('');
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
function reportarOk(prefijo, r, mensajeDefault) {
|
|
57
|
+
if (r.ok) {
|
|
58
|
+
console.log('[' + prefijo + ']', r.mensaje || mensajeDefault);
|
|
59
|
+
} else {
|
|
60
|
+
console.error('[' + prefijo + '] Error:', r.error);
|
|
61
|
+
process.exitCode = 1;
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
// Tabla de dispatch de los ops del bot daemon: [prefijo, fn, mensajeDefault].
|
|
66
|
+
const BOT_OPS = {
|
|
67
|
+
'bot-start': ['bot', () => nt.botStart(), 'Bot iniciado.'],
|
|
68
|
+
'bot-stop': ['bot', () => nt.botStop(), 'Bot detenido.'],
|
|
69
|
+
'bot-restart': ['bot', () => nt.botRestart(), 'Bot reiniciado.'],
|
|
70
|
+
'bot-enable-autostart': ['autostart', () => nt.botEnableAutostart(), 'Autostart habilitado.'],
|
|
71
|
+
'bot-disable-autostart': ['autostart', () => nt.botDisableAutostart(), 'Autostart deshabilitado.'],
|
|
72
|
+
};
|
|
73
|
+
|
|
74
|
+
async function notificaciones(opciones = {}) {
|
|
75
|
+
const op = (opciones._args && opciones._args[0]) || opciones.op || 'status';
|
|
76
|
+
|
|
77
|
+
if (BOT_OPS[op]) {
|
|
78
|
+
const [prefijo, fn, msg] = BOT_OPS[op];
|
|
79
|
+
reportarOk(prefijo, fn(), msg);
|
|
80
|
+
return;
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
switch (op) {
|
|
84
|
+
case 'init': {
|
|
85
|
+
if (!process.stdin.isTTY) {
|
|
86
|
+
console.log('Este subcomando requiere entrada interactiva (TTY).');
|
|
87
|
+
console.log('Ejecuta desde tu terminal:');
|
|
88
|
+
console.log(' swl-ses notificaciones init');
|
|
89
|
+
console.log('O usa el instalador:');
|
|
90
|
+
console.log(' npx -y @saulwade/swl-ses@latest install');
|
|
91
|
+
return;
|
|
92
|
+
}
|
|
93
|
+
const r = await nt.init({ esTty: true });
|
|
94
|
+
console.log('[resultado]', r.resultado, r.detalle || '');
|
|
95
|
+
if (r.resultado === 'error') process.exitCode = 1;
|
|
96
|
+
return;
|
|
97
|
+
}
|
|
98
|
+
case 'status':
|
|
99
|
+
imprimirStatus();
|
|
100
|
+
return;
|
|
101
|
+
case 'disable': {
|
|
102
|
+
const r = nt.disable({ confirmar: true, conservarEnv: !opciones.purge });
|
|
103
|
+
console.log('[resultado]', r.resultado, r.detalle || '');
|
|
104
|
+
if (r.resultado === 'error') process.exitCode = 1;
|
|
105
|
+
return;
|
|
106
|
+
}
|
|
107
|
+
case 'repair': {
|
|
108
|
+
const r = await nt.repair();
|
|
109
|
+
console.log('[repair]', r.resultado, r.detalle || '');
|
|
110
|
+
if (r.resultado === 'error') process.exitCode = 1;
|
|
111
|
+
return;
|
|
112
|
+
}
|
|
113
|
+
case 'bot-status':
|
|
114
|
+
imprimirBotStatus();
|
|
115
|
+
return;
|
|
116
|
+
default:
|
|
117
|
+
console.error('Op desconocida: ' + op);
|
|
118
|
+
console.error('Ops: init|status|disable|repair|bot-start|bot-stop|bot-status|bot-restart|bot-enable-autostart|bot-disable-autostart');
|
|
119
|
+
process.exitCode = 1;
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
module.exports = notificaciones;
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Wrapper CLI para `swl-ses propose-step` — anexo propositivo del RESUMEN de
|
|
5
|
+
* fase (ADR-0037). Envuelve hooks/lib/propose-step.js (que lee process.argv).
|
|
6
|
+
*
|
|
7
|
+
* Reemplaza el `node hooks/lib/propose-step.js --rango=...` relativo al proyecto
|
|
8
|
+
* que /swl:ejecutar-fase invocaba — roto downstream (ver
|
|
9
|
+
* docs/invocacion-cli-cross-scope.md).
|
|
10
|
+
*
|
|
11
|
+
* Uso: swl-ses propose-step --rango=<base>..HEAD (opt-out SWL_PROPOSE=0)
|
|
12
|
+
*/
|
|
13
|
+
|
|
14
|
+
const { main } = require('../../hooks/lib/propose-step.js');
|
|
15
|
+
|
|
16
|
+
function proposeStep(opciones = {}) {
|
|
17
|
+
opciones = opciones || {};
|
|
18
|
+
const args = ['node', 'propose-step'];
|
|
19
|
+
if (opciones.rango) args.push(`--rango=${opciones.rango}`);
|
|
20
|
+
const argvOriginal = process.argv;
|
|
21
|
+
process.argv = args;
|
|
22
|
+
try {
|
|
23
|
+
return main(process.argv);
|
|
24
|
+
} finally {
|
|
25
|
+
process.argv = argvOriginal;
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
module.exports = proposeStep;
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Wrapper CLI `swl-ses schedule-parse` — parsea una frase en inglés a expresión
|
|
5
|
+
* cron. Reemplaza el inline `require('./scripts/lib/schedule-parser')` relativo
|
|
6
|
+
* al proyecto (roto downstream; ver docs/invocacion-cli-cross-scope.md).
|
|
7
|
+
*
|
|
8
|
+
* Uso: swl-ses schedule-parse "every morning at 9am"
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
const { parseNaturalSchedule } = require('../lib/schedule-parser');
|
|
12
|
+
|
|
13
|
+
function scheduleParse(opciones = {}) {
|
|
14
|
+
const frase =
|
|
15
|
+
(opciones._args && opciones._args.join(' ')) || opciones.texto || '';
|
|
16
|
+
console.log(JSON.stringify(parseNaturalSchedule(frase), null, 2));
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
module.exports = scheduleParse;
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Wrapper CLI `swl-ses sugerir-modelo` — evalúa la complejidad de una tarea y
|
|
5
|
+
* sugiere el modelo (opus/sonnet/haiku). Reemplaza el inline
|
|
6
|
+
* `require('./hooks/lib/model-router.js')` relativo al proyecto (roto downstream;
|
|
7
|
+
* ver docs/invocacion-cli-cross-scope.md).
|
|
8
|
+
*
|
|
9
|
+
* Uso: swl-ses sugerir-modelo "<descripción de la tarea>"
|
|
10
|
+
*/
|
|
11
|
+
|
|
12
|
+
const { sugerirModelo } = require('../../hooks/lib/model-router.js');
|
|
13
|
+
|
|
14
|
+
function sugerirModeloCli(opciones = {}) {
|
|
15
|
+
const desc =
|
|
16
|
+
(opciones._args && opciones._args.join(' ')) || opciones.desc || '';
|
|
17
|
+
console.log(JSON.stringify(sugerirModelo(desc), null, 2));
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
module.exports = sugerirModeloCli;
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Subcomando `swl-ses verificar-plan --fase=N` — verificación read-only del lock
|
|
5
|
+
* de un PLAN (gate G1). Lo usa /swl:ejecutar-fase antes de implementar y
|
|
6
|
+
* /swl:aprobar-plan Paso 2 para detectar planes ya firmados/mutados/legacy.
|
|
7
|
+
*
|
|
8
|
+
* Reemplaza el `node -e "require('./scripts/lib/plan-lock')"` inline que se
|
|
9
|
+
* rompía downstream (ver docs/invocacion-cli-cross-scope.md).
|
|
10
|
+
*
|
|
11
|
+
* Exit 0 si ok (firmado o legacy); exit 1 si mutado/sin-firmar/corrupto.
|
|
12
|
+
*
|
|
13
|
+
* Uso: swl-ses verificar-plan --fase=N [--plan=<ruta>]
|
|
14
|
+
*/
|
|
15
|
+
|
|
16
|
+
const { verificarPlan } = require('../lib/plan-lock');
|
|
17
|
+
const { resolverPlanPath } = require('../lib/resolver-plan-fase');
|
|
18
|
+
|
|
19
|
+
function verificarPlanCli(opciones = {}) {
|
|
20
|
+
opciones = opciones || {};
|
|
21
|
+
const planPath = resolverPlanPath(opciones);
|
|
22
|
+
if (!planPath) {
|
|
23
|
+
console.error('Uso: swl-ses verificar-plan --fase=N [--plan=<ruta>]');
|
|
24
|
+
process.exit(1);
|
|
25
|
+
}
|
|
26
|
+
const r = verificarPlan(planPath);
|
|
27
|
+
console.log(JSON.stringify(r, null, 2));
|
|
28
|
+
process.exit(r.ok ? 0 : 1);
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
module.exports = verificarPlanCli;
|
|
32
|
+
|
|
33
|
+
if (require.main === module) {
|
|
34
|
+
const { parsearOpciones } = require('../lib/parsear-opciones');
|
|
35
|
+
verificarPlanCli(parsearOpciones(process.argv.slice(2)));
|
|
36
|
+
}
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Wrapper CLI para el subcomando `swl-ses verificar-trazabilidad`.
|
|
5
|
+
*
|
|
6
|
+
* Traduce las opciones parseadas del bin a las flags que espera el script
|
|
7
|
+
* standalone `scripts/verificar-trazabilidad.js` (que lee process.argv y hace
|
|
8
|
+
* su propio process.exit). Mismo patrón que scripts/cli/audit-claudemd.js.
|
|
9
|
+
*
|
|
10
|
+
* Reemplaza el `node scripts/verificar-trazabilidad.js` relativo al proyecto que
|
|
11
|
+
* /swl:aprobar-plan invocaba — roto downstream (ver
|
|
12
|
+
* docs/invocacion-cli-cross-scope.md).
|
|
13
|
+
*
|
|
14
|
+
* Uso: swl-ses verificar-trazabilidad --fase=N [--solo-plan] [--json]
|
|
15
|
+
*/
|
|
16
|
+
|
|
17
|
+
const { main } = require('../verificar-trazabilidad.js');
|
|
18
|
+
|
|
19
|
+
function verificarTrazabilidad(opciones = {}) {
|
|
20
|
+
opciones = opciones || {};
|
|
21
|
+
const args = [];
|
|
22
|
+
if (opciones.fase != null && opciones.fase !== true) args.push(`--fase=${opciones.fase}`);
|
|
23
|
+
if (opciones['solo-plan'] || opciones.solo_plan) args.push('--solo-plan');
|
|
24
|
+
if (opciones.json) args.push('--json');
|
|
25
|
+
|
|
26
|
+
const argvOriginal = process.argv;
|
|
27
|
+
process.argv = ['node', 'verificar-trazabilidad.js', ...args];
|
|
28
|
+
try {
|
|
29
|
+
main();
|
|
30
|
+
} finally {
|
|
31
|
+
process.argv = argvOriginal;
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
module.exports = verificarTrazabilidad;
|
|
@@ -0,0 +1,104 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* auditar-invocaciones-comandos.js
|
|
5
|
+
*
|
|
6
|
+
* Gate anti-regresión de la clase de bug "invocación relativa al proyecto en
|
|
7
|
+
* comandos user-facing" (ver docs/invocacion-cli-cross-scope.md).
|
|
8
|
+
*
|
|
9
|
+
* Un comando que corre en el proyecto del usuario NUNCA debe invocar scripts SWL
|
|
10
|
+
* con ruta relativa al proyecto (`node scripts/lib/...`, `node hooks/...`,
|
|
11
|
+
* `require('./scripts/...')`): esas rutas no existen downstream. Debe usar el
|
|
12
|
+
* subcomando del CLI (`swl-ses <sub>` / `npx ... <sub>` / `node scripts/cli/<sub>.js`).
|
|
13
|
+
*
|
|
14
|
+
* Solo escanea bloques de código (``` ... ```) para no marcar menciones
|
|
15
|
+
* prohibitivas en prosa. Excluye `scripts/cli/` (forma repo-madre sancionada).
|
|
16
|
+
*
|
|
17
|
+
* Función pura + entrypoint CLI con exit 1 si hay violaciones.
|
|
18
|
+
*/
|
|
19
|
+
|
|
20
|
+
const fs = require('node:fs');
|
|
21
|
+
const path = require('node:path');
|
|
22
|
+
|
|
23
|
+
// Comandos que corren SOLO en el repo madre de swl-ses (el CWD ES el repo, así
|
|
24
|
+
// que `node scripts/...` relativo es correcto). NO son user-facing.
|
|
25
|
+
const SELF_DEV = new Set([
|
|
26
|
+
'release',
|
|
27
|
+
'contribuir',
|
|
28
|
+
'evaluar-skill',
|
|
29
|
+
'reflect-skills',
|
|
30
|
+
]);
|
|
31
|
+
|
|
32
|
+
// Patrones de invocación relativa al proyecto que se rompen downstream.
|
|
33
|
+
// Se evalúan SOLO dentro de bloques de código.
|
|
34
|
+
const PATRONES = [
|
|
35
|
+
{ re: /\bnode\s+(?:\.\/)?scripts\/(?!cli\/)/, desc: 'node scripts/ (usa swl-ses <sub>)' },
|
|
36
|
+
{ re: /\bnode\s+(?:\.\/)?hooks\//, desc: 'node hooks/ (usa swl-ses <sub>)' },
|
|
37
|
+
{ re: /require\(\s*['"]\.\.?\/scripts\//, desc: "require('./scripts/ inline" },
|
|
38
|
+
{ re: /require\(\s*['"]\.\.?\/hooks\//, desc: "require('./hooks/ inline" },
|
|
39
|
+
];
|
|
40
|
+
|
|
41
|
+
/**
|
|
42
|
+
* @param {string} [comandosDir] default: comandos/swl relativo al repo de este script.
|
|
43
|
+
* @returns {{ok:boolean, violaciones:Array<{comando:string,linea:number,texto:string,patron:string}>, escaneados:number}}
|
|
44
|
+
*/
|
|
45
|
+
function auditarInvocacionesComandos(comandosDir) {
|
|
46
|
+
const dir = comandosDir || path.join(__dirname, '..', '..', 'comandos', 'swl');
|
|
47
|
+
const violaciones = [];
|
|
48
|
+
let escaneados = 0;
|
|
49
|
+
|
|
50
|
+
let archivos;
|
|
51
|
+
try {
|
|
52
|
+
archivos = fs.readdirSync(dir).filter((f) => f.endsWith('.md') && !f.startsWith('_'));
|
|
53
|
+
} catch {
|
|
54
|
+
return { ok: true, violaciones, escaneados: 0 };
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
for (const archivo of archivos) {
|
|
58
|
+
const comando = archivo.replace(/\.md$/, '');
|
|
59
|
+
if (SELF_DEV.has(comando)) continue;
|
|
60
|
+
escaneados++;
|
|
61
|
+
|
|
62
|
+
const contenido = fs.readFileSync(path.join(dir, archivo), 'utf8');
|
|
63
|
+
const lineas = contenido.split('\n');
|
|
64
|
+
let dentroCodigo = false;
|
|
65
|
+
|
|
66
|
+
for (let i = 0; i < lineas.length; i++) {
|
|
67
|
+
const linea = lineas[i];
|
|
68
|
+
if (/^\s*```/.test(linea)) {
|
|
69
|
+
dentroCodigo = !dentroCodigo;
|
|
70
|
+
continue;
|
|
71
|
+
}
|
|
72
|
+
if (!dentroCodigo) continue;
|
|
73
|
+
for (const { re, desc } of PATRONES) {
|
|
74
|
+
if (re.test(linea)) {
|
|
75
|
+
violaciones.push({
|
|
76
|
+
comando,
|
|
77
|
+
linea: i + 1,
|
|
78
|
+
texto: linea.trim(),
|
|
79
|
+
patron: desc,
|
|
80
|
+
});
|
|
81
|
+
break;
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
return { ok: violaciones.length === 0, violaciones, escaneados };
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
module.exports = { auditarInvocacionesComandos, SELF_DEV };
|
|
91
|
+
|
|
92
|
+
if (require.main === module) {
|
|
93
|
+
const r = auditarInvocacionesComandos();
|
|
94
|
+
if (r.ok) {
|
|
95
|
+
console.log(`[OK] invocaciones-comandos — ${r.escaneados} comandos user-facing sin rutas relativas al proyecto`);
|
|
96
|
+
process.exit(0);
|
|
97
|
+
}
|
|
98
|
+
console.error(`[FALLA] invocaciones-comandos — ${r.violaciones.length} invocación(es) relativa(s) al proyecto en comandos user-facing:`);
|
|
99
|
+
for (const v of r.violaciones) {
|
|
100
|
+
console.error(` ${v.comando}.md:${v.linea} [${v.patron}] ${v.texto}`);
|
|
101
|
+
}
|
|
102
|
+
console.error('\nUsa el patrón cross-scope (docs/invocacion-cli-cross-scope.md): swl-ses <sub> / npx ... <sub> / node scripts/cli/<sub>.js');
|
|
103
|
+
process.exit(1);
|
|
104
|
+
}
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* resolver-plan-fase.js
|
|
5
|
+
*
|
|
6
|
+
* Resuelve la ruta del PLAN.md de una fase a partir de las opciones del CLI.
|
|
7
|
+
* Compartido por los subcomandos `aprobar-plan` y `verificar-plan`.
|
|
8
|
+
*
|
|
9
|
+
* Acepta `--plan=<ruta>` explícito o `--fase=N` (deriva
|
|
10
|
+
* `.planning/fases/0N-PLAN.md`, con fallback sin cero a la izquierda).
|
|
11
|
+
*/
|
|
12
|
+
|
|
13
|
+
const fs = require('node:fs');
|
|
14
|
+
const path = require('node:path');
|
|
15
|
+
|
|
16
|
+
/**
|
|
17
|
+
* @param {{plan?:string, fase?:(string|number|boolean)}} [opciones]
|
|
18
|
+
* @returns {string|null} ruta al PLAN, o null si no se pudo derivar.
|
|
19
|
+
*/
|
|
20
|
+
function resolverPlanPath(opciones = {}) {
|
|
21
|
+
if (opciones && opciones.plan && typeof opciones.plan === 'string') {
|
|
22
|
+
return opciones.plan;
|
|
23
|
+
}
|
|
24
|
+
const fase = opciones ? opciones.fase : undefined;
|
|
25
|
+
if (fase == null || fase === true) return null;
|
|
26
|
+
const n = String(fase).padStart(2, '0');
|
|
27
|
+
const candidatos = [
|
|
28
|
+
path.join('.planning', 'fases', `${n}-PLAN.md`),
|
|
29
|
+
path.join('.planning', 'fases', `${fase}-PLAN.md`),
|
|
30
|
+
];
|
|
31
|
+
for (const c of candidatos) {
|
|
32
|
+
if (fs.existsSync(c)) return c;
|
|
33
|
+
}
|
|
34
|
+
return candidatos[0];
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
module.exports = { resolverPlanPath };
|