@dyyz1993/pi-coding-agent 0.74.23 → 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 (142) hide show
  1. package/dist/extensions/agent-permissions/index.ts +235 -0
  2. package/dist/extensions/ask-tools/index.ts +115 -0
  3. package/dist/extensions/auto-memory/contract.d.ts +51 -0
  4. package/dist/extensions/auto-memory/contract.d.ts.map +1 -0
  5. package/dist/extensions/auto-memory/contract.js +2 -0
  6. package/dist/extensions/auto-memory/contract.js.map +1 -0
  7. package/dist/extensions/auto-memory/contract.ts +56 -0
  8. package/dist/extensions/auto-memory/index.ts +969 -0
  9. package/dist/extensions/auto-memory/prompts.ts +202 -0
  10. package/dist/extensions/auto-memory/skip-rules.ts +297 -0
  11. package/dist/extensions/auto-memory/utils.ts +208 -0
  12. package/dist/extensions/auto-session-title/index.ts +83 -0
  13. package/dist/extensions/bash-ext/contract.d.ts +79 -0
  14. package/dist/extensions/bash-ext/contract.d.ts.map +1 -0
  15. package/dist/extensions/bash-ext/contract.js +2 -0
  16. package/dist/extensions/bash-ext/contract.js.map +1 -0
  17. package/dist/extensions/bash-ext/contract.ts +69 -0
  18. package/dist/extensions/bash-ext/index.ts +858 -0
  19. package/dist/extensions/claude-hooks-compat/config-loader.ts +49 -0
  20. package/dist/extensions/claude-hooks-compat/handler-runner.ts +377 -0
  21. package/dist/extensions/claude-hooks-compat/if-parser.ts +53 -0
  22. package/dist/extensions/claude-hooks-compat/index.ts +178 -0
  23. package/dist/extensions/claude-hooks-compat/matcher.ts +17 -0
  24. package/dist/extensions/claude-hooks-compat/stdin-builder.ts +27 -0
  25. package/dist/extensions/claude-hooks-compat/types.ts +77 -0
  26. package/dist/extensions/compaction-manager/config.ts +47 -0
  27. package/dist/extensions/compaction-manager/context-fold.ts +63 -0
  28. package/dist/extensions/compaction-manager/index.ts +151 -0
  29. package/dist/extensions/compaction-manager/microcompact.ts +49 -0
  30. package/dist/extensions/compaction-manager/reactive.ts +9 -0
  31. package/dist/extensions/compaction-manager/session-memory.ts +48 -0
  32. package/dist/extensions/coordinator/INTEGRATION.md +376 -0
  33. package/dist/extensions/coordinator/handler.test.ts +277 -0
  34. package/dist/extensions/coordinator/handler.ts +189 -0
  35. package/dist/extensions/coordinator/index.ts +261 -0
  36. package/dist/extensions/coordinator/types.d.ts +100 -0
  37. package/dist/extensions/coordinator/types.d.ts.map +1 -0
  38. package/dist/extensions/coordinator/types.js +2 -0
  39. package/dist/extensions/coordinator/types.js.map +1 -0
  40. package/dist/extensions/coordinator/types.ts +72 -0
  41. package/dist/extensions/file-snapshot/index.ts +131 -0
  42. package/dist/extensions/file-time-guard/README.md +133 -0
  43. package/dist/extensions/file-time-guard/config.ts +13 -0
  44. package/dist/extensions/file-time-guard/index.ts +171 -0
  45. package/dist/extensions/hooks-engine/index.ts +117 -0
  46. package/dist/extensions/lsp/lsp/client/file-tracker.ts +70 -0
  47. package/dist/extensions/lsp/lsp/client/registry.ts +305 -0
  48. package/dist/extensions/lsp/lsp/client/runtime.ts +832 -0
  49. package/dist/extensions/lsp/lsp/config/resolver.ts +573 -0
  50. package/dist/extensions/lsp/lsp/contract.d.ts +101 -0
  51. package/dist/extensions/lsp/lsp/contract.d.ts.map +1 -0
  52. package/dist/extensions/lsp/lsp/contract.js +2 -0
  53. package/dist/extensions/lsp/lsp/contract.js.map +1 -0
  54. package/dist/extensions/lsp/lsp/contract.ts +103 -0
  55. package/dist/extensions/lsp/lsp/hooks/agent-end.ts +169 -0
  56. package/dist/extensions/lsp/lsp/hooks/diagnostics-mode.d.ts +10 -0
  57. package/dist/extensions/lsp/lsp/hooks/diagnostics-mode.d.ts.map +1 -0
  58. package/dist/extensions/lsp/lsp/hooks/diagnostics-mode.js +30 -0
  59. package/dist/extensions/lsp/lsp/hooks/diagnostics-mode.js.map +1 -0
  60. package/dist/extensions/lsp/lsp/hooks/diagnostics-mode.ts +41 -0
  61. package/dist/extensions/lsp/lsp/hooks/writethrough.ts +342 -0
  62. package/dist/extensions/lsp/lsp/index.ts +307 -0
  63. package/dist/extensions/lsp/lsp/lsp.test.ts +684 -0
  64. package/dist/extensions/lsp/lsp/monitoring/server-metrics.ts +176 -0
  65. package/dist/extensions/lsp/lsp/tools/lsp-tool.ts +402 -0
  66. package/dist/extensions/lsp/lsp/utils/dependency-resolver.ts +147 -0
  67. package/dist/extensions/lsp/lsp/utils/diagnostics-wait.ts +41 -0
  68. package/dist/extensions/lsp/lsp/utils/lsp-helpers.d.ts +20 -0
  69. package/dist/extensions/lsp/lsp/utils/lsp-helpers.d.ts.map +1 -0
  70. package/dist/extensions/lsp/lsp/utils/lsp-helpers.js +64 -0
  71. package/dist/extensions/lsp/lsp/utils/lsp-helpers.js.map +1 -0
  72. package/dist/extensions/lsp/lsp/utils/lsp-helpers.ts +76 -0
  73. package/dist/extensions/message-bridge/GUIDE.md +210 -0
  74. package/dist/extensions/message-bridge/index.ts +222 -0
  75. package/dist/extensions/output-guard/index.ts +384 -0
  76. package/dist/extensions/preview/index.ts +278 -0
  77. package/dist/extensions/rules-engine/MATCH_HISTORY_RECONCILIATION.md +111 -0
  78. package/dist/extensions/rules-engine/RULES-ENGINE-GUIDE.md +470 -0
  79. package/dist/extensions/rules-engine/cache.js +232 -0
  80. package/dist/extensions/rules-engine/cache.ts +38 -0
  81. package/dist/extensions/rules-engine/config.js +63 -0
  82. package/dist/extensions/rules-engine/config.ts +70 -0
  83. package/dist/extensions/rules-engine/index.js +1530 -0
  84. package/dist/extensions/rules-engine/index.ts +552 -0
  85. package/dist/extensions/rules-engine/injector.js +68 -0
  86. package/dist/extensions/rules-engine/injector.ts +74 -0
  87. package/dist/extensions/rules-engine/loader.js +179 -0
  88. package/dist/extensions/rules-engine/loader.ts +205 -0
  89. package/dist/extensions/rules-engine/matcher.js +534 -0
  90. package/dist/extensions/rules-engine/matcher.ts +52 -0
  91. package/dist/extensions/rules-engine/types.d.ts +156 -0
  92. package/dist/extensions/rules-engine/types.d.ts.map +1 -0
  93. package/dist/extensions/rules-engine/types.js +2 -0
  94. package/dist/extensions/rules-engine/types.js.map +1 -0
  95. package/dist/extensions/rules-engine/types.ts +169 -0
  96. package/dist/extensions/session-supervisor/checker.ts +116 -0
  97. package/dist/extensions/session-supervisor/config.ts +45 -0
  98. package/dist/extensions/session-supervisor/index.ts +726 -0
  99. package/dist/extensions/session-supervisor/prompts.ts +132 -0
  100. package/dist/extensions/session-supervisor/scheduler.ts +69 -0
  101. package/dist/extensions/session-supervisor/types.ts +215 -0
  102. package/dist/extensions/subagent/README.md +172 -0
  103. package/dist/extensions/subagent/agents/explorer.md +25 -0
  104. package/dist/extensions/subagent/agents/guide.md +27 -0
  105. package/dist/extensions/subagent/agents/planner.md +37 -0
  106. package/dist/extensions/subagent/agents/reviewer.md +35 -0
  107. package/dist/extensions/subagent/agents/scout.md +50 -0
  108. package/dist/extensions/subagent/agents/verification.md +35 -0
  109. package/dist/extensions/subagent/agents/worker.md +24 -0
  110. package/dist/extensions/subagent/agents.ts +25 -0
  111. package/dist/extensions/subagent/index.ts +987 -0
  112. package/dist/extensions/subagent/prompts/implement-and-review.md +10 -0
  113. package/dist/extensions/subagent/prompts/implement.md +10 -0
  114. package/dist/extensions/subagent/prompts/scout-and-plan.md +9 -0
  115. package/dist/extensions/subagent-ext/contract.d.ts +2 -0
  116. package/dist/extensions/subagent-ext/contract.d.ts.map +1 -0
  117. package/dist/extensions/subagent-ext/contract.js +2 -0
  118. package/dist/extensions/subagent-ext/contract.js.map +1 -0
  119. package/dist/extensions/subagent-ext/contract.ts +1 -0
  120. package/dist/extensions/subagent-ext/index.ts +347 -0
  121. package/dist/extensions/subagent-shared/contract.d.ts +25 -0
  122. package/dist/extensions/subagent-shared/contract.d.ts.map +1 -0
  123. package/dist/extensions/subagent-shared/contract.js +2 -0
  124. package/dist/extensions/subagent-shared/contract.js.map +1 -0
  125. package/dist/extensions/subagent-shared/contract.ts +28 -0
  126. package/dist/extensions/subagent-shared/index.ts +4 -0
  127. package/dist/extensions/subagent-shared/render.ts +166 -0
  128. package/dist/extensions/subagent-shared/types.ts +35 -0
  129. package/dist/extensions/subagent-shared/utils.ts +112 -0
  130. package/dist/extensions/subagent-v2/contract.d.ts +2 -0
  131. package/dist/extensions/subagent-v2/contract.d.ts.map +1 -0
  132. package/dist/extensions/subagent-v2/contract.js +2 -0
  133. package/dist/extensions/subagent-v2/contract.js.map +1 -0
  134. package/dist/extensions/subagent-v2/contract.ts +1 -0
  135. package/dist/extensions/subagent-v2/index.ts +599 -0
  136. package/dist/extensions/todo-ext/contract.d.ts +27 -0
  137. package/dist/extensions/todo-ext/contract.d.ts.map +1 -0
  138. package/dist/extensions/todo-ext/contract.js +2 -0
  139. package/dist/extensions/todo-ext/contract.js.map +1 -0
  140. package/dist/extensions/todo-ext/contract.ts +30 -0
  141. package/dist/extensions/todo-ext/index.ts +419 -0
  142. package/package.json +3 -2
