@asifkibria/claude-code-toolkit 1.3.0 → 1.4.2
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 +21 -6
- package/dist/__tests__/alerts.test.d.ts +2 -0
- package/dist/__tests__/alerts.test.d.ts.map +1 -0
- package/dist/__tests__/alerts.test.js +195 -0
- package/dist/__tests__/alerts.test.js.map +1 -0
- package/dist/__tests__/bookmarks.test.d.ts +2 -0
- package/dist/__tests__/bookmarks.test.d.ts.map +1 -0
- package/dist/__tests__/bookmarks.test.js +170 -0
- package/dist/__tests__/bookmarks.test.js.map +1 -0
- package/dist/__tests__/bulk.test.d.ts +2 -0
- package/dist/__tests__/bulk.test.d.ts.map +1 -0
- package/dist/__tests__/bulk.test.js +245 -0
- package/dist/__tests__/bulk.test.js.map +1 -0
- package/dist/__tests__/dashboard.test.js +2 -2
- package/dist/__tests__/dashboard.test.js.map +1 -1
- package/dist/__tests__/git.test.d.ts +2 -0
- package/dist/__tests__/git.test.d.ts.map +1 -0
- package/dist/__tests__/git.test.js +216 -0
- package/dist/__tests__/git.test.js.map +1 -0
- package/dist/__tests__/logs.test.d.ts +2 -0
- package/dist/__tests__/logs.test.d.ts.map +1 -0
- package/dist/__tests__/logs.test.js +196 -0
- package/dist/__tests__/logs.test.js.map +1 -0
- package/dist/__tests__/search.test.d.ts +2 -0
- package/dist/__tests__/search.test.d.ts.map +1 -0
- package/dist/__tests__/search.test.js +155 -0
- package/dist/__tests__/search.test.js.map +1 -0
- package/dist/cli.js +269 -12
- package/dist/cli.js.map +1 -1
- package/dist/index.js +5 -2
- package/dist/index.js.map +1 -1
- package/dist/lib/bookmarks.d.ts +69 -0
- package/dist/lib/bookmarks.d.ts.map +1 -0
- package/dist/lib/bookmarks.js +225 -0
- package/dist/lib/bookmarks.js.map +1 -0
- package/dist/lib/bulk.d.ts +78 -0
- package/dist/lib/bulk.d.ts.map +1 -0
- package/dist/lib/bulk.js +257 -0
- package/dist/lib/bulk.js.map +1 -0
- package/dist/lib/dashboard-ui.d.ts.map +1 -1
- package/dist/lib/dashboard-ui.js +425 -19
- package/dist/lib/dashboard-ui.js.map +1 -1
- package/dist/lib/dashboard.d.ts.map +1 -1
- package/dist/lib/dashboard.js +398 -20
- package/dist/lib/dashboard.js.map +1 -1
- package/dist/lib/export.d.ts +41 -0
- package/dist/lib/export.d.ts.map +1 -0
- package/dist/lib/export.js +428 -0
- package/dist/lib/export.js.map +1 -0
- package/dist/lib/git.d.ts.map +1 -1
- package/dist/lib/git.js +2 -1
- package/dist/lib/git.js.map +1 -1
- package/dist/lib/mcp-validator.d.ts.map +1 -1
- package/dist/lib/mcp-validator.js +2 -1
- package/dist/lib/mcp-validator.js.map +1 -1
- package/dist/lib/scanner.d.ts.map +1 -1
- package/dist/lib/scanner.js +2 -12
- package/dist/lib/scanner.js.map +1 -1
- package/dist/lib/security.d.ts.map +1 -1
- package/dist/lib/security.js +12 -1
- package/dist/lib/security.js.map +1 -1
- package/dist/lib/session-recovery.d.ts.map +1 -1
- package/dist/lib/session-recovery.js +25 -2
- package/dist/lib/session-recovery.js.map +1 -1
- package/dist/lib/utils.d.ts +15 -0
- package/dist/lib/utils.d.ts.map +1 -0
- package/dist/lib/utils.js +78 -0
- package/dist/lib/utils.js.map +1 -0
- package/package.json +2 -2
|
@@ -0,0 +1,216 @@
|
|
|
1
|
+
import { describe, it, expect, beforeEach, afterEach } from "vitest";
|
|
2
|
+
import * as fs from "fs";
|
|
3
|
+
import * as path from "path";
|
|
4
|
+
import * as os from "os";
|
|
5
|
+
import { getGitInfo, formatGitLinkReport } from "../lib/git.js";
|
|
6
|
+
const TEST_DIR = path.join(os.tmpdir(), "cct-git-test");
|
|
7
|
+
describe("Git Module", () => {
|
|
8
|
+
beforeEach(() => {
|
|
9
|
+
if (fs.existsSync(TEST_DIR))
|
|
10
|
+
fs.rmSync(TEST_DIR, { recursive: true });
|
|
11
|
+
fs.mkdirSync(TEST_DIR, { recursive: true });
|
|
12
|
+
});
|
|
13
|
+
afterEach(() => {
|
|
14
|
+
if (fs.existsSync(TEST_DIR))
|
|
15
|
+
fs.rmSync(TEST_DIR, { recursive: true });
|
|
16
|
+
});
|
|
17
|
+
describe("getGitInfo", () => {
|
|
18
|
+
it("should return null for nonexistent path", () => {
|
|
19
|
+
const result = getGitInfo(path.join(TEST_DIR, "nonexistent"));
|
|
20
|
+
expect(result).toBeNull();
|
|
21
|
+
});
|
|
22
|
+
it("should return null for a directory that is not a git repo", () => {
|
|
23
|
+
const nonGitDir = path.join(TEST_DIR, "not-a-repo");
|
|
24
|
+
fs.mkdirSync(nonGitDir, { recursive: true });
|
|
25
|
+
fs.writeFileSync(path.join(nonGitDir, "file.txt"), "content");
|
|
26
|
+
const result = getGitInfo(nonGitDir);
|
|
27
|
+
expect(result).toBeNull();
|
|
28
|
+
});
|
|
29
|
+
it("should return GitInfo for a valid git repository", () => {
|
|
30
|
+
const { execFileSync } = require("child_process");
|
|
31
|
+
const repoDir = path.join(TEST_DIR, "my-repo");
|
|
32
|
+
fs.mkdirSync(repoDir, { recursive: true });
|
|
33
|
+
try {
|
|
34
|
+
execFileSync("git", ["init"], { cwd: repoDir, stdio: "pipe" });
|
|
35
|
+
execFileSync("git", ["config", "user.email", "test@test.com"], { cwd: repoDir, stdio: "pipe" });
|
|
36
|
+
execFileSync("git", ["config", "user.name", "Test"], { cwd: repoDir, stdio: "pipe" });
|
|
37
|
+
fs.writeFileSync(path.join(repoDir, "README.md"), "# Test");
|
|
38
|
+
execFileSync("git", ["add", "."], { cwd: repoDir, stdio: "pipe" });
|
|
39
|
+
execFileSync("git", ["commit", "-m", "Initial commit"], { cwd: repoDir, stdio: "pipe" });
|
|
40
|
+
const result = getGitInfo(repoDir);
|
|
41
|
+
expect(result).not.toBeNull();
|
|
42
|
+
expect(result.branch).toBeTruthy();
|
|
43
|
+
expect(result.commit).toBeTruthy();
|
|
44
|
+
expect(result.commitMessage).toBe("Initial commit");
|
|
45
|
+
}
|
|
46
|
+
catch {
|
|
47
|
+
// Skip if git not available in CI
|
|
48
|
+
}
|
|
49
|
+
});
|
|
50
|
+
it("should detect dirty working tree", () => {
|
|
51
|
+
const { execFileSync } = require("child_process");
|
|
52
|
+
const repoDir = path.join(TEST_DIR, "dirty-repo");
|
|
53
|
+
fs.mkdirSync(repoDir, { recursive: true });
|
|
54
|
+
try {
|
|
55
|
+
execFileSync("git", ["init"], { cwd: repoDir, stdio: "pipe" });
|
|
56
|
+
execFileSync("git", ["config", "user.email", "test@test.com"], { cwd: repoDir, stdio: "pipe" });
|
|
57
|
+
execFileSync("git", ["config", "user.name", "Test"], { cwd: repoDir, stdio: "pipe" });
|
|
58
|
+
fs.writeFileSync(path.join(repoDir, "README.md"), "# Test");
|
|
59
|
+
execFileSync("git", ["add", "."], { cwd: repoDir, stdio: "pipe" });
|
|
60
|
+
execFileSync("git", ["commit", "-m", "init"], { cwd: repoDir, stdio: "pipe" });
|
|
61
|
+
fs.writeFileSync(path.join(repoDir, "new-file.ts"), "code");
|
|
62
|
+
const result = getGitInfo(repoDir);
|
|
63
|
+
if (result) {
|
|
64
|
+
expect(result.isDirty).toBe(true);
|
|
65
|
+
expect(result.uncommittedChanges).toBeGreaterThan(0);
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
catch {
|
|
69
|
+
// Skip if git not available
|
|
70
|
+
}
|
|
71
|
+
});
|
|
72
|
+
it("should detect clean working tree", () => {
|
|
73
|
+
const { execFileSync } = require("child_process");
|
|
74
|
+
const repoDir = path.join(TEST_DIR, "clean-repo");
|
|
75
|
+
fs.mkdirSync(repoDir, { recursive: true });
|
|
76
|
+
try {
|
|
77
|
+
execFileSync("git", ["init"], { cwd: repoDir, stdio: "pipe" });
|
|
78
|
+
execFileSync("git", ["config", "user.email", "test@test.com"], { cwd: repoDir, stdio: "pipe" });
|
|
79
|
+
execFileSync("git", ["config", "user.name", "Test"], { cwd: repoDir, stdio: "pipe" });
|
|
80
|
+
fs.writeFileSync(path.join(repoDir, "README.md"), "# Test");
|
|
81
|
+
execFileSync("git", ["add", "."], { cwd: repoDir, stdio: "pipe" });
|
|
82
|
+
execFileSync("git", ["commit", "-m", "init"], { cwd: repoDir, stdio: "pipe" });
|
|
83
|
+
const result = getGitInfo(repoDir);
|
|
84
|
+
if (result) {
|
|
85
|
+
expect(result.isDirty).toBe(false);
|
|
86
|
+
expect(result.uncommittedChanges).toBe(0);
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
catch {
|
|
90
|
+
// Skip if git not available
|
|
91
|
+
}
|
|
92
|
+
});
|
|
93
|
+
});
|
|
94
|
+
describe("formatGitLinkReport", () => {
|
|
95
|
+
it("should include summary counts", () => {
|
|
96
|
+
const report = {
|
|
97
|
+
sessionsWithGit: 3,
|
|
98
|
+
sessionsWithoutGit: 2,
|
|
99
|
+
branches: new Map([["main", 2], ["feature/test", 1]]),
|
|
100
|
+
commits: new Map([["abc1234", ["session-1"]]]),
|
|
101
|
+
links: [],
|
|
102
|
+
};
|
|
103
|
+
const output = formatGitLinkReport(report);
|
|
104
|
+
expect(output).toContain("Sessions with Git: 3");
|
|
105
|
+
expect(output).toContain("Sessions without Git: 2");
|
|
106
|
+
});
|
|
107
|
+
it("should list branches with session counts", () => {
|
|
108
|
+
const report = {
|
|
109
|
+
sessionsWithGit: 2,
|
|
110
|
+
sessionsWithoutGit: 0,
|
|
111
|
+
branches: new Map([["main", 2], ["dev", 1]]),
|
|
112
|
+
commits: new Map(),
|
|
113
|
+
links: [],
|
|
114
|
+
};
|
|
115
|
+
const output = formatGitLinkReport(report);
|
|
116
|
+
expect(output).toContain("main: 2");
|
|
117
|
+
expect(output).toContain("dev: 1");
|
|
118
|
+
});
|
|
119
|
+
it("should show git-linked session details", () => {
|
|
120
|
+
const report = {
|
|
121
|
+
sessionsWithGit: 1,
|
|
122
|
+
sessionsWithoutGit: 0,
|
|
123
|
+
branches: new Map([["main", 1]]),
|
|
124
|
+
commits: new Map(),
|
|
125
|
+
links: [{
|
|
126
|
+
sessionId: "abcdef123456",
|
|
127
|
+
project: "my-project",
|
|
128
|
+
projectPath: "/tmp/my-project",
|
|
129
|
+
git: {
|
|
130
|
+
branch: "main",
|
|
131
|
+
commit: "abc1234",
|
|
132
|
+
commitMessage: "Add feature",
|
|
133
|
+
commitDate: new Date(),
|
|
134
|
+
isDirty: false,
|
|
135
|
+
uncommittedChanges: 0,
|
|
136
|
+
},
|
|
137
|
+
sessionCreated: new Date(),
|
|
138
|
+
sessionModified: new Date(),
|
|
139
|
+
hasGit: true,
|
|
140
|
+
}],
|
|
141
|
+
};
|
|
142
|
+
const output = formatGitLinkReport(report);
|
|
143
|
+
expect(output).toContain("main");
|
|
144
|
+
expect(output).toContain("abc1234");
|
|
145
|
+
});
|
|
146
|
+
it("should show sessions without git section", () => {
|
|
147
|
+
const report = {
|
|
148
|
+
sessionsWithGit: 0,
|
|
149
|
+
sessionsWithoutGit: 1,
|
|
150
|
+
branches: new Map(),
|
|
151
|
+
commits: new Map(),
|
|
152
|
+
links: [{
|
|
153
|
+
sessionId: "orphan-session-id",
|
|
154
|
+
project: "no-git-project",
|
|
155
|
+
projectPath: "/tmp/no-git",
|
|
156
|
+
git: undefined,
|
|
157
|
+
sessionCreated: new Date(),
|
|
158
|
+
sessionModified: new Date(),
|
|
159
|
+
hasGit: false,
|
|
160
|
+
}],
|
|
161
|
+
};
|
|
162
|
+
const output = formatGitLinkReport(report);
|
|
163
|
+
expect(output).toContain("Sessions without Git");
|
|
164
|
+
});
|
|
165
|
+
it("should handle dirty sessions with uncommitted changes warning", () => {
|
|
166
|
+
const report = {
|
|
167
|
+
sessionsWithGit: 1,
|
|
168
|
+
sessionsWithoutGit: 0,
|
|
169
|
+
branches: new Map([["main", 1]]),
|
|
170
|
+
commits: new Map(),
|
|
171
|
+
links: [{
|
|
172
|
+
sessionId: "dirty-session",
|
|
173
|
+
project: "my-project",
|
|
174
|
+
projectPath: "/tmp/my-project",
|
|
175
|
+
git: {
|
|
176
|
+
branch: "main",
|
|
177
|
+
commit: "def5678",
|
|
178
|
+
commitMessage: "WIP",
|
|
179
|
+
commitDate: new Date(),
|
|
180
|
+
isDirty: true,
|
|
181
|
+
uncommittedChanges: 3,
|
|
182
|
+
},
|
|
183
|
+
sessionCreated: new Date(),
|
|
184
|
+
sessionModified: new Date(),
|
|
185
|
+
hasGit: true,
|
|
186
|
+
}],
|
|
187
|
+
};
|
|
188
|
+
const output = formatGitLinkReport(report);
|
|
189
|
+
expect(output).toContain("3 uncommitted");
|
|
190
|
+
});
|
|
191
|
+
it("should handle empty report gracefully", () => {
|
|
192
|
+
const report = {
|
|
193
|
+
sessionsWithGit: 0,
|
|
194
|
+
sessionsWithoutGit: 0,
|
|
195
|
+
branches: new Map(),
|
|
196
|
+
commits: new Map(),
|
|
197
|
+
links: [],
|
|
198
|
+
};
|
|
199
|
+
const output = formatGitLinkReport(report);
|
|
200
|
+
expect(output).toContain("Git Integration Report");
|
|
201
|
+
});
|
|
202
|
+
it("should include unique branches and commits counts", () => {
|
|
203
|
+
const report = {
|
|
204
|
+
sessionsWithGit: 2,
|
|
205
|
+
sessionsWithoutGit: 0,
|
|
206
|
+
branches: new Map([["main", 1], ["feature", 1]]),
|
|
207
|
+
commits: new Map([["abc", ["s1"]], ["def", ["s2"]]]),
|
|
208
|
+
links: [],
|
|
209
|
+
};
|
|
210
|
+
const output = formatGitLinkReport(report);
|
|
211
|
+
expect(output).toContain("Unique branches: 2");
|
|
212
|
+
expect(output).toContain("Unique commits: 2");
|
|
213
|
+
});
|
|
214
|
+
});
|
|
215
|
+
});
|
|
216
|
+
//# sourceMappingURL=git.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"git.test.js","sourceRoot":"","sources":["../../src/__tests__/git.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"logs.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/logs.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,196 @@
|
|
|
1
|
+
import { describe, it, expect, beforeEach, afterEach } from "vitest";
|
|
2
|
+
import * as fs from "fs";
|
|
3
|
+
import * as path from "path";
|
|
4
|
+
import * as os from "os";
|
|
5
|
+
import { listLogFiles, parseLogLine, parseLogFile, getLogSummary, searchLogs, getErrorLogs, } from "../lib/logs.js";
|
|
6
|
+
const TEST_DIR = path.join(os.tmpdir(), "cct-logs-test");
|
|
7
|
+
const DEBUG_DIR = path.join(TEST_DIR, "debug");
|
|
8
|
+
function createLogFile(name, lines) {
|
|
9
|
+
fs.mkdirSync(DEBUG_DIR, { recursive: true });
|
|
10
|
+
const filePath = path.join(DEBUG_DIR, name);
|
|
11
|
+
fs.writeFileSync(filePath, lines.join("\n"), "utf-8");
|
|
12
|
+
return filePath;
|
|
13
|
+
}
|
|
14
|
+
const ts = "2026-01-01T10:00:00.000Z";
|
|
15
|
+
const logLine = (level, component, message) => `${ts} [${level}] [${component}] ${message}`;
|
|
16
|
+
describe("Logs Module", () => {
|
|
17
|
+
beforeEach(() => {
|
|
18
|
+
if (fs.existsSync(TEST_DIR))
|
|
19
|
+
fs.rmSync(TEST_DIR, { recursive: true });
|
|
20
|
+
fs.mkdirSync(DEBUG_DIR, { recursive: true });
|
|
21
|
+
});
|
|
22
|
+
afterEach(() => {
|
|
23
|
+
if (fs.existsSync(TEST_DIR))
|
|
24
|
+
fs.rmSync(TEST_DIR, { recursive: true });
|
|
25
|
+
});
|
|
26
|
+
describe("listLogFiles", () => {
|
|
27
|
+
it("should return empty array when debug dir does not exist", () => {
|
|
28
|
+
const result = listLogFiles(path.join(TEST_DIR, "nonexistent"));
|
|
29
|
+
expect(result).toHaveLength(0);
|
|
30
|
+
});
|
|
31
|
+
it("should list .txt log files", () => {
|
|
32
|
+
createLogFile("debug-2026-01-01.txt", [logLine("INFO", "App", "started")]);
|
|
33
|
+
const result = listLogFiles(TEST_DIR);
|
|
34
|
+
expect(result).toHaveLength(1);
|
|
35
|
+
expect(result[0].name).toBe("debug-2026-01-01.txt");
|
|
36
|
+
});
|
|
37
|
+
it("should not include non-.txt files", () => {
|
|
38
|
+
createLogFile("notes.md", ["not a log"]);
|
|
39
|
+
createLogFile("debug.txt", [logLine("INFO", "App", "started")]);
|
|
40
|
+
const result = listLogFiles(TEST_DIR);
|
|
41
|
+
expect(result.every(f => f.name.endsWith(".txt"))).toBe(true);
|
|
42
|
+
});
|
|
43
|
+
it("should sort log files by modified date, newest first", () => {
|
|
44
|
+
const f1 = createLogFile("old.txt", [logLine("INFO", "App", "first")]);
|
|
45
|
+
const f2 = createLogFile("new.txt", [logLine("INFO", "App", "second")]);
|
|
46
|
+
const old = new Date(Date.now() - 10000);
|
|
47
|
+
fs.utimesSync(f1, old, old);
|
|
48
|
+
const result = listLogFiles(TEST_DIR);
|
|
49
|
+
expect(result[0].name).toBe("new.txt");
|
|
50
|
+
});
|
|
51
|
+
it("should mark latest symlink target as isLatest", () => {
|
|
52
|
+
createLogFile("debug.txt", [logLine("INFO", "App", "msg")]);
|
|
53
|
+
const symlinkPath = path.join(DEBUG_DIR, "latest");
|
|
54
|
+
try {
|
|
55
|
+
fs.symlinkSync(path.join(DEBUG_DIR, "debug.txt"), symlinkPath);
|
|
56
|
+
const result = listLogFiles(TEST_DIR);
|
|
57
|
+
const latest = result.find(f => f.isLatest);
|
|
58
|
+
expect(latest).toBeDefined();
|
|
59
|
+
}
|
|
60
|
+
catch {
|
|
61
|
+
// Symlinks may not be supported in all environments, skip
|
|
62
|
+
}
|
|
63
|
+
});
|
|
64
|
+
});
|
|
65
|
+
describe("parseLogLine", () => {
|
|
66
|
+
it("should parse a valid log line", () => {
|
|
67
|
+
const line = logLine("INFO", "AppServer", "Server started on port 3000");
|
|
68
|
+
const entry = parseLogLine(line, "/tmp/debug.txt", 1);
|
|
69
|
+
expect(entry).not.toBeNull();
|
|
70
|
+
expect(entry.level).toBe("INFO");
|
|
71
|
+
expect(entry.component).toBe("AppServer");
|
|
72
|
+
expect(entry.message).toBe("Server started on port 3000");
|
|
73
|
+
});
|
|
74
|
+
it("should return null for invalid log lines", () => {
|
|
75
|
+
const entry = parseLogLine("not a valid log line", "/tmp/debug.txt", 1);
|
|
76
|
+
expect(entry).toBeNull();
|
|
77
|
+
});
|
|
78
|
+
it("should parse all log levels", () => {
|
|
79
|
+
for (const level of ["DEBUG", "INFO", "WARN", "ERROR"]) {
|
|
80
|
+
const entry = parseLogLine(logLine(level, "App", "test"), "/tmp/f.txt", 1);
|
|
81
|
+
expect(entry.level).toBe(level);
|
|
82
|
+
}
|
|
83
|
+
});
|
|
84
|
+
it("should parse timestamp into Date object", () => {
|
|
85
|
+
const entry = parseLogLine(logLine("INFO", "App", "test"), "/tmp/f.txt", 1);
|
|
86
|
+
expect(entry.timestamp).toBeInstanceOf(Date);
|
|
87
|
+
expect(entry.timestamp.getFullYear()).toBe(2026);
|
|
88
|
+
});
|
|
89
|
+
});
|
|
90
|
+
describe("parseLogFile", () => {
|
|
91
|
+
it("should return empty array for nonexistent file", () => {
|
|
92
|
+
const result = parseLogFile("/nonexistent/file.txt");
|
|
93
|
+
expect(result).toHaveLength(0);
|
|
94
|
+
});
|
|
95
|
+
it("should parse all valid log lines from file", () => {
|
|
96
|
+
const filePath = createLogFile("test.txt", [
|
|
97
|
+
logLine("INFO", "App", "message one"),
|
|
98
|
+
logLine("ERROR", "DB", "connection failed"),
|
|
99
|
+
"invalid line",
|
|
100
|
+
logLine("WARN", "Cache", "cache miss"),
|
|
101
|
+
]);
|
|
102
|
+
const result = parseLogFile(filePath);
|
|
103
|
+
expect(result).toHaveLength(3);
|
|
104
|
+
});
|
|
105
|
+
it("should filter by log level", () => {
|
|
106
|
+
const filePath = createLogFile("test.txt", [
|
|
107
|
+
logLine("INFO", "App", "info msg"),
|
|
108
|
+
logLine("ERROR", "App", "error msg"),
|
|
109
|
+
logLine("WARN", "App", "warn msg"),
|
|
110
|
+
]);
|
|
111
|
+
const result = parseLogFile(filePath, { level: "ERROR" });
|
|
112
|
+
expect(result.every(e => e.level === "ERROR")).toBe(true);
|
|
113
|
+
});
|
|
114
|
+
it("should filter by component", () => {
|
|
115
|
+
const filePath = createLogFile("test.txt", [
|
|
116
|
+
logLine("INFO", "Database", "db query"),
|
|
117
|
+
logLine("INFO", "AppServer", "request handled"),
|
|
118
|
+
]);
|
|
119
|
+
const result = parseLogFile(filePath, { component: "Database" });
|
|
120
|
+
expect(result.every(e => e.component === "Database")).toBe(true);
|
|
121
|
+
});
|
|
122
|
+
it("should filter by search term", () => {
|
|
123
|
+
const filePath = createLogFile("test.txt", [
|
|
124
|
+
logLine("INFO", "App", "user logged in"),
|
|
125
|
+
logLine("INFO", "App", "page loaded"),
|
|
126
|
+
]);
|
|
127
|
+
const result = parseLogFile(filePath, { search: "logged in" });
|
|
128
|
+
expect(result).toHaveLength(1);
|
|
129
|
+
expect(result[0].message).toContain("logged in");
|
|
130
|
+
});
|
|
131
|
+
it("should respect limit option", () => {
|
|
132
|
+
const filePath = createLogFile("test.txt", [
|
|
133
|
+
logLine("INFO", "App", "msg 1"),
|
|
134
|
+
logLine("INFO", "App", "msg 2"),
|
|
135
|
+
logLine("INFO", "App", "msg 3"),
|
|
136
|
+
logLine("INFO", "App", "msg 4"),
|
|
137
|
+
logLine("INFO", "App", "msg 5"),
|
|
138
|
+
]);
|
|
139
|
+
const result = parseLogFile(filePath, { limit: 3 });
|
|
140
|
+
expect(result).toHaveLength(3);
|
|
141
|
+
});
|
|
142
|
+
});
|
|
143
|
+
describe("getLogSummary", () => {
|
|
144
|
+
it("should return zero counts when no log files", () => {
|
|
145
|
+
const summary = getLogSummary(path.join(TEST_DIR, "nonexistent"));
|
|
146
|
+
expect(summary.totalFiles).toBe(0);
|
|
147
|
+
expect(summary.levelCounts.ERROR).toBe(0);
|
|
148
|
+
});
|
|
149
|
+
it("should count log entries by level", () => {
|
|
150
|
+
createLogFile("test.txt", [
|
|
151
|
+
logLine("INFO", "App", "a"),
|
|
152
|
+
logLine("ERROR", "App", "b"),
|
|
153
|
+
logLine("ERROR", "DB", "c"),
|
|
154
|
+
logLine("WARN", "App", "d"),
|
|
155
|
+
]);
|
|
156
|
+
const summary = getLogSummary(TEST_DIR);
|
|
157
|
+
expect(summary.levelCounts.INFO).toBe(1);
|
|
158
|
+
expect(summary.levelCounts.ERROR).toBe(2);
|
|
159
|
+
expect(summary.levelCounts.WARN).toBe(1);
|
|
160
|
+
});
|
|
161
|
+
it("should count entries by component", () => {
|
|
162
|
+
createLogFile("test.txt", [
|
|
163
|
+
logLine("INFO", "AppServer", "msg"),
|
|
164
|
+
logLine("INFO", "AppServer", "msg2"),
|
|
165
|
+
logLine("INFO", "Database", "msg3"),
|
|
166
|
+
]);
|
|
167
|
+
const summary = getLogSummary(TEST_DIR);
|
|
168
|
+
expect(summary.componentCounts["AppServer"]).toBe(2);
|
|
169
|
+
expect(summary.componentCounts["Database"]).toBe(1);
|
|
170
|
+
});
|
|
171
|
+
});
|
|
172
|
+
describe("searchLogs", () => {
|
|
173
|
+
it("should find matching log entries across files", () => {
|
|
174
|
+
createLogFile("test.txt", [
|
|
175
|
+
logLine("ERROR", "DB", "connection timeout occurred"),
|
|
176
|
+
logLine("INFO", "App", "startup complete"),
|
|
177
|
+
]);
|
|
178
|
+
const results = searchLogs("connection timeout", undefined, TEST_DIR);
|
|
179
|
+
expect(results.length).toBeGreaterThan(0);
|
|
180
|
+
expect(results[0].message).toContain("connection timeout");
|
|
181
|
+
});
|
|
182
|
+
});
|
|
183
|
+
describe("getErrorLogs", () => {
|
|
184
|
+
it("should return only ERROR and WARN entries", () => {
|
|
185
|
+
createLogFile("test.txt", [
|
|
186
|
+
logLine("INFO", "App", "info"),
|
|
187
|
+
logLine("ERROR", "App", "error"),
|
|
188
|
+
logLine("WARN", "App", "warn"),
|
|
189
|
+
logLine("DEBUG", "App", "debug"),
|
|
190
|
+
]);
|
|
191
|
+
const results = getErrorLogs(100, TEST_DIR);
|
|
192
|
+
expect(results.every(e => e.level === "ERROR" || e.level === "WARN")).toBe(true);
|
|
193
|
+
});
|
|
194
|
+
});
|
|
195
|
+
});
|
|
196
|
+
//# sourceMappingURL=logs.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"logs.test.js","sourceRoot":"","sources":["../../src/__tests__/logs.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"search.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/search.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,155 @@
|
|
|
1
|
+
import { describe, it, expect, beforeEach, afterEach } from "vitest";
|
|
2
|
+
import * as fs from "fs";
|
|
3
|
+
import * as path from "path";
|
|
4
|
+
import * as os from "os";
|
|
5
|
+
import { searchConversations, formatSearchReport, formatSessionDiff, } from "../lib/search.js";
|
|
6
|
+
const TEST_DIR = path.join(os.tmpdir(), "cct-search-test");
|
|
7
|
+
const PROJECTS_DIR = path.join(TEST_DIR, "projects");
|
|
8
|
+
function createSession(projectName, fileName, lines) {
|
|
9
|
+
const dir = path.join(PROJECTS_DIR, projectName);
|
|
10
|
+
fs.mkdirSync(dir, { recursive: true });
|
|
11
|
+
const filePath = path.join(dir, fileName);
|
|
12
|
+
fs.writeFileSync(filePath, lines.map(l => JSON.stringify(l)).join("\n"), "utf-8");
|
|
13
|
+
return filePath;
|
|
14
|
+
}
|
|
15
|
+
const userLine = (text) => ({
|
|
16
|
+
type: "user",
|
|
17
|
+
timestamp: "2026-01-01T10:00:00Z",
|
|
18
|
+
message: { role: "user", content: [{ type: "text", text }] },
|
|
19
|
+
});
|
|
20
|
+
const assistantLine = (text) => ({
|
|
21
|
+
type: "assistant",
|
|
22
|
+
timestamp: "2026-01-01T10:01:00Z",
|
|
23
|
+
message: { role: "assistant", content: [{ type: "text", text }] },
|
|
24
|
+
});
|
|
25
|
+
describe("Search Module", () => {
|
|
26
|
+
beforeEach(() => {
|
|
27
|
+
if (fs.existsSync(TEST_DIR))
|
|
28
|
+
fs.rmSync(TEST_DIR, { recursive: true });
|
|
29
|
+
fs.mkdirSync(PROJECTS_DIR, { recursive: true });
|
|
30
|
+
});
|
|
31
|
+
afterEach(() => {
|
|
32
|
+
if (fs.existsSync(TEST_DIR))
|
|
33
|
+
fs.rmSync(TEST_DIR, { recursive: true });
|
|
34
|
+
});
|
|
35
|
+
describe("searchConversations", () => {
|
|
36
|
+
it("should find matching text in user messages", () => {
|
|
37
|
+
createSession("proj1", "s1.jsonl", [userLine("Hello world from user")]);
|
|
38
|
+
const result = searchConversations({ query: "Hello world" }, TEST_DIR);
|
|
39
|
+
expect(result.totalMatches).toBeGreaterThan(0);
|
|
40
|
+
expect(result.results[0].preview).toContain("Hello world");
|
|
41
|
+
});
|
|
42
|
+
it("should find matching text in assistant messages", () => {
|
|
43
|
+
createSession("proj1", "s1.jsonl", [assistantLine("Here is the solution to your problem")]);
|
|
44
|
+
const result = searchConversations({ query: "solution" }, TEST_DIR);
|
|
45
|
+
expect(result.totalMatches).toBeGreaterThan(0);
|
|
46
|
+
expect(result.results[0].role).toBe("assistant");
|
|
47
|
+
});
|
|
48
|
+
it("should return empty results when no matches found", () => {
|
|
49
|
+
createSession("proj1", "s1.jsonl", [userLine("completely unrelated content")]);
|
|
50
|
+
const result = searchConversations({ query: "xyz_no_match_here" }, TEST_DIR);
|
|
51
|
+
expect(result.totalMatches).toBe(0);
|
|
52
|
+
expect(result.results).toHaveLength(0);
|
|
53
|
+
});
|
|
54
|
+
it("should respect case insensitive search by default", () => {
|
|
55
|
+
createSession("proj1", "s1.jsonl", [userLine("TypeScript is awesome")]);
|
|
56
|
+
const result = searchConversations({ query: "typescript" }, TEST_DIR);
|
|
57
|
+
expect(result.totalMatches).toBeGreaterThan(0);
|
|
58
|
+
});
|
|
59
|
+
it("should respect case sensitive search option", () => {
|
|
60
|
+
createSession("proj1", "s1.jsonl", [userLine("TypeScript is awesome")]);
|
|
61
|
+
const resultSensitive = searchConversations({ query: "typescript", caseSensitive: true }, TEST_DIR);
|
|
62
|
+
const resultInsensitive = searchConversations({ query: "typescript", caseSensitive: false }, TEST_DIR);
|
|
63
|
+
expect(resultSensitive.totalMatches).toBe(0);
|
|
64
|
+
expect(resultInsensitive.totalMatches).toBeGreaterThan(0);
|
|
65
|
+
});
|
|
66
|
+
it("should filter by role when specified", () => {
|
|
67
|
+
createSession("proj1", "s1.jsonl", [
|
|
68
|
+
userLine("user asks about testing"),
|
|
69
|
+
assistantLine("assistant discusses testing"),
|
|
70
|
+
]);
|
|
71
|
+
const result = searchConversations({ query: "testing", role: "user" }, TEST_DIR);
|
|
72
|
+
expect(result.results.every(r => r.role === "user")).toBe(true);
|
|
73
|
+
});
|
|
74
|
+
it("should respect result limit", () => {
|
|
75
|
+
createSession("proj1", "s1.jsonl", [
|
|
76
|
+
userLine("keyword one"),
|
|
77
|
+
userLine("keyword two"),
|
|
78
|
+
userLine("keyword three"),
|
|
79
|
+
]);
|
|
80
|
+
const result = searchConversations({ query: "keyword", limit: 2 }, TEST_DIR);
|
|
81
|
+
expect(result.results.length).toBeLessThanOrEqual(2);
|
|
82
|
+
});
|
|
83
|
+
it("should track filesSearched count", () => {
|
|
84
|
+
createSession("proj1", "s1.jsonl", [userLine("content a")]);
|
|
85
|
+
createSession("proj2", "s2.jsonl", [userLine("content b")]);
|
|
86
|
+
const result = searchConversations({ query: "content" }, TEST_DIR);
|
|
87
|
+
expect(result.filesSearched).toBe(2);
|
|
88
|
+
});
|
|
89
|
+
it("should mark truncated when more results exist than limit", () => {
|
|
90
|
+
for (let i = 0; i < 5; i++) {
|
|
91
|
+
createSession(`proj${i}`, "s.jsonl", [userLine("keyword in every session")]);
|
|
92
|
+
}
|
|
93
|
+
const result = searchConversations({ query: "keyword", limit: 3 }, TEST_DIR);
|
|
94
|
+
if (result.totalMatches > 3) {
|
|
95
|
+
expect(result.truncated).toBe(true);
|
|
96
|
+
}
|
|
97
|
+
});
|
|
98
|
+
});
|
|
99
|
+
describe("formatSearchReport", () => {
|
|
100
|
+
it("should include query in output", () => {
|
|
101
|
+
const report = {
|
|
102
|
+
query: "my search term",
|
|
103
|
+
results: [],
|
|
104
|
+
totalMatches: 0,
|
|
105
|
+
filesSearched: 5,
|
|
106
|
+
truncated: false,
|
|
107
|
+
};
|
|
108
|
+
const output = formatSearchReport(report);
|
|
109
|
+
expect(output).toContain("my search term");
|
|
110
|
+
expect(output).toContain("No matches found");
|
|
111
|
+
});
|
|
112
|
+
it("should show results grouped by project", () => {
|
|
113
|
+
const report = {
|
|
114
|
+
query: "test",
|
|
115
|
+
results: [
|
|
116
|
+
{
|
|
117
|
+
file: "proj1/s1.jsonl",
|
|
118
|
+
sessionId: "session-abc",
|
|
119
|
+
project: "myproject",
|
|
120
|
+
line: 1,
|
|
121
|
+
role: "user",
|
|
122
|
+
preview: "test content",
|
|
123
|
+
},
|
|
124
|
+
],
|
|
125
|
+
totalMatches: 1,
|
|
126
|
+
filesSearched: 1,
|
|
127
|
+
truncated: false,
|
|
128
|
+
};
|
|
129
|
+
const output = formatSearchReport(report);
|
|
130
|
+
expect(output).toContain("myproject");
|
|
131
|
+
expect(output).toContain("session-");
|
|
132
|
+
});
|
|
133
|
+
});
|
|
134
|
+
describe("formatSessionDiff", () => {
|
|
135
|
+
it("should format a session diff report", () => {
|
|
136
|
+
const diff = {
|
|
137
|
+
session1: { id: "session-aaa", file: "/tmp/s1.jsonl", messageCount: 10, modified: new Date("2026-01-01") },
|
|
138
|
+
session2: { id: "session-bbb", file: "/tmp/s2.jsonl", messageCount: 15, modified: new Date("2026-01-02") },
|
|
139
|
+
commonTopics: ["typescript", "testing"],
|
|
140
|
+
uniqueToSession1: ["authentication"],
|
|
141
|
+
uniqueToSession2: ["deployment"],
|
|
142
|
+
messagesDelta: 5,
|
|
143
|
+
sizeDelta: 1024,
|
|
144
|
+
toolsUsedBoth: ["Bash", "Read"],
|
|
145
|
+
toolsOnlyIn1: ["Write"],
|
|
146
|
+
toolsOnlyIn2: [],
|
|
147
|
+
};
|
|
148
|
+
const output = formatSessionDiff(diff);
|
|
149
|
+
expect(output).toContain("Session Comparison");
|
|
150
|
+
expect(output).toContain("Bash");
|
|
151
|
+
expect(output).toContain("+5");
|
|
152
|
+
});
|
|
153
|
+
});
|
|
154
|
+
});
|
|
155
|
+
//# sourceMappingURL=search.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"search.test.js","sourceRoot":"","sources":["../../src/__tests__/search.test.ts"],"names":[],"mappings":"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"}
|