bonecode 1.1.0 → 1.2.1
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/README.md +145 -9
- package/bin/bonecode +47 -42
- package/compat/opencode_adapter.ts +188 -17
- package/dist/bone/output/agent/src/algorithms.d.ts +1 -0
- package/dist/bone/output/agent/src/algorithms.js +3 -0
- package/dist/bone/output/agent/src/algorithms.js.map +1 -0
- package/dist/bone/output/agent/src/audit.d.ts +3 -0
- package/dist/bone/output/agent/src/audit.js +40 -0
- package/dist/bone/output/agent/src/audit.js.map +1 -0
- package/dist/bone/output/agent/src/auth.d.ts +8 -0
- package/dist/bone/output/agent/src/auth.js +56 -0
- package/dist/bone/output/agent/src/auth.js.map +1 -0
- package/dist/bone/output/agent/src/db.d.ts +6 -0
- package/dist/bone/output/agent/src/db.js +63 -0
- package/dist/bone/output/agent/src/db.js.map +1 -0
- package/dist/bone/output/agent/src/events.d.ts +25 -0
- package/dist/bone/output/agent/src/events.js +184 -0
- package/dist/bone/output/agent/src/events.js.map +1 -0
- package/dist/bone/output/agent/src/logger.d.ts +28 -0
- package/dist/bone/output/agent/src/logger.js +45 -0
- package/dist/bone/output/agent/src/logger.js.map +1 -0
- package/dist/bone/output/agent/src/metrics.d.ts +5 -0
- package/dist/bone/output/agent/src/metrics.js +60 -0
- package/dist/bone/output/agent/src/metrics.js.map +1 -0
- package/dist/bone/output/agent/src/routes/agent_instance.d.ts +1 -0
- package/dist/bone/output/agent/src/routes/agent_instance.js +253 -0
- package/dist/bone/output/agent/src/routes/agent_instance.js.map +1 -0
- package/dist/bone/output/agent/src/routes/build_step.d.ts +1 -0
- package/dist/bone/output/agent/src/routes/build_step.js +133 -0
- package/dist/bone/output/agent/src/routes/build_step.js.map +1 -0
- package/dist/bone/output/agent/src/routes/plan.d.ts +1 -0
- package/dist/bone/output/agent/src/routes/plan.js +119 -0
- package/dist/bone/output/agent/src/routes/plan.js.map +1 -0
- package/dist/bone/output/agent/src/routes/task.d.ts +1 -0
- package/dist/bone/output/agent/src/routes/task.js +133 -0
- package/dist/bone/output/agent/src/routes/task.js.map +1 -0
- package/dist/bone/output/agent/src/routes/tool_call.d.ts +1 -0
- package/dist/bone/output/agent/src/routes/tool_call.js +190 -0
- package/dist/bone/output/agent/src/routes/tool_call.js.map +1 -0
- package/dist/bone/output/agent/src/state_machines/agent_instance.d.ts +9 -0
- package/dist/bone/output/agent/src/state_machines/agent_instance.js +22 -0
- package/dist/bone/output/agent/src/state_machines/agent_instance.js.map +1 -0
- package/dist/bone/output/agent/src/state_machines/build_step.d.ts +9 -0
- package/dist/bone/output/agent/src/state_machines/build_step.js +20 -0
- package/dist/bone/output/agent/src/state_machines/build_step.js.map +1 -0
- package/dist/bone/output/agent/src/state_machines/plan.d.ts +9 -0
- package/dist/bone/output/agent/src/state_machines/plan.js +20 -0
- package/dist/bone/output/agent/src/state_machines/plan.js.map +1 -0
- package/dist/bone/output/agent/src/state_machines/task.d.ts +9 -0
- package/dist/bone/output/agent/src/state_machines/task.js +20 -0
- package/dist/bone/output/agent/src/state_machines/task.js.map +1 -0
- package/dist/bone/output/agent/src/state_machines/tool_call.d.ts +9 -0
- package/dist/bone/output/agent/src/state_machines/tool_call.js +20 -0
- package/dist/bone/output/agent/src/state_machines/tool_call.js.map +1 -0
- package/dist/bone/output/rag/src/algorithms.d.ts +1 -0
- package/dist/bone/output/rag/src/algorithms.js +3 -0
- package/dist/bone/output/rag/src/algorithms.js.map +1 -0
- package/dist/bone/output/rag/src/auth.d.ts +8 -0
- package/dist/bone/output/rag/src/auth.js +56 -0
- package/dist/bone/output/rag/src/auth.js.map +1 -0
- package/dist/bone/output/rag/src/db.d.ts +6 -0
- package/dist/bone/output/rag/src/db.js +63 -0
- package/dist/bone/output/rag/src/db.js.map +1 -0
- package/dist/bone/output/rag/src/events.d.ts +25 -0
- package/dist/bone/output/rag/src/events.js +184 -0
- package/dist/bone/output/rag/src/events.js.map +1 -0
- package/dist/bone/output/rag/src/extensions.d.ts +83 -0
- package/dist/bone/output/rag/src/extensions.js +329 -0
- package/dist/bone/output/rag/src/extensions.js.map +1 -0
- package/dist/bone/output/rag/src/flows.d.ts +24 -0
- package/dist/bone/output/rag/src/flows.js +236 -0
- package/dist/bone/output/rag/src/flows.js.map +1 -0
- package/dist/bone/output/rag/src/logger.d.ts +28 -0
- package/dist/bone/output/rag/src/logger.js +45 -0
- package/dist/bone/output/rag/src/logger.js.map +1 -0
- package/dist/bone/output/rag/src/metrics.d.ts +5 -0
- package/dist/bone/output/rag/src/metrics.js +60 -0
- package/dist/bone/output/rag/src/metrics.js.map +1 -0
- package/dist/bone/output/rag/src/routes/code_chunk.d.ts +1 -0
- package/dist/bone/output/rag/src/routes/code_chunk.js +100 -0
- package/dist/bone/output/rag/src/routes/code_chunk.js.map +1 -0
- package/dist/bone/output/rag/src/routes/code_file.d.ts +1 -0
- package/dist/bone/output/rag/src/routes/code_file.js +127 -0
- package/dist/bone/output/rag/src/routes/code_file.js.map +1 -0
- package/dist/bone/output/rag/src/routes/indexing_job.d.ts +1 -0
- package/dist/bone/output/rag/src/routes/indexing_job.js +113 -0
- package/dist/bone/output/rag/src/routes/indexing_job.js.map +1 -0
- package/dist/bone/output/rag/src/routes/knowledge_base.d.ts +1 -0
- package/dist/bone/output/rag/src/routes/knowledge_base.js +242 -0
- package/dist/bone/output/rag/src/routes/knowledge_base.js.map +1 -0
- package/dist/bone/output/rag/src/routes/memory_entry.d.ts +1 -0
- package/dist/bone/output/rag/src/routes/memory_entry.js +113 -0
- package/dist/bone/output/rag/src/routes/memory_entry.js.map +1 -0
- package/dist/bone/output/rag/src/state_machines/code_file.d.ts +9 -0
- package/dist/bone/output/rag/src/state_machines/code_file.js +21 -0
- package/dist/bone/output/rag/src/state_machines/code_file.js.map +1 -0
- package/dist/bone/output/rag/src/state_machines/indexing_job.d.ts +9 -0
- package/dist/bone/output/rag/src/state_machines/indexing_job.js +20 -0
- package/dist/bone/output/rag/src/state_machines/indexing_job.js.map +1 -0
- package/dist/bone/output/rag/src/state_machines/knowledge_base.d.ts +9 -0
- package/dist/bone/output/rag/src/state_machines/knowledge_base.js +21 -0
- package/dist/bone/output/rag/src/state_machines/knowledge_base.js.map +1 -0
- package/dist/bone/output/rag/src/state_machines/memory_entry.d.ts +9 -0
- package/dist/bone/output/rag/src/state_machines/memory_entry.js +18 -0
- package/dist/bone/output/rag/src/state_machines/memory_entry.js.map +1 -0
- package/dist/bone/output/session/src/algorithms.d.ts +1 -0
- package/dist/bone/output/session/src/algorithms.js +3 -0
- package/dist/bone/output/session/src/algorithms.js.map +1 -0
- package/dist/bone/output/session/src/audit.d.ts +3 -0
- package/dist/bone/output/session/src/audit.js +40 -0
- package/dist/bone/output/session/src/audit.js.map +1 -0
- package/dist/bone/output/session/src/auth.d.ts +8 -0
- package/dist/bone/output/session/src/auth.js +56 -0
- package/dist/bone/output/session/src/auth.js.map +1 -0
- package/dist/bone/output/session/src/db.d.ts +6 -0
- package/dist/bone/output/session/src/db.js +63 -0
- package/dist/bone/output/session/src/db.js.map +1 -0
- package/dist/bone/output/session/src/events.d.ts +26 -0
- package/dist/bone/output/session/src/events.js +212 -0
- package/dist/bone/output/session/src/events.js.map +1 -0
- package/dist/bone/output/session/src/extensions.d.ts +41 -0
- package/dist/bone/output/session/src/extensions.js +217 -0
- package/dist/bone/output/session/src/extensions.js.map +1 -0
- package/dist/bone/output/session/src/logger.d.ts +28 -0
- package/dist/bone/output/session/src/logger.js +44 -0
- package/dist/bone/output/session/src/logger.js.map +1 -0
- package/dist/bone/output/session/src/metrics.d.ts +5 -0
- package/dist/bone/output/session/src/metrics.js +60 -0
- package/dist/bone/output/session/src/metrics.js.map +1 -0
- package/dist/bone/output/session/src/routes/message.d.ts +1 -0
- package/dist/bone/output/session/src/routes/message.js +120 -0
- package/dist/bone/output/session/src/routes/message.js.map +1 -0
- package/dist/bone/output/session/src/routes/part.d.ts +1 -0
- package/dist/bone/output/session/src/routes/part.js +106 -0
- package/dist/bone/output/session/src/routes/part.js.map +1 -0
- package/dist/bone/output/session/src/routes/permission.d.ts +1 -0
- package/dist/bone/output/session/src/routes/permission.js +106 -0
- package/dist/bone/output/session/src/routes/permission.js.map +1 -0
- package/dist/bone/output/session/src/routes/project.d.ts +1 -0
- package/dist/bone/output/session/src/routes/project.js +106 -0
- package/dist/bone/output/session/src/routes/project.js.map +1 -0
- package/dist/bone/output/session/src/routes/session.d.ts +1 -0
- package/dist/bone/output/session/src/routes/session.js +308 -0
- package/dist/bone/output/session/src/routes/session.js.map +1 -0
- package/dist/bone/output/session/src/state_machines/session.d.ts +9 -0
- package/dist/bone/output/session/src/state_machines/session.js +21 -0
- package/dist/bone/output/session/src/state_machines/session.js.map +1 -0
- package/dist/bone/output/session/src/websocket.d.ts +15 -0
- package/dist/bone/output/session/src/websocket.js +215 -0
- package/dist/bone/output/session/src/websocket.js.map +1 -0
- package/dist/bone/output/workspace/src/algorithms.d.ts +1 -0
- package/dist/bone/output/workspace/src/algorithms.js +3 -0
- package/dist/bone/output/workspace/src/algorithms.js.map +1 -0
- package/dist/bone/output/workspace/src/auth.d.ts +8 -0
- package/dist/bone/output/workspace/src/auth.js +56 -0
- package/dist/bone/output/workspace/src/auth.js.map +1 -0
- package/dist/bone/output/workspace/src/db.d.ts +6 -0
- package/dist/bone/output/workspace/src/db.js +63 -0
- package/dist/bone/output/workspace/src/db.js.map +1 -0
- package/dist/bone/output/workspace/src/events.d.ts +25 -0
- package/dist/bone/output/workspace/src/events.js +184 -0
- package/dist/bone/output/workspace/src/events.js.map +1 -0
- package/dist/bone/output/workspace/src/logger.d.ts +28 -0
- package/dist/bone/output/workspace/src/logger.js +45 -0
- package/dist/bone/output/workspace/src/logger.js.map +1 -0
- package/dist/bone/output/workspace/src/metrics.d.ts +5 -0
- package/dist/bone/output/workspace/src/metrics.js +60 -0
- package/dist/bone/output/workspace/src/metrics.js.map +1 -0
- package/dist/bone/output/workspace/src/routes/codebase.d.ts +1 -0
- package/dist/bone/output/workspace/src/routes/codebase.js +113 -0
- package/dist/bone/output/workspace/src/routes/codebase.js.map +1 -0
- package/dist/bone/output/workspace/src/routes/snapshot.d.ts +1 -0
- package/dist/bone/output/workspace/src/routes/snapshot.js +151 -0
- package/dist/bone/output/workspace/src/routes/snapshot.js.map +1 -0
- package/dist/bone/output/workspace/src/routes/workspace.d.ts +1 -0
- package/dist/bone/output/workspace/src/routes/workspace.js +209 -0
- package/dist/bone/output/workspace/src/routes/workspace.js.map +1 -0
- package/dist/bone/output/workspace/src/state_machines/codebase.d.ts +9 -0
- package/dist/bone/output/workspace/src/state_machines/codebase.js +19 -0
- package/dist/bone/output/workspace/src/state_machines/codebase.js.map +1 -0
- package/dist/bone/output/workspace/src/state_machines/snapshot.d.ts +9 -0
- package/dist/bone/output/workspace/src/state_machines/snapshot.js +18 -0
- package/dist/bone/output/workspace/src/state_machines/snapshot.js.map +1 -0
- package/dist/bone/output/workspace/src/state_machines/workspace.d.ts +9 -0
- package/dist/bone/output/workspace/src/state_machines/workspace.js +19 -0
- package/dist/bone/output/workspace/src/state_machines/workspace.js.map +1 -0
- package/dist/compat/opencode_adapter.d.ts +25 -0
- package/dist/compat/opencode_adapter.js +599 -0
- package/dist/compat/opencode_adapter.js.map +1 -0
- package/dist/extensions/chunker.d.ts +24 -0
- package/dist/extensions/chunker.js +360 -0
- package/dist/extensions/chunker.js.map +1 -0
- package/dist/extensions/embedding_provider.d.ts +18 -0
- package/dist/extensions/embedding_provider.js +150 -0
- package/dist/extensions/embedding_provider.js.map +1 -0
- package/dist/extensions/llm_provider.d.ts +33 -0
- package/dist/extensions/llm_provider.js +338 -0
- package/dist/extensions/llm_provider.js.map +1 -0
- package/dist/extensions/mcp_bridge.d.ts +44 -0
- package/dist/extensions/mcp_bridge.js +151 -0
- package/dist/extensions/mcp_bridge.js.map +1 -0
- package/dist/extensions/rag_search.d.ts +38 -0
- package/dist/extensions/rag_search.js +242 -0
- package/dist/extensions/rag_search.js.map +1 -0
- package/dist/extensions/snapshot.d.ts +14 -0
- package/dist/extensions/snapshot.js +158 -0
- package/dist/extensions/snapshot.js.map +1 -0
- package/dist/extensions/tool_executor.d.ts +28 -0
- package/dist/extensions/tool_executor.js +268 -0
- package/dist/extensions/tool_executor.js.map +1 -0
- package/dist/src/cli.d.ts +15 -0
- package/dist/src/cli.js +687 -0
- package/dist/src/cli.js.map +1 -0
- package/dist/src/config.d.ts +44 -0
- package/dist/src/config.js +165 -0
- package/dist/src/config.js.map +1 -0
- package/dist/src/context_builder.d.ts +51 -0
- package/dist/src/context_builder.js +558 -0
- package/dist/src/context_builder.js.map +1 -0
- package/dist/src/db_adapter.d.ts +24 -0
- package/dist/src/db_adapter.js +341 -0
- package/dist/src/db_adapter.js.map +1 -0
- package/dist/src/engine/session/compaction_logic.d.ts +11 -0
- package/dist/src/engine/session/compaction_logic.js +113 -0
- package/dist/src/engine/session/compaction_logic.js.map +1 -0
- package/dist/src/engine/session/instruction_loader.d.ts +5 -0
- package/dist/src/engine/session/instruction_loader.js +78 -0
- package/dist/src/engine/session/instruction_loader.js.map +1 -0
- package/dist/src/engine/session/overflow_check.d.ts +14 -0
- package/dist/src/engine/session/overflow_check.js +45 -0
- package/dist/src/engine/session/overflow_check.js.map +1 -0
- package/dist/src/engine/session/prompt.d.ts +45 -0
- package/dist/src/engine/session/prompt.js +584 -0
- package/dist/src/engine/session/prompt.js.map +1 -0
- package/dist/src/engine/session/provider_transform.d.ts +59 -0
- package/dist/src/engine/session/provider_transform.js +193 -0
- package/dist/src/engine/session/provider_transform.js.map +1 -0
- package/dist/src/engine/session/retry_logic.d.ts +12 -0
- package/dist/src/engine/session/retry_logic.js +72 -0
- package/dist/src/engine/session/retry_logic.js.map +1 -0
- package/dist/src/engine/session/system_prompt.d.ts +9 -0
- package/dist/src/engine/session/system_prompt.js +96 -0
- package/dist/src/engine/session/system_prompt.js.map +1 -0
- package/dist/src/engine/session/tool_registry.d.ts +5 -0
- package/dist/src/engine/session/tool_registry.js +117 -0
- package/dist/src/engine/session/tool_registry.js.map +1 -0
- package/dist/src/export.d.ts +13 -0
- package/dist/src/export.js +103 -0
- package/dist/src/export.js.map +1 -0
- package/dist/src/mdns.d.ts +7 -0
- package/dist/src/mdns.js +60 -0
- package/dist/src/mdns.js.map +1 -0
- package/dist/src/rag_worker.d.ts +38 -0
- package/dist/src/rag_worker.js +435 -0
- package/dist/src/rag_worker.js.map +1 -0
- package/dist/src/server.d.ts +11 -0
- package/dist/src/server.js +214 -0
- package/dist/src/server.js.map +1 -0
- package/dist/src/stats.d.ts +45 -0
- package/dist/src/stats.js +233 -0
- package/dist/src/stats.js.map +1 -0
- package/dist/src/tui.d.ts +29 -0
- package/dist/src/tui.js +1053 -0
- package/dist/src/tui.js.map +1 -0
- package/package.json +7 -4
- package/src/cli.ts +247 -5
- package/src/export.ts +122 -0
- package/src/mdns.ts +53 -0
- package/src/server.ts +32 -0
- package/src/stats.ts +290 -0
- package/src/tui.ts +749 -248
package/dist/src/cli.js
ADDED
|
@@ -0,0 +1,687 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
"use strict";
|
|
3
|
+
/**
|
|
4
|
+
* BoneCode CLI
|
|
5
|
+
*
|
|
6
|
+
* Usage:
|
|
7
|
+
* bonecode Start server + open TUI (default)
|
|
8
|
+
* bonecode serve Start server only (headless)
|
|
9
|
+
* bonecode run "fix the login bug" Send a one-shot prompt, stream output
|
|
10
|
+
* bonecode run -i Interactive mode (stream to terminal)
|
|
11
|
+
* bonecode compile Recompile .bone files
|
|
12
|
+
* bonecode migrate Run database migrations
|
|
13
|
+
* bonecode status Show server status
|
|
14
|
+
* bonecode --help Show help
|
|
15
|
+
*/
|
|
16
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
17
|
+
if (k2 === undefined) k2 = k;
|
|
18
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
19
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
20
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
21
|
+
}
|
|
22
|
+
Object.defineProperty(o, k2, desc);
|
|
23
|
+
}) : (function(o, m, k, k2) {
|
|
24
|
+
if (k2 === undefined) k2 = k;
|
|
25
|
+
o[k2] = m[k];
|
|
26
|
+
}));
|
|
27
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
28
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
29
|
+
}) : function(o, v) {
|
|
30
|
+
o["default"] = v;
|
|
31
|
+
});
|
|
32
|
+
var __importStar = (this && this.__importStar) || function (mod) {
|
|
33
|
+
if (mod && mod.__esModule) return mod;
|
|
34
|
+
var result = {};
|
|
35
|
+
if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
|
|
36
|
+
__setModuleDefault(result, mod);
|
|
37
|
+
return result;
|
|
38
|
+
};
|
|
39
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
40
|
+
require("dotenv/config");
|
|
41
|
+
const path = __importStar(require("path"));
|
|
42
|
+
const fs = __importStar(require("fs"));
|
|
43
|
+
const http = __importStar(require("http"));
|
|
44
|
+
const child_process_1 = require("child_process");
|
|
45
|
+
const PKG_ROOT = (() => {
|
|
46
|
+
// When running from compiled JS: __dirname = dist/src/ → go up 2 levels
|
|
47
|
+
// When running from source via ts-node: __dirname = src/ → go up 1 level
|
|
48
|
+
const fromSrc = path.resolve(__dirname, "..");
|
|
49
|
+
const fromDist = path.resolve(__dirname, "..", "..");
|
|
50
|
+
// Whichever has package.json is the real root
|
|
51
|
+
if (fs.existsSync(path.join(fromDist, "package.json")))
|
|
52
|
+
return fromDist;
|
|
53
|
+
return fromSrc;
|
|
54
|
+
})();
|
|
55
|
+
// Also load .env from package root when running globally (dotenv/config only loads from cwd)
|
|
56
|
+
const pkgEnv = path.join(PKG_ROOT, ".env");
|
|
57
|
+
if (pkgEnv !== path.join(process.cwd(), ".env") && fs.existsSync(pkgEnv)) {
|
|
58
|
+
require("dotenv").config({ path: pkgEnv, override: false });
|
|
59
|
+
}
|
|
60
|
+
const VERSION = JSON.parse(fs.readFileSync(path.join(PKG_ROOT, "package.json"), "utf-8")).version;
|
|
61
|
+
// ─── ANSI helpers ─────────────────────────────────────────────────────────────
|
|
62
|
+
const isTTY = process.stdout.isTTY;
|
|
63
|
+
const c = {
|
|
64
|
+
reset: isTTY ? "\x1b[0m" : "",
|
|
65
|
+
bold: isTTY ? "\x1b[1m" : "",
|
|
66
|
+
dim: isTTY ? "\x1b[2m" : "",
|
|
67
|
+
cyan: isTTY ? "\x1b[36m" : "",
|
|
68
|
+
green: isTTY ? "\x1b[32m" : "",
|
|
69
|
+
yellow: isTTY ? "\x1b[33m" : "",
|
|
70
|
+
red: isTTY ? "\x1b[31m" : "",
|
|
71
|
+
blue: isTTY ? "\x1b[34m" : "",
|
|
72
|
+
magenta: isTTY ? "\x1b[35m" : "",
|
|
73
|
+
};
|
|
74
|
+
function logo() {
|
|
75
|
+
console.log(`
|
|
76
|
+
${c.cyan}${c.bold} ██████╗ ██████╗ ███╗ ██╗███████╗${c.reset}
|
|
77
|
+
${c.cyan}${c.bold} ██╔══██╗██╔═══██╗████╗ ██║██╔════╝${c.reset}
|
|
78
|
+
${c.cyan}${c.bold} ██████╔╝██║ ██║██╔██╗ ██║█████╗ ${c.reset}
|
|
79
|
+
${c.cyan}${c.bold} ██╔══██╗██║ ██║██║╚██╗██║██╔══╝ ${c.reset}
|
|
80
|
+
${c.cyan}${c.bold} ██████╔╝╚██████╔╝██║ ╚████║███████╗${c.reset}
|
|
81
|
+
${c.cyan}${c.bold} ╚═════╝ ╚═════╝ ╚═╝ ╚═══╝╚══════╝${c.reset}
|
|
82
|
+
${c.dim} BoneCode v${VERSION} — AI coding agent${c.reset}
|
|
83
|
+
`);
|
|
84
|
+
}
|
|
85
|
+
function help() {
|
|
86
|
+
logo();
|
|
87
|
+
console.log(`${c.bold}Usage:${c.reset}
|
|
88
|
+
${c.cyan}bonecode${c.reset} Start server and open interactive session
|
|
89
|
+
${c.cyan}bonecode serve${c.reset} Start headless server only
|
|
90
|
+
${c.cyan}bonecode run${c.reset} ${c.dim}[message]${c.reset} Send a prompt and stream the response
|
|
91
|
+
${c.cyan}bonecode run -i${c.reset} Interactive terminal mode
|
|
92
|
+
${c.cyan}bonecode web${c.reset} Start server and open web UI in browser
|
|
93
|
+
${c.cyan}bonecode stats${c.reset} Show token usage and cost statistics
|
|
94
|
+
${c.cyan}bonecode export${c.reset} ${c.dim}[sessionID]${c.reset} Export session as JSON
|
|
95
|
+
${c.cyan}bonecode pr${c.reset} ${c.dim}<number>${c.reset} Checkout a GitHub PR branch
|
|
96
|
+
${c.cyan}bonecode github${c.reset} GitHub Actions agent management
|
|
97
|
+
${c.cyan}bonecode compile${c.reset} Recompile .bone domain models
|
|
98
|
+
${c.cyan}bonecode migrate${c.reset} Run database migrations
|
|
99
|
+
${c.cyan}bonecode status${c.reset} Check if server is running
|
|
100
|
+
${c.cyan}bonecode --version${c.reset} Show version
|
|
101
|
+
|
|
102
|
+
${c.bold}Options for ${c.cyan}run${c.reset}${c.bold}:${c.reset}
|
|
103
|
+
${c.dim}-m, --model${c.reset} Model to use (e.g. claude-sonnet-4-5, gpt-4o)
|
|
104
|
+
${c.dim}-s, --session${c.reset} Continue a specific session ID
|
|
105
|
+
${c.dim}-c, --continue${c.reset} Continue the last session
|
|
106
|
+
${c.dim}--dir${c.reset} Working directory (default: current directory)
|
|
107
|
+
${c.dim}--port${c.reset} Server port (default: 3000)
|
|
108
|
+
${c.dim}-i, --interactive${c.reset} Stream output interactively
|
|
109
|
+
|
|
110
|
+
${c.bold}Options for ${c.cyan}stats${c.reset}${c.bold}:${c.reset}
|
|
111
|
+
${c.dim}--days N${c.reset} Show stats for last N days (default: all time)
|
|
112
|
+
${c.dim}--tools N${c.reset} Show top N tools (default: all)
|
|
113
|
+
${c.dim}--models N${c.reset} Show top N models (default: hidden)
|
|
114
|
+
${c.dim}--project ID${c.reset} Filter by project ID
|
|
115
|
+
|
|
116
|
+
${c.bold}Options for ${c.cyan}export${c.reset}${c.bold}:${c.reset}
|
|
117
|
+
${c.dim}--sanitize${c.reset} Redact sensitive data (file paths, tool outputs, etc.)
|
|
118
|
+
|
|
119
|
+
${c.bold}Environment:${c.reset}
|
|
120
|
+
DATABASE_URL PostgreSQL connection string
|
|
121
|
+
JWT_SECRET JWT signing secret
|
|
122
|
+
DEFAULT_MODEL Default LLM model
|
|
123
|
+
DEFAULT_PROVIDER Default LLM provider
|
|
124
|
+
PORT Server port (default: 3000)
|
|
125
|
+
MDNS Set to 'false' to disable mDNS discovery
|
|
126
|
+
MDNS_DOMAIN Custom mDNS domain (default: bonecode.local)
|
|
127
|
+
|
|
128
|
+
${c.bold}Examples:${c.reset}
|
|
129
|
+
${c.dim}# Start the server and open a session in the browser${c.reset}
|
|
130
|
+
bonecode
|
|
131
|
+
|
|
132
|
+
${c.dim}# Open the web UI${c.reset}
|
|
133
|
+
bonecode web
|
|
134
|
+
|
|
135
|
+
${c.dim}# Send a one-shot prompt${c.reset}
|
|
136
|
+
bonecode run "fix the TypeScript errors in src/auth.ts"
|
|
137
|
+
|
|
138
|
+
${c.dim}# Show cost and token stats${c.reset}
|
|
139
|
+
bonecode stats --days 7 --models 5
|
|
140
|
+
|
|
141
|
+
${c.dim}# Export a session${c.reset}
|
|
142
|
+
bonecode export --sanitize > session.json
|
|
143
|
+
|
|
144
|
+
${c.dim}# Checkout a GitHub PR${c.reset}
|
|
145
|
+
bonecode pr 42
|
|
146
|
+
`);
|
|
147
|
+
}
|
|
148
|
+
// ─── Server health check ──────────────────────────────────────────────────────
|
|
149
|
+
function checkServer(port) {
|
|
150
|
+
return new Promise((resolve) => {
|
|
151
|
+
const req = http.get(`http://localhost:${port}/health`, (res) => {
|
|
152
|
+
resolve(res.statusCode === 200);
|
|
153
|
+
});
|
|
154
|
+
req.on("error", () => resolve(false));
|
|
155
|
+
req.setTimeout(2000, () => { req.destroy(); resolve(false); });
|
|
156
|
+
});
|
|
157
|
+
}
|
|
158
|
+
// ─── Ensure server is running (in-process) ───────────────────────────────────
|
|
159
|
+
let serverStarted = false;
|
|
160
|
+
async function ensureServer(port) {
|
|
161
|
+
if (await checkServer(port))
|
|
162
|
+
return;
|
|
163
|
+
if (serverStarted)
|
|
164
|
+
return; // already starting
|
|
165
|
+
serverStarted = true;
|
|
166
|
+
await Promise.resolve().then(() => __importStar(require("./server")));
|
|
167
|
+
let attempts = 0;
|
|
168
|
+
while (attempts < 30) {
|
|
169
|
+
if (await checkServer(port))
|
|
170
|
+
return;
|
|
171
|
+
await new Promise(r => setTimeout(r, 500));
|
|
172
|
+
attempts++;
|
|
173
|
+
}
|
|
174
|
+
throw new Error("Server failed to start");
|
|
175
|
+
}
|
|
176
|
+
// ─── Commands ─────────────────────────────────────────────────────────────────
|
|
177
|
+
async function cmdServe(args) {
|
|
178
|
+
const port = parseInt(args.find(a => a.startsWith("--port="))?.split("=")[1] || process.env.PORT || "3000");
|
|
179
|
+
process.env.PORT = String(port);
|
|
180
|
+
console.log(`${c.cyan}${c.bold}BoneCode${c.reset} server on port ${port} — press Ctrl+C to stop\n`);
|
|
181
|
+
// Start in-process and block forever (until SIGINT/SIGTERM)
|
|
182
|
+
await Promise.resolve().then(() => __importStar(require("./server")));
|
|
183
|
+
await new Promise(() => { }); // keep alive
|
|
184
|
+
}
|
|
185
|
+
async function cmdRun(args) {
|
|
186
|
+
// Parse flags
|
|
187
|
+
let message = "";
|
|
188
|
+
let model = process.env.DEFAULT_MODEL;
|
|
189
|
+
let provider = process.env.DEFAULT_PROVIDER || "openai_compatible";
|
|
190
|
+
let sessionId;
|
|
191
|
+
let continueSession = false;
|
|
192
|
+
let interactive = false;
|
|
193
|
+
let port = parseInt(process.env.PORT || "3000");
|
|
194
|
+
let dir = process.cwd();
|
|
195
|
+
const positional = [];
|
|
196
|
+
for (let i = 0; i < args.length; i++) {
|
|
197
|
+
const a = args[i];
|
|
198
|
+
if (a === "-m" || a === "--model") {
|
|
199
|
+
const v = args[++i];
|
|
200
|
+
if (v?.includes("/")) {
|
|
201
|
+
const [p, ...m] = v.split("/");
|
|
202
|
+
provider = p;
|
|
203
|
+
model = m.join("/");
|
|
204
|
+
}
|
|
205
|
+
else {
|
|
206
|
+
model = v;
|
|
207
|
+
}
|
|
208
|
+
}
|
|
209
|
+
else if (a.startsWith("--model=")) {
|
|
210
|
+
const v = a.slice(8);
|
|
211
|
+
if (v.includes("/")) {
|
|
212
|
+
const [p, ...m] = v.split("/");
|
|
213
|
+
provider = p;
|
|
214
|
+
model = m.join("/");
|
|
215
|
+
}
|
|
216
|
+
else {
|
|
217
|
+
model = v;
|
|
218
|
+
}
|
|
219
|
+
}
|
|
220
|
+
else if (a === "-s" || a === "--session") {
|
|
221
|
+
sessionId = args[++i];
|
|
222
|
+
}
|
|
223
|
+
else if (a === "-c" || a === "--continue") {
|
|
224
|
+
continueSession = true;
|
|
225
|
+
}
|
|
226
|
+
else if (a === "-i" || a === "--interactive") {
|
|
227
|
+
interactive = true;
|
|
228
|
+
}
|
|
229
|
+
else if (a === "--port") {
|
|
230
|
+
port = parseInt(args[++i]);
|
|
231
|
+
}
|
|
232
|
+
else if (a.startsWith("--port=")) {
|
|
233
|
+
port = parseInt(a.slice(7));
|
|
234
|
+
}
|
|
235
|
+
else if (a === "--dir") {
|
|
236
|
+
dir = path.resolve(args[++i]);
|
|
237
|
+
}
|
|
238
|
+
else if (!a.startsWith("-")) {
|
|
239
|
+
positional.push(a);
|
|
240
|
+
}
|
|
241
|
+
}
|
|
242
|
+
message = positional.join(" ");
|
|
243
|
+
if (!message && !interactive && !continueSession && !sessionId) {
|
|
244
|
+
console.error(`${c.red}Error:${c.reset} provide a message or use ${c.cyan}-i${c.reset} for interactive mode`);
|
|
245
|
+
console.error(` Example: bonecode run "fix the login bug"`);
|
|
246
|
+
process.exit(1);
|
|
247
|
+
}
|
|
248
|
+
await ensureServer(port);
|
|
249
|
+
// Get JWT token for API calls
|
|
250
|
+
const jwt = require("jsonwebtoken");
|
|
251
|
+
const secret = process.env.JWT_SECRET || "bonecode-dev-secret-do-not-use-in-production-at-all-ever";
|
|
252
|
+
const token = jwt.sign({ sub: "cli-user" }, secret, { expiresIn: "1h" });
|
|
253
|
+
const headers = {
|
|
254
|
+
"Content-Type": "application/json",
|
|
255
|
+
"Authorization": `Bearer ${token}`,
|
|
256
|
+
};
|
|
257
|
+
const baseUrl = `http://localhost:${port}`;
|
|
258
|
+
// Resolve or create session
|
|
259
|
+
let sid = sessionId;
|
|
260
|
+
if (!sid && continueSession) {
|
|
261
|
+
const r = await fetch(`${baseUrl}/v2/session`, { headers });
|
|
262
|
+
const sessions = await r.json();
|
|
263
|
+
sid = sessions.find((s) => !s.parentID)?.id;
|
|
264
|
+
}
|
|
265
|
+
if (!sid) {
|
|
266
|
+
const r = await fetch(`${baseUrl}/v2/session`, {
|
|
267
|
+
method: "POST", headers,
|
|
268
|
+
body: JSON.stringify({ title: message.slice(0, 60) || "Interactive session", directory: dir }),
|
|
269
|
+
});
|
|
270
|
+
const sess = await r.json();
|
|
271
|
+
sid = sess.id;
|
|
272
|
+
if (!sid) {
|
|
273
|
+
console.error(`${c.red}Failed to create session${c.reset}`);
|
|
274
|
+
process.exit(1);
|
|
275
|
+
}
|
|
276
|
+
}
|
|
277
|
+
if (!message && interactive) {
|
|
278
|
+
// Interactive TUI mode
|
|
279
|
+
const jwtLib = require("jsonwebtoken");
|
|
280
|
+
const secret = process.env.JWT_SECRET || "bonecode-dev-secret-do-not-use-in-production-at-all-ever";
|
|
281
|
+
const token = jwtLib.sign({ sub: "tui-user" }, secret, { expiresIn: "24h" });
|
|
282
|
+
const { startInteractiveTUI } = await Promise.resolve().then(() => __importStar(require("./tui")));
|
|
283
|
+
process.env.BONECODE_TUI = "1";
|
|
284
|
+
await startInteractiveTUI({ port, token, model: model, provider, worktree: dir });
|
|
285
|
+
return;
|
|
286
|
+
}
|
|
287
|
+
// One-shot prompt with streaming output
|
|
288
|
+
console.log(`${c.dim}Session: ${sid}${c.reset}`);
|
|
289
|
+
console.log(`${c.dim}Model: ${provider}/${model}${c.reset}\n`);
|
|
290
|
+
await streamPrompt(baseUrl, sid, message, model, provider, headers, interactive);
|
|
291
|
+
}
|
|
292
|
+
async function streamPrompt(baseUrl, sessionId, message, model, provider, headers, interactive) {
|
|
293
|
+
const r = await fetch(`${baseUrl}/v2/session/${sessionId}/prompt`, {
|
|
294
|
+
method: "POST", headers,
|
|
295
|
+
body: JSON.stringify({ content: message, modelID: model, providerID: provider }),
|
|
296
|
+
});
|
|
297
|
+
if (!r.ok) {
|
|
298
|
+
console.error(`${c.red}Error:${c.reset} ${await r.text()}`);
|
|
299
|
+
process.exit(1);
|
|
300
|
+
}
|
|
301
|
+
// Stream SSE events
|
|
302
|
+
const reader = r.body.getReader();
|
|
303
|
+
const decoder = new TextDecoder();
|
|
304
|
+
let buffer = "";
|
|
305
|
+
let inAssistantTurn = false;
|
|
306
|
+
while (true) {
|
|
307
|
+
const { value, done } = await reader.read();
|
|
308
|
+
if (done)
|
|
309
|
+
break;
|
|
310
|
+
buffer += decoder.decode(value, { stream: true });
|
|
311
|
+
const lines = buffer.split("\n");
|
|
312
|
+
buffer = lines.pop() || "";
|
|
313
|
+
for (const line of lines) {
|
|
314
|
+
if (!line.startsWith("data: "))
|
|
315
|
+
continue;
|
|
316
|
+
try {
|
|
317
|
+
const event = JSON.parse(line.slice(6));
|
|
318
|
+
if (event.type === "part.delta" && event.delta?.type === "text") {
|
|
319
|
+
if (!inAssistantTurn) {
|
|
320
|
+
process.stdout.write(`\n${c.cyan}${c.bold}●${c.reset} `);
|
|
321
|
+
inAssistantTurn = true;
|
|
322
|
+
}
|
|
323
|
+
process.stdout.write(event.delta.text);
|
|
324
|
+
}
|
|
325
|
+
if (event.type === "session.retry") {
|
|
326
|
+
process.stdout.write(`\n${c.yellow}⟳ Retrying (attempt ${event.attempt}): ${event.message}${c.reset}\n`);
|
|
327
|
+
}
|
|
328
|
+
if (event.type === "error") {
|
|
329
|
+
process.stdout.write(`\n${c.red}✗ ${event.properties?.message || "Unknown error"}${c.reset}\n`);
|
|
330
|
+
}
|
|
331
|
+
if (event.type === "tool.requested") {
|
|
332
|
+
process.stdout.write(`\n${c.dim}⚙ ${event.tool_name}${c.reset}`);
|
|
333
|
+
}
|
|
334
|
+
}
|
|
335
|
+
catch { /* non-JSON line */ }
|
|
336
|
+
}
|
|
337
|
+
}
|
|
338
|
+
if (inAssistantTurn)
|
|
339
|
+
process.stdout.write("\n\n");
|
|
340
|
+
}
|
|
341
|
+
async function cmdStatus(args) {
|
|
342
|
+
const port = parseInt(args.find(a => a.startsWith("--port="))?.split("=")[1] || process.env.PORT || "3000");
|
|
343
|
+
const running = await checkServer(port);
|
|
344
|
+
if (running) {
|
|
345
|
+
console.log(`${c.green}${c.bold}●${c.reset} BoneCode server is ${c.green}running${c.reset} on port ${port}`);
|
|
346
|
+
console.log(` ${c.dim}http://localhost:${port}/health${c.reset}`);
|
|
347
|
+
}
|
|
348
|
+
else {
|
|
349
|
+
console.log(`${c.red}${c.bold}●${c.reset} BoneCode server is ${c.red}not running${c.reset}`);
|
|
350
|
+
console.log(` ${c.dim}Start with: bonecode serve${c.reset}`);
|
|
351
|
+
}
|
|
352
|
+
}
|
|
353
|
+
async function cmdCompile() {
|
|
354
|
+
console.log(`${c.cyan}Compiling .bone files...${c.reset}`);
|
|
355
|
+
try {
|
|
356
|
+
(0, child_process_1.execSync)(`node --max-old-space-size=512 ${path.join(PKG_ROOT, "scripts", "compile.js")}`, {
|
|
357
|
+
cwd: PKG_ROOT, stdio: "inherit",
|
|
358
|
+
});
|
|
359
|
+
}
|
|
360
|
+
catch {
|
|
361
|
+
process.exit(1);
|
|
362
|
+
}
|
|
363
|
+
}
|
|
364
|
+
async function cmdMigrate() {
|
|
365
|
+
console.log(`${c.cyan}Running migrations...${c.reset}`);
|
|
366
|
+
try {
|
|
367
|
+
(0, child_process_1.execSync)(`node --max-old-space-size=256 ${path.join(PKG_ROOT, "scripts", "migrate.js")}`, {
|
|
368
|
+
cwd: PKG_ROOT, stdio: "inherit",
|
|
369
|
+
env: { ...process.env },
|
|
370
|
+
});
|
|
371
|
+
}
|
|
372
|
+
catch {
|
|
373
|
+
process.exit(1);
|
|
374
|
+
}
|
|
375
|
+
}
|
|
376
|
+
async function cmdStats(args) {
|
|
377
|
+
const days = (() => {
|
|
378
|
+
const idx = args.findIndex(a => a === "--days");
|
|
379
|
+
if (idx >= 0)
|
|
380
|
+
return parseInt(args[idx + 1] || "0");
|
|
381
|
+
const match = args.find(a => a.startsWith("--days="));
|
|
382
|
+
if (match)
|
|
383
|
+
return parseInt(match.split("=")[1]);
|
|
384
|
+
return undefined;
|
|
385
|
+
})();
|
|
386
|
+
const toolLimit = (() => {
|
|
387
|
+
const idx = args.findIndex(a => a === "--tools");
|
|
388
|
+
if (idx >= 0)
|
|
389
|
+
return parseInt(args[idx + 1] || "10");
|
|
390
|
+
const match = args.find(a => a.startsWith("--tools="));
|
|
391
|
+
if (match)
|
|
392
|
+
return parseInt(match.split("=")[1]);
|
|
393
|
+
return undefined;
|
|
394
|
+
})();
|
|
395
|
+
const modelLimit = (() => {
|
|
396
|
+
const idx = args.findIndex(a => a === "--models");
|
|
397
|
+
if (idx >= 0) {
|
|
398
|
+
const val = args[idx + 1];
|
|
399
|
+
return val ? parseInt(val) : Infinity;
|
|
400
|
+
}
|
|
401
|
+
const match = args.find(a => a.startsWith("--models="));
|
|
402
|
+
if (match)
|
|
403
|
+
return parseInt(match.split("=")[1]);
|
|
404
|
+
return undefined;
|
|
405
|
+
})();
|
|
406
|
+
const projectFilter = (() => {
|
|
407
|
+
const idx = args.findIndex(a => a === "--project");
|
|
408
|
+
if (idx >= 0)
|
|
409
|
+
return args[idx + 1];
|
|
410
|
+
const match = args.find(a => a.startsWith("--project="));
|
|
411
|
+
if (match)
|
|
412
|
+
return match.split("=")[1];
|
|
413
|
+
return undefined;
|
|
414
|
+
})();
|
|
415
|
+
const port = parseInt(process.env.PORT || "3000");
|
|
416
|
+
await ensureServer(port);
|
|
417
|
+
// Use the stats module directly via the server's DB
|
|
418
|
+
const { aggregateStats, displayStats } = await Promise.resolve().then(() => __importStar(require("./stats")));
|
|
419
|
+
const { Pool } = await Promise.resolve().then(() => __importStar(require("pg")));
|
|
420
|
+
const pool = new Pool({ connectionString: process.env.DATABASE_URL });
|
|
421
|
+
try {
|
|
422
|
+
console.log(`${c.dim}Aggregating stats...${c.reset}`);
|
|
423
|
+
const stats = await aggregateStats(pool, { days, projectFilter });
|
|
424
|
+
displayStats(stats, toolLimit, modelLimit);
|
|
425
|
+
}
|
|
426
|
+
finally {
|
|
427
|
+
await pool.end();
|
|
428
|
+
}
|
|
429
|
+
}
|
|
430
|
+
async function cmdExport(args) {
|
|
431
|
+
const sessionId = args.find(a => !a.startsWith("-"));
|
|
432
|
+
const sanitize = args.includes("--sanitize");
|
|
433
|
+
const port = parseInt(process.env.PORT || "3000");
|
|
434
|
+
await ensureServer(port);
|
|
435
|
+
const { exportSession, listSessionsForExport } = await Promise.resolve().then(() => __importStar(require("./export")));
|
|
436
|
+
const { Pool } = await Promise.resolve().then(() => __importStar(require("pg")));
|
|
437
|
+
const pool = new Pool({ connectionString: process.env.DATABASE_URL });
|
|
438
|
+
try {
|
|
439
|
+
let sid = sessionId;
|
|
440
|
+
if (!sid) {
|
|
441
|
+
// Pick the most recent session
|
|
442
|
+
const sessions = await listSessionsForExport(pool);
|
|
443
|
+
if (sessions.length === 0) {
|
|
444
|
+
console.error(`${c.red}No sessions found${c.reset}`);
|
|
445
|
+
process.exit(1);
|
|
446
|
+
}
|
|
447
|
+
sid = sessions[0].id;
|
|
448
|
+
console.error(`${c.dim}Exporting most recent session: ${sid}${c.reset}`);
|
|
449
|
+
}
|
|
450
|
+
const json = await exportSession(pool, sid, { sanitize });
|
|
451
|
+
process.stdout.write(json + "\n");
|
|
452
|
+
}
|
|
453
|
+
finally {
|
|
454
|
+
await pool.end();
|
|
455
|
+
}
|
|
456
|
+
}
|
|
457
|
+
async function cmdWeb(args) {
|
|
458
|
+
const port = parseInt(args.find(a => a.startsWith("--port="))?.split("=")[1] || process.env.PORT || "3000");
|
|
459
|
+
process.env.PORT = String(port);
|
|
460
|
+
const alreadyRunning = await checkServer(port);
|
|
461
|
+
if (!alreadyRunning) {
|
|
462
|
+
console.log(`${c.cyan}${c.bold}BoneCode${c.reset} starting server on port ${port}...`);
|
|
463
|
+
await Promise.resolve().then(() => __importStar(require("./server")));
|
|
464
|
+
let attempts = 0;
|
|
465
|
+
while (attempts < 30) {
|
|
466
|
+
if (await checkServer(port))
|
|
467
|
+
break;
|
|
468
|
+
await new Promise(r => setTimeout(r, 500));
|
|
469
|
+
attempts++;
|
|
470
|
+
}
|
|
471
|
+
}
|
|
472
|
+
const url = `http://localhost:${port}/ui`;
|
|
473
|
+
console.log(`${c.cyan}${c.bold}BoneCode${c.reset} web UI: ${c.dim}${url}${c.reset}`);
|
|
474
|
+
// Open browser
|
|
475
|
+
const { exec } = require("child_process");
|
|
476
|
+
const openCmd = process.platform === "darwin" ? `open "${url}"`
|
|
477
|
+
: process.platform === "win32" ? `start "" "${url}"`
|
|
478
|
+
: `xdg-open "${url}"`;
|
|
479
|
+
exec(openCmd, (err) => {
|
|
480
|
+
if (err)
|
|
481
|
+
console.log(`${c.dim}Open manually: ${url}${c.reset}`);
|
|
482
|
+
});
|
|
483
|
+
// Keep alive
|
|
484
|
+
await new Promise(() => { });
|
|
485
|
+
}
|
|
486
|
+
async function cmdPr(args) {
|
|
487
|
+
const prNumber = parseInt(args[0] || "");
|
|
488
|
+
if (!prNumber || isNaN(prNumber)) {
|
|
489
|
+
console.error(`${c.red}Usage:${c.reset} bonecode pr <number>`);
|
|
490
|
+
process.exit(1);
|
|
491
|
+
}
|
|
492
|
+
const { execSync, spawnSync } = require("child_process");
|
|
493
|
+
const localBranch = `pr/${prNumber}`;
|
|
494
|
+
console.log(`${c.cyan}Fetching PR #${prNumber}...${c.reset}`);
|
|
495
|
+
// Check gh CLI is available
|
|
496
|
+
try {
|
|
497
|
+
execSync("gh --version", { stdio: "ignore" });
|
|
498
|
+
}
|
|
499
|
+
catch {
|
|
500
|
+
console.error(`${c.red}Error:${c.reset} gh CLI not found. Install from https://cli.github.com/`);
|
|
501
|
+
process.exit(1);
|
|
502
|
+
}
|
|
503
|
+
// Checkout the PR
|
|
504
|
+
const checkout = spawnSync("gh", ["pr", "checkout", String(prNumber), "--branch", localBranch, "--force"], {
|
|
505
|
+
stdio: "inherit",
|
|
506
|
+
});
|
|
507
|
+
if (checkout.status !== 0) {
|
|
508
|
+
console.error(`${c.red}Failed to checkout PR #${prNumber}${c.reset}`);
|
|
509
|
+
process.exit(1);
|
|
510
|
+
}
|
|
511
|
+
console.log(`${c.green}✓${c.reset} Checked out PR #${prNumber} as branch '${localBranch}'`);
|
|
512
|
+
console.log(`\n${c.dim}Starting BoneCode...${c.reset}\n`);
|
|
513
|
+
// Launch bonecode in the current directory
|
|
514
|
+
const result = spawnSync(process.execPath, [process.argv[1], "run", "-i"], {
|
|
515
|
+
stdio: "inherit",
|
|
516
|
+
cwd: process.cwd(),
|
|
517
|
+
});
|
|
518
|
+
process.exit(result.status || 0);
|
|
519
|
+
}
|
|
520
|
+
async function cmdGithub(args) {
|
|
521
|
+
const sub = args[0];
|
|
522
|
+
if (sub === "install") {
|
|
523
|
+
console.log(`${c.cyan}GitHub Agent Setup${c.reset}`);
|
|
524
|
+
console.log(`\nTo install the BoneCode GitHub agent:`);
|
|
525
|
+
console.log(`\n 1. Add a GitHub Actions workflow file:`);
|
|
526
|
+
console.log(`\n${c.dim}.github/workflows/bonecode.yml${c.reset}`);
|
|
527
|
+
console.log(`
|
|
528
|
+
${c.dim}name: bonecode
|
|
529
|
+
on:
|
|
530
|
+
issue_comment:
|
|
531
|
+
types: [created]
|
|
532
|
+
pull_request_review_comment:
|
|
533
|
+
types: [created]
|
|
534
|
+
|
|
535
|
+
jobs:
|
|
536
|
+
bonecode:
|
|
537
|
+
if: |
|
|
538
|
+
contains(github.event.comment.body, '/bc') ||
|
|
539
|
+
startsWith(github.event.comment.body, '/bc')
|
|
540
|
+
runs-on: ubuntu-latest
|
|
541
|
+
steps:
|
|
542
|
+
- uses: actions/checkout@v4
|
|
543
|
+
- name: Run BoneCode
|
|
544
|
+
run: npx bonecode run "\${{ github.event.comment.body }}"
|
|
545
|
+
env:
|
|
546
|
+
ANTHROPIC_API_KEY: \${{ secrets.ANTHROPIC_API_KEY }}
|
|
547
|
+
DATABASE_URL: \${{ secrets.DATABASE_URL }}${c.reset}`);
|
|
548
|
+
console.log(`\n 2. Add your API key as a GitHub secret`);
|
|
549
|
+
console.log(` 3. Comment '/bc <task>' on any issue or PR\n`);
|
|
550
|
+
}
|
|
551
|
+
else if (sub === "run") {
|
|
552
|
+
// Run the GitHub agent from a workflow context
|
|
553
|
+
const prompt = process.env.PROMPT || args.slice(1).join(" ");
|
|
554
|
+
if (!prompt) {
|
|
555
|
+
console.error(`${c.red}Error:${c.reset} PROMPT env var or message required`);
|
|
556
|
+
process.exit(1);
|
|
557
|
+
}
|
|
558
|
+
const port = parseInt(process.env.PORT || "3000");
|
|
559
|
+
await ensureServer(port);
|
|
560
|
+
await cmdRun(["--interactive=false", prompt]);
|
|
561
|
+
}
|
|
562
|
+
else {
|
|
563
|
+
console.log(`${c.bold}bonecode github${c.reset} — GitHub agent management`);
|
|
564
|
+
console.log(`\n ${c.cyan}bonecode github install${c.reset} Show setup instructions`);
|
|
565
|
+
console.log(` ${c.cyan}bonecode github run${c.reset} Run agent (for CI/CD use)`);
|
|
566
|
+
}
|
|
567
|
+
}
|
|
568
|
+
async function cmdDefault() {
|
|
569
|
+
const port = parseInt(process.env.PORT || "3000");
|
|
570
|
+
const model = process.env.DEFAULT_MODEL || "local-model";
|
|
571
|
+
const provider = process.env.DEFAULT_PROVIDER || "openai_compatible";
|
|
572
|
+
const worktree = process.cwd();
|
|
573
|
+
const jwtLib = require("jsonwebtoken");
|
|
574
|
+
const secret = process.env.JWT_SECRET || "bonecode-dev-secret-do-not-use-in-production-at-all-ever";
|
|
575
|
+
const token = jwtLib.sign({ sub: "tui-user" }, secret, { expiresIn: "24h" });
|
|
576
|
+
// If server is already running externally (e.g. bonecode serve), just attach
|
|
577
|
+
const alreadyRunning = await checkServer(port);
|
|
578
|
+
if (!alreadyRunning) {
|
|
579
|
+
// Start the server IN-PROCESS — same Node.js instance as the CLI.
|
|
580
|
+
// This means the server dies automatically when the TUI exits.
|
|
581
|
+
// Suppress server startup logs in default (TUI) mode.
|
|
582
|
+
const origWrite = process.stdout.write.bind(process.stdout);
|
|
583
|
+
const origErr = process.stderr.write.bind(process.stderr);
|
|
584
|
+
let startupDone = false;
|
|
585
|
+
const suppress = (chunk) => {
|
|
586
|
+
if (startupDone)
|
|
587
|
+
return false; // let it through after startup
|
|
588
|
+
const s = chunk.toString();
|
|
589
|
+
// Only suppress JSON log lines and the server banner
|
|
590
|
+
if (s.startsWith("{") || s.startsWith("[BoneCode]") || s.startsWith(" OpenCode") ||
|
|
591
|
+
s.startsWith(" Health") || s.startsWith(" Domain") || s.startsWith(" WebSocket")) {
|
|
592
|
+
return true;
|
|
593
|
+
}
|
|
594
|
+
return false;
|
|
595
|
+
};
|
|
596
|
+
process.stdout.write = (chunk, ...args) => suppress(chunk) ? true : origWrite(chunk, ...args);
|
|
597
|
+
process.stderr.write = (chunk, ...args) => suppress(chunk) ? true : origErr(chunk, ...args);
|
|
598
|
+
try {
|
|
599
|
+
process.stdout.write(`${c.dim}Starting BoneCode...${c.reset}\n`);
|
|
600
|
+
await Promise.resolve().then(() => __importStar(require("./server")));
|
|
601
|
+
let attempts = 0;
|
|
602
|
+
while (attempts < 30) {
|
|
603
|
+
if (await checkServer(port))
|
|
604
|
+
break;
|
|
605
|
+
await new Promise(r => setTimeout(r, 500));
|
|
606
|
+
attempts++;
|
|
607
|
+
}
|
|
608
|
+
if (!(await checkServer(port)))
|
|
609
|
+
throw new Error("Server failed to start");
|
|
610
|
+
}
|
|
611
|
+
catch (e) {
|
|
612
|
+
// Restore stdout before printing error
|
|
613
|
+
process.stdout.write = origWrite;
|
|
614
|
+
process.stderr.write = origErr;
|
|
615
|
+
console.error(`${c.red}Failed to start server:${c.reset} ${e.message}`);
|
|
616
|
+
process.exit(1);
|
|
617
|
+
}
|
|
618
|
+
// Restore stdout now that startup is done
|
|
619
|
+
startupDone = true;
|
|
620
|
+
process.stdout.write = origWrite;
|
|
621
|
+
process.stderr.write = origErr;
|
|
622
|
+
}
|
|
623
|
+
// Launch TUI — process stays alive until user exits
|
|
624
|
+
process.env.BONECODE_TUI = "1";
|
|
625
|
+
const { startInteractiveTUI } = await Promise.resolve().then(() => __importStar(require("./tui")));
|
|
626
|
+
await startInteractiveTUI({ port, token, model, provider, worktree });
|
|
627
|
+
// TUI exited — kill the process (takes the in-process server with it)
|
|
628
|
+
process.exit(0);
|
|
629
|
+
}
|
|
630
|
+
// ─── Main ─────────────────────────────────────────────────────────────────────
|
|
631
|
+
async function main() {
|
|
632
|
+
const args = process.argv.slice(2);
|
|
633
|
+
const cmd = args[0];
|
|
634
|
+
if (cmd === "--version" || cmd === "-v") {
|
|
635
|
+
console.log(`bonecode v${VERSION}`);
|
|
636
|
+
return;
|
|
637
|
+
}
|
|
638
|
+
if (cmd === "--help" || cmd === "-h" || cmd === "help") {
|
|
639
|
+
help();
|
|
640
|
+
return;
|
|
641
|
+
}
|
|
642
|
+
switch (cmd) {
|
|
643
|
+
case "serve":
|
|
644
|
+
await cmdServe(args.slice(1));
|
|
645
|
+
break;
|
|
646
|
+
case "run":
|
|
647
|
+
await cmdRun(args.slice(1));
|
|
648
|
+
break;
|
|
649
|
+
case "status":
|
|
650
|
+
await cmdStatus(args.slice(1));
|
|
651
|
+
break;
|
|
652
|
+
case "compile":
|
|
653
|
+
await cmdCompile();
|
|
654
|
+
break;
|
|
655
|
+
case "migrate":
|
|
656
|
+
await cmdMigrate();
|
|
657
|
+
break;
|
|
658
|
+
case "stats":
|
|
659
|
+
await cmdStats(args.slice(1));
|
|
660
|
+
break;
|
|
661
|
+
case "export":
|
|
662
|
+
await cmdExport(args.slice(1));
|
|
663
|
+
break;
|
|
664
|
+
case "web":
|
|
665
|
+
await cmdWeb(args.slice(1));
|
|
666
|
+
break;
|
|
667
|
+
case "pr":
|
|
668
|
+
await cmdPr(args.slice(1));
|
|
669
|
+
break;
|
|
670
|
+
case "github":
|
|
671
|
+
await cmdGithub(args.slice(1));
|
|
672
|
+
break;
|
|
673
|
+
default:
|
|
674
|
+
if (cmd && !cmd.startsWith("-")) {
|
|
675
|
+
// Treat unknown first arg as a message to `run`
|
|
676
|
+
await cmdRun(args);
|
|
677
|
+
}
|
|
678
|
+
else {
|
|
679
|
+
await cmdDefault();
|
|
680
|
+
}
|
|
681
|
+
}
|
|
682
|
+
}
|
|
683
|
+
main().catch((e) => {
|
|
684
|
+
console.error(`${c.red}Error:${c.reset} ${e.message}`);
|
|
685
|
+
process.exit(1);
|
|
686
|
+
});
|
|
687
|
+
//# sourceMappingURL=cli.js.map
|