Skip to content

Commit a2fb8c2

Browse files
authored
Merge pull request #13 from Agentic-Insights/linterfixes
fix: add the contextignore fix and linter usage
2 parents 97c8b49 + 3a88501 commit a2fb8c2

File tree

3 files changed

+105
-19
lines changed

3 files changed

+105
-19
lines changed

linters/typescript/src/cli.ts

+47-14
Original file line numberDiff line numberDiff line change
@@ -19,30 +19,63 @@ function parseLogLevel(logLevelArg: string): LogLevel {
1919
}
2020
}
2121

22+
function printUsage() {
23+
console.log(`
24+
Usage: codebase-context-lint <directory_to_lint> [options]
25+
26+
Codebase Context Linter
27+
This tool validates context files (.context.md, .context.yaml, .context.json),
28+
.contextdocs.md, and .contextignore according to the Codebase Context Specification.
29+
30+
Arguments:
31+
<directory_to_lint> The directory containing the files to lint
32+
33+
Options:
34+
--log-level <level> Set the logging level (error, warn, info, debug)
35+
Default: info
36+
--help, -h Show this help message
37+
38+
Examples:
39+
codebase-context-lint .
40+
codebase-context-lint /path/to/project --log-level debug
41+
42+
For more information, visit: https://github.com/Agentic-Insights/codebase-context-spec
43+
`);
44+
}
45+
2246
async function main() {
2347
const args = process.argv.slice(2);
2448
let directoryToLint: string | undefined;
2549
let logLevel = LogLevel.INFO;
2650

2751
for (let i = 0; i < args.length; i++) {
28-
if (args[i] === '--log-level' && i + 1 < args.length) {
29-
logLevel = parseLogLevel(args[i + 1]);
30-
i++; // Skip the next argument as it's the log level value
31-
} else if (!directoryToLint) {
32-
directoryToLint = args[i];
52+
switch (args[i]) {
53+
case '--log-level':
54+
if (i + 1 < args.length) {
55+
logLevel = parseLogLevel(args[++i]);
56+
} else {
57+
console.error('Error: --log-level requires a value');
58+
process.exit(1);
59+
}
60+
break;
61+
case '--help':
62+
case '-h':
63+
printUsage();
64+
process.exit(0);
65+
default:
66+
if (!directoryToLint) {
67+
directoryToLint = args[i];
68+
} else {
69+
console.error(`Error: Unexpected argument '${args[i]}'`);
70+
printUsage();
71+
process.exit(1);
72+
}
3373
}
3474
}
3575

3676
if (!directoryToLint) {
37-
console.error(`
38-
Usage: codebase-context-lint <directory_to_lint> [--log-level <level>]
39-
40-
Codebase Context Linter
41-
This tool validates context files, including .contextdocs.md and .contextignore, according to the Codebase Context Specification.
42-
43-
Options:
44-
--log-level <level> Set the logging level (error, warn, info, debug). Default: info
45-
`);
77+
console.error('Error: Directory to lint is required');
78+
printUsage();
4679
process.exit(1);
4780
}
4881

linters/typescript/src/context_linter.ts

+22-5
Original file line numberDiff line numberDiff line change
@@ -77,8 +77,8 @@ export class ContextLinter {
7777
isValid = await this.handleContextdocs(directoryPath) && isValid;
7878
isValid = await this.handleContextFilesRecursively(directoryPath) && isValid;
7979

80-
// Log ignored files
81-
this.logIgnoredFiles(directoryPath);
80+
// Report on .contextignore usage
81+
this.reportContextignoreUsage(directoryPath);
8282

8383
// Clear all caches after processing the directory
8484
this.clearAllCaches();
@@ -109,6 +109,9 @@ export class ContextLinter {
109109
if (await fileExists(contextignorePath)) {
110110
const content = await fs.promises.readFile(contextignorePath, 'utf-8');
111111
await this.contextignoreLinter.lintContextignoreFile(content, contextignorePath);
112+
this.log(LogLevel.INFO, `Found .contextignore file at ${this.normalizePath(contextignorePath)}`);
113+
} else {
114+
this.log(LogLevel.INFO, 'No .contextignore file found. All files will be processed.');
112115
}
113116
}
114117

@@ -174,19 +177,33 @@ export class ContextLinter {
174177
}
175178

176179
/**
177-
* Log ignored files in the directory
180+
* Report on .contextignore usage
178181
* @param directoryPath The path of the directory to check for ignored files
179182
*/
180-
private logIgnoredFiles(directoryPath: string): void {
183+
private reportContextignoreUsage(directoryPath: string): void {
184+
const ignoredFiles = this.contextignoreLinter.getIgnoredFiles(directoryPath);
185+
const ignoredDirectories = this.contextignoreLinter.getIgnoredDirectories(directoryPath);
186+
187+
this.log(LogLevel.INFO, '\n.contextignore Usage Report:');
188+
this.log(LogLevel.INFO, `Total ignored files: ${ignoredFiles.length}`);
189+
this.log(LogLevel.INFO, `Total ignored directories: ${ignoredDirectories.length}`);
190+
181191
if (this.logLevel === LogLevel.DEBUG) {
182-
const ignoredFiles = this.contextignoreLinter.getIgnoredFiles(directoryPath);
183192
if (ignoredFiles.length > 0) {
184193
this.log(LogLevel.DEBUG, '\nIgnored files:');
185194
for (const file of ignoredFiles) {
186195
this.log(LogLevel.DEBUG, ` ${this.normalizePath(file)}`);
187196
}
188197
}
198+
if (ignoredDirectories.length > 0) {
199+
this.log(LogLevel.DEBUG, '\nIgnored directories:');
200+
for (const dir of ignoredDirectories) {
201+
this.log(LogLevel.DEBUG, ` ${this.normalizePath(dir)}`);
202+
}
203+
}
189204
}
205+
206+
this.log(LogLevel.INFO, ''); // Add a blank line for better readability
190207
}
191208

192209
/**

linters/typescript/src/contextignore_linter.ts

+36
Original file line numberDiff line numberDiff line change
@@ -224,4 +224,40 @@ export class ContextignoreLinter {
224224
return [];
225225
}
226226
}
227+
228+
/**
229+
* Get a list of ignored directories in a directory
230+
* @param directoryPath The path of the directory to check
231+
* @returns An array of ignored directory paths
232+
*/
233+
public getIgnoredDirectories(directoryPath: string): string[] {
234+
try {
235+
const ig = this.ignoreCache.get(directoryPath);
236+
if (!ig) {
237+
return [];
238+
}
239+
240+
const ignoredDirectories: string[] = [];
241+
const walk = (dir: string) => {
242+
const dirents = fs.readdirSync(dir, { withFileTypes: true });
243+
for (const dirent of dirents) {
244+
if (dirent.isDirectory()) {
245+
const res = path.join(dir, dirent.name);
246+
const relativePath = path.relative(directoryPath, res);
247+
if (ig.ignores(relativePath)) {
248+
ignoredDirectories.push(res);
249+
} else {
250+
walk(res);
251+
}
252+
}
253+
}
254+
};
255+
256+
walk(directoryPath);
257+
return ignoredDirectories;
258+
} catch (error) {
259+
this.log(LogLevel.ERROR, `Error getting ignored directories for directory ${directoryPath}: ${error instanceof Error ? error.message : String(error)}`);
260+
return [];
261+
}
262+
}
227263
}

0 commit comments

Comments
 (0)