claude-code-sync 0.1.13 → 0.1.14
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cli.js +208 -28
- package/dist/index.d.ts +55 -24
- package/dist/index.js +169 -40
- package/package.json +1 -1
package/dist/index.d.ts
CHANGED
|
@@ -25,7 +25,7 @@ export interface SessionData {
|
|
|
25
25
|
gitRepo?: string;
|
|
26
26
|
model?: string;
|
|
27
27
|
startType?: "new" | "resume" | "continue";
|
|
28
|
-
endReason?: "user_stop" | "max_turns" | "error" | "completed";
|
|
28
|
+
endReason?: "user_stop" | "max_turns" | "error" | "completed" | "clear" | "logout" | "prompt_input_exit" | "other" | string;
|
|
29
29
|
thinkingEnabled?: boolean;
|
|
30
30
|
permissionMode?: string;
|
|
31
31
|
mcpServers?: string[];
|
|
@@ -39,6 +39,10 @@ export interface SessionData {
|
|
|
39
39
|
startedAt?: string;
|
|
40
40
|
endedAt?: string;
|
|
41
41
|
}
|
|
42
|
+
export interface MessagePart {
|
|
43
|
+
type: "text" | "tool-call" | "tool-result";
|
|
44
|
+
content: unknown;
|
|
45
|
+
}
|
|
42
46
|
export interface MessageData {
|
|
43
47
|
sessionId: string;
|
|
44
48
|
messageId: string;
|
|
@@ -49,6 +53,7 @@ export interface MessageData {
|
|
|
49
53
|
toolName?: string;
|
|
50
54
|
toolArgs?: Record<string, unknown>;
|
|
51
55
|
toolResult?: string;
|
|
56
|
+
parts?: MessagePart[];
|
|
52
57
|
durationMs?: number;
|
|
53
58
|
tokenCount?: number;
|
|
54
59
|
timestamp?: string;
|
|
@@ -71,46 +76,72 @@ export interface ClaudeCodeHooks {
|
|
|
71
76
|
SessionEnd?: (data: SessionEndEvent) => void | Promise<void>;
|
|
72
77
|
}
|
|
73
78
|
export interface SessionStartEvent {
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
79
|
+
session_id: string;
|
|
80
|
+
transcript_path: string;
|
|
81
|
+
permission_mode: string;
|
|
82
|
+
hook_event_name: "SessionStart";
|
|
83
|
+
source: "startup" | "resume" | "clear" | "compact";
|
|
84
|
+
sessionId?: string;
|
|
85
|
+
cwd?: string;
|
|
86
|
+
model?: string;
|
|
87
|
+
startType?: "new" | "resume" | "continue";
|
|
78
88
|
thinkingEnabled?: boolean;
|
|
79
89
|
permissionMode?: string;
|
|
80
90
|
mcpServers?: string[];
|
|
81
91
|
}
|
|
82
92
|
export interface UserPromptEvent {
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
93
|
+
session_id: string;
|
|
94
|
+
transcript_path: string;
|
|
95
|
+
permission_mode: string;
|
|
96
|
+
hook_event_name: "UserPromptSubmit";
|
|
97
|
+
prompt?: string;
|
|
98
|
+
sessionId?: string;
|
|
99
|
+
timestamp?: string;
|
|
86
100
|
}
|
|
87
101
|
export interface ToolUseEvent {
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
102
|
+
session_id: string;
|
|
103
|
+
transcript_path: string;
|
|
104
|
+
tool_name: string;
|
|
105
|
+
permission_mode: string;
|
|
106
|
+
hook_event_name: "PostToolUse";
|
|
107
|
+
sessionId?: string;
|
|
108
|
+
toolName?: string;
|
|
109
|
+
args?: Record<string, unknown>;
|
|
110
|
+
result?: string;
|
|
111
|
+
success?: boolean;
|
|
112
|
+
durationMs?: number;
|
|
94
113
|
}
|
|
95
114
|
export interface StopEvent {
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
115
|
+
session_id: string;
|
|
116
|
+
transcript_path: string;
|
|
117
|
+
permission_mode: string;
|
|
118
|
+
hook_event_name: "Stop";
|
|
119
|
+
stop_hook_active: boolean;
|
|
120
|
+
sessionId?: string;
|
|
121
|
+
response?: string;
|
|
122
|
+
tokenUsage?: {
|
|
99
123
|
input: number;
|
|
100
124
|
output: number;
|
|
101
125
|
};
|
|
102
|
-
durationMs
|
|
126
|
+
durationMs?: number;
|
|
127
|
+
model?: string;
|
|
103
128
|
}
|
|
104
129
|
export interface SessionEndEvent {
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
130
|
+
session_id: string;
|
|
131
|
+
transcript_path: string;
|
|
132
|
+
cwd: string;
|
|
133
|
+
permission_mode: string;
|
|
134
|
+
hook_event_name: "SessionEnd";
|
|
135
|
+
reason: "clear" | "logout" | "prompt_input_exit" | "other";
|
|
136
|
+
sessionId?: string;
|
|
137
|
+
endReason?: string;
|
|
138
|
+
messageCount?: number;
|
|
139
|
+
toolCallCount?: number;
|
|
140
|
+
totalTokenUsage?: {
|
|
110
141
|
input: number;
|
|
111
142
|
output: number;
|
|
112
143
|
};
|
|
113
|
-
costEstimate
|
|
144
|
+
costEstimate?: number;
|
|
114
145
|
}
|
|
115
146
|
export declare function loadConfig(): Config | null;
|
|
116
147
|
export declare function saveConfig(config: Config): void;
|
package/dist/index.js
CHANGED
|
@@ -50,6 +50,114 @@ exports.createPlugin = createPlugin;
|
|
|
50
50
|
const fs = __importStar(require("fs"));
|
|
51
51
|
const path = __importStar(require("path"));
|
|
52
52
|
const os = __importStar(require("os"));
|
|
53
|
+
/**
|
|
54
|
+
* Parse transcript file to extract model, token usage, and stats
|
|
55
|
+
*/
|
|
56
|
+
function parseTranscript(transcriptPath) {
|
|
57
|
+
const stats = {
|
|
58
|
+
model: undefined,
|
|
59
|
+
inputTokens: 0,
|
|
60
|
+
cacheCreationTokens: 0,
|
|
61
|
+
cacheReadTokens: 0,
|
|
62
|
+
outputTokens: 0,
|
|
63
|
+
messageCount: 0,
|
|
64
|
+
toolCallCount: 0,
|
|
65
|
+
title: undefined,
|
|
66
|
+
cwd: undefined,
|
|
67
|
+
};
|
|
68
|
+
try {
|
|
69
|
+
if (!fs.existsSync(transcriptPath)) {
|
|
70
|
+
return stats;
|
|
71
|
+
}
|
|
72
|
+
const content = fs.readFileSync(transcriptPath, "utf-8");
|
|
73
|
+
const lines = content.trim().split("\n");
|
|
74
|
+
for (const line of lines) {
|
|
75
|
+
try {
|
|
76
|
+
const entry = JSON.parse(line);
|
|
77
|
+
// Get cwd and title from first entry that has them
|
|
78
|
+
if (!stats.cwd && entry.cwd) {
|
|
79
|
+
stats.cwd = entry.cwd;
|
|
80
|
+
}
|
|
81
|
+
if (!stats.title && entry.slug) {
|
|
82
|
+
stats.title = entry.slug;
|
|
83
|
+
}
|
|
84
|
+
// Count messages
|
|
85
|
+
if (entry.type === "user") {
|
|
86
|
+
stats.messageCount++;
|
|
87
|
+
}
|
|
88
|
+
// Extract model and tokens from assistant messages
|
|
89
|
+
if (entry.type === "assistant" && entry.message) {
|
|
90
|
+
if (entry.message.model && !stats.model) {
|
|
91
|
+
stats.model = entry.message.model;
|
|
92
|
+
}
|
|
93
|
+
if (entry.message.usage) {
|
|
94
|
+
const usage = entry.message.usage;
|
|
95
|
+
stats.inputTokens += usage.input_tokens || 0;
|
|
96
|
+
stats.cacheCreationTokens += usage.cache_creation_input_tokens || 0;
|
|
97
|
+
stats.cacheReadTokens += usage.cache_read_input_tokens || 0;
|
|
98
|
+
stats.outputTokens += usage.output_tokens || 0;
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
// Count tool uses
|
|
102
|
+
if (entry.type === "assistant" && entry.message?.content) {
|
|
103
|
+
const msgContent = entry.message.content;
|
|
104
|
+
if (Array.isArray(msgContent)) {
|
|
105
|
+
for (const part of msgContent) {
|
|
106
|
+
if (part && typeof part === "object" && "type" in part && part.type === "tool_use") {
|
|
107
|
+
stats.toolCallCount++;
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
catch {
|
|
114
|
+
// Skip malformed lines
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
catch (error) {
|
|
119
|
+
console.error("[claude-code-sync] Error parsing transcript:", error);
|
|
120
|
+
}
|
|
121
|
+
return stats;
|
|
122
|
+
}
|
|
123
|
+
// ============================================================================
|
|
124
|
+
// Model Pricing & Cost Calculation
|
|
125
|
+
// ============================================================================
|
|
126
|
+
// Pricing per million tokens (USD) with cache pricing
|
|
127
|
+
// Source: https://www.anthropic.com/pricing
|
|
128
|
+
const MODEL_PRICING = {
|
|
129
|
+
"claude-sonnet-4-20250514": { input: 3.00, output: 15.00, cacheWrite: 3.75, cacheRead: 0.30 },
|
|
130
|
+
"claude-opus-4-20250514": { input: 15.00, output: 75.00, cacheWrite: 18.75, cacheRead: 1.50 },
|
|
131
|
+
"claude-opus-4-5-20251101": { input: 15.00, output: 75.00, cacheWrite: 18.75, cacheRead: 1.50 },
|
|
132
|
+
"claude-3-5-sonnet-20241022": { input: 3.00, output: 15.00, cacheWrite: 3.75, cacheRead: 0.30 },
|
|
133
|
+
"claude-3-opus-20240229": { input: 15.00, output: 75.00, cacheWrite: 18.75, cacheRead: 1.50 },
|
|
134
|
+
"claude-3-5-haiku-20241022": { input: 0.80, output: 4.00, cacheWrite: 1.00, cacheRead: 0.08 },
|
|
135
|
+
};
|
|
136
|
+
/**
|
|
137
|
+
* Calculate cost from model and token usage with proper cache pricing
|
|
138
|
+
* Returns 0 if model is unknown or pricing not available
|
|
139
|
+
*/
|
|
140
|
+
function calculateCost(model, stats) {
|
|
141
|
+
if (!model)
|
|
142
|
+
return 0;
|
|
143
|
+
// Try exact match first
|
|
144
|
+
let pricing = MODEL_PRICING[model];
|
|
145
|
+
// Try partial match if exact match fails
|
|
146
|
+
if (!pricing) {
|
|
147
|
+
const matchingKey = Object.keys(MODEL_PRICING).find(k => model.includes(k) || k.includes(model));
|
|
148
|
+
if (matchingKey) {
|
|
149
|
+
pricing = MODEL_PRICING[matchingKey];
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
if (!pricing)
|
|
153
|
+
return 0;
|
|
154
|
+
// Calculate cost with proper cache pricing
|
|
155
|
+
const inputCost = stats.inputTokens * pricing.input;
|
|
156
|
+
const cacheWriteCost = stats.cacheCreationTokens * pricing.cacheWrite;
|
|
157
|
+
const cacheReadCost = stats.cacheReadTokens * pricing.cacheRead;
|
|
158
|
+
const outputCost = stats.outputTokens * pricing.output;
|
|
159
|
+
return (inputCost + cacheWriteCost + cacheReadCost + outputCost) / 1_000_000;
|
|
160
|
+
}
|
|
53
161
|
// ============================================================================
|
|
54
162
|
// Configuration
|
|
55
163
|
// ============================================================================
|
|
@@ -306,24 +414,28 @@ function createPlugin() {
|
|
|
306
414
|
SessionStart: async (event) => {
|
|
307
415
|
messageCounter = 0;
|
|
308
416
|
toolCallCounter = 0;
|
|
417
|
+
// Use session_id (new) or sessionId (legacy)
|
|
418
|
+
const sessionId = event.session_id || event.sessionId || "";
|
|
419
|
+
const cwd = event.cwd;
|
|
420
|
+
const permissionMode = event.permission_mode || event.permissionMode;
|
|
309
421
|
const session = {
|
|
310
|
-
sessionId
|
|
422
|
+
sessionId,
|
|
311
423
|
source: "claude-code",
|
|
312
|
-
cwd
|
|
424
|
+
cwd,
|
|
313
425
|
model: event.model,
|
|
314
426
|
startType: event.startType,
|
|
315
427
|
thinkingEnabled: event.thinkingEnabled,
|
|
316
|
-
permissionMode
|
|
428
|
+
permissionMode,
|
|
317
429
|
mcpServers: event.mcpServers,
|
|
318
430
|
startedAt: new Date().toISOString(),
|
|
319
431
|
};
|
|
320
432
|
// Extract project info from cwd
|
|
321
|
-
if (
|
|
322
|
-
session.projectPath =
|
|
323
|
-
session.projectName = path.basename(
|
|
433
|
+
if (cwd) {
|
|
434
|
+
session.projectPath = cwd;
|
|
435
|
+
session.projectName = path.basename(cwd);
|
|
324
436
|
// Try to get git info
|
|
325
437
|
try {
|
|
326
|
-
const gitDir = path.join(
|
|
438
|
+
const gitDir = path.join(cwd, ".git");
|
|
327
439
|
if (fs.existsSync(gitDir)) {
|
|
328
440
|
const headFile = path.join(gitDir, "HEAD");
|
|
329
441
|
if (fs.existsSync(headFile)) {
|
|
@@ -338,7 +450,7 @@ function createPlugin() {
|
|
|
338
450
|
// Ignore git errors
|
|
339
451
|
}
|
|
340
452
|
}
|
|
341
|
-
client.updateSessionState(
|
|
453
|
+
client.updateSessionState(sessionId, session);
|
|
342
454
|
await client.syncSession(session);
|
|
343
455
|
},
|
|
344
456
|
/**
|
|
@@ -346,12 +458,14 @@ function createPlugin() {
|
|
|
346
458
|
*/
|
|
347
459
|
UserPromptSubmit: async (event) => {
|
|
348
460
|
messageCounter++;
|
|
461
|
+
const sessionId = event.session_id || event.sessionId || "";
|
|
462
|
+
const prompt = event.prompt || "";
|
|
349
463
|
const message = {
|
|
350
|
-
sessionId
|
|
351
|
-
messageId: `${
|
|
464
|
+
sessionId,
|
|
465
|
+
messageId: `${sessionId}-msg-${messageCounter}`,
|
|
352
466
|
source: "claude-code",
|
|
353
467
|
role: "user",
|
|
354
|
-
content:
|
|
468
|
+
content: prompt,
|
|
355
469
|
timestamp: event.timestamp || new Date().toISOString(),
|
|
356
470
|
};
|
|
357
471
|
await client.syncMessage(message);
|
|
@@ -364,12 +478,14 @@ function createPlugin() {
|
|
|
364
478
|
return;
|
|
365
479
|
toolCallCounter++;
|
|
366
480
|
messageCounter++;
|
|
481
|
+
const sessionId = event.session_id || event.sessionId || "";
|
|
482
|
+
const toolName = event.tool_name || event.toolName || "unknown";
|
|
367
483
|
const message = {
|
|
368
|
-
sessionId
|
|
369
|
-
messageId: `${
|
|
484
|
+
sessionId,
|
|
485
|
+
messageId: `${sessionId}-tool-${toolCallCounter}`,
|
|
370
486
|
source: "claude-code",
|
|
371
487
|
role: "assistant",
|
|
372
|
-
toolName
|
|
488
|
+
toolName,
|
|
373
489
|
toolArgs: event.args,
|
|
374
490
|
toolResult: event.result,
|
|
375
491
|
durationMs: event.durationMs,
|
|
@@ -381,38 +497,51 @@ function createPlugin() {
|
|
|
381
497
|
* Called when Claude stops responding
|
|
382
498
|
*/
|
|
383
499
|
Stop: async (event) => {
|
|
384
|
-
|
|
385
|
-
const
|
|
386
|
-
|
|
387
|
-
|
|
388
|
-
|
|
389
|
-
|
|
390
|
-
|
|
391
|
-
|
|
392
|
-
|
|
393
|
-
|
|
394
|
-
|
|
500
|
+
const sessionId = event.session_id || event.sessionId || "";
|
|
501
|
+
const tokenUsage = event.tokenUsage || { input: 0, output: 0 };
|
|
502
|
+
// Only create message if we have response content
|
|
503
|
+
if (event.response) {
|
|
504
|
+
messageCounter++;
|
|
505
|
+
const message = {
|
|
506
|
+
sessionId,
|
|
507
|
+
messageId: `${sessionId}-msg-${messageCounter}`,
|
|
508
|
+
source: "claude-code",
|
|
509
|
+
role: "assistant",
|
|
510
|
+
content: event.response,
|
|
511
|
+
model: event.model,
|
|
512
|
+
tokenCount: tokenUsage.input + tokenUsage.output,
|
|
513
|
+
durationMs: event.durationMs,
|
|
514
|
+
timestamp: new Date().toISOString(),
|
|
515
|
+
};
|
|
516
|
+
await client.syncMessage(message);
|
|
517
|
+
}
|
|
395
518
|
// Update session state with token usage
|
|
396
|
-
|
|
397
|
-
|
|
398
|
-
|
|
399
|
-
|
|
400
|
-
|
|
401
|
-
|
|
402
|
-
|
|
403
|
-
|
|
404
|
-
|
|
519
|
+
if (tokenUsage.input > 0 || tokenUsage.output > 0) {
|
|
520
|
+
const currentState = client.getSessionState(sessionId);
|
|
521
|
+
const currentTokens = currentState.tokenUsage || { input: 0, output: 0 };
|
|
522
|
+
client.updateSessionState(sessionId, {
|
|
523
|
+
tokenUsage: {
|
|
524
|
+
input: currentTokens.input + tokenUsage.input,
|
|
525
|
+
output: currentTokens.output + tokenUsage.output,
|
|
526
|
+
},
|
|
527
|
+
});
|
|
528
|
+
}
|
|
405
529
|
},
|
|
406
530
|
/**
|
|
407
531
|
* Called when session ends
|
|
408
532
|
*/
|
|
409
533
|
SessionEnd: async (event) => {
|
|
410
|
-
const
|
|
534
|
+
const sessionId = event.session_id || event.sessionId || "";
|
|
535
|
+
const currentState = client.getSessionState(sessionId);
|
|
536
|
+
const endReason = event.reason || event.endReason;
|
|
411
537
|
const session = {
|
|
412
538
|
...currentState,
|
|
413
|
-
sessionId
|
|
539
|
+
sessionId,
|
|
414
540
|
source: "claude-code",
|
|
415
|
-
|
|
541
|
+
cwd: event.cwd,
|
|
542
|
+
projectPath: event.cwd,
|
|
543
|
+
projectName: event.cwd ? path.basename(event.cwd) : undefined,
|
|
544
|
+
endReason,
|
|
416
545
|
messageCount: event.messageCount,
|
|
417
546
|
toolCallCount: event.toolCallCount,
|
|
418
547
|
tokenUsage: event.totalTokenUsage,
|
|
@@ -420,11 +549,11 @@ function createPlugin() {
|
|
|
420
549
|
endedAt: new Date().toISOString(),
|
|
421
550
|
};
|
|
422
551
|
await client.syncSession(session);
|
|
423
|
-
client.clearSessionState(
|
|
424
|
-
console.log(`[claude-code-sync] Session synced: ${event.messageCount} messages, ${event.toolCallCount} tool calls`);
|
|
552
|
+
client.clearSessionState(sessionId);
|
|
553
|
+
console.log(`[claude-code-sync] Session synced: ${event.messageCount || 0} messages, ${event.toolCallCount || 0} tool calls`);
|
|
425
554
|
},
|
|
426
555
|
};
|
|
427
556
|
}
|
|
428
557
|
// Default export for Claude Code plugin system
|
|
429
558
|
exports.default = createPlugin;
|
|
430
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Claude Code Sync Plugin\n *\n * Syncs Claude Code sessions to OpenSync dashboard.\n * Uses API Key authentication (no browser OAuth required).\n *\n * Install: npm install -g claude-code-sync\n * Configure: claude-code-sync login\n */\n\nimport * as fs from \"fs\";\nimport * as path from \"path\";\nimport * as os from \"os\";\n\n// ============================================================================\n// Types\n// ============================================================================\n\nexport interface Config {\n  convexUrl: string;\n  apiKey: string;\n  autoSync?: boolean;\n  syncToolCalls?: boolean;\n  syncThinking?: boolean;\n}\n\nexport interface SessionData {\n  sessionId: string;\n  source: \"claude-code\";\n  title?: string;\n  projectPath?: string;\n  projectName?: string;\n  cwd?: string;\n  gitBranch?: string;\n  gitRepo?: string;\n  model?: string;\n  startType?: \"new\" | \"resume\" | \"continue\";\n  endReason?: \"user_stop\" | \"max_turns\" | \"error\" | \"completed\";\n  thinkingEnabled?: boolean;\n  permissionMode?: string;\n  mcpServers?: string[];\n  messageCount?: number;\n  toolCallCount?: number;\n  tokenUsage?: {\n    input: number;\n    output: number;\n  };\n  costEstimate?: number;\n  startedAt?: string;\n  endedAt?: string;\n}\n\nexport interface MessageData {\n  sessionId: string;\n  messageId: string;\n  source: \"claude-code\";\n  role: \"user\" | \"assistant\" | \"system\";\n  content?: string;\n  thinkingContent?: string;\n  toolName?: string;\n  toolArgs?: Record<string, unknown>;\n  toolResult?: string;\n  durationMs?: number;\n  tokenCount?: number;\n  timestamp?: string;\n  model?: string;\n}\n\nexport interface ToolUseData {\n  sessionId: string;\n  toolName: string;\n  toolArgs?: Record<string, unknown>;\n  result?: string;\n  success?: boolean;\n  durationMs?: number;\n  timestamp?: string;\n}\n\n// Claude Code Hook Types\nexport interface ClaudeCodeHooks {\n  SessionStart?: (data: SessionStartEvent) => void | Promise<void>;\n  UserPromptSubmit?: (data: UserPromptEvent) => void | Promise<void>;\n  PostToolUse?: (data: ToolUseEvent) => void | Promise<void>;\n  Stop?: (data: StopEvent) => void | Promise<void>;\n  SessionEnd?: (data: SessionEndEvent) => void | Promise<void>;\n}\n\nexport interface SessionStartEvent {\n  sessionId: string;\n  cwd: string;\n  model: string;\n  startType: \"new\" | \"resume\" | \"continue\";\n  thinkingEnabled?: boolean;\n  permissionMode?: string;\n  mcpServers?: string[];\n}\n\nexport interface UserPromptEvent {\n  sessionId: string;\n  prompt: string;\n  timestamp: string;\n}\n\nexport interface ToolUseEvent {\n  sessionId: string;\n  toolName: string;\n  args: Record<string, unknown>;\n  result: string;\n  success: boolean;\n  durationMs: number;\n}\n\nexport interface StopEvent {\n  sessionId: string;\n  response: string;\n  tokenUsage: {\n    input: number;\n    output: number;\n  };\n  durationMs: number;\n}\n\nexport interface SessionEndEvent {\n  sessionId: string;\n  endReason: \"user_stop\" | \"max_turns\" | \"error\" | \"completed\";\n  messageCount: number;\n  toolCallCount: number;\n  totalTokenUsage: {\n    input: number;\n    output: number;\n  };\n  costEstimate: number;\n}\n\n// ============================================================================\n// Configuration\n// ============================================================================\n\nconst CONFIG_DIR = path.join(os.homedir(), \".config\", \"claude-code-sync\");\nconst CONFIG_FILE = path.join(CONFIG_DIR, \"config.json\");\n\nexport function loadConfig(): Config | null {\n  // Check environment variables first\n  const envUrl = process.env.CLAUDE_SYNC_CONVEX_URL;\n  const envKey = process.env.CLAUDE_SYNC_API_KEY;\n\n  if (envUrl && envKey) {\n    return {\n      convexUrl: normalizeConvexUrl(envUrl),\n      apiKey: envKey,\n      autoSync: process.env.CLAUDE_SYNC_AUTO_SYNC !== \"false\",\n      syncToolCalls: process.env.CLAUDE_SYNC_TOOL_CALLS !== \"false\",\n      syncThinking: process.env.CLAUDE_SYNC_THINKING === \"true\",\n    };\n  }\n\n  // Fall back to config file\n  try {\n    if (fs.existsSync(CONFIG_FILE)) {\n      const data = fs.readFileSync(CONFIG_FILE, \"utf-8\");\n      const config = JSON.parse(data) as Config;\n      config.convexUrl = normalizeConvexUrl(config.convexUrl);\n      return config;\n    }\n  } catch (error) {\n    console.error(\"Error loading config:\", error);\n  }\n\n  return null;\n}\n\nexport function saveConfig(config: Config): void {\n  try {\n    if (!fs.existsSync(CONFIG_DIR)) {\n      fs.mkdirSync(CONFIG_DIR, { recursive: true });\n    }\n    fs.writeFileSync(CONFIG_FILE, JSON.stringify(config, null, 2));\n  } catch (error) {\n    console.error(\"Error saving config:\", error);\n    throw error;\n  }\n}\n\nexport function clearConfig(): void {\n  try {\n    if (fs.existsSync(CONFIG_FILE)) {\n      fs.unlinkSync(CONFIG_FILE);\n    }\n  } catch (error) {\n    console.error(\"Error clearing config:\", error);\n  }\n}\n\nfunction normalizeConvexUrl(url: string): string {\n  // Convert .convex.cloud to .convex.site for API calls\n  return url.replace(\".convex.cloud\", \".convex.site\");\n}\n\n// ============================================================================\n// Sync Client\n// ============================================================================\n\nexport class SyncClient {\n  private config: Config;\n  private siteUrl: string;\n  private sessionCache: Map<string, Partial<SessionData>> = new Map();\n\n  constructor(config: Config) {\n    this.config = config;\n    // Normalize URL to .convex.site for HTTP endpoints\n    // Supports both .convex.cloud and .convex.site input URLs\n    this.siteUrl = config.convexUrl.replace(\".convex.cloud\", \".convex.site\");\n  }\n\n  private async request(endpoint: string, data: unknown): Promise<unknown> {\n    const url = `${this.siteUrl}${endpoint}`;\n\n    const response = await fetch(url, {\n      method: \"POST\",\n      headers: {\n        \"Content-Type\": \"application/json\",\n        Authorization: `Bearer ${this.config.apiKey}`,\n      },\n      body: JSON.stringify(data),\n    });\n\n    if (!response.ok) {\n      const text = await response.text();\n      throw new Error(`Sync failed: ${response.status} - ${text}`);\n    }\n\n    return response.json();\n  }\n\n  // Transform session data to backend format\n  private transformSession(session: SessionData): Record<string, unknown> {\n    const payload: Record<string, unknown> = {\n      externalId: session.sessionId,\n      source: session.source,\n    };\n    \n    // Only include fields that are defined\n    if (session.title !== undefined) payload.title = session.title;\n    if (session.projectPath || session.cwd) payload.projectPath = session.projectPath || session.cwd;\n    if (session.projectName) payload.projectName = session.projectName;\n    if (session.model) payload.model = session.model;\n    if (session.tokenUsage?.input !== undefined) payload.promptTokens = session.tokenUsage.input;\n    if (session.tokenUsage?.output !== undefined) payload.completionTokens = session.tokenUsage.output;\n    if (session.costEstimate !== undefined) payload.cost = session.costEstimate;\n    if (session.messageCount !== undefined) payload.messageCount = session.messageCount;\n    if (session.toolCallCount !== undefined) payload.toolCallCount = session.toolCallCount;\n    if (session.endReason) payload.endReason = session.endReason;\n    \n    // Calculate duration if both timestamps exist\n    if (session.endedAt && session.startedAt) {\n      payload.durationMs = new Date(session.endedAt).getTime() - new Date(session.startedAt).getTime();\n    }\n    \n    return payload;\n  }\n\n  // Transform message data to backend format\n  private transformMessage(message: MessageData): Record<string, unknown> {\n    const payload: Record<string, unknown> = {\n      sessionExternalId: message.sessionId,\n      externalId: message.messageId,\n      role: message.role,\n      source: message.source,\n    };\n    \n    // Set textContent for user prompts and assistant responses\n    if (message.content) {\n      payload.textContent = message.content;\n    } else if (message.toolResult && !message.toolName) {\n      // Only use toolResult as textContent if no toolName (fallback)\n      payload.textContent = message.toolResult;\n    }\n    \n    // Include duration if available\n    if (message.durationMs !== undefined) {\n      payload.durationMs = message.durationMs;\n    }\n    \n    // Include token count if available\n    if (message.tokenCount !== undefined) {\n      payload.promptTokens = message.tokenCount;\n    }\n    \n    // Include model if available\n    if (message.model) {\n      payload.model = message.model;\n    }\n    \n    // Tool use info stored in parts\n    if (message.toolName) {\n      payload.parts = [\n        {\n          type: \"tool-call\",\n          content: {\n            toolName: message.toolName,\n            args: message.toolArgs,\n          },\n        },\n      ];\n      // Add tool result as separate part\n      if (message.toolResult) {\n        (payload.parts as Array<{ type: string; content: unknown }>).push({\n          type: \"tool-result\",\n          content: message.toolResult,\n        });\n      }\n    }\n    \n    return payload;\n  }\n\n  async syncSession(session: SessionData): Promise<void> {\n    try {\n      const payload = this.transformSession(session);\n      await this.request(\"/sync/session\", payload);\n    } catch (error) {\n      console.error(\"Failed to sync session:\", error);\n      throw error;\n    }\n  }\n\n  async syncMessage(message: MessageData): Promise<void> {\n    try {\n      const payload = this.transformMessage(message);\n      await this.request(\"/sync/message\", payload);\n    } catch (error) {\n      console.error(\"Failed to sync message:\", error);\n      throw error;\n    }\n  }\n\n  async syncBatch(\n    sessions: SessionData[],\n    messages: MessageData[]\n  ): Promise<void> {\n    try {\n      const transformedSessions = sessions.map((s) => this.transformSession(s));\n      const transformedMessages = messages.map((m) => this.transformMessage(m));\n      await this.request(\"/sync/batch\", {\n        sessions: transformedSessions,\n        messages: transformedMessages,\n      });\n    } catch (error) {\n      console.error(\"Failed to sync batch:\", error);\n      throw error;\n    }\n  }\n\n  async testConnection(): Promise<boolean> {\n    try {\n      const url = `${this.siteUrl}/health`;\n      const response = await fetch(url);\n      return response.ok;\n    } catch {\n      return false;\n    }\n  }\n\n  // Session state management\n  getSessionState(sessionId: string): Partial<SessionData> {\n    return this.sessionCache.get(sessionId) || {};\n  }\n\n  updateSessionState(\n    sessionId: string,\n    updates: Partial<SessionData>\n  ): void {\n    const current = this.sessionCache.get(sessionId) || {};\n    this.sessionCache.set(sessionId, { ...current, ...updates });\n  }\n\n  clearSessionState(sessionId: string): void {\n    this.sessionCache.delete(sessionId);\n  }\n}\n\n// ============================================================================\n// Plugin Export\n// ============================================================================\n\n/**\n * Claude Code Plugin Entry Point\n *\n * This function is called by Claude Code to register the plugin.\n * It returns hook handlers that fire at key points in the session lifecycle.\n */\nexport function createPlugin(): ClaudeCodeHooks | null {\n  const config = loadConfig();\n\n  if (!config) {\n    console.log(\n      \"[claude-code-sync] Not configured. Run 'claude-code-sync login' to set up.\"\n    );\n    return null;\n  }\n\n  if (config.autoSync === false) {\n    console.log(\"[claude-code-sync] Auto-sync disabled in config.\");\n    return null;\n  }\n\n  const client = new SyncClient(config);\n  let messageCounter = 0;\n  let toolCallCounter = 0;\n\n  console.log(\"[claude-code-sync] Plugin loaded. Sessions will sync to OpenSync.\");\n\n  return {\n    /**\n     * Called when a new session starts\n     */\n    SessionStart: async (event: SessionStartEvent) => {\n      messageCounter = 0;\n      toolCallCounter = 0;\n\n      const session: SessionData = {\n        sessionId: event.sessionId,\n        source: \"claude-code\",\n        cwd: event.cwd,\n        model: event.model,\n        startType: event.startType,\n        thinkingEnabled: event.thinkingEnabled,\n        permissionMode: event.permissionMode,\n        mcpServers: event.mcpServers,\n        startedAt: new Date().toISOString(),\n      };\n\n      // Extract project info from cwd\n      if (event.cwd) {\n        session.projectPath = event.cwd;\n        session.projectName = path.basename(event.cwd);\n\n        // Try to get git info\n        try {\n          const gitDir = path.join(event.cwd, \".git\");\n          if (fs.existsSync(gitDir)) {\n            const headFile = path.join(gitDir, \"HEAD\");\n            if (fs.existsSync(headFile)) {\n              const head = fs.readFileSync(headFile, \"utf-8\").trim();\n              if (head.startsWith(\"ref: refs/heads/\")) {\n                session.gitBranch = head.replace(\"ref: refs/heads/\", \"\");\n              }\n            }\n          }\n        } catch {\n          // Ignore git errors\n        }\n      }\n\n      client.updateSessionState(event.sessionId, session);\n      await client.syncSession(session);\n    },\n\n    /**\n     * Called when user submits a prompt\n     */\n    UserPromptSubmit: async (event: UserPromptEvent) => {\n      messageCounter++;\n\n      const message: MessageData = {\n        sessionId: event.sessionId,\n        messageId: `${event.sessionId}-msg-${messageCounter}`,\n        source: \"claude-code\",\n        role: \"user\",\n        content: event.prompt,\n        timestamp: event.timestamp || new Date().toISOString(),\n      };\n\n      await client.syncMessage(message);\n    },\n\n    /**\n     * Called after each tool use\n     */\n    PostToolUse: async (event: ToolUseEvent) => {\n      if (!config.syncToolCalls) return;\n\n      toolCallCounter++;\n      messageCounter++;\n\n      const message: MessageData = {\n        sessionId: event.sessionId,\n        messageId: `${event.sessionId}-tool-${toolCallCounter}`,\n        source: \"claude-code\",\n        role: \"assistant\",\n        toolName: event.toolName,\n        toolArgs: event.args,\n        toolResult: event.result,\n        durationMs: event.durationMs,\n        timestamp: new Date().toISOString(),\n      };\n\n      await client.syncMessage(message);\n    },\n\n    /**\n     * Called when Claude stops responding\n     */\n    Stop: async (event: StopEvent) => {\n      messageCounter++;\n\n      const message: MessageData = {\n        sessionId: event.sessionId,\n        messageId: `${event.sessionId}-msg-${messageCounter}`,\n        source: \"claude-code\",\n        role: \"assistant\",\n        content: event.response,\n        tokenCount: event.tokenUsage.input + event.tokenUsage.output,\n        durationMs: event.durationMs,\n        timestamp: new Date().toISOString(),\n      };\n\n      // Update session state with token usage\n      const currentState = client.getSessionState(event.sessionId);\n      const currentTokens = currentState.tokenUsage || { input: 0, output: 0 };\n      client.updateSessionState(event.sessionId, {\n        tokenUsage: {\n          input: currentTokens.input + event.tokenUsage.input,\n          output: currentTokens.output + event.tokenUsage.output,\n        },\n      });\n\n      await client.syncMessage(message);\n    },\n\n    /**\n     * Called when session ends\n     */\n    SessionEnd: async (event: SessionEndEvent) => {\n      const currentState = client.getSessionState(event.sessionId);\n\n      const session: SessionData = {\n        ...currentState,\n        sessionId: event.sessionId,\n        source: \"claude-code\",\n        endReason: event.endReason,\n        messageCount: event.messageCount,\n        toolCallCount: event.toolCallCount,\n        tokenUsage: event.totalTokenUsage,\n        costEstimate: event.costEstimate,\n        endedAt: new Date().toISOString(),\n      };\n\n      await client.syncSession(session);\n      client.clearSessionState(event.sessionId);\n\n      console.log(\n        `[claude-code-sync] Session synced: ${event.messageCount} messages, ${event.toolCallCount} tool calls`\n      );\n    },\n  };\n}\n\n// Default export for Claude Code plugin system\nexport default createPlugin;\n"]}
|
|
559
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Claude Code Sync Plugin\n *\n * Syncs Claude Code sessions to OpenSync dashboard.\n * Uses API Key authentication (no browser OAuth required).\n *\n * Install: npm install -g claude-code-sync\n * Configure: claude-code-sync login\n */\n\nimport * as fs from \"fs\";\nimport * as path from \"path\";\nimport * as os from \"os\";\n\n// ============================================================================\n// Types\n// ============================================================================\n\nexport interface Config {\n  convexUrl: string;\n  apiKey: string;\n  autoSync?: boolean;\n  syncToolCalls?: boolean;\n  syncThinking?: boolean;\n}\n\nexport interface SessionData {\n  sessionId: string;\n  source: \"claude-code\";\n  title?: string;\n  projectPath?: string;\n  projectName?: string;\n  cwd?: string;\n  gitBranch?: string;\n  gitRepo?: string;\n  model?: string;\n  startType?: \"new\" | \"resume\" | \"continue\";\n  endReason?: \"user_stop\" | \"max_turns\" | \"error\" | \"completed\" | \"clear\" | \"logout\" | \"prompt_input_exit\" | \"other\" | string;\n  thinkingEnabled?: boolean;\n  permissionMode?: string;\n  mcpServers?: string[];\n  messageCount?: number;\n  toolCallCount?: number;\n  tokenUsage?: {\n    input: number;\n    output: number;\n  };\n  costEstimate?: number;\n  startedAt?: string;\n  endedAt?: string;\n}\n\n// Message part type for tool calls and results\nexport interface MessagePart {\n  type: \"text\" | \"tool-call\" | \"tool-result\";\n  content: unknown;\n}\n\nexport interface MessageData {\n  sessionId: string;\n  messageId: string;\n  source: \"claude-code\";\n  role: \"user\" | \"assistant\" | \"system\";\n  content?: string;\n  thinkingContent?: string;\n  toolName?: string;\n  toolArgs?: Record<string, unknown>;\n  toolResult?: string;\n  parts?: MessagePart[];\n  durationMs?: number;\n  tokenCount?: number;\n  timestamp?: string;\n  model?: string;\n}\n\nexport interface ToolUseData {\n  sessionId: string;\n  toolName: string;\n  toolArgs?: Record<string, unknown>;\n  result?: string;\n  success?: boolean;\n  durationMs?: number;\n  timestamp?: string;\n}\n\n// Claude Code Hook Types\nexport interface ClaudeCodeHooks {\n  SessionStart?: (data: SessionStartEvent) => void | Promise<void>;\n  UserPromptSubmit?: (data: UserPromptEvent) => void | Promise<void>;\n  PostToolUse?: (data: ToolUseEvent) => void | Promise<void>;\n  Stop?: (data: StopEvent) => void | Promise<void>;\n  SessionEnd?: (data: SessionEndEvent) => void | Promise<void>;\n}\n\n// Actual Claude Code hook event interfaces (from documentation)\n// Supports both new format (session_id, transcript_path) and legacy format (sessionId)\nexport interface SessionStartEvent {\n  session_id: string;\n  transcript_path: string;\n  permission_mode: string;\n  hook_event_name: \"SessionStart\";\n  source: \"startup\" | \"resume\" | \"clear\" | \"compact\";\n  // Legacy fields (may or may not be provided)\n  sessionId?: string;\n  cwd?: string;\n  model?: string;\n  startType?: \"new\" | \"resume\" | \"continue\";\n  thinkingEnabled?: boolean;\n  permissionMode?: string;\n  mcpServers?: string[];\n}\n\nexport interface UserPromptEvent {\n  session_id: string;\n  transcript_path: string;\n  permission_mode: string;\n  hook_event_name: \"UserPromptSubmit\";\n  prompt?: string;\n  // Legacy fields\n  sessionId?: string;\n  timestamp?: string;\n}\n\nexport interface ToolUseEvent {\n  session_id: string;\n  transcript_path: string;\n  tool_name: string;\n  permission_mode: string;\n  hook_event_name: \"PostToolUse\";\n  // Legacy fields\n  sessionId?: string;\n  toolName?: string;\n  args?: Record<string, unknown>;\n  result?: string;\n  success?: boolean;\n  durationMs?: number;\n}\n\nexport interface StopEvent {\n  session_id: string;\n  transcript_path: string;\n  permission_mode: string;\n  hook_event_name: \"Stop\";\n  stop_hook_active: boolean;\n  // Legacy fields\n  sessionId?: string;\n  response?: string;\n  tokenUsage?: { input: number; output: number };\n  durationMs?: number;\n  model?: string;\n}\n\nexport interface SessionEndEvent {\n  session_id: string;\n  transcript_path: string;\n  cwd: string;\n  permission_mode: string;\n  hook_event_name: \"SessionEnd\";\n  reason: \"clear\" | \"logout\" | \"prompt_input_exit\" | \"other\";\n  // Legacy fields\n  sessionId?: string;\n  endReason?: string;\n  messageCount?: number;\n  toolCallCount?: number;\n  totalTokenUsage?: { input: number; output: number };\n  costEstimate?: number;\n}\n\n// ============================================================================\n// Transcript Parsing\n// ============================================================================\n\n// Transcript entry types for parsing JSONL files\ninterface TranscriptEntry {\n  type: string;\n  message?: {\n    model?: string;\n    role?: string;\n    usage?: {\n      input_tokens?: number;\n      output_tokens?: number;\n      cache_creation_input_tokens?: number;\n      cache_read_input_tokens?: number;\n    };\n    content?: unknown;\n  };\n  sessionId?: string;\n  cwd?: string;\n  slug?: string;\n}\n\n// Stats extracted from transcript\ninterface TranscriptStats {\n  model: string | undefined;\n  inputTokens: number;\n  cacheCreationTokens: number;\n  cacheReadTokens: number;\n  outputTokens: number;\n  messageCount: number;\n  toolCallCount: number;\n  title: string | undefined;\n  cwd: string | undefined;\n}\n\n/**\n * Parse transcript file to extract model, token usage, and stats\n */\nfunction parseTranscript(transcriptPath: string): TranscriptStats {\n  const stats: TranscriptStats = {\n    model: undefined,\n    inputTokens: 0,\n    cacheCreationTokens: 0,\n    cacheReadTokens: 0,\n    outputTokens: 0,\n    messageCount: 0,\n    toolCallCount: 0,\n    title: undefined,\n    cwd: undefined,\n  };\n\n  try {\n    if (!fs.existsSync(transcriptPath)) {\n      return stats;\n    }\n\n    const content = fs.readFileSync(transcriptPath, \"utf-8\");\n    const lines = content.trim().split(\"\\n\");\n\n    for (const line of lines) {\n      try {\n        const entry: TranscriptEntry = JSON.parse(line);\n\n        // Get cwd and title from first entry that has them\n        if (!stats.cwd && entry.cwd) {\n          stats.cwd = entry.cwd;\n        }\n        if (!stats.title && entry.slug) {\n          stats.title = entry.slug;\n        }\n\n        // Count messages\n        if (entry.type === \"user\") {\n          stats.messageCount++;\n        }\n\n        // Extract model and tokens from assistant messages\n        if (entry.type === \"assistant\" && entry.message) {\n          if (entry.message.model && !stats.model) {\n            stats.model = entry.message.model;\n          }\n\n          if (entry.message.usage) {\n            const usage = entry.message.usage;\n            stats.inputTokens += usage.input_tokens || 0;\n            stats.cacheCreationTokens += usage.cache_creation_input_tokens || 0;\n            stats.cacheReadTokens += usage.cache_read_input_tokens || 0;\n            stats.outputTokens += usage.output_tokens || 0;\n          }\n        }\n\n        // Count tool uses\n        if (entry.type === \"assistant\" && entry.message?.content) {\n          const msgContent = entry.message.content;\n          if (Array.isArray(msgContent)) {\n            for (const part of msgContent) {\n              if (part && typeof part === \"object\" && \"type\" in part && part.type === \"tool_use\") {\n                stats.toolCallCount++;\n              }\n            }\n          }\n        }\n      } catch {\n        // Skip malformed lines\n      }\n    }\n  } catch (error) {\n    console.error(\"[claude-code-sync] Error parsing transcript:\", error);\n  }\n\n  return stats;\n}\n\n// ============================================================================\n// Model Pricing & Cost Calculation\n// ============================================================================\n\n// Pricing per million tokens (USD) with cache pricing\n// Source: https://www.anthropic.com/pricing\nconst MODEL_PRICING: Record<string, { input: number; output: number; cacheWrite: number; cacheRead: number }> = {\n  \"claude-sonnet-4-20250514\": { input: 3.00, output: 15.00, cacheWrite: 3.75, cacheRead: 0.30 },\n  \"claude-opus-4-20250514\": { input: 15.00, output: 75.00, cacheWrite: 18.75, cacheRead: 1.50 },\n  \"claude-opus-4-5-20251101\": { input: 15.00, output: 75.00, cacheWrite: 18.75, cacheRead: 1.50 },\n  \"claude-3-5-sonnet-20241022\": { input: 3.00, output: 15.00, cacheWrite: 3.75, cacheRead: 0.30 },\n  \"claude-3-opus-20240229\": { input: 15.00, output: 75.00, cacheWrite: 18.75, cacheRead: 1.50 },\n  \"claude-3-5-haiku-20241022\": { input: 0.80, output: 4.00, cacheWrite: 1.00, cacheRead: 0.08 },\n};\n\n/**\n * Calculate cost from model and token usage with proper cache pricing\n * Returns 0 if model is unknown or pricing not available\n */\nfunction calculateCost(model: string | undefined, stats: TranscriptStats): number {\n  if (!model) return 0;\n\n  // Try exact match first\n  let pricing = MODEL_PRICING[model];\n\n  // Try partial match if exact match fails\n  if (!pricing) {\n    const matchingKey = Object.keys(MODEL_PRICING).find(k => model.includes(k) || k.includes(model));\n    if (matchingKey) {\n      pricing = MODEL_PRICING[matchingKey];\n    }\n  }\n\n  if (!pricing) return 0;\n\n  // Calculate cost with proper cache pricing\n  const inputCost = stats.inputTokens * pricing.input;\n  const cacheWriteCost = stats.cacheCreationTokens * pricing.cacheWrite;\n  const cacheReadCost = stats.cacheReadTokens * pricing.cacheRead;\n  const outputCost = stats.outputTokens * pricing.output;\n\n  return (inputCost + cacheWriteCost + cacheReadCost + outputCost) / 1_000_000;\n}\n\n// ============================================================================\n// Configuration\n// ============================================================================\n\nconst CONFIG_DIR = path.join(os.homedir(), \".config\", \"claude-code-sync\");\nconst CONFIG_FILE = path.join(CONFIG_DIR, \"config.json\");\n\nexport function loadConfig(): Config | null {\n  // Check environment variables first\n  const envUrl = process.env.CLAUDE_SYNC_CONVEX_URL;\n  const envKey = process.env.CLAUDE_SYNC_API_KEY;\n\n  if (envUrl && envKey) {\n    return {\n      convexUrl: normalizeConvexUrl(envUrl),\n      apiKey: envKey,\n      autoSync: process.env.CLAUDE_SYNC_AUTO_SYNC !== \"false\",\n      syncToolCalls: process.env.CLAUDE_SYNC_TOOL_CALLS !== \"false\",\n      syncThinking: process.env.CLAUDE_SYNC_THINKING === \"true\",\n    };\n  }\n\n  // Fall back to config file\n  try {\n    if (fs.existsSync(CONFIG_FILE)) {\n      const data = fs.readFileSync(CONFIG_FILE, \"utf-8\");\n      const config = JSON.parse(data) as Config;\n      config.convexUrl = normalizeConvexUrl(config.convexUrl);\n      return config;\n    }\n  } catch (error) {\n    console.error(\"Error loading config:\", error);\n  }\n\n  return null;\n}\n\nexport function saveConfig(config: Config): void {\n  try {\n    if (!fs.existsSync(CONFIG_DIR)) {\n      fs.mkdirSync(CONFIG_DIR, { recursive: true });\n    }\n    fs.writeFileSync(CONFIG_FILE, JSON.stringify(config, null, 2));\n  } catch (error) {\n    console.error(\"Error saving config:\", error);\n    throw error;\n  }\n}\n\nexport function clearConfig(): void {\n  try {\n    if (fs.existsSync(CONFIG_FILE)) {\n      fs.unlinkSync(CONFIG_FILE);\n    }\n  } catch (error) {\n    console.error(\"Error clearing config:\", error);\n  }\n}\n\nfunction normalizeConvexUrl(url: string): string {\n  // Convert .convex.cloud to .convex.site for API calls\n  return url.replace(\".convex.cloud\", \".convex.site\");\n}\n\n// ============================================================================\n// Sync Client\n// ============================================================================\n\nexport class SyncClient {\n  private config: Config;\n  private siteUrl: string;\n  private sessionCache: Map<string, Partial<SessionData>> = new Map();\n\n  constructor(config: Config) {\n    this.config = config;\n    // Normalize URL to .convex.site for HTTP endpoints\n    // Supports both .convex.cloud and .convex.site input URLs\n    this.siteUrl = config.convexUrl.replace(\".convex.cloud\", \".convex.site\");\n  }\n\n  private async request(endpoint: string, data: unknown): Promise<unknown> {\n    const url = `${this.siteUrl}${endpoint}`;\n\n    const response = await fetch(url, {\n      method: \"POST\",\n      headers: {\n        \"Content-Type\": \"application/json\",\n        Authorization: `Bearer ${this.config.apiKey}`,\n      },\n      body: JSON.stringify(data),\n    });\n\n    if (!response.ok) {\n      const text = await response.text();\n      throw new Error(`Sync failed: ${response.status} - ${text}`);\n    }\n\n    return response.json();\n  }\n\n  // Transform session data to backend format\n  private transformSession(session: SessionData): Record<string, unknown> {\n    const payload: Record<string, unknown> = {\n      externalId: session.sessionId,\n      source: session.source,\n    };\n    \n    // Only include fields that are defined\n    if (session.title !== undefined) payload.title = session.title;\n    if (session.projectPath || session.cwd) payload.projectPath = session.projectPath || session.cwd;\n    if (session.projectName) payload.projectName = session.projectName;\n    if (session.model) payload.model = session.model;\n    if (session.tokenUsage?.input !== undefined) payload.promptTokens = session.tokenUsage.input;\n    if (session.tokenUsage?.output !== undefined) payload.completionTokens = session.tokenUsage.output;\n    if (session.costEstimate !== undefined) payload.cost = session.costEstimate;\n    if (session.messageCount !== undefined) payload.messageCount = session.messageCount;\n    if (session.toolCallCount !== undefined) payload.toolCallCount = session.toolCallCount;\n    if (session.endReason) payload.endReason = session.endReason;\n    \n    // Calculate duration if both timestamps exist\n    if (session.endedAt && session.startedAt) {\n      payload.durationMs = new Date(session.endedAt).getTime() - new Date(session.startedAt).getTime();\n    }\n    \n    return payload;\n  }\n\n  // Transform message data to backend format\n  private transformMessage(message: MessageData): Record<string, unknown> {\n    const payload: Record<string, unknown> = {\n      sessionExternalId: message.sessionId,\n      externalId: message.messageId,\n      role: message.role,\n      source: message.source,\n    };\n    \n    // Set textContent for user prompts and assistant responses\n    if (message.content) {\n      payload.textContent = message.content;\n    } else if (message.toolResult && !message.toolName) {\n      // Only use toolResult as textContent if no toolName (fallback)\n      payload.textContent = message.toolResult;\n    }\n    \n    // Include duration if available\n    if (message.durationMs !== undefined) {\n      payload.durationMs = message.durationMs;\n    }\n    \n    // Include token count if available\n    if (message.tokenCount !== undefined) {\n      payload.promptTokens = message.tokenCount;\n    }\n    \n    // Include model if available\n    if (message.model) {\n      payload.model = message.model;\n    }\n    \n    // Tool use info stored in parts\n    if (message.toolName) {\n      payload.parts = [\n        {\n          type: \"tool-call\",\n          content: {\n            toolName: message.toolName,\n            args: message.toolArgs,\n          },\n        },\n      ];\n      // Add tool result as separate part\n      if (message.toolResult) {\n        (payload.parts as Array<{ type: string; content: unknown }>).push({\n          type: \"tool-result\",\n          content: message.toolResult,\n        });\n      }\n    }\n    \n    return payload;\n  }\n\n  async syncSession(session: SessionData): Promise<void> {\n    try {\n      const payload = this.transformSession(session);\n      await this.request(\"/sync/session\", payload);\n    } catch (error) {\n      console.error(\"Failed to sync session:\", error);\n      throw error;\n    }\n  }\n\n  async syncMessage(message: MessageData): Promise<void> {\n    try {\n      const payload = this.transformMessage(message);\n      await this.request(\"/sync/message\", payload);\n    } catch (error) {\n      console.error(\"Failed to sync message:\", error);\n      throw error;\n    }\n  }\n\n  async syncBatch(\n    sessions: SessionData[],\n    messages: MessageData[]\n  ): Promise<void> {\n    try {\n      const transformedSessions = sessions.map((s) => this.transformSession(s));\n      const transformedMessages = messages.map((m) => this.transformMessage(m));\n      await this.request(\"/sync/batch\", {\n        sessions: transformedSessions,\n        messages: transformedMessages,\n      });\n    } catch (error) {\n      console.error(\"Failed to sync batch:\", error);\n      throw error;\n    }\n  }\n\n  async testConnection(): Promise<boolean> {\n    try {\n      const url = `${this.siteUrl}/health`;\n      const response = await fetch(url);\n      return response.ok;\n    } catch {\n      return false;\n    }\n  }\n\n  // Session state management\n  getSessionState(sessionId: string): Partial<SessionData> {\n    return this.sessionCache.get(sessionId) || {};\n  }\n\n  updateSessionState(\n    sessionId: string,\n    updates: Partial<SessionData>\n  ): void {\n    const current = this.sessionCache.get(sessionId) || {};\n    this.sessionCache.set(sessionId, { ...current, ...updates });\n  }\n\n  clearSessionState(sessionId: string): void {\n    this.sessionCache.delete(sessionId);\n  }\n}\n\n// ============================================================================\n// Plugin Export\n// ============================================================================\n\n/**\n * Claude Code Plugin Entry Point\n *\n * This function is called by Claude Code to register the plugin.\n * It returns hook handlers that fire at key points in the session lifecycle.\n */\nexport function createPlugin(): ClaudeCodeHooks | null {\n  const config = loadConfig();\n\n  if (!config) {\n    console.log(\n      \"[claude-code-sync] Not configured. Run 'claude-code-sync login' to set up.\"\n    );\n    return null;\n  }\n\n  if (config.autoSync === false) {\n    console.log(\"[claude-code-sync] Auto-sync disabled in config.\");\n    return null;\n  }\n\n  const client = new SyncClient(config);\n  let messageCounter = 0;\n  let toolCallCounter = 0;\n\n  console.log(\"[claude-code-sync] Plugin loaded. Sessions will sync to OpenSync.\");\n\n  return {\n    /**\n     * Called when a new session starts\n     */\n    SessionStart: async (event: SessionStartEvent) => {\n      messageCounter = 0;\n      toolCallCounter = 0;\n\n      // Use session_id (new) or sessionId (legacy)\n      const sessionId = event.session_id || event.sessionId || \"\";\n      const cwd = event.cwd;\n      const permissionMode = event.permission_mode || event.permissionMode;\n\n      const session: SessionData = {\n        sessionId,\n        source: \"claude-code\",\n        cwd,\n        model: event.model,\n        startType: event.startType,\n        thinkingEnabled: event.thinkingEnabled,\n        permissionMode,\n        mcpServers: event.mcpServers,\n        startedAt: new Date().toISOString(),\n      };\n\n      // Extract project info from cwd\n      if (cwd) {\n        session.projectPath = cwd;\n        session.projectName = path.basename(cwd);\n\n        // Try to get git info\n        try {\n          const gitDir = path.join(cwd, \".git\");\n          if (fs.existsSync(gitDir)) {\n            const headFile = path.join(gitDir, \"HEAD\");\n            if (fs.existsSync(headFile)) {\n              const head = fs.readFileSync(headFile, \"utf-8\").trim();\n              if (head.startsWith(\"ref: refs/heads/\")) {\n                session.gitBranch = head.replace(\"ref: refs/heads/\", \"\");\n              }\n            }\n          }\n        } catch {\n          // Ignore git errors\n        }\n      }\n\n      client.updateSessionState(sessionId, session);\n      await client.syncSession(session);\n    },\n\n    /**\n     * Called when user submits a prompt\n     */\n    UserPromptSubmit: async (event: UserPromptEvent) => {\n      messageCounter++;\n      const sessionId = event.session_id || event.sessionId || \"\";\n      const prompt = event.prompt || \"\";\n\n      const message: MessageData = {\n        sessionId,\n        messageId: `${sessionId}-msg-${messageCounter}`,\n        source: \"claude-code\",\n        role: \"user\",\n        content: prompt,\n        timestamp: event.timestamp || new Date().toISOString(),\n      };\n\n      await client.syncMessage(message);\n    },\n\n    /**\n     * Called after each tool use\n     */\n    PostToolUse: async (event: ToolUseEvent) => {\n      if (!config.syncToolCalls) return;\n\n      toolCallCounter++;\n      messageCounter++;\n      const sessionId = event.session_id || event.sessionId || \"\";\n      const toolName = event.tool_name || event.toolName || \"unknown\";\n\n      const message: MessageData = {\n        sessionId,\n        messageId: `${sessionId}-tool-${toolCallCounter}`,\n        source: \"claude-code\",\n        role: \"assistant\",\n        toolName,\n        toolArgs: event.args,\n        toolResult: event.result,\n        durationMs: event.durationMs,\n        timestamp: new Date().toISOString(),\n      };\n\n      await client.syncMessage(message);\n    },\n\n    /**\n     * Called when Claude stops responding\n     */\n    Stop: async (event: StopEvent) => {\n      const sessionId = event.session_id || event.sessionId || \"\";\n      const tokenUsage = event.tokenUsage || { input: 0, output: 0 };\n\n      // Only create message if we have response content\n      if (event.response) {\n        messageCounter++;\n\n        const message: MessageData = {\n          sessionId,\n          messageId: `${sessionId}-msg-${messageCounter}`,\n          source: \"claude-code\",\n          role: \"assistant\",\n          content: event.response,\n          model: event.model,\n          tokenCount: tokenUsage.input + tokenUsage.output,\n          durationMs: event.durationMs,\n          timestamp: new Date().toISOString(),\n        };\n\n        await client.syncMessage(message);\n      }\n\n      // Update session state with token usage\n      if (tokenUsage.input > 0 || tokenUsage.output > 0) {\n        const currentState = client.getSessionState(sessionId);\n        const currentTokens = currentState.tokenUsage || { input: 0, output: 0 };\n        client.updateSessionState(sessionId, {\n          tokenUsage: {\n            input: currentTokens.input + tokenUsage.input,\n            output: currentTokens.output + tokenUsage.output,\n          },\n        });\n      }\n    },\n\n    /**\n     * Called when session ends\n     */\n    SessionEnd: async (event: SessionEndEvent) => {\n      const sessionId = event.session_id || event.sessionId || \"\";\n      const currentState = client.getSessionState(sessionId);\n      const endReason = event.reason || event.endReason;\n\n      const session: SessionData = {\n        ...currentState,\n        sessionId,\n        source: \"claude-code\",\n        cwd: event.cwd,\n        projectPath: event.cwd,\n        projectName: event.cwd ? path.basename(event.cwd) : undefined,\n        endReason,\n        messageCount: event.messageCount,\n        toolCallCount: event.toolCallCount,\n        tokenUsage: event.totalTokenUsage,\n        costEstimate: event.costEstimate,\n        endedAt: new Date().toISOString(),\n      };\n\n      await client.syncSession(session);\n      client.clearSessionState(sessionId);\n\n      console.log(\n        `[claude-code-sync] Session synced: ${event.messageCount || 0} messages, ${event.toolCallCount || 0} tool calls`\n      );\n    },\n  };\n}\n\n// Default export for Claude Code plugin system\nexport default createPlugin;\n"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "claude-code-sync",
|
|
3
|
-
"version": "0.1.
|
|
3
|
+
"version": "0.1.14",
|
|
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",
|