claude-code-sync 0.1.8 → 0.1.10
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 +122 -6
- package/dist/index.js +65 -29
- package/package.json +1 -1
package/dist/cli.js
CHANGED
|
@@ -466,6 +466,43 @@ program
|
|
|
466
466
|
// ============================================================================
|
|
467
467
|
// Hook Command (for Claude Code integration)
|
|
468
468
|
// ============================================================================
|
|
469
|
+
// Track session state for title generation (first user prompt)
|
|
470
|
+
const SESSION_STATE_FILE = path.join(process.env.HOME || "~", ".config", "claude-code-sync", "session-state.json");
|
|
471
|
+
function loadSessionState() {
|
|
472
|
+
try {
|
|
473
|
+
if (fs.existsSync(SESSION_STATE_FILE)) {
|
|
474
|
+
return JSON.parse(fs.readFileSync(SESSION_STATE_FILE, "utf-8"));
|
|
475
|
+
}
|
|
476
|
+
}
|
|
477
|
+
catch {
|
|
478
|
+
// Ignore errors
|
|
479
|
+
}
|
|
480
|
+
return {};
|
|
481
|
+
}
|
|
482
|
+
function saveSessionState(state) {
|
|
483
|
+
try {
|
|
484
|
+
const dir = path.dirname(SESSION_STATE_FILE);
|
|
485
|
+
if (!fs.existsSync(dir)) {
|
|
486
|
+
fs.mkdirSync(dir, { recursive: true });
|
|
487
|
+
}
|
|
488
|
+
fs.writeFileSync(SESSION_STATE_FILE, JSON.stringify(state, null, 2));
|
|
489
|
+
}
|
|
490
|
+
catch {
|
|
491
|
+
// Ignore errors
|
|
492
|
+
}
|
|
493
|
+
}
|
|
494
|
+
function generateTitle(prompt) {
|
|
495
|
+
// Use first 80 chars of first prompt as title, trim at word boundary
|
|
496
|
+
const trimmed = prompt.slice(0, 80).trim();
|
|
497
|
+
if (prompt.length > 80) {
|
|
498
|
+
const lastSpace = trimmed.lastIndexOf(" ");
|
|
499
|
+
if (lastSpace > 40) {
|
|
500
|
+
return trimmed.slice(0, lastSpace) + "...";
|
|
501
|
+
}
|
|
502
|
+
return trimmed + "...";
|
|
503
|
+
}
|
|
504
|
+
return trimmed;
|
|
505
|
+
}
|
|
469
506
|
program
|
|
470
507
|
.command("hook <event>")
|
|
471
508
|
.description("Handle Claude Code hook events (reads stdin)")
|
|
@@ -488,14 +525,25 @@ program
|
|
|
488
525
|
}
|
|
489
526
|
try {
|
|
490
527
|
const client = new index_1.SyncClient(config);
|
|
528
|
+
const sessionState = loadSessionState();
|
|
491
529
|
switch (event) {
|
|
492
530
|
case "SessionStart": {
|
|
493
531
|
const data = JSON.parse(input);
|
|
532
|
+
// Initialize session state
|
|
533
|
+
sessionState[data.session_id] = {
|
|
534
|
+
model: data.model,
|
|
535
|
+
tokenUsage: { input: 0, output: 0 },
|
|
536
|
+
messageCount: 0,
|
|
537
|
+
};
|
|
538
|
+
saveSessionState(sessionState);
|
|
494
539
|
const session = {
|
|
495
540
|
sessionId: data.session_id,
|
|
496
541
|
source: "claude-code",
|
|
497
542
|
cwd: data.cwd,
|
|
543
|
+
model: data.model,
|
|
498
544
|
permissionMode: data.permission_mode,
|
|
545
|
+
thinkingEnabled: data.thinking_enabled,
|
|
546
|
+
mcpServers: data.mcp_servers,
|
|
499
547
|
startType: data.source === "startup" ? "new" : data.source,
|
|
500
548
|
startedAt: new Date().toISOString(),
|
|
501
549
|
projectPath: data.cwd,
|
|
@@ -506,24 +554,53 @@ program
|
|
|
506
554
|
}
|
|
507
555
|
case "SessionEnd": {
|
|
508
556
|
const data = JSON.parse(input);
|
|
557
|
+
const state = sessionState[data.session_id] || {};
|
|
558
|
+
// Calculate final token usage
|
|
559
|
+
const finalTokenUsage = data.total_token_usage || state.tokenUsage;
|
|
509
560
|
const session = {
|
|
510
561
|
sessionId: data.session_id,
|
|
511
562
|
source: "claude-code",
|
|
563
|
+
title: state.firstPrompt ? generateTitle(state.firstPrompt) : undefined,
|
|
512
564
|
endReason: data.reason,
|
|
565
|
+
messageCount: data.message_count || state.messageCount,
|
|
566
|
+
toolCallCount: data.tool_call_count,
|
|
567
|
+
tokenUsage: finalTokenUsage,
|
|
568
|
+
costEstimate: data.cost_estimate,
|
|
513
569
|
endedAt: new Date().toISOString(),
|
|
514
570
|
};
|
|
515
571
|
await client.syncSession(session);
|
|
572
|
+
// Clean up session state
|
|
573
|
+
delete sessionState[data.session_id];
|
|
574
|
+
saveSessionState(sessionState);
|
|
516
575
|
break;
|
|
517
576
|
}
|
|
518
577
|
case "UserPromptSubmit": {
|
|
519
578
|
const data = JSON.parse(input);
|
|
579
|
+
const state = sessionState[data.session_id] || {};
|
|
580
|
+
// Track first prompt for title generation
|
|
581
|
+
if (!state.firstPrompt) {
|
|
582
|
+
state.firstPrompt = data.prompt;
|
|
583
|
+
sessionState[data.session_id] = state;
|
|
584
|
+
saveSessionState(sessionState);
|
|
585
|
+
// Update session with title
|
|
586
|
+
const session = {
|
|
587
|
+
sessionId: data.session_id,
|
|
588
|
+
source: "claude-code",
|
|
589
|
+
title: generateTitle(data.prompt),
|
|
590
|
+
};
|
|
591
|
+
await client.syncSession(session);
|
|
592
|
+
}
|
|
593
|
+
// Increment message count
|
|
594
|
+
state.messageCount = (state.messageCount || 0) + 1;
|
|
595
|
+
sessionState[data.session_id] = state;
|
|
596
|
+
saveSessionState(sessionState);
|
|
520
597
|
const message = {
|
|
521
598
|
sessionId: data.session_id,
|
|
522
|
-
messageId: `${data.session_id}-${Date.now()}`,
|
|
599
|
+
messageId: `${data.session_id}-user-${Date.now()}`,
|
|
523
600
|
source: "claude-code",
|
|
524
601
|
role: "user",
|
|
525
602
|
content: data.prompt,
|
|
526
|
-
timestamp: new Date().toISOString(),
|
|
603
|
+
timestamp: data.timestamp || new Date().toISOString(),
|
|
527
604
|
};
|
|
528
605
|
await client.syncMessage(message);
|
|
529
606
|
break;
|
|
@@ -532,22 +609,61 @@ program
|
|
|
532
609
|
if (!config.syncToolCalls)
|
|
533
610
|
break;
|
|
534
611
|
const data = JSON.parse(input);
|
|
612
|
+
const state = sessionState[data.session_id] || {};
|
|
535
613
|
const message = {
|
|
536
614
|
sessionId: data.session_id,
|
|
537
|
-
messageId: `${data.session_id}-tool-${Date.now()}`,
|
|
615
|
+
messageId: data.tool_use_id || `${data.session_id}-tool-${Date.now()}`,
|
|
538
616
|
source: "claude-code",
|
|
539
617
|
role: "assistant",
|
|
540
618
|
toolName: data.tool_name,
|
|
541
619
|
toolArgs: data.tool_input,
|
|
542
620
|
toolResult: data.tool_result?.output || data.tool_result?.error,
|
|
621
|
+
durationMs: data.duration_ms,
|
|
543
622
|
timestamp: new Date().toISOString(),
|
|
544
623
|
};
|
|
545
624
|
await client.syncMessage(message);
|
|
546
625
|
break;
|
|
547
626
|
}
|
|
548
627
|
case "Stop": {
|
|
549
|
-
// Stop event
|
|
550
|
-
|
|
628
|
+
// Stop event contains the assistant's response and token usage
|
|
629
|
+
const data = JSON.parse(input);
|
|
630
|
+
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
|
+
};
|
|
638
|
+
}
|
|
639
|
+
// Increment message count for assistant response
|
|
640
|
+
state.messageCount = (state.messageCount || 0) + 1;
|
|
641
|
+
sessionState[data.session_id] = state;
|
|
642
|
+
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
|
|
660
|
+
const session = {
|
|
661
|
+
sessionId: data.session_id,
|
|
662
|
+
source: "claude-code",
|
|
663
|
+
tokenUsage: state.tokenUsage,
|
|
664
|
+
messageCount: state.messageCount,
|
|
665
|
+
};
|
|
666
|
+
await client.syncSession(session);
|
|
551
667
|
break;
|
|
552
668
|
}
|
|
553
669
|
default:
|
|
@@ -564,4 +680,4 @@ program
|
|
|
564
680
|
});
|
|
565
681
|
// Parse and run
|
|
566
682
|
program.parse();
|
|
567
|
-
//# 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  permission_mode?: string;\n  source?: string;\n}\n\ninterface HookSessionEndData {\n  session_id: string;\n  reason?: \"user_stop\" | \"max_turns\" | \"error\" | \"completed\";\n}\n\ninterface HookUserPromptData {\n  session_id: string;\n  prompt: string;\n}\n\ninterface HookToolUseData {\n  session_id: string;\n  tool_name: string;\n  tool_input?: Record<string, unknown>;\n  tool_result?: {\n    output?: string;\n    error?: string;\n  };\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\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\n      switch (event) {\n        case \"SessionStart\": {\n          const data = JSON.parse(input) as HookSessionStartData;\n          const session: SessionData = {\n            sessionId: data.session_id,\n            source: \"claude-code\",\n            cwd: data.cwd,\n            permissionMode: data.permission_mode,\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 session: SessionData = {\n            sessionId: data.session_id,\n            source: \"claude-code\",\n            endReason: data.reason,\n            endedAt: new Date().toISOString(),\n          };\n          await client.syncSession(session);\n          break;\n        }\n\n        case \"UserPromptSubmit\": {\n          const data = JSON.parse(input) as HookUserPromptData;\n          const message: MessageData = {\n            sessionId: data.session_id,\n            messageId: `${data.session_id}-${Date.now()}`,\n            source: \"claude-code\",\n            role: \"user\",\n            content: data.prompt,\n            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 message: MessageData = {\n            sessionId: data.session_id,\n            messageId: `${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            timestamp: new Date().toISOString(),\n          };\n          await client.syncMessage(message);\n          break;\n        }\n\n        case \"Stop\": {\n          // Stop event indicates Claude finished responding\n          // We could track this but for now just acknowledge\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"]}
|
|
683
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cli.js","sourceRoot":"","sources":["../src/cli.ts"],"names":[],"mappings":";;AAEA;;;;;;;;GAQG;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAEH,yCAAoC;AACpC,mDAAqC;AACrC,uCAAyB;AACzB,2CAA6B;AAC7B,mCAQiB;AAkEjB,iCAAiC;AACjC,SAAS,UAAU;IACjB,IAAI,CAAC;QACH,MAAM,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,EAAE,cAAc,CAAC,CAAC;QAC3D,MAAM,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,YAAY,CAAC,OAAO,EAAE,OAAO,CAAC,CAAgB,CAAC;QACzE,OAAO,GAAG,CAAC,OAAO,IAAI,OAAO,CAAC;IAChC,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,OAAO,CAAC;IACjB,CAAC;AACH,CAAC;AAED,MAAM,OAAO,GAAG,IAAI,mBAAO,EAAE,CAAC;AAE9B,OAAO;KACJ,IAAI,CAAC,kBAAkB,CAAC;KACxB,WAAW,CAAC,iDAAiD,CAAC;KAC9D,OAAO,CAAC,UAAU,EAAE,CAAC,CAAC;AAEzB,+EAA+E;AAC/E,mBAAmB;AACnB,+EAA+E;AAE/E,SAAS,MAAM,CAAC,QAAgB;IAC9B,MAAM,EAAE,GAAG,QAAQ,CAAC,eAAe,CAAC;QAClC,KAAK,EAAE,OAAO,CAAC,KAAK;QACpB,MAAM,EAAE,OAAO,CAAC,MAAM;KACvB,CAAC,CAAC;IAEH,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;QAC7B,EAAE,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC,MAAM,EAAE,EAAE;YAC/B,EAAE,CAAC,KAAK,EAAE,CAAC;YACX,OAAO,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC,CAAC;QACzB,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC;AAED,SAAS,UAAU,CAAC,GAAW;IAC7B,IAAI,GAAG,CAAC,MAAM,IAAI,CAAC;QAAE,OAAO,MAAM,CAAC;IACnC,OAAO,GAAG,CAAC,SAAS,CAAC,CAAC,EAAE,CAAC,CAAC,GAAG,MAAM,GAAG,GAAG,CAAC,SAAS,CAAC,GAAG,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;AACtE,CAAC;AAED,+EAA+E;AAC/E,WAAW;AACX,+EAA+E;AAE/E,OAAO;KACJ,OAAO,CAAC,OAAO,CAAC;KAChB,WAAW,CAAC,kCAAkC,CAAC;KAC/C,MAAM,CAAC,KAAK,IAAI,EAAE;IACjB,OAAO,CAAC,GAAG,CAAC,gCAAgC,CAAC,CAAC;IAC9C,OAAO,CAAC,GAAG,CAAC,yFAAyF,CAAC,CAAC;IACvG,OAAO,CAAC,GAAG,CAAC,qBAAqB,CAAC,CAAC;IACnC,OAAO,CAAC,GAAG,CAAC,+BAA+B,CAAC,CAAC;IAC7C,OAAO,CAAC,GAAG,CAAC,wCAAwC,CAAC,CAAC;IAEtD,MAAM,SAAS,GAAG,MAAM,MAAM,CAAC,wDAAwD,CAAC,CAAC;IAEzF,IAAI,CAAC,SAAS,EAAE,CAAC;QACf,OAAO,CAAC,KAAK,CAAC,+BAA+B,CAAC,CAAC;QAC/C,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAED,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,cAAc,CAAC,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,aAAa,CAAC,EAAE,CAAC;QAC9E,OAAO,CAAC,KAAK,CAAC,qEAAqE,CAAC,CAAC;QACrF,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAED,MAAM,MAAM,GAAG,MAAM,MAAM,CAAC,qBAAqB,CAAC,CAAC;IAEnD,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,OAAO,CAAC,KAAK,CAAC,4BAA4B,CAAC,CAAC;QAC5C,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAED,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,MAAM,CAAC,EAAE,CAAC;QAC/B,OAAO,CAAC,KAAK,CAAC,8CAA8C,CAAC,CAAC;QAC9D,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAED,MAAM,MAAM,GAAW;QACrB,SAAS;QACT,MAAM;QACN,QAAQ,EAAE,IAAI;QACd,aAAa,EAAE,IAAI;QACnB,YAAY,EAAE,KAAK;KACpB,CAAC;IAEF,kBAAkB;IAClB,OAAO,CAAC,GAAG,CAAC,yBAAyB,CAAC,CAAC;IACvC,MAAM,MAAM,GAAG,IAAI,kBAAU,CAAC,MAAM,CAAC,CAAC;IACtC,MAAM,SAAS,GAAG,MAAM,MAAM,CAAC,cAAc,EAAE,CAAC;IAEhD,IAAI,CAAC,SAAS,EAAE,CAAC;QACf,OAAO,CAAC,KAAK,CAAC,4CAA4C,CAAC,CAAC;QAC5D,OAAO,CAAC,KAAK,CAAC,iCAAiC,CAAC,CAAC;QACjD,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAED,cAAc;IACd,IAAA,kBAAU,EAAC,MAAM,CAAC,CAAC;IACnB,OAAO,CAAC,GAAG,CAAC,wBAAwB,CAAC,CAAC;IACtC,OAAO,CAAC,GAAG,CAAC,WAAW,SAAS,EAAE,CAAC,CAAC;IACpC,OAAO,CAAC,GAAG,CAAC,WAAW,UAAU,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;IAC7C,OAAO,CAAC,GAAG,CAAC,sEAAsE,CAAC,CAAC;IACpF,OAAO,CAAC,GAAG,CAAC,6BAA6B,CAAC,CAAC;AAC7C,CAAC,CAAC,CAAC;AAEL,OAAO;KACJ,OAAO,CAAC,QAAQ,CAAC;KACjB,WAAW,CAAC,0BAA0B,CAAC;KACvC,MAAM,CAAC,GAAG,EAAE;IACX,IAAA,mBAAW,GAAE,CAAC;IACd,OAAO,CAAC,GAAG,CAAC,qBAAqB,CAAC,CAAC;AACrC,CAAC,CAAC,CAAC;AAEL,OAAO;KACJ,OAAO,CAAC,QAAQ,CAAC;KACjB,WAAW,CAAC,wBAAwB,CAAC;KACrC,MAAM,CAAC,KAAK,IAAI,EAAE;IACjB,MAAM,MAAM,GAAG,IAAA,kBAAU,GAAE,CAAC;IAE5B,OAAO,CAAC,GAAG,CAAC,iCAAiC,CAAC,CAAC;IAE/C,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,OAAO,CAAC,GAAG,CAAC,gBAAgB,CAAC,CAAC;QAC9B,OAAO,CAAC,GAAG,CAAC,6CAA6C,CAAC,CAAC;QAC3D,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAED,OAAO,CAAC,GAAG,CAAC,gBAAgB,CAAC,CAAC;IAC9B,OAAO,CAAC,GAAG,CAAC,iBAAiB,MAAM,CAAC,SAAS,EAAE,CAAC,CAAC;IACjD,OAAO,CAAC,GAAG,CAAC,iBAAiB,UAAU,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;IAC1D,OAAO,CAAC,GAAG,CAAC,iBAAiB,MAAM,CAAC,QAAQ,KAAK,KAAK,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,UAAU,EAAE,CAAC,CAAC;IACnF,OAAO,CAAC,GAAG,CAAC,iBAAiB,MAAM,CAAC,aAAa,KAAK,KAAK,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,UAAU,EAAE,CAAC,CAAC;IACxF,OAAO,CAAC,GAAG,CAAC,iBAAiB,MAAM,CAAC,YAAY,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,UAAU,EAAE,CAAC,CAAC;IAE7E,OAAO,CAAC,GAAG,CAAC,yBAAyB,CAAC,CAAC;IACvC,MAAM,MAAM,GAAG,IAAI,kBAAU,CAAC,MAAM,CAAC,CAAC;IACtC,MAAM,SAAS,GAAG,MAAM,MAAM,CAAC,cAAc,EAAE,CAAC;IAEhD,IAAI,SAAS,EAAE,CAAC;QACd,OAAO,CAAC,GAAG,CAAC,+BAA+B,CAAC,CAAC;IAC/C,CAAC;SAAM,CAAC;QACN,OAAO,CAAC,GAAG,CAAC,8CAA8C,CAAC,CAAC;QAC5D,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;AACH,CAAC,CAAC,CAAC;AAEL,OAAO;KACJ,OAAO,CAAC,QAAQ,CAAC;KACjB,WAAW,CAAC,4BAA4B,CAAC;KACzC,MAAM,CAAC,QAAQ,EAAE,gBAAgB,CAAC;KAClC,MAAM,CAAC,CAAC,OAA2B,EAAE,EAAE;IACtC,MAAM,MAAM,GAAG,IAAA,kBAAU,GAAE,CAAC;IAE5B,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,IAAI,OAAO,CAAC,IAAI,EAAE,CAAC;YACjB,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,UAAU,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC;QACrD,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,GAAG,CAAC,yDAAyD,CAAC,CAAC;QACzE,CAAC;QACD,OAAO;IACT,CAAC;IAED,IAAI,OAAO,CAAC,IAAI,EAAE,CAAC;QACjB,OAAO,CAAC,GAAG,CACT,IAAI,CAAC,SAAS,CACZ;YACE,UAAU,EAAE,IAAI;YAChB,SAAS,EAAE,MAAM,CAAC,SAAS;YAC3B,MAAM,EAAE,UAAU,CAAC,MAAM,CAAC,MAAM,CAAC;YACjC,QAAQ,EAAE,MAAM,CAAC,QAAQ,KAAK,KAAK;YACnC,aAAa,EAAE,MAAM,CAAC,aAAa,KAAK,KAAK;YAC7C,YAAY,EAAE,MAAM,CAAC,YAAY,KAAK,IAAI;SAC3C,EACD,IAAI,EACJ,CAAC,CACF,CACF,CAAC;IACJ,CAAC;SAAM,CAAC;QACN,OAAO,CAAC,GAAG,CAAC,6BAA6B,CAAC,CAAC;QAC3C,OAAO,CAAC,GAAG,CAAC,gBAAgB,MAAM,CAAC,SAAS,EAAE,CAAC,CAAC;QAChD,OAAO,CAAC,GAAG,CAAC,gBAAgB,UAAU,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;QACzD,OAAO,CAAC,GAAG,CAAC,gBAAgB,MAAM,CAAC,QAAQ,KAAK,KAAK,EAAE,CAAC,CAAC;QACzD,OAAO,CAAC,GAAG,CAAC,gBAAgB,MAAM,CAAC,aAAa,KAAK,KAAK,EAAE,CAAC,CAAC;QAC9D,OAAO,CAAC,GAAG,CAAC,gBAAgB,MAAM,CAAC,YAAY,KAAK,IAAI,EAAE,CAAC,CAAC;QAC5D,OAAO,CAAC,GAAG,CAAC,yDAAyD,CAAC,CAAC;IACzE,CAAC;AACH,CAAC,CAAC,CAAC;AAEL,OAAO;KACJ,OAAO,CAAC,mBAAmB,CAAC;KAC5B,WAAW,CAAC,2BAA2B,CAAC;KACxC,MAAM,CAAC,CAAC,GAAW,EAAE,KAAa,EAAE,EAAE;IACrC,MAAM,MAAM,GAAG,IAAA,kBAAU,GAAE,CAAC;IAE5B,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,OAAO,CAAC,KAAK,CAAC,qDAAqD,CAAC,CAAC;QACrE,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAED,MAAM,SAAS,GAAG,CAAC,UAAU,EAAE,eAAe,EAAE,cAAc,CAAC,CAAC;IAChE,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,CAAC;QAC7B,OAAO,CAAC,KAAK,CAAC,4BAA4B,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAClE,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAED,MAAM,SAAS,GAAG,KAAK,KAAK,MAAM,IAAI,KAAK,KAAK,GAAG,IAAI,KAAK,KAAK,KAAK,CAAC;IAEvE,0BAA0B;IAC1B,IAAI,GAAG,KAAK,UAAU,EAAE,CAAC;QACvB,MAAM,CAAC,QAAQ,GAAG,SAAS,CAAC;IAC9B,CAAC;SAAM,IAAI,GAAG,KAAK,eAAe,EAAE,CAAC;QACnC,MAAM,CAAC,aAAa,GAAG,SAAS,CAAC;IACnC,CAAC;SAAM,IAAI,GAAG,KAAK,cAAc,EAAE,CAAC;QAClC,MAAM,CAAC,YAAY,GAAG,SAAS,CAAC;IAClC,CAAC;IAED,IAAA,kBAAU,EAAC,MAAM,CAAC,CAAC;IACnB,OAAO,CAAC,GAAG,CAAC,OAAO,GAAG,MAAM,SAAS,EAAE,CAAC,CAAC;AAC3C,CAAC,CAAC,CAAC;AAEL,+EAA+E;AAC/E,+CAA+C;AAC/C,+EAA+E;AAE/E,kCAAkC;AAClC,MAAM,mBAAmB,GAAG;IAC1B,KAAK,EAAE;QACL,YAAY,EAAE;YACZ;gBACE,KAAK,EAAE;oBACL;wBACE,IAAI,EAAE,SAAS;wBACf,OAAO,EAAE,oCAAoC;qBAC9C;iBACF;aACF;SACF;QACD,UAAU,EAAE;YACV;gBACE,KAAK,EAAE;oBACL;wBACE,IAAI,EAAE,SAAS;wBACf,OAAO,EAAE,kCAAkC;qBAC5C;iBACF;aACF;SACF;QACD,gBAAgB,EAAE;YAChB;gBACE,KAAK,EAAE;oBACL;wBACE,IAAI,EAAE,SAAS;wBACf,OAAO,EAAE,wCAAwC;qBAClD;iBACF;aACF;SACF;QACD,WAAW,EAAE;YACX;gBACE,OAAO,EAAE,GAAG;gBACZ,KAAK,EAAE;oBACL;wBACE,IAAI,EAAE,SAAS;wBACf,OAAO,EAAE,mCAAmC;qBAC7C;iBACF;aACF;SACF;QACD,IAAI,EAAE;YACJ;gBACE,OAAO,EAAE,GAAG;gBACZ,KAAK,EAAE;oBACL;wBACE,IAAI,EAAE,SAAS;wBACf,OAAO,EAAE,4BAA4B;qBACtC;iBACF;aACF;SACF;KACF;CACF,CAAC;AAEF,OAAO;KACJ,OAAO,CAAC,OAAO,CAAC;KAChB,WAAW,CAAC,wEAAwE,CAAC;KACrF,MAAM,CAAC,SAAS,EAAE,wCAAwC,CAAC;KAC3D,MAAM,CAAC,KAAK,EAAE,OAA4B,EAAE,EAAE;IAC7C,MAAM,SAAS,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,IAAI,GAAG,EAAE,SAAS,CAAC,CAAC;IAChE,MAAM,YAAY,GAAG,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,eAAe,CAAC,CAAC;IAE3D,OAAO,CAAC,GAAG,CAAC,gCAAgC,CAAC,CAAC;IAE9C,6CAA6C;IAC7C,MAAM,MAAM,GAAG,IAAA,kBAAU,GAAE,CAAC;IAC5B,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,OAAO,CAAC,GAAG,CAAC,qCAAqC,CAAC,CAAC;QACnD,OAAO,CAAC,GAAG,CAAC,gEAAgE,CAAC,CAAC;IAChF,CAAC;IAED,qCAAqC;IACrC,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,SAAS,CAAC,EAAE,CAAC;QAC9B,EAAE,CAAC,SAAS,CAAC,SAAS,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QAC7C,OAAO,CAAC,GAAG,CAAC,6BAA6B,CAAC,CAAC;IAC7C,CAAC;IAED,8BAA8B;IAC9B,IAAI,gBAAgB,GAAmB,EAAE,CAAC;IAC1C,IAAI,gBAAgB,GAAG,KAAK,CAAC;IAE7B,IAAI,EAAE,CAAC,UAAU,CAAC,YAAY,CAAC,EAAE,CAAC;QAChC,IAAI,CAAC;YACH,MAAM,OAAO,GAAG,EAAE,CAAC,YAAY,CAAC,YAAY,EAAE,OAAO,CAAC,CAAC;YACvD,gBAAgB,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,CAAmB,CAAC;YACzD,gBAAgB,GAAG,CAAC,CAAC,gBAAgB,CAAC,KAAK,CAAC;YAC5C,OAAO,CAAC,GAAG,CAAC,8BAA8B,CAAC,CAAC;QAC9C,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,CAAC,GAAG,CAAC,sEAAsE,CAAC,CAAC;QACtF,CAAC;IACH,CAAC;IAED,wBAAwB;IACxB,IAAI,gBAAgB,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC;QACvC,OAAO,CAAC,GAAG,CAAC,uCAAuC,CAAC,CAAC;QACrD,OAAO,CAAC,GAAG,CAAC,6DAA6D,CAAC,CAAC;QAC3E,OAAO,CAAC,GAAG,CAAC,6DAA6D,CAAC,CAAC;QAC3E,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,mBAAmB,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QAC1D,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAED,iBAAiB;IACjB,MAAM,WAAW,GAAG;QAClB,GAAG,gBAAgB;QACnB,GAAG,mBAAmB;KACvB,CAAC;IAEF,iBAAiB;IACjB,IAAI,CAAC;QACH,EAAE,CAAC,aAAa,CAAC,YAAY,EAAE,IAAI,CAAC,SAAS,CAAC,WAAW,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QACrE,OAAO,CAAC,GAAG,CAAC,iCAAiC,CAAC,CAAC;QAC/C,OAAO,CAAC,GAAG,CAAC,qBAAqB,YAAY,EAAE,CAAC,CAAC;QACjD,OAAO,CAAC,GAAG,CAAC,uDAAuD,CAAC,CAAC;IACvE,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,OAAO,CAAC,KAAK,CAAC,yBAAyB,EAAE,KAAK,CAAC,CAAC;QAChD,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;AACH,CAAC,CAAC,CAAC;AAEL,OAAO;KACJ,OAAO,CAAC,QAAQ,CAAC;KACjB,WAAW,CAAC,kDAAkD,CAAC;KAC/D,MAAM,CAAC,KAAK,IAAI,EAAE;IACjB,OAAO,CAAC,GAAG,CAAC,mCAAmC,CAAC,CAAC;IAEjD,oBAAoB;IACpB,MAAM,MAAM,GAAG,IAAA,kBAAU,GAAE,CAAC;IAC5B,IAAI,MAAM,EAAE,CAAC;QACX,OAAO,CAAC,GAAG,CAAC,iBAAiB,CAAC,CAAC;QAC/B,OAAO,CAAC,GAAG,CAAC,kBAAkB,MAAM,CAAC,SAAS,EAAE,CAAC,CAAC;QAClD,OAAO,CAAC,GAAG,CAAC,eAAe,UAAU,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;IAC1D,CAAC;SAAM,CAAC;QACN,OAAO,CAAC,GAAG,CAAC,6BAA6B,CAAC,CAAC;QAC3C,OAAO,CAAC,GAAG,CAAC,2CAA2C,CAAC,CAAC;IAC3D,CAAC;IAED,2BAA2B;IAC3B,MAAM,YAAY,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,IAAI,GAAG,EAAE,SAAS,EAAE,eAAe,CAAC,CAAC;IACpF,IAAI,eAAe,GAAG,KAAK,CAAC;IAE5B,IAAI,EAAE,CAAC,UAAU,CAAC,YAAY,CAAC,EAAE,CAAC;QAChC,IAAI,CAAC;YACH,MAAM,OAAO,GAAG,EAAE,CAAC,YAAY,CAAC,YAAY,EAAE,OAAO,CAAC,CAAC;YACvD,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,CAAmB,CAAC;YACvD,eAAe,GAAG,CAAC,CAAC,QAAQ,CAAC,KAAK,EAAE,YAAY,CAAC;QACnD,CAAC;QAAC,MAAM,CAAC;YACP,sBAAsB;QACxB,CAAC;IACH,CAAC;IAED,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;IAChB,IAAI,eAAe,EAAE,CAAC;QACpB,OAAO,CAAC,GAAG,CAAC,wBAAwB,CAAC,CAAC;QACtC,OAAO,CAAC,GAAG,CAAC,mBAAmB,YAAY,EAAE,CAAC,CAAC;QAC/C,OAAO,CAAC,GAAG,CAAC,uCAAuC,CAAC,CAAC;IACvD,CAAC;SAAM,CAAC;QACN,OAAO,CAAC,GAAG,CAAC,oCAAoC,CAAC,CAAC;QAClD,OAAO,CAAC,GAAG,CAAC,oDAAoD,CAAC,CAAC;IACpE,CAAC;IAED,uCAAuC;IACvC,IAAI,MAAM,EAAE,CAAC;QACX,OAAO,CAAC,GAAG,CAAC,yBAAyB,CAAC,CAAC;QACvC,MAAM,MAAM,GAAG,IAAI,kBAAU,CAAC,MAAM,CAAC,CAAC;QACtC,MAAM,SAAS,GAAG,MAAM,MAAM,CAAC,cAAc,EAAE,CAAC;QAChD,IAAI,SAAS,EAAE,CAAC;YACd,OAAO,CAAC,GAAG,CAAC,kBAAkB,CAAC,CAAC;QAClC,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,GAAG,CAAC,sBAAsB,CAAC,CAAC;YACpC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAClB,CAAC;IACH,CAAC;IAED,IAAI,MAAM,IAAI,eAAe,EAAE,CAAC;QAC9B,OAAO,CAAC,GAAG,CAAC,kEAAkE,CAAC,CAAC;IAClF,CAAC;SAAM,CAAC;QACN,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;QAChB,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;AACH,CAAC,CAAC,CAAC;AAEL,+EAA+E;AAC/E,wCAAwC;AACxC,+EAA+E;AAE/E,OAAO;KACJ,OAAO,CAAC,UAAU,CAAC;KACnB,WAAW,CAAC,6CAA6C,CAAC;KAC1D,MAAM,CAAC,KAAK,IAAI,EAAE;IACjB,MAAM,MAAM,GAAG,IAAA,kBAAU,GAAE,CAAC;IAE5B,OAAO,CAAC,GAAG,CAAC,0CAA0C,CAAC,CAAC;IAExD,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,OAAO,CAAC,GAAG,CAAC,gBAAgB,CAAC,CAAC;QAC9B,OAAO,CAAC,GAAG,CAAC,6CAA6C,CAAC,CAAC;QAC3D,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAED,OAAO,CAAC,GAAG,CAAC,gBAAgB,CAAC,CAAC;IAC9B,OAAO,CAAC,GAAG,CAAC,iBAAiB,MAAM,CAAC,SAAS,EAAE,CAAC,CAAC;IACjD,OAAO,CAAC,GAAG,CAAC,iBAAiB,UAAU,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;IAE1D,OAAO,CAAC,GAAG,CAAC,yBAAyB,CAAC,CAAC;IACvC,MAAM,MAAM,GAAG,IAAI,kBAAU,CAAC,MAAM,CAAC,CAAC;IACtC,MAAM,SAAS,GAAG,MAAM,MAAM,CAAC,cAAc,EAAE,CAAC;IAEhD,IAAI,SAAS,EAAE,CAAC;QACd,OAAO,CAAC,GAAG,CAAC,gBAAgB,CAAC,CAAC;QAE9B,kDAAkD;QAClD,OAAO,CAAC,GAAG,CAAC,4BAA4B,CAAC,CAAC;QAC1C,IAAI,CAAC;YACH,MAAM,WAAW,GAAG;gBAClB,SAAS,EAAE,QAAQ,IAAI,CAAC,GAAG,EAAE,EAAE;gBAC/B,MAAM,EAAE,aAAsB;gBAC9B,KAAK,EAAE,iBAAiB;gBACxB,WAAW,EAAE,UAAU;gBACvB,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;gBACnC,OAAO,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;aAClC,CAAC;YACF,MAAM,MAAM,CAAC,WAAW,CAAC,WAAW,CAAC,CAAC;YACtC,OAAO,CAAC,GAAG,CAAC,mCAAmC,CAAC,CAAC;YACjD,OAAO,CAAC,GAAG,CAAC,2DAA2D,CAAC,CAAC;QAC3E,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,GAAG,CAAC,wBAAwB,KAAK,EAAE,CAAC,CAAC;YAC7C,OAAO,CAAC,GAAG,CAAC,gDAAgD,CAAC,CAAC;YAC9D,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAClB,CAAC;IACH,CAAC;SAAM,CAAC;QACN,OAAO,CAAC,GAAG,CAAC,oBAAoB,CAAC,CAAC;QAClC,OAAO,CAAC,GAAG,CAAC,wCAAwC,CAAC,CAAC;QACtD,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;AACH,CAAC,CAAC,CAAC;AAEL,+EAA+E;AAC/E,6CAA6C;AAC7C,+EAA+E;AAE/E,+DAA+D;AAC/D,MAAM,kBAAkB,GAAG,IAAI,CAAC,IAAI,CAClC,OAAO,CAAC,GAAG,CAAC,IAAI,IAAI,GAAG,EACvB,SAAS,EACT,kBAAkB,EAClB,oBAAoB,CACrB,CAAC;AAWF,SAAS,gBAAgB;IACvB,IAAI,CAAC;QACH,IAAI,EAAE,CAAC,UAAU,CAAC,kBAAkB,CAAC,EAAE,CAAC;YACtC,OAAO,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,YAAY,CAAC,kBAAkB,EAAE,OAAO,CAAC,CAAiB,CAAC;QAClF,CAAC;IACH,CAAC;IAAC,MAAM,CAAC;QACP,gBAAgB;IAClB,CAAC;IACD,OAAO,EAAE,CAAC;AACZ,CAAC;AAED,SAAS,gBAAgB,CAAC,KAAmB;IAC3C,IAAI,CAAC;QACH,MAAM,GAAG,GAAG,IAAI,CAAC,OAAO,CAAC,kBAAkB,CAAC,CAAC;QAC7C,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC;YACxB,EAAE,CAAC,SAAS,CAAC,GAAG,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QACzC,CAAC;QACD,EAAE,CAAC,aAAa,CAAC,kBAAkB,EAAE,IAAI,CAAC,SAAS,CAAC,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;IACvE,CAAC;IAAC,MAAM,CAAC;QACP,gBAAgB;IAClB,CAAC;AACH,CAAC;AAED,SAAS,aAAa,CAAC,MAAc;IACnC,qEAAqE;IACrE,MAAM,OAAO,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,IAAI,EAAE,CAAC;IAC3C,IAAI,MAAM,CAAC,MAAM,GAAG,EAAE,EAAE,CAAC;QACvB,MAAM,SAAS,GAAG,OAAO,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC;QAC3C,IAAI,SAAS,GAAG,EAAE,EAAE,CAAC;YACnB,OAAO,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,SAAS,CAAC,GAAG,KAAK,CAAC;QAC7C,CAAC;QACD,OAAO,OAAO,GAAG,KAAK,CAAC;IACzB,CAAC;IACD,OAAO,OAAO,CAAC;AACjB,CAAC;AAED,OAAO;KACJ,OAAO,CAAC,cAAc,CAAC;KACvB,WAAW,CAAC,8CAA8C,CAAC;KAC3D,MAAM,CAAC,KAAK,EAAE,KAAa,EAAE,EAAE;IAC9B,MAAM,MAAM,GAAG,IAAA,kBAAU,GAAE,CAAC;IAE5B,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,4DAA4D;QAC5D,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAED,IAAI,MAAM,CAAC,QAAQ,KAAK,KAAK,EAAE,CAAC;QAC9B,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAED,6BAA6B;IAC7B,IAAI,KAAK,GAAG,EAAE,CAAC;IACf,IAAI,KAAK,EAAE,MAAM,KAAK,IAAI,OAAO,CAAC,KAAK,EAAE,CAAC;QACxC,KAAK,IAAI,KAAK,CAAC;IACjB,CAAC;IAED,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE,EAAE,CAAC;QAClB,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAED,IAAI,CAAC;QACH,MAAM,MAAM,GAAG,IAAI,kBAAU,CAAC,MAAM,CAAC,CAAC;QACtC,MAAM,YAAY,GAAG,gBAAgB,EAAE,CAAC;QAExC,QAAQ,KAAK,EAAE,CAAC;YACd,KAAK,cAAc,CAAC,CAAC,CAAC;gBACpB,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,CAAyB,CAAC;gBAEvD,2BAA2B;gBAC3B,YAAY,CAAC,IAAI,CAAC,UAAU,CAAC,GAAG;oBAC9B,KAAK,EAAE,IAAI,CAAC,KAAK;oBACjB,UAAU,EAAE,EAAE,KAAK,EAAE,CAAC,EAAE,MAAM,EAAE,CAAC,EAAE;oBACnC,YAAY,EAAE,CAAC;iBAChB,CAAC;gBACF,gBAAgB,CAAC,YAAY,CAAC,CAAC;gBAE/B,MAAM,OAAO,GAAgB;oBAC3B,SAAS,EAAE,IAAI,CAAC,UAAU;oBAC1B,MAAM,EAAE,aAAa;oBACrB,GAAG,EAAE,IAAI,CAAC,GAAG;oBACb,KAAK,EAAE,IAAI,CAAC,KAAK;oBACjB,cAAc,EAAE,IAAI,CAAC,eAAe;oBACpC,eAAe,EAAE,IAAI,CAAC,gBAAgB;oBACtC,UAAU,EAAE,IAAI,CAAC,WAAW;oBAC5B,SAAS,EAAE,IAAI,CAAC,MAAM,KAAK,SAAS,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAE,IAAI,CAAC,MAAmC;oBACxF,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;oBACnC,WAAW,EAAE,IAAI,CAAC,GAAG;oBACrB,WAAW,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,CAAC,CAAC,CAAC,SAAS;iBAC9D,CAAC;gBACF,MAAM,MAAM,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;gBAClC,MAAM;YACR,CAAC;YAED,KAAK,YAAY,CAAC,CAAC,CAAC;gBAClB,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,CAAuB,CAAC;gBACrD,MAAM,KAAK,GAAG,YAAY,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,EAAE,CAAC;gBAElD,8BAA8B;gBAC9B,MAAM,eAAe,GAAG,IAAI,CAAC,iBAAiB,IAAI,KAAK,CAAC,UAAU,CAAC;gBAEnE,MAAM,OAAO,GAAgB;oBAC3B,SAAS,EAAE,IAAI,CAAC,UAAU;oBAC1B,MAAM,EAAE,aAAa;oBACrB,KAAK,EAAE,KAAK,CAAC,WAAW,CAAC,CAAC,CAAC,aAAa,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,SAAS;oBACvE,SAAS,EAAE,IAAI,CAAC,MAAM;oBACtB,YAAY,EAAE,IAAI,CAAC,aAAa,IAAI,KAAK,CAAC,YAAY;oBACtD,aAAa,EAAE,IAAI,CAAC,eAAe;oBACnC,UAAU,EAAE,eAAe;oBAC3B,YAAY,EAAE,IAAI,CAAC,aAAa;oBAChC,OAAO,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;iBAClC,CAAC;gBACF,MAAM,MAAM,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;gBAElC,yBAAyB;gBACzB,OAAO,YAAY,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;gBACrC,gBAAgB,CAAC,YAAY,CAAC,CAAC;gBAC/B,MAAM;YACR,CAAC;YAED,KAAK,kBAAkB,CAAC,CAAC,CAAC;gBACxB,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,CAAuB,CAAC;gBACrD,MAAM,KAAK,GAAG,YAAY,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,EAAE,CAAC;gBAElD,0CAA0C;gBAC1C,IAAI,CAAC,KAAK,CAAC,WAAW,EAAE,CAAC;oBACvB,KAAK,CAAC,WAAW,GAAG,IAAI,CAAC,MAAM,CAAC;oBAChC,YAAY,CAAC,IAAI,CAAC,UAAU,CAAC,GAAG,KAAK,CAAC;oBACtC,gBAAgB,CAAC,YAAY,CAAC,CAAC;oBAE/B,4BAA4B;oBAC5B,MAAM,OAAO,GAAgB;wBAC3B,SAAS,EAAE,IAAI,CAAC,UAAU;wBAC1B,MAAM,EAAE,aAAa;wBACrB,KAAK,EAAE,aAAa,CAAC,IAAI,CAAC,MAAM,CAAC;qBAClC,CAAC;oBACF,MAAM,MAAM,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;gBACpC,CAAC;gBAED,0BAA0B;gBAC1B,KAAK,CAAC,YAAY,GAAG,CAAC,KAAK,CAAC,YAAY,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC;gBACnD,YAAY,CAAC,IAAI,CAAC,UAAU,CAAC,GAAG,KAAK,CAAC;gBACtC,gBAAgB,CAAC,YAAY,CAAC,CAAC;gBAE/B,MAAM,OAAO,GAAgB;oBAC3B,SAAS,EAAE,IAAI,CAAC,UAAU;oBAC1B,SAAS,EAAE,GAAG,IAAI,CAAC,UAAU,SAAS,IAAI,CAAC,GAAG,EAAE,EAAE;oBAClD,MAAM,EAAE,aAAa;oBACrB,IAAI,EAAE,MAAM;oBACZ,OAAO,EAAE,IAAI,CAAC,MAAM;oBACpB,SAAS,EAAE,IAAI,CAAC,SAAS,IAAI,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;iBACtD,CAAC;gBACF,MAAM,MAAM,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;gBAClC,MAAM;YACR,CAAC;YAED,KAAK,aAAa,CAAC,CAAC,CAAC;gBACnB,IAAI,CAAC,MAAM,CAAC,aAAa;oBAAE,MAAM;gBACjC,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,CAAoB,CAAC;gBAClD,MAAM,KAAK,GAAG,YAAY,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,EAAE,CAAC;gBAElD,MAAM,OAAO,GAAgB;oBAC3B,SAAS,EAAE,IAAI,CAAC,UAAU;oBAC1B,SAAS,EAAE,IAAI,CAAC,WAAW,IAAI,GAAG,IAAI,CAAC,UAAU,SAAS,IAAI,CAAC,GAAG,EAAE,EAAE;oBACtE,MAAM,EAAE,aAAa;oBACrB,IAAI,EAAE,WAAW;oBACjB,QAAQ,EAAE,IAAI,CAAC,SAAS;oBACxB,QAAQ,EAAE,IAAI,CAAC,UAAU;oBACzB,UAAU,EAAE,IAAI,CAAC,WAAW,EAAE,MAAM,IAAI,IAAI,CAAC,WAAW,EAAE,KAAK;oBAC/D,UAAU,EAAE,IAAI,CAAC,WAAW;oBAC5B,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;iBACpC,CAAC;gBACF,MAAM,MAAM,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;gBAClC,MAAM;YACR,CAAC;YAED,KAAK,MAAM,CAAC,CAAC,CAAC;gBACZ,+DAA+D;gBAC/D,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,CAAiB,CAAC;gBAC/C,MAAM,KAAK,GAAG,YAAY,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,EAAE,CAAC;gBAElD,iCAAiC;gBACjC,IAAI,IAAI,CAAC,WAAW,EAAE,CAAC;oBACrB,MAAM,OAAO,GAAG,KAAK,CAAC,UAAU,IAAI,EAAE,KAAK,EAAE,CAAC,EAAE,MAAM,EAAE,CAAC,EAAE,CAAC;oBAC5D,KAAK,CAAC,UAAU,GAAG;wBACjB,KAAK,EAAE,OAAO,CAAC,KAAK,GAAG,IAAI,CAAC,WAAW,CAAC,KAAK;wBAC7C,MAAM,EAAE,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,WAAW,CAAC,MAAM;qBACjD,CAAC;gBACJ,CAAC;gBAED,iDAAiD;gBACjD,KAAK,CAAC,YAAY,GAAG,CAAC,KAAK,CAAC,YAAY,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC;gBACnD,YAAY,CAAC,IAAI,CAAC,UAAU,CAAC,GAAG,KAAK,CAAC;gBACtC,gBAAgB,CAAC,YAAY,CAAC,CAAC;gBAE/B,6CAA6C;gBAC7C,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC;oBAClB,MAAM,OAAO,GAAgB;wBAC3B,SAAS,EAAE,IAAI,CAAC,UAAU;wBAC1B,SAAS,EAAE,GAAG,IAAI,CAAC,UAAU,cAAc,IAAI,CAAC,GAAG,EAAE,EAAE;wBACvD,MAAM,EAAE,aAAa;wBACrB,IAAI,EAAE,WAAW;wBACjB,OAAO,EAAE,IAAI,CAAC,QAAQ;wBACtB,UAAU,EAAE,IAAI,CAAC,WAAW;4BAC1B,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,KAAK,GAAG,IAAI,CAAC,WAAW,CAAC,MAAM;4BAClD,CAAC,CAAC,SAAS;wBACb,UAAU,EAAE,IAAI,CAAC,WAAW;wBAC5B,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;qBACpC,CAAC;oBACF,MAAM,MAAM,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;gBACpC,CAAC;gBAED,8CAA8C;gBAC9C,MAAM,OAAO,GAAgB;oBAC3B,SAAS,EAAE,IAAI,CAAC,UAAU;oBAC1B,MAAM,EAAE,aAAa;oBACrB,UAAU,EAAE,KAAK,CAAC,UAAU;oBAC5B,YAAY,EAAE,KAAK,CAAC,YAAY;iBACjC,CAAC;gBACF,MAAM,MAAM,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC;gBAClC,MAAM;YACR,CAAC;YAED;gBACE,wBAAwB;gBACxB,MAAM;QACV,CAAC;QAED,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,4CAA4C;QAC5C,OAAO,CAAC,KAAK,CAAC,6BAA6B,KAAK,EAAE,CAAC,CAAC;QACpD,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;AACH,CAAC,CAAC,CAAC;AAEL,gBAAgB;AAChB,OAAO,CAAC,KAAK,EAAE,CAAC","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"]}
|
package/dist/index.js
CHANGED
|
@@ -139,45 +139,81 @@ class SyncClient {
|
|
|
139
139
|
}
|
|
140
140
|
// Transform session data to backend format
|
|
141
141
|
transformSession(session) {
|
|
142
|
-
|
|
142
|
+
const payload = {
|
|
143
143
|
externalId: session.sessionId,
|
|
144
|
-
title: session.title,
|
|
145
|
-
projectPath: session.projectPath || session.cwd,
|
|
146
|
-
projectName: session.projectName,
|
|
147
|
-
model: session.model,
|
|
148
144
|
source: session.source,
|
|
149
|
-
promptTokens: session.tokenUsage?.input,
|
|
150
|
-
completionTokens: session.tokenUsage?.output,
|
|
151
|
-
cost: session.costEstimate,
|
|
152
|
-
durationMs: session.endedAt && session.startedAt
|
|
153
|
-
? new Date(session.endedAt).getTime() - new Date(session.startedAt).getTime()
|
|
154
|
-
: undefined,
|
|
155
145
|
};
|
|
146
|
+
// Only include fields that are defined
|
|
147
|
+
if (session.title !== undefined)
|
|
148
|
+
payload.title = session.title;
|
|
149
|
+
if (session.projectPath || session.cwd)
|
|
150
|
+
payload.projectPath = session.projectPath || session.cwd;
|
|
151
|
+
if (session.projectName)
|
|
152
|
+
payload.projectName = session.projectName;
|
|
153
|
+
if (session.model)
|
|
154
|
+
payload.model = session.model;
|
|
155
|
+
if (session.tokenUsage?.input !== undefined)
|
|
156
|
+
payload.promptTokens = session.tokenUsage.input;
|
|
157
|
+
if (session.tokenUsage?.output !== undefined)
|
|
158
|
+
payload.completionTokens = session.tokenUsage.output;
|
|
159
|
+
if (session.costEstimate !== undefined)
|
|
160
|
+
payload.cost = session.costEstimate;
|
|
161
|
+
if (session.messageCount !== undefined)
|
|
162
|
+
payload.messageCount = session.messageCount;
|
|
163
|
+
if (session.toolCallCount !== undefined)
|
|
164
|
+
payload.toolCallCount = session.toolCallCount;
|
|
165
|
+
if (session.endReason)
|
|
166
|
+
payload.endReason = session.endReason;
|
|
167
|
+
// Calculate duration if both timestamps exist
|
|
168
|
+
if (session.endedAt && session.startedAt) {
|
|
169
|
+
payload.durationMs = new Date(session.endedAt).getTime() - new Date(session.startedAt).getTime();
|
|
170
|
+
}
|
|
171
|
+
return payload;
|
|
156
172
|
}
|
|
157
173
|
// Transform message data to backend format
|
|
158
174
|
transformMessage(message) {
|
|
159
|
-
|
|
175
|
+
const payload = {
|
|
160
176
|
sessionExternalId: message.sessionId,
|
|
161
177
|
externalId: message.messageId,
|
|
162
178
|
role: message.role,
|
|
163
|
-
textContent: message.content || message.toolResult,
|
|
164
|
-
model: undefined,
|
|
165
|
-
durationMs: message.durationMs,
|
|
166
179
|
source: message.source,
|
|
167
|
-
// Tool use info stored in parts
|
|
168
|
-
parts: message.toolName
|
|
169
|
-
? [
|
|
170
|
-
{
|
|
171
|
-
type: "tool_use",
|
|
172
|
-
content: {
|
|
173
|
-
toolName: message.toolName,
|
|
174
|
-
args: message.toolArgs,
|
|
175
|
-
result: message.toolResult,
|
|
176
|
-
},
|
|
177
|
-
},
|
|
178
|
-
]
|
|
179
|
-
: undefined,
|
|
180
180
|
};
|
|
181
|
+
// Set textContent for user prompts and assistant responses
|
|
182
|
+
if (message.content) {
|
|
183
|
+
payload.textContent = message.content;
|
|
184
|
+
}
|
|
185
|
+
else if (message.toolResult && !message.toolName) {
|
|
186
|
+
// Only use toolResult as textContent if no toolName (fallback)
|
|
187
|
+
payload.textContent = message.toolResult;
|
|
188
|
+
}
|
|
189
|
+
// Include duration if available
|
|
190
|
+
if (message.durationMs !== undefined) {
|
|
191
|
+
payload.durationMs = message.durationMs;
|
|
192
|
+
}
|
|
193
|
+
// Include token count if available
|
|
194
|
+
if (message.tokenCount !== undefined) {
|
|
195
|
+
payload.promptTokens = message.tokenCount;
|
|
196
|
+
}
|
|
197
|
+
// Tool use info stored in parts
|
|
198
|
+
if (message.toolName) {
|
|
199
|
+
payload.parts = [
|
|
200
|
+
{
|
|
201
|
+
type: "tool-call",
|
|
202
|
+
content: {
|
|
203
|
+
toolName: message.toolName,
|
|
204
|
+
args: message.toolArgs,
|
|
205
|
+
},
|
|
206
|
+
},
|
|
207
|
+
];
|
|
208
|
+
// Add tool result as separate part
|
|
209
|
+
if (message.toolResult) {
|
|
210
|
+
payload.parts.push({
|
|
211
|
+
type: "tool-result",
|
|
212
|
+
content: message.toolResult,
|
|
213
|
+
});
|
|
214
|
+
}
|
|
215
|
+
}
|
|
216
|
+
return payload;
|
|
181
217
|
}
|
|
182
218
|
async syncSession(session) {
|
|
183
219
|
try {
|
|
@@ -387,4 +423,4 @@ function createPlugin() {
|
|
|
387
423
|
}
|
|
388
424
|
// Default export for Claude Code plugin system
|
|
389
425
|
exports.default = createPlugin;
|
|
390
|
-
//# 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    return {\n      externalId: session.sessionId,\n      title: session.title,\n      projectPath: session.projectPath || session.cwd,\n      projectName: session.projectName,\n      model: session.model,\n      source: session.source,\n      promptTokens: session.tokenUsage?.input,\n      completionTokens: session.tokenUsage?.output,\n      cost: session.costEstimate,\n      durationMs: session.endedAt && session.startedAt\n        ? new Date(session.endedAt).getTime() - new Date(session.startedAt).getTime()\n        : undefined,\n    };\n  }\n\n  // Transform message data to backend format\n  private transformMessage(message: MessageData): Record<string, unknown> {\n    return {\n      sessionExternalId: message.sessionId,\n      externalId: message.messageId,\n      role: message.role,\n      textContent: message.content || message.toolResult,\n      model: undefined,\n      durationMs: message.durationMs,\n      source: message.source,\n      // Tool use info stored in parts\n      parts: message.toolName\n        ? [\n            {\n              type: \"tool_use\",\n              content: {\n                toolName: message.toolName,\n                args: message.toolArgs,\n                result: message.toolResult,\n              },\n            },\n          ]\n        : undefined,\n    };\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"]}
|
|
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"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "claude-code-sync",
|
|
3
|
-
"version": "0.1.
|
|
3
|
+
"version": "0.1.10",
|
|
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",
|