tribunal-kit 4.4.1 → 4.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 (44) hide show
  1. package/.agent/history/architecture-graph.yaml +140 -0
  2. package/.agent/history/graph-cache.json +262 -0
  3. package/.agent/history/snapshots/bin__tribunal-kit.js.json +19 -0
  4. package/.agent/history/snapshots/eslint.config.js.json +9 -0
  5. package/.agent/history/snapshots/migrate_refs.js.json +11 -0
  6. package/.agent/history/snapshots/scripts__changelog.js.json +13 -0
  7. package/.agent/history/snapshots/scripts__sync-version.js.json +12 -0
  8. package/.agent/history/snapshots/scripts__validate-payload.js.json +12 -0
  9. package/.agent/history/snapshots/test__integration__bridges.test.js.json +14 -0
  10. package/.agent/history/snapshots/test__integration__init.test.js.json +14 -0
  11. package/.agent/history/snapshots/test__integration__routing.test.js.json +12 -0
  12. package/.agent/history/snapshots/test__integration__swarm_dispatcher.test.js.json +14 -0
  13. package/.agent/history/snapshots/test__integration__wave2.test.js.json +14 -0
  14. package/.agent/history/snapshots/test__unit__args.test.js.json +20 -0
  15. package/.agent/history/snapshots/test__unit__case_law_manager.test.js.json +11 -0
  16. package/.agent/history/snapshots/test__unit__context_broker.test.js.json +11 -0
  17. package/.agent/history/snapshots/test__unit__copyDir.test.js.json +23 -0
  18. package/.agent/history/snapshots/test__unit__graph_tools.test.js.json +12 -0
  19. package/.agent/history/snapshots/test__unit__inner_loop_validator.test.js.json +11 -0
  20. package/.agent/history/snapshots/test__unit__selfInstall.test.js.json +23 -0
  21. package/.agent/history/snapshots/test__unit__semver.test.js.json +20 -0
  22. package/.agent/history/snapshots/test__unit__swarm_dispatcher.test.js.json +12 -0
  23. package/.agent/scripts/_colors.js +170 -18
  24. package/.agent/scripts/_utils.js +244 -42
  25. package/.agent/scripts/bundle_analyzer.js +261 -290
  26. package/.agent/scripts/case_law_manager.js +1 -7
  27. package/.agent/scripts/checklist.js +278 -266
  28. package/.agent/scripts/colors.js +11 -17
  29. package/.agent/scripts/context_broker.js +1 -7
  30. package/.agent/scripts/dependency_analyzer.js +234 -272
  31. package/.agent/scripts/graph_builder.js +46 -18
  32. package/.agent/scripts/graph_visualizer.js +10 -4
  33. package/.agent/scripts/graph_zoom.js +6 -4
  34. package/.agent/scripts/inner_loop_validator.js +2 -8
  35. package/.agent/scripts/lint_runner.js +186 -187
  36. package/.agent/scripts/schema_validator.js +8 -25
  37. package/.agent/scripts/security_scan.js +276 -303
  38. package/.agent/scripts/session_manager.js +1 -7
  39. package/.agent/scripts/skill_evolution.js +1 -8
  40. package/.agent/scripts/skill_integrator.js +1 -7
  41. package/.agent/scripts/test_runner.js +186 -193
  42. package/.agent/scripts/utils.js +17 -32
  43. package/.agent/scripts/verify_all.js +248 -257
  44. package/package.json +1 -1
