@johpaz/hive 1.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CONTRIBUTING.md +44 -0
- package/README.md +310 -0
- package/package.json +96 -0
- package/packages/cli/package.json +28 -0
- package/packages/cli/src/commands/agent-run.ts +168 -0
- package/packages/cli/src/commands/agents.ts +398 -0
- package/packages/cli/src/commands/chat.ts +142 -0
- package/packages/cli/src/commands/config.ts +50 -0
- package/packages/cli/src/commands/cron.ts +161 -0
- package/packages/cli/src/commands/dev.ts +95 -0
- package/packages/cli/src/commands/doctor.ts +133 -0
- package/packages/cli/src/commands/gateway.ts +443 -0
- package/packages/cli/src/commands/logs.ts +57 -0
- package/packages/cli/src/commands/mcp.ts +175 -0
- package/packages/cli/src/commands/message.ts +77 -0
- package/packages/cli/src/commands/onboard.ts +1868 -0
- package/packages/cli/src/commands/security.ts +144 -0
- package/packages/cli/src/commands/service.ts +50 -0
- package/packages/cli/src/commands/sessions.ts +116 -0
- package/packages/cli/src/commands/skills.ts +187 -0
- package/packages/cli/src/commands/update.ts +25 -0
- package/packages/cli/src/index.ts +185 -0
- package/packages/cli/src/utils/token.ts +6 -0
- package/packages/code-bridge/README.md +78 -0
- package/packages/code-bridge/package.json +18 -0
- package/packages/code-bridge/src/index.ts +95 -0
- package/packages/code-bridge/src/process-manager.ts +212 -0
- package/packages/code-bridge/src/schemas.ts +133 -0
- package/packages/core/package.json +46 -0
- package/packages/core/src/agent/agent-loop.ts +369 -0
- package/packages/core/src/agent/compaction.ts +140 -0
- package/packages/core/src/agent/context-compiler.ts +378 -0
- package/packages/core/src/agent/context-guard.ts +91 -0
- package/packages/core/src/agent/context.ts +138 -0
- package/packages/core/src/agent/conversation-store.ts +198 -0
- package/packages/core/src/agent/curator.ts +158 -0
- package/packages/core/src/agent/hooks.ts +166 -0
- package/packages/core/src/agent/index.ts +116 -0
- package/packages/core/src/agent/llm-client.ts +503 -0
- package/packages/core/src/agent/native-tools.ts +505 -0
- package/packages/core/src/agent/prompt-builder.ts +532 -0
- package/packages/core/src/agent/providers/index.ts +167 -0
- package/packages/core/src/agent/providers.ts +1 -0
- package/packages/core/src/agent/reflector.ts +170 -0
- package/packages/core/src/agent/service.ts +64 -0
- package/packages/core/src/agent/stuck-loop.ts +133 -0
- package/packages/core/src/agent/supervisor.ts +39 -0
- package/packages/core/src/agent/tracer.ts +102 -0
- package/packages/core/src/agent/workspace.ts +110 -0
- package/packages/core/src/canvas/canvas-manager.test.ts +161 -0
- package/packages/core/src/canvas/canvas-manager.ts +319 -0
- package/packages/core/src/canvas/canvas-tools.ts +420 -0
- package/packages/core/src/canvas/emitter.ts +115 -0
- package/packages/core/src/canvas/index.ts +2 -0
- package/packages/core/src/channels/base.ts +138 -0
- package/packages/core/src/channels/discord.ts +260 -0
- package/packages/core/src/channels/index.ts +7 -0
- package/packages/core/src/channels/manager.ts +383 -0
- package/packages/core/src/channels/slack.ts +287 -0
- package/packages/core/src/channels/telegram.ts +502 -0
- package/packages/core/src/channels/webchat.ts +128 -0
- package/packages/core/src/channels/whatsapp.ts +375 -0
- package/packages/core/src/config/index.ts +12 -0
- package/packages/core/src/config/loader.ts +529 -0
- package/packages/core/src/events/event-bus.ts +169 -0
- package/packages/core/src/gateway/index.ts +5 -0
- package/packages/core/src/gateway/initializer.ts +290 -0
- package/packages/core/src/gateway/lane-queue.ts +169 -0
- package/packages/core/src/gateway/resolver.ts +108 -0
- package/packages/core/src/gateway/router.ts +124 -0
- package/packages/core/src/gateway/server.ts +3317 -0
- package/packages/core/src/gateway/session.ts +95 -0
- package/packages/core/src/gateway/slash-commands.ts +192 -0
- package/packages/core/src/heartbeat/index.ts +157 -0
- package/packages/core/src/index.ts +19 -0
- package/packages/core/src/integrations/catalog.ts +286 -0
- package/packages/core/src/integrations/env.ts +64 -0
- package/packages/core/src/integrations/index.ts +2 -0
- package/packages/core/src/memory/index.ts +1 -0
- package/packages/core/src/memory/notes.ts +68 -0
- package/packages/core/src/plugins/api.ts +128 -0
- package/packages/core/src/plugins/index.ts +2 -0
- package/packages/core/src/plugins/loader.ts +365 -0
- package/packages/core/src/resilience/circuit-breaker.ts +225 -0
- package/packages/core/src/security/google-chat.ts +269 -0
- package/packages/core/src/security/index.ts +192 -0
- package/packages/core/src/security/pairing.ts +250 -0
- package/packages/core/src/security/rate-limit.ts +270 -0
- package/packages/core/src/security/signal.ts +321 -0
- package/packages/core/src/state/store.ts +312 -0
- package/packages/core/src/storage/bun-sqlite-store.ts +188 -0
- package/packages/core/src/storage/crypto.ts +101 -0
- package/packages/core/src/storage/db-context.ts +333 -0
- package/packages/core/src/storage/onboarding.ts +1087 -0
- package/packages/core/src/storage/schema.ts +541 -0
- package/packages/core/src/storage/seed.ts +571 -0
- package/packages/core/src/storage/sqlite.ts +387 -0
- package/packages/core/src/storage/usage.ts +212 -0
- package/packages/core/src/tools/bridge-events.ts +74 -0
- package/packages/core/src/tools/browser.ts +275 -0
- package/packages/core/src/tools/codebridge.ts +421 -0
- package/packages/core/src/tools/coordinator-tools.ts +179 -0
- package/packages/core/src/tools/cron.ts +611 -0
- package/packages/core/src/tools/exec.ts +140 -0
- package/packages/core/src/tools/fs.ts +364 -0
- package/packages/core/src/tools/index.ts +12 -0
- package/packages/core/src/tools/memory.ts +176 -0
- package/packages/core/src/tools/notify.ts +113 -0
- package/packages/core/src/tools/project-management.ts +376 -0
- package/packages/core/src/tools/project.ts +375 -0
- package/packages/core/src/tools/read.ts +158 -0
- package/packages/core/src/tools/web.ts +436 -0
- package/packages/core/src/tools/workspace.ts +171 -0
- package/packages/core/src/utils/benchmark.ts +80 -0
- package/packages/core/src/utils/crypto.ts +73 -0
- package/packages/core/src/utils/date.ts +42 -0
- package/packages/core/src/utils/index.ts +4 -0
- package/packages/core/src/utils/logger.ts +388 -0
- package/packages/core/src/utils/retry.ts +70 -0
- package/packages/core/src/voice/index.ts +583 -0
- package/packages/core/tsconfig.json +9 -0
- package/packages/mcp/package.json +26 -0
- package/packages/mcp/src/config.ts +13 -0
- package/packages/mcp/src/index.ts +1 -0
- package/packages/mcp/src/logger.ts +42 -0
- package/packages/mcp/src/manager.ts +434 -0
- package/packages/mcp/src/transports/index.ts +67 -0
- package/packages/mcp/src/transports/sse.ts +241 -0
- package/packages/mcp/src/transports/websocket.ts +159 -0
- package/packages/skills/package.json +21 -0
- package/packages/skills/src/bundled/agent_management/SKILL.md +24 -0
- package/packages/skills/src/bundled/browser_automation/SKILL.md +30 -0
- package/packages/skills/src/bundled/context_compact/SKILL.md +35 -0
- package/packages/skills/src/bundled/cron_manager/SKILL.md +52 -0
- package/packages/skills/src/bundled/file_manager/SKILL.md +76 -0
- package/packages/skills/src/bundled/http_client/SKILL.md +24 -0
- package/packages/skills/src/bundled/memory/SKILL.md +42 -0
- package/packages/skills/src/bundled/project_management/SKILL.md +26 -0
- package/packages/skills/src/bundled/shell/SKILL.md +43 -0
- package/packages/skills/src/bundled/system_notify/SKILL.md +52 -0
- package/packages/skills/src/bundled/voice/SKILL.md +25 -0
- package/packages/skills/src/bundled/web_search/SKILL.md +29 -0
- package/packages/skills/src/index.ts +1 -0
- package/packages/skills/src/loader.ts +282 -0
- package/packages/tools/package.json +43 -0
- package/packages/tools/src/browser/browser.test.ts +111 -0
- package/packages/tools/src/browser/index.ts +272 -0
- package/packages/tools/src/canvas/index.ts +220 -0
- package/packages/tools/src/cron/cron.test.ts +164 -0
- package/packages/tools/src/cron/index.ts +304 -0
- package/packages/tools/src/filesystem/filesystem.test.ts +240 -0
- package/packages/tools/src/filesystem/index.ts +379 -0
- package/packages/tools/src/git/index.ts +239 -0
- package/packages/tools/src/index.ts +4 -0
- package/packages/tools/src/shell/detect-env.ts +70 -0
- package/packages/tools/tsconfig.json +9 -0
|
@@ -0,0 +1,398 @@
|
|
|
1
|
+
import * as p from "@clack/prompts";
|
|
2
|
+
import * as fs from "fs";
|
|
3
|
+
import * as path from "path";
|
|
4
|
+
import { loadConfig, getHiveDir } from "@johpaz/hive-core/config/loader";
|
|
5
|
+
|
|
6
|
+
const getAgentsPath = () => path.join(getHiveDir(), "agents");
|
|
7
|
+
|
|
8
|
+
interface AgentConfig {
|
|
9
|
+
id: string;
|
|
10
|
+
name: string;
|
|
11
|
+
default?: boolean;
|
|
12
|
+
workspace: string;
|
|
13
|
+
agentDir: string;
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
interface AgentState {
|
|
17
|
+
id: string;
|
|
18
|
+
name: string;
|
|
19
|
+
type: "primary" | "child" | "sibling";
|
|
20
|
+
state: "active" | "hibernated" | "terminated";
|
|
21
|
+
purpose: string;
|
|
22
|
+
parentId?: string;
|
|
23
|
+
childrenIds: string[];
|
|
24
|
+
workspacePath: string;
|
|
25
|
+
createdAt: string;
|
|
26
|
+
lastActivity: string;
|
|
27
|
+
capabilities: string[];
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
async function loadLifecycleAgents(): Promise<AgentState[]> {
|
|
31
|
+
const agents: AgentState[] = [];
|
|
32
|
+
try {
|
|
33
|
+
if (!fs.existsSync(getAgentsPath())) return [];
|
|
34
|
+
const entries = fs.readdirSync(getAgentsPath(), { withFileTypes: true });
|
|
35
|
+
for (const entry of entries) {
|
|
36
|
+
if (!entry.isDirectory()) continue;
|
|
37
|
+
const statePath = path.join(getAgentsPath(), entry.name, "state.json");
|
|
38
|
+
try {
|
|
39
|
+
const content = fs.readFileSync(statePath, "utf-8");
|
|
40
|
+
agents.push(JSON.parse(content) as AgentState);
|
|
41
|
+
} catch { }
|
|
42
|
+
}
|
|
43
|
+
} catch { }
|
|
44
|
+
return agents;
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
export async function agents(subcommand: string | undefined, args: string[]): Promise<void> {
|
|
48
|
+
switch (subcommand) {
|
|
49
|
+
case "add":
|
|
50
|
+
await addAgent(args[0]);
|
|
51
|
+
break;
|
|
52
|
+
case "create":
|
|
53
|
+
await createAgent(args);
|
|
54
|
+
break;
|
|
55
|
+
case "list":
|
|
56
|
+
case "ls":
|
|
57
|
+
await listAgents(args.includes("--bindings"));
|
|
58
|
+
break;
|
|
59
|
+
case "logs":
|
|
60
|
+
await showLogs(args[0]);
|
|
61
|
+
break;
|
|
62
|
+
case "hibernate":
|
|
63
|
+
await hibernateAgent(args[0]);
|
|
64
|
+
break;
|
|
65
|
+
case "wake":
|
|
66
|
+
await wakeAgent(args[0]);
|
|
67
|
+
break;
|
|
68
|
+
case "terminate":
|
|
69
|
+
case "kill":
|
|
70
|
+
await terminateAgent(args[0], args.includes("--cascade"));
|
|
71
|
+
break;
|
|
72
|
+
case "tree":
|
|
73
|
+
await showTree();
|
|
74
|
+
break;
|
|
75
|
+
case "remove":
|
|
76
|
+
await removeAgent(args[0]);
|
|
77
|
+
break;
|
|
78
|
+
default:
|
|
79
|
+
console.log(`
|
|
80
|
+
Usage: hive agents <command> [arguments]
|
|
81
|
+
|
|
82
|
+
Commands:
|
|
83
|
+
list, ls List all agents
|
|
84
|
+
add <id> Create agent (legacy)
|
|
85
|
+
create Create agent with new lifecycle system
|
|
86
|
+
logs <agent-id> Show agent logs
|
|
87
|
+
hibernate <agent-id> Put agent to sleep
|
|
88
|
+
wake <agent-id> Wake a hibernated agent
|
|
89
|
+
terminate <agent-id> [--cascade] Terminate an agent
|
|
90
|
+
tree Show agent hierarchy tree
|
|
91
|
+
remove <id> Remove agent (legacy)
|
|
92
|
+
|
|
93
|
+
Examples:
|
|
94
|
+
hive agents list
|
|
95
|
+
hive agents create
|
|
96
|
+
hive agents tree
|
|
97
|
+
hive agents hibernate agent-abc123
|
|
98
|
+
hive agents wake agent-abc123
|
|
99
|
+
hive agents terminate agent-abc123 --cascade
|
|
100
|
+
`);
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
async function createAgent(args: string[]): Promise<void> {
|
|
105
|
+
const name = args[0] || await promptInput("Agent name:", `agent-${Date.now()}`);
|
|
106
|
+
const purpose = args[1] || await promptInput("Purpose:", "General AI assistant");
|
|
107
|
+
|
|
108
|
+
const agentId = `agent-${crypto.randomUUID().slice(0, 8)}`;
|
|
109
|
+
const workspacePath = path.join(getAgentsPath(), agentId);
|
|
110
|
+
|
|
111
|
+
fs.mkdirSync(path.join(workspacePath, "memory"), { recursive: true });
|
|
112
|
+
fs.mkdirSync(path.join(workspacePath, "skills"), { recursive: true });
|
|
113
|
+
fs.mkdirSync(path.join(workspacePath, "logs"), { recursive: true });
|
|
114
|
+
|
|
115
|
+
const soulMd = `# SOUL.md - ${name}
|
|
116
|
+
|
|
117
|
+
## Identity
|
|
118
|
+
|
|
119
|
+
I am ${name}, created to ${purpose.toLowerCase()}.
|
|
120
|
+
|
|
121
|
+
## Personality
|
|
122
|
+
|
|
123
|
+
- Helpful and responsive
|
|
124
|
+
- Clear communicator
|
|
125
|
+
- Adaptable to user needs
|
|
126
|
+
`;
|
|
127
|
+
|
|
128
|
+
const ethicsMd = `# ETHICS.md
|
|
129
|
+
|
|
130
|
+
## Core Principles
|
|
131
|
+
|
|
132
|
+
1. Honesty - Never deceive the user
|
|
133
|
+
2. Privacy - Protect user data
|
|
134
|
+
3. Safety - Prevent harm
|
|
135
|
+
4. Consent - Ask before significant actions
|
|
136
|
+
`;
|
|
137
|
+
|
|
138
|
+
fs.writeFileSync(path.join(workspacePath, "SOUL.md"), soulMd);
|
|
139
|
+
fs.writeFileSync(path.join(workspacePath, "ETHICS.md"), ethicsMd);
|
|
140
|
+
fs.writeFileSync(path.join(workspacePath, "USER.md"), "# USER.md\n\n## Preferences\n\n");
|
|
141
|
+
|
|
142
|
+
const state: AgentState = {
|
|
143
|
+
id: agentId,
|
|
144
|
+
name,
|
|
145
|
+
type: "sibling",
|
|
146
|
+
state: "active",
|
|
147
|
+
purpose,
|
|
148
|
+
childrenIds: [],
|
|
149
|
+
workspacePath,
|
|
150
|
+
createdAt: new Date().toISOString(),
|
|
151
|
+
lastActivity: new Date().toISOString(),
|
|
152
|
+
capabilities: ["general"],
|
|
153
|
+
};
|
|
154
|
+
|
|
155
|
+
fs.writeFileSync(path.join(workspacePath, "state.json"), JSON.stringify(state, null, 2));
|
|
156
|
+
|
|
157
|
+
console.log(`\n✅ Agent created: ${name} (${agentId})`);
|
|
158
|
+
console.log(` Workspace: ${workspacePath}\n`);
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
async function promptInput(message: string, defaultVal: string): Promise<string> {
|
|
162
|
+
const result = await p.text({
|
|
163
|
+
message,
|
|
164
|
+
placeholder: defaultVal,
|
|
165
|
+
defaultValue: defaultVal,
|
|
166
|
+
});
|
|
167
|
+
if (p.isCancel(result)) {
|
|
168
|
+
p.cancel("Cancelled");
|
|
169
|
+
process.exit(0);
|
|
170
|
+
}
|
|
171
|
+
return result as string;
|
|
172
|
+
}
|
|
173
|
+
|
|
174
|
+
async function showLogs(agentId: string | undefined): Promise<void> {
|
|
175
|
+
if (!agentId) {
|
|
176
|
+
console.log("Usage: hive agents logs <agent-id>");
|
|
177
|
+
return;
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
const logPath = path.join(getAgentsPath(), agentId, "logs");
|
|
181
|
+
try {
|
|
182
|
+
const files = fs.readdirSync(logPath).filter((f) => f.endsWith(".log")).sort().reverse();
|
|
183
|
+
if (files.length === 0) {
|
|
184
|
+
console.log("\n📋 No logs found.\n");
|
|
185
|
+
return;
|
|
186
|
+
}
|
|
187
|
+
const content = fs.readFileSync(path.join(logPath, files[0]!), "utf-8");
|
|
188
|
+
console.log(`\n📋 Logs for ${agentId}:\n`);
|
|
189
|
+
console.log(content.slice(-3000));
|
|
190
|
+
} catch {
|
|
191
|
+
console.log(`\n❌ Agent not found: ${agentId}\n`);
|
|
192
|
+
}
|
|
193
|
+
}
|
|
194
|
+
|
|
195
|
+
async function hibernateAgent(agentId: string | undefined): Promise<void> {
|
|
196
|
+
if (!agentId) {
|
|
197
|
+
console.log("Usage: hive agents hibernate <agent-id>");
|
|
198
|
+
return;
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
const statePath = path.join(getAgentsPath(), agentId, "state.json");
|
|
202
|
+
try {
|
|
203
|
+
const state = JSON.parse(fs.readFileSync(statePath, "utf-8")) as AgentState;
|
|
204
|
+
state.state = "hibernated";
|
|
205
|
+
(state as any).hibernatedAt = new Date().toISOString();
|
|
206
|
+
fs.writeFileSync(statePath, JSON.stringify(state, null, 2));
|
|
207
|
+
console.log(`\n💤 Agent ${state.name} hibernated.\n`);
|
|
208
|
+
} catch {
|
|
209
|
+
console.log(`\n❌ Agent not found: ${agentId}\n`);
|
|
210
|
+
}
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
async function wakeAgent(agentId: string | undefined): Promise<void> {
|
|
214
|
+
if (!agentId) {
|
|
215
|
+
console.log("Usage: hive agents wake <agent-id>");
|
|
216
|
+
return;
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
const statePath = path.join(getAgentsPath(), agentId, "state.json");
|
|
220
|
+
try {
|
|
221
|
+
const state = JSON.parse(fs.readFileSync(statePath, "utf-8")) as AgentState;
|
|
222
|
+
state.state = "active";
|
|
223
|
+
state.lastActivity = new Date().toISOString();
|
|
224
|
+
fs.writeFileSync(statePath, JSON.stringify(state, null, 2));
|
|
225
|
+
console.log(`\n⚡ Agent ${state.name} activated.\n`);
|
|
226
|
+
} catch {
|
|
227
|
+
console.log(`\n❌ Agent not found: ${agentId}\n`);
|
|
228
|
+
}
|
|
229
|
+
}
|
|
230
|
+
|
|
231
|
+
async function terminateAgent(agentId: string | undefined, cascade: boolean): Promise<void> {
|
|
232
|
+
if (!agentId) {
|
|
233
|
+
console.log("Usage: hive agents terminate <agent-id> [--cascade]");
|
|
234
|
+
return;
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
const statePath = path.join(getAgentsPath(), agentId, "state.json");
|
|
238
|
+
try {
|
|
239
|
+
const state = JSON.parse(fs.readFileSync(statePath, "utf-8")) as AgentState;
|
|
240
|
+
state.state = "terminated";
|
|
241
|
+
(state as any).terminatedAt = new Date().toISOString();
|
|
242
|
+
(state as any).terminationReason = "User requested";
|
|
243
|
+
fs.writeFileSync(statePath, JSON.stringify(state, null, 2));
|
|
244
|
+
console.log(`\n⚫ Agent ${state.name} terminated.\n`);
|
|
245
|
+
|
|
246
|
+
if (cascade && state.childrenIds?.length > 0) {
|
|
247
|
+
for (const childId of state.childrenIds) {
|
|
248
|
+
await terminateAgent(childId, true);
|
|
249
|
+
}
|
|
250
|
+
}
|
|
251
|
+
} catch {
|
|
252
|
+
console.log(`\n❌ Agent not found: ${agentId}\n`);
|
|
253
|
+
}
|
|
254
|
+
}
|
|
255
|
+
|
|
256
|
+
async function showTree(): Promise<void> {
|
|
257
|
+
const agents = await loadLifecycleAgents();
|
|
258
|
+
if (agents.length === 0) {
|
|
259
|
+
console.log("\n📋 No agents found. Run 'hive onboard' to create your first agent.\n");
|
|
260
|
+
return;
|
|
261
|
+
}
|
|
262
|
+
|
|
263
|
+
const primaries = agents.filter((a) => a.type === "primary" && a.state !== "terminated");
|
|
264
|
+
|
|
265
|
+
const printTree = (agent: AgentState, indent = "") => {
|
|
266
|
+
const icon = agent.state === "active" ? "🟢" : agent.state === "hibernated" ? "💤" : "⚫";
|
|
267
|
+
console.log(`${indent}${icon} ${agent.name} (${agent.id.slice(0, 12)}...)`);
|
|
268
|
+
const children = agents.filter((a) => a.parentId === agent.id);
|
|
269
|
+
for (const child of children) {
|
|
270
|
+
printTree(child, indent + " ");
|
|
271
|
+
}
|
|
272
|
+
};
|
|
273
|
+
|
|
274
|
+
console.log("\n🌳 Agent Hierarchy:\n");
|
|
275
|
+
for (const primary of primaries) {
|
|
276
|
+
printTree(primary);
|
|
277
|
+
}
|
|
278
|
+
console.log("");
|
|
279
|
+
}
|
|
280
|
+
|
|
281
|
+
async function addAgent(id: string | undefined): Promise<void> {
|
|
282
|
+
const config = loadConfig();
|
|
283
|
+
|
|
284
|
+
if (!id) {
|
|
285
|
+
const result = await p.text({
|
|
286
|
+
message: "ID del nuevo agente:",
|
|
287
|
+
placeholder: "work",
|
|
288
|
+
validate: (v) => (!v ? "El ID es requerido" : undefined),
|
|
289
|
+
});
|
|
290
|
+
if (p.isCancel(result)) {
|
|
291
|
+
p.cancel("Cancelado");
|
|
292
|
+
return;
|
|
293
|
+
}
|
|
294
|
+
id = result;
|
|
295
|
+
}
|
|
296
|
+
|
|
297
|
+
const agentsList = ((config.agents as Record<string, unknown>)?.list as AgentConfig[]) || [];
|
|
298
|
+
if (agentsList.find((a) => a.id === id)) {
|
|
299
|
+
console.log(`❌ Ya existe un agente con ID "${id}"`);
|
|
300
|
+
return;
|
|
301
|
+
}
|
|
302
|
+
|
|
303
|
+
const name = await p.text({
|
|
304
|
+
message: "Nombre del agente:",
|
|
305
|
+
placeholder: id,
|
|
306
|
+
defaultValue: id,
|
|
307
|
+
});
|
|
308
|
+
|
|
309
|
+
if (p.isCancel(name)) {
|
|
310
|
+
p.cancel("Cancelado");
|
|
311
|
+
return;
|
|
312
|
+
}
|
|
313
|
+
|
|
314
|
+
const workspace = path.join(getHiveDir(), "agents", id as string, "workspace");
|
|
315
|
+
const agentDir = path.join(getHiveDir(), "agents", id as string, "agent");
|
|
316
|
+
|
|
317
|
+
const newAgent: AgentConfig = {
|
|
318
|
+
id: id as string,
|
|
319
|
+
name: name as string,
|
|
320
|
+
workspace,
|
|
321
|
+
agentDir,
|
|
322
|
+
};
|
|
323
|
+
|
|
324
|
+
fs.mkdirSync(workspace, { recursive: true });
|
|
325
|
+
fs.mkdirSync(agentDir, { recursive: true });
|
|
326
|
+
|
|
327
|
+
const soulPath = path.join(agentDir, "SOUL.md");
|
|
328
|
+
if (!fs.existsSync(soulPath)) {
|
|
329
|
+
fs.writeFileSync(soulPath, `# ${name} — Soul\n\nEres ${name}, un agente especializado.\n`, "utf-8");
|
|
330
|
+
}
|
|
331
|
+
|
|
332
|
+
const userPath = path.join(agentDir, "USER.md");
|
|
333
|
+
if (!fs.existsSync(userPath)) {
|
|
334
|
+
fs.writeFileSync(userPath, `# User Profile\n\nNotas sobre el usuario.\n`, "utf-8");
|
|
335
|
+
}
|
|
336
|
+
|
|
337
|
+
agentsList.push(newAgent);
|
|
338
|
+
|
|
339
|
+
console.log(`✅ Agente "${id}" creado`);
|
|
340
|
+
console.log(` Workspace: ${workspace}`);
|
|
341
|
+
console.log(` AgentDir: ${agentDir}`);
|
|
342
|
+
}
|
|
343
|
+
|
|
344
|
+
async function listAgents(showBindings: boolean): Promise<void> {
|
|
345
|
+
const config = loadConfig();
|
|
346
|
+
const agentsList = ((config.agents as Record<string, unknown>)?.list as AgentConfig[]) || [];
|
|
347
|
+
|
|
348
|
+
if (agentsList.length === 0) {
|
|
349
|
+
console.log("No hay agentes configurados");
|
|
350
|
+
return;
|
|
351
|
+
}
|
|
352
|
+
|
|
353
|
+
console.log("\n🐝 Agentes:\n");
|
|
354
|
+
for (const agent of agentsList) {
|
|
355
|
+
const defaultTag = agent.default ? " (default)" : "";
|
|
356
|
+
console.log(` ${agent.id}${defaultTag}`);
|
|
357
|
+
console.log(` Nombre: ${agent.name}`);
|
|
358
|
+
console.log(` Workspace: ${agent.workspace}`);
|
|
359
|
+
if (showBindings) {
|
|
360
|
+
const bindings = ((config as any).bindings as any[])?.filter((b: any) => (b.agentId || b.agent) === agent.id) || [];
|
|
361
|
+
console.log(` Bindings: ${bindings.length}`);
|
|
362
|
+
}
|
|
363
|
+
console.log();
|
|
364
|
+
}
|
|
365
|
+
}
|
|
366
|
+
|
|
367
|
+
async function removeAgent(id: string | undefined): Promise<void> {
|
|
368
|
+
if (!id) {
|
|
369
|
+
console.log("❌ Especifica el ID del agente: hive agents remove <id>");
|
|
370
|
+
return;
|
|
371
|
+
}
|
|
372
|
+
|
|
373
|
+
const config = loadConfig();
|
|
374
|
+
const agentsList = ((config.agents as Record<string, unknown>)?.list as AgentConfig[]) || [];
|
|
375
|
+
const agent = agentsList.find((a) => a.id === id);
|
|
376
|
+
|
|
377
|
+
if (!agent) {
|
|
378
|
+
console.log(`❌ No existe el agente "${id}"`);
|
|
379
|
+
return;
|
|
380
|
+
}
|
|
381
|
+
|
|
382
|
+
if (agent.default) {
|
|
383
|
+
console.log("❌ No puedes eliminar el agente por defecto");
|
|
384
|
+
return;
|
|
385
|
+
}
|
|
386
|
+
|
|
387
|
+
const confirm = await p.confirm({
|
|
388
|
+
message: `¿Eliminar el agente "${id}"?`,
|
|
389
|
+
initialValue: false,
|
|
390
|
+
});
|
|
391
|
+
|
|
392
|
+
if (p.isCancel(confirm) || !confirm) {
|
|
393
|
+
console.log("Cancelado");
|
|
394
|
+
return;
|
|
395
|
+
}
|
|
396
|
+
|
|
397
|
+
console.log(`✅ Agente "${id}" eliminado`);
|
|
398
|
+
}
|
|
@@ -0,0 +1,142 @@
|
|
|
1
|
+
import * as p from "@clack/prompts";
|
|
2
|
+
import * as fs from "fs";
|
|
3
|
+
import * as path from "path";
|
|
4
|
+
import { spawn } from "child_process";
|
|
5
|
+
import * as readline from "readline";
|
|
6
|
+
import { loadConfig, getHiveDir } from "@johpaz/hive-core/config/loader";
|
|
7
|
+
|
|
8
|
+
const getHiveDirConst = () => getHiveDir();
|
|
9
|
+
|
|
10
|
+
export async function chat(flags: string[]): Promise<void> {
|
|
11
|
+
const config = loadConfig();
|
|
12
|
+
const HIVE_DIR = getHiveDirConst();
|
|
13
|
+
|
|
14
|
+
if (!fs.existsSync(HIVE_DIR)) {
|
|
15
|
+
console.log("❌ Directorio Hive no encontrado. Ejecuta: hive onboard");
|
|
16
|
+
return;
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
const agentId = flags.find((f) => f.startsWith("--agent"))?.split("=")[1] ||
|
|
20
|
+
flags[flags.indexOf("--agent") + 1] ||
|
|
21
|
+
"main";
|
|
22
|
+
|
|
23
|
+
console.log(`\n🐝 Chat con agente: ${agentId}`);
|
|
24
|
+
console.log(" Escribe /exit para salir, /new para nueva sesión\n");
|
|
25
|
+
|
|
26
|
+
const rl = readline.createInterface({
|
|
27
|
+
input: process.stdin,
|
|
28
|
+
output: process.stdout,
|
|
29
|
+
});
|
|
30
|
+
|
|
31
|
+
const question = (prompt: string): Promise<string> => {
|
|
32
|
+
return new Promise((resolve) => {
|
|
33
|
+
rl.question(prompt, resolve);
|
|
34
|
+
});
|
|
35
|
+
};
|
|
36
|
+
|
|
37
|
+
const messages: Array<{ role: string; content: string }> = [];
|
|
38
|
+
|
|
39
|
+
while (true) {
|
|
40
|
+
const input = await question("> ");
|
|
41
|
+
const trimmed = input.trim();
|
|
42
|
+
|
|
43
|
+
if (!trimmed) continue;
|
|
44
|
+
|
|
45
|
+
if (trimmed === "/exit" || trimmed === "/quit") {
|
|
46
|
+
console.log("👋 ¡Hasta luego!");
|
|
47
|
+
rl.close();
|
|
48
|
+
break;
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
if (trimmed === "/new") {
|
|
52
|
+
messages.length = 0;
|
|
53
|
+
console.log("📝 Nueva sesión iniciada\n");
|
|
54
|
+
continue;
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
if (trimmed === "/help") {
|
|
58
|
+
console.log(`
|
|
59
|
+
Comandos disponibles:
|
|
60
|
+
/exit, /quit - Salir del chat
|
|
61
|
+
/new - Iniciar nueva sesión
|
|
62
|
+
/help - Mostrar esta ayuda
|
|
63
|
+
`);
|
|
64
|
+
continue;
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
messages.push({ role: "user", content: trimmed });
|
|
68
|
+
|
|
69
|
+
console.log("\n🤖 Pensando...\n");
|
|
70
|
+
|
|
71
|
+
try {
|
|
72
|
+
const response = await callLLM(config, messages);
|
|
73
|
+
messages.push({ role: "assistant", content: response });
|
|
74
|
+
console.log(`\n${response}\n`);
|
|
75
|
+
} catch (error) {
|
|
76
|
+
console.log(`❌ Error: ${(error as Error).message}\n`);
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
async function callLLM(config: any, messages: Array<{ role: string; content: string }>): Promise<string> {
|
|
82
|
+
const models = config.models as Record<string, unknown> | undefined;
|
|
83
|
+
const provider = models?.defaultProvider as string || "anthropic";
|
|
84
|
+
const defaults = (models as any)?.defaults as Record<string, string> | undefined;
|
|
85
|
+
const model = defaults?.default || (defaults as any)?.[provider] || "claude-sonnet-4-5";
|
|
86
|
+
const providers = models?.providers as Record<string, Record<string, unknown>> | undefined;
|
|
87
|
+
const providerConfig = providers?.[provider] as Record<string, unknown> | undefined;
|
|
88
|
+
|
|
89
|
+
// Try to get API key from config, then process.env
|
|
90
|
+
const apiKey = (providerConfig?.apiKey as string) ||
|
|
91
|
+
process.env[`${provider.toUpperCase()}_API_KEY`] ||
|
|
92
|
+
"";
|
|
93
|
+
|
|
94
|
+
if (provider === "anthropic") {
|
|
95
|
+
const response = await fetch("https://api.anthropic.com/v1/messages", {
|
|
96
|
+
method: "POST",
|
|
97
|
+
headers: {
|
|
98
|
+
"Content-Type": "application/json",
|
|
99
|
+
"x-api-key": apiKey,
|
|
100
|
+
"anthropic-version": "2023-06-01",
|
|
101
|
+
"anthropic-dangerous-direct-browser-access": "true",
|
|
102
|
+
},
|
|
103
|
+
body: JSON.stringify({
|
|
104
|
+
model,
|
|
105
|
+
max_tokens: 4096,
|
|
106
|
+
messages: messages.map((m) => ({ role: m.role, content: m.content })),
|
|
107
|
+
}),
|
|
108
|
+
});
|
|
109
|
+
|
|
110
|
+
if (!response.ok) {
|
|
111
|
+
const errorData = await response.json().catch(() => ({}));
|
|
112
|
+
throw new Error(`Anthropic API error: ${response.status} - ${JSON.stringify(errorData)}`);
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
const data = (await response.json()) as { content: Array<{ text: string }> };
|
|
116
|
+
return data.content[0]?.text || "Sin respuesta";
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
if (provider === "openai") {
|
|
120
|
+
const response = await fetch("https://api.openai.com/v1/chat/completions", {
|
|
121
|
+
method: "POST",
|
|
122
|
+
headers: {
|
|
123
|
+
"Content-Type": "application/json",
|
|
124
|
+
Authorization: `Bearer ${apiKey}`,
|
|
125
|
+
},
|
|
126
|
+
body: JSON.stringify({
|
|
127
|
+
model,
|
|
128
|
+
messages: messages.map((m) => ({ role: m.role, content: m.content })),
|
|
129
|
+
}),
|
|
130
|
+
});
|
|
131
|
+
|
|
132
|
+
if (!response.ok) {
|
|
133
|
+
const errorData = await response.json().catch(() => ({}));
|
|
134
|
+
throw new Error(`OpenAI API error: ${response.status} - ${JSON.stringify(errorData)}`);
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
const data = (await response.json()) as { choices: Array<{ message: { content: string } }> };
|
|
138
|
+
return data.choices[0]?.message?.content || "Sin respuesta";
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
return `Proveedor '${provider}' no soportado en modo chat CLI. Usa 'hive start' para el Gateway completo.`;
|
|
142
|
+
}
|
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
import * as p from "@clack/prompts";
|
|
2
|
+
import { loadConfig } from "@johpaz/hive-core/config/loader";
|
|
3
|
+
|
|
4
|
+
export async function config(subcommand: string | undefined, args: string[]): Promise<void> {
|
|
5
|
+
switch (subcommand) {
|
|
6
|
+
case "show":
|
|
7
|
+
await showConfig();
|
|
8
|
+
break;
|
|
9
|
+
case "edit":
|
|
10
|
+
console.log("⚠️ La edición manual de archivos YAML ha sido deshabilitada.");
|
|
11
|
+
console.log("Configura Hive a través de variables de entorno o la base de datos.");
|
|
12
|
+
break;
|
|
13
|
+
default:
|
|
14
|
+
console.log(`
|
|
15
|
+
Usage: hive config <command>
|
|
16
|
+
|
|
17
|
+
Commands:
|
|
18
|
+
show Mostrar la configuración actual (redactada)
|
|
19
|
+
edit (Deshabilitado) Edición manual de configuración
|
|
20
|
+
`);
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
function redactSecrets(obj: any, depth = 0): any {
|
|
25
|
+
if (depth > 10) return obj;
|
|
26
|
+
if (typeof obj !== "object" || obj === null) return obj;
|
|
27
|
+
|
|
28
|
+
const sensitiveKeys = ["apiKey", "token", "botToken", "password", "secret", "key"];
|
|
29
|
+
|
|
30
|
+
if (Array.isArray(obj)) {
|
|
31
|
+
return obj.map((item) => redactSecrets(item, depth + 1));
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
const result: any = {};
|
|
35
|
+
for (const [key, value] of Object.entries(obj)) {
|
|
36
|
+
if (sensitiveKeys.some((sk) => key.toLowerCase().includes(sk.toLowerCase()))) {
|
|
37
|
+
result[key] = "***REDACTED***";
|
|
38
|
+
} else {
|
|
39
|
+
result[key] = redactSecrets(value, depth + 1);
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
return result;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
async function showConfig(): Promise<void> {
|
|
46
|
+
const config = loadConfig();
|
|
47
|
+
console.log("\n⚙️ Configuración Actual (Redactada):\n");
|
|
48
|
+
console.log(JSON.stringify(redactSecrets(config), null, 2));
|
|
49
|
+
console.log();
|
|
50
|
+
}
|