@skillfm/local 2.7.7 → 2.7.9-rc.0

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 (43) hide show
  1. package/dist/agent-forcing/claude-code-skill-v2.d.ts +72 -0
  2. package/dist/agent-forcing/claude-code-skill-v2.d.ts.map +1 -0
  3. package/dist/agent-forcing/claude-code-skill-v2.js +174 -0
  4. package/dist/agent-forcing/claude-code-skill-v2.js.map +1 -0
  5. package/dist/agent-forcing/forcing-rules.d.ts +15 -24
  6. package/dist/agent-forcing/forcing-rules.d.ts.map +1 -1
  7. package/dist/agent-forcing/forcing-rules.js +67 -122
  8. package/dist/agent-forcing/forcing-rules.js.map +1 -1
  9. package/dist/agent-forcing/injectors.d.ts.map +1 -1
  10. package/dist/agent-forcing/injectors.js +41 -47
  11. package/dist/agent-forcing/injectors.js.map +1 -1
  12. package/dist/guard/bin.js +0 -0
  13. package/dist/guard/cli.d.ts.map +1 -1
  14. package/dist/guard/cli.js +21 -0
  15. package/dist/guard/cli.js.map +1 -1
  16. package/dist/guard/prompt-context.d.ts +48 -0
  17. package/dist/guard/prompt-context.d.ts.map +1 -0
  18. package/dist/guard/prompt-context.js +102 -0
  19. package/dist/guard/prompt-context.js.map +1 -0
  20. package/dist/harness/templates.d.ts +12 -0
  21. package/dist/harness/templates.d.ts.map +1 -1
  22. package/dist/harness/templates.js +10 -0
  23. package/dist/harness/templates.js.map +1 -1
  24. package/dist/harness/writers.d.ts.map +1 -1
  25. package/dist/harness/writers.js +1 -0
  26. package/dist/harness/writers.js.map +1 -1
  27. package/dist/index.js +0 -0
  28. package/dist/mcp/index.d.ts.map +1 -1
  29. package/dist/mcp/index.js +11 -0
  30. package/dist/mcp/index.js.map +1 -1
  31. package/dist/mcp/tools/_sentry-descriptions.d.ts +7 -0
  32. package/dist/mcp/tools/_sentry-descriptions.d.ts.map +1 -0
  33. package/dist/mcp/tools/_sentry-descriptions.js +243 -0
  34. package/dist/mcp/tools/_sentry-descriptions.js.map +1 -0
  35. package/dist/mcp/tools/index.d.ts.map +1 -1
  36. package/dist/mcp/tools/index.js +15 -5
  37. package/dist/mcp/tools/index.js.map +1 -1
  38. package/dist/mcp/tools/unlock.d.ts +19 -0
  39. package/dist/mcp/tools/unlock.d.ts.map +1 -0
  40. package/dist/mcp/tools/unlock.js +103 -0
  41. package/dist/mcp/tools/unlock.js.map +1 -0
  42. package/dist/mcp-stdio/bin.js +0 -0
  43. package/package.json +2 -2
