claude-code-sync 0.1.2 → 0.1.5

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.
Files changed (3) hide show
  1. package/README.md +93 -5
  2. package/dist/cli.js +327 -1
  3. package/package.json +1 -1
package/README.md CHANGED
@@ -3,12 +3,20 @@
3
3
  [![npm version](https://img.shields.io/npm/v/claude-code-sync.svg)](https://www.npmjs.com/package/claude-code-sync)
4
4
  [![License: MIT](https://img.shields.io/badge/License-MIT-blue.svg)](https://opensource.org/licenses/MIT)
5
5
 
6
- Sync your Claude Code sessions to [OpenSync](https://github.com/waynesutton/opensync) dashboard. Track coding sessions, analyze tool usage, and monitor token consumption across projects.
6
+ Sync your Claude Code sessions to [OpenSync](https://opensync.dev/) dashboard. Track coding sessions, analyze tool usage, and monitor token consumption across projects.
7
7
 
8
8
  **GitHub:** [github.com/waynesutton/claude-code-sync](https://github.com/waynesutton/claude-code-sync)
9
9
 
10
10
  **npm:** [npmjs.com/package/claude-code-sync](https://www.npmjs.com/package/claude-code-sync)
11
11
 
12
+ ## OpenSync Ecosystem
13
+
14
+ | Package | Description | Install |
15
+ |---------|-------------|---------|
16
+ | [OpenSync](https://opensync.dev/) | Beautiful dashboards for OpenCode and Claude Code sessions synced to the cloud | [GitHub](https://github.com/waynesutton/opensync) |
17
+ | [opencode-sync-plugin](https://www.npmjs.com/package/opencode-sync-plugin) | Sync your OpenCode sessions to OpenSync dashboard | `npm install -g opencode-sync-plugin` |
18
+ | [claude-code-sync](https://www.npmjs.com/package/claude-code-sync) | Sync your Claude Code sessions to OpenSync dashboard | `npm install -g claude-code-sync` |
19
+
12
20
  ## Installation
13
21
 
14
22
  ```bash
@@ -36,23 +44,85 @@ Enter when prompted:
36
44
 
37
45
  ### 3. Add to Claude Code
38
46
 
39
- Add the plugin to your Claude Code configuration. Sessions will sync automatically.
47
+ **Option A: Use the setup command (recommended)**
48
+
49
+ ```bash
50
+ claude-code-sync setup
51
+ ```
52
+
53
+ This automatically configures the hooks in `~/.claude/settings.json`.
54
+
55
+ **Option B: One-liner (copy and paste)**
56
+
57
+ ```bash
58
+ mkdir -p ~/.claude && cat > ~/.claude/settings.json << 'EOF'
59
+ {
60
+ "hooks": {
61
+ "SessionStart": [{ "hooks": [{ "type": "command", "command": "claude-code-sync hook SessionStart" }] }],
62
+ "SessionEnd": [{ "hooks": [{ "type": "command", "command": "claude-code-sync hook SessionEnd" }] }],
63
+ "UserPromptSubmit": [{ "hooks": [{ "type": "command", "command": "claude-code-sync hook UserPromptSubmit" }] }],
64
+ "PostToolUse": [{ "matcher": "*", "hooks": [{ "type": "command", "command": "claude-code-sync hook PostToolUse" }] }],
65
+ "Stop": [{ "matcher": "*", "hooks": [{ "type": "command", "command": "claude-code-sync hook Stop" }] }]
66
+ }
67
+ }
68
+ EOF
69
+ ```
70
+
71
+ ### 4. Verify Setup
72
+
73
+ ```bash
74
+ claude-code-sync verify
75
+ ```
76
+
77
+ You should see:
78
+
79
+ ```
80
+ OpenSync Setup Verification
81
+
82
+ Credentials: OK
83
+ Convex URL: https://your-project.convex.cloud
84
+ API Key: osk_****...****
85
+
86
+ Claude Code Config: OK
87
+ Config file: ~/.claude/settings.json
88
+ Hooks registered: claude-code-sync
89
+
90
+ Ready! Start Claude Code and sessions will sync automatically.
91
+ ```
92
+
93
+ Sessions will now sync automatically when you use Claude Code.
40
94
 
41
95
  ## CLI Commands
42
96
 
43
97
  | Command | Description |
44
98
  |---------|-------------|
45
99
  | `claude-code-sync login` | Configure Convex URL and API Key |
100
+ | `claude-code-sync setup` | Add hooks to Claude Code settings |
101
+ | `claude-code-sync verify` | Verify credentials and Claude Code config |
102
+ | `claude-code-sync synctest` | Test connectivity and create a test session |
46
103
  | `claude-code-sync logout` | Clear stored credentials |
47
104
  | `claude-code-sync status` | Show connection status |
48
105
  | `claude-code-sync config` | Show current configuration |
49
106
  | `claude-code-sync config --json` | Show config as JSON |
50
107
  | `claude-code-sync set <key> <value>` | Update a config value |
108
+ | `claude-code-sync hook <event>` | Handle Claude Code hook events (internal) |
51
109
  | `claude-code-sync --version` | Show version number |
52
110
  | `claude-code-sync --help` | Show help |
53
111
 
54
112
  See [full command reference](docs/commands.md) for detailed usage, troubleshooting, and examples.
55
113
 
114
+ ## Hook Events
115
+
116
+ The plugin captures these Claude Code events:
117
+
118
+ | Event | Description |
119
+ |-------|-------------|
120
+ | `SessionStart` | Fires when a coding session begins |
121
+ | `SessionEnd` | Fires when a session terminates |
122
+ | `UserPromptSubmit` | Fires when you submit a prompt |
123
+ | `PostToolUse` | Fires after each tool execution |
124
+ | `Stop` | Fires when Claude finishes responding |
125
+
56
126
  ### Configuration Options
57
127
 
58
128
  | Option | Type | Default | Description |
@@ -117,6 +187,13 @@ claude-code-sync login
117
187
 
118
188
  See [troubleshooting guide](docs/commands.md#troubleshooting) for more solutions.
119
189
 
190
+ ### Need Help?
191
+
192
+ If you run into issues or have questions:
193
+
194
+ - **Report a bug or request a feature:** [GitHub Issues](https://github.com/waynesutton/claude-code-sync/issues)
195
+ - Check existing issues for solutions to common problems
196
+
120
197
  ## Requirements
121
198
 
122
199
  - Node.js 18 or later
@@ -125,10 +202,21 @@ See [troubleshooting guide](docs/commands.md#troubleshooting) for more solutions
125
202
 
126
203
  ## Links
127
204
 
128
- - [claude-code-sync Repository](https://github.com/waynesutton/claude-code-sync)
129
- - [OpenSync Backend](https://github.com/waynesutton/opensync)
130
- - [OpenSync Dashboard](https://opensyncsessions.netlify.app)
205
+ ### OpenSync
206
+
207
+ - [OpenSync](https://opensync.dev/) - Beautiful dashboards for OpenCode and Claude Code sessions
208
+ - [OpenSync Repository](https://github.com/waynesutton/opensync)
209
+
210
+ ### claude-code-sync (this package)
211
+
212
+ - [GitHub Repository](https://github.com/waynesutton/claude-code-sync)
131
213
  - [npm Package](https://www.npmjs.com/package/claude-code-sync)
214
+ - [Issues and Support](https://github.com/waynesutton/claude-code-sync/issues)
215
+
216
+ ### opencode-sync-plugin
217
+
218
+ - [GitHub Repository](https://github.com/waynesutton/opencode-sync-plugin)
219
+ - [npm Package](https://www.npmjs.com/package/opencode-sync-plugin)
132
220
 
133
221
  ## License
134
222
 
package/dist/cli.js CHANGED
@@ -235,6 +235,332 @@ program
235
235
  (0, index_1.saveConfig)(config);
236
236
  console.log(`Set ${key} = ${boolValue}`);
237
237
  });
238
+ // ============================================================================
239
+ // Setup Command (configures Claude Code hooks)
240
+ // ============================================================================
241
+ // Claude Code hooks configuration
242
+ const CLAUDE_HOOKS_CONFIG = {
243
+ hooks: {
244
+ SessionStart: [
245
+ {
246
+ hooks: [
247
+ {
248
+ type: "command",
249
+ command: "claude-code-sync hook SessionStart",
250
+ },
251
+ ],
252
+ },
253
+ ],
254
+ SessionEnd: [
255
+ {
256
+ hooks: [
257
+ {
258
+ type: "command",
259
+ command: "claude-code-sync hook SessionEnd",
260
+ },
261
+ ],
262
+ },
263
+ ],
264
+ UserPromptSubmit: [
265
+ {
266
+ hooks: [
267
+ {
268
+ type: "command",
269
+ command: "claude-code-sync hook UserPromptSubmit",
270
+ },
271
+ ],
272
+ },
273
+ ],
274
+ PostToolUse: [
275
+ {
276
+ matcher: "*",
277
+ hooks: [
278
+ {
279
+ type: "command",
280
+ command: "claude-code-sync hook PostToolUse",
281
+ },
282
+ ],
283
+ },
284
+ ],
285
+ Stop: [
286
+ {
287
+ matcher: "*",
288
+ hooks: [
289
+ {
290
+ type: "command",
291
+ command: "claude-code-sync hook Stop",
292
+ },
293
+ ],
294
+ },
295
+ ],
296
+ },
297
+ };
298
+ program
299
+ .command("setup")
300
+ .description("Add hooks to Claude Code settings (configures ~/.claude/settings.json)")
301
+ .option("--force", "Overwrite existing hooks configuration")
302
+ .action(async (options) => {
303
+ const claudeDir = path.join(process.env.HOME || "~", ".claude");
304
+ const settingsPath = path.join(claudeDir, "settings.json");
305
+ console.log("\n Claude Code Sync - Setup\n");
306
+ // Check if plugin credentials are configured
307
+ const config = (0, index_1.loadConfig)();
308
+ if (!config) {
309
+ console.log("Warning: Plugin not configured yet.");
310
+ console.log(" Run 'claude-code-sync login' first to set up credentials.\n");
311
+ }
312
+ // Create .claude directory if needed
313
+ if (!fs.existsSync(claudeDir)) {
314
+ fs.mkdirSync(claudeDir, { recursive: true });
315
+ console.log("Created ~/.claude directory");
316
+ }
317
+ // Check for existing settings
318
+ let existingSettings = {};
319
+ let hasExistingHooks = false;
320
+ if (fs.existsSync(settingsPath)) {
321
+ try {
322
+ const content = fs.readFileSync(settingsPath, "utf-8");
323
+ existingSettings = JSON.parse(content);
324
+ hasExistingHooks = !!existingSettings.hooks;
325
+ console.log("Found existing settings.json");
326
+ }
327
+ catch {
328
+ console.log("Warning: Could not parse existing settings.json, will create new one");
329
+ }
330
+ }
331
+ // Handle existing hooks
332
+ if (hasExistingHooks && !options.force) {
333
+ console.log("\nExisting hooks configuration found.");
334
+ console.log(" Use --force to overwrite, or manually merge the hooks.\n");
335
+ console.log("To manually add, include these hooks in your settings.json:");
336
+ console.log(JSON.stringify(CLAUDE_HOOKS_CONFIG, null, 2));
337
+ process.exit(1);
338
+ }
339
+ // Merge settings
340
+ const newSettings = {
341
+ ...existingSettings,
342
+ ...CLAUDE_HOOKS_CONFIG,
343
+ };
344
+ // Write settings
345
+ try {
346
+ fs.writeFileSync(settingsPath, JSON.stringify(newSettings, null, 2));
347
+ console.log("\nClaude Code hooks configured!");
348
+ console.log(` Settings file: ${settingsPath}`);
349
+ console.log("\nSetup complete. Sessions will sync automatically.\n");
350
+ }
351
+ catch (error) {
352
+ console.error("Error writing settings:", error);
353
+ process.exit(1);
354
+ }
355
+ });
356
+ program
357
+ .command("verify")
358
+ .description("Verify credentials and Claude Code configuration")
359
+ .action(async () => {
360
+ console.log("\n OpenSync Setup Verification\n");
361
+ // Check credentials
362
+ const config = (0, index_1.loadConfig)();
363
+ if (config) {
364
+ console.log("Credentials: OK");
365
+ console.log(` Convex URL: ${config.convexUrl}`);
366
+ console.log(` API Key: ${maskApiKey(config.apiKey)}`);
367
+ }
368
+ else {
369
+ console.log("Credentials: NOT CONFIGURED");
370
+ console.log(" Run 'claude-code-sync login' to set up");
371
+ }
372
+ // Check Claude Code config
373
+ const settingsPath = path.join(process.env.HOME || "~", ".claude", "settings.json");
374
+ let hooksConfigured = false;
375
+ if (fs.existsSync(settingsPath)) {
376
+ try {
377
+ const content = fs.readFileSync(settingsPath, "utf-8");
378
+ const settings = JSON.parse(content);
379
+ hooksConfigured = !!settings.hooks?.SessionStart;
380
+ }
381
+ catch {
382
+ // Ignore parse errors
383
+ }
384
+ }
385
+ console.log("");
386
+ if (hooksConfigured) {
387
+ console.log("Claude Code Config: OK");
388
+ console.log(` Config file: ${settingsPath}`);
389
+ console.log(" Hooks registered: claude-code-sync");
390
+ }
391
+ else {
392
+ console.log("Claude Code Config: NOT CONFIGURED");
393
+ console.log(" Run 'claude-code-sync setup' to configure hooks");
394
+ }
395
+ // Test connection if credentials exist
396
+ if (config) {
397
+ console.log("\nTesting connection...");
398
+ const client = new index_1.SyncClient(config);
399
+ const connected = await client.testConnection();
400
+ if (connected) {
401
+ console.log("Connection: OK\n");
402
+ }
403
+ else {
404
+ console.log("Connection: FAILED\n");
405
+ process.exit(1);
406
+ }
407
+ }
408
+ if (config && hooksConfigured) {
409
+ console.log("Ready! Start Claude Code and sessions will sync automatically.\n");
410
+ }
411
+ else {
412
+ console.log("");
413
+ process.exit(1);
414
+ }
415
+ });
416
+ // ============================================================================
417
+ // Sync Test Command (test connectivity)
418
+ // ============================================================================
419
+ program
420
+ .command("synctest")
421
+ .description("Test connectivity and create a test session")
422
+ .action(async () => {
423
+ const config = (0, index_1.loadConfig)();
424
+ console.log("\n Claude Code Sync - Connection Test\n");
425
+ if (!config) {
426
+ console.log("Not configured");
427
+ console.log(" Run 'claude-code-sync login' to set up\n");
428
+ process.exit(1);
429
+ }
430
+ console.log("Configuration:");
431
+ console.log(` Convex URL: ${config.convexUrl}`);
432
+ console.log(` API Key: ${maskApiKey(config.apiKey)}`);
433
+ console.log("\nTesting connection...");
434
+ const client = new index_1.SyncClient(config);
435
+ const connected = await client.testConnection();
436
+ if (connected) {
437
+ console.log("Connection: OK");
438
+ // Create a test session to verify full sync works
439
+ console.log("\nCreating test session...");
440
+ try {
441
+ const testSession = {
442
+ sessionId: `test-${Date.now()}`,
443
+ source: "claude-code",
444
+ title: "Connection Test",
445
+ projectName: "synctest",
446
+ startedAt: new Date().toISOString(),
447
+ endedAt: new Date().toISOString(),
448
+ };
449
+ await client.syncSession(testSession);
450
+ console.log("Test session created successfully");
451
+ console.log("\nSync test passed. Ready to sync Claude Code sessions.\n");
452
+ }
453
+ catch (error) {
454
+ console.log(`Test session failed: ${error}`);
455
+ console.log("\nConnection works but sync may have issues.\n");
456
+ process.exit(1);
457
+ }
458
+ }
459
+ else {
460
+ console.log("Connection: FAILED");
461
+ console.log("\nCheck your Convex URL and API key.\n");
462
+ process.exit(1);
463
+ }
464
+ });
465
+ // ============================================================================
466
+ // Hook Command (for Claude Code integration)
467
+ // ============================================================================
468
+ program
469
+ .command("hook <event>")
470
+ .description("Handle Claude Code hook events (reads stdin)")
471
+ .action(async (event) => {
472
+ const config = (0, index_1.loadConfig)();
473
+ if (!config) {
474
+ // Exit silently if not configured (don't block Claude Code)
475
+ process.exit(0);
476
+ }
477
+ if (config.autoSync === false) {
478
+ process.exit(0);
479
+ }
480
+ // Read JSON input from stdin
481
+ let input = "";
482
+ for await (const chunk of process.stdin) {
483
+ input += chunk;
484
+ }
485
+ if (!input.trim()) {
486
+ process.exit(0);
487
+ }
488
+ try {
489
+ const client = new index_1.SyncClient(config);
490
+ switch (event) {
491
+ case "SessionStart": {
492
+ const data = JSON.parse(input);
493
+ const session = {
494
+ sessionId: data.session_id,
495
+ source: "claude-code",
496
+ cwd: data.cwd,
497
+ permissionMode: data.permission_mode,
498
+ startType: data.source === "startup" ? "new" : data.source,
499
+ startedAt: new Date().toISOString(),
500
+ projectPath: data.cwd,
501
+ projectName: data.cwd ? data.cwd.split("/").pop() : undefined,
502
+ };
503
+ await client.syncSession(session);
504
+ break;
505
+ }
506
+ case "SessionEnd": {
507
+ const data = JSON.parse(input);
508
+ const session = {
509
+ sessionId: data.session_id,
510
+ source: "claude-code",
511
+ endReason: data.reason,
512
+ endedAt: new Date().toISOString(),
513
+ };
514
+ await client.syncSession(session);
515
+ break;
516
+ }
517
+ case "UserPromptSubmit": {
518
+ const data = JSON.parse(input);
519
+ const message = {
520
+ sessionId: data.session_id,
521
+ messageId: `${data.session_id}-${Date.now()}`,
522
+ source: "claude-code",
523
+ role: "user",
524
+ content: data.prompt,
525
+ timestamp: new Date().toISOString(),
526
+ };
527
+ await client.syncMessage(message);
528
+ break;
529
+ }
530
+ case "PostToolUse": {
531
+ if (!config.syncToolCalls)
532
+ break;
533
+ const data = JSON.parse(input);
534
+ const message = {
535
+ sessionId: data.session_id,
536
+ messageId: `${data.session_id}-tool-${Date.now()}`,
537
+ source: "claude-code",
538
+ role: "assistant",
539
+ toolName: data.tool_name,
540
+ toolArgs: data.tool_input,
541
+ toolResult: data.tool_result?.output || data.tool_result?.error,
542
+ timestamp: new Date().toISOString(),
543
+ };
544
+ await client.syncMessage(message);
545
+ break;
546
+ }
547
+ case "Stop": {
548
+ // Stop event indicates Claude finished responding
549
+ // We could track this but for now just acknowledge
550
+ break;
551
+ }
552
+ default:
553
+ // Unknown event, ignore
554
+ break;
555
+ }
556
+ process.exit(0);
557
+ }
558
+ catch (error) {
559
+ // Log to stderr but don't block Claude Code
560
+ console.error(`[claude-code-sync] Error: ${error}`);
561
+ process.exit(0);
562
+ }
563
+ });
238
564
  // Parse and run
239
565
  program.parse();
240
- //# 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} from \"./index\";\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\"));\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 dashboard:\");\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(\"\\nAdd the plugin to your Claude Code config to start syncing.\\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// Parse and run\nprogram.parse();\n"]}
566
+ //# 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 dashboard:\");\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(\"\\nAdd the plugin to your Claude Code config to start syncing.\\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"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "claude-code-sync",
3
- "version": "0.1.2",
3
+ "version": "0.1.5",
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",