claude-code-sync 0.1.13 → 0.1.14

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/cli.js CHANGED
@@ -48,6 +48,38 @@ const readline = __importStar(require("readline"));
48
48
  const fs = __importStar(require("fs"));
49
49
  const path = __importStar(require("path"));
50
50
  const index_1 = require("./index");
51
+ // Pricing per million tokens (USD) with cache pricing
52
+ // Source: https://www.anthropic.com/pricing
53
+ const MODEL_PRICING = {
54
+ "claude-sonnet-4-20250514": { input: 3.00, output: 15.00, cacheWrite: 3.75, cacheRead: 0.30 },
55
+ "claude-opus-4-20250514": { input: 15.00, output: 75.00, cacheWrite: 18.75, cacheRead: 1.50 },
56
+ "claude-opus-4-5-20251101": { input: 15.00, output: 75.00, cacheWrite: 18.75, cacheRead: 1.50 },
57
+ "claude-3-5-sonnet-20241022": { input: 3.00, output: 15.00, cacheWrite: 3.75, cacheRead: 0.30 },
58
+ "claude-3-opus-20240229": { input: 15.00, output: 75.00, cacheWrite: 18.75, cacheRead: 1.50 },
59
+ "claude-3-5-haiku-20241022": { input: 0.80, output: 4.00, cacheWrite: 1.00, cacheRead: 0.08 },
60
+ };
61
+ // Calculate cost from model and token usage with proper cache pricing
62
+ function calculateCost(model, stats) {
63
+ if (!model)
64
+ return 0;
65
+ // Try exact match first
66
+ let pricing = MODEL_PRICING[model];
67
+ // Try partial match if exact match fails
68
+ if (!pricing) {
69
+ const matchingKey = Object.keys(MODEL_PRICING).find(k => model.includes(k) || k.includes(model));
70
+ if (matchingKey) {
71
+ pricing = MODEL_PRICING[matchingKey];
72
+ }
73
+ }
74
+ if (!pricing)
75
+ return 0;
76
+ // Calculate cost with proper cache pricing
77
+ const inputCost = stats.inputTokens * pricing.input;
78
+ const cacheWriteCost = stats.cacheCreationTokens * pricing.cacheWrite;
79
+ const cacheReadCost = stats.cacheReadTokens * pricing.cacheRead;
80
+ const outputCost = stats.outputTokens * pricing.output;
81
+ return (inputCost + cacheWriteCost + cacheReadCost + outputCost) / 1_000_000;
82
+ }
51
83
  // Read version from package.json