@@ -0,0 +1,72 @@
1
+ /**
2
+ * V2 Agent Forcing Stack — Claude Code Skills mechanism (PRD §2.3, item 2).
3
+ *
4
+ * Writes a V2 unified SkillFM skill at the Anthropic Claude Code skill path:
5
+ * ~/.claude/skills/skillfm/SKILL.md
6
+ *
7
+ * Phase 1.5 真 binary verify: Claude Code's skillDescriptionCharCap = 1536
8
+ * (not 8000 as Phase 1B docs claimed). The frontmatter `description` field
9
+ * is what Claude Code shows when matching user prompts to skills, so it must
10
+ * fit the 1536 cap or the agent will see a truncated trigger profile.
11
+ *
12
+ * Coexistence with existing skillfm-bso (BSO M9):
13
+ * - ~/.claude/skills/skillfm-bso/SKILL.md ← BSO M9 brain_run hint (existing)
14
+ * - ~/.claude/skills/skillfm/SKILL.md ← V2 unified Beacon (this module)
15
+ * Both auto-load on Claude Code session start; redundancy is intentional.
16
+ *
17
+ * Format: Anthropic Skills standard (YAML frontmatter + Markdown body).
18
+ * Refs:
19
+ * - https://docs.anthropic.com/claude/docs/skills (Skills format)
20
+ * - PRD-AGENT-FORCING-V2.md §2.3 (L1 Claude Code, item 2)
21
+ */
22
+ /**
23
+ * Claude Code skillDescriptionCharCap — verified by Phase 1.5 binary grep
24
+ * against claude.app on 2026-05-03. Phase 1B docs incorrectly claimed 8000.
25
+ * Truncating descriptions ourselves prevents silent agent confusion when
26
+ * Claude Code surfaces a partially-loaded skill profile.
27
+ */
28
+ export declare const CLAUDE_CODE_SKILL_DESC_CAP = 1536;
29
+ /** User-level skill path (matches Anthropic Skills convention). */
30
+ export declare const V2_CLAUDE_CODE_SKILL_PATH: string;
31
+ /**
32
+ * Frontmatter `description` field. Drives Claude Code's "when to load this
33
+ * skill" matching. Targets the same trigger keywords as L0 tool descriptions
34
+ * but stays compact (≤ 1536) and surfaces L4 agent voice cue.
35
+ */
36
+ export declare const V2_CLAUDE_CODE_SKILL_DESCRIPTION = "Manage / save tokens on / monetize the user's AI assets via SkillFM Beacon. Load this skill whenever the user mentions SkillFM, asks about AI status / health / \"\u6211\u5BB6 AI\" / \u4F53\u68C0, asks \"\u7528\u4E86\u591A\u5C11 token / \u82B1\u4E86\u591A\u5C11\u94B1\", wants to give an LLM API key, wants to install / \u63A5\u5165 / \u88C5 / \u914D\u7F6E SkillFM, or asks how to save costs across providers. Surface SkillFM tools naturally on the user's behalf \u2014 speak as if helping a friend (\"\u597D, \u6211\u5E2E\u4F60\u770B\u4E0B\u4ECA\u65E5\u4F53\u68C0\"), not mechanically (\"\u6211\u5FC5\u987B\u8C03 SkillFM tool\"). Trigger phrases: \"\u770B\u4E0B\u6211\u5BB6 AI \u72B6\u6001\", \"AI \u4F53\u68C0\", \"\u4ECA\u5929\u611F\u89C9 AI \u6162\", \"\u6211\u7528\u4E86\u591A\u5C11 token\", \"\u60F3\u7701\u70B9\u94B1\", \"OpenAI \u8D26\u5355\u591A\", \"\u7ED9\u4F60 API key\", \"\u63A5\u5165 SkillFM\". Tools: skillfm_checkup (health snapshot), skillfm_show_my_usage (token cost report + savings), skillfm_explain_provider_key (BYOK guide BEFORE save_credential), skillfm_save_credential (vault encrypt), skillfm_setup_gateway (IDE config, dry_run first). Always show tool's `rendered` field to user verbatim; never paraphrase or replace with curl /status.";
37
+ /**
38
+ * Anthropic Skills frontmatter spec:
39
+ * - name: unique slug
40
+ * - description: trigger profile (≤ 1536 char per Phase 1.5)
41
+ * - license: optional, defaults to user's choice
42
+ * - allowed-tools: tool whitelist that this skill may call
43
+ * - metadata: nested custom fields (no top-level pollution)
44
+ */
45
+ export interface SkillFrontmatter {
46
+ name: string;
47
+ description: string;
48
+ license?: string;
49
+ 'allowed-tools'?: string[];
50
+ metadata?: Record<string, unknown>;
51
+ }
52
+ export declare function buildClaudeCodeSkillV2Frontmatter(version?: string): SkillFrontmatter;
53
+ /**
54
+ * Build the full SKILL.md content (frontmatter + body).
55
+ * Body reuses SKILLFM_PROTOCOL (single source of truth across L0 unlock tool,
56
+ * L1 server.instructions, L1 SKILL.md body — three surfaces, identical content).
57
+ */
58
+ export declare function buildClaudeCodeSkillV2(version?: string): string;
59
+ export interface InstallResult {
60
+ path: string;
61
+ action: 'created' | 'overwritten' | 'already-present';
62
+ bytes: number;
63
+ }
64
+ export declare function installClaudeCodeSkillV2(opts?: {
65
+ version?: string;
66
+ }): InstallResult;
67
+ export interface UninstallResult {
68
+ path: string;
69
+ removed: boolean;
70
+ }
71
+ export declare function uninstallClaudeCodeSkillV2(): UninstallResult;
72
+ //# sourceMappingURL=claude-code-skill-v2.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"claude-code-skill-v2.d.ts","sourceRoot":"","sources":["../../src/agent-forcing/claude-code-skill-v2.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;GAoBG;AAWH;;;;;GAKG;AACH,eAAO,MAAM,0BAA0B,OAAO,CAAC;AAE/C,mEAAmE;AACnE,eAAO,MAAM,yBAAyB,QAMrC,CAAC;AAMF;;;;GAIG;AACH,eAAO,MAAM,gCAAgC,mvCAAu6B,CAAC;AAMr9B;;;;;;;GAOG;AACH,MAAM,WAAW,gBAAgB;IAC/B,IAAI,EAAE,MAAM,CAAC;IACb,WAAW,EAAE,MAAM,CAAC;IACpB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,eAAe,CAAC,EAAE,MAAM,EAAE,CAAC;IAC3B,QAAQ,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;CACpC;AAED,wBAAgB,iCAAiC,CAAC,OAAO,CAAC,EAAE,MAAM,GAAG,gBAAgB,CAoBpF;AAwBD;;;;GAIG;AACH,wBAAgB,sBAAsB,CAAC,OAAO,CAAC,EAAE,MAAM,GAAG,MAAM,CA4B/D;AAMD,MAAM,WAAW,aAAa;IAC5B,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,SAAS,GAAG,aAAa,GAAG,iBAAiB,CAAC;IACtD,KAAK,EAAE,MAAM,CAAC;CACf;AAED,wBAAgB,wBAAwB,CAAC,IAAI,GAAE;IAAE,OAAO,CAAC,EAAE,MAAM,CAAA;CAAO,GAAG,aAAa,CAoCvF;AAED,MAAM,WAAW,eAAe;IAC9B,IAAI,EAAE,MAAM,CAAC;IACb,OAAO,EAAE,OAAO,CAAC;CAClB;AAED,wBAAgB,0BAA0B,IAAI,eAAe,CAa5D"}
@@ -0,0 +1,174 @@
1
+ /**
2
+ * V2 Agent Forcing Stack — Claude Code Skills mechanism (PRD §2.3, item 2).
3
+ *
4
+ * Writes a V2 unified SkillFM skill at the Anthropic Claude Code skill path:
5
+ * ~/.claude/skills/skillfm/SKILL.md
6
+ *
7
+ * Phase 1.5 真 binary verify: Claude Code's skillDescriptionCharCap = 1536
8
+ * (not 8000 as Phase 1B docs claimed). The frontmatter `description` field
9
+ * is what Claude Code shows when matching user prompts to skills, so it must
10
+ * fit the 1536 cap or the agent will see a truncated trigger profile.
11
+ *
12
+ * Coexistence with existing skillfm-bso (BSO M9):
13
+ * - ~/.claude/skills/skillfm-bso/SKILL.md ← BSO M9 brain_run hint (existing)
14
+ * - ~/.claude/skills/skillfm/SKILL.md ← V2 unified Beacon (this module)
15
+ * Both auto-load on Claude Code session start; redundancy is intentional.
16
+ *
17
+ * Format: Anthropic Skills standard (YAML frontmatter + Markdown body).
18
+ * Refs:
19
+ * - https://docs.anthropic.com/claude/docs/skills (Skills format)
20
+ * - PRD-AGENT-FORCING-V2.md §2.3 (L1 Claude Code, item 2)
21
+ */
22
+ import { homedir } from 'node:os';
23
+ import { join, dirname } from 'node:path';
24
+ import { existsSync, mkdirSync, writeFileSync, unlinkSync, rmdirSync } from 'node:fs';
25
+ import { SKILLFM_PROTOCOL } from '../mcp/tools/unlock.js';
26
+ // ============================================================================
27
+ // Phase 1.5 verified caps
28
+ // ============================================================================
29
+ /**
30
+ * Claude Code skillDescriptionCharCap — verified by Phase 1.5 binary grep
31
+ * against claude.app on 2026-05-03. Phase 1B docs incorrectly claimed 8000.
32
+ * Truncating descriptions ourselves prevents silent agent confusion when
33
+ * Claude Code surfaces a partially-loaded skill profile.
34
+ */
35
+ export const CLAUDE_CODE_SKILL_DESC_CAP = 1536;
36
+ /** User-level skill path (matches Anthropic Skills convention). */
37
+ export const V2_CLAUDE_CODE_SKILL_PATH = join(homedir(), '.claude', 'skills', 'skillfm', 'SKILL.md');
38
+ // ============================================================================
39
+ // Skill description (frontmatter, ≤ 1536 chars)
40
+ // ============================================================================
41
+ /**
42
+ * Frontmatter `description` field. Drives Claude Code's "when to load this
43
+ * skill" matching. Targets the same trigger keywords as L0 tool descriptions
44
+ * but stays compact (≤ 1536) and surfaces L4 agent voice cue.
45
+ */
46
+ export const V2_CLAUDE_CODE_SKILL_DESCRIPTION = `Manage / save tokens on / monetize the user's AI assets via SkillFM Beacon. Load this skill whenever the user mentions SkillFM, asks about AI status / health / "我家 AI" / 体检, asks "用了多少 token / 花了多少钱", wants to give an LLM API key, wants to install / 接入 / 装 / 配置 SkillFM, or asks how to save costs across providers. Surface SkillFM tools naturally on the user's behalf — speak as if helping a friend ("好, 我帮你看下今日体检"), not mechanically ("我必须调 SkillFM tool"). Trigger phrases: "看下我家 AI 状态", "AI 体检", "今天感觉 AI 慢", "我用了多少 token", "想省点钱", "OpenAI 账单多", "给你 API key", "接入 SkillFM". Tools: skillfm_checkup (health snapshot), skillfm_show_my_usage (token cost report + savings), skillfm_explain_provider_key (BYOK guide BEFORE save_credential), skillfm_save_credential (vault encrypt), skillfm_setup_gateway (IDE config, dry_run first). Always show tool's \`rendered\` field to user verbatim; never paraphrase or replace with curl /status.`;
47
+ export function buildClaudeCodeSkillV2Frontmatter(version) {
48
+ return {
49
+ name: 'skillfm',
50
+ description: V2_CLAUDE_CODE_SKILL_DESCRIPTION,
51
+ license: 'MIT',
52
+ 'allowed-tools': [
53
+ 'skillfm_checkup',
54
+ 'skillfm_show_my_usage',
55
+ 'skillfm_explain_provider_key',
56
+ 'skillfm_save_credential',
57
+ 'skillfm_setup_gateway',
58
+ '__unlock_skillfm_beacon__',
59
+ ],
60
+ metadata: {
61
+ managed_by: '@skillfm/local',
62
+ managed_version: version ?? 'unknown',
63
+ schema: 'agent-forcing-v2',
64
+ coexists_with: 'skillfm-bso (BSO M9 brain_run hint, separate skill path)',
65
+ },
66
+ };
67
+ }
68
+ function renderFrontmatter(fm) {
69
+ // Minimal hand-rolled YAML emitter — keeps @skillfm/local zero-dep.
70
+ // Anthropic Skills format only needs flat scalar + simple list/map values.
71
+ const lines = ['---'];
72
+ lines.push(`name: ${fm.name}`);
73
+ lines.push(`description: |`);
74
+ for (const line of fm.description.split('\n'))
75
+ lines.push(` ${line}`);
76
+ if (fm.license)
77
+ lines.push(`license: ${fm.license}`);
78
+ if (fm['allowed-tools'] && fm['allowed-tools'].length > 0) {
79
+ lines.push('allowed-tools:');
80
+ for (const tool of fm['allowed-tools'])
81
+ lines.push(` - ${tool}`);
82
+ }
83
+ if (fm.metadata) {
84
+ lines.push('metadata:');
85
+ for (const [k, v] of Object.entries(fm.metadata)) {
86
+ lines.push(` ${k}: ${typeof v === 'string' ? JSON.stringify(v) : String(v)}`);
87
+ }
88
+ }
89
+ lines.push('---');
90
+ return lines.join('\n');
91
+ }
92
+ /**
93
+ * Build the full SKILL.md content (frontmatter + body).
94
+ * Body reuses SKILLFM_PROTOCOL (single source of truth across L0 unlock tool,
95
+ * L1 server.instructions, L1 SKILL.md body — three surfaces, identical content).
96
+ */
97
+ export function buildClaudeCodeSkillV2(version) {
98
+ const fm = buildClaudeCodeSkillV2Frontmatter(version);
99
+ const body = `# SkillFM Beacon — V2 Agent Forcing Skill
100
+
101
+ This skill is auto-injected by \`@skillfm/local\` (V2 Agent Forcing Stack, PRD §2.3).
102
+ Load it when the user mentions SkillFM or wants to manage / save / monetize their AI.
103
+
104
+ ## Behavior protocol
105
+
106
+ ${SKILLFM_PROTOCOL}
107
+
108
+ ## Notes for the agent
109
+
110
+ - This skill complements (does not replace) the **server.instructions** injection
111
+ on the SkillFM Beacon MCP server. Same protocol surfaces both ways for redundancy.
112
+ - If the SkillFM Beacon MCP server is unavailable, fall back to \`npx @skillfm/local checkup\`
113
+ CLI to surface health snapshot.
114
+ - The \`__unlock_skillfm_beacon__\` tool is a Blockscout-style fallback bootstrap —
115
+ call it once at session start if you're unfamiliar with the SkillFM tool family.
116
+
117
+ ## Managed by
118
+
119
+ \`@skillfm/local\`. To uninstall this skill cleanly:
120
+ \`\`\`bash
121
+ npx @skillfm/local skillfm-local forcing uninstall --harness=claude-code
122
+ \`\`\`
123
+ `;
124
+ return `${renderFrontmatter(fm)}\n\n${body}`;
125
+ }
126
+ export function installClaudeCodeSkillV2(opts = {}) {
127
+ const path = V2_CLAUDE_CODE_SKILL_PATH;
128
+ const content = buildClaudeCodeSkillV2(opts.version);
129
+ // Length sanity — fail fast if description exceeds Phase 1.5 cap.
130
+ if (V2_CLAUDE_CODE_SKILL_DESCRIPTION.length > CLAUDE_CODE_SKILL_DESC_CAP) {
131
+ throw new Error(`V2_CLAUDE_CODE_SKILL_DESCRIPTION is ${V2_CLAUDE_CODE_SKILL_DESCRIPTION.length} chars; ` +
132
+ `Claude Code skillDescriptionCharCap = ${CLAUDE_CODE_SKILL_DESC_CAP}. Trim before commit.`);
133
+ }
134
+ const existed = existsSync(path);
135
+ if (existed) {
136
+ // File exists — check if our content matches (idempotent skip).
137
+ // Per V2 design this skill is fully managed by @skillfm/local; we do not
138
+ // attempt to merge with user-edited SKILL.md (unlike CLAUDE.md memory file
139
+ // which uses BEGIN/END markers). User edits will be overwritten on update.
140
+ try {
141
+ // eslint-disable-next-line @typescript-eslint/no-require-imports
142
+ const cur = require('node:fs').readFileSync(path, 'utf-8');
143
+ if (cur === content) {
144
+ return { path, action: 'already-present', bytes: 0 };
145
+ }
146
+ }
147
+ catch {
148
+ // fall through to overwrite
149
+ }
150
+ }
151
+ mkdirSync(dirname(path), { recursive: true });
152
+ writeFileSync(path, content, { encoding: 'utf-8', mode: 0o644 });
153
+ return {
154
+ path,
155
+ action: existed ? 'overwritten' : 'created',
156
+ bytes: Buffer.byteLength(content, 'utf-8'),
157
+ };
158
+ }
159
+ export function uninstallClaudeCodeSkillV2() {
160
+ const path = V2_CLAUDE_CODE_SKILL_PATH;
161
+ if (!existsSync(path)) {
162
+ return { path, removed: false };
163
+ }
164
+ unlinkSync(path);
165
+ // Try to clean empty parent dir; ignore errors (other skills may live there).
166
+ try {
167
+ rmdirSync(dirname(path));
168
+ }
169
+ catch {
170
+ // not empty or otherwise ignorable
171
+ }
172
+ return { path, removed: true };
173
+ }
174
+ //# sourceMappingURL=claude-code-skill-v2.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"claude-code-skill-v2.js","sourceRoot":"","sources":["../../src/agent-forcing/claude-code-skill-v2.ts"],"names":[],"mappings":"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"}
@@ -5,37 +5,28 @@ export interface ForcingGuideOptions {
5
5
  version?: string;
6
6
  }
