skillo 0.2.5 → 0.2.6

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.
@@ -1096,15 +1096,35 @@ async function main() {
1096
1096
  cleanup();
1097
1097
  return;
1098
1098
  }
1099
+ try {
1100
+ const authResult = await client.authenticate();
1101
+ if (authResult.success && authResult.data?.user) {
1102
+ statusWriter.update({ user: authResult.data.user.name });
1103
+ log("INFO", `Authenticated as: ${authResult.data.user.name}`);
1104
+ }
1105
+ } catch (err) {
1106
+ log("WARN", `Could not fetch user info: ${err}`);
1107
+ }
1099
1108
  ensureDirectory(getActiveSessionsDir());
1100
1109
  await updateTrackedProjectsCache(client);
1110
+ try {
1111
+ const cacheData = JSON.parse(readFileSync4(getTrackedProjectsCacheFile(), "utf-8"));
1112
+ statusWriter.update({ trackedProjects: cacheData.projects || [] });
1113
+ } catch {
1114
+ }
1101
1115
  await cleanupStaleSessions(client);
1102
1116
  const watchInterval = (config.daemon?.conversationCheckInterval || 5) * 1e3;
1103
1117
  const watcher = new ClaudeWatcher(client, {
1104
1118
  intervalMs: watchInterval,
1105
1119
  callbacks: {
1106
- onSync: (count) => log("INFO", `Synced ${count} Claude prompt(s)`),
1107
- onError: (err) => log("ERROR", `Watcher error: ${err.message}`),
1120
+ onSync: (count) => {
1121
+ log("INFO", `Synced ${count} Claude prompt(s)`);
1122
+ statusWriter.update({ claudeWatcher: { promptsSynced: count, lastSync: (/* @__PURE__ */ new Date()).toISOString(), lastError: null } });
1123
+ },
1124
+ onError: (err) => {
1125
+ log("ERROR", `Watcher error: ${err.message}`);
1126
+ statusWriter.update({ claudeWatcher: { lastError: err.message } });
1127
+ },
1108
1128
  log: (level, msg) => log(level, msg)
1109
1129
  }
1110
1130
  });
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/daemon-runner.ts","../src/core/config.ts","../src/utils/paths.ts","../src/core/api-client.ts","../src/core/claude-watcher.ts","../src/core/skill-usage-detector.ts","../src/utils/status-writer.ts"],"sourcesContent":["/**\n * Daemon Runner - Entry point for the forked daemon child process.\n *\n * Runs:\n * 1. Claude history watcher (auto-syncs conversations for tracked projects)\n * 2. Tracked projects cache writer (shell hooks read this for fast detection)\n * 3. Stale session cleanup (ends sessions where terminal PID is dead)\n */\n\nimport { existsSync, writeFileSync, unlinkSync, readFileSync, readdirSync, appendFileSync } from \"fs\";\nimport { loadConfig } from \"./core/config.js\";\nimport { getApiClient } from \"./core/api-client.js\";\nimport { getPidFile, getLogFile, getTrackedProjectsCacheFile, getActiveSessionsDir, ensureDirectory } from \"./utils/paths.js\";\nimport { ClaudeWatcher, loadTrackedProjects } from \"./core/claude-watcher.js\";\nimport { SkillUsageDetector } from \"./core/skill-usage-detector.js\";\nimport { StatusWriter } from \"./utils/status-writer.js\";\n\n// ── Daemon Logger ──\n\nfunction log(level: string, msg: string) {\n const line = `[${new Date().toISOString()}] [${level}] ${msg}\\n`;\n try {\n appendFileSync(getLogFile(), line);\n } catch {\n // Can't log, ignore\n }\n}\n\n// ── Tracked Projects Cache Writer ──\n\nasync function updateTrackedProjectsCache(client: ReturnType<typeof getApiClient>) {\n try {\n const result = await client.listProjects(true);\n if (result.success && result.data?.projects) {\n const cacheData = {\n updatedAt: Date.now(),\n projects: result.data.projects\n .filter((p: { trackingEnabled: boolean }) => p.trackingEnabled)\n .map((p: { path: string; name: string }) => ({ path: p.path, name: p.name })),\n };\n writeFileSync(getTrackedProjectsCacheFile(), JSON.stringify(cacheData, null, 2));\n log(\"DEBUG\", `Updated tracked projects cache: ${cacheData.projects.length} project(s)`);\n }\n } catch (error) {\n log(\"ERROR\", `Failed to update tracked projects cache: ${error}`);\n }\n}\n\n// ── Stale Session Cleanup ──\n\nasync function cleanupStaleSessions(client: ReturnType<typeof getApiClient>) {\n const sessionsDir = getActiveSessionsDir();\n if (!existsSync(sessionsDir)) return;\n\n try {\n const files = readdirSync(sessionsDir);\n for (const file of files) {\n if (!file.endsWith(\".json\")) continue;\n\n const pid = parseInt(file.replace(\".json\", \"\"), 10);\n if (isNaN(pid)) continue;\n\n // Check if process is still running\n let isRunning = false;\n try {\n process.kill(pid, 0);\n isRunning = true;\n } catch {\n // Process doesn't exist\n }\n\n if (!isRunning) {\n try {\n const sessionData = JSON.parse(readFileSync(`${sessionsDir}/${file}`, \"utf-8\"));\n if (sessionData.sessionId) {\n await client.endSession(sessionData.sessionId);\n log(\"INFO\", `Ended stale session ${sessionData.sessionId.slice(0, 8)} (PID ${pid} dead)`);\n }\n unlinkSync(`${sessionsDir}/${file}`);\n } catch {\n // Clean up the file even if API call fails\n try { unlinkSync(`${sessionsDir}/${file}`); } catch { /* ignore */ }\n }\n }\n }\n } catch (error) {\n log(\"ERROR\", `Stale session cleanup error: ${error}`);\n }\n}\n\n// ── Main Daemon Loop ──\n\nasync function main() {\n const pidFile = getPidFile();\n writeFileSync(pidFile, String(process.pid));\n\n log(\"INFO\", `Daemon starting (PID: ${process.pid})`);\n\n // Status writer\n const statusWriter = new StatusWriter();\n\n // Signal handlers\n const cleanup = () => {\n log(\"INFO\", \"Daemon shutting down\");\n statusWriter.stop();\n if (existsSync(pidFile)) {\n try { unlinkSync(pidFile); } catch { /* ignore */ }\n }\n process.exit(0);\n };\n process.on(\"SIGINT\", cleanup);\n process.on(\"SIGTERM\", cleanup);\n\n // Load config\n let config;\n try {\n config = loadConfig();\n } catch {\n log(\"WARN\", \"Could not load config, using defaults\");\n config = { daemon: { conversationCheckInterval: 5, patternCheckInterval: 60 } };\n }\n\n // Check API key\n const client = getApiClient();\n if (!client.hasApiKey()) {\n log(\"ERROR\", \"Not logged in. Daemon cannot run without API key.\");\n cleanup();\n return;\n }\n\n // Ensure directories\n ensureDirectory(getActiveSessionsDir());\n\n // Initial cache update\n await updateTrackedProjectsCache(client);\n\n // Clean up stale sessions from previous crashes\n await cleanupStaleSessions(client);\n\n // Start Claude watcher\n const watchInterval = ((config as { daemon?: { conversationCheckInterval?: number } }).daemon?.conversationCheckInterval || 5) * 1000;\n const watcher = new ClaudeWatcher(client, {\n intervalMs: watchInterval,\n callbacks: {\n onSync: (count) => log(\"INFO\", `Synced ${count} Claude prompt(s)`),\n onError: (err) => log(\"ERROR\", `Watcher error: ${err.message}`),\n log: (level, msg) => log(level, msg),\n },\n });\n await watcher.start();\n\n // Start skill usage detector\n const skillDetector = new SkillUsageDetector(client, {\n intervalMs: 30000,\n callbacks: {\n onDetection: (count) => {\n log(\"INFO\", `Detected ${count} skill usage(s)`);\n statusWriter.update({ skillDetector: { usagesDetected: count, lastDetection: new Date().toISOString(), lastError: null } });\n },\n onError: (err) => {\n log(\"ERROR\", `Skill detection error: ${err.message}`);\n statusWriter.update({ skillDetector: { lastError: err.message } });\n },\n log: (level, msg) => log(level, msg),\n },\n });\n await skillDetector.start();\n\n // Start status writer\n statusWriter.start();\n\n // Refresh tracked projects cache every 60s\n setInterval(async () => {\n await updateTrackedProjectsCache(client);\n // Update status writer with project count\n try {\n const cacheData = JSON.parse(readFileSync(getTrackedProjectsCacheFile(), \"utf-8\"));\n statusWriter.update({\n trackedProjects: cacheData.projects || [],\n });\n } catch { /* ignore */ }\n }, 60000);\n\n // Clean up stale sessions every 5 minutes\n setInterval(() => cleanupStaleSessions(client), 300000);\n\n log(\"INFO\", \"Daemon started successfully — watching Claude conversations & skill usage\");\n\n // Keep alive\n await new Promise(() => {});\n}\n\nmain().catch((err) => {\n log(\"ERROR\", `Daemon crashed: ${err}`);\n process.exit(1);\n});\n","/**\n * Configuration management for Skillo.\n */\n\nimport { readFileSync, writeFileSync, existsSync } from \"fs\";\nimport { dirname } from \"path\";\nimport YAML from \"yaml\";\nimport { ensureDirectory, getConfigFile } from \"../utils/paths.js\";\n\nexport interface SkilloConfig {\n // Shell settings\n defaultShell: string | null;\n\n // Pattern detection\n patternDetection: {\n minCount: number;\n sessionTimeout: number;\n similarityThreshold: number;\n maxSequenceLength: number;\n };\n\n // Privacy settings\n privacy: {\n autoRedact: boolean;\n trackOutput: boolean;\n neverTrack: string[];\n redactionPatterns: string[];\n };\n\n // Notification settings\n notifications: {\n enabled: boolean;\n style: \"inline\" | \"desktop\" | \"both\";\n sound: boolean;\n };\n\n // Skill generation\n skillGeneration: {\n outputDir: string | null;\n includeScripts: boolean;\n includeExamples: boolean;\n autoGenerate: boolean;\n };\n\n // Claude Code integration\n claudeCode: {\n watchConversations: boolean;\n conversationDir: string | null;\n };\n\n // Daemon settings\n daemon: {\n logLevel: \"DEBUG\" | \"INFO\" | \"WARN\" | \"ERROR\";\n patternCheckInterval: number;\n conversationCheckInterval: number;\n };\n\n // Team settings\n team: {\n enabled: boolean;\n slug: string | null;\n autoSync: boolean;\n syncInterval: number;\n };\n\n // API settings\n api: {\n baseUrl: string | null;\n timeout: number;\n };\n\n // API key (stored separately, not in YAML for security)\n apiKey?: string;\n}\n\nexport function getDefaultConfig(): SkilloConfig {\n return {\n defaultShell: null,\n\n patternDetection: {\n minCount: 3,\n sessionTimeout: 30,\n similarityThreshold: 0.8,\n maxSequenceLength: 10,\n },\n\n privacy: {\n autoRedact: true,\n trackOutput: false,\n neverTrack: [\n \"*password*\",\n \"*secret*\",\n \"vault *\",\n \"1password *\",\n \"op *\",\n \"export *_KEY=*\",\n \"export *_SECRET=*\",\n \"export *_TOKEN=*\",\n ],\n redactionPatterns: [\n \"password[=:]\\\\s*\\\\S+\",\n \"token[=:]\\\\s*\\\\S+\",\n \"secret[=:]\\\\s*\\\\S+\",\n \"api[_-]?key[=:]\\\\s*\\\\S+\",\n \"auth[=:]\\\\s*\\\\S+\",\n \"bearer\\\\s+\\\\S+\",\n \"-----BEGIN.*PRIVATE KEY-----\",\n \"AKIA[0-9A-Z]{16}\",\n '[\"\\'][A-Za-z0-9+/]{40,}[\"\\']',\n ],\n },\n\n notifications: {\n enabled: true,\n style: \"inline\",\n sound: false,\n },\n\n skillGeneration: {\n outputDir: null,\n includeScripts: true,\n includeExamples: true,\n autoGenerate: false,\n },\n\n claudeCode: {\n watchConversations: true,\n conversationDir: null,\n },\n\n daemon: {\n logLevel: \"INFO\",\n patternCheckInterval: 60,\n conversationCheckInterval: 5,\n },\n\n team: {\n enabled: false,\n slug: null,\n autoSync: true,\n syncInterval: 300,\n },\n\n api: {\n baseUrl: null,\n timeout: 30000,\n },\n };\n}\n\nfunction deepMerge<T extends object>(base: T, override: Partial<T>): T {\n const result = { ...base };\n\n for (const key of Object.keys(override) as Array<keyof T>) {\n const value = override[key];\n if (\n value !== undefined &&\n typeof result[key] === \"object\" &&\n result[key] !== null &&\n typeof value === \"object\" &&\n value !== null &&\n !Array.isArray(value)\n ) {\n result[key] = deepMerge(\n result[key] as Record<string, unknown>,\n value as Record<string, unknown>\n ) as T[keyof T];\n } else if (value !== undefined) {\n result[key] = value as T[keyof T];\n }\n }\n\n return result;\n}\n\nexport function loadConfig(path?: string): SkilloConfig {\n const configPath = path || getConfigFile();\n\n if (!existsSync(configPath)) {\n return getDefaultConfig();\n }\n\n try {\n const content = readFileSync(configPath, \"utf-8\");\n const parsed = YAML.parse(content) || {};\n\n // Convert snake_case to camelCase for compatibility\n const converted = convertKeysToCamelCase(parsed);\n const defaultConfig = getDefaultConfig();\n\n // Deep merge with default config\n return deepMerge(defaultConfig, converted as Partial<SkilloConfig>);\n } catch {\n return getDefaultConfig();\n }\n}\n\nexport function saveConfig(config: SkilloConfig, path?: string): void {\n const configPath = path || getConfigFile();\n\n ensureDirectory(dirname(configPath));\n\n // Convert camelCase to snake_case for YAML file\n const converted = convertKeysToSnakeCase(config);\n\n const content = YAML.stringify(converted, {\n indent: 2,\n lineWidth: 0,\n });\n\n writeFileSync(configPath, content, \"utf-8\");\n}\n\nexport function getConfigValue(config: SkilloConfig, key: string): unknown {\n const keys = key.split(\".\");\n let current: unknown = config;\n\n for (const k of keys) {\n if (current && typeof current === \"object\" && k in current) {\n current = (current as Record<string, unknown>)[k];\n } else {\n return undefined;\n }\n }\n\n return current;\n}\n\nexport function setConfigValue(config: SkilloConfig, key: string, value: unknown): SkilloConfig {\n const keys = key.split(\".\");\n const result = JSON.parse(JSON.stringify(config)) as SkilloConfig;\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n let current: any = result;\n\n for (let i = 0; i < keys.length - 1; i++) {\n const k = keys[i];\n if (!(k in current)) {\n current[k] = {};\n }\n current = current[k] as Record<string, unknown>;\n }\n\n current[keys[keys.length - 1]] = value;\n return result;\n}\n\n// Helper functions for key conversion\nfunction toCamelCase(str: string): string {\n return str.replace(/_([a-z])/g, (_, letter) => letter.toUpperCase());\n}\n\nfunction toSnakeCase(str: string): string {\n return str.replace(/[A-Z]/g, (letter) => `_${letter.toLowerCase()}`);\n}\n\nfunction convertKeysToCamelCase(obj: unknown): unknown {\n if (Array.isArray(obj)) {\n return obj.map(convertKeysToCamelCase);\n }\n\n if (obj !== null && typeof obj === \"object\") {\n const result: Record<string, unknown> = {};\n for (const [key, value] of Object.entries(obj)) {\n result[toCamelCase(key)] = convertKeysToCamelCase(value);\n }\n return result;\n }\n\n return obj;\n}\n\nfunction convertKeysToSnakeCase(obj: unknown): unknown {\n if (Array.isArray(obj)) {\n return obj.map(convertKeysToSnakeCase);\n }\n\n if (obj !== null && typeof obj === \"object\") {\n const result: Record<string, unknown> = {};\n for (const [key, value] of Object.entries(obj)) {\n result[toSnakeCase(key)] = convertKeysToSnakeCase(value);\n }\n return result;\n }\n\n return obj;\n}\n","/**\n * Path utilities for Skillo.\n *\n * Handles platform-specific path resolution for data directories,\n * configuration files, and skill storage locations.\n */\n\nimport { homedir } from \"os\";\nimport { join } from \"path\";\nimport { mkdirSync, existsSync } from \"fs\";\n\nexport function getHomeDir(): string {\n return homedir();\n}\n\n/**\n * Get Skillo data directory.\n * This is where the database and other data files are stored.\n *\n * - Linux/macOS: ~/.skillo/\n * - Windows: %USERPROFILE%/.skillo/\n */\nexport function getDataDir(): string {\n const envPath = process.env.SKILLO_DATA_DIR;\n if (envPath) return envPath;\n\n return join(getHomeDir(), \".skillo\");\n}\n\n/**\n * Get Skillo configuration directory.\n * This is where config.yaml and other configuration files are stored.\n *\n * - Linux: ~/.config/skillo/\n * - macOS: ~/.config/skillo/\n * - Windows: %USERPROFILE%/.config/skillo/\n */\nexport function getConfigDir(): string {\n const envPath = process.env.SKILLO_CONFIG_DIR;\n if (envPath) return envPath;\n\n const xdgConfig = process.env.XDG_CONFIG_HOME;\n if (xdgConfig) return join(xdgConfig, \"skillo\");\n\n return join(getHomeDir(), \".config\", \"skillo\");\n}\n\n/**\n * Get personal skills directory.\n * This is where generated skills are stored and where Claude Code\n * reads personal skills from.\n *\n * - All platforms: ~/.claude/skills/\n */\nexport function getSkillsDir(): string {\n const envPath = process.env.SKILLO_SKILLS_DIR;\n if (envPath) return envPath;\n\n return join(getClaudeDir(), \"skills\");\n}\n\n/**\n * Get Claude Code directory.\n * This is where Claude Code stores its data, including conversations.\n *\n * - All platforms: ~/.claude/\n */\nexport function getClaudeDir(): string {\n return join(getHomeDir(), \".claude\");\n}\n\n/**\n * Get project-specific skills directory.\n * This is where project-specific skills are stored.\n *\n * - Path: <project>/.claude/skills/\n */\nexport function getProjectSkillsDir(projectPath?: string): string {\n const project = projectPath || process.cwd();\n return join(project, \".claude\", \"skills\");\n}\n\n/**\n * Get team skills directory.\n * This is where team skills are synced to.\n *\n * - Path: ~/.claude/skills/team/<team-slug>/\n */\nexport function getTeamSkillsDir(teamSlug?: string): string {\n const base = join(getSkillsDir(), \"team\");\n if (teamSlug) return join(base, teamSlug);\n return base;\n}\n\n/**\n * Ensure a directory exists, creating it if necessary.\n *\n * @returns True if directory was created, False if it already existed.\n */\nexport function ensureDirectory(path: string): boolean {\n if (existsSync(path)) {\n return false;\n }\n\n mkdirSync(path, { recursive: true });\n return true;\n}\n\n/**\n * Get path to daemon log file.\n */\nexport function getLogFile(): string {\n return join(getDataDir(), \"daemon.log\");\n}\n\n/**\n * Get path to daemon PID file.\n */\nexport function getPidFile(): string {\n return join(getDataDir(), \"daemon.pid\");\n}\n\n/**\n * Get path to SQLite database file.\n */\nexport function getDbPath(): string {\n return join(getDataDir(), \"skillo.db\");\n}\n\n/**\n * Get path to config file.\n */\nexport function getConfigFile(): string {\n return join(getConfigDir(), \"config.yaml\");\n}\n\n/**\n * Get path to tracked projects cache file.\n * Written by daemon, read by shell hooks for fast project detection.\n */\nexport function getTrackedProjectsCacheFile(): string {\n return join(getDataDir(), \"tracked-projects.json\");\n}\n\n/**\n * Get path to active sessions directory.\n * Each terminal gets a file named by its PID.\n */\nexport function getActiveSessionsDir(): string {\n return join(getDataDir(), \"active-sessions\");\n}\n\n/**\n * Get path to shell integration directory.\n */\nexport function getShellIntegrationDir(): string {\n return join(getDataDir(), \"shell-integration\");\n}\n","/**\n * Skillo Platform API Client\n *\n * Handles communication with the Skillo platform API.\n */\n\nimport { loadConfig, saveConfig } from \"./config.js\";\n\nconst DEFAULT_API_URL = \"https://www.skillo.one/api/cli\";\n\ninterface ApiResponse<T = unknown> {\n success: boolean;\n data?: T;\n error?: string;\n message?: string;\n}\n\ninterface AuthResponse {\n success: boolean;\n user: {\n id: string;\n email: string;\n name: string;\n };\n}\n\ninterface SyncResponse {\n success: boolean;\n skills?: Array<{\n id: string;\n name: string;\n slug: string;\n description: string;\n content: string;\n commands: string[];\n }>;\n patterns?: Array<{\n id: string;\n name: string;\n description: string;\n commands: string[];\n status: string;\n }>;\n}\n\ninterface GenerateResponse {\n success: boolean;\n skill: {\n id: string;\n name: string;\n slug: string;\n description: string;\n content: string;\n commands: string[];\n };\n}\n\ninterface DeviceAuthResponse {\n code: string;\n verification_url: string;\n expires_in: number;\n interval: number;\n}\n\ninterface TokenStatusResponse {\n status: \"pending\" | \"ready\" | \"expired\" | \"used\" | \"not_found\";\n}\n\ninterface TokenExchangeResponse {\n success: boolean;\n api_key: string;\n user: {\n id: string;\n };\n}\n\nexport class ApiClient {\n private baseUrl: string;\n private apiKey: string | null;\n\n constructor() {\n const config = loadConfig();\n this.baseUrl = config.api?.baseUrl || DEFAULT_API_URL;\n this.apiKey = this.loadApiKey();\n }\n\n /**\n * Load API key from config\n */\n private loadApiKey(): string | null {\n const config = loadConfig();\n return (config as { apiKey?: string }).apiKey || null;\n }\n\n /**\n * Save API key to config\n */\n saveApiKey(key: string): void {\n const config = loadConfig();\n (config as { apiKey?: string }).apiKey = key;\n saveConfig(config);\n this.apiKey = key;\n }\n\n /**\n * Clear API key from config\n */\n clearApiKey(): void {\n const config = loadConfig();\n delete (config as { apiKey?: string }).apiKey;\n saveConfig(config);\n this.apiKey = null;\n }\n\n /**\n * Check if API key is configured\n */\n hasApiKey(): boolean {\n return !!this.apiKey;\n }\n\n /**\n * Get the configured API key (masked)\n */\n getMaskedApiKey(): string | null {\n if (!this.apiKey) return null;\n return `${this.apiKey.substring(0, 12)}...${this.apiKey.substring(this.apiKey.length - 4)}`;\n }\n\n /**\n * Set the API base URL\n */\n setBaseUrl(url: string): void {\n const config = loadConfig();\n if (!config.api) {\n (config as { api?: { baseUrl: string } }).api = { baseUrl: url };\n } else {\n config.api.baseUrl = url;\n }\n saveConfig(config);\n this.baseUrl = url;\n }\n\n /**\n * Make an authenticated API request\n */\n private async request<T>(\n endpoint: string,\n options: RequestInit = {}\n ): Promise<ApiResponse<T>> {\n if (!this.apiKey) {\n return { success: false, error: \"No API key configured. Run 'skillo login' first.\" };\n }\n\n const url = `${this.baseUrl}${endpoint}`;\n const headers: HeadersInit = {\n \"Content-Type\": \"application/json\",\n Authorization: `Bearer ${this.apiKey}`,\n ...options.headers,\n };\n\n try {\n const response = await fetch(url, {\n ...options,\n headers,\n });\n\n const data = await response.json();\n\n if (process.env.SKILLO_DEBUG) {\n console.log('[DEBUG] Request:', url);\n console.log('[DEBUG] Status:', response.status);\n console.log('[DEBUG] Response:', JSON.stringify(data));\n }\n\n if (!response.ok) {\n return {\n success: false,\n error: data.error || `Request failed with status ${response.status}`,\n };\n }\n\n // If the response already has success/data structure, return as-is\n // Otherwise wrap the data\n if (data.success !== undefined && data.data !== undefined) {\n return { success: data.success, data: data.data as T, message: data.message };\n }\n\n // If response just has success field (no data wrapper), return as-is\n if (data.success !== undefined) {\n return data;\n }\n\n // Legacy: wrap raw data\n return { success: true, data: data as T };\n } catch (error) {\n if (process.env.SKILLO_DEBUG) {\n console.log('[DEBUG] Error:', error);\n }\n if (error instanceof Error) {\n if (error.message.includes(\"ECONNREFUSED\")) {\n return {\n success: false,\n error: \"Cannot connect to Skillo platform. Is the server running?\",\n };\n }\n return { success: false, error: error.message };\n }\n return { success: false, error: \"Unknown error occurred\" };\n }\n }\n\n /**\n * Authenticate with the platform\n */\n async authenticate(): Promise<ApiResponse<AuthResponse>> {\n return this.request<AuthResponse>(\"/auth\", { method: \"POST\" });\n }\n\n /**\n * Sync commands to platform\n */\n async syncCommands(\n commands: Array<{\n timestamp: string;\n command: string;\n normalized: string;\n cwd: string;\n exitCode?: number | null;\n durationMs?: number | null;\n sessionId?: string;\n variables?: Record<string, string> | null;\n }>\n ): Promise<ApiResponse> {\n return this.request(\"/sync\", {\n method: \"POST\",\n body: JSON.stringify({\n type: \"commands\",\n data: commands,\n }),\n });\n }\n\n /**\n * Sync a pattern to platform\n */\n async syncPattern(pattern: {\n sourceType: string;\n name?: string;\n description?: string;\n commands: string[];\n category?: string;\n frequency?: number;\n score?: number;\n firstSeen?: string;\n lastSeen?: string;\n context?: Record<string, unknown>;\n }): Promise<ApiResponse<{ pattern: { id: string; name: string; status: string } }>> {\n return this.request(\"/sync\", {\n method: \"POST\",\n body: JSON.stringify({\n type: \"pattern\",\n data: pattern,\n }),\n });\n }\n\n /**\n * Create a session on the platform\n */\n async createSession(session: {\n startedAt: string;\n shell: string;\n endedAt?: string;\n commandCount?: number;\n }): Promise<ApiResponse<{ session: { id: string } }>> {\n return this.request(\"/sync\", {\n method: \"POST\",\n body: JSON.stringify({\n type: \"session\",\n data: session,\n }),\n });\n }\n\n /**\n * End a session on the platform\n */\n async endSession(sessionId: string): Promise<ApiResponse<{ success: boolean }>> {\n return this.request(\"/sessions\", {\n method: \"PATCH\",\n body: JSON.stringify({ sessionId }),\n });\n }\n\n /**\n * Start a new session on the platform (for standalone terminals)\n */\n async startSession(shell: string, projectPath?: string): Promise<ApiResponse<{ sessionId: string }>> {\n return this.request(\"/sessions\", {\n method: \"POST\",\n body: JSON.stringify({ shell, projectPath }),\n });\n }\n\n /**\n * Download skills and patterns from platform\n */\n async downloadData(type: \"all\" | \"skills\" | \"patterns\" = \"all\"): Promise<ApiResponse<SyncResponse>> {\n return this.request<SyncResponse>(`/sync?type=${type}`, { method: \"GET\" });\n }\n\n /**\n * Generate a skill from a pattern\n */\n async generateSkill(options: {\n patternId?: string;\n commands?: string[];\n name?: string;\n description?: string;\n category?: string;\n context?: Record<string, unknown>;\n }): Promise<ApiResponse<GenerateResponse>> {\n return this.request<GenerateResponse>(\"/generate\", {\n method: \"POST\",\n body: JSON.stringify(options),\n });\n }\n\n /**\n * Start device authorization flow (no auth required)\n */\n async startDeviceAuth(deviceName?: string): Promise<ApiResponse<DeviceAuthResponse>> {\n const url = `${this.baseUrl}/auth/device`;\n try {\n const response = await fetch(url, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({ device_name: deviceName }),\n });\n\n const data = await response.json();\n\n if (!response.ok) {\n return {\n success: false,\n error: data.error || `Request failed with status ${response.status}`,\n };\n }\n\n return { success: true, data };\n } catch (error) {\n if (error instanceof Error) {\n if (error.message.includes(\"ECONNREFUSED\")) {\n return {\n success: false,\n error: \"Cannot connect to Skillo platform. Is the server running?\",\n };\n }\n return { success: false, error: error.message };\n }\n return { success: false, error: \"Unknown error occurred\" };\n }\n }\n\n /**\n * Check token status (polling)\n */\n async checkTokenStatus(code: string): Promise<ApiResponse<TokenStatusResponse>> {\n const url = `${this.baseUrl}/token?code=${encodeURIComponent(code)}`;\n try {\n const response = await fetch(url, { method: \"GET\" });\n const data = await response.json();\n\n if (!response.ok) {\n return {\n success: false,\n error: data.error || `Request failed with status ${response.status}`,\n };\n }\n\n return { success: true, data };\n } catch (error) {\n if (error instanceof Error) {\n return { success: false, error: error.message };\n }\n return { success: false, error: \"Unknown error occurred\" };\n }\n }\n\n /**\n * Exchange code for API key\n */\n async exchangeToken(code: string, deviceName?: string): Promise<ApiResponse<TokenExchangeResponse>> {\n const url = `${this.baseUrl}/token`;\n try {\n const response = await fetch(url, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({ code, device_name: deviceName }),\n });\n\n const data = await response.json();\n\n if (!response.ok) {\n return {\n success: false,\n error: data.error || `Request failed with status ${response.status}`,\n };\n }\n\n return { success: true, data };\n } catch (error) {\n if (error instanceof Error) {\n return { success: false, error: error.message };\n }\n return { success: false, error: \"Unknown error occurred\" };\n }\n }\n\n /**\n * Sync Claude Code prompts to platform\n */\n async syncClaudePrompts(\n prompts: Array<{\n display: string;\n timestamp: number;\n project: string;\n sessionId: string;\n pastedContents?: Record<string, unknown>;\n }>,\n options?: {\n terminalSessionId?: string;\n projectPath?: string;\n }\n ): Promise<ApiResponse<{\n sessionsCreated: number;\n promptsCreated: number;\n promptsSkipped: number;\n }>> {\n return this.request(\"/claude\", {\n method: \"POST\",\n body: JSON.stringify({\n prompts,\n terminalSessionId: options?.terminalSessionId,\n projectPath: options?.projectPath,\n }),\n });\n }\n\n /**\n * Get Claude Code sessions from platform\n */\n async getClaudeSessions(limit = 20): Promise<ApiResponse<{\n sessions: Array<{\n id: string;\n claudeSessionId: string;\n projectPath: string;\n projectName: string;\n startedAt: string;\n endedAt?: string;\n promptCount: number;\n prompts?: Array<{\n id: string;\n prompt: string;\n timestamp: string;\n category?: string;\n }>;\n }>;\n }>> {\n return this.request(`/claude?limit=${limit}`, { method: \"GET\" });\n }\n\n /**\n * Report skill usage detections to platform\n */\n async reportSkillUsage(\n usages: Array<{\n skillSlug: string;\n claudeSessionId: string;\n projectPath: string;\n timestamp: string;\n }>\n ): Promise<ApiResponse<{ logged: number }>> {\n return this.request(\"/skill-usage\", {\n method: \"POST\",\n body: JSON.stringify({ usages }),\n });\n }\n\n // ============================================================================\n // PROJECT TRACKING\n // ============================================================================\n\n /**\n * Connect a project for tracking\n */\n async connectProject(params: {\n path: string;\n name?: string;\n gitRemote?: string;\n gitRemoteNormalized?: string;\n language?: string;\n framework?: string;\n }): Promise<ApiResponse<{\n id: string;\n name: string;\n path: string;\n trackingEnabled: boolean;\n connectedAt: string;\n }>> {\n return this.request(\"/projects/connect\", {\n method: \"POST\",\n body: JSON.stringify(params),\n });\n }\n\n /**\n * Disconnect a project from tracking\n */\n async disconnectProject(path: string): Promise<ApiResponse<{\n id: string;\n name: string;\n trackingEnabled: boolean;\n }>> {\n return this.request(`/projects/connect?path=${encodeURIComponent(path)}`, {\n method: \"DELETE\",\n });\n }\n\n /**\n * Get tracking status for a project\n */\n async getProjectStatus(path: string): Promise<ApiResponse<{\n connected: boolean;\n tracked: boolean;\n connectedAt?: string;\n project?: {\n id: string;\n name: string;\n path: string;\n gitRemote?: string;\n gitRemoteNormalized?: string;\n trackingEnabled: boolean;\n };\n }>> {\n return this.request(`/projects/connect?path=${encodeURIComponent(path)}`, {\n method: \"GET\",\n });\n }\n\n /**\n * List all tracked projects\n */\n async listProjects(includeDisabled = false): Promise<ApiResponse<{\n projects: Array<{\n id: string;\n name: string;\n path: string;\n gitRemote?: string;\n gitRemoteNormalized?: string;\n trackingEnabled: boolean;\n connectedAt?: string;\n }>;\n totalTracked: number;\n totalProjects: number;\n }>> {\n return this.request(`/projects/tracked?includeDisabled=${includeDisabled}`, {\n method: \"GET\",\n });\n }\n\n /**\n * Check if a path is in a tracked project\n * Returns the project if tracked, null if not\n */\n async isProjectTracked(path: string): Promise<{\n tracked: boolean;\n project?: {\n id: string;\n name: string;\n path: string;\n };\n }> {\n const result = await this.getProjectStatus(path);\n if (result.success && result.data?.connected) {\n return {\n tracked: true,\n project: result.data.project ? {\n id: result.data.project.id,\n name: result.data.project.name,\n path: result.data.project.path,\n } : undefined,\n };\n }\n return { tracked: false };\n }\n}\n\n// Singleton instance\nlet clientInstance: ApiClient | null = null;\n\nexport function getApiClient(): ApiClient {\n if (!clientInstance) {\n clientInstance = new ApiClient();\n }\n return clientInstance;\n}\n","/**\n * Claude Watcher - Reusable module for watching Claude Code history.\n *\n * Polls ~/.claude/history.jsonl for new entries and syncs them to the platform.\n * Used by both the daemon (background) and the `claude watch` command (foreground).\n */\n\nimport * as fs from \"fs\";\nimport * as path from \"path\";\nimport * as os from \"os\";\nimport * as readline from \"readline\";\n\n// Types\nexport interface ClaudeHistoryEntry {\n display: string;\n timestamp: number;\n project: string;\n sessionId: string;\n pastedContents?: Record<string, unknown>;\n}\n\ninterface TrackedProjectInfo {\n tracked: boolean;\n name?: string;\n}\n\nexport interface ClaudeWatcherCallbacks {\n onSync?: (promptsCreated: number) => void;\n onError?: (error: Error) => void;\n log?: (level: string, msg: string) => void;\n}\n\n// ── Path helpers ──\n\nfunction getClaudeHistoryPath(): string {\n return path.join(os.homedir(), \".claude\", \"history.jsonl\");\n}\n\nfunction normalizePath(p: string): string {\n return p.toLowerCase().replace(/\\\\/g, \"/\").replace(/\\/+$/, \"\");\n}\n\nfunction matchesProjectPath(claudeProjectPath: string, filterPath: string): boolean {\n const normalizedClaude = normalizePath(claudeProjectPath);\n const normalizedFilter = normalizePath(filterPath);\n return normalizedClaude === normalizedFilter ||\n normalizedClaude.startsWith(normalizedFilter + \"/\");\n}\n\n// ── Project tracking cache ──\n\nconst projectCache = {\n projects: new Map<string, TrackedProjectInfo>(),\n lastFetched: 0,\n};\n\nconst CACHE_TTL_MS = 60000;\n\nexport async function loadTrackedProjects(\n client: { listProjects: (includeDisabled: boolean) => Promise<{ success: boolean; data?: { projects: Array<{ path: string; name: string; trackingEnabled: boolean }> } }> }\n): Promise<Map<string, TrackedProjectInfo>> {\n const now = Date.now();\n\n if (projectCache.lastFetched > 0 && now - projectCache.lastFetched < CACHE_TTL_MS) {\n return projectCache.projects;\n }\n\n try {\n const result = await client.listProjects(true);\n if (result.success && result.data?.projects) {\n projectCache.projects.clear();\n for (const project of result.data.projects) {\n const normalizedPath = normalizePath(project.path);\n projectCache.projects.set(normalizedPath, {\n tracked: project.trackingEnabled,\n name: project.name,\n });\n }\n projectCache.lastFetched = now;\n }\n } catch {\n // Use stale cache on failure\n }\n\n return projectCache.projects;\n}\n\nexport function getTrackedProjectCount(): number {\n return Array.from(projectCache.projects.values()).filter(p => p.tracked).length;\n}\n\nasync function isProjectTracked(projectPath: string): Promise<TrackedProjectInfo> {\n const normalizedPath = normalizePath(projectPath);\n\n if (projectCache.projects.has(normalizedPath)) {\n return projectCache.projects.get(normalizedPath)!;\n }\n\n for (const [trackedPath, info] of projectCache.projects.entries()) {\n if (normalizedPath.startsWith(trackedPath + \"/\")) {\n return info;\n }\n }\n\n return { tracked: false };\n}\n\n// ── Watcher ──\n\nexport class ClaudeWatcher {\n private lastSize = 0;\n private lastMtime = 0;\n private intervalId: ReturnType<typeof setInterval> | null = null;\n private historyPath: string;\n private intervalMs: number;\n private projectFilter?: string;\n private sessionFilter?: string;\n private callbacks: ClaudeWatcherCallbacks;\n private client: {\n syncClaudePrompts: (prompts: ClaudeHistoryEntry[], options?: { terminalSessionId?: string; projectPath?: string }) => Promise<{ success: boolean; data?: { promptsCreated: number; sessionsCreated: number; promptsSkipped: number }; error?: string }>;\n listProjects: (includeDisabled: boolean) => Promise<{ success: boolean; data?: { projects: Array<{ path: string; name: string; trackingEnabled: boolean }> } }>;\n };\n\n constructor(\n client: ClaudeWatcher[\"client\"],\n options: {\n intervalMs?: number;\n projectFilter?: string;\n sessionFilter?: string;\n callbacks?: ClaudeWatcherCallbacks;\n } = {}\n ) {\n this.client = client;\n this.historyPath = getClaudeHistoryPath();\n this.intervalMs = options.intervalMs || 5000;\n this.projectFilter = options.projectFilter;\n this.sessionFilter = options.sessionFilter;\n this.callbacks = options.callbacks || {};\n }\n\n private log(level: string, msg: string) {\n if (this.callbacks.log) {\n this.callbacks.log(level, msg);\n }\n }\n\n /** Initialize file position to current end (don't sync old entries). */\n initPosition() {\n try {\n if (fs.existsSync(this.historyPath)) {\n const stats = fs.statSync(this.historyPath);\n this.lastSize = stats.size;\n this.lastMtime = stats.mtimeMs;\n }\n } catch {\n // File doesn't exist yet\n }\n }\n\n /** Start watching. Runs an initial check then polls on interval. */\n async start() {\n this.initPosition();\n\n // Load tracked projects\n await loadTrackedProjects(this.client);\n const trackedCount = getTrackedProjectCount();\n this.log(\"INFO\", `Watching ${trackedCount} tracked project(s)`);\n\n // Initial check\n await this.checkAndSync();\n\n // Poll\n this.intervalId = setInterval(() => this.checkAndSync(), this.intervalMs);\n }\n\n /** Stop watching. */\n stop() {\n if (this.intervalId) {\n clearInterval(this.intervalId);\n this.intervalId = null;\n }\n }\n\n /** Check for new entries and sync them. */\n async checkAndSync() {\n try {\n if (!fs.existsSync(this.historyPath)) return;\n\n const stats = fs.statSync(this.historyPath);\n\n // Skip if no changes\n if (stats.size === this.lastSize && stats.mtimeMs === this.lastMtime) {\n return;\n }\n\n // Refresh tracked projects cache\n await loadTrackedProjects(this.client);\n\n // Read new content\n const newPrompts: ClaudeHistoryEntry[] = [];\n\n if (stats.size > this.lastSize) {\n const stream = fs.createReadStream(this.historyPath, {\n start: this.lastSize,\n encoding: \"utf-8\",\n });\n\n const rl = readline.createInterface({\n input: stream,\n crlfDelay: Infinity,\n });\n\n for await (const line of rl) {\n if (!line.trim()) continue;\n\n try {\n const entry = JSON.parse(line) as ClaudeHistoryEntry;\n if (!entry.display || !entry.timestamp || !entry.project || !entry.sessionId) {\n continue;\n }\n\n // Project path filter\n if (this.projectFilter && !matchesProjectPath(entry.project, this.projectFilter)) {\n continue;\n }\n\n // Privacy: only tracked projects\n const trackingStatus = await isProjectTracked(entry.project);\n if (!trackingStatus.tracked) {\n continue;\n }\n\n newPrompts.push(entry);\n } catch {\n // Skip invalid JSON\n }\n }\n }\n\n this.lastSize = stats.size;\n this.lastMtime = stats.mtimeMs;\n\n if (newPrompts.length > 0) {\n this.log(\"INFO\", `Detected ${newPrompts.length} new prompt(s), syncing...`);\n\n const response = await this.client.syncClaudePrompts(newPrompts, {\n terminalSessionId: this.sessionFilter,\n projectPath: this.projectFilter,\n });\n\n if (response.success && response.data) {\n this.log(\"INFO\", `Synced ${response.data.promptsCreated} prompt(s)`);\n this.callbacks.onSync?.(response.data.promptsCreated);\n } else {\n this.log(\"ERROR\", `Sync failed: ${response.error}`);\n }\n }\n } catch (error) {\n const err = error instanceof Error ? error : new Error(String(error));\n this.log(\"ERROR\", `Watch error: ${err.message}`);\n this.callbacks.onError?.(err);\n }\n }\n}\n","/**\n * Skill Usage Detector — detects when Claude Code uses skills.\n *\n * Scans Claude Code session JSONL files for:\n * - Read tool calls to ~/.claude/skills/{slug}/SKILL.md\n * - Bash tool calls matching skill command arrays\n *\n * Tracks byte offsets per file to avoid re-processing.\n * Reports detections to the Skillo platform via batch API.\n */\n\nimport * as fs from \"fs\";\nimport * as path from \"path\";\nimport * as readline from \"readline\";\nimport { getSkillsDir, getClaudeDir, getDataDir, ensureDirectory } from \"../utils/paths.js\";\n\nexport interface SkillUsageEvent {\n skillSlug: string;\n claudeSessionId: string;\n projectPath: string;\n timestamp: string;\n}\n\ninterface OffsetState {\n [filePath: string]: {\n byteOffset: number;\n lastModified: number;\n };\n}\n\nexport interface SkillUsageDetectorCallbacks {\n onDetection?: (count: number) => void;\n onError?: (error: Error) => void;\n log?: (level: string, msg: string) => void;\n}\n\nconst MAX_NEW_BYTES_PER_FILE = 2 * 1024 * 1024; // 2MB cap per cycle\nconst OFFSETS_FILE = \"skill-usage-offsets.json\";\n\nexport class SkillUsageDetector {\n private intervalId: ReturnType<typeof setInterval> | null = null;\n private intervalMs: number;\n private callbacks: SkillUsageDetectorCallbacks;\n private client: {\n reportSkillUsage: (usages: SkillUsageEvent[]) => Promise<{\n success: boolean;\n data?: { logged: number };\n error?: string;\n }>;\n };\n\n constructor(\n client: SkillUsageDetector[\"client\"],\n options: {\n intervalMs?: number;\n callbacks?: SkillUsageDetectorCallbacks;\n } = {}\n ) {\n this.client = client;\n this.intervalMs = options.intervalMs || 30000;\n this.callbacks = options.callbacks || {};\n }\n\n private log(level: string, msg: string) {\n this.callbacks.log?.(level, msg);\n }\n\n async start() {\n // Initialize offsets to current file sizes\n this.initOffsets();\n this.log(\"INFO\", \"Skill usage detector started\");\n\n // Poll\n this.intervalId = setInterval(() => this.detect(), this.intervalMs);\n }\n\n stop() {\n if (this.intervalId) {\n clearInterval(this.intervalId);\n this.intervalId = null;\n }\n }\n\n /** Build slug inventory from ~/.claude/skills/ */\n private getDeployedSkillSlugs(): Set<string> {\n const slugs = new Set<string>();\n const skillsDir = getSkillsDir();\n\n if (!fs.existsSync(skillsDir)) return slugs;\n\n try {\n const entries = fs.readdirSync(skillsDir, { withFileTypes: true });\n for (const entry of entries) {\n if (!entry.isDirectory() || entry.name.startsWith(\".\")) continue;\n const skillFile = path.join(skillsDir, entry.name, \"SKILL.md\");\n if (fs.existsSync(skillFile)) {\n slugs.add(entry.name);\n }\n }\n } catch {\n // ignore\n }\n\n return slugs;\n }\n\n /** Encode project path to Claude's directory name format */\n private encodeProjectPath(projectPath: string): string {\n // /Users/foo/bar → -Users-foo-bar\n return projectPath.replace(/\\//g, \"-\");\n }\n\n /** Find session JSONL files for tracked projects */\n private getSessionFiles(): string[] {\n const claudeDir = getClaudeDir();\n const projectsDir = path.join(claudeDir, \"projects\");\n\n if (!fs.existsSync(projectsDir)) return [];\n\n const files: string[] = [];\n try {\n const projectDirs = fs.readdirSync(projectsDir, { withFileTypes: true });\n for (const dir of projectDirs) {\n if (!dir.isDirectory()) continue;\n const dirPath = path.join(projectsDir, dir.name);\n const jsonlFiles = fs.readdirSync(dirPath).filter(\n (f) => f.endsWith(\".jsonl\") && !f.startsWith(\"agent-\")\n );\n for (const f of jsonlFiles) {\n files.push(path.join(dirPath, f));\n }\n }\n } catch {\n // ignore\n }\n\n return files;\n }\n\n /** Load persisted offsets */\n private loadOffsets(): OffsetState {\n const offsetsPath = path.join(getDataDir(), OFFSETS_FILE);\n try {\n if (fs.existsSync(offsetsPath)) {\n return JSON.parse(fs.readFileSync(offsetsPath, \"utf-8\"));\n }\n } catch {\n // ignore\n }\n return {};\n }\n\n /** Save offsets */\n private saveOffsets(state: OffsetState) {\n ensureDirectory(getDataDir());\n const offsetsPath = path.join(getDataDir(), OFFSETS_FILE);\n try {\n fs.writeFileSync(offsetsPath, JSON.stringify(state), \"utf-8\");\n } catch {\n // ignore\n }\n }\n\n /** Initialize offsets to current file sizes (skip existing data) */\n private initOffsets() {\n const files = this.getSessionFiles();\n const state = this.loadOffsets();\n\n for (const file of files) {\n if (!state[file]) {\n try {\n const stats = fs.statSync(file);\n state[file] = { byteOffset: stats.size, lastModified: stats.mtimeMs };\n } catch {\n // ignore\n }\n }\n }\n\n this.saveOffsets(state);\n }\n\n /** Main detection cycle */\n async detect() {\n try {\n const skillSlugs = this.getDeployedSkillSlugs();\n if (skillSlugs.size === 0) return; // No skills deployed\n\n const skillsDir = getSkillsDir();\n const files = this.getSessionFiles();\n const offsets = this.loadOffsets();\n const events: SkillUsageEvent[] = [];\n\n for (const file of files) {\n try {\n const stats = fs.statSync(file);\n const currentOffset = offsets[file]?.byteOffset ?? 0;\n const currentMtime = offsets[file]?.lastModified ?? 0;\n\n // Skip unchanged files\n if (stats.size <= currentOffset && stats.mtimeMs === currentMtime) continue;\n\n // Cap how much we read per cycle\n const startOffset = Math.max(currentOffset, stats.size - MAX_NEW_BYTES_PER_FILE);\n\n if (stats.size <= startOffset) {\n offsets[file] = { byteOffset: stats.size, lastModified: stats.mtimeMs };\n continue;\n }\n\n // Read new content\n const stream = fs.createReadStream(file, {\n start: startOffset,\n encoding: \"utf-8\",\n });\n\n const rl = readline.createInterface({ input: stream, crlfDelay: Infinity });\n\n for await (const line of rl) {\n if (!line.trim()) continue;\n\n try {\n const entry = JSON.parse(line);\n const msg = entry?.message;\n if (!msg || msg.role !== \"assistant\") continue;\n\n const content = msg.content;\n if (!Array.isArray(content)) continue;\n\n const sessionId = entry.sessionId || \"\";\n const timestamp = entry.timestamp || new Date().toISOString();\n const cwd = entry.cwd || \"\";\n\n for (const block of content) {\n if (block?.type !== \"tool_use\") continue;\n\n const toolName = block.name;\n const toolInput = block.input || {};\n\n // Signal 1: Read calls to skill files\n if (toolName === \"Read\") {\n const filePath = toolInput.file_path || \"\";\n // Match paths like ~/.claude/skills/{slug}/SKILL.md or full path\n if (filePath.includes(\"/skills/\") && filePath.endsWith(\"/SKILL.md\")) {\n // Extract slug from path\n const parts = filePath.split(\"/skills/\");\n if (parts.length >= 2) {\n const afterSkills = parts[parts.length - 1]; // \"my-skill/SKILL.md\"\n const slug = afterSkills.split(\"/\")[0];\n if (slug && skillSlugs.has(slug)) {\n events.push({\n skillSlug: slug,\n claudeSessionId: sessionId,\n projectPath: cwd,\n timestamp,\n });\n }\n }\n }\n }\n\n // Signal 2: Bash commands matching skill commands\n // (skipped for v1 — Read detection is sufficient)\n }\n } catch {\n // Skip invalid JSON lines\n }\n }\n\n offsets[file] = { byteOffset: stats.size, lastModified: stats.mtimeMs };\n } catch {\n // Skip inaccessible files\n }\n }\n\n // Clean up offsets for deleted files\n for (const key of Object.keys(offsets)) {\n if (!fs.existsSync(key)) {\n delete offsets[key];\n }\n }\n\n this.saveOffsets(offsets);\n\n // Deduplicate: same slug+session within batch\n const seen = new Set<string>();\n const uniqueEvents = events.filter((e) => {\n const key = `${e.skillSlug}:${e.claudeSessionId}:${e.timestamp}`;\n if (seen.has(key)) return false;\n seen.add(key);\n return true;\n });\n\n // Report to server\n if (uniqueEvents.length > 0) {\n this.log(\"INFO\", `Detected ${uniqueEvents.length} skill usage(s), reporting...`);\n\n const response = await this.client.reportSkillUsage(uniqueEvents);\n if (response.success) {\n this.log(\"INFO\", `Reported ${response.data?.logged ?? 0} skill usage(s)`);\n this.callbacks.onDetection?.(response.data?.logged ?? 0);\n } else {\n this.log(\"ERROR\", `Report failed: ${response.error}`);\n }\n }\n } catch (error) {\n const err = error instanceof Error ? error : new Error(String(error));\n this.log(\"ERROR\", `Detection error: ${err.message}`);\n this.callbacks.onError?.(err);\n }\n }\n}\n","/**\n * StatusWriter — writes daemon status JSON for tray icon and diagnostics.\n *\n * Writes ~/.skillo/daemon-status.json every 10s and on key events.\n * Tray icon and `skillo status` read this file.\n */\n\nimport { writeFileSync, readFileSync, existsSync } from \"fs\";\nimport { join } from \"path\";\nimport { getDataDir, ensureDirectory } from \"./paths.js\";\n\nexport interface DaemonStatus {\n running: boolean;\n pid: number | null;\n startedAt: string | null;\n updatedAt: string;\n user?: string;\n claudeWatcher?: {\n lastSync?: string | null;\n promptsSynced?: number;\n lastError?: string | null;\n };\n skillDetector?: {\n deployedSkills?: number;\n usagesDetected?: number;\n lastDetection?: string | null;\n lastError?: string | null;\n };\n trackedProjects?: Array<{ name: string; path: string }>;\n activeSessions?: number;\n}\n\nconst STATUS_FILE = \"daemon-status.json\";\nconst WRITE_INTERVAL_MS = 10000;\n\nexport class StatusWriter {\n private intervalId: ReturnType<typeof setInterval> | null = null;\n private status: DaemonStatus;\n private filePath: string;\n\n constructor() {\n this.filePath = join(getDataDir(), STATUS_FILE);\n this.status = {\n running: true,\n pid: process.pid,\n startedAt: new Date().toISOString(),\n updatedAt: new Date().toISOString(),\n claudeWatcher: { lastSync: null, promptsSynced: 0, lastError: null },\n skillDetector: { deployedSkills: 0, usagesDetected: 0, lastDetection: null, lastError: null },\n trackedProjects: [],\n activeSessions: 0,\n };\n }\n\n /** Start periodic writes */\n start() {\n ensureDirectory(getDataDir());\n this.write();\n this.intervalId = setInterval(() => this.write(), WRITE_INTERVAL_MS);\n }\n\n /** Stop writing and mark as not running */\n stop() {\n if (this.intervalId) {\n clearInterval(this.intervalId);\n this.intervalId = null;\n }\n this.status.running = false;\n this.status.pid = null;\n this.status.updatedAt = new Date().toISOString();\n this.write();\n }\n\n /** Merge partial status updates */\n update(partial: Partial<DaemonStatus>) {\n // Deep merge for nested objects\n if (partial.claudeWatcher) {\n this.status.claudeWatcher = { ...this.status.claudeWatcher, ...partial.claudeWatcher };\n delete partial.claudeWatcher;\n }\n if (partial.skillDetector) {\n this.status.skillDetector = { ...this.status.skillDetector, ...partial.skillDetector };\n delete partial.skillDetector;\n }\n Object.assign(this.status, partial);\n }\n\n /** Get the status file path */\n static getStatusFilePath(): string {\n return join(getDataDir(), STATUS_FILE);\n }\n\n /** Read current status from disk (static, for tray/status commands) */\n static read(): DaemonStatus | null {\n const filePath = StatusWriter.getStatusFilePath();\n try {\n if (existsSync(filePath)) {\n return JSON.parse(readFileSync(filePath, \"utf-8\"));\n }\n } catch {\n // ignore\n }\n return null;\n }\n\n private write() {\n this.status.updatedAt = new Date().toISOString();\n try {\n writeFileSync(this.filePath, JSON.stringify(this.status, null, 2), \"utf-8\");\n } catch {\n // Can't write status, ignore\n }\n }\n}\n"],"mappings":";AASA,SAAS,cAAAA,aAAY,iBAAAC,gBAAe,YAAY,gBAAAC,eAAc,eAAAC,cAAa,sBAAsB;;;ACLjG,SAAS,cAAc,eAAe,cAAAC,mBAAkB;AACxD,SAAS,eAAe;AACxB,OAAO,UAAU;;;ACCjB,SAAS,eAAe;AACxB,SAAS,YAAY;AACrB,SAAS,WAAW,kBAAkB;AAE/B,SAAS,aAAqB;AACnC,SAAO,QAAQ;AACjB;AASO,SAAS,aAAqB;AACnC,QAAM,UAAU,QAAQ,IAAI;AAC5B,MAAI,QAAS,QAAO;AAEpB,SAAO,KAAK,WAAW,GAAG,SAAS;AACrC;AAUO,SAAS,eAAuB;AACrC,QAAM,UAAU,QAAQ,IAAI;AAC5B,MAAI,QAAS,QAAO;AAEpB,QAAM,YAAY,QAAQ,IAAI;AAC9B,MAAI,UAAW,QAAO,KAAK,WAAW,QAAQ;AAE9C,SAAO,KAAK,WAAW,GAAG,WAAW,QAAQ;AAC/C;AASO,SAAS,eAAuB;AACrC,QAAM,UAAU,QAAQ,IAAI;AAC5B,MAAI,QAAS,QAAO;AAEpB,SAAO,KAAK,aAAa,GAAG,QAAQ;AACtC;AAQO,SAAS,eAAuB;AACrC,SAAO,KAAK,WAAW,GAAG,SAAS;AACrC;AA8BO,SAAS,gBAAgBC,OAAuB;AACrD,MAAI,WAAWA,KAAI,GAAG;AACpB,WAAO;AAAA,EACT;AAEA,YAAUA,OAAM,EAAE,WAAW,KAAK,CAAC;AACnC,SAAO;AACT;AAKO,SAAS,aAAqB;AACnC,SAAO,KAAK,WAAW,GAAG,YAAY;AACxC;AAKO,SAAS,aAAqB;AACnC,SAAO,KAAK,WAAW,GAAG,YAAY;AACxC;AAYO,SAAS,gBAAwB;AACtC,SAAO,KAAK,aAAa,GAAG,aAAa;AAC3C;AAMO,SAAS,8BAAsC;AACpD,SAAO,KAAK,WAAW,GAAG,uBAAuB;AACnD;AAMO,SAAS,uBAA+B;AAC7C,SAAO,KAAK,WAAW,GAAG,iBAAiB;AAC7C;;;AD3EO,SAAS,mBAAiC;AAC/C,SAAO;AAAA,IACL,cAAc;AAAA,IAEd,kBAAkB;AAAA,MAChB,UAAU;AAAA,MACV,gBAAgB;AAAA,MAChB,qBAAqB;AAAA,MACrB,mBAAmB;AAAA,IACrB;AAAA,IAEA,SAAS;AAAA,MACP,YAAY;AAAA,MACZ,aAAa;AAAA,MACb,YAAY;AAAA,QACV;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,MACF;AAAA,MACA,mBAAmB;AAAA,QACjB;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,MACF;AAAA,IACF;AAAA,IAEA,eAAe;AAAA,MACb,SAAS;AAAA,MACT,OAAO;AAAA,MACP,OAAO;AAAA,IACT;AAAA,IAEA,iBAAiB;AAAA,MACf,WAAW;AAAA,MACX,gBAAgB;AAAA,MAChB,iBAAiB;AAAA,MACjB,cAAc;AAAA,IAChB;AAAA,IAEA,YAAY;AAAA,MACV,oBAAoB;AAAA,MACpB,iBAAiB;AAAA,IACnB;AAAA,IAEA,QAAQ;AAAA,MACN,UAAU;AAAA,MACV,sBAAsB;AAAA,MACtB,2BAA2B;AAAA,IAC7B;AAAA,IAEA,MAAM;AAAA,MACJ,SAAS;AAAA,MACT,MAAM;AAAA,MACN,UAAU;AAAA,MACV,cAAc;AAAA,IAChB;AAAA,IAEA,KAAK;AAAA,MACH,SAAS;AAAA,MACT,SAAS;AAAA,IACX;AAAA,EACF;AACF;AAEA,SAAS,UAA4B,MAAS,UAAyB;AACrE,QAAM,SAAS,EAAE,GAAG,KAAK;AAEzB,aAAW,OAAO,OAAO,KAAK,QAAQ,GAAqB;AACzD,UAAM,QAAQ,SAAS,GAAG;AAC1B,QACE,UAAU,UACV,OAAO,OAAO,GAAG,MAAM,YACvB,OAAO,GAAG,MAAM,QAChB,OAAO,UAAU,YACjB,UAAU,QACV,CAAC,MAAM,QAAQ,KAAK,GACpB;AACA,aAAO,GAAG,IAAI;AAAA,QACZ,OAAO,GAAG;AAAA,QACV;AAAA,MACF;AAAA,IACF,WAAW,UAAU,QAAW;AAC9B,aAAO,GAAG,IAAI;AAAA,IAChB;AAAA,EACF;AAEA,SAAO;AACT;AAEO,SAAS,WAAWC,OAA6B;AACtD,QAAM,aAAaA,SAAQ,cAAc;AAEzC,MAAI,CAACC,YAAW,UAAU,GAAG;AAC3B,WAAO,iBAAiB;AAAA,EAC1B;AAEA,MAAI;AACF,UAAM,UAAU,aAAa,YAAY,OAAO;AAChD,UAAM,SAAS,KAAK,MAAM,OAAO,KAAK,CAAC;AAGvC,UAAM,YAAY,uBAAuB,MAAM;AAC/C,UAAM,gBAAgB,iBAAiB;AAGvC,WAAO,UAAU,eAAe,SAAkC;AAAA,EACpE,QAAQ;AACN,WAAO,iBAAiB;AAAA,EAC1B;AACF;AAEO,SAAS,WAAW,QAAsBD,OAAqB;AACpE,QAAM,aAAaA,SAAQ,cAAc;AAEzC,kBAAgB,QAAQ,UAAU,CAAC;AAGnC,QAAM,YAAY,uBAAuB,MAAM;AAE/C,QAAM,UAAU,KAAK,UAAU,WAAW;AAAA,IACxC,QAAQ;AAAA,IACR,WAAW;AAAA,EACb,CAAC;AAED,gBAAc,YAAY,SAAS,OAAO;AAC5C;AAoCA,SAAS,YAAY,KAAqB;AACxC,SAAO,IAAI,QAAQ,aAAa,CAAC,GAAG,WAAW,OAAO,YAAY,CAAC;AACrE;AAEA,SAAS,YAAY,KAAqB;AACxC,SAAO,IAAI,QAAQ,UAAU,CAAC,WAAW,IAAI,OAAO,YAAY,CAAC,EAAE;AACrE;AAEA,SAAS,uBAAuB,KAAuB;AACrD,MAAI,MAAM,QAAQ,GAAG,GAAG;AACtB,WAAO,IAAI,IAAI,sBAAsB;AAAA,EACvC;AAEA,MAAI,QAAQ,QAAQ,OAAO,QAAQ,UAAU;AAC3C,UAAM,SAAkC,CAAC;AACzC,eAAW,CAAC,KAAK,KAAK,KAAK,OAAO,QAAQ,GAAG,GAAG;AAC9C,aAAO,YAAY,GAAG,CAAC,IAAI,uBAAuB,KAAK;AAAA,IACzD;AACA,WAAO;AAAA,EACT;AAEA,SAAO;AACT;AAEA,SAAS,uBAAuB,KAAuB;AACrD,MAAI,MAAM,QAAQ,GAAG,GAAG;AACtB,WAAO,IAAI,IAAI,sBAAsB;AAAA,EACvC;AAEA,MAAI,QAAQ,QAAQ,OAAO,QAAQ,UAAU;AAC3C,UAAM,SAAkC,CAAC;AACzC,eAAW,CAAC,KAAK,KAAK,KAAK,OAAO,QAAQ,GAAG,GAAG;AAC9C,aAAO,YAAY,GAAG,CAAC,IAAI,uBAAuB,KAAK;AAAA,IACzD;AACA,WAAO;AAAA,EACT;AAEA,SAAO;AACT;;;AErRA,IAAM,kBAAkB;AAoEjB,IAAM,YAAN,MAAgB;AAAA,EACb;AAAA,EACA;AAAA,EAER,cAAc;AACZ,UAAM,SAAS,WAAW;AAC1B,SAAK,UAAU,OAAO,KAAK,WAAW;AACtC,SAAK,SAAS,KAAK,WAAW;AAAA,EAChC;AAAA;AAAA;AAAA;AAAA,EAKQ,aAA4B;AAClC,UAAM,SAAS,WAAW;AAC1B,WAAQ,OAA+B,UAAU;AAAA,EACnD;AAAA;AAAA;AAAA;AAAA,EAKA,WAAW,KAAmB;AAC5B,UAAM,SAAS,WAAW;AAC1B,IAAC,OAA+B,SAAS;AACzC,eAAW,MAAM;AACjB,SAAK,SAAS;AAAA,EAChB;AAAA;AAAA;AAAA;AAAA,EAKA,cAAoB;AAClB,UAAM,SAAS,WAAW;AAC1B,WAAQ,OAA+B;AACvC,eAAW,MAAM;AACjB,SAAK,SAAS;AAAA,EAChB;AAAA;AAAA;AAAA;AAAA,EAKA,YAAqB;AACnB,WAAO,CAAC,CAAC,KAAK;AAAA,EAChB;AAAA;AAAA;AAAA;AAAA,EAKA,kBAAiC;AAC/B,QAAI,CAAC,KAAK,OAAQ,QAAO;AACzB,WAAO,GAAG,KAAK,OAAO,UAAU,GAAG,EAAE,CAAC,MAAM,KAAK,OAAO,UAAU,KAAK,OAAO,SAAS,CAAC,CAAC;AAAA,EAC3F;AAAA;AAAA;AAAA;AAAA,EAKA,WAAW,KAAmB;AAC5B,UAAM,SAAS,WAAW;AAC1B,QAAI,CAAC,OAAO,KAAK;AACf,MAAC,OAAyC,MAAM,EAAE,SAAS,IAAI;AAAA,IACjE,OAAO;AACL,aAAO,IAAI,UAAU;AAAA,IACvB;AACA,eAAW,MAAM;AACjB,SAAK,UAAU;AAAA,EACjB;AAAA;AAAA;AAAA;AAAA,EAKA,MAAc,QACZ,UACA,UAAuB,CAAC,GACC;AACzB,QAAI,CAAC,KAAK,QAAQ;AAChB,aAAO,EAAE,SAAS,OAAO,OAAO,mDAAmD;AAAA,IACrF;AAEA,UAAM,MAAM,GAAG,KAAK,OAAO,GAAG,QAAQ;AACtC,UAAM,UAAuB;AAAA,MAC3B,gBAAgB;AAAA,MAChB,eAAe,UAAU,KAAK,MAAM;AAAA,MACpC,GAAG,QAAQ;AAAA,IACb;AAEA,QAAI;AACF,YAAM,WAAW,MAAM,MAAM,KAAK;AAAA,QAChC,GAAG;AAAA,QACH;AAAA,MACF,CAAC;AAED,YAAM,OAAO,MAAM,SAAS,KAAK;AAEjC,UAAI,QAAQ,IAAI,cAAc;AAC5B,gBAAQ,IAAI,oBAAoB,GAAG;AACnC,gBAAQ,IAAI,mBAAmB,SAAS,MAAM;AAC9C,gBAAQ,IAAI,qBAAqB,KAAK,UAAU,IAAI,CAAC;AAAA,MACvD;AAEA,UAAI,CAAC,SAAS,IAAI;AAChB,eAAO;AAAA,UACL,SAAS;AAAA,UACT,OAAO,KAAK,SAAS,8BAA8B,SAAS,MAAM;AAAA,QACpE;AAAA,MACF;AAIA,UAAI,KAAK,YAAY,UAAa,KAAK,SAAS,QAAW;AACzD,eAAO,EAAE,SAAS,KAAK,SAAS,MAAM,KAAK,MAAW,SAAS,KAAK,QAAQ;AAAA,MAC9E;AAGA,UAAI,KAAK,YAAY,QAAW;AAC9B,eAAO;AAAA,MACT;AAGA,aAAO,EAAE,SAAS,MAAM,KAAgB;AAAA,IAC1C,SAAS,OAAO;AACd,UAAI,QAAQ,IAAI,cAAc;AAC5B,gBAAQ,IAAI,kBAAkB,KAAK;AAAA,MACrC;AACA,UAAI,iBAAiB,OAAO;AAC1B,YAAI,MAAM,QAAQ,SAAS,cAAc,GAAG;AAC1C,iBAAO;AAAA,YACL,SAAS;AAAA,YACT,OAAO;AAAA,UACT;AAAA,QACF;AACA,eAAO,EAAE,SAAS,OAAO,OAAO,MAAM,QAAQ;AAAA,MAChD;AACA,aAAO,EAAE,SAAS,OAAO,OAAO,yBAAyB;AAAA,IAC3D;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,eAAmD;AACvD,WAAO,KAAK,QAAsB,SAAS,EAAE,QAAQ,OAAO,CAAC;AAAA,EAC/D;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,aACJ,UAUsB;AACtB,WAAO,KAAK,QAAQ,SAAS;AAAA,MAC3B,QAAQ;AAAA,MACR,MAAM,KAAK,UAAU;AAAA,QACnB,MAAM;AAAA,QACN,MAAM;AAAA,MACR,CAAC;AAAA,IACH,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,YAAY,SAWkE;AAClF,WAAO,KAAK,QAAQ,SAAS;AAAA,MAC3B,QAAQ;AAAA,MACR,MAAM,KAAK,UAAU;AAAA,QACnB,MAAM;AAAA,QACN,MAAM;AAAA,MACR,CAAC;AAAA,IACH,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,cAAc,SAKkC;AACpD,WAAO,KAAK,QAAQ,SAAS;AAAA,MAC3B,QAAQ;AAAA,MACR,MAAM,KAAK,UAAU;AAAA,QACnB,MAAM;AAAA,QACN,MAAM;AAAA,MACR,CAAC;AAAA,IACH,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,WAAW,WAA+D;AAC9E,WAAO,KAAK,QAAQ,aAAa;AAAA,MAC/B,QAAQ;AAAA,MACR,MAAM,KAAK,UAAU,EAAE,UAAU,CAAC;AAAA,IACpC,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,aAAa,OAAe,aAAmE;AACnG,WAAO,KAAK,QAAQ,aAAa;AAAA,MAC/B,QAAQ;AAAA,MACR,MAAM,KAAK,UAAU,EAAE,OAAO,YAAY,CAAC;AAAA,IAC7C,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,aAAa,OAAsC,OAA2C;AAClG,WAAO,KAAK,QAAsB,cAAc,IAAI,IAAI,EAAE,QAAQ,MAAM,CAAC;AAAA,EAC3E;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,cAAc,SAOuB;AACzC,WAAO,KAAK,QAA0B,aAAa;AAAA,MACjD,QAAQ;AAAA,MACR,MAAM,KAAK,UAAU,OAAO;AAAA,IAC9B,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,gBAAgB,YAA+D;AACnF,UAAM,MAAM,GAAG,KAAK,OAAO;AAC3B,QAAI;AACF,YAAM,WAAW,MAAM,MAAM,KAAK;AAAA,QAChC,QAAQ;AAAA,QACR,SAAS,EAAE,gBAAgB,mBAAmB;AAAA,QAC9C,MAAM,KAAK,UAAU,EAAE,aAAa,WAAW,CAAC;AAAA,MAClD,CAAC;AAED,YAAM,OAAO,MAAM,SAAS,KAAK;AAEjC,UAAI,CAAC,SAAS,IAAI;AAChB,eAAO;AAAA,UACL,SAAS;AAAA,UACT,OAAO,KAAK,SAAS,8BAA8B,SAAS,MAAM;AAAA,QACpE;AAAA,MACF;AAEA,aAAO,EAAE,SAAS,MAAM,KAAK;AAAA,IAC/B,SAAS,OAAO;AACd,UAAI,iBAAiB,OAAO;AAC1B,YAAI,MAAM,QAAQ,SAAS,cAAc,GAAG;AAC1C,iBAAO;AAAA,YACL,SAAS;AAAA,YACT,OAAO;AAAA,UACT;AAAA,QACF;AACA,eAAO,EAAE,SAAS,OAAO,OAAO,MAAM,QAAQ;AAAA,MAChD;AACA,aAAO,EAAE,SAAS,OAAO,OAAO,yBAAyB;AAAA,IAC3D;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,iBAAiB,MAAyD;AAC9E,UAAM,MAAM,GAAG,KAAK,OAAO,eAAe,mBAAmB,IAAI,CAAC;AAClE,QAAI;AACF,YAAM,WAAW,MAAM,MAAM,KAAK,EAAE,QAAQ,MAAM,CAAC;AACnD,YAAM,OAAO,MAAM,SAAS,KAAK;AAEjC,UAAI,CAAC,SAAS,IAAI;AAChB,eAAO;AAAA,UACL,SAAS;AAAA,UACT,OAAO,KAAK,SAAS,8BAA8B,SAAS,MAAM;AAAA,QACpE;AAAA,MACF;AAEA,aAAO,EAAE,SAAS,MAAM,KAAK;AAAA,IAC/B,SAAS,OAAO;AACd,UAAI,iBAAiB,OAAO;AAC1B,eAAO,EAAE,SAAS,OAAO,OAAO,MAAM,QAAQ;AAAA,MAChD;AACA,aAAO,EAAE,SAAS,OAAO,OAAO,yBAAyB;AAAA,IAC3D;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,cAAc,MAAc,YAAkE;AAClG,UAAM,MAAM,GAAG,KAAK,OAAO;AAC3B,QAAI;AACF,YAAM,WAAW,MAAM,MAAM,KAAK;AAAA,QAChC,QAAQ;AAAA,QACR,SAAS,EAAE,gBAAgB,mBAAmB;AAAA,QAC9C,MAAM,KAAK,UAAU,EAAE,MAAM,aAAa,WAAW,CAAC;AAAA,MACxD,CAAC;AAED,YAAM,OAAO,MAAM,SAAS,KAAK;AAEjC,UAAI,CAAC,SAAS,IAAI;AAChB,eAAO;AAAA,UACL,SAAS;AAAA,UACT,OAAO,KAAK,SAAS,8BAA8B,SAAS,MAAM;AAAA,QACpE;AAAA,MACF;AAEA,aAAO,EAAE,SAAS,MAAM,KAAK;AAAA,IAC/B,SAAS,OAAO;AACd,UAAI,iBAAiB,OAAO;AAC1B,eAAO,EAAE,SAAS,OAAO,OAAO,MAAM,QAAQ;AAAA,MAChD;AACA,aAAO,EAAE,SAAS,OAAO,OAAO,yBAAyB;AAAA,IAC3D;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,kBACJ,SAOA,SAQE;AACF,WAAO,KAAK,QAAQ,WAAW;AAAA,MAC7B,QAAQ;AAAA,MACR,MAAM,KAAK,UAAU;AAAA,QACnB;AAAA,QACA,mBAAmB,SAAS;AAAA,QAC5B,aAAa,SAAS;AAAA,MACxB,CAAC;AAAA,IACH,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,kBAAkB,QAAQ,IAgB5B;AACF,WAAO,KAAK,QAAQ,iBAAiB,KAAK,IAAI,EAAE,QAAQ,MAAM,CAAC;AAAA,EACjE;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,iBACJ,QAM0C;AAC1C,WAAO,KAAK,QAAQ,gBAAgB;AAAA,MAClC,QAAQ;AAAA,MACR,MAAM,KAAK,UAAU,EAAE,OAAO,CAAC;AAAA,IACjC,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASA,MAAM,eAAe,QAajB;AACF,WAAO,KAAK,QAAQ,qBAAqB;AAAA,MACvC,QAAQ;AAAA,MACR,MAAM,KAAK,UAAU,MAAM;AAAA,IAC7B,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,kBAAkBE,OAIpB;AACF,WAAO,KAAK,QAAQ,0BAA0B,mBAAmBA,KAAI,CAAC,IAAI;AAAA,MACxE,QAAQ;AAAA,IACV,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,iBAAiBA,OAYnB;AACF,WAAO,KAAK,QAAQ,0BAA0B,mBAAmBA,KAAI,CAAC,IAAI;AAAA,MACxE,QAAQ;AAAA,IACV,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,aAAa,kBAAkB,OAYjC;AACF,WAAO,KAAK,QAAQ,qCAAqC,eAAe,IAAI;AAAA,MAC1E,QAAQ;AAAA,IACV,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,MAAM,iBAAiBA,OAOpB;AACD,UAAM,SAAS,MAAM,KAAK,iBAAiBA,KAAI;AAC/C,QAAI,OAAO,WAAW,OAAO,MAAM,WAAW;AAC5C,aAAO;AAAA,QACL,SAAS;AAAA,QACT,SAAS,OAAO,KAAK,UAAU;AAAA,UAC7B,IAAI,OAAO,KAAK,QAAQ;AAAA,UACxB,MAAM,OAAO,KAAK,QAAQ;AAAA,UAC1B,MAAM,OAAO,KAAK,QAAQ;AAAA,QAC5B,IAAI;AAAA,MACN;AAAA,IACF;AACA,WAAO,EAAE,SAAS,MAAM;AAAA,EAC1B;AACF;AAGA,IAAI,iBAAmC;AAEhC,SAAS,eAA0B;AACxC,MAAI,CAAC,gBAAgB;AACnB,qBAAiB,IAAI,UAAU;AAAA,EACjC;AACA,SAAO;AACT;;;ACxlBA,YAAY,QAAQ;AACpB,YAAY,UAAU;AACtB,YAAY,QAAQ;AACpB,YAAY,cAAc;AAwB1B,SAAS,uBAA+B;AACtC,SAAY,UAAQ,WAAQ,GAAG,WAAW,eAAe;AAC3D;AAEA,SAAS,cAAc,GAAmB;AACxC,SAAO,EAAE,YAAY,EAAE,QAAQ,OAAO,GAAG,EAAE,QAAQ,QAAQ,EAAE;AAC/D;AAEA,SAAS,mBAAmB,mBAA2B,YAA6B;AAClF,QAAM,mBAAmB,cAAc,iBAAiB;AACxD,QAAM,mBAAmB,cAAc,UAAU;AACjD,SAAO,qBAAqB,oBACrB,iBAAiB,WAAW,mBAAmB,GAAG;AAC3D;AAIA,IAAM,eAAe;AAAA,EACnB,UAAU,oBAAI,IAAgC;AAAA,EAC9C,aAAa;AACf;AAEA,IAAM,eAAe;AAErB,eAAsB,oBACpB,QAC0C;AAC1C,QAAM,MAAM,KAAK,IAAI;AAErB,MAAI,aAAa,cAAc,KAAK,MAAM,aAAa,cAAc,cAAc;AACjF,WAAO,aAAa;AAAA,EACtB;AAEA,MAAI;AACF,UAAM,SAAS,MAAM,OAAO,aAAa,IAAI;AAC7C,QAAI,OAAO,WAAW,OAAO,MAAM,UAAU;AAC3C,mBAAa,SAAS,MAAM;AAC5B,iBAAW,WAAW,OAAO,KAAK,UAAU;AAC1C,cAAM,iBAAiB,cAAc,QAAQ,IAAI;AACjD,qBAAa,SAAS,IAAI,gBAAgB;AAAA,UACxC,SAAS,QAAQ;AAAA,UACjB,MAAM,QAAQ;AAAA,QAChB,CAAC;AAAA,MACH;AACA,mBAAa,cAAc;AAAA,IAC7B;AAAA,EACF,QAAQ;AAAA,EAER;AAEA,SAAO,aAAa;AACtB;AAEO,SAAS,yBAAiC;AAC/C,SAAO,MAAM,KAAK,aAAa,SAAS,OAAO,CAAC,EAAE,OAAO,OAAK,EAAE,OAAO,EAAE;AAC3E;AAEA,eAAe,iBAAiB,aAAkD;AAChF,QAAM,iBAAiB,cAAc,WAAW;AAEhD,MAAI,aAAa,SAAS,IAAI,cAAc,GAAG;AAC7C,WAAO,aAAa,SAAS,IAAI,cAAc;AAAA,EACjD;AAEA,aAAW,CAAC,aAAa,IAAI,KAAK,aAAa,SAAS,QAAQ,GAAG;AACjE,QAAI,eAAe,WAAW,cAAc,GAAG,GAAG;AAChD,aAAO;AAAA,IACT;AAAA,EACF;AAEA,SAAO,EAAE,SAAS,MAAM;AAC1B;AAIO,IAAM,gBAAN,MAAoB;AAAA,EACjB,WAAW;AAAA,EACX,YAAY;AAAA,EACZ,aAAoD;AAAA,EACpD;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EAKR,YACE,QACA,UAKI,CAAC,GACL;AACA,SAAK,SAAS;AACd,SAAK,cAAc,qBAAqB;AACxC,SAAK,aAAa,QAAQ,cAAc;AACxC,SAAK,gBAAgB,QAAQ;AAC7B,SAAK,gBAAgB,QAAQ;AAC7B,SAAK,YAAY,QAAQ,aAAa,CAAC;AAAA,EACzC;AAAA,EAEQ,IAAI,OAAe,KAAa;AACtC,QAAI,KAAK,UAAU,KAAK;AACtB,WAAK,UAAU,IAAI,OAAO,GAAG;AAAA,IAC/B;AAAA,EACF;AAAA;AAAA,EAGA,eAAe;AACb,QAAI;AACF,UAAO,cAAW,KAAK,WAAW,GAAG;AACnC,cAAM,QAAW,YAAS,KAAK,WAAW;AAC1C,aAAK,WAAW,MAAM;AACtB,aAAK,YAAY,MAAM;AAAA,MACzB;AAAA,IACF,QAAQ;AAAA,IAER;AAAA,EACF;AAAA;AAAA,EAGA,MAAM,QAAQ;AACZ,SAAK,aAAa;AAGlB,UAAM,oBAAoB,KAAK,MAAM;AACrC,UAAM,eAAe,uBAAuB;AAC5C,SAAK,IAAI,QAAQ,YAAY,YAAY,qBAAqB;AAG9D,UAAM,KAAK,aAAa;AAGxB,SAAK,aAAa,YAAY,MAAM,KAAK,aAAa,GAAG,KAAK,UAAU;AAAA,EAC1E;AAAA;AAAA,EAGA,OAAO;AACL,QAAI,KAAK,YAAY;AACnB,oBAAc,KAAK,UAAU;AAC7B,WAAK,aAAa;AAAA,IACpB;AAAA,EACF;AAAA;AAAA,EAGA,MAAM,eAAe;AACnB,QAAI;AACF,UAAI,CAAI,cAAW,KAAK,WAAW,EAAG;AAEtC,YAAM,QAAW,YAAS,KAAK,WAAW;AAG1C,UAAI,MAAM,SAAS,KAAK,YAAY,MAAM,YAAY,KAAK,WAAW;AACpE;AAAA,MACF;AAGA,YAAM,oBAAoB,KAAK,MAAM;AAGrC,YAAM,aAAmC,CAAC;AAE1C,UAAI,MAAM,OAAO,KAAK,UAAU;AAC9B,cAAM,SAAY,oBAAiB,KAAK,aAAa;AAAA,UACnD,OAAO,KAAK;AAAA,UACZ,UAAU;AAAA,QACZ,CAAC;AAED,cAAM,KAAc,yBAAgB;AAAA,UAClC,OAAO;AAAA,UACP,WAAW;AAAA,QACb,CAAC;AAED,yBAAiB,QAAQ,IAAI;AAC3B,cAAI,CAAC,KAAK,KAAK,EAAG;AAElB,cAAI;AACF,kBAAM,QAAQ,KAAK,MAAM,IAAI;AAC7B,gBAAI,CAAC,MAAM,WAAW,CAAC,MAAM,aAAa,CAAC,MAAM,WAAW,CAAC,MAAM,WAAW;AAC5E;AAAA,YACF;AAGA,gBAAI,KAAK,iBAAiB,CAAC,mBAAmB,MAAM,SAAS,KAAK,aAAa,GAAG;AAChF;AAAA,YACF;AAGA,kBAAM,iBAAiB,MAAM,iBAAiB,MAAM,OAAO;AAC3D,gBAAI,CAAC,eAAe,SAAS;AAC3B;AAAA,YACF;AAEA,uBAAW,KAAK,KAAK;AAAA,UACvB,QAAQ;AAAA,UAER;AAAA,QACF;AAAA,MACF;AAEA,WAAK,WAAW,MAAM;AACtB,WAAK,YAAY,MAAM;AAEvB,UAAI,WAAW,SAAS,GAAG;AACzB,aAAK,IAAI,QAAQ,YAAY,WAAW,MAAM,4BAA4B;AAE1E,cAAM,WAAW,MAAM,KAAK,OAAO,kBAAkB,YAAY;AAAA,UAC/D,mBAAmB,KAAK;AAAA,UACxB,aAAa,KAAK;AAAA,QACpB,CAAC;AAED,YAAI,SAAS,WAAW,SAAS,MAAM;AACrC,eAAK,IAAI,QAAQ,UAAU,SAAS,KAAK,cAAc,YAAY;AACnE,eAAK,UAAU,SAAS,SAAS,KAAK,cAAc;AAAA,QACtD,OAAO;AACL,eAAK,IAAI,SAAS,gBAAgB,SAAS,KAAK,EAAE;AAAA,QACpD;AAAA,MACF;AAAA,IACF,SAAS,OAAO;AACd,YAAM,MAAM,iBAAiB,QAAQ,QAAQ,IAAI,MAAM,OAAO,KAAK,CAAC;AACpE,WAAK,IAAI,SAAS,gBAAgB,IAAI,OAAO,EAAE;AAC/C,WAAK,UAAU,UAAU,GAAG;AAAA,IAC9B;AAAA,EACF;AACF;;;AC5PA,YAAYC,SAAQ;AACpB,YAAYC,WAAU;AACtB,YAAYC,eAAc;AAuB1B,IAAM,yBAAyB,IAAI,OAAO;AAC1C,IAAM,eAAe;AAEd,IAAM,qBAAN,MAAyB;AAAA,EACtB,aAAoD;AAAA,EACpD;AAAA,EACA;AAAA,EACA;AAAA,EAQR,YACE,QACA,UAGI,CAAC,GACL;AACA,SAAK,SAAS;AACd,SAAK,aAAa,QAAQ,cAAc;AACxC,SAAK,YAAY,QAAQ,aAAa,CAAC;AAAA,EACzC;AAAA,EAEQ,IAAI,OAAe,KAAa;AACtC,SAAK,UAAU,MAAM,OAAO,GAAG;AAAA,EACjC;AAAA,EAEA,MAAM,QAAQ;AAEZ,SAAK,YAAY;AACjB,SAAK,IAAI,QAAQ,8BAA8B;AAG/C,SAAK,aAAa,YAAY,MAAM,KAAK,OAAO,GAAG,KAAK,UAAU;AAAA,EACpE;AAAA,EAEA,OAAO;AACL,QAAI,KAAK,YAAY;AACnB,oBAAc,KAAK,UAAU;AAC7B,WAAK,aAAa;AAAA,IACpB;AAAA,EACF;AAAA;AAAA,EAGQ,wBAAqC;AAC3C,UAAM,QAAQ,oBAAI,IAAY;AAC9B,UAAM,YAAY,aAAa;AAE/B,QAAI,CAAI,eAAW,SAAS,EAAG,QAAO;AAEtC,QAAI;AACF,YAAM,UAAa,gBAAY,WAAW,EAAE,eAAe,KAAK,CAAC;AACjE,iBAAW,SAAS,SAAS;AAC3B,YAAI,CAAC,MAAM,YAAY,KAAK,MAAM,KAAK,WAAW,GAAG,EAAG;AACxD,cAAM,YAAiB,WAAK,WAAW,MAAM,MAAM,UAAU;AAC7D,YAAO,eAAW,SAAS,GAAG;AAC5B,gBAAM,IAAI,MAAM,IAAI;AAAA,QACtB;AAAA,MACF;AAAA,IACF,QAAQ;AAAA,IAER;AAEA,WAAO;AAAA,EACT;AAAA;AAAA,EAGQ,kBAAkB,aAA6B;AAErD,WAAO,YAAY,QAAQ,OAAO,GAAG;AAAA,EACvC;AAAA;AAAA,EAGQ,kBAA4B;AAClC,UAAM,YAAY,aAAa;AAC/B,UAAM,cAAmB,WAAK,WAAW,UAAU;AAEnD,QAAI,CAAI,eAAW,WAAW,EAAG,QAAO,CAAC;AAEzC,UAAM,QAAkB,CAAC;AACzB,QAAI;AACF,YAAM,cAAiB,gBAAY,aAAa,EAAE,eAAe,KAAK,CAAC;AACvE,iBAAW,OAAO,aAAa;AAC7B,YAAI,CAAC,IAAI,YAAY,EAAG;AACxB,cAAM,UAAe,WAAK,aAAa,IAAI,IAAI;AAC/C,cAAM,aAAgB,gBAAY,OAAO,EAAE;AAAA,UACzC,CAAC,MAAM,EAAE,SAAS,QAAQ,KAAK,CAAC,EAAE,WAAW,QAAQ;AAAA,QACvD;AACA,mBAAW,KAAK,YAAY;AAC1B,gBAAM,KAAU,WAAK,SAAS,CAAC,CAAC;AAAA,QAClC;AAAA,MACF;AAAA,IACF,QAAQ;AAAA,IAER;AAEA,WAAO;AAAA,EACT;AAAA;AAAA,EAGQ,cAA2B;AACjC,UAAM,cAAmB,WAAK,WAAW,GAAG,YAAY;AACxD,QAAI;AACF,UAAO,eAAW,WAAW,GAAG;AAC9B,eAAO,KAAK,MAAS,iBAAa,aAAa,OAAO,CAAC;AAAA,MACzD;AAAA,IACF,QAAQ;AAAA,IAER;AACA,WAAO,CAAC;AAAA,EACV;AAAA;AAAA,EAGQ,YAAY,OAAoB;AACtC,oBAAgB,WAAW,CAAC;AAC5B,UAAM,cAAmB,WAAK,WAAW,GAAG,YAAY;AACxD,QAAI;AACF,MAAG,kBAAc,aAAa,KAAK,UAAU,KAAK,GAAG,OAAO;AAAA,IAC9D,QAAQ;AAAA,IAER;AAAA,EACF;AAAA;AAAA,EAGQ,cAAc;AACpB,UAAM,QAAQ,KAAK,gBAAgB;AACnC,UAAM,QAAQ,KAAK,YAAY;AAE/B,eAAW,QAAQ,OAAO;AACxB,UAAI,CAAC,MAAM,IAAI,GAAG;AAChB,YAAI;AACF,gBAAM,QAAW,aAAS,IAAI;AAC9B,gBAAM,IAAI,IAAI,EAAE,YAAY,MAAM,MAAM,cAAc,MAAM,QAAQ;AAAA,QACtE,QAAQ;AAAA,QAER;AAAA,MACF;AAAA,IACF;AAEA,SAAK,YAAY,KAAK;AAAA,EACxB;AAAA;AAAA,EAGA,MAAM,SAAS;AACb,QAAI;AACF,YAAM,aAAa,KAAK,sBAAsB;AAC9C,UAAI,WAAW,SAAS,EAAG;AAE3B,YAAM,YAAY,aAAa;AAC/B,YAAM,QAAQ,KAAK,gBAAgB;AACnC,YAAM,UAAU,KAAK,YAAY;AACjC,YAAM,SAA4B,CAAC;AAEnC,iBAAW,QAAQ,OAAO;AACxB,YAAI;AACF,gBAAM,QAAW,aAAS,IAAI;AAC9B,gBAAM,gBAAgB,QAAQ,IAAI,GAAG,cAAc;AACnD,gBAAM,eAAe,QAAQ,IAAI,GAAG,gBAAgB;AAGpD,cAAI,MAAM,QAAQ,iBAAiB,MAAM,YAAY,aAAc;AAGnE,gBAAM,cAAc,KAAK,IAAI,eAAe,MAAM,OAAO,sBAAsB;AAE/E,cAAI,MAAM,QAAQ,aAAa;AAC7B,oBAAQ,IAAI,IAAI,EAAE,YAAY,MAAM,MAAM,cAAc,MAAM,QAAQ;AACtE;AAAA,UACF;AAGA,gBAAM,SAAY,qBAAiB,MAAM;AAAA,YACvC,OAAO;AAAA,YACP,UAAU;AAAA,UACZ,CAAC;AAED,gBAAM,KAAc,0BAAgB,EAAE,OAAO,QAAQ,WAAW,SAAS,CAAC;AAE1E,2BAAiB,QAAQ,IAAI;AAC3B,gBAAI,CAAC,KAAK,KAAK,EAAG;AAElB,gBAAI;AACF,oBAAM,QAAQ,KAAK,MAAM,IAAI;AAC7B,oBAAM,MAAM,OAAO;AACnB,kBAAI,CAAC,OAAO,IAAI,SAAS,YAAa;AAEtC,oBAAM,UAAU,IAAI;AACpB,kBAAI,CAAC,MAAM,QAAQ,OAAO,EAAG;AAE7B,oBAAM,YAAY,MAAM,aAAa;AACrC,oBAAM,YAAY,MAAM,cAAa,oBAAI,KAAK,GAAE,YAAY;AAC5D,oBAAM,MAAM,MAAM,OAAO;AAEzB,yBAAW,SAAS,SAAS;AAC3B,oBAAI,OAAO,SAAS,WAAY;AAEhC,sBAAM,WAAW,MAAM;AACvB,sBAAM,YAAY,MAAM,SAAS,CAAC;AAGlC,oBAAI,aAAa,QAAQ;AACvB,wBAAM,WAAW,UAAU,aAAa;AAExC,sBAAI,SAAS,SAAS,UAAU,KAAK,SAAS,SAAS,WAAW,GAAG;AAEnE,0BAAM,QAAQ,SAAS,MAAM,UAAU;AACvC,wBAAI,MAAM,UAAU,GAAG;AACrB,4BAAM,cAAc,MAAM,MAAM,SAAS,CAAC;AAC1C,4BAAM,OAAO,YAAY,MAAM,GAAG,EAAE,CAAC;AACrC,0BAAI,QAAQ,WAAW,IAAI,IAAI,GAAG;AAChC,+BAAO,KAAK;AAAA,0BACV,WAAW;AAAA,0BACX,iBAAiB;AAAA,0BACjB,aAAa;AAAA,0BACb;AAAA,wBACF,CAAC;AAAA,sBACH;AAAA,oBACF;AAAA,kBACF;AAAA,gBACF;AAAA,cAIF;AAAA,YACF,QAAQ;AAAA,YAER;AAAA,UACF;AAEA,kBAAQ,IAAI,IAAI,EAAE,YAAY,MAAM,MAAM,cAAc,MAAM,QAAQ;AAAA,QACxE,QAAQ;AAAA,QAER;AAAA,MACF;AAGA,iBAAW,OAAO,OAAO,KAAK,OAAO,GAAG;AACtC,YAAI,CAAI,eAAW,GAAG,GAAG;AACvB,iBAAO,QAAQ,GAAG;AAAA,QACpB;AAAA,MACF;AAEA,WAAK,YAAY,OAAO;AAGxB,YAAM,OAAO,oBAAI,IAAY;AAC7B,YAAM,eAAe,OAAO,OAAO,CAAC,MAAM;AACxC,cAAM,MAAM,GAAG,EAAE,SAAS,IAAI,EAAE,eAAe,IAAI,EAAE,SAAS;AAC9D,YAAI,KAAK,IAAI,GAAG,EAAG,QAAO;AAC1B,aAAK,IAAI,GAAG;AACZ,eAAO;AAAA,MACT,CAAC;AAGD,UAAI,aAAa,SAAS,GAAG;AAC3B,aAAK,IAAI,QAAQ,YAAY,aAAa,MAAM,+BAA+B;AAE/E,cAAM,WAAW,MAAM,KAAK,OAAO,iBAAiB,YAAY;AAChE,YAAI,SAAS,SAAS;AACpB,eAAK,IAAI,QAAQ,YAAY,SAAS,MAAM,UAAU,CAAC,iBAAiB;AACxE,eAAK,UAAU,cAAc,SAAS,MAAM,UAAU,CAAC;AAAA,QACzD,OAAO;AACL,eAAK,IAAI,SAAS,kBAAkB,SAAS,KAAK,EAAE;AAAA,QACtD;AAAA,MACF;AAAA,IACF,SAAS,OAAO;AACd,YAAM,MAAM,iBAAiB,QAAQ,QAAQ,IAAI,MAAM,OAAO,KAAK,CAAC;AACpE,WAAK,IAAI,SAAS,oBAAoB,IAAI,OAAO,EAAE;AACnD,WAAK,UAAU,UAAU,GAAG;AAAA,IAC9B;AAAA,EACF;AACF;;;AChTA,SAAS,iBAAAC,gBAAe,gBAAAC,eAAc,cAAAC,mBAAkB;AACxD,SAAS,QAAAC,aAAY;AAwBrB,IAAM,cAAc;AACpB,IAAM,oBAAoB;AAEnB,IAAM,eAAN,MAAM,cAAa;AAAA,EAChB,aAAoD;AAAA,EACpD;AAAA,EACA;AAAA,EAER,cAAc;AACZ,SAAK,WAAWC,MAAK,WAAW,GAAG,WAAW;AAC9C,SAAK,SAAS;AAAA,MACZ,SAAS;AAAA,MACT,KAAK,QAAQ;AAAA,MACb,YAAW,oBAAI,KAAK,GAAE,YAAY;AAAA,MAClC,YAAW,oBAAI,KAAK,GAAE,YAAY;AAAA,MAClC,eAAe,EAAE,UAAU,MAAM,eAAe,GAAG,WAAW,KAAK;AAAA,MACnE,eAAe,EAAE,gBAAgB,GAAG,gBAAgB,GAAG,eAAe,MAAM,WAAW,KAAK;AAAA,MAC5F,iBAAiB,CAAC;AAAA,MAClB,gBAAgB;AAAA,IAClB;AAAA,EACF;AAAA;AAAA,EAGA,QAAQ;AACN,oBAAgB,WAAW,CAAC;AAC5B,SAAK,MAAM;AACX,SAAK,aAAa,YAAY,MAAM,KAAK,MAAM,GAAG,iBAAiB;AAAA,EACrE;AAAA;AAAA,EAGA,OAAO;AACL,QAAI,KAAK,YAAY;AACnB,oBAAc,KAAK,UAAU;AAC7B,WAAK,aAAa;AAAA,IACpB;AACA,SAAK,OAAO,UAAU;AACtB,SAAK,OAAO,MAAM;AAClB,SAAK,OAAO,aAAY,oBAAI,KAAK,GAAE,YAAY;AAC/C,SAAK,MAAM;AAAA,EACb;AAAA;AAAA,EAGA,OAAO,SAAgC;AAErC,QAAI,QAAQ,eAAe;AACzB,WAAK,OAAO,gBAAgB,EAAE,GAAG,KAAK,OAAO,eAAe,GAAG,QAAQ,cAAc;AACrF,aAAO,QAAQ;AAAA,IACjB;AACA,QAAI,QAAQ,eAAe;AACzB,WAAK,OAAO,gBAAgB,EAAE,GAAG,KAAK,OAAO,eAAe,GAAG,QAAQ,cAAc;AACrF,aAAO,QAAQ;AAAA,IACjB;AACA,WAAO,OAAO,KAAK,QAAQ,OAAO;AAAA,EACpC;AAAA;AAAA,EAGA,OAAO,oBAA4B;AACjC,WAAOA,MAAK,WAAW,GAAG,WAAW;AAAA,EACvC;AAAA;AAAA,EAGA,OAAO,OAA4B;AACjC,UAAM,WAAW,cAAa,kBAAkB;AAChD,QAAI;AACF,UAAIC,YAAW,QAAQ,GAAG;AACxB,eAAO,KAAK,MAAMC,cAAa,UAAU,OAAO,CAAC;AAAA,MACnD;AAAA,IACF,QAAQ;AAAA,IAER;AACA,WAAO;AAAA,EACT;AAAA,EAEQ,QAAQ;AACd,SAAK,OAAO,aAAY,oBAAI,KAAK,GAAE,YAAY;AAC/C,QAAI;AACF,MAAAC,eAAc,KAAK,UAAU,KAAK,UAAU,KAAK,QAAQ,MAAM,CAAC,GAAG,OAAO;AAAA,IAC5E,QAAQ;AAAA,IAER;AAAA,EACF;AACF;;;AN9FA,SAAS,IAAI,OAAe,KAAa;AACvC,QAAM,OAAO,KAAI,oBAAI,KAAK,GAAE,YAAY,CAAC,MAAM,KAAK,KAAK,GAAG;AAAA;AAC5D,MAAI;AACF,mBAAe,WAAW,GAAG,IAAI;AAAA,EACnC,QAAQ;AAAA,EAER;AACF;AAIA,eAAe,2BAA2B,QAAyC;AACjF,MAAI;AACF,UAAM,SAAS,MAAM,OAAO,aAAa,IAAI;AAC7C,QAAI,OAAO,WAAW,OAAO,MAAM,UAAU;AAC3C,YAAM,YAAY;AAAA,QAChB,WAAW,KAAK,IAAI;AAAA,QACpB,UAAU,OAAO,KAAK,SACnB,OAAO,CAAC,MAAoC,EAAE,eAAe,EAC7D,IAAI,CAAC,OAAuC,EAAE,MAAM,EAAE,MAAM,MAAM,EAAE,KAAK,EAAE;AAAA,MAChF;AACA,MAAAC,eAAc,4BAA4B,GAAG,KAAK,UAAU,WAAW,MAAM,CAAC,CAAC;AAC/E,UAAI,SAAS,mCAAmC,UAAU,SAAS,MAAM,aAAa;AAAA,IACxF;AAAA,EACF,SAAS,OAAO;AACd,QAAI,SAAS,4CAA4C,KAAK,EAAE;AAAA,EAClE;AACF;AAIA,eAAe,qBAAqB,QAAyC;AAC3E,QAAM,cAAc,qBAAqB;AACzC,MAAI,CAACC,YAAW,WAAW,EAAG;AAE9B,MAAI;AACF,UAAM,QAAQC,aAAY,WAAW;AACrC,eAAW,QAAQ,OAAO;AACxB,UAAI,CAAC,KAAK,SAAS,OAAO,EAAG;AAE7B,YAAM,MAAM,SAAS,KAAK,QAAQ,SAAS,EAAE,GAAG,EAAE;AAClD,UAAI,MAAM,GAAG,EAAG;AAGhB,UAAI,YAAY;AAChB,UAAI;AACF,gBAAQ,KAAK,KAAK,CAAC;AACnB,oBAAY;AAAA,MACd,QAAQ;AAAA,MAER;AAEA,UAAI,CAAC,WAAW;AACd,YAAI;AACF,gBAAM,cAAc,KAAK,MAAMC,cAAa,GAAG,WAAW,IAAI,IAAI,IAAI,OAAO,CAAC;AAC9E,cAAI,YAAY,WAAW;AACzB,kBAAM,OAAO,WAAW,YAAY,SAAS;AAC7C,gBAAI,QAAQ,uBAAuB,YAAY,UAAU,MAAM,GAAG,CAAC,CAAC,SAAS,GAAG,QAAQ;AAAA,UAC1F;AACA,qBAAW,GAAG,WAAW,IAAI,IAAI,EAAE;AAAA,QACrC,QAAQ;AAEN,cAAI;AAAE,uBAAW,GAAG,WAAW,IAAI,IAAI,EAAE;AAAA,UAAG,QAAQ;AAAA,UAAe;AAAA,QACrE;AAAA,MACF;AAAA,IACF;AAAA,EACF,SAAS,OAAO;AACd,QAAI,SAAS,gCAAgC,KAAK,EAAE;AAAA,EACtD;AACF;AAIA,eAAe,OAAO;AACpB,QAAM,UAAU,WAAW;AAC3B,EAAAH,eAAc,SAAS,OAAO,QAAQ,GAAG,CAAC;AAE1C,MAAI,QAAQ,yBAAyB,QAAQ,GAAG,GAAG;AAGnD,QAAM,eAAe,IAAI,aAAa;AAGtC,QAAM,UAAU,MAAM;AACpB,QAAI,QAAQ,sBAAsB;AAClC,iBAAa,KAAK;AAClB,QAAIC,YAAW,OAAO,GAAG;AACvB,UAAI;AAAE,mBAAW,OAAO;AAAA,MAAG,QAAQ;AAAA,MAAe;AAAA,IACpD;AACA,YAAQ,KAAK,CAAC;AAAA,EAChB;AACA,UAAQ,GAAG,UAAU,OAAO;AAC5B,UAAQ,GAAG,WAAW,OAAO;AAG7B,MAAI;AACJ,MAAI;AACF,aAAS,WAAW;AAAA,EACtB,QAAQ;AACN,QAAI,QAAQ,uCAAuC;AACnD,aAAS,EAAE,QAAQ,EAAE,2BAA2B,GAAG,sBAAsB,GAAG,EAAE;AAAA,EAChF;AAGA,QAAM,SAAS,aAAa;AAC5B,MAAI,CAAC,OAAO,UAAU,GAAG;AACvB,QAAI,SAAS,mDAAmD;AAChE,YAAQ;AACR;AAAA,EACF;AAGA,kBAAgB,qBAAqB,CAAC;AAGtC,QAAM,2BAA2B,MAAM;AAGvC,QAAM,qBAAqB,MAAM;AAGjC,QAAM,iBAAkB,OAA+D,QAAQ,6BAA6B,KAAK;AACjI,QAAM,UAAU,IAAI,cAAc,QAAQ;AAAA,IACxC,YAAY;AAAA,IACZ,WAAW;AAAA,MACT,QAAQ,CAAC,UAAU,IAAI,QAAQ,UAAU,KAAK,mBAAmB;AAAA,MACjE,SAAS,CAAC,QAAQ,IAAI,SAAS,kBAAkB,IAAI,OAAO,EAAE;AAAA,MAC9D,KAAK,CAAC,OAAO,QAAQ,IAAI,OAAO,GAAG;AAAA,IACrC;AAAA,EACF,CAAC;AACD,QAAM,QAAQ,MAAM;AAGpB,QAAM,gBAAgB,IAAI,mBAAmB,QAAQ;AAAA,IACnD,YAAY;AAAA,IACZ,WAAW;AAAA,MACT,aAAa,CAAC,UAAU;AACtB,YAAI,QAAQ,YAAY,KAAK,iBAAiB;AAC9C,qBAAa,OAAO,EAAE,eAAe,EAAE,gBAAgB,OAAO,gBAAe,oBAAI,KAAK,GAAE,YAAY,GAAG,WAAW,KAAK,EAAE,CAAC;AAAA,MAC5H;AAAA,MACA,SAAS,CAAC,QAAQ;AAChB,YAAI,SAAS,0BAA0B,IAAI,OAAO,EAAE;AACpD,qBAAa,OAAO,EAAE,eAAe,EAAE,WAAW,IAAI,QAAQ,EAAE,CAAC;AAAA,MACnE;AAAA,MACA,KAAK,CAAC,OAAO,QAAQ,IAAI,OAAO,GAAG;AAAA,IACrC;AAAA,EACF,CAAC;AACD,QAAM,cAAc,MAAM;AAG1B,eAAa,MAAM;AAGnB,cAAY,YAAY;AACtB,UAAM,2BAA2B,MAAM;AAEvC,QAAI;AACF,YAAM,YAAY,KAAK,MAAME,cAAa,4BAA4B,GAAG,OAAO,CAAC;AACjF,mBAAa,OAAO;AAAA,QAClB,iBAAiB,UAAU,YAAY,CAAC;AAAA,MAC1C,CAAC;AAAA,IACH,QAAQ;AAAA,IAAe;AAAA,EACzB,GAAG,GAAK;AAGR,cAAY,MAAM,qBAAqB,MAAM,GAAG,GAAM;AAEtD,MAAI,QAAQ,gFAA2E;AAGvF,QAAM,IAAI,QAAQ,MAAM;AAAA,EAAC,CAAC;AAC5B;AAEA,KAAK,EAAE,MAAM,CAAC,QAAQ;AACpB,MAAI,SAAS,mBAAmB,GAAG,EAAE;AACrC,UAAQ,KAAK,CAAC;AAChB,CAAC;","names":["existsSync","writeFileSync","readFileSync","readdirSync","existsSync","path","path","existsSync","path","fs","path","readline","writeFileSync","readFileSync","existsSync","join","join","existsSync","readFileSync","writeFileSync","writeFileSync","existsSync","readdirSync","readFileSync"]}
1
+ {"version":3,"sources":["../src/daemon-runner.ts","../src/core/config.ts","../src/utils/paths.ts","../src/core/api-client.ts","../src/core/claude-watcher.ts","../src/core/skill-usage-detector.ts","../src/utils/status-writer.ts"],"sourcesContent":["/**\n * Daemon Runner - Entry point for the forked daemon child process.\n *\n * Runs:\n * 1. Claude history watcher (auto-syncs conversations for tracked projects)\n * 2. Tracked projects cache writer (shell hooks read this for fast detection)\n * 3. Stale session cleanup (ends sessions where terminal PID is dead)\n */\n\nimport { existsSync, writeFileSync, unlinkSync, readFileSync, readdirSync, appendFileSync } from \"fs\";\nimport { loadConfig } from \"./core/config.js\";\nimport { getApiClient } from \"./core/api-client.js\";\nimport { getPidFile, getLogFile, getTrackedProjectsCacheFile, getActiveSessionsDir, ensureDirectory } from \"./utils/paths.js\";\nimport { ClaudeWatcher, loadTrackedProjects } from \"./core/claude-watcher.js\";\nimport { SkillUsageDetector } from \"./core/skill-usage-detector.js\";\nimport { StatusWriter } from \"./utils/status-writer.js\";\n\n// ── Daemon Logger ──\n\nfunction log(level: string, msg: string) {\n const line = `[${new Date().toISOString()}] [${level}] ${msg}\\n`;\n try {\n appendFileSync(getLogFile(), line);\n } catch {\n // Can't log, ignore\n }\n}\n\n// ── Tracked Projects Cache Writer ──\n\nasync function updateTrackedProjectsCache(client: ReturnType<typeof getApiClient>) {\n try {\n const result = await client.listProjects(true);\n if (result.success && result.data?.projects) {\n const cacheData = {\n updatedAt: Date.now(),\n projects: result.data.projects\n .filter((p: { trackingEnabled: boolean }) => p.trackingEnabled)\n .map((p: { path: string; name: string }) => ({ path: p.path, name: p.name })),\n };\n writeFileSync(getTrackedProjectsCacheFile(), JSON.stringify(cacheData, null, 2));\n log(\"DEBUG\", `Updated tracked projects cache: ${cacheData.projects.length} project(s)`);\n }\n } catch (error) {\n log(\"ERROR\", `Failed to update tracked projects cache: ${error}`);\n }\n}\n\n// ── Stale Session Cleanup ──\n\nasync function cleanupStaleSessions(client: ReturnType<typeof getApiClient>) {\n const sessionsDir = getActiveSessionsDir();\n if (!existsSync(sessionsDir)) return;\n\n try {\n const files = readdirSync(sessionsDir);\n for (const file of files) {\n if (!file.endsWith(\".json\")) continue;\n\n const pid = parseInt(file.replace(\".json\", \"\"), 10);\n if (isNaN(pid)) continue;\n\n // Check if process is still running\n let isRunning = false;\n try {\n process.kill(pid, 0);\n isRunning = true;\n } catch {\n // Process doesn't exist\n }\n\n if (!isRunning) {\n try {\n const sessionData = JSON.parse(readFileSync(`${sessionsDir}/${file}`, \"utf-8\"));\n if (sessionData.sessionId) {\n await client.endSession(sessionData.sessionId);\n log(\"INFO\", `Ended stale session ${sessionData.sessionId.slice(0, 8)} (PID ${pid} dead)`);\n }\n unlinkSync(`${sessionsDir}/${file}`);\n } catch {\n // Clean up the file even if API call fails\n try { unlinkSync(`${sessionsDir}/${file}`); } catch { /* ignore */ }\n }\n }\n }\n } catch (error) {\n log(\"ERROR\", `Stale session cleanup error: ${error}`);\n }\n}\n\n// ── Main Daemon Loop ──\n\nasync function main() {\n const pidFile = getPidFile();\n writeFileSync(pidFile, String(process.pid));\n\n log(\"INFO\", `Daemon starting (PID: ${process.pid})`);\n\n // Status writer\n const statusWriter = new StatusWriter();\n\n // Signal handlers\n const cleanup = () => {\n log(\"INFO\", \"Daemon shutting down\");\n statusWriter.stop();\n if (existsSync(pidFile)) {\n try { unlinkSync(pidFile); } catch { /* ignore */ }\n }\n process.exit(0);\n };\n process.on(\"SIGINT\", cleanup);\n process.on(\"SIGTERM\", cleanup);\n\n // Load config\n let config;\n try {\n config = loadConfig();\n } catch {\n log(\"WARN\", \"Could not load config, using defaults\");\n config = { daemon: { conversationCheckInterval: 5, patternCheckInterval: 60 } };\n }\n\n // Check API key\n const client = getApiClient();\n if (!client.hasApiKey()) {\n log(\"ERROR\", \"Not logged in. Daemon cannot run without API key.\");\n cleanup();\n return;\n }\n\n // Fetch user info for status display\n try {\n const authResult = await client.authenticate();\n if (authResult.success && authResult.data?.user) {\n statusWriter.update({ user: authResult.data.user.name });\n log(\"INFO\", `Authenticated as: ${authResult.data.user.name}`);\n }\n } catch (err) {\n log(\"WARN\", `Could not fetch user info: ${err}`);\n }\n\n // Ensure directories\n ensureDirectory(getActiveSessionsDir());\n\n // Initial cache update + populate status with tracked projects\n await updateTrackedProjectsCache(client);\n try {\n const cacheData = JSON.parse(readFileSync(getTrackedProjectsCacheFile(), \"utf-8\"));\n statusWriter.update({ trackedProjects: cacheData.projects || [] });\n } catch { /* ignore */ }\n\n // Clean up stale sessions from previous crashes\n await cleanupStaleSessions(client);\n\n // Start Claude watcher\n const watchInterval = ((config as { daemon?: { conversationCheckInterval?: number } }).daemon?.conversationCheckInterval || 5) * 1000;\n const watcher = new ClaudeWatcher(client, {\n intervalMs: watchInterval,\n callbacks: {\n onSync: (count) => {\n log(\"INFO\", `Synced ${count} Claude prompt(s)`);\n statusWriter.update({ claudeWatcher: { promptsSynced: count, lastSync: new Date().toISOString(), lastError: null } });\n },\n onError: (err) => {\n log(\"ERROR\", `Watcher error: ${err.message}`);\n statusWriter.update({ claudeWatcher: { lastError: err.message } });\n },\n log: (level, msg) => log(level, msg),\n },\n });\n await watcher.start();\n\n // Start skill usage detector\n const skillDetector = new SkillUsageDetector(client, {\n intervalMs: 30000,\n callbacks: {\n onDetection: (count) => {\n log(\"INFO\", `Detected ${count} skill usage(s)`);\n statusWriter.update({ skillDetector: { usagesDetected: count, lastDetection: new Date().toISOString(), lastError: null } });\n },\n onError: (err) => {\n log(\"ERROR\", `Skill detection error: ${err.message}`);\n statusWriter.update({ skillDetector: { lastError: err.message } });\n },\n log: (level, msg) => log(level, msg),\n },\n });\n await skillDetector.start();\n\n // Start status writer\n statusWriter.start();\n\n // Refresh tracked projects cache every 60s\n setInterval(async () => {\n await updateTrackedProjectsCache(client);\n // Update status writer with project count\n try {\n const cacheData = JSON.parse(readFileSync(getTrackedProjectsCacheFile(), \"utf-8\"));\n statusWriter.update({\n trackedProjects: cacheData.projects || [],\n });\n } catch { /* ignore */ }\n }, 60000);\n\n // Clean up stale sessions every 5 minutes\n setInterval(() => cleanupStaleSessions(client), 300000);\n\n log(\"INFO\", \"Daemon started successfully — watching Claude conversations & skill usage\");\n\n // Keep alive\n await new Promise(() => {});\n}\n\nmain().catch((err) => {\n log(\"ERROR\", `Daemon crashed: ${err}`);\n process.exit(1);\n});\n","/**\n * Configuration management for Skillo.\n */\n\nimport { readFileSync, writeFileSync, existsSync } from \"fs\";\nimport { dirname } from \"path\";\nimport YAML from \"yaml\";\nimport { ensureDirectory, getConfigFile } from \"../utils/paths.js\";\n\nexport interface SkilloConfig {\n // Shell settings\n defaultShell: string | null;\n\n // Pattern detection\n patternDetection: {\n minCount: number;\n sessionTimeout: number;\n similarityThreshold: number;\n maxSequenceLength: number;\n };\n\n // Privacy settings\n privacy: {\n autoRedact: boolean;\n trackOutput: boolean;\n neverTrack: string[];\n redactionPatterns: string[];\n };\n\n // Notification settings\n notifications: {\n enabled: boolean;\n style: \"inline\" | \"desktop\" | \"both\";\n sound: boolean;\n };\n\n // Skill generation\n skillGeneration: {\n outputDir: string | null;\n includeScripts: boolean;\n includeExamples: boolean;\n autoGenerate: boolean;\n };\n\n // Claude Code integration\n claudeCode: {\n watchConversations: boolean;\n conversationDir: string | null;\n };\n\n // Daemon settings\n daemon: {\n logLevel: \"DEBUG\" | \"INFO\" | \"WARN\" | \"ERROR\";\n patternCheckInterval: number;\n conversationCheckInterval: number;\n };\n\n // Team settings\n team: {\n enabled: boolean;\n slug: string | null;\n autoSync: boolean;\n syncInterval: number;\n };\n\n // API settings\n api: {\n baseUrl: string | null;\n timeout: number;\n };\n\n // API key (stored separately, not in YAML for security)\n apiKey?: string;\n}\n\nexport function getDefaultConfig(): SkilloConfig {\n return {\n defaultShell: null,\n\n patternDetection: {\n minCount: 3,\n sessionTimeout: 30,\n similarityThreshold: 0.8,\n maxSequenceLength: 10,\n },\n\n privacy: {\n autoRedact: true,\n trackOutput: false,\n neverTrack: [\n \"*password*\",\n \"*secret*\",\n \"vault *\",\n \"1password *\",\n \"op *\",\n \"export *_KEY=*\",\n \"export *_SECRET=*\",\n \"export *_TOKEN=*\",\n ],\n redactionPatterns: [\n \"password[=:]\\\\s*\\\\S+\",\n \"token[=:]\\\\s*\\\\S+\",\n \"secret[=:]\\\\s*\\\\S+\",\n \"api[_-]?key[=:]\\\\s*\\\\S+\",\n \"auth[=:]\\\\s*\\\\S+\",\n \"bearer\\\\s+\\\\S+\",\n \"-----BEGIN.*PRIVATE KEY-----\",\n \"AKIA[0-9A-Z]{16}\",\n '[\"\\'][A-Za-z0-9+/]{40,}[\"\\']',\n ],\n },\n\n notifications: {\n enabled: true,\n style: \"inline\",\n sound: false,\n },\n\n skillGeneration: {\n outputDir: null,\n includeScripts: true,\n includeExamples: true,\n autoGenerate: false,\n },\n\n claudeCode: {\n watchConversations: true,\n conversationDir: null,\n },\n\n daemon: {\n logLevel: \"INFO\",\n patternCheckInterval: 60,\n conversationCheckInterval: 5,\n },\n\n team: {\n enabled: false,\n slug: null,\n autoSync: true,\n syncInterval: 300,\n },\n\n api: {\n baseUrl: null,\n timeout: 30000,\n },\n };\n}\n\nfunction deepMerge<T extends object>(base: T, override: Partial<T>): T {\n const result = { ...base };\n\n for (const key of Object.keys(override) as Array<keyof T>) {\n const value = override[key];\n if (\n value !== undefined &&\n typeof result[key] === \"object\" &&\n result[key] !== null &&\n typeof value === \"object\" &&\n value !== null &&\n !Array.isArray(value)\n ) {\n result[key] = deepMerge(\n result[key] as Record<string, unknown>,\n value as Record<string, unknown>\n ) as T[keyof T];\n } else if (value !== undefined) {\n result[key] = value as T[keyof T];\n }\n }\n\n return result;\n}\n\nexport function loadConfig(path?: string): SkilloConfig {\n const configPath = path || getConfigFile();\n\n if (!existsSync(configPath)) {\n return getDefaultConfig();\n }\n\n try {\n const content = readFileSync(configPath, \"utf-8\");\n const parsed = YAML.parse(content) || {};\n\n // Convert snake_case to camelCase for compatibility\n const converted = convertKeysToCamelCase(parsed);\n const defaultConfig = getDefaultConfig();\n\n // Deep merge with default config\n return deepMerge(defaultConfig, converted as Partial<SkilloConfig>);\n } catch {\n return getDefaultConfig();\n }\n}\n\nexport function saveConfig(config: SkilloConfig, path?: string): void {\n const configPath = path || getConfigFile();\n\n ensureDirectory(dirname(configPath));\n\n // Convert camelCase to snake_case for YAML file\n const converted = convertKeysToSnakeCase(config);\n\n const content = YAML.stringify(converted, {\n indent: 2,\n lineWidth: 0,\n });\n\n writeFileSync(configPath, content, \"utf-8\");\n}\n\nexport function getConfigValue(config: SkilloConfig, key: string): unknown {\n const keys = key.split(\".\");\n let current: unknown = config;\n\n for (const k of keys) {\n if (current && typeof current === \"object\" && k in current) {\n current = (current as Record<string, unknown>)[k];\n } else {\n return undefined;\n }\n }\n\n return current;\n}\n\nexport function setConfigValue(config: SkilloConfig, key: string, value: unknown): SkilloConfig {\n const keys = key.split(\".\");\n const result = JSON.parse(JSON.stringify(config)) as SkilloConfig;\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n let current: any = result;\n\n for (let i = 0; i < keys.length - 1; i++) {\n const k = keys[i];\n if (!(k in current)) {\n current[k] = {};\n }\n current = current[k] as Record<string, unknown>;\n }\n\n current[keys[keys.length - 1]] = value;\n return result;\n}\n\n// Helper functions for key conversion\nfunction toCamelCase(str: string): string {\n return str.replace(/_([a-z])/g, (_, letter) => letter.toUpperCase());\n}\n\nfunction toSnakeCase(str: string): string {\n return str.replace(/[A-Z]/g, (letter) => `_${letter.toLowerCase()}`);\n}\n\nfunction convertKeysToCamelCase(obj: unknown): unknown {\n if (Array.isArray(obj)) {\n return obj.map(convertKeysToCamelCase);\n }\n\n if (obj !== null && typeof obj === \"object\") {\n const result: Record<string, unknown> = {};\n for (const [key, value] of Object.entries(obj)) {\n result[toCamelCase(key)] = convertKeysToCamelCase(value);\n }\n return result;\n }\n\n return obj;\n}\n\nfunction convertKeysToSnakeCase(obj: unknown): unknown {\n if (Array.isArray(obj)) {\n return obj.map(convertKeysToSnakeCase);\n }\n\n if (obj !== null && typeof obj === \"object\") {\n const result: Record<string, unknown> = {};\n for (const [key, value] of Object.entries(obj)) {\n result[toSnakeCase(key)] = convertKeysToSnakeCase(value);\n }\n return result;\n }\n\n return obj;\n}\n","/**\n * Path utilities for Skillo.\n *\n * Handles platform-specific path resolution for data directories,\n * configuration files, and skill storage locations.\n */\n\nimport { homedir } from \"os\";\nimport { join } from \"path\";\nimport { mkdirSync, existsSync } from \"fs\";\n\nexport function getHomeDir(): string {\n return homedir();\n}\n\n/**\n * Get Skillo data directory.\n * This is where the database and other data files are stored.\n *\n * - Linux/macOS: ~/.skillo/\n * - Windows: %USERPROFILE%/.skillo/\n */\nexport function getDataDir(): string {\n const envPath = process.env.SKILLO_DATA_DIR;\n if (envPath) return envPath;\n\n return join(getHomeDir(), \".skillo\");\n}\n\n/**\n * Get Skillo configuration directory.\n * This is where config.yaml and other configuration files are stored.\n *\n * - Linux: ~/.config/skillo/\n * - macOS: ~/.config/skillo/\n * - Windows: %USERPROFILE%/.config/skillo/\n */\nexport function getConfigDir(): string {\n const envPath = process.env.SKILLO_CONFIG_DIR;\n if (envPath) return envPath;\n\n const xdgConfig = process.env.XDG_CONFIG_HOME;\n if (xdgConfig) return join(xdgConfig, \"skillo\");\n\n return join(getHomeDir(), \".config\", \"skillo\");\n}\n\n/**\n * Get personal skills directory.\n * This is where generated skills are stored and where Claude Code\n * reads personal skills from.\n *\n * - All platforms: ~/.claude/skills/\n */\nexport function getSkillsDir(): string {\n const envPath = process.env.SKILLO_SKILLS_DIR;\n if (envPath) return envPath;\n\n return join(getClaudeDir(), \"skills\");\n}\n\n/**\n * Get Claude Code directory.\n * This is where Claude Code stores its data, including conversations.\n *\n * - All platforms: ~/.claude/\n */\nexport function getClaudeDir(): string {\n return join(getHomeDir(), \".claude\");\n}\n\n/**\n * Get project-specific skills directory.\n * This is where project-specific skills are stored.\n *\n * - Path: <project>/.claude/skills/\n */\nexport function getProjectSkillsDir(projectPath?: string): string {\n const project = projectPath || process.cwd();\n return join(project, \".claude\", \"skills\");\n}\n\n/**\n * Get team skills directory.\n * This is where team skills are synced to.\n *\n * - Path: ~/.claude/skills/team/<team-slug>/\n */\nexport function getTeamSkillsDir(teamSlug?: string): string {\n const base = join(getSkillsDir(), \"team\");\n if (teamSlug) return join(base, teamSlug);\n return base;\n}\n\n/**\n * Ensure a directory exists, creating it if necessary.\n *\n * @returns True if directory was created, False if it already existed.\n */\nexport function ensureDirectory(path: string): boolean {\n if (existsSync(path)) {\n return false;\n }\n\n mkdirSync(path, { recursive: true });\n return true;\n}\n\n/**\n * Get path to daemon log file.\n */\nexport function getLogFile(): string {\n return join(getDataDir(), \"daemon.log\");\n}\n\n/**\n * Get path to daemon PID file.\n */\nexport function getPidFile(): string {\n return join(getDataDir(), \"daemon.pid\");\n}\n\n/**\n * Get path to SQLite database file.\n */\nexport function getDbPath(): string {\n return join(getDataDir(), \"skillo.db\");\n}\n\n/**\n * Get path to config file.\n */\nexport function getConfigFile(): string {\n return join(getConfigDir(), \"config.yaml\");\n}\n\n/**\n * Get path to tracked projects cache file.\n * Written by daemon, read by shell hooks for fast project detection.\n */\nexport function getTrackedProjectsCacheFile(): string {\n return join(getDataDir(), \"tracked-projects.json\");\n}\n\n/**\n * Get path to active sessions directory.\n * Each terminal gets a file named by its PID.\n */\nexport function getActiveSessionsDir(): string {\n return join(getDataDir(), \"active-sessions\");\n}\n\n/**\n * Get path to shell integration directory.\n */\nexport function getShellIntegrationDir(): string {\n return join(getDataDir(), \"shell-integration\");\n}\n","/**\n * Skillo Platform API Client\n *\n * Handles communication with the Skillo platform API.\n */\n\nimport { loadConfig, saveConfig } from \"./config.js\";\n\nconst DEFAULT_API_URL = \"https://www.skillo.one/api/cli\";\n\ninterface ApiResponse<T = unknown> {\n success: boolean;\n data?: T;\n error?: string;\n message?: string;\n}\n\ninterface AuthResponse {\n success: boolean;\n user: {\n id: string;\n email: string;\n name: string;\n };\n}\n\ninterface SyncResponse {\n success: boolean;\n skills?: Array<{\n id: string;\n name: string;\n slug: string;\n description: string;\n content: string;\n commands: string[];\n }>;\n patterns?: Array<{\n id: string;\n name: string;\n description: string;\n commands: string[];\n status: string;\n }>;\n}\n\ninterface GenerateResponse {\n success: boolean;\n skill: {\n id: string;\n name: string;\n slug: string;\n description: string;\n content: string;\n commands: string[];\n };\n}\n\ninterface DeviceAuthResponse {\n code: string;\n verification_url: string;\n expires_in: number;\n interval: number;\n}\n\ninterface TokenStatusResponse {\n status: \"pending\" | \"ready\" | \"expired\" | \"used\" | \"not_found\";\n}\n\ninterface TokenExchangeResponse {\n success: boolean;\n api_key: string;\n user: {\n id: string;\n };\n}\n\nexport class ApiClient {\n private baseUrl: string;\n private apiKey: string | null;\n\n constructor() {\n const config = loadConfig();\n this.baseUrl = config.api?.baseUrl || DEFAULT_API_URL;\n this.apiKey = this.loadApiKey();\n }\n\n /**\n * Load API key from config\n */\n private loadApiKey(): string | null {\n const config = loadConfig();\n return (config as { apiKey?: string }).apiKey || null;\n }\n\n /**\n * Save API key to config\n */\n saveApiKey(key: string): void {\n const config = loadConfig();\n (config as { apiKey?: string }).apiKey = key;\n saveConfig(config);\n this.apiKey = key;\n }\n\n /**\n * Clear API key from config\n */\n clearApiKey(): void {\n const config = loadConfig();\n delete (config as { apiKey?: string }).apiKey;\n saveConfig(config);\n this.apiKey = null;\n }\n\n /**\n * Check if API key is configured\n */\n hasApiKey(): boolean {\n return !!this.apiKey;\n }\n\n /**\n * Get the configured API key (masked)\n */\n getMaskedApiKey(): string | null {\n if (!this.apiKey) return null;\n return `${this.apiKey.substring(0, 12)}...${this.apiKey.substring(this.apiKey.length - 4)}`;\n }\n\n /**\n * Set the API base URL\n */\n setBaseUrl(url: string): void {\n const config = loadConfig();\n if (!config.api) {\n (config as { api?: { baseUrl: string } }).api = { baseUrl: url };\n } else {\n config.api.baseUrl = url;\n }\n saveConfig(config);\n this.baseUrl = url;\n }\n\n /**\n * Make an authenticated API request\n */\n private async request<T>(\n endpoint: string,\n options: RequestInit = {}\n ): Promise<ApiResponse<T>> {\n if (!this.apiKey) {\n return { success: false, error: \"No API key configured. Run 'skillo login' first.\" };\n }\n\n const url = `${this.baseUrl}${endpoint}`;\n const headers: HeadersInit = {\n \"Content-Type\": \"application/json\",\n Authorization: `Bearer ${this.apiKey}`,\n ...options.headers,\n };\n\n try {\n const response = await fetch(url, {\n ...options,\n headers,\n });\n\n const data = await response.json();\n\n if (process.env.SKILLO_DEBUG) {\n console.log('[DEBUG] Request:', url);\n console.log('[DEBUG] Status:', response.status);\n console.log('[DEBUG] Response:', JSON.stringify(data));\n }\n\n if (!response.ok) {\n return {\n success: false,\n error: data.error || `Request failed with status ${response.status}`,\n };\n }\n\n // If the response already has success/data structure, return as-is\n // Otherwise wrap the data\n if (data.success !== undefined && data.data !== undefined) {\n return { success: data.success, data: data.data as T, message: data.message };\n }\n\n // If response just has success field (no data wrapper), return as-is\n if (data.success !== undefined) {\n return data;\n }\n\n // Legacy: wrap raw data\n return { success: true, data: data as T };\n } catch (error) {\n if (process.env.SKILLO_DEBUG) {\n console.log('[DEBUG] Error:', error);\n }\n if (error instanceof Error) {\n if (error.message.includes(\"ECONNREFUSED\")) {\n return {\n success: false,\n error: \"Cannot connect to Skillo platform. Is the server running?\",\n };\n }\n return { success: false, error: error.message };\n }\n return { success: false, error: \"Unknown error occurred\" };\n }\n }\n\n /**\n * Authenticate with the platform\n */\n async authenticate(): Promise<ApiResponse<AuthResponse>> {\n return this.request<AuthResponse>(\"/auth\", { method: \"POST\" });\n }\n\n /**\n * Sync commands to platform\n */\n async syncCommands(\n commands: Array<{\n timestamp: string;\n command: string;\n normalized: string;\n cwd: string;\n exitCode?: number | null;\n durationMs?: number | null;\n sessionId?: string;\n variables?: Record<string, string> | null;\n }>\n ): Promise<ApiResponse> {\n return this.request(\"/sync\", {\n method: \"POST\",\n body: JSON.stringify({\n type: \"commands\",\n data: commands,\n }),\n });\n }\n\n /**\n * Sync a pattern to platform\n */\n async syncPattern(pattern: {\n sourceType: string;\n name?: string;\n description?: string;\n commands: string[];\n category?: string;\n frequency?: number;\n score?: number;\n firstSeen?: string;\n lastSeen?: string;\n context?: Record<string, unknown>;\n }): Promise<ApiResponse<{ pattern: { id: string; name: string; status: string } }>> {\n return this.request(\"/sync\", {\n method: \"POST\",\n body: JSON.stringify({\n type: \"pattern\",\n data: pattern,\n }),\n });\n }\n\n /**\n * Create a session on the platform\n */\n async createSession(session: {\n startedAt: string;\n shell: string;\n endedAt?: string;\n commandCount?: number;\n }): Promise<ApiResponse<{ session: { id: string } }>> {\n return this.request(\"/sync\", {\n method: \"POST\",\n body: JSON.stringify({\n type: \"session\",\n data: session,\n }),\n });\n }\n\n /**\n * End a session on the platform\n */\n async endSession(sessionId: string): Promise<ApiResponse<{ success: boolean }>> {\n return this.request(\"/sessions\", {\n method: \"PATCH\",\n body: JSON.stringify({ sessionId }),\n });\n }\n\n /**\n * Start a new session on the platform (for standalone terminals)\n */\n async startSession(shell: string, projectPath?: string): Promise<ApiResponse<{ sessionId: string }>> {\n return this.request(\"/sessions\", {\n method: \"POST\",\n body: JSON.stringify({ shell, projectPath }),\n });\n }\n\n /**\n * Download skills and patterns from platform\n */\n async downloadData(type: \"all\" | \"skills\" | \"patterns\" = \"all\"): Promise<ApiResponse<SyncResponse>> {\n return this.request<SyncResponse>(`/sync?type=${type}`, { method: \"GET\" });\n }\n\n /**\n * Generate a skill from a pattern\n */\n async generateSkill(options: {\n patternId?: string;\n commands?: string[];\n name?: string;\n description?: string;\n category?: string;\n context?: Record<string, unknown>;\n }): Promise<ApiResponse<GenerateResponse>> {\n return this.request<GenerateResponse>(\"/generate\", {\n method: \"POST\",\n body: JSON.stringify(options),\n });\n }\n\n /**\n * Start device authorization flow (no auth required)\n */\n async startDeviceAuth(deviceName?: string): Promise<ApiResponse<DeviceAuthResponse>> {\n const url = `${this.baseUrl}/auth/device`;\n try {\n const response = await fetch(url, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({ device_name: deviceName }),\n });\n\n const data = await response.json();\n\n if (!response.ok) {\n return {\n success: false,\n error: data.error || `Request failed with status ${response.status}`,\n };\n }\n\n return { success: true, data };\n } catch (error) {\n if (error instanceof Error) {\n if (error.message.includes(\"ECONNREFUSED\")) {\n return {\n success: false,\n error: \"Cannot connect to Skillo platform. Is the server running?\",\n };\n }\n return { success: false, error: error.message };\n }\n return { success: false, error: \"Unknown error occurred\" };\n }\n }\n\n /**\n * Check token status (polling)\n */\n async checkTokenStatus(code: string): Promise<ApiResponse<TokenStatusResponse>> {\n const url = `${this.baseUrl}/token?code=${encodeURIComponent(code)}`;\n try {\n const response = await fetch(url, { method: \"GET\" });\n const data = await response.json();\n\n if (!response.ok) {\n return {\n success: false,\n error: data.error || `Request failed with status ${response.status}`,\n };\n }\n\n return { success: true, data };\n } catch (error) {\n if (error instanceof Error) {\n return { success: false, error: error.message };\n }\n return { success: false, error: \"Unknown error occurred\" };\n }\n }\n\n /**\n * Exchange code for API key\n */\n async exchangeToken(code: string, deviceName?: string): Promise<ApiResponse<TokenExchangeResponse>> {\n const url = `${this.baseUrl}/token`;\n try {\n const response = await fetch(url, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({ code, device_name: deviceName }),\n });\n\n const data = await response.json();\n\n if (!response.ok) {\n return {\n success: false,\n error: data.error || `Request failed with status ${response.status}`,\n };\n }\n\n return { success: true, data };\n } catch (error) {\n if (error instanceof Error) {\n return { success: false, error: error.message };\n }\n return { success: false, error: \"Unknown error occurred\" };\n }\n }\n\n /**\n * Sync Claude Code prompts to platform\n */\n async syncClaudePrompts(\n prompts: Array<{\n display: string;\n timestamp: number;\n project: string;\n sessionId: string;\n pastedContents?: Record<string, unknown>;\n }>,\n options?: {\n terminalSessionId?: string;\n projectPath?: string;\n }\n ): Promise<ApiResponse<{\n sessionsCreated: number;\n promptsCreated: number;\n promptsSkipped: number;\n }>> {\n return this.request(\"/claude\", {\n method: \"POST\",\n body: JSON.stringify({\n prompts,\n terminalSessionId: options?.terminalSessionId,\n projectPath: options?.projectPath,\n }),\n });\n }\n\n /**\n * Get Claude Code sessions from platform\n */\n async getClaudeSessions(limit = 20): Promise<ApiResponse<{\n sessions: Array<{\n id: string;\n claudeSessionId: string;\n projectPath: string;\n projectName: string;\n startedAt: string;\n endedAt?: string;\n promptCount: number;\n prompts?: Array<{\n id: string;\n prompt: string;\n timestamp: string;\n category?: string;\n }>;\n }>;\n }>> {\n return this.request(`/claude?limit=${limit}`, { method: \"GET\" });\n }\n\n /**\n * Report skill usage detections to platform\n */\n async reportSkillUsage(\n usages: Array<{\n skillSlug: string;\n claudeSessionId: string;\n projectPath: string;\n timestamp: string;\n }>\n ): Promise<ApiResponse<{ logged: number }>> {\n return this.request(\"/skill-usage\", {\n method: \"POST\",\n body: JSON.stringify({ usages }),\n });\n }\n\n // ============================================================================\n // PROJECT TRACKING\n // ============================================================================\n\n /**\n * Connect a project for tracking\n */\n async connectProject(params: {\n path: string;\n name?: string;\n gitRemote?: string;\n gitRemoteNormalized?: string;\n language?: string;\n framework?: string;\n }): Promise<ApiResponse<{\n id: string;\n name: string;\n path: string;\n trackingEnabled: boolean;\n connectedAt: string;\n }>> {\n return this.request(\"/projects/connect\", {\n method: \"POST\",\n body: JSON.stringify(params),\n });\n }\n\n /**\n * Disconnect a project from tracking\n */\n async disconnectProject(path: string): Promise<ApiResponse<{\n id: string;\n name: string;\n trackingEnabled: boolean;\n }>> {\n return this.request(`/projects/connect?path=${encodeURIComponent(path)}`, {\n method: \"DELETE\",\n });\n }\n\n /**\n * Get tracking status for a project\n */\n async getProjectStatus(path: string): Promise<ApiResponse<{\n connected: boolean;\n tracked: boolean;\n connectedAt?: string;\n project?: {\n id: string;\n name: string;\n path: string;\n gitRemote?: string;\n gitRemoteNormalized?: string;\n trackingEnabled: boolean;\n };\n }>> {\n return this.request(`/projects/connect?path=${encodeURIComponent(path)}`, {\n method: \"GET\",\n });\n }\n\n /**\n * List all tracked projects\n */\n async listProjects(includeDisabled = false): Promise<ApiResponse<{\n projects: Array<{\n id: string;\n name: string;\n path: string;\n gitRemote?: string;\n gitRemoteNormalized?: string;\n trackingEnabled: boolean;\n connectedAt?: string;\n }>;\n totalTracked: number;\n totalProjects: number;\n }>> {\n return this.request(`/projects/tracked?includeDisabled=${includeDisabled}`, {\n method: \"GET\",\n });\n }\n\n /**\n * Check if a path is in a tracked project\n * Returns the project if tracked, null if not\n */\n async isProjectTracked(path: string): Promise<{\n tracked: boolean;\n project?: {\n id: string;\n name: string;\n path: string;\n };\n }> {\n const result = await this.getProjectStatus(path);\n if (result.success && result.data?.connected) {\n return {\n tracked: true,\n project: result.data.project ? {\n id: result.data.project.id,\n name: result.data.project.name,\n path: result.data.project.path,\n } : undefined,\n };\n }\n return { tracked: false };\n }\n}\n\n// Singleton instance\nlet clientInstance: ApiClient | null = null;\n\nexport function getApiClient(): ApiClient {\n if (!clientInstance) {\n clientInstance = new ApiClient();\n }\n return clientInstance;\n}\n","/**\n * Claude Watcher - Reusable module for watching Claude Code history.\n *\n * Polls ~/.claude/history.jsonl for new entries and syncs them to the platform.\n * Used by both the daemon (background) and the `claude watch` command (foreground).\n */\n\nimport * as fs from \"fs\";\nimport * as path from \"path\";\nimport * as os from \"os\";\nimport * as readline from \"readline\";\n\n// Types\nexport interface ClaudeHistoryEntry {\n display: string;\n timestamp: number;\n project: string;\n sessionId: string;\n pastedContents?: Record<string, unknown>;\n}\n\ninterface TrackedProjectInfo {\n tracked: boolean;\n name?: string;\n}\n\nexport interface ClaudeWatcherCallbacks {\n onSync?: (promptsCreated: number) => void;\n onError?: (error: Error) => void;\n log?: (level: string, msg: string) => void;\n}\n\n// ── Path helpers ──\n\nfunction getClaudeHistoryPath(): string {\n return path.join(os.homedir(), \".claude\", \"history.jsonl\");\n}\n\nfunction normalizePath(p: string): string {\n return p.toLowerCase().replace(/\\\\/g, \"/\").replace(/\\/+$/, \"\");\n}\n\nfunction matchesProjectPath(claudeProjectPath: string, filterPath: string): boolean {\n const normalizedClaude = normalizePath(claudeProjectPath);\n const normalizedFilter = normalizePath(filterPath);\n return normalizedClaude === normalizedFilter ||\n normalizedClaude.startsWith(normalizedFilter + \"/\");\n}\n\n// ── Project tracking cache ──\n\nconst projectCache = {\n projects: new Map<string, TrackedProjectInfo>(),\n lastFetched: 0,\n};\n\nconst CACHE_TTL_MS = 60000;\n\nexport async function loadTrackedProjects(\n client: { listProjects: (includeDisabled: boolean) => Promise<{ success: boolean; data?: { projects: Array<{ path: string; name: string; trackingEnabled: boolean }> } }> }\n): Promise<Map<string, TrackedProjectInfo>> {\n const now = Date.now();\n\n if (projectCache.lastFetched > 0 && now - projectCache.lastFetched < CACHE_TTL_MS) {\n return projectCache.projects;\n }\n\n try {\n const result = await client.listProjects(true);\n if (result.success && result.data?.projects) {\n projectCache.projects.clear();\n for (const project of result.data.projects) {\n const normalizedPath = normalizePath(project.path);\n projectCache.projects.set(normalizedPath, {\n tracked: project.trackingEnabled,\n name: project.name,\n });\n }\n projectCache.lastFetched = now;\n }\n } catch {\n // Use stale cache on failure\n }\n\n return projectCache.projects;\n}\n\nexport function getTrackedProjectCount(): number {\n return Array.from(projectCache.projects.values()).filter(p => p.tracked).length;\n}\n\nasync function isProjectTracked(projectPath: string): Promise<TrackedProjectInfo> {\n const normalizedPath = normalizePath(projectPath);\n\n if (projectCache.projects.has(normalizedPath)) {\n return projectCache.projects.get(normalizedPath)!;\n }\n\n for (const [trackedPath, info] of projectCache.projects.entries()) {\n if (normalizedPath.startsWith(trackedPath + \"/\")) {\n return info;\n }\n }\n\n return { tracked: false };\n}\n\n// ── Watcher ──\n\nexport class ClaudeWatcher {\n private lastSize = 0;\n private lastMtime = 0;\n private intervalId: ReturnType<typeof setInterval> | null = null;\n private historyPath: string;\n private intervalMs: number;\n private projectFilter?: string;\n private sessionFilter?: string;\n private callbacks: ClaudeWatcherCallbacks;\n private client: {\n syncClaudePrompts: (prompts: ClaudeHistoryEntry[], options?: { terminalSessionId?: string; projectPath?: string }) => Promise<{ success: boolean; data?: { promptsCreated: number; sessionsCreated: number; promptsSkipped: number }; error?: string }>;\n listProjects: (includeDisabled: boolean) => Promise<{ success: boolean; data?: { projects: Array<{ path: string; name: string; trackingEnabled: boolean }> } }>;\n };\n\n constructor(\n client: ClaudeWatcher[\"client\"],\n options: {\n intervalMs?: number;\n projectFilter?: string;\n sessionFilter?: string;\n callbacks?: ClaudeWatcherCallbacks;\n } = {}\n ) {\n this.client = client;\n this.historyPath = getClaudeHistoryPath();\n this.intervalMs = options.intervalMs || 5000;\n this.projectFilter = options.projectFilter;\n this.sessionFilter = options.sessionFilter;\n this.callbacks = options.callbacks || {};\n }\n\n private log(level: string, msg: string) {\n if (this.callbacks.log) {\n this.callbacks.log(level, msg);\n }\n }\n\n /** Initialize file position to current end (don't sync old entries). */\n initPosition() {\n try {\n if (fs.existsSync(this.historyPath)) {\n const stats = fs.statSync(this.historyPath);\n this.lastSize = stats.size;\n this.lastMtime = stats.mtimeMs;\n }\n } catch {\n // File doesn't exist yet\n }\n }\n\n /** Start watching. Runs an initial check then polls on interval. */\n async start() {\n this.initPosition();\n\n // Load tracked projects\n await loadTrackedProjects(this.client);\n const trackedCount = getTrackedProjectCount();\n this.log(\"INFO\", `Watching ${trackedCount} tracked project(s)`);\n\n // Initial check\n await this.checkAndSync();\n\n // Poll\n this.intervalId = setInterval(() => this.checkAndSync(), this.intervalMs);\n }\n\n /** Stop watching. */\n stop() {\n if (this.intervalId) {\n clearInterval(this.intervalId);\n this.intervalId = null;\n }\n }\n\n /** Check for new entries and sync them. */\n async checkAndSync() {\n try {\n if (!fs.existsSync(this.historyPath)) return;\n\n const stats = fs.statSync(this.historyPath);\n\n // Skip if no changes\n if (stats.size === this.lastSize && stats.mtimeMs === this.lastMtime) {\n return;\n }\n\n // Refresh tracked projects cache\n await loadTrackedProjects(this.client);\n\n // Read new content\n const newPrompts: ClaudeHistoryEntry[] = [];\n\n if (stats.size > this.lastSize) {\n const stream = fs.createReadStream(this.historyPath, {\n start: this.lastSize,\n encoding: \"utf-8\",\n });\n\n const rl = readline.createInterface({\n input: stream,\n crlfDelay: Infinity,\n });\n\n for await (const line of rl) {\n if (!line.trim()) continue;\n\n try {\n const entry = JSON.parse(line) as ClaudeHistoryEntry;\n if (!entry.display || !entry.timestamp || !entry.project || !entry.sessionId) {\n continue;\n }\n\n // Project path filter\n if (this.projectFilter && !matchesProjectPath(entry.project, this.projectFilter)) {\n continue;\n }\n\n // Privacy: only tracked projects\n const trackingStatus = await isProjectTracked(entry.project);\n if (!trackingStatus.tracked) {\n continue;\n }\n\n newPrompts.push(entry);\n } catch {\n // Skip invalid JSON\n }\n }\n }\n\n this.lastSize = stats.size;\n this.lastMtime = stats.mtimeMs;\n\n if (newPrompts.length > 0) {\n this.log(\"INFO\", `Detected ${newPrompts.length} new prompt(s), syncing...`);\n\n const response = await this.client.syncClaudePrompts(newPrompts, {\n terminalSessionId: this.sessionFilter,\n projectPath: this.projectFilter,\n });\n\n if (response.success && response.data) {\n this.log(\"INFO\", `Synced ${response.data.promptsCreated} prompt(s)`);\n this.callbacks.onSync?.(response.data.promptsCreated);\n } else {\n this.log(\"ERROR\", `Sync failed: ${response.error}`);\n }\n }\n } catch (error) {\n const err = error instanceof Error ? error : new Error(String(error));\n this.log(\"ERROR\", `Watch error: ${err.message}`);\n this.callbacks.onError?.(err);\n }\n }\n}\n","/**\n * Skill Usage Detector — detects when Claude Code uses skills.\n *\n * Scans Claude Code session JSONL files for:\n * - Read tool calls to ~/.claude/skills/{slug}/SKILL.md\n * - Bash tool calls matching skill command arrays\n *\n * Tracks byte offsets per file to avoid re-processing.\n * Reports detections to the Skillo platform via batch API.\n */\n\nimport * as fs from \"fs\";\nimport * as path from \"path\";\nimport * as readline from \"readline\";\nimport { getSkillsDir, getClaudeDir, getDataDir, ensureDirectory } from \"../utils/paths.js\";\n\nexport interface SkillUsageEvent {\n skillSlug: string;\n claudeSessionId: string;\n projectPath: string;\n timestamp: string;\n}\n\ninterface OffsetState {\n [filePath: string]: {\n byteOffset: number;\n lastModified: number;\n };\n}\n\nexport interface SkillUsageDetectorCallbacks {\n onDetection?: (count: number) => void;\n onError?: (error: Error) => void;\n log?: (level: string, msg: string) => void;\n}\n\nconst MAX_NEW_BYTES_PER_FILE = 2 * 1024 * 1024; // 2MB cap per cycle\nconst OFFSETS_FILE = \"skill-usage-offsets.json\";\n\nexport class SkillUsageDetector {\n private intervalId: ReturnType<typeof setInterval> | null = null;\n private intervalMs: number;\n private callbacks: SkillUsageDetectorCallbacks;\n private client: {\n reportSkillUsage: (usages: SkillUsageEvent[]) => Promise<{\n success: boolean;\n data?: { logged: number };\n error?: string;\n }>;\n };\n\n constructor(\n client: SkillUsageDetector[\"client\"],\n options: {\n intervalMs?: number;\n callbacks?: SkillUsageDetectorCallbacks;\n } = {}\n ) {\n this.client = client;\n this.intervalMs = options.intervalMs || 30000;\n this.callbacks = options.callbacks || {};\n }\n\n private log(level: string, msg: string) {\n this.callbacks.log?.(level, msg);\n }\n\n async start() {\n // Initialize offsets to current file sizes\n this.initOffsets();\n this.log(\"INFO\", \"Skill usage detector started\");\n\n // Poll\n this.intervalId = setInterval(() => this.detect(), this.intervalMs);\n }\n\n stop() {\n if (this.intervalId) {\n clearInterval(this.intervalId);\n this.intervalId = null;\n }\n }\n\n /** Build slug inventory from ~/.claude/skills/ */\n private getDeployedSkillSlugs(): Set<string> {\n const slugs = new Set<string>();\n const skillsDir = getSkillsDir();\n\n if (!fs.existsSync(skillsDir)) return slugs;\n\n try {\n const entries = fs.readdirSync(skillsDir, { withFileTypes: true });\n for (const entry of entries) {\n if (!entry.isDirectory() || entry.name.startsWith(\".\")) continue;\n const skillFile = path.join(skillsDir, entry.name, \"SKILL.md\");\n if (fs.existsSync(skillFile)) {\n slugs.add(entry.name);\n }\n }\n } catch {\n // ignore\n }\n\n return slugs;\n }\n\n /** Encode project path to Claude's directory name format */\n private encodeProjectPath(projectPath: string): string {\n // /Users/foo/bar → -Users-foo-bar\n return projectPath.replace(/\\//g, \"-\");\n }\n\n /** Find session JSONL files for tracked projects */\n private getSessionFiles(): string[] {\n const claudeDir = getClaudeDir();\n const projectsDir = path.join(claudeDir, \"projects\");\n\n if (!fs.existsSync(projectsDir)) return [];\n\n const files: string[] = [];\n try {\n const projectDirs = fs.readdirSync(projectsDir, { withFileTypes: true });\n for (const dir of projectDirs) {\n if (!dir.isDirectory()) continue;\n const dirPath = path.join(projectsDir, dir.name);\n const jsonlFiles = fs.readdirSync(dirPath).filter(\n (f) => f.endsWith(\".jsonl\") && !f.startsWith(\"agent-\")\n );\n for (const f of jsonlFiles) {\n files.push(path.join(dirPath, f));\n }\n }\n } catch {\n // ignore\n }\n\n return files;\n }\n\n /** Load persisted offsets */\n private loadOffsets(): OffsetState {\n const offsetsPath = path.join(getDataDir(), OFFSETS_FILE);\n try {\n if (fs.existsSync(offsetsPath)) {\n return JSON.parse(fs.readFileSync(offsetsPath, \"utf-8\"));\n }\n } catch {\n // ignore\n }\n return {};\n }\n\n /** Save offsets */\n private saveOffsets(state: OffsetState) {\n ensureDirectory(getDataDir());\n const offsetsPath = path.join(getDataDir(), OFFSETS_FILE);\n try {\n fs.writeFileSync(offsetsPath, JSON.stringify(state), \"utf-8\");\n } catch {\n // ignore\n }\n }\n\n /** Initialize offsets to current file sizes (skip existing data) */\n private initOffsets() {\n const files = this.getSessionFiles();\n const state = this.loadOffsets();\n\n for (const file of files) {\n if (!state[file]) {\n try {\n const stats = fs.statSync(file);\n state[file] = { byteOffset: stats.size, lastModified: stats.mtimeMs };\n } catch {\n // ignore\n }\n }\n }\n\n this.saveOffsets(state);\n }\n\n /** Main detection cycle */\n async detect() {\n try {\n const skillSlugs = this.getDeployedSkillSlugs();\n if (skillSlugs.size === 0) return; // No skills deployed\n\n const skillsDir = getSkillsDir();\n const files = this.getSessionFiles();\n const offsets = this.loadOffsets();\n const events: SkillUsageEvent[] = [];\n\n for (const file of files) {\n try {\n const stats = fs.statSync(file);\n const currentOffset = offsets[file]?.byteOffset ?? 0;\n const currentMtime = offsets[file]?.lastModified ?? 0;\n\n // Skip unchanged files\n if (stats.size <= currentOffset && stats.mtimeMs === currentMtime) continue;\n\n // Cap how much we read per cycle\n const startOffset = Math.max(currentOffset, stats.size - MAX_NEW_BYTES_PER_FILE);\n\n if (stats.size <= startOffset) {\n offsets[file] = { byteOffset: stats.size, lastModified: stats.mtimeMs };\n continue;\n }\n\n // Read new content\n const stream = fs.createReadStream(file, {\n start: startOffset,\n encoding: \"utf-8\",\n });\n\n const rl = readline.createInterface({ input: stream, crlfDelay: Infinity });\n\n for await (const line of rl) {\n if (!line.trim()) continue;\n\n try {\n const entry = JSON.parse(line);\n const msg = entry?.message;\n if (!msg || msg.role !== \"assistant\") continue;\n\n const content = msg.content;\n if (!Array.isArray(content)) continue;\n\n const sessionId = entry.sessionId || \"\";\n const timestamp = entry.timestamp || new Date().toISOString();\n const cwd = entry.cwd || \"\";\n\n for (const block of content) {\n if (block?.type !== \"tool_use\") continue;\n\n const toolName = block.name;\n const toolInput = block.input || {};\n\n // Signal 1: Read calls to skill files\n if (toolName === \"Read\") {\n const filePath = toolInput.file_path || \"\";\n // Match paths like ~/.claude/skills/{slug}/SKILL.md or full path\n if (filePath.includes(\"/skills/\") && filePath.endsWith(\"/SKILL.md\")) {\n // Extract slug from path\n const parts = filePath.split(\"/skills/\");\n if (parts.length >= 2) {\n const afterSkills = parts[parts.length - 1]; // \"my-skill/SKILL.md\"\n const slug = afterSkills.split(\"/\")[0];\n if (slug && skillSlugs.has(slug)) {\n events.push({\n skillSlug: slug,\n claudeSessionId: sessionId,\n projectPath: cwd,\n timestamp,\n });\n }\n }\n }\n }\n\n // Signal 2: Bash commands matching skill commands\n // (skipped for v1 — Read detection is sufficient)\n }\n } catch {\n // Skip invalid JSON lines\n }\n }\n\n offsets[file] = { byteOffset: stats.size, lastModified: stats.mtimeMs };\n } catch {\n // Skip inaccessible files\n }\n }\n\n // Clean up offsets for deleted files\n for (const key of Object.keys(offsets)) {\n if (!fs.existsSync(key)) {\n delete offsets[key];\n }\n }\n\n this.saveOffsets(offsets);\n\n // Deduplicate: same slug+session within batch\n const seen = new Set<string>();\n const uniqueEvents = events.filter((e) => {\n const key = `${e.skillSlug}:${e.claudeSessionId}:${e.timestamp}`;\n if (seen.has(key)) return false;\n seen.add(key);\n return true;\n });\n\n // Report to server\n if (uniqueEvents.length > 0) {\n this.log(\"INFO\", `Detected ${uniqueEvents.length} skill usage(s), reporting...`);\n\n const response = await this.client.reportSkillUsage(uniqueEvents);\n if (response.success) {\n this.log(\"INFO\", `Reported ${response.data?.logged ?? 0} skill usage(s)`);\n this.callbacks.onDetection?.(response.data?.logged ?? 0);\n } else {\n this.log(\"ERROR\", `Report failed: ${response.error}`);\n }\n }\n } catch (error) {\n const err = error instanceof Error ? error : new Error(String(error));\n this.log(\"ERROR\", `Detection error: ${err.message}`);\n this.callbacks.onError?.(err);\n }\n }\n}\n","/**\n * StatusWriter — writes daemon status JSON for tray icon and diagnostics.\n *\n * Writes ~/.skillo/daemon-status.json every 10s and on key events.\n * Tray icon and `skillo status` read this file.\n */\n\nimport { writeFileSync, readFileSync, existsSync } from \"fs\";\nimport { join } from \"path\";\nimport { getDataDir, ensureDirectory } from \"./paths.js\";\n\nexport interface DaemonStatus {\n running: boolean;\n pid: number | null;\n startedAt: string | null;\n updatedAt: string;\n user?: string;\n claudeWatcher?: {\n lastSync?: string | null;\n promptsSynced?: number;\n lastError?: string | null;\n };\n skillDetector?: {\n deployedSkills?: number;\n usagesDetected?: number;\n lastDetection?: string | null;\n lastError?: string | null;\n };\n trackedProjects?: Array<{ name: string; path: string }>;\n activeSessions?: number;\n}\n\nconst STATUS_FILE = \"daemon-status.json\";\nconst WRITE_INTERVAL_MS = 10000;\n\nexport class StatusWriter {\n private intervalId: ReturnType<typeof setInterval> | null = null;\n private status: DaemonStatus;\n private filePath: string;\n\n constructor() {\n this.filePath = join(getDataDir(), STATUS_FILE);\n this.status = {\n running: true,\n pid: process.pid,\n startedAt: new Date().toISOString(),\n updatedAt: new Date().toISOString(),\n claudeWatcher: { lastSync: null, promptsSynced: 0, lastError: null },\n skillDetector: { deployedSkills: 0, usagesDetected: 0, lastDetection: null, lastError: null },\n trackedProjects: [],\n activeSessions: 0,\n };\n }\n\n /** Start periodic writes */\n start() {\n ensureDirectory(getDataDir());\n this.write();\n this.intervalId = setInterval(() => this.write(), WRITE_INTERVAL_MS);\n }\n\n /** Stop writing and mark as not running */\n stop() {\n if (this.intervalId) {\n clearInterval(this.intervalId);\n this.intervalId = null;\n }\n this.status.running = false;\n this.status.pid = null;\n this.status.updatedAt = new Date().toISOString();\n this.write();\n }\n\n /** Merge partial status updates */\n update(partial: Partial<DaemonStatus>) {\n // Deep merge for nested objects\n if (partial.claudeWatcher) {\n this.status.claudeWatcher = { ...this.status.claudeWatcher, ...partial.claudeWatcher };\n delete partial.claudeWatcher;\n }\n if (partial.skillDetector) {\n this.status.skillDetector = { ...this.status.skillDetector, ...partial.skillDetector };\n delete partial.skillDetector;\n }\n Object.assign(this.status, partial);\n }\n\n /** Get the status file path */\n static getStatusFilePath(): string {\n return join(getDataDir(), STATUS_FILE);\n }\n\n /** Read current status from disk (static, for tray/status commands) */\n static read(): DaemonStatus | null {\n const filePath = StatusWriter.getStatusFilePath();\n try {\n if (existsSync(filePath)) {\n return JSON.parse(readFileSync(filePath, \"utf-8\"));\n }\n } catch {\n // ignore\n }\n return null;\n }\n\n private write() {\n this.status.updatedAt = new Date().toISOString();\n try {\n writeFileSync(this.filePath, JSON.stringify(this.status, null, 2), \"utf-8\");\n } catch {\n // Can't write status, ignore\n }\n }\n}\n"],"mappings":"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","names":["existsSync","writeFileSync","readFileSync","readdirSync","existsSync","path","path","existsSync","path","fs","path","readline","writeFileSync","readFileSync","existsSync","join","join","existsSync","readFileSync","writeFileSync","writeFileSync","existsSync","readdirSync","readFileSync"]}
@@ -90,7 +90,7 @@ var StatusWriter = class _StatusWriter {
90
90
  // src/tray/tray.ts
91
91
  var SysTray = SysTrayModule.default || SysTrayModule;
92
92
  var SEPARATOR = { title: "<SEPARATOR>", tooltip: "", enabled: true };
93
- var ICON_BASE64 = "iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAAhklEQVQ4T2NkoBAwUqifgWoGMDIyNjAyMP5nYGD4T8gFjHADGBkZAxgZGQMIGQK3gYmJqYCRkfE/IyPj/4KCgv9whzAyMDIwMDL+Z2Bk/M/AwMjwn4GBsaCgoOA/uhehuwBuACMjI9wQBgYGBkZGxv8FBQUBcBfgC0e4C4gJR7gBxIQj1QwAAFbxMBHleBQjAAAAAElFTkSuQmCC";
93
+ var ICON_BASE64 = "iVBORw0KGgoAAAANSUhEUgAAABYAAAAWCAYAAADEtGw7AAAAAXNSR0IArs4c6QAAADhlWElmTU0AKgAAAAgAAYdpAAQAAAABAAAAGgAAAAAAAqACAAQAAAABAAAAFqADAAQAAAABAAAAFgAAAAA/6RFgAAABm2lUWHRYTUw6Y29tLmFkb2JlLnhtcAAAAAAAPHg6eG1wbWV0YSB4bWxuczp4PSJhZG9iZTpuczptZXRhLyIgeDp4bXB0az0iWE1QIENvcmUgNi4wLjAiPgogICA8cmRmOlJERiB4bWxuczpyZGY9Imh0dHA6Ly93d3cudzMub3JnLzE5OTkvMDIvMjItcmRmLXN5bnRheC1ucyMiPgogICAgICA8cmRmOkRlc2NyaXB0aW9uIHJkZjphYm91dD0iIgogICAgICAgICAgICB4bWxuczpleGlmPSJodHRwOi8vbnMuYWRvYmUuY29tL2V4aWYvMS4wLyI+CiAgICAgICAgIDxleGlmOlBpeGVsWERpbWVuc2lvbj4yMjwvZXhpZjpQaXhlbFhEaW1lbnNpb24+CiAgICAgICAgIDxleGlmOlBpeGVsWURpbWVuc2lvbj4yMjwvZXhpZjpQaXhlbFlEaW1lbnNpb24+CiAgICAgIDwvcmRmOkRlc2NyaXB0aW9uPgogICA8L3JkZjpSREY+CjwveDp4bXBtZXRhPgolQXt+AAABg0lEQVQ4EaWUPyhFURzHH4/UK38KA5OBlGKQUWQRmQ1vMSmb2WKwSqKkUBaLUhabTfmTjMhiV0qKouTf59tz6nTeuefc637r07nn/H7f7zvvnntvsZBPndiXoQHu8kVV3G0MK/AGPzANudSMewleQIGG1MHDmFy1snAMJsweq4JrXTfzQZjwrD+xNg79sAufkFo1dF7BLZQiLv2AerXzqh273sm/RjVfQK/b4MzrmW9CNHiNJvvefTDX6TdCSE2homrbYAeb6wfWZ0C36l8q4zJhvvGcug43s/SUHIIv1Kx9Ud+AFsikIt0L8A4mzDc+Up8D9WdSF90H4Au11/TI9UFmjeG4BjvMvX6l3pM5GUMdzMMzuKFmvkUtUbETb8e5AzpAE2jGk8RUCmewDnqjQhqieAkmVONeyKBgNd3ASKiRmp6GRfgGeaYgUXoB7F0cMR9N7K4UVhlO3R4dSkjdFDtAL4525pO+hMHbIJPZ8T3X+jakefgHZIxpn4ZZiP2TWE7hF4oBfI7ofv98AAAAAElFTkSuQmCC";
94
94
  var TRAY_PID_FILE = "tray.pid";
95
95
  var STATUS_POLL_MS = 3e3;
96
96
  function formatTimeAgo(isoString) {
@@ -405,4 +405,4 @@ async function startTray() {
405
405
  export {
406
406
  startTray
407
407
  };
408
- //# sourceMappingURL=tray-YOL4R2RH.js.map
408
+ //# sourceMappingURL=tray-UCAI2U2C.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/tray/tray.ts","../src/utils/status-writer.ts"],"sourcesContent":["/**\n * Skillo System Tray Icon\n *\n * Shows daemon status, tracked projects, sync info in the system tray.\n * Reads ~/.skillo/daemon-status.json (written by StatusWriter in daemon).\n *\n * macOS: native Swift helper (scripts/tray-helper-darwin) — arm64+x86_64.\n * Linux/Windows: systray2 (Go-based binary).\n */\n\nimport SysTrayModule from \"systray2\";\nimport { readFileSync, existsSync, writeFileSync, unlinkSync, chmodSync, readdirSync } from \"fs\";\nimport { exec, execSync, spawn } from \"child_process\";\nimport { join, dirname } from \"path\";\nimport { platform, homedir } from \"os\";\nimport { fileURLToPath } from \"url\";\nimport { StatusWriter, type DaemonStatus } from \"../utils/status-writer.js\";\nimport { getDataDir, ensureDirectory } from \"../utils/paths.js\";\n\n// Handle CJS/ESM interop — systray2 is CJS, so default import may be wrapped\nconst SysTray = (SysTrayModule as any).default || SysTrayModule;\n\ntype MenuItem = {\n title: string;\n tooltip: string;\n checked?: boolean;\n enabled?: boolean;\n hidden?: boolean;\n items?: MenuItem[];\n};\n\ntype Menu = {\n icon: string;\n title: string;\n tooltip: string;\n items: MenuItem[];\n};\n\nconst SEPARATOR: MenuItem = { title: \"<SEPARATOR>\", tooltip: \"\", enabled: true };\n\n// Skillo logo as 22x22 black template PNG (auto-inverts for dark mode on macOS)\nconst ICON_BASE64 =\n \"iVBORw0KGgoAAAANSUhEUgAAABYAAAAWCAYAAADEtGw7AAAAAXNSR0IArs4c6QAAADhlWElm\" +\n \"TU0AKgAAAAgAAYdpAAQAAAABAAAAGgAAAAAAAqACAAQAAAABAAAAFqADAAQAAAABAAAAFgAAAAA/\" +\n \"6RFgAAABm2lUWHRYTUw6Y29tLmFkb2JlLnhtcAAAAAAAPHg6eG1wbWV0YSB4bWxuczp4PSJh\" +\n \"ZG9iZTpuczptZXRhLyIgeDp4bXB0az0iWE1QIENvcmUgNi4wLjAiPgogICA8cmRmOlJERiB4\" +\n \"bWxuczpyZGY9Imh0dHA6Ly93d3cudzMub3JnLzE5OTkvMDIvMjItcmRmLXN5bnRheC1ucyMi\" +\n \"PgogICAgICA8cmRmOkRlc2NyaXB0aW9uIHJkZjphYm91dD0iIgogICAgICAgICAgICB4bWxu\" +\n \"czpleGlmPSJodHRwOi8vbnMuYWRvYmUuY29tL2V4aWYvMS4wLyI+CiAgICAgICAgIDxleGlm\" +\n \"OlBpeGVsWERpbWVuc2lvbj4yMjwvZXhpZjpQaXhlbFhEaW1lbnNpb24+CiAgICAgICAgIDxl\" +\n \"eGlmOlBpeGVsWURpbWVuc2lvbj4yMjwvZXhpZjpQaXhlbFlEaW1lbnNpb24+CiAgICAgIDwv\" +\n \"cmRmOkRlc2NyaXB0aW9uPgogICA8L3JkZjpSREY+CjwveDp4bXBtZXRhPgolQXt+AAABg0lE\" +\n \"QVQ4EaWUPyhFURzHH4/UK38KA5OBlGKQUWQRmQ1vMSmb2WKwSqKkUBaLUhabTfmTjMhiV0qK\" +\n \"ouTf59tz6nTeuefc637r07nn/H7f7zvvnntvsZBPndiXoQHu8kVV3G0MK/AGPzANudSMewle\" +\n \"QIGG1MHDmFy1snAMJsweq4JrXTfzQZjwrD+xNg79sAufkFo1dF7BLZQiLv2AerXzqh273sm/\" +\n \"RjVfQK/b4MzrmW9CNHiNJvvefTDX6TdCSE2homrbYAeb6wfWZ0C36l8q4zJhvvGcug43s/SU\" +\n \"HIIv1Kx9Ud+AFsikIt0L8A4mzDc+Up8D9WdSF90H4Au11/TI9UFmjeG4BjvMvX6l3pM5GUMd\" +\n \"zMMzuKFmvkUtUbETb8e5AzpAE2jGk8RUCmewDnqjQhqieAkmVONeyKBgNd3ASKiRmp6GRfgG\" +\n \"eaYgUXoB7F0cMR9N7K4UVhlO3R4dSkjdFDtAL4525pO+hMHbIJPZ8T3X+jakefgHZIxpn4ZZ\" +\n \"iP2TWE7hF4oBfI7ofv98AAAAAElFTkSuQmCC\";\n\nconst TRAY_PID_FILE = \"tray.pid\";\nconst STATUS_POLL_MS = 3000;\n\nfunction formatTimeAgo(isoString: string | null | undefined): string {\n if (!isoString) return \"Never\";\n const diff = Date.now() - new Date(isoString).getTime();\n const seconds = Math.floor(diff / 1000);\n if (seconds < 60) return \"Just now\";\n const minutes = Math.floor(seconds / 60);\n if (minutes < 60) return `${minutes} min ago`;\n const hours = Math.floor(minutes / 60);\n if (hours < 24) return `${hours}h ago`;\n const days = Math.floor(hours / 24);\n return `${days}d ago`;\n}\n\nfunction readStatus(): DaemonStatus | null {\n return StatusWriter.read();\n}\n\nfunction buildMenu(status: DaemonStatus | null): Menu {\n const running = status?.running ?? false;\n const statusIcon = running ? \"\\u25CF\" : \"\\u25CB\"; // ● or ○\n const statusText = running ? \"Running\" : \"Stopped\";\n\n const projects = status?.trackedProjects ?? [];\n const projectItems: MenuItem[] = projects.length > 0\n ? projects.map((p) => ({\n title: ` ${p.name}`,\n tooltip: p.path,\n enabled: false,\n checked: false,\n }))\n : [{\n title: \" (none)\",\n tooltip: \"No tracked projects\",\n enabled: false,\n checked: false,\n }];\n\n const items: MenuItem[] = [\n {\n title: `${statusIcon} Skillo \\u2014 ${statusText}`,\n tooltip: \"Daemon status\",\n enabled: false,\n checked: false,\n },\n SEPARATOR,\n {\n title: status?.user ? `Logged in as: ${status.user}` : \"Not logged in\",\n tooltip: \"User info\",\n enabled: false,\n checked: false,\n },\n SEPARATOR,\n {\n title: `Tracked Projects (${projects.length})`,\n tooltip: \"Projects being tracked\",\n enabled: false,\n checked: false,\n },\n ...projectItems,\n SEPARATOR,\n {\n title: `Last sync: ${formatTimeAgo(status?.claudeWatcher?.lastSync)}`,\n tooltip: \"Last Claude conversation sync\",\n enabled: false,\n checked: false,\n },\n {\n title: `Prompts synced: ${status?.claudeWatcher?.promptsSynced ?? 0}`,\n tooltip: \"Total prompts synced\",\n enabled: false,\n checked: false,\n },\n {\n title: `Skills detected: ${status?.skillDetector?.usagesDetected ?? 0}`,\n tooltip: \"Skill usages detected\",\n enabled: false,\n checked: false,\n },\n SEPARATOR,\n {\n title: running ? \"Stop Daemon\" : \"Start Daemon\",\n tooltip: running ? \"Stop the Skillo daemon\" : \"Start the Skillo daemon\",\n enabled: true,\n checked: false,\n },\n {\n title: \"Open Dashboard\",\n tooltip: \"Open Skillo dashboard in browser\",\n enabled: true,\n checked: false,\n },\n SEPARATOR,\n {\n title: \"Quit Tray\",\n tooltip: \"Hide tray icon (daemon keeps running)\",\n enabled: true,\n checked: false,\n },\n ];\n\n return {\n icon: ICON_BASE64,\n title: \"\",\n tooltip: `Skillo \\u2014 ${statusText}`,\n items,\n };\n}\n\nfunction openBrowser(url: string) {\n const os = platform();\n let cmd: string;\n if (os === \"darwin\") {\n cmd = `open \"${url}\"`;\n } else if (os === \"win32\") {\n cmd = `start \"\" \"${url}\"`;\n } else {\n cmd = `xdg-open \"${url}\"`;\n }\n exec(cmd, () => {});\n}\n\nfunction execSkillo(args: string) {\n exec(`skillo ${args}`, () => {});\n}\n\nfunction handleClick(title: string, cleanup: () => void) {\n if (title === \"Stop Daemon\" || title === \"Start Daemon\") {\n execSkillo(title === \"Stop Daemon\" ? \"stop\" : \"start\");\n } else if (title === \"Open Dashboard\") {\n openBrowser(\"https://www.skillo.one/dashboard\");\n } else if (title === \"Quit Tray\") {\n cleanup();\n }\n}\n\n// ── macOS native tray (Swift helper) ─────────────────────────────────────────\n\nfunction findDarwinHelper(): string | null {\n // Look for the compiled helper next to the running script, or in scripts/\n const candidates = [\n join(getDataDir(), \"tray-helper-darwin\"),\n join(dirname(fileURLToPath(import.meta.url)), \"..\", \"scripts\", \"tray-helper-darwin\"),\n join(dirname(fileURLToPath(import.meta.url)), \"scripts\", \"tray-helper-darwin\"),\n ];\n for (const p of candidates) {\n if (existsSync(p)) return p;\n }\n return null;\n}\n\nfunction compileDarwinHelper(): string | null {\n // Compile from source if Swift is available\n const srcCandidates = [\n join(dirname(fileURLToPath(import.meta.url)), \"..\", \"scripts\", \"tray-helper-darwin.swift\"),\n join(dirname(fileURLToPath(import.meta.url)), \"scripts\", \"tray-helper-darwin.swift\"),\n ];\n let src: string | null = null;\n for (const p of srcCandidates) {\n if (existsSync(p)) { src = p; break; }\n }\n if (!src) return null;\n\n const out = join(getDataDir(), \"tray-helper-darwin\");\n try {\n execSync(`swiftc \"${src}\" -o \"${out}\" -O 2>/dev/null`, { timeout: 60000 });\n chmodSync(out, 0o755);\n return out;\n } catch {\n return null;\n }\n}\n\nasync function startDarwinTray(pidFile: string): Promise<void> {\n let helperPath = findDarwinHelper();\n if (!helperPath) {\n helperPath = compileDarwinHelper();\n }\n if (!helperPath) {\n // Fall back to systray2\n await startSystray2Tray(pidFile);\n return;\n }\n\n const initialStatus = readStatus();\n const menu = buildMenu(initialStatus);\n\n const helper = spawn(helperPath, [], {\n stdio: [\"pipe\", \"pipe\", \"ignore\"],\n });\n\n const cleanup = () => {\n try { unlinkSync(pidFile); } catch { /* ignore */ }\n helper.kill();\n process.exit(0);\n };\n process.on(\"SIGINT\", cleanup);\n process.on(\"SIGTERM\", cleanup);\n helper.on(\"exit\", () => {\n try { unlinkSync(pidFile); } catch { /* ignore */ }\n process.exit(0);\n });\n\n // Wait for ready, then send menu\n await new Promise<void>((resolve, reject) => {\n let buf = \"\";\n const onData = (data: Buffer) => {\n buf += data.toString();\n const lines = buf.split(\"\\n\");\n buf = lines.pop() || \"\";\n for (const line of lines) {\n if (!line.trim()) continue;\n try {\n const msg = JSON.parse(line);\n if (msg.type === \"ready\") {\n helper.stdout!.removeListener(\"data\", onData);\n resolve();\n }\n } catch { /* ignore parse errors */ }\n }\n };\n helper.stdout!.on(\"data\", onData);\n setTimeout(() => reject(new Error(\"Tray helper did not become ready\")), 10000);\n });\n\n // Send initial menu\n helper.stdin!.write(JSON.stringify(menu) + \"\\n\");\n\n // Listen for clicks\n let clickBuf = \"\";\n helper.stdout!.on(\"data\", (data: Buffer) => {\n clickBuf += data.toString();\n const lines = clickBuf.split(\"\\n\");\n clickBuf = lines.pop() || \"\";\n for (const line of lines) {\n if (!line.trim()) continue;\n try {\n const msg = JSON.parse(line);\n if (msg.type === \"clicked\" && msg.item) {\n handleClick(msg.item.title, cleanup);\n }\n } catch { /* ignore */ }\n }\n });\n\n // Poll and update\n setInterval(() => {\n try {\n const status = readStatus();\n const updatedMenu = buildMenu(status);\n const action = { type: \"update-menu\", menu: updatedMenu };\n helper.stdin!.write(JSON.stringify(action) + \"\\n\");\n } catch { /* ignore */ }\n }, STATUS_POLL_MS);\n\n await new Promise(() => {});\n}\n\n// ── systray2 tray (Linux/Windows) ────────────────────────────────────────────\n\nfunction fixSystrayBinaryPermissions() {\n if (platform() === \"win32\") return;\n try {\n const cacheDir = join(homedir(), \".cache\", \"node-systray\");\n if (!existsSync(cacheDir)) return;\n for (const ver of readdirSync(cacheDir)) {\n const dir = join(cacheDir, ver);\n for (const file of readdirSync(dir)) {\n if (file.startsWith(\"tray_\")) {\n chmodSync(join(dir, file), 0o755);\n }\n }\n }\n } catch { /* best-effort */ }\n}\n\nasync function startSystray2Tray(pidFile: string): Promise<void> {\n fixSystrayBinaryPermissions();\n\n const initialStatus = readStatus();\n const menu = buildMenu(initialStatus);\n\n const systray = new SysTray({\n menu,\n debug: false,\n copyDir: true,\n });\n\n const cleanup = () => {\n try { unlinkSync(pidFile); } catch { /* ignore */ }\n systray.kill(false);\n process.exit(0);\n };\n process.on(\"SIGINT\", cleanup);\n process.on(\"SIGTERM\", cleanup);\n\n await systray.ready();\n\n systray.onClick(async (action: { item: MenuItem }) => {\n handleClick(action.item.title, cleanup);\n });\n\n setInterval(async () => {\n try {\n const status = readStatus();\n const updatedMenu = buildMenu(status);\n await systray.sendAction({ type: \"update-menu\", menu: updatedMenu });\n } catch { /* ignore */ }\n }, STATUS_POLL_MS);\n\n await new Promise(() => {});\n}\n\n// ── Entry point ──────────────────────────────────────────────────────────────\n\nexport async function startTray(): Promise<void> {\n const pidFile = join(getDataDir(), TRAY_PID_FILE);\n ensureDirectory(getDataDir());\n\n // Check if another tray is already running\n if (existsSync(pidFile)) {\n try {\n const pid = parseInt(readFileSync(pidFile, \"utf-8\").trim(), 10);\n process.kill(pid, 0);\n console.error(\"Tray icon is already running (PID: \" + pid + \")\");\n process.exit(1);\n } catch {\n // Stale PID file, continue\n }\n }\n\n writeFileSync(pidFile, String(process.pid));\n\n if (platform() === \"darwin\") {\n await startDarwinTray(pidFile);\n } else {\n await startSystray2Tray(pidFile);\n }\n}\n","/**\n * StatusWriter — writes daemon status JSON for tray icon and diagnostics.\n *\n * Writes ~/.skillo/daemon-status.json every 10s and on key events.\n * Tray icon and `skillo status` read this file.\n */\n\nimport { writeFileSync, readFileSync, existsSync } from \"fs\";\nimport { join } from \"path\";\nimport { getDataDir, ensureDirectory } from \"./paths.js\";\n\nexport interface DaemonStatus {\n running: boolean;\n pid: number | null;\n startedAt: string | null;\n updatedAt: string;\n user?: string;\n claudeWatcher?: {\n lastSync?: string | null;\n promptsSynced?: number;\n lastError?: string | null;\n };\n skillDetector?: {\n deployedSkills?: number;\n usagesDetected?: number;\n lastDetection?: string | null;\n lastError?: string | null;\n };\n trackedProjects?: Array<{ name: string; path: string }>;\n activeSessions?: number;\n}\n\nconst STATUS_FILE = \"daemon-status.json\";\nconst WRITE_INTERVAL_MS = 10000;\n\nexport class StatusWriter {\n private intervalId: ReturnType<typeof setInterval> | null = null;\n private status: DaemonStatus;\n private filePath: string;\n\n constructor() {\n this.filePath = join(getDataDir(), STATUS_FILE);\n this.status = {\n running: true,\n pid: process.pid,\n startedAt: new Date().toISOString(),\n updatedAt: new Date().toISOString(),\n claudeWatcher: { lastSync: null, promptsSynced: 0, lastError: null },\n skillDetector: { deployedSkills: 0, usagesDetected: 0, lastDetection: null, lastError: null },\n trackedProjects: [],\n activeSessions: 0,\n };\n }\n\n /** Start periodic writes */\n start() {\n ensureDirectory(getDataDir());\n this.write();\n this.intervalId = setInterval(() => this.write(), WRITE_INTERVAL_MS);\n }\n\n /** Stop writing and mark as not running */\n stop() {\n if (this.intervalId) {\n clearInterval(this.intervalId);\n this.intervalId = null;\n }\n this.status.running = false;\n this.status.pid = null;\n this.status.updatedAt = new Date().toISOString();\n this.write();\n }\n\n /** Merge partial status updates */\n update(partial: Partial<DaemonStatus>) {\n // Deep merge for nested objects\n if (partial.claudeWatcher) {\n this.status.claudeWatcher = { ...this.status.claudeWatcher, ...partial.claudeWatcher };\n delete partial.claudeWatcher;\n }\n if (partial.skillDetector) {\n this.status.skillDetector = { ...this.status.skillDetector, ...partial.skillDetector };\n delete partial.skillDetector;\n }\n Object.assign(this.status, partial);\n }\n\n /** Get the status file path */\n static getStatusFilePath(): string {\n return join(getDataDir(), STATUS_FILE);\n }\n\n /** Read current status from disk (static, for tray/status commands) */\n static read(): DaemonStatus | null {\n const filePath = StatusWriter.getStatusFilePath();\n try {\n if (existsSync(filePath)) {\n return JSON.parse(readFileSync(filePath, \"utf-8\"));\n }\n } catch {\n // ignore\n }\n return null;\n }\n\n private write() {\n this.status.updatedAt = new Date().toISOString();\n try {\n writeFileSync(this.filePath, JSON.stringify(this.status, null, 2), \"utf-8\");\n } catch {\n // Can't write status, ignore\n }\n }\n}\n"],"mappings":"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","names":["readFileSync","existsSync","writeFileSync","join","join","existsSync","readFileSync","writeFileSync"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "skillo",
3
- "version": "0.2.5",
3
+ "version": "0.2.6",
4
4
  "description": "Autonomous workflow learning & skill generation system - Learn workflows by observation, not explanation",
5
5
  "keywords": [
6
6
  "cli",
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../src/tray/tray.ts","../src/utils/status-writer.ts"],"sourcesContent":["/**\n * Skillo System Tray Icon\n *\n * Shows daemon status, tracked projects, sync info in the system tray.\n * Reads ~/.skillo/daemon-status.json (written by StatusWriter in daemon).\n *\n * macOS: native Swift helper (scripts/tray-helper-darwin) — arm64+x86_64.\n * Linux/Windows: systray2 (Go-based binary).\n */\n\nimport SysTrayModule from \"systray2\";\nimport { readFileSync, existsSync, writeFileSync, unlinkSync, chmodSync, readdirSync } from \"fs\";\nimport { exec, execSync, spawn } from \"child_process\";\nimport { join, dirname } from \"path\";\nimport { platform, homedir } from \"os\";\nimport { fileURLToPath } from \"url\";\nimport { StatusWriter, type DaemonStatus } from \"../utils/status-writer.js\";\nimport { getDataDir, ensureDirectory } from \"../utils/paths.js\";\n\n// Handle CJS/ESM interop — systray2 is CJS, so default import may be wrapped\nconst SysTray = (SysTrayModule as any).default || SysTrayModule;\n\ntype MenuItem = {\n title: string;\n tooltip: string;\n checked?: boolean;\n enabled?: boolean;\n hidden?: boolean;\n items?: MenuItem[];\n};\n\ntype Menu = {\n icon: string;\n title: string;\n tooltip: string;\n items: MenuItem[];\n};\n\nconst SEPARATOR: MenuItem = { title: \"<SEPARATOR>\", tooltip: \"\", enabled: true };\n\n// Minimal 16x16 template icon (black circle on transparent, suitable for macOS menu bar)\nconst ICON_BASE64 =\n \"iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAAhklEQVQ4T2NkoBAwUqifgWoG\" +\n \"MDIyNjAyMP5nYGD4T8gFjHADGBkZAxgZGQMIGQK3gYmJqYCRkfE/IyPj/4KCgv9whzAyMDIw\" +\n \"MDL+Z2Bk/M/AwMjwn4GBsaCgoOA/uhehuwBuACMjI9wQBgYGBkZGxv8FBQUBcBfgC0e4C4gJ\" +\n \"R7gBxIQj1QwAAFbxMBHleBQjAAAAAElFTkSuQmCC\";\n\nconst TRAY_PID_FILE = \"tray.pid\";\nconst STATUS_POLL_MS = 3000;\n\nfunction formatTimeAgo(isoString: string | null | undefined): string {\n if (!isoString) return \"Never\";\n const diff = Date.now() - new Date(isoString).getTime();\n const seconds = Math.floor(diff / 1000);\n if (seconds < 60) return \"Just now\";\n const minutes = Math.floor(seconds / 60);\n if (minutes < 60) return `${minutes} min ago`;\n const hours = Math.floor(minutes / 60);\n if (hours < 24) return `${hours}h ago`;\n const days = Math.floor(hours / 24);\n return `${days}d ago`;\n}\n\nfunction readStatus(): DaemonStatus | null {\n return StatusWriter.read();\n}\n\nfunction buildMenu(status: DaemonStatus | null): Menu {\n const running = status?.running ?? false;\n const statusIcon = running ? \"\\u25CF\" : \"\\u25CB\"; // ● or ○\n const statusText = running ? \"Running\" : \"Stopped\";\n\n const projects = status?.trackedProjects ?? [];\n const projectItems: MenuItem[] = projects.length > 0\n ? projects.map((p) => ({\n title: ` ${p.name}`,\n tooltip: p.path,\n enabled: false,\n checked: false,\n }))\n : [{\n title: \" (none)\",\n tooltip: \"No tracked projects\",\n enabled: false,\n checked: false,\n }];\n\n const items: MenuItem[] = [\n {\n title: `${statusIcon} Skillo \\u2014 ${statusText}`,\n tooltip: \"Daemon status\",\n enabled: false,\n checked: false,\n },\n SEPARATOR,\n {\n title: status?.user ? `Logged in as: ${status.user}` : \"Not logged in\",\n tooltip: \"User info\",\n enabled: false,\n checked: false,\n },\n SEPARATOR,\n {\n title: `Tracked Projects (${projects.length})`,\n tooltip: \"Projects being tracked\",\n enabled: false,\n checked: false,\n },\n ...projectItems,\n SEPARATOR,\n {\n title: `Last sync: ${formatTimeAgo(status?.claudeWatcher?.lastSync)}`,\n tooltip: \"Last Claude conversation sync\",\n enabled: false,\n checked: false,\n },\n {\n title: `Prompts synced: ${status?.claudeWatcher?.promptsSynced ?? 0}`,\n tooltip: \"Total prompts synced\",\n enabled: false,\n checked: false,\n },\n {\n title: `Skills detected: ${status?.skillDetector?.usagesDetected ?? 0}`,\n tooltip: \"Skill usages detected\",\n enabled: false,\n checked: false,\n },\n SEPARATOR,\n {\n title: running ? \"Stop Daemon\" : \"Start Daemon\",\n tooltip: running ? \"Stop the Skillo daemon\" : \"Start the Skillo daemon\",\n enabled: true,\n checked: false,\n },\n {\n title: \"Open Dashboard\",\n tooltip: \"Open Skillo dashboard in browser\",\n enabled: true,\n checked: false,\n },\n SEPARATOR,\n {\n title: \"Quit Tray\",\n tooltip: \"Hide tray icon (daemon keeps running)\",\n enabled: true,\n checked: false,\n },\n ];\n\n return {\n icon: ICON_BASE64,\n title: \"\",\n tooltip: `Skillo \\u2014 ${statusText}`,\n items,\n };\n}\n\nfunction openBrowser(url: string) {\n const os = platform();\n let cmd: string;\n if (os === \"darwin\") {\n cmd = `open \"${url}\"`;\n } else if (os === \"win32\") {\n cmd = `start \"\" \"${url}\"`;\n } else {\n cmd = `xdg-open \"${url}\"`;\n }\n exec(cmd, () => {});\n}\n\nfunction execSkillo(args: string) {\n exec(`skillo ${args}`, () => {});\n}\n\nfunction handleClick(title: string, cleanup: () => void) {\n if (title === \"Stop Daemon\" || title === \"Start Daemon\") {\n execSkillo(title === \"Stop Daemon\" ? \"stop\" : \"start\");\n } else if (title === \"Open Dashboard\") {\n openBrowser(\"https://www.skillo.one/dashboard\");\n } else if (title === \"Quit Tray\") {\n cleanup();\n }\n}\n\n// ── macOS native tray (Swift helper) ─────────────────────────────────────────\n\nfunction findDarwinHelper(): string | null {\n // Look for the compiled helper next to the running script, or in scripts/\n const candidates = [\n join(getDataDir(), \"tray-helper-darwin\"),\n join(dirname(fileURLToPath(import.meta.url)), \"..\", \"scripts\", \"tray-helper-darwin\"),\n join(dirname(fileURLToPath(import.meta.url)), \"scripts\", \"tray-helper-darwin\"),\n ];\n for (const p of candidates) {\n if (existsSync(p)) return p;\n }\n return null;\n}\n\nfunction compileDarwinHelper(): string | null {\n // Compile from source if Swift is available\n const srcCandidates = [\n join(dirname(fileURLToPath(import.meta.url)), \"..\", \"scripts\", \"tray-helper-darwin.swift\"),\n join(dirname(fileURLToPath(import.meta.url)), \"scripts\", \"tray-helper-darwin.swift\"),\n ];\n let src: string | null = null;\n for (const p of srcCandidates) {\n if (existsSync(p)) { src = p; break; }\n }\n if (!src) return null;\n\n const out = join(getDataDir(), \"tray-helper-darwin\");\n try {\n execSync(`swiftc \"${src}\" -o \"${out}\" -O 2>/dev/null`, { timeout: 60000 });\n chmodSync(out, 0o755);\n return out;\n } catch {\n return null;\n }\n}\n\nasync function startDarwinTray(pidFile: string): Promise<void> {\n let helperPath = findDarwinHelper();\n if (!helperPath) {\n helperPath = compileDarwinHelper();\n }\n if (!helperPath) {\n // Fall back to systray2\n await startSystray2Tray(pidFile);\n return;\n }\n\n const initialStatus = readStatus();\n const menu = buildMenu(initialStatus);\n\n const helper = spawn(helperPath, [], {\n stdio: [\"pipe\", \"pipe\", \"ignore\"],\n });\n\n const cleanup = () => {\n try { unlinkSync(pidFile); } catch { /* ignore */ }\n helper.kill();\n process.exit(0);\n };\n process.on(\"SIGINT\", cleanup);\n process.on(\"SIGTERM\", cleanup);\n helper.on(\"exit\", () => {\n try { unlinkSync(pidFile); } catch { /* ignore */ }\n process.exit(0);\n });\n\n // Wait for ready, then send menu\n await new Promise<void>((resolve, reject) => {\n let buf = \"\";\n const onData = (data: Buffer) => {\n buf += data.toString();\n const lines = buf.split(\"\\n\");\n buf = lines.pop() || \"\";\n for (const line of lines) {\n if (!line.trim()) continue;\n try {\n const msg = JSON.parse(line);\n if (msg.type === \"ready\") {\n helper.stdout!.removeListener(\"data\", onData);\n resolve();\n }\n } catch { /* ignore parse errors */ }\n }\n };\n helper.stdout!.on(\"data\", onData);\n setTimeout(() => reject(new Error(\"Tray helper did not become ready\")), 10000);\n });\n\n // Send initial menu\n helper.stdin!.write(JSON.stringify(menu) + \"\\n\");\n\n // Listen for clicks\n let clickBuf = \"\";\n helper.stdout!.on(\"data\", (data: Buffer) => {\n clickBuf += data.toString();\n const lines = clickBuf.split(\"\\n\");\n clickBuf = lines.pop() || \"\";\n for (const line of lines) {\n if (!line.trim()) continue;\n try {\n const msg = JSON.parse(line);\n if (msg.type === \"clicked\" && msg.item) {\n handleClick(msg.item.title, cleanup);\n }\n } catch { /* ignore */ }\n }\n });\n\n // Poll and update\n setInterval(() => {\n try {\n const status = readStatus();\n const updatedMenu = buildMenu(status);\n const action = { type: \"update-menu\", menu: updatedMenu };\n helper.stdin!.write(JSON.stringify(action) + \"\\n\");\n } catch { /* ignore */ }\n }, STATUS_POLL_MS);\n\n await new Promise(() => {});\n}\n\n// ── systray2 tray (Linux/Windows) ────────────────────────────────────────────\n\nfunction fixSystrayBinaryPermissions() {\n if (platform() === \"win32\") return;\n try {\n const cacheDir = join(homedir(), \".cache\", \"node-systray\");\n if (!existsSync(cacheDir)) return;\n for (const ver of readdirSync(cacheDir)) {\n const dir = join(cacheDir, ver);\n for (const file of readdirSync(dir)) {\n if (file.startsWith(\"tray_\")) {\n chmodSync(join(dir, file), 0o755);\n }\n }\n }\n } catch { /* best-effort */ }\n}\n\nasync function startSystray2Tray(pidFile: string): Promise<void> {\n fixSystrayBinaryPermissions();\n\n const initialStatus = readStatus();\n const menu = buildMenu(initialStatus);\n\n const systray = new SysTray({\n menu,\n debug: false,\n copyDir: true,\n });\n\n const cleanup = () => {\n try { unlinkSync(pidFile); } catch { /* ignore */ }\n systray.kill(false);\n process.exit(0);\n };\n process.on(\"SIGINT\", cleanup);\n process.on(\"SIGTERM\", cleanup);\n\n await systray.ready();\n\n systray.onClick(async (action: { item: MenuItem }) => {\n handleClick(action.item.title, cleanup);\n });\n\n setInterval(async () => {\n try {\n const status = readStatus();\n const updatedMenu = buildMenu(status);\n await systray.sendAction({ type: \"update-menu\", menu: updatedMenu });\n } catch { /* ignore */ }\n }, STATUS_POLL_MS);\n\n await new Promise(() => {});\n}\n\n// ── Entry point ──────────────────────────────────────────────────────────────\n\nexport async function startTray(): Promise<void> {\n const pidFile = join(getDataDir(), TRAY_PID_FILE);\n ensureDirectory(getDataDir());\n\n // Check if another tray is already running\n if (existsSync(pidFile)) {\n try {\n const pid = parseInt(readFileSync(pidFile, \"utf-8\").trim(), 10);\n process.kill(pid, 0);\n console.error(\"Tray icon is already running (PID: \" + pid + \")\");\n process.exit(1);\n } catch {\n // Stale PID file, continue\n }\n }\n\n writeFileSync(pidFile, String(process.pid));\n\n if (platform() === \"darwin\") {\n await startDarwinTray(pidFile);\n } else {\n await startSystray2Tray(pidFile);\n }\n}\n","/**\n * StatusWriter — writes daemon status JSON for tray icon and diagnostics.\n *\n * Writes ~/.skillo/daemon-status.json every 10s and on key events.\n * Tray icon and `skillo status` read this file.\n */\n\nimport { writeFileSync, readFileSync, existsSync } from \"fs\";\nimport { join } from \"path\";\nimport { getDataDir, ensureDirectory } from \"./paths.js\";\n\nexport interface DaemonStatus {\n running: boolean;\n pid: number | null;\n startedAt: string | null;\n updatedAt: string;\n user?: string;\n claudeWatcher?: {\n lastSync?: string | null;\n promptsSynced?: number;\n lastError?: string | null;\n };\n skillDetector?: {\n deployedSkills?: number;\n usagesDetected?: number;\n lastDetection?: string | null;\n lastError?: string | null;\n };\n trackedProjects?: Array<{ name: string; path: string }>;\n activeSessions?: number;\n}\n\nconst STATUS_FILE = \"daemon-status.json\";\nconst WRITE_INTERVAL_MS = 10000;\n\nexport class StatusWriter {\n private intervalId: ReturnType<typeof setInterval> | null = null;\n private status: DaemonStatus;\n private filePath: string;\n\n constructor() {\n this.filePath = join(getDataDir(), STATUS_FILE);\n this.status = {\n running: true,\n pid: process.pid,\n startedAt: new Date().toISOString(),\n updatedAt: new Date().toISOString(),\n claudeWatcher: { lastSync: null, promptsSynced: 0, lastError: null },\n skillDetector: { deployedSkills: 0, usagesDetected: 0, lastDetection: null, lastError: null },\n trackedProjects: [],\n activeSessions: 0,\n };\n }\n\n /** Start periodic writes */\n start() {\n ensureDirectory(getDataDir());\n this.write();\n this.intervalId = setInterval(() => this.write(), WRITE_INTERVAL_MS);\n }\n\n /** Stop writing and mark as not running */\n stop() {\n if (this.intervalId) {\n clearInterval(this.intervalId);\n this.intervalId = null;\n }\n this.status.running = false;\n this.status.pid = null;\n this.status.updatedAt = new Date().toISOString();\n this.write();\n }\n\n /** Merge partial status updates */\n update(partial: Partial<DaemonStatus>) {\n // Deep merge for nested objects\n if (partial.claudeWatcher) {\n this.status.claudeWatcher = { ...this.status.claudeWatcher, ...partial.claudeWatcher };\n delete partial.claudeWatcher;\n }\n if (partial.skillDetector) {\n this.status.skillDetector = { ...this.status.skillDetector, ...partial.skillDetector };\n delete partial.skillDetector;\n }\n Object.assign(this.status, partial);\n }\n\n /** Get the status file path */\n static getStatusFilePath(): string {\n return join(getDataDir(), STATUS_FILE);\n }\n\n /** Read current status from disk (static, for tray/status commands) */\n static read(): DaemonStatus | null {\n const filePath = StatusWriter.getStatusFilePath();\n try {\n if (existsSync(filePath)) {\n return JSON.parse(readFileSync(filePath, \"utf-8\"));\n }\n } catch {\n // ignore\n }\n return null;\n }\n\n private write() {\n this.status.updatedAt = new Date().toISOString();\n try {\n writeFileSync(this.filePath, JSON.stringify(this.status, null, 2), \"utf-8\");\n } catch {\n // Can't write status, ignore\n }\n }\n}\n"],"mappings":"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","names":["readFileSync","existsSync","writeFileSync","join","join","existsSync","readFileSync","writeFileSync"]}