@dyyz1993/pi-coding-agent 0.74.24 → 0.74.25
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/dist/core/agent-session.d.ts.map +1 -1
- package/dist/core/agent-session.js +3 -0
- package/dist/core/agent-session.js.map +1 -1
- package/dist/extensions/agent-permissions/index.ts +235 -0
- package/dist/extensions/ask-tools/index.ts +115 -0
- package/dist/extensions/auto-memory/contract.d.ts +51 -0
- package/dist/extensions/auto-memory/contract.d.ts.map +1 -0
- package/dist/extensions/auto-memory/contract.js +2 -0
- package/dist/extensions/auto-memory/contract.js.map +1 -0
- package/dist/extensions/auto-memory/contract.ts +56 -0
- package/dist/extensions/auto-memory/index.ts +969 -0
- package/dist/extensions/auto-memory/prompts.ts +202 -0
- package/dist/extensions/auto-memory/skip-rules.ts +297 -0
- package/dist/extensions/auto-memory/utils.ts +208 -0
- package/dist/extensions/auto-session-title/index.ts +83 -0
- package/dist/extensions/bash-ext/contract.d.ts +79 -0
- package/dist/extensions/bash-ext/contract.d.ts.map +1 -0
- package/dist/extensions/bash-ext/contract.js +2 -0
- package/dist/extensions/bash-ext/contract.js.map +1 -0
- package/dist/extensions/bash-ext/contract.ts +69 -0
- package/dist/extensions/bash-ext/index.ts +858 -0
- package/dist/extensions/claude-hooks-compat/config-loader.ts +49 -0
- package/dist/extensions/claude-hooks-compat/handler-runner.ts +377 -0
- package/dist/extensions/claude-hooks-compat/if-parser.ts +53 -0
- package/dist/extensions/claude-hooks-compat/index.ts +178 -0
- package/dist/extensions/claude-hooks-compat/matcher.ts +17 -0
- package/dist/extensions/claude-hooks-compat/stdin-builder.ts +27 -0
- package/dist/extensions/claude-hooks-compat/types.ts +77 -0
- package/dist/extensions/compaction-manager/config.ts +47 -0
- package/dist/extensions/compaction-manager/context-fold.ts +63 -0
- package/dist/extensions/compaction-manager/index.ts +151 -0
- package/dist/extensions/compaction-manager/microcompact.ts +49 -0
- package/dist/extensions/compaction-manager/reactive.ts +9 -0
- package/dist/extensions/compaction-manager/session-memory.ts +48 -0
- package/dist/extensions/coordinator/INTEGRATION.md +376 -0
- package/dist/extensions/coordinator/handler.test.ts +277 -0
- package/dist/extensions/coordinator/handler.ts +189 -0
- package/dist/extensions/coordinator/index.ts +261 -0
- package/dist/extensions/coordinator/types.d.ts +100 -0
- package/dist/extensions/coordinator/types.d.ts.map +1 -0
- package/dist/extensions/coordinator/types.js +2 -0
- package/dist/extensions/coordinator/types.js.map +1 -0
- package/dist/extensions/coordinator/types.ts +72 -0
- package/dist/extensions/file-snapshot/index.ts +131 -0
- package/dist/extensions/file-time-guard/README.md +133 -0
- package/dist/extensions/file-time-guard/config.ts +13 -0
- package/dist/extensions/file-time-guard/index.ts +171 -0
- package/dist/extensions/hooks-engine/index.ts +117 -0
- package/dist/extensions/lsp/lsp/client/file-tracker.ts +70 -0
- package/dist/extensions/lsp/lsp/client/registry.ts +305 -0
- package/dist/extensions/lsp/lsp/client/runtime.ts +832 -0
- package/dist/extensions/lsp/lsp/config/resolver.ts +573 -0
- package/dist/extensions/lsp/lsp/contract.d.ts +101 -0
- package/dist/extensions/lsp/lsp/contract.d.ts.map +1 -0
- package/dist/extensions/lsp/lsp/contract.js +2 -0
- package/dist/extensions/lsp/lsp/contract.js.map +1 -0
- package/dist/extensions/lsp/lsp/contract.ts +103 -0
- package/dist/extensions/lsp/lsp/hooks/agent-end.ts +169 -0
- package/dist/extensions/lsp/lsp/hooks/diagnostics-mode.d.ts +10 -0
- package/dist/extensions/lsp/lsp/hooks/diagnostics-mode.d.ts.map +1 -0
- package/dist/extensions/lsp/lsp/hooks/diagnostics-mode.js +30 -0
- package/dist/extensions/lsp/lsp/hooks/diagnostics-mode.js.map +1 -0
- package/dist/extensions/lsp/lsp/hooks/diagnostics-mode.ts +41 -0
- package/dist/extensions/lsp/lsp/hooks/writethrough.ts +342 -0
- package/dist/extensions/lsp/lsp/index.ts +307 -0
- package/dist/extensions/lsp/lsp/lsp.test.ts +684 -0
- package/dist/extensions/lsp/lsp/monitoring/server-metrics.ts +176 -0
- package/dist/extensions/lsp/lsp/tools/lsp-tool.ts +402 -0
- package/dist/extensions/lsp/lsp/utils/dependency-resolver.ts +147 -0
- package/dist/extensions/lsp/lsp/utils/diagnostics-wait.ts +41 -0
- package/dist/extensions/lsp/lsp/utils/lsp-helpers.d.ts +20 -0
- package/dist/extensions/lsp/lsp/utils/lsp-helpers.d.ts.map +1 -0
- package/dist/extensions/lsp/lsp/utils/lsp-helpers.js +64 -0
- package/dist/extensions/lsp/lsp/utils/lsp-helpers.js.map +1 -0
- package/dist/extensions/lsp/lsp/utils/lsp-helpers.ts +76 -0
- package/dist/extensions/message-bridge/GUIDE.md +210 -0
- package/dist/extensions/message-bridge/index.ts +222 -0
- package/dist/extensions/output-guard/index.ts +384 -0
- package/dist/extensions/preview/index.ts +278 -0
- package/dist/extensions/rules-engine/MATCH_HISTORY_RECONCILIATION.md +111 -0
- package/dist/extensions/rules-engine/RULES-ENGINE-GUIDE.md +470 -0
- package/dist/extensions/rules-engine/cache.js +232 -0
- package/dist/extensions/rules-engine/cache.ts +38 -0
- package/dist/extensions/rules-engine/config.js +63 -0
- package/dist/extensions/rules-engine/config.ts +70 -0
- package/dist/extensions/rules-engine/index.js +1530 -0
- package/dist/extensions/rules-engine/index.ts +552 -0
- package/dist/extensions/rules-engine/injector.js +68 -0
- package/dist/extensions/rules-engine/injector.ts +74 -0
- package/dist/extensions/rules-engine/loader.js +179 -0
- package/dist/extensions/rules-engine/loader.ts +205 -0
- package/dist/extensions/rules-engine/matcher.js +534 -0
- package/dist/extensions/rules-engine/matcher.ts +52 -0
- package/dist/extensions/rules-engine/types.d.ts +156 -0
- package/dist/extensions/rules-engine/types.d.ts.map +1 -0
- package/dist/extensions/rules-engine/types.js +2 -0
- package/dist/extensions/rules-engine/types.js.map +1 -0
- package/dist/extensions/rules-engine/types.ts +169 -0
- package/dist/extensions/session-supervisor/checker.ts +116 -0
- package/dist/extensions/session-supervisor/config.ts +45 -0
- package/dist/extensions/session-supervisor/index.ts +726 -0
- package/dist/extensions/session-supervisor/prompts.ts +132 -0
- package/dist/extensions/session-supervisor/scheduler.ts +69 -0
- package/dist/extensions/session-supervisor/types.ts +215 -0
- package/dist/extensions/subagent/README.md +172 -0
- package/dist/extensions/subagent/agents/explorer.md +25 -0
- package/dist/extensions/subagent/agents/guide.md +27 -0
- package/dist/extensions/subagent/agents/planner.md +37 -0
- package/dist/extensions/subagent/agents/reviewer.md +35 -0
- package/dist/extensions/subagent/agents/scout.md +50 -0
- package/dist/extensions/subagent/agents/verification.md +35 -0
- package/dist/extensions/subagent/agents/worker.md +24 -0
- package/dist/extensions/subagent/agents.ts +25 -0
- package/dist/extensions/subagent/index.ts +987 -0
- package/dist/extensions/subagent/prompts/implement-and-review.md +10 -0
- package/dist/extensions/subagent/prompts/implement.md +10 -0
- package/dist/extensions/subagent/prompts/scout-and-plan.md +9 -0
- package/dist/extensions/subagent-ext/contract.d.ts +2 -0
- package/dist/extensions/subagent-ext/contract.d.ts.map +1 -0
- package/dist/extensions/subagent-ext/contract.js +2 -0
- package/dist/extensions/subagent-ext/contract.js.map +1 -0
- package/dist/extensions/subagent-ext/contract.ts +1 -0
- package/dist/extensions/subagent-ext/index.ts +347 -0
- package/dist/extensions/subagent-shared/contract.d.ts +25 -0
- package/dist/extensions/subagent-shared/contract.d.ts.map +1 -0
- package/dist/extensions/subagent-shared/contract.js +2 -0
- package/dist/extensions/subagent-shared/contract.js.map +1 -0
- package/dist/extensions/subagent-shared/contract.ts +28 -0
- package/dist/extensions/subagent-shared/index.ts +4 -0
- package/dist/extensions/subagent-shared/render.ts +166 -0
- package/dist/extensions/subagent-shared/types.ts +35 -0
- package/dist/extensions/subagent-shared/utils.ts +112 -0
- package/dist/extensions/subagent-v2/contract.d.ts +2 -0
- package/dist/extensions/subagent-v2/contract.d.ts.map +1 -0
- package/dist/extensions/subagent-v2/contract.js +2 -0
- package/dist/extensions/subagent-v2/contract.js.map +1 -0
- package/dist/extensions/subagent-v2/contract.ts +1 -0
- package/dist/extensions/subagent-v2/index.ts +599 -0
- package/dist/extensions/todo-ext/contract.d.ts +27 -0
- package/dist/extensions/todo-ext/contract.d.ts.map +1 -0
- package/dist/extensions/todo-ext/contract.js +2 -0
- package/dist/extensions/todo-ext/contract.js.map +1 -0
- package/dist/extensions/todo-ext/contract.ts +30 -0
- package/dist/extensions/todo-ext/index.ts +419 -0
- package/package.json +6 -5
|
@@ -0,0 +1,179 @@
|
|
|
1
|
+
// packages/coding-agent/extensions/rules-engine/loader.ts
|
|
2
|
+
import * as fs from "node:fs";
|
|
3
|
+
import * as path from "node:path";
|
|
4
|
+
function splitComma(val) {
|
|
5
|
+
const result = [];
|
|
6
|
+
let depth = 0;
|
|
7
|
+
let current = "";
|
|
8
|
+
for (const ch of val) {
|
|
9
|
+
if (ch === "{" || ch === "(" || ch === "[") depth++;
|
|
10
|
+
else if (ch === "}" || ch === ")" || ch === "]") depth--;
|
|
11
|
+
if (ch === "," && depth === 0) {
|
|
12
|
+
const trimmed2 = current.trim().replace(/^["']|["']$/g, "");
|
|
13
|
+
if (trimmed2) result.push(trimmed2);
|
|
14
|
+
current = "";
|
|
15
|
+
} else {
|
|
16
|
+
current += ch;
|
|
17
|
+
}
|
|
18
|
+
}
|
|
19
|
+
const trimmed = current.trim().replace(/^["']|["']$/g, "");
|
|
20
|
+
if (trimmed) result.push(trimmed);
|
|
21
|
+
return result;
|
|
22
|
+
}
|
|
23
|
+
function parseFrontmatter(content) {
|
|
24
|
+
const frontmatterRegex = /^---\r?\n([\s\S]*?)\n?---\r?\n([\s\S]*)$/;
|
|
25
|
+
const match = content.match(frontmatterRegex);
|
|
26
|
+
if (!match) {
|
|
27
|
+
return { data: {}, body: content };
|
|
28
|
+
}
|
|
29
|
+
const [, frontmatterStr, body] = match;
|
|
30
|
+
const data = {};
|
|
31
|
+
const lines = frontmatterStr.split("\n");
|
|
32
|
+
let i = 0;
|
|
33
|
+
while (i < lines.length) {
|
|
34
|
+
const line = lines[i];
|
|
35
|
+
const colonIndex = line.indexOf(":");
|
|
36
|
+
if (colonIndex === -1) {
|
|
37
|
+
i++;
|
|
38
|
+
continue;
|
|
39
|
+
}
|
|
40
|
+
const rawKey = line.slice(0, colonIndex).trim();
|
|
41
|
+
let value = line.slice(colonIndex + 1).trim();
|
|
42
|
+
if (value === "" || value === "null" || value === "undefined") {
|
|
43
|
+
const listItems = [];
|
|
44
|
+
let j = i + 1;
|
|
45
|
+
while (j < lines.length) {
|
|
46
|
+
const subLine = lines[j];
|
|
47
|
+
if (subLine.match(/^\s*-\s+/)) {
|
|
48
|
+
listItems.push(
|
|
49
|
+
subLine.replace(/^\s*-\s+/, "").trim().replace(/^["']|["']$/g, "")
|
|
50
|
+
);
|
|
51
|
+
j++;
|
|
52
|
+
} else if (subLine.trim() === "" || subLine.match(/^\s+/)) {
|
|
53
|
+
j++;
|
|
54
|
+
} else {
|
|
55
|
+
break;
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
if (listItems.length > 0) {
|
|
59
|
+
const camelKey2 = rawKey.replace(/-([a-z])/g, (_, letter) => letter.toUpperCase());
|
|
60
|
+
data[camelKey2] = listItems;
|
|
61
|
+
i = j;
|
|
62
|
+
continue;
|
|
63
|
+
}
|
|
64
|
+
value = null;
|
|
65
|
+
} else if (value.startsWith("[") && value.endsWith("]")) {
|
|
66
|
+
try {
|
|
67
|
+
value = JSON.parse(value.replace(/'/g, '"'));
|
|
68
|
+
} catch (err) {
|
|
69
|
+
console.debug("[rules-engine] JSON array parse failed:", err instanceof Error ? err.message : err);
|
|
70
|
+
value = value.slice(1, -1).split(",").map((v) => v.trim().replace(/^["']|["']$/g, ""));
|
|
71
|
+
}
|
|
72
|
+
} else if (value.startsWith('"') && value.endsWith('"')) {
|
|
73
|
+
value = value.slice(1, -1);
|
|
74
|
+
} else if (value.startsWith("'") && value.endsWith("'")) {
|
|
75
|
+
value = value.slice(1, -1);
|
|
76
|
+
}
|
|
77
|
+
const camelKey = rawKey.replace(/-([a-z])/g, (_, letter) => letter.toUpperCase());
|
|
78
|
+
if ((camelKey === "paths" || camelKey === "globs") && typeof value === "string") {
|
|
79
|
+
data[camelKey] = splitComma(value);
|
|
80
|
+
} else {
|
|
81
|
+
data[camelKey] = value;
|
|
82
|
+
}
|
|
83
|
+
i++;
|
|
84
|
+
}
|
|
85
|
+
return { data, body: body.trim() };
|
|
86
|
+
}
|
|
87
|
+
function extractTitle(body) {
|
|
88
|
+
for (const line of body.split("\n")) {
|
|
89
|
+
const trimmed = line.trim();
|
|
90
|
+
if (trimmed) {
|
|
91
|
+
return trimmed.replace(/^#+\s*/, "").replace(/\*\*/g, "");
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
return "Untitled Rule";
|
|
95
|
+
}
|
|
96
|
+
function parsePaths(raw) {
|
|
97
|
+
if (!raw) return [];
|
|
98
|
+
if (typeof raw === "string") {
|
|
99
|
+
return raw.split(",").map((g) => g.trim()).filter(Boolean);
|
|
100
|
+
}
|
|
101
|
+
if (Array.isArray(raw)) return raw;
|
|
102
|
+
return [];
|
|
103
|
+
}
|
|
104
|
+
function parseRuleFile(filePath, content) {
|
|
105
|
+
const { data, body } = parseFrontmatter(content);
|
|
106
|
+
const rawGlobs = data.globs ?? data.paths;
|
|
107
|
+
const globs = parsePaths(rawGlobs);
|
|
108
|
+
const rawPaths = data.paths;
|
|
109
|
+
const paths = parsePaths(rawPaths);
|
|
110
|
+
const isUnconditional = globs.length === 0 || globs.length === 1 && globs[0] === "**";
|
|
111
|
+
const frontmatter = {};
|
|
112
|
+
if (rawGlobs) {
|
|
113
|
+
frontmatter.globs = globs;
|
|
114
|
+
if (rawPaths) {
|
|
115
|
+
frontmatter.paths = paths;
|
|
116
|
+
} else {
|
|
117
|
+
frontmatter.paths = globs;
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
if (data.description && typeof data.description === "string") frontmatter.description = data.description;
|
|
121
|
+
if (data.severity && typeof data.severity === "string")
|
|
122
|
+
frontmatter.severity = data.severity;
|
|
123
|
+
if (data.allowedTools)
|
|
124
|
+
frontmatter.allowedTools = typeof data.allowedTools === "string" ? [data.allowedTools] : data.allowedTools;
|
|
125
|
+
if (data.whenToUse && typeof data.whenToUse === "string") frontmatter.whenToUse = data.whenToUse;
|
|
126
|
+
if (data.notifyOnMatch !== void 0)
|
|
127
|
+
frontmatter.notifyOnMatch = data.notifyOnMatch === "true" || data.notifyOnMatch === true;
|
|
128
|
+
if (data.skipInPrompt !== void 0)
|
|
129
|
+
frontmatter.skipInPrompt = data.skipInPrompt === "true" || data.skipInPrompt === true;
|
|
130
|
+
return {
|
|
131
|
+
name: path.basename(filePath, path.extname(filePath)),
|
|
132
|
+
filePath,
|
|
133
|
+
title: extractTitle(body),
|
|
134
|
+
content: body.trim(),
|
|
135
|
+
scope: "project",
|
|
136
|
+
source: "",
|
|
137
|
+
frontmatter,
|
|
138
|
+
isUnconditional
|
|
139
|
+
};
|
|
140
|
+
}
|
|
141
|
+
function scanDir(dir, files = []) {
|
|
142
|
+
if (!fs.existsSync(dir)) return files;
|
|
143
|
+
const entries = fs.readdirSync(dir, { withFileTypes: true });
|
|
144
|
+
for (const entry of entries) {
|
|
145
|
+
const fullPath = path.join(dir, entry.name);
|
|
146
|
+
if (entry.isDirectory()) {
|
|
147
|
+
scanDir(fullPath, files);
|
|
148
|
+
} else if (entry.isFile() && (entry.name.endsWith(".md") || entry.name.endsWith(".mdc"))) {
|
|
149
|
+
files.push(fullPath);
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
return files;
|
|
153
|
+
}
|
|
154
|
+
function loadRules(rulesDir) {
|
|
155
|
+
const rules = [];
|
|
156
|
+
const unconditional = [];
|
|
157
|
+
const conditional = [];
|
|
158
|
+
if (!fs.existsSync(rulesDir)) {
|
|
159
|
+
return { rules, unconditional, conditional, loadedAt: Date.now() };
|
|
160
|
+
}
|
|
161
|
+
const files = scanDir(rulesDir);
|
|
162
|
+
for (const filePath of files) {
|
|
163
|
+
const content = fs.readFileSync(filePath, "utf-8");
|
|
164
|
+
const rule = parseRuleFile(filePath, content);
|
|
165
|
+
rules.push(rule);
|
|
166
|
+
if (rule.isUnconditional) {
|
|
167
|
+
unconditional.push(rule);
|
|
168
|
+
} else {
|
|
169
|
+
conditional.push(rule);
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
return { rules, unconditional, conditional, loadedAt: Date.now() };
|
|
173
|
+
}
|
|
174
|
+
export {
|
|
175
|
+
loadRules,
|
|
176
|
+
parseFrontmatter,
|
|
177
|
+
parseRuleFile
|
|
178
|
+
};
|
|
179
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["loader.ts"],
  "sourcesContent": ["import * as fs from \"node:fs\";\nimport * as path from \"node:path\";\nimport type { ParsedRule, RuleCache, RuleFrontmatter } from \"./types.js\";\n\nfunction splitComma(val: string): string[] {\n\tconst result: string[] = [];\n\tlet depth = 0;\n\tlet current = \"\";\n\tfor (const ch of val) {\n\t\tif (ch === \"{\" || ch === \"(\" || ch === \"[\") depth++;\n\t\telse if (ch === \"}\" || ch === \")\" || ch === \"]\") depth--;\n\n\t\tif (ch === \",\" && depth === 0) {\n\t\t\tconst trimmed = current.trim().replace(/^[\"']|[\"']$/g, \"\");\n\t\t\tif (trimmed) result.push(trimmed);\n\t\t\tcurrent = \"\";\n\t\t} else {\n\t\t\tcurrent += ch;\n\t\t}\n\t}\n\tconst trimmed = current.trim().replace(/^[\"']|[\"']$/g, \"\");\n\tif (trimmed) result.push(trimmed);\n\treturn result;\n}\n\nexport function parseFrontmatter(content: string): { data: Record<string, unknown>; body: string } {\n\tconst frontmatterRegex = /^---\\r?\\n([\\s\\S]*?)\\n?---\\r?\\n([\\s\\S]*)$/;\n\tconst match = content.match(frontmatterRegex);\n\n\tif (!match) {\n\t\treturn { data: {}, body: content };\n\t}\n\n\tconst [, frontmatterStr, body] = match;\n\tconst data: Record<string, unknown> = {};\n\n\tconst lines = frontmatterStr.split(\"\\n\");\n\tlet i = 0;\n\twhile (i < lines.length) {\n\t\tconst line = lines[i];\n\t\tconst colonIndex = line.indexOf(\":\");\n\t\tif (colonIndex === -1) {\n\t\t\ti++;\n\t\t\tcontinue;\n\t\t}\n\n\t\tconst rawKey = line.slice(0, colonIndex).trim();\n\t\tlet value: string | string[] | null = line.slice(colonIndex + 1).trim();\n\n\t\tif (value === \"\" || value === \"null\" || value === \"undefined\") {\n\t\t\tconst listItems: string[] = [];\n\t\t\tlet j = i + 1;\n\t\t\twhile (j < lines.length) {\n\t\t\t\tconst subLine = lines[j];\n\t\t\t\tif (subLine.match(/^\\s*-\\s+/)) {\n\t\t\t\t\tlistItems.push(\n\t\t\t\t\t\tsubLine\n\t\t\t\t\t\t\t.replace(/^\\s*-\\s+/, \"\")\n\t\t\t\t\t\t\t.trim()\n\t\t\t\t\t\t\t.replace(/^[\"']|[\"']$/g, \"\"),\n\t\t\t\t\t);\n\t\t\t\t\tj++;\n\t\t\t\t} else if (subLine.trim() === \"\" || subLine.match(/^\\s+/)) {\n\t\t\t\t\tj++;\n\t\t\t\t} else {\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t}\n\t\t\tif (listItems.length > 0) {\n\t\t\t\tconst camelKey = rawKey.replace(/-([a-z])/g, (_, letter) => letter.toUpperCase());\n\t\t\t\tdata[camelKey] = listItems;\n\t\t\t\ti = j;\n\t\t\t\tcontinue;\n\t\t\t}\n\t\t\tvalue = null;\n\t\t} else if (value.startsWith(\"[\") && value.endsWith(\"]\")) {\n\t\t\ttry {\n\t\t\t\tvalue = JSON.parse(value.replace(/'/g, '\"'));\n\t\t\t} catch (err) {\n\t\t\t\tconsole.debug(\"[rules-engine] JSON array parse failed:\", err instanceof Error ? err.message : err);\n\t\t\t\tvalue = (value as string)\n\t\t\t\t\t.slice(1, -1)\n\t\t\t\t\t.split(\",\")\n\t\t\t\t\t.map((v: string) => v.trim().replace(/^[\"']|[\"']$/g, \"\"));\n\t\t\t}\n\t\t} else if (value.startsWith('\"') && value.endsWith('\"')) {\n\t\t\tvalue = value.slice(1, -1);\n\t\t} else if (value.startsWith(\"'\") && value.endsWith(\"'\")) {\n\t\t\tvalue = value.slice(1, -1);\n\t\t}\n\n\t\tconst camelKey = rawKey.replace(/-([a-z])/g, (_, letter) => letter.toUpperCase());\n\n\t\tif ((camelKey === \"paths\" || camelKey === \"globs\") && typeof value === \"string\") {\n\t\t\tdata[camelKey] = splitComma(value);\n\t\t} else {\n\t\t\tdata[camelKey] = value;\n\t\t}\n\t\ti++;\n\t}\n\n\treturn { data, body: body.trim() };\n}\n\nfunction extractTitle(body: string): string {\n\tfor (const line of body.split(\"\\n\")) {\n\t\tconst trimmed = line.trim();\n\t\tif (trimmed) {\n\t\t\treturn trimmed.replace(/^#+\\s*/, \"\").replace(/\\*\\*/g, \"\");\n\t\t}\n\t}\n\treturn \"Untitled Rule\";\n}\n\nfunction parsePaths(raw: unknown): string[] {\n\tif (!raw) return [];\n\tif (typeof raw === \"string\") {\n\t\treturn raw\n\t\t\t.split(\",\")\n\t\t\t.map((g) => g.trim())\n\t\t\t.filter(Boolean);\n\t}\n\tif (Array.isArray(raw)) return raw as string[];\n\treturn [];\n}\n\nexport function parseRuleFile(filePath: string, content: string): ParsedRule {\n\tconst { data, body } = parseFrontmatter(content);\n\tconst rawGlobs = data.globs ?? data.paths;\n\tconst globs = parsePaths(rawGlobs);\n\tconst rawPaths = data.paths;\n\tconst paths = parsePaths(rawPaths);\n\tconst isUnconditional = globs.length === 0 || (globs.length === 1 && globs[0] === \"**\");\n\n\tconst frontmatter: RuleFrontmatter = {};\n\tif (rawGlobs) {\n\t\tfrontmatter.globs = globs;\n\t\tif (rawPaths) {\n\t\t\tfrontmatter.paths = paths;\n\t\t} else {\n\t\t\tfrontmatter.paths = globs;\n\t\t}\n\t}\n\tif (data.description && typeof data.description === \"string\") frontmatter.description = data.description;\n\tif (data.severity && typeof data.severity === \"string\")\n\t\tfrontmatter.severity = data.severity as ParsedRule[\"frontmatter\"][\"severity\"];\n\tif (data.allowedTools)\n\t\tfrontmatter.allowedTools =\n\t\t\ttypeof data.allowedTools === \"string\" ? [data.allowedTools] : (data.allowedTools as string[]);\n\tif (data.whenToUse && typeof data.whenToUse === \"string\") frontmatter.whenToUse = data.whenToUse;\n\tif (data.notifyOnMatch !== undefined)\n\t\tfrontmatter.notifyOnMatch = data.notifyOnMatch === \"true\" || data.notifyOnMatch === true;\n\tif (data.skipInPrompt !== undefined)\n\t\tfrontmatter.skipInPrompt = data.skipInPrompt === \"true\" || data.skipInPrompt === true;\n\n\treturn {\n\t\tname: path.basename(filePath, path.extname(filePath)),\n\t\tfilePath,\n\t\ttitle: extractTitle(body),\n\t\tcontent: body.trim(),\n\t\tscope: \"project\",\n\t\tsource: \"\",\n\t\tfrontmatter,\n\t\tisUnconditional,\n\t};\n}\n\nfunction scanDir(dir: string, files: string[] = []): string[] {\n\tif (!fs.existsSync(dir)) return files;\n\tconst entries = fs.readdirSync(dir, { withFileTypes: true });\n\tfor (const entry of entries) {\n\t\tconst fullPath = path.join(dir, entry.name);\n\t\tif (entry.isDirectory()) {\n\t\t\tscanDir(fullPath, files);\n\t\t} else if (entry.isFile() && (entry.name.endsWith(\".md\") || entry.name.endsWith(\".mdc\"))) {\n\t\t\tfiles.push(fullPath);\n\t\t}\n\t}\n\treturn files;\n}\n\nexport function loadRules(rulesDir: string): RuleCache {\n\tconst rules: ParsedRule[] = [];\n\tconst unconditional: ParsedRule[] = [];\n\tconst conditional: ParsedRule[] = [];\n\n\tif (!fs.existsSync(rulesDir)) {\n\t\treturn { rules, unconditional, conditional, loadedAt: Date.now() };\n\t}\n\n\tconst files = scanDir(rulesDir);\n\n\tfor (const filePath of files) {\n\t\tconst content = fs.readFileSync(filePath, \"utf-8\");\n\t\tconst rule = parseRuleFile(filePath, content);\n\t\trules.push(rule);\n\t\tif (rule.isUnconditional) {\n\t\t\tunconditional.push(rule);\n\t\t} else {\n\t\t\tconditional.push(rule);\n\t\t}\n\t}\n\n\treturn { rules, unconditional, conditional, loadedAt: Date.now() };\n}\n"],
  "mappings": "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",
  "names": ["trimmed", "camelKey"]
}

|
|
@@ -0,0 +1,205 @@
|
|
|
1
|
+
import * as fs from "node:fs";
|
|
2
|
+
import * as path from "node:path";
|
|
3
|
+
import type { ParsedRule, RuleCache, RuleFrontmatter } from "./types.js";
|
|
4
|
+
|
|
5
|
+
function splitComma(val: string): string[] {
|
|
6
|
+
const result: string[] = [];
|
|
7
|
+
let depth = 0;
|
|
8
|
+
let current = "";
|
|
9
|
+
for (const ch of val) {
|
|
10
|
+
if (ch === "{" || ch === "(" || ch === "[") depth++;
|
|
11
|
+
else if (ch === "}" || ch === ")" || ch === "]") depth--;
|
|
12
|
+
|
|
13
|
+
if (ch === "," && depth === 0) {
|
|
14
|
+
const trimmed = current.trim().replace(/^["']|["']$/g, "");
|
|
15
|
+
if (trimmed) result.push(trimmed);
|
|
16
|
+
current = "";
|
|
17
|
+
} else {
|
|
18
|
+
current += ch;
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
const trimmed = current.trim().replace(/^["']|["']$/g, "");
|
|
22
|
+
if (trimmed) result.push(trimmed);
|
|
23
|
+
return result;
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
export function parseFrontmatter(content: string): { data: Record<string, unknown>; body: string } {
|
|
27
|
+
const frontmatterRegex = /^---\r?\n([\s\S]*?)\n?---\r?\n([\s\S]*)$/;
|
|
28
|
+
const match = content.match(frontmatterRegex);
|
|
29
|
+
|
|
30
|
+
if (!match) {
|
|
31
|
+
return { data: {}, body: content };
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
const [, frontmatterStr, body] = match;
|
|
35
|
+
const data: Record<string, unknown> = {};
|
|
36
|
+
|
|
37
|
+
const lines = frontmatterStr.split("\n");
|
|
38
|
+
let i = 0;
|
|
39
|
+
while (i < lines.length) {
|
|
40
|
+
const line = lines[i];
|
|
41
|
+
const colonIndex = line.indexOf(":");
|
|
42
|
+
if (colonIndex === -1) {
|
|
43
|
+
i++;
|
|
44
|
+
continue;
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
const rawKey = line.slice(0, colonIndex).trim();
|
|
48
|
+
let value: string | string[] | null = line.slice(colonIndex + 1).trim();
|
|
49
|
+
|
|
50
|
+
if (value === "" || value === "null" || value === "undefined") {
|
|
51
|
+
const listItems: string[] = [];
|
|
52
|
+
let j = i + 1;
|
|
53
|
+
while (j < lines.length) {
|
|
54
|
+
const subLine = lines[j];
|
|
55
|
+
if (subLine.match(/^\s*-\s+/)) {
|
|
56
|
+
listItems.push(
|
|
57
|
+
subLine
|
|
58
|
+
.replace(/^\s*-\s+/, "")
|
|
59
|
+
.trim()
|
|
60
|
+
.replace(/^["']|["']$/g, ""),
|
|
61
|
+
);
|
|
62
|
+
j++;
|
|
63
|
+
} else if (subLine.trim() === "" || subLine.match(/^\s+/)) {
|
|
64
|
+
j++;
|
|
65
|
+
} else {
|
|
66
|
+
break;
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
if (listItems.length > 0) {
|
|
70
|
+
const camelKey = rawKey.replace(/-([a-z])/g, (_, letter) => letter.toUpperCase());
|
|
71
|
+
data[camelKey] = listItems;
|
|
72
|
+
i = j;
|
|
73
|
+
continue;
|
|
74
|
+
}
|
|
75
|
+
value = null;
|
|
76
|
+
} else if (value.startsWith("[") && value.endsWith("]")) {
|
|
77
|
+
try {
|
|
78
|
+
value = JSON.parse(value.replace(/'/g, '"'));
|
|
79
|
+
} catch (err) {
|
|
80
|
+
console.debug("[rules-engine] JSON array parse failed:", err instanceof Error ? err.message : err);
|
|
81
|
+
value = (value as string)
|
|
82
|
+
.slice(1, -1)
|
|
83
|
+
.split(",")
|
|
84
|
+
.map((v: string) => v.trim().replace(/^["']|["']$/g, ""));
|
|
85
|
+
}
|
|
86
|
+
} else if (value.startsWith('"') && value.endsWith('"')) {
|
|
87
|
+
value = value.slice(1, -1);
|
|
88
|
+
} else if (value.startsWith("'") && value.endsWith("'")) {
|
|
89
|
+
value = value.slice(1, -1);
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
const camelKey = rawKey.replace(/-([a-z])/g, (_, letter) => letter.toUpperCase());
|
|
93
|
+
|
|
94
|
+
if ((camelKey === "paths" || camelKey === "globs") && typeof value === "string") {
|
|
95
|
+
data[camelKey] = splitComma(value);
|
|
96
|
+
} else {
|
|
97
|
+
data[camelKey] = value;
|
|
98
|
+
}
|
|
99
|
+
i++;
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
return { data, body: body.trim() };
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
function extractTitle(body: string): string {
|
|
106
|
+
for (const line of body.split("\n")) {
|
|
107
|
+
const trimmed = line.trim();
|
|
108
|
+
if (trimmed) {
|
|
109
|
+
return trimmed.replace(/^#+\s*/, "").replace(/\*\*/g, "");
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
return "Untitled Rule";
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
function parsePaths(raw: unknown): string[] {
|
|
116
|
+
if (!raw) return [];
|
|
117
|
+
if (typeof raw === "string") {
|
|
118
|
+
return raw
|
|
119
|
+
.split(",")
|
|
120
|
+
.map((g) => g.trim())
|
|
121
|
+
.filter(Boolean);
|
|
122
|
+
}
|
|
123
|
+
if (Array.isArray(raw)) return raw as string[];
|
|
124
|
+
return [];
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
export function parseRuleFile(filePath: string, content: string): ParsedRule {
|
|
128
|
+
const { data, body } = parseFrontmatter(content);
|
|
129
|
+
const rawGlobs = data.globs ?? data.paths;
|
|
130
|
+
const globs = parsePaths(rawGlobs);
|
|
131
|
+
const rawPaths = data.paths;
|
|
132
|
+
const paths = parsePaths(rawPaths);
|
|
133
|
+
const isUnconditional = globs.length === 0 || (globs.length === 1 && globs[0] === "**");
|
|
134
|
+
|
|
135
|
+
const frontmatter: RuleFrontmatter = {};
|
|
136
|
+
if (rawGlobs) {
|
|
137
|
+
frontmatter.globs = globs;
|
|
138
|
+
if (rawPaths) {
|
|
139
|
+
frontmatter.paths = paths;
|
|
140
|
+
} else {
|
|
141
|
+
frontmatter.paths = globs;
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
if (data.description && typeof data.description === "string") frontmatter.description = data.description;
|
|
145
|
+
if (data.severity && typeof data.severity === "string")
|
|
146
|
+
frontmatter.severity = data.severity as ParsedRule["frontmatter"]["severity"];
|
|
147
|
+
if (data.allowedTools)
|
|
148
|
+
frontmatter.allowedTools =
|
|
149
|
+
typeof data.allowedTools === "string" ? [data.allowedTools] : (data.allowedTools as string[]);
|
|
150
|
+
if (data.whenToUse && typeof data.whenToUse === "string") frontmatter.whenToUse = data.whenToUse;
|
|
151
|
+
if (data.notifyOnMatch !== undefined)
|
|
152
|
+
frontmatter.notifyOnMatch = data.notifyOnMatch === "true" || data.notifyOnMatch === true;
|
|
153
|
+
if (data.skipInPrompt !== undefined)
|
|
154
|
+
frontmatter.skipInPrompt = data.skipInPrompt === "true" || data.skipInPrompt === true;
|
|
155
|
+
|
|
156
|
+
return {
|
|
157
|
+
name: path.basename(filePath, path.extname(filePath)),
|
|
158
|
+
filePath,
|
|
159
|
+
title: extractTitle(body),
|
|
160
|
+
content: body.trim(),
|
|
161
|
+
scope: "project",
|
|
162
|
+
source: "",
|
|
163
|
+
frontmatter,
|
|
164
|
+
isUnconditional,
|
|
165
|
+
};
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
function scanDir(dir: string, files: string[] = []): string[] {
|
|
169
|
+
if (!fs.existsSync(dir)) return files;
|
|
170
|
+
const entries = fs.readdirSync(dir, { withFileTypes: true });
|
|
171
|
+
for (const entry of entries) {
|
|
172
|
+
const fullPath = path.join(dir, entry.name);
|
|
173
|
+
if (entry.isDirectory()) {
|
|
174
|
+
scanDir(fullPath, files);
|
|
175
|
+
} else if (entry.isFile() && (entry.name.endsWith(".md") || entry.name.endsWith(".mdc"))) {
|
|
176
|
+
files.push(fullPath);
|
|
177
|
+
}
|
|
178
|
+
}
|
|
179
|
+
return files;
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
export function loadRules(rulesDir: string): RuleCache {
|
|
183
|
+
const rules: ParsedRule[] = [];
|
|
184
|
+
const unconditional: ParsedRule[] = [];
|
|
185
|
+
const conditional: ParsedRule[] = [];
|
|
186
|
+
|
|
187
|
+
if (!fs.existsSync(rulesDir)) {
|
|
188
|
+
return { rules, unconditional, conditional, loadedAt: Date.now() };
|
|
189
|
+
}
|
|
190
|
+
|
|
191
|
+
const files = scanDir(rulesDir);
|
|
192
|
+
|
|
193
|
+
for (const filePath of files) {
|
|
194
|
+
const content = fs.readFileSync(filePath, "utf-8");
|
|
195
|
+
const rule = parseRuleFile(filePath, content);
|
|
196
|
+
rules.push(rule);
|
|
197
|
+
if (rule.isUnconditional) {
|
|
198
|
+
unconditional.push(rule);
|
|
199
|
+
} else {
|
|
200
|
+
conditional.push(rule);
|
|
201
|
+
}
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
return { rules, unconditional, conditional, loadedAt: Date.now() };
|
|
205
|
+
}
|