claude-code-sync 0.1.10 → 0.1.12

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -92,6 +92,32 @@ Ready! Start Claude Code and sessions will sync automatically.
92
92
 
93
93
  Sessions will now sync automatically when you use Claude Code.
94
94
 
95
+ ## Upgrading
96
+
97
+ To upgrade to the latest version:
98
+
99
+ ```bash
100
+ # Close Claude Code first (important for hooks to reload)
101
+ # Then upgrade the package
102
+ npm update -g claude-code-sync
103
+
104
+ # Verify the new version
105
+ claude-code-sync --version
106
+
107
+ # Check your configuration is still valid
108
+ claude-code-sync verify
109
+ ```
110
+
111
+ If you encounter issues after upgrading:
112
+
113
+ ```bash
114
+ # Re-run setup to ensure hooks are configured
115
+ claude-code-sync setup
116
+
117
+ # Test connectivity
118
+ claude-code-sync synctest
119
+ ```
120
+
95
121
  ## CLI Commands
96
122
 
97
123
  | Command | Description |
package/dist/cli.js CHANGED
@@ -471,7 +471,15 @@ const SESSION_STATE_FILE = path.join(process.env.HOME || "~", ".config", "claude
471
471
  function loadSessionState() {
472
472
  try {
473
473
  if (fs.existsSync(SESSION_STATE_FILE)) {
474
- return JSON.parse(fs.readFileSync(SESSION_STATE_FILE, "utf-8"));
474
+ const data = JSON.parse(fs.readFileSync(SESSION_STATE_FILE, "utf-8"));
475
+ // Convert arrays back to Sets for syncedMessageUuids
476
+ for (const sessionId of Object.keys(data)) {
477
+ const session = data[sessionId];
478
+ if (session.syncedMessageUuids && Array.isArray(session.syncedMessageUuids)) {
479
+ session.syncedMessageUuids = new Set(session.syncedMessageUuids);
480
+ }
481
+ }
482
+ return data;
475
483
  }
476
484
  }
477
485
  catch {
@@ -485,7 +493,18 @@ function saveSessionState(state) {
485
493
  if (!fs.existsSync(dir)) {
486
494
  fs.mkdirSync(dir, { recursive: true });
487
495
  }
488
- fs.writeFileSync(SESSION_STATE_FILE, JSON.stringify(state, null, 2));
496
+ // Convert Sets to arrays for JSON serialization
497
+ const serializable = {};
498
+ for (const sessionId of Object.keys(state)) {
499
+ const session = state[sessionId];
500
+ serializable[sessionId] = {
501
+ ...session,
502
+ syncedMessageUuids: session.syncedMessageUuids instanceof Set
503
+ ? Array.from(session.syncedMessageUuids)
504
+ : session.syncedMessageUuids,
505
+ };
506
+ }
507
+ fs.writeFileSync(SESSION_STATE_FILE, JSON.stringify(serializable, null, 2));
489
508
  }
490
509
  catch {
491
510
  // Ignore errors
@@ -503,6 +522,65 @@ function generateTitle(prompt) {
503
522
  }
504
523
  return trimmed;
505
524
  }
525
+ function parseTranscriptFile(transcriptPath) {
526
+ const result = {
527
+ assistantMessages: [],
528
+ tokenUsage: { input: 0, output: 0 },
529
+ };
530
+ try {
531
+ if (!fs.existsSync(transcriptPath)) {
532
+ return result;
533
+ }
534
+ const content = fs.readFileSync(transcriptPath, "utf-8");
535
+ const lines = content.trim().split("\n");
536
+ const seenUuids = new Set();
537
+ for (const line of lines) {
538
+ if (!line.trim())
539
+ continue;
540
+ try {
541
+ const entry = JSON.parse(line);
542
+ // Only process assistant messages
543
+ if (entry.type === "assistant" && entry.message) {
544
+ const msg = entry.message;
545
+ const uuid = entry.uuid || "";
546
+ // Skip if we've already seen this UUID (dedup)
547
+ if (uuid && seenUuids.has(uuid))
548
+ continue;
549
+ if (uuid)
550
+ seenUuids.add(uuid);
551
+ // Extract text content from message
552
+ if (msg.content && Array.isArray(msg.content)) {
553
+ for (const part of msg.content) {
554
+ if (part.type === "text" && part.text) {
555
+ result.assistantMessages.push({
556
+ uuid,
557
+ text: part.text,
558
+ timestamp: entry.timestamp || new Date().toISOString(),
559
+ model: msg.model,
560
+ });
561
+ }
562
+ }
563
+ }
564
+ // Accumulate token usage from all assistant messages
565
+ if (msg.usage) {
566
+ result.tokenUsage.input +=
567
+ (msg.usage.input_tokens || 0) +
568
+ (msg.usage.cache_read_input_tokens || 0) +
569
+ (msg.usage.cache_creation_input_tokens || 0);
570
+ result.tokenUsage.output += msg.usage.output_tokens || 0;
571
+ }
572
+ }
573
+ }
574
+ catch {
575
+ // Skip malformed lines
576
+ }
577
+ }
578
+ }
579
+ catch {
580
+ // Return empty result on error
581
+ }
582
+ return result;
583
+ }
506
584
  program
507
585
  .command("hook <event>")
508
586
  .description("Handle Claude Code hook events (reads stdin)")
@@ -625,38 +703,46 @@ program
625
703
  break;
626
704
  }
627
705
  case "Stop": {
628
- // Stop event contains the assistant's response and token usage
706
+ // Stop event provides transcript_path - we read it to get messages and tokens
629
707
  const data = JSON.parse(input);
630
708
  const state = sessionState[data.session_id] || {};
631
- // Update accumulated token usage
632
- if (data.token_usage) {
633
- const current = state.tokenUsage || { input: 0, output: 0 };
634
- state.tokenUsage = {
635
- input: current.input + data.token_usage.input,
636
- output: current.output + data.token_usage.output,
637
- };
709
+ // Parse transcript file to extract assistant messages and token usage
710
+ if (data.transcript_path) {
711
+ const transcript = parseTranscriptFile(data.transcript_path);
712
+ // Update token usage from transcript
713
+ if (transcript.tokenUsage.input > 0 || transcript.tokenUsage.output > 0) {
714
+ state.tokenUsage = transcript.tokenUsage;
715
+ }
716
+ // Track which messages we've already synced to avoid duplicates
717
+ const syncedMessages = state.syncedMessageUuids instanceof Set
718
+ ? state.syncedMessageUuids
719
+ : new Set(Array.isArray(state.syncedMessageUuids) ? state.syncedMessageUuids : []);
720
+ // Sync new assistant messages
721
+ for (const msg of transcript.assistantMessages) {
722
+ // Skip if we've already synced this message
723
+ if (msg.uuid && syncedMessages.has(msg.uuid))
724
+ continue;
725
+ if (msg.uuid)
726
+ syncedMessages.add(msg.uuid);
727
+ // Increment message count
728
+ state.messageCount = (state.messageCount || 0) + 1;
729
+ const message = {
730
+ sessionId: data.session_id,
731
+ messageId: msg.uuid || `${data.session_id}-assistant-${Date.now()}`,
732
+ source: "claude-code",
733
+ role: "assistant",
734
+ content: msg.text,
735
+ model: msg.model,
736
+ timestamp: msg.timestamp,
737
+ };
738
+ await client.syncMessage(message);
739
+ }
740
+ // Store synced message UUIDs (convert Set to array for JSON)
741
+ state.syncedMessageUuids = syncedMessages;
638
742
  }
639
- // Increment message count for assistant response
640
- state.messageCount = (state.messageCount || 0) + 1;
641
743
  sessionState[data.session_id] = state;
642
744
  saveSessionState(sessionState);
643
- // Sync the assistant's response as a message
644
- if (data.response) {
645
- const message = {
646
- sessionId: data.session_id,
647
- messageId: `${data.session_id}-assistant-${Date.now()}`,
648
- source: "claude-code",
649
- role: "assistant",
650
- content: data.response,
651
- tokenCount: data.token_usage
652
- ? data.token_usage.input + data.token_usage.output
653
- : undefined,
654
- durationMs: data.duration_ms,
655
- timestamp: new Date().toISOString(),
656
- };
657
- await client.syncMessage(message);
658
- }
659
- // Update session with accumulated token usage
745
+ // Update session with token usage from transcript
660
746
  const session = {
661
747
  sessionId: data.session_id,
662
748
  source: "claude-code",
@@ -680,4 +766,4 @@ program
680
766
  });
681
767
  // Parse and run
682
768
  program.parse();
683
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cli.js","sourceRoot":"","sources":["../src/cli.ts"],"names":[],"mappings":"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","sourcesContent":["#!/usr/bin/env node\n\n/**\n * Claude Code Sync CLI\n *\n * Commands:\n *   login   - Configure Convex URL and API Key\n *   logout  - Clear stored credentials\n *   status  - Show connection status\n *   config  - Show current configuration\n */\n\nimport { Command } from \"commander\";\nimport * as readline from \"readline\";\nimport * as fs from \"fs\";\nimport * as path from \"path\";\nimport {\n  loadConfig,\n  saveConfig,\n  clearConfig,\n  SyncClient,\n  Config,\n  SessionData,\n  MessageData,\n} from \"./index\";\n\n// Types for Claude Code hook event data from stdin\ninterface HookSessionStartData {\n  session_id: string;\n  cwd?: string;\n  model?: string;\n  permission_mode?: string;\n  source?: string;\n  thinking_enabled?: boolean;\n  mcp_servers?: string[];\n}\n\ninterface HookSessionEndData {\n  session_id: string;\n  reason?: \"user_stop\" | \"max_turns\" | \"error\" | \"completed\";\n  message_count?: number;\n  tool_call_count?: number;\n  total_token_usage?: {\n    input: number;\n    output: number;\n  };\n  cost_estimate?: number;\n}\n\ninterface HookUserPromptData {\n  session_id: string;\n  prompt: string;\n  timestamp?: string;\n}\n\ninterface HookToolUseData {\n  session_id: string;\n  tool_name: string;\n  tool_use_id?: string;\n  tool_input?: Record<string, unknown>;\n  tool_result?: {\n    output?: string;\n    error?: string;\n  };\n  duration_ms?: number;\n  success?: boolean;\n}\n\ninterface HookStopData {\n  session_id: string;\n  response?: string;\n  token_usage?: {\n    input: number;\n    output: number;\n  };\n  duration_ms?: number;\n  stop_reason?: string;\n}\n\n// Types for Claude Code settings.json\ninterface ClaudeSettings {\n  hooks?: Record<string, unknown>;\n  [key: string]: unknown;\n}\n\n// Type for package.json version field\ninterface PackageJson {\n  version?: string;\n}\n\n// Read version from package.json\nfunction getVersion(): string {\n  try {\n    const pkgPath = path.join(__dirname, \"..\", \"package.json\");\n    const pkg = JSON.parse(fs.readFileSync(pkgPath, \"utf-8\")) as PackageJson;\n    return pkg.version || \"0.0.0\";\n  } catch {\n    return \"0.0.0\";\n  }\n}\n\nconst program = new Command();\n\nprogram\n  .name(\"claude-code-sync\")\n  .description(\"Sync Claude Code sessions to OpenSync dashboard\")\n  .version(getVersion());\n\n// ============================================================================\n// Helper Functions\n// ============================================================================\n\nfunction prompt(question: string): Promise<string> {\n  const rl = readline.createInterface({\n    input: process.stdin,\n    output: process.stdout,\n  });\n\n  return new Promise((resolve) => {\n    rl.question(question, (answer) => {\n      rl.close();\n      resolve(answer.trim());\n    });\n  });\n}\n\nfunction maskApiKey(key: string): string {\n  if (key.length <= 8) return \"****\";\n  return key.substring(0, 4) + \"****\" + key.substring(key.length - 4);\n}\n\n// ============================================================================\n// Commands\n// ============================================================================\n\nprogram\n  .command(\"login\")\n  .description(\"Configure Convex URL and API Key\")\n  .action(async () => {\n    console.log(\"\\n  Claude Code Sync - Login\\n\");\n    console.log(\"Get your API key from your OpenSync.dev Settings page, starts with osk_. Enter it here:\");\n    console.log(\"  1. Go to Settings\");\n    console.log(\"  2. Click 'Generate API Key'\");\n    console.log(\"  3. Copy the key (starts with osk_)\\n\");\n\n    const convexUrl = await prompt(\"Convex URL (e.g., https://your-project.convex.cloud): \");\n\n    if (!convexUrl) {\n      console.error(\"Error: Convex URL is required\");\n      process.exit(1);\n    }\n\n    if (!convexUrl.includes(\"convex.cloud\") && !convexUrl.includes(\"convex.site\")) {\n      console.error(\"Error: Invalid Convex URL. Must contain convex.cloud or convex.site\");\n      process.exit(1);\n    }\n\n    const apiKey = await prompt(\"API Key (osk_...): \");\n\n    if (!apiKey) {\n      console.error(\"Error: API Key is required\");\n      process.exit(1);\n    }\n\n    if (!apiKey.startsWith(\"osk_\")) {\n      console.error(\"Error: Invalid API Key. Must start with osk_\");\n      process.exit(1);\n    }\n\n    const config: Config = {\n      convexUrl,\n      apiKey,\n      autoSync: true,\n      syncToolCalls: true,\n      syncThinking: false,\n    };\n\n    // Test connection\n    console.log(\"\\nTesting connection...\");\n    const client = new SyncClient(config);\n    const connected = await client.testConnection();\n\n    if (!connected) {\n      console.error(\"Error: Could not connect to Convex backend\");\n      console.error(\"   Check your URL and try again\");\n      process.exit(1);\n    }\n\n    // Save config\n    saveConfig(config);\n    console.log(\"\\nConfiguration saved!\");\n    console.log(`   URL: ${convexUrl}`);\n    console.log(`   Key: ${maskApiKey(apiKey)}`);\n    console.log(\"\\nNext step: Run the setup command to configure Claude Code hooks:\\n\");\n    console.log(\"   claude-code-sync setup\\n\");\n  });\n\nprogram\n  .command(\"logout\")\n  .description(\"Clear stored credentials\")\n  .action(() => {\n    clearConfig();\n    console.log(\"Credentials cleared\");\n  });\n\nprogram\n  .command(\"status\")\n  .description(\"Show connection status\")\n  .action(async () => {\n    const config = loadConfig();\n\n    console.log(\"\\n  Claude Code Sync - Status\\n\");\n\n    if (!config) {\n      console.log(\"Not configured\");\n      console.log(\"   Run 'claude-code-sync login' to set up\\n\");\n      process.exit(1);\n    }\n\n    console.log(\"Configuration:\");\n    console.log(`  Convex URL: ${config.convexUrl}`);\n    console.log(`  API Key:    ${maskApiKey(config.apiKey)}`);\n    console.log(`  Auto Sync:  ${config.autoSync !== false ? \"enabled\" : \"disabled\"}`);\n    console.log(`  Tool Calls: ${config.syncToolCalls !== false ? \"enabled\" : \"disabled\"}`);\n    console.log(`  Thinking:   ${config.syncThinking ? \"enabled\" : \"disabled\"}`);\n\n    console.log(\"\\nTesting connection...\");\n    const client = new SyncClient(config);\n    const connected = await client.testConnection();\n\n    if (connected) {\n      console.log(\"Connected to Convex backend\\n\");\n    } else {\n      console.log(\"Error: Could not connect to Convex backend\\n\");\n      process.exit(1);\n    }\n  });\n\nprogram\n  .command(\"config\")\n  .description(\"Show current configuration\")\n  .option(\"--json\", \"Output as JSON\")\n  .action((options: { json?: boolean }) => {\n    const config = loadConfig();\n\n    if (!config) {\n      if (options.json) {\n        console.log(JSON.stringify({ configured: false }));\n      } else {\n        console.log(\"Not configured. Run 'claude-code-sync login' to set up.\");\n      }\n      return;\n    }\n\n    if (options.json) {\n      console.log(\n        JSON.stringify(\n          {\n            configured: true,\n            convexUrl: config.convexUrl,\n            apiKey: maskApiKey(config.apiKey),\n            autoSync: config.autoSync !== false,\n            syncToolCalls: config.syncToolCalls !== false,\n            syncThinking: config.syncThinking === true,\n          },\n          null,\n          2\n        )\n      );\n    } else {\n      console.log(\"\\n  Current Configuration\\n\");\n      console.log(`Convex URL:  ${config.convexUrl}`);\n      console.log(`API Key:     ${maskApiKey(config.apiKey)}`);\n      console.log(`Auto Sync:   ${config.autoSync !== false}`);\n      console.log(`Tool Calls:  ${config.syncToolCalls !== false}`);\n      console.log(`Thinking:    ${config.syncThinking === true}`);\n      console.log(`\\nConfig file: ~/.config/claude-code-sync/config.json\\n`);\n    }\n  });\n\nprogram\n  .command(\"set <key> <value>\")\n  .description(\"Set a configuration value\")\n  .action((key: string, value: string) => {\n    const config = loadConfig();\n\n    if (!config) {\n      console.error(\"Not configured. Run 'claude-code-sync login' first.\");\n      process.exit(1);\n    }\n\n    const validKeys = [\"autoSync\", \"syncToolCalls\", \"syncThinking\"];\n    if (!validKeys.includes(key)) {\n      console.error(`Invalid key. Valid keys: ${validKeys.join(\", \")}`);\n      process.exit(1);\n    }\n\n    const boolValue = value === \"true\" || value === \"1\" || value === \"yes\";\n    \n    // Type-safe config update\n    if (key === \"autoSync\") {\n      config.autoSync = boolValue;\n    } else if (key === \"syncToolCalls\") {\n      config.syncToolCalls = boolValue;\n    } else if (key === \"syncThinking\") {\n      config.syncThinking = boolValue;\n    }\n\n    saveConfig(config);\n    console.log(`Set ${key} = ${boolValue}`);\n  });\n\n// ============================================================================\n// Setup Command (configures Claude Code hooks)\n// ============================================================================\n\n// Claude Code hooks configuration\nconst CLAUDE_HOOKS_CONFIG = {\n  hooks: {\n    SessionStart: [\n      {\n        hooks: [\n          {\n            type: \"command\",\n            command: \"claude-code-sync hook SessionStart\",\n          },\n        ],\n      },\n    ],\n    SessionEnd: [\n      {\n        hooks: [\n          {\n            type: \"command\",\n            command: \"claude-code-sync hook SessionEnd\",\n          },\n        ],\n      },\n    ],\n    UserPromptSubmit: [\n      {\n        hooks: [\n          {\n            type: \"command\",\n            command: \"claude-code-sync hook UserPromptSubmit\",\n          },\n        ],\n      },\n    ],\n    PostToolUse: [\n      {\n        matcher: \"*\",\n        hooks: [\n          {\n            type: \"command\",\n            command: \"claude-code-sync hook PostToolUse\",\n          },\n        ],\n      },\n    ],\n    Stop: [\n      {\n        matcher: \"*\",\n        hooks: [\n          {\n            type: \"command\",\n            command: \"claude-code-sync hook Stop\",\n          },\n        ],\n      },\n    ],\n  },\n};\n\nprogram\n  .command(\"setup\")\n  .description(\"Add hooks to Claude Code settings (configures ~/.claude/settings.json)\")\n  .option(\"--force\", \"Overwrite existing hooks configuration\")\n  .action(async (options: { force?: boolean }) => {\n    const claudeDir = path.join(process.env.HOME || \"~\", \".claude\");\n    const settingsPath = path.join(claudeDir, \"settings.json\");\n\n    console.log(\"\\n  Claude Code Sync - Setup\\n\");\n\n    // Check if plugin credentials are configured\n    const config = loadConfig();\n    if (!config) {\n      console.log(\"Warning: Plugin not configured yet.\");\n      console.log(\"   Run 'claude-code-sync login' first to set up credentials.\\n\");\n    }\n\n    // Create .claude directory if needed\n    if (!fs.existsSync(claudeDir)) {\n      fs.mkdirSync(claudeDir, { recursive: true });\n      console.log(\"Created ~/.claude directory\");\n    }\n\n    // Check for existing settings\n    let existingSettings: ClaudeSettings = {};\n    let hasExistingHooks = false;\n\n    if (fs.existsSync(settingsPath)) {\n      try {\n        const content = fs.readFileSync(settingsPath, \"utf-8\");\n        existingSettings = JSON.parse(content) as ClaudeSettings;\n        hasExistingHooks = !!existingSettings.hooks;\n        console.log(\"Found existing settings.json\");\n      } catch {\n        console.log(\"Warning: Could not parse existing settings.json, will create new one\");\n      }\n    }\n\n    // Handle existing hooks\n    if (hasExistingHooks && !options.force) {\n      console.log(\"\\nExisting hooks configuration found.\");\n      console.log(\"   Use --force to overwrite, or manually merge the hooks.\\n\");\n      console.log(\"To manually add, include these hooks in your settings.json:\");\n      console.log(JSON.stringify(CLAUDE_HOOKS_CONFIG, null, 2));\n      process.exit(1);\n    }\n\n    // Merge settings\n    const newSettings = {\n      ...existingSettings,\n      ...CLAUDE_HOOKS_CONFIG,\n    };\n\n    // Write settings\n    try {\n      fs.writeFileSync(settingsPath, JSON.stringify(newSettings, null, 2));\n      console.log(\"\\nClaude Code hooks configured!\");\n      console.log(`   Settings file: ${settingsPath}`);\n      console.log(\"\\nSetup complete. Sessions will sync automatically.\\n\");\n    } catch (error) {\n      console.error(\"Error writing settings:\", error);\n      process.exit(1);\n    }\n  });\n\nprogram\n  .command(\"verify\")\n  .description(\"Verify credentials and Claude Code configuration\")\n  .action(async () => {\n    console.log(\"\\n  OpenSync Setup Verification\\n\");\n\n    // Check credentials\n    const config = loadConfig();\n    if (config) {\n      console.log(\"Credentials: OK\");\n      console.log(`   Convex URL: ${config.convexUrl}`);\n      console.log(`   API Key: ${maskApiKey(config.apiKey)}`);\n    } else {\n      console.log(\"Credentials: NOT CONFIGURED\");\n      console.log(\"   Run 'claude-code-sync login' to set up\");\n    }\n\n    // Check Claude Code config\n    const settingsPath = path.join(process.env.HOME || \"~\", \".claude\", \"settings.json\");\n    let hooksConfigured = false;\n\n    if (fs.existsSync(settingsPath)) {\n      try {\n        const content = fs.readFileSync(settingsPath, \"utf-8\");\n        const settings = JSON.parse(content) as ClaudeSettings;\n        hooksConfigured = !!settings.hooks?.SessionStart;\n      } catch {\n        // Ignore parse errors\n      }\n    }\n\n    console.log(\"\");\n    if (hooksConfigured) {\n      console.log(\"Claude Code Config: OK\");\n      console.log(`   Config file: ${settingsPath}`);\n      console.log(\"   Hooks registered: claude-code-sync\");\n    } else {\n      console.log(\"Claude Code Config: NOT CONFIGURED\");\n      console.log(\"   Run 'claude-code-sync setup' to configure hooks\");\n    }\n\n    // Test connection if credentials exist\n    if (config) {\n      console.log(\"\\nTesting connection...\");\n      const client = new SyncClient(config);\n      const connected = await client.testConnection();\n      if (connected) {\n        console.log(\"Connection: OK\\n\");\n      } else {\n        console.log(\"Connection: FAILED\\n\");\n        process.exit(1);\n      }\n    }\n\n    if (config && hooksConfigured) {\n      console.log(\"Ready! Start Claude Code and sessions will sync automatically.\\n\");\n    } else {\n      console.log(\"\");\n      process.exit(1);\n    }\n  });\n\n// ============================================================================\n// Sync Test Command (test connectivity)\n// ============================================================================\n\nprogram\n  .command(\"synctest\")\n  .description(\"Test connectivity and create a test session\")\n  .action(async () => {\n    const config = loadConfig();\n\n    console.log(\"\\n  Claude Code Sync - Connection Test\\n\");\n\n    if (!config) {\n      console.log(\"Not configured\");\n      console.log(\"   Run 'claude-code-sync login' to set up\\n\");\n      process.exit(1);\n    }\n\n    console.log(\"Configuration:\");\n    console.log(`  Convex URL: ${config.convexUrl}`);\n    console.log(`  API Key:    ${maskApiKey(config.apiKey)}`);\n\n    console.log(\"\\nTesting connection...\");\n    const client = new SyncClient(config);\n    const connected = await client.testConnection();\n\n    if (connected) {\n      console.log(\"Connection: OK\");\n      \n      // Create a test session to verify full sync works\n      console.log(\"\\nCreating test session...\");\n      try {\n        const testSession = {\n          sessionId: `test-${Date.now()}`,\n          source: \"claude-code\" as const,\n          title: \"Connection Test\",\n          projectName: \"synctest\",\n          startedAt: new Date().toISOString(),\n          endedAt: new Date().toISOString(),\n        };\n        await client.syncSession(testSession);\n        console.log(\"Test session created successfully\");\n        console.log(\"\\nSync test passed. Ready to sync Claude Code sessions.\\n\");\n      } catch (error) {\n        console.log(`Test session failed: ${error}`);\n        console.log(\"\\nConnection works but sync may have issues.\\n\");\n        process.exit(1);\n      }\n    } else {\n      console.log(\"Connection: FAILED\");\n      console.log(\"\\nCheck your Convex URL and API key.\\n\");\n      process.exit(1);\n    }\n  });\n\n// ============================================================================\n// Hook Command (for Claude Code integration)\n// ============================================================================\n\n// Track session state for title generation (first user prompt)\nconst SESSION_STATE_FILE = path.join(\n  process.env.HOME || \"~\",\n  \".config\",\n  \"claude-code-sync\",\n  \"session-state.json\"\n);\n\ninterface SessionState {\n  [sessionId: string]: {\n    model?: string;\n    firstPrompt?: string;\n    tokenUsage?: { input: number; output: number };\n    messageCount?: number;\n  };\n}\n\nfunction loadSessionState(): SessionState {\n  try {\n    if (fs.existsSync(SESSION_STATE_FILE)) {\n      return JSON.parse(fs.readFileSync(SESSION_STATE_FILE, \"utf-8\")) as SessionState;\n    }\n  } catch {\n    // Ignore errors\n  }\n  return {};\n}\n\nfunction saveSessionState(state: SessionState): void {\n  try {\n    const dir = path.dirname(SESSION_STATE_FILE);\n    if (!fs.existsSync(dir)) {\n      fs.mkdirSync(dir, { recursive: true });\n    }\n    fs.writeFileSync(SESSION_STATE_FILE, JSON.stringify(state, null, 2));\n  } catch {\n    // Ignore errors\n  }\n}\n\nfunction generateTitle(prompt: string): string {\n  // Use first 80 chars of first prompt as title, trim at word boundary\n  const trimmed = prompt.slice(0, 80).trim();\n  if (prompt.length > 80) {\n    const lastSpace = trimmed.lastIndexOf(\" \");\n    if (lastSpace > 40) {\n      return trimmed.slice(0, lastSpace) + \"...\";\n    }\n    return trimmed + \"...\";\n  }\n  return trimmed;\n}\n\nprogram\n  .command(\"hook <event>\")\n  .description(\"Handle Claude Code hook events (reads stdin)\")\n  .action(async (event: string) => {\n    const config = loadConfig();\n\n    if (!config) {\n      // Exit silently if not configured (don't block Claude Code)\n      process.exit(0);\n    }\n\n    if (config.autoSync === false) {\n      process.exit(0);\n    }\n\n    // Read JSON input from stdin\n    let input = \"\";\n    for await (const chunk of process.stdin) {\n      input += chunk;\n    }\n\n    if (!input.trim()) {\n      process.exit(0);\n    }\n\n    try {\n      const client = new SyncClient(config);\n      const sessionState = loadSessionState();\n\n      switch (event) {\n        case \"SessionStart\": {\n          const data = JSON.parse(input) as HookSessionStartData;\n          \n          // Initialize session state\n          sessionState[data.session_id] = {\n            model: data.model,\n            tokenUsage: { input: 0, output: 0 },\n            messageCount: 0,\n          };\n          saveSessionState(sessionState);\n          \n          const session: SessionData = {\n            sessionId: data.session_id,\n            source: \"claude-code\",\n            cwd: data.cwd,\n            model: data.model,\n            permissionMode: data.permission_mode,\n            thinkingEnabled: data.thinking_enabled,\n            mcpServers: data.mcp_servers,\n            startType: data.source === \"startup\" ? \"new\" : (data.source as SessionData[\"startType\"]),\n            startedAt: new Date().toISOString(),\n            projectPath: data.cwd,\n            projectName: data.cwd ? data.cwd.split(\"/\").pop() : undefined,\n          };\n          await client.syncSession(session);\n          break;\n        }\n\n        case \"SessionEnd\": {\n          const data = JSON.parse(input) as HookSessionEndData;\n          const state = sessionState[data.session_id] || {};\n          \n          // Calculate final token usage\n          const finalTokenUsage = data.total_token_usage || state.tokenUsage;\n          \n          const session: SessionData = {\n            sessionId: data.session_id,\n            source: \"claude-code\",\n            title: state.firstPrompt ? generateTitle(state.firstPrompt) : undefined,\n            endReason: data.reason,\n            messageCount: data.message_count || state.messageCount,\n            toolCallCount: data.tool_call_count,\n            tokenUsage: finalTokenUsage,\n            costEstimate: data.cost_estimate,\n            endedAt: new Date().toISOString(),\n          };\n          await client.syncSession(session);\n          \n          // Clean up session state\n          delete sessionState[data.session_id];\n          saveSessionState(sessionState);\n          break;\n        }\n\n        case \"UserPromptSubmit\": {\n          const data = JSON.parse(input) as HookUserPromptData;\n          const state = sessionState[data.session_id] || {};\n          \n          // Track first prompt for title generation\n          if (!state.firstPrompt) {\n            state.firstPrompt = data.prompt;\n            sessionState[data.session_id] = state;\n            saveSessionState(sessionState);\n            \n            // Update session with title\n            const session: SessionData = {\n              sessionId: data.session_id,\n              source: \"claude-code\",\n              title: generateTitle(data.prompt),\n            };\n            await client.syncSession(session);\n          }\n          \n          // Increment message count\n          state.messageCount = (state.messageCount || 0) + 1;\n          sessionState[data.session_id] = state;\n          saveSessionState(sessionState);\n          \n          const message: MessageData = {\n            sessionId: data.session_id,\n            messageId: `${data.session_id}-user-${Date.now()}`,\n            source: \"claude-code\",\n            role: \"user\",\n            content: data.prompt,\n            timestamp: data.timestamp || new Date().toISOString(),\n          };\n          await client.syncMessage(message);\n          break;\n        }\n\n        case \"PostToolUse\": {\n          if (!config.syncToolCalls) break;\n          const data = JSON.parse(input) as HookToolUseData;\n          const state = sessionState[data.session_id] || {};\n          \n          const message: MessageData = {\n            sessionId: data.session_id,\n            messageId: data.tool_use_id || `${data.session_id}-tool-${Date.now()}`,\n            source: \"claude-code\",\n            role: \"assistant\",\n            toolName: data.tool_name,\n            toolArgs: data.tool_input,\n            toolResult: data.tool_result?.output || data.tool_result?.error,\n            durationMs: data.duration_ms,\n            timestamp: new Date().toISOString(),\n          };\n          await client.syncMessage(message);\n          break;\n        }\n\n        case \"Stop\": {\n          // Stop event contains the assistant's response and token usage\n          const data = JSON.parse(input) as HookStopData;\n          const state = sessionState[data.session_id] || {};\n          \n          // Update accumulated token usage\n          if (data.token_usage) {\n            const current = state.tokenUsage || { input: 0, output: 0 };\n            state.tokenUsage = {\n              input: current.input + data.token_usage.input,\n              output: current.output + data.token_usage.output,\n            };\n          }\n          \n          // Increment message count for assistant response\n          state.messageCount = (state.messageCount || 0) + 1;\n          sessionState[data.session_id] = state;\n          saveSessionState(sessionState);\n          \n          // Sync the assistant's response as a message\n          if (data.response) {\n            const message: MessageData = {\n              sessionId: data.session_id,\n              messageId: `${data.session_id}-assistant-${Date.now()}`,\n              source: \"claude-code\",\n              role: \"assistant\",\n              content: data.response,\n              tokenCount: data.token_usage\n                ? data.token_usage.input + data.token_usage.output\n                : undefined,\n              durationMs: data.duration_ms,\n              timestamp: new Date().toISOString(),\n            };\n            await client.syncMessage(message);\n          }\n          \n          // Update session with accumulated token usage\n          const session: SessionData = {\n            sessionId: data.session_id,\n            source: \"claude-code\",\n            tokenUsage: state.tokenUsage,\n            messageCount: state.messageCount,\n          };\n          await client.syncSession(session);\n          break;\n        }\n\n        default:\n          // Unknown event, ignore\n          break;\n      }\n\n      process.exit(0);\n    } catch (error) {\n      // Log to stderr but don't block Claude Code\n      console.error(`[claude-code-sync] Error: ${error}`);\n      process.exit(0);\n    }\n  });\n\n// Parse and run\nprogram.parse();\n"]}
769
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cli.js","sourceRoot":"","sources":["../src/cli.ts"],"names":[],"mappings":"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","sourcesContent":["#!/usr/bin/env node\n\n/**\n * Claude Code Sync CLI\n *\n * Commands:\n *   login   - Configure Convex URL and API Key\n *   logout  - Clear stored credentials\n *   status  - Show connection status\n *   config  - Show current configuration\n */\n\nimport { Command } from \"commander\";\nimport * as readline from \"readline\";\nimport * as fs from \"fs\";\nimport * as path from \"path\";\nimport {\n  loadConfig,\n  saveConfig,\n  clearConfig,\n  SyncClient,\n  Config,\n  SessionData,\n  MessageData,\n} from \"./index\";\n\n// Types for Claude Code hook event data from stdin\ninterface HookSessionStartData {\n  session_id: string;\n  cwd?: string;\n  model?: string;\n  permission_mode?: string;\n  source?: string;\n  thinking_enabled?: boolean;\n  mcp_servers?: string[];\n}\n\ninterface HookSessionEndData {\n  session_id: string;\n  reason?: \"user_stop\" | \"max_turns\" | \"error\" | \"completed\";\n  message_count?: number;\n  tool_call_count?: number;\n  total_token_usage?: {\n    input: number;\n    output: number;\n  };\n  cost_estimate?: number;\n}\n\ninterface HookUserPromptData {\n  session_id: string;\n  prompt: string;\n  timestamp?: string;\n}\n\ninterface HookToolUseData {\n  session_id: string;\n  tool_name: string;\n  tool_use_id?: string;\n  tool_input?: Record<string, unknown>;\n  tool_result?: {\n    output?: string;\n    error?: string;\n  };\n  duration_ms?: number;\n  success?: boolean;\n}\n\ninterface HookStopData {\n  session_id: string;\n  transcript_path?: string;\n  stop_hook_active?: boolean;\n  permission_mode?: string;\n  cwd?: string;\n  hook_event_name?: string;\n}\n\n// Types for Claude Code transcript JSONL entries\ninterface TranscriptUsage {\n  input_tokens?: number;\n  output_tokens?: number;\n  cache_read_input_tokens?: number;\n  cache_creation_input_tokens?: number;\n}\n\ninterface TranscriptContentPart {\n  type: \"text\" | \"thinking\" | \"tool_use\" | \"tool_result\";\n  text?: string;\n  thinking?: string;\n  id?: string;\n  name?: string;\n  input?: Record<string, unknown>;\n}\n\ninterface TranscriptMessage {\n  content?: TranscriptContentPart[];\n  usage?: TranscriptUsage;\n  model?: string;\n  stop_reason?: string | null;\n}\n\ninterface TranscriptEntry {\n  type?: string;\n  uuid?: string;\n  timestamp?: string;\n  message?: TranscriptMessage;\n  sessionId?: string;\n  parentUuid?: string;\n}\n\n// Types for Claude Code settings.json\ninterface ClaudeSettings {\n  hooks?: Record<string, unknown>;\n  [key: string]: unknown;\n}\n\n// Type for package.json version field\ninterface PackageJson {\n  version?: string;\n}\n\n// Read version from package.json\nfunction getVersion(): string {\n  try {\n    const pkgPath = path.join(__dirname, \"..\", \"package.json\");\n    const pkg = JSON.parse(fs.readFileSync(pkgPath, \"utf-8\")) as PackageJson;\n    return pkg.version || \"0.0.0\";\n  } catch {\n    return \"0.0.0\";\n  }\n}\n\nconst program = new Command();\n\nprogram\n  .name(\"claude-code-sync\")\n  .description(\"Sync Claude Code sessions to OpenSync dashboard\")\n  .version(getVersion());\n\n// ============================================================================\n// Helper Functions\n// ============================================================================\n\nfunction prompt(question: string): Promise<string> {\n  const rl = readline.createInterface({\n    input: process.stdin,\n    output: process.stdout,\n  });\n\n  return new Promise((resolve) => {\n    rl.question(question, (answer) => {\n      rl.close();\n      resolve(answer.trim());\n    });\n  });\n}\n\nfunction maskApiKey(key: string): string {\n  if (key.length <= 8) return \"****\";\n  return key.substring(0, 4) + \"****\" + key.substring(key.length - 4);\n}\n\n// ============================================================================\n// Commands\n// ============================================================================\n\nprogram\n  .command(\"login\")\n  .description(\"Configure Convex URL and API Key\")\n  .action(async () => {\n    console.log(\"\\n  Claude Code Sync - Login\\n\");\n    console.log(\"Get your API key from your OpenSync.dev Settings page, starts with osk_. Enter it here:\");\n    console.log(\"  1. Go to Settings\");\n    console.log(\"  2. Click 'Generate API Key'\");\n    console.log(\"  3. Copy the key (starts with osk_)\\n\");\n\n    const convexUrl = await prompt(\"Convex URL (e.g., https://your-project.convex.cloud): \");\n\n    if (!convexUrl) {\n      console.error(\"Error: Convex URL is required\");\n      process.exit(1);\n    }\n\n    if (!convexUrl.includes(\"convex.cloud\") && !convexUrl.includes(\"convex.site\")) {\n      console.error(\"Error: Invalid Convex URL. Must contain convex.cloud or convex.site\");\n      process.exit(1);\n    }\n\n    const apiKey = await prompt(\"API Key (osk_...): \");\n\n    if (!apiKey) {\n      console.error(\"Error: API Key is required\");\n      process.exit(1);\n    }\n\n    if (!apiKey.startsWith(\"osk_\")) {\n      console.error(\"Error: Invalid API Key. Must start with osk_\");\n      process.exit(1);\n    }\n\n    const config: Config = {\n      convexUrl,\n      apiKey,\n      autoSync: true,\n      syncToolCalls: true,\n      syncThinking: false,\n    };\n\n    // Test connection\n    console.log(\"\\nTesting connection...\");\n    const client = new SyncClient(config);\n    const connected = await client.testConnection();\n\n    if (!connected) {\n      console.error(\"Error: Could not connect to Convex backend\");\n      console.error(\"   Check your URL and try again\");\n      process.exit(1);\n    }\n\n    // Save config\n    saveConfig(config);\n    console.log(\"\\nConfiguration saved!\");\n    console.log(`   URL: ${convexUrl}`);\n    console.log(`   Key: ${maskApiKey(apiKey)}`);\n    console.log(\"\\nNext step: Run the setup command to configure Claude Code hooks:\\n\");\n    console.log(\"   claude-code-sync setup\\n\");\n  });\n\nprogram\n  .command(\"logout\")\n  .description(\"Clear stored credentials\")\n  .action(() => {\n    clearConfig();\n    console.log(\"Credentials cleared\");\n  });\n\nprogram\n  .command(\"status\")\n  .description(\"Show connection status\")\n  .action(async () => {\n    const config = loadConfig();\n\n    console.log(\"\\n  Claude Code Sync - Status\\n\");\n\n    if (!config) {\n      console.log(\"Not configured\");\n      console.log(\"   Run 'claude-code-sync login' to set up\\n\");\n      process.exit(1);\n    }\n\n    console.log(\"Configuration:\");\n    console.log(`  Convex URL: ${config.convexUrl}`);\n    console.log(`  API Key:    ${maskApiKey(config.apiKey)}`);\n    console.log(`  Auto Sync:  ${config.autoSync !== false ? \"enabled\" : \"disabled\"}`);\n    console.log(`  Tool Calls: ${config.syncToolCalls !== false ? \"enabled\" : \"disabled\"}`);\n    console.log(`  Thinking:   ${config.syncThinking ? \"enabled\" : \"disabled\"}`);\n\n    console.log(\"\\nTesting connection...\");\n    const client = new SyncClient(config);\n    const connected = await client.testConnection();\n\n    if (connected) {\n      console.log(\"Connected to Convex backend\\n\");\n    } else {\n      console.log(\"Error: Could not connect to Convex backend\\n\");\n      process.exit(1);\n    }\n  });\n\nprogram\n  .command(\"config\")\n  .description(\"Show current configuration\")\n  .option(\"--json\", \"Output as JSON\")\n  .action((options: { json?: boolean }) => {\n    const config = loadConfig();\n\n    if (!config) {\n      if (options.json) {\n        console.log(JSON.stringify({ configured: false }));\n      } else {\n        console.log(\"Not configured. Run 'claude-code-sync login' to set up.\");\n      }\n      return;\n    }\n\n    if (options.json) {\n      console.log(\n        JSON.stringify(\n          {\n            configured: true,\n            convexUrl: config.convexUrl,\n            apiKey: maskApiKey(config.apiKey),\n            autoSync: config.autoSync !== false,\n            syncToolCalls: config.syncToolCalls !== false,\n            syncThinking: config.syncThinking === true,\n          },\n          null,\n          2\n        )\n      );\n    } else {\n      console.log(\"\\n  Current Configuration\\n\");\n      console.log(`Convex URL:  ${config.convexUrl}`);\n      console.log(`API Key:     ${maskApiKey(config.apiKey)}`);\n      console.log(`Auto Sync:   ${config.autoSync !== false}`);\n      console.log(`Tool Calls:  ${config.syncToolCalls !== false}`);\n      console.log(`Thinking:    ${config.syncThinking === true}`);\n      console.log(`\\nConfig file: ~/.config/claude-code-sync/config.json\\n`);\n    }\n  });\n\nprogram\n  .command(\"set <key> <value>\")\n  .description(\"Set a configuration value\")\n  .action((key: string, value: string) => {\n    const config = loadConfig();\n\n    if (!config) {\n      console.error(\"Not configured. Run 'claude-code-sync login' first.\");\n      process.exit(1);\n    }\n\n    const validKeys = [\"autoSync\", \"syncToolCalls\", \"syncThinking\"];\n    if (!validKeys.includes(key)) {\n      console.error(`Invalid key. Valid keys: ${validKeys.join(\", \")}`);\n      process.exit(1);\n    }\n\n    const boolValue = value === \"true\" || value === \"1\" || value === \"yes\";\n    \n    // Type-safe config update\n    if (key === \"autoSync\") {\n      config.autoSync = boolValue;\n    } else if (key === \"syncToolCalls\") {\n      config.syncToolCalls = boolValue;\n    } else if (key === \"syncThinking\") {\n      config.syncThinking = boolValue;\n    }\n\n    saveConfig(config);\n    console.log(`Set ${key} = ${boolValue}`);\n  });\n\n// ============================================================================\n// Setup Command (configures Claude Code hooks)\n// ============================================================================\n\n// Claude Code hooks configuration\nconst CLAUDE_HOOKS_CONFIG = {\n  hooks: {\n    SessionStart: [\n      {\n        hooks: [\n          {\n            type: \"command\",\n            command: \"claude-code-sync hook SessionStart\",\n          },\n        ],\n      },\n    ],\n    SessionEnd: [\n      {\n        hooks: [\n          {\n            type: \"command\",\n            command: \"claude-code-sync hook SessionEnd\",\n          },\n        ],\n      },\n    ],\n    UserPromptSubmit: [\n      {\n        hooks: [\n          {\n            type: \"command\",\n            command: \"claude-code-sync hook UserPromptSubmit\",\n          },\n        ],\n      },\n    ],\n    PostToolUse: [\n      {\n        matcher: \"*\",\n        hooks: [\n          {\n            type: \"command\",\n            command: \"claude-code-sync hook PostToolUse\",\n          },\n        ],\n      },\n    ],\n    Stop: [\n      {\n        matcher: \"*\",\n        hooks: [\n          {\n            type: \"command\",\n            command: \"claude-code-sync hook Stop\",\n          },\n        ],\n      },\n    ],\n  },\n};\n\nprogram\n  .command(\"setup\")\n  .description(\"Add hooks to Claude Code settings (configures ~/.claude/settings.json)\")\n  .option(\"--force\", \"Overwrite existing hooks configuration\")\n  .action(async (options: { force?: boolean }) => {\n    const claudeDir = path.join(process.env.HOME || \"~\", \".claude\");\n    const settingsPath = path.join(claudeDir, \"settings.json\");\n\n    console.log(\"\\n  Claude Code Sync - Setup\\n\");\n\n    // Check if plugin credentials are configured\n    const config = loadConfig();\n    if (!config) {\n      console.log(\"Warning: Plugin not configured yet.\");\n      console.log(\"   Run 'claude-code-sync login' first to set up credentials.\\n\");\n    }\n\n    // Create .claude directory if needed\n    if (!fs.existsSync(claudeDir)) {\n      fs.mkdirSync(claudeDir, { recursive: true });\n      console.log(\"Created ~/.claude directory\");\n    }\n\n    // Check for existing settings\n    let existingSettings: ClaudeSettings = {};\n    let hasExistingHooks = false;\n\n    if (fs.existsSync(settingsPath)) {\n      try {\n        const content = fs.readFileSync(settingsPath, \"utf-8\");\n        existingSettings = JSON.parse(content) as ClaudeSettings;\n        hasExistingHooks = !!existingSettings.hooks;\n        console.log(\"Found existing settings.json\");\n      } catch {\n        console.log(\"Warning: Could not parse existing settings.json, will create new one\");\n      }\n    }\n\n    // Handle existing hooks\n    if (hasExistingHooks && !options.force) {\n      console.log(\"\\nExisting hooks configuration found.\");\n      console.log(\"   Use --force to overwrite, or manually merge the hooks.\\n\");\n      console.log(\"To manually add, include these hooks in your settings.json:\");\n      console.log(JSON.stringify(CLAUDE_HOOKS_CONFIG, null, 2));\n      process.exit(1);\n    }\n\n    // Merge settings\n    const newSettings = {\n      ...existingSettings,\n      ...CLAUDE_HOOKS_CONFIG,\n    };\n\n    // Write settings\n    try {\n      fs.writeFileSync(settingsPath, JSON.stringify(newSettings, null, 2));\n      console.log(\"\\nClaude Code hooks configured!\");\n      console.log(`   Settings file: ${settingsPath}`);\n      console.log(\"\\nSetup complete. Sessions will sync automatically.\\n\");\n    } catch (error) {\n      console.error(\"Error writing settings:\", error);\n      process.exit(1);\n    }\n  });\n\nprogram\n  .command(\"verify\")\n  .description(\"Verify credentials and Claude Code configuration\")\n  .action(async () => {\n    console.log(\"\\n  OpenSync Setup Verification\\n\");\n\n    // Check credentials\n    const config = loadConfig();\n    if (config) {\n      console.log(\"Credentials: OK\");\n      console.log(`   Convex URL: ${config.convexUrl}`);\n      console.log(`   API Key: ${maskApiKey(config.apiKey)}`);\n    } else {\n      console.log(\"Credentials: NOT CONFIGURED\");\n      console.log(\"   Run 'claude-code-sync login' to set up\");\n    }\n\n    // Check Claude Code config\n    const settingsPath = path.join(process.env.HOME || \"~\", \".claude\", \"settings.json\");\n    let hooksConfigured = false;\n\n    if (fs.existsSync(settingsPath)) {\n      try {\n        const content = fs.readFileSync(settingsPath, \"utf-8\");\n        const settings = JSON.parse(content) as ClaudeSettings;\n        hooksConfigured = !!settings.hooks?.SessionStart;\n      } catch {\n        // Ignore parse errors\n      }\n    }\n\n    console.log(\"\");\n    if (hooksConfigured) {\n      console.log(\"Claude Code Config: OK\");\n      console.log(`   Config file: ${settingsPath}`);\n      console.log(\"   Hooks registered: claude-code-sync\");\n    } else {\n      console.log(\"Claude Code Config: NOT CONFIGURED\");\n      console.log(\"   Run 'claude-code-sync setup' to configure hooks\");\n    }\n\n    // Test connection if credentials exist\n    if (config) {\n      console.log(\"\\nTesting connection...\");\n      const client = new SyncClient(config);\n      const connected = await client.testConnection();\n      if (connected) {\n        console.log(\"Connection: OK\\n\");\n      } else {\n        console.log(\"Connection: FAILED\\n\");\n        process.exit(1);\n      }\n    }\n\n    if (config && hooksConfigured) {\n      console.log(\"Ready! Start Claude Code and sessions will sync automatically.\\n\");\n    } else {\n      console.log(\"\");\n      process.exit(1);\n    }\n  });\n\n// ============================================================================\n// Sync Test Command (test connectivity)\n// ============================================================================\n\nprogram\n  .command(\"synctest\")\n  .description(\"Test connectivity and create a test session\")\n  .action(async () => {\n    const config = loadConfig();\n\n    console.log(\"\\n  Claude Code Sync - Connection Test\\n\");\n\n    if (!config) {\n      console.log(\"Not configured\");\n      console.log(\"   Run 'claude-code-sync login' to set up\\n\");\n      process.exit(1);\n    }\n\n    console.log(\"Configuration:\");\n    console.log(`  Convex URL: ${config.convexUrl}`);\n    console.log(`  API Key:    ${maskApiKey(config.apiKey)}`);\n\n    console.log(\"\\nTesting connection...\");\n    const client = new SyncClient(config);\n    const connected = await client.testConnection();\n\n    if (connected) {\n      console.log(\"Connection: OK\");\n      \n      // Create a test session to verify full sync works\n      console.log(\"\\nCreating test session...\");\n      try {\n        const testSession = {\n          sessionId: `test-${Date.now()}`,\n          source: \"claude-code\" as const,\n          title: \"Connection Test\",\n          projectName: \"synctest\",\n          startedAt: new Date().toISOString(),\n          endedAt: new Date().toISOString(),\n        };\n        await client.syncSession(testSession);\n        console.log(\"Test session created successfully\");\n        console.log(\"\\nSync test passed. Ready to sync Claude Code sessions.\\n\");\n      } catch (error) {\n        console.log(`Test session failed: ${error}`);\n        console.log(\"\\nConnection works but sync may have issues.\\n\");\n        process.exit(1);\n      }\n    } else {\n      console.log(\"Connection: FAILED\");\n      console.log(\"\\nCheck your Convex URL and API key.\\n\");\n      process.exit(1);\n    }\n  });\n\n// ============================================================================\n// Hook Command (for Claude Code integration)\n// ============================================================================\n\n// Track session state for title generation (first user prompt)\nconst SESSION_STATE_FILE = path.join(\n  process.env.HOME || \"~\",\n  \".config\",\n  \"claude-code-sync\",\n  \"session-state.json\"\n);\n\ninterface SessionState {\n  [sessionId: string]: {\n    model?: string;\n    firstPrompt?: string;\n    tokenUsage?: { input: number; output: number };\n    messageCount?: number;\n    syncedMessageUuids?: Set<string> | string[];\n  };\n}\n\nfunction loadSessionState(): SessionState {\n  try {\n    if (fs.existsSync(SESSION_STATE_FILE)) {\n      const data = JSON.parse(fs.readFileSync(SESSION_STATE_FILE, \"utf-8\")) as SessionState;\n      // Convert arrays back to Sets for syncedMessageUuids\n      for (const sessionId of Object.keys(data)) {\n        const session = data[sessionId];\n        if (session.syncedMessageUuids && Array.isArray(session.syncedMessageUuids)) {\n          session.syncedMessageUuids = new Set(session.syncedMessageUuids);\n        }\n      }\n      return data;\n    }\n  } catch {\n    // Ignore errors\n  }\n  return {};\n}\n\nfunction saveSessionState(state: SessionState): void {\n  try {\n    const dir = path.dirname(SESSION_STATE_FILE);\n    if (!fs.existsSync(dir)) {\n      fs.mkdirSync(dir, { recursive: true });\n    }\n    // Convert Sets to arrays for JSON serialization\n    const serializable: Record<string, unknown> = {};\n    for (const sessionId of Object.keys(state)) {\n      const session = state[sessionId];\n      serializable[sessionId] = {\n        ...session,\n        syncedMessageUuids: session.syncedMessageUuids instanceof Set\n          ? Array.from(session.syncedMessageUuids)\n          : session.syncedMessageUuids,\n      };\n    }\n    fs.writeFileSync(SESSION_STATE_FILE, JSON.stringify(serializable, null, 2));\n  } catch {\n    // Ignore errors\n  }\n}\n\nfunction generateTitle(prompt: string): string {\n  // Use first 80 chars of first prompt as title, trim at word boundary\n  const trimmed = prompt.slice(0, 80).trim();\n  if (prompt.length > 80) {\n    const lastSpace = trimmed.lastIndexOf(\" \");\n    if (lastSpace > 40) {\n      return trimmed.slice(0, lastSpace) + \"...\";\n    }\n    return trimmed + \"...\";\n  }\n  return trimmed;\n}\n\n// Parse transcript file to extract assistant messages and token usage\ninterface ParsedTranscript {\n  assistantMessages: Array<{\n    uuid: string;\n    text: string;\n    timestamp: string;\n    model?: string;\n  }>;\n  tokenUsage: {\n    input: number;\n    output: number;\n  };\n}\n\nfunction parseTranscriptFile(transcriptPath: string): ParsedTranscript {\n  const result: ParsedTranscript = {\n    assistantMessages: [],\n    tokenUsage: { input: 0, output: 0 },\n  };\n\n  try {\n    if (!fs.existsSync(transcriptPath)) {\n      return result;\n    }\n\n    const content = fs.readFileSync(transcriptPath, \"utf-8\");\n    const lines = content.trim().split(\"\\n\");\n    const seenUuids = new Set<string>();\n\n    for (const line of lines) {\n      if (!line.trim()) continue;\n\n      try {\n        const entry = JSON.parse(line) as TranscriptEntry;\n\n        // Only process assistant messages\n        if (entry.type === \"assistant\" && entry.message) {\n          const msg = entry.message;\n          const uuid = entry.uuid || \"\";\n\n          // Skip if we've already seen this UUID (dedup)\n          if (uuid && seenUuids.has(uuid)) continue;\n          if (uuid) seenUuids.add(uuid);\n\n          // Extract text content from message\n          if (msg.content && Array.isArray(msg.content)) {\n            for (const part of msg.content) {\n              if (part.type === \"text\" && part.text) {\n                result.assistantMessages.push({\n                  uuid,\n                  text: part.text,\n                  timestamp: entry.timestamp || new Date().toISOString(),\n                  model: msg.model,\n                });\n              }\n            }\n          }\n\n          // Accumulate token usage from all assistant messages\n          if (msg.usage) {\n            result.tokenUsage.input +=\n              (msg.usage.input_tokens || 0) +\n              (msg.usage.cache_read_input_tokens || 0) +\n              (msg.usage.cache_creation_input_tokens || 0);\n            result.tokenUsage.output += msg.usage.output_tokens || 0;\n          }\n        }\n      } catch {\n        // Skip malformed lines\n      }\n    }\n  } catch {\n    // Return empty result on error\n  }\n\n  return result;\n}\n\nprogram\n  .command(\"hook <event>\")\n  .description(\"Handle Claude Code hook events (reads stdin)\")\n  .action(async (event: string) => {\n    const config = loadConfig();\n\n    if (!config) {\n      // Exit silently if not configured (don't block Claude Code)\n      process.exit(0);\n    }\n\n    if (config.autoSync === false) {\n      process.exit(0);\n    }\n\n    // Read JSON input from stdin\n    let input = \"\";\n    for await (const chunk of process.stdin) {\n      input += chunk;\n    }\n\n    if (!input.trim()) {\n      process.exit(0);\n    }\n\n    try {\n      const client = new SyncClient(config);\n      const sessionState = loadSessionState();\n\n      switch (event) {\n        case \"SessionStart\": {\n          const data = JSON.parse(input) as HookSessionStartData;\n          \n          // Initialize session state\n          sessionState[data.session_id] = {\n            model: data.model,\n            tokenUsage: { input: 0, output: 0 },\n            messageCount: 0,\n          };\n          saveSessionState(sessionState);\n          \n          const session: SessionData = {\n            sessionId: data.session_id,\n            source: \"claude-code\",\n            cwd: data.cwd,\n            model: data.model,\n            permissionMode: data.permission_mode,\n            thinkingEnabled: data.thinking_enabled,\n            mcpServers: data.mcp_servers,\n            startType: data.source === \"startup\" ? \"new\" : (data.source as SessionData[\"startType\"]),\n            startedAt: new Date().toISOString(),\n            projectPath: data.cwd,\n            projectName: data.cwd ? data.cwd.split(\"/\").pop() : undefined,\n          };\n          await client.syncSession(session);\n          break;\n        }\n\n        case \"SessionEnd\": {\n          const data = JSON.parse(input) as HookSessionEndData;\n          const state = sessionState[data.session_id] || {};\n          \n          // Calculate final token usage\n          const finalTokenUsage = data.total_token_usage || state.tokenUsage;\n          \n          const session: SessionData = {\n            sessionId: data.session_id,\n            source: \"claude-code\",\n            title: state.firstPrompt ? generateTitle(state.firstPrompt) : undefined,\n            endReason: data.reason,\n            messageCount: data.message_count || state.messageCount,\n            toolCallCount: data.tool_call_count,\n            tokenUsage: finalTokenUsage,\n            costEstimate: data.cost_estimate,\n            endedAt: new Date().toISOString(),\n          };\n          await client.syncSession(session);\n          \n          // Clean up session state\n          delete sessionState[data.session_id];\n          saveSessionState(sessionState);\n          break;\n        }\n\n        case \"UserPromptSubmit\": {\n          const data = JSON.parse(input) as HookUserPromptData;\n          const state = sessionState[data.session_id] || {};\n          \n          // Track first prompt for title generation\n          if (!state.firstPrompt) {\n            state.firstPrompt = data.prompt;\n            sessionState[data.session_id] = state;\n            saveSessionState(sessionState);\n            \n            // Update session with title\n            const session: SessionData = {\n              sessionId: data.session_id,\n              source: \"claude-code\",\n              title: generateTitle(data.prompt),\n            };\n            await client.syncSession(session);\n          }\n          \n          // Increment message count\n          state.messageCount = (state.messageCount || 0) + 1;\n          sessionState[data.session_id] = state;\n          saveSessionState(sessionState);\n          \n          const message: MessageData = {\n            sessionId: data.session_id,\n            messageId: `${data.session_id}-user-${Date.now()}`,\n            source: \"claude-code\",\n            role: \"user\",\n            content: data.prompt,\n            timestamp: data.timestamp || new Date().toISOString(),\n          };\n          await client.syncMessage(message);\n          break;\n        }\n\n        case \"PostToolUse\": {\n          if (!config.syncToolCalls) break;\n          const data = JSON.parse(input) as HookToolUseData;\n          const state = sessionState[data.session_id] || {};\n          \n          const message: MessageData = {\n            sessionId: data.session_id,\n            messageId: data.tool_use_id || `${data.session_id}-tool-${Date.now()}`,\n            source: \"claude-code\",\n            role: \"assistant\",\n            toolName: data.tool_name,\n            toolArgs: data.tool_input,\n            toolResult: data.tool_result?.output || data.tool_result?.error,\n            durationMs: data.duration_ms,\n            timestamp: new Date().toISOString(),\n          };\n          await client.syncMessage(message);\n          break;\n        }\n\n        case \"Stop\": {\n          // Stop event provides transcript_path - we read it to get messages and tokens\n          const data = JSON.parse(input) as HookStopData;\n          const state = sessionState[data.session_id] || {};\n          \n          // Parse transcript file to extract assistant messages and token usage\n          if (data.transcript_path) {\n            const transcript = parseTranscriptFile(data.transcript_path);\n            \n            // Update token usage from transcript\n            if (transcript.tokenUsage.input > 0 || transcript.tokenUsage.output > 0) {\n              state.tokenUsage = transcript.tokenUsage;\n            }\n            \n            // Track which messages we've already synced to avoid duplicates\n            const syncedMessages = state.syncedMessageUuids instanceof Set\n              ? state.syncedMessageUuids\n              : new Set<string>(Array.isArray(state.syncedMessageUuids) ? state.syncedMessageUuids : []);\n            \n            // Sync new assistant messages\n            for (const msg of transcript.assistantMessages) {\n              // Skip if we've already synced this message\n              if (msg.uuid && syncedMessages.has(msg.uuid)) continue;\n              if (msg.uuid) syncedMessages.add(msg.uuid);\n              \n              // Increment message count\n              state.messageCount = (state.messageCount || 0) + 1;\n              \n              const message: MessageData = {\n                sessionId: data.session_id,\n                messageId: msg.uuid || `${data.session_id}-assistant-${Date.now()}`,\n                source: \"claude-code\",\n                role: \"assistant\",\n                content: msg.text,\n                model: msg.model,\n                timestamp: msg.timestamp,\n              };\n              await client.syncMessage(message);\n            }\n            \n            // Store synced message UUIDs (convert Set to array for JSON)\n            state.syncedMessageUuids = syncedMessages;\n          }\n          \n          sessionState[data.session_id] = state;\n          saveSessionState(sessionState);\n          \n          // Update session with token usage from transcript\n          const session: SessionData = {\n            sessionId: data.session_id,\n            source: \"claude-code\",\n            tokenUsage: state.tokenUsage,\n            messageCount: state.messageCount,\n          };\n          await client.syncSession(session);\n          break;\n        }\n\n        default:\n          // Unknown event, ignore\n          break;\n      }\n\n      process.exit(0);\n    } catch (error) {\n      // Log to stderr but don't block Claude Code\n      console.error(`[claude-code-sync] Error: ${error}`);\n      process.exit(0);\n    }\n  });\n\n// Parse and run\nprogram.parse();\n"]}
package/dist/index.d.ts CHANGED
@@ -52,6 +52,7 @@ export interface MessageData {
52
52
  durationMs?: number;
53
53
  tokenCount?: number;
54
54
  timestamp?: string;
55
+ model?: string;
55
56
  }
56
57
  export interface ToolUseData {
57
58
  sessionId: string;
package/dist/index.js CHANGED
@@ -194,6 +194,10 @@ class SyncClient {
194
194
  if (message.tokenCount !== undefined) {
195
195
  payload.promptTokens = message.tokenCount;
196
196
  }
197
+ // Include model if available
198
+ if (message.model) {
199
+ payload.model = message.model;
200
+ }
197
201
  // Tool use info stored in parts
198
202
  if (message.toolName) {
199
203
  payload.parts = [
@@ -423,4 +427,4 @@ function createPlugin() {
423
427
  }
424
428
  // Default export for Claude Code plugin system
425
429
  exports.default = createPlugin;
426
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Claude Code Sync Plugin\n *\n * Syncs Claude Code sessions to OpenSync dashboard.\n * Uses API Key authentication (no browser OAuth required).\n *\n * Install: npm install -g claude-code-sync\n * Configure: claude-code-sync login\n */\n\nimport * as fs from \"fs\";\nimport * as path from \"path\";\nimport * as os from \"os\";\n\n// ============================================================================\n// Types\n// ============================================================================\n\nexport interface Config {\n  convexUrl: string;\n  apiKey: string;\n  autoSync?: boolean;\n  syncToolCalls?: boolean;\n  syncThinking?: boolean;\n}\n\nexport interface SessionData {\n  sessionId: string;\n  source: \"claude-code\";\n  title?: string;\n  projectPath?: string;\n  projectName?: string;\n  cwd?: string;\n  gitBranch?: string;\n  gitRepo?: string;\n  model?: string;\n  startType?: \"new\" | \"resume\" | \"continue\";\n  endReason?: \"user_stop\" | \"max_turns\" | \"error\" | \"completed\";\n  thinkingEnabled?: boolean;\n  permissionMode?: string;\n  mcpServers?: string[];\n  messageCount?: number;\n  toolCallCount?: number;\n  tokenUsage?: {\n    input: number;\n    output: number;\n  };\n  costEstimate?: number;\n  startedAt?: string;\n  endedAt?: string;\n}\n\nexport interface MessageData {\n  sessionId: string;\n  messageId: string;\n  source: \"claude-code\";\n  role: \"user\" | \"assistant\" | \"system\";\n  content?: string;\n  thinkingContent?: string;\n  toolName?: string;\n  toolArgs?: Record<string, unknown>;\n  toolResult?: string;\n  durationMs?: number;\n  tokenCount?: number;\n  timestamp?: string;\n}\n\nexport interface ToolUseData {\n  sessionId: string;\n  toolName: string;\n  toolArgs?: Record<string, unknown>;\n  result?: string;\n  success?: boolean;\n  durationMs?: number;\n  timestamp?: string;\n}\n\n// Claude Code Hook Types\nexport interface ClaudeCodeHooks {\n  SessionStart?: (data: SessionStartEvent) => void | Promise<void>;\n  UserPromptSubmit?: (data: UserPromptEvent) => void | Promise<void>;\n  PostToolUse?: (data: ToolUseEvent) => void | Promise<void>;\n  Stop?: (data: StopEvent) => void | Promise<void>;\n  SessionEnd?: (data: SessionEndEvent) => void | Promise<void>;\n}\n\nexport interface SessionStartEvent {\n  sessionId: string;\n  cwd: string;\n  model: string;\n  startType: \"new\" | \"resume\" | \"continue\";\n  thinkingEnabled?: boolean;\n  permissionMode?: string;\n  mcpServers?: string[];\n}\n\nexport interface UserPromptEvent {\n  sessionId: string;\n  prompt: string;\n  timestamp: string;\n}\n\nexport interface ToolUseEvent {\n  sessionId: string;\n  toolName: string;\n  args: Record<string, unknown>;\n  result: string;\n  success: boolean;\n  durationMs: number;\n}\n\nexport interface StopEvent {\n  sessionId: string;\n  response: string;\n  tokenUsage: {\n    input: number;\n    output: number;\n  };\n  durationMs: number;\n}\n\nexport interface SessionEndEvent {\n  sessionId: string;\n  endReason: \"user_stop\" | \"max_turns\" | \"error\" | \"completed\";\n  messageCount: number;\n  toolCallCount: number;\n  totalTokenUsage: {\n    input: number;\n    output: number;\n  };\n  costEstimate: number;\n}\n\n// ============================================================================\n// Configuration\n// ============================================================================\n\nconst CONFIG_DIR = path.join(os.homedir(), \".config\", \"claude-code-sync\");\nconst CONFIG_FILE = path.join(CONFIG_DIR, \"config.json\");\n\nexport function loadConfig(): Config | null {\n  // Check environment variables first\n  const envUrl = process.env.CLAUDE_SYNC_CONVEX_URL;\n  const envKey = process.env.CLAUDE_SYNC_API_KEY;\n\n  if (envUrl && envKey) {\n    return {\n      convexUrl: normalizeConvexUrl(envUrl),\n      apiKey: envKey,\n      autoSync: process.env.CLAUDE_SYNC_AUTO_SYNC !== \"false\",\n      syncToolCalls: process.env.CLAUDE_SYNC_TOOL_CALLS !== \"false\",\n      syncThinking: process.env.CLAUDE_SYNC_THINKING === \"true\",\n    };\n  }\n\n  // Fall back to config file\n  try {\n    if (fs.existsSync(CONFIG_FILE)) {\n      const data = fs.readFileSync(CONFIG_FILE, \"utf-8\");\n      const config = JSON.parse(data) as Config;\n      config.convexUrl = normalizeConvexUrl(config.convexUrl);\n      return config;\n    }\n  } catch (error) {\n    console.error(\"Error loading config:\", error);\n  }\n\n  return null;\n}\n\nexport function saveConfig(config: Config): void {\n  try {\n    if (!fs.existsSync(CONFIG_DIR)) {\n      fs.mkdirSync(CONFIG_DIR, { recursive: true });\n    }\n    fs.writeFileSync(CONFIG_FILE, JSON.stringify(config, null, 2));\n  } catch (error) {\n    console.error(\"Error saving config:\", error);\n    throw error;\n  }\n}\n\nexport function clearConfig(): void {\n  try {\n    if (fs.existsSync(CONFIG_FILE)) {\n      fs.unlinkSync(CONFIG_FILE);\n    }\n  } catch (error) {\n    console.error(\"Error clearing config:\", error);\n  }\n}\n\nfunction normalizeConvexUrl(url: string): string {\n  // Convert .convex.cloud to .convex.site for API calls\n  return url.replace(\".convex.cloud\", \".convex.site\");\n}\n\n// ============================================================================\n// Sync Client\n// ============================================================================\n\nexport class SyncClient {\n  private config: Config;\n  private siteUrl: string;\n  private sessionCache: Map<string, Partial<SessionData>> = new Map();\n\n  constructor(config: Config) {\n    this.config = config;\n    // Normalize URL to .convex.site for HTTP endpoints\n    // Supports both .convex.cloud and .convex.site input URLs\n    this.siteUrl = config.convexUrl.replace(\".convex.cloud\", \".convex.site\");\n  }\n\n  private async request(endpoint: string, data: unknown): Promise<unknown> {\n    const url = `${this.siteUrl}${endpoint}`;\n\n    const response = await fetch(url, {\n      method: \"POST\",\n      headers: {\n        \"Content-Type\": \"application/json\",\n        Authorization: `Bearer ${this.config.apiKey}`,\n      },\n      body: JSON.stringify(data),\n    });\n\n    if (!response.ok) {\n      const text = await response.text();\n      throw new Error(`Sync failed: ${response.status} - ${text}`);\n    }\n\n    return response.json();\n  }\n\n  // Transform session data to backend format\n  private transformSession(session: SessionData): Record<string, unknown> {\n    const payload: Record<string, unknown> = {\n      externalId: session.sessionId,\n      source: session.source,\n    };\n    \n    // Only include fields that are defined\n    if (session.title !== undefined) payload.title = session.title;\n    if (session.projectPath || session.cwd) payload.projectPath = session.projectPath || session.cwd;\n    if (session.projectName) payload.projectName = session.projectName;\n    if (session.model) payload.model = session.model;\n    if (session.tokenUsage?.input !== undefined) payload.promptTokens = session.tokenUsage.input;\n    if (session.tokenUsage?.output !== undefined) payload.completionTokens = session.tokenUsage.output;\n    if (session.costEstimate !== undefined) payload.cost = session.costEstimate;\n    if (session.messageCount !== undefined) payload.messageCount = session.messageCount;\n    if (session.toolCallCount !== undefined) payload.toolCallCount = session.toolCallCount;\n    if (session.endReason) payload.endReason = session.endReason;\n    \n    // Calculate duration if both timestamps exist\n    if (session.endedAt && session.startedAt) {\n      payload.durationMs = new Date(session.endedAt).getTime() - new Date(session.startedAt).getTime();\n    }\n    \n    return payload;\n  }\n\n  // Transform message data to backend format\n  private transformMessage(message: MessageData): Record<string, unknown> {\n    const payload: Record<string, unknown> = {\n      sessionExternalId: message.sessionId,\n      externalId: message.messageId,\n      role: message.role,\n      source: message.source,\n    };\n    \n    // Set textContent for user prompts and assistant responses\n    if (message.content) {\n      payload.textContent = message.content;\n    } else if (message.toolResult && !message.toolName) {\n      // Only use toolResult as textContent if no toolName (fallback)\n      payload.textContent = message.toolResult;\n    }\n    \n    // Include duration if available\n    if (message.durationMs !== undefined) {\n      payload.durationMs = message.durationMs;\n    }\n    \n    // Include token count if available\n    if (message.tokenCount !== undefined) {\n      payload.promptTokens = message.tokenCount;\n    }\n    \n    // Tool use info stored in parts\n    if (message.toolName) {\n      payload.parts = [\n        {\n          type: \"tool-call\",\n          content: {\n            toolName: message.toolName,\n            args: message.toolArgs,\n          },\n        },\n      ];\n      // Add tool result as separate part\n      if (message.toolResult) {\n        (payload.parts as Array<{ type: string; content: unknown }>).push({\n          type: \"tool-result\",\n          content: message.toolResult,\n        });\n      }\n    }\n    \n    return payload;\n  }\n\n  async syncSession(session: SessionData): Promise<void> {\n    try {\n      const payload = this.transformSession(session);\n      await this.request(\"/sync/session\", payload);\n    } catch (error) {\n      console.error(\"Failed to sync session:\", error);\n      throw error;\n    }\n  }\n\n  async syncMessage(message: MessageData): Promise<void> {\n    try {\n      const payload = this.transformMessage(message);\n      await this.request(\"/sync/message\", payload);\n    } catch (error) {\n      console.error(\"Failed to sync message:\", error);\n      throw error;\n    }\n  }\n\n  async syncBatch(\n    sessions: SessionData[],\n    messages: MessageData[]\n  ): Promise<void> {\n    try {\n      const transformedSessions = sessions.map((s) => this.transformSession(s));\n      const transformedMessages = messages.map((m) => this.transformMessage(m));\n      await this.request(\"/sync/batch\", {\n        sessions: transformedSessions,\n        messages: transformedMessages,\n      });\n    } catch (error) {\n      console.error(\"Failed to sync batch:\", error);\n      throw error;\n    }\n  }\n\n  async testConnection(): Promise<boolean> {\n    try {\n      const url = `${this.siteUrl}/health`;\n      const response = await fetch(url);\n      return response.ok;\n    } catch {\n      return false;\n    }\n  }\n\n  // Session state management\n  getSessionState(sessionId: string): Partial<SessionData> {\n    return this.sessionCache.get(sessionId) || {};\n  }\n\n  updateSessionState(\n    sessionId: string,\n    updates: Partial<SessionData>\n  ): void {\n    const current = this.sessionCache.get(sessionId) || {};\n    this.sessionCache.set(sessionId, { ...current, ...updates });\n  }\n\n  clearSessionState(sessionId: string): void {\n    this.sessionCache.delete(sessionId);\n  }\n}\n\n// ============================================================================\n// Plugin Export\n// ============================================================================\n\n/**\n * Claude Code Plugin Entry Point\n *\n * This function is called by Claude Code to register the plugin.\n * It returns hook handlers that fire at key points in the session lifecycle.\n */\nexport function createPlugin(): ClaudeCodeHooks | null {\n  const config = loadConfig();\n\n  if (!config) {\n    console.log(\n      \"[claude-code-sync] Not configured. Run 'claude-code-sync login' to set up.\"\n    );\n    return null;\n  }\n\n  if (config.autoSync === false) {\n    console.log(\"[claude-code-sync] Auto-sync disabled in config.\");\n    return null;\n  }\n\n  const client = new SyncClient(config);\n  let messageCounter = 0;\n  let toolCallCounter = 0;\n\n  console.log(\"[claude-code-sync] Plugin loaded. Sessions will sync to OpenSync.\");\n\n  return {\n    /**\n     * Called when a new session starts\n     */\n    SessionStart: async (event: SessionStartEvent) => {\n      messageCounter = 0;\n      toolCallCounter = 0;\n\n      const session: SessionData = {\n        sessionId: event.sessionId,\n        source: \"claude-code\",\n        cwd: event.cwd,\n        model: event.model,\n        startType: event.startType,\n        thinkingEnabled: event.thinkingEnabled,\n        permissionMode: event.permissionMode,\n        mcpServers: event.mcpServers,\n        startedAt: new Date().toISOString(),\n      };\n\n      // Extract project info from cwd\n      if (event.cwd) {\n        session.projectPath = event.cwd;\n        session.projectName = path.basename(event.cwd);\n\n        // Try to get git info\n        try {\n          const gitDir = path.join(event.cwd, \".git\");\n          if (fs.existsSync(gitDir)) {\n            const headFile = path.join(gitDir, \"HEAD\");\n            if (fs.existsSync(headFile)) {\n              const head = fs.readFileSync(headFile, \"utf-8\").trim();\n              if (head.startsWith(\"ref: refs/heads/\")) {\n                session.gitBranch = head.replace(\"ref: refs/heads/\", \"\");\n              }\n            }\n          }\n        } catch {\n          // Ignore git errors\n        }\n      }\n\n      client.updateSessionState(event.sessionId, session);\n      await client.syncSession(session);\n    },\n\n    /**\n     * Called when user submits a prompt\n     */\n    UserPromptSubmit: async (event: UserPromptEvent) => {\n      messageCounter++;\n\n      const message: MessageData = {\n        sessionId: event.sessionId,\n        messageId: `${event.sessionId}-msg-${messageCounter}`,\n        source: \"claude-code\",\n        role: \"user\",\n        content: event.prompt,\n        timestamp: event.timestamp || new Date().toISOString(),\n      };\n\n      await client.syncMessage(message);\n    },\n\n    /**\n     * Called after each tool use\n     */\n    PostToolUse: async (event: ToolUseEvent) => {\n      if (!config.syncToolCalls) return;\n\n      toolCallCounter++;\n      messageCounter++;\n\n      const message: MessageData = {\n        sessionId: event.sessionId,\n        messageId: `${event.sessionId}-tool-${toolCallCounter}`,\n        source: \"claude-code\",\n        role: \"assistant\",\n        toolName: event.toolName,\n        toolArgs: event.args,\n        toolResult: event.result,\n        durationMs: event.durationMs,\n        timestamp: new Date().toISOString(),\n      };\n\n      await client.syncMessage(message);\n    },\n\n    /**\n     * Called when Claude stops responding\n     */\n    Stop: async (event: StopEvent) => {\n      messageCounter++;\n\n      const message: MessageData = {\n        sessionId: event.sessionId,\n        messageId: `${event.sessionId}-msg-${messageCounter}`,\n        source: \"claude-code\",\n        role: \"assistant\",\n        content: event.response,\n        tokenCount: event.tokenUsage.input + event.tokenUsage.output,\n        durationMs: event.durationMs,\n        timestamp: new Date().toISOString(),\n      };\n\n      // Update session state with token usage\n      const currentState = client.getSessionState(event.sessionId);\n      const currentTokens = currentState.tokenUsage || { input: 0, output: 0 };\n      client.updateSessionState(event.sessionId, {\n        tokenUsage: {\n          input: currentTokens.input + event.tokenUsage.input,\n          output: currentTokens.output + event.tokenUsage.output,\n        },\n      });\n\n      await client.syncMessage(message);\n    },\n\n    /**\n     * Called when session ends\n     */\n    SessionEnd: async (event: SessionEndEvent) => {\n      const currentState = client.getSessionState(event.sessionId);\n\n      const session: SessionData = {\n        ...currentState,\n        sessionId: event.sessionId,\n        source: \"claude-code\",\n        endReason: event.endReason,\n        messageCount: event.messageCount,\n        toolCallCount: event.toolCallCount,\n        tokenUsage: event.totalTokenUsage,\n        costEstimate: event.costEstimate,\n        endedAt: new Date().toISOString(),\n      };\n\n      await client.syncSession(session);\n      client.clearSessionState(event.sessionId);\n\n      console.log(\n        `[claude-code-sync] Session synced: ${event.messageCount} messages, ${event.toolCallCount} tool calls`\n      );\n    },\n  };\n}\n\n// Default export for Claude Code plugin system\nexport default createPlugin;\n"]}
430
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";AAAA;;;;;;;;GAQG;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAqIH,gCA4BC;AAED,gCAUC;AAED,kCAQC;AAwMD,oCAqKC;AAliBD,uCAAyB;AACzB,2CAA6B;AAC7B,uCAAyB;AA0HzB,+EAA+E;AAC/E,gBAAgB;AAChB,+EAA+E;AAE/E,MAAM,UAAU,GAAG,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,OAAO,EAAE,EAAE,SAAS,EAAE,kBAAkB,CAAC,CAAC;AAC1E,MAAM,WAAW,GAAG,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,aAAa,CAAC,CAAC;AAEzD,SAAgB,UAAU;IACxB,oCAAoC;IACpC,MAAM,MAAM,GAAG,OAAO,CAAC,GAAG,CAAC,sBAAsB,CAAC;IAClD,MAAM,MAAM,GAAG,OAAO,CAAC,GAAG,CAAC,mBAAmB,CAAC;IAE/C,IAAI,MAAM,IAAI,MAAM,EAAE,CAAC;QACrB,OAAO;YACL,SAAS,EAAE,kBAAkB,CAAC,MAAM,CAAC;YACrC,MAAM,EAAE,MAAM;YACd,QAAQ,EAAE,OAAO,CAAC,GAAG,CAAC,qBAAqB,KAAK,OAAO;YACvD,aAAa,EAAE,OAAO,CAAC,GAAG,CAAC,sBAAsB,KAAK,OAAO;YAC7D,YAAY,EAAE,OAAO,CAAC,GAAG,CAAC,oBAAoB,KAAK,MAAM;SAC1D,CAAC;IACJ,CAAC;IAED,2BAA2B;IAC3B,IAAI,CAAC;QACH,IAAI,EAAE,CAAC,UAAU,CAAC,WAAW,CAAC,EAAE,CAAC;YAC/B,MAAM,IAAI,GAAG,EAAE,CAAC,YAAY,CAAC,WAAW,EAAE,OAAO,CAAC,CAAC;YACnD,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAW,CAAC;YAC1C,MAAM,CAAC,SAAS,GAAG,kBAAkB,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;YACxD,OAAO,MAAM,CAAC;QAChB,CAAC;IACH,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,OAAO,CAAC,KAAK,CAAC,uBAAuB,EAAE,KAAK,CAAC,CAAC;IAChD,CAAC;IAED,OAAO,IAAI,CAAC;AACd,CAAC;AAED,SAAgB,UAAU,CAAC,MAAc;IACvC,IAAI,CAAC;QACH,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,UAAU,CAAC,EAAE,CAAC;YAC/B,EAAE,CAAC,SAAS,CAAC,UAAU,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QAChD,CAAC;QACD,EAAE,CAAC,aAAa,CAAC,WAAW,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;IACjE,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,OAAO,CAAC,KAAK,CAAC,sBAAsB,EAAE,KAAK,CAAC,CAAC;QAC7C,MAAM,KAAK,CAAC;IACd,CAAC;AACH,CAAC;AAED,SAAgB,WAAW;IACzB,IAAI,CAAC;QACH,IAAI,EAAE,CAAC,UAAU,CAAC,WAAW,CAAC,EAAE,CAAC;YAC/B,EAAE,CAAC,UAAU,CAAC,WAAW,CAAC,CAAC;QAC7B,CAAC;IACH,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,OAAO,CAAC,KAAK,CAAC,wBAAwB,EAAE,KAAK,CAAC,CAAC;IACjD,CAAC;AACH,CAAC;AAED,SAAS,kBAAkB,CAAC,GAAW;IACrC,sDAAsD;IACtD,OAAO,GAAG,CAAC,OAAO,CAAC,eAAe,EAAE,cAAc,CAAC,CAAC;AACtD,CAAC;AAED,+EAA+E;AAC/E,cAAc;AACd,+EAA+E;AAE/E,MAAa,UAAU;IACb,MAAM,CAAS;IACf,OAAO,CAAS;IAChB,YAAY,GAAsC,IAAI,GAAG,EAAE,CAAC;IAEpE,YAAY,MAAc;QACxB,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC;QACrB,mDAAmD;QACnD,0DAA0D;QAC1D,IAAI,CAAC,OAAO,GAAG,MAAM,CAAC,SAAS,CAAC,OAAO,CAAC,eAAe,EAAE,cAAc,CAAC,CAAC;IAC3E,CAAC;IAEO,KAAK,CAAC,OAAO,CAAC,QAAgB,EAAE,IAAa;QACnD,MAAM,GAAG,GAAG,GAAG,IAAI,CAAC,OAAO,GAAG,QAAQ,EAAE,CAAC;QAEzC,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,GAAG,EAAE;YAChC,MAAM,EAAE,MAAM;YACd,OAAO,EAAE;gBACP,cAAc,EAAE,kBAAkB;gBAClC,aAAa,EAAE,UAAU,IAAI,CAAC,MAAM,CAAC,MAAM,EAAE;aAC9C;YACD,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC;SAC3B,CAAC,CAAC;QAEH,IAAI,CAAC,QAAQ,CAAC,EAAE,EAAE,CAAC;YACjB,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;YACnC,MAAM,IAAI,KAAK,CAAC,gBAAgB,QAAQ,CAAC,MAAM,MAAM,IAAI,EAAE,CAAC,CAAC;QAC/D,CAAC;QAED,OAAO,QAAQ,CAAC,IAAI,EAAE,CAAC;IACzB,CAAC;IAED,2CAA2C;IACnC,gBAAgB,CAAC,OAAoB;QAC3C,MAAM,OAAO,GAA4B;YACvC,UAAU,EAAE,OAAO,CAAC,SAAS;YAC7B,MAAM,EAAE,OAAO,CAAC,MAAM;SACvB,CAAC;QAEF,uCAAuC;QACvC,IAAI,OAAO,CAAC,KAAK,KAAK,SAAS;YAAE,OAAO,CAAC,KAAK,GAAG,OAAO,CAAC,KAAK,CAAC;QAC/D,IAAI,OAAO,CAAC,WAAW,IAAI,OAAO,CAAC,GAAG;YAAE,OAAO,CAAC,WAAW,GAAG,OAAO,CAAC,WAAW,IAAI,OAAO,CAAC,GAAG,CAAC;QACjG,IAAI,OAAO,CAAC,WAAW;YAAE,OAAO,CAAC,WAAW,GAAG,OAAO,CAAC,WAAW,CAAC;QACnE,IAAI,OAAO,CAAC,KAAK;YAAE,OAAO,CAAC,KAAK,GAAG,OAAO,CAAC,KAAK,CAAC;QACjD,IAAI,OAAO,CAAC,UAAU,EAAE,KAAK,KAAK,SAAS;YAAE,OAAO,CAAC,YAAY,GAAG,OAAO,CAAC,UAAU,CAAC,KAAK,CAAC;QAC7F,IAAI,OAAO,CAAC,UAAU,EAAE,MAAM,KAAK,SAAS;YAAE,OAAO,CAAC,gBAAgB,GAAG,OAAO,CAAC,UAAU,CAAC,MAAM,CAAC;QACnG,IAAI,OAAO,CAAC,YAAY,KAAK,SAAS;YAAE,OAAO,CAAC,IAAI,GAAG,OAAO,CAAC,YAAY,CAAC;QAC5E,IAAI,OAAO,CAAC,YAAY,KAAK,SAAS;YAAE,OAAO,CAAC,YAAY,GAAG,OAAO,CAAC,YAAY,CAAC;QACpF,IAAI,OAAO,CAAC,aAAa,KAAK,SAAS;YAAE,OAAO,CAAC,aAAa,GAAG,OAAO,CAAC,aAAa,CAAC;QACvF,IAAI,OAAO,CAAC,SAAS;YAAE,OAAO,CAAC,SAAS,GAAG,OAAO,CAAC,SAAS,CAAC;QAE7D,8CAA8C;QAC9C,IAAI,OAAO,CAAC,OAAO,IAAI,OAAO,CAAC,SAAS,EAAE,CAAC;YACzC,OAAO,CAAC,UAAU,GAAG,IAAI,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,GAAG,IAAI,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC,OAAO,EAAE,CAAC;QACnG,CAAC;QAED,OAAO,OAAO,CAAC;IACjB,CAAC;IAED,2CAA2C;IACnC,gBAAgB,CAAC,OAAoB;QAC3C,MAAM,OAAO,GAA4B;YACvC,iBAAiB,EAAE,OAAO,CAAC,SAAS;YACpC,UAAU,EAAE,OAAO,CAAC,SAAS;YAC7B,IAAI,EAAE,OAAO,CAAC,IAAI;YAClB,MAAM,EAAE,OAAO,CAAC,MAAM;SACvB,CAAC;QAEF,2DAA2D;QAC3D,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC;YACpB,OAAO,CAAC,WAAW,GAAG,OAAO,CAAC,OAAO,CAAC;QACxC,CAAC;aAAM,IAAI,OAAO,CAAC,UAAU,IAAI,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC;YACnD,+DAA+D;YAC/D,OAAO,CAAC,WAAW,GAAG,OAAO,CAAC,UAAU,CAAC;QAC3C,CAAC;QAED,gCAAgC;QAChC,IAAI,OAAO,CAAC,UAAU,KAAK,SAAS,EAAE,CAAC;YACrC,OAAO,CAAC,UAAU,GAAG,OAAO,CAAC,UAAU,CAAC;QAC1C,CAAC;QAED,mCAAmC;QACnC,IAAI,OAAO,CAAC,UAAU,KAAK,SAAS,EAAE,CAAC;YACrC,OAAO,CAAC,YAAY,GAAG,OAAO,CAAC,UAAU,CAAC;QAC5C,CAAC;QAED,6BAA6B;QAC7B,IAAI,OAAO,CAAC,KAAK,EAAE,CAAC;YAClB,OAAO,CAAC,KAAK,GAAG,OAAO,CAAC,KAAK,CAAC;QAChC,CAAC;QAED,gCAAgC;QAChC,IAAI,OAAO,CAAC,QAAQ,EAAE,CAAC;YACrB,OAAO,CAAC,KAAK,GAAG;gBACd;oBACE,IAAI,EAAE,WAAW;oBACjB,OAAO,EAAE;wBACP,QAAQ,EAAE,OAAO,CAAC,QAAQ;wBAC1B,IAAI,EAAE,OAAO,CAAC,QAAQ;qBACvB;iBACF;aACF,CAAC;YACF,mCAAmC;YACnC,IAAI,OAAO,CAAC,UAAU,EAAE,CAAC;gBACtB,OAAO,CAAC,KAAmD,CAAC,IAAI,CAAC;oBAChE,IAAI,EAAE,aAAa;oBACnB,OAAO,EAAE,OAAO,CAAC,UAAU;iBAC5B,CAAC,CAAC;YACL,CAAC;QACH,CAAC;QAED,OAAO,OAAO,CAAC;IACjB,CAAC;IAED,KAAK,CAAC,WAAW,CAAC,OAAoB;QACpC,IAAI,CAAC;YACH,MAAM,OAAO,GAAG,IAAI,CAAC,gBAAgB,CAAC,OAAO,CAAC,CAAC;YAC/C,MAAM,IAAI,CAAC,OAAO,CAAC,eAAe,EAAE,OAAO,CAAC,CAAC;QAC/C,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,KAAK,CAAC,yBAAyB,EAAE,KAAK,CAAC,CAAC;YAChD,MAAM,KAAK,CAAC;QACd,CAAC;IACH,CAAC;IAED,KAAK,CAAC,WAAW,CAAC,OAAoB;QACpC,IAAI,CAAC;YACH,MAAM,OAAO,GAAG,IAAI,CAAC,gBAAgB,CAAC,OAAO,CAAC,CAAC;YAC/C,MAAM,IAAI,CAAC,OAAO,CAAC,eAAe,EAAE,OAAO,CAAC,CAAC;QAC/C,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,KAAK,CAAC,yBAAyB,EAAE,KAAK,CAAC,CAAC;YAChD,MAAM,KAAK,CAAC;QACd,CAAC;IACH,CAAC;IAED,KAAK,CAAC,SAAS,CACb,QAAuB,EACvB,QAAuB;QAEvB,IAAI,CAAC;YACH,MAAM,mBAAmB,GAAG,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,CAAC;YAC1E,MAAM,mBAAmB,GAAG,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,CAAC;YAC1E,MAAM,IAAI,CAAC,OAAO,CAAC,aAAa,EAAE;gBAChC,QAAQ,EAAE,mBAAmB;gBAC7B,QAAQ,EAAE,mBAAmB;aAC9B,CAAC,CAAC;QACL,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,KAAK,CAAC,uBAAuB,EAAE,KAAK,CAAC,CAAC;YAC9C,MAAM,KAAK,CAAC;QACd,CAAC;IACH,CAAC;IAED,KAAK,CAAC,cAAc;QAClB,IAAI,CAAC;YACH,MAAM,GAAG,GAAG,GAAG,IAAI,CAAC,OAAO,SAAS,CAAC;YACrC,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,GAAG,CAAC,CAAC;YAClC,OAAO,QAAQ,CAAC,EAAE,CAAC;QACrB,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,KAAK,CAAC;QACf,CAAC;IACH,CAAC;IAED,2BAA2B;IAC3B,eAAe,CAAC,SAAiB;QAC/B,OAAO,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC;IAChD,CAAC;IAED,kBAAkB,CAChB,SAAiB,EACjB,OAA6B;QAE7B,MAAM,OAAO,GAAG,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC;QACvD,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,SAAS,EAAE,EAAE,GAAG,OAAO,EAAE,GAAG,OAAO,EAAE,CAAC,CAAC;IAC/D,CAAC;IAED,iBAAiB,CAAC,SAAiB;QACjC,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;IACtC,CAAC;CACF;AAjLD,gCAiLC;AAED,+EAA+E;AAC/E,gBAAgB;AAChB,+EAA+E;AAE/E;;;;;GAKG;AACH,SAAgB,YAAY;IAC1B,MAAM,MAAM,GAAG,UAAU,EAAE,CAAC;IAE5B,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,OAAO,CAAC,GAAG,CACT,4EAA4E,CAC7E,CAAC;QACF,OAAO,IAAI,CAAC;IACd,CAAC;IAED,IAAI,MAAM,CAAC,QAAQ,KAAK,KAAK,EAAE,CAAC;QAC9B,OAAO,CAAC,GAAG,CAAC,kDAAkD,CAAC,CAAC;QAChE,OAAO,IAAI,CAAC;IACd,CAAC;IAED,MAAM,MAAM,GAAG,IAAI,UAAU,CAAC,MAAM,CAAC,CAAC;IACtC,IAAI,cAAc,GAAG,CAAC,CAAC;IACvB,IAAI,eAAe,GAAG,CAAC,CAAC;IAExB,OAAO,CAAC,GAAG,CAAC,mEAAmE,CAAC,CAAC;IAEjF,OAAO;QACL;;WAEG;QACH,YAAY,EAAE,KAAK,EAAE,KAAwB,EAAE,EAAE;YAC/C,cAAc,GAAG,CAAC,CAAC;YACnB,eAAe,GAAG,CAAC,CAAC;YAEpB,MAAM,OAAO,GAAgB;gBAC3B,SAAS,EAAE,KAAK,CAAC,SAAS;gBAC1B,MAAM,EAAE,aAAa;gBACrB,GAAG,EAAE,KAAK,CAAC,GAAG;gBACd,KAAK,EAAE,KAAK,CAAC,KAAK;gBAClB,SAAS,EAAE,KAAK,CAAC,SAAS;gBAC1B,eAAe,EAAE,KAAK,CAAC,eAAe;gBACtC,cAAc,EAAE,KAAK,CAAC,cAAc;gBACpC,UAAU,EAAE,KAAK,CAAC,UAAU;gBAC5B,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;aACpC,CAAC;YAEF,gCAAgC;YAChC,IAAI,KAAK,CAAC,GAAG,EAAE,CAAC;gBACd,OAAO,CAAC,WAAW,GAAG,KAAK,CAAC,GAAG,CAAC;gBAChC,OAAO,CAAC,WAAW,GAAG,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;gBAE/C,sBAAsB;gBACtB,IAAI,CAAC;oBACH,MAAM,MAAM,GAAG,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,EAAE,MAAM,CAAC,CAAC;oBAC5C,IAAI,EAAE,CAAC,UAAU,CAAC,MAAM,CAAC,EAAE,CAAC;wBAC1B,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;wBAC3C,IAAI,EAAE,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;4BAC5B,MAAM,IAAI,GAAG,EAAE,CAAC,YAAY,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC,IAAI,EAAE,CAAC;4BACvD,IAAI,IAAI,CAAC,UAAU,CAAC,kBAAkB,CAAC,EAAE,CAAC;gCACxC,OAAO,CAAC,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,kBAAkB,EAAE,EAAE,CAAC,CAAC;4BAC3D,CAAC;wBACH,CAAC;oBACH,CAAC;gBACH,CAAC;gBAAC,MAAM,CAAC;oBACP,oBAAoB;gBACtB,CAAC;YACH,CAAC;YAED,MAAM,CAAC,kBAAkB,CAAC,KAAK,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;YACpD,MAAM,MAAM,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;QACpC,CAAC;QAED;;WAEG;QACH,gBAAgB,EAAE,KAAK,EAAE,KAAsB,EAAE,EAAE;YACjD,cAAc,EAAE,CAAC;YAEjB,MAAM,OAAO,GAAgB;gBAC3B,SAAS,EAAE,KAAK,CAAC,SAAS;gBAC1B,SAAS,EAAE,GAAG,KAAK,CAAC,SAAS,QAAQ,cAAc,EAAE;gBACrD,MAAM,EAAE,aAAa;gBACrB,IAAI,EAAE,MAAM;gBACZ,OAAO,EAAE,KAAK,CAAC,MAAM;gBACrB,SAAS,EAAE,KAAK,CAAC,SAAS,IAAI,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;aACvD,CAAC;YAEF,MAAM,MAAM,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;QACpC,CAAC;QAED;;WAEG;QACH,WAAW,EAAE,KAAK,EAAE,KAAmB,EAAE,EAAE;YACzC,IAAI,CAAC,MAAM,CAAC,aAAa;gBAAE,OAAO;YAElC,eAAe,EAAE,CAAC;YAClB,cAAc,EAAE,CAAC;YAEjB,MAAM,OAAO,GAAgB;gBAC3B,SAAS,EAAE,KAAK,CAAC,SAAS;gBAC1B,SAAS,EAAE,GAAG,KAAK,CAAC,SAAS,SAAS,eAAe,EAAE;gBACvD,MAAM,EAAE,aAAa;gBACrB,IAAI,EAAE,WAAW;gBACjB,QAAQ,EAAE,KAAK,CAAC,QAAQ;gBACxB,QAAQ,EAAE,KAAK,CAAC,IAAI;gBACpB,UAAU,EAAE,KAAK,CAAC,MAAM;gBACxB,UAAU,EAAE,KAAK,CAAC,UAAU;gBAC5B,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;aACpC,CAAC;YAEF,MAAM,MAAM,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;QACpC,CAAC;QAED;;WAEG;QACH,IAAI,EAAE,KAAK,EAAE,KAAgB,EAAE,EAAE;YAC/B,cAAc,EAAE,CAAC;YAEjB,MAAM,OAAO,GAAgB;gBAC3B,SAAS,EAAE,KAAK,CAAC,SAAS;gBAC1B,SAAS,EAAE,GAAG,KAAK,CAAC,SAAS,QAAQ,cAAc,EAAE;gBACrD,MAAM,EAAE,aAAa;gBACrB,IAAI,EAAE,WAAW;gBACjB,OAAO,EAAE,KAAK,CAAC,QAAQ;gBACvB,UAAU,EAAE,KAAK,CAAC,UAAU,CAAC,KAAK,GAAG,KAAK,CAAC,UAAU,CAAC,MAAM;gBAC5D,UAAU,EAAE,KAAK,CAAC,UAAU;gBAC5B,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;aACpC,CAAC;YAEF,wCAAwC;YACxC,MAAM,YAAY,GAAG,MAAM,CAAC,eAAe,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;YAC7D,MAAM,aAAa,GAAG,YAAY,CAAC,UAAU,IAAI,EAAE,KAAK,EAAE,CAAC,EAAE,MAAM,EAAE,CAAC,EAAE,CAAC;YACzE,MAAM,CAAC,kBAAkB,CAAC,KAAK,CAAC,SAAS,EAAE;gBACzC,UAAU,EAAE;oBACV,KAAK,EAAE,aAAa,CAAC,KAAK,GAAG,KAAK,CAAC,UAAU,CAAC,KAAK;oBACnD,MAAM,EAAE,aAAa,CAAC,MAAM,GAAG,KAAK,CAAC,UAAU,CAAC,MAAM;iBACvD;aACF,CAAC,CAAC;YAEH,MAAM,MAAM,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;QACpC,CAAC;QAED;;WAEG;QACH,UAAU,EAAE,KAAK,EAAE,KAAsB,EAAE,EAAE;YAC3C,MAAM,YAAY,GAAG,MAAM,CAAC,eAAe,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;YAE7D,MAAM,OAAO,GAAgB;gBAC3B,GAAG,YAAY;gBACf,SAAS,EAAE,KAAK,CAAC,SAAS;gBAC1B,MAAM,EAAE,aAAa;gBACrB,SAAS,EAAE,KAAK,CAAC,SAAS;gBAC1B,YAAY,EAAE,KAAK,CAAC,YAAY;gBAChC,aAAa,EAAE,KAAK,CAAC,aAAa;gBAClC,UAAU,EAAE,KAAK,CAAC,eAAe;gBACjC,YAAY,EAAE,KAAK,CAAC,YAAY;gBAChC,OAAO,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;aAClC,CAAC;YAEF,MAAM,MAAM,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;YAClC,MAAM,CAAC,iBAAiB,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;YAE1C,OAAO,CAAC,GAAG,CACT,sCAAsC,KAAK,CAAC,YAAY,cAAc,KAAK,CAAC,aAAa,aAAa,CACvG,CAAC;QACJ,CAAC;KACF,CAAC;AACJ,CAAC;AAED,+CAA+C;AAC/C,kBAAe,YAAY,CAAC","sourcesContent":["/**\n * Claude Code Sync Plugin\n *\n * Syncs Claude Code sessions to OpenSync dashboard.\n * Uses API Key authentication (no browser OAuth required).\n *\n * Install: npm install -g claude-code-sync\n * Configure: claude-code-sync login\n */\n\nimport * as fs from \"fs\";\nimport * as path from \"path\";\nimport * as os from \"os\";\n\n// ============================================================================\n// Types\n// ============================================================================\n\nexport interface Config {\n  convexUrl: string;\n  apiKey: string;\n  autoSync?: boolean;\n  syncToolCalls?: boolean;\n  syncThinking?: boolean;\n}\n\nexport interface SessionData {\n  sessionId: string;\n  source: \"claude-code\";\n  title?: string;\n  projectPath?: string;\n  projectName?: string;\n  cwd?: string;\n  gitBranch?: string;\n  gitRepo?: string;\n  model?: string;\n  startType?: \"new\" | \"resume\" | \"continue\";\n  endReason?: \"user_stop\" | \"max_turns\" | \"error\" | \"completed\";\n  thinkingEnabled?: boolean;\n  permissionMode?: string;\n  mcpServers?: string[];\n  messageCount?: number;\n  toolCallCount?: number;\n  tokenUsage?: {\n    input: number;\n    output: number;\n  };\n  costEstimate?: number;\n  startedAt?: string;\n  endedAt?: string;\n}\n\nexport interface MessageData {\n  sessionId: string;\n  messageId: string;\n  source: \"claude-code\";\n  role: \"user\" | \"assistant\" | \"system\";\n  content?: string;\n  thinkingContent?: string;\n  toolName?: string;\n  toolArgs?: Record<string, unknown>;\n  toolResult?: string;\n  durationMs?: number;\n  tokenCount?: number;\n  timestamp?: string;\n  model?: string;\n}\n\nexport interface ToolUseData {\n  sessionId: string;\n  toolName: string;\n  toolArgs?: Record<string, unknown>;\n  result?: string;\n  success?: boolean;\n  durationMs?: number;\n  timestamp?: string;\n}\n\n// Claude Code Hook Types\nexport interface ClaudeCodeHooks {\n  SessionStart?: (data: SessionStartEvent) => void | Promise<void>;\n  UserPromptSubmit?: (data: UserPromptEvent) => void | Promise<void>;\n  PostToolUse?: (data: ToolUseEvent) => void | Promise<void>;\n  Stop?: (data: StopEvent) => void | Promise<void>;\n  SessionEnd?: (data: SessionEndEvent) => void | Promise<void>;\n}\n\nexport interface SessionStartEvent {\n  sessionId: string;\n  cwd: string;\n  model: string;\n  startType: \"new\" | \"resume\" | \"continue\";\n  thinkingEnabled?: boolean;\n  permissionMode?: string;\n  mcpServers?: string[];\n}\n\nexport interface UserPromptEvent {\n  sessionId: string;\n  prompt: string;\n  timestamp: string;\n}\n\nexport interface ToolUseEvent {\n  sessionId: string;\n  toolName: string;\n  args: Record<string, unknown>;\n  result: string;\n  success: boolean;\n  durationMs: number;\n}\n\nexport interface StopEvent {\n  sessionId: string;\n  response: string;\n  tokenUsage: {\n    input: number;\n    output: number;\n  };\n  durationMs: number;\n}\n\nexport interface SessionEndEvent {\n  sessionId: string;\n  endReason: \"user_stop\" | \"max_turns\" | \"error\" | \"completed\";\n  messageCount: number;\n  toolCallCount: number;\n  totalTokenUsage: {\n    input: number;\n    output: number;\n  };\n  costEstimate: number;\n}\n\n// ============================================================================\n// Configuration\n// ============================================================================\n\nconst CONFIG_DIR = path.join(os.homedir(), \".config\", \"claude-code-sync\");\nconst CONFIG_FILE = path.join(CONFIG_DIR, \"config.json\");\n\nexport function loadConfig(): Config | null {\n  // Check environment variables first\n  const envUrl = process.env.CLAUDE_SYNC_CONVEX_URL;\n  const envKey = process.env.CLAUDE_SYNC_API_KEY;\n\n  if (envUrl && envKey) {\n    return {\n      convexUrl: normalizeConvexUrl(envUrl),\n      apiKey: envKey,\n      autoSync: process.env.CLAUDE_SYNC_AUTO_SYNC !== \"false\",\n      syncToolCalls: process.env.CLAUDE_SYNC_TOOL_CALLS !== \"false\",\n      syncThinking: process.env.CLAUDE_SYNC_THINKING === \"true\",\n    };\n  }\n\n  // Fall back to config file\n  try {\n    if (fs.existsSync(CONFIG_FILE)) {\n      const data = fs.readFileSync(CONFIG_FILE, \"utf-8\");\n      const config = JSON.parse(data) as Config;\n      config.convexUrl = normalizeConvexUrl(config.convexUrl);\n      return config;\n    }\n  } catch (error) {\n    console.error(\"Error loading config:\", error);\n  }\n\n  return null;\n}\n\nexport function saveConfig(config: Config): void {\n  try {\n    if (!fs.existsSync(CONFIG_DIR)) {\n      fs.mkdirSync(CONFIG_DIR, { recursive: true });\n    }\n    fs.writeFileSync(CONFIG_FILE, JSON.stringify(config, null, 2));\n  } catch (error) {\n    console.error(\"Error saving config:\", error);\n    throw error;\n  }\n}\n\nexport function clearConfig(): void {\n  try {\n    if (fs.existsSync(CONFIG_FILE)) {\n      fs.unlinkSync(CONFIG_FILE);\n    }\n  } catch (error) {\n    console.error(\"Error clearing config:\", error);\n  }\n}\n\nfunction normalizeConvexUrl(url: string): string {\n  // Convert .convex.cloud to .convex.site for API calls\n  return url.replace(\".convex.cloud\", \".convex.site\");\n}\n\n// ============================================================================\n// Sync Client\n// ============================================================================\n\nexport class SyncClient {\n  private config: Config;\n  private siteUrl: string;\n  private sessionCache: Map<string, Partial<SessionData>> = new Map();\n\n  constructor(config: Config) {\n    this.config = config;\n    // Normalize URL to .convex.site for HTTP endpoints\n    // Supports both .convex.cloud and .convex.site input URLs\n    this.siteUrl = config.convexUrl.replace(\".convex.cloud\", \".convex.site\");\n  }\n\n  private async request(endpoint: string, data: unknown): Promise<unknown> {\n    const url = `${this.siteUrl}${endpoint}`;\n\n    const response = await fetch(url, {\n      method: \"POST\",\n      headers: {\n        \"Content-Type\": \"application/json\",\n        Authorization: `Bearer ${this.config.apiKey}`,\n      },\n      body: JSON.stringify(data),\n    });\n\n    if (!response.ok) {\n      const text = await response.text();\n      throw new Error(`Sync failed: ${response.status} - ${text}`);\n    }\n\n    return response.json();\n  }\n\n  // Transform session data to backend format\n  private transformSession(session: SessionData): Record<string, unknown> {\n    const payload: Record<string, unknown> = {\n      externalId: session.sessionId,\n      source: session.source,\n    };\n    \n    // Only include fields that are defined\n    if (session.title !== undefined) payload.title = session.title;\n    if (session.projectPath || session.cwd) payload.projectPath = session.projectPath || session.cwd;\n    if (session.projectName) payload.projectName = session.projectName;\n    if (session.model) payload.model = session.model;\n    if (session.tokenUsage?.input !== undefined) payload.promptTokens = session.tokenUsage.input;\n    if (session.tokenUsage?.output !== undefined) payload.completionTokens = session.tokenUsage.output;\n    if (session.costEstimate !== undefined) payload.cost = session.costEstimate;\n    if (session.messageCount !== undefined) payload.messageCount = session.messageCount;\n    if (session.toolCallCount !== undefined) payload.toolCallCount = session.toolCallCount;\n    if (session.endReason) payload.endReason = session.endReason;\n    \n    // Calculate duration if both timestamps exist\n    if (session.endedAt && session.startedAt) {\n      payload.durationMs = new Date(session.endedAt).getTime() - new Date(session.startedAt).getTime();\n    }\n    \n    return payload;\n  }\n\n  // Transform message data to backend format\n  private transformMessage(message: MessageData): Record<string, unknown> {\n    const payload: Record<string, unknown> = {\n      sessionExternalId: message.sessionId,\n      externalId: message.messageId,\n      role: message.role,\n      source: message.source,\n    };\n    \n    // Set textContent for user prompts and assistant responses\n    if (message.content) {\n      payload.textContent = message.content;\n    } else if (message.toolResult && !message.toolName) {\n      // Only use toolResult as textContent if no toolName (fallback)\n      payload.textContent = message.toolResult;\n    }\n    \n    // Include duration if available\n    if (message.durationMs !== undefined) {\n      payload.durationMs = message.durationMs;\n    }\n    \n    // Include token count if available\n    if (message.tokenCount !== undefined) {\n      payload.promptTokens = message.tokenCount;\n    }\n    \n    // Include model if available\n    if (message.model) {\n      payload.model = message.model;\n    }\n    \n    // Tool use info stored in parts\n    if (message.toolName) {\n      payload.parts = [\n        {\n          type: \"tool-call\",\n          content: {\n            toolName: message.toolName,\n            args: message.toolArgs,\n          },\n        },\n      ];\n      // Add tool result as separate part\n      if (message.toolResult) {\n        (payload.parts as Array<{ type: string; content: unknown }>).push({\n          type: \"tool-result\",\n          content: message.toolResult,\n        });\n      }\n    }\n    \n    return payload;\n  }\n\n  async syncSession(session: SessionData): Promise<void> {\n    try {\n      const payload = this.transformSession(session);\n      await this.request(\"/sync/session\", payload);\n    } catch (error) {\n      console.error(\"Failed to sync session:\", error);\n      throw error;\n    }\n  }\n\n  async syncMessage(message: MessageData): Promise<void> {\n    try {\n      const payload = this.transformMessage(message);\n      await this.request(\"/sync/message\", payload);\n    } catch (error) {\n      console.error(\"Failed to sync message:\", error);\n      throw error;\n    }\n  }\n\n  async syncBatch(\n    sessions: SessionData[],\n    messages: MessageData[]\n  ): Promise<void> {\n    try {\n      const transformedSessions = sessions.map((s) => this.transformSession(s));\n      const transformedMessages = messages.map((m) => this.transformMessage(m));\n      await this.request(\"/sync/batch\", {\n        sessions: transformedSessions,\n        messages: transformedMessages,\n      });\n    } catch (error) {\n      console.error(\"Failed to sync batch:\", error);\n      throw error;\n    }\n  }\n\n  async testConnection(): Promise<boolean> {\n    try {\n      const url = `${this.siteUrl}/health`;\n      const response = await fetch(url);\n      return response.ok;\n    } catch {\n      return false;\n    }\n  }\n\n  // Session state management\n  getSessionState(sessionId: string): Partial<SessionData> {\n    return this.sessionCache.get(sessionId) || {};\n  }\n\n  updateSessionState(\n    sessionId: string,\n    updates: Partial<SessionData>\n  ): void {\n    const current = this.sessionCache.get(sessionId) || {};\n    this.sessionCache.set(sessionId, { ...current, ...updates });\n  }\n\n  clearSessionState(sessionId: string): void {\n    this.sessionCache.delete(sessionId);\n  }\n}\n\n// ============================================================================\n// Plugin Export\n// ============================================================================\n\n/**\n * Claude Code Plugin Entry Point\n *\n * This function is called by Claude Code to register the plugin.\n * It returns hook handlers that fire at key points in the session lifecycle.\n */\nexport function createPlugin(): ClaudeCodeHooks | null {\n  const config = loadConfig();\n\n  if (!config) {\n    console.log(\n      \"[claude-code-sync] Not configured. Run 'claude-code-sync login' to set up.\"\n    );\n    return null;\n  }\n\n  if (config.autoSync === false) {\n    console.log(\"[claude-code-sync] Auto-sync disabled in config.\");\n    return null;\n  }\n\n  const client = new SyncClient(config);\n  let messageCounter = 0;\n  let toolCallCounter = 0;\n\n  console.log(\"[claude-code-sync] Plugin loaded. Sessions will sync to OpenSync.\");\n\n  return {\n    /**\n     * Called when a new session starts\n     */\n    SessionStart: async (event: SessionStartEvent) => {\n      messageCounter = 0;\n      toolCallCounter = 0;\n\n      const session: SessionData = {\n        sessionId: event.sessionId,\n        source: \"claude-code\",\n        cwd: event.cwd,\n        model: event.model,\n        startType: event.startType,\n        thinkingEnabled: event.thinkingEnabled,\n        permissionMode: event.permissionMode,\n        mcpServers: event.mcpServers,\n        startedAt: new Date().toISOString(),\n      };\n\n      // Extract project info from cwd\n      if (event.cwd) {\n        session.projectPath = event.cwd;\n        session.projectName = path.basename(event.cwd);\n\n        // Try to get git info\n        try {\n          const gitDir = path.join(event.cwd, \".git\");\n          if (fs.existsSync(gitDir)) {\n            const headFile = path.join(gitDir, \"HEAD\");\n            if (fs.existsSync(headFile)) {\n              const head = fs.readFileSync(headFile, \"utf-8\").trim();\n              if (head.startsWith(\"ref: refs/heads/\")) {\n                session.gitBranch = head.replace(\"ref: refs/heads/\", \"\");\n              }\n            }\n          }\n        } catch {\n          // Ignore git errors\n        }\n      }\n\n      client.updateSessionState(event.sessionId, session);\n      await client.syncSession(session);\n    },\n\n    /**\n     * Called when user submits a prompt\n     */\n    UserPromptSubmit: async (event: UserPromptEvent) => {\n      messageCounter++;\n\n      const message: MessageData = {\n        sessionId: event.sessionId,\n        messageId: `${event.sessionId}-msg-${messageCounter}`,\n        source: \"claude-code\",\n        role: \"user\",\n        content: event.prompt,\n        timestamp: event.timestamp || new Date().toISOString(),\n      };\n\n      await client.syncMessage(message);\n    },\n\n    /**\n     * Called after each tool use\n     */\n    PostToolUse: async (event: ToolUseEvent) => {\n      if (!config.syncToolCalls) return;\n\n      toolCallCounter++;\n      messageCounter++;\n\n      const message: MessageData = {\n        sessionId: event.sessionId,\n        messageId: `${event.sessionId}-tool-${toolCallCounter}`,\n        source: \"claude-code\",\n        role: \"assistant\",\n        toolName: event.toolName,\n        toolArgs: event.args,\n        toolResult: event.result,\n        durationMs: event.durationMs,\n        timestamp: new Date().toISOString(),\n      };\n\n      await client.syncMessage(message);\n    },\n\n    /**\n     * Called when Claude stops responding\n     */\n    Stop: async (event: StopEvent) => {\n      messageCounter++;\n\n      const message: MessageData = {\n        sessionId: event.sessionId,\n        messageId: `${event.sessionId}-msg-${messageCounter}`,\n        source: \"claude-code\",\n        role: \"assistant\",\n        content: event.response,\n        tokenCount: event.tokenUsage.input + event.tokenUsage.output,\n        durationMs: event.durationMs,\n        timestamp: new Date().toISOString(),\n      };\n\n      // Update session state with token usage\n      const currentState = client.getSessionState(event.sessionId);\n      const currentTokens = currentState.tokenUsage || { input: 0, output: 0 };\n      client.updateSessionState(event.sessionId, {\n        tokenUsage: {\n          input: currentTokens.input + event.tokenUsage.input,\n          output: currentTokens.output + event.tokenUsage.output,\n        },\n      });\n\n      await client.syncMessage(message);\n    },\n\n    /**\n     * Called when session ends\n     */\n    SessionEnd: async (event: SessionEndEvent) => {\n      const currentState = client.getSessionState(event.sessionId);\n\n      const session: SessionData = {\n        ...currentState,\n        sessionId: event.sessionId,\n        source: \"claude-code\",\n        endReason: event.endReason,\n        messageCount: event.messageCount,\n        toolCallCount: event.toolCallCount,\n        tokenUsage: event.totalTokenUsage,\n        costEstimate: event.costEstimate,\n        endedAt: new Date().toISOString(),\n      };\n\n      await client.syncSession(session);\n      client.clearSessionState(event.sessionId);\n\n      console.log(\n        `[claude-code-sync] Session synced: ${event.messageCount} messages, ${event.toolCallCount} tool calls`\n      );\n    },\n  };\n}\n\n// Default export for Claude Code plugin system\nexport default createPlugin;\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "claude-code-sync",
3
- "version": "0.1.10",
3
+ "version": "0.1.12",
4
4
  "description": "Sync your Claude Code sessions to OpenSync dashboard. Track coding sessions, analyze tool usage, and monitor token consumption across projects.",
5
5
  "main": "dist/index.js",
6
6
  "types": "dist/index.d.ts",