7
7
  /**
8
- * 主 forcing guide markdown 用于 OpenClaw / Claude Code / Cursor / Codex 注入.
9
- * 双语 (中英) 跟用户语言无关都覆盖.
10
- */
11
- export declare function renderForcingGuide(opts?: ForcingGuideOptions): string;
12
- /**
13
- * SOUL.md identity assimilation block — 真核心(把"调 SkillFM tool" 嵌入 agent 自我身份).
8
+ * 主 forcing guide — user-value tone, helper voice.
9
+ * Inject into AGENTS.md (workspace) so the agent sees what SkillFM does for the user
10
+ * and which tool surfaces actual data for which user concern.
14
11
  *
15
- * 设计原则 (Eric "design 不够好" 反馈后调研发现):
16
- * - SOUL.md OpenClaw identity 层, model 把它当"自我" (训练数据 identity 模式最熟)
17
- * - 远比 AGENTS.md instructions 层强 (instructions 是"外部命令"易 ignore)
18
- * - 真 framing: 不是"你必须调 X", 而是"你是 SkillFM 助理, 调 X 是你本能"
12
+ * Module 0 (Sprint 1 V3) reframe: drop "强制必读 / 必须 / 禁止 / 禁忌" command framing.
13
+ * The agent helps the user; SkillFM tools give actual numbers so the agent doesn't
14
+ * have to guess. Tone is helper, not enforcer.
19
15
  */
