@planckspace/cli 0.0.2 → 0.1.0
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 +109 -0
- package/dist/__tests__/correlate.test.d.ts +2 -0
- package/dist/__tests__/correlate.test.d.ts.map +1 -0
- package/dist/__tests__/correlate.test.js +204 -0
- package/dist/__tests__/correlate.test.js.map +1 -0
- package/dist/commands/init.d.ts +2 -0
- package/dist/commands/init.d.ts.map +1 -0
- package/dist/commands/init.js +12 -0
- package/dist/commands/init.js.map +1 -0
- package/dist/commands/login.d.ts +2 -0
- package/dist/commands/login.d.ts.map +1 -0
- package/dist/commands/login.js +45 -0
- package/dist/commands/login.js.map +1 -0
- package/dist/commands/logout.d.ts +2 -0
- package/dist/commands/logout.d.ts.map +1 -0
- package/dist/commands/logout.js +7 -0
- package/dist/commands/logout.js.map +1 -0
- package/dist/commands/scan.d.ts +5 -0
- package/dist/commands/scan.d.ts.map +1 -0
- package/dist/commands/scan.js +51 -0
- package/dist/commands/scan.js.map +1 -0
- package/dist/commands/status.d.ts +2 -0
- package/dist/commands/status.d.ts.map +1 -0
- package/dist/commands/status.js +31 -0
- package/dist/commands/status.js.map +1 -0
- package/dist/commands/sync.d.ts +15 -0
- package/dist/commands/sync.d.ts.map +1 -0
- package/dist/commands/sync.js +77 -0
- package/dist/commands/sync.js.map +1 -0
- package/dist/config.d.ts +6 -0
- package/dist/config.d.ts.map +1 -0
- package/dist/config.js +40 -0
- package/dist/config.js.map +1 -0
- package/dist/correlate.d.ts +21 -0
- package/dist/correlate.d.ts.map +1 -0
- package/dist/correlate.js +204 -0
- package/dist/correlate.js.map +1 -0
- package/dist/db/store.d.ts +46 -0
- package/dist/db/store.d.ts.map +1 -0
- package/dist/db/store.js +210 -0
- package/dist/db/store.js.map +1 -0
- package/dist/env.d.ts +6 -0
- package/dist/env.d.ts.map +1 -0
- package/dist/env.js +7 -0
- package/dist/env.js.map +1 -0
- package/dist/index.d.ts +3 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +97 -0
- package/dist/index.js.map +1 -0
- package/dist/scrapers/__tests__/claudeCode.test.d.ts +2 -0
- package/dist/scrapers/__tests__/claudeCode.test.d.ts.map +1 -0
- package/dist/scrapers/__tests__/claudeCode.test.js +115 -0
- package/dist/scrapers/__tests__/claudeCode.test.js.map +1 -0
- package/dist/scrapers/__tests__/cursor.test.d.ts +2 -0
- package/dist/scrapers/__tests__/cursor.test.d.ts.map +1 -0
- package/dist/scrapers/__tests__/cursor.test.js +173 -0
- package/dist/scrapers/__tests__/cursor.test.js.map +1 -0
- package/dist/scrapers/__tests__/windsurf.test.d.ts +2 -0
- package/dist/scrapers/__tests__/windsurf.test.d.ts.map +1 -0
- package/dist/scrapers/__tests__/windsurf.test.js +87 -0
- package/dist/scrapers/__tests__/windsurf.test.js.map +1 -0
- package/dist/scrapers/claudeCode.d.ts +4 -0
- package/dist/scrapers/claudeCode.d.ts.map +1 -0
- package/dist/scrapers/claudeCode.js +211 -0
- package/dist/scrapers/claudeCode.js.map +1 -0
- package/dist/scrapers/cursor.d.ts +9 -0
- package/dist/scrapers/cursor.d.ts.map +1 -0
- package/dist/scrapers/cursor.js +280 -0
- package/dist/scrapers/cursor.js.map +1 -0
- package/dist/scrapers/repo.d.ts +12 -0
- package/dist/scrapers/repo.d.ts.map +1 -0
- package/dist/scrapers/repo.js +40 -0
- package/dist/scrapers/repo.js.map +1 -0
- package/dist/scrapers/types.d.ts +36 -0
- package/dist/scrapers/types.d.ts.map +1 -0
- package/dist/scrapers/types.js +5 -0
- package/dist/scrapers/types.js.map +1 -0
- package/dist/scrapers/windsurf.d.ts +9 -0
- package/dist/scrapers/windsurf.d.ts.map +1 -0
- package/dist/scrapers/windsurf.js +255 -0
- package/dist/scrapers/windsurf.js.map +1 -0
- package/dist/scripts/scanTest.d.ts +3 -0
- package/dist/scripts/scanTest.d.ts.map +1 -0
- package/dist/scripts/scanTest.js +146 -0
- package/dist/scripts/scanTest.js.map +1 -0
- package/dist/sync/__tests__/payload.privacy.test.d.ts +2 -0
- package/dist/sync/__tests__/payload.privacy.test.d.ts.map +1 -0
- package/dist/sync/__tests__/payload.privacy.test.js +100 -0
- package/dist/sync/__tests__/payload.privacy.test.js.map +1 -0
- package/dist/sync/payload.d.ts +14 -0
- package/dist/sync/payload.d.ts.map +1 -0
- package/dist/sync/payload.js +36 -0
- package/dist/sync/payload.js.map +1 -0
- package/dist/sync/syncEngine.d.ts +16 -0
- package/dist/sync/syncEngine.d.ts.map +1 -0
- package/dist/sync/syncEngine.js +76 -0
- package/dist/sync/syncEngine.js.map +1 -0
- package/install.sh +126 -0
- package/package.json +39 -7
- package/index.js +0 -1
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"env.d.ts","sourceRoot":"","sources":["../src/env.ts"],"names":[],"mappings":"AAAA,OAAO,eAAe,CAAC;AAIvB,eAAO,MAAM,GAAG;;;CAGN,CAAC"}
|
package/dist/env.js
ADDED
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import "dotenv/config";
|
|
2
|
+
// .env is dev-only. At runtime, user tokens come from ~/.planckspace/config.json.
|
|
3
|
+
export const env = {
|
|
4
|
+
apiUrl: process.env.PLANCKSPACE_API_URL ?? "https://api.planckspace.dev",
|
|
5
|
+
testToken: process.env.PLANCKSPACE_TEST_TOKEN ?? null,
|
|
6
|
+
};
|
|
7
|
+
//# sourceMappingURL=env.js.map
|
package/dist/env.js.map
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"env.js","sourceRoot":"","sources":["../src/env.ts"],"names":[],"mappings":"AAAA,OAAO,eAAe,CAAC;AAEvB,kFAAkF;AAElF,MAAM,CAAC,MAAM,GAAG,GAAG;IACjB,MAAM,EAAE,OAAO,CAAC,GAAG,CAAC,mBAAmB,IAAI,6BAA6B;IACxE,SAAS,EAAE,OAAO,CAAC,GAAG,CAAC,sBAAsB,IAAI,IAAI;CAC7C,CAAC"}
|
package/dist/index.d.ts
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";AACA,OAAO,eAAe,CAAC"}
|
package/dist/index.js
ADDED
|
@@ -0,0 +1,97 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
import "dotenv/config";
|
|
3
|
+
import { Command } from "commander";
|
|
4
|
+
import { readFileSync } from "fs";
|
|
5
|
+
import { fileURLToPath } from "url";
|
|
6
|
+
import { dirname, join } from "path";
|
|
7
|
+
import { runInit } from "./commands/init.js";
|
|
8
|
+
import { runScan } from "./commands/scan.js";
|
|
9
|
+
import { runStatus } from "./commands/status.js";
|
|
10
|
+
import { runLogin } from "./commands/login.js";
|
|
11
|
+
import { runLogout } from "./commands/logout.js";
|
|
12
|
+
import { runSync } from "./commands/sync.js";
|
|
13
|
+
const __dirname = dirname(fileURLToPath(import.meta.url));
|
|
14
|
+
const pkg = JSON.parse(readFileSync(join(__dirname, "..", "package.json"), "utf-8"));
|
|
15
|
+
const program = new Command();
|
|
16
|
+
program
|
|
17
|
+
.name("planck")
|
|
18
|
+
.description("PlanckSpace CLI — sync AI token usage to the team ledger")
|
|
19
|
+
.version(pkg.version);
|
|
20
|
+
program
|
|
21
|
+
.command("init")
|
|
22
|
+
.description("Initialize ~/.planckspace workspace and local database")
|
|
23
|
+
.action(() => {
|
|
24
|
+
try {
|
|
25
|
+
runInit();
|
|
26
|
+
}
|
|
27
|
+
catch (err) {
|
|
28
|
+
console.error(String(err));
|
|
29
|
+
process.exit(1);
|
|
30
|
+
}
|
|
31
|
+
});
|
|
32
|
+
program
|
|
33
|
+
.command("scan")
|
|
34
|
+
.description("Scan Claude Code logs and save sessions to local database")
|
|
35
|
+
.option("--sync", "Sync to the team ledger after scanning")
|
|
36
|
+
.action(async (options) => {
|
|
37
|
+
try {
|
|
38
|
+
await runScan({ sync: options.sync });
|
|
39
|
+
}
|
|
40
|
+
catch (err) {
|
|
41
|
+
console.error(String(err));
|
|
42
|
+
process.exit(1);
|
|
43
|
+
}
|
|
44
|
+
});
|
|
45
|
+
program
|
|
46
|
+
.command("sync")
|
|
47
|
+
.description("Push unsynced session metadata to the team ledger")
|
|
48
|
+
.option("-w, --watch", "Keep running and re-sync on an interval")
|
|
49
|
+
.option("-i, --interval <seconds>", "Watch interval in seconds (default 60)", (v) => parseInt(v, 10))
|
|
50
|
+
.action(async (options) => {
|
|
51
|
+
try {
|
|
52
|
+
await runSync({ watch: options.watch, interval: options.interval });
|
|
53
|
+
}
|
|
54
|
+
catch (err) {
|
|
55
|
+
console.error(String(err));
|
|
56
|
+
process.exit(1);
|
|
57
|
+
}
|
|
58
|
+
});
|
|
59
|
+
program
|
|
60
|
+
.command("status")
|
|
61
|
+
.description("Show workspace status and session statistics")
|
|
62
|
+
.action(() => {
|
|
63
|
+
try {
|
|
64
|
+
runStatus();
|
|
65
|
+
}
|
|
66
|
+
catch (err) {
|
|
67
|
+
console.error(String(err));
|
|
68
|
+
process.exit(1);
|
|
69
|
+
}
|
|
70
|
+
});
|
|
71
|
+
program
|
|
72
|
+
.command("login [token]")
|
|
73
|
+
.description("Connect to a PlanckSpace workspace")
|
|
74
|
+
.option("-t, --token <token>", "API token (for non-interactive/script installs)")
|
|
75
|
+
.action(async (tokenArg, options) => {
|
|
76
|
+
try {
|
|
77
|
+
await runLogin(options.token ?? tokenArg);
|
|
78
|
+
}
|
|
79
|
+
catch (err) {
|
|
80
|
+
console.error(String(err));
|
|
81
|
+
process.exit(1);
|
|
82
|
+
}
|
|
83
|
+
});
|
|
84
|
+
program
|
|
85
|
+
.command("logout")
|
|
86
|
+
.description("Disconnect from PlanckSpace (local data retained)")
|
|
87
|
+
.action(() => {
|
|
88
|
+
try {
|
|
89
|
+
runLogout();
|
|
90
|
+
}
|
|
91
|
+
catch (err) {
|
|
92
|
+
console.error(String(err));
|
|
93
|
+
process.exit(1);
|
|
94
|
+
}
|
|
95
|
+
});
|
|
96
|
+
program.parse();
|
|
97
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"claudeCode.test.d.ts","sourceRoot":"","sources":["../../../src/scrapers/__tests__/claudeCode.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,115 @@
|
|
|
1
|
+
import { describe, it, expect } from "vitest";
|
|
2
|
+
import fs from "fs";
|
|
3
|
+
import path from "path";
|
|
4
|
+
import { fileURLToPath } from "url";
|
|
5
|
+
// We test the internal parser by re-exporting it through a thin helper so we
|
|
6
|
+
// don't have to call the real scanClaudeCodeLogs() (which hits the filesystem).
|
|
7
|
+
// Instead we parse fixture files directly using the same logic.
|
|
8
|
+
import { scanClaudeCodeLogs } from "../claudeCode.js";
|
|
9
|
+
const __dirname = path.dirname(fileURLToPath(import.meta.url));
|
|
10
|
+
const fixturesDir = path.join(__dirname, "fixtures");
|
|
11
|
+
// Helper: run the parser over a single fixture file using the exported public
|
|
12
|
+
// API by monkey-patching os.homedir to point at a temp dir we control.
|
|
13
|
+
import os from "os";
|
|
14
|
+
import { vi } from "vitest";
|
|
15
|
+
function makeFakeHome(sessions) {
|
|
16
|
+
const tmp = fs.mkdtempSync(path.join(os.tmpdir(), "planck-test-"));
|
|
17
|
+
const projectsDir = path.join(tmp, ".claude", "projects", "test-project");
|
|
18
|
+
fs.mkdirSync(projectsDir, { recursive: true });
|
|
19
|
+
for (const [name, content] of Object.entries(sessions)) {
|
|
20
|
+
fs.writeFileSync(path.join(projectsDir, name), content, "utf-8");
|
|
21
|
+
}
|
|
22
|
+
return tmp;
|
|
23
|
+
}
|
|
24
|
+
function readFixture(name) {
|
|
25
|
+
return fs.readFileSync(path.join(fixturesDir, name), "utf-8");
|
|
26
|
+
}
|
|
27
|
+
describe("scanClaudeCodeLogs", () => {
|
|
28
|
+
it("parses a single-turn session correctly", async () => {
|
|
29
|
+
const content = readFixture("session-single-turn.jsonl");
|
|
30
|
+
const fakeHome = makeFakeHome({ "aaaaaaaa-0000-0000-0000-000000000001.jsonl": content });
|
|
31
|
+
vi.spyOn(os, "homedir").mockReturnValue(fakeHome);
|
|
32
|
+
const sessions = await scanClaudeCodeLogs();
|
|
33
|
+
vi.restoreAllMocks();
|
|
34
|
+
expect(sessions).toHaveLength(1);
|
|
35
|
+
const s = sessions[0];
|
|
36
|
+
expect(s.sessionId).toBe("aaaaaaaa-0000-0000-0000-000000000001");
|
|
37
|
+
expect(s.tool).toBe("claude_code");
|
|
38
|
+
expect(s.model).toBe("claude-sonnet-4-6");
|
|
39
|
+
expect(s.turnCount).toBe(1);
|
|
40
|
+
expect(s.inputTokens).toBe(1200);
|
|
41
|
+
expect(s.outputTokens).toBe(350);
|
|
42
|
+
expect(s.cacheWriteTokens).toBe(800);
|
|
43
|
+
expect(s.cacheReadTokens).toBe(0);
|
|
44
|
+
expect(s.costUsd).toBeGreaterThan(0);
|
|
45
|
+
expect(s.startedAt).toBe("2026-05-01T10:00:00.000Z");
|
|
46
|
+
expect(s.endedAt).toBe("2026-05-01T10:00:05.000Z");
|
|
47
|
+
expect(s.turns).toHaveLength(1);
|
|
48
|
+
expect(s.turns[0].sequence).toBe(1);
|
|
49
|
+
expect(s.turns[0].model).toBe("claude-sonnet-4-6");
|
|
50
|
+
expect(s.turns[0].costUsd).toBeGreaterThan(0);
|
|
51
|
+
expect(s.turns[0].cacheReadTokens).toBe(0);
|
|
52
|
+
expect(s.turns[0].cacheWriteTokens).toBe(800);
|
|
53
|
+
expect(s.turns[0].toolCalls).toHaveLength(0);
|
|
54
|
+
expect(s.turns[0].filesTouched).toHaveLength(0);
|
|
55
|
+
});
|
|
56
|
+
it("deduplicates events sharing the same message.id when counting tokens", async () => {
|
|
57
|
+
const content = readFixture("session-multi-turn-tools.jsonl");
|
|
58
|
+
const fakeHome = makeFakeHome({ "bbbbbbbb-0000-0000-0000-000000000002.jsonl": content });
|
|
59
|
+
vi.spyOn(os, "homedir").mockReturnValue(fakeHome);
|
|
60
|
+
const sessions = await scanClaudeCodeLogs();
|
|
61
|
+
vi.restoreAllMocks();
|
|
62
|
+
expect(sessions).toHaveLength(1);
|
|
63
|
+
const s = sessions[0];
|
|
64
|
+
expect(s.turnCount).toBe(3); // msg_BBBB0001, msg_BBBB0002, msg_BBBB0003
|
|
65
|
+
// msg_BBBB0001 appears in 2 events but tokens should be counted once
|
|
66
|
+
expect(s.inputTokens).toBe(2000 + 3000 + 3200);
|
|
67
|
+
expect(s.outputTokens).toBe(120 + 200 + 180);
|
|
68
|
+
expect(s.cacheReadTokens).toBe(0 + 3500 + 5500);
|
|
69
|
+
expect(s.cacheWriteTokens).toBe(1500 + 0 + 0);
|
|
70
|
+
});
|
|
71
|
+
it("aggregates tool calls and filesTouched per turn", async () => {
|
|
72
|
+
const content = readFixture("session-multi-turn-tools.jsonl");
|
|
73
|
+
const fakeHome = makeFakeHome({ "bbbbbbbb-0000-0000-0000-000000000002.jsonl": content });
|
|
74
|
+
vi.spyOn(os, "homedir").mockReturnValue(fakeHome);
|
|
75
|
+
const sessions = await scanClaudeCodeLogs();
|
|
76
|
+
vi.restoreAllMocks();
|
|
77
|
+
const s = sessions[0];
|
|
78
|
+
const turn1 = s.turns.find((t) => t.sequence === 1);
|
|
79
|
+
const turn2 = s.turns.find((t) => t.sequence === 2);
|
|
80
|
+
const turn3 = s.turns.find((t) => t.sequence === 3);
|
|
81
|
+
expect(turn1.toolCalls).toEqual([{ name: "Read", count: 1 }]);
|
|
82
|
+
expect(turn1.filesTouched).toEqual(["d:\\projects\\my-app\\src\\index.ts"]);
|
|
83
|
+
expect(turn2.toolCalls).toEqual([{ name: "Edit", count: 1 }]);
|
|
84
|
+
expect(turn2.filesTouched).toEqual(["d:\\projects\\my-app\\src\\index.ts"]);
|
|
85
|
+
expect(turn3.toolCalls).toHaveLength(0);
|
|
86
|
+
expect(turn3.filesTouched).toHaveLength(0);
|
|
87
|
+
// filesTouchedCount is unique file count across all turns
|
|
88
|
+
expect(s.filesTouchedCount).toBe(1);
|
|
89
|
+
});
|
|
90
|
+
it("skips malformed lines without crashing and deduplicates same message.id", async () => {
|
|
91
|
+
const content = readFixture("session-malformed-lines.jsonl");
|
|
92
|
+
const fakeHome = makeFakeHome({ "cccccccc-0000-0000-0000-000000000003.jsonl": content });
|
|
93
|
+
vi.spyOn(os, "homedir").mockReturnValue(fakeHome);
|
|
94
|
+
const sessions = await scanClaudeCodeLogs();
|
|
95
|
+
vi.restoreAllMocks();
|
|
96
|
+
expect(sessions).toHaveLength(1);
|
|
97
|
+
const s = sessions[0];
|
|
98
|
+
// msg_CCCC0001 appears twice but tokens counted once
|
|
99
|
+
expect(s.inputTokens).toBe(500);
|
|
100
|
+
expect(s.outputTokens).toBe(150);
|
|
101
|
+
expect(s.cacheReadTokens).toBe(2000);
|
|
102
|
+
expect(s.model).toBe("claude-opus-4-8");
|
|
103
|
+
// The Bash tool_use is collected (not a file tool, so no filesTouched)
|
|
104
|
+
expect(s.turns[0].toolCalls).toEqual([{ name: "Bash", count: 1 }]);
|
|
105
|
+
expect(s.turns[0].filesTouched).toHaveLength(0);
|
|
106
|
+
expect(s.filesTouchedCount).toBe(0);
|
|
107
|
+
});
|
|
108
|
+
it("returns empty array when projects dir does not exist", async () => {
|
|
109
|
+
vi.spyOn(os, "homedir").mockReturnValue("/nonexistent-path-xyz");
|
|
110
|
+
const sessions = await scanClaudeCodeLogs();
|
|
111
|
+
vi.restoreAllMocks();
|
|
112
|
+
expect(sessions).toEqual([]);
|
|
113
|
+
});
|
|
114
|
+
});
|
|
115
|
+
//# sourceMappingURL=claudeCode.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"claudeCode.test.js","sourceRoot":"","sources":["../../../src/scrapers/__tests__/claudeCode.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"cursor.test.d.ts","sourceRoot":"","sources":["../../../src/scrapers/__tests__/cursor.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,173 @@
|
|
|
1
|
+
import { describe, it, expect, afterEach } from "vitest";
|
|
2
|
+
import fs from "fs";
|
|
3
|
+
import os from "os";
|
|
4
|
+
import path from "path";
|
|
5
|
+
import Database from "better-sqlite3";
|
|
6
|
+
import { scanCursorLogs } from "../cursor.js";
|
|
7
|
+
// Build a throwaway Cursor `state.vscdb` that mirrors the real schema we verified:
|
|
8
|
+
// ItemTable["composer.composerHeaders"] + cursorDiskKV["composerData:*"]/["bubbleId:*"].
|
|
9
|
+
function makeCursorDb(headers, kv) {
|
|
10
|
+
const dir = fs.mkdtempSync(path.join(os.tmpdir(), "planck-cursor-"));
|
|
11
|
+
const dbPath = path.join(dir, "state.vscdb");
|
|
12
|
+
const db = new Database(dbPath);
|
|
13
|
+
db.exec("CREATE TABLE ItemTable (key TEXT PRIMARY KEY, value BLOB)");
|
|
14
|
+
db.exec("CREATE TABLE cursorDiskKV (key TEXT PRIMARY KEY, value BLOB)");
|
|
15
|
+
db.prepare("INSERT INTO ItemTable (key, value) VALUES (?, ?)").run("composer.composerHeaders", JSON.stringify({ allComposers: headers }));
|
|
16
|
+
const ins = db.prepare("INSERT INTO cursorDiskKV (key, value) VALUES (?, ?)");
|
|
17
|
+
for (const [k, v] of Object.entries(kv))
|
|
18
|
+
ins.run(k, JSON.stringify(v));
|
|
19
|
+
db.close();
|
|
20
|
+
return dbPath;
|
|
21
|
+
}
|
|
22
|
+
const tempDirs = [];
|
|
23
|
+
function track(dbPath) {
|
|
24
|
+
tempDirs.push(path.dirname(dbPath));
|
|
25
|
+
return dbPath;
|
|
26
|
+
}
|
|
27
|
+
afterEach(() => {
|
|
28
|
+
while (tempDirs.length) {
|
|
29
|
+
const d = tempDirs.pop();
|
|
30
|
+
try {
|
|
31
|
+
fs.rmSync(d, { recursive: true, force: true });
|
|
32
|
+
}
|
|
33
|
+
catch {
|
|
34
|
+
/* ignore */
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
});
|
|
38
|
+
describe("scanCursorLogs", () => {
|
|
39
|
+
it("returns [] when the DB does not exist", () => {
|
|
40
|
+
expect(scanCursorLogs(path.join(os.tmpdir(), "does-not-exist-xyz", "state.vscdb"))).toEqual([]);
|
|
41
|
+
});
|
|
42
|
+
it("parses a composer into a session: groups user prompts into turns, uses filesChangedCount, captures tool calls", () => {
|
|
43
|
+
const id = "11111111-1111-1111-1111-111111111111";
|
|
44
|
+
const headers = [
|
|
45
|
+
{
|
|
46
|
+
composerId: id,
|
|
47
|
+
name: "Add feature",
|
|
48
|
+
createdAt: 1_700_000_000_000,
|
|
49
|
+
lastUpdatedAt: 1_700_000_900_000,
|
|
50
|
+
filesChangedCount: 7,
|
|
51
|
+
workspaceIdentifier: { uri: { fsPath: "/tmp/no-such-repo" } },
|
|
52
|
+
},
|
|
53
|
+
];
|
|
54
|
+
const order = [
|
|
55
|
+
{ bubbleId: "b1", type: 1 },
|
|
56
|
+
{ bubbleId: "b2", type: 2 },
|
|
57
|
+
{ bubbleId: "b3", type: 2 },
|
|
58
|
+
{ bubbleId: "b4", type: 1 },
|
|
59
|
+
{ bubbleId: "b5", type: 2 },
|
|
60
|
+
];
|
|
61
|
+
const kv = {
|
|
62
|
+
[`composerData:${id}`]: {
|
|
63
|
+
composerId: id,
|
|
64
|
+
createdAt: 1_700_000_000_000,
|
|
65
|
+
modelConfig: { modelName: "default" },
|
|
66
|
+
fullConversationHeadersOnly: order,
|
|
67
|
+
},
|
|
68
|
+
[`bubbleId:${id}:b1`]: { type: 1, createdAt: "2026-05-01T10:00:00.000Z" },
|
|
69
|
+
[`bubbleId:${id}:b2`]: {
|
|
70
|
+
type: 2,
|
|
71
|
+
createdAt: "2026-05-01T10:00:05.000Z",
|
|
72
|
+
tokenCount: { inputTokens: 0, outputTokens: 0 },
|
|
73
|
+
modelInfo: { modelName: "default" },
|
|
74
|
+
toolFormerData: { name: "read_file_v2", params: JSON.stringify({ targetFile: "src/a.ts" }) },
|
|
75
|
+
},
|
|
76
|
+
[`bubbleId:${id}:b3`]: {
|
|
77
|
+
type: 2,
|
|
78
|
+
createdAt: "2026-05-01T10:00:09.000Z",
|
|
79
|
+
toolFormerData: { name: "edit_file_v2", params: JSON.stringify({ relativeWorkspacePath: "src/a.ts" }) },
|
|
80
|
+
},
|
|
81
|
+
[`bubbleId:${id}:b4`]: { type: 1, createdAt: "2026-05-01T10:00:20.000Z" },
|
|
82
|
+
[`bubbleId:${id}:b5`]: {
|
|
83
|
+
type: 2,
|
|
84
|
+
createdAt: "2026-05-01T10:00:25.000Z",
|
|
85
|
+
toolFormerData: { name: "edit_file_v2", params: JSON.stringify({ relativeWorkspacePath: "src/b.ts" }) },
|
|
86
|
+
},
|
|
87
|
+
};
|
|
88
|
+
const dbPath = track(makeCursorDb(headers, kv));
|
|
89
|
+
const sessions = scanCursorLogs(dbPath);
|
|
90
|
+
expect(sessions).toHaveLength(1);
|
|
91
|
+
const s = sessions[0];
|
|
92
|
+
expect(s.sessionId).toBe(`cursor:${id}`);
|
|
93
|
+
expect(s.tool).toBe("cursor");
|
|
94
|
+
expect(s.model).toBe("default");
|
|
95
|
+
// Two user prompts → two exchanges.
|
|
96
|
+
expect(s.turnCount).toBe(2);
|
|
97
|
+
expect(s.turns).toHaveLength(2);
|
|
98
|
+
// Timing comes from the bubbles.
|
|
99
|
+
expect(s.startedAt).toBe("2026-05-01T10:00:00.000Z");
|
|
100
|
+
expect(s.endedAt).toBe("2026-05-01T10:00:25.000Z");
|
|
101
|
+
// Cursor stores no tokens locally → honest zeros, cost $0.
|
|
102
|
+
expect(s.inputTokens).toBe(0);
|
|
103
|
+
expect(s.outputTokens).toBe(0);
|
|
104
|
+
expect(s.costUsd).toBe(0);
|
|
105
|
+
// filesTouchedCount is Cursor's authoritative count, not our scrape.
|
|
106
|
+
expect(s.filesTouchedCount).toBe(7);
|
|
107
|
+
// Tool calls are captured per turn (versioned names recognised).
|
|
108
|
+
expect(s.turns[0].toolCalls).toEqual([
|
|
109
|
+
{ name: "read_file_v2", count: 1 },
|
|
110
|
+
{ name: "edit_file_v2", count: 1 },
|
|
111
|
+
]);
|
|
112
|
+
expect(s.turns[0].filesTouched).toEqual(["src/a.ts"]);
|
|
113
|
+
expect(s.turns[1].toolCalls).toEqual([{ name: "edit_file_v2", count: 1 }]);
|
|
114
|
+
expect(s.turns[1].filesTouched).toEqual(["src/b.ts"]);
|
|
115
|
+
});
|
|
116
|
+
it("computes cost when a real model and token counts ARE present (so we don't hardcode 0)", () => {
|
|
117
|
+
const id = "22222222-2222-2222-2222-222222222222";
|
|
118
|
+
const headers = [{ composerId: id, createdAt: 1, filesChangedCount: 1, workspaceIdentifier: { uri: { fsPath: "/tmp/x" } } }];
|
|
119
|
+
const kv = {
|
|
120
|
+
[`composerData:${id}`]: {
|
|
121
|
+
composerId: id,
|
|
122
|
+
modelConfig: { modelName: "claude-sonnet-4-6" },
|
|
123
|
+
fullConversationHeadersOnly: [
|
|
124
|
+
{ bubbleId: "u", type: 1 },
|
|
125
|
+
{ bubbleId: "a", type: 2 },
|
|
126
|
+
],
|
|
127
|
+
},
|
|
128
|
+
[`bubbleId:${id}:u`]: { type: 1, createdAt: "2026-05-01T10:00:00.000Z" },
|
|
129
|
+
[`bubbleId:${id}:a`]: {
|
|
130
|
+
type: 2,
|
|
131
|
+
createdAt: "2026-05-01T10:00:05.000Z",
|
|
132
|
+
modelInfo: { modelName: "claude-sonnet-4-6" },
|
|
133
|
+
tokenCount: { inputTokens: 1000, outputTokens: 500 },
|
|
134
|
+
},
|
|
135
|
+
};
|
|
136
|
+
const dbPath = track(makeCursorDb(headers, kv));
|
|
137
|
+
const s = scanCursorLogs(dbPath)[0];
|
|
138
|
+
expect(s.inputTokens).toBe(1000);
|
|
139
|
+
expect(s.outputTokens).toBe(500);
|
|
140
|
+
expect(s.model).toBe("claude-sonnet-4-6");
|
|
141
|
+
// sonnet: (1000*3 + 500*15)/1e6 = 0.0105
|
|
142
|
+
expect(s.costUsd).toBeCloseTo(0.0105, 6);
|
|
143
|
+
});
|
|
144
|
+
it("skips empty/placeholder composers that have no user prompt", () => {
|
|
145
|
+
const id = "33333333-3333-3333-3333-333333333333";
|
|
146
|
+
const headers = [{ composerId: id, createdAt: 1, workspaceIdentifier: { uri: { fsPath: "/tmp/x" } } }];
|
|
147
|
+
const kv = {
|
|
148
|
+
[`composerData:${id}`]: { composerId: id, fullConversationHeadersOnly: [] },
|
|
149
|
+
};
|
|
150
|
+
const dbPath = track(makeCursorDb(headers, kv));
|
|
151
|
+
expect(scanCursorLogs(dbPath)).toEqual([]);
|
|
152
|
+
});
|
|
153
|
+
it("resolves repo name and branch from the workspace path's .git", () => {
|
|
154
|
+
const id = "44444444-4444-4444-4444-444444444444";
|
|
155
|
+
const repoDir = fs.mkdtempSync(path.join(os.tmpdir(), "planck-repo-"));
|
|
156
|
+
tempDirs.push(repoDir);
|
|
157
|
+
fs.mkdirSync(path.join(repoDir, ".git"), { recursive: true });
|
|
158
|
+
fs.writeFileSync(path.join(repoDir, ".git", "HEAD"), "ref: refs/heads/feature-x\n");
|
|
159
|
+
const headers = [{ composerId: id, createdAt: 1, filesChangedCount: 0, workspaceIdentifier: { uri: { fsPath: repoDir } } }];
|
|
160
|
+
const kv = {
|
|
161
|
+
[`composerData:${id}`]: {
|
|
162
|
+
composerId: id,
|
|
163
|
+
fullConversationHeadersOnly: [{ bubbleId: "u", type: 1 }],
|
|
164
|
+
},
|
|
165
|
+
[`bubbleId:${id}:u`]: { type: 1, createdAt: "2026-05-01T10:00:00.000Z" },
|
|
166
|
+
};
|
|
167
|
+
const dbPath = track(makeCursorDb(headers, kv));
|
|
168
|
+
const s = scanCursorLogs(dbPath)[0];
|
|
169
|
+
expect(s.repoName).toBe(path.basename(repoDir));
|
|
170
|
+
expect(s.gitBranch).toBe("feature-x");
|
|
171
|
+
});
|
|
172
|
+
});
|
|
173
|
+
//# sourceMappingURL=cursor.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"cursor.test.js","sourceRoot":"","sources":["../../../src/scrapers/__tests__/cursor.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"windsurf.test.d.ts","sourceRoot":"","sources":["../../../src/scrapers/__tests__/windsurf.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
import { describe, it, expect, afterEach, vi } from "vitest";
|
|
2
|
+
import fs from "fs";
|
|
3
|
+
import os from "os";
|
|
4
|
+
import path from "path";
|
|
5
|
+
import Database from "better-sqlite3";
|
|
6
|
+
import { scanWindsurfLogs } from "../windsurf.js";
|
|
7
|
+
// Windsurf is a VS Code fork: its state DB is an ItemTable(key, value) KV store.
|
|
8
|
+
function makeWindsurfDb(items) {
|
|
9
|
+
const dir = fs.mkdtempSync(path.join(os.tmpdir(), "planck-windsurf-"));
|
|
10
|
+
const dbPath = path.join(dir, "state.vscdb");
|
|
11
|
+
const db = new Database(dbPath);
|
|
12
|
+
db.exec("CREATE TABLE ItemTable (key TEXT PRIMARY KEY, value BLOB)");
|
|
13
|
+
const ins = db.prepare("INSERT INTO ItemTable (key, value) VALUES (?, ?)");
|
|
14
|
+
for (const [k, v] of Object.entries(items))
|
|
15
|
+
ins.run(k, JSON.stringify(v));
|
|
16
|
+
db.close();
|
|
17
|
+
return dbPath;
|
|
18
|
+
}
|
|
19
|
+
const tempDirs = [];
|
|
20
|
+
function track(p) {
|
|
21
|
+
tempDirs.push(path.dirname(p));
|
|
22
|
+
return p;
|
|
23
|
+
}
|
|
24
|
+
afterEach(() => {
|
|
25
|
+
vi.restoreAllMocks();
|
|
26
|
+
while (tempDirs.length) {
|
|
27
|
+
try {
|
|
28
|
+
fs.rmSync(tempDirs.pop(), { recursive: true, force: true });
|
|
29
|
+
}
|
|
30
|
+
catch {
|
|
31
|
+
/* ignore */
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
});
|
|
35
|
+
describe("scanWindsurfLogs", () => {
|
|
36
|
+
it("returns [] when no Windsurf DB exists", () => {
|
|
37
|
+
expect(scanWindsurfLogs([path.join(os.tmpdir(), "nope-xyz", "state.vscdb")])).toEqual([]);
|
|
38
|
+
});
|
|
39
|
+
it("parses a recognised Cascade conversation when the schema is present", () => {
|
|
40
|
+
const dbPath = track(makeWindsurfDb({
|
|
41
|
+
"cascade.conversations": [
|
|
42
|
+
{
|
|
43
|
+
id: "conv-1",
|
|
44
|
+
workspacePath: "/tmp/no-such-repo",
|
|
45
|
+
messages: [
|
|
46
|
+
{ role: "user", createdAt: "2026-05-02T09:00:00.000Z" },
|
|
47
|
+
{
|
|
48
|
+
role: "assistant",
|
|
49
|
+
createdAt: "2026-05-02T09:00:08.000Z",
|
|
50
|
+
model: "claude-sonnet-4-6",
|
|
51
|
+
usage: { inputTokens: 200, outputTokens: 100 },
|
|
52
|
+
},
|
|
53
|
+
],
|
|
54
|
+
},
|
|
55
|
+
],
|
|
56
|
+
}));
|
|
57
|
+
const sessions = scanWindsurfLogs([dbPath]);
|
|
58
|
+
expect(sessions).toHaveLength(1);
|
|
59
|
+
const s = sessions[0];
|
|
60
|
+
expect(s.tool).toBe("windsurf");
|
|
61
|
+
expect(s.sessionId).toBe("windsurf:conv-1");
|
|
62
|
+
expect(s.turnCount).toBe(1);
|
|
63
|
+
expect(s.startedAt).toBe("2026-05-02T09:00:00.000Z");
|
|
64
|
+
expect(s.endedAt).toBe("2026-05-02T09:00:08.000Z");
|
|
65
|
+
expect(s.model).toBe("claude-sonnet-4-6");
|
|
66
|
+
expect(s.inputTokens).toBe(200);
|
|
67
|
+
expect(s.outputTokens).toBe(100);
|
|
68
|
+
// sonnet pricing: (200*3 + 100*15)/1e6
|
|
69
|
+
expect(s.costUsd).toBeCloseTo(0.0021, 6);
|
|
70
|
+
});
|
|
71
|
+
it("does NOT fabricate sessions when a Cascade-like key exists but the schema is unrecognised", () => {
|
|
72
|
+
const warn = vi.spyOn(console, "warn").mockImplementation(() => { });
|
|
73
|
+
const dbPath = track(makeWindsurfDb({
|
|
74
|
+
// Key matches a Cascade pattern, but the value is not a conversation.
|
|
75
|
+
"cascade.settings": { theme: "dark", enabled: true },
|
|
76
|
+
}));
|
|
77
|
+
expect(scanWindsurfLogs([dbPath])).toEqual([]);
|
|
78
|
+
expect(warn).toHaveBeenCalledWith(expect.stringContaining("schema"));
|
|
79
|
+
});
|
|
80
|
+
it("ignores unrelated keys entirely (no warning, no sessions)", () => {
|
|
81
|
+
const warn = vi.spyOn(console, "warn").mockImplementation(() => { });
|
|
82
|
+
const dbPath = track(makeWindsurfDb({ "workbench.layout": { sidebar: 200 } }));
|
|
83
|
+
expect(scanWindsurfLogs([dbPath])).toEqual([]);
|
|
84
|
+
expect(warn).not.toHaveBeenCalled();
|
|
85
|
+
});
|
|
86
|
+
});
|
|
87
|
+
//# sourceMappingURL=windsurf.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"windsurf.test.js","sourceRoot":"","sources":["../../../src/scrapers/__tests__/windsurf.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"claudeCode.d.ts","sourceRoot":"","sources":["../../src/scrapers/claudeCode.ts"],"names":[],"mappings":"AAKA,OAAO,KAAK,EAAQ,aAAa,EAAE,MAAM,YAAY,CAAC;AAGtD,YAAY,EAAE,IAAI,EAAE,aAAa,EAAE,MAAM,YAAY,CAAC;AAoPtD,wBAAsB,kBAAkB,IAAI,OAAO,CAAC,aAAa,EAAE,CAAC,CAsCnE"}
|