@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.
- 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 +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
|
+
}
|