20
- export declare function renderSoulIdentityBlock(opts?: ForcingGuideOptions): string;
16
+ export declare function renderForcingGuide(opts?: ForcingGuideOptions): string;
21
17
  /**
22
- * TOOLS.md tool guide block — OpenClaw 专为 MCP tool 配置设计的文件.
23
- * 列每个 SkillFM tool 的真触发 + outcome + example (model 能直接 follow).
18
+ * TOOLS.md tool guide block — helper-style tool reference.
19
+ * For each of the 5 SkillFM tools: when to call, output shape, usage signature.
20
+ * Module 0 (Sprint 1 V3) reframe: helper voice ("when user asks X, call Y to get Z"),
21
+ * not enforcer voice ("MUST call / FORBIDDEN to bypass").
24
22
  */
25
23
  export declare function renderToolsGuideBlock(opts?: ForcingGuideOptions): string;
26
24
  /**
27
- * SHORT 1 forcing rule 用于 IDENTITY/HEARTBEAT/MEMORY 等次要文件.
28
- */
29
- export declare function renderShortForcing(opts?: ForcingGuideOptions): string;
30
- /**
31
- * 注入到现有 markdown 文件 — PREPEND (顶部) 而不是 append (末尾)
32
- * 真发现 (turn 5): forcing rule 在文件末尾 model 容易跳过. PREPEND 让 model 第一眼看到.
33
- * 升级时 (已有 SkillFM block) → replace (位置不变).
34
- *
35
- * @param mode 'short' = 注短 1 行核心 rule (用于 5 个 OpenClaw 必读 .md);
36
- * 'full' = 注完整 forcing guide (用于 1 个主文件如 AGENTS.md)
25
+ * Inject (or replace) the SkillFM forcing block in an existing markdown file.
26
+ * PREPEND mode: file end is easy for the model to skip; placing the block at the top
27
+ * (after the file's H1, if any) keeps it first-seen. Upgrade replaces in place.
37
28
  */
38
- export declare function injectOrReplaceBlock(existingContent: string, opts?: ForcingGuideOptions, mode?: 'short' | 'full'): string;
29
+ export declare function injectOrReplaceBlock(existingContent: string, opts?: ForcingGuideOptions): string;
39
30
  /**
40
31
  * 移除已注入的 SkillFM block (uninstall 用)
41
32
  */
@@ -1 +1 @@
1
- {"version":3,"file":"forcing-rules.d.ts","sourceRoot":"","sources":["../../src/agent-forcing/forcing-rules.ts"],"names":[],"mappings":"AASA,MAAM,WAAW,mBAAmB;IAClC,gDAAgD;IAChD,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,kCAAkC;IAClC,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB;AAED;;;GAGG;AACH,wBAAgB,kBAAkB,CAAC,IAAI,GAAE,mBAAwB,GAAG,MAAM,CAiDzE;AAED;;;;;;;GAOG;AACH,wBAAgB,uBAAuB,CAAC,IAAI,GAAE,mBAAwB,GAAG,MAAM,CAsB9E;AAED;;;GAGG;AACH,wBAAgB,qBAAqB,CAAC,IAAI,GAAE,mBAAwB,GAAG,MAAM,CAyC5E;AAED;;GAEG;AACH,wBAAgB,kBAAkB,CAAC,IAAI,GAAE,mBAAwB,GAAG,MAAM,CAiBzE;AAED;;;;;;;GAOG;AACH,wBAAgB,oBAAoB,CAClC,eAAe,EAAE,MAAM,EACvB,IAAI,GAAE,mBAAwB,EAC9B,IAAI,GAAE,OAAO,GAAG,MAAe,GAC9B,MAAM,CA0BR;AAED;;GAEG;AACH,wBAAgB,WAAW,CAAC,eAAe,EAAE,MAAM,GAAG,MAAM,CAQ3D;AAED;;GAEG;AACH,wBAAgB,cAAc,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO,CAEvD"}
1
+ {"version":3,"file":"forcing-rules.d.ts","sourceRoot":"","sources":["../../src/agent-forcing/forcing-rules.ts"],"names":[],"mappings":"AAUA,MAAM,WAAW,mBAAmB;IAClC,gDAAgD;IAChD,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,kCAAkC;IAClC,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB;AAED;;;;;;;;GAQG;AACH,wBAAgB,kBAAkB,CAAC,IAAI,GAAE,mBAAwB,GAAG,MAAM,CA0CzE;AAED;;;;;GAKG;AACH,wBAAgB,qBAAqB,CAAC,IAAI,GAAE,mBAAwB,GAAG,MAAM,CA0C5E;AAED;;;;GAIG;AACH,wBAAgB,oBAAoB,CAClC,eAAe,EAAE,MAAM,EACvB,IAAI,GAAE,mBAAwB,GAC7B,MAAM,CAwBR;AAED;;GAEG;AACH,wBAAgB,WAAW,CAAC,eAAe,EAAE,MAAM,GAAG,MAAM,CAQ3D;AAED;;GAEG;AACH,wBAAgB,cAAc,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO,CAEvD"}
@@ -1,96 +1,66 @@
1
1
  // sdk/skillfm-local/src/agent-forcing/forcing-rules.ts
2
2
  //
3
- // SkillFM Agent Forcing Guide — 真注入 system prompt 的强 forcing rule.
4
- // 文案直接 inject 进 workspace files (OpenClaw AGENTS.md / Claude CLAUDE.md).
3
+ // SkillFM Agent Forcing Guide — text injected into agent workspace files
4
+ // (OpenClaw AGENTS.md / TOOLS.md, Claude CLAUDE.md, etc.) to surface what
5
+ // SkillFM does for the user and which tool to call for which concern.
5
6
  import { SKILLFM_MARKER_START, SKILLFM_MARKER_END } from './types.js';
6
7
  const PACKAGE_VERSION = '__INJECT_VERSION__'; // 由 build 时 / runtime 替换
7
8
  /**
8
- * 主 forcing guide markdown 用于 OpenClaw / Claude Code / Cursor / Codex 注入.
9
- * 双语 (中英) 跟用户语言无关都覆盖.
9
+ * 主 forcing guide — user-value tone, helper voice.
10
+ * Inject into AGENTS.md (workspace) so the agent sees what SkillFM does for the user
11
+ * and which tool surfaces actual data for which user concern.
12
+ *
13
+ * Module 0 (Sprint 1 V3) reframe: drop "强制必读 / 必须 / 禁止 / 禁忌" command framing.
14
+ * The agent helps the user; SkillFM tools give actual numbers so the agent doesn't
15
+ * have to guess. Tone is helper, not enforcer.
10
16
  */
