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.
- package/README.md +93 -5
- package/dist/cli.js +327 -1
- package/package.json +1 -1
package/README.md
CHANGED
|
@@ -3,12 +3,20 @@
|
|
|
3
3
|
[](https://www.npmjs.com/package/claude-code-sync)
|
|
4
4
|
[](https://opensource.org/licenses/MIT)
|
|
5
5
|
|
|
6
|
-
Sync your Claude Code sessions to [OpenSync](https://
|
|
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
|
-
|
|
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
|
-
|
|
129
|
-
|
|
130
|
-
- [OpenSync
|
|
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.
|
|
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",
|