frappe-builder 1.1.0-dev.22 → 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 +0 -0
- package/.frappe-builder/po-approval/implementation-artifacts/sprint-status.yaml +2 -2
- package/VrWyV9740LA8sENuDokCo/ssr/196abb03d76072784b62b46fb8bc3145b285d048 +318 -0
- package/VrWyV9740LA8sENuDokCo/ssr/38bf1c2ec078b4df1b452507602034ccf39ce1b1 +207 -0
- package/config/defaults.ts +0 -1
- package/config/loader.ts +18 -84
- package/dist/cli.mjs +7 -6
- package/dist/{init-BUkSYk2l.mjs → init-CkLSZ_3g.mjs} +57 -123
- package/extensions/frappe-state.ts +0 -13
- package/extensions/frappe-tools.ts +25 -3
- package/extensions/frappe-workflow.ts +1 -1
- package/package.json +1 -1
- package/state/db.ts +14 -2
- package/state/schema.ts +6 -0
- package/tools/frappe-context7.ts +28 -32
- package/tools/frappe-query-tools.ts +36 -20
- package/tools/project-tools.ts +12 -11
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:
|
|
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:
|
|
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==
|
package/config/defaults.ts
CHANGED
|
@@ -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
|