tribunal-kit 4.0.1 → 4.3.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.
Files changed (196) hide show
  1. package/.agent/ARCHITECTURE.md +21 -14
  2. package/.agent/GEMINI.md +4 -2
  3. package/.agent/agents/api-architect.md +66 -0
  4. package/.agent/agents/db-latency-auditor.md +216 -0
  5. package/.agent/agents/precedence-reviewer.md +41 -4
  6. package/.agent/agents/resilience-reviewer.md +88 -0
  7. package/.agent/agents/schema-reviewer.md +67 -0
  8. package/.agent/agents/swarm-worker-contracts.md +5 -5
  9. package/.agent/agents/throughput-optimizer.md +299 -0
  10. package/.agent/agents/ui-ux-auditor.md +292 -0
  11. package/.agent/agents/vitals-reviewer.md +223 -0
  12. package/.agent/history/case-law/cases/case-0001.json +33 -0
  13. package/.agent/history/case-law/index.json +35 -0
  14. package/.agent/rules/GEMINI.md +28 -11
  15. package/.agent/scripts/__pycache__/_colors.cpython-311.pyc +0 -0
  16. package/.agent/scripts/__pycache__/_utils.cpython-311.pyc +0 -0
  17. package/.agent/scripts/__pycache__/case_law_manager.cpython-311.pyc +0 -0
  18. package/.agent/scripts/_colors.js +18 -0
  19. package/.agent/scripts/_utils.js +42 -0
  20. package/.agent/scripts/auto_preview.js +197 -0
  21. package/.agent/scripts/bundle_analyzer.js +290 -0
  22. package/.agent/scripts/case_law_manager.js +684 -0
  23. package/.agent/scripts/checklist.js +266 -0
  24. package/.agent/scripts/colors.js +17 -0
  25. package/.agent/scripts/compress_skills.js +141 -0
  26. package/.agent/scripts/consolidate_skills.js +149 -0
  27. package/.agent/scripts/context_broker.js +609 -0
  28. package/.agent/scripts/deep_compress.js +150 -0
  29. package/.agent/scripts/dependency_analyzer.js +272 -0
  30. package/.agent/scripts/inner_loop_validator.js +465 -0
  31. package/.agent/scripts/lint_runner.js +187 -0
  32. package/.agent/scripts/minify_context.js +100 -0
  33. package/.agent/scripts/patch_skills_meta.js +156 -0
  34. package/.agent/scripts/patch_skills_output.js +244 -0
  35. package/.agent/scripts/schema_validator.js +297 -0
  36. package/.agent/scripts/security_scan.js +303 -0
  37. package/.agent/scripts/session_manager.js +276 -0
  38. package/.agent/scripts/skill_evolution.js +644 -0
  39. package/.agent/scripts/skill_integrator.js +313 -0
  40. package/.agent/scripts/strengthen_skills.js +193 -0
  41. package/.agent/scripts/strip_tribunal.js +47 -0
  42. package/.agent/scripts/swarm_dispatcher.js +360 -0
  43. package/.agent/scripts/test_runner.js +193 -0
  44. package/.agent/scripts/utils.js +32 -0
  45. package/.agent/scripts/verify_all.js +256 -0
  46. package/.agent/skills/agent-organizer/SKILL.md +42 -0
  47. package/.agent/skills/agentic-patterns/SKILL.md +42 -0
  48. package/.agent/skills/ai-prompt-injection-defense/SKILL.md +42 -0
  49. package/.agent/skills/api-patterns/SKILL.md +42 -0
  50. package/.agent/skills/api-security-auditor/SKILL.md +42 -0
  51. package/.agent/skills/app-builder/SKILL.md +42 -0
  52. package/.agent/skills/app-builder/templates/SKILL.md +70 -0
  53. package/.agent/skills/app-builder/templates/astro-static/TEMPLATE.md +1 -1
  54. package/.agent/skills/app-builder/templates/chrome-extension/TEMPLATE.md +1 -1
  55. package/.agent/skills/app-builder/templates/cli-tool/TEMPLATE.md +1 -1
  56. package/.agent/skills/app-builder/templates/electron-desktop/TEMPLATE.md +1 -1
  57. package/.agent/skills/app-builder/templates/express-api/TEMPLATE.md +1 -1
  58. package/.agent/skills/app-builder/templates/flutter-app/TEMPLATE.md +1 -1
  59. package/.agent/skills/app-builder/templates/monorepo-turborepo/TEMPLATE.md +1 -1
  60. package/.agent/skills/app-builder/templates/nextjs-fullstack/TEMPLATE.md +1 -1
  61. package/.agent/skills/app-builder/templates/nextjs-saas/TEMPLATE.md +1 -1
  62. package/.agent/skills/app-builder/templates/nextjs-static/TEMPLATE.md +1 -1
  63. package/.agent/skills/app-builder/templates/nuxt-app/TEMPLATE.md +1 -1
  64. package/.agent/skills/app-builder/templates/python-fastapi/TEMPLATE.md +1 -1
  65. package/.agent/skills/app-builder/templates/react-native-app/TEMPLATE.md +1 -1
  66. package/.agent/skills/appflow-wireframe/SKILL.md +42 -0
  67. package/.agent/skills/architecture/SKILL.md +42 -0
  68. package/.agent/skills/authentication-best-practices/SKILL.md +42 -0
  69. package/.agent/skills/bash-linux/SKILL.md +42 -0
  70. package/.agent/skills/behavioral-modes/SKILL.md +42 -0
  71. package/.agent/skills/brainstorming/SKILL.md +42 -0
  72. package/.agent/skills/building-native-ui/SKILL.md +42 -0
  73. package/.agent/skills/clean-code/SKILL.md +42 -0
  74. package/.agent/skills/code-review-checklist/SKILL.md +42 -0
  75. package/.agent/skills/config-validator/SKILL.md +42 -0
  76. package/.agent/skills/csharp-developer/SKILL.md +42 -0
  77. package/.agent/skills/data-validation-schemas/SKILL.md +320 -0
  78. package/.agent/skills/database-design/SKILL.md +42 -0
  79. package/.agent/skills/deployment-procedures/SKILL.md +42 -0
  80. package/.agent/skills/devops-engineer/SKILL.md +42 -0
  81. package/.agent/skills/devops-incident-responder/SKILL.md +42 -0
  82. package/.agent/skills/doc.md +1 -1
  83. package/.agent/skills/documentation-templates/SKILL.md +42 -0
  84. package/.agent/skills/edge-computing/SKILL.md +42 -0
  85. package/.agent/skills/error-resilience/SKILL.md +420 -0
  86. package/.agent/skills/extract-design-system/SKILL.md +42 -0
  87. package/.agent/skills/framer-motion-expert/SKILL.md +42 -1
  88. package/.agent/skills/frontend-design/SKILL.md +42 -0
  89. package/.agent/skills/game-design-expert/SKILL.md +42 -0
  90. package/.agent/skills/game-engineering-expert/SKILL.md +42 -0
  91. package/.agent/skills/geo-fundamentals/SKILL.md +42 -0
  92. package/.agent/skills/github-operations/SKILL.md +42 -0
  93. package/.agent/skills/gsap-core/SKILL.md +300 -0
  94. package/.agent/skills/gsap-frameworks/SKILL.md +199 -0
  95. package/.agent/skills/gsap-performance/SKILL.md +125 -0
  96. package/.agent/skills/gsap-plugins/SKILL.md +472 -0
  97. package/.agent/skills/gsap-react/SKILL.md +181 -0
  98. package/.agent/skills/gsap-scrolltrigger/SKILL.md +342 -0
  99. package/.agent/skills/gsap-timeline/SKILL.md +153 -0
  100. package/.agent/skills/gsap-utils/SKILL.md +330 -0
  101. package/.agent/skills/i18n-localization/SKILL.md +42 -0
  102. package/.agent/skills/intelligent-routing/SKILL.md +72 -1
  103. package/.agent/skills/lint-and-validate/SKILL.md +42 -0
  104. package/.agent/skills/llm-engineering/SKILL.md +42 -0
  105. package/.agent/skills/local-first/SKILL.md +42 -0
  106. package/.agent/skills/mcp-builder/SKILL.md +42 -0
  107. package/.agent/skills/mobile-design/SKILL.md +42 -0
  108. package/.agent/skills/monorepo-management/SKILL.md +326 -0
  109. package/.agent/skills/motion-engineering/SKILL.md +42 -0
  110. package/.agent/skills/nextjs-react-expert/SKILL.md +42 -0
  111. package/.agent/skills/nodejs-best-practices/SKILL.md +42 -0
  112. package/.agent/skills/observability/SKILL.md +42 -0
  113. package/.agent/skills/parallel-agents/SKILL.md +42 -0
  114. package/.agent/skills/performance-profiling/SKILL.md +42 -0
  115. package/.agent/skills/plan-writing/SKILL.md +42 -0
  116. package/.agent/skills/platform-engineer/SKILL.md +42 -0
  117. package/.agent/skills/playwright-best-practices/SKILL.md +42 -0
  118. package/.agent/skills/powershell-windows/SKILL.md +42 -0
  119. package/.agent/skills/project-idioms/SKILL.md +42 -0
  120. package/.agent/skills/python-patterns/SKILL.md +42 -0
  121. package/.agent/skills/python-pro/SKILL.md +42 -0
  122. package/.agent/skills/react-specialist/SKILL.md +42 -0
  123. package/.agent/skills/readme-builder/SKILL.md +42 -0
  124. package/.agent/skills/realtime-patterns/SKILL.md +42 -0
  125. package/.agent/skills/red-team-tactics/SKILL.md +42 -0
  126. package/.agent/skills/rust-pro/SKILL.md +42 -0
  127. package/.agent/skills/seo-fundamentals/SKILL.md +42 -0
  128. package/.agent/skills/server-management/SKILL.md +42 -0
  129. package/.agent/skills/shadcn-ui-expert/SKILL.md +42 -0
  130. package/.agent/skills/skill-creator/SKILL.md +42 -0
  131. package/.agent/skills/sql-pro/SKILL.md +42 -0
  132. package/.agent/skills/supabase-postgres-best-practices/SKILL.md +42 -0
  133. package/.agent/skills/swiftui-expert/SKILL.md +42 -0
  134. package/.agent/skills/systematic-debugging/SKILL.md +42 -0
  135. package/.agent/skills/tailwind-patterns/SKILL.md +42 -0
  136. package/.agent/skills/tdd-workflow/SKILL.md +42 -0
  137. package/.agent/skills/test-result-analyzer/SKILL.md +42 -0
  138. package/.agent/skills/testing-patterns/SKILL.md +42 -0
  139. package/.agent/skills/trend-researcher/SKILL.md +42 -0
  140. package/.agent/skills/typescript-advanced/SKILL.md +327 -0
  141. package/.agent/skills/ui-ux-pro-max/SKILL.md +42 -0
  142. package/.agent/skills/ui-ux-researcher/SKILL.md +42 -0
  143. package/.agent/skills/vue-expert/SKILL.md +42 -0
  144. package/.agent/skills/vulnerability-scanner/SKILL.md +42 -0
  145. package/.agent/skills/web-accessibility-auditor/SKILL.md +42 -0
  146. package/.agent/skills/web-design-guidelines/SKILL.md +42 -0
  147. package/.agent/skills/webapp-testing/SKILL.md +42 -0
  148. package/.agent/skills/whimsy-injector/SKILL.md +42 -0
  149. package/.agent/skills/workflow-optimizer/SKILL.md +42 -0
  150. package/.agent/workflows/audit.md +6 -6
  151. package/.agent/workflows/deploy.md +1 -1
  152. package/.agent/workflows/generate.md +23 -6
  153. package/.agent/workflows/session.md +5 -5
  154. package/.agent/workflows/swarm.md +2 -2
  155. package/.agent/workflows/tribunal-backend.md +13 -2
  156. package/.agent/workflows/tribunal-full.md +15 -8
  157. package/.agent/workflows/tribunal-speed.md +183 -0
  158. package/README.md +64 -8
  159. package/bin/tribunal-kit.js +281 -41
  160. package/package.json +9 -6
  161. package/scripts/changelog.js +167 -0
  162. package/scripts/sync-version.js +81 -0
  163. package/.agent/scripts/__pycache__/auto_preview.cpython-311.pyc +0 -0
  164. package/.agent/scripts/__pycache__/bundle_analyzer.cpython-311.pyc +0 -0
  165. package/.agent/scripts/__pycache__/checklist.cpython-311.pyc +0 -0
  166. package/.agent/scripts/__pycache__/dependency_analyzer.cpython-311.pyc +0 -0
  167. package/.agent/scripts/__pycache__/security_scan.cpython-311.pyc +0 -0
  168. package/.agent/scripts/__pycache__/session_manager.cpython-311.pyc +0 -0
  169. package/.agent/scripts/__pycache__/skill_integrator.cpython-311.pyc +0 -0
  170. package/.agent/scripts/__pycache__/swarm_dispatcher.cpython-311.pyc +0 -0
  171. package/.agent/scripts/__pycache__/test_runner.cpython-311.pyc +0 -0
  172. package/.agent/scripts/__pycache__/verify_all.cpython-311.pyc +0 -0
  173. package/.agent/scripts/auto_preview.py +0 -180
  174. package/.agent/scripts/bundle_analyzer.py +0 -259
  175. package/.agent/scripts/case_law_manager.py +0 -525
  176. package/.agent/scripts/checklist.py +0 -209
  177. package/.agent/scripts/compress_skills.py +0 -167
  178. package/.agent/scripts/consolidate_skills.py +0 -173
  179. package/.agent/scripts/deep_compress.py +0 -202
  180. package/.agent/scripts/dependency_analyzer.py +0 -247
  181. package/.agent/scripts/lint_runner.py +0 -188
  182. package/.agent/scripts/minify_context.py +0 -80
  183. package/.agent/scripts/patch_skills_meta.py +0 -177
  184. package/.agent/scripts/patch_skills_output.py +0 -285
  185. package/.agent/scripts/schema_validator.py +0 -279
  186. package/.agent/scripts/security_scan.py +0 -224
  187. package/.agent/scripts/session_manager.py +0 -261
  188. package/.agent/scripts/skill_evolution.py +0 -563
  189. package/.agent/scripts/skill_integrator.py +0 -234
  190. package/.agent/scripts/strengthen_skills.py +0 -220
  191. package/.agent/scripts/strip_tribunal.py +0 -41
  192. package/.agent/scripts/swarm_dispatcher.py +0 -350
  193. package/.agent/scripts/test_runner.py +0 -192
  194. package/.agent/scripts/test_swarm_dispatcher.py +0 -163
  195. package/.agent/scripts/verify_all.py +0 -195
  196. package/.agent/skills/gsap-expert/SKILL.md +0 -194
@@ -0,0 +1,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
+ 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
+ }