52
84
  function getVersion() {
53
85
  try {
@@ -525,7 +557,20 @@ function generateTitle(prompt) {
525
557
  function parseTranscriptFile(transcriptPath) {
526
558
  const result = {
527
559
  assistantMessages: [],
528
- tokenUsage: { input: 0, output: 0 },
560
+ stats: {
561
+ model: undefined,
562
+ inputTokens: 0,
563
+ cacheCreationTokens: 0,
564
+ cacheReadTokens: 0,
565
+ outputTokens: 0,
566
+ messageCount: 0,
567
+ toolCallCount: 0,
568
+ title: undefined,
569
+ cwd: undefined,
570
+ startedAt: undefined,
571
+ endedAt: undefined,
572
+ durationMs: undefined,
573
+ },
529
574
  };
530
575
  try {
531
576
  if (!fs.existsSync(transcriptPath)) {
@@ -534,16 +579,40 @@ function parseTranscriptFile(transcriptPath) {
534
579
  const content = fs.readFileSync(transcriptPath, "utf-8");
535
580
  const lines = content.trim().split("\n");
536
581
  const seenUuids = new Set();
582
+ let firstTimestamp;
583
+ let lastTimestamp;
537
584
  for (const line of lines) {
538
585
  if (!line.trim())
539
586
  continue;
540
587
  try {
541
588
  const entry = JSON.parse(line);
542
- // Only process assistant messages
589
+ // Track timestamps for duration calculation
590
+ if (entry.timestamp) {
591
+ if (!firstTimestamp) {
592
+ firstTimestamp = entry.timestamp;
593
+ }
594
+ lastTimestamp = entry.timestamp;
595
+ }
596
+ // Get cwd and title from first entry that has them
597
+ if (!result.stats.cwd && entry.cwd) {
598
+ result.stats.cwd = entry.cwd;
599
+ }
600
+ if (!result.stats.title && entry.slug) {
601
+ result.stats.title = entry.slug;
602
+ }
603
+ // Count user messages
604
+ if (entry.type === "user") {
605
+ result.stats.messageCount++;
606
+ }
607
+ // Process assistant messages
543
608
  if (entry.type === "assistant" && entry.message) {
544
609
  const msg = entry.message;
545
610
  const uuid = entry.uuid || "";
546
- // Skip if we've already seen this UUID (dedup)
611
+ // Get model from first assistant message
612
+ if (msg.model && !result.stats.model) {
613
+ result.stats.model = msg.model;
614
+ }
615
+ // Skip duplicate UUIDs for message extraction
547
616
  if (uuid && seenUuids.has(uuid))
548
617
  continue;
549
618
  if (uuid)
@@ -559,15 +628,18 @@ function parseTranscriptFile(transcriptPath) {
559
628
  model: msg.model,
560
629
  });
561
630
  }
631
+ // Count tool uses
632
+ if (part.type === "tool_use") {
633
+ result.stats.toolCallCount++;
634
+ }
562
635
  }
563
636
  }
564
- // Accumulate token usage from all assistant messages
637
+ // Accumulate token usage (separate cache tokens for cost calculation)
565
638
  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;
639
+ result.stats.inputTokens += msg.usage.input_tokens || 0;
640
+ result.stats.cacheCreationTokens += msg.usage.cache_creation_input_tokens || 0;
641
+ result.stats.cacheReadTokens += msg.usage.cache_read_input_tokens || 0;
642
+ result.stats.outputTokens += msg.usage.output_tokens || 0;
571
643
  }
572
644
  }
573
645
  }
@@ -575,6 +647,16 @@ function parseTranscriptFile(transcriptPath) {
575
647
  // Skip malformed lines
576
648
  }
577
649
  }
650
+ // Calculate duration from timestamps
651
+ if (firstTimestamp && lastTimestamp) {
652
+ result.stats.startedAt = firstTimestamp;
653
+ result.stats.endedAt = lastTimestamp;
654
+ const startMs = new Date(firstTimestamp).getTime();
655
+ const endMs = new Date(lastTimestamp).getTime();
656
+ if (!isNaN(startMs) && !isNaN(endMs)) {
657
+ result.stats.durationMs = endMs - startMs;
658
+ }
659
+ }
578
660
  }
579
661
  catch {
580
662
  // Return empty result on error
@@ -607,44 +689,112 @@ program
607
689
  switch (event) {
608
690
  case "SessionStart": {
609
691
  const data = JSON.parse(input);
692
+ // Parse transcript if available to get initial info
693
+ let stats;
694
+ if (data.transcript_path && fs.existsSync(data.transcript_path)) {
695
+ const parsed = parseTranscriptFile(data.transcript_path);
696
+ stats = parsed.stats;
697
+ }
698
+ const cwd = stats?.cwd || data.cwd;
699
+ const startedAt = new Date().toISOString();
610
700
  // Initialize session state
611
701
  sessionState[data.session_id] = {
612
- model: data.model,
702
+ model: stats?.model || data.model,
613
703
  tokenUsage: { input: 0, output: 0 },
704
+ cacheCreationTokens: 0,
705
+ cacheReadTokens: 0,
614
706
  messageCount: 0,
707
+ toolCallCount: 0,
708
+ startedAt,
709
+ title: stats?.title,
710
+ cwd,
615
711
  };
616
712
  saveSessionState(sessionState);
617
713
  const session = {
618
714
  sessionId: data.session_id,
619
715
  source: "claude-code",
620
- cwd: data.cwd,
621
- model: data.model,
716
+ cwd,
717
+ model: stats?.model || data.model,
718
+ title: stats?.title,
622
719
  permissionMode: data.permission_mode,
623
720
  thinkingEnabled: data.thinking_enabled,
624
721
  mcpServers: data.mcp_servers,
625
722
  startType: data.source === "startup" ? "new" : data.source,
626
- startedAt: new Date().toISOString(),
627
- projectPath: data.cwd,
628
- projectName: data.cwd ? data.cwd.split("/").pop() : undefined,
723
+ startedAt,
724
+ projectPath: cwd,
725
+ projectName: cwd ? path.basename(cwd) : undefined,
629
726
  };
727
+ // Try to get git branch from cwd
728
+ if (cwd) {
729
+ try {
730
+ const gitDir = path.join(cwd, ".git");
731
+ if (fs.existsSync(gitDir)) {
732
+ const headFile = path.join(gitDir, "HEAD");
733
+ if (fs.existsSync(headFile)) {
734
+ const head = fs.readFileSync(headFile, "utf-8").trim();
735
+ if (head.startsWith("ref: refs/heads/")) {
736
+ session.gitBranch = head.replace("ref: refs/heads/", "");
737
+ }
738
+ }
739
+ }
740
+ }
741
+ catch {
742
+ // Ignore git errors
743
+ }
744
+ }
630
745
  await client.syncSession(session);
631
746
  break;
632
747
  }
633
748
  case "SessionEnd": {
634
749
  const data = JSON.parse(input);
635
750
  const state = sessionState[data.session_id] || {};
636
- // Calculate final token usage
637
- const finalTokenUsage = data.total_token_usage || state.tokenUsage;
751
+ // Parse transcript to get final stats including model, tokens, cost
752
+ let stats = {
753
+ model: state.model,
754
+ inputTokens: state.tokenUsage?.input || 0,
755
+ cacheCreationTokens: state.cacheCreationTokens || 0,
756
+ cacheReadTokens: state.cacheReadTokens || 0,
757
+ outputTokens: state.tokenUsage?.output || 0,
758
+ messageCount: state.messageCount || 0,
759
+ toolCallCount: state.toolCallCount || 0,
760
+ title: state.title,
761
+ cwd: state.cwd || data.cwd,
762
+ startedAt: state.startedAt,
763
+ endedAt: state.endedAt,
764
+ durationMs: state.durationMs,
765
+ };
766
+ if (data.transcript_path && fs.existsSync(data.transcript_path)) {
767
+ const parsed = parseTranscriptFile(data.transcript_path);
768
+ stats = parsed.stats;
769
+ }
770
+ // Calculate cost from tokens (with proper cache pricing)
771
+ let cost = data.cost_estimate;
772
+ if ((cost === undefined || cost === null || cost === 0) && stats.model) {
773
+ if (stats.inputTokens || stats.outputTokens || stats.cacheReadTokens || stats.cacheCreationTokens) {
774
+ cost = calculateCost(stats.model, stats);
775
+ }
776
+ }
777
+ const cwd = stats.cwd || data.cwd;
778
+ const totalInputTokens = stats.inputTokens + stats.cacheCreationTokens + stats.cacheReadTokens;
779
+ const endedAt = stats.endedAt || new Date().toISOString();
638
780
  const session = {
639
781
  sessionId: data.session_id,
640
782
  source: "claude-code",
641
- title: state.firstPrompt ? generateTitle(state.firstPrompt) : undefined,
783
+ model: stats.model,
784
+ title: stats.title || (state.firstPrompt ? generateTitle(state.firstPrompt) : undefined),
785
+ cwd,
786
+ projectPath: cwd,
787
+ projectName: cwd ? path.basename(cwd) : undefined,
642
788
  endReason: data.reason,
643
- messageCount: data.message_count || state.messageCount,
644
- toolCallCount: data.tool_call_count,
645
- tokenUsage: finalTokenUsage,
646
- costEstimate: data.cost_estimate,
647
- endedAt: new Date().toISOString(),
789
+ messageCount: data.message_count || stats.messageCount,
790
+ toolCallCount: data.tool_call_count || stats.toolCallCount,
791
+ tokenUsage: {
792
+ input: totalInputTokens,
793
+ output: stats.outputTokens,
794
+ },
795
+ costEstimate: cost,
796
+ startedAt: stats.startedAt,
797
+ endedAt,
648
798
  };
649
799
  await client.syncSession(session);
650
800
  // Clean up session state
@@ -709,9 +859,36 @@ program
709
859
  // Parse transcript file to extract assistant messages and token usage
710
860
  if (data.transcript_path) {
711
861
  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;
862
+ const { stats } = transcript;
863
+ // Update token usage from transcript (track cache tokens separately)
864
+ if (stats.inputTokens > 0 || stats.outputTokens > 0 || stats.cacheReadTokens > 0) {
865
+ const totalInput = stats.inputTokens + stats.cacheCreationTokens + stats.cacheReadTokens;
866
+ state.tokenUsage = { input: totalInput, output: stats.outputTokens };
867
+ state.cacheCreationTokens = stats.cacheCreationTokens;
868
+ state.cacheReadTokens = stats.cacheReadTokens;
869
+ }
870
+ // Update model if found
871
+ if (stats.model && !state.model) {
872
+ state.model = stats.model;
873
+ }
874
+ // Update title if found from transcript slug
875
+ if (stats.title && !state.title) {
876
+ state.title = stats.title;
877
+ }
878
+ // Update duration info
879
+ if (stats.startedAt)
880
+ state.startedAt = stats.startedAt;
881
+ if (stats.endedAt)
882
+ state.endedAt = stats.endedAt;
883
+ if (stats.durationMs)
884
+ state.durationMs = stats.durationMs;
885
+ // Update tool call count
886
+ if (stats.toolCallCount > 0) {
887
+ state.toolCallCount = stats.toolCallCount;
888
+ }
889
+ // Calculate cost if we have model and tokens
890
+ if (stats.model && (stats.inputTokens || stats.outputTokens || stats.cacheReadTokens)) {
891
+ state.costEstimate = calculateCost(stats.model, stats);
715
892
  }
716
893
  // Track which messages we've already synced to avoid duplicates
717
894
  const syncedMessages = state.syncedMessageUuids instanceof Set
@@ -742,12 +919,15 @@ program
742
919
  }
743
920
  sessionState[data.session_id] = state;
744
921
  saveSessionState(sessionState);
745
- // Update session with token usage from transcript
922
+ // Update session with token usage, model, and cost from transcript
746
923
  const session = {
747
924
  sessionId: data.session_id,
748
925
  source: "claude-code",
926
+ model: state.model,
749
927
  tokenUsage: state.tokenUsage,
750
928
  messageCount: state.messageCount,
929
+ toolCallCount: state.toolCallCount,
930
+ costEstimate: state.costEstimate,
751
931
  };
752
932
  await client.syncSession(session);
753
933
  break;
@@ -766,4 +946,4 @@ program
766
946
  });
767
947
  // Parse and run
768
948
  program.parse();
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"]}
949
+ //# 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  transcript_path?: 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  transcript_path?: string;\n  cwd?: string;\n  reason?: string;\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  cwd?: string;\n  slug?: string;\n}\n\n// Extended stats from transcript parsing (includes cache tokens and duration)\ninterface TranscriptStats {\n  model: string | undefined;\n  inputTokens: number;\n  cacheCreationTokens: number;\n  cacheReadTokens: number;\n  outputTokens: number;\n  messageCount: number;\n  toolCallCount: number;\n  title: string | undefined;\n  cwd: string | undefined;\n  startedAt: string | undefined;\n  endedAt: string | undefined;\n  durationMs: number | undefined;\n}\n\n// Pricing per million tokens (USD) with cache pricing\n// Source: https://www.anthropic.com/pricing\nconst MODEL_PRICING: Record<string, { input: number; output: number; cacheWrite: number; cacheRead: number }> = {\n  \"claude-sonnet-4-20250514\": { input: 3.00, output: 15.00, cacheWrite: 3.75, cacheRead: 0.30 },\n  \"claude-opus-4-20250514\": { input: 15.00, output: 75.00, cacheWrite: 18.75, cacheRead: 1.50 },\n  \"claude-opus-4-5-20251101\": { input: 15.00, output: 75.00, cacheWrite: 18.75, cacheRead: 1.50 },\n  \"claude-3-5-sonnet-20241022\": { input: 3.00, output: 15.00, cacheWrite: 3.75, cacheRead: 0.30 },\n  \"claude-3-opus-20240229\": { input: 15.00, output: 75.00, cacheWrite: 18.75, cacheRead: 1.50 },\n  \"claude-3-5-haiku-20241022\": { input: 0.80, output: 4.00, cacheWrite: 1.00, cacheRead: 0.08 },\n};\n\n// Calculate cost from model and token usage with proper cache pricing\nfunction calculateCost(model: string | undefined, stats: TranscriptStats): number {\n  if (!model) return 0;\n\n  // Try exact match first\n  let pricing = MODEL_PRICING[model];\n\n  // Try partial match if exact match fails\n  if (!pricing) {\n    const matchingKey = Object.keys(MODEL_PRICING).find(k => model.includes(k) || k.includes(model));\n    if (matchingKey) {\n      pricing = MODEL_PRICING[matchingKey];\n    }\n  }\n\n  if (!pricing) return 0;\n\n  // Calculate cost with proper cache pricing\n  const inputCost = stats.inputTokens * pricing.input;\n  const cacheWriteCost = stats.cacheCreationTokens * pricing.cacheWrite;\n  const cacheReadCost = stats.cacheReadTokens * pricing.cacheRead;\n  const outputCost = stats.outputTokens * pricing.output;\n\n  return (inputCost + cacheWriteCost + cacheReadCost + outputCost) / 1_000_000;\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    cacheCreationTokens?: number;\n    cacheReadTokens?: number;\n    messageCount?: number;\n    toolCallCount?: number;\n    syncedMessageUuids?: Set<string> | string[];\n    startedAt?: string;\n    endedAt?: string;\n    durationMs?: number;\n    title?: string;\n    cwd?: string;\n    costEstimate?: number;\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, token usage, and stats\ninterface ParsedTranscript {\n  assistantMessages: Array<{\n    uuid: string;\n    text: string;\n    timestamp: string;\n    model?: string;\n  }>;\n  stats: TranscriptStats;\n}\n\nfunction parseTranscriptFile(transcriptPath: string): ParsedTranscript {\n  const result: ParsedTranscript = {\n    assistantMessages: [],\n    stats: {\n      model: undefined,\n      inputTokens: 0,\n      cacheCreationTokens: 0,\n      cacheReadTokens: 0,\n      outputTokens: 0,\n      messageCount: 0,\n      toolCallCount: 0,\n      title: undefined,\n      cwd: undefined,\n      startedAt: undefined,\n      endedAt: undefined,\n      durationMs: undefined,\n    },\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    let firstTimestamp: string | undefined;\n    let lastTimestamp: string | undefined;\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        // Track timestamps for duration calculation\n        if (entry.timestamp) {\n          if (!firstTimestamp) {\n            firstTimestamp = entry.timestamp;\n          }\n          lastTimestamp = entry.timestamp;\n        }\n\n        // Get cwd and title from first entry that has them\n        if (!result.stats.cwd && entry.cwd) {\n          result.stats.cwd = entry.cwd;\n        }\n        if (!result.stats.title && entry.slug) {\n          result.stats.title = entry.slug;\n        }\n\n        // Count user messages\n        if (entry.type === \"user\") {\n          result.stats.messageCount++;\n        }\n\n        // Process assistant messages\n        if (entry.type === \"assistant\" && entry.message) {\n          const msg = entry.message;\n          const uuid = entry.uuid || \"\";\n\n          // Get model from first assistant message\n          if (msg.model && !result.stats.model) {\n            result.stats.model = msg.model;\n          }\n\n          // Skip duplicate UUIDs for message extraction\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              // Count tool uses\n              if (part.type === \"tool_use\") {\n                result.stats.toolCallCount++;\n              }\n            }\n          }\n\n          // Accumulate token usage (separate cache tokens for cost calculation)\n          if (msg.usage) {\n            result.stats.inputTokens += msg.usage.input_tokens || 0;\n            result.stats.cacheCreationTokens += msg.usage.cache_creation_input_tokens || 0;\n            result.stats.cacheReadTokens += msg.usage.cache_read_input_tokens || 0;\n            result.stats.outputTokens += msg.usage.output_tokens || 0;\n          }\n        }\n      } catch {\n        // Skip malformed lines\n      }\n    }\n\n    // Calculate duration from timestamps\n    if (firstTimestamp && lastTimestamp) {\n      result.stats.startedAt = firstTimestamp;\n      result.stats.endedAt = lastTimestamp;\n      const startMs = new Date(firstTimestamp).getTime();\n      const endMs = new Date(lastTimestamp).getTime();\n      if (!isNaN(startMs) && !isNaN(endMs)) {\n        result.stats.durationMs = endMs - startMs;\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          // Parse transcript if available to get initial info\n          let stats: TranscriptStats | undefined;\n          if (data.transcript_path && fs.existsSync(data.transcript_path)) {\n            const parsed = parseTranscriptFile(data.transcript_path);\n            stats = parsed.stats;\n          }\n\n          const cwd = stats?.cwd || data.cwd;\n          const startedAt = new Date().toISOString();\n          \n          // Initialize session state\n          sessionState[data.session_id] = {\n            model: stats?.model || data.model,\n            tokenUsage: { input: 0, output: 0 },\n            cacheCreationTokens: 0,\n            cacheReadTokens: 0,\n            messageCount: 0,\n            toolCallCount: 0,\n            startedAt,\n            title: stats?.title,\n            cwd,\n          };\n          saveSessionState(sessionState);\n          \n          const session: SessionData = {\n            sessionId: data.session_id,\n            source: \"claude-code\",\n            cwd,\n            model: stats?.model || data.model,\n            title: stats?.title,\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,\n            projectPath: cwd,\n            projectName: cwd ? path.basename(cwd) : undefined,\n          };\n\n          // Try to get git branch from cwd\n          if (cwd) {\n            try {\n              const gitDir = path.join(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          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          // Parse transcript to get final stats including model, tokens, cost\n          let stats: TranscriptStats = {\n            model: state.model,\n            inputTokens: state.tokenUsage?.input || 0,\n            cacheCreationTokens: state.cacheCreationTokens || 0,\n            cacheReadTokens: state.cacheReadTokens || 0,\n            outputTokens: state.tokenUsage?.output || 0,\n            messageCount: state.messageCount || 0,\n            toolCallCount: state.toolCallCount || 0,\n            title: state.title,\n            cwd: state.cwd || data.cwd,\n            startedAt: state.startedAt,\n            endedAt: state.endedAt,\n            durationMs: state.durationMs,\n          };\n\n          if (data.transcript_path && fs.existsSync(data.transcript_path)) {\n            const parsed = parseTranscriptFile(data.transcript_path);\n            stats = parsed.stats;\n          }\n\n          // Calculate cost from tokens (with proper cache pricing)\n          let cost = data.cost_estimate;\n          if ((cost === undefined || cost === null || cost === 0) && stats.model) {\n            if (stats.inputTokens || stats.outputTokens || stats.cacheReadTokens || stats.cacheCreationTokens) {\n              cost = calculateCost(stats.model, stats);\n            }\n          }\n\n          const cwd = stats.cwd || data.cwd;\n          const totalInputTokens = stats.inputTokens + stats.cacheCreationTokens + stats.cacheReadTokens;\n          const endedAt = stats.endedAt || new Date().toISOString();\n\n          const session: SessionData = {\n            sessionId: data.session_id,\n            source: \"claude-code\",\n            model: stats.model,\n            title: stats.title || (state.firstPrompt ? generateTitle(state.firstPrompt) : undefined),\n            cwd,\n            projectPath: cwd,\n            projectName: cwd ? path.basename(cwd) : undefined,\n            endReason: data.reason,\n            messageCount: data.message_count || stats.messageCount,\n            toolCallCount: data.tool_call_count || stats.toolCallCount,\n            tokenUsage: {\n              input: totalInputTokens,\n              output: stats.outputTokens,\n            },\n            costEstimate: cost,\n            startedAt: stats.startedAt,\n            endedAt,\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            const { stats } = transcript;\n            \n            // Update token usage from transcript (track cache tokens separately)\n            if (stats.inputTokens > 0 || stats.outputTokens > 0 || stats.cacheReadTokens > 0) {\n              const totalInput = stats.inputTokens + stats.cacheCreationTokens + stats.cacheReadTokens;\n              state.tokenUsage = { input: totalInput, output: stats.outputTokens };\n              state.cacheCreationTokens = stats.cacheCreationTokens;\n              state.cacheReadTokens = stats.cacheReadTokens;\n            }\n\n            // Update model if found\n            if (stats.model && !state.model) {\n              state.model = stats.model;\n            }\n\n            // Update title if found from transcript slug\n            if (stats.title && !state.title) {\n              state.title = stats.title;\n            }\n\n            // Update duration info\n            if (stats.startedAt) state.startedAt = stats.startedAt;\n            if (stats.endedAt) state.endedAt = stats.endedAt;\n            if (stats.durationMs) state.durationMs = stats.durationMs;\n\n            // Update tool call count\n            if (stats.toolCallCount > 0) {\n              state.toolCallCount = stats.toolCallCount;\n            }\n\n            // Calculate cost if we have model and tokens\n            if (stats.model && (stats.inputTokens || stats.outputTokens || stats.cacheReadTokens)) {\n              state.costEstimate = calculateCost(stats.model, stats);\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, model, and cost from transcript\n          const session: SessionData = {\n            sessionId: data.session_id,\n            source: \"claude-code\",\n            model: state.model,\n            tokenUsage: state.tokenUsage,\n            messageCount: state.messageCount,\n            toolCallCount: state.toolCallCount,\n            costEstimate: state.costEstimate,\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"]}