@fleetagent/pi-coding-agent 0.0.4 → 0.0.6

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 (95) hide show
  1. package/CHANGELOG.md +48 -0
  2. package/README.md +28 -5
  3. package/dist/cli/args.d.ts +2 -0
  4. package/dist/cli/args.d.ts.map +1 -1
  5. package/dist/cli/args.js +9 -0
  6. package/dist/cli/args.js.map +1 -1
  7. package/dist/core/agent-session.d.ts +4 -3
  8. package/dist/core/agent-session.d.ts.map +1 -1
  9. package/dist/core/agent-session.js +44 -8
  10. package/dist/core/agent-session.js.map +1 -1
  11. package/dist/core/diagnostics.d.ts +1 -1
  12. package/dist/core/diagnostics.d.ts.map +1 -1
  13. package/dist/core/diagnostics.js.map +1 -1
  14. package/dist/core/extensions/runner.d.ts +4 -0
  15. package/dist/core/extensions/runner.d.ts.map +1 -1
  16. package/dist/core/extensions/runner.js +5 -1
  17. package/dist/core/extensions/runner.js.map +1 -1
  18. package/dist/core/extensions/types.d.ts +2 -1
  19. package/dist/core/extensions/types.d.ts.map +1 -1
  20. package/dist/core/extensions/types.js.map +1 -1
  21. package/dist/core/package-manager.d.ts +1 -0
  22. package/dist/core/package-manager.d.ts.map +1 -1
  23. package/dist/core/package-manager.js +130 -12
  24. package/dist/core/package-manager.js.map +1 -1
  25. package/dist/core/resource-loader.d.ts +30 -0
  26. package/dist/core/resource-loader.d.ts.map +1 -1
  27. package/dist/core/resource-loader.js +94 -0
  28. package/dist/core/resource-loader.js.map +1 -1
  29. package/dist/core/rules.d.ts +57 -0
  30. package/dist/core/rules.d.ts.map +1 -0
  31. package/dist/core/rules.js +384 -0
  32. package/dist/core/rules.js.map +1 -0
  33. package/dist/core/settings-manager.d.ts +5 -0
  34. package/dist/core/settings-manager.d.ts.map +1 -1
  35. package/dist/core/settings-manager.js +14 -0
  36. package/dist/core/settings-manager.js.map +1 -1
  37. package/dist/core/slash-commands.d.ts +1 -1
  38. package/dist/core/slash-commands.d.ts.map +1 -1
  39. package/dist/core/slash-commands.js +1 -1
  40. package/dist/core/slash-commands.js.map +1 -1
  41. package/dist/core/system-prompt.d.ts +3 -0
  42. package/dist/core/system-prompt.d.ts.map +1 -1
  43. package/dist/core/system-prompt.js +11 -3
  44. package/dist/core/system-prompt.js.map +1 -1
  45. package/dist/core/tools/read.d.ts.map +1 -1
  46. package/dist/core/tools/read.js +6 -2
  47. package/dist/core/tools/read.js.map +1 -1
  48. package/dist/index.d.ts +1 -0
  49. package/dist/index.d.ts.map +1 -1
  50. package/dist/index.js +2 -0
  51. package/dist/index.js.map +1 -1
  52. package/dist/main.d.ts.map +1 -1
  53. package/dist/main.js +3 -0
  54. package/dist/main.js.map +1 -1
  55. package/dist/modes/interactive/components/config-selector.d.ts +1 -1
  56. package/dist/modes/interactive/components/config-selector.d.ts.map +1 -1
  57. package/dist/modes/interactive/components/config-selector.js +12 -3
  58. package/dist/modes/interactive/components/config-selector.js.map +1 -1
  59. package/dist/modes/interactive/components/settings-selector.d.ts.map +1 -1
  60. package/dist/modes/interactive/components/settings-selector.js +1 -1
  61. package/dist/modes/interactive/components/settings-selector.js.map +1 -1
  62. package/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  63. package/dist/modes/interactive/interactive-mode.js +34 -4
  64. package/dist/modes/interactive/interactive-mode.js.map +1 -1
  65. package/dist/modes/rpc/rpc-client.d.ts +14 -1
  66. package/dist/modes/rpc/rpc-client.d.ts.map +1 -1
  67. package/dist/modes/rpc/rpc-client.js +34 -11
  68. package/dist/modes/rpc/rpc-client.js.map +1 -1
  69. package/dist/modes/rpc/rpc-mode.d.ts.map +1 -1
  70. package/dist/modes/rpc/rpc-mode.js +18 -0
  71. package/dist/modes/rpc/rpc-mode.js.map +1 -1
  72. package/dist/modes/rpc/rpc-types.d.ts +17 -2
  73. package/dist/modes/rpc/rpc-types.d.ts.map +1 -1
  74. package/dist/modes/rpc/rpc-types.js.map +1 -1
  75. package/docs/extensions.md +9 -8
  76. package/docs/index.md +3 -2
  77. package/docs/packages.md +6 -4
  78. package/docs/quickstart.md +1 -1
  79. package/docs/rpc.md +4 -2
  80. package/docs/rules.md +102 -0
  81. package/docs/sdk.md +1 -1
  82. package/docs/settings.md +3 -2
  83. package/docs/usage.md +4 -2
  84. package/examples/extensions/README.md +1 -1
  85. package/examples/extensions/custom-provider-anthropic/package.json +1 -1
  86. package/examples/extensions/custom-provider-gitlab-duo/package.json +1 -1
  87. package/examples/extensions/dynamic-resources/RULES.md +8 -0
  88. package/examples/extensions/dynamic-resources/index.ts +1 -0
  89. package/examples/extensions/reload-runtime.ts +2 -2
  90. package/examples/extensions/sandbox/package.json +1 -1
  91. package/examples/extensions/with-deps/package.json +1 -1
  92. package/examples/sdk/12-full-control.ts +1 -0
  93. package/examples/sdk/README.md +1 -1
  94. package/npm-shrinkwrap.json +12 -12
  95. package/package.json +4 -4
