@ebowwa/daemons 0.5.1 → 0.7.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (121) hide show
  1. package/dist/core.d.ts +89 -0
  2. package/dist/core.d.ts.map +1 -0
  3. package/dist/core.js +346 -0
  4. package/dist/core.js.map +1 -0
  5. package/dist/index.d.ts +16 -17
  6. package/dist/index.d.ts.map +1 -1
  7. package/dist/index.js +25 -125305
  8. package/dist/index.js.map +1 -0
  9. package/dist/types.d.ts +28 -273
  10. package/dist/types.d.ts.map +1 -1
  11. package/dist/types.js +7 -0
  12. package/dist/types.js.map +1 -0
  13. package/package.json +14 -65
  14. package/src/core.ts +476 -0
  15. package/src/index.ts +23 -101
  16. package/src/types.ts +24 -301
  17. package/dist/agent.d.ts +0 -37
  18. package/dist/agent.d.ts.map +0 -1
  19. package/dist/bin/discord-cli.js +0 -124083
  20. package/dist/bin/manager.js +0 -143
  21. package/dist/bin/telegram-cli.js +0 -124079
  22. package/dist/channels/base.d.ts +0 -163
  23. package/dist/channels/base.d.ts.map +0 -1
  24. package/dist/channels/discord.d.ts +0 -61
  25. package/dist/channels/discord.d.ts.map +0 -1
  26. package/dist/channels/index.d.ts +0 -55
  27. package/dist/channels/index.d.ts.map +0 -1
  28. package/dist/channels/telegram.d.ts +0 -113
  29. package/dist/channels/telegram.d.ts.map +0 -1
  30. package/dist/daemon.d.ts +0 -119
  31. package/dist/daemon.d.ts.map +0 -1
  32. package/dist/hooks.d.ts +0 -34
  33. package/dist/hooks.d.ts.map +0 -1
  34. package/dist/memory.d.ts +0 -151
  35. package/dist/memory.d.ts.map +0 -1
  36. package/dist/skills/coding/commit.d.ts +0 -31
  37. package/dist/skills/coding/commit.d.ts.map +0 -1
  38. package/dist/skills/coding/execute-subtask.d.ts +0 -30
  39. package/dist/skills/coding/execute-subtask.d.ts.map +0 -1
  40. package/dist/skills/coding/fix-issues.d.ts +0 -29
  41. package/dist/skills/coding/fix-issues.d.ts.map +0 -1
  42. package/dist/skills/coding/index.d.ts +0 -21
  43. package/dist/skills/coding/index.d.ts.map +0 -1
  44. package/dist/skills/coding/plan-task.d.ts +0 -38
  45. package/dist/skills/coding/plan-task.d.ts.map +0 -1
  46. package/dist/skills/coding/quality-check.d.ts +0 -35
  47. package/dist/skills/coding/quality-check.d.ts.map +0 -1
  48. package/dist/skills/index.d.ts +0 -38
  49. package/dist/skills/index.d.ts.map +0 -1
  50. package/dist/skills/registry.d.ts +0 -95
  51. package/dist/skills/registry.d.ts.map +0 -1
  52. package/dist/skills/shared/index.d.ts +0 -16
  53. package/dist/skills/shared/index.d.ts.map +0 -1
  54. package/dist/skills/shared/reflect.d.ts +0 -42
  55. package/dist/skills/shared/reflect.d.ts.map +0 -1
  56. package/dist/skills/shared/review.d.ts +0 -42
  57. package/dist/skills/shared/review.d.ts.map +0 -1
  58. package/dist/skills/shared/trajectory.d.ts +0 -80
  59. package/dist/skills/shared/trajectory.d.ts.map +0 -1
  60. package/dist/skills/trading/analyze-market.d.ts +0 -43
  61. package/dist/skills/trading/analyze-market.d.ts.map +0 -1
  62. package/dist/skills/trading/check-risk.d.ts +0 -33
  63. package/dist/skills/trading/check-risk.d.ts.map +0 -1
  64. package/dist/skills/trading/execute-trade.d.ts +0 -38
  65. package/dist/skills/trading/execute-trade.d.ts.map +0 -1
  66. package/dist/skills/trading/generate-signal.d.ts +0 -57
  67. package/dist/skills/trading/generate-signal.d.ts.map +0 -1
  68. package/dist/skills/trading/index.d.ts +0 -21
  69. package/dist/skills/trading/index.d.ts.map +0 -1
  70. package/dist/skills/trading/monitor-position.d.ts +0 -37
  71. package/dist/skills/trading/monitor-position.d.ts.map +0 -1
  72. package/dist/skills/types.d.ts +0 -173
  73. package/dist/skills/types.d.ts.map +0 -1
  74. package/dist/skills/workflows.d.ts +0 -36
  75. package/dist/skills/workflows.d.ts.map +0 -1
  76. package/dist/state.d.ts +0 -31
  77. package/dist/state.d.ts.map +0 -1
  78. package/dist/tools.d.ts +0 -42
  79. package/dist/tools.d.ts.map +0 -1
  80. package/dist/workflow.d.ts +0 -182
  81. package/dist/workflow.d.ts.map +0 -1
  82. package/dist/workflows/coding.d.ts +0 -79
  83. package/dist/workflows/coding.d.ts.map +0 -1
  84. package/dist/workflows/index.d.ts +0 -24
  85. package/dist/workflows/index.d.ts.map +0 -1
  86. package/dist/workflows/trading.d.ts +0 -72
  87. package/dist/workflows/trading.d.ts.map +0 -1
  88. package/src/agent.ts +0 -111
  89. package/src/channels/base.ts +0 -574
  90. package/src/channels/discord.ts +0 -306
  91. package/src/channels/index.ts +0 -169
  92. package/src/channels/telegram.ts +0 -316
  93. package/src/daemon.ts +0 -534
  94. package/src/hooks.ts +0 -97
  95. package/src/memory.ts +0 -369
  96. package/src/skills/coding/commit.ts +0 -202
  97. package/src/skills/coding/execute-subtask.ts +0 -136
  98. package/src/skills/coding/fix-issues.ts +0 -126
  99. package/src/skills/coding/index.ts +0 -26
  100. package/src/skills/coding/plan-task.ts +0 -158
  101. package/src/skills/coding/quality-check.ts +0 -155
  102. package/src/skills/index.ts +0 -65
  103. package/src/skills/registry.ts +0 -380
  104. package/src/skills/shared/index.ts +0 -21
  105. package/src/skills/shared/reflect.ts +0 -156
  106. package/src/skills/shared/review.ts +0 -201
  107. package/src/skills/shared/trajectory.ts +0 -326
  108. package/src/skills/trading/analyze-market.ts +0 -144
  109. package/src/skills/trading/check-risk.ts +0 -176
  110. package/src/skills/trading/execute-trade.ts +0 -185
  111. package/src/skills/trading/generate-signal.ts +0 -160
  112. package/src/skills/trading/index.ts +0 -26
  113. package/src/skills/trading/monitor-position.ts +0 -179
  114. package/src/skills/types.ts +0 -235
  115. package/src/skills/workflows.ts +0 -340
  116. package/src/state.ts +0 -77
  117. package/src/tools.ts +0 -134
  118. package/src/workflow.ts +0 -341
  119. package/src/workflows/coding.ts +0 -580
  120. package/src/workflows/index.ts +0 -61
  121. package/src/workflows/trading.ts +0 -608