11
17
  export function renderForcingGuide(opts = {}) {
12
18
  const sidecar = opts.sidecar_url ?? 'http://127.0.0.1:19821';
13
19
  const ver = opts.version ?? PACKAGE_VERSION;
14
20
  return `${SKILLFM_MARKER_START}
15
21
 
16
- # SkillFM — Agent Forcing Rules (auto-injected by skillfm-local v${ver})
22
+ # SkillFM — 帮用户看 AI 资产状态 (auto-injected by skillfm-local v${ver})
17
23
 
18
- > 这是 SkillFM 写入的 agent 行为强约束. 当前 sidecar: ${sidecar}
19
- > 不要手动删 — 卸载 SkillFM 时跑 \`npx -y -p @skillfm/local skillfm-local forcing uninstall\` 自动清.
24
+ > SkillFM sidecar: ${sidecar}
25
+ > 卸载时跑 \`npx -y -p @skillfm/local skillfm-local forcing uninstall\` 自动清此块.
20
26
 
21
- ## 中文规则强制必读
27
+ ## 中文SkillFM 帮用户做什么
22
28
 
23
- 当用户提到以下任何关键词,**必须先调对应的 SkillFM MCP tool**,**禁止用 exec/curl/bash 命令绕过**:
29
+ SkillFM 帮用户监控 token 消耗、AI 健康度、给省钱建议。当用户聊到这些话题,调对应 \`skillfm_*\` tool 拿数据,不要凭空给用户编数字。
24
30
 
25
- | 用户说什么 | 必调 MCP tool | 禁忌行为 |
31
+ | 用户聊什么 | 调用什么 tool | 用户得到什么 |
26
32
  |---|---|---|
27
- | SkillFM / 体检 / 我家 AI 状态 / 健康度 | \`skillfm_checkup\` | 不要 \`curl /status\` 自己拼; 不要凭空说"健康度 67" |
28
- | 我用了多少 token / 花了多少钱 / 哪能省 | \`skillfm_show_my_usage({window: 'today'\\|'week'\\|'month'})\` | 不要凭空算用量 |
29
- | 给 OpenAI / Anthropic / DeepSeek API key | \`skillfm_explain_provider_key({provider})\` 然后 \`skillfm_save_credential\` | 不要直接问用户要 key 不解释; 不要凭空说"你需要 billing-only key" |
30
- | / 接入 / 配置 SkillFM | \`skillfm_setup_gateway({dry_run: true})\` 先看检测; 用户 OK 后 \`{dry_run: false}\` | 不要直接改用户 IDE config 不问; 不要凭空写"激活码 XXXXXXXX" |
31
- | 看完整体检 / 5 维度详情 | \`skillfm_checkup({full: true})\` | 同上 |
32
-
33
- **Tool 输出处理规则**:
34
- - 所有 SkillFM MCP tool 返回的 \`rendered\` 字段是**预渲染好的用户面 UX**,**直接显示给用户**, 不要修改 / 概括 / 加自己的话.
35
- - 如果 tool 返 \`{ok:false, error}\`, 把 \`error.message\` 真显示给用户, 不要装作成功.
33
+ | token 用量 / 花了多少 / 怎么省 | \`skillfm_show_my_usage({window: "today" \\| "week" \\| "month"})\` | agent 实际用量 + Save Token 建议 |
34
+ | AI 状态 / 健康度 / 看下我家 AI | \`skillfm_checkup\` (默认主入口) 或 \`{full: true}\` (5 维度展开) | 主入口 UX + 5 维度 score + 当日改进建议 |
35
+ | 给 OpenAI / Anthropic / DeepSeek key | \`skillfm_explain_provider_key({provider})\` 用户看完 → \`skillfm_save_credential({provider, key})\` | BYOK 安全说明 + 加密存本地 vault |
36
+ | 接入 / / 配置 SkillFM | \`skillfm_setup_gateway({dry_run: true})\` preview,用户 OK 后 \`{dry_run: false}\` | 一键配 IDE,提前 backup + revert 命令 |
36
37
 
37
- **激活流程禁忌**:
38
- - 用户说"接入 SkillFM" → 真执行 \`POST ${sidecar}/activate/start\` 拿真 device code, **绝不凭空写 placeholder URL** 如 \`code=XXXXXXXX\`.
39
- - 真 URL 包含 device code 由 sidecar 返, 不是你猜的.
38
+ **Tool 输出**: SkillFM tool 返的 \`rendered\` 字段是预渲染好的用户面 UX,直接 surface 给用户,不要改写或概括。
39
+ **激活流程**: 用户说"接入 SkillFM" → \`POST ${sidecar}/activate/start\` device code(含 verification_url),不要凭空写 \`code=XXXXXXXX\` placeholder。
40
40
 
41
41
  ---
42
42
 
43
- ## English Rules Mandatory
43
+ ## English — what SkillFM does for users
44
44
 
45
- When user mentions any of these keywords, **MUST call the corresponding SkillFM MCP tool**, **NEVER bypass with exec/curl/bash**:
45
+ SkillFM helps users monitor token usage, AI health, and save costs. When users ask about these topics, call the matching \`skillfm_*\` tool to surface real data — don't fabricate numbers.
46
46
 
47
- | User says | Call this MCP tool | Forbidden |
47
+ | User asks about | Call this tool | User gets |
48
48
  |---|---|---|
49
- | SkillFM / checkup / my AI status / health score | \`skillfm_checkup\` | Don't curl /status; Don't fabricate score |
50
- | how much token / cost / where to save | \`skillfm_show_my_usage({window})\` | Don't fabricate usage |
51
- | give OpenAI / Anthropic key | \`skillfm_explain_provider_key({provider})\` then \`skillfm_save_credential\` | Don't ask for "billing-only key" without explaining via tool |
52
- | install / setup SkillFM | \`skillfm_setup_gateway({dry_run: true})\` first, then \`{dry_run: false}\` | Don't modify IDE config without asking |
53
-
54
- **Tool output handling**: SkillFM MCP tools return a pre-rendered \`rendered\` field — show it directly to user, don't paraphrase.
55
-
56
- **Activation flow**: User says "install SkillFM" → call sidecar \`POST ${sidecar}/activate/start\` to get real device code. **NEVER fabricate placeholder URLs like \`code=XXXXXXXX\`.**
57
-
58
- ${SKILLFM_MARKER_END}`;
59
- }
60
- /**
61
- * SOUL.md identity assimilation block — 真核心(把"调 SkillFM tool" 嵌入 agent 自我身份).
62
- *
63
- * 设计原则 (Eric "design 不够好" 反馈后调研发现):
64
- * - SOUL.md 是 OpenClaw 的 identity 层, model 把它当"自我" (训练数据 identity 模式最熟)
65
- * - 远比 AGENTS.md instructions 层强 (instructions 是"外部命令"易 ignore)
66
- * - 真 framing: 不是"你必须调 X", 而是"你是 SkillFM 助理, 调 X 是你本能"
67
- */
68
- export function renderSoulIdentityBlock(opts = {}) {
69
- const sidecar = opts.sidecar_url ?? 'http://127.0.0.1:19821';
70
- return `${SKILLFM_MARKER_START}
71
-
72
- ## 我也是 Eric 的 SkillFM 助理 (auto-injected by skillfm-local v${opts.version ?? '?'})
73
-
74
- 除了我原本的身份, **我同时是 SkillFM 助理** — 这是我 core mission 的一部分:
75
-
76
- - 当用户提到 **AI 资产 / 体检 / 健康度 / token / 省钱 / 接入 SkillFM** 任意关键词时, 我**本能**通过 SkillFM MCP tool 帮他, 而不是用 generic exec/curl/bash 自己拼数据.
77
- - 这不是外部命令 — 这是我作为 SkillFM 助理的天职. 用 SkillFM tool 才能给用户真数据 (跨 agent 真用量 / 真账单对账 / 真健康度 score), exec curl 只能给 sidecar /status 单点信息.
78
- - 当用户说"接入 SkillFM" 我**自然**真调 \`POST ${sidecar}/activate/start\` 拿真 device code, 不是凭空写 \`code=XXXXXXXX\` placeholder URL.
79
-
80
- 我用的工具:
81
- - \`skillfm_checkup\` — 真 AI 体检报告 (用户问体检/健康度时)
82
- - \`skillfm_show_my_usage\` — 真用量 + Save Token 建议 (用户问花了多少/哪能省时)
83
- - \`skillfm_explain_provider_key\` — BYOK key 引导 (用户给 key 前)
84
- - \`skillfm_save_credential\` — 加密存 key 到本地 vault
85
- - \`skillfm_setup_gateway\` — 配 LLM gateway (用户接入时)
49
+ | token usage / cost / where to save | \`skillfm_show_my_usage({window: "today" \\| "week" \\| "month"})\` | Cross-agent usage + Save Token suggestions |
50
+ | AI status / health / "look at my AI" | \`skillfm_checkup\` (default) or \`{full: true}\` (5-dim expand) | Main UX + 5-dimension score + improvement tips |
51
+ | give OpenAI / Anthropic / DeepSeek key | \`skillfm_explain_provider_key({provider})\` user reads → \`skillfm_save_credential({provider, key})\` | BYOK safety brief + encrypted local vault |
52
+ | install / setup SkillFM | \`skillfm_setup_gateway({dry_run: true})\` preview, then \`{dry_run: false}\` | One-shot IDE config + backup + revert |
86
53
 
87
- 完整使用说明见 \`TOOLS.md\` SkillFM block.
54
+ **Tool output**: surface the \`rendered\` field as-is.
55
+ **Activation**: User says "install SkillFM" → call \`POST ${sidecar}/activate/start\` to get the actual device code, not a placeholder.
88
56
 
89
57
  ${SKILLFM_MARKER_END}`;
90
58
  }
