@gallop.software/canon 2.22.0 → 2.23.0
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/dist/cli/index.js
CHANGED
|
@@ -1,7 +1,6 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
2
|
import { audit } from './commands/audit.js';
|
|
3
3
|
import { generate } from './commands/generate.js';
|
|
4
|
-
import { validate } from './commands/validate.js';
|
|
5
4
|
import { version } from '../index.js';
|
|
6
5
|
const args = process.argv.slice(2);
|
|
7
6
|
const command = args[0];
|
|
@@ -28,7 +27,6 @@ ${colors.bold}Usage:${colors.reset}
|
|
|
28
27
|
${colors.bold}Commands:${colors.reset}
|
|
29
28
|
audit [path] Check Canon compliance (default: src/blocks/)
|
|
30
29
|
generate [output] Generate AI rules from Canon (default: .cursorrules)
|
|
31
|
-
validate [path] Validate project folder structure (default: .)
|
|
32
30
|
version Show version information
|
|
33
31
|
help Show this help message
|
|
34
32
|
|
|
@@ -39,18 +37,12 @@ ${colors.bold}Audit Options:${colors.reset}
|
|
|
39
37
|
${colors.bold}Generate Options:${colors.reset}
|
|
40
38
|
--output, -o Output file path (default: .cursorrules)
|
|
41
39
|
|
|
42
|
-
${colors.bold}Validate Options:${colors.reset}
|
|
43
|
-
--strict Exit with error code on violations
|
|
44
|
-
--json Output as JSON
|
|
45
|
-
|
|
46
40
|
${colors.bold}Examples:${colors.reset}
|
|
47
41
|
gallop audit
|
|
48
42
|
gallop audit src/blocks/ --strict
|
|
49
43
|
gallop generate
|
|
50
44
|
gallop generate .cursorrules
|
|
51
45
|
gallop generate --output .github/copilot-instructions.md
|
|
52
|
-
gallop validate
|
|
53
|
-
gallop validate . --strict
|
|
54
46
|
`);
|
|
55
47
|
}
|
|
56
48
|
function showVersion() {
|
|
@@ -88,14 +80,6 @@ async function main() {
|
|
|
88
80
|
};
|
|
89
81
|
await generate(generateOptions);
|
|
90
82
|
break;
|
|
91
|
-
case 'validate':
|
|
92
|
-
const validatePath = args[1] && !args[1].startsWith('--') ? args[1] : '.';
|
|
93
|
-
const validateOptions = {
|
|
94
|
-
strict: args.includes('--strict'),
|
|
95
|
-
json: args.includes('--json'),
|
|
96
|
-
};
|
|
97
|
-
await validate(validatePath, validateOptions);
|
|
98
|
-
break;
|
|
99
83
|
case 'version':
|
|
100
84
|
case '-v':
|
|
101
85
|
case '--version':
|
|
@@ -117,4 +101,4 @@ main().catch((error) => {
|
|
|
117
101
|
console.error('Error:', error.message);
|
|
118
102
|
process.exit(1);
|
|
119
103
|
});
|
|
120
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
104
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY2xpL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFFQSxPQUFPLEVBQUUsS0FBSyxFQUFFLE1BQU0scUJBQXFCLENBQUE7QUFDM0MsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLHdCQUF3QixDQUFBO0FBQ2pELE9BQU8sRUFBRSxPQUFPLEVBQUUsTUFBTSxhQUFhLENBQUE7QUFFckMsTUFBTSxJQUFJLEdBQUcsT0FBTyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUE7QUFDbEMsTUFBTSxPQUFPLEdBQUcsSUFBSSxDQUFDLENBQUMsQ0FBQyxDQUFBO0FBRXZCLDZCQUE2QjtBQUM3QixNQUFNLE1BQU0sR0FBRztJQUNiLEtBQUssRUFBRSxTQUFTO0lBQ2hCLElBQUksRUFBRSxTQUFTO0lBQ2YsR0FBRyxFQUFFLFNBQVM7SUFDZCxHQUFHLEVBQUUsVUFBVTtJQUNmLEtBQUssRUFBRSxVQUFVO0lBQ2pCLE1BQU0sRUFBRSxVQUFVO0lBQ2xCLElBQUksRUFBRSxVQUFVO0lBQ2hCLE9BQU8sRUFBRSxVQUFVO0lBQ25CLElBQUksRUFBRSxVQUFVO0NBQ2pCLENBQUE7QUFFRCxTQUFTLFFBQVE7SUFDZixPQUFPLENBQUMsR0FBRyxDQUFDO0VBQ1osTUFBTSxDQUFDLElBQUksYUFBYSxNQUFNLENBQUMsS0FBSztFQUNwQyxNQUFNLENBQUMsR0FBRyxrQkFBa0IsT0FBTyxHQUFHLE1BQU0sQ0FBQyxLQUFLOztFQUVsRCxNQUFNLENBQUMsSUFBSSxTQUFTLE1BQU0sQ0FBQyxLQUFLOzs7RUFHaEMsTUFBTSxDQUFDLElBQUksWUFBWSxNQUFNLENBQUMsS0FBSzs7Ozs7O0VBTW5DLE1BQU0sQ0FBQyxJQUFJLGlCQUFpQixNQUFNLENBQUMsS0FBSzs7OztFQUl4QyxNQUFNLENBQUMsSUFBSSxvQkFBb0IsTUFBTSxDQUFDLEtBQUs7OztFQUczQyxNQUFN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QUMsT0FBTyxDQUFDLENBQUE7SUFDdEMsT0FBTyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQTtBQUNqQixDQUFDLENBQUMsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbIiMhL3Vzci9iaW4vZW52IG5vZGVcblxuaW1wb3J0IHsgYXVkaXQgfSBmcm9tICcuL2NvbW1hbmRzL2F1ZGl0LmpzJ1xuaW1wb3J0IHsgZ2VuZXJhdGUgfSBmcm9tICcuL2NvbW1hbmRzL2dlbmVyYXRlLmpzJ1xuaW1wb3J0IHsgdmVyc2lvbiB9IGZyb20gJy4uL2luZGV4LmpzJ1xuXG5jb25zdCBhcmdzID0gcHJvY2Vzcy5hcmd2LnNsaWNlKDIpXG5jb25zdCBjb21tYW5kID0gYXJnc1swXVxuXG4vLyBDb2xvcnMgZm9yIHRlcm1pbmFsIG91dHB1dFxuY29uc3QgY29sb3JzID0ge1xuICByZXNldDogJ1xceDFiWzBtJyxcbiAgYm9sZDogJ1xceDFiWzFtJyxcbiAgZGltOiAnXFx4MWJbMm0nLFxuICByZWQ6ICdcXHgxYlszMW0nLFxuICBncmVlbjogJ1xceDFiWzMybScsXG4gIHllbGxvdzogJ1xceDFiWzMzbScsXG4gIGJsdWU6ICdcXHgxYlszNG0nLFxuICBtYWdlbnRhOiAnXFx4MWJbMzVtJyxcbiAgY3lhbjogJ1xceDFiWzM2bScsXG59XG5cbmZ1bmN0aW9uIHNob3dIZWxwKCkge1xuICBjb25zb2xlLmxvZyhgXG4ke2NvbG9ycy5ib2xkfUdhbGxvcCBDTEkke2NvbG9ycy5yZXNldH0gLSBDYW5vbiBDb21wbGlhbmNlIFRvb2xpbmdcbiR7Y29sb3JzLmRpbX1DYW5vbiBWZXJzaW9uOiAke3ZlcnNpb259JHtjb2xvcnMucmVzZXR9XG5cbiR7Y29sb3JzLmJvbGR9VXNhZ2U6JHtjb2xvcnMucmVzZXR9XG4gIGdhbGxvcCA8Y29tbWFuZD4gW29wdGlvbnNdXG5cbiR7Y29sb3JzLmJvbGR9Q29tbWFuZHM6JHtjb2xvcnMucmVzZXR9XG4gIGF1ZGl0IFtwYXRoXSAgICAgICBDaGVjayBDYW5vbiBjb21wbGlhbmNlIChkZWZhdWx0OiBzcmMvYmxvY2tzLylcbiAgZ2VuZXJhdGUgW291dHB1dF0gIEdlbmVyYXRlIEFJIHJ1bGVzIGZyb20gQ2Fub24gKGRlZmF1bHQ6IC5jdXJzb3JydWxlcylcbiAgdmVyc2lvbiAgICAgICAgICAgIFNob3cgdmVyc2lvbiBpbmZvcm1hdGlvblxuICBoZWxwICAgICAgICAgICAgICAgU2hvdyB0aGlzIGhlbHAgbWVzc2FnZVxuXG4ke2NvbG9ycy5ib2xkfUF1ZGl0IE9wdGlvbnM6JHtjb2xvcnMucmVzZXR9XG4gIC0tc3RyaWN0ICAgICAgICAgICBFeGl0IHdpdGggZXJyb3IgY29kZSBvbiB2aW9sYXRpb25zXG4gIC0tanNvbiAgICAgICAgICAgICBPdXRwdXQgYXMgSlNPTlxuXG4ke2NvbG9ycy5ib2xkfUdlbmVyYXRlIE9wdGlvbnM6JHtjb2xvcnMucmVzZXR9XG4gIC0tb3V0cHV0LCAtbyAgICAgICBPdXRwdXQgZmlsZSBwYXRoIChkZWZhdWx0OiAuY3Vyc29ycnVsZXMpXG5cbiR7Y29sb3JzLmJvbGR9RXhhbXBsZXM6JHtjb2xvcnMucmVzZXR9XG4gIGdhbGxvcCBhdWRpdFxuICBnYWxsb3AgYXVkaXQgc3JjL2Jsb2Nrcy8gLS1zdHJpY3RcbiAgZ2FsbG9wIGdlbmVyYXRlXG4gIGdhbGxvcCBnZW5lcmF0ZSAuY3Vyc29ycnVsZXNcbiAgZ2FsbG9wIGdlbmVyYXRlIC0tb3V0cHV0IC5naXRodWIvY29waWxvdC1pbnN0cnVjdGlvbnMubWRcbmApXG59XG5cbmZ1bmN0aW9uIHNob3dWZXJzaW9uKCkge1xuICBjb25zb2xlLmxvZyhgR2FsbG9wIENMSSB2MS4wLjBgKVxuICBjb25zb2xlLmxvZyhgQ2Fub24gdiR7dmVyc2lvbn1gKVxufVxuXG5hc3luYyBmdW5jdGlvbiBtYWluKCkge1xuICBzd2l0Y2ggKGNvbW1hbmQpIHtcbiAgICBjYXNlICdhdWRpdCc6XG4gICAgICBjb25zdCBhdWRpdFBhdGggPVxuICAgICAgICBhcmdzWzFdICYmICFhcmdzWzFdLnN0YXJ0c1dpdGgoJy0tJykgPyBhcmdzWzFdIDogJ3NyYy9ibG9ja3MvJ1xuICAgICAgY29uc3QgYXVkaXRPcHRpb25zID0ge1xuICAgICAgICBzdHJpY3Q6IGFyZ3MuaW5jbHVkZXMoJy0tc3RyaWN0JyksXG4gICAgICAgIGpzb246IGFyZ3MuaW5jbHVkZXMoJy0tanNvbicpLFxuICAgICAgICBmaXg6IGFyZ3MuaW5jbHVkZXMoJy0tZml4JyksXG4gICAgICB9XG4gICAgICBhd2FpdCBhdWRpdChhdWRpdFBhdGgsIGF1ZGl0T3B0aW9ucylcbiAgICAgIGJyZWFrXG5cbiAgICBjYXNlICdnZW5lcmF0ZSc6XG4gICAgICAvLyBGaW5kIG91dHB1dCBwYXRoIGZyb20gYXJnc1xuICAgICAgbGV0IG91dHB1dFBhdGggPSAnLmN1cnNvcnJ1bGVzJ1xuICAgICAgY29uc3Qgb3V0cHV0SW5kZXggPSBhcmdzLmluZGV4T2YoJy0tb3V0cHV0JylcbiAgICAgIGNvbnN0IG91dHB1dEluZGV4U2hvcnQgPSBhcmdzLmluZGV4T2YoJy1vJylcbiAgICAgIGlmIChvdXRwdXRJbmRleCAhPT0gLTEgJiYgYXJnc1tvdXRwdXRJbmRleCArIDFdKSB7XG4gICAgICAgIG91dHB1dFBhdGggPSBhcmdzW291dHB1dEluZGV4ICsgMV1cbiAgICAgIH0gZWxzZSBpZiAob3V0cHV0SW5kZXhTaG9ydCAhPT0gLTEgJiYgYXJnc1tvdXRwdXRJbmRleFNob3J0ICsgMV0pIHtcbiAgICAgICAgb3V0cHV0UGF0aCA9IGFyZ3Nbb3V0cHV0SW5kZXhTaG9ydCArIDFdXG4gICAgICB9IGVsc2UgaWYgKGFyZ3NbMV0gJiYgIWFyZ3NbMV0uc3RhcnRzV2l0aCgnLS0nKSkge1xuICAgICAgICBvdXRwdXRQYXRoID0gYXJnc1sxXVxuICAgICAgfVxuICAgICAgY29uc3QgZ2VuZXJhdGVPcHRpb25zID0ge1xuICAgICAgICBvdXRwdXQ6IG91dHB1dFBhdGgsXG4gICAgICAgIGZvcm1hdDogJ2N1cnNvcnJ1bGVzJyBhcyBjb25zdCxcbiAgICAgIH1cbiAgICAgIGF3YWl0IGdlbmVyYXRlKGdlbmVyYXRlT3B0aW9ucylcbiAgICAgIGJyZWFrXG5cbiAgICBjYXNlICd2ZXJzaW9uJzpcbiAgICBjYXNlICctdic6XG4gICAgY2FzZSAnLS12ZXJzaW9uJzpcbiAgICAgIHNob3dWZXJzaW9uKClcbiAgICAgIGJyZWFrXG5cbiAgICBjYXNlICdoZWxwJzpcbiAgICBjYXNlICctaCc6XG4gICAgY2FzZSAnLS1oZWxwJzpcbiAgICBjYXNlIHVuZGVmaW5lZDpcbiAgICAgIHNob3dIZWxwKClcbiAgICAgIGJyZWFrXG5cbiAgICBkZWZhdWx0OlxuICAgICAgY29uc29sZS5lcnJvcihgVW5rbm93biBjb21tYW5kOiAke2NvbW1hbmR9YClcbiAgICAgIGNvbnNvbGUuZXJyb3IoYFJ1biAnZ2FsbG9wIGhlbHAnIGZvciB1c2FnZSBpbmZvcm1hdGlvbi5gKVxuICAgICAgcHJvY2Vzcy5leGl0KDEpXG4gIH1cbn1cblxubWFpbigpLmNhdGNoKChlcnJvcikgPT4ge1xuICBjb25zb2xlLmVycm9yKCdFcnJvcjonLCBlcnJvci5tZXNzYWdlKVxuICBwcm9jZXNzLmV4aXQoMSlcbn0pXG4iXX0=
|
package/package.json
CHANGED
|
@@ -1,228 +0,0 @@
|
|
|
1
|
-
import * as fs from 'fs';
|
|
2
|
-
import * as path from 'path';
|
|
3
|
-
// Required dev dependencies
|
|
4
|
-
const REQUIRED_DEV_DEPENDENCIES = ['knip', '@gallop.software/canon'];
|
|
5
|
-
// Required npm scripts
|
|
6
|
-
const REQUIRED_SCRIPTS = {
|
|
7
|
-
unused: 'knip',
|
|
8
|
-
check: 'npm run lint && npm run ts && npm run unused',
|
|
9
|
-
lint: 'eslint',
|
|
10
|
-
ts: 'tsc',
|
|
11
|
-
audit: 'gallop audit',
|
|
12
|
-
'generate:ai-rules': 'gallop generate',
|
|
13
|
-
};
|
|
14
|
-
// Allowed top-level directories (non-dotfiles)
|
|
15
|
-
const ALLOWED_TOP_LEVEL = [
|
|
16
|
-
'src',
|
|
17
|
-
'public',
|
|
18
|
-
'_scripts',
|
|
19
|
-
'_data',
|
|
20
|
-
'_docs',
|
|
21
|
-
'node_modules',
|
|
22
|
-
];
|
|
23
|
-
// Allowed folders directly under /src
|
|
24
|
-
const ALLOWED_SRC_FOLDERS = [
|
|
25
|
-
'app',
|
|
26
|
-
'blocks',
|
|
27
|
-
'blog',
|
|
28
|
-
'components',
|
|
29
|
-
'hooks',
|
|
30
|
-
'styles',
|
|
31
|
-
'template',
|
|
32
|
-
'tools',
|
|
33
|
-
'types',
|
|
34
|
-
'utils',
|
|
35
|
-
];
|
|
36
|
-
// Files allowed at top level
|
|
37
|
-
const ALLOWED_TOP_LEVEL_FILES = [
|
|
38
|
-
'package.json',
|
|
39
|
-
'package-lock.json',
|
|
40
|
-
'tsconfig.json',
|
|
41
|
-
'tsconfig.tsbuildinfo',
|
|
42
|
-
'next.config.mjs',
|
|
43
|
-
'next.config.js',
|
|
44
|
-
'next-env.d.ts',
|
|
45
|
-
'eslint.config.mjs',
|
|
46
|
-
'eslint.config.js',
|
|
47
|
-
'.eslintrc.js',
|
|
48
|
-
'.eslintrc.json',
|
|
49
|
-
'postcss.config.js',
|
|
50
|
-
'tailwind.config.js',
|
|
51
|
-
'tailwind.config.ts',
|
|
52
|
-
'README.md',
|
|
53
|
-
'LICENSE',
|
|
54
|
-
'CHANGELOG.md',
|
|
55
|
-
'.gitignore',
|
|
56
|
-
'.cursorrules',
|
|
57
|
-
];
|
|
58
|
-
/**
|
|
59
|
-
* Check if a path is a dotfile or dotfolder
|
|
60
|
-
*/
|
|
61
|
-
function isDotfile(name) {
|
|
62
|
-
return name.startsWith('.');
|
|
63
|
-
}
|
|
64
|
-
/**
|
|
65
|
-
* Check if a path is an archive content folder (allowed new folders in /src)
|
|
66
|
-
*/
|
|
67
|
-
function isArchiveContentFolder(name) {
|
|
68
|
-
// Archive folders typically have plural names for content collections
|
|
69
|
-
// We allow any folder that could reasonably be archive content
|
|
70
|
-
// This is a heuristic - we check if it looks like a content collection
|
|
71
|
-
return !ALLOWED_SRC_FOLDERS.includes(name);
|
|
72
|
-
}
|
|
73
|
-
/**
|
|
74
|
-
* Validate project structure
|
|
75
|
-
*/
|
|
76
|
-
export async function validate(projectPath, options) {
|
|
77
|
-
const violations = [];
|
|
78
|
-
const absolutePath = path.resolve(process.cwd(), projectPath);
|
|
79
|
-
if (!fs.existsSync(absolutePath)) {
|
|
80
|
-
console.error(`Path does not exist: ${projectPath}`);
|
|
81
|
-
process.exit(1);
|
|
82
|
-
}
|
|
83
|
-
// Check top-level directories
|
|
84
|
-
const topLevelItems = fs.readdirSync(absolutePath);
|
|
85
|
-
for (const item of topLevelItems) {
|
|
86
|
-
const itemPath = path.join(absolutePath, item);
|
|
87
|
-
const stat = fs.statSync(itemPath);
|
|
88
|
-
if (stat.isDirectory()) {
|
|
89
|
-
// Dotfolders are exempt
|
|
90
|
-
if (isDotfile(item)) {
|
|
91
|
-
continue;
|
|
92
|
-
}
|
|
93
|
-
// Check if it's an allowed top-level directory
|
|
94
|
-
if (!ALLOWED_TOP_LEVEL.includes(item)) {
|
|
95
|
-
violations.push({
|
|
96
|
-
type: 'invalid-top-level',
|
|
97
|
-
path: item,
|
|
98
|
-
message: `Invalid top-level directory: ${item}. Allowed: ${ALLOWED_TOP_LEVEL.join(', ')} (dotfolders exempt)`,
|
|
99
|
-
});
|
|
100
|
-
}
|
|
101
|
-
}
|
|
102
|
-
else {
|
|
103
|
-
// It's a file - check if it's allowed at top level
|
|
104
|
-
if (!isDotfile(item) && !ALLOWED_TOP_LEVEL_FILES.includes(item)) {
|
|
105
|
-
// Allow common config file patterns
|
|
106
|
-
const isConfigFile = item.endsWith('.config.js') ||
|
|
107
|
-
item.endsWith('.config.mjs') ||
|
|
108
|
-
item.endsWith('.config.ts') ||
|
|
109
|
-
item.endsWith('.json') ||
|
|
110
|
-
item.endsWith('.md') ||
|
|
111
|
-
item.endsWith('.sh');
|
|
112
|
-
if (!isConfigFile) {
|
|
113
|
-
violations.push({
|
|
114
|
-
type: 'orphan-file',
|
|
115
|
-
path: item,
|
|
116
|
-
message: `Orphan file at project root: ${item}. Files should be in defined zones.`,
|
|
117
|
-
});
|
|
118
|
-
}
|
|
119
|
-
}
|
|
120
|
-
}
|
|
121
|
-
}
|
|
122
|
-
// Check /src folder structure
|
|
123
|
-
const srcPath = path.join(absolutePath, 'src');
|
|
124
|
-
if (fs.existsSync(srcPath)) {
|
|
125
|
-
const srcItems = fs.readdirSync(srcPath);
|
|
126
|
-
for (const item of srcItems) {
|
|
127
|
-
const itemPath = path.join(srcPath, item);
|
|
128
|
-
const stat = fs.statSync(itemPath);
|
|
129
|
-
if (stat.isDirectory()) {
|
|
130
|
-
// Check if it's an allowed /src folder
|
|
131
|
-
if (!ALLOWED_SRC_FOLDERS.includes(item) &&
|
|
132
|
-
!isArchiveContentFolder(item)) {
|
|
133
|
-
violations.push({
|
|
134
|
-
type: 'invalid-src-folder',
|
|
135
|
-
path: `src/${item}`,
|
|
136
|
-
message: `Invalid folder in /src: ${item}. Allowed: ${ALLOWED_SRC_FOLDERS.join(', ')} or archive content folders`,
|
|
137
|
-
});
|
|
138
|
-
}
|
|
139
|
-
}
|
|
140
|
-
}
|
|
141
|
-
// Check that src/app has route groups
|
|
142
|
-
const appPath = path.join(srcPath, 'app');
|
|
143
|
-
if (fs.existsSync(appPath)) {
|
|
144
|
-
const appItems = fs.readdirSync(appPath);
|
|
145
|
-
const hasDefaultGroup = appItems.some((item) => item === '(default)' || item.startsWith('('));
|
|
146
|
-
if (!hasDefaultGroup) {
|
|
147
|
-
violations.push({
|
|
148
|
-
type: 'invalid-src-folder',
|
|
149
|
-
path: 'src/app',
|
|
150
|
-
message: 'src/app should have at least one route group folder (e.g., (default)/)',
|
|
151
|
-
});
|
|
152
|
-
}
|
|
153
|
-
}
|
|
154
|
-
}
|
|
155
|
-
// Check package.json for required dependencies and scripts
|
|
156
|
-
const packageJsonPath = path.join(absolutePath, 'package.json');
|
|
157
|
-
if (fs.existsSync(packageJsonPath)) {
|
|
158
|
-
try {
|
|
159
|
-
const packageJson = JSON.parse(fs.readFileSync(packageJsonPath, 'utf8'));
|
|
160
|
-
// Check dev dependencies
|
|
161
|
-
const devDeps = packageJson.devDependencies || {};
|
|
162
|
-
const deps = packageJson.dependencies || {};
|
|
163
|
-
const allDeps = { ...deps, ...devDeps };
|
|
164
|
-
for (const dep of REQUIRED_DEV_DEPENDENCIES) {
|
|
165
|
-
if (!allDeps[dep]) {
|
|
166
|
-
violations.push({
|
|
167
|
-
type: 'missing-dependency',
|
|
168
|
-
path: 'package.json',
|
|
169
|
-
message: `Missing required dependency: ${dep}. Run: npm install -D ${dep}`,
|
|
170
|
-
});
|
|
171
|
-
}
|
|
172
|
-
}
|
|
173
|
-
// Check scripts
|
|
174
|
-
const scripts = packageJson.scripts || {};
|
|
175
|
-
for (const [scriptName, expectedPattern] of Object.entries(REQUIRED_SCRIPTS)) {
|
|
176
|
-
if (!scripts[scriptName]) {
|
|
177
|
-
violations.push({
|
|
178
|
-
type: 'missing-script',
|
|
179
|
-
path: 'package.json',
|
|
180
|
-
message: `Missing required script: "${scriptName}". Expected pattern: "${expectedPattern}"`,
|
|
181
|
-
});
|
|
182
|
-
}
|
|
183
|
-
else if (!scripts[scriptName].includes(expectedPattern.split(' ')[0])) {
|
|
184
|
-
// Check if the script contains the main command (first word of expected pattern)
|
|
185
|
-
violations.push({
|
|
186
|
-
type: 'missing-script',
|
|
187
|
-
path: 'package.json',
|
|
188
|
-
message: `Script "${scriptName}" should contain "${expectedPattern.split(' ')[0]}". Found: "${scripts[scriptName]}"`,
|
|
189
|
-
});
|
|
190
|
-
}
|
|
191
|
-
}
|
|
192
|
-
}
|
|
193
|
-
catch (e) {
|
|
194
|
-
console.error('Failed to parse package.json');
|
|
195
|
-
}
|
|
196
|
-
}
|
|
197
|
-
else {
|
|
198
|
-
violations.push({
|
|
199
|
-
type: 'orphan-file',
|
|
200
|
-
path: 'package.json',
|
|
201
|
-
message: 'Missing package.json file',
|
|
202
|
-
});
|
|
203
|
-
}
|
|
204
|
-
// Output results
|
|
205
|
-
if (options.json) {
|
|
206
|
-
console.log(JSON.stringify({
|
|
207
|
-
valid: violations.length === 0,
|
|
208
|
-
violations,
|
|
209
|
-
}, null, 2));
|
|
210
|
-
}
|
|
211
|
-
else {
|
|
212
|
-
if (violations.length === 0) {
|
|
213
|
-
console.log('✓ Project structure is valid');
|
|
214
|
-
}
|
|
215
|
-
else {
|
|
216
|
-
console.log(`Found ${violations.length} violation(s):\n`);
|
|
217
|
-
for (const v of violations) {
|
|
218
|
-
console.log(` ✗ ${v.message}`);
|
|
219
|
-
}
|
|
220
|
-
console.log('');
|
|
221
|
-
}
|
|
222
|
-
}
|
|
223
|
-
// Exit with error code if strict mode and violations found
|
|
224
|
-
if (options.strict && violations.length > 0) {
|
|
225
|
-
process.exit(1);
|
|
226
|
-
}
|
|
227
|
-
}
|
|
228
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"validate.js","sourceRoot":"","sources":["../../../src/cli/commands/validate.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs'\nimport * as path from 'path'\n\ninterface ValidateOptions {\n  strict: boolean\n  json: boolean\n}\n\ninterface Violation {\n  type: 'orphan-file' | 'invalid-top-level' | 'invalid-src-folder' | 'missing-dependency' | 'missing-script'\n  path: string\n  message: string\n}\n\n// Required dev dependencies\nconst REQUIRED_DEV_DEPENDENCIES = ['knip', '@gallop.software/canon']\n\n// Required npm scripts\nconst REQUIRED_SCRIPTS: Record<string, string> = {\n  unused: 'knip',\n  check: 'npm run lint && npm run ts && npm run unused',\n  lint: 'eslint',\n  ts: 'tsc',\n  audit: 'gallop audit',\n  'generate:ai-rules': 'gallop generate',\n}\n\n// Allowed top-level directories (non-dotfiles)\nconst ALLOWED_TOP_LEVEL = [\n  'src',\n  'public',\n  '_scripts',\n  '_data',\n  '_docs',\n  'node_modules',\n]\n\n// Allowed folders directly under /src\nconst ALLOWED_SRC_FOLDERS = [\n  'app',\n  'blocks',\n  'blog',\n  'components',\n  'hooks',\n  'styles',\n  'template',\n  'tools',\n  'types',\n  'utils',\n]\n\n// Files allowed at top level\nconst ALLOWED_TOP_LEVEL_FILES = [\n  'package.json',\n  'package-lock.json',\n  'tsconfig.json',\n  'tsconfig.tsbuildinfo',\n  'next.config.mjs',\n  'next.config.js',\n  'next-env.d.ts',\n  'eslint.config.mjs',\n  'eslint.config.js',\n  '.eslintrc.js',\n  '.eslintrc.json',\n  'postcss.config.js',\n  'tailwind.config.js',\n  'tailwind.config.ts',\n  'README.md',\n  'LICENSE',\n  'CHANGELOG.md',\n  '.gitignore',\n  '.cursorrules',\n]\n\n/**\n * Check if a path is a dotfile or dotfolder\n */\nfunction isDotfile(name: string): boolean {\n  return name.startsWith('.')\n}\n\n/**\n * Check if a path is an archive content folder (allowed new folders in /src)\n */\nfunction isArchiveContentFolder(name: string): boolean {\n  // Archive folders typically have plural names for content collections\n  // We allow any folder that could reasonably be archive content\n  // This is a heuristic - we check if it looks like a content collection\n  return !ALLOWED_SRC_FOLDERS.includes(name)\n}\n\n/**\n * Validate project structure\n */\nexport async function validate(\n  projectPath: string,\n  options: ValidateOptions\n): Promise<void> {\n  const violations: Violation[] = []\n  const absolutePath = path.resolve(process.cwd(), projectPath)\n\n  if (!fs.existsSync(absolutePath)) {\n    console.error(`Path does not exist: ${projectPath}`)\n    process.exit(1)\n  }\n\n  // Check top-level directories\n  const topLevelItems = fs.readdirSync(absolutePath)\n  for (const item of topLevelItems) {\n    const itemPath = path.join(absolutePath, item)\n    const stat = fs.statSync(itemPath)\n\n    if (stat.isDirectory()) {\n      // Dotfolders are exempt\n      if (isDotfile(item)) {\n        continue\n      }\n\n      // Check if it's an allowed top-level directory\n      if (!ALLOWED_TOP_LEVEL.includes(item)) {\n        violations.push({\n          type: 'invalid-top-level',\n          path: item,\n          message: `Invalid top-level directory: ${item}. Allowed: ${ALLOWED_TOP_LEVEL.join(', ')} (dotfolders exempt)`,\n        })\n      }\n    } else {\n      // It's a file - check if it's allowed at top level\n      if (!isDotfile(item) && !ALLOWED_TOP_LEVEL_FILES.includes(item)) {\n        // Allow common config file patterns\n        const isConfigFile =\n          item.endsWith('.config.js') ||\n          item.endsWith('.config.mjs') ||\n          item.endsWith('.config.ts') ||\n          item.endsWith('.json') ||\n          item.endsWith('.md') ||\n          item.endsWith('.sh')\n\n        if (!isConfigFile) {\n          violations.push({\n            type: 'orphan-file',\n            path: item,\n            message: `Orphan file at project root: ${item}. Files should be in defined zones.`,\n          })\n        }\n      }\n    }\n  }\n\n  // Check /src folder structure\n  const srcPath = path.join(absolutePath, 'src')\n  if (fs.existsSync(srcPath)) {\n    const srcItems = fs.readdirSync(srcPath)\n    for (const item of srcItems) {\n      const itemPath = path.join(srcPath, item)\n      const stat = fs.statSync(itemPath)\n\n      if (stat.isDirectory()) {\n        // Check if it's an allowed /src folder\n        if (\n          !ALLOWED_SRC_FOLDERS.includes(item) &&\n          !isArchiveContentFolder(item)\n        ) {\n          violations.push({\n            type: 'invalid-src-folder',\n            path: `src/${item}`,\n            message: `Invalid folder in /src: ${item}. Allowed: ${ALLOWED_SRC_FOLDERS.join(', ')} or archive content folders`,\n          })\n        }\n      }\n    }\n\n    // Check that src/app has route groups\n    const appPath = path.join(srcPath, 'app')\n    if (fs.existsSync(appPath)) {\n      const appItems = fs.readdirSync(appPath)\n      const hasDefaultGroup = appItems.some(\n        (item) => item === '(default)' || item.startsWith('(')\n      )\n\n      if (!hasDefaultGroup) {\n        violations.push({\n          type: 'invalid-src-folder',\n          path: 'src/app',\n          message:\n            'src/app should have at least one route group folder (e.g., (default)/)',\n        })\n      }\n    }\n  }\n\n  // Check package.json for required dependencies and scripts\n  const packageJsonPath = path.join(absolutePath, 'package.json')\n  if (fs.existsSync(packageJsonPath)) {\n    try {\n      const packageJson = JSON.parse(fs.readFileSync(packageJsonPath, 'utf8'))\n      \n      // Check dev dependencies\n      const devDeps = packageJson.devDependencies || {}\n      const deps = packageJson.dependencies || {}\n      const allDeps = { ...deps, ...devDeps }\n      \n      for (const dep of REQUIRED_DEV_DEPENDENCIES) {\n        if (!allDeps[dep]) {\n          violations.push({\n            type: 'missing-dependency',\n            path: 'package.json',\n            message: `Missing required dependency: ${dep}. Run: npm install -D ${dep}`,\n          })\n        }\n      }\n      \n      // Check scripts\n      const scripts = packageJson.scripts || {}\n      \n      for (const [scriptName, expectedPattern] of Object.entries(REQUIRED_SCRIPTS)) {\n        if (!scripts[scriptName]) {\n          violations.push({\n            type: 'missing-script',\n            path: 'package.json',\n            message: `Missing required script: \"${scriptName}\". Expected pattern: \"${expectedPattern}\"`,\n          })\n        } else if (!scripts[scriptName].includes(expectedPattern.split(' ')[0])) {\n          // Check if the script contains the main command (first word of expected pattern)\n          violations.push({\n            type: 'missing-script',\n            path: 'package.json',\n            message: `Script \"${scriptName}\" should contain \"${expectedPattern.split(' ')[0]}\". Found: \"${scripts[scriptName]}\"`,\n          })\n        }\n      }\n    } catch (e) {\n      console.error('Failed to parse package.json')\n    }\n  } else {\n    violations.push({\n      type: 'orphan-file',\n      path: 'package.json',\n      message: 'Missing package.json file',\n    })\n  }\n\n  // Output results\n  if (options.json) {\n    console.log(\n      JSON.stringify(\n        {\n          valid: violations.length === 0,\n          violations,\n        },\n        null,\n        2\n      )\n    )\n  } else {\n    if (violations.length === 0) {\n      console.log('✓ Project structure is valid')\n    } else {\n      console.log(`Found ${violations.length} violation(s):\\n`)\n      for (const v of violations) {\n        console.log(`  ✗ ${v.message}`)\n      }\n      console.log('')\n    }\n  }\n\n  // Exit with error code if strict mode and violations found\n  if (options.strict && violations.length > 0) {\n    process.exit(1)\n  }\n}\n"]}
|