@@ -0,0 +1,232 @@
1
+ // packages/coding-agent/extensions/rules-engine/config.ts
2
+ import { homedir } from "node:os";
3
+ import * as path from "node:path";
4
+ var DEFAULT_DIRS = {
5
+ managed: ["/etc/claude-code/.claude/rules"],
6
+ user: [path.join(homedir(), ".claude", "rules"), path.join(homedir(), ".config", "opencode", "rules")],
7
+ pi: [".pi/rules"],
8
+ project: [".claude/rules", ".opencode/rules", ".trae/rules"]
9
+ };
10
+ function resolveDirs(projectDir, config) {
11
+ const sources = config.dirs || DEFAULT_DIRS;
12
+ const result = [];
13
+ for (const [scope, paths] of Object.entries(sources)) {
14
+ for (const p of paths) {
15
+ result.push({
16
+ scope,
17
+ dir: p.startsWith("/") || p.startsWith("~") ? p.replace(/^~/, homedir()) : path.resolve(projectDir, p),
18
+ source: p
19
+ });
20
+ }
21
+ }
22
+ return result;
23
+ }
24
+
25
+ // packages/coding-agent/extensions/rules-engine/loader.ts
26
+ import * as fs from "node:fs";
27
+ import * as path2 from "node:path";
28
+ function splitComma(val) {
29
+ const result = [];
30
+ let depth = 0;
31
+ let current = "";
32
+ for (const ch of val) {
33
+ if (ch === "{" || ch === "(" || ch === "[") depth++;
34
+ else if (ch === "}" || ch === ")" || ch === "]") depth--;
35
+ if (ch === "," && depth === 0) {
36
+ const trimmed2 = current.trim().replace(/^["']|["']$/g, "");
37
+ if (trimmed2) result.push(trimmed2);
38
+ current = "";
39
+ } else {
40
+ current += ch;
41
+ }
42
+ }
43
+ const trimmed = current.trim().replace(/^["']|["']$/g, "");
44
+ if (trimmed) result.push(trimmed);
45
+ return result;
46
+ }
47
+ function parseFrontmatter(content) {
48
+ const frontmatterRegex = /^---\r?\n([\s\S]*?)\n?---\r?\n([\s\S]*)$/;
49
+ const match = content.match(frontmatterRegex);
50
+ if (!match) {
51
+ return { data: {}, body: content };
52
+ }
53
+ const [, frontmatterStr, body] = match;
54
+ const data = {};
55
+ const lines = frontmatterStr.split("\n");
56
+ let i = 0;
57
+ while (i < lines.length) {
58
+ const line = lines[i];
59
+ const colonIndex = line.indexOf(":");
60
+ if (colonIndex === -1) {
61
+ i++;
62
+ continue;
63
+ }
64
+ const rawKey = line.slice(0, colonIndex).trim();
65
+ let value = line.slice(colonIndex + 1).trim();
66
+ if (value === "" || value === "null" || value === "undefined") {
67
+ const listItems = [];
68
+ let j = i + 1;
69
+ while (j < lines.length) {
70
+ const subLine = lines[j];
71
+ if (subLine.match(/^\s*-\s+/)) {
72
+ listItems.push(
73
+ subLine.replace(/^\s*-\s+/, "").trim().replace(/^["']|["']$/g, "")
74
+ );
75
+ j++;
76
+ } else if (subLine.trim() === "" || subLine.match(/^\s+/)) {
77
+ j++;
78
+ } else {
79
+ break;
80
+ }
81
+ }
82
+ if (listItems.length > 0) {
83
+ const camelKey2 = rawKey.replace(/-([a-z])/g, (_, letter) => letter.toUpperCase());
84
+ data[camelKey2] = listItems;
85
+ i = j;
86
+ continue;
87
+ }
88
+ value = null;
89
+ } else if (value.startsWith("[") && value.endsWith("]")) {
90
+ try {
91
+ value = JSON.parse(value.replace(/'/g, '"'));
92
+ } catch (err) {
93
+ console.debug("[rules-engine] JSON array parse failed:", err instanceof Error ? err.message : err);
94
+ value = value.slice(1, -1).split(",").map((v) => v.trim().replace(/^["']|["']$/g, ""));
95
+ }
96
+ } else if (value.startsWith('"') && value.endsWith('"')) {
97
+ value = value.slice(1, -1);
98
+ } else if (value.startsWith("'") && value.endsWith("'")) {
99
+ value = value.slice(1, -1);
100
+ }
101
+ const camelKey = rawKey.replace(/-([a-z])/g, (_, letter) => letter.toUpperCase());
102
+ if ((camelKey === "paths" || camelKey === "globs") && typeof value === "string") {
103
+ data[camelKey] = splitComma(value);
104
+ } else {
105
+ data[camelKey] = value;
106
+ }
107
+ i++;
108
+ }
109
+ return { data, body: body.trim() };
110
+ }
111
+ function extractTitle(body) {
112
+ for (const line of body.split("\n")) {
113
+ const trimmed = line.trim();
114
+ if (trimmed) {
115
+ return trimmed.replace(/^#+\s*/, "").replace(/\*\*/g, "");
116
+ }
117
+ }
118
+ return "Untitled Rule";
119
+ }
120
+ function parsePaths(raw) {
121
+ if (!raw) return [];
122
+ if (typeof raw === "string") {
123
+ return raw.split(",").map((g) => g.trim()).filter(Boolean);
124
+ }
125
+ if (Array.isArray(raw)) return raw;
126
+ return [];
127
+ }
128
+ function parseRuleFile(filePath, content) {
129
+ const { data, body } = parseFrontmatter(content);
130
+ const rawGlobs = data.globs ?? data.paths;
131
+ const globs = parsePaths(rawGlobs);
132
+ const rawPaths = data.paths;
133
+ const paths = parsePaths(rawPaths);
134
+ const isUnconditional = globs.length === 0 || globs.length === 1 && globs[0] === "**";
135
+ const frontmatter = {};
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;
147
+ if (data.allowedTools)
148
+ frontmatter.allowedTools = typeof data.allowedTools === "string" ? [data.allowedTools] : data.allowedTools;
149
+ if (data.whenToUse && typeof data.whenToUse === "string") frontmatter.whenToUse = data.whenToUse;
150
+ if (data.notifyOnMatch !== void 0)
151
+ frontmatter.notifyOnMatch = data.notifyOnMatch === "true" || data.notifyOnMatch === true;
152
+ if (data.skipInPrompt !== void 0)
153
+ frontmatter.skipInPrompt = data.skipInPrompt === "true" || data.skipInPrompt === true;
154
+ return {
155
+ name: path2.basename(filePath, path2.extname(filePath)),
156
+ filePath,
157
+ title: extractTitle(body),
158
+ content: body.trim(),
159
+ scope: "project",
160
+ source: "",
161
+ frontmatter,
162
+ isUnconditional
163
+ };
164
+ }
165
+ function scanDir(dir, files = []) {
166
+ if (!fs.existsSync(dir)) return files;
167
+ const entries = fs.readdirSync(dir, { withFileTypes: true });
168
+ for (const entry of entries) {
169
+ const fullPath = path2.join(dir, entry.name);
170
+ if (entry.isDirectory()) {
171
+ scanDir(fullPath, files);
172
+ } else if (entry.isFile() && (entry.name.endsWith(".md") || entry.name.endsWith(".mdc"))) {
173
+ files.push(fullPath);
174
+ }
175
+ }
176
+ return files;
177
+ }
178
+ function loadRules(rulesDir) {
179
+ const rules = [];
180
+ const unconditional = [];
181
+ const conditional = [];
182
+ if (!fs.existsSync(rulesDir)) {
183
+ return { rules, unconditional, conditional, loadedAt: Date.now() };
184
+ }
185
+ const files = scanDir(rulesDir);
186
+ for (const filePath of files) {
187
+ const content = fs.readFileSync(filePath, "utf-8");
188
+ const rule = parseRuleFile(filePath, content);
189
+ rules.push(rule);
190
+ if (rule.isUnconditional) {
191
+ unconditional.push(rule);
192
+ } else {
193
+ conditional.push(rule);
194
+ }
195
+ }
196
+ return { rules, unconditional, conditional, loadedAt: Date.now() };
197
+ }
198
+
199
+ // packages/coding-agent/extensions/rules-engine/cache.ts
200
+ var cache = null;
201
+ async function getRules(projectDir, config) {
202
+ if (cache && Date.now() - cache.loadedAt < config.cacheTTL) {
203
+ return cache.rules;
204
+ }
205
+ const rules = await loadAllRules(projectDir, config);
206
+ cache = { rules, loadedAt: Date.now() };
207
+ return rules;
208
+ }
209
+ function invalidateCache() {
210
+ cache = null;
211
+ }
212
+ async function loadAllRules(projectDir, config) {
213
+ const sources = resolveDirs(projectDir, config);
214
+ const result = [];
215
+ const seen = /* @__PURE__ */ new Set();
216
+ for (const { scope, dir, source } of sources) {
217
+ const ruleCache = loadRules(dir);
218
+ for (const rule of ruleCache.rules) {
219
+ if (seen.has(rule.filePath)) continue;
220
+ seen.add(rule.filePath);
221
+ rule.scope = scope;
222
+ rule.source = source;
223
+ result.push(rule);
224
+ }
225
+ }
226
+ return result;
227
+ }
228
+ export {
229
+ getRules,
230
+ invalidateCache
231
+ };
232
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["config.ts", "loader.ts", "cache.ts"],
  "sourcesContent": ["import * as fs from \"node:fs\";\nimport { homedir } from \"node:os\";\nimport * as path from \"node:path\";\nimport type { RulesConfig } from \"./types.js\";\n\nconst DEFAULT_CACHE_TTL = 30_000;\n\nconst DEFAULT_DIRS = {\n\tmanaged: [\"/etc/claude-code/.claude/rules\"],\n\tuser: [path.join(homedir(), \".claude\", \"rules\"), path.join(homedir(), \".config\", \"opencode\", \"rules\")],\n\tpi: [\".pi/rules\"],\n\tproject: [\".claude/rules\", \".opencode/rules\", \".trae/rules\"],\n};\n\nfunction defaultConfig(): RulesConfig {\n\treturn {\n\t\tcacheTTL: DEFAULT_CACHE_TTL,\n\t\tnotifyOnLoad: true,\n\t\tnotifyOnMatch: true,\n\t};\n}\n\nexport async function loadConfig(projectDir: string): Promise<RulesConfig> {\n\tconst configFiles = [\n\t\t\".rules-config.json\",\n\t\t\".pi/rules-config.json\",\n\t\t\".claude/rules-config.json\",\n\t\t\".opencode/rules-config.json\",\n\t];\n\n\tfor (const name of configFiles) {\n\t\tconst fp = path.resolve(projectDir, name);\n\t\tif (!fs.existsSync(fp)) continue;\n\t\ttry {\n\t\t\tconst raw = fs.readFileSync(fp, \"utf-8\");\n\t\t\tconst parsed = JSON.parse(raw);\n\t\t\treturn {\n\t\t\t\t...defaultConfig(),\n\t\t\t\t...parsed,\n\t\t\t\tcacheTTL: parsed.cacheTTL ?? DEFAULT_CACHE_TTL,\n\t\t\t\tnotifyOnLoad: parsed.notifyOnLoad ?? true,\n\t\t\t\tnotifyOnMatch: parsed.notifyOnMatch ?? true,\n\t\t\t};\n\t\t} catch (err) {\n\t\t\tconsole.debug(\"[rules-engine] config parse failed:\", err instanceof Error ? err.message : err);\n\t\t}\n\t}\n\n\treturn defaultConfig();\n}\n\nexport function resolveDirs(\n\tprojectDir: string,\n\tconfig: RulesConfig,\n): Array<{ scope: string; dir: string; source: string }> {\n\tconst sources = config.dirs || DEFAULT_DIRS;\n\tconst result: Array<{ scope: string; dir: string; source: string }> = [];\n\n\tfor (const [scope, paths] of Object.entries(sources)) {\n\t\tfor (const p of paths) {\n\t\t\tresult.push({\n\t\t\t\tscope,\n\t\t\t\tdir: p.startsWith(\"/\") || p.startsWith(\"~\") ? p.replace(/^~/, homedir()) : path.resolve(projectDir, p),\n\t\t\t\tsource: p,\n\t\t\t});\n\t\t}\n\t}\n\n\treturn result;\n}\n", "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", "import { resolveDirs } from \"./config.js\";\nimport { loadRules } from \"./loader.js\";\nimport type { ParsedRule, RulesConfig } from \"./types.js\";\n\nlet cache: { rules: ParsedRule[]; loadedAt: number } | null = null;\n\nexport async function getRules(projectDir: string, config: RulesConfig): Promise<ParsedRule[]> {\n\tif (cache && Date.now() - cache.loadedAt < config.cacheTTL) {\n\t\treturn cache.rules;\n\t}\n\n\tconst rules = await loadAllRules(projectDir, config);\n\tcache = { rules, loadedAt: Date.now() };\n\treturn rules;\n}\n\nexport function invalidateCache(): void {\n\tcache = null;\n}\n\nasync function loadAllRules(projectDir: string, config: RulesConfig): Promise<ParsedRule[]> {\n\tconst sources = resolveDirs(projectDir, config);\n\tconst result: ParsedRule[] = [];\n\tconst seen = new Set<string>();\n\n\tfor (const { scope, dir, source } of sources) {\n\t\tconst ruleCache = loadRules(dir);\n\t\tfor (const rule of ruleCache.rules) {\n\t\t\tif (seen.has(rule.filePath)) continue;\n\t\t\tseen.add(rule.filePath);\n\t\t\trule.scope = scope as ParsedRule[\"scope\"];\n\t\t\trule.source = source;\n\t\t\tresult.push(rule);\n\t\t}\n\t}\n\n\treturn result;\n}\n"],
  "mappings": "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",
  "names": ["path", "trimmed", "camelKey"]
}

@@ -0,0 +1,38 @@
1
+ import { resolveDirs } from "./config.js";
2
+ import { loadRules } from "./loader.js";
3
+ import type { ParsedRule, RulesConfig } from "./types.js";
4
+
5
+ let cache: { rules: ParsedRule[]; loadedAt: number } | null = null;
6
+
7
+ export async function getRules(projectDir: string, config: RulesConfig): Promise<ParsedRule[]> {
8
+ if (cache && Date.now() - cache.loadedAt < config.cacheTTL) {
9
+ return cache.rules;
10
+ }
11
+
12
+ const rules = await loadAllRules(projectDir, config);
13
+ cache = { rules, loadedAt: Date.now() };
14
+ return rules;
15
+ }
16
+
17
+ export function invalidateCache(): void {
18
+ cache = null;
19
+ }
20
+
21
+ async function loadAllRules(projectDir: string, config: RulesConfig): Promise<ParsedRule[]> {
22
+ const sources = resolveDirs(projectDir, config);
23
+ const result: ParsedRule[] = [];
24
+ const seen = new Set<string>();
25
+
26
+ for (const { scope, dir, source } of sources) {
27
+ const ruleCache = loadRules(dir);
28
+ for (const rule of ruleCache.rules) {
29
+ if (seen.has(rule.filePath)) continue;
30
+ seen.add(rule.filePath);
31
+ rule.scope = scope as ParsedRule["scope"];
32
+ rule.source = source;
33
+ result.push(rule);
34
+ }
35
+ }
36
+
37
+ return result;
38
+ }
@@ -0,0 +1,63 @@
1
+ // packages/coding-agent/extensions/rules-engine/config.ts
2
+ import * as fs from "node:fs";
3
+ import { homedir } from "node:os";
4
+ import * as path from "node:path";
5
+ var DEFAULT_CACHE_TTL = 3e4;
6
+ var DEFAULT_DIRS = {
7
+ managed: ["/etc/claude-code/.claude/rules"],
8
+ user: [path.join(homedir(), ".claude", "rules"), path.join(homedir(), ".config", "opencode", "rules")],
9
+ pi: [".pi/rules"],
10
+ project: [".claude/rules", ".opencode/rules", ".trae/rules"]
11
+ };
12
+ function defaultConfig() {
13
+ return {
14
+ cacheTTL: DEFAULT_CACHE_TTL,
15
+ notifyOnLoad: true,
16
+ notifyOnMatch: true
17
+ };
18
+ }
19
+ async function loadConfig(projectDir) {
20
+ const configFiles = [
21
+ ".rules-config.json",
22
+ ".pi/rules-config.json",
23
+ ".claude/rules-config.json",
24
+ ".opencode/rules-config.json"
25
+ ];
26
+ for (const name of configFiles) {
27
+ const fp = path.resolve(projectDir, name);
28
+ if (!fs.existsSync(fp)) continue;
29
+ try {
30
+ const raw = fs.readFileSync(fp, "utf-8");
31
+ const parsed = JSON.parse(raw);
32
+ return {
33
+ ...defaultConfig(),
34
+ ...parsed,
35
+ cacheTTL: parsed.cacheTTL ?? DEFAULT_CACHE_TTL,
36
+ notifyOnLoad: parsed.notifyOnLoad ?? true,
37
+ notifyOnMatch: parsed.notifyOnMatch ?? true
38
+ };
39
+ } catch (err) {
40
+ console.debug("[rules-engine] config parse failed:", err instanceof Error ? err.message : err);
41
+ }
42
+ }
43
+ return defaultConfig();
44
+ }
45
+ function resolveDirs(projectDir, config) {
46
+ const sources = config.dirs || DEFAULT_DIRS;
47
+ const result = [];
48
+ for (const [scope, paths] of Object.entries(sources)) {
49
+ for (const p of paths) {
50
+ result.push({
51
+ scope,
52
+ dir: p.startsWith("/") || p.startsWith("~") ? p.replace(/^~/, homedir()) : path.resolve(projectDir, p),
53
+ source: p
54
+ });
55
+ }
56
+ }
57
+ return result;
58
+ }
59
+ export {
60
+ loadConfig,
61
+ resolveDirs
62
+ };
63
+ //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiY29uZmlnLnRzIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyJpbXBvcnQgKiBhcyBmcyBmcm9tIFwibm9kZTpmc1wiO1xuaW1wb3J0IHsgaG9tZWRpciB9IGZyb20gXCJub2RlOm9zXCI7XG5pbXBvcnQgKiBhcyBwYXRoIGZyb20gXCJub2RlOnBhdGhcIjtcbmltcG9ydCB0eXBlIHsgUnVsZXNDb25maWcgfSBmcm9tIFwiLi90eXBlcy5qc1wiO1xuXG5jb25zdCBERUZBVUxUX0NBQ0hFX1RUTCA9IDMwXzAwMDtcblxuY29uc3QgREVGQVVMVF9ESVJTID0ge1xuXHRtYW5hZ2VkOiBbXCIvZXRjL2NsYXVkZS1jb2RlLy5jbGF1ZGUvcnVsZXNcIl0sXG5cdHVzZXI6IFtwYXRoLmpvaW4oaG9tZWRpcigpLCBcIi5jbGF1ZGVcIiwgXCJydWxlc1wiKSwgcGF0aC5qb2luKGhvbWVkaXIoKSwgXCIuY29uZmlnXCIsIFwib3BlbmNvZGVcIiwgXCJydWxlc1wiKV0sXG5cdHBpOiBbXCIucGkvcnVsZXNcIl0sXG5cdHByb2plY3Q6IFtcIi5jbGF1ZGUvcnVsZXNcIiwgXCIub3BlbmNvZGUvcnVsZXNcIiwgXCIudHJhZS9ydWxlc1wiXSxcbn07XG5cbmZ1bmN0aW9uIGRlZmF1bHRDb25maWcoKTogUnVsZXNDb25maWcge1xuXHRyZXR1cm4ge1xuXHRcdGNhY2hlVFRMOiBERUZBVUxUX0NBQ0hFX1RUTCxcblx0XHRub3RpZnlPbkxvYWQ6IHRydWUsXG5cdFx0bm90aWZ5T25NYXRjaDogdHJ1ZSxcblx0fTtcbn1cblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGxvYWRDb25maWcocHJvamVjdERpcjogc3RyaW5nKTogUHJvbWlzZTxSdWxlc0NvbmZpZz4ge1xuXHRjb25zdCBjb25maWdGaWxlcyA9IFtcblx0XHRcIi5ydWxlcy1jb25maWcuanNvblwiLFxuXHRcdFwiLnBpL3J1bGVzLWNvbmZpZy5qc29uXCIsXG5cdFx0XCIuY2xhdWRlL3J1bGVzLWNvbmZpZy5qc29uXCIsXG5cdFx0XCIub3BlbmNvZGUvcnVsZXMtY29uZmlnLmpzb25cIixcblx0XTtcblxuXHRmb3IgKGNvbnN0IG5hbWUgb2YgY29uZmlnRmlsZXMpIHtcblx0XHRjb25zdCBmcCA9IHBhdGgucmVzb2x2ZShwcm9qZWN0RGlyLCBuYW1lKTtcblx0XHRpZiAoIWZzLmV4aXN0c1N5bmMoZnApKSBjb250aW51ZTtcblx0XHR0cnkge1xuXHRcdFx0Y29uc3QgcmF3ID0gZnMucmVhZEZpbGVTeW5jKGZwLCBcInV0Zi04XCIpO1xuXHRcdFx0Y29uc3QgcGFyc2VkID0gSlNPTi5wYXJzZShyYXcpO1xuXHRcdFx0cmV0dXJuIHtcblx0XHRcdFx0Li4uZGVmYXVsdENvbmZpZygpLFxuXHRcdFx0XHQuLi5wYXJzZWQsXG5cdFx0XHRcdGNhY2hlVFRMOiBwYXJzZWQuY2FjaGVUVEwgPz8gREVGQVVMVF9DQUNIRV9UVEwsXG5cdFx0XHRcdG5vdGlmeU9uTG9hZDogcGFyc2VkLm5vdGlmeU9uTG9hZCA/PyB0cnVlLFxuXHRcdFx0XHRub3RpZnlPbk1hdGNoOiBwYXJzZWQubm90aWZ5T25NYXRjaCA/PyB0cnVlLFxuXHRcdFx0fTtcblx0XHR9IGNhdGNoIChlcnIpIHtcblx0XHRcdGNvbnNvbGUuZGVidWcoXCJbcnVsZXMtZW5naW5lXSBjb25maWcgcGFyc2UgZmFpbGVkOlwiLCBlcnIgaW5zdGFuY2VvZiBFcnJvciA/IGVyci5tZXNzYWdlIDogZXJyKTtcblx0XHR9XG5cdH1cblxuXHRyZXR1cm4gZGVmYXVsdENvbmZpZygpO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gcmVzb2x2ZURpcnMoXG5cdHByb2plY3REaXI6IHN0cmluZyxcblx0Y29uZmlnOiBSdWxlc0NvbmZpZyxcbik6IEFycmF5PHsgc2NvcGU6IHN0cmluZzsgZGlyOiBzdHJpbmc7IHNvdXJjZTogc3RyaW5nIH0+IHtcblx0Y29uc3Qgc291cmNlcyA9IGNvbmZpZy5kaXJzIHx8IERFRkFVTFRfRElSUztcblx0Y29uc3QgcmVzdWx0OiBBcnJheTx7IHNjb3BlOiBzdHJpbmc7IGRpcjogc3RyaW5nOyBzb3VyY2U6IHN0cmluZyB9PiA9IFtdO1xuXG5cdGZvciAoY29uc3QgW3Njb3BlLCBwYXRoc10gb2YgT2JqZWN0LmVudHJpZXMoc291cmNlcykpIHtcblx0XHRmb3IgKGNvbnN0IHAgb2YgcGF0aHMpIHtcblx0XHRcdHJlc3VsdC5wdXNoKHtcblx0XHRcdFx0c2NvcGUsXG5cdFx0XHRcdGRpcjogcC5zdGFydHNXaXRoKFwiL1wiKSB8fCBwLnN0YXJ0c1dpdGgoXCJ+XCIpID8gcC5yZXBsYWNlKC9efi8sIGhvbWVkaXIoKSkgOiBwYXRoLnJlc29sdmUocHJvamVjdERpciwgcCksXG5cdFx0XHRcdHNvdXJjZTogcCxcblx0XHRcdH0pO1xuXHRcdH1cblx0fVxuXG5cdHJldHVybiByZXN1bHQ7XG59XG4iXSwKICAibWFwcGluZ3MiOiAiO0FBQUEsWUFBWSxRQUFRO0FBQ3BCLFNBQVMsZUFBZTtBQUN4QixZQUFZLFVBQVU7QUFHdEIsSUFBTSxvQkFBb0I7QUFFMUIsSUFBTSxlQUFlO0FBQUEsRUFDcEIsU0FBUyxDQUFDLGdDQUFnQztBQUFBLEVBQzFDLE1BQU0sQ0FBTSxVQUFLLFFBQVEsR0FBRyxXQUFXLE9BQU8sR0FBUSxVQUFLLFFBQVEsR0FBRyxXQUFXLFlBQVksT0FBTyxDQUFDO0FBQUEsRUFDckcsSUFBSSxDQUFDLFdBQVc7QUFBQSxFQUNoQixTQUFTLENBQUMsaUJBQWlCLG1CQUFtQixhQUFhO0FBQzVEO0FBRUEsU0FBUyxnQkFBNkI7QUFDckMsU0FBTztBQUFBLElBQ04sVUFBVTtBQUFBLElBQ1YsY0FBYztBQUFBLElBQ2QsZUFBZTtBQUFBLEVBQ2hCO0FBQ0Q7QUFFQSxlQUFzQixXQUFXLFlBQTBDO0FBQzFFLFFBQU0sY0FBYztBQUFBLElBQ25CO0FBQUEsSUFDQTtBQUFBLElBQ0E7QUFBQSxJQUNBO0FBQUEsRUFDRDtBQUVBLGFBQVcsUUFBUSxhQUFhO0FBQy9CLFVBQU0sS0FBVSxhQUFRLFlBQVksSUFBSTtBQUN4QyxRQUFJLENBQUksY0FBVyxFQUFFLEVBQUc7QUFDeEIsUUFBSTtBQUNILFlBQU0sTUFBUyxnQkFBYSxJQUFJLE9BQU87QUFDdkMsWUFBTSxTQUFTLEtBQUssTUFBTSxHQUFHO0FBQzdCLGFBQU87QUFBQSxRQUNOLEdBQUcsY0FBYztBQUFBLFFBQ2pCLEdBQUc7QUFBQSxRQUNILFVBQVUsT0FBTyxZQUFZO0FBQUEsUUFDN0IsY0FBYyxPQUFPLGdCQUFnQjtBQUFBLFFBQ3JDLGVBQWUsT0FBTyxpQkFBaUI7QUFBQSxNQUN4QztBQUFBLElBQ0QsU0FBUyxLQUFLO0FBQ2IsY0FBUSxNQUFNLHVDQUF1QyxlQUFlLFFBQVEsSUFBSSxVQUFVLEdBQUc7QUFBQSxJQUM5RjtBQUFBLEVBQ0Q7QUFFQSxTQUFPLGNBQWM7QUFDdEI7QUFFTyxTQUFTLFlBQ2YsWUFDQSxRQUN3RDtBQUN4RCxRQUFNLFVBQVUsT0FBTyxRQUFRO0FBQy9CLFFBQU0sU0FBZ0UsQ0FBQztBQUV2RSxhQUFXLENBQUMsT0FBTyxLQUFLLEtBQUssT0FBTyxRQUFRLE9BQU8sR0FBRztBQUNyRCxlQUFXLEtBQUssT0FBTztBQUN0QixhQUFPLEtBQUs7QUFBQSxRQUNYO0FBQUEsUUFDQSxLQUFLLEVBQUUsV0FBVyxHQUFHLEtBQUssRUFBRSxXQUFXLEdBQUcsSUFBSSxFQUFFLFFBQVEsTUFBTSxRQUFRLENBQUMsSUFBUyxhQUFRLFlBQVksQ0FBQztBQUFBLFFBQ3JHLFFBQVE7QUFBQSxNQUNULENBQUM7QUFBQSxJQUNGO0FBQUEsRUFDRDtBQUVBLFNBQU87QUFDUjsiLAogICJuYW1lcyI6IFtdCn0K
@@ -0,0 +1,70 @@
1
+ import * as fs from "node:fs";
2
+ import { homedir } from "node:os";
3
+ import * as path from "node:path";
4
+ import type { RulesConfig } from "./types.js";
5
+
6
+ const DEFAULT_CACHE_TTL = 30_000;
7
+
8
+ const DEFAULT_DIRS = {
9
+ managed: ["/etc/claude-code/.claude/rules"],
10
+ user: [path.join(homedir(), ".claude", "rules"), path.join(homedir(), ".config", "opencode", "rules")],
11
+ pi: [".pi/rules"],
12
+ project: [".claude/rules", ".opencode/rules", ".trae/rules"],
13
+ };
14
+
15
+ function defaultConfig(): RulesConfig {
16
+ return {
17
+ cacheTTL: DEFAULT_CACHE_TTL,
18
+ notifyOnLoad: true,
19
+ notifyOnMatch: true,
20
+ };
21
+ }
22
+
23
+ export async function loadConfig(projectDir: string): Promise<RulesConfig> {
24
+ const configFiles = [
25
+ ".rules-config.json",
26
+ ".pi/rules-config.json",
27
+ ".claude/rules-config.json",
28
+ ".opencode/rules-config.json",
29
+ ];
30
+
31
+ for (const name of configFiles) {
32
+ const fp = path.resolve(projectDir, name);
33
+ if (!fs.existsSync(fp)) continue;
34
+ try {
35
+ const raw = fs.readFileSync(fp, "utf-8");
36
+ const parsed = JSON.parse(raw);
37
+ return {
38
+ ...defaultConfig(),
39
+ ...parsed,
40
+ cacheTTL: parsed.cacheTTL ?? DEFAULT_CACHE_TTL,
41
+ notifyOnLoad: parsed.notifyOnLoad ?? true,
42
+ notifyOnMatch: parsed.notifyOnMatch ?? true,
43
+ };
44
+ } catch (err) {
45
+ console.debug("[rules-engine] config parse failed:", err instanceof Error ? err.message : err);
46
+ }
47
+ }
48
+
49
+ return defaultConfig();
50
+ }
51
+
52
+ export function resolveDirs(
53
+ projectDir: string,
54
+ config: RulesConfig,
55
+ ): Array<{ scope: string; dir: string; source: string }> {
56
+ const sources = config.dirs || DEFAULT_DIRS;
57
+ const result: Array<{ scope: string; dir: string; source: string }> = [];
58
+
59
+ for (const [scope, paths] of Object.entries(sources)) {
60
+ for (const p of paths) {
61
+ result.push({
62
+ scope,
63
+ dir: p.startsWith("/") || p.startsWith("~") ? p.replace(/^~/, homedir()) : path.resolve(projectDir, p),
64
+ source: p,
65
+ });
66
+ }
67
+ }
68
+
69
+ return result;
70
+ }