@@ -1,290 +1,261 @@
1
- #!/usr/bin/env node
2
- /**
3
- * bundle_analyzer.js — JS/TS bundle size analyzer for the Tribunal Agent Kit.
4
- *
5
- * Analyzes build output for:
6
- * - Total bundle size
7
- * - Largest files in dist/
8
- * - Suggested tree-shaking opportunities
9
- * - Bundler-specific analysis (Vite / Webpack)
10
- *
11
- * Usage:
12
- * node .agent/scripts/bundle_analyzer.js .
13
- * node .agent/scripts/bundle_analyzer.js . --build
14
- * node .agent/scripts/bundle_analyzer.js . --threshold 500
15
- */
16
-
17
- 'use strict';
18
-
19
- const fs = require('fs');
20
- const path = require('path');
21
- const { spawnSync } = require('child_process');
22
-
23
- const { RED, GREEN, YELLOW, BLUE, BOLD, RESET } = require('./colors.js');
24
-
25
- const HEAVY_PACKAGES = {
26
- "moment": "Use date-fns or dayjs instead (~2KB vs ~230KB)",
27
- "lodash": "Import specific functions: lodash/debounce instead of full lodash",
28
- "rxjs": "Import specific operators to enable tree-shaking",
29
- "aws-sdk": "Use @aws-sdk/client-* v3 modular imports",
30
- "firebase": "Use modular imports: firebase/auth, firebase/firestore",
31
- "chart.js": "Register only needed components",
32
- "three": "Import specific modules from three/examples/jsm/",
33
- "@mui/material": "Ensure babel-plugin-import or modular imports",
34
- "@mui/icons-material": "Import specific icons, never the barrel",
35
- "antd": "Use modular imports with babel-plugin-import",
36
- };
37
-
38
- function header(title) {
39
- console.log(`\n${BOLD}${BLUE}━━━ ${title} ━━━${RESET}`);
40
- }
41
-
42
- function ok(msg) {
43
- console.log(` ${GREEN}✅ ${msg}${RESET}`);
44
- }
45
-
46
- function failPrint(msg) {
47
- console.log(` ${RED}❌ ${msg}${RESET}`);
48
- }
49
-
50
- function warn(msg) {
51
- console.log(` ${YELLOW}⚠️ ${msg}${RESET}`);
52
- }
53
-
54
- function skip(msg) {
55
- console.log(` ${YELLOW}⏭️ ${msg}${RESET}`);
56
- }
57
-
58
- function formatSize(sizeBytes) {
59
- if (sizeBytes < 1024) return `${sizeBytes}B`;
60
- if (sizeBytes < 1024 * 1024) return `${(sizeBytes / 1024).toFixed(1)}KB`;
61
- return `${(sizeBytes / (1024 * 1024)).toFixed(1)}MB`;
62
- }
63
-
64
- function detectBundler(projectRoot) {
65
- const pkgPath = path.join(projectRoot, "package.json");
66
- if (!fs.existsSync(pkgPath)) return null;
67
-
68
- try {
69
- const pkg = JSON.parse(fs.readFileSync(pkgPath, 'utf8'));
70
- const deps = { ...(pkg.dependencies || {}), ...(pkg.devDependencies || {}) };
71
-
72
- if (deps.vite) return "vite";
73
- if (deps.next) return "next";
74
- if (deps.webpack) return "webpack";
75
-
76
- if (fs.existsSync(path.join(projectRoot, "webpack.config.js")) ||
77
- fs.existsSync(path.join(projectRoot, "webpack.config.ts"))) {
78
- return "webpack";
79
- }
80
- } catch {}
81
-
82
- return null;
83
- }
84
-
85
- function findDistDir(projectRoot) {
86
- const candidates = ["dist", "build", ".next", "out", "public/build"];
87
- for (const c of candidates) {
88
- const d = path.join(projectRoot, c);
89
- if (fs.existsSync(d) && fs.statSync(d).isDirectory()) return d;
90
- }
91
- return null;
92
- }
93
-
94
- function analyzeDist(distDir, _thresholdKb) {
95
- const files = [];
96
- let total = 0;
97
-
98
- function walkDir(dir) {
99
- const items = fs.readdirSync(dir, { withFileTypes: true });
100
- for (const item of items) {
101
- const fpath = path.join(dir, item.name);
102
- if (item.isDirectory()) {
103
- walkDir(fpath);
104
- } else {
105
- const size = fs.statSync(fpath).size;
106
- total += size;
107
- files.push([path.relative(distDir, fpath), size]);
108
- }
109
- }
110
- }
111
-
112
- try {
113
- walkDir(distDir);
114
- } catch {}
115
-
116
- files.sort((a, b) => b[1] - a[1]);
117
- return { total, files };
118
- }
119
-
120
- function checkHeavyDependencies(projectRoot) {
121
- const pkgPath = path.join(projectRoot, "package.json");
122
- if (!fs.existsSync(pkgPath)) return [];
123
-
124
- try {
125
- const pkg = JSON.parse(fs.readFileSync(pkgPath, 'utf8'));
126
- const deps = Object.keys(pkg.dependencies || {});
127
- const found = [];
128
-
129
- for (const [pkgName, suggestion] of Object.entries(HEAVY_PACKAGES)) {
130
- if (deps.includes(pkgName)) {
131
- found.push([pkgName, suggestion]);
132
- }
133
- }
134
- return found;
135
- } catch {
136
- return [];
137
- }
138
- }
139
-
140
- function runBuild(projectRoot) {
141
- const pkgPath = path.join(projectRoot, "package.json");
142
- if (fs.existsSync(pkgPath)) {
143
- try {
144
- const pkg = JSON.parse(fs.readFileSync(pkgPath, 'utf8'));
145
- if (!pkg.scripts || !pkg.scripts.build) {
146
- skip("No 'build' script found in package.json");
147
- return true;
148
- }
149
- } catch {}
150
- }
151
-
152
- try {
153
- const executable = process.platform === 'win32' ? 'npm.cmd' : 'npm';
154
- const result = spawnSync(executable, ["run", "build"], {
155
- cwd: projectRoot,
156
- encoding: 'utf8',
157
- timeout: 300000,
158
- shell: process.platform === 'win32'
159
- });
160
-
161
- if (result.status === 0) {
162
- ok("Build completed successfully");
163
- return true;
164
- }
165
-
166
- failPrint("Build failed");
167
- if (result.error) {
168
- console.log(` Error: ${result.error.message}`);
169
- }
170
- const out = result.stdout ? result.stdout.toString() : '';
171
- const err = result.stderr ? result.stderr.toString() : '';
172
- const output = (out + "\n" + err).trim();
173
- if (output) {
174
- for (const line of output.split("\n").slice(0, 10)) {
175
- console.log(` ${line}`);
176
- }
177
- }
178
- return false;
179
- } catch (e) {
180
- failPrint(`Execution error: ${e.message}`);
181
- return false;
182
- }
183
- }
184
-
185
- function main() {
186
- const args = process.argv.slice(2);
187
-
188
- let targetPath = null;
189
- let buildFlag = false;
190
- let threshold = 250;
191
-
192
- for (let i = 0; i < args.length; i++) {
193
- if (args[i] === '--build') buildFlag = true;
194
- else if (args[i] === '--threshold' && i + 1 < args.length) {
195
- threshold = parseInt(args[++i], 10);
196
- } else if (args[i].startsWith('-')) {
197
- console.log("Usage: node bundle_analyzer.js <path> [--build] [--threshold <kb>]");
198
- process.exit(1);
199
- } else if (!targetPath) {
200
- targetPath = args[i];
201
- }
202
- }
203
-
204
- if (!targetPath) {
205
- console.log("Usage: node bundle_analyzer.js <path> [--build] [--threshold <kb>]");
206
- process.exit(1);
207
- }
208
-
209
- const projectRoot = path.resolve(targetPath);
210
- if (!fs.existsSync(projectRoot) || !fs.statSync(projectRoot).isDirectory()) {
211
- failPrint(`Directory not found: ${projectRoot}`);
212
- process.exit(1);
213
- }
214
-
215
- console.log(`${BOLD}Tribunal bundle_analyzer.js${RESET}`);
216
- console.log(`Project: ${projectRoot}`);
217
-
218
- const bundler = detectBundler(projectRoot);
219
- if (bundler) console.log(` Bundler: ${bundler}`);
220
-
221
- if (buildFlag) {
222
- header("Building project");
223
- if (!runBuild(projectRoot)) {
224
- process.exit(1);
225
- }
226
- }
227
-
228
- const distDir = findDistDir(projectRoot);
229
- const heavy = checkHeavyDependencies(projectRoot);
230
-
231
- if (!distDir) {
232
- skip("No build output directory found (dist/, build/, .next/, out/)");
233
- skip("Run with --build to create a build first, or build manually");
234
- } else {
235
- header(`Bundle Size Analysis (${path.relative(projectRoot, distDir)}/)`);
236
- const { total, files } = analyzeDist(distDir, threshold);
237
- console.log(`\n Total bundle size: ${BOLD}${formatSize(total)}${RESET}`);
238
-
239
- const thresholdBytes = threshold * 1024;
240
- console.log(`\n ${BOLD}Top files by size:${RESET}`);
241
- let count = 0;
242
- for (const [filepath, size] of files) {
243
- if (count++ >= 10) break;
244
- const sizeStr = formatSize(size).padStart(10, ' ');
245
- if (size > thresholdBytes) {
246
- warn(`${sizeStr} ${filepath}`);
247
- } else {
248
- console.log(` ${sizeStr} ${filepath}`);
249
- }
250
- }
251
-
252
- const largeJs = files.filter(([f, s]) => (f.endsWith('.js') || f.endsWith('.mjs')) && s > thresholdBytes);
253
- if (largeJs.length > 0) {
254
- console.log(`\n ${YELLOW}${largeJs.length} JS file(s) exceed ${threshold}KB threshold${RESET}`);
255
- }
256
- }
257
-
258
- header("Dependency Weight Check");
259
- if (heavy.length > 0) {
260
- for (const [pkgName, suggestion] of heavy) {
261
- warn(`'${pkgName}' is a heavy dependency`);
262
- console.log(` → ${suggestion}`);
263
- }
264
- } else {
265
- ok("No known-heavy packages detected");
266
- }
267
-
268
- console.log(`\n${BOLD}━━━ Bundle Analysis Summary ━━━${RESET}`);
269
- if (distDir) {
270
- const { total } = analyzeDist(distDir, threshold);
271
- const sizeStr = formatSize(total);
272
- if (total > 5 * 1024 * 1024) {
273
- failPrint(`Total bundle: ${sizeStr} — consider code splitting`);
274
- } else if (total > 2 * 1024 * 1024) {
275
- warn(`Total bundle: ${sizeStr} — review for optimization opportunities`);
276
- } else {
277
- ok(`Total bundle: ${sizeStr}`);
278
- }
279
- }
280
-
281
- if (heavy.length > 0) {
282
- warn(`${heavy.length} heavy dependency suggestion(s) — see above`);
283
- } else if (distDir && heavy.length === 0) {
284
- ok("No optimization suggestions");
285
- }
286
- }
287
-
288
- if (require.main === module) {
289
- main();
290
- }
1
+ #!/usr/bin/env node
2
+ /**
3
+ * bundle_analyzer.js — JS/TS bundle size analyzer for the Tribunal Agent Kit.
4
+ *
5
+ * Analyzes build output for:
6
+ * - Total bundle size
7
+ * - Largest files in dist/
8
+ * - Suggested tree-shaking opportunities
9
+ * - Bundler-specific analysis (Vite / Webpack)
10
+ *
11
+ * Usage:
12
+ * node .agent/scripts/bundle_analyzer.js .
13
+ * node .agent/scripts/bundle_analyzer.js . --build
14
+ * node .agent/scripts/bundle_analyzer.js . --threshold 500
15
+ */
16
+
17
+ 'use strict';
18
+
19
+ const fs = require('fs');
20
+ const path = require('path');
21
+
22
+ const {
23
+ RED, GREEN, YELLOW, BLUE, BOLD, DIM, CYAN, RESET,
24
+ banner, sectionHeader, timer, formatMs,
25
+ ok, fail, warn, skip,
26
+ } = require('./_colors');
27
+
28
+ const { loadJson, runCommand } = require('./_utils');
29
+
30
+ const HEAVY_PACKAGES = {
31
+ "moment": "Use date-fns or dayjs instead (~2KB vs ~230KB)",
32
+ "lodash": "Import specific functions: lodash/debounce instead of full lodash",
33
+ "rxjs": "Import specific operators to enable tree-shaking",
34
+ "aws-sdk": "Use @aws-sdk/client-* v3 modular imports",
35
+ "firebase": "Use modular imports: firebase/auth, firebase/firestore",
36
+ "chart.js": "Register only needed components",
37
+ "three": "Import specific modules from three/examples/jsm/",
38
+ "@mui/material": "Ensure babel-plugin-import or modular imports",
39
+ "@mui/icons-material": "Import specific icons, never the barrel",
40
+ "antd": "Use modular imports with babel-plugin-import",
41
+ };
42
+
43
+ function formatSize(sizeBytes) {
44
+ if (sizeBytes < 1024) return `${sizeBytes}B`;
45
+ if (sizeBytes < 1024 * 1024) return `${(sizeBytes / 1024).toFixed(1)}KB`;
46
+ return `${(sizeBytes / (1024 * 1024)).toFixed(1)}MB`;
47
+ }
48
+
49
+ function detectBundler(projectRoot) {
50
+ const pkg = loadJson(path.join(projectRoot, "package.json"));
51
+ if (!pkg) return null;
52
+
53
+ const deps = { ...(pkg.dependencies || {}), ...(pkg.devDependencies || {}) };
54
+
55
+ if (deps.vite) return "vite";
56
+ if (deps.next) return "next";
57
+ if (deps.webpack) return "webpack";
58
+
59
+ if (fs.existsSync(path.join(projectRoot, "webpack.config.js")) ||
60
+ fs.existsSync(path.join(projectRoot, "webpack.config.ts"))) {
61
+ return "webpack";
62
+ }
63
+
64
+ return null;
65
+ }
66
+
67
+ function findDistDir(projectRoot) {
68
+ const candidates = ["dist", "build", ".next", "out", "public/build"];
69
+ for (const c of candidates) {
70
+ const d = path.join(projectRoot, c);
71
+ if (fs.existsSync(d) && fs.statSync(d).isDirectory()) return d;
72
+ }
73
+ return null;
74
+ }
75
+
76
+ /**
77
+ * Analyze dist directory. Uses inline walker (not shared _utils.walkDir)
78
+ * because it needs to collect file sizes and totals in one pass.
79
+ */
80
+ function analyzeDist(distDir) {
81
+ const files = [];
82
+ let total = 0;
83
+
84
+ function _walk(dir) {
85
+ let items;
86
+ try { items = fs.readdirSync(dir, { withFileTypes: true }); } catch { return; }
87
+ for (const item of items) {
88
+ const fpath = path.join(dir, item.name);
89
+ if (item.isDirectory()) {
90
+ _walk(fpath);
91
+ } else {
92
+ const size = fs.statSync(fpath).size;
93
+ total += size;
94
+ files.push([path.relative(distDir, fpath), size]);
95
+ }
96
+ }
97
+ }
98
+
99
+ _walk(distDir);
100
+ files.sort((a, b) => b[1] - a[1]);
101
+ return { total, files };
102
+ }
103
+
104
+ function checkHeavyDependencies(projectRoot) {
105
+ const pkg = loadJson(path.join(projectRoot, "package.json"));
106
+ if (!pkg) return [];
107
+
108
+ const deps = Object.keys(pkg.dependencies || {});
109
+ const found = [];
110
+
111
+ for (const [pkgName, suggestion] of Object.entries(HEAVY_PACKAGES)) {
112
+ if (deps.includes(pkgName)) {
113
+ found.push([pkgName, suggestion]);
114
+ }
115
+ }
116
+ return found;
117
+ }
118
+
119
+ function runBuild(projectRoot) {
120
+ const pkg = loadJson(path.join(projectRoot, "package.json"));
121
+ if (pkg && (!pkg.scripts || !pkg.scripts.build)) {
122
+ skip("No 'build' script found in package.json");
123
+ return true;
124
+ }
125
+
126
+ const elapsed = timer();
127
+ const result = runCommand('npm', ['run', 'build'], {
128
+ cwd: projectRoot,
129
+ timeout: 300000,
130
+ });
131
+
132
+ const ms = elapsed();
133
+ if (result.ok) {
134
+ ok(`Build completed successfully ${DIM}(${formatMs(ms)})${RESET}`);
135
+ return true;
136
+ }
137
+
138
+ fail(`Build failed ${DIM}(${formatMs(ms)})${RESET}`);
139
+ const output = (result.stdout + "\n" + result.stderr).trim();
140
+ if (output) {
141
+ for (const line of output.split("\n").slice(0, 10)) {
142
+ console.log(` ${line}`);
143
+ }
144
+ }
145
+ return false;
146
+ }
147
+
148
+ function main() {
149
+ const args = process.argv.slice(2);
150
+
151
+ let targetPath = null;
152
+ let buildFlag = false;
153
+ let threshold = 250;
154
+
155
+ for (let i = 0; i < args.length; i++) {
156
+ if (args[i] === '--build') buildFlag = true;
157
+ else if (args[i] === '--threshold' && i + 1 < args.length) {
158
+ threshold = parseInt(args[++i], 10);
159
+ } else if (args[i] === '-h' || args[i] === '--help') {
160
+ console.log("Usage: node bundle_analyzer.js <path> [--build] [--threshold <kb>]");
161
+ process.exit(0);
162
+ } else if (args[i].startsWith('-')) {
163
+ console.log("Usage: node bundle_analyzer.js <path> [--build] [--threshold <kb>]");
164
+ process.exit(1);
165
+ } else if (!targetPath) {
166
+ targetPath = args[i];
167
+ }
168
+ }
169
+
170
+ if (!targetPath) {
171
+ console.log("Usage: node bundle_analyzer.js <path> [--build] [--threshold <kb>]");
172
+ process.exit(1);
173
+ }
174
+
175
+ const projectRoot = path.resolve(targetPath);
176
+ if (!fs.existsSync(projectRoot) || !fs.statSync(projectRoot).isDirectory()) {
177
+ fail(`Directory not found: ${projectRoot}`);
178
+ process.exit(1);
179
+ }
180
+
181
+ const bundler = detectBundler(projectRoot);
182
+ console.log(banner('bundle_analyzer.js', {
183
+ Project: projectRoot,
184
+ Bundler: bundler || 'auto-detect',
185
+ Threshold: `${threshold}KB`,
186
+ }));
187
+
188
+ if (buildFlag) {
189
+ console.log(sectionHeader('Building Project'));
190
+ if (!runBuild(projectRoot)) {
191
+ process.exit(1);
192
+ }
193
+ }
194
+
195
+ const distDir = findDistDir(projectRoot);
196
+ const heavy = checkHeavyDependencies(projectRoot);
197
+
198
+ // PERFORMANCE FIX: Cache analyzeDist result — was called TWICE before
199
+ let distResult = null;
200
+
201
+ if (!distDir) {
202
+ skip("No build output directory found (dist/, build/, .next/, out/)");
203
+ skip("Run with --build to create a build first, or build manually");
204
+ } else {
205
+ console.log(sectionHeader(`Bundle Size Analysis (${path.relative(projectRoot, distDir)}/)`));
206
+ distResult = analyzeDist(distDir);
207
+ console.log(`\n Total bundle size: ${BOLD}${formatSize(distResult.total)}${RESET}`);
208
+
209
+ const thresholdBytes = threshold * 1024;
210
+ console.log(`\n ${BOLD}Top files by size:${RESET}`);
211
+ let count = 0;
212
+ for (const [filepath, size] of distResult.files) {
213
+ if (count++ >= 10) break;
214
+ const sizeStr = formatSize(size).padStart(10, ' ');
215
+ if (size > thresholdBytes) {
216
+ warn(`${sizeStr} ${filepath}`);
217
+ } else {
218
+ console.log(` ${sizeStr} ${filepath}`);
219
+ }
220
+ }
221
+
222
+ const largeJs = distResult.files.filter(([f, s]) => (f.endsWith('.js') || f.endsWith('.mjs')) && s > thresholdBytes);
223
+ if (largeJs.length > 0) {
224
+ console.log(`\n ${YELLOW}${largeJs.length} JS file(s) exceed ${threshold}KB threshold${RESET}`);
225
+ }
226
+ }
227
+
228
+ console.log(sectionHeader('Dependency Weight Check'));
229
+ if (heavy.length > 0) {
230
+ for (const [pkgName, suggestion] of heavy) {
231
+ warn(`'${pkgName}' is a heavy dependency`);
232
+ console.log(` → ${suggestion}`);
233
+ }
234
+ } else {
235
+ ok("No known-heavy packages detected");
236
+ }
237
+
238
+ // ━━━ Summary ━━━ (reuses cached distResult instead of re-scanning)
239
+ console.log(`\n${BOLD}${CYAN}━━━ Bundle Analysis Summary ━━━${RESET}`);
240
+ if (distResult) {
241
+ const sizeStr = formatSize(distResult.total);
242
+ if (distResult.total > 5 * 1024 * 1024) {
243
+ fail(`Total bundle: ${sizeStr} — consider code splitting`);
244
+ } else if (distResult.total > 2 * 1024 * 1024) {
245
+ warn(`Total bundle: ${sizeStr} — review for optimization opportunities`);
246
+ } else {
247
+ ok(`Total bundle: ${sizeStr}`);
248
+ }
249
+ }
250
+
251
+ if (heavy.length > 0) {
252
+ warn(`${heavy.length} heavy dependency suggestion(s) see above`);
253
+ } else if (distResult && heavy.length === 0) {
254
+ ok("No optimization suggestions");
255
+ }
256
+ console.log();
257
+ }
258
+
259
+ if (require.main === module) {
260
+ main();
261
+ }
@@ -26,13 +26,7 @@ const crypto = require('crypto');
26
26
  const readline = require('readline');
27
27
 
28
28
  // ── Colours ──────────────────────────────────────────────────────────────────
29
- const GREEN = '\x1b[92m';
30
- const YELLOW = '\x1b[93m';
31
- const CYAN = '\x1b[96m';
32
- const RED = '\x1b[91m';
33
- const BOLD = '\x1b[1m';
34
- const DIM = '\x1b[2m';
35
- const RESET = '\x1b[0m';
29
+ const { GREEN, YELLOW, CYAN, RED, BOLD, DIM, RESET } = require('./_colors');
36
30
 
37
31
  // ── Find .agent directory ─────────────────────────────────────────────────────
38
32
  function findAgentDir() {