tribunal-kit 4.3.1 → 4.4.1

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 (67) hide show
  1. package/.agent/agents/api-architect.md +66 -66
  2. package/.agent/agents/db-latency-auditor.md +216 -216
  3. package/.agent/agents/precedence-reviewer.md +250 -250
  4. package/.agent/agents/resilience-reviewer.md +88 -88
  5. package/.agent/agents/schema-reviewer.md +67 -67
  6. package/.agent/agents/throughput-optimizer.md +299 -299
  7. package/.agent/agents/ui-ux-auditor.md +292 -292
  8. package/.agent/agents/vitals-reviewer.md +223 -223
  9. package/.agent/scripts/_colors.js +18 -18
  10. package/.agent/scripts/_utils.js +42 -42
  11. package/.agent/scripts/append_flow.js +72 -72
  12. package/.agent/scripts/auto_preview.js +197 -197
  13. package/.agent/scripts/bundle_analyzer.js +290 -290
  14. package/.agent/scripts/case_law_manager.js +17 -6
  15. package/.agent/scripts/checklist.js +266 -266
  16. package/.agent/scripts/colors.js +17 -17
  17. package/.agent/scripts/compress_skills.js +141 -141
  18. package/.agent/scripts/consolidate_skills.js +149 -149
  19. package/.agent/scripts/context_broker.js +611 -609
  20. package/.agent/scripts/deep_compress.js +150 -150
  21. package/.agent/scripts/dependency_analyzer.js +272 -272
  22. package/.agent/scripts/graph_builder.js +151 -37
  23. package/.agent/scripts/graph_visualizer.js +384 -0
  24. package/.agent/scripts/inner_loop_validator.js +451 -465
  25. package/.agent/scripts/lint_runner.js +187 -187
  26. package/.agent/scripts/minify_context.js +100 -100
  27. package/.agent/scripts/mutation_runner.js +280 -0
  28. package/.agent/scripts/patch_skills_meta.js +156 -156
  29. package/.agent/scripts/patch_skills_output.js +244 -244
  30. package/.agent/scripts/schema_validator.js +297 -297
  31. package/.agent/scripts/security_scan.js +303 -303
  32. package/.agent/scripts/session_manager.js +276 -276
  33. package/.agent/scripts/skill_evolution.js +644 -644
  34. package/.agent/scripts/skill_integrator.js +313 -313
  35. package/.agent/scripts/strengthen_skills.js +193 -193
  36. package/.agent/scripts/strip_tribunal.js +47 -47
  37. package/.agent/scripts/swarm_dispatcher.js +360 -360
  38. package/.agent/scripts/test_runner.js +193 -193
  39. package/.agent/scripts/utils.js +32 -32
  40. package/.agent/scripts/verify_all.js +257 -256
  41. package/.agent/skills/app-builder/templates/astro-static/TEMPLATE.md +1 -1
  42. package/.agent/skills/app-builder/templates/chrome-extension/TEMPLATE.md +1 -1
  43. package/.agent/skills/app-builder/templates/cli-tool/TEMPLATE.md +1 -1
  44. package/.agent/skills/app-builder/templates/electron-desktop/TEMPLATE.md +1 -1
  45. package/.agent/skills/app-builder/templates/express-api/TEMPLATE.md +1 -1
  46. package/.agent/skills/app-builder/templates/flutter-app/TEMPLATE.md +1 -1
  47. package/.agent/skills/app-builder/templates/monorepo-turborepo/TEMPLATE.md +1 -1
  48. package/.agent/skills/app-builder/templates/nextjs-fullstack/TEMPLATE.md +1 -1
  49. package/.agent/skills/app-builder/templates/nextjs-saas/TEMPLATE.md +1 -1
  50. package/.agent/skills/app-builder/templates/nextjs-static/TEMPLATE.md +1 -1
  51. package/.agent/skills/app-builder/templates/nuxt-app/TEMPLATE.md +1 -1
  52. package/.agent/skills/app-builder/templates/python-fastapi/TEMPLATE.md +1 -1
  53. package/.agent/skills/app-builder/templates/react-native-app/TEMPLATE.md +1 -1
  54. package/.agent/skills/doc.md +1 -1
  55. package/.agent/skills/knowledge-graph/SKILL.md +32 -16
  56. package/.agent/skills/testing-patterns/SKILL.md +19 -2
  57. package/.agent/skills/ui-ux-pro-max/SKILL.md +480 -43
  58. package/.agent/workflows/generate.md +183 -183
  59. package/.agent/workflows/tribunal-speed.md +183 -183
  60. package/README.md +1 -1
  61. package/bin/tribunal-kit.js +134 -17
  62. package/package.json +6 -3
  63. package/scripts/changelog.js +167 -167
  64. package/scripts/sync-version.js +81 -81
  65. package/.agent/scripts/__pycache__/_colors.cpython-311.pyc +0 -0
  66. package/.agent/scripts/__pycache__/_utils.cpython-311.pyc +0 -0
  67. package/.agent/scripts/__pycache__/case_law_manager.cpython-311.pyc +0 -0
