@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.
Files changed (145) hide show
  1. package/dist/core/agent-session.d.ts.map +1 -1
  2. package/dist/core/agent-session.js +3 -0
  3. package/dist/core/agent-session.js.map +1 -1
  4. package/dist/extensions/agent-permissions/index.ts +235 -0
  5. package/dist/extensions/ask-tools/index.ts +115 -0
  6. package/dist/extensions/auto-memory/contract.d.ts +51 -0
  7. package/dist/extensions/auto-memory/contract.d.ts.map +1 -0
  8. package/dist/extensions/auto-memory/contract.js +2 -0
  9. package/dist/extensions/auto-memory/contract.js.map +1 -0
  10. package/dist/extensions/auto-memory/contract.ts +56 -0
  11. package/dist/extensions/auto-memory/index.ts +969 -0
  12. package/dist/extensions/auto-memory/prompts.ts +202 -0
  13. package/dist/extensions/auto-memory/skip-rules.ts +297 -0
  14. package/dist/extensions/auto-memory/utils.ts +208 -0
  15. package/dist/extensions/auto-session-title/index.ts +83 -0
  16. package/dist/extensions/bash-ext/contract.d.ts +79 -0
  17. package/dist/extensions/bash-ext/contract.d.ts.map +1 -0
  18. package/dist/extensions/bash-ext/contract.js +2 -0
  19. package/dist/extensions/bash-ext/contract.js.map +1 -0
  20. package/dist/extensions/bash-ext/contract.ts +69 -0
  21. package/dist/extensions/bash-ext/index.ts +858 -0
  22. package/dist/extensions/claude-hooks-compat/config-loader.ts +49 -0
  23. package/dist/extensions/claude-hooks-compat/handler-runner.ts +377 -0
  24. package/dist/extensions/claude-hooks-compat/if-parser.ts +53 -0
  25. package/dist/extensions/claude-hooks-compat/index.ts +178 -0
  26. package/dist/extensions/claude-hooks-compat/matcher.ts +17 -0
  27. package/dist/extensions/claude-hooks-compat/stdin-builder.ts +27 -0
  28. package/dist/extensions/claude-hooks-compat/types.ts +77 -0
  29. package/dist/extensions/compaction-manager/config.ts +47 -0
  30. package/dist/extensions/compaction-manager/context-fold.ts +63 -0
  31. package/dist/extensions/compaction-manager/index.ts +151 -0
  32. package/dist/extensions/compaction-manager/microcompact.ts +49 -0
  33. package/dist/extensions/compaction-manager/reactive.ts +9 -0
  34. package/dist/extensions/compaction-manager/session-memory.ts +48 -0
  35. package/dist/extensions/coordinator/INTEGRATION.md +376 -0
  36. package/dist/extensions/coordinator/handler.test.ts +277 -0
  37. package/dist/extensions/coordinator/handler.ts +189 -0
  38. package/dist/extensions/coordinator/index.ts +261 -0
  39. package/dist/extensions/coordinator/types.d.ts +100 -0
  40. package/dist/extensions/coordinator/types.d.ts.map +1 -0
  41. package/dist/extensions/coordinator/types.js +2 -0
  42. package/dist/extensions/coordinator/types.js.map +1 -0
  43. package/dist/extensions/coordinator/types.ts +72 -0
  44. package/dist/extensions/file-snapshot/index.ts +131 -0
  45. package/dist/extensions/file-time-guard/README.md +133 -0
  46. package/dist/extensions/file-time-guard/config.ts +13 -0
  47. package/dist/extensions/file-time-guard/index.ts +171 -0
  48. package/dist/extensions/hooks-engine/index.ts +117 -0
  49. package/dist/extensions/lsp/lsp/client/file-tracker.ts +70 -0
  50. package/dist/extensions/lsp/lsp/client/registry.ts +305 -0
  51. package/dist/extensions/lsp/lsp/client/runtime.ts +832 -0
  52. package/dist/extensions/lsp/lsp/config/resolver.ts +573 -0
  53. package/dist/extensions/lsp/lsp/contract.d.ts +101 -0
  54. package/dist/extensions/lsp/lsp/contract.d.ts.map +1 -0
  55. package/dist/extensions/lsp/lsp/contract.js +2 -0
  56. package/dist/extensions/lsp/lsp/contract.js.map +1 -0
  57. package/dist/extensions/lsp/lsp/contract.ts +103 -0
  58. package/dist/extensions/lsp/lsp/hooks/agent-end.ts +169 -0
  59. package/dist/extensions/lsp/lsp/hooks/diagnostics-mode.d.ts +10 -0
  60. package/dist/extensions/lsp/lsp/hooks/diagnostics-mode.d.ts.map +1 -0
  61. package/dist/extensions/lsp/lsp/hooks/diagnostics-mode.js +30 -0
  62. package/dist/extensions/lsp/lsp/hooks/diagnostics-mode.js.map +1 -0
  63. package/dist/extensions/lsp/lsp/hooks/diagnostics-mode.ts +41 -0
  64. package/dist/extensions/lsp/lsp/hooks/writethrough.ts +342 -0
  65. package/dist/extensions/lsp/lsp/index.ts +307 -0
  66. package/dist/extensions/lsp/lsp/lsp.test.ts +684 -0
  67. package/dist/extensions/lsp/lsp/monitoring/server-metrics.ts +176 -0
  68. package/dist/extensions/lsp/lsp/tools/lsp-tool.ts +402 -0
  69. package/dist/extensions/lsp/lsp/utils/dependency-resolver.ts +147 -0
  70. package/dist/extensions/lsp/lsp/utils/diagnostics-wait.ts +41 -0
  71. package/dist/extensions/lsp/lsp/utils/lsp-helpers.d.ts +20 -0
  72. package/dist/extensions/lsp/lsp/utils/lsp-helpers.d.ts.map +1 -0
  73. package/dist/extensions/lsp/lsp/utils/lsp-helpers.js +64 -0
  74. package/dist/extensions/lsp/lsp/utils/lsp-helpers.js.map +1 -0
  75. package/dist/extensions/lsp/lsp/utils/lsp-helpers.ts +76 -0
  76. package/dist/extensions/message-bridge/GUIDE.md +210 -0
  77. package/dist/extensions/message-bridge/index.ts +222 -0
  78. package/dist/extensions/output-guard/index.ts +384 -0
  79. package/dist/extensions/preview/index.ts +278 -0
  80. package/dist/extensions/rules-engine/MATCH_HISTORY_RECONCILIATION.md +111 -0
  81. package/dist/extensions/rules-engine/RULES-ENGINE-GUIDE.md +470 -0
  82. package/dist/extensions/rules-engine/cache.js +232 -0
  83. package/dist/extensions/rules-engine/cache.ts +38 -0
  84. package/dist/extensions/rules-engine/config.js +63 -0
  85. package/dist/extensions/rules-engine/config.ts +70 -0
  86. package/dist/extensions/rules-engine/index.js +1530 -0
  87. package/dist/extensions/rules-engine/index.ts +552 -0
  88. package/dist/extensions/rules-engine/injector.js +68 -0
  89. package/dist/extensions/rules-engine/injector.ts +74 -0
  90. package/dist/extensions/rules-engine/loader.js +179 -0
  91. package/dist/extensions/rules-engine/loader.ts +205 -0
  92. package/dist/extensions/rules-engine/matcher.js +534 -0
  93. package/dist/extensions/rules-engine/matcher.ts +52 -0
  94. package/dist/extensions/rules-engine/types.d.ts +156 -0
  95. package/dist/extensions/rules-engine/types.d.ts.map +1 -0
  96. package/dist/extensions/rules-engine/types.js +2 -0
  97. package/dist/extensions/rules-engine/types.js.map +1 -0
  98. package/dist/extensions/rules-engine/types.ts +169 -0
  99. package/dist/extensions/session-supervisor/checker.ts +116 -0
  100. package/dist/extensions/session-supervisor/config.ts +45 -0
  101. package/dist/extensions/session-supervisor/index.ts +726 -0
  102. package/dist/extensions/session-supervisor/prompts.ts +132 -0
  103. package/dist/extensions/session-supervisor/scheduler.ts +69 -0
  104. package/dist/extensions/session-supervisor/types.ts +215 -0
  105. package/dist/extensions/subagent/README.md +172 -0
  106. package/dist/extensions/subagent/agents/explorer.md +25 -0
  107. package/dist/extensions/subagent/agents/guide.md +27 -0
  108. package/dist/extensions/subagent/agents/planner.md +37 -0
  109. package/dist/extensions/subagent/agents/reviewer.md +35 -0
  110. package/dist/extensions/subagent/agents/scout.md +50 -0
  111. package/dist/extensions/subagent/agents/verification.md +35 -0
  112. package/dist/extensions/subagent/agents/worker.md +24 -0
  113. package/dist/extensions/subagent/agents.ts +25 -0
  114. package/dist/extensions/subagent/index.ts +987 -0
  115. package/dist/extensions/subagent/prompts/implement-and-review.md +10 -0
  116. package/dist/extensions/subagent/prompts/implement.md +10 -0
  117. package/dist/extensions/subagent/prompts/scout-and-plan.md +9 -0
  118. package/dist/extensions/subagent-ext/contract.d.ts +2 -0
  119. package/dist/extensions/subagent-ext/contract.d.ts.map +1 -0
  120. package/dist/extensions/subagent-ext/contract.js +2 -0
  121. package/dist/extensions/subagent-ext/contract.js.map +1 -0
  122. package/dist/extensions/subagent-ext/contract.ts +1 -0
  123. package/dist/extensions/subagent-ext/index.ts +347 -0
  124. package/dist/extensions/subagent-shared/contract.d.ts +25 -0
  125. package/dist/extensions/subagent-shared/contract.d.ts.map +1 -0
  126. package/dist/extensions/subagent-shared/contract.js +2 -0
  127. package/dist/extensions/subagent-shared/contract.js.map +1 -0
  128. package/dist/extensions/subagent-shared/contract.ts +28 -0
  129. package/dist/extensions/subagent-shared/index.ts +4 -0
  130. package/dist/extensions/subagent-shared/render.ts +166 -0
  131. package/dist/extensions/subagent-shared/types.ts +35 -0
  132. package/dist/extensions/subagent-shared/utils.ts +112 -0
  133. package/dist/extensions/subagent-v2/contract.d.ts +2 -0
  134. package/dist/extensions/subagent-v2/contract.d.ts.map +1 -0
  135. package/dist/extensions/subagent-v2/contract.js +2 -0
  136. package/dist/extensions/subagent-v2/contract.js.map +1 -0
  137. package/dist/extensions/subagent-v2/contract.ts +1 -0
  138. package/dist/extensions/subagent-v2/index.ts +599 -0
  139. package/dist/extensions/todo-ext/contract.d.ts +27 -0
  140. package/dist/extensions/todo-ext/contract.d.ts.map +1 -0
  141. package/dist/extensions/todo-ext/contract.js +2 -0
  142. package/dist/extensions/todo-ext/contract.js.map +1 -0
  143. package/dist/extensions/todo-ext/contract.ts +30 -0
  144. package/dist/extensions/todo-ext/index.ts +419 -0
  145. 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": ";AAAA,YAAY,QAAQ;AACpB,YAAY,UAAU;AAGtB,SAAS,WAAW,KAAuB;AAC1C,QAAM,SAAmB,CAAC;AAC1B,MAAI,QAAQ;AACZ,MAAI,UAAU;AACd,aAAW,MAAM,KAAK;AACrB,QAAI,OAAO,OAAO,OAAO,OAAO,OAAO,IAAK;AAAA,aACnC,OAAO,OAAO,OAAO,OAAO,OAAO,IAAK;AAEjD,QAAI,OAAO,OAAO,UAAU,GAAG;AAC9B,YAAMA,WAAU,QAAQ,KAAK,EAAE,QAAQ,gBAAgB,EAAE;AACzD,UAAIA,SAAS,QAAO,KAAKA,QAAO;AAChC,gBAAU;AAAA,IACX,OAAO;AACN,iBAAW;AAAA,IACZ;AAAA,EACD;AACA,QAAM,UAAU,QAAQ,KAAK,EAAE,QAAQ,gBAAgB,EAAE;AACzD,MAAI,QAAS,QAAO,KAAK,OAAO;AAChC,SAAO;AACR;AAEO,SAAS,iBAAiB,SAAkE;AAClG,QAAM,mBAAmB;AACzB,QAAM,QAAQ,QAAQ,MAAM,gBAAgB;AAE5C,MAAI,CAAC,OAAO;AACX,WAAO,EAAE,MAAM,CAAC,GAAG,MAAM,QAAQ;AAAA,EAClC;AAEA,QAAM,CAAC,EAAE,gBAAgB,IAAI,IAAI;AACjC,QAAM,OAAgC,CAAC;AAEvC,QAAM,QAAQ,eAAe,MAAM,IAAI;AACvC,MAAI,IAAI;AACR,SAAO,IAAI,MAAM,QAAQ;AACxB,UAAM,OAAO,MAAM,CAAC;AACpB,UAAM,aAAa,KAAK,QAAQ,GAAG;AACnC,QAAI,eAAe,IAAI;AACtB;AACA;AAAA,IACD;AAEA,UAAM,SAAS,KAAK,MAAM,GAAG,UAAU,EAAE,KAAK;AAC9C,QAAI,QAAkC,KAAK,MAAM,aAAa,CAAC,EAAE,KAAK;AAEtE,QAAI,UAAU,MAAM,UAAU,UAAU,UAAU,aAAa;AAC9D,YAAM,YAAsB,CAAC;AAC7B,UAAI,IAAI,IAAI;AACZ,aAAO,IAAI,MAAM,QAAQ;AACxB,cAAM,UAAU,MAAM,CAAC;AACvB,YAAI,QAAQ,MAAM,UAAU,GAAG;AAC9B,oBAAU;AAAA,YACT,QACE,QAAQ,YAAY,EAAE,EACtB,KAAK,EACL,QAAQ,gBAAgB,EAAE;AAAA,UAC7B;AACA;AAAA,QACD,WAAW,QAAQ,KAAK,MAAM,MAAM,QAAQ,MAAM,MAAM,GAAG;AAC1D;AAAA,QACD,OAAO;AACN;AAAA,QACD;AAAA,MACD;AACA,UAAI,UAAU,SAAS,GAAG;AACzB,cAAMC,YAAW,OAAO,QAAQ,aAAa,CAAC,GAAG,WAAW,OAAO,YAAY,CAAC;AAChF,aAAKA,SAAQ,IAAI;AACjB,YAAI;AACJ;AAAA,MACD;AACA,cAAQ;AAAA,IACT,WAAW,MAAM,WAAW,GAAG,KAAK,MAAM,SAAS,GAAG,GAAG;AACxD,UAAI;AACH,gBAAQ,KAAK,MAAM,MAAM,QAAQ,MAAM,GAAG,CAAC;AAAA,MAC5C,SAAS,KAAK;AACb,gBAAQ,MAAM,2CAA2C,eAAe,QAAQ,IAAI,UAAU,GAAG;AACjG,gBAAS,MACP,MAAM,GAAG,EAAE,EACX,MAAM,GAAG,EACT,IAAI,CAAC,MAAc,EAAE,KAAK,EAAE,QAAQ,gBAAgB,EAAE,CAAC;AAAA,MAC1D;AAAA,IACD,WAAW,MAAM,WAAW,GAAG,KAAK,MAAM,SAAS,GAAG,GAAG;AACxD,cAAQ,MAAM,MAAM,GAAG,EAAE;AAAA,IAC1B,WAAW,MAAM,WAAW,GAAG,KAAK,MAAM,SAAS,GAAG,GAAG;AACxD,cAAQ,MAAM,MAAM,GAAG,EAAE;AAAA,IAC1B;AAEA,UAAM,WAAW,OAAO,QAAQ,aAAa,CAAC,GAAG,WAAW,OAAO,YAAY,CAAC;AAEhF,SAAK,aAAa,WAAW,aAAa,YAAY,OAAO,UAAU,UAAU;AAChF,WAAK,QAAQ,IAAI,WAAW,KAAK;AAAA,IAClC,OAAO;AACN,WAAK,QAAQ,IAAI;AAAA,IAClB;AACA;AAAA,EACD;AAEA,SAAO,EAAE,MAAM,MAAM,KAAK,KAAK,EAAE;AAClC;AAEA,SAAS,aAAa,MAAsB;AAC3C,aAAW,QAAQ,KAAK,MAAM,IAAI,GAAG;AACpC,UAAM,UAAU,KAAK,KAAK;AAC1B,QAAI,SAAS;AACZ,aAAO,QAAQ,QAAQ,UAAU,EAAE,EAAE,QAAQ,SAAS,EAAE;AAAA,IACzD;AAAA,EACD;AACA,SAAO;AACR;AAEA,SAAS,WAAW,KAAwB;AAC3C,MAAI,CAAC,IAAK,QAAO,CAAC;AAClB,MAAI,OAAO,QAAQ,UAAU;AAC5B,WAAO,IACL,MAAM,GAAG,EACT,IAAI,CAAC,MAAM,EAAE,KAAK,CAAC,EACnB,OAAO,OAAO;AAAA,EACjB;AACA,MAAI,MAAM,QAAQ,GAAG,EAAG,QAAO;AAC/B,SAAO,CAAC;AACT;AAEO,SAAS,cAAc,UAAkB,SAA6B;AAC5E,QAAM,EAAE,MAAM,KAAK,IAAI,iBAAiB,OAAO;AAC/C,QAAM,WAAW,KAAK,SAAS,KAAK;AACpC,QAAM,QAAQ,WAAW,QAAQ;AACjC,QAAM,WAAW,KAAK;AACtB,QAAM,QAAQ,WAAW,QAAQ;AACjC,QAAM,kBAAkB,MAAM,WAAW,KAAM,MAAM,WAAW,KAAK,MAAM,CAAC,MAAM;AAElF,QAAM,cAA+B,CAAC;AACtC,MAAI,UAAU;AACb,gBAAY,QAAQ;AACpB,QAAI,UAAU;AACb,kBAAY,QAAQ;AAAA,IACrB,OAAO;AACN,kBAAY,QAAQ;AAAA,IACrB;AAAA,EACD;AACA,MAAI,KAAK,eAAe,OAAO,KAAK,gBAAgB,SAAU,aAAY,cAAc,KAAK;AAC7F,MAAI,KAAK,YAAY,OAAO,KAAK,aAAa;AAC7C,gBAAY,WAAW,KAAK;AAC7B,MAAI,KAAK;AACR,gBAAY,eACX,OAAO,KAAK,iBAAiB,WAAW,CAAC,KAAK,YAAY,IAAK,KAAK;AACtE,MAAI,KAAK,aAAa,OAAO,KAAK,cAAc,SAAU,aAAY,YAAY,KAAK;AACvF,MAAI,KAAK,kBAAkB;AAC1B,gBAAY,gBAAgB,KAAK,kBAAkB,UAAU,KAAK,kBAAkB;AACrF,MAAI,KAAK,iBAAiB;AACzB,gBAAY,eAAe,KAAK,iBAAiB,UAAU,KAAK,iBAAiB;AAElF,SAAO;AAAA,IACN,MAAW,cAAS,UAAe,aAAQ,QAAQ,CAAC;AAAA,IACpD;AAAA,IACA,OAAO,aAAa,IAAI;AAAA,IACxB,SAAS,KAAK,KAAK;AAAA,IACnB,OAAO;AAAA,IACP,QAAQ;AAAA,IACR;AAAA,IACA;AAAA,EACD;AACD;AAEA,SAAS,QAAQ,KAAa,QAAkB,CAAC,GAAa;AAC7D,MAAI,CAAI,cAAW,GAAG,EAAG,QAAO;AAChC,QAAM,UAAa,eAAY,KAAK,EAAE,eAAe,KAAK,CAAC;AAC3D,aAAW,SAAS,SAAS;AAC5B,UAAM,WAAgB,UAAK,KAAK,MAAM,IAAI;AAC1C,QAAI,MAAM,YAAY,GAAG;AACxB,cAAQ,UAAU,KAAK;AAAA,IACxB,WAAW,MAAM,OAAO,MAAM,MAAM,KAAK,SAAS,KAAK,KAAK,MAAM,KAAK,SAAS,MAAM,IAAI;AACzF,YAAM,KAAK,QAAQ;AAAA,IACpB;AAAA,EACD;AACA,SAAO;AACR;AAEO,SAAS,UAAU,UAA6B;AACtD,QAAM,QAAsB,CAAC;AAC7B,QAAM,gBAA8B,CAAC;AACrC,QAAM,cAA4B,CAAC;AAEnC,MAAI,CAAI,cAAW,QAAQ,GAAG;AAC7B,WAAO,EAAE,OAAO,eAAe,aAAa,UAAU,KAAK,IAAI,EAAE;AAAA,EAClE;AAEA,QAAM,QAAQ,QAAQ,QAAQ;AAE9B,aAAW,YAAY,OAAO;AAC7B,UAAM,UAAa,gBAAa,UAAU,OAAO;AACjD,UAAM,OAAO,cAAc,UAAU,OAAO;AAC5C,UAAM,KAAK,IAAI;AACf,QAAI,KAAK,iBAAiB;AACzB,oBAAc,KAAK,IAAI;AAAA,IACxB,OAAO;AACN,kBAAY,KAAK,IAAI;AAAA,IACtB;AAAA,EACD;AAEA,SAAO,EAAE,OAAO,eAAe,aAAa,UAAU,KAAK,IAAI,EAAE;AAClE;",
  "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
+ }