@saulwade/swl-ses 1.4.1 → 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.
Files changed (114) hide show
  1. package/CLAUDE.md +1 -1
  2. package/README.md +1 -1
  3. package/agentes/nemesis-auditor-swl.md +161 -161
  4. package/bin/swl-mcp-server.js +187 -187
  5. package/comandos/swl/.evolved.json +22 -22
  6. package/comandos/swl/contribuir.md +233 -233
  7. package/comandos/swl/nemesis.md +122 -122
  8. package/gateway/lib/event-channel.js +191 -191
  9. package/habilidades/backend-production-resilience/SKILL.md +288 -288
  10. package/habilidades/benchmark-memoria/SKILL.md +186 -186
  11. package/habilidades/diagrama-arquitectura/assets/template.html +276 -276
  12. package/habilidades/doubt-driven-review/SKILL.md +171 -171
  13. package/habilidades/doubt-driven-review/recursos/EXAMPLES.md +130 -130
  14. package/habilidades/eval-framework/SKILL.md +212 -212
  15. package/habilidades/feynman-auditor-swl/SKILL.md +123 -123
  16. package/habilidades/feynman-auditor-swl/recursos/preguntas-language-agnostic.md +108 -108
  17. package/habilidades/harness-claude-code/SKILL.md +299 -299
  18. package/habilidades/infra-github-actions/SKILL.md +166 -166
  19. package/habilidades/legacy-code-rescue/SKILL.md +267 -267
  20. package/habilidades/manejo-errores/.evolved.json +8 -8
  21. package/habilidades/meta-skills-estandar/recursos/convencion-examples.md +93 -93
  22. package/habilidades/meta-skills-estandar/recursos/skills-as-agents.md +163 -163
  23. package/habilidades/patrones-python/SKILL.md +229 -229
  24. package/habilidades/patrones-python/recursos/patrones-avanzados.md +469 -469
  25. package/habilidades/planear-fase/SKILL.md +319 -319
  26. package/habilidades/release-semver/.evolved.json +8 -8
  27. package/habilidades/state-inconsistency-auditor-swl/SKILL.md +166 -166
  28. package/habilidades/state-inconsistency-auditor-swl/recursos/coupled-state-patterns.md +147 -147
  29. package/habilidades/testing-python/SKILL.md +340 -340
  30. package/habilidades/web-fetcher-routing/SKILL.md +75 -75
  31. package/hooks/claudemd-bloat-detector.js +161 -161
  32. package/hooks/lib/agent-routing.js +107 -107
  33. package/hooks/lib/auto-consolidator.js +335 -335
  34. package/hooks/lib/error-classifier.js +308 -308
  35. package/hooks/lib/merkle-audit.js +96 -96
  36. package/hooks/lib/provenance-tracker.js +191 -191
  37. package/hooks/lib/rate-limit-tracker.js +253 -253
  38. package/hooks/lib/resource-quota.js +122 -122
  39. package/hooks/lib/retry-jitter.js +165 -165
  40. package/hooks/lib/security-net.js +201 -201
  41. package/hooks/lib/skill-auditor.js +588 -588
  42. package/hooks/lib/sync-status.js +228 -228
  43. package/hooks/lib/taint-tracker.js +107 -107
  44. package/hooks/lib/text-similarity.js +241 -241
  45. package/hooks/lib/toon-compressor.js +245 -245
  46. package/hooks/registro-turnos.js +209 -209
  47. package/hooks/sugerir-regenerar-inventario.js +170 -170
  48. package/hooks/validar-formato-post-subagente.js +140 -140
  49. package/hooks/validar-memoria-hook.js +218 -218
  50. package/instintos/prompt-appendices.yaml +57 -57
  51. package/manifiestos/agent-output-schemas.json +57 -57
  52. package/manifiestos/modulos.json +11 -6
  53. package/manifiestos/perfiles.json +2 -1
  54. package/manifiestos/skills-lock.json +1114 -1114
  55. package/package.json +1 -1
  56. package/plantillas/auditor-veto-template.md +105 -105
  57. package/plantillas/github-workflows/README.md +47 -47
  58. package/plantillas/github-workflows/release-please.yml +44 -44
  59. package/plantillas/github-workflows/swl-ci.yml +107 -107
  60. package/plantillas/github-workflows/swl-security.yml +51 -51
  61. package/plugin.json +9 -1
  62. package/reglas/analisis-previo-tareas-grandes.md +172 -172
  63. package/reglas/arreglar-al-detectar.md +147 -147
  64. package/reglas/fragmentos-compartidos.md +152 -152
  65. package/reglas/harness-claude-code.md +213 -213
  66. package/reglas/usar-context7.md +226 -226
  67. package/schemas/diary-entry.schema.json +80 -80
  68. package/scripts/audit-tools/audit-history.js +330 -330
  69. package/scripts/audit-tools/bundle-tracker.js +290 -290
  70. package/scripts/audit-tools/canary-monitor.js +352 -352
  71. package/scripts/audit-tools/code-profiler.js +605 -605
  72. package/scripts/audit-tools/dep-doctor.js +320 -320
  73. package/scripts/audit-tools/env-validator.js +206 -206
  74. package/scripts/audit-tools/lib/fs-walk.js +48 -48
  75. package/scripts/audit-tools/lib/output.js +23 -23
  76. package/scripts/audit-tools/migration-checker.js +392 -392
  77. package/scripts/audit-tools/pentest-scanner.js +1436 -1436
  78. package/scripts/benchmark-memoria.js +167 -167
  79. package/scripts/configurar-branch-protection.js +418 -418
  80. package/scripts/detectar-aprendizajes-duplicados.js +151 -151
  81. package/scripts/field-report.js +199 -199
  82. package/scripts/generar-checklists-consolidados.js +273 -273
  83. package/scripts/generar-inventario.js +420 -420
  84. package/scripts/generar-matriz-lenguajes.js +271 -271
  85. package/scripts/lib/artefactos-python.js +43 -43
  86. package/scripts/lib/benchmark-metrics.js +160 -160
  87. package/scripts/lib/budget-enforcer.js +252 -252
  88. package/scripts/lib/configurar-ci.js +380 -380
  89. package/scripts/lib/contadores-inventario.js +217 -217
  90. package/scripts/lib/detectar-stack-detallado.js +307 -307
  91. package/scripts/lib/diary-entry.js +234 -234
  92. package/scripts/lib/eval-metrics-store.js +218 -218
  93. package/scripts/lib/eval-quality.js +171 -171
  94. package/scripts/lib/eval-schemas.js +144 -144
  95. package/scripts/lib/eval-self-correct.js +106 -106
  96. package/scripts/lib/eval-validator.js +185 -185
  97. package/scripts/lib/jaccard-similarity.js +98 -98
  98. package/scripts/lib/longmemeval-runner.js +125 -125
  99. package/scripts/lib/manifiestos.js +42 -1
  100. package/scripts/lib/npm-version.js +261 -261
  101. package/scripts/lib/paquetes-conocidos.js +50 -50
  102. package/scripts/lib/prompt-builder.js +264 -264
  103. package/scripts/lib/rrf-fusion.js +175 -175
  104. package/scripts/lib/scoring-instintos.js +277 -277
  105. package/scripts/lib/semantic-search.js +252 -252
  106. package/scripts/limpiar-artefactos-python.js +131 -131
  107. package/scripts/mcp-server/README.md +128 -128
  108. package/scripts/mcp-server/handlers.js +206 -206
  109. package/scripts/migrar-csv-a-array.js +168 -168
  110. package/scripts/migrar-fase-dominio.js +201 -201
  111. package/scripts/publicar.js +511 -511
  112. package/scripts/run-eval.js +141 -141
  113. package/scripts/validar-manifest.js +231 -195
  114. package/scripts/validar-userland-vacio.js +110 -110