@@ -0,0 +1,384 @@
1
+ import { existsSync, readdirSync, readFileSync, statSync } from "fs";
2
+ import ignore from "ignore";
3
+ import { basename, dirname, join, relative, resolve, sep } from "path";
4
+ import { CONFIG_DIR_NAME, getAgentDir } from "../config.js";
5
+ import { parseFrontmatter } from "../utils/frontmatter.js";
6
+ import { canonicalizePath, resolvePath } from "../utils/paths.js";
7
+ import { createSyntheticSourceInfo } from "./source-info.js";
8
+ /** Max name length per spec */
9
+ const MAX_NAME_LENGTH = 64;
10
+ /** Max description length per spec */
11
+ const MAX_DESCRIPTION_LENGTH = 1024;
12
+ const IGNORE_FILE_NAMES = [".gitignore", ".ignore", ".fdignore"];
13
+ function toPosixPath(p) {
14
+ return p.split(sep).join("/");
15
+ }
16
+ function prefixIgnorePattern(line, prefix) {
17
+ const trimmed = line.trim();
18
+ if (!trimmed)
19
+ return null;
20
+ if (trimmed.startsWith("#") && !trimmed.startsWith("\\#"))
21
+ return null;
22
+ let pattern = line;
23
+ let negated = false;
24
+ if (pattern.startsWith("!")) {
25
+ negated = true;
26
+ pattern = pattern.slice(1);
27
+ }
28
+ else if (pattern.startsWith("\\!")) {
29
+ pattern = pattern.slice(1);
30
+ }
31
+ if (pattern.startsWith("/")) {
32
+ pattern = pattern.slice(1);
33
+ }
34
+ const prefixed = prefix ? `${prefix}${pattern}` : pattern;
35
+ return negated ? `!${prefixed}` : prefixed;
36
+ }
37
+ function addIgnoreRules(ig, dir, rootDir) {
38
+ const relativeDir = relative(rootDir, dir);
39
+ const prefix = relativeDir ? `${toPosixPath(relativeDir)}/` : "";
40
+ for (const filename of IGNORE_FILE_NAMES) {
41
+ const ignorePath = join(dir, filename);
42
+ if (!existsSync(ignorePath))
43
+ continue;
44
+ try {
45
+ const content = readFileSync(ignorePath, "utf-8");
46
+ const patterns = content
47
+ .split(/\r?\n/)
48
+ .map((line) => prefixIgnorePattern(line, prefix))
49
+ .filter((line) => Boolean(line));
50
+ if (patterns.length > 0) {
51
+ ig.add(patterns);
52
+ }
53
+ }
54
+ catch { }
55
+ }
56
+ }
57
+ /**
58
+ * Validate rule name.
59
+ * Returns array of validation error messages (empty if valid).
60
+ */
61
+ function validateName(name) {
62
+ const errors = [];
63
+ if (name.length > MAX_NAME_LENGTH) {
64
+ errors.push(`name exceeds ${MAX_NAME_LENGTH} characters (${name.length})`);
65
+ }
66
+ if (!/^[a-z0-9-]+$/.test(name)) {
67
+ errors.push(`name contains invalid characters (must be lowercase a-z, 0-9, hyphens only)`);
68
+ }
69
+ if (name.startsWith("-") || name.endsWith("-")) {
70
+ errors.push(`name must not start or end with a hyphen`);
71
+ }
72
+ if (name.includes("--")) {
73
+ errors.push(`name must not contain consecutive hyphens`);
74
+ }
75
+ return errors;
76
+ }
77
+ /**
78
+ * Validate description.
79
+ */
80
+ function validateDescription(description) {
81
+ const errors = [];
82
+ if (!description || description.trim() === "") {
83
+ errors.push("description is required");
84
+ }
85
+ else if (description.length > MAX_DESCRIPTION_LENGTH) {
86
+ errors.push(`description exceeds ${MAX_DESCRIPTION_LENGTH} characters (${description.length})`);
87
+ }
88
+ return errors;
89
+ }
90
+ function createRuleSourceInfo(filePath, baseDir, source) {
91
+ switch (source) {
92
+ case "user":
93
+ return createSyntheticSourceInfo(filePath, {
94
+ source: "local",
95
+ scope: "user",
96
+ baseDir,
97
+ });
98
+ case "project":
99
+ return createSyntheticSourceInfo(filePath, {
100
+ source: "local",
101
+ scope: "project",
102
+ baseDir,
103
+ });
104
+ case "path":
105
+ return createSyntheticSourceInfo(filePath, {
106
+ source: "local",
107
+ baseDir,
108
+ });
109
+ default:
110
+ return createSyntheticSourceInfo(filePath, { source, baseDir });
111
+ }
112
+ }
113
+ /**
114
+ * Load rules from a directory.
115
+ *
116
+ * Discovery rules:
117
+ * - if a directory contains RULES.md, treat it as a rule root and do not recurse further
118
+ * - otherwise, load direct .md children in the root
119
+ * - recurse into subdirectories to find RULES.md
120
+ */
121
+ export function loadRulesFromDir(options) {
122
+ const { dir, source } = options;
123
+ return loadRulesFromDirInternal(dir, source, true);
124
+ }
125
+ function loadRulesFromDirInternal(dir, source, includeRootFiles, ignoreMatcher, rootDir) {
126
+ const rules = [];
127
+ const diagnostics = [];
128
+ if (!existsSync(dir)) {
129
+ return { rules, diagnostics };
130
+ }
131
+ const root = rootDir ?? dir;
132
+ const ig = ignoreMatcher ?? ignore();
133
+ addIgnoreRules(ig, dir, root);
134
+ try {
135
+ const entries = readdirSync(dir, { withFileTypes: true });
136
+ for (const entry of entries) {
137
+ if (entry.name !== "RULES.md") {
138
+ continue;
139
+ }
140
+ const fullPath = join(dir, entry.name);
141
+ let isFile = entry.isFile();
142
+ if (entry.isSymbolicLink()) {
143
+ try {
144
+ isFile = statSync(fullPath).isFile();
145
+ }
146
+ catch {
147
+ continue;
148
+ }
149
+ }
150
+ const relPath = toPosixPath(relative(root, fullPath));
151
+ if (!isFile || ig.ignores(relPath)) {
152
+ continue;
153
+ }
154
+ const result = loadRuleFromFile(fullPath, source);
155
+ if (result.rule) {
156
+ rules.push(result.rule);
157
+ }
158
+ diagnostics.push(...result.diagnostics);
159
+ return { rules, diagnostics };
160
+ }
161
+ for (const entry of entries) {
162
+ if (entry.name.startsWith(".")) {
163
+ continue;
164
+ }
165
+ // Skip node_modules to avoid scanning dependencies
166
+ if (entry.name === "node_modules") {
167
+ continue;
168
+ }
169
+ const fullPath = join(dir, entry.name);
170
+ // For symlinks, check if they point to a directory and follow them
171
+ let isDirectory = entry.isDirectory();
172
+ let isFile = entry.isFile();
173
+ if (entry.isSymbolicLink()) {
174
+ try {
175
+ const stats = statSync(fullPath);
176
+ isDirectory = stats.isDirectory();
177
+ isFile = stats.isFile();
178
+ }
179
+ catch {
180
+ // Broken symlink, skip it
181
+ continue;
182
+ }
183
+ }
184
+ const relPath = toPosixPath(relative(root, fullPath));
185
+ const ignorePath = isDirectory ? `${relPath}/` : relPath;
186
+ if (ig.ignores(ignorePath)) {
187
+ continue;
188
+ }
189
+ if (isDirectory) {
190
+ const subResult = loadRulesFromDirInternal(fullPath, source, false, ig, root);
191
+ rules.push(...subResult.rules);
192
+ diagnostics.push(...subResult.diagnostics);
193
+ continue;
194
+ }
195
+ if (!isFile || !includeRootFiles || !entry.name.endsWith(".md")) {
196
+ continue;
197
+ }
198
+ const result = loadRuleFromFile(fullPath, source);
199
+ if (result.rule) {
200
+ rules.push(result.rule);
201
+ }
202
+ diagnostics.push(...result.diagnostics);
203
+ }
204
+ }
205
+ catch { }
206
+ return { rules, diagnostics };
207
+ }
208
+ function loadRuleFromFile(filePath, source) {
209
+ const diagnostics = [];
210
+ try {
211
+ const rawContent = readFileSync(filePath, "utf-8");
212
+ const { frontmatter } = parseFrontmatter(rawContent);
213
+ const ruleDir = dirname(filePath);
214
+ const parentDirName = basename(ruleDir);
215
+ // Validate description
216
+ const descErrors = validateDescription(frontmatter.description);
217
+ for (const error of descErrors) {
218
+ diagnostics.push({ type: "warning", message: error, path: filePath });
219
+ }
220
+ // Use name from frontmatter, or fall back to parent directory name
221
+ const name = frontmatter.name || parentDirName;
222
+ // Validate name
223
+ const nameErrors = validateName(name);
224
+ for (const error of nameErrors) {
225
+ diagnostics.push({ type: "warning", message: error, path: filePath });
226
+ }
227
+ // Still load the rule even with warnings (unless description is completely missing)
228
+ if (!frontmatter.description || frontmatter.description.trim() === "") {
229
+ return { rule: null, diagnostics };
230
+ }
231
+ return {
232
+ rule: {
233
+ name,
234
+ description: frontmatter.description,
235
+ filePath,
236
+ baseDir: ruleDir,
237
+ sourceInfo: createRuleSourceInfo(filePath, ruleDir, source),
238
+ disableModelInvocation: frontmatter["disable-model-invocation"] === true,
239
+ },
240
+ diagnostics,
241
+ };
242
+ }
243
+ catch (error) {
244
+ const message = error instanceof Error ? error.message : "failed to parse rule file";
245
+ diagnostics.push({ type: "warning", message, path: filePath });
246
+ return { rule: null, diagnostics };
247
+ }
248
+ }
249
+ /**
250
+ * Format rules for inclusion in a system prompt.
251
+ * Rules with disableModelInvocation=true are excluded from the prompt
252
+ * (they can only be invoked explicitly via /rule:name commands).
253
+ */
254
+ export function formatRulesForPrompt(rules) {
255
+ const visibleRules = rules.filter((s) => !s.disableModelInvocation);
256
+ if (visibleRules.length === 0) {
257
+ return "";
258
+ }
259
+ const lines = [
260
+ "\n\nThe following rules provide mandatory constraints and policies.",
261
+ "Use the read tool to load a rule's file when the task or files match its description; applicable rules are mandatory.",
262
+ "When a rule file references a relative path, resolve it against the rule directory (parent of RULES.md / dirname of the path) and use that absolute path in tool commands.",
263
+ "",
264
+ "<available_rules>",
265
+ ];
266
+ for (const rule of visibleRules) {
267
+ lines.push(" <rule>");
268
+ lines.push(` <name>${escapeXml(rule.name)}</name>`);
269
+ lines.push(` <description>${escapeXml(rule.description)}</description>`);
270
+ lines.push(` <location>${escapeXml(rule.filePath)}</location>`);
271
+ lines.push(" </rule>");
272
+ }
273
+ lines.push("</available_rules>");
274
+ return lines.join("\n");
275
+ }
276
+ function escapeXml(str) {
277
+ return str
278
+ .replace(/&/g, "&amp;")
279
+ .replace(/</g, "&lt;")
280
+ .replace(/>/g, "&gt;")
281
+ .replace(/"/g, "&quot;")
282
+ .replace(/'/g, "&apos;");
283
+ }
284
+ /**
285
+ * Load rules from all configured locations.
286
+ * Returns rules and any validation diagnostics.
287
+ */
288
+ export function loadRules(options) {
289
+ const { agentDir, rulePaths, includeDefaults } = options;
290
+ // Resolve agentDir - if not provided, use default from config
291
+ const resolvedCwd = resolvePath(options.cwd);
292
+ const resolvedAgentDir = resolvePath(agentDir ?? getAgentDir());
293
+ const ruleMap = new Map();
294
+ const realPathSet = new Set();
295
+ const allDiagnostics = [];
296
+ const collisionDiagnostics = [];
297
+ function addRules(result) {
298
+ allDiagnostics.push(...result.diagnostics);
299
+ for (const rule of result.rules) {
300
+ // Resolve symlinks to detect duplicate files
301
+ const realPath = canonicalizePath(rule.filePath);
302
+ // Skip silently if we've already loaded this exact file (via symlink)
303
+ if (realPathSet.has(realPath)) {
304
+ continue;
305
+ }
306
+ const existing = ruleMap.get(rule.name);
307
+ if (existing) {
308
+ collisionDiagnostics.push({
309
+ type: "collision",
310
+ message: `name "${rule.name}" collision`,
311
+ path: rule.filePath,
312
+ collision: {
313
+ resourceType: "rule",
314
+ name: rule.name,
315
+ winnerPath: existing.filePath,
316
+ loserPath: rule.filePath,
317
+ },
318
+ });
319
+ }
320
+ else {
321
+ ruleMap.set(rule.name, rule);
322
+ realPathSet.add(realPath);
323
+ }
324
+ }
325
+ }
326
+ if (includeDefaults) {
327
+ addRules(loadRulesFromDirInternal(join(resolvedAgentDir, "rules"), "user", true));
328
+ addRules(loadRulesFromDirInternal(resolve(resolvedCwd, CONFIG_DIR_NAME, "rules"), "project", true));
329
+ }
330
+ const userRulesDir = join(resolvedAgentDir, "rules");
331
+ const projectRulesDir = resolve(resolvedCwd, CONFIG_DIR_NAME, "rules");
332
+ const isUnderPath = (target, root) => {
333
+ const normalizedRoot = resolve(root);
334
+ if (target === normalizedRoot) {
335
+ return true;
336
+ }
337
+ const prefix = normalizedRoot.endsWith(sep) ? normalizedRoot : `${normalizedRoot}${sep}`;
338
+ return target.startsWith(prefix);
339
+ };
340
+ const getSource = (resolvedPath) => {
341
+ if (!includeDefaults) {
342
+ if (isUnderPath(resolvedPath, userRulesDir))
343
+ return "user";
344
+ if (isUnderPath(resolvedPath, projectRulesDir))
345
+ return "project";
346
+ }
347
+ return "path";
348
+ };
349
+ for (const rawPath of rulePaths) {
350
+ const resolvedPath = resolvePath(rawPath, resolvedCwd, { trim: true });
351
+ if (!existsSync(resolvedPath)) {
352
+ allDiagnostics.push({ type: "warning", message: "rule path does not exist", path: resolvedPath });
353
+ continue;
354
+ }
355
+ try {
356
+ const stats = statSync(resolvedPath);
357
+ const source = getSource(resolvedPath);
358
+ if (stats.isDirectory()) {
359
+ addRules(loadRulesFromDirInternal(resolvedPath, source, true));
360
+ }
361
+ else if (stats.isFile() && resolvedPath.endsWith(".md")) {
362
+ const result = loadRuleFromFile(resolvedPath, source);
363
+ if (result.rule) {
364
+ addRules({ rules: [result.rule], diagnostics: result.diagnostics });
365
+ }
366
+ else {
367
+ allDiagnostics.push(...result.diagnostics);
368
+ }
369
+ }
370
+ else {
371
+ allDiagnostics.push({ type: "warning", message: "rule path is not a markdown file", path: resolvedPath });
372
+ }
373
+ }
374
+ catch (error) {
375
+ const message = error instanceof Error ? error.message : "failed to read rule path";
376
+ allDiagnostics.push({ type: "warning", message, path: resolvedPath });
377
+ }
378
+ }
379
+ return {
380
+ rules: Array.from(ruleMap.values()),
381
+ diagnostics: [...allDiagnostics, ...collisionDiagnostics],
382
+ };
383
+ }
384
+ //# sourceMappingURL=rules.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"rules.js","sourceRoot":"","sources":["../../src/core/rules.ts"],"names":[],"mappings":"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","sourcesContent":["import { existsSync, readdirSync, readFileSync, statSync } from \"fs\";\nimport ignore from \"ignore\";\nimport { basename, dirname, join, relative, resolve, sep } from \"path\";\nimport { CONFIG_DIR_NAME, getAgentDir } from \"../config.ts\";\nimport { parseFrontmatter } from \"../utils/frontmatter.ts\";\nimport { canonicalizePath, resolvePath } from \"../utils/paths.ts\";\nimport type { ResourceDiagnostic } from \"./diagnostics.ts\";\nimport { createSyntheticSourceInfo, type SourceInfo } from \"./source-info.ts\";\n\n/** Max name length per spec */\nconst MAX_NAME_LENGTH = 64;\n\n/** Max description length per spec */\nconst MAX_DESCRIPTION_LENGTH = 1024;\n\nconst IGNORE_FILE_NAMES = [\".gitignore\", \".ignore\", \".fdignore\"];\n\ntype IgnoreMatcher = ReturnType<typeof ignore>;\n\nfunction toPosixPath(p: string): string {\n\treturn p.split(sep).join(\"/\");\n}\n\nfunction prefixIgnorePattern(line: string, prefix: string): string | null {\n\tconst trimmed = line.trim();\n\tif (!trimmed) return null;\n\tif (trimmed.startsWith(\"#\") && !trimmed.startsWith(\"\\\\#\")) return null;\n\n\tlet pattern = line;\n\tlet negated = false;\n\n\tif (pattern.startsWith(\"!\")) {\n\t\tnegated = true;\n\t\tpattern = pattern.slice(1);\n\t} else if (pattern.startsWith(\"\\\\!\")) {\n\t\tpattern = pattern.slice(1);\n\t}\n\n\tif (pattern.startsWith(\"/\")) {\n\t\tpattern = pattern.slice(1);\n\t}\n\n\tconst prefixed = prefix ? `${prefix}${pattern}` : pattern;\n\treturn negated ? `!${prefixed}` : prefixed;\n}\n\nfunction addIgnoreRules(ig: IgnoreMatcher, dir: string, rootDir: string): void {\n\tconst relativeDir = relative(rootDir, dir);\n\tconst prefix = relativeDir ? `${toPosixPath(relativeDir)}/` : \"\";\n\n\tfor (const filename of IGNORE_FILE_NAMES) {\n\t\tconst ignorePath = join(dir, filename);\n\t\tif (!existsSync(ignorePath)) continue;\n\t\ttry {\n\t\t\tconst content = readFileSync(ignorePath, \"utf-8\");\n\t\t\tconst patterns = content\n\t\t\t\t.split(/\\r?\\n/)\n\t\t\t\t.map((line) => prefixIgnorePattern(line, prefix))\n\t\t\t\t.filter((line): line is string => Boolean(line));\n\t\t\tif (patterns.length > 0) {\n\t\t\t\tig.add(patterns);\n\t\t\t}\n\t\t} catch {}\n\t}\n}\n\nexport interface RuleFrontmatter {\n\tname?: string;\n\tdescription?: string;\n\t\"disable-model-invocation\"?: boolean;\n\t[key: string]: unknown;\n}\n\nexport interface Rule {\n\tname: string;\n\tdescription: string;\n\tfilePath: string;\n\tbaseDir: string;\n\tsourceInfo: SourceInfo;\n\tdisableModelInvocation: boolean;\n}\n\nexport interface LoadRulesResult {\n\trules: Rule[];\n\tdiagnostics: ResourceDiagnostic[];\n}\n\n/**\n * Validate rule name.\n * Returns array of validation error messages (empty if valid).\n */\nfunction validateName(name: string): string[] {\n\tconst errors: string[] = [];\n\n\tif (name.length > MAX_NAME_LENGTH) {\n\t\terrors.push(`name exceeds ${MAX_NAME_LENGTH} characters (${name.length})`);\n\t}\n\n\tif (!/^[a-z0-9-]+$/.test(name)) {\n\t\terrors.push(`name contains invalid characters (must be lowercase a-z, 0-9, hyphens only)`);\n\t}\n\n\tif (name.startsWith(\"-\") || name.endsWith(\"-\")) {\n\t\terrors.push(`name must not start or end with a hyphen`);\n\t}\n\n\tif (name.includes(\"--\")) {\n\t\terrors.push(`name must not contain consecutive hyphens`);\n\t}\n\n\treturn errors;\n}\n\n/**\n * Validate description.\n */\nfunction validateDescription(description: string | undefined): string[] {\n\tconst errors: string[] = [];\n\n\tif (!description || description.trim() === \"\") {\n\t\terrors.push(\"description is required\");\n\t} else if (description.length > MAX_DESCRIPTION_LENGTH) {\n\t\terrors.push(`description exceeds ${MAX_DESCRIPTION_LENGTH} characters (${description.length})`);\n\t}\n\n\treturn errors;\n}\n\nexport interface LoadRulesFromDirOptions {\n\t/** Directory to scan for rules */\n\tdir: string;\n\t/** Source identifier for these rules */\n\tsource: string;\n}\n\nfunction createRuleSourceInfo(filePath: string, baseDir: string, source: string): SourceInfo {\n\tswitch (source) {\n\t\tcase \"user\":\n\t\t\treturn createSyntheticSourceInfo(filePath, {\n\t\t\t\tsource: \"local\",\n\t\t\t\tscope: \"user\",\n\t\t\t\tbaseDir,\n\t\t\t});\n\t\tcase \"project\":\n\t\t\treturn createSyntheticSourceInfo(filePath, {\n\t\t\t\tsource: \"local\",\n\t\t\t\tscope: \"project\",\n\t\t\t\tbaseDir,\n\t\t\t});\n\t\tcase \"path\":\n\t\t\treturn createSyntheticSourceInfo(filePath, {\n\t\t\t\tsource: \"local\",\n\t\t\t\tbaseDir,\n\t\t\t});\n\t\tdefault:\n\t\t\treturn createSyntheticSourceInfo(filePath, { source, baseDir });\n\t}\n}\n\n/**\n * Load rules from a directory.\n *\n * Discovery rules:\n * - if a directory contains RULES.md, treat it as a rule root and do not recurse further\n * - otherwise, load direct .md children in the root\n * - recurse into subdirectories to find RULES.md\n */\nexport function loadRulesFromDir(options: LoadRulesFromDirOptions): LoadRulesResult {\n\tconst { dir, source } = options;\n\treturn loadRulesFromDirInternal(dir, source, true);\n}\n\nfunction loadRulesFromDirInternal(\n\tdir: string,\n\tsource: string,\n\tincludeRootFiles: boolean,\n\tignoreMatcher?: IgnoreMatcher,\n\trootDir?: string,\n): LoadRulesResult {\n\tconst rules: Rule[] = [];\n\tconst diagnostics: ResourceDiagnostic[] = [];\n\n\tif (!existsSync(dir)) {\n\t\treturn { rules, diagnostics };\n\t}\n\n\tconst root = rootDir ?? dir;\n\tconst ig = ignoreMatcher ?? ignore();\n\taddIgnoreRules(ig, dir, root);\n\n\ttry {\n\t\tconst entries = readdirSync(dir, { withFileTypes: true });\n\n\t\tfor (const entry of entries) {\n\t\t\tif (entry.name !== \"RULES.md\") {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tconst fullPath = join(dir, entry.name);\n\n\t\t\tlet isFile = entry.isFile();\n\t\t\tif (entry.isSymbolicLink()) {\n\t\t\t\ttry {\n\t\t\t\t\tisFile = statSync(fullPath).isFile();\n\t\t\t\t} catch {\n\t\t\t\t\tcontinue;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tconst relPath = toPosixPath(relative(root, fullPath));\n\t\t\tif (!isFile || ig.ignores(relPath)) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tconst result = loadRuleFromFile(fullPath, source);\n\t\t\tif (result.rule) {\n\t\t\t\trules.push(result.rule);\n\t\t\t}\n\t\t\tdiagnostics.push(...result.diagnostics);\n\t\t\treturn { rules, diagnostics };\n\t\t}\n\n\t\tfor (const entry of entries) {\n\t\t\tif (entry.name.startsWith(\".\")) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\t// Skip node_modules to avoid scanning dependencies\n\t\t\tif (entry.name === \"node_modules\") {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tconst fullPath = join(dir, entry.name);\n\n\t\t\t// For symlinks, check if they point to a directory and follow them\n\t\t\tlet isDirectory = entry.isDirectory();\n\t\t\tlet isFile = entry.isFile();\n\t\t\tif (entry.isSymbolicLink()) {\n\t\t\t\ttry {\n\t\t\t\t\tconst stats = statSync(fullPath);\n\t\t\t\t\tisDirectory = stats.isDirectory();\n\t\t\t\t\tisFile = stats.isFile();\n\t\t\t\t} catch {\n\t\t\t\t\t// Broken symlink, skip it\n\t\t\t\t\tcontinue;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tconst relPath = toPosixPath(relative(root, fullPath));\n\t\t\tconst ignorePath = isDirectory ? `${relPath}/` : relPath;\n\t\t\tif (ig.ignores(ignorePath)) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tif (isDirectory) {\n\t\t\t\tconst subResult = loadRulesFromDirInternal(fullPath, source, false, ig, root);\n\t\t\t\trules.push(...subResult.rules);\n\t\t\t\tdiagnostics.push(...subResult.diagnostics);\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tif (!isFile || !includeRootFiles || !entry.name.endsWith(\".md\")) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tconst result = loadRuleFromFile(fullPath, source);\n\t\t\tif (result.rule) {\n\t\t\t\trules.push(result.rule);\n\t\t\t}\n\t\t\tdiagnostics.push(...result.diagnostics);\n\t\t}\n\t} catch {}\n\n\treturn { rules, diagnostics };\n}\n\nfunction loadRuleFromFile(filePath: string, source: string): { rule: Rule | null; diagnostics: ResourceDiagnostic[] } {\n\tconst diagnostics: ResourceDiagnostic[] = [];\n\n\ttry {\n\t\tconst rawContent = readFileSync(filePath, \"utf-8\");\n\t\tconst { frontmatter } = parseFrontmatter<RuleFrontmatter>(rawContent);\n\t\tconst ruleDir = dirname(filePath);\n\t\tconst parentDirName = basename(ruleDir);\n\n\t\t// Validate description\n\t\tconst descErrors = validateDescription(frontmatter.description);\n\t\tfor (const error of descErrors) {\n\t\t\tdiagnostics.push({ type: \"warning\", message: error, path: filePath });\n\t\t}\n\n\t\t// Use name from frontmatter, or fall back to parent directory name\n\t\tconst name = frontmatter.name || parentDirName;\n\n\t\t// Validate name\n\t\tconst nameErrors = validateName(name);\n\t\tfor (const error of nameErrors) {\n\t\t\tdiagnostics.push({ type: \"warning\", message: error, path: filePath });\n\t\t}\n\n\t\t// Still load the rule even with warnings (unless description is completely missing)\n\t\tif (!frontmatter.description || frontmatter.description.trim() === \"\") {\n\t\t\treturn { rule: null, diagnostics };\n\t\t}\n\n\t\treturn {\n\t\t\trule: {\n\t\t\t\tname,\n\t\t\t\tdescription: frontmatter.description,\n\t\t\t\tfilePath,\n\t\t\t\tbaseDir: ruleDir,\n\t\t\t\tsourceInfo: createRuleSourceInfo(filePath, ruleDir, source),\n\t\t\t\tdisableModelInvocation: frontmatter[\"disable-model-invocation\"] === true,\n\t\t\t},\n\t\t\tdiagnostics,\n\t\t};\n\t} catch (error) {\n\t\tconst message = error instanceof Error ? error.message : \"failed to parse rule file\";\n\t\tdiagnostics.push({ type: \"warning\", message, path: filePath });\n\t\treturn { rule: null, diagnostics };\n\t}\n}\n\n/**\n * Format rules for inclusion in a system prompt.\n * Rules with disableModelInvocation=true are excluded from the prompt\n * (they can only be invoked explicitly via /rule:name commands).\n */\nexport function formatRulesForPrompt(rules: Rule[]): string {\n\tconst visibleRules = rules.filter((s) => !s.disableModelInvocation);\n\n\tif (visibleRules.length === 0) {\n\t\treturn \"\";\n\t}\n\n\tconst lines = [\n\t\t\"\\n\\nThe following rules provide mandatory constraints and policies.\",\n\t\t\"Use the read tool to load a rule's file when the task or files match its description; applicable rules are mandatory.\",\n\t\t\"When a rule file references a relative path, resolve it against the rule directory (parent of RULES.md / dirname of the path) and use that absolute path in tool commands.\",\n\t\t\"\",\n\t\t\"<available_rules>\",\n\t];\n\n\tfor (const rule of visibleRules) {\n\t\tlines.push(\" <rule>\");\n\t\tlines.push(` <name>${escapeXml(rule.name)}</name>`);\n\t\tlines.push(` <description>${escapeXml(rule.description)}</description>`);\n\t\tlines.push(` <location>${escapeXml(rule.filePath)}</location>`);\n\t\tlines.push(\" </rule>\");\n\t}\n\n\tlines.push(\"</available_rules>\");\n\n\treturn lines.join(\"\\n\");\n}\n\nfunction escapeXml(str: string): string {\n\treturn str\n\t\t.replace(/&/g, \"&amp;\")\n\t\t.replace(/</g, \"&lt;\")\n\t\t.replace(/>/g, \"&gt;\")\n\t\t.replace(/\"/g, \"&quot;\")\n\t\t.replace(/'/g, \"&apos;\");\n}\n\nexport interface LoadRulesOptions {\n\t/** Working directory for project-local rules. */\n\tcwd: string;\n\t/** Agent config directory for global rules. */\n\tagentDir: string;\n\t/** Explicit rule paths (files or directories) */\n\trulePaths: string[];\n\t/** Include default rules directories. */\n\tincludeDefaults: boolean;\n}\n\n/**\n * Load rules from all configured locations.\n * Returns rules and any validation diagnostics.\n */\nexport function loadRules(options: LoadRulesOptions): LoadRulesResult {\n\tconst { agentDir, rulePaths, includeDefaults } = options;\n\n\t// Resolve agentDir - if not provided, use default from config\n\tconst resolvedCwd = resolvePath(options.cwd);\n\tconst resolvedAgentDir = resolvePath(agentDir ?? getAgentDir());\n\n\tconst ruleMap = new Map<string, Rule>();\n\tconst realPathSet = new Set<string>();\n\tconst allDiagnostics: ResourceDiagnostic[] = [];\n\tconst collisionDiagnostics: ResourceDiagnostic[] = [];\n\n\tfunction addRules(result: LoadRulesResult) {\n\t\tallDiagnostics.push(...result.diagnostics);\n\t\tfor (const rule of result.rules) {\n\t\t\t// Resolve symlinks to detect duplicate files\n\t\t\tconst realPath = canonicalizePath(rule.filePath);\n\n\t\t\t// Skip silently if we've already loaded this exact file (via symlink)\n\t\t\tif (realPathSet.has(realPath)) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tconst existing = ruleMap.get(rule.name);\n\t\t\tif (existing) {\n\t\t\t\tcollisionDiagnostics.push({\n\t\t\t\t\ttype: \"collision\",\n\t\t\t\t\tmessage: `name \"${rule.name}\" collision`,\n\t\t\t\t\tpath: rule.filePath,\n\t\t\t\t\tcollision: {\n\t\t\t\t\t\tresourceType: \"rule\",\n\t\t\t\t\t\tname: rule.name,\n\t\t\t\t\t\twinnerPath: existing.filePath,\n\t\t\t\t\t\tloserPath: rule.filePath,\n\t\t\t\t\t},\n\t\t\t\t});\n\t\t\t} else {\n\t\t\t\truleMap.set(rule.name, rule);\n\t\t\t\trealPathSet.add(realPath);\n\t\t\t}\n\t\t}\n\t}\n\n\tif (includeDefaults) {\n\t\taddRules(loadRulesFromDirInternal(join(resolvedAgentDir, \"rules\"), \"user\", true));\n\t\taddRules(loadRulesFromDirInternal(resolve(resolvedCwd, CONFIG_DIR_NAME, \"rules\"), \"project\", true));\n\t}\n\n\tconst userRulesDir = join(resolvedAgentDir, \"rules\");\n\tconst projectRulesDir = resolve(resolvedCwd, CONFIG_DIR_NAME, \"rules\");\n\n\tconst isUnderPath = (target: string, root: string): boolean => {\n\t\tconst normalizedRoot = resolve(root);\n\t\tif (target === normalizedRoot) {\n\t\t\treturn true;\n\t\t}\n\t\tconst prefix = normalizedRoot.endsWith(sep) ? normalizedRoot : `${normalizedRoot}${sep}`;\n\t\treturn target.startsWith(prefix);\n\t};\n\n\tconst getSource = (resolvedPath: string): \"user\" | \"project\" | \"path\" => {\n\t\tif (!includeDefaults) {\n\t\t\tif (isUnderPath(resolvedPath, userRulesDir)) return \"user\";\n\t\t\tif (isUnderPath(resolvedPath, projectRulesDir)) return \"project\";\n\t\t}\n\t\treturn \"path\";\n\t};\n\n\tfor (const rawPath of rulePaths) {\n\t\tconst resolvedPath = resolvePath(rawPath, resolvedCwd, { trim: true });\n\t\tif (!existsSync(resolvedPath)) {\n\t\t\tallDiagnostics.push({ type: \"warning\", message: \"rule path does not exist\", path: resolvedPath });\n\t\t\tcontinue;\n\t\t}\n\n\t\ttry {\n\t\t\tconst stats = statSync(resolvedPath);\n\t\t\tconst source = getSource(resolvedPath);\n\t\t\tif (stats.isDirectory()) {\n\t\t\t\taddRules(loadRulesFromDirInternal(resolvedPath, source, true));\n\t\t\t} else if (stats.isFile() && resolvedPath.endsWith(\".md\")) {\n\t\t\t\tconst result = loadRuleFromFile(resolvedPath, source);\n\t\t\t\tif (result.rule) {\n\t\t\t\t\taddRules({ rules: [result.rule], diagnostics: result.diagnostics });\n\t\t\t\t} else {\n\t\t\t\t\tallDiagnostics.push(...result.diagnostics);\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\tallDiagnostics.push({ type: \"warning\", message: \"rule path is not a markdown file\", path: resolvedPath });\n\t\t\t}\n\t\t} catch (error) {\n\t\t\tconst message = error instanceof Error ? error.message : \"failed to read rule path\";\n\t\t\tallDiagnostics.push({ type: \"warning\", message, path: resolvedPath });\n\t\t}\n\t}\n\n\treturn {\n\t\trules: Array.from(ruleMap.values()),\n\t\tdiagnostics: [...allDiagnostics, ...collisionDiagnostics],\n\t};\n}\n"]}
@@ -51,6 +51,7 @@ export type PackageSource = string | {
51
51
  source: string;
52
52
  extensions?: string[];
53
53
  skills?: string[];
54
+ rules?: string[];
54
55
  prompts?: string[];
55
56
  themes?: string[];
56
57
  };
@@ -76,6 +77,7 @@ export interface Settings {
76
77
  packages?: PackageSource[];
77
78
  extensions?: string[];
78
79
  skills?: string[];
80
+ rules?: string[];
79
81
  prompts?: string[];
80
82
  themes?: string[];
81
83
  enableSkillCommands?: boolean;
@@ -224,6 +226,9 @@ export declare class SettingsManager {
224
226
  getSkillPaths(): string[];
225
227
  setSkillPaths(paths: string[]): void;
226
228
  setProjectSkillPaths(paths: string[]): void;
229
+ getRulePaths(): string[];
230
+ setRulePaths(paths: string[]): void;
231
+ setProjectRulePaths(paths: string[]): void;
227
232
  getPromptTemplatePaths(): string[];
228
233
  setPromptTemplatePaths(paths: string[]): void;
229
234
  setProjectPromptTemplatePaths(paths: string[]): void;