@@ -0,0 +1,280 @@
1
+ /**
2
+ * Tribunal-Kit: Testing Patterns 2.0 (Mutation Engine)
3
+ *
4
+ * Safely mutates source code and runs tests to detect false positives.
5
+ * Includes absolute safety net for file restoration.
6
+ *
7
+ * v2.1 — Context-aware mutations (skips strings/comments),
8
+ * line number reporting, configurable --max-mutants.
9
+ */
10
+
11
+ const fs = require('fs');
12
+ const path = require('path');
13
+ const { spawnSync } = require('child_process');
14
+
15
+ // ── Mutation Definitions ──────────────────────────────────────────────────────
16
+ const MUTATIONS = [
17
+ { name: 'Strict Equality', pattern: /===/g, replacement: '!==' },
18
+ { name: 'Strict Inequality', pattern: /!==/g, replacement: '===' },
19
+ { name: 'Logical AND', pattern: /&&/g, replacement: '||' },
20
+ { name: 'Logical OR', pattern: /\|\|/g, replacement: '&&' },
21
+ { name: 'True -> False', pattern: /\btrue\b/g, replacement: 'false' },
22
+ { name: 'False -> True', pattern: /\bfalse\b/g, replacement: 'true' },
23
+ { name: 'Greater Than', pattern: /(?<!=)>(?!=)/g, replacement: '<' },
24
+ { name: 'Less Than', pattern: /(?<!=)<(?!=)/g, replacement: '>' },
25
+ { name: 'Return Early Removal', pattern: /\breturn\b/g, replacement: '/* return */' },
26
+ ];
27
+
28
+ // ── Context-Aware Token Map ───────────────────────────────────────────────────
29
+ // Builds a boolean mask: true = "live code", false = "inside string or comment"
30
+ function buildCodeMask(source) {
31
+ const mask = new Array(source.length).fill(true);
32
+ let inString = false;
33
+ let stringChar = '';
34
+ let inBlockComment = false;
35
+ let inLineComment = false;
36
+
37
+ for (let i = 0; i < source.length; i++) {
38
+ const ch = source[i];
39
+ const next = source[i + 1] || '';
40
+
41
+ if (inBlockComment) {
42
+ mask[i] = false;
43
+ if (ch === '*' && next === '/') {
44
+ mask[i + 1] = false;
45
+ inBlockComment = false;
46
+ i++;
47
+ }
48
+ continue;
49
+ }
50
+
51
+ if (inLineComment) {
52
+ mask[i] = false;
53
+ if (ch === '\n') {
54
+ inLineComment = false;
55
+ }
56
+ continue;
57
+ }
58
+
59
+ if (inString) {
60
+ mask[i] = false;
61
+ if (ch === '\\') {
62
+ i++;
63
+ if (i < source.length) mask[i] = false;
64
+ continue;
65
+ }
66
+ if (ch === stringChar) {
67
+ inString = false;
68
+ }
69
+ continue;
70
+ }
71
+
72
+ // Entering block comment
73
+ if (ch === '/' && next === '*') {
74
+ mask[i] = false;
75
+ mask[i + 1] = false;
76
+ inBlockComment = true;
77
+ i++;
78
+ continue;
79
+ }
80
+
81
+ // Entering line comment
82
+ if (ch === '/' && next === '/') {
83
+ mask[i] = false;
84
+ mask[i + 1] = false;
85
+ inLineComment = true;
86
+ i++;
87
+ continue;
88
+ }
89
+
90
+ // Entering string
91
+ if (ch === '"' || ch === "'" || ch === '`') {
92
+ mask[i] = false;
93
+ inString = true;
94
+ stringChar = ch;
95
+ continue;
96
+ }
97
+
98
+ // Everything else is live code
99
+ mask[i] = true;
100
+ }
101
+
102
+ return mask;
103
+ }
104
+
105
+ // ── Line Number Lookup ────────────────────────────────────────────────────────
106
+ function getLineNumber(source, charIndex) {
107
+ let line = 1;
108
+ for (let i = 0; i < charIndex && i < source.length; i++) {
109
+ if (source[i] === '\n') line++;
110
+ }
111
+ return line;
112
+ }
113
+
114
+ // ── Safe Restore Globals ──────────────────────────────────────────────────────
115
+ let targetFile = null;
116
+ let originalContent = null;
117
+ let backupPath = null;
118
+
119
+ function safeRestore() {
120
+ if (targetFile && originalContent) {
121
+ try {
122
+ fs.writeFileSync(targetFile, originalContent, 'utf-8');
123
+ } catch {
124
+ // Last resort: tell user where the backup is
125
+ if (backupPath) {
126
+ console.error(`[Tribunal] CRITICAL: Could not restore file. Manual backup at: ${backupPath}`);
127
+ }
128
+ }
129
+ if (backupPath && fs.existsSync(backupPath)) {
130
+ try { fs.unlinkSync(backupPath); } catch {}
131
+ }
132
+ }
133
+ }
134
+
135
+ // 🛑 ABSOLUTE SAFETY NET
136
+ process.on('SIGINT', () => {
137
+ safeRestore();
138
+ console.error('\n[Tribunal] Mutation Engine aborted. Target file restored.');
139
+ process.exit(1);
140
+ });
141
+
142
+ process.on('uncaughtException', (err) => {
143
+ safeRestore();
144
+ console.error('\n[Tribunal] Critical error. File restored.', err);
145
+ process.exit(1);
146
+ });
147
+
148
+ process.on('exit', safeRestore);
149
+
150
+ // ── CLI Argument Parsing ──────────────────────────────────────────────────────
151
+ function parseCliArgs(argv) {
152
+ const args = argv.slice(2);
153
+ let maxMutants = 5; // default per mutation type
154
+ let fileToMutate = null;
155
+ let testCommandParts = [];
156
+
157
+ for (let i = 0; i < args.length; i++) {
158
+ if (args[i] === '--max-mutants' && args[i + 1]) {
159
+ maxMutants = parseInt(args[i + 1], 10) || 5;
160
+ i++; // skip next
161
+ } else if (!fileToMutate) {
162
+ fileToMutate = args[i];
163
+ } else {
164
+ testCommandParts.push(args[i]);
165
+ }
166
+ }
167
+
168
+ return { fileToMutate, testCommand: testCommandParts.join(' '), maxMutants };
169
+ }
170
+
171
+ // ── Main Engine ───────────────────────────────────────────────────────────────
172
+ function runMutationTesting(fileToMutate, testCommand, maxMutantsPerType) {
173
+ targetFile = path.resolve(fileToMutate);
174
+
175
+ if (!fs.existsSync(targetFile)) {
176
+ console.error(`ERROR: File not found: ${targetFile}`);
177
+ process.exit(1);
178
+ }
179
+
180
+ originalContent = fs.readFileSync(targetFile, 'utf-8');
181
+ backupPath = targetFile + '.bak';
182
+ fs.writeFileSync(backupPath, originalContent, 'utf-8');
183
+
184
+ console.log(`\n━━━ Tribunal Mutation Engine v2.1 ━━━`);
185
+ console.log(`Target: ${fileToMutate}`);
186
+ console.log(`Test cmd: ${testCommand}`);
187
+ console.log(`Max/type: ${maxMutantsPerType}`);
188
+ console.log(`\nExecuting baseline test run...`);
189
+
190
+ const baseline = spawnSync(testCommand, { shell: true, stdio: 'pipe' });
191
+ if (baseline.status !== 0) {
192
+ console.error(`ERROR: Baseline test failed! Fix your tests before mutating.`);
193
+ console.error(baseline.stderr.toString());
194
+ safeRestore();
195
+ process.exit(1);
196
+ }
197
+
198
+ console.log(`Baseline passed. Building code mask & generating mutants...\n`);
199
+
200
+ // Build the context mask once
201
+ const codeMask = buildCodeMask(originalContent);
202
+
203
+ let totalMutants = 0;
204
+ let killedMutants = 0;
205
+ let survivedMutants = 0;
206
+ const survivors = []; // Track survivors for the report
207
+
208
+ for (const mutation of MUTATIONS) {
209
+ const regex = new RegExp(mutation.pattern.source, mutation.pattern.flags);
210
+ const matchIndices = [];
211
+ let m;
212
+
213
+ while ((m = regex.exec(originalContent)) !== null && matchIndices.length < maxMutantsPerType) {
214
+ // Context-aware: skip matches that are inside strings or comments
215
+ const matchStart = m.index;
216
+ const matchEnd = m.index + m[0].length - 1;
217
+ const isLiveCode = codeMask[matchStart] && codeMask[matchEnd];
218
+
219
+ if (isLiveCode) {
220
+ matchIndices.push({ index: m.index, length: m[0].length, string: m[0] });
221
+ }
222
+ }
223
+
224
+ for (const { index, length, string } of matchIndices) {
225
+ totalMutants++;
226
+ const lineNum = getLineNumber(originalContent, index);
227
+ const mutatedString = string.replace(new RegExp(mutation.pattern.source), mutation.replacement);
228
+ const mutatedContent = originalContent.substring(0, index) + mutatedString + originalContent.substring(index + length);
229
+
230
+ fs.writeFileSync(targetFile, mutatedContent, 'utf-8');
231
+
232
+ process.stdout.write(` [Mutant #${totalMutants}] ${mutation.name} (L${lineNum}) ... `);
233
+ const run = spawnSync(testCommand, { shell: true, stdio: 'pipe' });
234
+
235
+ if (run.status !== 0) {
236
+ console.log(`✅ KILLED`);
237
+ killedMutants++;
238
+ } else {
239
+ console.log(`❌ SURVIVED`);
240
+ survivedMutants++;
241
+ survivors.push({ type: mutation.name, line: lineNum, original: string, mutated: mutatedString });
242
+ }
243
+
244
+ // Restore immediately after each mutant
245
+ fs.writeFileSync(targetFile, originalContent, 'utf-8');
246
+ }
247
+ }
248
+
249
+ const score = totalMutants > 0 ? Math.round((killedMutants / totalMutants) * 100) : 100;
250
+
251
+ console.log(`\n━━━ Mutation Summary ━━━`);
252
+ console.log(` Total Mutants: ${totalMutants}`);
253
+ console.log(` Killed: ${killedMutants}`);
254
+ console.log(` Survived: ${survivedMutants}`);
255
+ console.log(` Score: ${score}%`);
256
+
257
+ if (survivors.length > 0) {
258
+ console.log(`\n━━━ Surviving Mutants (Weak Test Coverage) ━━━`);
259
+ survivors.forEach((s, i) => {
260
+ console.log(` ${i + 1}. Line ${s.line}: ${s.type} (${s.original} → ${s.mutated})`);
261
+ });
262
+ console.log(`\n ⚠ These lines have no test that catches the mutation.`);
263
+ console.log(` Add assertions that would FAIL if the operator were swapped.`);
264
+ }
265
+
266
+ process.exit(score < 80 ? 1 : 0);
267
+ }
268
+
269
+ // ── Entry Point ───────────────────────────────────────────────────────────────
270
+ const { fileToMutate, testCommand, maxMutants } = parseCliArgs(process.argv);
271
+
272
+ if (!fileToMutate || !testCommand) {
273
+ console.log(`Usage: node mutation_runner.js <target_file> [--max-mutants N] <test_command>`);
274
+ console.log(`\nExamples:`);
275
+ console.log(` node mutation_runner.js src/math.js "npx jest src/math.test.js"`);
276
+ console.log(` node mutation_runner.js src/auth.js --max-mutants 10 "npx jest test/auth.test.js"`);
277
+ process.exit(1);
278
+ }
279
+
280
+ runMutationTesting(fileToMutate, testCommand, maxMutants);
@@ -1,156 +1,156 @@
1
- #!/usr/bin/env node
2
- /**
3
- * patch_skills_meta.js — Injects version/freshness metadata into SKILL.md frontmatter.
4
- */
5
-
6
- 'use strict';
7
-
8
- const fs = require('fs');
9
- const path = require('path');
10
-
11
- const { RED, GREEN, YELLOW, BLUE, BOLD, RESET } = require('./colors.js');
12
-
13
- const META_FIELDS = {
14
- "version": "1.0.0",
15
- "last-updated": "2026-03-12",
16
- "applies-to-model": "gemini-2.5-pro, claude-3-7-sonnet"
17
- };
18
-
19
- function header(title) { console.log(`\n${BOLD}${BLUE}━━━ ${title} ━━━${RESET}`); }
20
- function ok(msg) { console.log(` ${GREEN}✅ ${msg}${RESET}`); }
21
- function skip(msg) { console.log(` ${YELLOW}⏭️ ${msg}${RESET}`); }
22
- function warn(msg) { console.log(` ${YELLOW}⚠️ ${msg}${RESET}`); }
23
- function fail(msg) { console.log(` ${RED}❌ ${msg}${RESET}`); }
24
-
25
- function patchFrontmatter(content) {
26
- const added = [];
27
- const fmMatch = content.match(/^---\r?\n([\s\S]*?)\r?\n---/);
28
-
29
- if (!fmMatch) {
30
- const newFmLines = ["---"];
31
- for (const [key, value] of Object.entries(META_FIELDS)) {
32
- newFmLines.push(`${key}: ${value}`);
33
- added.push(key);
34
- }
35
- newFmLines.push("---");
36
- return [newFmLines.join("\n") + "\n\n" + content, added];
37
- }
38
-
39
- const fmText = fmMatch[1];
40
- const fmEnd = fmMatch[0].length;
41
-
42
- const existingKeys = new Set();
43
- const lines = fmText.split("\n");
44
- for (const line of lines) {
45
- const m = line.match(/^([a-zA-Z0-9_-]+)\s*:/);
46
- if (m) existingKeys.add(m[1]);
47
- }
48
-
49
- const newFmLines = fmText.trimEnd().split("\n");
50
- for (const [key, value] of Object.entries(META_FIELDS)) {
51
- if (!existingKeys.has(key)) {
52
- newFmLines.push(`${key}: ${value}`);
53
- added.push(key);
54
- }
55
- }
56
-
57
- if (added.length === 0) return [content, []];
58
-
59
- const patched = "---\n" + newFmLines.join("\n") + "\n---" + content.slice(fmEnd);
60
- return [patched, added];
61
- }
62
-
63
- function processSkill(skillPath, dryRun) {
64
- const skillName = path.basename(path.dirname(skillPath));
65
- try {
66
- const content = fs.readFileSync(skillPath, 'utf8');
67
- const [patched, added] = patchFrontmatter(content);
68
-
69
- if (added.length === 0) {
70
- skip(`${skillName} — all meta fields present`);
71
- return "skipped";
72
- }
73
-
74
- const fieldList = added.join(', ');
75
- if (dryRun) {
76
- warn(`[DRY RUN] ${skillName} — would add: ${fieldList}`);
77
- return "updated";
78
- }
79
-
80
- fs.writeFileSync(skillPath, patched, 'utf8');
81
- ok(`${skillName} — added: ${fieldList}`);
82
- return "updated";
83
- } catch (e) {
84
- fail(`${skillName} — ${e.message}`);
85
- return "error";
86
- }
87
- }
88
-
89
- function main() {
90
- const args = process.argv.slice(2);
91
- let targetPath = null;
92
- let dryRun = false;
93
- let skillArg = null;
94
-
95
- let i = 0;
96
- while (i < args.length) {
97
- if (args[i] === '--dry-run') dryRun = true;
98
- else if (args[i] === '--skill' && i + 1 < args.length) skillArg = args[++i];
99
- else if (args[i] === '-h' || args[i] === '--help') {
100
- console.log("Usage: node patch_skills_meta.js <path> [--dry-run] [--skill <name>]");
101
- process.exit(0);
102
- } else if (!args[i].startsWith('-') && !targetPath) {
103
- targetPath = args[i];
104
- }
105
- i++;
106
- }
107
-
108
- if (!targetPath) {
109
- console.log("Usage: node patch_skills_meta.js <path> [--dry-run] [--skill <name>]");
110
- process.exit(1);
111
- }
112
-
113
- const projectRoot = path.resolve(targetPath);
114
- const skillsDir = path.join(projectRoot, ".agent", "skills");
115
-
116
- if (!fs.existsSync(skillsDir) || !fs.statSync(skillsDir).isDirectory()) {
117
- fail(`Skills directory not found: ${skillsDir}`);
118
- process.exit(1);
119
- }
120
-
121
- console.log(`${BOLD}Tribunal — patch_skills_meta.js${RESET}`);
122
- if (dryRun) console.log(` ${YELLOW}DRY RUN — no files will be written${RESET}`);
123
- console.log(`Skills dir: ${skillsDir}\n`);
124
-
125
- const counts = { updated: 0, skipped: 0, error: 0 };
126
- header("Patching Frontmatter");
127
-
128
- const dirs = fs.readdirSync(skillsDir, { withFileTypes: true });
129
- dirs.sort((a, b) => a.name.localeCompare(b.name));
130
-
131
- for (const dir of dirs) {
132
- if (!dir.isDirectory()) continue;
133
- if (skillArg && dir.name !== skillArg) continue;
134
-
135
- const skillMd = path.join(skillsDir, dir.name, "SKILL.md");
136
- if (!fs.existsSync(skillMd)) {
137
- warn(`${dir.name} — no SKILL.md found`);
138
- continue;
139
- }
140
-
141
- const result = processSkill(skillMd, dryRun);
142
- counts[result]++;
143
- }
144
-
145
- console.log(`\n${BOLD}━━━ Summary ━━━${RESET}`);
146
- console.log(` ${GREEN}✅ Updated: ${counts.updated}${RESET}`);
147
- console.log(` ${YELLOW}⏭️ Skipped: ${counts.skipped}${RESET}`);
148
- if (counts.error > 0) console.log(` ${RED}❌ Errors: ${counts.error}${RESET}`);
149
- if (dryRun) console.log(` ${YELLOW}(dry-run — nothing written)${RESET}`);
150
-
151
- process.exit(counts.error > 0 ? 1 : 0);
152
- }
153
-
154
- if (require.main === module) {
155
- main();
156
- }
1
+ #!/usr/bin/env node
2
+ /**
3
+ * patch_skills_meta.js — Injects version/freshness metadata into SKILL.md frontmatter.
4
+ */
5
+
6
+ 'use strict';
7
+
8
+ const fs = require('fs');
9
+ const path = require('path');
10
+
11
+ const { RED, GREEN, YELLOW, BLUE, BOLD, RESET } = require('./colors.js');
12
+
13
+ const META_FIELDS = {
14
+ "version": "1.0.0",
15
+ "last-updated": "2026-03-12",
16
+ "applies-to-model": "gemini-2.5-pro, claude-3-7-sonnet"
17
+ };
18
+
19
+ function header(title) { console.log(`\n${BOLD}${BLUE}━━━ ${title} ━━━${RESET}`); }
20
+ function ok(msg) { console.log(` ${GREEN}✅ ${msg}${RESET}`); }
21
+ function skip(msg) { console.log(` ${YELLOW}⏭️ ${msg}${RESET}`); }
22
+ function warn(msg) { console.log(` ${YELLOW}⚠️ ${msg}${RESET}`); }
23
+ function fail(msg) { console.log(` ${RED}❌ ${msg}${RESET}`); }
24
+
25
+ function patchFrontmatter(content) {
26
+ const added = [];
27
+ const fmMatch = content.match(/^---\r?\n([\s\S]*?)\r?\n---/);
28
+
29
+ if (!fmMatch) {
30
+ const newFmLines = ["---"];
31
+ for (const [key, value] of Object.entries(META_FIELDS)) {
32
+ newFmLines.push(`${key}: ${value}`);
33
+ added.push(key);
34
+ }
35
+ newFmLines.push("---");
36
+ return [newFmLines.join("\n") + "\n\n" + content, added];
37
+ }
38
+
39
+ const fmText = fmMatch[1];
40
+ const fmEnd = fmMatch[0].length;
41
+
42
+ const existingKeys = new Set();
43
+ const lines = fmText.split("\n");
44
+ for (const line of lines) {
45
+ const m = line.match(/^([a-zA-Z0-9_-]+)\s*:/);
46
+ if (m) existingKeys.add(m[1]);
47
+ }
48
+
49
+ const newFmLines = fmText.trimEnd().split("\n");
50
+ for (const [key, value] of Object.entries(META_FIELDS)) {
51
+ if (!existingKeys.has(key)) {
52
+ newFmLines.push(`${key}: ${value}`);
53
+ added.push(key);
54
+ }
55
+ }
56
+
57
+ if (added.length === 0) return [content, []];
58
+
59
+ const patched = "---\n" + newFmLines.join("\n") + "\n---" + content.slice(fmEnd);
60
+ return [patched, added];
61
+ }
62
+
63
+ function processSkill(skillPath, dryRun) {
64
+ const skillName = path.basename(path.dirname(skillPath));
65
+ try {
66
+ const content = fs.readFileSync(skillPath, 'utf8');
67
+ const [patched, added] = patchFrontmatter(content);
68
+
69
+ if (added.length === 0) {
70
+ skip(`${skillName} — all meta fields present`);
71
+ return "skipped";
72
+ }
73
+
74
+ const fieldList = added.join(', ');
75
+ if (dryRun) {
76
+ warn(`[DRY RUN] ${skillName} — would add: ${fieldList}`);
77
+ return "updated";
78
+ }
79
+
80
+ fs.writeFileSync(skillPath, patched, 'utf8');
81
+ ok(`${skillName} — added: ${fieldList}`);
82
+ return "updated";
83
+ } catch (e) {
84
+ fail(`${skillName} — ${e.message}`);
85
+ return "error";
86
+ }
87
+ }
88
+
89
+ function main() {
90
+ const args = process.argv.slice(2);
91
+ let targetPath = null;
92
+ let dryRun = false;
93
+ let skillArg = null;
94
+
95
+ let i = 0;
96
+ while (i < args.length) {
97
+ if (args[i] === '--dry-run') dryRun = true;
98
+ else if (args[i] === '--skill' && i + 1 < args.length) skillArg = args[++i];
99
+ else if (args[i] === '-h' || args[i] === '--help') {
100
+ console.log("Usage: node patch_skills_meta.js <path> [--dry-run] [--skill <name>]");
101
+ process.exit(0);
102
+ } else if (!args[i].startsWith('-') && !targetPath) {
103
+ targetPath = args[i];
104
+ }
105
+ i++;
106
+ }
107
+
108
+ if (!targetPath) {
109
+ console.log("Usage: node patch_skills_meta.js <path> [--dry-run] [--skill <name>]");
110
+ process.exit(1);
111
+ }
112
+
113
+ const projectRoot = path.resolve(targetPath);
114
+ const skillsDir = path.join(projectRoot, ".agent", "skills");
115
+
116
+ if (!fs.existsSync(skillsDir) || !fs.statSync(skillsDir).isDirectory()) {
117
+ fail(`Skills directory not found: ${skillsDir}`);
118
+ process.exit(1);
119
+ }
120
+
121
+ console.log(`${BOLD}Tribunal — patch_skills_meta.js${RESET}`);
122
+ if (dryRun) console.log(` ${YELLOW}DRY RUN — no files will be written${RESET}`);
123
+ console.log(`Skills dir: ${skillsDir}\n`);
124
+
125
+ const counts = { updated: 0, skipped: 0, error: 0 };
126
+ header("Patching Frontmatter");
127
+
128
+ const dirs = fs.readdirSync(skillsDir, { withFileTypes: true });
129
+ dirs.sort((a, b) => a.name.localeCompare(b.name));
130
+
131
+ for (const dir of dirs) {
132
+ if (!dir.isDirectory()) continue;
133
+ if (skillArg && dir.name !== skillArg) continue;
134
+
135
+ const skillMd = path.join(skillsDir, dir.name, "SKILL.md");
136
+ if (!fs.existsSync(skillMd)) {
137
+ warn(`${dir.name} — no SKILL.md found`);
138
+ continue;
139
+ }
140
+
141
+ const result = processSkill(skillMd, dryRun);
142
+ counts[result]++;
143
+ }
144
+
145
+ console.log(`\n${BOLD}━━━ Summary ━━━${RESET}`);
146
+ console.log(` ${GREEN}✅ Updated: ${counts.updated}${RESET}`);
147
+ console.log(` ${YELLOW}⏭️ Skipped: ${counts.skipped}${RESET}`);
148
+ if (counts.error > 0) console.log(` ${RED}❌ Errors: ${counts.error}${RESET}`);
149
+ if (dryRun) console.log(` ${YELLOW}(dry-run — nothing written)${RESET}`);
150
+
151
+ process.exit(counts.error > 0 ? 1 : 0);
152
+ }
153
+
154
+ if (require.main === module) {
155
+ main();
156
+ }