@sesamespace/hivemind 0.3.1 → 0.4.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/dist/chunk-4LHKQFQT.js +731 -0
- package/dist/chunk-4LHKQFQT.js.map +1 -0
- package/dist/chunk-MBS5A6BZ.js +132 -0
- package/dist/chunk-MBS5A6BZ.js.map +1 -0
- package/dist/chunk-NLMP7MYR.js +44 -0
- package/dist/chunk-NLMP7MYR.js.map +1 -0
- package/dist/chunk-O4CT7SDB.js +186 -0
- package/dist/chunk-O4CT7SDB.js.map +1 -0
- package/dist/chunk-VRDYBNAA.js +191 -0
- package/dist/chunk-VRDYBNAA.js.map +1 -0
- package/dist/chunk-WSOOI3AY.js +1598 -0
- package/dist/chunk-WSOOI3AY.js.map +1 -0
- package/dist/commands/fleet.js +9 -0
- package/dist/commands/fleet.js.map +1 -0
- package/dist/commands/init.js +7 -0
- package/dist/commands/init.js.map +1 -0
- package/dist/commands/service.js +7 -0
- package/dist/commands/service.js.map +1 -0
- package/dist/commands/start.js +9 -0
- package/dist/commands/start.js.map +1 -0
- package/dist/index.js +41 -0
- package/dist/index.js.map +1 -0
- package/dist/main.js +195 -0
- package/dist/main.js.map +1 -0
- package/dist/start.js +22 -0
- package/dist/start.js.map +1 -0
- package/package.json +4 -1
- package/packages/runtime/src/pipeline.ts +36 -0
- package/packages/runtime/src/sesame.ts +52 -0
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../packages/runtime/src/fleet/primary-client.ts","../packages/runtime/src/fleet/fleet-manager.ts","../packages/runtime/src/fleet/memory-sync.ts"],"sourcesContent":["/**\n * Primary-side client for managing Workers.\n *\n * The Primary uses this to:\n * - Track registered workers\n * - Poll worker health\n * - Assign/unassign contexts to workers\n * - Collect status reports\n */\n\nimport type {\n WorkerInfo,\n WorkerRegistrationRequest,\n WorkerRegistrationResponse,\n WorkerHealthResponse,\n WorkerHealthStatus,\n WorkerStatus,\n WorkerStatusReport,\n ContextAssignRequest,\n ContextAssignResponse,\n SyncPushRequest,\n SyncPushResponse,\n SyncPullRequest,\n SyncPullResponse,\n} from \"./worker-protocol.js\";\n\nimport {\n PRIMARY_ROUTES,\n WORKER_ROUTES,\n DEFAULT_HEALTH_INTERVAL_MS,\n HEALTH_TIMEOUT_MS,\n} from \"./worker-protocol.js\";\n\nexport class PrimaryClient {\n private workers: Map<string, WorkerInfo> = new Map();\n private healthTimer: ReturnType<typeof setInterval> | null = null;\n private nextId = 1;\n private onSyncPullCallback: ((req: SyncPullRequest) => Promise<SyncPullResponse>) | null = null;\n\n /** All registered workers. */\n getWorkers(): WorkerInfo[] {\n return Array.from(this.workers.values());\n }\n\n /** Get a single worker by ID. */\n getWorker(workerId: string): WorkerInfo | undefined {\n return this.workers.get(workerId);\n }\n\n /** Find which worker owns a context, if any. */\n findWorkerForContext(contextName: string): WorkerInfo | undefined {\n for (const w of this.workers.values()) {\n if (w.assigned_contexts.includes(contextName)) return w;\n }\n return undefined;\n }\n\n // --- Registration (called when Worker POSTs to Primary) ---\n\n handleRegistration(req: WorkerRegistrationRequest): WorkerRegistrationResponse {\n const id = `worker-${this.nextId++}`;\n const now = new Date().toISOString();\n\n const info: WorkerInfo = {\n id,\n url: req.url,\n capabilities: req.capabilities,\n assigned_contexts: [],\n registered_at: now,\n last_heartbeat: now,\n };\n\n this.workers.set(id, info);\n return { worker_id: id, registered_at: now };\n }\n\n /** Remove a worker from the registry. */\n deregister(workerId: string): boolean {\n return this.workers.delete(workerId);\n }\n\n // --- Health Checking ---\n\n /** Poll a single worker's health endpoint. */\n async checkHealth(workerId: string): Promise<WorkerHealthResponse | null> {\n const worker = this.workers.get(workerId);\n if (!worker) return null;\n\n try {\n const controller = new AbortController();\n const timeout = setTimeout(() => controller.abort(), HEALTH_TIMEOUT_MS);\n\n const resp = await fetch(`${worker.url}${WORKER_ROUTES.health}`, {\n signal: controller.signal,\n });\n clearTimeout(timeout);\n\n if (!resp.ok) {\n this.markUnreachable(workerId);\n return null;\n }\n\n const health = (await resp.json()) as WorkerHealthResponse;\n worker.last_heartbeat = new Date().toISOString();\n return health;\n } catch {\n this.markUnreachable(workerId);\n return null;\n }\n }\n\n /** Poll all workers and return their statuses. */\n async checkAllHealth(): Promise<Map<string, WorkerHealthStatus>> {\n const results = new Map<string, WorkerHealthStatus>();\n\n const checks = Array.from(this.workers.keys()).map(async (id) => {\n const health = await this.checkHealth(id);\n results.set(id, health?.status ?? \"unreachable\");\n });\n\n await Promise.all(checks);\n return results;\n }\n\n /** Start periodic health polling. */\n startHealthPolling(intervalMs = DEFAULT_HEALTH_INTERVAL_MS): void {\n this.stopHealthPolling();\n this.healthTimer = setInterval(() => {\n this.checkAllHealth().catch(() => {});\n }, intervalMs);\n }\n\n /** Stop periodic health polling. */\n stopHealthPolling(): void {\n if (this.healthTimer) {\n clearInterval(this.healthTimer);\n this.healthTimer = null;\n }\n }\n\n // --- Context Assignment ---\n\n /** Assign a context to a worker. */\n async assignContext(\n workerId: string,\n contextName: string,\n contextDescription = \"\",\n ): Promise<ContextAssignResponse> {\n const worker = this.workers.get(workerId);\n if (!worker) {\n return { context_name: contextName, accepted: false, reason: \"Worker not found\" };\n }\n\n if (worker.assigned_contexts.length >= worker.capabilities.max_contexts) {\n return { context_name: contextName, accepted: false, reason: \"Worker at capacity\" };\n }\n\n const body: ContextAssignRequest = {\n context_name: contextName,\n context_description: contextDescription,\n };\n\n try {\n const resp = await fetch(`${worker.url}${WORKER_ROUTES.assign}`, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify(body),\n });\n\n if (!resp.ok) {\n const text = await resp.text();\n return { context_name: contextName, accepted: false, reason: `Worker rejected: ${text}` };\n }\n\n const result = (await resp.json()) as ContextAssignResponse;\n\n if (result.accepted) {\n worker.assigned_contexts.push(contextName);\n }\n\n return result;\n } catch (err) {\n const msg = err instanceof Error ? err.message : String(err);\n return { context_name: contextName, accepted: false, reason: `Request failed: ${msg}` };\n }\n }\n\n /** Unassign a context from a worker. */\n async unassignContext(workerId: string, contextName: string): Promise<boolean> {\n const worker = this.workers.get(workerId);\n if (!worker) return false;\n\n try {\n const resp = await fetch(`${worker.url}${WORKER_ROUTES.unassign(contextName)}`, {\n method: \"DELETE\",\n });\n\n if (resp.ok) {\n worker.assigned_contexts = worker.assigned_contexts.filter((c) => c !== contextName);\n return true;\n }\n return false;\n } catch {\n return false;\n }\n }\n\n // --- Status Collection ---\n\n /** Handle an incoming status report from a Worker. */\n handleStatusReport(workerId: string, report: WorkerStatusReport): WorkerStatus | null {\n const worker = this.workers.get(workerId);\n if (!worker) return null;\n\n worker.last_heartbeat = new Date().toISOString();\n\n return {\n worker_id: workerId,\n activity: report.activity,\n current_context: report.current_context,\n current_task: report.current_task,\n error: report.error,\n reported_at: new Date().toISOString(),\n };\n }\n\n // --- Memory Sync ---\n\n /** Register a handler for incoming sync pull requests from Workers. */\n onSyncPull(cb: (req: SyncPullRequest) => Promise<SyncPullResponse>): void {\n this.onSyncPullCallback = cb;\n }\n\n /** Handle an incoming sync pull (Worker sends L3 knowledge to Primary). */\n async handleSyncPull(req: SyncPullRequest): Promise<SyncPullResponse> {\n if (!this.onSyncPullCallback) {\n return { accepted: 0, rejected: req.entries.length };\n }\n return this.onSyncPullCallback(req);\n }\n\n /** Push Global context updates to a specific worker. */\n async pushSyncToWorker(workerId: string, payload: SyncPushRequest): Promise<SyncPushResponse | null> {\n const worker = this.workers.get(workerId);\n if (!worker) return null;\n\n try {\n const controller = new AbortController();\n const timeout = setTimeout(() => controller.abort(), HEALTH_TIMEOUT_MS);\n\n const resp = await fetch(`${worker.url}${WORKER_ROUTES.syncPush}`, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify(payload),\n signal: controller.signal,\n });\n clearTimeout(timeout);\n\n if (!resp.ok) return null;\n return (await resp.json()) as SyncPushResponse;\n } catch {\n return null;\n }\n }\n\n /** Push Global context updates to all registered workers. */\n async pushSyncToAll(payload: SyncPushRequest): Promise<Map<string, SyncPushResponse | null>> {\n const results = new Map<string, SyncPushResponse | null>();\n\n const pushes = Array.from(this.workers.keys()).map(async (id) => {\n const result = await this.pushSyncToWorker(id, payload);\n results.set(id, result);\n });\n\n await Promise.all(pushes);\n return results;\n }\n\n // --- Internal ---\n\n private markUnreachable(workerId: string): void {\n // Worker stays registered but health is tracked via last_heartbeat staleness.\n // Fleet manager (Phase 3.3) will decide when to evict.\n }\n}\n","/**\n * Fleet Manager — high-level orchestration layer on top of PrimaryClient.\n *\n * Provides:\n * - Worker provisioning (register by URL)\n * - Context assignment and migration between workers\n * - Aggregate fleet status / dashboard data\n * - Worker discovery and removal\n */\n\nimport { PrimaryClient } from \"./primary-client.js\";\nimport {\n WORKER_ROUTES,\n HEALTH_TIMEOUT_MS,\n} from \"./worker-protocol.js\";\nimport type {\n WorkerInfo,\n WorkerHealthResponse,\n WorkerHealthStatus,\n WorkerCapabilities,\n ContextAssignResponse,\n WorkerStatus,\n WorkerActivity,\n} from \"./worker-protocol.js\";\n\n// --- Dashboard Types ---\n\nexport interface WorkerSummary {\n id: string;\n url: string;\n health: WorkerHealthStatus;\n contexts: string[];\n current_task: string | null;\n activity: WorkerActivity;\n uptime_seconds: number | null;\n capabilities: WorkerCapabilities;\n last_heartbeat: string;\n}\n\nexport interface FleetDashboard {\n total_workers: number;\n healthy: number;\n degraded: number;\n unreachable: number;\n total_contexts: number;\n workers: WorkerSummary[];\n unassigned_contexts: string[];\n generated_at: string;\n}\n\nexport interface MigrationResult {\n context_name: string;\n from_worker: string;\n to_worker: string;\n success: boolean;\n reason?: string;\n}\n\nexport class FleetManager {\n private primary: PrimaryClient;\n private latestHealth: Map<string, WorkerHealthResponse> = new Map();\n private latestStatus: Map<string, WorkerStatus> = new Map();\n private knownContexts: Set<string> = new Set();\n\n constructor(primary?: PrimaryClient) {\n this.primary = primary ?? new PrimaryClient();\n }\n\n getPrimary(): PrimaryClient {\n return this.primary;\n }\n\n // --- Worker Provisioning ---\n\n /**\n * Register a new worker by URL. Probes the worker's health endpoint\n * to discover capabilities, then registers it with the Primary.\n */\n async addWorker(url: string): Promise<WorkerInfo> {\n // Normalise trailing slash\n const baseUrl = url.replace(/\\/+$/, \"\");\n\n // Probe worker health to discover capabilities\n const capabilities = await this.probeWorker(baseUrl);\n\n const reg = this.primary.handleRegistration({ url: baseUrl, capabilities });\n\n const worker = this.primary.getWorker(reg.worker_id);\n if (!worker) {\n throw new Error(\"Worker registered but not found in registry\");\n }\n\n return worker;\n }\n\n /** Remove a worker from the fleet, unassigning its contexts first. */\n async removeWorker(workerId: string): Promise<boolean> {\n const worker = this.primary.getWorker(workerId);\n if (!worker) return false;\n\n // Unassign all contexts from this worker\n for (const ctx of [...worker.assigned_contexts]) {\n await this.primary.unassignContext(workerId, ctx);\n }\n\n return this.primary.deregister(workerId);\n }\n\n // --- Context Assignment ---\n\n /** Assign a context to a specific worker. */\n async assignContext(\n workerId: string,\n contextName: string,\n description = \"\",\n ): Promise<ContextAssignResponse> {\n const result = await this.primary.assignContext(workerId, contextName, description);\n if (result.accepted) {\n this.knownContexts.add(contextName);\n }\n return result;\n }\n\n /** Migrate a context from one worker to another. */\n async migrateContext(\n contextName: string,\n toWorkerId: string,\n ): Promise<MigrationResult> {\n const fromWorker = this.primary.findWorkerForContext(contextName);\n\n if (!fromWorker) {\n // Context isn't assigned anywhere — just assign to target\n const resp = await this.primary.assignContext(toWorkerId, contextName);\n return {\n context_name: contextName,\n from_worker: \"(none)\",\n to_worker: toWorkerId,\n success: resp.accepted,\n reason: resp.accepted ? undefined : resp.reason,\n };\n }\n\n if (fromWorker.id === toWorkerId) {\n return {\n context_name: contextName,\n from_worker: fromWorker.id,\n to_worker: toWorkerId,\n success: false,\n reason: \"Context already assigned to this worker\",\n };\n }\n\n // Assign to new worker first, then unassign from old\n const assignResult = await this.primary.assignContext(toWorkerId, contextName);\n if (!assignResult.accepted) {\n return {\n context_name: contextName,\n from_worker: fromWorker.id,\n to_worker: toWorkerId,\n success: false,\n reason: assignResult.reason ?? \"Target worker rejected assignment\",\n };\n }\n\n await this.primary.unassignContext(fromWorker.id, contextName);\n\n return {\n context_name: contextName,\n from_worker: fromWorker.id,\n to_worker: toWorkerId,\n success: true,\n };\n }\n\n // --- Status & Health ---\n\n /** Refresh health for all workers. */\n async refreshHealth(): Promise<void> {\n const workers = this.primary.getWorkers();\n\n const checks = workers.map(async (w) => {\n const health = await this.primary.checkHealth(w.id);\n if (health) {\n this.latestHealth.set(w.id, health);\n } else {\n // Mark unreachable but keep stale data\n this.latestHealth.set(w.id, {\n worker_id: w.id,\n status: \"unreachable\",\n uptime_seconds: 0,\n assigned_contexts: w.assigned_contexts,\n active_context: null,\n memory_daemon_ok: false,\n ollama_ok: false,\n });\n }\n });\n\n await Promise.all(checks);\n }\n\n /** Record a status report (called when worker POSTs status to Primary). */\n recordStatus(workerId: string, status: WorkerStatus): void {\n this.latestStatus.set(workerId, status);\n }\n\n /** Get the full fleet dashboard. */\n async getDashboard(): Promise<FleetDashboard> {\n await this.refreshHealth();\n\n const workers = this.primary.getWorkers();\n let healthy = 0;\n let degraded = 0;\n let unreachable = 0;\n let totalContexts = 0;\n\n const workerSummaries: WorkerSummary[] = workers.map((w) => {\n const health = this.latestHealth.get(w.id);\n const status = this.latestStatus.get(w.id);\n\n const healthStatus: WorkerHealthStatus = health?.status ?? \"unreachable\";\n if (healthStatus === \"healthy\") healthy++;\n else if (healthStatus === \"degraded\") degraded++;\n else unreachable++;\n\n totalContexts += w.assigned_contexts.length;\n\n return {\n id: w.id,\n url: w.url,\n health: healthStatus,\n contexts: w.assigned_contexts,\n current_task: status?.current_task ?? null,\n activity: status?.activity ?? \"idle\",\n uptime_seconds: health?.uptime_seconds ?? null,\n capabilities: w.capabilities,\n last_heartbeat: w.last_heartbeat,\n };\n });\n\n // Determine unassigned contexts\n const assignedContexts = new Set(workers.flatMap((w) => w.assigned_contexts));\n const unassigned = [...this.knownContexts].filter((c) => !assignedContexts.has(c));\n\n return {\n total_workers: workers.length,\n healthy,\n degraded,\n unreachable,\n total_contexts: totalContexts,\n workers: workerSummaries,\n unassigned_contexts: unassigned,\n generated_at: new Date().toISOString(),\n };\n }\n\n /** Get a quick status summary without refreshing health. */\n getStatusSnapshot(): FleetDashboard {\n const workers = this.primary.getWorkers();\n let healthy = 0;\n let degraded = 0;\n let unreachable = 0;\n let totalContexts = 0;\n\n const workerSummaries: WorkerSummary[] = workers.map((w) => {\n const health = this.latestHealth.get(w.id);\n const status = this.latestStatus.get(w.id);\n\n const healthStatus: WorkerHealthStatus = health?.status ?? \"unreachable\";\n if (healthStatus === \"healthy\") healthy++;\n else if (healthStatus === \"degraded\") degraded++;\n else unreachable++;\n\n totalContexts += w.assigned_contexts.length;\n\n return {\n id: w.id,\n url: w.url,\n health: healthStatus,\n contexts: w.assigned_contexts,\n current_task: status?.current_task ?? null,\n activity: status?.activity ?? \"idle\",\n uptime_seconds: health?.uptime_seconds ?? null,\n capabilities: w.capabilities,\n last_heartbeat: w.last_heartbeat,\n };\n });\n\n const assignedContexts = new Set(workers.flatMap((w) => w.assigned_contexts));\n const unassigned = [...this.knownContexts].filter((c) => !assignedContexts.has(c));\n\n return {\n total_workers: workers.length,\n healthy,\n degraded,\n unreachable,\n total_contexts: totalContexts,\n workers: workerSummaries,\n unassigned_contexts: unassigned,\n generated_at: new Date().toISOString(),\n };\n }\n\n // --- Worker Discovery ---\n\n /**\n * Scan a list of candidate URLs for workers. Returns the URLs\n * that responded to a health probe. Useful for auto-detecting\n * workers on a local network.\n */\n async discoverWorkers(candidateUrls: string[]): Promise<string[]> {\n const found: string[] = [];\n\n const probes = candidateUrls.map(async (url) => {\n const baseUrl = url.replace(/\\/+$/, \"\");\n try {\n const controller = new AbortController();\n const timeout = setTimeout(() => controller.abort(), HEALTH_TIMEOUT_MS);\n\n const resp = await fetch(`${baseUrl}${WORKER_ROUTES.health}`, {\n signal: controller.signal,\n });\n clearTimeout(timeout);\n\n if (resp.ok) {\n found.push(baseUrl);\n }\n } catch {\n // Not reachable, skip\n }\n });\n\n await Promise.all(probes);\n return found;\n }\n\n /**\n * Scan a subnet-style range for workers. Generates URLs for\n * a base IP with ports and probes them.\n * Example: scanSubnet(\"192.168.1\", [10, 11, 12], 3100)\n */\n async scanSubnet(\n baseIp: string,\n hostIds: number[],\n port: number,\n ): Promise<string[]> {\n const urls = hostIds.map((id) => `http://${baseIp}.${id}:${port}`);\n return this.discoverWorkers(urls);\n }\n\n /** Track a context name so it appears in unassigned lists. */\n registerContext(contextName: string): void {\n this.knownContexts.add(contextName);\n }\n\n /** Start health polling (delegates to PrimaryClient). */\n startHealthPolling(intervalMs?: number): void {\n this.primary.startHealthPolling(intervalMs);\n }\n\n /** Stop health polling. */\n stopHealthPolling(): void {\n this.primary.stopHealthPolling();\n }\n\n // --- Internal ---\n\n private async probeWorker(baseUrl: string): Promise<WorkerCapabilities> {\n try {\n const controller = new AbortController();\n const timeout = setTimeout(() => controller.abort(), HEALTH_TIMEOUT_MS);\n\n const resp = await fetch(`${baseUrl}${WORKER_ROUTES.health}`, {\n signal: controller.signal,\n });\n clearTimeout(timeout);\n\n if (resp.ok) {\n const health = (await resp.json()) as WorkerHealthResponse;\n return {\n max_contexts: health.assigned_contexts.length + 4, // assume capacity\n has_ollama: health.ollama_ok,\n has_memory_daemon: health.memory_daemon_ok,\n available_models: [],\n };\n }\n } catch {\n // Fall through to defaults\n }\n\n // Return default capabilities if probe fails\n return {\n max_contexts: 4,\n has_ollama: false,\n has_memory_daemon: false,\n available_models: [],\n };\n }\n}\n","/**\n * Cross-machine memory sync.\n *\n * Workers periodically push key L3 knowledge back to the Primary.\n * Primary can push Global context updates (L3 + L2 episodes) to workers.\n *\n * Conflict resolution:\n * - L3: last-write-wins (compared by updated_at timestamp)\n * - L2: append-only (episodes are never overwritten, only added)\n *\n * Sync protocol:\n * POST /sync/pull — Worker sends L3 knowledge to Primary\n * POST /sync/push — Primary sends Global context updates to Worker\n */\n\nimport type { MemoryClient, L3Entry, Episode } from \"../memory-client.js\";\nimport type { PrimaryClient } from \"./primary-client.js\";\nimport type { WorkerServer } from \"./worker-server.js\";\nimport type {\n SyncPullRequest,\n SyncPullResponse,\n SyncPushRequest,\n SyncPushResponse,\n SyncL3Entry,\n SyncL2Episode,\n} from \"./worker-protocol.js\";\nimport { DEFAULT_SYNC_INTERVAL_MS, HEALTH_TIMEOUT_MS, PRIMARY_ROUTES } from \"./worker-protocol.js\";\n\n// --- Helpers ---\n\nfunction l3ToSync(entry: L3Entry): SyncL3Entry {\n return {\n id: entry.id,\n source_episode_id: entry.source_episode_id,\n context_name: entry.context_name,\n content: entry.content,\n promoted_at: entry.promoted_at,\n access_count: entry.access_count,\n connection_density: entry.connection_density,\n updated_at: entry.promoted_at,\n };\n}\n\nfunction episodeToSync(ep: Episode): SyncL2Episode {\n return {\n id: ep.id,\n timestamp: ep.timestamp,\n context_name: ep.context_name,\n role: ep.role,\n content: ep.content,\n };\n}\n\n// --- MemorySync (Worker side) ---\n\nexport interface WorkerSyncOptions {\n workerId: string;\n primaryUrl: string;\n memory: MemoryClient;\n server: WorkerServer;\n syncIntervalMs?: number;\n}\n\n/**\n * Worker-side memory sync. Periodically sends L3 knowledge from assigned\n * contexts back to the Primary, and accepts Global context pushes.\n */\nexport class WorkerMemorySync {\n private workerId: string;\n private primaryUrl: string;\n private memory: MemoryClient;\n private syncTimer: ReturnType<typeof setInterval> | null = null;\n private syncIntervalMs: number;\n private knownL2Ids: Set<string> = new Set();\n private knownL3Ids: Map<string, string> = new Map(); // id -> updated_at\n\n constructor(opts: WorkerSyncOptions) {\n this.workerId = opts.workerId;\n this.primaryUrl = opts.primaryUrl;\n this.memory = opts.memory;\n this.syncIntervalMs = opts.syncIntervalMs ?? DEFAULT_SYNC_INTERVAL_MS;\n\n // Register handler for incoming Global context pushes from Primary\n opts.server.onSyncPush((req) => this.handlePush(req));\n }\n\n /** Start periodic sync (Worker -> Primary). */\n start(contextNames: () => string[]): void {\n this.stop();\n this.syncTimer = setInterval(() => {\n this.pullTowardsPrimary(contextNames()).catch((err) => {\n console.warn(\"[sync] Pull cycle failed:\", (err as Error).message);\n });\n }, this.syncIntervalMs);\n }\n\n /** Stop periodic sync. */\n stop(): void {\n if (this.syncTimer) {\n clearInterval(this.syncTimer);\n this.syncTimer = null;\n }\n }\n\n /**\n * Pull L3 knowledge from all assigned contexts and send to Primary.\n * Called periodically by the sync timer.\n */\n async pullTowardsPrimary(contextNames: string[]): Promise<void> {\n for (const contextName of contextNames) {\n try {\n const entries = await this.memory.getL3Knowledge(contextName);\n if (entries.length === 0) continue;\n\n const syncEntries = entries.map(l3ToSync);\n\n const req: SyncPullRequest = {\n worker_id: this.workerId,\n context_name: contextName,\n entries: syncEntries,\n };\n\n const controller = new AbortController();\n const timeout = setTimeout(() => controller.abort(), HEALTH_TIMEOUT_MS);\n\n const resp = await fetch(`${this.primaryUrl}${PRIMARY_ROUTES.syncPull}`, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify(req),\n signal: controller.signal,\n });\n clearTimeout(timeout);\n\n if (resp.ok) {\n const result = (await resp.json()) as SyncPullResponse;\n if (result.accepted > 0) {\n console.log(`[sync] Pushed ${result.accepted} L3 entries from \"${contextName}\" to Primary`);\n }\n }\n } catch {\n // Primary unreachable — will retry next interval\n }\n }\n }\n\n /**\n * Handle an incoming push of Global context updates from the Primary.\n * L3: last-write-wins. L2: append-only.\n */\n async handlePush(req: SyncPushRequest): Promise<SyncPushResponse> {\n let l3Accepted = 0;\n let l2Appended = 0;\n\n // Process L3 entries with last-write-wins\n for (const entry of req.entries) {\n const existing = this.knownL3Ids.get(entry.id);\n if (existing && existing >= entry.updated_at) {\n // Local version is same or newer — skip\n continue;\n }\n\n // Store in local memory daemon (the memory daemon's L3 store handles upsert)\n try {\n // Store as an episode in the global context so the worker has access\n await this.memory.storeEpisode({\n context_name: entry.context_name,\n role: \"system\",\n content: entry.content,\n });\n this.knownL3Ids.set(entry.id, entry.updated_at);\n l3Accepted++;\n } catch {\n // Non-fatal — entry will be retried on next push\n }\n }\n\n // Process L2 episodes as append-only\n for (const episode of req.episodes) {\n if (this.knownL2Ids.has(episode.id)) {\n continue; // Already stored\n }\n\n try {\n await this.memory.storeEpisode({\n context_name: episode.context_name,\n role: episode.role,\n content: episode.content,\n });\n this.knownL2Ids.add(episode.id);\n l2Appended++;\n } catch {\n // Non-fatal\n }\n }\n\n if (l3Accepted > 0 || l2Appended > 0) {\n console.log(`[sync] Received push: ${l3Accepted} L3 accepted, ${l2Appended} L2 appended`);\n }\n\n return { l3_accepted: l3Accepted, l2_appended: l2Appended };\n }\n\n getSyncIntervalMs(): number {\n return this.syncIntervalMs;\n }\n}\n\n// --- PrimaryMemorySync ---\n\nexport interface PrimarySyncOptions {\n primary: PrimaryClient;\n memory: MemoryClient;\n syncIntervalMs?: number;\n}\n\n/**\n * Primary-side memory sync. Handles incoming L3 pulls from workers and\n * can push Global context updates out to all workers.\n */\nexport class PrimaryMemorySync {\n private primary: PrimaryClient;\n private memory: MemoryClient;\n private syncIntervalMs: number;\n private pushTimer: ReturnType<typeof setInterval> | null = null;\n private l3Store: Map<string, SyncL3Entry> = new Map(); // id -> latest entry\n private lastPushAt: string = new Date(0).toISOString();\n\n constructor(opts: PrimarySyncOptions) {\n this.primary = opts.primary;\n this.memory = opts.memory;\n this.syncIntervalMs = opts.syncIntervalMs ?? DEFAULT_SYNC_INTERVAL_MS;\n\n // Register handler for incoming L3 pulls from workers\n this.primary.onSyncPull((req) => this.handlePull(req));\n }\n\n /**\n * Handle an incoming pull — Worker sending L3 knowledge to Primary.\n * Uses last-write-wins for conflict resolution.\n */\n async handlePull(req: SyncPullRequest): Promise<SyncPullResponse> {\n let accepted = 0;\n let rejected = 0;\n\n for (const entry of req.entries) {\n const existing = this.l3Store.get(entry.id);\n\n if (existing && existing.updated_at >= entry.updated_at) {\n // Primary has same or newer version — reject\n rejected++;\n continue;\n }\n\n // Accept the entry — store locally\n this.l3Store.set(entry.id, entry);\n\n // Persist to Primary's memory daemon\n try {\n await this.memory.storeEpisode({\n context_name: entry.context_name,\n role: \"system\",\n content: entry.content,\n });\n } catch {\n // Non-fatal — the in-memory store is the source of truth for sync\n }\n\n accepted++;\n }\n\n if (accepted > 0) {\n console.log(\n `[sync] Accepted ${accepted} L3 entries from worker \"${req.worker_id}\" context \"${req.context_name}\"`,\n );\n }\n\n return { accepted, rejected };\n }\n\n /**\n * Push Global context updates to all workers.\n * Fetches current Global L3 knowledge and recent L2 episodes,\n * then pushes to every registered worker.\n */\n async pushGlobalToAll(): Promise<Map<string, SyncPushResponse | null>> {\n // Gather Global context L3 knowledge\n let l3Entries: SyncL3Entry[] = [];\n try {\n const knowledge = await this.memory.getL3Knowledge(\"global\");\n l3Entries = knowledge.map(l3ToSync);\n } catch {\n // Memory daemon may not have global context yet\n }\n\n // Gather recent Global L2 episodes (append-only)\n let l2Episodes: SyncL2Episode[] = [];\n try {\n const episodes = await this.memory.getContext(\"global\");\n // Only send episodes newer than last push\n l2Episodes = episodes\n .filter((ep) => ep.timestamp > this.lastPushAt)\n .map(episodeToSync);\n } catch {\n // Memory daemon may not have global context yet\n }\n\n if (l3Entries.length === 0 && l2Episodes.length === 0) {\n return new Map();\n }\n\n const payload: SyncPushRequest = {\n entries: l3Entries,\n episodes: l2Episodes,\n };\n\n this.lastPushAt = new Date().toISOString();\n\n const results = await this.primary.pushSyncToAll(payload);\n\n let totalL3 = 0;\n let totalL2 = 0;\n for (const resp of results.values()) {\n if (resp) {\n totalL3 += resp.l3_accepted;\n totalL2 += resp.l2_appended;\n }\n }\n\n if (totalL3 > 0 || totalL2 > 0) {\n console.log(`[sync] Pushed Global updates: ${totalL3} L3, ${totalL2} L2 across ${results.size} workers`);\n }\n\n return results;\n }\n\n /** Start periodic Global push to all workers. */\n startPushLoop(): void {\n this.stopPushLoop();\n this.pushTimer = setInterval(() => {\n this.pushGlobalToAll().catch((err) => {\n console.warn(\"[sync] Push cycle failed:\", (err as Error).message);\n });\n }, this.syncIntervalMs);\n }\n\n /** Stop periodic push. */\n stopPushLoop(): void {\n if (this.pushTimer) {\n clearInterval(this.pushTimer);\n this.pushTimer = null;\n }\n }\n\n /** Get all synced L3 entries the Primary has received. */\n getSyncedEntries(): SyncL3Entry[] {\n return Array.from(this.l3Store.values());\n }\n\n getSyncIntervalMs(): number {\n return this.syncIntervalMs;\n }\n}\n"],"mappings":"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","names":[]}
|
|
@@ -0,0 +1,132 @@
|
|
|
1
|
+
// packages/cli/src/commands/service.ts
|
|
2
|
+
import { resolve } from "path";
|
|
3
|
+
import { readFileSync, writeFileSync, existsSync, mkdirSync } from "fs";
|
|
4
|
+
import { execSync } from "child_process";
|
|
5
|
+
import { homedir } from "os";
|
|
6
|
+
import { fileURLToPath } from "url";
|
|
7
|
+
import { dirname } from "path";
|
|
8
|
+
var LAUNCH_AGENTS_DIR = resolve(homedir(), "Library/LaunchAgents");
|
|
9
|
+
var AGENT_LABEL = "com.hivemind.agent";
|
|
10
|
+
var MEMORY_LABEL = "com.hivemind.memory";
|
|
11
|
+
async function runServiceCommand(args) {
|
|
12
|
+
const subcommand = args[0];
|
|
13
|
+
switch (subcommand) {
|
|
14
|
+
case "install":
|
|
15
|
+
await installServices(args.slice(1));
|
|
16
|
+
break;
|
|
17
|
+
case "uninstall":
|
|
18
|
+
await uninstallServices();
|
|
19
|
+
break;
|
|
20
|
+
case "status":
|
|
21
|
+
showStatus();
|
|
22
|
+
break;
|
|
23
|
+
case "logs":
|
|
24
|
+
showLogs(args[1]);
|
|
25
|
+
break;
|
|
26
|
+
default:
|
|
27
|
+
printHelp();
|
|
28
|
+
if (subcommand) {
|
|
29
|
+
console.error(`Unknown subcommand: ${subcommand}`);
|
|
30
|
+
process.exit(1);
|
|
31
|
+
}
|
|
32
|
+
break;
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
async function installServices(_args) {
|
|
36
|
+
const hivemindHome = process.env.HIVEMIND_HOME || process.cwd();
|
|
37
|
+
console.log(`
|
|
38
|
+
\u2192 Installing launchd services for ${hivemindHome}
|
|
39
|
+
`);
|
|
40
|
+
mkdirSync(LAUNCH_AGENTS_DIR, { recursive: true });
|
|
41
|
+
const scriptDirs = [
|
|
42
|
+
resolve(hivemindHome, "scripts"),
|
|
43
|
+
resolve(dirname(dirname(dirname(fileURLToPath(import.meta.url)))), "scripts")
|
|
44
|
+
];
|
|
45
|
+
for (const label of [MEMORY_LABEL, AGENT_LABEL]) {
|
|
46
|
+
const plistFile = `${label}.plist`;
|
|
47
|
+
let templatePath = "";
|
|
48
|
+
for (const dir of scriptDirs) {
|
|
49
|
+
const p = resolve(dir, plistFile);
|
|
50
|
+
if (existsSync(p)) {
|
|
51
|
+
templatePath = p;
|
|
52
|
+
break;
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
if (!templatePath) {
|
|
56
|
+
console.error(` \u2717 Template not found: ${plistFile}`);
|
|
57
|
+
continue;
|
|
58
|
+
}
|
|
59
|
+
let content = readFileSync(templatePath, "utf-8");
|
|
60
|
+
content = content.replace(/__HIVEMIND_HOME__/g, hivemindHome);
|
|
61
|
+
const destPath = resolve(LAUNCH_AGENTS_DIR, plistFile);
|
|
62
|
+
writeFileSync(destPath, content);
|
|
63
|
+
try {
|
|
64
|
+
execSync(`launchctl unload ${destPath} 2>/dev/null`);
|
|
65
|
+
} catch {
|
|
66
|
+
}
|
|
67
|
+
execSync(`launchctl load ${destPath}`);
|
|
68
|
+
console.log(` \u2713 ${label} installed and started`);
|
|
69
|
+
}
|
|
70
|
+
console.log("\n Services will auto-start on boot.\n");
|
|
71
|
+
}
|
|
72
|
+
async function uninstallServices() {
|
|
73
|
+
console.log("\n\u2192 Uninstalling launchd services\n");
|
|
74
|
+
for (const label of [AGENT_LABEL, MEMORY_LABEL]) {
|
|
75
|
+
const plistPath = resolve(LAUNCH_AGENTS_DIR, `${label}.plist`);
|
|
76
|
+
if (existsSync(plistPath)) {
|
|
77
|
+
try {
|
|
78
|
+
execSync(`launchctl unload ${plistPath} 2>/dev/null`);
|
|
79
|
+
} catch {
|
|
80
|
+
}
|
|
81
|
+
const { unlinkSync } = await import("fs");
|
|
82
|
+
unlinkSync(plistPath);
|
|
83
|
+
console.log(` \u2713 ${label} uninstalled`);
|
|
84
|
+
} else {
|
|
85
|
+
console.log(` - ${label} not installed`);
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
console.log("");
|
|
89
|
+
}
|
|
90
|
+
function showStatus() {
|
|
91
|
+
console.log("\n\u2192 Service status\n");
|
|
92
|
+
for (const label of [MEMORY_LABEL, AGENT_LABEL]) {
|
|
93
|
+
try {
|
|
94
|
+
const out = execSync(`launchctl list ${label} 2>/dev/null`, { encoding: "utf-8" });
|
|
95
|
+
const pidMatch = out.match(/"PID"\s*=\s*(\d+)/);
|
|
96
|
+
const pid = pidMatch ? pidMatch[1] : "unknown";
|
|
97
|
+
console.log(` \u2713 ${label}: running (PID ${pid})`);
|
|
98
|
+
} catch {
|
|
99
|
+
console.log(` - ${label}: not running`);
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
console.log("");
|
|
103
|
+
}
|
|
104
|
+
function showLogs(which) {
|
|
105
|
+
const logFile = which === "memory" ? "/tmp/hivemind-memory.log" : "/tmp/hivemind-agent.log";
|
|
106
|
+
try {
|
|
107
|
+
execSync(`tail -30 ${logFile}`, { stdio: "inherit" });
|
|
108
|
+
} catch {
|
|
109
|
+
console.error(`No log file at ${logFile}`);
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
function printHelp() {
|
|
113
|
+
console.log(`hivemind service \u2014 Manage launchd services
|
|
114
|
+
|
|
115
|
+
Usage: hivemind service <subcommand>
|
|
116
|
+
|
|
117
|
+
Subcommands:
|
|
118
|
+
install Install and start launchd services (survives reboots)
|
|
119
|
+
uninstall Stop and remove launchd services
|
|
120
|
+
status Show service status
|
|
121
|
+
logs [agent|memory] Show recent logs
|
|
122
|
+
|
|
123
|
+
Services:
|
|
124
|
+
com.hivemind.memory \u2014 Memory daemon (Rust, port 3434)
|
|
125
|
+
com.hivemind.agent \u2014 Agent runtime (Node.js, Sesame)
|
|
126
|
+
`);
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
export {
|
|
130
|
+
runServiceCommand
|
|
131
|
+
};
|
|
132
|
+
//# sourceMappingURL=chunk-MBS5A6BZ.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../packages/cli/src/commands/service.ts"],"sourcesContent":["import { resolve } from \"path\";\nimport { readFileSync, writeFileSync, existsSync, mkdirSync } from \"fs\";\nimport { execSync } from \"child_process\";\nimport { homedir } from \"os\";\n\nconst LAUNCH_AGENTS_DIR = resolve(homedir(), \"Library/LaunchAgents\");\nconst AGENT_LABEL = \"com.hivemind.agent\";\nconst MEMORY_LABEL = \"com.hivemind.memory\";\n\nexport async function runServiceCommand(args: string[]): Promise<void> {\n const subcommand = args[0];\n\n switch (subcommand) {\n case \"install\":\n await installServices(args.slice(1));\n break;\n case \"uninstall\":\n await uninstallServices();\n break;\n case \"status\":\n showStatus();\n break;\n case \"logs\":\n showLogs(args[1]);\n break;\n default:\n printHelp();\n if (subcommand) {\n console.error(`Unknown subcommand: ${subcommand}`);\n process.exit(1);\n }\n break;\n }\n}\n\nasync function installServices(_args: string[]): Promise<void> {\n const hivemindHome = process.env.HIVEMIND_HOME || process.cwd();\n\n console.log(`\\n→ Installing launchd services for ${hivemindHome}\\n`);\n\n mkdirSync(LAUNCH_AGENTS_DIR, { recursive: true });\n\n // Find plist templates\n const scriptDirs = [\n resolve(hivemindHome, \"scripts\"),\n resolve(dirname(dirname(dirname(fileURLToPath(import.meta.url)))), \"scripts\"),\n ];\n\n for (const label of [MEMORY_LABEL, AGENT_LABEL]) {\n const plistFile = `${label}.plist`;\n let templatePath = \"\";\n\n for (const dir of scriptDirs) {\n const p = resolve(dir, plistFile);\n if (existsSync(p)) { templatePath = p; break; }\n }\n\n if (!templatePath) {\n console.error(` ✗ Template not found: ${plistFile}`);\n continue;\n }\n\n let content = readFileSync(templatePath, \"utf-8\");\n content = content.replace(/__HIVEMIND_HOME__/g, hivemindHome);\n\n const destPath = resolve(LAUNCH_AGENTS_DIR, plistFile);\n writeFileSync(destPath, content);\n\n // Unload if already loaded, then load\n try { execSync(`launchctl unload ${destPath} 2>/dev/null`); } catch {}\n execSync(`launchctl load ${destPath}`);\n\n console.log(` ✓ ${label} installed and started`);\n }\n\n console.log(\"\\n Services will auto-start on boot.\\n\");\n}\n\nasync function uninstallServices(): Promise<void> {\n console.log(\"\\n→ Uninstalling launchd services\\n\");\n\n for (const label of [AGENT_LABEL, MEMORY_LABEL]) {\n const plistPath = resolve(LAUNCH_AGENTS_DIR, `${label}.plist`);\n if (existsSync(plistPath)) {\n try { execSync(`launchctl unload ${plistPath} 2>/dev/null`); } catch {}\n const { unlinkSync } = await import(\"fs\");\n unlinkSync(plistPath);\n console.log(` ✓ ${label} uninstalled`);\n } else {\n console.log(` - ${label} not installed`);\n }\n }\n console.log(\"\");\n}\n\nfunction showStatus(): void {\n console.log(\"\\n→ Service status\\n\");\n for (const label of [MEMORY_LABEL, AGENT_LABEL]) {\n try {\n const out = execSync(`launchctl list ${label} 2>/dev/null`, { encoding: \"utf-8\" });\n const pidMatch = out.match(/\"PID\"\\s*=\\s*(\\d+)/);\n const pid = pidMatch ? pidMatch[1] : \"unknown\";\n console.log(` ✓ ${label}: running (PID ${pid})`);\n } catch {\n console.log(` - ${label}: not running`);\n }\n }\n console.log(\"\");\n}\n\nfunction showLogs(which?: string): void {\n const logFile = which === \"memory\"\n ? \"/tmp/hivemind-memory.log\"\n : \"/tmp/hivemind-agent.log\";\n try {\n execSync(`tail -30 ${logFile}`, { stdio: \"inherit\" });\n } catch {\n console.error(`No log file at ${logFile}`);\n }\n}\n\nfunction printHelp(): void {\n console.log(`hivemind service — Manage launchd services\n\nUsage: hivemind service <subcommand>\n\nSubcommands:\n install Install and start launchd services (survives reboots)\n uninstall Stop and remove launchd services\n status Show service status\n logs [agent|memory] Show recent logs\n\nServices:\n com.hivemind.memory — Memory daemon (Rust, port 3434)\n com.hivemind.agent — Agent runtime (Node.js, Sesame)\n`);\n}\n\n// Helpers for ESM\nimport { fileURLToPath } from \"url\";\nimport { dirname } from \"path\";\n"],"mappings":"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","names":[]}
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
import {
|
|
2
|
+
startPipeline
|
|
3
|
+
} from "./chunk-WSOOI3AY.js";
|
|
4
|
+
|
|
5
|
+
// packages/cli/src/commands/start.ts
|
|
6
|
+
import { resolve } from "path";
|
|
7
|
+
import { existsSync } from "fs";
|
|
8
|
+
var DEFAULT_CONFIG = "config/default.toml";
|
|
9
|
+
async function runStartCommand(args) {
|
|
10
|
+
let configPath = DEFAULT_CONFIG;
|
|
11
|
+
for (let i = 0; i < args.length; i++) {
|
|
12
|
+
if ((args[i] === "--config" || args[i] === "-c") && args[i + 1]) {
|
|
13
|
+
configPath = args[++i];
|
|
14
|
+
} else if (args[i] === "--help" || args[i] === "-h") {
|
|
15
|
+
printHelp();
|
|
16
|
+
return;
|
|
17
|
+
} else {
|
|
18
|
+
console.error(`Unknown argument: ${args[i]}`);
|
|
19
|
+
printHelp();
|
|
20
|
+
process.exit(1);
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
const resolved = resolve(configPath);
|
|
24
|
+
if (!existsSync(resolved)) {
|
|
25
|
+
console.error(`Config not found: ${resolved}`);
|
|
26
|
+
process.exit(1);
|
|
27
|
+
}
|
|
28
|
+
await startPipeline(resolved);
|
|
29
|
+
}
|
|
30
|
+
function printHelp() {
|
|
31
|
+
console.log(`hivemind start \u2014 Start the Hivemind agent
|
|
32
|
+
|
|
33
|
+
Usage: hivemind start [options]
|
|
34
|
+
|
|
35
|
+
Options:
|
|
36
|
+
-c, --config <path> Config file (default: config/default.toml)
|
|
37
|
+
-h, --help Show this help
|
|
38
|
+
`);
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
export {
|
|
42
|
+
runStartCommand
|
|
43
|
+
};
|
|
44
|
+
//# sourceMappingURL=chunk-NLMP7MYR.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../packages/cli/src/commands/start.ts"],"sourcesContent":["import { resolve } from \"path\";\nimport { existsSync } from \"fs\";\nimport { startPipeline } from \"@hivemind/runtime\";\n\nconst DEFAULT_CONFIG = \"config/default.toml\";\n\nexport async function runStartCommand(args: string[]): Promise<void> {\n let configPath = DEFAULT_CONFIG;\n\n // Parse args\n for (let i = 0; i < args.length; i++) {\n if ((args[i] === \"--config\" || args[i] === \"-c\") && args[i + 1]) {\n configPath = args[++i];\n } else if (args[i] === \"--help\" || args[i] === \"-h\") {\n printHelp();\n return;\n } else {\n console.error(`Unknown argument: ${args[i]}`);\n printHelp();\n process.exit(1);\n }\n }\n\n const resolved = resolve(configPath);\n if (!existsSync(resolved)) {\n console.error(`Config not found: ${resolved}`);\n process.exit(1);\n }\n\n await startPipeline(resolved);\n}\n\nfunction printHelp(): void {\n console.log(`hivemind start — Start the Hivemind agent\n\nUsage: hivemind start [options]\n\nOptions:\n -c, --config <path> Config file (default: config/default.toml)\n -h, --help Show this help\n`);\n}\n"],"mappings":";;;;;AAAA,SAAS,eAAe;AACxB,SAAS,kBAAkB;AAG3B,IAAM,iBAAiB;AAEvB,eAAsB,gBAAgB,MAA+B;AACnE,MAAI,aAAa;AAGjB,WAAS,IAAI,GAAG,IAAI,KAAK,QAAQ,KAAK;AACpC,SAAK,KAAK,CAAC,MAAM,cAAc,KAAK,CAAC,MAAM,SAAS,KAAK,IAAI,CAAC,GAAG;AAC/D,mBAAa,KAAK,EAAE,CAAC;AAAA,IACvB,WAAW,KAAK,CAAC,MAAM,YAAY,KAAK,CAAC,MAAM,MAAM;AACnD,gBAAU;AACV;AAAA,IACF,OAAO;AACL,cAAQ,MAAM,qBAAqB,KAAK,CAAC,CAAC,EAAE;AAC5C,gBAAU;AACV,cAAQ,KAAK,CAAC;AAAA,IAChB;AAAA,EACF;AAEA,QAAM,WAAW,QAAQ,UAAU;AACnC,MAAI,CAAC,WAAW,QAAQ,GAAG;AACzB,YAAQ,MAAM,qBAAqB,QAAQ,EAAE;AAC7C,YAAQ,KAAK,CAAC;AAAA,EAChB;AAEA,QAAM,cAAc,QAAQ;AAC9B;AAEA,SAAS,YAAkB;AACzB,UAAQ,IAAI;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,CAOb;AACD;","names":[]}
|
|
@@ -0,0 +1,186 @@
|
|
|
1
|
+
import {
|
|
2
|
+
FleetManager
|
|
3
|
+
} from "./chunk-4LHKQFQT.js";
|
|
4
|
+
|
|
5
|
+
// packages/cli/src/commands/fleet.ts
|
|
6
|
+
function formatUptime(seconds) {
|
|
7
|
+
if (seconds == null) return "\u2014";
|
|
8
|
+
if (seconds < 60) return `${seconds}s`;
|
|
9
|
+
if (seconds < 3600) return `${Math.floor(seconds / 60)}m`;
|
|
10
|
+
const h = Math.floor(seconds / 3600);
|
|
11
|
+
const m = Math.floor(seconds % 3600 / 60);
|
|
12
|
+
return `${h}h ${m}m`;
|
|
13
|
+
}
|
|
14
|
+
function formatHealth(status) {
|
|
15
|
+
switch (status) {
|
|
16
|
+
case "healthy":
|
|
17
|
+
return "OK";
|
|
18
|
+
case "degraded":
|
|
19
|
+
return "DEGRADED";
|
|
20
|
+
case "unreachable":
|
|
21
|
+
return "DOWN";
|
|
22
|
+
default:
|
|
23
|
+
return status.toUpperCase();
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
function printDashboard(dashboard) {
|
|
27
|
+
console.log("\n=== Hivemind Fleet ===\n");
|
|
28
|
+
console.log(`Workers: ${dashboard.total_workers} (${dashboard.healthy} healthy, ${dashboard.degraded} degraded, ${dashboard.unreachable} unreachable)`);
|
|
29
|
+
console.log(`Contexts assigned: ${dashboard.total_contexts}`);
|
|
30
|
+
if (dashboard.unassigned_contexts.length > 0) {
|
|
31
|
+
console.log(`Unassigned contexts: ${dashboard.unassigned_contexts.join(", ")}`);
|
|
32
|
+
}
|
|
33
|
+
console.log("");
|
|
34
|
+
if (dashboard.workers.length === 0) {
|
|
35
|
+
console.log("No workers registered. Use 'fleet add-worker <url>' to add one.");
|
|
36
|
+
return;
|
|
37
|
+
}
|
|
38
|
+
const cols = { id: 12, url: 30, health: 8, activity: 10, contexts: 25, task: 20, uptime: 8 };
|
|
39
|
+
const header = [
|
|
40
|
+
"ID".padEnd(cols.id),
|
|
41
|
+
"URL".padEnd(cols.url),
|
|
42
|
+
"HEALTH".padEnd(cols.health),
|
|
43
|
+
"ACTIVITY".padEnd(cols.activity),
|
|
44
|
+
"CONTEXTS".padEnd(cols.contexts),
|
|
45
|
+
"TASK".padEnd(cols.task),
|
|
46
|
+
"UPTIME".padEnd(cols.uptime)
|
|
47
|
+
].join(" ");
|
|
48
|
+
console.log(header);
|
|
49
|
+
console.log("-".repeat(header.length));
|
|
50
|
+
for (const w of dashboard.workers) {
|
|
51
|
+
const row = [
|
|
52
|
+
w.id.padEnd(cols.id),
|
|
53
|
+
w.url.slice(0, cols.url).padEnd(cols.url),
|
|
54
|
+
formatHealth(w.health).padEnd(cols.health),
|
|
55
|
+
w.activity.padEnd(cols.activity),
|
|
56
|
+
(w.contexts.join(", ") || "\u2014").slice(0, cols.contexts).padEnd(cols.contexts),
|
|
57
|
+
(w.current_task ?? "\u2014").slice(0, cols.task).padEnd(cols.task),
|
|
58
|
+
formatUptime(w.uptime_seconds).padEnd(cols.uptime)
|
|
59
|
+
].join(" ");
|
|
60
|
+
console.log(row);
|
|
61
|
+
}
|
|
62
|
+
console.log(`
|
|
63
|
+
Generated: ${dashboard.generated_at}`);
|
|
64
|
+
}
|
|
65
|
+
async function runFleetCommand(args) {
|
|
66
|
+
const subcommand = args[0];
|
|
67
|
+
if (!subcommand) {
|
|
68
|
+
printUsage();
|
|
69
|
+
process.exit(1);
|
|
70
|
+
}
|
|
71
|
+
const fleet = new FleetManager();
|
|
72
|
+
switch (subcommand) {
|
|
73
|
+
case "status": {
|
|
74
|
+
const dashboard = await fleet.getDashboard();
|
|
75
|
+
printDashboard(dashboard);
|
|
76
|
+
break;
|
|
77
|
+
}
|
|
78
|
+
case "add-worker": {
|
|
79
|
+
const url = args[1];
|
|
80
|
+
if (!url) {
|
|
81
|
+
console.error("Usage: hivemind fleet add-worker <url>");
|
|
82
|
+
process.exit(1);
|
|
83
|
+
}
|
|
84
|
+
try {
|
|
85
|
+
const worker = await fleet.addWorker(url);
|
|
86
|
+
console.log(`Worker registered: ${worker.id}`);
|
|
87
|
+
console.log(` URL: ${worker.url}`);
|
|
88
|
+
console.log(` Max contexts: ${worker.capabilities.max_contexts}`);
|
|
89
|
+
console.log(` Ollama: ${worker.capabilities.has_ollama ? "yes" : "no"}`);
|
|
90
|
+
console.log(` Memory daemon: ${worker.capabilities.has_memory_daemon ? "yes" : "no"}`);
|
|
91
|
+
} catch (err) {
|
|
92
|
+
const msg = err instanceof Error ? err.message : String(err);
|
|
93
|
+
console.error(`Failed to add worker: ${msg}`);
|
|
94
|
+
process.exit(1);
|
|
95
|
+
}
|
|
96
|
+
break;
|
|
97
|
+
}
|
|
98
|
+
case "remove-worker": {
|
|
99
|
+
const workerId = args[1];
|
|
100
|
+
if (!workerId) {
|
|
101
|
+
console.error("Usage: hivemind fleet remove-worker <worker-id>");
|
|
102
|
+
process.exit(1);
|
|
103
|
+
}
|
|
104
|
+
const removed = await fleet.removeWorker(workerId);
|
|
105
|
+
if (removed) {
|
|
106
|
+
console.log(`Worker ${workerId} removed.`);
|
|
107
|
+
} else {
|
|
108
|
+
console.error(`Worker ${workerId} not found.`);
|
|
109
|
+
process.exit(1);
|
|
110
|
+
}
|
|
111
|
+
break;
|
|
112
|
+
}
|
|
113
|
+
case "assign": {
|
|
114
|
+
const workerId = args[1];
|
|
115
|
+
const context = args[2];
|
|
116
|
+
if (!workerId || !context) {
|
|
117
|
+
console.error("Usage: hivemind fleet assign <worker-id> <context>");
|
|
118
|
+
process.exit(1);
|
|
119
|
+
}
|
|
120
|
+
const result = await fleet.assignContext(workerId, context);
|
|
121
|
+
if (result.accepted) {
|
|
122
|
+
console.log(`Context '${context}' assigned to ${workerId}.`);
|
|
123
|
+
} else {
|
|
124
|
+
console.error(`Assignment rejected: ${result.reason}`);
|
|
125
|
+
process.exit(1);
|
|
126
|
+
}
|
|
127
|
+
break;
|
|
128
|
+
}
|
|
129
|
+
case "migrate": {
|
|
130
|
+
const context = args[1];
|
|
131
|
+
const toWorker = args[2];
|
|
132
|
+
if (!context || !toWorker) {
|
|
133
|
+
console.error("Usage: hivemind fleet migrate <context> <worker-id>");
|
|
134
|
+
process.exit(1);
|
|
135
|
+
}
|
|
136
|
+
const result = await fleet.migrateContext(context, toWorker);
|
|
137
|
+
if (result.success) {
|
|
138
|
+
console.log(`Context '${context}' migrated: ${result.from_worker} -> ${result.to_worker}`);
|
|
139
|
+
} else {
|
|
140
|
+
console.error(`Migration failed: ${result.reason}`);
|
|
141
|
+
process.exit(1);
|
|
142
|
+
}
|
|
143
|
+
break;
|
|
144
|
+
}
|
|
145
|
+
case "discover": {
|
|
146
|
+
const urls = args.slice(1);
|
|
147
|
+
if (urls.length === 0) {
|
|
148
|
+
console.error("Usage: hivemind fleet discover <url1> [url2] ...");
|
|
149
|
+
process.exit(1);
|
|
150
|
+
}
|
|
151
|
+
console.log(`Probing ${urls.length} URL(s)...`);
|
|
152
|
+
const found = await fleet.discoverWorkers(urls);
|
|
153
|
+
if (found.length === 0) {
|
|
154
|
+
console.log("No workers found.");
|
|
155
|
+
} else {
|
|
156
|
+
console.log(`Found ${found.length} worker(s):`);
|
|
157
|
+
for (const url of found) {
|
|
158
|
+
console.log(` ${url}`);
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
break;
|
|
162
|
+
}
|
|
163
|
+
default:
|
|
164
|
+
console.error(`Unknown fleet subcommand: ${subcommand}`);
|
|
165
|
+
printUsage();
|
|
166
|
+
process.exit(1);
|
|
167
|
+
}
|
|
168
|
+
}
|
|
169
|
+
function printUsage() {
|
|
170
|
+
console.log(`
|
|
171
|
+
Usage: hivemind fleet <command> [args]
|
|
172
|
+
|
|
173
|
+
Commands:
|
|
174
|
+
status Show fleet dashboard
|
|
175
|
+
add-worker <url> Register a worker by URL
|
|
176
|
+
remove-worker <worker-id> Remove a worker from the fleet
|
|
177
|
+
assign <worker-id> <context> Assign a context to a worker
|
|
178
|
+
migrate <context> <worker-id> Migrate a context to another worker
|
|
179
|
+
discover <url1> [url2] ... Probe URLs for workers
|
|
180
|
+
`.trim());
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
export {
|
|
184
|
+
runFleetCommand
|
|
185
|
+
};
|
|
186
|
+
//# sourceMappingURL=chunk-O4CT7SDB.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../packages/cli/src/commands/fleet.ts"],"sourcesContent":["/**\n * CLI commands for fleet management.\n *\n * Usage:\n * hivemind fleet status — Show fleet dashboard\n * hivemind fleet add-worker <url> — Register a worker by URL\n * hivemind fleet remove-worker <worker-id> — Remove a worker from the fleet\n * hivemind fleet assign <worker-id> <context> — Assign a context to a worker\n * hivemind fleet migrate <context> <worker-id> — Migrate a context to another worker\n * hivemind fleet discover <url1> [url2] ... — Probe URLs for workers\n */\n\nimport { FleetManager } from \"@hivemind/runtime\";\nimport type { FleetDashboard, WorkerSummary } from \"@hivemind/runtime\";\n\nfunction formatUptime(seconds: number | null): string {\n if (seconds == null) return \"—\";\n if (seconds < 60) return `${seconds}s`;\n if (seconds < 3600) return `${Math.floor(seconds / 60)}m`;\n const h = Math.floor(seconds / 3600);\n const m = Math.floor((seconds % 3600) / 60);\n return `${h}h ${m}m`;\n}\n\nfunction formatHealth(status: string): string {\n switch (status) {\n case \"healthy\": return \"OK\";\n case \"degraded\": return \"DEGRADED\";\n case \"unreachable\": return \"DOWN\";\n default: return status.toUpperCase();\n }\n}\n\nfunction printDashboard(dashboard: FleetDashboard): void {\n console.log(\"\\n=== Hivemind Fleet ===\\n\");\n console.log(`Workers: ${dashboard.total_workers} (${dashboard.healthy} healthy, ${dashboard.degraded} degraded, ${dashboard.unreachable} unreachable)`);\n console.log(`Contexts assigned: ${dashboard.total_contexts}`);\n\n if (dashboard.unassigned_contexts.length > 0) {\n console.log(`Unassigned contexts: ${dashboard.unassigned_contexts.join(\", \")}`);\n }\n\n console.log(\"\");\n\n if (dashboard.workers.length === 0) {\n console.log(\"No workers registered. Use 'fleet add-worker <url>' to add one.\");\n return;\n }\n\n // Table header\n const cols = { id: 12, url: 30, health: 8, activity: 10, contexts: 25, task: 20, uptime: 8 };\n const header = [\n \"ID\".padEnd(cols.id),\n \"URL\".padEnd(cols.url),\n \"HEALTH\".padEnd(cols.health),\n \"ACTIVITY\".padEnd(cols.activity),\n \"CONTEXTS\".padEnd(cols.contexts),\n \"TASK\".padEnd(cols.task),\n \"UPTIME\".padEnd(cols.uptime),\n ].join(\" \");\n console.log(header);\n console.log(\"-\".repeat(header.length));\n\n for (const w of dashboard.workers) {\n const row = [\n w.id.padEnd(cols.id),\n w.url.slice(0, cols.url).padEnd(cols.url),\n formatHealth(w.health).padEnd(cols.health),\n w.activity.padEnd(cols.activity),\n (w.contexts.join(\", \") || \"—\").slice(0, cols.contexts).padEnd(cols.contexts),\n (w.current_task ?? \"—\").slice(0, cols.task).padEnd(cols.task),\n formatUptime(w.uptime_seconds).padEnd(cols.uptime),\n ].join(\" \");\n console.log(row);\n }\n\n console.log(`\\nGenerated: ${dashboard.generated_at}`);\n}\n\nexport async function runFleetCommand(args: string[]): Promise<void> {\n const subcommand = args[0];\n\n if (!subcommand) {\n printUsage();\n process.exit(1);\n }\n\n const fleet = new FleetManager();\n\n switch (subcommand) {\n case \"status\": {\n const dashboard = await fleet.getDashboard();\n printDashboard(dashboard);\n break;\n }\n\n case \"add-worker\": {\n const url = args[1];\n if (!url) {\n console.error(\"Usage: hivemind fleet add-worker <url>\");\n process.exit(1);\n }\n try {\n const worker = await fleet.addWorker(url);\n console.log(`Worker registered: ${worker.id}`);\n console.log(` URL: ${worker.url}`);\n console.log(` Max contexts: ${worker.capabilities.max_contexts}`);\n console.log(` Ollama: ${worker.capabilities.has_ollama ? \"yes\" : \"no\"}`);\n console.log(` Memory daemon: ${worker.capabilities.has_memory_daemon ? \"yes\" : \"no\"}`);\n } catch (err) {\n const msg = err instanceof Error ? err.message : String(err);\n console.error(`Failed to add worker: ${msg}`);\n process.exit(1);\n }\n break;\n }\n\n case \"remove-worker\": {\n const workerId = args[1];\n if (!workerId) {\n console.error(\"Usage: hivemind fleet remove-worker <worker-id>\");\n process.exit(1);\n }\n const removed = await fleet.removeWorker(workerId);\n if (removed) {\n console.log(`Worker ${workerId} removed.`);\n } else {\n console.error(`Worker ${workerId} not found.`);\n process.exit(1);\n }\n break;\n }\n\n case \"assign\": {\n const workerId = args[1];\n const context = args[2];\n if (!workerId || !context) {\n console.error(\"Usage: hivemind fleet assign <worker-id> <context>\");\n process.exit(1);\n }\n const result = await fleet.assignContext(workerId, context);\n if (result.accepted) {\n console.log(`Context '${context}' assigned to ${workerId}.`);\n } else {\n console.error(`Assignment rejected: ${result.reason}`);\n process.exit(1);\n }\n break;\n }\n\n case \"migrate\": {\n const context = args[1];\n const toWorker = args[2];\n if (!context || !toWorker) {\n console.error(\"Usage: hivemind fleet migrate <context> <worker-id>\");\n process.exit(1);\n }\n const result = await fleet.migrateContext(context, toWorker);\n if (result.success) {\n console.log(`Context '${context}' migrated: ${result.from_worker} -> ${result.to_worker}`);\n } else {\n console.error(`Migration failed: ${result.reason}`);\n process.exit(1);\n }\n break;\n }\n\n case \"discover\": {\n const urls = args.slice(1);\n if (urls.length === 0) {\n console.error(\"Usage: hivemind fleet discover <url1> [url2] ...\");\n process.exit(1);\n }\n console.log(`Probing ${urls.length} URL(s)...`);\n const found = await fleet.discoverWorkers(urls);\n if (found.length === 0) {\n console.log(\"No workers found.\");\n } else {\n console.log(`Found ${found.length} worker(s):`);\n for (const url of found) {\n console.log(` ${url}`);\n }\n }\n break;\n }\n\n default:\n console.error(`Unknown fleet subcommand: ${subcommand}`);\n printUsage();\n process.exit(1);\n }\n}\n\nfunction printUsage(): void {\n console.log(`\nUsage: hivemind fleet <command> [args]\n\nCommands:\n status Show fleet dashboard\n add-worker <url> Register a worker by URL\n remove-worker <worker-id> Remove a worker from the fleet\n assign <worker-id> <context> Assign a context to a worker\n migrate <context> <worker-id> Migrate a context to another worker\n discover <url1> [url2] ... Probe URLs for workers\n`.trim());\n}\n"],"mappings":"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","names":[]}
|