@gallop.software/canon 2.21.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 +1 -17
- package/dist/eslint/index.d.ts +2 -0
- package/dist/eslint/index.js +4 -1
- package/dist/eslint/rules/require-canon-setup.d.ts +4 -0
- package/dist/eslint/rules/require-canon-setup.js +106 -0
- package/package.json +1 -1
- package/dist/cli/commands/validate.d.ts +0 -9
- package/dist/cli/commands/validate.js +0 -228
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/dist/eslint/index.d.ts
CHANGED
|
@@ -28,6 +28,7 @@ declare const plugin: {
|
|
|
28
28
|
'no-component-in-blocks': import("eslint").Rule.RuleModule;
|
|
29
29
|
'prefer-list-components': import("eslint").Rule.RuleModule;
|
|
30
30
|
'no-native-date': import("eslint").Rule.RuleModule;
|
|
31
|
+
'require-canon-setup': import("eslint").Rule.RuleModule;
|
|
31
32
|
};
|
|
32
33
|
/**
|
|
33
34
|
* Recommended rule configurations - spread into your ESLint config
|
|
@@ -46,6 +47,7 @@ declare const plugin: {
|
|
|
46
47
|
readonly 'gallop/no-component-in-blocks': "warn";
|
|
47
48
|
readonly 'gallop/prefer-list-components': "warn";
|
|
48
49
|
readonly 'gallop/no-native-date': "warn";
|
|
50
|
+
readonly 'gallop/require-canon-setup': "warn";
|
|
49
51
|
};
|
|
50
52
|
};
|
|
51
53
|
export default plugin;
|
package/dist/eslint/index.js
CHANGED
|
@@ -10,6 +10,7 @@ import noComponentInBlocks from './rules/no-component-in-blocks.js';
|
|
|
10
10
|
import preferListComponents from './rules/prefer-list-components.js';
|
|
11
11
|
import noNativeDate from './rules/no-native-date.js';
|
|
12
12
|
import blockNamingConvention from './rules/block-naming-convention.js';
|
|
13
|
+
import requireCanonSetup from './rules/require-canon-setup.js';
|
|
13
14
|
/**
|
|
14
15
|
* All Canon ESLint rules with recommended severity levels
|
|
15
16
|
*/
|
|
@@ -26,6 +27,7 @@ const recommended = {
|
|
|
26
27
|
'gallop/no-component-in-blocks': 'warn',
|
|
27
28
|
'gallop/prefer-list-components': 'warn',
|
|
28
29
|
'gallop/no-native-date': 'warn',
|
|
30
|
+
'gallop/require-canon-setup': 'warn',
|
|
29
31
|
};
|
|
30
32
|
const plugin = {
|
|
31
33
|
meta: {
|
|
@@ -45,6 +47,7 @@ const plugin = {
|
|
|
45
47
|
'no-component-in-blocks': noComponentInBlocks,
|
|
46
48
|
'prefer-list-components': preferListComponents,
|
|
47
49
|
'no-native-date': noNativeDate,
|
|
50
|
+
'require-canon-setup': requireCanonSetup,
|
|
48
51
|
},
|
|
49
52
|
/**
|
|
50
53
|
* Recommended rule configurations - spread into your ESLint config
|
|
@@ -53,4 +56,4 @@ const plugin = {
|
|
|
53
56
|
recommended,
|
|
54
57
|
};
|
|
55
58
|
export default plugin;
|
|
56
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
59
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvZXNsaW50L2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sY0FBYyxNQUFNLDZCQUE2QixDQUFBO0FBQ3hELE9BQU8sb0JBQW9CLE1BQU0sb0NBQW9DLENBQUE7QUFDckUsT0FBTyxvQkFBb0IsTUFBTSxtQ0FBbUMsQ0FBQTtBQUNwRSxPQUFPLDBCQUEwQixNQUFNLHlDQUF5QyxDQUFBO0FBQ2hGLE9BQU8sc0JBQXNCLE1BQU0scUNBQXFDLENBQUE7QUFDeEUsT0FBTyxpQkFBaUIsTUFBTSxnQ0FBZ0MsQ0FBQTtBQUM5RCxPQUFPLGtCQUFrQixNQUFNLGtDQUFrQyxDQUFBO0FBQ2pFLE9BQU8sNEJBQTRCLE1BQU0sNENBQTRDLENBQUE7QUFDckYsT0FBTyxtQkFBbUIsTUFBTSxtQ0FBbUMsQ0FBQTtBQUNuRSxPQUFPLG9CQUFvQixNQUFNLG1DQUFtQyxDQUFBO0FBQ3BFLE9BQU8sWUFBWSxNQUFNLDJCQUEyQixDQUFBO0FBQ3BELE9BQU8scUJBQXFCLE1BQU0sb0NBQW9DLENBQUE7QUFDdEUsT0FBTyxpQkFBaUIsTUFBTSxnQ0FBZ0MsQ0FBQTtBQUU5RDs7R0FFRztBQUNILE1BQU0sV0FBVyxHQUFHO0lBQ2xCLHlCQUF5QixFQUFFLE1BQU07SUFDakMsZ0NBQWdDLEVBQUUsTUFBTTtJQUN4QyxnQ0FBZ0MsRUFBRSxNQUFNO0lBQ3hDLCtCQUErQixFQUFFLE1BQU07SUFDdkMscUNBQXFDLEVBQUUsTUFBTTtJQUM3QyxpQ0FBaUMsRUFBRSxNQUFNO0lBQ3pDLDRCQUE0QixFQUFFLE1BQU07SUFDcEMsOEJBQThCLEVBQUUsTUFBTTtJQUN0Qyx3Q0FBd0MsRUFBRSxNQUFNO0lBQ2hELCtCQUErQixFQUFFLE1BQU07SUFDdkMsK0JBQStCLEVBQUUsTUFBTTtJQUN2Qyx1QkFBdUIsRUFBRSxNQUFNO0lBQy9CLDRCQUE0QixFQUFFLE1BQU07Q0FDNUIsQ0FBQTtBQUVWLE1BQU0sTUFBTSxHQUFHO0lBQ2IsSUFBSSxFQUFFO1FBQ0osSUFBSSxFQUFFLHNCQUFzQjtRQUM1QixPQUFPLEVBQUUsUUFBUTtLQUNsQjtJQUNELEtBQUssRUFBRTtRQUNMLGtCQUFrQixFQUFFLGNBQWM7UUFDbEMseUJBQXlCLEVBQUUscUJBQXFCO1FBQ2hELHlCQUF5QixFQUFFLG9CQUFvQjtRQUMvQyx3QkFBd0IsRUFBRSxvQkFBb0I7UUFDOUMsOEJBQThCLEVBQUUsMEJBQTBCO1FBQzFELDBCQUEwQixFQUFFLHNCQUFzQjtRQUNsRCxxQkFBcUIsRUFBRSxpQkFBaUI7UUFDeEMsdUJBQXVCLEVBQUUsa0JBQWtCO1FBQzNDLGlDQUFpQyxFQUFFLDRCQUE0QjtRQUMvRCx3QkFBd0IsRUFBRSxtQkFBbUI7UUFDN0Msd0JBQXdCLEVBQUUsb0JBQW9CO1FBQzlDLGdCQUFnQixFQUFFLFlBQVk7UUFDOUIscUJBQXFCLEVBQUUsaUJBQWlCO0tBQ3pDO0lBQ0Q7OztPQUdHO0lBQ0gsV0FBVztDQUNaLENBQUE7QUFFRCxlQUFlLE1BQU0sQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBub0NsaWVudEJsb2NrcyBmcm9tICcuL3J1bGVzL25vLWNsaWVudC1ibG9ja3MuanMnXG5pbXBvcnQgbm9Db250YWluZXJJblNlY3Rpb24gZnJvbSAnLi9ydWxlcy9uby1jb250YWluZXItaW4tc2VjdGlvbi5qcydcbmltcG9ydCBwcmVmZXJDb21wb25lbnRQcm9wcyBmcm9tICcuL3J1bGVzL3ByZWZlci1jb21wb25lbnQtcHJvcHMuanMnXG5pbXBvcnQgcHJlZmVyVHlwb2dyYXBoeUNvbXBvbmVudHMgZnJvbSAnLi9ydWxlcy9wcmVmZXItdHlwb2dyYXBoeS1jb21wb25lbnRzLmpzJ1xuaW1wb3J0IHByZWZlckxheW91dENvbXBvbmVudHMgZnJvbSAnLi9ydWxlcy9wcmVmZXItbGF5b3V0LWNvbXBvbmVudHMuanMnXG5pbXBvcnQgbm9BcmJpdHJhcnlDb2xvcnMgZnJvbSAnLi9ydWxlcy9uby1hcmJpdHJhcnktY29sb3JzLmpzJ1xuaW1wb3J0IG5vQ3Jvc3Nab25lSW1wb3J0cyBmcm9tICcuL3J1bGVzL25vLWNyb3NzLXpvbmUtaW1wb3J0cy5qcydcbmltcG9ydCBub05hdGl2ZUludGVyc2VjdGlvbk9ic2VydmVyIGZyb20gJy4vcnVsZXMvbm8tbmF0aXZlLWludGVyc2VjdGlvbi1vYnNlcnZlci5qcydcbmltcG9ydCBub0NvbXBvbmVudEluQmxvY2tzIGZyb20gJy4vcnVsZXMvbm8tY29tcG9uZW50LWluLWJsb2Nrcy5qcydcbmltcG9ydCBwcmVmZXJMaXN0Q29tcG9uZW50cyBmcm9tICcuL3J1bGVzL3ByZWZlci1saXN0LWNvbXBvbmVudHMuanMnXG5pbXBvcnQgbm9OYXRpdmVEYXRlIGZyb20gJy4vcnVsZXMvbm8tbmF0aXZlLWRhdGUuanMnXG5pbXBvcnQgYmxvY2tOYW1pbmdDb252ZW50aW9uIGZyb20gJy4vcnVsZXMvYmxvY2stbmFtaW5nLWNvbnZlbnRpb24uanMnXG5pbXBvcnQgcmVxdWlyZUNhbm9uU2V0dXAgZnJvbSAnLi9ydWxlcy9yZXF1aXJlLWNhbm9uLXNldHVwLmpzJ1xuXG4vKipcbiAqIEFsbCBDYW5vbiBFU0xpbnQgcnVsZXMgd2l0aCByZWNvbW1lbmRlZCBzZXZlcml0eSBsZXZlbHNcbiAqL1xuY29uc3QgcmVjb21tZW5kZWQgPSB7XG4gICdnYWxsb3Avbm8tY2xpZW50LWJsb2Nrcyc6ICd3YXJuJyxcbiAgJ2dhbGxvcC9ibG9jay1uYW1pbmctY29udmVudGlvbic6ICd3YXJuJyxcbiAgJ2dhbGxvcC9uby1jb250YWluZXItaW4tc2VjdGlvbic6ICd3YXJuJyxcbiAgJ2dhbGxvcC9wcmVmZXItY29tcG9uZW50LXByb3BzJzogJ3dhcm4nLFxuICAnZ2FsbG9wL3ByZWZlci10eXBvZ3JhcGh5LWNvbXBvbmVudHMnOiAnd2FybicsXG4gICdnYWxsb3AvcHJlZmVyLWxheW91dC1jb21wb25lbnRzJzogJ3dhcm4nLFxuICAnZ2FsbG9wL25vLWFyYml0cmFyeS1jb2xvcnMnOiAnd2FybicsXG4gICdnYWxsb3Avbm8tY3Jvc3Mtem9uZS1pbXBvcnRzJzogJ3dhcm4nLFxuICAnZ2FsbG9wL25vLW5hdGl2ZS1pbnRlcnNlY3Rpb24tb2JzZXJ2ZXInOiAnd2FybicsXG4gICdnYWxsb3Avbm8tY29tcG9uZW50LWluLWJsb2Nrcyc6ICd3YXJuJyxcbiAgJ2dhbGxvcC9wcmVmZXItbGlzdC1jb21wb25lbnRzJzogJ3dhcm4nLFxuICAnZ2FsbG9wL25vLW5hdGl2ZS1kYXRlJzogJ3dhcm4nLFxuICAnZ2FsbG9wL3JlcXVpcmUtY2Fub24tc2V0dXAnOiAnd2FybicsXG59IGFzIGNvbnN0XG5cbmNvbnN0IHBsdWdpbiA9IHtcbiAgbWV0YToge1xuICAgIG5hbWU6ICdlc2xpbnQtcGx1Z2luLWdhbGxvcCcsXG4gICAgdmVyc2lvbjogJzIuMTIuMCcsXG4gIH0sXG4gIHJ1bGVzOiB7XG4gICAgJ25vLWNsaWVudC1ibG9ja3MnOiBub0NsaWVudEJsb2NrcyxcbiAgICAnYmxvY2stbmFtaW5nLWNvbnZlbnRpb24nOiBibG9ja05hbWluZ0NvbnZlbnRpb24sXG4gICAgJ25vLWNvbnRhaW5lci1pbi1zZWN0aW9uJzogbm9Db250YWluZXJJblNlY3Rpb24sXG4gICAgJ3ByZWZlci1jb21wb25lbnQtcHJvcHMnOiBwcmVmZXJDb21wb25lbnRQcm9wcyxcbiAgICAncHJlZmVyLXR5cG9ncmFwaHktY29tcG9uZW50cyc6IHByZWZlclR5cG9ncmFwaHlDb21wb25lbnRzLFxuICAgICdwcmVmZXItbGF5b3V0LWNvbXBvbmVudHMnOiBwcmVmZXJMYXlvdXRDb21wb25lbnRzLFxuICAgICduby1hcmJpdHJhcnktY29sb3JzJzogbm9BcmJpdHJhcnlDb2xvcnMsXG4gICAgJ25vLWNyb3NzLXpvbmUtaW1wb3J0cyc6IG5vQ3Jvc3Nab25lSW1wb3J0cyxcbiAgICAnbm8tbmF0aXZlLWludGVyc2VjdGlvbi1vYnNlcnZlcic6IG5vTmF0aXZlSW50ZXJzZWN0aW9uT2JzZXJ2ZXIsXG4gICAgJ25vLWNvbXBvbmVudC1pbi1ibG9ja3MnOiBub0NvbXBvbmVudEluQmxvY2tzLFxuICAgICdwcmVmZXItbGlzdC1jb21wb25lbnRzJzogcHJlZmVyTGlzdENvbXBvbmVudHMsXG4gICAgJ25vLW5hdGl2ZS1kYXRlJzogbm9OYXRpdmVEYXRlLFxuICAgICdyZXF1aXJlLWNhbm9uLXNldHVwJzogcmVxdWlyZUNhbm9uU2V0dXAsXG4gIH0sXG4gIC8qKlxuICAgKiBSZWNvbW1lbmRlZCBydWxlIGNvbmZpZ3VyYXRpb25zIC0gc3ByZWFkIGludG8geW91ciBFU0xpbnQgY29uZmlnXG4gICAqIEBleGFtcGxlIHJ1bGVzOiB7IC4uLmdhbGxvcC5yZWNvbW1lbmRlZCB9XG4gICAqL1xuICByZWNvbW1lbmRlZCxcbn1cblxuZXhwb3J0IGRlZmF1bHQgcGx1Z2luXG4iXX0=
|
|
@@ -0,0 +1,106 @@
|
|
|
1
|
+
import * as fs from 'fs';
|
|
2
|
+
import * as path from 'path';
|
|
3
|
+
import { getCanonUrl, getCanonPattern } from '../utils/canon.js';
|
|
4
|
+
const RULE_NAME = 'require-canon-setup';
|
|
5
|
+
const pattern = getCanonPattern(RULE_NAME);
|
|
6
|
+
// Track if we've already reported for this lint run
|
|
7
|
+
let hasReported = false;
|
|
8
|
+
// Required dev dependencies
|
|
9
|
+
const REQUIRED_DEPENDENCIES = ['knip', '@gallop.software/canon'];
|
|
10
|
+
// Required npm scripts (key = script name, value = must contain this string)
|
|
11
|
+
const REQUIRED_SCRIPTS = {
|
|
12
|
+
unused: 'knip',
|
|
13
|
+
check: 'npm run',
|
|
14
|
+
lint: 'eslint',
|
|
15
|
+
ts: 'tsc',
|
|
16
|
+
audit: 'gallop',
|
|
17
|
+
'generate:ai-rules': 'gallop',
|
|
18
|
+
};
|
|
19
|
+
const rule = {
|
|
20
|
+
meta: {
|
|
21
|
+
type: 'suggestion',
|
|
22
|
+
docs: {
|
|
23
|
+
description: pattern?.summary || 'Require Canon setup in package.json',
|
|
24
|
+
recommended: true,
|
|
25
|
+
url: getCanonUrl(RULE_NAME),
|
|
26
|
+
},
|
|
27
|
+
messages: {
|
|
28
|
+
missingDependency: `[Canon] Missing required dependency: "{{dep}}". Run: npm install -D {{dep}}`,
|
|
29
|
+
missingScript: `[Canon] Missing required npm script: "{{script}}". Add to package.json scripts.`,
|
|
30
|
+
invalidScript: `[Canon] Script "{{script}}" should contain "{{expected}}".`,
|
|
31
|
+
},
|
|
32
|
+
schema: [],
|
|
33
|
+
},
|
|
34
|
+
create(context) {
|
|
35
|
+
// Only check once per lint run, on the first file
|
|
36
|
+
if (hasReported) {
|
|
37
|
+
return {};
|
|
38
|
+
}
|
|
39
|
+
// Find the project root (where package.json is)
|
|
40
|
+
const filename = context.filename || context.getFilename();
|
|
41
|
+
let dir = path.dirname(filename);
|
|
42
|
+
let packageJsonPath = '';
|
|
43
|
+
// Walk up to find package.json
|
|
44
|
+
while (dir !== path.dirname(dir)) {
|
|
45
|
+
const candidate = path.join(dir, 'package.json');
|
|
46
|
+
if (fs.existsSync(candidate)) {
|
|
47
|
+
packageJsonPath = candidate;
|
|
48
|
+
break;
|
|
49
|
+
}
|
|
50
|
+
dir = path.dirname(dir);
|
|
51
|
+
}
|
|
52
|
+
if (!packageJsonPath) {
|
|
53
|
+
return {};
|
|
54
|
+
}
|
|
55
|
+
return {
|
|
56
|
+
Program(node) {
|
|
57
|
+
if (hasReported)
|
|
58
|
+
return;
|
|
59
|
+
hasReported = true;
|
|
60
|
+
try {
|
|
61
|
+
const packageJson = JSON.parse(fs.readFileSync(packageJsonPath, 'utf8'));
|
|
62
|
+
const devDeps = packageJson.devDependencies || {};
|
|
63
|
+
const deps = packageJson.dependencies || {};
|
|
64
|
+
const allDeps = { ...deps, ...devDeps };
|
|
65
|
+
const scripts = packageJson.scripts || {};
|
|
66
|
+
// Check dependencies
|
|
67
|
+
for (const dep of REQUIRED_DEPENDENCIES) {
|
|
68
|
+
if (!allDeps[dep]) {
|
|
69
|
+
context.report({
|
|
70
|
+
node,
|
|
71
|
+
messageId: 'missingDependency',
|
|
72
|
+
data: { dep },
|
|
73
|
+
});
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
// Check scripts
|
|
77
|
+
for (const [scriptName, expectedContains] of Object.entries(REQUIRED_SCRIPTS)) {
|
|
78
|
+
if (!scripts[scriptName]) {
|
|
79
|
+
context.report({
|
|
80
|
+
node,
|
|
81
|
+
messageId: 'missingScript',
|
|
82
|
+
data: { script: scriptName },
|
|
83
|
+
});
|
|
84
|
+
}
|
|
85
|
+
else if (!scripts[scriptName].includes(expectedContains)) {
|
|
86
|
+
context.report({
|
|
87
|
+
node,
|
|
88
|
+
messageId: 'invalidScript',
|
|
89
|
+
data: { script: scriptName, expected: expectedContains },
|
|
90
|
+
});
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
catch {
|
|
95
|
+
// Ignore parse errors
|
|
96
|
+
}
|
|
97
|
+
},
|
|
98
|
+
};
|
|
99
|
+
},
|
|
100
|
+
};
|
|
101
|
+
// Reset the flag when the module is reloaded (for watch mode)
|
|
102
|
+
export function resetReported() {
|
|
103
|
+
hasReported = false;
|
|
104
|
+
}
|
|
105
|
+
export default rule;
|
|
106
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"require-canon-setup.js","sourceRoot":"","sources":["../../../src/eslint/rules/require-canon-setup.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,MAAM,IAAI,CAAA;AACxB,OAAO,KAAK,IAAI,MAAM,MAAM,CAAA;AAC5B,OAAO,EAAE,WAAW,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAA;AAEhE,MAAM,SAAS,GAAG,qBAAqB,CAAA;AACvC,MAAM,OAAO,GAAG,eAAe,CAAC,SAAS,CAAC,CAAA;AAE1C,oDAAoD;AACpD,IAAI,WAAW,GAAG,KAAK,CAAA;AAEvB,4BAA4B;AAC5B,MAAM,qBAAqB,GAAG,CAAC,MAAM,EAAE,wBAAwB,CAAC,CAAA;AAEhE,6EAA6E;AAC7E,MAAM,gBAAgB,GAA2B;IAC/C,MAAM,EAAE,MAAM;IACd,KAAK,EAAE,SAAS;IAChB,IAAI,EAAE,QAAQ;IACd,EAAE,EAAE,KAAK;IACT,KAAK,EAAE,QAAQ;IACf,mBAAmB,EAAE,QAAQ;CAC9B,CAAA;AAED,MAAM,IAAI,GAAoB;IAC5B,IAAI,EAAE;QACJ,IAAI,EAAE,YAAY;QAClB,IAAI,EAAE;YACJ,WAAW,EAAE,OAAO,EAAE,OAAO,IAAI,qCAAqC;YACtE,WAAW,EAAE,IAAI;YACjB,GAAG,EAAE,WAAW,CAAC,SAAS,CAAC;SAC5B;QACD,QAAQ,EAAE;YACR,iBAAiB,EAAE,6EAA6E;YAChG,aAAa,EAAE,iFAAiF;YAChG,aAAa,EAAE,4DAA4D;SAC5E;QACD,MAAM,EAAE,EAAE;KACX;IAED,MAAM,CAAC,OAAO;QACZ,kDAAkD;QAClD,IAAI,WAAW,EAAE,CAAC;YAChB,OAAO,EAAE,CAAA;QACX,CAAC;QAED,gDAAgD;QAChD,MAAM,QAAQ,GAAG,OAAO,CAAC,QAAQ,IAAI,OAAO,CAAC,WAAW,EAAE,CAAA;QAC1D,IAAI,GAAG,GAAG,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAA;QAChC,IAAI,eAAe,GAAG,EAAE,CAAA;QAExB,+BAA+B;QAC/B,OAAO,GAAG,KAAK,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC;YACjC,MAAM,SAAS,GAAG,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,cAAc,CAAC,CAAA;YAChD,IAAI,EAAE,CAAC,UAAU,CAAC,SAAS,CAAC,EAAE,CAAC;gBAC7B,eAAe,GAAG,SAAS,CAAA;gBAC3B,MAAK;YACP,CAAC;YACD,GAAG,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAA;QACzB,CAAC;QAED,IAAI,CAAC,eAAe,EAAE,CAAC;YACrB,OAAO,EAAE,CAAA;QACX,CAAC;QAED,OAAO;YACL,OAAO,CAAC,IAAI;gBACV,IAAI,WAAW;oBAAE,OAAM;gBACvB,WAAW,GAAG,IAAI,CAAA;gBAElB,IAAI,CAAC;oBACH,MAAM,WAAW,GAAG,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,YAAY,CAAC,eAAe,EAAE,MAAM,CAAC,CAAC,CAAA;oBACxE,MAAM,OAAO,GAAG,WAAW,CAAC,eAAe,IAAI,EAAE,CAAA;oBACjD,MAAM,IAAI,GAAG,WAAW,CAAC,YAAY,IAAI,EAAE,CAAA;oBAC3C,MAAM,OAAO,GAAG,EAAE,GAAG,IAAI,EAAE,GAAG,OAAO,EAAE,CAAA;oBACvC,MAAM,OAAO,GAAG,WAAW,CAAC,OAAO,IAAI,EAAE,CAAA;oBAEzC,qBAAqB;oBACrB,KAAK,MAAM,GAAG,IAAI,qBAAqB,EAAE,CAAC;wBACxC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC;4BAClB,OAAO,CAAC,MAAM,CAAC;gCACb,IAAI;gCACJ,SAAS,EAAE,mBAAmB;gCAC9B,IAAI,EAAE,EAAE,GAAG,EAAE;6BACd,CAAC,CAAA;wBACJ,CAAC;oBACH,CAAC;oBAED,gBAAgB;oBAChB,KAAK,MAAM,CAAC,UAAU,EAAE,gBAAgB,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,gBAAgB,CAAC,EAAE,CAAC;wBAC9E,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,EAAE,CAAC;4BACzB,OAAO,CAAC,MAAM,CAAC;gCACb,IAAI;gCACJ,SAAS,EAAE,eAAe;gCAC1B,IAAI,EAAE,EAAE,MAAM,EAAE,UAAU,EAAE;6BAC7B,CAAC,CAAA;wBACJ,CAAC;6BAAM,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC,QAAQ,CAAC,gBAAgB,CAAC,EAAE,CAAC;4BAC3D,OAAO,CAAC,MAAM,CAAC;gCACb,IAAI;gCACJ,SAAS,EAAE,eAAe;gCAC1B,IAAI,EAAE,EAAE,MAAM,EAAE,UAAU,EAAE,QAAQ,EAAE,gBAAgB,EAAE;6BACzD,CAAC,CAAA;wBACJ,CAAC;oBACH,CAAC;gBACH,CAAC;gBAAC,MAAM,CAAC;oBACP,sBAAsB;gBACxB,CAAC;YACH,CAAC;SACF,CAAA;IACH,CAAC;CACF,CAAA;AAED,8DAA8D;AAC9D,MAAM,UAAU,aAAa;IAC3B,WAAW,GAAG,KAAK,CAAA;AACrB,CAAC;AAED,eAAe,IAAI,CAAA","sourcesContent":["import type { Rule } from 'eslint'\nimport * as fs from 'fs'\nimport * as path from 'path'\nimport { getCanonUrl, getCanonPattern } from '../utils/canon.js'\n\nconst RULE_NAME = 'require-canon-setup'\nconst pattern = getCanonPattern(RULE_NAME)\n\n// Track if we've already reported for this lint run\nlet hasReported = false\n\n// Required dev dependencies\nconst REQUIRED_DEPENDENCIES = ['knip', '@gallop.software/canon']\n\n// Required npm scripts (key = script name, value = must contain this string)\nconst REQUIRED_SCRIPTS: Record<string, string> = {\n  unused: 'knip',\n  check: 'npm run',\n  lint: 'eslint',\n  ts: 'tsc',\n  audit: 'gallop',\n  'generate:ai-rules': 'gallop',\n}\n\nconst rule: Rule.RuleModule = {\n  meta: {\n    type: 'suggestion',\n    docs: {\n      description: pattern?.summary || 'Require Canon setup in package.json',\n      recommended: true,\n      url: getCanonUrl(RULE_NAME),\n    },\n    messages: {\n      missingDependency: `[Canon] Missing required dependency: \"{{dep}}\". Run: npm install -D {{dep}}`,\n      missingScript: `[Canon] Missing required npm script: \"{{script}}\". Add to package.json scripts.`,\n      invalidScript: `[Canon] Script \"{{script}}\" should contain \"{{expected}}\".`,\n    },\n    schema: [],\n  },\n\n  create(context) {\n    // Only check once per lint run, on the first file\n    if (hasReported) {\n      return {}\n    }\n\n    // Find the project root (where package.json is)\n    const filename = context.filename || context.getFilename()\n    let dir = path.dirname(filename)\n    let packageJsonPath = ''\n    \n    // Walk up to find package.json\n    while (dir !== path.dirname(dir)) {\n      const candidate = path.join(dir, 'package.json')\n      if (fs.existsSync(candidate)) {\n        packageJsonPath = candidate\n        break\n      }\n      dir = path.dirname(dir)\n    }\n\n    if (!packageJsonPath) {\n      return {}\n    }\n\n    return {\n      Program(node) {\n        if (hasReported) return\n        hasReported = true\n\n        try {\n          const packageJson = JSON.parse(fs.readFileSync(packageJsonPath, 'utf8'))\n          const devDeps = packageJson.devDependencies || {}\n          const deps = packageJson.dependencies || {}\n          const allDeps = { ...deps, ...devDeps }\n          const scripts = packageJson.scripts || {}\n\n          // Check dependencies\n          for (const dep of REQUIRED_DEPENDENCIES) {\n            if (!allDeps[dep]) {\n              context.report({\n                node,\n                messageId: 'missingDependency',\n                data: { dep },\n              })\n            }\n          }\n\n          // Check scripts\n          for (const [scriptName, expectedContains] of Object.entries(REQUIRED_SCRIPTS)) {\n            if (!scripts[scriptName]) {\n              context.report({\n                node,\n                messageId: 'missingScript',\n                data: { script: scriptName },\n              })\n            } else if (!scripts[scriptName].includes(expectedContains)) {\n              context.report({\n                node,\n                messageId: 'invalidScript',\n                data: { script: scriptName, expected: expectedContains },\n              })\n            }\n          }\n        } catch {\n          // Ignore parse errors\n        }\n      },\n    }\n  },\n}\n\n// Reset the flag when the module is reloaded (for watch mode)\nexport function resetReported() {\n  hasReported = false\n}\n\nexport default rule\n"]}
|
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"]}
|