sqlew 3.2.4 → 3.5.3
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/CHANGELOG.md +843 -0
- package/README.md +53 -2
- package/assets/schema.sql +6 -1
- package/dist/config/loader.d.ts +46 -0
- package/dist/config/loader.d.ts.map +1 -0
- package/dist/config/loader.js +151 -0
- package/dist/config/loader.js.map +1 -0
- package/dist/config/types.d.ts +77 -0
- package/dist/config/types.d.ts.map +1 -0
- package/dist/config/types.js +28 -0
- package/dist/config/types.js.map +1 -0
- package/dist/constants.d.ts +9 -0
- package/dist/constants.d.ts.map +1 -1
- package/dist/constants.js +10 -0
- package/dist/constants.js.map +1 -1
- package/dist/database.d.ts +1 -1
- package/dist/database.d.ts.map +1 -1
- package/dist/database.js +84 -15
- package/dist/database.js.map +1 -1
- package/dist/index.js +21 -3
- package/dist/index.js.map +1 -1
- package/dist/migrations/add-v3.5.0-pruned-files.d.ts +26 -0
- package/dist/migrations/add-v3.5.0-pruned-files.d.ts.map +1 -0
- package/dist/migrations/add-v3.5.0-pruned-files.js +107 -0
- package/dist/migrations/add-v3.5.0-pruned-files.js.map +1 -0
- package/dist/migrations/index.d.ts +2 -1
- package/dist/migrations/index.d.ts.map +1 -1
- package/dist/migrations/index.js +16 -1
- package/dist/migrations/index.js.map +1 -1
- package/dist/tests/git-aware-completion.test.d.ts +6 -0
- package/dist/tests/git-aware-completion.test.d.ts.map +1 -0
- package/dist/tests/git-aware-completion.test.js +141 -0
- package/dist/tests/git-aware-completion.test.js.map +1 -0
- package/dist/tests/tasks.auto-pruning-decision-link.test.d.ts +6 -0
- package/dist/tests/tasks.auto-pruning-decision-link.test.d.ts.map +1 -0
- package/dist/tests/tasks.auto-pruning-decision-link.test.js +250 -0
- package/dist/tests/tasks.auto-pruning-decision-link.test.js.map +1 -0
- package/dist/tests/tasks.auto-pruning-partial.test.d.ts +6 -0
- package/dist/tests/tasks.auto-pruning-partial.test.d.ts.map +1 -0
- package/dist/tests/tasks.auto-pruning-partial.test.js +274 -0
- package/dist/tests/tasks.auto-pruning-partial.test.js.map +1 -0
- package/dist/tests/tasks.auto-pruning-persistence.test.d.ts +6 -0
- package/dist/tests/tasks.auto-pruning-persistence.test.d.ts.map +1 -0
- package/dist/tests/tasks.auto-pruning-persistence.test.js +232 -0
- package/dist/tests/tasks.auto-pruning-persistence.test.js.map +1 -0
- package/dist/tests/tasks.auto-pruning-safety.test.d.ts +12 -0
- package/dist/tests/tasks.auto-pruning-safety.test.d.ts.map +1 -0
- package/dist/tests/tasks.auto-pruning-safety.test.js +196 -0
- package/dist/tests/tasks.auto-pruning-safety.test.js.map +1 -0
- package/dist/tests/tasks.link-file-backward-compat.test.d.ts +6 -0
- package/dist/tests/tasks.link-file-backward-compat.test.d.ts.map +1 -0
- package/dist/tests/tasks.link-file-backward-compat.test.js +235 -0
- package/dist/tests/tasks.link-file-backward-compat.test.js.map +1 -0
- package/dist/tests/tasks.watch-files-action.test.d.ts +6 -0
- package/dist/tests/tasks.watch-files-action.test.d.ts.map +1 -0
- package/dist/tests/tasks.watch-files-action.test.js +351 -0
- package/dist/tests/tasks.watch-files-action.test.js.map +1 -0
- package/dist/tests/tasks.watch-files-parameter.test.d.ts +6 -0
- package/dist/tests/tasks.watch-files-parameter.test.d.ts.map +1 -0
- package/dist/tests/tasks.watch-files-parameter.test.js +249 -0
- package/dist/tests/tasks.watch-files-parameter.test.js.map +1 -0
- package/dist/tests/two-step-git-completion.test.d.ts +6 -0
- package/dist/tests/two-step-git-completion.test.d.ts.map +1 -0
- package/dist/tests/two-step-git-completion.test.js +283 -0
- package/dist/tests/two-step-git-completion.test.js.map +1 -0
- package/dist/tests/vcs-staging.test.d.ts +6 -0
- package/dist/tests/vcs-staging.test.d.ts.map +1 -0
- package/dist/tests/vcs-staging.test.js +137 -0
- package/dist/tests/vcs-staging.test.js.map +1 -0
- package/dist/tools/config.d.ts +4 -2
- package/dist/tools/config.d.ts.map +1 -1
- package/dist/tools/config.js +13 -11
- package/dist/tools/config.js.map +1 -1
- package/dist/tools/constraints.d.ts +7 -4
- package/dist/tools/constraints.d.ts.map +1 -1
- package/dist/tools/constraints.js +19 -16
- package/dist/tools/constraints.js.map +1 -1
- package/dist/tools/context.d.ts +33 -17
- package/dist/tools/context.d.ts.map +1 -1
- package/dist/tools/context.js +84 -68
- package/dist/tools/context.js.map +1 -1
- package/dist/tools/files.d.ts +9 -5
- package/dist/tools/files.d.ts.map +1 -1
- package/dist/tools/files.js +19 -15
- package/dist/tools/files.js.map +1 -1
- package/dist/tools/messaging.d.ts +9 -5
- package/dist/tools/messaging.d.ts.map +1 -1
- package/dist/tools/messaging.js +20 -16
- package/dist/tools/messaging.js.map +1 -1
- package/dist/tools/tasks.d.ts +40 -12
- package/dist/tools/tasks.d.ts.map +1 -1
- package/dist/tools/tasks.js +475 -87
- package/dist/tools/tasks.js.map +1 -1
- package/dist/tools/utils.d.ts +11 -6
- package/dist/tools/utils.d.ts.map +1 -1
- package/dist/tools/utils.js +56 -44
- package/dist/tools/utils.js.map +1 -1
- package/dist/types.d.ts +4 -0
- package/dist/types.d.ts.map +1 -1
- package/dist/utils/file-pruning.d.ts +69 -0
- package/dist/utils/file-pruning.d.ts.map +1 -0
- package/dist/utils/file-pruning.js +185 -0
- package/dist/utils/file-pruning.js.map +1 -0
- package/dist/utils/quality-checks.d.ts +60 -0
- package/dist/utils/quality-checks.d.ts.map +1 -0
- package/dist/utils/quality-checks.js +228 -0
- package/dist/utils/quality-checks.js.map +1 -0
- package/dist/utils/retention.d.ts +8 -0
- package/dist/utils/retention.d.ts.map +1 -1
- package/dist/utils/retention.js +12 -0
- package/dist/utils/retention.js.map +1 -1
- package/dist/utils/task-stale-detection.d.ts +69 -1
- package/dist/utils/task-stale-detection.d.ts.map +1 -1
- package/dist/utils/task-stale-detection.js +494 -17
- package/dist/utils/task-stale-detection.js.map +1 -1
- package/dist/utils/validators.d.ts +2 -2
- package/dist/utils/validators.d.ts.map +1 -1
- package/dist/utils/validators.js +11 -2
- package/dist/utils/validators.js.map +1 -1
- package/dist/utils/vcs-adapter.d.ts +68 -0
- package/dist/utils/vcs-adapter.d.ts.map +1 -0
- package/dist/utils/vcs-adapter.js +187 -0
- package/dist/utils/vcs-adapter.js.map +1 -0
- package/dist/watcher/file-watcher.d.ts +54 -4
- package/dist/watcher/file-watcher.d.ts.map +1 -1
- package/dist/watcher/file-watcher.js +312 -30
- package/dist/watcher/file-watcher.js.map +1 -1
- package/dist/watcher/gitignore-parser.d.ts +70 -0
- package/dist/watcher/gitignore-parser.d.ts.map +1 -0
- package/dist/watcher/gitignore-parser.js +191 -0
- package/dist/watcher/gitignore-parser.js.map +1 -0
- package/dist/watcher/index.d.ts +1 -0
- package/dist/watcher/index.d.ts.map +1 -1
- package/dist/watcher/index.js +1 -0
- package/dist/watcher/index.js.map +1 -1
- package/docs/AI_AGENT_GUIDE.md +1 -1
- package/docs/ARCHITECTURE.md +12 -0
- package/docs/AUTO_FILE_TRACKING.md +486 -82
- package/docs/CONFIGURATION.md +908 -0
- package/docs/GIT_AWARE_AUTO_COMPLETE.md +645 -0
- package/docs/MIGRATION_v3.3.md +602 -0
- package/docs/SHARED_CONCEPTS.md +2 -1
- package/docs/TASK_ACTIONS.md +12 -0
- package/docs/TASK_OVERVIEW.md +124 -23
- package/docs/TASK_PRUNING.md +589 -0
- package/docs/TASK_SYSTEM.md +83 -13
- package/docs/TOOL_REFERENCE.md +94 -6
- package/package.json +8 -6
|
@@ -0,0 +1,274 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Integration test for v3.5.0 Auto-Pruning feature
|
|
3
|
+
* Tests partial pruning scenario: keeps existing files, removes non-existent ones
|
|
4
|
+
*/
|
|
5
|
+
import { describe, it, beforeEach, afterEach } from 'node:test';
|
|
6
|
+
import assert from 'node:assert/strict';
|
|
7
|
+
import Database from 'better-sqlite3';
|
|
8
|
+
import { initializeSchema } from '../schema.js';
|
|
9
|
+
import { runAllMigrations } from '../migrations/index.js';
|
|
10
|
+
import { detectAndTransitionToReview } from '../utils/task-stale-detection.js';
|
|
11
|
+
import { getOrCreateAgent, getOrCreateFile } from '../database.js';
|
|
12
|
+
import fs from 'fs';
|
|
13
|
+
import path from 'path';
|
|
14
|
+
import os from 'os';
|
|
15
|
+
describe('Auto-pruning: Partial file existence', () => {
|
|
16
|
+
let db;
|
|
17
|
+
let tempDir;
|
|
18
|
+
beforeEach(() => {
|
|
19
|
+
db = new Database(':memory:');
|
|
20
|
+
db.pragma('foreign_keys = ON');
|
|
21
|
+
initializeSchema(db);
|
|
22
|
+
runAllMigrations(db);
|
|
23
|
+
// Skip the explicit migration result check
|
|
24
|
+
if (false) {
|
|
25
|
+
throw new Error(`Migration failed`);
|
|
26
|
+
}
|
|
27
|
+
// Create temp directory for test files
|
|
28
|
+
tempDir = fs.mkdtempSync(path.join(os.tmpdir(), 'sqlew-test-'));
|
|
29
|
+
});
|
|
30
|
+
afterEach(() => {
|
|
31
|
+
db.close();
|
|
32
|
+
// Cleanup temp directory
|
|
33
|
+
if (fs.existsSync(tempDir)) {
|
|
34
|
+
fs.rmSync(tempDir, { recursive: true, force: true });
|
|
35
|
+
}
|
|
36
|
+
});
|
|
37
|
+
it('should prune non-existent files and keep existing ones', async () => {
|
|
38
|
+
// 1. Create test task
|
|
39
|
+
const taskId = createTestTask(db);
|
|
40
|
+
// 2. Create 4 watched files (2 exist, 2 don't)
|
|
41
|
+
const existingFile1 = path.join(tempDir, 'exists1.ts');
|
|
42
|
+
const existingFile2 = path.join(tempDir, 'exists2.ts');
|
|
43
|
+
const nonExistentFile1 = path.join(tempDir, 'missing1.ts');
|
|
44
|
+
const nonExistentFile2 = path.join(tempDir, 'missing2.ts');
|
|
45
|
+
fs.writeFileSync(existingFile1, '// test file 1');
|
|
46
|
+
fs.writeFileSync(existingFile2, '// test file 2');
|
|
47
|
+
// Don't create nonExistentFile1 and nonExistentFile2
|
|
48
|
+
// Add all 4 files to watch list
|
|
49
|
+
addWatchedFiles(db, taskId, [
|
|
50
|
+
existingFile1,
|
|
51
|
+
existingFile2,
|
|
52
|
+
nonExistentFile1,
|
|
53
|
+
nonExistentFile2,
|
|
54
|
+
]);
|
|
55
|
+
// Set task to in_progress status and update timestamp to be old enough
|
|
56
|
+
// for auto-transition (older than 15 minutes, which is the default review_idle_minutes)
|
|
57
|
+
const fifteenMinutesAgo = Math.floor(Date.now() / 1000) - (15 * 60 + 10);
|
|
58
|
+
db.prepare(`
|
|
59
|
+
UPDATE t_tasks
|
|
60
|
+
SET status_id = 2, updated_ts = ?
|
|
61
|
+
WHERE id = ?
|
|
62
|
+
`).run(fifteenMinutesAgo, taskId);
|
|
63
|
+
// 3. Trigger auto-pruning by detecting ready for review
|
|
64
|
+
// Change cwd to tempDir so file existence checks work correctly
|
|
65
|
+
const originalCwd = process.cwd();
|
|
66
|
+
try {
|
|
67
|
+
process.chdir(tempDir);
|
|
68
|
+
const transitioned = await detectAndTransitionToReview(db);
|
|
69
|
+
// Task should have transitioned
|
|
70
|
+
assert.equal(transitioned, 1, 'Should have transitioned 1 task');
|
|
71
|
+
// 4. Verify results
|
|
72
|
+
// Check watch list - should have only 2 files (the existing ones)
|
|
73
|
+
const watchedFiles = getWatchedFiles(db, taskId);
|
|
74
|
+
assert.equal(watchedFiles.length, 2, 'Should have 2 files in watch list');
|
|
75
|
+
assert.ok(watchedFiles.includes('exists1.ts'), 'Should keep existing file 1');
|
|
76
|
+
assert.ok(watchedFiles.includes('exists2.ts'), 'Should keep existing file 2');
|
|
77
|
+
// Check pruned files table - should have 2 records
|
|
78
|
+
const prunedFiles = db
|
|
79
|
+
.prepare(`
|
|
80
|
+
SELECT file_path FROM t_task_pruned_files WHERE task_id = ?
|
|
81
|
+
`)
|
|
82
|
+
.all(taskId);
|
|
83
|
+
assert.equal(prunedFiles.length, 2, 'Should have 2 pruned file records');
|
|
84
|
+
// Verify pruned files are the correct ones
|
|
85
|
+
const prunedPaths = prunedFiles.map((f) => f.file_path);
|
|
86
|
+
assert.ok(prunedPaths.includes('missing1.ts'), 'Should have pruned missing1.ts');
|
|
87
|
+
assert.ok(prunedPaths.includes('missing2.ts'), 'Should have pruned missing2.ts');
|
|
88
|
+
// Check task status transitioned to waiting_review
|
|
89
|
+
const task = db
|
|
90
|
+
.prepare('SELECT status_id FROM t_tasks WHERE id = ?')
|
|
91
|
+
.get(taskId);
|
|
92
|
+
assert.equal(task.status_id, 3, 'Task should be in waiting_review status (3)');
|
|
93
|
+
}
|
|
94
|
+
finally {
|
|
95
|
+
// Restore original cwd
|
|
96
|
+
process.chdir(originalCwd);
|
|
97
|
+
}
|
|
98
|
+
});
|
|
99
|
+
it('should not transition if all files are non-existent', async () => {
|
|
100
|
+
// 1. Create test task
|
|
101
|
+
const taskId = createTestTask(db);
|
|
102
|
+
// 2. Create 2 watched files that don't exist
|
|
103
|
+
const nonExistentFile1 = path.join(tempDir, 'missing1.ts');
|
|
104
|
+
const nonExistentFile2 = path.join(tempDir, 'missing2.ts');
|
|
105
|
+
// Don't create these files
|
|
106
|
+
// Add files to watch list
|
|
107
|
+
addWatchedFiles(db, taskId, [nonExistentFile1, nonExistentFile2]);
|
|
108
|
+
// Set task to in_progress status and update timestamp to be old enough
|
|
109
|
+
const fifteenMinutesAgo = Math.floor(Date.now() / 1000) - (15 * 60 + 10);
|
|
110
|
+
db.prepare(`
|
|
111
|
+
UPDATE t_tasks
|
|
112
|
+
SET status_id = 2, updated_ts = ?
|
|
113
|
+
WHERE id = ?
|
|
114
|
+
`).run(fifteenMinutesAgo, taskId);
|
|
115
|
+
// 3. Trigger auto-pruning
|
|
116
|
+
const originalCwd = process.cwd();
|
|
117
|
+
try {
|
|
118
|
+
process.chdir(tempDir);
|
|
119
|
+
const transitioned = await detectAndTransitionToReview(db);
|
|
120
|
+
// Task should NOT have transitioned (safety check prevents it)
|
|
121
|
+
assert.equal(transitioned, 0, 'Should not have transitioned any tasks');
|
|
122
|
+
// 4. Verify task is still in in_progress
|
|
123
|
+
const task = db
|
|
124
|
+
.prepare('SELECT status_id FROM t_tasks WHERE id = ?')
|
|
125
|
+
.get(taskId);
|
|
126
|
+
assert.equal(task.status_id, 2, 'Task should still be in in_progress (2)');
|
|
127
|
+
// 5. Verify no pruning occurred (safety check blocked it)
|
|
128
|
+
const prunedFiles = db
|
|
129
|
+
.prepare('SELECT COUNT(*) as count FROM t_task_pruned_files WHERE task_id = ?')
|
|
130
|
+
.get(taskId);
|
|
131
|
+
assert.equal(prunedFiles.count, 0, 'Should have 0 pruned records (safety check prevented pruning)');
|
|
132
|
+
}
|
|
133
|
+
finally {
|
|
134
|
+
process.chdir(originalCwd);
|
|
135
|
+
}
|
|
136
|
+
});
|
|
137
|
+
it('should not prune when all files exist', async () => {
|
|
138
|
+
// 1. Create test task
|
|
139
|
+
const taskId = createTestTask(db);
|
|
140
|
+
// 2. Create 3 watched files that all exist
|
|
141
|
+
const existingFile1 = path.join(tempDir, 'exists1.ts');
|
|
142
|
+
const existingFile2 = path.join(tempDir, 'exists2.ts');
|
|
143
|
+
const existingFile3 = path.join(tempDir, 'exists3.ts');
|
|
144
|
+
fs.writeFileSync(existingFile1, '// test file 1');
|
|
145
|
+
fs.writeFileSync(existingFile2, '// test file 2');
|
|
146
|
+
fs.writeFileSync(existingFile3, '// test file 3');
|
|
147
|
+
// Add files to watch list
|
|
148
|
+
addWatchedFiles(db, taskId, [
|
|
149
|
+
existingFile1,
|
|
150
|
+
existingFile2,
|
|
151
|
+
existingFile3,
|
|
152
|
+
]);
|
|
153
|
+
// Set task to in_progress status and update timestamp to be old enough
|
|
154
|
+
const fifteenMinutesAgo = Math.floor(Date.now() / 1000) - (15 * 60 + 10);
|
|
155
|
+
db.prepare(`
|
|
156
|
+
UPDATE t_tasks
|
|
157
|
+
SET status_id = 2, updated_ts = ?
|
|
158
|
+
WHERE id = ?
|
|
159
|
+
`).run(fifteenMinutesAgo, taskId);
|
|
160
|
+
// 3. Trigger auto-pruning
|
|
161
|
+
const originalCwd = process.cwd();
|
|
162
|
+
try {
|
|
163
|
+
process.chdir(tempDir);
|
|
164
|
+
const transitioned = await detectAndTransitionToReview(db);
|
|
165
|
+
// Task should have transitioned (all files exist)
|
|
166
|
+
assert.equal(transitioned, 1, 'Should have transitioned 1 task');
|
|
167
|
+
// 4. Verify no pruning occurred
|
|
168
|
+
const prunedFiles = db
|
|
169
|
+
.prepare('SELECT COUNT(*) as count FROM t_task_pruned_files WHERE task_id = ?')
|
|
170
|
+
.get(taskId);
|
|
171
|
+
assert.equal(prunedFiles.count, 0, 'Should have 0 pruned records');
|
|
172
|
+
// 5. Verify all files still in watch list
|
|
173
|
+
const watchedFiles = getWatchedFiles(db, taskId);
|
|
174
|
+
assert.equal(watchedFiles.length, 3, 'Should still have 3 files in watch list');
|
|
175
|
+
// 6. Verify task transitioned to waiting_review
|
|
176
|
+
const task = db
|
|
177
|
+
.prepare('SELECT status_id FROM t_tasks WHERE id = ?')
|
|
178
|
+
.get(taskId);
|
|
179
|
+
assert.equal(task.status_id, 3, 'Task should be in waiting_review status (3)');
|
|
180
|
+
}
|
|
181
|
+
finally {
|
|
182
|
+
process.chdir(originalCwd);
|
|
183
|
+
}
|
|
184
|
+
});
|
|
185
|
+
it('should handle mixed ratios (3 exist, 1 missing)', async () => {
|
|
186
|
+
// 1. Create test task
|
|
187
|
+
const taskId = createTestTask(db);
|
|
188
|
+
// 2. Create 4 watched files (3 exist, 1 doesn't)
|
|
189
|
+
const existingFile1 = path.join(tempDir, 'exists1.ts');
|
|
190
|
+
const existingFile2 = path.join(tempDir, 'exists2.ts');
|
|
191
|
+
const existingFile3 = path.join(tempDir, 'exists3.ts');
|
|
192
|
+
const nonExistentFile = path.join(tempDir, 'missing.ts');
|
|
193
|
+
fs.writeFileSync(existingFile1, '// test file 1');
|
|
194
|
+
fs.writeFileSync(existingFile2, '// test file 2');
|
|
195
|
+
fs.writeFileSync(existingFile3, '// test file 3');
|
|
196
|
+
// Don't create nonExistentFile
|
|
197
|
+
// Add all files to watch list
|
|
198
|
+
addWatchedFiles(db, taskId, [
|
|
199
|
+
existingFile1,
|
|
200
|
+
existingFile2,
|
|
201
|
+
existingFile3,
|
|
202
|
+
nonExistentFile,
|
|
203
|
+
]);
|
|
204
|
+
// Set task to in_progress status and update timestamp to be old enough
|
|
205
|
+
const fifteenMinutesAgo = Math.floor(Date.now() / 1000) - (15 * 60 + 10);
|
|
206
|
+
db.prepare(`
|
|
207
|
+
UPDATE t_tasks
|
|
208
|
+
SET status_id = 2, updated_ts = ?
|
|
209
|
+
WHERE id = ?
|
|
210
|
+
`).run(fifteenMinutesAgo, taskId);
|
|
211
|
+
// 3. Trigger auto-pruning
|
|
212
|
+
const originalCwd = process.cwd();
|
|
213
|
+
try {
|
|
214
|
+
process.chdir(tempDir);
|
|
215
|
+
const transitioned = await detectAndTransitionToReview(db);
|
|
216
|
+
// Task should have transitioned
|
|
217
|
+
assert.equal(transitioned, 1, 'Should have transitioned 1 task');
|
|
218
|
+
// 4. Verify results
|
|
219
|
+
const watchedFiles = getWatchedFiles(db, taskId);
|
|
220
|
+
assert.equal(watchedFiles.length, 3, 'Should have 3 files in watch list');
|
|
221
|
+
const prunedFiles = db
|
|
222
|
+
.prepare('SELECT file_path FROM t_task_pruned_files WHERE task_id = ?')
|
|
223
|
+
.all(taskId);
|
|
224
|
+
assert.equal(prunedFiles.length, 1, 'Should have 1 pruned file record');
|
|
225
|
+
assert.equal(prunedFiles[0].file_path, 'missing.ts', 'Should have pruned missing.ts');
|
|
226
|
+
}
|
|
227
|
+
finally {
|
|
228
|
+
process.chdir(originalCwd);
|
|
229
|
+
}
|
|
230
|
+
});
|
|
231
|
+
});
|
|
232
|
+
// Helper functions
|
|
233
|
+
/**
|
|
234
|
+
* Create a test task in in_progress status
|
|
235
|
+
*/
|
|
236
|
+
function createTestTask(db) {
|
|
237
|
+
const agentId = getOrCreateAgent(db, 'test-agent');
|
|
238
|
+
const result = db
|
|
239
|
+
.prepare(`
|
|
240
|
+
INSERT INTO t_tasks (title, status_id, priority, created_by_agent_id, assigned_agent_id)
|
|
241
|
+
VALUES (?, 2, 2, ?, ?)
|
|
242
|
+
`)
|
|
243
|
+
.run('Test Task for Auto-Pruning', agentId, agentId);
|
|
244
|
+
return result.lastInsertRowid;
|
|
245
|
+
}
|
|
246
|
+
/**
|
|
247
|
+
* Add watched files to a task
|
|
248
|
+
*/
|
|
249
|
+
function addWatchedFiles(db, taskId, filePaths) {
|
|
250
|
+
const insertFileLinkStmt = db.prepare(`
|
|
251
|
+
INSERT OR IGNORE INTO t_task_file_links (task_id, file_id)
|
|
252
|
+
VALUES (?, ?)
|
|
253
|
+
`);
|
|
254
|
+
for (const filePath of filePaths) {
|
|
255
|
+
// Extract just the filename (last segment) for relative paths
|
|
256
|
+
const fileName = path.basename(filePath);
|
|
257
|
+
const fileId = getOrCreateFile(db, fileName);
|
|
258
|
+
insertFileLinkStmt.run(taskId, fileId);
|
|
259
|
+
}
|
|
260
|
+
}
|
|
261
|
+
/**
|
|
262
|
+
* Get watched files for a task
|
|
263
|
+
*/
|
|
264
|
+
function getWatchedFiles(db, taskId) {
|
|
265
|
+
return db
|
|
266
|
+
.prepare(`
|
|
267
|
+
SELECT mf.path
|
|
268
|
+
FROM t_task_file_links tfl
|
|
269
|
+
JOIN m_files mf ON tfl.file_id = mf.id
|
|
270
|
+
WHERE tfl.task_id = ?
|
|
271
|
+
`)
|
|
272
|
+
.all(taskId).map((row) => row.path);
|
|
273
|
+
}
|
|
274
|
+
//# sourceMappingURL=tasks.auto-pruning-partial.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"tasks.auto-pruning-partial.test.js","sourceRoot":"","sources":["../../src/tests/tasks.auto-pruning-partial.test.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,UAAU,EAAE,SAAS,EAAE,MAAM,WAAW,CAAC;AAChE,OAAO,MAAM,MAAM,oBAAoB,CAAC;AACxC,OAAO,QAAQ,MAAM,gBAAgB,CAAC;AACtC,OAAO,EAAE,gBAAgB,EAAE,MAAM,cAAc,CAAC;AAChD,OAAO,EAAE,gBAAgB,EAAE,MAAM,wBAAwB,CAAC;AAC1D,OAAO,EAAE,2BAA2B,EAAE,MAAM,kCAAkC,CAAC;AAC/E,OAAO,EAAE,gBAAgB,EAAE,eAAe,EAAE,MAAM,gBAAgB,CAAC;AAEnE,OAAO,EAAE,MAAM,IAAI,CAAC;AACpB,OAAO,IAAI,MAAM,MAAM,CAAC;AACxB,OAAO,EAAE,MAAM,IAAI,CAAC;AAEpB,QAAQ,CAAC,sCAAsC,EAAE,GAAG,EAAE;IACpD,IAAI,EAAgB,CAAC;IACrB,IAAI,OAAe,CAAC;IAEpB,UAAU,CAAC,GAAG,EAAE;QACd,EAAE,GAAG,IAAI,QAAQ,CAAC,UAAU,CAAC,CAAC;QAC9B,EAAE,CAAC,MAAM,CAAC,mBAAmB,CAAC,CAAC;QAC/B,gBAAgB,CAAC,EAAE,CAAC,CAAC;QACrB,gBAAgB,CAAC,EAAE,CAAC,CAAC;QAErB,2CAA2C;QAC3C,IAAI,KAAK,EAAE,CAAC;YACV,MAAM,IAAI,KAAK,CAAC,kBAAkB,CAAC,CAAC;QACtC,CAAC;QAED,uCAAuC;QACvC,OAAO,GAAG,EAAE,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,MAAM,EAAE,EAAE,aAAa,CAAC,CAAC,CAAC;IAClE,CAAC,CAAC,CAAC;IAEH,SAAS,CAAC,GAAG,EAAE;QACb,EAAE,CAAC,KAAK,EAAE,CAAC;QACX,yBAAyB;QACzB,IAAI,EAAE,CAAC,UAAU,CAAC,OAAO,CAAC,EAAE,CAAC;YAC3B,EAAE,CAAC,MAAM,CAAC,OAAO,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;QACvD,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,wDAAwD,EAAE,KAAK,IAAI,EAAE;QACtE,sBAAsB;QACtB,MAAM,MAAM,GAAG,cAAc,CAAC,EAAE,CAAC,CAAC;QAElC,+CAA+C;QAC/C,MAAM,aAAa,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,YAAY,CAAC,CAAC;QACvD,MAAM,aAAa,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,YAAY,CAAC,CAAC;QACvD,MAAM,gBAAgB,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,aAAa,CAAC,CAAC;QAC3D,MAAM,gBAAgB,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,aAAa,CAAC,CAAC;QAE3D,EAAE,CAAC,aAAa,CAAC,aAAa,EAAE,gBAAgB,CAAC,CAAC;QAClD,EAAE,CAAC,aAAa,CAAC,aAAa,EAAE,gBAAgB,CAAC,CAAC;QAClD,qDAAqD;QAErD,gCAAgC;QAChC,eAAe,CAAC,EAAE,EAAE,MAAM,EAAE;YAC1B,aAAa;YACb,aAAa;YACb,gBAAgB;YAChB,gBAAgB;SACjB,CAAC,CAAC;QAEH,uEAAuE;QACvE,wFAAwF;QACxF,MAAM,iBAAiB,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,GAAG,CAAC,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC,CAAC;QACzE,EAAE,CAAC,OAAO,CAAC;;;;KAIV,CAAC,CAAC,GAAG,CAAC,iBAAiB,EAAE,MAAM,CAAC,CAAC;QAElC,wDAAwD;QACxD,gEAAgE;QAChE,MAAM,WAAW,GAAG,OAAO,CAAC,GAAG,EAAE,CAAC;QAClC,IAAI,CAAC;YACH,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;YAEvB,MAAM,YAAY,GAAG,MAAM,2BAA2B,CAAC,EAAE,CAAC,CAAC;YAE3D,gCAAgC;YAChC,MAAM,CAAC,KAAK,CAAC,YAAY,EAAE,CAAC,EAAE,iCAAiC,CAAC,CAAC;YAEjE,oBAAoB;YAEpB,kEAAkE;YAClE,MAAM,YAAY,GAAG,eAAe,CAAC,EAAE,EAAE,MAAM,CAAC,CAAC;YACjD,MAAM,CAAC,KAAK,CAAC,YAAY,CAAC,MAAM,EAAE,CAAC,EAAE,mCAAmC,CAAC,CAAC;YAC1E,MAAM,CAAC,EAAE,CACP,YAAY,CAAC,QAAQ,CAAC,YAAY,CAAC,EACnC,6BAA6B,CAC9B,CAAC;YACF,MAAM,CAAC,EAAE,CACP,YAAY,CAAC,QAAQ,CAAC,YAAY,CAAC,EACnC,6BAA6B,CAC9B,CAAC;YAEF,mDAAmD;YACnD,MAAM,WAAW,GAAG,EAAE;iBACnB,OAAO,CACN;;OAEH,CACE;iBACA,GAAG,CAAC,MAAM,CAAiC,CAAC;YAE/C,MAAM,CAAC,KAAK,CAAC,WAAW,CAAC,MAAM,EAAE,CAAC,EAAE,mCAAmC,CAAC,CAAC;YAEzE,2CAA2C;YAC3C,MAAM,WAAW,GAAG,WAAW,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC;YACxD,MAAM,CAAC,EAAE,CACP,WAAW,CAAC,QAAQ,CAAC,aAAa,CAAC,EACnC,gCAAgC,CACjC,CAAC;YACF,MAAM,CAAC,EAAE,CACP,WAAW,CAAC,QAAQ,CAAC,aAAa,CAAC,EACnC,gCAAgC,CACjC,CAAC;YAEF,mDAAmD;YACnD,MAAM,IAAI,GAAG,EAAE;iBACZ,OAAO,CAAC,4CAA4C,CAAC;iBACrD,GAAG,CAAC,MAAM,CAA0B,CAAC;YACxC,MAAM,CAAC,KAAK,CACV,IAAI,CAAC,SAAS,EACd,CAAC,EACD,6CAA6C,CAC9C,CAAC;QACJ,CAAC;gBAAS,CAAC;YACT,uBAAuB;YACvB,OAAO,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC;QAC7B,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,qDAAqD,EAAE,KAAK,IAAI,EAAE;QACnE,sBAAsB;QACtB,MAAM,MAAM,GAAG,cAAc,CAAC,EAAE,CAAC,CAAC;QAElC,6CAA6C;QAC7C,MAAM,gBAAgB,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,aAAa,CAAC,CAAC;QAC3D,MAAM,gBAAgB,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,aAAa,CAAC,CAAC;QAC3D,2BAA2B;QAE3B,0BAA0B;QAC1B,eAAe,CAAC,EAAE,EAAE,MAAM,EAAE,CAAC,gBAAgB,EAAE,gBAAgB,CAAC,CAAC,CAAC;QAElE,uEAAuE;QACvE,MAAM,iBAAiB,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,GAAG,CAAC,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC,CAAC;QACzE,EAAE,CAAC,OAAO,CAAC;;;;KAIV,CAAC,CAAC,GAAG,CAAC,iBAAiB,EAAE,MAAM,CAAC,CAAC;QAElC,0BAA0B;QAC1B,MAAM,WAAW,GAAG,OAAO,CAAC,GAAG,EAAE,CAAC;QAClC,IAAI,CAAC;YACH,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;YAEvB,MAAM,YAAY,GAAG,MAAM,2BAA2B,CAAC,EAAE,CAAC,CAAC;YAE3D,+DAA+D;YAC/D,MAAM,CAAC,KAAK,CAAC,YAAY,EAAE,CAAC,EAAE,wCAAwC,CAAC,CAAC;YAExE,yCAAyC;YACzC,MAAM,IAAI,GAAG,EAAE;iBACZ,OAAO,CAAC,4CAA4C,CAAC;iBACrD,GAAG,CAAC,MAAM,CAA0B,CAAC;YACxC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,EAAE,CAAC,EAAE,yCAAyC,CAAC,CAAC;YAE3E,0DAA0D;YAC1D,MAAM,WAAW,GAAG,EAAE;iBACnB,OAAO,CAAC,qEAAqE,CAAC;iBAC9E,GAAG,CAAC,MAAM,CAAsB,CAAC;YACpC,MAAM,CAAC,KAAK,CACV,WAAW,CAAC,KAAK,EACjB,CAAC,EACD,+DAA+D,CAChE,CAAC;QACJ,CAAC;gBAAS,CAAC;YACT,OAAO,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC;QAC7B,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,uCAAuC,EAAE,KAAK,IAAI,EAAE;QACrD,sBAAsB;QACtB,MAAM,MAAM,GAAG,cAAc,CAAC,EAAE,CAAC,CAAC;QAElC,2CAA2C;QAC3C,MAAM,aAAa,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,YAAY,CAAC,CAAC;QACvD,MAAM,aAAa,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,YAAY,CAAC,CAAC;QACvD,MAAM,aAAa,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,YAAY,CAAC,CAAC;QAEvD,EAAE,CAAC,aAAa,CAAC,aAAa,EAAE,gBAAgB,CAAC,CAAC;QAClD,EAAE,CAAC,aAAa,CAAC,aAAa,EAAE,gBAAgB,CAAC,CAAC;QAClD,EAAE,CAAC,aAAa,CAAC,aAAa,EAAE,gBAAgB,CAAC,CAAC;QAElD,0BAA0B;QAC1B,eAAe,CAAC,EAAE,EAAE,MAAM,EAAE;YAC1B,aAAa;YACb,aAAa;YACb,aAAa;SACd,CAAC,CAAC;QAEH,uEAAuE;QACvE,MAAM,iBAAiB,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,GAAG,CAAC,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC,CAAC;QACzE,EAAE,CAAC,OAAO,CAAC;;;;KAIV,CAAC,CAAC,GAAG,CAAC,iBAAiB,EAAE,MAAM,CAAC,CAAC;QAElC,0BAA0B;QAC1B,MAAM,WAAW,GAAG,OAAO,CAAC,GAAG,EAAE,CAAC;QAClC,IAAI,CAAC;YACH,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;YAEvB,MAAM,YAAY,GAAG,MAAM,2BAA2B,CAAC,EAAE,CAAC,CAAC;YAE3D,kDAAkD;YAClD,MAAM,CAAC,KAAK,CAAC,YAAY,EAAE,CAAC,EAAE,iCAAiC,CAAC,CAAC;YAEjE,gCAAgC;YAChC,MAAM,WAAW,GAAG,EAAE;iBACnB,OAAO,CAAC,qEAAqE,CAAC;iBAC9E,GAAG,CAAC,MAAM,CAAsB,CAAC;YACpC,MAAM,CAAC,KAAK,CAAC,WAAW,CAAC,KAAK,EAAE,CAAC,EAAE,8BAA8B,CAAC,CAAC;YAEnE,0CAA0C;YAC1C,MAAM,YAAY,GAAG,eAAe,CAAC,EAAE,EAAE,MAAM,CAAC,CAAC;YACjD,MAAM,CAAC,KAAK,CAAC,YAAY,CAAC,MAAM,EAAE,CAAC,EAAE,yCAAyC,CAAC,CAAC;YAEhF,gDAAgD;YAChD,MAAM,IAAI,GAAG,EAAE;iBACZ,OAAO,CAAC,4CAA4C,CAAC;iBACrD,GAAG,CAAC,MAAM,CAA0B,CAAC;YACxC,MAAM,CAAC,KAAK,CACV,IAAI,CAAC,SAAS,EACd,CAAC,EACD,6CAA6C,CAC9C,CAAC;QACJ,CAAC;gBAAS,CAAC;YACT,OAAO,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC;QAC7B,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,iDAAiD,EAAE,KAAK,IAAI,EAAE;QAC/D,sBAAsB;QACtB,MAAM,MAAM,GAAG,cAAc,CAAC,EAAE,CAAC,CAAC;QAElC,iDAAiD;QACjD,MAAM,aAAa,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,YAAY,CAAC,CAAC;QACvD,MAAM,aAAa,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,YAAY,CAAC,CAAC;QACvD,MAAM,aAAa,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,YAAY,CAAC,CAAC;QACvD,MAAM,eAAe,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,YAAY,CAAC,CAAC;QAEzD,EAAE,CAAC,aAAa,CAAC,aAAa,EAAE,gBAAgB,CAAC,CAAC;QAClD,EAAE,CAAC,aAAa,CAAC,aAAa,EAAE,gBAAgB,CAAC,CAAC;QAClD,EAAE,CAAC,aAAa,CAAC,aAAa,EAAE,gBAAgB,CAAC,CAAC;QAClD,+BAA+B;QAE/B,8BAA8B;QAC9B,eAAe,CAAC,EAAE,EAAE,MAAM,EAAE;YAC1B,aAAa;YACb,aAAa;YACb,aAAa;YACb,eAAe;SAChB,CAAC,CAAC;QAEH,uEAAuE;QACvE,MAAM,iBAAiB,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,GAAG,CAAC,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC,CAAC;QACzE,EAAE,CAAC,OAAO,CAAC;;;;KAIV,CAAC,CAAC,GAAG,CAAC,iBAAiB,EAAE,MAAM,CAAC,CAAC;QAElC,0BAA0B;QAC1B,MAAM,WAAW,GAAG,OAAO,CAAC,GAAG,EAAE,CAAC;QAClC,IAAI,CAAC;YACH,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;YAEvB,MAAM,YAAY,GAAG,MAAM,2BAA2B,CAAC,EAAE,CAAC,CAAC;YAE3D,gCAAgC;YAChC,MAAM,CAAC,KAAK,CAAC,YAAY,EAAE,CAAC,EAAE,iCAAiC,CAAC,CAAC;YAEjE,oBAAoB;YACpB,MAAM,YAAY,GAAG,eAAe,CAAC,EAAE,EAAE,MAAM,CAAC,CAAC;YACjD,MAAM,CAAC,KAAK,CAAC,YAAY,CAAC,MAAM,EAAE,CAAC,EAAE,mCAAmC,CAAC,CAAC;YAE1E,MAAM,WAAW,GAAG,EAAE;iBACnB,OAAO,CAAC,6DAA6D,CAAC;iBACtE,GAAG,CAAC,MAAM,CAAiC,CAAC;YAC/C,MAAM,CAAC,KAAK,CAAC,WAAW,CAAC,MAAM,EAAE,CAAC,EAAE,kCAAkC,CAAC,CAAC;YACxE,MAAM,CAAC,KAAK,CACV,WAAW,CAAC,CAAC,CAAC,CAAC,SAAS,EACxB,YAAY,EACZ,+BAA+B,CAChC,CAAC;QACJ,CAAC;gBAAS,CAAC;YACT,OAAO,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC;QAC7B,CAAC;IACH,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAEH,mBAAmB;AAEnB;;GAEG;AACH,SAAS,cAAc,CAAC,EAAgB;IACtC,MAAM,OAAO,GAAG,gBAAgB,CAAC,EAAE,EAAE,YAAY,CAAC,CAAC;IAEnD,MAAM,MAAM,GAAG,EAAE;SACd,OAAO,CACN;;;GAGH,CACE;SACA,GAAG,CAAC,4BAA4B,EAAE,OAAO,EAAE,OAAO,CAAC,CAAC;IAEvD,OAAO,MAAM,CAAC,eAAyB,CAAC;AAC1C,CAAC;AAED;;GAEG;AACH,SAAS,eAAe,CACtB,EAAgB,EAChB,MAAc,EACd,SAAmB;IAEnB,MAAM,kBAAkB,GAAG,EAAE,CAAC,OAAO,CAAC;;;GAGrC,CAAC,CAAC;IAEH,KAAK,MAAM,QAAQ,IAAI,SAAS,EAAE,CAAC;QACjC,8DAA8D;QAC9D,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC;QACzC,MAAM,MAAM,GAAG,eAAe,CAAC,EAAE,EAAE,QAAQ,CAAC,CAAC;QAC7C,kBAAkB,CAAC,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;IACzC,CAAC;AACH,CAAC;AAED;;GAEG;AACH,SAAS,eAAe,CAAC,EAAgB,EAAE,MAAc;IACvD,OACE,EAAE;SACC,OAAO,CACN;;;;;GAKL,CACI;SACA,GAAG,CAAC,MAAM,CACd,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;AAC3B,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"tasks.auto-pruning-persistence.test.d.ts","sourceRoot":"","sources":["../../src/tests/tasks.auto-pruning-persistence.test.ts"],"names":[],"mappings":"AAAA;;;GAGG"}
|
|
@@ -0,0 +1,232 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Integration tests for v3.5.0 Auto-Pruning persistence
|
|
3
|
+
* Tests audit trail persistence after task archival (no cascade deletion)
|
|
4
|
+
*/
|
|
5
|
+
import { describe, it, beforeEach, afterEach } from 'node:test';
|
|
6
|
+
import assert from 'node:assert/strict';
|
|
7
|
+
import Database from 'better-sqlite3';
|
|
8
|
+
import { initializeSchema } from '../schema.js';
|
|
9
|
+
import { runAllMigrations } from '../migrations/index.js';
|
|
10
|
+
import { getOrCreateAgent, transaction } from '../database.js';
|
|
11
|
+
/**
|
|
12
|
+
* Test database instance
|
|
13
|
+
*/
|
|
14
|
+
let testDb;
|
|
15
|
+
/**
|
|
16
|
+
* Create an in-memory test database with schema and migrations
|
|
17
|
+
*/
|
|
18
|
+
function createTestDatabase() {
|
|
19
|
+
const db = new Database(':memory:');
|
|
20
|
+
db.pragma('foreign_keys = ON');
|
|
21
|
+
initializeSchema(db);
|
|
22
|
+
// Run migrations to add t_task_pruned_files table (v3.5.0)
|
|
23
|
+
runAllMigrations(db);
|
|
24
|
+
return db;
|
|
25
|
+
}
|
|
26
|
+
/**
|
|
27
|
+
* Helper: Create a test task in 'done' status (ready to archive)
|
|
28
|
+
*/
|
|
29
|
+
function createTestTask(db, title) {
|
|
30
|
+
const agentId = getOrCreateAgent(db, 'test-agent');
|
|
31
|
+
const statusId = 5; // done (ready to archive)
|
|
32
|
+
const result = db.prepare(`
|
|
33
|
+
INSERT INTO t_tasks (title, status_id, priority, created_by_agent_id, assigned_agent_id)
|
|
34
|
+
VALUES (?, ?, 2, ?, ?)
|
|
35
|
+
`).run(title, statusId, agentId, agentId);
|
|
36
|
+
return result.lastInsertRowid;
|
|
37
|
+
}
|
|
38
|
+
/**
|
|
39
|
+
* Helper: Create a pruned file record in audit table
|
|
40
|
+
*/
|
|
41
|
+
function createPrunedFileRecord(db, taskId, filePath) {
|
|
42
|
+
const result = db.prepare(`
|
|
43
|
+
INSERT INTO t_task_pruned_files (task_id, file_path, pruned_ts)
|
|
44
|
+
VALUES (?, ?, unixepoch())
|
|
45
|
+
`).run(taskId, filePath);
|
|
46
|
+
return result.lastInsertRowid;
|
|
47
|
+
}
|
|
48
|
+
/**
|
|
49
|
+
* Helper: Get task status by ID
|
|
50
|
+
*/
|
|
51
|
+
function getTaskStatus(db, taskId) {
|
|
52
|
+
const row = db.prepare('SELECT status_id FROM t_tasks WHERE id = ?').get(taskId);
|
|
53
|
+
if (!row) {
|
|
54
|
+
throw new Error(`Task not found: ${taskId}`);
|
|
55
|
+
}
|
|
56
|
+
return row.status_id;
|
|
57
|
+
}
|
|
58
|
+
/**
|
|
59
|
+
* Helper: Count pruned file records for a task
|
|
60
|
+
*/
|
|
61
|
+
function countPrunedFiles(db, taskId) {
|
|
62
|
+
const row = db.prepare(`
|
|
63
|
+
SELECT COUNT(*) as count FROM t_task_pruned_files WHERE task_id = ?
|
|
64
|
+
`).get(taskId);
|
|
65
|
+
return row.count;
|
|
66
|
+
}
|
|
67
|
+
/**
|
|
68
|
+
* Helper: Archive a task (test version)
|
|
69
|
+
*/
|
|
70
|
+
function archiveTask(db, taskId) {
|
|
71
|
+
const TASK_STATUS_DONE = 5;
|
|
72
|
+
const TASK_STATUS_ARCHIVED = 6;
|
|
73
|
+
return transaction(db, () => {
|
|
74
|
+
// Check if task is in 'done' status
|
|
75
|
+
const taskRow = db.prepare('SELECT status_id FROM t_tasks WHERE id = ?').get(taskId);
|
|
76
|
+
if (!taskRow) {
|
|
77
|
+
throw new Error(`Task with id ${taskId} not found`);
|
|
78
|
+
}
|
|
79
|
+
if (taskRow.status_id !== TASK_STATUS_DONE) {
|
|
80
|
+
throw new Error(`Task ${taskId} must be in 'done' status to archive`);
|
|
81
|
+
}
|
|
82
|
+
// Update to archived
|
|
83
|
+
db.prepare('UPDATE t_tasks SET status_id = ? WHERE id = ?').run(TASK_STATUS_ARCHIVED, taskId);
|
|
84
|
+
return { success: true, task_id: taskId };
|
|
85
|
+
});
|
|
86
|
+
}
|
|
87
|
+
/**
|
|
88
|
+
* Helper: Get pruned files for a task (test version)
|
|
89
|
+
*/
|
|
90
|
+
function getPrunedFiles(db, taskId) {
|
|
91
|
+
const rows = db.prepare(`
|
|
92
|
+
SELECT file_path, pruned_ts
|
|
93
|
+
FROM t_task_pruned_files
|
|
94
|
+
WHERE task_id = ?
|
|
95
|
+
ORDER BY pruned_ts DESC
|
|
96
|
+
`).all(taskId);
|
|
97
|
+
return {
|
|
98
|
+
success: true,
|
|
99
|
+
count: rows.length,
|
|
100
|
+
pruned_files: rows
|
|
101
|
+
};
|
|
102
|
+
}
|
|
103
|
+
describe('Auto-pruning: Audit trail persistence after archival', () => {
|
|
104
|
+
beforeEach(() => {
|
|
105
|
+
testDb = createTestDatabase();
|
|
106
|
+
});
|
|
107
|
+
afterEach(() => {
|
|
108
|
+
testDb.close();
|
|
109
|
+
});
|
|
110
|
+
it('should preserve audit trail after task archival', () => {
|
|
111
|
+
// 1. Setup: Create task with pruned files
|
|
112
|
+
const taskId = createTestTask(testDb, 'Task with pruned files');
|
|
113
|
+
// Create multiple pruned file records
|
|
114
|
+
const prunedFileIds = [
|
|
115
|
+
createPrunedFileRecord(testDb, taskId, '/tmp/file1.ts'),
|
|
116
|
+
createPrunedFileRecord(testDb, taskId, '/tmp/file2.ts'),
|
|
117
|
+
createPrunedFileRecord(testDb, taskId, '/tmp/file3.ts')
|
|
118
|
+
];
|
|
119
|
+
assert.equal(prunedFileIds.length, 3, 'Should create 3 pruned file records');
|
|
120
|
+
// 2. Verify pruned files exist before archival
|
|
121
|
+
const beforeCount = countPrunedFiles(testDb, taskId);
|
|
122
|
+
assert.equal(beforeCount, 3, 'Should have 3 pruned file records before archival');
|
|
123
|
+
// 3. Archive the task
|
|
124
|
+
const archiveResult = archiveTask(testDb, taskId);
|
|
125
|
+
assert.ok(archiveResult.success, 'Task archival should succeed');
|
|
126
|
+
assert.equal(archiveResult.task_id, taskId);
|
|
127
|
+
// 4. Verify task is archived
|
|
128
|
+
const taskStatus = getTaskStatus(testDb, taskId);
|
|
129
|
+
assert.equal(taskStatus, 6, 'Task should be archived (status_id 6)');
|
|
130
|
+
// 5. Verify pruned files still exist (NOT cascade deleted)
|
|
131
|
+
const afterCount = countPrunedFiles(testDb, taskId);
|
|
132
|
+
assert.equal(afterCount, 3, 'Should still have 3 pruned file records after archival');
|
|
133
|
+
// 6. Verify get_pruned_files works for archived tasks
|
|
134
|
+
const getPrunedResult = getPrunedFiles(testDb, taskId);
|
|
135
|
+
assert.ok(getPrunedResult.success, 'get_pruned_files should work for archived tasks');
|
|
136
|
+
assert.equal(getPrunedResult.count, 3, 'Should return count of 3 pruned files');
|
|
137
|
+
assert.equal(getPrunedResult.pruned_files.length, 3, 'Should return all 3 pruned files');
|
|
138
|
+
// Verify file paths are preserved
|
|
139
|
+
const filePaths = getPrunedResult.pruned_files.map((f) => f.file_path);
|
|
140
|
+
assert.ok(filePaths.includes('/tmp/file1.ts'), 'Should include file1.ts');
|
|
141
|
+
assert.ok(filePaths.includes('/tmp/file2.ts'), 'Should include file2.ts');
|
|
142
|
+
assert.ok(filePaths.includes('/tmp/file3.ts'), 'Should include file3.ts');
|
|
143
|
+
});
|
|
144
|
+
it('should maintain foreign key integrity after archival', () => {
|
|
145
|
+
const taskId = createTestTask(testDb, 'Task for FK test');
|
|
146
|
+
createPrunedFileRecord(testDb, taskId, '/tmp/test.ts');
|
|
147
|
+
// Archive task
|
|
148
|
+
const archiveResult = archiveTask(testDb, taskId);
|
|
149
|
+
assert.ok(archiveResult.success, 'Task archival should succeed');
|
|
150
|
+
// Verify foreign key still valid (can JOIN successfully)
|
|
151
|
+
const result = testDb.prepare(`
|
|
152
|
+
SELECT tpf.id, tpf.task_id, t.status_id
|
|
153
|
+
FROM t_task_pruned_files tpf
|
|
154
|
+
JOIN t_tasks t ON tpf.task_id = t.id
|
|
155
|
+
WHERE tpf.task_id = ?
|
|
156
|
+
`).get(taskId);
|
|
157
|
+
assert.ok(result, 'Foreign key join should succeed');
|
|
158
|
+
assert.equal(result.task_id, taskId, 'Task ID should match');
|
|
159
|
+
assert.equal(result.status_id, 6, 'Task status should be archived (6)');
|
|
160
|
+
});
|
|
161
|
+
it('should handle zero pruned files for archived task', () => {
|
|
162
|
+
const taskId = createTestTask(testDb, 'Task with no pruned files');
|
|
163
|
+
// Archive without any pruned files
|
|
164
|
+
const archiveResult = archiveTask(testDb, taskId);
|
|
165
|
+
assert.ok(archiveResult.success);
|
|
166
|
+
// Query pruned files - should return empty array
|
|
167
|
+
const getPrunedResult = getPrunedFiles(testDb, taskId);
|
|
168
|
+
assert.ok(getPrunedResult.success);
|
|
169
|
+
assert.equal(getPrunedResult.count, 0);
|
|
170
|
+
assert.equal(getPrunedResult.pruned_files.length, 0);
|
|
171
|
+
});
|
|
172
|
+
it('should preserve pruned file timestamps after archival', () => {
|
|
173
|
+
const taskId = createTestTask(testDb, 'Task for timestamp test');
|
|
174
|
+
// Create pruned file with explicit timestamp check
|
|
175
|
+
const beforeTs = Math.floor(Date.now() / 1000);
|
|
176
|
+
createPrunedFileRecord(testDb, taskId, '/tmp/timestamped.ts');
|
|
177
|
+
// Archive task
|
|
178
|
+
archiveTask(testDb, taskId);
|
|
179
|
+
// Get pruned files and verify timestamp
|
|
180
|
+
const getPrunedResult = getPrunedFiles(testDb, taskId);
|
|
181
|
+
assert.equal(getPrunedResult.pruned_files.length, 1);
|
|
182
|
+
const prunedFile = getPrunedResult.pruned_files[0];
|
|
183
|
+
assert.ok(prunedFile.pruned_ts, 'Should have pruned_ts timestamp');
|
|
184
|
+
assert.ok(prunedFile.file_path === '/tmp/timestamped.ts', 'File path should match');
|
|
185
|
+
// Verify timestamp is reasonable (within last few seconds)
|
|
186
|
+
const prunedAt = prunedFile.pruned_ts;
|
|
187
|
+
assert.ok(prunedAt >= beforeTs, 'Timestamp should be after or equal to test start');
|
|
188
|
+
assert.ok(prunedAt <= beforeTs + 5, 'Timestamp should be within 5 seconds');
|
|
189
|
+
});
|
|
190
|
+
it('should handle multiple archived tasks with pruned files', () => {
|
|
191
|
+
// Create multiple tasks with pruned files
|
|
192
|
+
const task1Id = createTestTask(testDb, 'Task 1');
|
|
193
|
+
const task2Id = createTestTask(testDb, 'Task 2');
|
|
194
|
+
const task3Id = createTestTask(testDb, 'Task 3');
|
|
195
|
+
createPrunedFileRecord(testDb, task1Id, '/tmp/task1-file1.ts');
|
|
196
|
+
createPrunedFileRecord(testDb, task1Id, '/tmp/task1-file2.ts');
|
|
197
|
+
createPrunedFileRecord(testDb, task2Id, '/tmp/task2-file1.ts');
|
|
198
|
+
createPrunedFileRecord(testDb, task3Id, '/tmp/task3-file1.ts');
|
|
199
|
+
createPrunedFileRecord(testDb, task3Id, '/tmp/task3-file2.ts');
|
|
200
|
+
createPrunedFileRecord(testDb, task3Id, '/tmp/task3-file3.ts');
|
|
201
|
+
// Archive all tasks
|
|
202
|
+
archiveTask(testDb, task1Id);
|
|
203
|
+
archiveTask(testDb, task2Id);
|
|
204
|
+
archiveTask(testDb, task3Id);
|
|
205
|
+
// Verify each task's pruned files are isolated and preserved
|
|
206
|
+
const task1Pruned = getPrunedFiles(testDb, task1Id);
|
|
207
|
+
assert.equal(task1Pruned.count, 2, 'Task 1 should have 2 pruned files');
|
|
208
|
+
const task2Pruned = getPrunedFiles(testDb, task2Id);
|
|
209
|
+
assert.equal(task2Pruned.count, 1, 'Task 2 should have 1 pruned file');
|
|
210
|
+
const task3Pruned = getPrunedFiles(testDb, task3Id);
|
|
211
|
+
assert.equal(task3Pruned.count, 3, 'Task 3 should have 3 pruned files');
|
|
212
|
+
// Verify file paths are isolated (no cross-contamination)
|
|
213
|
+
const task1Paths = task1Pruned.pruned_files.map((f) => f.file_path);
|
|
214
|
+
assert.ok(!task1Paths.includes('/tmp/task2-file1.ts'), 'Task 1 should not include Task 2 files');
|
|
215
|
+
assert.ok(!task1Paths.includes('/tmp/task3-file1.ts'), 'Task 1 should not include Task 3 files');
|
|
216
|
+
});
|
|
217
|
+
it('should verify CASCADE constraint when task is deleted (not archived)', () => {
|
|
218
|
+
// This test verifies the ON DELETE CASCADE behavior
|
|
219
|
+
const taskId = createTestTask(testDb, 'Task for deletion test');
|
|
220
|
+
createPrunedFileRecord(testDb, taskId, '/tmp/cascade-test.ts');
|
|
221
|
+
// Verify pruned file exists
|
|
222
|
+
const beforeCount = countPrunedFiles(testDb, taskId);
|
|
223
|
+
assert.equal(beforeCount, 1);
|
|
224
|
+
// DELETE task (not archive) - should cascade delete pruned files
|
|
225
|
+
testDb.prepare('DELETE FROM t_tasks WHERE id = ?').run(taskId);
|
|
226
|
+
// Verify pruned file was cascade deleted
|
|
227
|
+
const afterCount = countPrunedFiles(testDb, taskId);
|
|
228
|
+
assert.equal(afterCount, 0, 'Pruned files should be CASCADE deleted when task is deleted');
|
|
229
|
+
});
|
|
230
|
+
console.log('\n✅ All auto-pruning persistence tests passed!\n');
|
|
231
|
+
});
|
|
232
|
+
//# sourceMappingURL=tasks.auto-pruning-persistence.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"tasks.auto-pruning-persistence.test.js","sourceRoot":"","sources":["../../src/tests/tasks.auto-pruning-persistence.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Integration tests for v3.5.0 Auto-Pruning Safety Check
|
|
3
|
+
* Tests that the system blocks task transition when ALL watched files are non-existent
|
|
4
|
+
*
|
|
5
|
+
* Test Scenario: Zero Work Done Protection
|
|
6
|
+
* - Create task with 3 watched files
|
|
7
|
+
* - Make ALL 3 files non-existent
|
|
8
|
+
* - Trigger detectAndTransitionToReview()
|
|
9
|
+
* - Expected: Error thrown, task status unchanged, transaction rolled back
|
|
10
|
+
*/
|
|
11
|
+
export {};
|
|
12
|
+
//# sourceMappingURL=tasks.auto-pruning-safety.test.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"tasks.auto-pruning-safety.test.d.ts","sourceRoot":"","sources":["../../src/tests/tasks.auto-pruning-safety.test.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG"}
|