frappe-builder 1.1.0-dev.24 → 1.1.0-dev.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/.fb/state.db CHANGED
Binary file
@@ -2,13 +2,13 @@ feature_id: po-approval
2
2
  feature_name: "PO Approval"
3
3
  mode: full
4
4
  phase: testing
5
- updated_at: 2026-03-28T14:23:24.878Z
5
+ updated_at: 2026-03-28T14:37:54.966Z
6
6
 
7
7
  components:
8
8
  - id: final-comp
9
9
  sort_order: 0
10
10
  status: complete
11
- completed_at: 2026-03-28T14:23:24.878Z
11
+ completed_at: 2026-03-28T14:37:54.965Z
12
12
 
13
13
  progress:
14
14
  done: 1
@@ -0,0 +1,318 @@
1
+ __vite_ssr_exportName__("patchGitignore", () => { try { return patchGitignore } catch {} });
2
+ __vite_ssr_exportName__("runInit", () => { try { return runInit } catch {} });
3
+ __vite_ssr_exportName__("setupContextMode", () => { try { return setupContextMode } catch {} });
4
+ __vite_ssr_exportName__("setupMcp2cli", () => { try { return setupMcp2cli } catch {} });
5
+ const __vite_ssr_import_0__ = await __vite_ssr_import__("node:fs", {"importedNames":["mkdirSync","existsSync","readFileSync","writeFileSync","renameSync"]});
6
+ const __vite_ssr_import_1__ = await __vite_ssr_import__("node:path", {"importedNames":["join"]});
7
+ const __vite_ssr_import_2__ = await __vite_ssr_import__("node:os", {"importedNames":["homedir"]});
8
+ const __vite_ssr_import_3__ = await __vite_ssr_import__("node:readline", {"importedNames":["createInterface"]});
9
+ const __vite_ssr_import_4__ = await __vite_ssr_import__("node:child_process", {"importedNames":["spawnSync"]});
10
+ /**
11
+ * src/init.ts — interactive setup wizard for frappe-builder
12
+ *
13
+ * Handles: global config (~/.frappe-builder/config.json),
14
+ * project config (.frappe-builder-config.json),
15
+ * and .gitignore patching.
16
+ *
17
+ * No imports from state/, extensions/, or gates/.
18
+ * Uses Node.js built-ins only — no external prompt libraries.
19
+ */
20
+
21
+
22
+
23
+
24
+
25
+ let cancelled = false;
26
+ process.on("SIGINT", () => {
27
+ cancelled = true;
28
+ });
29
+ function promptLine(question) {
30
+ return new Promise((resolve) => {
31
+ if (cancelled) {
32
+ resolve("");
33
+ return;
34
+ };
35
+ const rl = (0,__vite_ssr_import_3__.createInterface)({
36
+ input: process.stdin,
37
+ output: process.stdout
38
+ });
39
+ rl.question(question, (answer) => {
40
+ rl.close();
41
+ resolve(answer);
42
+ });
43
+ });
44
+ }
45
+ async function promptYN(question) {
46
+ const answer = await promptLine(question + " (y/N): ");
47
+ return answer.trim().toLowerCase() === "y" || answer.trim().toLowerCase() === "yes";
48
+ }
49
+ function writeAtomic(filePath, content) {
50
+ const tmp = filePath + ".tmp";
51
+ (0,__vite_ssr_import_0__.writeFileSync)(tmp, content, "utf-8");
52
+ (0,__vite_ssr_import_0__.renameSync)(tmp, filePath);
53
+ }
54
+ /** Patches .gitignore to include the exact entry if not already present. */
55
+ function patchGitignore(projectRoot, entry) {
56
+ const gitignorePath = (0,__vite_ssr_import_1__.join)(projectRoot, ".gitignore");
57
+ if (!(0,__vite_ssr_import_0__.existsSync)(gitignorePath)) {
58
+ (0,__vite_ssr_import_0__.writeFileSync)(gitignorePath, entry + "\n", "utf-8");
59
+ return "created";
60
+ };
61
+ const content = (0,__vite_ssr_import_0__.readFileSync)(gitignorePath, "utf-8");
62
+ const lines = content.split("\n");
63
+ if (lines.includes(entry)) {
64
+ return "already-present";
65
+ };
66
+ const patched = content.endsWith("\n") ? content + entry + "\n" : content + "\n" + entry + "\n";
67
+ (0,__vite_ssr_import_0__.writeFileSync)(gitignorePath, patched, "utf-8");
68
+ return "patched";
69
+ };
70
+ async function runInit(opts = {}) {
71
+ const projectRoot = opts.projectRoot ?? process.cwd();
72
+ const homeDir = (0,__vite_ssr_import_2__.homedir)();
73
+ const globalConfigDir = (0,__vite_ssr_import_1__.join)(homeDir, ".frappe-builder");
74
+ const globalConfigPath = (0,__vite_ssr_import_1__.join)(globalConfigDir, "config.json");
75
+ const projectConfigPath = (0,__vite_ssr_import_1__.join)(projectRoot, ".frappe-builder-config.json");
76
+ console.log("\n=== frappe-builder Setup ===\n");
77
+ // ── Global config ────────────────────────────────────────────────────────
78
+ console.log(`[Global config: ${globalConfigPath}]`);
79
+ let globalConfig = {};
80
+ let globalAction = "written";
81
+ if ((0,__vite_ssr_import_0__.existsSync)(globalConfigPath)) {
82
+ try {
83
+ globalConfig = JSON.parse((0,__vite_ssr_import_0__.readFileSync)(globalConfigPath, "utf-8"));
84
+ } catch {};
85
+ if (!cancelled) {
86
+ const overwrite = await promptYN(`Overwrite existing ${globalConfigPath}?`);
87
+ if (cancelled) {
88
+ printCancelled();
89
+ return;
90
+ };
91
+ if (!overwrite) {
92
+ globalAction = "skipped";
93
+ console.log(" Keeping existing global config.\n");
94
+ }
95
+ }
96
+ };
97
+ if (!cancelled && globalAction === "written") {
98
+ const llmKey = await promptLine("LLM API key (leave blank to skip): ");
99
+ if (cancelled) {
100
+ printCancelled();
101
+ return;
102
+ };
103
+ globalConfig.llm_api_key = llmKey.trim();
104
+ };
105
+ // ── Project config ───────────────────────────────────────────────────────
106
+ console.log(`\n[Project config: ${projectConfigPath}]`);
107
+ let projectConfig = {};
108
+ let projectAction = "written";
109
+ if ((0,__vite_ssr_import_0__.existsSync)(projectConfigPath)) {
110
+ try {
111
+ projectConfig = JSON.parse((0,__vite_ssr_import_0__.readFileSync)(projectConfigPath, "utf-8"));
112
+ } catch {};
113
+ if (!cancelled) {
114
+ const overwrite = await promptYN(`Overwrite existing ${projectConfigPath}?`);
115
+ if (cancelled) {
116
+ printCancelled();
117
+ return;
118
+ };
119
+ if (!overwrite) {
120
+ projectAction = "skipped";
121
+ console.log(" Keeping existing project config.\n");
122
+ }
123
+ }
124
+ };
125
+ if (!cancelled && projectAction === "written") {
126
+ const siteUrl = await promptLine("Frappe site URL (e.g. http://site1.localhost): ");
127
+ if (cancelled) {
128
+ printCancelled();
129
+ return;
130
+ };
131
+ const apiKey = await promptLine("Frappe API key: ");
132
+ if (cancelled) {
133
+ printCancelled();
134
+ return;
135
+ };
136
+ const apiSecret = await promptLine("Frappe API secret: ");
137
+ if (cancelled) {
138
+ printCancelled();
139
+ return;
140
+ };
141
+ projectConfig = {
142
+ site_url: siteUrl.trim(),
143
+ api_key: apiKey.trim(),
144
+ api_secret: apiSecret.trim()
145
+ };
146
+ };
147
+ // ── All prompts collected — now write files ──────────────────────────────
148
+ const written = [];
149
+ const skipped = [];
150
+ // Global config
151
+ if (globalAction === "written") {
152
+ (0,__vite_ssr_import_0__.mkdirSync)(globalConfigDir, { recursive: true });
153
+ writeAtomic(globalConfigPath, JSON.stringify(globalConfig, null, 2) + "\n");
154
+ written.push(`~/.frappe-builder/config.json`);
155
+ } else {
156
+ skipped.push(`~/.frappe-builder/config.json`);
157
+ };
158
+ // Project config
159
+ if (projectAction === "written") {
160
+ writeAtomic(projectConfigPath, JSON.stringify(projectConfig, null, 2) + "\n");
161
+ written.push(`.frappe-builder-config.json`);
162
+ } else {
163
+ skipped.push(`.frappe-builder-config.json`);
164
+ };
165
+ // Gitignore patch
166
+ const gitignoreResult = patchGitignore(projectRoot, ".frappe-builder-config.json");
167
+ if (gitignoreResult === "patched") {
168
+ written.push(".gitignore (patched)");
169
+ } else if (gitignoreResult === "created") {
170
+ written.push(".gitignore (created)");
171
+ } else {
172
+ skipped.push(".gitignore (entry already present)");
173
+ };
174
+ // ── context-mode MCP extension ───────────────────────────────────────────
175
+ await setupContextMode(homeDir);
176
+ // ── mcp2cli skill + context-mode bake ────────────────────────────────────
177
+ setupMcp2cli(homeDir);
178
+ // ── Summary ──────────────────────────────────────────────────────────────
179
+ console.log("\nFiles written:");
180
+ for (const f of written) {
181
+ console.log(` ✓ ${f}`);
182
+ };
183
+ if (skipped.length > 0) {
184
+ console.log("Skipped:");
185
+ for (const f of skipped) {
186
+ console.log(` - ${f}`);
187
+ }
188
+ };
189
+ console.log("\nReady. Run: frappe-builder\n");
190
+ };
191
+ function printCancelled() {
192
+ console.log("\nSetup cancelled. No files were written.\n");
193
+ }
194
+ /**
195
+ * Installs and configures the context-mode pi MCP extension.
196
+ * Clones https://github.com/mksglu/context-mode into ~/.pi/extensions/context-mode,
197
+ * builds it, and patches ~/.pi/settings/mcp.json with the server entry.
198
+ *
199
+ * Non-fatal — failures are logged as warnings, never abort init.
200
+ */
201
+ async function setupContextMode(homeDir) {
202
+ const extDir = (0,__vite_ssr_import_1__.join)(homeDir, ".pi", "extensions", "context-mode");
203
+ const mcpSettingsDir = (0,__vite_ssr_import_1__.join)(homeDir, ".pi", "settings");
204
+ const mcpSettingsPath = (0,__vite_ssr_import_1__.join)(mcpSettingsDir, "mcp.json");
205
+ const startScript = (0,__vite_ssr_import_1__.join)(extDir, "node_modules", "context-mode", "start.mjs");
206
+ console.log("\n[context-mode MCP extension]");
207
+ // ── Already installed? ──────────────────────────────────────────────────
208
+ if ((0,__vite_ssr_import_0__.existsSync)(extDir)) {
209
+ console.log(" ✓ context-mode already installed at ~/.pi/extensions/context-mode");
210
+ } else {
211
+ console.log(" context-mode not found — installing (requires git + Node.js)...");
212
+ (0,__vite_ssr_import_0__.mkdirSync)((0,__vite_ssr_import_1__.join)(homeDir, ".pi", "extensions"), { recursive: true });
213
+ const clone = (0,__vite_ssr_import_4__.spawnSync)("git", [
214
+ "clone",
215
+ "https://github.com/mksglu/context-mode.git",
216
+ extDir
217
+ ], { stdio: "pipe" });
218
+ if (clone.status !== 0) {
219
+ console.warn(` ⚠ git clone failed: ${clone.stderr?.toString().trim()}`);
220
+ console.warn(" Skipping context-mode setup. Install manually: https://github.com/mksglu/context-mode");
221
+ return;
222
+ };
223
+ const install = (0,__vite_ssr_import_4__.spawnSync)("npm", ["install"], {
224
+ cwd: extDir,
225
+ stdio: "pipe"
226
+ });
227
+ if (install.status !== 0) {
228
+ console.warn(` ⚠ npm install failed: ${install.stderr?.toString().trim()}`);
229
+ return;
230
+ };
231
+ const build = (0,__vite_ssr_import_4__.spawnSync)("npm", ["run", "build"], {
232
+ cwd: extDir,
233
+ stdio: "pipe"
234
+ });
235
+ if (build.status !== 0) {
236
+ console.warn(` ⚠ npm run build failed: ${build.stderr?.toString().trim()}`);
237
+ return;
238
+ };
239
+ console.log(" ✓ context-mode installed and built");
240
+ };
241
+ // ── Patch ~/.pi/settings/mcp.json ──────────────────────────────────────
242
+ (0,__vite_ssr_import_0__.mkdirSync)(mcpSettingsDir, { recursive: true });
243
+ let mcpConfig = {};
244
+ if ((0,__vite_ssr_import_0__.existsSync)(mcpSettingsPath)) {
245
+ try {
246
+ mcpConfig = JSON.parse((0,__vite_ssr_import_0__.readFileSync)(mcpSettingsPath, "utf-8"));
247
+ } catch {}
248
+ };
249
+ const servers = mcpConfig.mcpServers ?? {};
250
+ if (servers["context-mode"]) {
251
+ console.log(" ✓ context-mode already in ~/.pi/settings/mcp.json");
252
+ return;
253
+ };
254
+ servers["context-mode"] = {
255
+ command: "node",
256
+ args: [startScript]
257
+ };
258
+ mcpConfig.mcpServers = servers;
259
+ writeAtomic(mcpSettingsPath, JSON.stringify(mcpConfig, null, 2) + "\n");
260
+ console.log(" ✓ Added context-mode to ~/.pi/settings/mcp.json");
261
+ console.log(" Restart pi (or frappe-builder) for context-mode to activate.");
262
+ };
263
+ /**
264
+ * Installs the mcp2cli Claude Code skill and bakes the context-mode connection
265
+ * so the agent can call `mcp2cli @context-mode <tool>` without repeating flags.
266
+ *
267
+ * Non-fatal — failures are logged as warnings, never abort init.
268
+ */
269
+ function setupMcp2cli(homeDir) {
270
+ const startScript = (0,__vite_ssr_import_1__.join)(homeDir, ".pi", "extensions", "context-mode", "node_modules", "context-mode", "start.mjs");
271
+ console.log("\n[mcp2cli skill + context-mode bake]");
272
+ // ── Install mcp2cli Claude Code skill ───────────────────────────────────
273
+ const skillAdd = (0,__vite_ssr_import_4__.spawnSync)("npx", [
274
+ "skills",
275
+ "add",
276
+ "knowsuchagency/mcp2cli",
277
+ "--skill",
278
+ "mcp2cli"
279
+ ], { stdio: "pipe" });
280
+ if (skillAdd.status !== 0) {
281
+ console.warn(` ⚠ mcp2cli skill install failed: ${skillAdd.stderr?.toString().trim()}`);
282
+ console.warn(" Install manually: npx skills add knowsuchagency/mcp2cli --skill mcp2cli");
283
+ } else {
284
+ console.log(" ✓ mcp2cli skill installed");
285
+ };
286
+ // ── Bake context-mode connection ─────────────────────────────────────────
287
+ // Check if already baked
288
+ const bakeShow = (0,__vite_ssr_import_4__.spawnSync)("mcp2cli", [
289
+ "bake",
290
+ "show",
291
+ "context-mode"
292
+ ], { stdio: "pipe" });
293
+ if (bakeShow.status === 0) {
294
+ console.log(" ✓ mcp2cli @context-mode already baked");
295
+ return;
296
+ };
297
+ if (!(0,__vite_ssr_import_0__.existsSync)(startScript)) {
298
+ console.warn(" ⚠ context-mode start.mjs not found — skipping bake (run init again after context-mode installs)");
299
+ return;
300
+ };
301
+ const bakeCreate = (0,__vite_ssr_import_4__.spawnSync)("mcp2cli", [
302
+ "bake",
303
+ "create",
304
+ "context-mode",
305
+ "--mcp-stdio",
306
+ `node ${startScript}`
307
+ ], { stdio: "pipe" });
308
+ if (bakeCreate.status !== 0) {
309
+ console.warn(` ⚠ mcp2cli bake failed: ${bakeCreate.stderr?.toString().trim()}`);
310
+ console.warn(" Install mcp2cli first: pip install mcp2cli");
311
+ } else {
312
+ console.log(" ✓ mcp2cli @context-mode baked — agent can now call: mcp2cli @context-mode <tool>");
313
+ }
314
+ }
315
+ //# sourceMappingSource=vite-generated
316
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"mappings":"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","names":[],"ignoreList":[],"sources":["init.ts"],"sourcesContent":["/**\n * src/init.ts — interactive setup wizard for frappe-builder\n *\n * Handles: global config (~/.frappe-builder/config.json),\n *          project config (.frappe-builder-config.json),\n *          and .gitignore patching.\n *\n * No imports from state/, extensions/, or gates/.\n * Uses Node.js built-ins only — no external prompt libraries.\n */\n\nimport { mkdirSync, existsSync, readFileSync, writeFileSync, renameSync } from \"node:fs\";\nimport { join } from \"node:path\";\nimport { homedir } from \"node:os\";\nimport { createInterface } from \"node:readline\";\nimport { spawnSync } from \"node:child_process\";\n\nlet cancelled = false;\n\nprocess.on(\"SIGINT\", () => {\n  cancelled = true;\n});\n\nfunction promptLine(question: string): Promise<string> {\n  return new Promise((resolve) => {\n    if (cancelled) { resolve(\"\"); return; }\n    const rl = createInterface({ input: process.stdin, output: process.stdout });\n    rl.question(question, (answer) => {\n      rl.close();\n      resolve(answer);\n    });\n  });\n}\n\nasync function promptYN(question: string): Promise<boolean> {\n  const answer = await promptLine(question + \" (y/N): \");\n  return answer.trim().toLowerCase() === \"y\" || answer.trim().toLowerCase() === \"yes\";\n}\n\nfunction writeAtomic(filePath: string, content: string): void {\n  const tmp = filePath + \".tmp\";\n  writeFileSync(tmp, content, \"utf-8\");\n  renameSync(tmp, filePath);\n}\n\n/** Patches .gitignore to include the exact entry if not already present. */\nexport function patchGitignore(projectRoot: string, entry: string): \"patched\" | \"already-present\" | \"created\" {\n  const gitignorePath = join(projectRoot, \".gitignore\");\n  if (!existsSync(gitignorePath)) {\n    writeFileSync(gitignorePath, entry + \"\\n\", \"utf-8\");\n    return \"created\";\n  }\n  const content = readFileSync(gitignorePath, \"utf-8\");\n  const lines = content.split(\"\\n\");\n  if (lines.includes(entry)) {\n    return \"already-present\";\n  }\n  const patched = content.endsWith(\"\\n\") ? content + entry + \"\\n\" : content + \"\\n\" + entry + \"\\n\";\n  writeFileSync(gitignorePath, patched, \"utf-8\");\n  return \"patched\";\n}\n\nexport async function runInit(opts: { projectRoot?: string } = {}): Promise<void> {\n  const projectRoot = opts.projectRoot ?? process.cwd();\n  const homeDir = homedir();\n  const globalConfigDir = join(homeDir, \".frappe-builder\");\n  const globalConfigPath = join(globalConfigDir, \"config.json\");\n  const projectConfigPath = join(projectRoot, \".frappe-builder-config.json\");\n\n  console.log(\"\\n=== frappe-builder Setup ===\\n\");\n\n  // ── Global config ────────────────────────────────────────────────────────\n  console.log(`[Global config: ${globalConfigPath}]`);\n\n  let globalConfig: Record<string, unknown> = {};\n  let globalAction: \"written\" | \"skipped\" = \"written\";\n\n  if (existsSync(globalConfigPath)) {\n    try {\n      globalConfig = JSON.parse(readFileSync(globalConfigPath, \"utf-8\")) as Record<string, unknown>;\n    } catch { /* ignore malformed file — overwrite */ }\n    if (!cancelled) {\n      const overwrite = await promptYN(`Overwrite existing ${globalConfigPath}?`);\n      if (cancelled) { printCancelled(); return; }\n      if (!overwrite) {\n        globalAction = \"skipped\";\n        console.log(\"  Keeping existing global config.\\n\");\n      }\n    }\n  }\n\n  if (!cancelled && globalAction === \"written\") {\n    const llmKey = await promptLine(\"LLM API key (leave blank to skip): \");\n    if (cancelled) { printCancelled(); return; }\n    globalConfig.llm_api_key = llmKey.trim();\n  }\n\n  // ── Project config ───────────────────────────────────────────────────────\n  console.log(`\\n[Project config: ${projectConfigPath}]`);\n\n  let projectConfig: Record<string, unknown> = {};\n  let projectAction: \"written\" | \"skipped\" = \"written\";\n\n  if (existsSync(projectConfigPath)) {\n    try {\n      projectConfig = JSON.parse(readFileSync(projectConfigPath, \"utf-8\")) as Record<string, unknown>;\n    } catch { /* ignore malformed file — overwrite */ }\n    if (!cancelled) {\n      const overwrite = await promptYN(`Overwrite existing ${projectConfigPath}?`);\n      if (cancelled) { printCancelled(); return; }\n      if (!overwrite) {\n        projectAction = \"skipped\";\n        console.log(\"  Keeping existing project config.\\n\");\n      }\n    }\n  }\n\n  if (!cancelled && projectAction === \"written\") {\n    const siteUrl = await promptLine(\"Frappe site URL (e.g. http://site1.localhost): \");\n    if (cancelled) { printCancelled(); return; }\n    const apiKey = await promptLine(\"Frappe API key: \");\n    if (cancelled) { printCancelled(); return; }\n    const apiSecret = await promptLine(\"Frappe API secret: \");\n    if (cancelled) { printCancelled(); return; }\n    projectConfig = {\n      site_url: siteUrl.trim(),\n      api_key: apiKey.trim(),\n      api_secret: apiSecret.trim(),\n    };\n  }\n\n  // ── All prompts collected — now write files ──────────────────────────────\n  const written: string[] = [];\n  const skipped: string[] = [];\n\n  // Global config\n  if (globalAction === \"written\") {\n    mkdirSync(globalConfigDir, { recursive: true });\n    writeAtomic(globalConfigPath, JSON.stringify(globalConfig, null, 2) + \"\\n\");\n    written.push(`~/.frappe-builder/config.json`);\n  } else {\n    skipped.push(`~/.frappe-builder/config.json`);\n  }\n\n  // Project config\n  if (projectAction === \"written\") {\n    writeAtomic(projectConfigPath, JSON.stringify(projectConfig, null, 2) + \"\\n\");\n    written.push(`.frappe-builder-config.json`);\n  } else {\n    skipped.push(`.frappe-builder-config.json`);\n  }\n\n  // Gitignore patch\n  const gitignoreResult = patchGitignore(projectRoot, \".frappe-builder-config.json\");\n  if (gitignoreResult === \"patched\") {\n    written.push(\".gitignore (patched)\");\n  } else if (gitignoreResult === \"created\") {\n    written.push(\".gitignore (created)\");\n  } else {\n    skipped.push(\".gitignore (entry already present)\");\n  }\n\n  // ── context-mode MCP extension ───────────────────────────────────────────\n  await setupContextMode(homeDir);\n\n  // ── mcp2cli skill + context-mode bake ────────────────────────────────────\n  setupMcp2cli(homeDir);\n\n  // ── Summary ──────────────────────────────────────────────────────────────\n  console.log(\"\\nFiles written:\");\n  for (const f of written) {\n    console.log(`  ✓ ${f}`);\n  }\n  if (skipped.length > 0) {\n    console.log(\"Skipped:\");\n    for (const f of skipped) {\n      console.log(`  - ${f}`);\n    }\n  }\n\n  console.log(\"\\nReady. Run: frappe-builder\\n\");\n}\n\nfunction printCancelled(): void {\n  console.log(\"\\nSetup cancelled. No files were written.\\n\");\n}\n\n/**\n * Installs and configures the context-mode pi MCP extension.\n * Clones https://github.com/mksglu/context-mode into ~/.pi/extensions/context-mode,\n * builds it, and patches ~/.pi/settings/mcp.json with the server entry.\n *\n * Non-fatal — failures are logged as warnings, never abort init.\n */\nexport async function setupContextMode(homeDir: string): Promise<void> {\n  const extDir = join(homeDir, \".pi\", \"extensions\", \"context-mode\");\n  const mcpSettingsDir = join(homeDir, \".pi\", \"settings\");\n  const mcpSettingsPath = join(mcpSettingsDir, \"mcp.json\");\n  const startScript = join(extDir, \"node_modules\", \"context-mode\", \"start.mjs\");\n\n  console.log(\"\\n[context-mode MCP extension]\");\n\n  // ── Already installed? ──────────────────────────────────────────────────\n  if (existsSync(extDir)) {\n    console.log(\"  ✓ context-mode already installed at ~/.pi/extensions/context-mode\");\n  } else {\n    console.log(\"  context-mode not found — installing (requires git + Node.js)...\");\n    mkdirSync(join(homeDir, \".pi\", \"extensions\"), { recursive: true });\n\n    const clone = spawnSync(\"git\", [\n      \"clone\", \"https://github.com/mksglu/context-mode.git\", extDir,\n    ], { stdio: \"pipe\" });\n\n    if (clone.status !== 0) {\n      console.warn(`  ⚠ git clone failed: ${clone.stderr?.toString().trim()}`);\n      console.warn(\"  Skipping context-mode setup. Install manually: https://github.com/mksglu/context-mode\");\n      return;\n    }\n\n    const install = spawnSync(\"npm\", [\"install\"], { cwd: extDir, stdio: \"pipe\" });\n    if (install.status !== 0) {\n      console.warn(`  ⚠ npm install failed: ${install.stderr?.toString().trim()}`);\n      return;\n    }\n\n    const build = spawnSync(\"npm\", [\"run\", \"build\"], { cwd: extDir, stdio: \"pipe\" });\n    if (build.status !== 0) {\n      console.warn(`  ⚠ npm run build failed: ${build.stderr?.toString().trim()}`);\n      return;\n    }\n\n    console.log(\"  ✓ context-mode installed and built\");\n  }\n\n  // ── Patch ~/.pi/settings/mcp.json ──────────────────────────────────────\n  mkdirSync(mcpSettingsDir, { recursive: true });\n\n  let mcpConfig: Record<string, unknown> = {};\n  if (existsSync(mcpSettingsPath)) {\n    try {\n      mcpConfig = JSON.parse(readFileSync(mcpSettingsPath, \"utf-8\")) as Record<string, unknown>;\n    } catch { /* overwrite malformed file */ }\n  }\n\n  const servers = (mcpConfig.mcpServers ?? {}) as Record<string, unknown>;\n  if (servers[\"context-mode\"]) {\n    console.log(\"  ✓ context-mode already in ~/.pi/settings/mcp.json\");\n    return;\n  }\n\n  servers[\"context-mode\"] = {\n    command: \"node\",\n    args: [startScript],\n  };\n  mcpConfig.mcpServers = servers;\n\n  writeAtomic(mcpSettingsPath, JSON.stringify(mcpConfig, null, 2) + \"\\n\");\n  console.log(\"  ✓ Added context-mode to ~/.pi/settings/mcp.json\");\n  console.log(\"  Restart pi (or frappe-builder) for context-mode to activate.\");\n}\n\n/**\n * Installs the mcp2cli Claude Code skill and bakes the context-mode connection\n * so the agent can call `mcp2cli @context-mode <tool>` without repeating flags.\n *\n * Non-fatal — failures are logged as warnings, never abort init.\n */\nexport function setupMcp2cli(homeDir: string): void {\n  const startScript = join(\n    homeDir, \".pi\", \"extensions\", \"context-mode\",\n    \"node_modules\", \"context-mode\", \"start.mjs\"\n  );\n\n  console.log(\"\\n[mcp2cli skill + context-mode bake]\");\n\n  // ── Install mcp2cli Claude Code skill ───────────────────────────────────\n  const skillAdd = spawnSync(\n    \"npx\",\n    [\"skills\", \"add\", \"knowsuchagency/mcp2cli\", \"--skill\", \"mcp2cli\"],\n    { stdio: \"pipe\" }\n  );\n  if (skillAdd.status !== 0) {\n    console.warn(`  ⚠ mcp2cli skill install failed: ${skillAdd.stderr?.toString().trim()}`);\n    console.warn(\"  Install manually: npx skills add knowsuchagency/mcp2cli --skill mcp2cli\");\n  } else {\n    console.log(\"  ✓ mcp2cli skill installed\");\n  }\n\n  // ── Bake context-mode connection ─────────────────────────────────────────\n  // Check if already baked\n  const bakeShow = spawnSync(\"mcp2cli\", [\"bake\", \"show\", \"context-mode\"], { stdio: \"pipe\" });\n  if (bakeShow.status === 0) {\n    console.log(\"  ✓ mcp2cli @context-mode already baked\");\n    return;\n  }\n\n  if (!existsSync(startScript)) {\n    console.warn(\"  ⚠ context-mode start.mjs not found — skipping bake (run init again after context-mode installs)\");\n    return;\n  }\n\n  const bakeCreate = spawnSync(\n    \"mcp2cli\",\n    [\"bake\", \"create\", \"context-mode\", \"--mcp-stdio\", `node ${startScript}`],\n    { stdio: \"pipe\" }\n  );\n  if (bakeCreate.status !== 0) {\n    console.warn(`  ⚠ mcp2cli bake failed: ${bakeCreate.stderr?.toString().trim()}`);\n    console.warn(\"  Install mcp2cli first: pip install mcp2cli\");\n  } else {\n    console.log(\"  ✓ mcp2cli @context-mode baked — agent can now call: mcp2cli @context-mode <tool>\");\n  }\n}\n"],"file":"/src/init.ts"}
317
+
318
+ //# vitestCache=W3siZmlsZSI6IjEiLCJpZCI6IjEiLCJ1cmwiOiIyIiwiaW1wb3J0ZWRVcmxzIjoiMyIsIm1hcHBpbmdzIjpmYWxzZX0sIi9ob21lL3Jpei9mcmFwcGUtYnVpbGRlci9zcmMvaW5pdC50cyIsIi9zcmMvaW5pdC50cyIsW11d
@@ -0,0 +1,207 @@
1
+ const __vite_ssr_import_0__ = await __vite_ssr_import__("/node_modules/vitest/dist/index.js", {"importedNames":["describe","it","expect","vi","beforeEach","afterEach"]});
2
+ __vite_ssr_import_0__.vi.mock("node:os", async (importOriginal) => {
3
+ const actual = await importOriginal();
4
+ return {
5
+ ...actual,
6
+ homedir: __vite_ssr_import_0__.vi.fn(() => actual.homedir())
7
+ };
8
+ });
9
+ __vite_ssr_import_0__.vi.mock("node:readline", () => ({ createInterface: __vite_ssr_import_0__.vi.fn(() => ({
10
+ question: __vite_ssr_import_0__.vi.fn((_q, cb) => {
11
+ cb(mockAnswers.shift() ?? "");
12
+ }),
13
+ close: __vite_ssr_import_0__.vi.fn()
14
+ })) }));
15
+ const __vi_import_0__ = await __vite_ssr_dynamic_import__("node:fs");
16
+ const __vi_import_1__ = await __vite_ssr_dynamic_import__("node:path");
17
+ const __vi_import_2__ = await __vite_ssr_dynamic_import__("node:os");
18
+ const __vi_import_3__ = await __vite_ssr_dynamic_import__("/src/init.ts");
19
+
20
+
21
+
22
+ // ── Top-level mocks — must be hoisted before any imports that use them ───────
23
+ // Shared answer queue — tests push answers before calling runInit
24
+ const mockAnswers = [];
25
+
26
+
27
+ // ── patchGitignore ───────────────────────────────────────────────────────────
28
+ (0,__vite_ssr_import_0__.describe)("patchGitignore", () => {
29
+ let tmpDir;
30
+ (0,__vite_ssr_import_0__.beforeEach)(() => {
31
+ tmpDir = __vi_import_0__.mkdtempSync(__vi_import_1__.join("/tmp", "fb-init-gitignore-"));
32
+ });
33
+ (0,__vite_ssr_import_0__.afterEach)(() => {
34
+ __vi_import_0__.rmSync(tmpDir, {
35
+ recursive: true,
36
+ force: true
37
+ });
38
+ });
39
+ (0,__vite_ssr_import_0__.it)("creates .gitignore from scratch when absent and adds the entry", () => {
40
+ const result = __vi_import_3__.patchGitignore(tmpDir, ".frappe-builder-config.json");
41
+ (0,__vite_ssr_import_0__.expect)(result).toBe("created");
42
+ const content = __vi_import_0__.readFileSync(__vi_import_1__.join(tmpDir, ".gitignore"), "utf-8");
43
+ (0,__vite_ssr_import_0__.expect)(content).toContain(".frappe-builder-config.json");
44
+ });
45
+ (0,__vite_ssr_import_0__.it)("appends line when .gitignore exists but entry is absent", () => {
46
+ __vi_import_0__.writeFileSync(__vi_import_1__.join(tmpDir, ".gitignore"), "node_modules\n.env\n", "utf-8");
47
+ const result = __vi_import_3__.patchGitignore(tmpDir, ".frappe-builder-config.json");
48
+ (0,__vite_ssr_import_0__.expect)(result).toBe("patched");
49
+ const content = __vi_import_0__.readFileSync(__vi_import_1__.join(tmpDir, ".gitignore"), "utf-8");
50
+ const lines = content.split("\n");
51
+ (0,__vite_ssr_import_0__.expect)(lines).toContain(".frappe-builder-config.json");
52
+ (0,__vite_ssr_import_0__.expect)(lines).toContain("node_modules");
53
+ (0,__vite_ssr_import_0__.expect)(lines).toContain(".env");
54
+ });
55
+ (0,__vite_ssr_import_0__.it)("does NOT duplicate entry when already present", () => {
56
+ __vi_import_0__.writeFileSync(__vi_import_1__.join(tmpDir, ".gitignore"), "node_modules\n.frappe-builder-config.json\n.env\n", "utf-8");
57
+ const result = __vi_import_3__.patchGitignore(tmpDir, ".frappe-builder-config.json");
58
+ (0,__vite_ssr_import_0__.expect)(result).toBe("already-present");
59
+ const content = __vi_import_0__.readFileSync(__vi_import_1__.join(tmpDir, ".gitignore"), "utf-8");
60
+ const count = content.split("\n").filter((l) => l === ".frappe-builder-config.json").length;
61
+ (0,__vite_ssr_import_0__.expect)(count).toBe(1);
62
+ });
63
+ (0,__vite_ssr_import_0__.it)("does NOT accept a glob pattern as a match — requires exact filename", () => {
64
+ __vi_import_0__.writeFileSync(__vi_import_1__.join(tmpDir, ".gitignore"), "*.json\n", "utf-8");
65
+ const result = __vi_import_3__.patchGitignore(tmpDir, ".frappe-builder-config.json");
66
+ (0,__vite_ssr_import_0__.expect)(result).toBe("patched");
67
+ });
68
+ });
69
+ // ── runInit — file writes ────────────────────────────────────────────────────
70
+ (0,__vite_ssr_import_0__.describe)("runInit — file writes", () => {
71
+ let projectDir;
72
+ let homeDir;
73
+ (0,__vite_ssr_import_0__.beforeEach)(() => {
74
+ projectDir = __vi_import_0__.mkdtempSync(__vi_import_1__.join("/tmp", "fb-init-proj-"));
75
+ homeDir = __vi_import_0__.mkdtempSync(__vi_import_1__.join("/tmp", "fb-init-home-"));
76
+ __vite_ssr_import_0__.vi.mocked(__vi_import_2__.homedir).mockReturnValue(homeDir);
77
+ mockAnswers.length = 0;
78
+ });
79
+ (0,__vite_ssr_import_0__.afterEach)(() => {
80
+ __vite_ssr_import_0__.vi.clearAllMocks();
81
+ __vi_import_0__.rmSync(projectDir, {
82
+ recursive: true,
83
+ force: true
84
+ });
85
+ __vi_import_0__.rmSync(homeDir, {
86
+ recursive: true,
87
+ force: true
88
+ });
89
+ });
90
+ (0,__vite_ssr_import_0__.it)("writes global config to ~/.frappe-builder/config.json", async () => {
91
+ // Answers: llm_api_key, site_url, api_key, api_secret
92
+ mockAnswers.push("sk-test", "http://erp.localhost", "key123", "secret456");
93
+ await __vi_import_3__.runInit({ projectRoot: projectDir });
94
+ const configPath = __vi_import_1__.join(homeDir, ".frappe-builder", "config.json");
95
+ (0,__vite_ssr_import_0__.expect)(__vi_import_0__.existsSync(configPath)).toBe(true);
96
+ const cfg = JSON.parse(__vi_import_0__.readFileSync(configPath, "utf-8"));
97
+ (0,__vite_ssr_import_0__.expect)(cfg.llm_api_key).toBe("sk-test");
98
+ });
99
+ (0,__vite_ssr_import_0__.it)("writes project config to {projectRoot}/.frappe-builder-config.json", async () => {
100
+ mockAnswers.push("sk-test", "http://erp.localhost", "key123", "secret456");
101
+ await __vi_import_3__.runInit({ projectRoot: projectDir });
102
+ const configPath = __vi_import_1__.join(projectDir, ".frappe-builder-config.json");
103
+ (0,__vite_ssr_import_0__.expect)(__vi_import_0__.existsSync(configPath)).toBe(true);
104
+ const cfg = JSON.parse(__vi_import_0__.readFileSync(configPath, "utf-8"));
105
+ (0,__vite_ssr_import_0__.expect)(cfg.site_url).toBe("http://erp.localhost");
106
+ (0,__vite_ssr_import_0__.expect)(cfg.api_key).toBe("key123");
107
+ (0,__vite_ssr_import_0__.expect)(cfg.api_secret).toBe("secret456");
108
+ });
109
+ (0,__vite_ssr_import_0__.it)("patches .gitignore automatically", async () => {
110
+ mockAnswers.push("sk-test", "http://erp.localhost", "key123", "secret456");
111
+ await __vi_import_3__.runInit({ projectRoot: projectDir });
112
+ const content = __vi_import_0__.readFileSync(__vi_import_1__.join(projectDir, ".gitignore"), "utf-8");
113
+ (0,__vite_ssr_import_0__.expect)(content).toContain(".frappe-builder-config.json");
114
+ });
115
+ (0,__vite_ssr_import_0__.it)("creates ~/.frappe-builder/ directory if it does not exist", async () => {
116
+ mockAnswers.push("", "http://site.localhost", "k", "s");
117
+ await __vi_import_3__.runInit({ projectRoot: projectDir });
118
+ (0,__vite_ssr_import_0__.expect)(__vi_import_0__.existsSync(__vi_import_1__.join(homeDir, ".frappe-builder"))).toBe(true);
119
+ });
120
+ });
121
+ // ── runInit — overwrite guard ────────────────────────────────────────────────
122
+ (0,__vite_ssr_import_0__.describe)("runInit — overwrite guard", () => {
123
+ let projectDir;
124
+ let homeDir;
125
+ (0,__vite_ssr_import_0__.beforeEach)(() => {
126
+ projectDir = __vi_import_0__.mkdtempSync(__vi_import_1__.join("/tmp", "fb-init-guard-proj-"));
127
+ homeDir = __vi_import_0__.mkdtempSync(__vi_import_1__.join("/tmp", "fb-init-guard-home-"));
128
+ __vite_ssr_import_0__.vi.mocked(__vi_import_2__.homedir).mockReturnValue(homeDir);
129
+ mockAnswers.length = 0;
130
+ });
131
+ (0,__vite_ssr_import_0__.afterEach)(() => {
132
+ __vite_ssr_import_0__.vi.clearAllMocks();
133
+ __vi_import_0__.rmSync(projectDir, {
134
+ recursive: true,
135
+ force: true
136
+ });
137
+ __vi_import_0__.rmSync(homeDir, {
138
+ recursive: true,
139
+ force: true
140
+ });
141
+ });
142
+ (0,__vite_ssr_import_0__.it)("does NOT overwrite existing global config when user declines", async () => {
143
+ const configDir = __vi_import_1__.join(homeDir, ".frappe-builder");
144
+ __vi_import_0__.mkdirSync(configDir, { recursive: true });
145
+ const globalPath = __vi_import_1__.join(configDir, "config.json");
146
+ __vi_import_0__.writeFileSync(globalPath, JSON.stringify({ llm_api_key: "original-key" }), "utf-8");
147
+ // Overwrite global? → "n" | Overwrite project? → "n"
148
+ mockAnswers.push("n", "n");
149
+ await __vi_import_3__.runInit({ projectRoot: projectDir });
150
+ const cfg = JSON.parse(__vi_import_0__.readFileSync(globalPath, "utf-8"));
151
+ (0,__vite_ssr_import_0__.expect)(cfg.llm_api_key).toBe("original-key");
152
+ });
153
+ (0,__vite_ssr_import_0__.it)("overwrites existing global config when user confirms", async () => {
154
+ const configDir = __vi_import_1__.join(homeDir, ".frappe-builder");
155
+ __vi_import_0__.mkdirSync(configDir, { recursive: true });
156
+ const globalPath = __vi_import_1__.join(configDir, "config.json");
157
+ __vi_import_0__.writeFileSync(globalPath, JSON.stringify({ llm_api_key: "old-key" }), "utf-8");
158
+ // Overwrite global? → "y", new llm_key, site_url, api_key, api_secret
159
+ mockAnswers.push("y", "new-key", "http://site.localhost", "k", "s");
160
+ await __vi_import_3__.runInit({ projectRoot: projectDir });
161
+ const cfg = JSON.parse(__vi_import_0__.readFileSync(globalPath, "utf-8"));
162
+ (0,__vite_ssr_import_0__.expect)(cfg.llm_api_key).toBe("new-key");
163
+ });
164
+ (0,__vite_ssr_import_0__.it)("does NOT overwrite existing project config when user declines", async () => {
165
+ const projConfigPath = __vi_import_1__.join(projectDir, ".frappe-builder-config.json");
166
+ __vi_import_0__.writeFileSync(projConfigPath, JSON.stringify({
167
+ site_url: "http://original.localhost",
168
+ api_key: "orig",
169
+ api_secret: "s"
170
+ }), "utf-8");
171
+ // llm_api_key prompt (no existing global), overwrite project? → "n"
172
+ mockAnswers.push("sk-new", "n");
173
+ await __vi_import_3__.runInit({ projectRoot: projectDir });
174
+ const cfg = JSON.parse(__vi_import_0__.readFileSync(projConfigPath, "utf-8"));
175
+ (0,__vite_ssr_import_0__.expect)(cfg.site_url).toBe("http://original.localhost");
176
+ });
177
+ });
178
+ // ── runInit — clean exit ─────────────────────────────────────────────────────
179
+ (0,__vite_ssr_import_0__.describe)("runInit — clean exit", () => {
180
+ let projectDir;
181
+ let homeDir;
182
+ (0,__vite_ssr_import_0__.beforeEach)(() => {
183
+ projectDir = __vi_import_0__.mkdtempSync(__vi_import_1__.join("/tmp", "fb-init-exit-"));
184
+ homeDir = __vi_import_0__.mkdtempSync(__vi_import_1__.join("/tmp", "fb-init-exit-home-"));
185
+ __vite_ssr_import_0__.vi.mocked(__vi_import_2__.homedir).mockReturnValue(homeDir);
186
+ mockAnswers.length = 0;
187
+ });
188
+ (0,__vite_ssr_import_0__.afterEach)(() => {
189
+ __vite_ssr_import_0__.vi.clearAllMocks();
190
+ __vi_import_0__.rmSync(projectDir, {
191
+ recursive: true,
192
+ force: true
193
+ });
194
+ __vi_import_0__.rmSync(homeDir, {
195
+ recursive: true,
196
+ force: true
197
+ });
198
+ });
199
+ (0,__vite_ssr_import_0__.it)("resolves without throwing when all prompts return empty strings", async () => {
200
+ mockAnswers.push("", "", "", "");
201
+ await (0,__vite_ssr_import_0__.expect)(__vi_import_3__.runInit({ projectRoot: projectDir })).resolves.toBeUndefined();
202
+ });
203
+ });
204
+ //# sourceMappingSource=vite-generated
205
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"mappings":"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","names":[],"ignoreList":[],"sources":["init.test.ts"],"sourcesContent":["import { describe, it, expect, vi, beforeEach, afterEach } from \"vitest\";\nimport { mkdtempSync, rmSync, readFileSync, writeFileSync, existsSync, mkdirSync } from \"node:fs\";\nimport { join } from \"node:path\";\n\n// ── Top-level mocks — must be hoisted before any imports that use them ───────\n\nvi.mock(\"node:os\", async (importOriginal) => {\n  const actual = await importOriginal<typeof import(\"node:os\")>();\n  return { ...actual, homedir: vi.fn(() => actual.homedir()) };\n});\n\n// Shared answer queue — tests push answers before calling runInit\nconst mockAnswers: string[] = [];\n\nvi.mock(\"node:readline\", () => ({\n  createInterface: vi.fn(() => ({\n    question: vi.fn((_q: string, cb: (a: string) => void) => {\n      cb(mockAnswers.shift() ?? \"\");\n    }),\n    close: vi.fn(),\n  })),\n}));\n\nimport * as os from \"node:os\";\nimport { patchGitignore, runInit } from \"./init.js\";\n\n// ── patchGitignore ───────────────────────────────────────────────────────────\n\ndescribe(\"patchGitignore\", () => {\n  let tmpDir: string;\n\n  beforeEach(() => {\n    tmpDir = mkdtempSync(join(\"/tmp\", \"fb-init-gitignore-\"));\n  });\n\n  afterEach(() => {\n    rmSync(tmpDir, { recursive: true, force: true });\n  });\n\n  it(\"creates .gitignore from scratch when absent and adds the entry\", () => {\n    const result = patchGitignore(tmpDir, \".frappe-builder-config.json\");\n    expect(result).toBe(\"created\");\n    const content = readFileSync(join(tmpDir, \".gitignore\"), \"utf-8\");\n    expect(content).toContain(\".frappe-builder-config.json\");\n  });\n\n  it(\"appends line when .gitignore exists but entry is absent\", () => {\n    writeFileSync(join(tmpDir, \".gitignore\"), \"node_modules\\n.env\\n\", \"utf-8\");\n    const result = patchGitignore(tmpDir, \".frappe-builder-config.json\");\n    expect(result).toBe(\"patched\");\n    const content = readFileSync(join(tmpDir, \".gitignore\"), \"utf-8\");\n    const lines = content.split(\"\\n\");\n    expect(lines).toContain(\".frappe-builder-config.json\");\n    expect(lines).toContain(\"node_modules\");\n    expect(lines).toContain(\".env\");\n  });\n\n  it(\"does NOT duplicate entry when already present\", () => {\n    writeFileSync(\n      join(tmpDir, \".gitignore\"),\n      \"node_modules\\n.frappe-builder-config.json\\n.env\\n\",\n      \"utf-8\"\n    );\n    const result = patchGitignore(tmpDir, \".frappe-builder-config.json\");\n    expect(result).toBe(\"already-present\");\n    const content = readFileSync(join(tmpDir, \".gitignore\"), \"utf-8\");\n    const count = content.split(\"\\n\").filter((l) => l === \".frappe-builder-config.json\").length;\n    expect(count).toBe(1);\n  });\n\n  it(\"does NOT accept a glob pattern as a match — requires exact filename\", () => {\n    writeFileSync(join(tmpDir, \".gitignore\"), \"*.json\\n\", \"utf-8\");\n    const result = patchGitignore(tmpDir, \".frappe-builder-config.json\");\n    expect(result).toBe(\"patched\");\n  });\n});\n\n// ── runInit — file writes ────────────────────────────────────────────────────\n\ndescribe(\"runInit — file writes\", () => {\n  let projectDir: string;\n  let homeDir: string;\n\n  beforeEach(() => {\n    projectDir = mkdtempSync(join(\"/tmp\", \"fb-init-proj-\"));\n    homeDir = mkdtempSync(join(\"/tmp\", \"fb-init-home-\"));\n    vi.mocked(os.homedir).mockReturnValue(homeDir);\n    mockAnswers.length = 0;\n  });\n\n  afterEach(() => {\n    vi.clearAllMocks();\n    rmSync(projectDir, { recursive: true, force: true });\n    rmSync(homeDir, { recursive: true, force: true });\n  });\n\n  it(\"writes global config to ~/.frappe-builder/config.json\", async () => {\n    // Answers: llm_api_key, site_url, api_key, api_secret\n    mockAnswers.push(\"sk-test\", \"http://erp.localhost\", \"key123\", \"secret456\");\n    await runInit({ projectRoot: projectDir });\n\n    const configPath = join(homeDir, \".frappe-builder\", \"config.json\");\n    expect(existsSync(configPath)).toBe(true);\n    const cfg = JSON.parse(readFileSync(configPath, \"utf-8\")) as { llm_api_key: string };\n    expect(cfg.llm_api_key).toBe(\"sk-test\");\n  });\n\n  it(\"writes project config to {projectRoot}/.frappe-builder-config.json\", async () => {\n    mockAnswers.push(\"sk-test\", \"http://erp.localhost\", \"key123\", \"secret456\");\n    await runInit({ projectRoot: projectDir });\n\n    const configPath = join(projectDir, \".frappe-builder-config.json\");\n    expect(existsSync(configPath)).toBe(true);\n    const cfg = JSON.parse(readFileSync(configPath, \"utf-8\")) as {\n      site_url: string;\n      api_key: string;\n      api_secret: string;\n    };\n    expect(cfg.site_url).toBe(\"http://erp.localhost\");\n    expect(cfg.api_key).toBe(\"key123\");\n    expect(cfg.api_secret).toBe(\"secret456\");\n  });\n\n  it(\"patches .gitignore automatically\", async () => {\n    mockAnswers.push(\"sk-test\", \"http://erp.localhost\", \"key123\", \"secret456\");\n    await runInit({ projectRoot: projectDir });\n\n    const content = readFileSync(join(projectDir, \".gitignore\"), \"utf-8\");\n    expect(content).toContain(\".frappe-builder-config.json\");\n  });\n\n  it(\"creates ~/.frappe-builder/ directory if it does not exist\", async () => {\n    mockAnswers.push(\"\", \"http://site.localhost\", \"k\", \"s\");\n    await runInit({ projectRoot: projectDir });\n\n    expect(existsSync(join(homeDir, \".frappe-builder\"))).toBe(true);\n  });\n});\n\n// ── runInit — overwrite guard ────────────────────────────────────────────────\n\ndescribe(\"runInit — overwrite guard\", () => {\n  let projectDir: string;\n  let homeDir: string;\n\n  beforeEach(() => {\n    projectDir = mkdtempSync(join(\"/tmp\", \"fb-init-guard-proj-\"));\n    homeDir = mkdtempSync(join(\"/tmp\", \"fb-init-guard-home-\"));\n    vi.mocked(os.homedir).mockReturnValue(homeDir);\n    mockAnswers.length = 0;\n  });\n\n  afterEach(() => {\n    vi.clearAllMocks();\n    rmSync(projectDir, { recursive: true, force: true });\n    rmSync(homeDir, { recursive: true, force: true });\n  });\n\n  it(\"does NOT overwrite existing global config when user declines\", async () => {\n    const configDir = join(homeDir, \".frappe-builder\");\n    mkdirSync(configDir, { recursive: true });\n    const globalPath = join(configDir, \"config.json\");\n    writeFileSync(globalPath, JSON.stringify({ llm_api_key: \"original-key\" }), \"utf-8\");\n\n    // Overwrite global? → \"n\" | Overwrite project? → \"n\"\n    mockAnswers.push(\"n\", \"n\");\n    await runInit({ projectRoot: projectDir });\n\n    const cfg = JSON.parse(readFileSync(globalPath, \"utf-8\")) as { llm_api_key: string };\n    expect(cfg.llm_api_key).toBe(\"original-key\");\n  });\n\n  it(\"overwrites existing global config when user confirms\", async () => {\n    const configDir = join(homeDir, \".frappe-builder\");\n    mkdirSync(configDir, { recursive: true });\n    const globalPath = join(configDir, \"config.json\");\n    writeFileSync(globalPath, JSON.stringify({ llm_api_key: \"old-key\" }), \"utf-8\");\n\n    // Overwrite global? → \"y\", new llm_key, site_url, api_key, api_secret\n    mockAnswers.push(\"y\", \"new-key\", \"http://site.localhost\", \"k\", \"s\");\n    await runInit({ projectRoot: projectDir });\n\n    const cfg = JSON.parse(readFileSync(globalPath, \"utf-8\")) as { llm_api_key: string };\n    expect(cfg.llm_api_key).toBe(\"new-key\");\n  });\n\n  it(\"does NOT overwrite existing project config when user declines\", async () => {\n    const projConfigPath = join(projectDir, \".frappe-builder-config.json\");\n    writeFileSync(\n      projConfigPath,\n      JSON.stringify({ site_url: \"http://original.localhost\", api_key: \"orig\", api_secret: \"s\" }),\n      \"utf-8\"\n    );\n\n    // llm_api_key prompt (no existing global), overwrite project? → \"n\"\n    mockAnswers.push(\"sk-new\", \"n\");\n    await runInit({ projectRoot: projectDir });\n\n    const cfg = JSON.parse(readFileSync(projConfigPath, \"utf-8\")) as { site_url: string };\n    expect(cfg.site_url).toBe(\"http://original.localhost\");\n  });\n});\n\n// ── runInit — clean exit ─────────────────────────────────────────────────────\n\ndescribe(\"runInit — clean exit\", () => {\n  let projectDir: string;\n  let homeDir: string;\n\n  beforeEach(() => {\n    projectDir = mkdtempSync(join(\"/tmp\", \"fb-init-exit-\"));\n    homeDir = mkdtempSync(join(\"/tmp\", \"fb-init-exit-home-\"));\n    vi.mocked(os.homedir).mockReturnValue(homeDir);\n    mockAnswers.length = 0;\n  });\n\n  afterEach(() => {\n    vi.clearAllMocks();\n    rmSync(projectDir, { recursive: true, force: true });\n    rmSync(homeDir, { recursive: true, force: true });\n  });\n\n  it(\"resolves without throwing when all prompts return empty strings\", async () => {\n    mockAnswers.push(\"\", \"\", \"\", \"\");\n    await expect(runInit({ projectRoot: projectDir })).resolves.toBeUndefined();\n  });\n});\n"],"file":"/src/init.test.ts"}
206
+
207
+ //# vitestCache=W3siZmlsZSI6IjEiLCJpZCI6IjEiLCJ1cmwiOiIyIiwiaW1wb3J0ZWRVcmxzIjoiMyIsIm1hcHBpbmdzIjpmYWxzZX0sIi9ob21lL3Jpei9mcmFwcGUtYnVpbGRlci9zcmMvaW5pdC50ZXN0LnRzIiwiL3NyYy9pbml0LnRlc3QudHMiLFtdXQ==
@@ -11,7 +11,6 @@ export interface AppConfig {
11
11
  allowSubAgents: boolean;
12
12
  requirePermission: boolean;
13
13
  rules: SpawnRule[];
14
- frappeMcpUrl?: string; // URL of Frappe MCP server for frappe_query (e.g. "http://localhost:8000")
15
14
  defaultMode?: "full" | "quick"; // default feature mode: "quick" skips planning phases
16
15
  chainModel?: string; // model for chain subprocess agents (inherits parent model when unset)
17
16
  permissionMode?: PermissionMode; // agent autonomy level: auto | default | plan
package/config/loader.ts CHANGED
@@ -1,96 +1,16 @@
1
1
  import { homedir } from "node:os";
2
2
  import { join } from "node:path";
3
- import { readFileSync, existsSync, mkdirSync } from "node:fs";
3
+ import { readFileSync, writeFileSync, existsSync, mkdirSync } from "node:fs";
4
4
  import { defaults } from "./defaults.js";
5
5
  import type { AppConfig } from "./defaults.js";
6
6
 
7
- // ── Credential config types (NFR6, NFR9) ────────────────────────────────────
8
-
9
- export interface GlobalConfig {
10
- llm_api_key: string;
11
- llm_provider?: string;
12
- [key: string]: unknown;
13
- }
14
-
15
- export interface ProjectConfig {
16
- site_url: string;
17
- api_key: string;
18
- api_secret: string;
19
- [key: string]: unknown;
20
- }
21
-
22
- export interface CredentialConfig {
23
- global: GlobalConfig;
24
- project: ProjectConfig;
25
- sessionLogDir: string;
26
- }
27
-
28
- /** Always ~/.frappe-builder/sessions/ — never the project directory (NFR9). */
7
+ /** Always ~/.frappe-builder/sessions/ never the project directory. */
29
8
  export const SESSION_LOG_DIR = join(homedir(), ".frappe-builder", "sessions");
30
9
 
31
- const GITIGNORE_ERROR =
32
- "Site credentials file is not gitignored. Add .frappe-builder-config.json to .gitignore before proceeding.";
33
-
34
- /**
35
- * Validates that {projectRoot}/.gitignore lists .frappe-builder-config.json.
36
- * Throws with the exact AC error message if missing or absent.
37
- */
38
- export function validateGitignore(projectRoot: string): void {
39
- const gitignorePath = join(projectRoot, ".gitignore");
40
- if (!existsSync(gitignorePath)) {
41
- throw new Error(GITIGNORE_ERROR);
42
- }
43
- const contents = readFileSync(gitignorePath, "utf8");
44
- const lines = contents.split("\n").map((l) => l.trim());
45
- if (!lines.includes(".frappe-builder-config.json")) {
46
- throw new Error(GITIGNORE_ERROR);
47
- }
48
- }
49
-
50
- /**
51
- * Reads LLM API keys from ~/.frappe-builder/config.json (user-global, never project).
52
- * Creates ~/.frappe-builder/ on first run. Returns empty defaults if file absent.
53
- */
54
- export function loadGlobalConfig(): GlobalConfig {
55
- const configDir = join(homedir(), ".frappe-builder");
56
- mkdirSync(configDir, { recursive: true });
57
- const configPath = join(configDir, "config.json");
58
- if (!existsSync(configPath)) return { llm_api_key: "" };
59
- try {
60
- return JSON.parse(readFileSync(configPath, "utf8")) as GlobalConfig;
61
- } catch {
62
- return { llm_api_key: "" };
63
- }
64
- }
65
-
66
- /**
67
- * Reads Frappe site credentials from {projectRoot}/.frappe-builder-config.json.
68
- * Returns empty defaults if file absent (caller should surface a warning).
69
- */
70
- export function loadProjectConfig(projectRoot: string): ProjectConfig {
71
- const configPath = join(projectRoot, ".frappe-builder-config.json");
72
- if (!existsSync(configPath)) return { site_url: "", api_key: "", api_secret: "" };
73
- try {
74
- return JSON.parse(readFileSync(configPath, "utf8")) as ProjectConfig;
75
- } catch {
76
- return { site_url: "", api_key: "", api_secret: "" };
77
- }
78
- }
79
-
80
- /**
81
- * Full credential loader: validates gitignore, loads global + project configs.
82
- * Throws if .frappe-builder-config.json is not gitignored.
83
- * API keys are only ever read from ~/.frappe-builder/ — never from projectRoot.
84
- */
85
- export function loadCredentials(projectRoot: string): CredentialConfig {
86
- validateGitignore(projectRoot);
87
- const global = loadGlobalConfig();
88
- const project = loadProjectConfig(projectRoot);
89
- return { global, project, sessionLogDir: SESSION_LOG_DIR };
90
- }
91
-
92
10
  /**
93
11
  * Loads ~/.frappe-builder/config.json and merges with defaults.
12
+ * Stores agent behaviour preferences (permissionMode, requirePermission, etc.).
13
+ * Site credentials live in the session DB — not in this file.
94
14
  * Never throws — returns defaults on any read/parse failure.
95
15
  */
96
16
  export function loadConfig(): AppConfig {
@@ -103,3 +23,17 @@ export function loadConfig(): AppConfig {
103
23
  return { ...defaults };
104
24
  }
105
25
  }
26
+
27
+ /**
28
+ * Writes a partial config merged over the current config to ~/.frappe-builder/config.json.
29
+ * Creates the directory if needed. Non-fatal — never throws.
30
+ */
31
+ export function saveConfig(partial: Partial<AppConfig>): void {
32
+ try {
33
+ const configDir = join(homedir(), ".frappe-builder");
34
+ mkdirSync(configDir, { recursive: true });
35
+ const configPath = join(configDir, "config.json");
36
+ const merged = { ...loadConfig(), ...partial };
37
+ writeFileSync(configPath, JSON.stringify(merged, null, 2) + "\n", "utf-8");
38
+ } catch { /* non-fatal */ }
39
+ }
package/dist/cli.mjs CHANGED
@@ -1,20 +1,21 @@
1
1
  #!/usr/bin/env node
2
2
  import { createRequire } from "node:module";
3
3
  import { fileURLToPath } from "node:url";
4
- import { resolve } from "node:path";
4
+ import { join, resolve } from "node:path";
5
5
  import { spawnSync } from "node:child_process";
6
6
  import { existsSync } from "node:fs";
7
+ import { homedir } from "node:os";
7
8
  //#region src/cli.ts
8
9
  /**
9
10
  * src/cli.ts — frappe-builder CLI entry point
10
11
  *
11
12
  * Commands:
12
- * frappe-builder init — interactive setup wizard (run once per project)
13
+ * frappe-builder init — bootstrap agent toolchain (run once per machine)
13
14
  * frappe-builder — start a pi session with frappe-builder extensions loaded
14
15
  */
15
16
  const cmd = process.argv[2];
16
17
  if (cmd === "init") {
17
- const { runInit } = await import("./init-Gp1MgJD2.mjs");
18
+ const { runInit } = await import("./init-CkLSZ_3g.mjs");
18
19
  await runInit();
19
20
  process.exit(0);
20
21
  }
@@ -23,7 +24,7 @@ if (cmd === "--help" || cmd === "-h") {
23
24
  Usage: frappe-builder [command]
24
25
 
25
26
  Commands:
26
- init Configure frappe-builder for this project (run once per project)
27
+ init Bootstrap agent toolchain: context-mode, mcp2cli, context7 (run once per machine)
27
28
  (none) Start a frappe-builder session
28
29
 
29
30
  Options:
@@ -31,8 +32,8 @@ Options:
31
32
  `);
32
33
  process.exit(0);
33
34
  }
34
- if (!existsSync(resolve(process.cwd(), ".frappe-builder-config.json"))) {
35
- console.error("frappe-builder is not configured in this directory.");
35
+ if (!existsSync(join(homedir(), ".frappe-builder", ".initialized"))) {
36
+ console.error("frappe-builder toolchain not set up.");
36
37
  console.error("Run: frappe-builder init");
37
38
  process.exit(1);
38
39
  }
@@ -3,42 +3,16 @@ import { join } from "node:path";
3
3
  import { spawnSync } from "node:child_process";
4
4
  import { existsSync, mkdirSync, readFileSync, renameSync, writeFileSync } from "node:fs";
5
5
  import { homedir } from "node:os";
6
- import { createInterface } from "node:readline";
7
6
  //#region src/init.ts
8
7
  /**
9
- * src/init.ts — interactive setup wizard for frappe-builder
8
+ * src/init.ts — toolchain setup for frappe-builder
10
9
  *
11
- * Handles: global config (~/.frappe-builder/config.json),
12
- * project config (.frappe-builder-config.json),
13
- * and .gitignore patching.
10
+ * Bootstraps the agent environment: context-mode, mcp2cli, context7.
11
+ * No credential prompts — site credentials are collected at runtime via set_active_project.
14
12
  *
15
13
  * No imports from state/, extensions/, or gates/.
16
- * Uses Node.js built-ins only — no external prompt libraries.
14
+ * Uses Node.js built-ins only.
17
15
  */
18
- let cancelled = false;
19
- process.on("SIGINT", () => {
20
- cancelled = true;
21
- });
22
- function promptLine(question) {
23
- return new Promise((resolve) => {
24
- if (cancelled) {
25
- resolve("");
26
- return;
27
- }
28
- const rl = createInterface({
29
- input: process.stdin,
30
- output: process.stdout
31
- });
32
- rl.question(question, (answer) => {
33
- rl.close();
34
- resolve(answer);
35
- });
36
- });
37
- }
38
- async function promptYN(question) {
39
- const answer = await promptLine(question + " (y/N): ");
40
- return answer.trim().toLowerCase() === "y" || answer.trim().toLowerCase() === "yes";
41
- }
42
16
  function writeAtomic(filePath, content) {
43
17
  const tmp = filePath + ".tmp";
44
18
  writeFileSync(tmp, content, "utf-8");
@@ -59,107 +33,23 @@ function patchGitignore(projectRoot, entry) {
59
33
  async function runInit(opts = {}) {
60
34
  const projectRoot = opts.projectRoot ?? process.cwd();
61
35
  const homeDir = homedir();
62
- const globalConfigDir = join(homeDir, ".frappe-builder");
63
- const globalConfigPath = join(globalConfigDir, "config.json");
64
- const projectConfigPath = join(projectRoot, ".frappe-builder-config.json");
36
+ const markerPath = join(homeDir, ".frappe-builder", ".initialized");
65
37
  console.log("\n=== frappe-builder Setup ===\n");
66
- console.log(`[Global config: ${globalConfigPath}]`);
67
- let globalConfig = {};
68
- let globalAction = "written";
69
- if (existsSync(globalConfigPath)) {
70
- try {
71
- globalConfig = JSON.parse(readFileSync(globalConfigPath, "utf-8"));
72
- } catch {}
73
- if (!cancelled) {
74
- const overwrite = await promptYN(`Overwrite existing ${globalConfigPath}?`);
75
- if (cancelled) {
76
- printCancelled();
77
- return;
78
- }
79
- if (!overwrite) {
80
- globalAction = "skipped";
81
- console.log(" Keeping existing global config.\n");
82
- }
83
- }
84
- }
85
- if (!cancelled && globalAction === "written") {
86
- const llmKey = await promptLine("LLM API key (leave blank to skip): ");
87
- if (cancelled) {
88
- printCancelled();
89
- return;
90
- }
91
- globalConfig.llm_api_key = llmKey.trim();
92
- }
93
- console.log(`\n[Project config: ${projectConfigPath}]`);
94
- let projectConfig = {};
95
- let projectAction = "written";
96
- if (existsSync(projectConfigPath)) {
97
- try {
98
- projectConfig = JSON.parse(readFileSync(projectConfigPath, "utf-8"));
99
- } catch {}
100
- if (!cancelled) {
101
- const overwrite = await promptYN(`Overwrite existing ${projectConfigPath}?`);
102
- if (cancelled) {
103
- printCancelled();
104
- return;
105
- }
106
- if (!overwrite) {
107
- projectAction = "skipped";
108
- console.log(" Keeping existing project config.\n");
109
- }
110
- }
111
- }
112
- if (!cancelled && projectAction === "written") {
113
- const siteUrl = await promptLine("Frappe site URL (e.g. http://site1.localhost): ");
114
- if (cancelled) {
115
- printCancelled();
116
- return;
117
- }
118
- const apiKey = await promptLine("Frappe API key: ");
119
- if (cancelled) {
120
- printCancelled();
121
- return;
122
- }
123
- const apiSecret = await promptLine("Frappe API secret: ");
124
- if (cancelled) {
125
- printCancelled();
126
- return;
127
- }
128
- projectConfig = {
129
- site_url: siteUrl.trim(),
130
- api_key: apiKey.trim(),
131
- api_secret: apiSecret.trim()
132
- };
133
- }
38
+ mkdirSync(join(homeDir, ".frappe-builder"), { recursive: true });
39
+ writeFileSync(markerPath, (/* @__PURE__ */ new Date()).toISOString() + "\n", "utf-8");
40
+ const gitignoreResult = patchGitignore(projectRoot, ".fb/");
134
41
  const written = [];
135
- const skipped = [];
136
- if (globalAction === "written") {
137
- mkdirSync(globalConfigDir, { recursive: true });
138
- writeAtomic(globalConfigPath, JSON.stringify(globalConfig, null, 2) + "\n");
139
- written.push(`~/.frappe-builder/config.json`);
140
- } else skipped.push(`~/.frappe-builder/config.json`);
141
- if (projectAction === "written") {
142
- writeAtomic(projectConfigPath, JSON.stringify(projectConfig, null, 2) + "\n");
143
- written.push(`.frappe-builder-config.json`);
144
- } else skipped.push(`.frappe-builder-config.json`);
145
- const gitignoreResult = patchGitignore(projectRoot, ".frappe-builder-config.json");
146
- if (gitignoreResult === "patched") written.push(".gitignore (patched)");
147
- else if (gitignoreResult === "created") written.push(".gitignore (created)");
148
- else skipped.push(".gitignore (entry already present)");
42
+ if (gitignoreResult === "patched") written.push(".gitignore (patched with .fb/)");
43
+ else if (gitignoreResult === "created") written.push(".gitignore (created with .fb/)");
149
44
  await setupContextMode(homeDir);
150
45
  setupMcp2cli(homeDir);
151
46
  setupContext7();
152
- console.log("\nFiles written:");
153
- for (const f of written) console.log(` ✓ ${f}`);
154
- if (skipped.length > 0) {
155
- console.log("Skipped:");
156
- for (const f of skipped) console.log(` - ${f}`);
47
+ if (written.length > 0) {
48
+ console.log("\nFiles written:");
49
+ for (const f of written) console.log(` ✓ ${f}`);
157
50
  }
158
51
  console.log("\nReady. Run: frappe-builder\n");
159
52
  }
160
- function printCancelled() {
161
- console.log("\nSetup cancelled. No files were written.\n");
162
- }
163
53
  /**
164
54
  * Installs and configures the context-mode pi MCP extension.
165
55
  * Clones https://github.com/mksglu/context-mode into ~/.pi/extensions/context-mode,
@@ -3,7 +3,6 @@ import type { TextContent } from "@mariozechner/pi-ai";
3
3
  import { db, getCurrentPhase } from "../state/db.js";
4
4
  import { appendEntry } from "../state/journal.js";
5
5
  import { buildStateContext, loadSpecialist, loadDebuggerSpecialist } from "./frappe-session.js";
6
- import { loadCredentials } from "../config/loader.js";
7
6
 
8
7
  // eslint-disable-next-line @typescript-eslint/no-explicit-any
9
8
  export default function (pi: any) {
@@ -110,18 +109,6 @@ function applyPhaseTransition(featureId: string, newPhase: string): void {
110
109
  })();
111
110
  }
112
111
 
113
- /**
114
- * Session start hook — validates gitignore and loads credentials before any session setup.
115
- * Throws with AC error message if .frappe-builder-config.json is not gitignored.
116
- * Call this early in the session_start lifecycle; on throw, halt and surface the message.
117
- * projectRoot defaults to process.cwd() when not supplied by the pi extension system.
118
- *
119
- * TODO: wire this into the pi extension system's session_start event once the
120
- * session_start hook type is confirmed in @mariozechner/pi-agent-core.
121
- */
122
- export function handleSessionStart(projectRoot: string = process.cwd()): void {
123
- loadCredentials(projectRoot);
124
- }
125
112
 
126
113
  /**
127
114
  * afterToolCall hook — appends a JSONL journal entry and updates the sessions
@@ -82,6 +82,9 @@ export default function (pi: any) {
82
82
  parameters: Type.Object({
83
83
  projectId: Type.String({ description: "Project identifier (e.g. 'my-frappe-site')" }),
84
84
  sitePath: Type.String({ description: "Absolute path to the Frappe bench site (e.g. '/home/user/frappe-bench/sites/site1.local')" }),
85
+ siteUrl: Type.Optional(Type.String({ description: "Frappe site URL (e.g. 'http://site1.localhost'). Required for frappe_query." })),
86
+ apiKey: Type.Optional(Type.String({ description: "Frappe API key from User > API Access. Required for frappe_query." })),
87
+ apiSecret: Type.Optional(Type.String({ description: "Frappe API secret from User > API Access. Required for frappe_query." })),
85
88
  }),
86
89
  execute: async (_toolCallId: string, params: ToolParams) => {
87
90
  const result = await setActiveProject(params);
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "frappe-builder",
3
- "version": "1.1.0-dev.24",
3
+ "version": "1.1.0-dev.25",
4
4
  "description": "Frappe-native AI co-pilot for building and customising Frappe/ERPNext applications",
5
5
  "type": "module",
6
6
  "bin": {
package/state/db.ts CHANGED
@@ -27,7 +27,16 @@ export function setDb(instance: DatabaseType): void {
27
27
  *
28
28
  * sitePath and appPath are optional so existing callers without them continue to work.
29
29
  */
30
- export function switchProject(newProjectId: string, sitePath?: string, appPath?: string): void {
30
+ export interface ProjectCredentials {
31
+ sitePath?: string;
32
+ appPath?: string;
33
+ siteUrl?: string;
34
+ apiKey?: string;
35
+ apiSecret?: string;
36
+ }
37
+
38
+ export function switchProject(newProjectId: string, creds: ProjectCredentials = {}): void {
39
+ const { sitePath, appPath, siteUrl, apiKey, apiSecret } = creds;
31
40
  db.transaction(() => {
32
41
  // 1. Read current active session before closing
33
42
  const current = db
@@ -63,13 +72,16 @@ export function switchProject(newProjectId: string, sitePath?: string, appPath?:
63
72
 
64
73
  // 5. Create new session, restoring prior phase if available; feature_id defaults to NULL
65
74
  db.prepare(
66
- "INSERT INTO sessions (session_id, project_id, current_phase, site_path, app_path, started_at, is_active) VALUES (?, ?, ?, ?, ?, ?, 1)"
75
+ "INSERT INTO sessions (session_id, project_id, current_phase, site_path, app_path, site_url, api_key, api_secret, started_at, is_active) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, 1)"
67
76
  ).run(
68
77
  crypto.randomUUID(),
69
78
  newProjectId,
70
79
  prior?.current_phase ?? "idle",
71
80
  sitePath ?? null,
72
81
  appPath ?? null,
82
+ siteUrl ?? null,
83
+ apiKey ?? null,
84
+ apiSecret ?? null,
73
85
  new Date().toISOString()
74
86
  );
75
87
  })();
package/state/schema.ts CHANGED
@@ -36,6 +36,9 @@ export function initSchema(db: Database): void {
36
36
  current_phase TEXT NOT NULL DEFAULT 'idle',
37
37
  site_path TEXT,
38
38
  app_path TEXT,
39
+ site_url TEXT,
40
+ api_key TEXT,
41
+ api_secret TEXT,
39
42
  chain_step TEXT,
40
43
  chain_pid INTEGER,
41
44
  feature_id TEXT,
@@ -62,6 +65,9 @@ export function migrateSchema(db: Database): void {
62
65
  "ALTER TABLE sessions ADD COLUMN app_path TEXT",
63
66
  "ALTER TABLE sessions ADD COLUMN chain_step TEXT",
64
67
  "ALTER TABLE sessions ADD COLUMN chain_pid INTEGER",
68
+ "ALTER TABLE sessions ADD COLUMN site_url TEXT",
69
+ "ALTER TABLE sessions ADD COLUMN api_key TEXT",
70
+ "ALTER TABLE sessions ADD COLUMN api_secret TEXT",
65
71
  ];
66
72
  for (const sql of alters) {
67
73
  try { db.exec(sql); } catch { /* column already exists — safe to ignore */ }
@@ -1,6 +1,5 @@
1
- import { execa } from "execa";
2
- import { loadConfig } from "../config/loader.js";
3
- import { routeThroughContextMode } from "./context-sandbox.js";
1
+ import { db } from "../state/db.js";
2
+ import { applyTruncationFallback } from "./context-sandbox.js";
4
3
 
5
4
  export interface FrappeQueryArgs {
6
5
  doctype: string;
@@ -12,33 +11,50 @@ export interface FrappeQueryResult {
12
11
  error?: string;
13
12
  }
14
13
 
14
+ interface SessionCredentials {
15
+ site_url: string | null;
16
+ api_key: string | null;
17
+ api_secret: string | null;
18
+ }
19
+
15
20
  /**
16
- * Dispatches a Frappe get_list query via mcp2cli subprocess.
17
- * Raw output is routed through context-mode sandbox — never returned directly.
21
+ * Queries Frappe data via direct REST API call using credentials stored in the
22
+ * active session (set via set_active_project).
23
+ * Output is truncation-guarded — never returns raw payloads over 8K tokens.
18
24
  * Returns structured error on failure, never throws.
19
- *
20
- * Note: mcp2cli command syntax uses --mcp <url> --raw <tool_name> <json_args>.
21
- * Adapt args array if actual mcp2cli CLI syntax differs.
22
25
  */
23
26
  export async function frappeQuery({ doctype, filters }: FrappeQueryArgs): Promise<FrappeQueryResult> {
24
- const config = loadConfig();
27
+ const session = db
28
+ .prepare("SELECT site_url, api_key, api_secret FROM sessions WHERE is_active = 1 LIMIT 1")
29
+ .get() as SessionCredentials | undefined;
25
30
 
26
- if (!config.frappeMcpUrl) {
27
- return { error: "frappeMcpUrl not configured in ~/.frappe-builder/config.json" };
31
+ if (!session?.site_url) {
32
+ return { error: "No site_url configured. Call set_active_project with siteUrl, apiKey, and apiSecret first." };
33
+ }
34
+ if (!session.api_key || !session.api_secret) {
35
+ return { error: "No API credentials configured. Call set_active_project with apiKey and apiSecret." };
28
36
  }
29
37
 
30
38
  try {
31
- const payload = JSON.stringify({ doctype, filters: filters ?? {} });
32
- const { stdout, stderr } = await execa("mcp2cli", [
33
- "--mcp", config.frappeMcpUrl,
34
- "--raw",
35
- "frappe_get_list",
36
- payload,
37
- ]);
39
+ const params = new URLSearchParams();
40
+ if (filters && Object.keys(filters).length > 0) {
41
+ params.set("filters", JSON.stringify(filters));
42
+ }
43
+
44
+ const url = `${session.site_url}/api/resource/${encodeURIComponent(doctype)}?${params.toString()}`;
45
+ const response = await fetch(url, {
46
+ headers: {
47
+ Authorization: `token ${session.api_key}:${session.api_secret}`,
48
+ "Content-Type": "application/json",
49
+ },
50
+ });
38
51
 
39
- if (stderr) console.warn(`[frappe_query mcp2cli stderr: ${stderr}]`);
52
+ if (!response.ok) {
53
+ return { error: `Frappe API error ${response.status}: ${response.statusText}` };
54
+ }
40
55
 
41
- const summary = await routeThroughContextMode(stdout);
56
+ const raw = await response.text();
57
+ const summary = applyTruncationFallback(raw);
42
58
  return { summary };
43
59
  } catch (err) {
44
60
  const msg = err instanceof Error ? err.message : String(err);
@@ -1,4 +1,4 @@
1
- import { db, switchProject } from "../state/db.js";
1
+ import { db, switchProject, type ProjectCredentials } from "../state/db.js";
2
2
  import { reloadSessionContext } from "../extensions/frappe-session.js";
3
3
 
4
4
  export interface ComponentStatus {
@@ -78,31 +78,32 @@ interface SetActiveProjectArgs {
78
78
  projectId: string;
79
79
  sitePath: string;
80
80
  appPath?: string;
81
+ siteUrl?: string;
82
+ apiKey?: string;
83
+ apiSecret?: string;
81
84
  }
82
85
 
83
86
  /**
84
87
  * Switches the active Frappe project and site, flushes current session state,
85
88
  * creates a new session, and reloads the system prompt context.
86
- *
87
- * The state_transition JSONL entry is written inside switchProject()
88
- * no second appendEntry() call here.
89
+ * Site credentials (siteUrl, apiKey, apiSecret) are stored in the session row
90
+ * and used by frappe_query for direct REST API calls.
89
91
  */
90
- export async function setActiveProject({ projectId, sitePath, appPath }: SetActiveProjectArgs) {
91
- // Flush current state + create new session (JSONL entry written internally)
92
- switchProject(projectId, sitePath, appPath);
92
+ export async function setActiveProject({ projectId, sitePath, appPath, siteUrl, apiKey, apiSecret }: SetActiveProjectArgs) {
93
+ const creds: ProjectCredentials = { sitePath, appPath, siteUrl, apiKey, apiSecret };
94
+ switchProject(projectId, creds);
93
95
 
94
- // Reload system prompt with new project context
95
96
  await reloadSessionContext();
96
97
 
97
- // Read restored phase for return value
98
98
  const session = db
99
- .prepare("SELECT current_phase FROM sessions WHERE is_active = 1 LIMIT 1")
100
- .get() as { current_phase: string } | undefined;
99
+ .prepare("SELECT current_phase, site_url FROM sessions WHERE is_active = 1 LIMIT 1")
100
+ .get() as { current_phase: string; site_url: string | null } | undefined;
101
101
 
102
102
  return {
103
103
  project_id: projectId,
104
104
  site_path: sitePath,
105
105
  app_path: appPath ?? null,
106
+ site_url: session?.site_url ?? null,
106
107
  phase: session?.current_phase ?? "idle",
107
108
  context_reloaded: true,
108
109
  };