package/src/core.ts ADDED
@@ -0,0 +1,476 @@
1
+ /**
2
+ * Daemon Core - Generic Daemon Primitive
3
+ *
4
+ * Base class for all daemons. Provides:
5
+ * - Lifecycle management (start/stop)
6
+ * - Signal handling (SIGINT, SIGTERM, SIGHUP)
7
+ * - Channel registration and routing
8
+ * - HTTP health/status API
9
+ * - State persistence
10
+ * - Hook system
11
+ *
12
+ * @module @ebowwa/daemons/core
13
+ */
14
+
15
+ import { promises as fs } from "fs";
16
+ import type {
17
+ Channel,
18
+ ChannelMessage,
19
+ HealthStatus,
20
+ DaemonStatus,
21
+ } from "./types.js";
22
+
23
+ // ============================================================
24
+ // Core Types
25
+ // ============================================================
26
+
27
+ export interface DaemonCoreConfig {
28
+ /** Unique daemon identifier */
29
+ id: string;
30
+ /** Human-readable name */
31
+ name: string;
32
+ /** HTTP API config */
33
+ api?: {
34
+ port?: number;
35
+ host?: string;
36
+ enabled?: boolean;
37
+ };
38
+ /** State persistence */
39
+ state?: {
40
+ path?: string;
41
+ autoSave?: boolean;
42
+ saveInterval?: number;
43
+ };
44
+ /** Graceful shutdown timeout (ms) */
45
+ shutdownTimeout?: number;
46
+ }
47
+
48
+ export interface DaemonCoreState {
49
+ status: DaemonStatus;
50
+ startedAt: string | null;
51
+ stoppedAt: string | null;
52
+ error: string | null;
53
+ metadata: Record<string, unknown>;
54
+ }
55
+
56
+ export type CoreHookName =
57
+ | "onStart"
58
+ | "onStop"
59
+ | "onError"
60
+ | "onChannelRegister"
61
+ | "onChannelStart"
62
+ | "onChannelStop"
63
+ | "onMessage"
64
+ | "onHealthCheck";
65
+
66
+ export type CoreHookCallback<P = unknown, R = void> = (payload: P) => Promise<R> | R;
67
+
68
+ export interface DaemonPlugin {
69
+ id: string;
70
+ name: string;
71
+ init(daemon: DaemonCore): Promise<void>;
72
+ destroy?(): Promise<void>;
73
+ }
74
+
75
+ // ============================================================
76
+ // Daemon Core Class
77
+ // ============================================================
78
+
79
+ const DEFAULT_STATE_PATH = ".daemon-state.json";
80
+ const DEFAULT_SHUTDOWN_TIMEOUT = 10000;
81
+
82
+ export class DaemonCore {
83
+ protected config: DaemonCoreConfig;
84
+ protected state: DaemonCoreState;
85
+ protected channels: Map<string, Channel> = new Map();
86
+ private hooks: Map<CoreHookName, Set<CoreHookCallback>> = new Map();
87
+ private plugins: Map<string, DaemonPlugin> = new Map();
88
+ private messageHandler: ((msg: ChannelMessage) => Promise<void>) | null = null;
89
+ private httpServer: ReturnType<typeof Bun.serve> | null = null;
90
+ private stateSaveInterval: Timer | null = null;
91
+ protected shuttingDown = false;
92
+
93
+ constructor(config: DaemonCoreConfig) {
94
+ this.config = {
95
+ ...config,
96
+ api: config.api ?? { enabled: false },
97
+ state: config.state ?? { autoSave: false },
98
+ shutdownTimeout: config.shutdownTimeout ?? DEFAULT_SHUTDOWN_TIMEOUT,
99
+ };
100
+
101
+ this.state = {
102
+ status: "stopped",
103
+ startedAt: null,
104
+ stoppedAt: null,
105
+ error: null,
106
+ metadata: {},
107
+ };
108
+
109
+ this.setupSignalHandlers();
110
+ }
111
+
112
+ // ============================================================
113
+ // Lifecycle
114
+ // ============================================================
115
+
116
+ async start(): Promise<void> {
117
+ if (this.state.status === "running") {
118
+ throw new Error("Daemon already running");
119
+ }
120
+
121
+ this.state.status = "starting";
122
+ this.log("Starting...");
123
+
124
+ try {
125
+ await this.loadState();
126
+ await this.executeHooks("onStart", this.state);
127
+
128
+ for (const [id, channel] of this.channels) {
129
+ await this.startChannel(id, channel);
130
+ }
131
+
132
+ if (this.config.api?.enabled !== false && this.config.api?.port) {
133
+ this.startHTTP();
134
+ }
135
+
136
+ if (this.config.state?.autoSave && this.config.state.saveInterval) {
137
+ this.stateSaveInterval = setInterval(
138
+ () => this.saveState(),
139
+ this.config.state.saveInterval
140
+ );
141
+ }
142
+
143
+ this.state.status = "running";
144
+ this.state.startedAt = new Date().toISOString();
145
+ this.state.error = null;
146
+
147
+ this.log("Running");
148
+ } catch (error) {
149
+ this.state.status = "error";
150
+ this.state.error = error instanceof Error ? error.message : String(error);
151
+ await this.executeHooks("onError", error);
152
+ throw error;
153
+ }
154
+ }
155
+
156
+ async stop(): Promise<void> {
157
+ if (this.state.status !== "running") return;
158
+ if (this.shuttingDown) return;
159
+
160
+ this.shuttingDown = true;
161
+ this.state.status = "stopping";
162
+ this.log("Stopping...");
163
+
164
+ const timeout = this.config.shutdownTimeout!;
165
+
166
+ try {
167
+ if (this.stateSaveInterval) {
168
+ clearInterval(this.stateSaveInterval);
169
+ this.stateSaveInterval = null;
170
+ }
171
+
172
+ await Promise.race([
173
+ this.stopAllChannels(),
174
+ new Promise((_, reject) =>
175
+ setTimeout(() => reject(new Error("Channel stop timeout")), timeout / 2)
176
+ ),
177
+ ]);
178
+
179
+ if (this.httpServer) {
180
+ this.httpServer.stop();
181
+ this.httpServer = null;
182
+ }
183
+
184
+ await this.executeHooks("onStop", this.state);
185
+
186
+ for (const plugin of this.plugins.values()) {
187
+ await plugin.destroy?.();
188
+ }
189
+
190
+ await this.saveState();
191
+
192
+ this.state.status = "stopped";
193
+ this.state.stoppedAt = new Date().toISOString();
194
+ this.log("Stopped");
195
+ } catch (error) {
196
+ this.state.status = "error";
197
+ this.state.error = error instanceof Error ? error.message : String(error);
198
+ this.log(`Stop error: ${error}`);
199
+ } finally {
200
+ this.shuttingDown = false;
201
+ }
202
+ }
203
+
204
+ async restart(): Promise<void> {
205
+ await this.stop();
206
+ await this.start();
207
+ }
208
+
209
+ // ============================================================
210
+ // Status & Health
211
+ // ============================================================
212
+
213
+ getStatus(): DaemonStatus {
214
+ return this.state.status;
215
+ }
216
+
217
+ getState(): Readonly<DaemonCoreState> {
218
+ return { ...this.state };
219
+ }
220
+
221
+ async getHealth(): Promise<HealthStatus> {
222
+ const channelStatus: Record<string, { running: boolean; error?: string }> = {};
223
+
224
+ for (const [id, channel] of this.channels) {
225
+ channelStatus[id] = { running: channel.isRunning() };
226
+ }
227
+
228
+ const hookResults = await this.executeHooks("onHealthCheck", null) as unknown[];
229
+ const checks: Record<string, { passed: boolean; message?: string }> = {};
230
+
231
+ hookResults.forEach((result, i) => {
232
+ if (result && typeof result === "object" && "passed" in result) {
233
+ checks[`check-${i}`] = result as { passed: boolean; message?: string };
234
+ }
235
+ });
236
+
237
+ const allChannelsRunning = Object.values(channelStatus).every(c => c.running);
238
+ const allChecksPassed = Object.values(checks).every(c => c.passed);
239
+
240
+ let status: "healthy" | "degraded" | "unhealthy";
241
+ if (!allChannelsRunning) status = "unhealthy";
242
+ else if (!allChecksPassed) status = "degraded";
243
+ else status = "healthy";
244
+
245
+ return {
246
+ status,
247
+ uptime: this.state.startedAt ? Date.now() - new Date(this.state.startedAt).getTime() : 0,
248
+ channels: channelStatus,
249
+ checks,
250
+ };
251
+ }
252
+
253
+ // ============================================================
254
+ // Channels
255
+ // ============================================================
256
+
257
+ registerChannel(channel: Channel): void {
258
+ if (this.channels.has(channel.id)) {
259
+ throw new Error(`Channel already registered: ${channel.id}`);
260
+ }
261
+
262
+ this.channels.set(channel.id, channel);
263
+ channel.onMessage((msg) => this.handleChannelMessage(channel.id, msg));
264
+
265
+ this.executeHooks("onChannelRegister", channel);
266
+ this.log(`Channel registered: ${channel.id}`);
267
+ }
268
+
269
+ getChannel(id: string): Channel | undefined {
270
+ return this.channels.get(id);
271
+ }
272
+
273
+ getChannels(): Channel[] {
274
+ return Array.from(this.channels.values());
275
+ }
276
+
277
+ private async startChannel(id: string, channel: Channel): Promise<void> {
278
+ await channel.start();
279
+ await this.executeHooks("onChannelStart", { id, channel });
280
+ this.log(`Channel started: ${id}`);
281
+ }
282
+
283
+ private async stopAllChannels(): Promise<void> {
284
+ const stops = Array.from(this.channels.entries()).map(async ([id, channel]) => {
285
+ try {
286
+ await channel.stop();
287
+ await this.executeHooks("onChannelStop", { id, channel });
288
+ } catch (error) {
289
+ this.log(`Channel stop failed: ${id}`);
290
+ }
291
+ });
292
+
293
+ await Promise.allSettled(stops);
294
+ }
295
+
296
+ private async handleChannelMessage(channelId: string, msg: unknown): Promise<void> {
297
+ const message = msg as ChannelMessage;
298
+ await this.executeHooks("onMessage", { channelId, message });
299
+
300
+ if (this.messageHandler) {
301
+ try {
302
+ await this.messageHandler(message);
303
+ } catch (error) {
304
+ this.log(`Message handler error: ${error}`);
305
+ }
306
+ }
307
+ }
308
+
309
+ // ============================================================
310
+ // Message Handler
311
+ // ============================================================
312
+
313
+ onMessage(handler: (msg: ChannelMessage) => Promise<void>): void {
314
+ this.messageHandler = handler;
315
+ }
316
+
317
+ // ============================================================
318
+ // Hooks
319
+ // ============================================================
320
+
321
+ registerHook(name: CoreHookName, callback: CoreHookCallback): void {
322
+ if (!this.hooks.has(name)) this.hooks.set(name, new Set());
323
+ this.hooks.get(name)!.add(callback);
324
+ }
325
+
326
+ onStart(cb: CoreHookCallback): void { this.registerHook("onStart", cb); }
327
+ onStop(cb: CoreHookCallback): void { this.registerHook("onStop", cb); }
328
+ onError(cb: CoreHookCallback): void { this.registerHook("onError", cb); }
329
+ onHealthCheck(cb: CoreHookCallback): void { this.registerHook("onHealthCheck", cb); }
330
+
331
+ protected async executeHooks(name: CoreHookName, payload: unknown): Promise<unknown[]> {
332
+ const callbacks = this.hooks.get(name);
333
+ if (!callbacks) return [];
334
+
335
+ const results = [];
336
+ for (const cb of callbacks) {
337
+ try {
338
+ results.push(await cb(payload));
339
+ } catch (error) {
340
+ this.log(`Hook ${name} error: ${error}`);
341
+ }
342
+ }
343
+ return results;
344
+ }
345
+
346
+ // ============================================================
347
+ // Plugins
348
+ // ============================================================
349
+
350
+ async use(plugin: DaemonPlugin): Promise<void> {
351
+ if (this.plugins.has(plugin.id)) {
352
+ throw new Error(`Plugin already registered: ${plugin.id}`);
353
+ }
354
+ await plugin.init(this);
355
+ this.plugins.set(plugin.id, plugin);
356
+ this.log(`Plugin loaded: ${plugin.name}`);
357
+ }
358
+
359
+ // ============================================================
360
+ // State Persistence
361
+ // ============================================================
362
+
363
+ updateState(updates: Partial<DaemonCoreState["metadata"]>): void {
364
+ this.state.metadata = { ...this.state.metadata, ...updates };
365
+ }
366
+
367
+ private async loadState(): Promise<void> {
368
+ const path = this.config.state?.path ?? DEFAULT_STATE_PATH;
369
+ try {
370
+ const data = await fs.readFile(path, "utf-8");
371
+ const saved = JSON.parse(data);
372
+ this.state.metadata = saved.metadata ?? {};
373
+ } catch {
374
+ // No saved state
375
+ }
376
+ }
377
+
378
+ private async saveState(): Promise<void> {
379
+ const path = this.config.state?.path ?? DEFAULT_STATE_PATH;
380
+ try {
381
+ await fs.writeFile(path, JSON.stringify({
382
+ status: this.state.status,
383
+ metadata: this.state.metadata,
384
+ savedAt: new Date().toISOString(),
385
+ }, null, 2));
386
+ } catch (error) {
387
+ this.log(`State save failed: ${error}`);
388
+ }
389
+ }
390
+
391
+ // ============================================================
392
+ // HTTP API
393
+ // ============================================================
394
+
395
+ private startHTTP(): void {
396
+ const port = this.config.api?.port ?? 8911;
397
+ const hostname = this.config.api?.host ?? "0.0.0.0";
398
+
399
+ this.httpServer = Bun.serve({
400
+ port,
401
+ hostname,
402
+ fetch: async (req) => {
403
+ const url = new URL(req.url);
404
+ const path = url.pathname;
405
+
406
+ const corsHeaders = {
407
+ "Access-Control-Allow-Origin": "*",
408
+ "Access-Control-Allow-Methods": "GET, OPTIONS",
409
+ "Access-Control-Allow-Headers": "Content-Type",
410
+ };
411
+
412
+ if (req.method === "OPTIONS") {
413
+ return new Response(null, { headers: corsHeaders });
414
+ }
415
+
416
+ if (path === "/health") {
417
+ const health = await this.getHealth();
418
+ return Response.json(health, { headers: corsHeaders });
419
+ }
420
+
421
+ if (path === "/api/status" || path === "/status") {
422
+ return Response.json({
423
+ id: this.config.id,
424
+ name: this.config.name,
425
+ status: this.state.status,
426
+ startedAt: this.state.startedAt,
427
+ channels: Object.fromEntries(
428
+ Array.from(this.channels.entries()).map(([id, ch]) => [id, ch.isRunning()])
429
+ ),
430
+ }, { headers: corsHeaders });
431
+ }
432
+
433
+ if (path === "/api/channels") {
434
+ return Response.json(
435
+ this.getChannels().map(ch => ({ id: ch.id, type: ch.type, running: ch.isRunning() })),
436
+ { headers: corsHeaders }
437
+ );
438
+ }
439
+
440
+ return Response.json({ error: "Not found" }, { status: 404, headers: corsHeaders });
441
+ },
442
+ });
443
+
444
+ this.log(`API: http://${hostname}:${port}`);
445
+ }
446
+
447
+ // ============================================================
448
+ // Signal Handlers
449
+ // ============================================================
450
+
451
+ private setupSignalHandlers(): void {
452
+ const handleSignal = async (signal: string) => {
453
+ this.log(`Received ${signal}`);
454
+ await this.stop();
455
+ process.exit(0);
456
+ };
457
+
458
+ process.on("SIGINT", () => handleSignal("SIGINT"));
459
+ process.on("SIGTERM", () => handleSignal("SIGTERM"));
460
+
461
+ process.on("SIGHUP", async () => {
462
+ this.log("Received SIGHUP - saving state");
463
+ await this.saveState();
464
+ });
465
+ }
466
+
467
+ // ============================================================
468
+ // Logging
469
+ // ============================================================
470
+
471
+ protected log(message: string): void {
472
+ console.log(`[${this.config.name}] ${message}`);
473
+ }
474
+ }
475
+
476
+ export default DaemonCore;
package/src/index.ts CHANGED
@@ -1,111 +1,33 @@
1
1
  /**
2
- * Daemons - Autonomous Agent Daemon Framework
2
+ * @ebowwa/daemons
3
3
  *
4
- * A daemon framework for autonomous AI agents.
5
- * Implements SLAM pattern (State Loop Action Memory).
6
- * Supports hooks, tools via MCP, multi-agent coordination, and communication channels.
4
+ * Framework for building daemons.
5
+ * Provides DaemonCore primitive with lifecycle, signals, channels, HTTP API, state persistence.
6
+ *
7
+ * @example
8
+ * ```ts
9
+ * import { DaemonCore } from "@ebowwa/daemons";
10
+ *
11
+ * class MyDaemon extends DaemonCore {
12
+ * async start() {
13
+ * // Register channels, set up handlers
14
+ * await super.start();
15
+ * }
16
+ * }
17
+ * ```
7
18
  *
8
19
  * @module @ebowwa/daemons
9
20
  */
