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.
- package/.agent/ARCHITECTURE.md +21 -14
- package/.agent/GEMINI.md +4 -2
- package/.agent/agents/api-architect.md +66 -0
- package/.agent/agents/db-latency-auditor.md +216 -0
- package/.agent/agents/precedence-reviewer.md +41 -4
- package/.agent/agents/resilience-reviewer.md +88 -0
- package/.agent/agents/schema-reviewer.md +67 -0
- package/.agent/agents/swarm-worker-contracts.md +5 -5
- package/.agent/agents/throughput-optimizer.md +299 -0
- package/.agent/agents/ui-ux-auditor.md +292 -0
- package/.agent/agents/vitals-reviewer.md +223 -0
- package/.agent/history/case-law/cases/case-0001.json +33 -0
- package/.agent/history/case-law/index.json +35 -0
- package/.agent/rules/GEMINI.md +28 -11
- package/.agent/scripts/__pycache__/_colors.cpython-311.pyc +0 -0
- package/.agent/scripts/__pycache__/_utils.cpython-311.pyc +0 -0
- package/.agent/scripts/__pycache__/case_law_manager.cpython-311.pyc +0 -0
- package/.agent/scripts/_colors.js +18 -0
- package/.agent/scripts/_utils.js +42 -0
- package/.agent/scripts/auto_preview.js +197 -0
- package/.agent/scripts/bundle_analyzer.js +290 -0
- package/.agent/scripts/case_law_manager.js +684 -0
- package/.agent/scripts/checklist.js +266 -0
- package/.agent/scripts/colors.js +17 -0
- package/.agent/scripts/compress_skills.js +141 -0
- package/.agent/scripts/consolidate_skills.js +149 -0
- package/.agent/scripts/context_broker.js +609 -0
- package/.agent/scripts/deep_compress.js +150 -0
- package/.agent/scripts/dependency_analyzer.js +272 -0
- package/.agent/scripts/inner_loop_validator.js +465 -0
- package/.agent/scripts/lint_runner.js +187 -0
- package/.agent/scripts/minify_context.js +100 -0
- package/.agent/scripts/patch_skills_meta.js +156 -0
- package/.agent/scripts/patch_skills_output.js +244 -0
- package/.agent/scripts/schema_validator.js +297 -0
- package/.agent/scripts/security_scan.js +303 -0
- package/.agent/scripts/session_manager.js +276 -0
- package/.agent/scripts/skill_evolution.js +644 -0
- package/.agent/scripts/skill_integrator.js +313 -0
- package/.agent/scripts/strengthen_skills.js +193 -0
- package/.agent/scripts/strip_tribunal.js +47 -0
- package/.agent/scripts/swarm_dispatcher.js +360 -0
- package/.agent/scripts/test_runner.js +193 -0
- package/.agent/scripts/utils.js +32 -0
- package/.agent/scripts/verify_all.js +256 -0
- package/.agent/skills/agent-organizer/SKILL.md +42 -0
- package/.agent/skills/agentic-patterns/SKILL.md +42 -0
- package/.agent/skills/ai-prompt-injection-defense/SKILL.md +42 -0
- package/.agent/skills/api-patterns/SKILL.md +42 -0
- package/.agent/skills/api-security-auditor/SKILL.md +42 -0
- package/.agent/skills/app-builder/SKILL.md +42 -0
- package/.agent/skills/app-builder/templates/SKILL.md +70 -0
- package/.agent/skills/app-builder/templates/astro-static/TEMPLATE.md +1 -1
- package/.agent/skills/app-builder/templates/chrome-extension/TEMPLATE.md +1 -1
- package/.agent/skills/app-builder/templates/cli-tool/TEMPLATE.md +1 -1
- package/.agent/skills/app-builder/templates/electron-desktop/TEMPLATE.md +1 -1
- package/.agent/skills/app-builder/templates/express-api/TEMPLATE.md +1 -1
- package/.agent/skills/app-builder/templates/flutter-app/TEMPLATE.md +1 -1
- package/.agent/skills/app-builder/templates/monorepo-turborepo/TEMPLATE.md +1 -1
- package/.agent/skills/app-builder/templates/nextjs-fullstack/TEMPLATE.md +1 -1
- package/.agent/skills/app-builder/templates/nextjs-saas/TEMPLATE.md +1 -1
- package/.agent/skills/app-builder/templates/nextjs-static/TEMPLATE.md +1 -1
- package/.agent/skills/app-builder/templates/nuxt-app/TEMPLATE.md +1 -1
- package/.agent/skills/app-builder/templates/python-fastapi/TEMPLATE.md +1 -1
- package/.agent/skills/app-builder/templates/react-native-app/TEMPLATE.md +1 -1
- package/.agent/skills/appflow-wireframe/SKILL.md +42 -0
- package/.agent/skills/architecture/SKILL.md +42 -0
- package/.agent/skills/authentication-best-practices/SKILL.md +42 -0
- package/.agent/skills/bash-linux/SKILL.md +42 -0
- package/.agent/skills/behavioral-modes/SKILL.md +42 -0
- package/.agent/skills/brainstorming/SKILL.md +42 -0
- package/.agent/skills/building-native-ui/SKILL.md +42 -0
- package/.agent/skills/clean-code/SKILL.md +42 -0
- package/.agent/skills/code-review-checklist/SKILL.md +42 -0
- package/.agent/skills/config-validator/SKILL.md +42 -0
- package/.agent/skills/csharp-developer/SKILL.md +42 -0
- package/.agent/skills/data-validation-schemas/SKILL.md +320 -0
- package/.agent/skills/database-design/SKILL.md +42 -0
- package/.agent/skills/deployment-procedures/SKILL.md +42 -0
- package/.agent/skills/devops-engineer/SKILL.md +42 -0
- package/.agent/skills/devops-incident-responder/SKILL.md +42 -0
- package/.agent/skills/doc.md +1 -1
- package/.agent/skills/documentation-templates/SKILL.md +42 -0
- package/.agent/skills/edge-computing/SKILL.md +42 -0
- package/.agent/skills/error-resilience/SKILL.md +420 -0
- package/.agent/skills/extract-design-system/SKILL.md +42 -0
- package/.agent/skills/framer-motion-expert/SKILL.md +42 -1
- package/.agent/skills/frontend-design/SKILL.md +42 -0
- package/.agent/skills/game-design-expert/SKILL.md +42 -0
- package/.agent/skills/game-engineering-expert/SKILL.md +42 -0
- package/.agent/skills/geo-fundamentals/SKILL.md +42 -0
- package/.agent/skills/github-operations/SKILL.md +42 -0
- package/.agent/skills/gsap-core/SKILL.md +300 -0
- package/.agent/skills/gsap-frameworks/SKILL.md +199 -0
- package/.agent/skills/gsap-performance/SKILL.md +125 -0
- package/.agent/skills/gsap-plugins/SKILL.md +472 -0
- package/.agent/skills/gsap-react/SKILL.md +181 -0
- package/.agent/skills/gsap-scrolltrigger/SKILL.md +342 -0
- package/.agent/skills/gsap-timeline/SKILL.md +153 -0
- package/.agent/skills/gsap-utils/SKILL.md +330 -0
- package/.agent/skills/i18n-localization/SKILL.md +42 -0
- package/.agent/skills/intelligent-routing/SKILL.md +72 -1
- package/.agent/skills/lint-and-validate/SKILL.md +42 -0
- package/.agent/skills/llm-engineering/SKILL.md +42 -0
- package/.agent/skills/local-first/SKILL.md +42 -0
- package/.agent/skills/mcp-builder/SKILL.md +42 -0
- package/.agent/skills/mobile-design/SKILL.md +42 -0
- package/.agent/skills/monorepo-management/SKILL.md +326 -0
- package/.agent/skills/motion-engineering/SKILL.md +42 -0
- package/.agent/skills/nextjs-react-expert/SKILL.md +42 -0
- package/.agent/skills/nodejs-best-practices/SKILL.md +42 -0
- package/.agent/skills/observability/SKILL.md +42 -0
- package/.agent/skills/parallel-agents/SKILL.md +42 -0
- package/.agent/skills/performance-profiling/SKILL.md +42 -0
- package/.agent/skills/plan-writing/SKILL.md +42 -0
- package/.agent/skills/platform-engineer/SKILL.md +42 -0
- package/.agent/skills/playwright-best-practices/SKILL.md +42 -0
- package/.agent/skills/powershell-windows/SKILL.md +42 -0
- package/.agent/skills/project-idioms/SKILL.md +42 -0
- package/.agent/skills/python-patterns/SKILL.md +42 -0
- package/.agent/skills/python-pro/SKILL.md +42 -0
- package/.agent/skills/react-specialist/SKILL.md +42 -0
- package/.agent/skills/readme-builder/SKILL.md +42 -0
- package/.agent/skills/realtime-patterns/SKILL.md +42 -0
- package/.agent/skills/red-team-tactics/SKILL.md +42 -0
- package/.agent/skills/rust-pro/SKILL.md +42 -0
- package/.agent/skills/seo-fundamentals/SKILL.md +42 -0
- package/.agent/skills/server-management/SKILL.md +42 -0
- package/.agent/skills/shadcn-ui-expert/SKILL.md +42 -0
- package/.agent/skills/skill-creator/SKILL.md +42 -0
- package/.agent/skills/sql-pro/SKILL.md +42 -0
- package/.agent/skills/supabase-postgres-best-practices/SKILL.md +42 -0
- package/.agent/skills/swiftui-expert/SKILL.md +42 -0
- package/.agent/skills/systematic-debugging/SKILL.md +42 -0
- package/.agent/skills/tailwind-patterns/SKILL.md +42 -0
- package/.agent/skills/tdd-workflow/SKILL.md +42 -0
- package/.agent/skills/test-result-analyzer/SKILL.md +42 -0
- package/.agent/skills/testing-patterns/SKILL.md +42 -0
- package/.agent/skills/trend-researcher/SKILL.md +42 -0
- package/.agent/skills/typescript-advanced/SKILL.md +327 -0
- package/.agent/skills/ui-ux-pro-max/SKILL.md +42 -0
- package/.agent/skills/ui-ux-researcher/SKILL.md +42 -0
- package/.agent/skills/vue-expert/SKILL.md +42 -0
- package/.agent/skills/vulnerability-scanner/SKILL.md +42 -0
- package/.agent/skills/web-accessibility-auditor/SKILL.md +42 -0
- package/.agent/skills/web-design-guidelines/SKILL.md +42 -0
- package/.agent/skills/webapp-testing/SKILL.md +42 -0
- package/.agent/skills/whimsy-injector/SKILL.md +42 -0
- package/.agent/skills/workflow-optimizer/SKILL.md +42 -0
- package/.agent/workflows/audit.md +6 -6
- package/.agent/workflows/deploy.md +1 -1
- package/.agent/workflows/generate.md +23 -6
- package/.agent/workflows/session.md +5 -5
- package/.agent/workflows/swarm.md +2 -2
- package/.agent/workflows/tribunal-backend.md +13 -2
- package/.agent/workflows/tribunal-full.md +15 -8
- package/.agent/workflows/tribunal-speed.md +183 -0
- package/README.md +64 -8
- package/bin/tribunal-kit.js +281 -41
- package/package.json +9 -6
- package/scripts/changelog.js +167 -0
- package/scripts/sync-version.js +81 -0
- package/.agent/scripts/__pycache__/auto_preview.cpython-311.pyc +0 -0
- package/.agent/scripts/__pycache__/bundle_analyzer.cpython-311.pyc +0 -0
- package/.agent/scripts/__pycache__/checklist.cpython-311.pyc +0 -0
- package/.agent/scripts/__pycache__/dependency_analyzer.cpython-311.pyc +0 -0
- package/.agent/scripts/__pycache__/security_scan.cpython-311.pyc +0 -0
- package/.agent/scripts/__pycache__/session_manager.cpython-311.pyc +0 -0
- package/.agent/scripts/__pycache__/skill_integrator.cpython-311.pyc +0 -0
- package/.agent/scripts/__pycache__/swarm_dispatcher.cpython-311.pyc +0 -0
- package/.agent/scripts/__pycache__/test_runner.cpython-311.pyc +0 -0
- package/.agent/scripts/__pycache__/verify_all.cpython-311.pyc +0 -0
- package/.agent/scripts/auto_preview.py +0 -180
- package/.agent/scripts/bundle_analyzer.py +0 -259
- package/.agent/scripts/case_law_manager.py +0 -525
- package/.agent/scripts/checklist.py +0 -209
- package/.agent/scripts/compress_skills.py +0 -167
- package/.agent/scripts/consolidate_skills.py +0 -173
- package/.agent/scripts/deep_compress.py +0 -202
- package/.agent/scripts/dependency_analyzer.py +0 -247
- package/.agent/scripts/lint_runner.py +0 -188
- package/.agent/scripts/minify_context.py +0 -80
- package/.agent/scripts/patch_skills_meta.py +0 -177
- package/.agent/scripts/patch_skills_output.py +0 -285
- package/.agent/scripts/schema_validator.py +0 -279
- package/.agent/scripts/security_scan.py +0 -224
- package/.agent/scripts/session_manager.py +0 -261
- package/.agent/scripts/skill_evolution.py +0 -563
- package/.agent/scripts/skill_integrator.py +0 -234
- package/.agent/scripts/strengthen_skills.py +0 -220
- package/.agent/scripts/strip_tribunal.py +0 -41
- package/.agent/scripts/swarm_dispatcher.py +0 -350
- package/.agent/scripts/test_runner.py +0 -192
- package/.agent/scripts/test_swarm_dispatcher.py +0 -163
- package/.agent/scripts/verify_all.py +0 -195
- 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
|
+
}
|