ai-sdlc 0.2.0-alpha.5 → 0.2.0-alpha.50

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.
Files changed (130) hide show
  1. package/README.md +53 -1058
  2. package/dist/agents/implementation.d.ts +6 -0
  3. package/dist/agents/implementation.d.ts.map +1 -1
  4. package/dist/agents/implementation.js +151 -13
  5. package/dist/agents/implementation.js.map +1 -1
  6. package/dist/agents/index.d.ts +2 -0
  7. package/dist/agents/index.d.ts.map +1 -1
  8. package/dist/agents/index.js +2 -0
  9. package/dist/agents/index.js.map +1 -1
  10. package/dist/agents/orchestrator.d.ts +61 -0
  11. package/dist/agents/orchestrator.d.ts.map +1 -0
  12. package/dist/agents/orchestrator.js +443 -0
  13. package/dist/agents/orchestrator.js.map +1 -0
  14. package/dist/agents/planning.d.ts +1 -1
  15. package/dist/agents/planning.d.ts.map +1 -1
  16. package/dist/agents/planning.js +55 -4
  17. package/dist/agents/planning.js.map +1 -1
  18. package/dist/agents/refinement.d.ts.map +1 -1
  19. package/dist/agents/refinement.js +22 -3
  20. package/dist/agents/refinement.js.map +1 -1
  21. package/dist/agents/research.d.ts +85 -1
  22. package/dist/agents/research.d.ts.map +1 -1
  23. package/dist/agents/research.js +506 -16
  24. package/dist/agents/research.js.map +1 -1
  25. package/dist/agents/review.d.ts +77 -2
  26. package/dist/agents/review.d.ts.map +1 -1
  27. package/dist/agents/review.js +615 -93
  28. package/dist/agents/review.js.map +1 -1
  29. package/dist/agents/rework.d.ts.map +1 -1
  30. package/dist/agents/rework.js +22 -3
  31. package/dist/agents/rework.js.map +1 -1
  32. package/dist/agents/single-task.d.ts +41 -0
  33. package/dist/agents/single-task.d.ts.map +1 -0
  34. package/dist/agents/single-task.js +357 -0
  35. package/dist/agents/single-task.js.map +1 -0
  36. package/dist/agents/state-assessor.d.ts +3 -3
  37. package/dist/agents/state-assessor.d.ts.map +1 -1
  38. package/dist/agents/state-assessor.js +6 -6
  39. package/dist/agents/state-assessor.js.map +1 -1
  40. package/dist/agents/test-pattern-detector.d.ts +49 -0
  41. package/dist/agents/test-pattern-detector.d.ts.map +1 -0
  42. package/dist/agents/test-pattern-detector.js +273 -0
  43. package/dist/agents/test-pattern-detector.js.map +1 -0
  44. package/dist/agents/verification.d.ts +11 -0
  45. package/dist/agents/verification.d.ts.map +1 -1
  46. package/dist/agents/verification.js +97 -12
  47. package/dist/agents/verification.js.map +1 -1
  48. package/dist/cli/commands/migrate.js +1 -1
  49. package/dist/cli/commands/migrate.js.map +1 -1
  50. package/dist/cli/commands.d.ts +65 -3
  51. package/dist/cli/commands.d.ts.map +1 -1
  52. package/dist/cli/commands.js +1108 -204
  53. package/dist/cli/commands.js.map +1 -1
  54. package/dist/cli/daemon.d.ts.map +1 -1
  55. package/dist/cli/daemon.js +20 -3
  56. package/dist/cli/daemon.js.map +1 -1
  57. package/dist/cli/runner.d.ts.map +1 -1
  58. package/dist/cli/runner.js +19 -11
  59. package/dist/cli/runner.js.map +1 -1
  60. package/dist/core/auth.d.ts +43 -0
  61. package/dist/core/auth.d.ts.map +1 -1
  62. package/dist/core/auth.js +105 -1
  63. package/dist/core/auth.js.map +1 -1
  64. package/dist/core/client.d.ts +6 -0
  65. package/dist/core/client.d.ts.map +1 -1
  66. package/dist/core/client.js +57 -3
  67. package/dist/core/client.js.map +1 -1
  68. package/dist/core/config.d.ts +24 -1
  69. package/dist/core/config.d.ts.map +1 -1
  70. package/dist/core/config.js +100 -3
  71. package/dist/core/config.js.map +1 -1
  72. package/dist/core/conflict-detector.d.ts +108 -0
  73. package/dist/core/conflict-detector.d.ts.map +1 -0
  74. package/dist/core/conflict-detector.js +413 -0
  75. package/dist/core/conflict-detector.js.map +1 -0
  76. package/dist/core/git-utils.d.ts +28 -0
  77. package/dist/core/git-utils.d.ts.map +1 -0
  78. package/dist/core/git-utils.js +146 -0
  79. package/dist/core/git-utils.js.map +1 -0
  80. package/dist/core/index.d.ts +19 -0
  81. package/dist/core/index.d.ts.map +1 -0
  82. package/dist/core/index.js +19 -0
  83. package/dist/core/index.js.map +1 -0
  84. package/dist/core/kanban.d.ts +1 -1
  85. package/dist/core/kanban.d.ts.map +1 -1
  86. package/dist/core/kanban.js +7 -6
  87. package/dist/core/kanban.js.map +1 -1
  88. package/dist/core/llm-utils.d.ts +103 -0
  89. package/dist/core/llm-utils.d.ts.map +1 -0
  90. package/dist/core/llm-utils.js +368 -0
  91. package/dist/core/llm-utils.js.map +1 -0
  92. package/dist/core/logger.d.ts +92 -0
  93. package/dist/core/logger.d.ts.map +1 -0
  94. package/dist/core/logger.js +221 -0
  95. package/dist/core/logger.js.map +1 -0
  96. package/dist/core/story-logger.d.ts +102 -0
  97. package/dist/core/story-logger.d.ts.map +1 -0
  98. package/dist/core/story-logger.js +265 -0
  99. package/dist/core/story-logger.js.map +1 -0
  100. package/dist/core/story.d.ts +89 -20
  101. package/dist/core/story.d.ts.map +1 -1
  102. package/dist/core/story.js +297 -52
  103. package/dist/core/story.js.map +1 -1
  104. package/dist/core/task-parser.d.ts +59 -0
  105. package/dist/core/task-parser.d.ts.map +1 -0
  106. package/dist/core/task-parser.js +235 -0
  107. package/dist/core/task-parser.js.map +1 -0
  108. package/dist/core/task-progress.d.ts +92 -0
  109. package/dist/core/task-progress.d.ts.map +1 -0
  110. package/dist/core/task-progress.js +280 -0
  111. package/dist/core/task-progress.js.map +1 -0
  112. package/dist/core/workflow-state.d.ts +45 -6
  113. package/dist/core/workflow-state.d.ts.map +1 -1
  114. package/dist/core/workflow-state.js +201 -12
  115. package/dist/core/workflow-state.js.map +1 -1
  116. package/dist/core/worktree.d.ts +77 -0
  117. package/dist/core/worktree.d.ts.map +1 -0
  118. package/dist/core/worktree.js +246 -0
  119. package/dist/core/worktree.js.map +1 -0
  120. package/dist/index.js +135 -5
  121. package/dist/index.js.map +1 -1
  122. package/dist/services/error-classifier.d.ts +119 -0
  123. package/dist/services/error-classifier.d.ts.map +1 -0
  124. package/dist/services/error-classifier.js +182 -0
  125. package/dist/services/error-classifier.js.map +1 -0
  126. package/dist/types/index.d.ts +336 -1
  127. package/dist/types/index.d.ts.map +1 -1
  128. package/dist/types/index.js +1 -0
  129. package/dist/types/index.js.map +1 -1
  130. package/package.json +4 -1
@@ -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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"}