10
21
 
11
- // Main daemon class
12
- export { GLMDaemon, type GLMDaemonConfigWithWorkflow } from "./daemon.js";
13
-
14
- // Agent class
15
- export { GLMAgent } from "./agent.js";
16
-
17
- // Supporting systems
18
- export { HookSystem } from "./hooks.js";
19
- export { ToolBridge } from "./tools.js";
20
- export { StateManager } from "./state.js";
21
-
22
- // Workflow system
23
- export {
24
- // Classes
25
- BaseWorkflow,
26
- WorkflowRegistry,
27
- workflowRegistry,
28
- // Functions
29
- initializeWorkflows,
30
- getWorkflow,
31
- getDefaultWorkflow,
32
- // Built-in workflows
33
- CodingWorkflow,
34
- codingWorkflow,
35
- CODING_PHASES,
36
- CODING_TRANSITIONS,
37
- CODING_WORKFLOW_CONFIG,
38
- TradingWorkflow,
39
- tradingWorkflow,
40
- TRADING_PHASES,
41
- TRADING_TRANSITIONS,
42
- TRADING_WORKFLOW_CONFIG,
43
- // Types
44
- type GLMWorkflowPhase,
45
- type GLMWorkflowContext,
46
- type GLMWorkflowPhaseResult,
47
- type GLMWorkflowTransition,
48
- type GLMWorkflowConfig,
49
- type GLMWorkflowExecutor,
50
- type CodingPhase,
51
- type TradingPhase,
52
- } from "./workflows/index.js";
53
-
54
- // Built-in tools - re-exported from @ebowwa/ai/tools for convenience
22
+ // Generic daemon core
55
23
  export {
56
- BUILTIN_TOOLS,
57
- getBuiltinTool,
58
- getBuiltinToolNames,
59
- toGLMFormat,
60
- executeBuiltinTool,
61
- ToolExecutor,
62
- type ToolDefinition,
63
- type ToolCall,
64
- type ToolExecutorOptions,
65
- type ToolExecutionResult,
66
- } from "@ebowwa/ai/tools";
67
-
68
- // Memory systems
69
- export {
70
- ConversationMemory,
71
- NumericConversationMemory,
72
- StringConversationMemory,
73
- type ConversationMessage,
74
- type ConversationMemoryConfig,
75
- } from "./memory.js";
76
-
77
- // Communication channels
78
- export {
79
- BaseChannel,
80
- TelegramChannel,
81
- DiscordChannel,
82
- ChannelRegistry,
83
- type BaseChannelConfig,
84
- type TelegramChannelConfig,
85
- type DiscordChannelConfig,
86
- type MessageContext,
87
- type RouteResult,
88
- type MessageClassification,
89
- } from "./channels/index.js";
24
+ DaemonCore,
25
+ type DaemonCoreConfig,
26
+ type DaemonCoreState,
27
+ type CoreHookName,
28
+ type CoreHookCallback,
29
+ type DaemonPlugin,
30
+ } from "./core.js";
90
31
 
91
32
  // Types
92
33
  export * from "./types.js";
93
-
94
- // Convenience function to create and start a daemon
95
- import { GLMDaemon } from "./daemon.js";
96
- import type { GLMDaemonConfig } from "./types.js";
97
-
98
- export async function createGLMDaemon(
99
- config: GLMDaemonConfig
100
- ): Promise<GLMDaemon> {
101
- const daemon = new GLMDaemon(config);
102
- return daemon;
103
- }
104
-
105
- export async function startGLMDaemon(
106
- config: GLMDaemonConfig,
107
- prompt: string
108
- ): Promise<string> {
109
- const daemon = new GLMDaemon(config);
110
- return await daemon.start(prompt);
111
- }