ai-sdlc 0.2.0-alpha.5 → 0.2.0-alpha.51
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 +53 -1058
- package/dist/agents/implementation.d.ts +6 -0
- package/dist/agents/implementation.d.ts.map +1 -1
- package/dist/agents/implementation.js +151 -13
- package/dist/agents/implementation.js.map +1 -1
- package/dist/agents/index.d.ts +2 -0
- package/dist/agents/index.d.ts.map +1 -1
- package/dist/agents/index.js +2 -0
- package/dist/agents/index.js.map +1 -1
- package/dist/agents/orchestrator.d.ts +61 -0
- package/dist/agents/orchestrator.d.ts.map +1 -0
- package/dist/agents/orchestrator.js +443 -0
- package/dist/agents/orchestrator.js.map +1 -0
- package/dist/agents/planning.d.ts +1 -1
- package/dist/agents/planning.d.ts.map +1 -1
- package/dist/agents/planning.js +55 -4
- package/dist/agents/planning.js.map +1 -1
- package/dist/agents/refinement.d.ts.map +1 -1
- package/dist/agents/refinement.js +22 -3
- package/dist/agents/refinement.js.map +1 -1
- package/dist/agents/research.d.ts +85 -1
- package/dist/agents/research.d.ts.map +1 -1
- package/dist/agents/research.js +506 -16
- package/dist/agents/research.js.map +1 -1
- package/dist/agents/review.d.ts +103 -2
- package/dist/agents/review.d.ts.map +1 -1
- package/dist/agents/review.js +775 -93
- package/dist/agents/review.js.map +1 -1
- package/dist/agents/rework.d.ts.map +1 -1
- package/dist/agents/rework.js +22 -3
- package/dist/agents/rework.js.map +1 -1
- package/dist/agents/single-task.d.ts +41 -0
- package/dist/agents/single-task.d.ts.map +1 -0
- package/dist/agents/single-task.js +357 -0
- package/dist/agents/single-task.js.map +1 -0
- package/dist/agents/state-assessor.d.ts +3 -3
- package/dist/agents/state-assessor.d.ts.map +1 -1
- package/dist/agents/state-assessor.js +6 -6
- package/dist/agents/state-assessor.js.map +1 -1
- package/dist/agents/test-pattern-detector.d.ts +49 -0
- package/dist/agents/test-pattern-detector.d.ts.map +1 -0
- package/dist/agents/test-pattern-detector.js +273 -0
- package/dist/agents/test-pattern-detector.js.map +1 -0
- package/dist/agents/verification.d.ts +11 -0
- package/dist/agents/verification.d.ts.map +1 -1
- package/dist/agents/verification.js +97 -12
- package/dist/agents/verification.js.map +1 -1
- package/dist/cli/commands/migrate.js +1 -1
- package/dist/cli/commands/migrate.js.map +1 -1
- package/dist/cli/commands.d.ts +65 -3
- package/dist/cli/commands.d.ts.map +1 -1
- package/dist/cli/commands.js +1108 -204
- package/dist/cli/commands.js.map +1 -1
- package/dist/cli/daemon.d.ts.map +1 -1
- package/dist/cli/daemon.js +20 -3
- package/dist/cli/daemon.js.map +1 -1
- package/dist/cli/runner.d.ts.map +1 -1
- package/dist/cli/runner.js +19 -11
- package/dist/cli/runner.js.map +1 -1
- package/dist/core/auth.d.ts +43 -0
- package/dist/core/auth.d.ts.map +1 -1
- package/dist/core/auth.js +105 -1
- package/dist/core/auth.js.map +1 -1
- package/dist/core/client.d.ts +6 -0
- package/dist/core/client.d.ts.map +1 -1
- package/dist/core/client.js +57 -3
- package/dist/core/client.js.map +1 -1
- package/dist/core/config.d.ts +24 -1
- package/dist/core/config.d.ts.map +1 -1
- package/dist/core/config.js +100 -3
- package/dist/core/config.js.map +1 -1
- package/dist/core/conflict-detector.d.ts +108 -0
- package/dist/core/conflict-detector.d.ts.map +1 -0
- package/dist/core/conflict-detector.js +413 -0
- package/dist/core/conflict-detector.js.map +1 -0
- package/dist/core/git-utils.d.ts +28 -0
- package/dist/core/git-utils.d.ts.map +1 -0
- package/dist/core/git-utils.js +146 -0
- package/dist/core/git-utils.js.map +1 -0
- package/dist/core/index.d.ts +19 -0
- package/dist/core/index.d.ts.map +1 -0
- package/dist/core/index.js +19 -0
- package/dist/core/index.js.map +1 -0
- package/dist/core/kanban.d.ts +1 -1
- package/dist/core/kanban.d.ts.map +1 -1
- package/dist/core/kanban.js +7 -6
- package/dist/core/kanban.js.map +1 -1
- package/dist/core/llm-utils.d.ts +103 -0
- package/dist/core/llm-utils.d.ts.map +1 -0
- package/dist/core/llm-utils.js +368 -0
- package/dist/core/llm-utils.js.map +1 -0
- package/dist/core/logger.d.ts +92 -0
- package/dist/core/logger.d.ts.map +1 -0
- package/dist/core/logger.js +221 -0
- package/dist/core/logger.js.map +1 -0
- package/dist/core/story-logger.d.ts +102 -0
- package/dist/core/story-logger.d.ts.map +1 -0
- package/dist/core/story-logger.js +265 -0
- package/dist/core/story-logger.js.map +1 -0
- package/dist/core/story.d.ts +89 -20
- package/dist/core/story.d.ts.map +1 -1
- package/dist/core/story.js +300 -52
- package/dist/core/story.js.map +1 -1
- package/dist/core/task-parser.d.ts +59 -0
- package/dist/core/task-parser.d.ts.map +1 -0
- package/dist/core/task-parser.js +235 -0
- package/dist/core/task-parser.js.map +1 -0
- package/dist/core/task-progress.d.ts +92 -0
- package/dist/core/task-progress.d.ts.map +1 -0
- package/dist/core/task-progress.js +280 -0
- package/dist/core/task-progress.js.map +1 -0
- package/dist/core/workflow-state.d.ts +45 -6
- package/dist/core/workflow-state.d.ts.map +1 -1
- package/dist/core/workflow-state.js +201 -12
- package/dist/core/workflow-state.js.map +1 -1
- package/dist/core/worktree.d.ts +77 -0
- package/dist/core/worktree.d.ts.map +1 -0
- package/dist/core/worktree.js +246 -0
- package/dist/core/worktree.js.map +1 -0
- package/dist/index.js +135 -5
- package/dist/index.js.map +1 -1
- package/dist/services/error-classifier.d.ts +119 -0
- package/dist/services/error-classifier.d.ts.map +1 -0
- package/dist/services/error-classifier.js +182 -0
- package/dist/services/error-classifier.js.map +1 -0
- package/dist/types/index.d.ts +362 -1
- package/dist/types/index.d.ts.map +1 -1
- package/dist/types/index.js +1 -0
- package/dist/types/index.js.map +1 -1
- package/package.json +4 -1
- package/templates/story.md +5 -0
|
@@ -0,0 +1,221 @@
|
|
|
1
|
+
import fs from 'fs';
|
|
2
|
+
import path from 'path';
|
|
3
|
+
/**
|
|
4
|
+
* Log level priority for filtering
|
|
5
|
+
*/
|
|
6
|
+
const LOG_LEVEL_PRIORITY = {
|
|
7
|
+
debug: 0,
|
|
8
|
+
info: 1,
|
|
9
|
+
warn: 2,
|
|
10
|
+
error: 3,
|
|
11
|
+
};
|
|
12
|
+
/**
|
|
13
|
+
* Rolling file logger for ai-sdlc operations
|
|
14
|
+
*
|
|
15
|
+
* Features:
|
|
16
|
+
* - JSON Lines format (one JSON object per line)
|
|
17
|
+
* - Rolling by size (configurable, default 10MB)
|
|
18
|
+
* - Retains last N files (configurable, default 5)
|
|
19
|
+
* - Location: .ai-sdlc/logs/ai-sdlc-YYYY-MM-DD.log
|
|
20
|
+
*/
|
|
21
|
+
export class Logger {
|
|
22
|
+
projectRoot;
|
|
23
|
+
config;
|
|
24
|
+
logDir;
|
|
25
|
+
currentLogFile = null;
|
|
26
|
+
constructor(projectRoot, config) {
|
|
27
|
+
this.projectRoot = projectRoot;
|
|
28
|
+
this.config = config;
|
|
29
|
+
this.logDir = path.join(projectRoot, '.ai-sdlc', 'logs');
|
|
30
|
+
if (this.config.enabled) {
|
|
31
|
+
this.ensureLogDirectory();
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
/**
|
|
35
|
+
* Ensure the log directory exists
|
|
36
|
+
*/
|
|
37
|
+
ensureLogDirectory() {
|
|
38
|
+
if (!fs.existsSync(this.logDir)) {
|
|
39
|
+
fs.mkdirSync(this.logDir, { recursive: true });
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
/**
|
|
43
|
+
* Get the current log file path based on today's date
|
|
44
|
+
*/
|
|
45
|
+
getCurrentLogFile() {
|
|
46
|
+
const today = new Date().toISOString().split('T')[0]; // YYYY-MM-DD
|
|
47
|
+
return path.join(this.logDir, `ai-sdlc-${today}.log`);
|
|
48
|
+
}
|
|
49
|
+
/**
|
|
50
|
+
* Check if the current log file needs rotation based on size
|
|
51
|
+
*/
|
|
52
|
+
needsRotation(logFile) {
|
|
53
|
+
if (!fs.existsSync(logFile)) {
|
|
54
|
+
return false;
|
|
55
|
+
}
|
|
56
|
+
const stats = fs.statSync(logFile);
|
|
57
|
+
const maxSizeBytes = this.config.maxFileSizeMb * 1024 * 1024;
|
|
58
|
+
return stats.size >= maxSizeBytes;
|
|
59
|
+
}
|
|
60
|
+
/**
|
|
61
|
+
* Rotate the log file by renaming it with a sequence number
|
|
62
|
+
*/
|
|
63
|
+
rotateLogFile(logFile) {
|
|
64
|
+
const baseName = path.basename(logFile, '.log');
|
|
65
|
+
const dir = path.dirname(logFile);
|
|
66
|
+
// Find the next available sequence number
|
|
67
|
+
let seq = 1;
|
|
68
|
+
while (fs.existsSync(path.join(dir, `${baseName}.${seq}.log`))) {
|
|
69
|
+
seq++;
|
|
70
|
+
}
|
|
71
|
+
// Rename current file
|
|
72
|
+
fs.renameSync(logFile, path.join(dir, `${baseName}.${seq}.log`));
|
|
73
|
+
// Clean up old files
|
|
74
|
+
this.cleanupOldFiles();
|
|
75
|
+
}
|
|
76
|
+
/**
|
|
77
|
+
* Remove old log files beyond the retention limit
|
|
78
|
+
*/
|
|
79
|
+
cleanupOldFiles() {
|
|
80
|
+
if (!fs.existsSync(this.logDir)) {
|
|
81
|
+
return;
|
|
82
|
+
}
|
|
83
|
+
const files = fs.readdirSync(this.logDir)
|
|
84
|
+
.filter(f => f.startsWith('ai-sdlc-') && f.endsWith('.log'))
|
|
85
|
+
.map(f => ({
|
|
86
|
+
name: f,
|
|
87
|
+
path: path.join(this.logDir, f),
|
|
88
|
+
mtime: fs.statSync(path.join(this.logDir, f)).mtime.getTime(),
|
|
89
|
+
}))
|
|
90
|
+
.sort((a, b) => b.mtime - a.mtime); // Newest first
|
|
91
|
+
// Keep only the configured number of files
|
|
92
|
+
const filesToDelete = files.slice(this.config.maxFiles);
|
|
93
|
+
for (const file of filesToDelete) {
|
|
94
|
+
try {
|
|
95
|
+
fs.unlinkSync(file.path);
|
|
96
|
+
}
|
|
97
|
+
catch {
|
|
98
|
+
// Ignore deletion errors
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
/**
|
|
103
|
+
* Check if the given level should be logged based on config
|
|
104
|
+
*/
|
|
105
|
+
shouldLog(level) {
|
|
106
|
+
if (!this.config.enabled) {
|
|
107
|
+
return false;
|
|
108
|
+
}
|
|
109
|
+
return LOG_LEVEL_PRIORITY[level] >= LOG_LEVEL_PRIORITY[this.config.level];
|
|
110
|
+
}
|
|
111
|
+
/**
|
|
112
|
+
* Write a log entry to the current log file
|
|
113
|
+
*/
|
|
114
|
+
write(entry) {
|
|
115
|
+
if (!this.config.enabled) {
|
|
116
|
+
return;
|
|
117
|
+
}
|
|
118
|
+
const logFile = this.getCurrentLogFile();
|
|
119
|
+
// Check for rotation
|
|
120
|
+
if (this.needsRotation(logFile)) {
|
|
121
|
+
this.rotateLogFile(logFile);
|
|
122
|
+
}
|
|
123
|
+
// Append log entry as JSON line
|
|
124
|
+
const line = JSON.stringify(entry) + '\n';
|
|
125
|
+
fs.appendFileSync(logFile, line, 'utf-8');
|
|
126
|
+
}
|
|
127
|
+
/**
|
|
128
|
+
* Create a log entry with common fields
|
|
129
|
+
*/
|
|
130
|
+
createEntry(level, category, message, data) {
|
|
131
|
+
const entry = {
|
|
132
|
+
timestamp: new Date().toISOString(),
|
|
133
|
+
level,
|
|
134
|
+
category,
|
|
135
|
+
message,
|
|
136
|
+
};
|
|
137
|
+
if (data !== undefined) {
|
|
138
|
+
entry.data = data;
|
|
139
|
+
}
|
|
140
|
+
return entry;
|
|
141
|
+
}
|
|
142
|
+
/**
|
|
143
|
+
* Log a debug message
|
|
144
|
+
*/
|
|
145
|
+
debug(category, message, data) {
|
|
146
|
+
if (this.shouldLog('debug')) {
|
|
147
|
+
this.write(this.createEntry('debug', category, message, data));
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
/**
|
|
151
|
+
* Log an info message
|
|
152
|
+
*/
|
|
153
|
+
info(category, message, data) {
|
|
154
|
+
if (this.shouldLog('info')) {
|
|
155
|
+
this.write(this.createEntry('info', category, message, data));
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
/**
|
|
159
|
+
* Log a warning message
|
|
160
|
+
*/
|
|
161
|
+
warn(category, message, data) {
|
|
162
|
+
if (this.shouldLog('warn')) {
|
|
163
|
+
this.write(this.createEntry('warn', category, message, data));
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
/**
|
|
167
|
+
* Log an error message
|
|
168
|
+
*/
|
|
169
|
+
error(category, message, data) {
|
|
170
|
+
if (this.shouldLog('error')) {
|
|
171
|
+
this.write(this.createEntry('error', category, message, data));
|
|
172
|
+
}
|
|
173
|
+
}
|
|
174
|
+
/**
|
|
175
|
+
* Get the path to the current log file
|
|
176
|
+
*/
|
|
177
|
+
getLogFilePath() {
|
|
178
|
+
return this.getCurrentLogFile();
|
|
179
|
+
}
|
|
180
|
+
/**
|
|
181
|
+
* Get all log file paths
|
|
182
|
+
*/
|
|
183
|
+
getLogFiles() {
|
|
184
|
+
if (!fs.existsSync(this.logDir)) {
|
|
185
|
+
return [];
|
|
186
|
+
}
|
|
187
|
+
return fs.readdirSync(this.logDir)
|
|
188
|
+
.filter(f => f.startsWith('ai-sdlc-') && f.endsWith('.log'))
|
|
189
|
+
.map(f => path.join(this.logDir, f))
|
|
190
|
+
.sort((a, b) => {
|
|
191
|
+
const statA = fs.statSync(a);
|
|
192
|
+
const statB = fs.statSync(b);
|
|
193
|
+
return statB.mtime.getTime() - statA.mtime.getTime();
|
|
194
|
+
});
|
|
195
|
+
}
|
|
196
|
+
}
|
|
197
|
+
// Singleton logger instance
|
|
198
|
+
let globalLogger = null;
|
|
199
|
+
/**
|
|
200
|
+
* Initialize the global logger instance
|
|
201
|
+
*/
|
|
202
|
+
export function initLogger(projectRoot, config) {
|
|
203
|
+
globalLogger = new Logger(projectRoot, config);
|
|
204
|
+
return globalLogger;
|
|
205
|
+
}
|
|
206
|
+
/**
|
|
207
|
+
* Get the global logger instance (or create a disabled one if not initialized)
|
|
208
|
+
*/
|
|
209
|
+
export function getLogger() {
|
|
210
|
+
if (!globalLogger) {
|
|
211
|
+
// Return a disabled logger if not initialized
|
|
212
|
+
globalLogger = new Logger(process.cwd(), {
|
|
213
|
+
enabled: false,
|
|
214
|
+
level: 'info',
|
|
215
|
+
maxFileSizeMb: 10,
|
|
216
|
+
maxFiles: 5,
|
|
217
|
+
});
|
|
218
|
+
}
|
|
219
|
+
return globalLogger;
|
|
220
|
+
}
|
|
221
|
+
//# sourceMappingURL=logger.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"logger.js","sourceRoot":"","sources":["../../src/core/logger.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Per-story logging for concurrent execution
|
|
3
|
+
*
|
|
4
|
+
* Each story execution creates a new timestamped log file with dual output
|
|
5
|
+
* (console + file) for debugging and audit trail.
|
|
6
|
+
*/
|
|
7
|
+
import { LogLevel } from '../types/index.js';
|
|
8
|
+
/**
|
|
9
|
+
* Per-story logger that writes to timestamped files
|
|
10
|
+
*
|
|
11
|
+
* Features:
|
|
12
|
+
* - Dual output: console + timestamped file per execution
|
|
13
|
+
* - Automatic log rotation (keeps last N logs per story)
|
|
14
|
+
* - Crash-safe synchronous writes
|
|
15
|
+
* - Location: stories/{id}/logs/{timestamp}.log
|
|
16
|
+
*/
|
|
17
|
+
export declare class StoryLogger {
|
|
18
|
+
private logStream;
|
|
19
|
+
private storyId;
|
|
20
|
+
private logPath;
|
|
21
|
+
private closed;
|
|
22
|
+
/**
|
|
23
|
+
* Initialize a per-story logger
|
|
24
|
+
*
|
|
25
|
+
* @param storyId - Story ID (sanitized automatically)
|
|
26
|
+
* @param sdlcRoot - Path to .ai-sdlc directory
|
|
27
|
+
* @param maxLogs - Maximum number of log files to retain per story (default: 5)
|
|
28
|
+
*/
|
|
29
|
+
constructor(storyId: string, sdlcRoot: string, maxLogs?: number);
|
|
30
|
+
/**
|
|
31
|
+
* Log a message with specified level
|
|
32
|
+
*
|
|
33
|
+
* Writes to both console and file. Sanitizes message to prevent issues
|
|
34
|
+
* with very long lines or non-printable characters.
|
|
35
|
+
*
|
|
36
|
+
* @param level - Log level (INFO, AGENT, ERROR, WARN, DEBUG)
|
|
37
|
+
* @param message - Message to log
|
|
38
|
+
*/
|
|
39
|
+
log(level: LogLevel, message: string): void;
|
|
40
|
+
/**
|
|
41
|
+
* Close the log stream
|
|
42
|
+
*
|
|
43
|
+
* Should be called when story execution completes or on process exit.
|
|
44
|
+
* Flushes any buffered data and closes the file handle.
|
|
45
|
+
*
|
|
46
|
+
* @returns Promise that resolves when the stream is fully closed
|
|
47
|
+
*/
|
|
48
|
+
close(): Promise<void>;
|
|
49
|
+
/**
|
|
50
|
+
* Get the path to the current log file
|
|
51
|
+
*
|
|
52
|
+
* @returns Absolute path to the current log file
|
|
53
|
+
*/
|
|
54
|
+
getLogPath(): string;
|
|
55
|
+
/**
|
|
56
|
+
* Sanitize log message to prevent issues
|
|
57
|
+
*
|
|
58
|
+
* - Truncates messages longer than 10KB
|
|
59
|
+
* - Strips or escapes non-printable characters (except newlines/tabs)
|
|
60
|
+
* - Preserves ANSI color codes for console output
|
|
61
|
+
*
|
|
62
|
+
* @param message - Raw message to sanitize
|
|
63
|
+
* @returns Sanitized message
|
|
64
|
+
*/
|
|
65
|
+
private sanitizeMessage;
|
|
66
|
+
/**
|
|
67
|
+
* Rotate old log files, keeping only the most recent N logs
|
|
68
|
+
*
|
|
69
|
+
* Sorting is lexicographic (filename-based) since timestamps are in ISO 8601 format.
|
|
70
|
+
* Deletes oldest logs beyond the retention limit.
|
|
71
|
+
*
|
|
72
|
+
* @param logDir - Directory containing log files
|
|
73
|
+
* @param keep - Number of logs to retain
|
|
74
|
+
*/
|
|
75
|
+
private rotateOldLogs;
|
|
76
|
+
}
|
|
77
|
+
/**
|
|
78
|
+
* Get the latest log file path for a story
|
|
79
|
+
*
|
|
80
|
+
* @param sdlcRoot - Path to .ai-sdlc directory
|
|
81
|
+
* @param storyId - Story ID (sanitized automatically)
|
|
82
|
+
* @returns Path to latest log file, or null if no logs exist
|
|
83
|
+
*/
|
|
84
|
+
export declare function getLatestLogPath(sdlcRoot: string, storyId: string): string | null;
|
|
85
|
+
/**
|
|
86
|
+
* Read the last N lines from a log file
|
|
87
|
+
*
|
|
88
|
+
* @param filePath - Path to log file
|
|
89
|
+
* @param lines - Number of lines to read (default: 50)
|
|
90
|
+
* @returns Last N lines as a string
|
|
91
|
+
*/
|
|
92
|
+
export declare function readLastLines(filePath: string, lines?: number): Promise<string>;
|
|
93
|
+
/**
|
|
94
|
+
* Tail a log file (follow mode, like tail -f)
|
|
95
|
+
*
|
|
96
|
+
* Watches the file for changes and outputs new lines as they're written.
|
|
97
|
+
* Press Ctrl+C to exit.
|
|
98
|
+
*
|
|
99
|
+
* @param filePath - Path to log file
|
|
100
|
+
*/
|
|
101
|
+
export declare function tailLog(filePath: string): void;
|
|
102
|
+
//# sourceMappingURL=story-logger.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"story-logger.d.ts","sourceRoot":"","sources":["../../src/core/story-logger.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAKH,OAAO,EAAkB,QAAQ,EAAE,MAAM,mBAAmB,CAAC;AAI7D;;;;;;;;GAQG;AACH,qBAAa,WAAW;IACtB,OAAO,CAAC,SAAS,CAAiB;IAClC,OAAO,CAAC,OAAO,CAAS;IACxB,OAAO,CAAC,OAAO,CAAS;IACxB,OAAO,CAAC,MAAM,CAAkB;IAEhC;;;;;;OAMG;gBACS,OAAO,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,EAAE,OAAO,GAAE,MAAU;IAwClE;;;;;;;;OAQG;IACH,GAAG,CAAC,KAAK,EAAE,QAAQ,EAAE,OAAO,EAAE,MAAM,GAAG,IAAI;IAwC3C;;;;;;;OAOG;IACH,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAatB;;;;OAIG;IACH,UAAU,IAAI,MAAM;IAIpB;;;;;;;;;OASG;IACH,OAAO,CAAC,eAAe;IAgBvB;;;;;;;;OAQG;IACH,OAAO,CAAC,aAAa;CAsBtB;AAED;;;;;;GAMG;AACH,wBAAgB,gBAAgB,CAAC,QAAQ,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,GAAG,MAAM,GAAG,IAAI,CAmBjF;AAED;;;;;;GAMG;AACH,wBAAsB,aAAa,CAAC,QAAQ,EAAE,MAAM,EAAE,KAAK,GAAE,MAAW,GAAG,OAAO,CAAC,MAAM,CAAC,CAKzF;AAED;;;;;;;GAOG;AACH,wBAAgB,OAAO,CAAC,QAAQ,EAAE,MAAM,GAAG,IAAI,CAiC9C"}
|
|
@@ -0,0 +1,265 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Per-story logging for concurrent execution
|
|
3
|
+
*
|
|
4
|
+
* Each story execution creates a new timestamped log file with dual output
|
|
5
|
+
* (console + file) for debugging and audit trail.
|
|
6
|
+
*/
|
|
7
|
+
import fs from 'fs';
|
|
8
|
+
import path from 'path';
|
|
9
|
+
import { sanitizeStoryId } from './story.js';
|
|
10
|
+
import { STORIES_FOLDER } from '../types/index.js';
|
|
11
|
+
const MAX_MESSAGE_LENGTH = 10 * 1024; // 10KB per log entry
|
|
12
|
+
/**
|
|
13
|
+
* Per-story logger that writes to timestamped files
|
|
14
|
+
*
|
|
15
|
+
* Features:
|
|
16
|
+
* - Dual output: console + timestamped file per execution
|
|
17
|
+
* - Automatic log rotation (keeps last N logs per story)
|
|
18
|
+
* - Crash-safe synchronous writes
|
|
19
|
+
* - Location: stories/{id}/logs/{timestamp}.log
|
|
20
|
+
*/
|
|
21
|
+
export class StoryLogger {
|
|
22
|
+
logStream;
|
|
23
|
+
storyId;
|
|
24
|
+
logPath;
|
|
25
|
+
closed = false;
|
|
26
|
+
/**
|
|
27
|
+
* Initialize a per-story logger
|
|
28
|
+
*
|
|
29
|
+
* @param storyId - Story ID (sanitized automatically)
|
|
30
|
+
* @param sdlcRoot - Path to .ai-sdlc directory
|
|
31
|
+
* @param maxLogs - Maximum number of log files to retain per story (default: 5)
|
|
32
|
+
*/
|
|
33
|
+
constructor(storyId, sdlcRoot, maxLogs = 5) {
|
|
34
|
+
// SECURITY: Sanitize story ID to prevent path traversal
|
|
35
|
+
this.storyId = sanitizeStoryId(storyId);
|
|
36
|
+
const logDir = path.join(sdlcRoot, STORIES_FOLDER, this.storyId, 'logs');
|
|
37
|
+
// Ensure log directory exists
|
|
38
|
+
if (!fs.existsSync(logDir)) {
|
|
39
|
+
fs.mkdirSync(logDir, { recursive: true });
|
|
40
|
+
}
|
|
41
|
+
// Generate timestamp for log filename (ISO 8601 with safe characters)
|
|
42
|
+
// Split first to remove milliseconds (.000Z), then replace colons
|
|
43
|
+
const timestamp = new Date()
|
|
44
|
+
.toISOString()
|
|
45
|
+
.split('.')[0] // Remove milliseconds: 2026-01-15T10:30:00
|
|
46
|
+
.replace(/:/g, '-'); // Replace colons: 2026-01-15T10-30-00
|
|
47
|
+
this.logPath = path.join(logDir, `${timestamp}.log`);
|
|
48
|
+
// Ensure file exists before creating write stream (createWriteStream may not create immediately)
|
|
49
|
+
if (!fs.existsSync(this.logPath)) {
|
|
50
|
+
fs.writeFileSync(this.logPath, '');
|
|
51
|
+
}
|
|
52
|
+
// Create write stream in append mode
|
|
53
|
+
this.logStream = fs.createWriteStream(this.logPath, { flags: 'a' });
|
|
54
|
+
// Handle stream errors gracefully to prevent uncaught exceptions during cleanup
|
|
55
|
+
this.logStream.on('error', (err) => {
|
|
56
|
+
// ENOENT errors during cleanup are expected and can be ignored
|
|
57
|
+
if (err.code !== 'ENOENT') {
|
|
58
|
+
console.warn(`Warning: Log stream error: ${err.message}`);
|
|
59
|
+
}
|
|
60
|
+
});
|
|
61
|
+
// Rotate old logs (cleanup happens at initialization, not during writes)
|
|
62
|
+
this.rotateOldLogs(logDir, maxLogs);
|
|
63
|
+
}
|
|
64
|
+
/**
|
|
65
|
+
* Log a message with specified level
|
|
66
|
+
*
|
|
67
|
+
* Writes to both console and file. Sanitizes message to prevent issues
|
|
68
|
+
* with very long lines or non-printable characters.
|
|
69
|
+
*
|
|
70
|
+
* @param level - Log level (INFO, AGENT, ERROR, WARN, DEBUG)
|
|
71
|
+
* @param message - Message to log
|
|
72
|
+
*/
|
|
73
|
+
log(level, message) {
|
|
74
|
+
if (this.closed) {
|
|
75
|
+
console.warn('Warning: Attempted to log to closed logger');
|
|
76
|
+
return;
|
|
77
|
+
}
|
|
78
|
+
// Sanitize and truncate message
|
|
79
|
+
const sanitized = this.sanitizeMessage(message);
|
|
80
|
+
// Format log entry with ISO 8601 timestamp
|
|
81
|
+
const timestamp = new Date().toISOString();
|
|
82
|
+
const entry = `[${timestamp}] [${level}] ${sanitized}\n`;
|
|
83
|
+
// Write to file synchronously (crash-safe)
|
|
84
|
+
try {
|
|
85
|
+
this.logStream.write(entry);
|
|
86
|
+
}
|
|
87
|
+
catch (error) {
|
|
88
|
+
// Graceful degradation: if file write fails, continue with console-only
|
|
89
|
+
console.warn(`Warning: Failed to write to log file: ${error instanceof Error ? error.message : String(error)}`);
|
|
90
|
+
}
|
|
91
|
+
// Write to console based on level
|
|
92
|
+
switch (level) {
|
|
93
|
+
case 'ERROR':
|
|
94
|
+
console.error(sanitized);
|
|
95
|
+
break;
|
|
96
|
+
case 'WARN':
|
|
97
|
+
console.warn(sanitized);
|
|
98
|
+
break;
|
|
99
|
+
case 'DEBUG':
|
|
100
|
+
// Only log debug to file, not console (unless explicitly debugging)
|
|
101
|
+
if (process.env.DEBUG) {
|
|
102
|
+
console.log(sanitized);
|
|
103
|
+
}
|
|
104
|
+
break;
|
|
105
|
+
default:
|
|
106
|
+
console.log(sanitized);
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
/**
|
|
110
|
+
* Close the log stream
|
|
111
|
+
*
|
|
112
|
+
* Should be called when story execution completes or on process exit.
|
|
113
|
+
* Flushes any buffered data and closes the file handle.
|
|
114
|
+
*
|
|
115
|
+
* @returns Promise that resolves when the stream is fully closed
|
|
116
|
+
*/
|
|
117
|
+
close() {
|
|
118
|
+
return new Promise((resolve) => {
|
|
119
|
+
if (!this.closed) {
|
|
120
|
+
this.logStream.end(() => {
|
|
121
|
+
this.closed = true;
|
|
122
|
+
resolve();
|
|
123
|
+
});
|
|
124
|
+
}
|
|
125
|
+
else {
|
|
126
|
+
resolve();
|
|
127
|
+
}
|
|
128
|
+
});
|
|
129
|
+
}
|
|
130
|
+
/**
|
|
131
|
+
* Get the path to the current log file
|
|
132
|
+
*
|
|
133
|
+
* @returns Absolute path to the current log file
|
|
134
|
+
*/
|
|
135
|
+
getLogPath() {
|
|
136
|
+
return this.logPath;
|
|
137
|
+
}
|
|
138
|
+
/**
|
|
139
|
+
* Sanitize log message to prevent issues
|
|
140
|
+
*
|
|
141
|
+
* - Truncates messages longer than 10KB
|
|
142
|
+
* - Strips or escapes non-printable characters (except newlines/tabs)
|
|
143
|
+
* - Preserves ANSI color codes for console output
|
|
144
|
+
*
|
|
145
|
+
* @param message - Raw message to sanitize
|
|
146
|
+
* @returns Sanitized message
|
|
147
|
+
*/
|
|
148
|
+
sanitizeMessage(message) {
|
|
149
|
+
let sanitized = message;
|
|
150
|
+
// Truncate very long messages
|
|
151
|
+
if (sanitized.length > MAX_MESSAGE_LENGTH) {
|
|
152
|
+
sanitized = sanitized.substring(0, MAX_MESSAGE_LENGTH) + '\n... [message truncated]';
|
|
153
|
+
}
|
|
154
|
+
// Replace non-printable characters (except newlines, tabs, and ANSI escape codes)
|
|
155
|
+
// ANSI escape codes start with \x1b[ and are followed by formatting codes
|
|
156
|
+
// We preserve these for console output (they're stripped when written to most log viewers)
|
|
157
|
+
sanitized = sanitized.replace(/[\x00-\x08\x0B-\x0C\x0E-\x1F\x7F]/g, '');
|
|
158
|
+
return sanitized;
|
|
159
|
+
}
|
|
160
|
+
/**
|
|
161
|
+
* Rotate old log files, keeping only the most recent N logs
|
|
162
|
+
*
|
|
163
|
+
* Sorting is lexicographic (filename-based) since timestamps are in ISO 8601 format.
|
|
164
|
+
* Deletes oldest logs beyond the retention limit.
|
|
165
|
+
*
|
|
166
|
+
* @param logDir - Directory containing log files
|
|
167
|
+
* @param keep - Number of logs to retain
|
|
168
|
+
*/
|
|
169
|
+
rotateOldLogs(logDir, keep) {
|
|
170
|
+
try {
|
|
171
|
+
const logs = fs
|
|
172
|
+
.readdirSync(logDir)
|
|
173
|
+
.filter((f) => f.endsWith('.log'))
|
|
174
|
+
.sort()
|
|
175
|
+
.reverse(); // Newest first (lexicographic sort of ISO 8601 timestamps)
|
|
176
|
+
// Delete logs beyond the keep limit
|
|
177
|
+
const toDelete = logs.slice(keep);
|
|
178
|
+
for (const log of toDelete) {
|
|
179
|
+
try {
|
|
180
|
+
fs.unlinkSync(path.join(logDir, log));
|
|
181
|
+
}
|
|
182
|
+
catch {
|
|
183
|
+
// Ignore deletion errors (file may have been deleted by another process)
|
|
184
|
+
}
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
catch {
|
|
188
|
+
// Ignore rotation errors (directory may not exist or be readable)
|
|
189
|
+
// Rotation is a best-effort cleanup, not critical
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
}
|
|
193
|
+
/**
|
|
194
|
+
* Get the latest log file path for a story
|
|
195
|
+
*
|
|
196
|
+
* @param sdlcRoot - Path to .ai-sdlc directory
|
|
197
|
+
* @param storyId - Story ID (sanitized automatically)
|
|
198
|
+
* @returns Path to latest log file, or null if no logs exist
|
|
199
|
+
*/
|
|
200
|
+
export function getLatestLogPath(sdlcRoot, storyId) {
|
|
201
|
+
const sanitized = sanitizeStoryId(storyId);
|
|
202
|
+
const logDir = path.join(sdlcRoot, STORIES_FOLDER, sanitized, 'logs');
|
|
203
|
+
if (!fs.existsSync(logDir)) {
|
|
204
|
+
return null;
|
|
205
|
+
}
|
|
206
|
+
const logs = fs
|
|
207
|
+
.readdirSync(logDir)
|
|
208
|
+
.filter((f) => f.endsWith('.log'))
|
|
209
|
+
.sort()
|
|
210
|
+
.reverse(); // Newest first
|
|
211
|
+
if (logs.length === 0) {
|
|
212
|
+
return null;
|
|
213
|
+
}
|
|
214
|
+
return path.join(logDir, logs[0]);
|
|
215
|
+
}
|
|
216
|
+
/**
|
|
217
|
+
* Read the last N lines from a log file
|
|
218
|
+
*
|
|
219
|
+
* @param filePath - Path to log file
|
|
220
|
+
* @param lines - Number of lines to read (default: 50)
|
|
221
|
+
* @returns Last N lines as a string
|
|
222
|
+
*/
|
|
223
|
+
export async function readLastLines(filePath, lines = 50) {
|
|
224
|
+
const content = await fs.promises.readFile(filePath, 'utf-8');
|
|
225
|
+
const allLines = content.split('\n').filter((line) => line.trim() !== '');
|
|
226
|
+
const lastLines = allLines.slice(-lines);
|
|
227
|
+
return lastLines.join('\n');
|
|
228
|
+
}
|
|
229
|
+
/**
|
|
230
|
+
* Tail a log file (follow mode, like tail -f)
|
|
231
|
+
*
|
|
232
|
+
* Watches the file for changes and outputs new lines as they're written.
|
|
233
|
+
* Press Ctrl+C to exit.
|
|
234
|
+
*
|
|
235
|
+
* @param filePath - Path to log file
|
|
236
|
+
*/
|
|
237
|
+
export function tailLog(filePath) {
|
|
238
|
+
// Read existing content first
|
|
239
|
+
const existingContent = fs.readFileSync(filePath, 'utf-8');
|
|
240
|
+
process.stdout.write(existingContent);
|
|
241
|
+
// Track current file size
|
|
242
|
+
let lastSize = existingContent.length;
|
|
243
|
+
// Watch for changes
|
|
244
|
+
const watcher = fs.watchFile(filePath, { interval: 100 }, (curr) => {
|
|
245
|
+
if (curr.size > lastSize) {
|
|
246
|
+
// File grew - read new content
|
|
247
|
+
const stream = fs.createReadStream(filePath, {
|
|
248
|
+
start: lastSize,
|
|
249
|
+
encoding: 'utf-8',
|
|
250
|
+
});
|
|
251
|
+
stream.on('data', (chunk) => {
|
|
252
|
+
process.stdout.write(chunk);
|
|
253
|
+
});
|
|
254
|
+
lastSize = curr.size;
|
|
255
|
+
}
|
|
256
|
+
});
|
|
257
|
+
// Handle Ctrl+C gracefully
|
|
258
|
+
process.on('SIGINT', () => {
|
|
259
|
+
fs.unwatchFile(filePath);
|
|
260
|
+
process.exit(0);
|
|
261
|
+
});
|
|
262
|
+
// Keep process alive
|
|
263
|
+
console.log('\n[Following log file - Press Ctrl+C to exit]');
|
|
264
|
+
}
|
|
265
|
+
//# sourceMappingURL=story-logger.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"story-logger.js","sourceRoot":"","sources":["../../src/core/story-logger.ts"],"names":[],"mappings":"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"}
|