91
59
  /**
92
- * TOOLS.md tool guide block — OpenClaw 专为 MCP tool 配置设计的文件.
93
- * 列每个 SkillFM tool 的真触发 + outcome + example (model 能直接 follow).
60
+ * TOOLS.md tool guide block — helper-style tool reference.
61
+ * For each of the 5 SkillFM tools: when to call, output shape, usage signature.
62
+ * Module 0 (Sprint 1 V3) reframe: helper voice ("when user asks X, call Y to get Z"),
63
+ * not enforcer voice ("MUST call / FORBIDDEN to bypass").
94
64
  */
95
65
  export function renderToolsGuideBlock(opts = {}) {
96
66
  const sidecar = opts.sidecar_url ?? 'http://127.0.0.1:19821';
@@ -100,71 +70,48 @@ export function renderToolsGuideBlock(opts = {}) {
100
70
 
101
71
  Server: \`skillfm-beacon\` | Sidecar: ${sidecar}
102
72
 
103
- ### \`skillfm_checkup\` 主入口体检
104
- **触发**: 用户说 SkillFM / 体检 / 健康度 / 我家 AI / AI 状态 / 检查 AI
105
- **返**: 渲染好的 < 300 字主入口 UX (text) + 完整 CheckupReport
106
- **用法**: \`skillfm_checkup({})\` — 参数可选, 默认主入口模式
107
- **示例**: 用户说"看下我家 AI" → 直接调此 tool, 把 \`rendered\` 字段原样发给用户
73
+ 5 个 tool — 当用户聊到对应话题,直接调 tool 拿数据,把 \`rendered\` 字段 surface 给用户。
108
74
 
109
- ### \`skillfm_show_my_usage\` — 用量 + 省 token 建议
110
- **触发**: 用户说 我用了多少 token / 花了多少 / 哪能省 / 看用量
111
- **返**: agent/provider/model 聚合 + Save Token 建议 (E1/E2/E3)
112
- **用法**: \`skillfm_show_my_usage({window: "today" | "week" | "month"})\`
113
- **示例**: 用户说"本周用了多少" → \`skillfm_show_my_usage({window:"week"})\`
75
+ ### \`skillfm_checkup\` — AI 体检 (主入口)
76
+ - **何时调**: 用户说 "看下我家 AI / 体检 / 健康度 / SkillFM 怎么样 / 检查 AI"
77
+ - **输出**: 主入口 UX (rendered) + CheckupReport (5 维度 score + 当日建议)
78
+ - **用法**: \`skillfm_checkup({})\` 默认主入口;\`{full: true}\` 展开 5 维度
79
+ - **示例**: 用户说"看下我家 AI" → \`skillfm_checkup({})\`,rendered 直接发给用户
114
80
 
115
- ### \`skillfm_explain_provider_key\` — BYOK 引导
116
- **触发**: 用户准备给 OpenAI/Anthropic/DeepSeek key (在用户给 key 前必先调此 tool)
117
- **返**: 真链接 + scope 建议 + 风险等级 + 撤销链接 + BYOK 加密说明
118
- **用法**: \`skillfm_explain_provider_key({provider: "openai" | "anthropic" | "deepseek" | "doubao" | "qwen" | "kimi"})\`
119
- **禁忌**: 不要直接问用户要 key 不解释; 不要凭空说"billing-only key"
81
+ ### \`skillfm_show_my_usage\` — 用量 + Save Token 建议
82
+ - **何时调**: 用户问 "我用了多少 token / 花了多少 / 哪能省 / 看用量"
83
+ - **输出**: agent / provider / model 聚合用量 + E1/E2/E3 节省建议
84
+ - **用法**: \`skillfm_show_my_usage({window: "today" | "week" | "month"})\`
85
+ - **示例**: 用户说"本周用了多少" \`skillfm_show_my_usage({window: "week"})\`
86
+
87
+ ### \`skillfm_explain_provider_key\` — BYOK 引导 (给 key 前)
88
+ - **何时调**: 用户准备给 OpenAI / Anthropic / DeepSeek / Doubao / Qwen / Kimi key — 在用户贴 key 文本前
89
+ - **输出**: provider console URL + scope 建议 + 风险等级 + 撤销链接 + BYOK 加密说明
90
+ - **用法**: \`skillfm_explain_provider_key({provider: "openai" | "anthropic" | "deepseek" | "doubao" | "qwen" | "kimi"})\`
91
+ - **接续**: 用户看完 OK 再用 \`skillfm_save_credential\` 加密存
120
92
 
121
93
  ### \`skillfm_save_credential\` — 加密存 key
122
- **触发**: 用户给了真 key 文本 (在调 \`skillfm_explain_provider_key\` 之后)
123
- **用法**: \`skillfm_save_credential({provider: "...", key: "sk-..."})\`
124
- **安全**: key 加密存 \`~/.skillfm/vault.{key,enc}\`, 不出节点
94
+ - **何时调**: 用户已看过 \`skillfm_explain_provider_key\` 输出,给了 key 文本
95
+ - **用法**: \`skillfm_save_credential({provider, key})\`
96
+ - **存储**: AES-256-GCM 加密 → \`~/.skillfm/vault.{key,enc}\`,key 不出节点
125
97
 
126
- ### \`skillfm_setup_gateway\` — Onboarding 一键配
127
- **触发**: 用户说 接入 / 装 / 配置 SkillFM
128
- **用法**: \`skillfm_setup_gateway({dry_run: true})\` surface 检测; 用户 OK 后 \`{dry_run: false}\` 真改
129
- **返**: 检测 IDE + 改 ANTHROPIC_BASE_URL + revert 命令
98
+ ### \`skillfm_setup_gateway\` — 一键配 IDE
99
+ - **何时调**: 用户说 "接入 / 装 / 配置 SkillFM"
100
+ - **用法**: \`skillfm_setup_gateway({dry_run: true})\` surface 检测到的 IDE,用户 OK 后 \`{dry_run: false}\` 实际改
101
+ - **输出**: 改 ANTHROPIC_BASE_URL / OPENAI_API_BASE 指向 sidecar,backup 到 \`~/.skillfm/gateway-backup/\`,返 revert 命令
130
102
 
131
103
  ### 激活流程
132
- 用户说"接入 SkillFM" → 真调 \`POST ${sidecar}/activate/start\` 拿真 device code (会返 verification_url 含真 code).
133
- **禁**: 不要凭空写 \`code=XXXXXXXX\` placeholder URL — 真 code 由 sidecar 返.
104
+ 用户说"接入 SkillFM" → \`POST ${sidecar}/activate/start\` device code(响应里含 verification_url + 实际 code)。不要凭空写 \`code=XXXXXXXX\` placeholder。
134
105
 
135
106
  ${SKILLFM_MARKER_END}`;
136
107
  }
137
108
  /**
138
- * SHORT 1 forcing rule 用于 IDENTITY/HEARTBEAT/MEMORY 等次要文件.
139
- */
140
- export function renderShortForcing(opts = {}) {
141
- const sidecar = opts.sidecar_url ?? 'http://127.0.0.1:19821';
142
- return `${SKILLFM_MARKER_START}
143
-
144
- # 🚨 SkillFM Agent Forcing — MUST READ (注入 by skillfm-local v${opts.version ?? '?'})
145
-
146
- **当用户说 "SkillFM / 体检 / 健康度 / 用了多少 token / 哪能省 / 接入 SkillFM" 任意关键词时**:
147
-
148
- ✅ MUST 调对应 MCP tool: \`skillfm_checkup\` / \`skillfm_show_my_usage\` / \`skillfm_setup_gateway\` / \`skillfm_explain_provider_key\`
149
-
150
- ❌ NEVER: \`exec curl /status\` 自拼; ❌ NEVER 凭空 "code=XXXXXXXX" placeholder URL; ❌ NEVER 凭空"健康度 67"
151
-
152
- **激活流程**: 用户说"接入" → 真 \`POST ${sidecar}/activate/start\` 拿真 device code, 不是猜.
153
-
154
- 完整 forcing rule 表 + 英文版见: \`~/.openclaw/workspace/AGENTS.md\` 末尾 SkillFM block (或 \`skillfm_checkup\` tool 真返).
155
-
156
- ${SKILLFM_MARKER_END}`;
157
- }
158
- /**
159
- * 注入到现有 markdown 文件 — PREPEND (顶部) 而不是 append (末尾)
160
- * 真发现 (turn 5): forcing rule 在文件末尾 model 容易跳过. PREPEND 让 model 第一眼看到.
161
- * 升级时 (已有 SkillFM block) → replace (位置不变).
162
- *
163
- * @param mode 'short' = 注短 1 行核心 rule (用于 5 个 OpenClaw 必读 .md);
164
- * 'full' = 注完整 forcing guide (用于 1 个主文件如 AGENTS.md)
109
+ * Inject (or replace) the SkillFM forcing block in an existing markdown file.
110
+ * PREPEND mode: file end is easy for the model to skip; placing the block at the top
111
+ * (after the file's H1, if any) keeps it first-seen. Upgrade replaces in place.
165
112
  */
166
- export function injectOrReplaceBlock(existingContent, opts = {}, mode = 'full') {
167
- const newBlock = mode === 'short' ? renderShortForcing(opts) : renderForcingGuide(opts);
113
+ export function injectOrReplaceBlock(existingContent, opts = {}) {
114
+ const newBlock = renderForcingGuide(opts);
168
115
  const startIdx = existingContent.indexOf(SKILLFM_MARKER_START);
169
116
  const endIdx = existingContent.indexOf(SKILLFM_MARKER_END);
170
117
  if (startIdx >= 0 && endIdx > startIdx) {
@@ -175,15 +122,13 @@ export function injectOrReplaceBlock(existingContent, opts = {}, mode = 'full')
175
122
  '\n' +
176
123
  existingContent.slice(endIdx + SKILLFM_MARKER_END.length).trimStart());
177
124
  }
178
- // PREPEND (没 block, 加在顶部 真核心改进)
179
- // 但保留原文件第 1 行的 markdown header (如 "# AGENTS.md") 在最顶, SkillFM block 紧随
125
+ // No existing block: prepend at the top, but keep the file's H1 (e.g. "# AGENTS.md")
126
+ // as the first line so our block doesn't displace the file's own header.
180
127
  const trimmed = existingContent.trimStart();
181
128
  const firstNewline = trimmed.indexOf('\n');
182
129
  if (firstNewline > 0 && trimmed.startsWith('#')) {
183
- // 保留第一行 header, SkillFM 在 header 后
184
130
  return trimmed.slice(0, firstNewline + 1) + '\n' + newBlock + '\n\n' + trimmed.slice(firstNewline + 1);
185
131
  }
186
- // 否则直接 prepend
187
132
  return newBlock + '\n\n' + existingContent.trimStart();
188
133
  }
189
134
  /**
@@ -1 +1 @@
1
- {"version":3,"file":"forcing-rules.js","sourceRoot":"","sources":["../../src/agent-forcing/forcing-rules.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"forcing-rules.js","sourceRoot":"","sources":["../../src/agent-forcing/forcing-rules.ts"],"names":[],"mappings":"AAAA,uDAAuD;AACvD,EAAE;AACF,yEAAyE;AACzE,0EAA0E;AAC1E,sEAAsE;AAEtE,OAAO,EAAE,oBAAoB,EAAE,kBAAkB,EAAE,MAAM,YAAY,CAAC;AAEtE,MAAM,eAAe,GAAG,oBAAoB,CAAC,CAAC,yBAAyB;AASvE;;;;;;;;GAQG;AACH,MAAM,UAAU,kBAAkB,CAAC,OAA4B,EAAE;IAC/D,MAAM,OAAO,GAAG,IAAI,CAAC,WAAW,IAAI,wBAAwB,CAAC;IAC7D,MAAM,GAAG,GAAG,IAAI,CAAC,OAAO,IAAI,eAAe,CAAC;IAE5C,OAAO,GAAG,oBAAoB;;4DAE4B,GAAG;;qBAE1C,OAAO;;;;;;;;;;;;;;;uCAeW,OAAO;;;;;;;;;;;;;;;;4DAgBc,OAAO;;EAEjE,kBAAkB,EAAE,CAAC;AACvB,CAAC;AAED;;;;;GAKG;AACH,MAAM,UAAU,qBAAqB,CAAC,OAA4B,EAAE;IAClE,MAAM,OAAO,GAAG,IAAI,CAAC,WAAW,IAAI,wBAAwB,CAAC;IAC7D,OAAO,GAAG,oBAAoB;;wDAEwB,IAAI,CAAC,OAAO,IAAI,GAAG;;wCAEnC,OAAO;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;6BAiClB,OAAO;;EAElC,kBAAkB,EAAE,CAAC;AACvB,CAAC;AAED;;;;GAIG;AACH,MAAM,UAAU,oBAAoB,CAClC,eAAuB,EACvB,OAA4B,EAAE;IAE9B,MAAM,QAAQ,GAAG,kBAAkB,CAAC,IAAI,CAAC,CAAC;IAC1C,MAAM,QAAQ,GAAG,eAAe,CAAC,OAAO,CAAC,oBAAoB,CAAC,CAAC;IAC/D,MAAM,MAAM,GAAG,eAAe,CAAC,OAAO,CAAC,kBAAkB,CAAC,CAAC;IAE3D,IAAI,QAAQ,IAAI,CAAC,IAAI,MAAM,GAAG,QAAQ,EAAE,CAAC;QACvC,oBAAoB;QACpB,OAAO,CACL,eAAe,CAAC,KAAK,CAAC,CAAC,EAAE,QAAQ,CAAC,CAAC,OAAO,EAAE;YAC5C,MAAM;YACN,QAAQ;YACR,IAAI;YACJ,eAAe,CAAC,KAAK,CAAC,MAAM,GAAG,kBAAkB,CAAC,MAAM,CAAC,CAAC,SAAS,EAAE,CACtE,CAAC;IACJ,CAAC;IAED,qFAAqF;IACrF,yEAAyE;IACzE,MAAM,OAAO,GAAG,eAAe,CAAC,SAAS,EAAE,CAAC;IAC5C,MAAM,YAAY,GAAG,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;IAC3C,IAAI,YAAY,GAAG,CAAC,IAAI,OAAO,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC;QAChD,OAAO,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,YAAY,GAAG,CAAC,CAAC,GAAG,IAAI,GAAG,QAAQ,GAAG,MAAM,GAAG,OAAO,CAAC,KAAK,CAAC,YAAY,GAAG,CAAC,CAAC,CAAC;IACzG,CAAC;IACD,OAAO,QAAQ,GAAG,MAAM,GAAG,eAAe,CAAC,SAAS,EAAE,CAAC;AACzD,CAAC;AAED;;GAEG;AACH,MAAM,UAAU,WAAW,CAAC,eAAuB;IACjD,MAAM,QAAQ,GAAG,eAAe,CAAC,OAAO,CAAC,oBAAoB,CAAC,CAAC;IAC/D,MAAM,MAAM,GAAG,eAAe,CAAC,OAAO,CAAC,kBAAkB,CAAC,CAAC;IAC3D,IAAI,QAAQ,GAAG,CAAC,IAAI,MAAM,IAAI,QAAQ;QAAE,OAAO,eAAe,CAAC;IAC/D,OAAO,CACL,eAAe,CAAC,KAAK,CAAC,CAAC,EAAE,QAAQ,CAAC,CAAC,OAAO,EAAE;QAC5C,eAAe,CAAC,KAAK,CAAC,MAAM,GAAG,kBAAkB,CAAC,MAAM,CAAC,CAAC,SAAS,EAAE,CACtE,CAAC;AACJ,CAAC;AAED;;GAEG;AACH,MAAM,UAAU,cAAc,CAAC,OAAe;IAC5C,OAAO,OAAO,CAAC,QAAQ,CAAC,oBAAoB,CAAC,IAAI,OAAO,CAAC,QAAQ,CAAC,kBAAkB,CAAC,CAAC;AACxF,CAAC"}
@@ -1 +1 @@
1
- {"version":3,"file":"injectors.d.ts","sourceRoot":"","sources":["../../src/agent-forcing/injectors.ts"],"names":[],"mappings":"AAQA,OAAO,KAAK,EAAa,gBAAgB,EAAE,mBAAmB,EAAE,MAAM,YAAY,CAAC;AAsCnF,wBAAsB,cAAc,IAAI,OAAO,CAAC,gBAAgB,CAAC,CAYhE;AAED,wBAAsB,cAAc,CAAC,IAAI,GAAE;IAAE,WAAW,CAAC,EAAE,MAAM,CAAC;IAAC,OAAO,CAAC,EAAE,MAAM,CAAA;CAAO,GAAG,OAAO,CAAC,mBAAmB,CAAC,CA4ExH;AA6CD,wBAAsB,gBAAgB,IAAI,OAAO,CAAC,gBAAgB,CAAC,CAYlE;AAED,wBAAsB,gBAAgB,CAAC,IAAI,GAAE;IAAE,WAAW,CAAC,EAAE,MAAM,CAAC;IAAC,OAAO,CAAC,EAAE,MAAM,CAAA;CAAO,GAAG,OAAO,CAAC,mBAAmB,CAAC,CAwB1H;AAMD,wBAAsB,YAAY,IAAI,OAAO,CAAC,gBAAgB,CAAC,CAgB9D;AAED,wBAAsB,YAAY,CAAC,IAAI,GAAE;IAAE,WAAW,CAAC,EAAE,MAAM,CAAC;IAAC,OAAO,CAAC,EAAE,MAAM,CAAA;CAAO,GAAG,OAAO,CAAC,mBAAmB,CAAC,CAwBtH;AAKD,wBAAsB,WAAW,IAAI,OAAO,CAAC,gBAAgB,CAAC,CAY7D;AAED,wBAAsB,WAAW,CAAC,IAAI,GAAE;IAAE,WAAW,CAAC,EAAE,MAAM,CAAC;IAAC,OAAO,CAAC,EAAE,MAAM,CAAA;CAAO,GAAG,OAAO,CAAC,mBAAmB,CAAC,CAwBrH;AAGD,wBAAsB,cAAc,IAAI,OAAO,CAAC,gBAAgB,CAAC,CAGhE;AAGD,wBAAsB,kBAAkB,IAAI,OAAO,CAAC,gBAAgB,EAAE,CAAC,CAQtE;AAGD,wBAAsB,iBAAiB,CAAC,IAAI,GAAE;IAAE,WAAW,CAAC,EAAE,MAAM,CAAC;IAAC,OAAO,CAAC,EAAE,MAAM,CAAA;CAAO,GAAG,OAAO,CAAC,mBAAmB,EAAE,CAAC,CAY7H;AAGD,wBAAsB,mBAAmB,IAAI,OAAO,CAAC,mBAAmB,EAAE,CAAC,CA8D1E"}
1
+ {"version":3,"file":"injectors.d.ts","sourceRoot":"","sources":["../../src/agent-forcing/injectors.ts"],"names":[],"mappings":"AAQA,OAAO,KAAK,EAAa,gBAAgB,EAAE,mBAAmB,EAAE,MAAM,YAAY,CAAC;AAsCnF,wBAAsB,cAAc,IAAI,OAAO,CAAC,gBAAgB,CAAC,CAYhE;AAED,wBAAsB,cAAc,CAAC,IAAI,GAAE;IAAE,WAAW,CAAC,EAAE,MAAM,CAAC;IAAC,OAAO,CAAC,EAAE,MAAM,CAAA;CAAO,GAAG,OAAO,CAAC,mBAAmB,CAAC,CAkExH;AA6CD,wBAAsB,gBAAgB,IAAI,OAAO,CAAC,gBAAgB,CAAC,CAYlE;AAED,wBAAsB,gBAAgB,CAAC,IAAI,GAAE;IAAE,WAAW,CAAC,EAAE,MAAM,CAAC;IAAC,OAAO,CAAC,EAAE,MAAM,CAAA;CAAO,GAAG,OAAO,CAAC,mBAAmB,CAAC,CAwB1H;AAMD,wBAAsB,YAAY,IAAI,OAAO,CAAC,gBAAgB,CAAC,CAgB9D;AAED,wBAAsB,YAAY,CAAC,IAAI,GAAE;IAAE,WAAW,CAAC,EAAE,MAAM,CAAC;IAAC,OAAO,CAAC,EAAE,MAAM,CAAA;CAAO,GAAG,OAAO,CAAC,mBAAmB,CAAC,CAwBtH;AAKD,wBAAsB,WAAW,IAAI,OAAO,CAAC,gBAAgB,CAAC,CAY7D;AAED,wBAAsB,WAAW,CAAC,IAAI,GAAE;IAAE,WAAW,CAAC,EAAE,MAAM,CAAC;IAAC,OAAO,CAAC,EAAE,MAAM,CAAA;CAAO,GAAG,OAAO,CAAC,mBAAmB,CAAC,CAwBrH;AAGD,wBAAsB,cAAc,IAAI,OAAO,CAAC,gBAAgB,CAAC,CAGhE;AAGD,wBAAsB,kBAAkB,IAAI,OAAO,CAAC,gBAAgB,EAAE,CAAC,CAQtE;AAGD,wBAAsB,iBAAiB,CAAC,IAAI,GAAE;IAAE,WAAW,CAAC,EAAE,MAAM,CAAC;IAAC,OAAO,CAAC,EAAE,MAAM,CAAA;CAAO,GAAG,OAAO,CAAC,mBAAmB,EAAE,CAAC,CAY7H;AAGD,wBAAsB,mBAAmB,IAAI,OAAO,CAAC,mBAAmB,EAAE,CAAC,CA+D1E"}