@@ -1,290 +1,290 @@
1
- // Adaptado de temp/ultraship-main/tools/bundle-tracker.mjs bajo MIT License
2
- // Fuente: Houseofmvps/ultraship (https://github.com/Houseofmvps/ultraship)
3
- 'use strict';
4
-
5
- const { readFileSync, appendFileSync, existsSync, readdirSync, statSync, mkdirSync } = require('fs');
6
- const { join, relative, extname, resolve } = require('path');
7
- const { outputJSON, outputError } = require('./lib/output');
8
-
9
- /**
10
- * Formatea bytes a unidad legible.
11
- * @param {number} bytes
12
- * @returns {string}
13
- */
14
- function formatSize(bytes) {
15
- if (bytes < 1024) return `${bytes}B`;
16
- if (bytes < 1024 * 1024) return `${Math.round(bytes / 1024 * 10) / 10}KB`;
17
- return `${Math.round(bytes / (1024 * 1024) * 10) / 10}MB`;
18
- }
19
-
20
- /**
21
- * Detecta el directorio de build de un proyecto.
22
- * @param {string} dir
23
- * @returns {{ dir: string, path: string } | null}
24
- */
25
- function findBuildOutput(dir) {
26
- const candidates = ['dist', 'build', '.next', 'out', '.output', '.vercel/output'];
27
- for (const d of candidates) {
28
- const p = join(dir, d);
29
- try {
30
- if (existsSync(p) && statSync(p).isDirectory()) return { dir: d, path: p };
31
- } catch { /* skip */ }
32
- }
33
- return null;
34
- }
35
-
36
- /**
37
- * Recorre recursivamente un directorio y devuelve archivos con las extensiones indicadas.
38
- * @param {string} dir
39
- * @param {string[]} extensions
40
- * @returns {{ path: string, size: number }[]}
41
- */
42
- function walkFiles(dir, extensions) {
43
- const extSet = new Set(extensions.map(e => e.toLowerCase()));
44
- const files = [];
45
-
46
- function walk(d) {
47
- try {
48
- for (const entry of readdirSync(d)) {
49
- if (entry.startsWith('.')) continue;
50
- const p = join(d, entry);
51
- try {
52
- const s = statSync(p);
53
- if (s.isDirectory()) walk(p);
54
- else if (extSet.has(extname(entry).toLowerCase())) {
55
- files.push({ path: p, size: s.size });
56
- }
57
- } catch { /* skip */ }
58
- }
59
- } catch { /* skip */ }
60
- }
61
-
62
- walk(dir);
63
- return files;
64
- }
65
-
66
- /** Dependencias pesadas conocidas con estimación de tamaño y alternativas. */
67
- const HEAVY_DEPS = {
68
- 'moment': { size: '300KB+', alternative: 'dayjs (2KB)' },
69
- 'lodash': { size: '70KB+', alternative: 'lodash-es o imports individuales' },
70
- 'jquery': { size: '90KB+', alternative: 'APIs DOM nativas' },
71
- 'axios': { size: '30KB+', alternative: 'fetch nativo' },
72
- 'underscore': { size: '30KB+', alternative: 'métodos nativos de Array' },
73
- 'core-js': { size: '150KB+', alternative: 'polyfills específicos únicamente' },
74
- 'date-fns': { size: '75KB+ (completo)', alternative: 'importar solo funciones necesarias' },
75
- 'validator': { size: '50KB+', alternative: 'zod o comprobaciones individuales' },
76
- 'bluebird': { size: '80KB+', alternative: 'Promises nativas' },
77
- 'request': { size: '50KB+', alternative: 'fetch nativo o undici' },
78
- };
79
-
80
- /**
81
- * Analiza las dependencias del package.json en un directorio.
82
- * @param {string} dir
83
- * @returns {{ production_deps: number, dev_deps: number, heavy_deps: object[] } | null}
84
- */
85
- function analyzeDependencies(dir) {
86
- const pkgPath = join(dir, 'package.json');
87
- if (!existsSync(pkgPath)) return null;
88
-
89
- let pkg;
90
- try {
91
- pkg = JSON.parse(readFileSync(pkgPath, 'utf8'));
92
- } catch {
93
- return null;
94
- }
95
-
96
- const deps = Object.keys(pkg.dependencies || {});
97
- const devDeps = Object.keys(pkg.devDependencies || {});
98
-
99
- const heavyFound = [];
100
- for (const dep of deps) {
101
- if (HEAVY_DEPS[dep]) {
102
- heavyFound.push({
103
- dependency: dep,
104
- estimated_size: HEAVY_DEPS[dep].size,
105
- alternative: HEAVY_DEPS[dep].alternative,
106
- severity: 'medium',
107
- });
108
- }
109
- }
110
-
111
- return {
112
- production_deps: deps.length,
113
- dev_deps: devDeps.length,
114
- heavy_deps: heavyFound,
115
- };
116
- }
117
-
118
- /**
119
- * Lee la última entrada del historial JSONL.
120
- * @param {string} historyPath
121
- * @returns {object | null}
122
- */
123
- function readLastHistoryEntry(historyPath) {
124
- if (!existsSync(historyPath)) return null;
125
- try {
126
- const lines = readFileSync(historyPath, 'utf8')
127
- .split('\n')
128
- .filter(Boolean);
129
- if (lines.length === 0) return null;
130
- return JSON.parse(lines[lines.length - 1]);
131
- } catch {
132
- return null;
133
- }
134
- }
135
-
136
- /**
137
- * Agrega una entrada al historial JSONL.
138
- * @param {string} historyPath
139
- * @param {object} entry
140
- */
141
- function appendHistoryEntry(historyPath, entry) {
142
- const dir = require('path').dirname(historyPath);
143
- mkdirSync(dir, { recursive: true, mode: 0o700 });
144
- appendFileSync(historyPath, JSON.stringify(entry) + '\n', 'utf8');
145
- }
146
-
147
- function main() {
148
- const args = process.argv.slice(2);
149
- const dir = resolve(args.find(a => !a.startsWith('--')) || process.cwd());
150
- const shouldSave = args.includes('--save');
151
-
152
- if (!existsSync(dir)) {
153
- outputError(`Directorio no encontrado: ${dir}`);
154
- process.exit(0);
155
- }
156
-
157
- // Historial JSONL dentro del proyecto analizado
158
- const historyPath = join(dir, '.planning', 'bundles', 'history.jsonl');
159
-
160
- const findings = [];
161
-
162
- // ── Análisis de bundle ─────────────────────────────────────────────────────
163
- const buildOutput = findBuildOutput(dir);
164
- let bundleAnalysis = null;
165
-
166
- if (buildOutput) {
167
- const jsFiles = walkFiles(buildOutput.path, ['.js', '.mjs', '.cjs']);
168
- const cssFiles = walkFiles(buildOutput.path, ['.css']);
169
- const imageFiles = walkFiles(buildOutput.path, ['.png', '.jpg', '.jpeg', '.gif', '.svg', '.webp', '.avif']);
170
- const allFiles = walkFiles(buildOutput.path, [
171
- '.js', '.mjs', '.cjs', '.css', '.html', '.htm',
172
- '.json', '.map', '.png', '.jpg', '.jpeg', '.gif',
173
- '.svg', '.webp', '.avif',
174
- ]);
175
-
176
- const totalJsSize = jsFiles.reduce((s, f) => s + f.size, 0);
177
- const totalCssSize = cssFiles.reduce((s, f) => s + f.size, 0);
178
- const totalImageSize = imageFiles.reduce((s, f) => s + f.size, 0);
179
- const totalSize = allFiles.reduce((s, f) => s + f.size, 0);
180
-
181
- const sourceMaps = allFiles.filter(f => f.path.endsWith('.map'));
182
- const sourceMapSize = sourceMaps.reduce((s, f) => s + f.size, 0);
183
-
184
- const largestJs = jsFiles
185
- .sort((a, b) => b.size - a.size)
186
- .slice(0, 10)
187
- .map(f => ({ file: relative(dir, f.path), size: formatSize(f.size), bytes: f.size }));
188
-
189
- bundleAnalysis = {
190
- build_dir: buildOutput.dir,
191
- total_size: formatSize(totalSize),
192
- total_bytes: totalSize,
193
- js: { files: jsFiles.length, size: formatSize(totalJsSize), bytes: totalJsSize },
194
- css: { files: cssFiles.length, size: formatSize(totalCssSize), bytes: totalCssSize },
195
- images: { files: imageFiles.length, size: formatSize(totalImageSize), bytes: totalImageSize },
196
- source_maps: { files: sourceMaps.length, size: formatSize(sourceMapSize), bytes: sourceMapSize },
197
- largest_js_files: largestJs,
198
- };
199
-
200
- // Advertencias de tamaño
201
- if (totalJsSize > 500 * 1024) {
202
- findings.push({ severity: 'high', message: `Bundle JS total es ${formatSize(totalJsSize)} — objetivo: menos de 500 KB para buen rendimiento` });
203
- } else if (totalJsSize > 250 * 1024) {
204
- findings.push({ severity: 'medium', message: `Bundle JS total es ${formatSize(totalJsSize)} — considera code splitting para bundles mayores de 250 KB` });
205
- }
206
-
207
- for (const f of largestJs) {
208
- if (f.bytes > 200 * 1024) {
209
- findings.push({ severity: 'high', message: `${f.file} pesa ${f.size} — dividir en fragmentos más pequeños con dynamic imports` });
210
- }
211
- }
212
-
213
- if (sourceMaps.length > 0) {
214
- findings.push({ severity: 'low', message: `${sourceMaps.length} archivos de source map (${formatSize(sourceMapSize)}) en el build — considera excluirlos del despliegue` });
215
- }
216
-
217
- const largeImages = imageFiles.filter(f => f.size > 200 * 1024);
218
- if (largeImages.length > 0) {
219
- findings.push({ severity: 'medium', message: `${largeImages.length} imágenes mayores de 200 KB en el build — optimizar con conversión WebP/AVIF` });
220
- }
221
- } else {
222
- findings.push({ severity: 'info', message: 'No se encontró directorio de build (dist/, build/, .next/, out/) — ejecuta el comando de build primero' });
223
- }
224
-
225
- // ── Análisis de dependencias ───────────────────────────────────────────────
226
- const depAnalysis = analyzeDependencies(dir);
227
- if (depAnalysis && depAnalysis.heavy_deps.length > 0) {
228
- for (const dep of depAnalysis.heavy_deps) {
229
- findings.push({
230
- severity: dep.severity,
231
- message: `Dependencia pesada: ${dep.dependency} (~${dep.estimated_size}) — considera ${dep.alternative}`,
232
- });
233
- }
234
- }
235
-
236
- // ── Comparación con historial ──────────────────────────────────────────────
237
- let comparison = null;
238
-
239
- if (shouldSave) {
240
- const prev = readLastHistoryEntry(historyPath);
241
-
242
- if (prev && prev.bundle && bundleAnalysis) {
243
- const diff = bundleAnalysis.total_bytes - prev.bundle.total_bytes;
244
- comparison = {
245
- previous_size: prev.bundle.total_size,
246
- current_size: bundleAnalysis.total_size,
247
- diff_bytes: diff,
248
- diff_formatted: (diff >= 0 ? '+' : '') + formatSize(Math.abs(diff)),
249
- grew: diff > 0,
250
- previous_date: prev.timestamp,
251
- };
252
-
253
- if (diff > 50 * 1024) {
254
- findings.push({ severity: 'high', message: `Bundle creció ${formatSize(diff)} desde la última revisión — investiga nuevas dependencias o tree-shaking ausente` });
255
- } else if (diff > 10 * 1024) {
256
- findings.push({ severity: 'medium', message: `Bundle creció ${formatSize(diff)} desde la última revisión` });
257
- }
258
- }
259
-
260
- // Persistir entrada en JSONL
261
- const entry = {
262
- timestamp: new Date().toISOString(),
263
- bundle: bundleAnalysis,
264
- dependencies: depAnalysis,
265
- findings_count: findings.length,
266
- };
267
- try {
268
- appendHistoryEntry(historyPath, entry);
269
- } catch (err) {
270
- outputError('Error al guardar historial de bundle', { message: err.message });
271
- }
272
- }
273
-
274
- outputJSON({
275
- success: true,
276
- packages_scanned: 1,
277
- bundle: bundleAnalysis,
278
- dependencies: depAnalysis,
279
- comparison,
280
- findings,
281
- report_saved: shouldSave,
282
- history_path: shouldSave ? historyPath : null,
283
- });
284
- }
285
-
286
- if (require.main === module) {
287
- main();
288
- }
289
-
290
- module.exports = { findBuildOutput, walkFiles, analyzeDependencies, formatSize, HEAVY_DEPS };
1
+ // Adaptado de temp/ultraship-main/tools/bundle-tracker.mjs bajo MIT License
2
+ // Fuente: Houseofmvps/ultraship (https://github.com/Houseofmvps/ultraship)
3
+ 'use strict';
4
+
5
+ const { readFileSync, appendFileSync, existsSync, readdirSync, statSync, mkdirSync } = require('fs');
6
+ const { join, relative, extname, resolve } = require('path');
7
+ const { outputJSON, outputError } = require('./lib/output');
8
+
9
+ /**
10
+ * Formatea bytes a unidad legible.
11
+ * @param {number} bytes
12
+ * @returns {string}
13
+ */
14
+ function formatSize(bytes) {
15
+ if (bytes < 1024) return `${bytes}B`;
16
+ if (bytes < 1024 * 1024) return `${Math.round(bytes / 1024 * 10) / 10}KB`;
17
+ return `${Math.round(bytes / (1024 * 1024) * 10) / 10}MB`;
18
+ }
19
+
20
+ /**
21
+ * Detecta el directorio de build de un proyecto.
22
+ * @param {string} dir
23
+ * @returns {{ dir: string, path: string } | null}
24
+ */
25
+ function findBuildOutput(dir) {
26
+ const candidates = ['dist', 'build', '.next', 'out', '.output', '.vercel/output'];
27
+ for (const d of candidates) {
28
+ const p = join(dir, d);
29
+ try {
30
+ if (existsSync(p) && statSync(p).isDirectory()) return { dir: d, path: p };
31
+ } catch { /* skip */ }
32
+ }
33
+ return null;
34
+ }
35
+
36
+ /**
37
+ * Recorre recursivamente un directorio y devuelve archivos con las extensiones indicadas.
38
+ * @param {string} dir
39
+ * @param {string[]} extensions
40
+ * @returns {{ path: string, size: number }[]}
41
+ */
42
+ function walkFiles(dir, extensions) {
43
+ const extSet = new Set(extensions.map(e => e.toLowerCase()));
44
+ const files = [];
45
+
46
+ function walk(d) {
47
+ try {
48
+ for (const entry of readdirSync(d)) {
49
+ if (entry.startsWith('.')) continue;
50
+ const p = join(d, entry);
51
+ try {
52
+ const s = statSync(p);
53
+ if (s.isDirectory()) walk(p);
54
+ else if (extSet.has(extname(entry).toLowerCase())) {
55
+ files.push({ path: p, size: s.size });
56
+ }
57
+ } catch { /* skip */ }
58
+ }
59
+ } catch { /* skip */ }
60
+ }
61
+
62
+ walk(dir);
63
+ return files;
64
+ }
65
+
66
+ /** Dependencias pesadas conocidas con estimación de tamaño y alternativas. */
67
+ const HEAVY_DEPS = {
68
+ 'moment': { size: '300KB+', alternative: 'dayjs (2KB)' },
69
+ 'lodash': { size: '70KB+', alternative: 'lodash-es o imports individuales' },
70
+ 'jquery': { size: '90KB+', alternative: 'APIs DOM nativas' },
71
+ 'axios': { size: '30KB+', alternative: 'fetch nativo' },
72
+ 'underscore': { size: '30KB+', alternative: 'métodos nativos de Array' },
73
+ 'core-js': { size: '150KB+', alternative: 'polyfills específicos únicamente' },
74
+ 'date-fns': { size: '75KB+ (completo)', alternative: 'importar solo funciones necesarias' },
75
+ 'validator': { size: '50KB+', alternative: 'zod o comprobaciones individuales' },
76
+ 'bluebird': { size: '80KB+', alternative: 'Promises nativas' },
77
+ 'request': { size: '50KB+', alternative: 'fetch nativo o undici' },
78
+ };
79
+
80
+ /**
81
+ * Analiza las dependencias del package.json en un directorio.
82
+ * @param {string} dir
83
+ * @returns {{ production_deps: number, dev_deps: number, heavy_deps: object[] } | null}
84
+ */
85
+ function analyzeDependencies(dir) {
86
+ const pkgPath = join(dir, 'package.json');
87
+ if (!existsSync(pkgPath)) return null;
88
+
89
+ let pkg;
90
+ try {
91
+ pkg = JSON.parse(readFileSync(pkgPath, 'utf8'));
92
+ } catch {
93
+ return null;
94
+ }
95
+
96
+ const deps = Object.keys(pkg.dependencies || {});
97
+ const devDeps = Object.keys(pkg.devDependencies || {});
98
+
99
+ const heavyFound = [];
100
+ for (const dep of deps) {
101
+ if (HEAVY_DEPS[dep]) {
102
+ heavyFound.push({
103
+ dependency: dep,
104
+ estimated_size: HEAVY_DEPS[dep].size,
105
+ alternative: HEAVY_DEPS[dep].alternative,
106
+ severity: 'medium',
107
+ });
108
+ }
109
+ }
110
+
111
+ return {
112
+ production_deps: deps.length,
113
+ dev_deps: devDeps.length,
114
+ heavy_deps: heavyFound,
115
+ };
116
+ }
117
+
118
+ /**
119
+ * Lee la última entrada del historial JSONL.
120
+ * @param {string} historyPath
121
+ * @returns {object | null}
122
+ */
123
+ function readLastHistoryEntry(historyPath) {
124
+ if (!existsSync(historyPath)) return null;
125
+ try {
126
+ const lines = readFileSync(historyPath, 'utf8')
127
+ .split('\n')
128
+ .filter(Boolean);
129
+ if (lines.length === 0) return null;
130
+ return JSON.parse(lines[lines.length - 1]);
131
+ } catch {
132
+ return null;
133
+ }
134
+ }
135
+
136
+ /**
137
+ * Agrega una entrada al historial JSONL.
138
+ * @param {string} historyPath
139
+ * @param {object} entry
140
+ */
141
+ function appendHistoryEntry(historyPath, entry) {
142
+ const dir = require('path').dirname(historyPath);
143
+ mkdirSync(dir, { recursive: true, mode: 0o700 });
144
+ appendFileSync(historyPath, JSON.stringify(entry) + '\n', 'utf8');
145
+ }
146
+
147
+ function main() {
148
+ const args = process.argv.slice(2);
149
+ const dir = resolve(args.find(a => !a.startsWith('--')) || process.cwd());
150
+ const shouldSave = args.includes('--save');
151
+
152
+ if (!existsSync(dir)) {
153
+ outputError(`Directorio no encontrado: ${dir}`);
154
+ process.exit(0);
155
+ }
156
+
157
+ // Historial JSONL dentro del proyecto analizado
158
+ const historyPath = join(dir, '.planning', 'bundles', 'history.jsonl');
159
+
160
+ const findings = [];
161
+
162
+ // ── Análisis de bundle ─────────────────────────────────────────────────────
163
+ const buildOutput = findBuildOutput(dir);
164
+ let bundleAnalysis = null;
165
+
166
+ if (buildOutput) {
167
+ const jsFiles = walkFiles(buildOutput.path, ['.js', '.mjs', '.cjs']);
168
+ const cssFiles = walkFiles(buildOutput.path, ['.css']);
169
+ const imageFiles = walkFiles(buildOutput.path, ['.png', '.jpg', '.jpeg', '.gif', '.svg', '.webp', '.avif']);
170
+ const allFiles = walkFiles(buildOutput.path, [
171
+ '.js', '.mjs', '.cjs', '.css', '.html', '.htm',
172
+ '.json', '.map', '.png', '.jpg', '.jpeg', '.gif',
173
+ '.svg', '.webp', '.avif',
174
+ ]);
175
+
176
+ const totalJsSize = jsFiles.reduce((s, f) => s + f.size, 0);
177
+ const totalCssSize = cssFiles.reduce((s, f) => s + f.size, 0);
178
+ const totalImageSize = imageFiles.reduce((s, f) => s + f.size, 0);
179
+ const totalSize = allFiles.reduce((s, f) => s + f.size, 0);
180
+
181
+ const sourceMaps = allFiles.filter(f => f.path.endsWith('.map'));
182
+ const sourceMapSize = sourceMaps.reduce((s, f) => s + f.size, 0);
183
+
184
+ const largestJs = jsFiles
185
+ .sort((a, b) => b.size - a.size)
186
+ .slice(0, 10)
187
+ .map(f => ({ file: relative(dir, f.path), size: formatSize(f.size), bytes: f.size }));
188
+
189
+ bundleAnalysis = {
190
+ build_dir: buildOutput.dir,
191
+ total_size: formatSize(totalSize),
192
+ total_bytes: totalSize,
193
+ js: { files: jsFiles.length, size: formatSize(totalJsSize), bytes: totalJsSize },
194
+ css: { files: cssFiles.length, size: formatSize(totalCssSize), bytes: totalCssSize },
195
+ images: { files: imageFiles.length, size: formatSize(totalImageSize), bytes: totalImageSize },
196
+ source_maps: { files: sourceMaps.length, size: formatSize(sourceMapSize), bytes: sourceMapSize },
197
+ largest_js_files: largestJs,
198
+ };
199
+
200
+ // Advertencias de tamaño
201
+ if (totalJsSize > 500 * 1024) {
202
+ findings.push({ severity: 'high', message: `Bundle JS total es ${formatSize(totalJsSize)} — objetivo: menos de 500 KB para buen rendimiento` });
203
+ } else if (totalJsSize > 250 * 1024) {
204
+ findings.push({ severity: 'medium', message: `Bundle JS total es ${formatSize(totalJsSize)} — considera code splitting para bundles mayores de 250 KB` });
205
+ }
206
+
207
+ for (const f of largestJs) {
208
+ if (f.bytes > 200 * 1024) {
209
+ findings.push({ severity: 'high', message: `${f.file} pesa ${f.size} — dividir en fragmentos más pequeños con dynamic imports` });
210
+ }
211
+ }
212
+
213
+ if (sourceMaps.length > 0) {
214
+ findings.push({ severity: 'low', message: `${sourceMaps.length} archivos de source map (${formatSize(sourceMapSize)}) en el build — considera excluirlos del despliegue` });
215
+ }
216
+
217
+ const largeImages = imageFiles.filter(f => f.size > 200 * 1024);
218
+ if (largeImages.length > 0) {
219
+ findings.push({ severity: 'medium', message: `${largeImages.length} imágenes mayores de 200 KB en el build — optimizar con conversión WebP/AVIF` });
220
+ }
221
+ } else {
222
+ findings.push({ severity: 'info', message: 'No se encontró directorio de build (dist/, build/, .next/, out/) — ejecuta el comando de build primero' });
223
+ }
224
+
225
+ // ── Análisis de dependencias ───────────────────────────────────────────────
226
+ const depAnalysis = analyzeDependencies(dir);
227
+ if (depAnalysis && depAnalysis.heavy_deps.length > 0) {
228
+ for (const dep of depAnalysis.heavy_deps) {
229
+ findings.push({
230
+ severity: dep.severity,
231
+ message: `Dependencia pesada: ${dep.dependency} (~${dep.estimated_size}) — considera ${dep.alternative}`,
232
+ });
233
+ }
234
+ }
235
+
236
+ // ── Comparación con historial ──────────────────────────────────────────────
237
+ let comparison = null;
238
+
239
+ if (shouldSave) {
240
+ const prev = readLastHistoryEntry(historyPath);
241
+
242
+ if (prev && prev.bundle && bundleAnalysis) {
243
+ const diff = bundleAnalysis.total_bytes - prev.bundle.total_bytes;
244
+ comparison = {
245
+ previous_size: prev.bundle.total_size,
246
+ current_size: bundleAnalysis.total_size,
247
+ diff_bytes: diff,
248
+ diff_formatted: (diff >= 0 ? '+' : '') + formatSize(Math.abs(diff)),
249
+ grew: diff > 0,
250
+ previous_date: prev.timestamp,
251
+ };
252
+
253
+ if (diff > 50 * 1024) {
254
+ findings.push({ severity: 'high', message: `Bundle creció ${formatSize(diff)} desde la última revisión — investiga nuevas dependencias o tree-shaking ausente` });
255
+ } else if (diff > 10 * 1024) {
256
+ findings.push({ severity: 'medium', message: `Bundle creció ${formatSize(diff)} desde la última revisión` });
257
+ }
258
+ }
259
+
260
+ // Persistir entrada en JSONL
261
+ const entry = {
262
+ timestamp: new Date().toISOString(),
263
+ bundle: bundleAnalysis,
264
+ dependencies: depAnalysis,
265
+ findings_count: findings.length,
266
+ };
267
+ try {
268
+ appendHistoryEntry(historyPath, entry);
269
+ } catch (err) {
270
+ outputError('Error al guardar historial de bundle', { message: err.message });
271
+ }
272
+ }
273
+
274
+ outputJSON({
275
+ success: true,
276
+ packages_scanned: 1,
277
+ bundle: bundleAnalysis,
278
+ dependencies: depAnalysis,
279
+ comparison,
280
+ findings,
281
+ report_saved: shouldSave,
282
+ history_path: shouldSave ? historyPath : null,
283
+ });
284
+ }
285
+
286
+ if (require.main === module) {
287
+ main();
288
+ }
289
+
290
+ module.exports = { findBuildOutput, walkFiles, analyzeDependencies, formatSize, HEAVY_DEPS };