@gallop.software/canon 2.25.0 → 2.29.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/commands/generate.d.ts +2 -1
- package/dist/cli/commands/generate.js +337 -76
- package/dist/cli/config.d.ts +6 -0
- package/dist/cli/config.js +17 -0
- package/dist/cli/index.js +36 -10
- package/dist/eslint/index.d.ts +8 -0
- package/dist/eslint/index.js +13 -1
- package/dist/eslint/rules/no-classnames-package.d.ts +3 -0
- package/dist/eslint/rules/no-classnames-package.js +46 -0
- package/dist/eslint/rules/no-inline-svg.d.ts +3 -0
- package/dist/eslint/rules/no-inline-svg.js +37 -0
- package/dist/eslint/rules/no-raw-colors.d.ts +3 -0
- package/dist/eslint/rules/no-raw-colors.js +139 -0
- package/dist/eslint/rules/prefer-alias-imports.d.ts +3 -0
- package/dist/eslint/rules/prefer-alias-imports.js +105 -0
- package/dist/eslint/rules/prefer-list-components.js +3 -3
- package/dist/eslint/rules/prefer-typography-components.js +7 -1
- package/dist/eslint/rules/require-canon-setup.js +5 -1
- package/dist/index.d.ts +1 -0
- package/dist/index.js +1 -1
- package/dist/types/config.d.ts +36 -0
- package/dist/types/config.js +2 -0
- package/package.json +5 -1
- package/patterns/007-import-paths.md +4 -3
- package/patterns/009-color-tokens.md +4 -3
- package/patterns/012-icon-system.md +4 -3
- package/patterns/014-clsx-not-classnames.md +3 -3
- package/schema.json +8 -8
|
@@ -0,0 +1,105 @@
|
|
|
1
|
+
import { getCanonUrl, getCanonPattern } from '../utils/canon.js';
|
|
2
|
+
const RULE_NAME = 'prefer-alias-imports';
|
|
3
|
+
const pattern = getCanonPattern(RULE_NAME);
|
|
4
|
+
const DEFAULT_ZONES = [
|
|
5
|
+
'components',
|
|
6
|
+
'blocks',
|
|
7
|
+
'hooks',
|
|
8
|
+
'utils',
|
|
9
|
+
'tools',
|
|
10
|
+
'template',
|
|
11
|
+
'types',
|
|
12
|
+
'styles',
|
|
13
|
+
];
|
|
14
|
+
/**
|
|
15
|
+
* Count how many "../" segments a relative path starts with
|
|
16
|
+
*/
|
|
17
|
+
function countParentSegments(importPath) {
|
|
18
|
+
const matches = importPath.match(/^\.\.\//g) || importPath.match(/^(\.\.\/)+/);
|
|
19
|
+
if (!matches)
|
|
20
|
+
return 0;
|
|
21
|
+
// Count occurrences of ../ at the start
|
|
22
|
+
let count = 0;
|
|
23
|
+
let remaining = importPath;
|
|
24
|
+
while (remaining.startsWith('../')) {
|
|
25
|
+
count++;
|
|
26
|
+
remaining = remaining.slice(3);
|
|
27
|
+
}
|
|
28
|
+
return count;
|
|
29
|
+
}
|
|
30
|
+
/**
|
|
31
|
+
* Check if an import path targets a Canon zone
|
|
32
|
+
*/
|
|
33
|
+
function getTargetZone(importPath, zones) {
|
|
34
|
+
// Walk past all ../ segments and check if the next segment is a zone
|
|
35
|
+
let remaining = importPath;
|
|
36
|
+
while (remaining.startsWith('../')) {
|
|
37
|
+
remaining = remaining.slice(3);
|
|
38
|
+
}
|
|
39
|
+
const firstSegment = remaining.split('/')[0];
|
|
40
|
+
if (zones.includes(firstSegment)) {
|
|
41
|
+
return firstSegment;
|
|
42
|
+
}
|
|
43
|
+
return null;
|
|
44
|
+
}
|
|
45
|
+
const rule = {
|
|
46
|
+
meta: {
|
|
47
|
+
type: 'suggestion',
|
|
48
|
+
docs: {
|
|
49
|
+
description: pattern?.summary || 'Use @/ alias imports instead of deep relative paths',
|
|
50
|
+
recommended: true,
|
|
51
|
+
url: getCanonUrl(RULE_NAME),
|
|
52
|
+
},
|
|
53
|
+
messages: {
|
|
54
|
+
useAlias: `[Canon ${pattern?.id || '007'}] Use "{{alias}}{{zone}}/..." instead of "{{importPath}}".`,
|
|
55
|
+
},
|
|
56
|
+
schema: [
|
|
57
|
+
{
|
|
58
|
+
type: 'object',
|
|
59
|
+
properties: {
|
|
60
|
+
alias: {
|
|
61
|
+
type: 'string',
|
|
62
|
+
},
|
|
63
|
+
zones: {
|
|
64
|
+
type: 'array',
|
|
65
|
+
items: { type: 'string' },
|
|
66
|
+
},
|
|
67
|
+
},
|
|
68
|
+
additionalProperties: false,
|
|
69
|
+
},
|
|
70
|
+
],
|
|
71
|
+
},
|
|
72
|
+
create(context) {
|
|
73
|
+
const options = context.options[0] || {};
|
|
74
|
+
const alias = options.alias || '@/';
|
|
75
|
+
const zones = options.zones || DEFAULT_ZONES;
|
|
76
|
+
return {
|
|
77
|
+
ImportDeclaration(node) {
|
|
78
|
+
const importPath = node.source?.value;
|
|
79
|
+
if (typeof importPath !== 'string')
|
|
80
|
+
return;
|
|
81
|
+
// Only check relative imports
|
|
82
|
+
if (!importPath.startsWith('.'))
|
|
83
|
+
return;
|
|
84
|
+
// Same-directory imports are fine
|
|
85
|
+
if (importPath.startsWith('./'))
|
|
86
|
+
return;
|
|
87
|
+
// Check if 2+ parent segments
|
|
88
|
+
const parentCount = countParentSegments(importPath);
|
|
89
|
+
if (parentCount < 2)
|
|
90
|
+
return;
|
|
91
|
+
// Check if targeting a Canon zone
|
|
92
|
+
const zone = getTargetZone(importPath, zones);
|
|
93
|
+
if (zone) {
|
|
94
|
+
context.report({
|
|
95
|
+
node,
|
|
96
|
+
messageId: 'useAlias',
|
|
97
|
+
data: { alias, zone, importPath },
|
|
98
|
+
});
|
|
99
|
+
}
|
|
100
|
+
},
|
|
101
|
+
};
|
|
102
|
+
},
|
|
103
|
+
};
|
|
104
|
+
export default rule;
|
|
105
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJlZmVyLWFsaWFzLWltcG9ydHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvZXNsaW50L3J1bGVzL3ByZWZlci1hbGlhcy1pbXBvcnRzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUNBLE9BQU8sRUFBRSxXQUFXLEVBQUUsZUFBZSxFQUFFLE1BQU0sbUJBQW1CLENBQUE7QUFFaEUsTUFBTSxTQUFTLEdBQUcsc0JBQXNCLENBQUE7QUFDeEMsTUFBTSxPQUFPLEdBQUcsZUFBZSxDQUFDLFNBQVMsQ0FBQyxDQUFBO0FBRTFDLE1BQU0sYUFBYSxHQUFHO0lBQ3BCLFlBQVk7SUFDWixRQUFRO0lBQ1IsT0FBTztJQUNQLE9BQU87SUFDUCxPQUFPO0lBQ1AsVUFBVTtJQUNWLE9BQU87SUFDUCxRQUFRO0NBQ1QsQ0FBQTtBQUVEOztHQUVHO0FBQ0gsU0FBUyxtQkFBbUIsQ0FBQyxVQUFrQjtJQUM3QyxNQUFNLE9BQU8sR0FBRyxVQUFVLENBQUMsS0FBSyxDQUFDLFVBQVUsQ0FBQyxJQUFJLFVBQVUsQ0FBQyxLQUFLLENBQUMsWUFBWSxDQUFDLENBQUE7SUFDOUUsSUFBSSxDQUFDLE9BQU87UUFBRSxPQUFPLENBQUMsQ0FBQTtJQUN0Qix3Q0FBd0M7SUFDeEMsSUFBSSxLQUFLLEdBQUcsQ0FBQyxDQUFBO0lBQ2IsSUFBSSxTQUFTLEdBQUcsVUFBVSxDQUFBO0lBQzFCLE9BQU8sU0FBUyxDQUFDLFVBQVUsQ0FBQyxLQUFLLENBQUMsRUFBRSxDQUFDO1FBQ25DLEtBQUssRUFBRSxDQUFBO1FBQ1AsU0FBUyxHQUFHLFNBQVMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUE7SUFDaEMsQ0FBQztJQUNELE9BQU8sS0FBSyxDQUFBO0FBQ2QsQ0FBQztBQUVEOztHQUVHO0FBQ0gsU0FBUyxhQUFhLENBQUMsVUFBa0IsRUFBRSxLQUFlO0lBQ3hELHFFQUFxRTtJQUNyRSxJQUFJLFNBQVMsR0FBRyxVQUFVLENBQUE7SUFDMUIsT0FBTyxTQUFTLENBQUMsVUFBVSxDQUFDLEtBQUssQ0FBQyxFQUFFLENBQUM7UUFDbkMsU0FBUyxHQUFHLFNBQVMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUE7SUFDaEMsQ0FBQztJQUNELE1BQU0sWUFBWSxHQUFHLFNBQVMsQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUE7SUFDNUMsSUFBSSxLQUFLLENBQUMsUUFBUSxDQUFDLFlBQVksQ0FBQyxFQUFFLENBQUM7UUFDakMsT0FBTyxZQUFZLENBQUE7SUFDckIsQ0FBQztJQUNELE9BQU8sSUFBSSxDQUFBO0FBQ2IsQ0FBQztBQUVELE1BQU0sSUFBSSxHQUFvQjtJQUM1QixJQUFJLEVBQUU7UUFDSixJQUFJLEVBQUUsWUFBWTtRQUNsQixJQUFJLEVBQUU7WUFDSixXQUFXLEVBQUUsT0FBTyxFQUFFLE9BQU8sSUFBSSxxREFBcUQ7WUFDdEYsV0FBVyxFQUFFLElBQUk7WUFDakIsR0FBRyxFQUFFLFdBQVcsQ0FBQyxTQUFTLENBQUM7U0FDNUI7UUFDRCxRQUFRLEVBQUU7WUFDUixRQUFRLEVBQUUsVUFBVSxPQUFPLEVBQUUsRUFBRSxJQUFJLEtBQUssNERBQTREO1NBQ3JHO1FBQ0QsTUFBTSxFQUFFO1lBQ047Z0JBQ0UsSUFBSSxFQUFFLFFBQVE7Z0JBQ2QsVUFBVSxFQUFFO29CQUNWLEtBQUssRUFBRTt3QkFDTCxJQUFJLEVBQUUsUUFBUTtxQkFDZjtvQkFDRCxLQUFLLEVBQUU7d0JBQ0wsSUFBSSxFQUFFLE9BQU87d0JBQ2IsS0FBSyxFQUFFLEVBQUUsSUFBSSxFQUFFLFFBQVEsRUFBRTtxQkFDMUI7aUJBQ0Y7Z0JBQ0Qsb0JBQW9CLEVBQUUsS0FBSzthQUM1QjtTQUNGO0tBQ0Y7SUFFRCxNQUFNLENBQUMsT0FBTztRQUNaLE1BQU0sT0FBTyxHQUFHLE9BQU8sQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLElBQUksRUFBRSxDQUFBO1FBQ3hDLE1BQU0sS0FBSyxHQUFHLE9BQU8sQ0FBQyxLQUFLLElBQUksSUFBSSxDQUFBO1FBQ25DLE1BQU0sS0FBSyxHQUFHLE9BQU8sQ0FBQyxLQUFLLElBQUksYUFBYSxDQUFBO1FBRTVDLE9BQU87WUFDTCxpQkFBaUIsQ0FBQyxJQUFTO2dCQUN6QixNQUFNLFVBQVUsR0FBRyxJQUFJLENBQUMsTUFBTSxFQUFFLEtBQUssQ0FBQTtnQkFDckMsSUFBSSxPQUFPLFVBQVUsS0FBSyxRQUFRO29CQUFFLE9BQU07Z0JBRTFDLDhCQUE4QjtnQkFDOUIsSUFBSSxDQUFDLFVBQVUsQ0FBQyxVQUFVLENBQUMsR0FBRyxDQUFDO29CQUFFLE9BQU07Z0JBRXZDLGtDQUFrQztnQkFDbEMsSUFBSSxVQUFVLENBQUMsVUFBVSxDQUFDLElBQUksQ0FBQztvQkFBRSxPQUFNO2dCQUV2Qyw4QkFBOEI7Z0JBQzlCLE1BQU0sV0FBVyxHQUFHLG1CQUFtQixDQUFDLFVBQVUsQ0FBQyxDQUFBO2dCQUNuRCxJQUFJLFdBQVcsR0FBRyxDQUFDO29CQUFFLE9BQU07Z0JBRTNCLGtDQUFrQztnQkFDbEMsTUFBTSxJQUFJLEdBQUcsYUFBYSxDQUFDLFVBQVUsRUFBRSxLQUFLLENBQUMsQ0FBQTtnQkFDN0MsSUFBSSxJQUFJLEVBQUUsQ0FBQztvQkFDVCxPQUFPLENBQUMsTUFBTSxDQUFDO3dCQUNiLElBQUk7d0JBQ0osU0FBUyxFQUFFLFVBQVU7d0JBQ3JCLElBQUksRUFBRSxFQUFFLEtBQUssRUFBRSxJQUFJLEVBQUUsVUFBVSxFQUFFO3FCQUNsQyxDQUFDLENBQUE7Z0JBQ0osQ0FBQztZQUNILENBQUM7U0FDRixDQUFBO0lBQ0gsQ0FBQztDQUNGLENBQUE7QUFFRCxlQUFlLElBQUksQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIHsgUnVsZSB9IGZyb20gJ2VzbGludCdcbmltcG9ydCB7IGdldENhbm9uVXJsLCBnZXRDYW5vblBhdHRlcm4gfSBmcm9tICcuLi91dGlscy9jYW5vbi5qcydcblxuY29uc3QgUlVMRV9OQU1FID0gJ3ByZWZlci1hbGlhcy1pbXBvcnRzJ1xuY29uc3QgcGF0dGVybiA9IGdldENhbm9uUGF0dGVybihSVUxFX05BTUUpXG5cbmNvbnN0IERFRkFVTFRfWk9ORVMgPSBbXG4gICdjb21wb25lbnRzJyxcbiAgJ2Jsb2NrcycsXG4gICdob29rcycsXG4gICd1dGlscycsXG4gICd0b29scycsXG4gICd0ZW1wbGF0ZScsXG4gICd0eXBlcycsXG4gICdzdHlsZXMnLFxuXVxuXG4vKipcbiAqIENvdW50IGhvdyBtYW55IFwiLi4vXCIgc2VnbWVudHMgYSByZWxhdGl2ZSBwYXRoIHN0YXJ0cyB3aXRoXG4gKi9cbmZ1bmN0aW9uIGNvdW50UGFyZW50U2VnbWVudHMoaW1wb3J0UGF0aDogc3RyaW5nKTogbnVtYmVyIHtcbiAgY29uc3QgbWF0Y2hlcyA9IGltcG9ydFBhdGgubWF0Y2goL15cXC5cXC5cXC8vZykgfHwgaW1wb3J0UGF0aC5tYXRjaCgvXihcXC5cXC5cXC8pKy8pXG4gIGlmICghbWF0Y2hlcykgcmV0dXJuIDBcbiAgLy8gQ291bnQgb2NjdXJyZW5jZXMgb2YgLi4vIGF0IHRoZSBzdGFydFxuICBsZXQgY291bnQgPSAwXG4gIGxldCByZW1haW5pbmcgPSBpbXBvcnRQYXRoXG4gIHdoaWxlIChyZW1haW5pbmcuc3RhcnRzV2l0aCgnLi4vJykpIHtcbiAgICBjb3VudCsrXG4gICAgcmVtYWluaW5nID0gcmVtYWluaW5nLnNsaWNlKDMpXG4gIH1cbiAgcmV0dXJuIGNvdW50XG59XG5cbi8qKlxuICogQ2hlY2sgaWYgYW4gaW1wb3J0IHBhdGggdGFyZ2V0cyBhIENhbm9uIHpvbmVcbiAqL1xuZnVuY3Rpb24gZ2V0VGFyZ2V0Wm9uZShpbXBvcnRQYXRoOiBzdHJpbmcsIHpvbmVzOiBzdHJpbmdbXSk6IHN0cmluZyB8IG51bGwge1xuICAvLyBXYWxrIHBhc3QgYWxsIC4uLyBzZWdtZW50cyBhbmQgY2hlY2sgaWYgdGhlIG5leHQgc2VnbWVudCBpcyBhIHpvbmVcbiAgbGV0IHJlbWFpbmluZyA9IGltcG9ydFBhdGhcbiAgd2hpbGUgKHJlbWFpbmluZy5zdGFydHNXaXRoKCcuLi8nKSkge1xuICAgIHJlbWFpbmluZyA9IHJlbWFpbmluZy5zbGljZSgzKVxuICB9XG4gIGNvbnN0IGZpcnN0U2VnbWVudCA9IHJlbWFpbmluZy5zcGxpdCgnLycpWzBdXG4gIGlmICh6b25lcy5pbmNsdWRlcyhmaXJzdFNlZ21lbnQpKSB7XG4gICAgcmV0dXJuIGZpcnN0U2VnbWVudFxuICB9XG4gIHJldHVybiBudWxsXG59XG5cbmNvbnN0IHJ1bGU6IFJ1bGUuUnVsZU1vZHVsZSA9IHtcbiAgbWV0YToge1xuICAgIHR5cGU6ICdzdWdnZXN0aW9uJyxcbiAgICBkb2NzOiB7XG4gICAgICBkZXNjcmlwdGlvbjogcGF0dGVybj8uc3VtbWFyeSB8fCAnVXNlIEAvIGFsaWFzIGltcG9ydHMgaW5zdGVhZCBvZiBkZWVwIHJlbGF0aXZlIHBhdGhzJyxcbiAgICAgIHJlY29tbWVuZGVkOiB0cnVlLFxuICAgICAgdXJsOiBnZXRDYW5vblVybChSVUxFX05BTUUpLFxuICAgIH0sXG4gICAgbWVzc2FnZXM6IHtcbiAgICAgIHVzZUFsaWFzOiBgW0Nhbm9uICR7cGF0dGVybj8uaWQgfHwgJzAwNyd9XSBVc2UgXCJ7e2FsaWFzfX17e3pvbmV9fS8uLi5cIiBpbnN0ZWFkIG9mIFwie3tpbXBvcnRQYXRofX1cIi5gLFxuICAgIH0sXG4gICAgc2NoZW1hOiBbXG4gICAgICB7XG4gICAgICAgIHR5cGU6ICdvYmplY3QnLFxuICAgICAgICBwcm9wZXJ0aWVzOiB7XG4gICAgICAgICAgYWxpYXM6IHtcbiAgICAgICAgICAgIHR5cGU6ICdzdHJpbmcnLFxuICAgICAgICAgIH0sXG4gICAgICAgICAgem9uZXM6IHtcbiAgICAgICAgICAgIHR5cGU6ICdhcnJheScsXG4gICAgICAgICAgICBpdGVtczogeyB0eXBlOiAnc3RyaW5nJyB9LFxuICAgICAgICAgIH0sXG4gICAgICAgIH0sXG4gICAgICAgIGFkZGl0aW9uYWxQcm9wZXJ0aWVzOiBmYWxzZSxcbiAgICAgIH0sXG4gICAgXSxcbiAgfSxcblxuICBjcmVhdGUoY29udGV4dCkge1xuICAgIGNvbnN0IG9wdGlvbnMgPSBjb250ZXh0Lm9wdGlvbnNbMF0gfHwge31cbiAgICBjb25zdCBhbGlhcyA9IG9wdGlvbnMuYWxpYXMgfHwgJ0AvJ1xuICAgIGNvbnN0IHpvbmVzID0gb3B0aW9ucy56b25lcyB8fCBERUZBVUxUX1pPTkVTXG5cbiAgICByZXR1cm4ge1xuICAgICAgSW1wb3J0RGVjbGFyYXRpb24obm9kZTogYW55KSB7XG4gICAgICAgIGNvbnN0IGltcG9ydFBhdGggPSBub2RlLnNvdXJjZT8udmFsdWVcbiAgICAgICAgaWYgKHR5cGVvZiBpbXBvcnRQYXRoICE9PSAnc3RyaW5nJykgcmV0dXJuXG5cbiAgICAgICAgLy8gT25seSBjaGVjayByZWxhdGl2ZSBpbXBvcnRzXG4gICAgICAgIGlmICghaW1wb3J0UGF0aC5zdGFydHNXaXRoKCcuJykpIHJldHVyblxuXG4gICAgICAgIC8vIFNhbWUtZGlyZWN0b3J5IGltcG9ydHMgYXJlIGZpbmVcbiAgICAgICAgaWYgKGltcG9ydFBhdGguc3RhcnRzV2l0aCgnLi8nKSkgcmV0dXJuXG5cbiAgICAgICAgLy8gQ2hlY2sgaWYgMisgcGFyZW50IHNlZ21lbnRzXG4gICAgICAgIGNvbnN0IHBhcmVudENvdW50ID0gY291bnRQYXJlbnRTZWdtZW50cyhpbXBvcnRQYXRoKVxuICAgICAgICBpZiAocGFyZW50Q291bnQgPCAyKSByZXR1cm5cblxuICAgICAgICAvLyBDaGVjayBpZiB0YXJnZXRpbmcgYSBDYW5vbiB6b25lXG4gICAgICAgIGNvbnN0IHpvbmUgPSBnZXRUYXJnZXRab25lKGltcG9ydFBhdGgsIHpvbmVzKVxuICAgICAgICBpZiAoem9uZSkge1xuICAgICAgICAgIGNvbnRleHQucmVwb3J0KHtcbiAgICAgICAgICAgIG5vZGUsXG4gICAgICAgICAgICBtZXNzYWdlSWQ6ICd1c2VBbGlhcycsXG4gICAgICAgICAgICBkYXRhOiB7IGFsaWFzLCB6b25lLCBpbXBvcnRQYXRoIH0sXG4gICAgICAgICAgfSlcbiAgICAgICAgfVxuICAgICAgfSxcbiAgICB9XG4gIH0sXG59XG5cbmV4cG9ydCBkZWZhdWx0IHJ1bGVcbiJdfQ==
|
|
@@ -24,8 +24,8 @@ const rule = {
|
|
|
24
24
|
return {
|
|
25
25
|
JSXOpeningElement(node) {
|
|
26
26
|
const elementName = node.name?.name;
|
|
27
|
-
// Check <ul> tags
|
|
28
|
-
if (elementName === 'ul') {
|
|
27
|
+
// Check <ul> and <ol> tags
|
|
28
|
+
if (elementName === 'ul' || elementName === 'ol') {
|
|
29
29
|
context.report({
|
|
30
30
|
node,
|
|
31
31
|
messageId: 'useList',
|
|
@@ -45,4 +45,4 @@ const rule = {
|
|
|
45
45
|
},
|
|
46
46
|
};
|
|
47
47
|
export default rule;
|
|
48
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
48
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJlZmVyLWxpc3QtY29tcG9uZW50cy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9lc2xpbnQvcnVsZXMvcHJlZmVyLWxpc3QtY29tcG9uZW50cy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFDQSxPQUFPLEVBQUUsV0FBVyxFQUFFLGVBQWUsRUFBRSxNQUFNLG1CQUFtQixDQUFBO0FBRWhFLE1BQU0sU0FBUyxHQUFHLHdCQUF3QixDQUFBO0FBQzFDLE1BQU0sT0FBTyxHQUFHLGVBQWUsQ0FBQyxTQUFTLENBQUMsQ0FBQTtBQUUxQyxNQUFNLElBQUksR0FBb0I7SUFDNUIsSUFBSSxFQUFFO1FBQ0osSUFBSSxFQUFFLFlBQVk7UUFDbEIsSUFBSSxFQUFFO1lBQ0osV0FBVyxFQUFFLE9BQU8sRUFBRSxPQUFPLElBQUksaUNBQWlDO1lBQ2xFLFdBQVcsRUFBRSxJQUFJO1lBQ2pCLEdBQUcsRUFBRSxXQUFXLENBQUMsU0FBUyxDQUFDO1NBQzVCO1FBQ0QsUUFBUSxFQUFFO1lBQ1IsT0FBTyxFQUFFLFVBQVUsT0FBTyxFQUFFLEVBQUUsSUFBSSxLQUFLLDRGQUE0RjtZQUNuSSxLQUFLLEVBQUUsVUFBVSxPQUFPLEVBQUUsRUFBRSxJQUFJLEtBQUssd0ZBQXdGO1NBQzlIO1FBQ0QsTUFBTSxFQUFFLEVBQUU7S0FDWDtJQUVELE1BQU0sQ0FBQyxPQUFPO1FBQ1osTUFBTSxRQUFRLEdBQUcsT0FBTyxDQUFDLFFBQVEsSUFBSSxPQUFPLENBQUMsV0FBVyxFQUFFLENBQUE7UUFFMUQsNEJBQTRCO1FBQzVCLElBQUksQ0FBQyxRQUFRLENBQUMsUUFBUSxDQUFDLFVBQVUsQ0FBQyxFQUFFLENBQUM7WUFDbkMsT0FBTyxFQUFFLENBQUE7UUFDWCxDQUFDO1FBRUQsT0FBTztZQUNMLGlCQUFpQixDQUFDLElBQVM7Z0JBQ3pCLE1BQU0sV0FBVyxHQUFHLElBQUksQ0FBQyxJQUFJLEVBQUUsSUFBSSxDQUFBO2dCQUVuQywyQkFBMkI7Z0JBQzNCLElBQUksV0FBVyxLQUFLLElBQUksSUFBSSxXQUFXLEtBQUssSUFBSSxFQUFFLENBQUM7b0JBQ2pELE9BQU8sQ0FBQyxNQUFNLENBQUM7d0JBQ2IsSUFBSTt3QkFDSixTQUFTLEVBQUUsU0FBUztxQkFDckIsQ0FBQyxDQUFBO29CQUNGLE9BQU07Z0JBQ1IsQ0FBQztnQkFFRCxrQkFBa0I7Z0JBQ2xCLElBQUksV0FBVyxLQUFLLElBQUksRUFBRSxDQUFDO29CQUN6QixPQUFPLENBQUMsTUFBTSxDQUFDO3dCQUNiLElBQUk7d0JBQ0osU0FBUyxFQUFFLE9BQU87cUJBQ25CLENBQUMsQ0FBQTtvQkFDRixPQUFNO2dCQUNSLENBQUM7WUFDSCxDQUFDO1NBQ0YsQ0FBQTtJQUNILENBQUM7Q0FDRixDQUFBO0FBRUQsZUFBZSxJQUFJLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgdHlwZSB7IFJ1bGUgfSBmcm9tICdlc2xpbnQnXG5pbXBvcnQgeyBnZXRDYW5vblVybCwgZ2V0Q2Fub25QYXR0ZXJuIH0gZnJvbSAnLi4vdXRpbHMvY2Fub24uanMnXG5cbmNvbnN0IFJVTEVfTkFNRSA9ICdwcmVmZXItbGlzdC1jb21wb25lbnRzJ1xuY29uc3QgcGF0dGVybiA9IGdldENhbm9uUGF0dGVybihSVUxFX05BTUUpXG5cbmNvbnN0IHJ1bGU6IFJ1bGUuUnVsZU1vZHVsZSA9IHtcbiAgbWV0YToge1xuICAgIHR5cGU6ICdzdWdnZXN0aW9uJyxcbiAgICBkb2NzOiB7XG4gICAgICBkZXNjcmlwdGlvbjogcGF0dGVybj8uc3VtbWFyeSB8fCAnVXNlIExpc3QvTGksIG5vdCByYXcgdWwvbGkgdGFncycsXG4gICAgICByZWNvbW1lbmRlZDogdHJ1ZSxcbiAgICAgIHVybDogZ2V0Q2Fub25VcmwoUlVMRV9OQU1FKSxcbiAgICB9LFxuICAgIG1lc3NhZ2VzOiB7XG4gICAgICB1c2VMaXN0OiBgW0Nhbm9uICR7cGF0dGVybj8uaWQgfHwgJzAyNid9XSBVc2UgdGhlIExpc3QgY29tcG9uZW50IGluc3RlYWQgb2YgPHVsPi4gSW1wb3J0OiBpbXBvcnQgeyBMaXN0IH0gZnJvbSBcIkAvY29tcG9uZW50cy9saXN0XCJgLFxuICAgICAgdXNlTGk6IGBbQ2Fub24gJHtwYXR0ZXJuPy5pZCB8fCAnMDI2J31dIFVzZSB0aGUgTGkgY29tcG9uZW50IGluc3RlYWQgb2YgPGxpPi4gSW1wb3J0OiBpbXBvcnQgeyBMaSB9IGZyb20gXCJAL2NvbXBvbmVudHMvbGlzdFwiYCxcbiAgICB9LFxuICAgIHNjaGVtYTogW10sXG4gIH0sXG5cbiAgY3JlYXRlKGNvbnRleHQpIHtcbiAgICBjb25zdCBmaWxlbmFtZSA9IGNvbnRleHQuZmlsZW5hbWUgfHwgY29udGV4dC5nZXRGaWxlbmFtZSgpXG5cbiAgICAvLyBPbmx5IGFwcGx5IHRvIGJsb2NrIGZpbGVzXG4gICAgaWYgKCFmaWxlbmFtZS5pbmNsdWRlcygnL2Jsb2Nrcy8nKSkge1xuICAgICAgcmV0dXJuIHt9XG4gICAgfVxuXG4gICAgcmV0dXJuIHtcbiAgICAgIEpTWE9wZW5pbmdFbGVtZW50KG5vZGU6IGFueSkge1xuICAgICAgICBjb25zdCBlbGVtZW50TmFtZSA9IG5vZGUubmFtZT8ubmFtZVxuXG4gICAgICAgIC8vIENoZWNrIDx1bD4gYW5kIDxvbD4gdGFnc1xuICAgICAgICBpZiAoZWxlbWVudE5hbWUgPT09ICd1bCcgfHwgZWxlbWVudE5hbWUgPT09ICdvbCcpIHtcbiAgICAgICAgICBjb250ZXh0LnJlcG9ydCh7XG4gICAgICAgICAgICBub2RlLFxuICAgICAgICAgICAgbWVzc2FnZUlkOiAndXNlTGlzdCcsXG4gICAgICAgICAgfSlcbiAgICAgICAgICByZXR1cm5cbiAgICAgICAgfVxuXG4gICAgICAgIC8vIENoZWNrIDxsaT4gdGFnc1xuICAgICAgICBpZiAoZWxlbWVudE5hbWUgPT09ICdsaScpIHtcbiAgICAgICAgICBjb250ZXh0LnJlcG9ydCh7XG4gICAgICAgICAgICBub2RlLFxuICAgICAgICAgICAgbWVzc2FnZUlkOiAndXNlTGknLFxuICAgICAgICAgIH0pXG4gICAgICAgICAgcmV0dXJuXG4gICAgICAgIH1cbiAgICAgIH0sXG4gICAgfVxuICB9LFxufVxuXG5leHBvcnQgZGVmYXVsdCBydWxlXG4iXX0=
|
|
@@ -10,6 +10,7 @@ const rule = {
|
|
|
10
10
|
url: getCanonUrl(RULE_NAME),
|
|
11
11
|
},
|
|
12
12
|
messages: {
|
|
13
|
+
useHeading: `[Canon ${pattern?.id || '003'}] Use the Heading component instead of <{{tag}}>. Import: import { Heading } from "@/components/heading"`,
|
|
13
14
|
useParagraph: `[Canon ${pattern?.id || '003'}] Use the Paragraph component instead of <p>. Import: import { Paragraph } from "@/components/paragraph"`,
|
|
14
15
|
useSpan: `[Canon ${pattern?.id || '003'}] Use the Span component instead of <span> for text content. Import: import { Span } from "@/components/span"`,
|
|
15
16
|
useQuote: `[Canon ${pattern?.id || '003'}] Use the Quote component instead of <blockquote>. Import: import { Quote } from "@/components/quote"`,
|
|
@@ -88,6 +89,11 @@ const rule = {
|
|
|
88
89
|
return {
|
|
89
90
|
JSXOpeningElement(node) {
|
|
90
91
|
const elementName = node.name?.name;
|
|
92
|
+
// Check <h1>–<h6> tags
|
|
93
|
+
if (['h1', 'h2', 'h3', 'h4', 'h5', 'h6'].includes(elementName)) {
|
|
94
|
+
context.report({ node, messageId: 'useHeading', data: { tag: elementName } });
|
|
95
|
+
return;
|
|
96
|
+
}
|
|
91
97
|
// Check <p> tags
|
|
92
98
|
if (elementName === 'p') {
|
|
93
99
|
context.report({
|
|
@@ -141,4 +147,4 @@ const rule = {
|
|
|
141
147
|
},
|
|
142
148
|
};
|
|
143
149
|
export default rule;
|
|
144
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"prefer-typography-components.js","sourceRoot":"","sources":["../../../src/eslint/rules/prefer-typography-components.ts"],"names":[],"mappings":"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","sourcesContent":["import type { Rule } from 'eslint'\nimport { getCanonUrl, getCanonPattern } from '../utils/canon.js'\n\nconst RULE_NAME = 'prefer-typography-components'\nconst pattern = getCanonPattern(RULE_NAME)\n\nconst rule: Rule.RuleModule = {\n  meta: {\n    type: 'suggestion',\n    docs: {\n      description: pattern?.summary || 'Use Paragraph/Span, not raw tags',\n      recommended: true,\n      url: getCanonUrl(RULE_NAME),\n    },\n    messages: {\n      useParagraph: `[Canon ${pattern?.id || '003'}] Use the Paragraph component instead of <p>. Import: import { Paragraph } from \"@/components/paragraph\"`,\n      useSpan: `[Canon ${pattern?.id || '003'}] Use the Span component instead of <span> for text content. Import: import { Span } from \"@/components/span\"`,\n      useQuote: `[Canon ${pattern?.id || '003'}] Use the Quote component instead of <blockquote>. Import: import { Quote } from \"@/components/quote\"`,\n      useTypographyForDiv: `[Canon ${pattern?.id || '003'}] Use a typography component (Heading, Paragraph, Label, etc.) instead of <div> with text content.`,\n    },\n    schema: [],\n  },\n\n  create(context) {\n    const filename = context.filename || context.getFilename()\n\n    // Only apply to block files\n    if (!filename.includes('/blocks/')) {\n      return {}\n    }\n\n    const typographyComponents = ['Heading', 'Paragraph', 'Label', 'Span', 'Quote', 'Subheading', 'Accent']\n\n    /**\n     * Check if element is inside a typography component\n     */\n    function isInsideTypographyComponent(node: any): boolean {\n      let parent = node.parent\n      while (parent) {\n        if (\n          parent.type === 'JSXElement' &&\n          parent.openingElement?.name?.name &&\n          typographyComponents.includes(parent.openingElement.name.name)\n        ) {\n          return true\n        }\n        parent = parent.parent\n      }\n      return false\n    }\n\n    /**\n     * Check if element has direct text content\n     */\n    function hasDirectTextContent(node: any): boolean {\n      const jsxElement = node.parent\n      if (jsxElement?.type !== 'JSXElement') return false\n\n      const children = jsxElement.children || []\n      return children.some((child: any) => {\n        // Check for direct text content\n        if (child.type === 'JSXText') {\n          return child.value.trim().length > 0\n        }\n        // Check for expression with literal string\n        if (child.type === 'JSXExpressionContainer' && child.expression?.type === 'Literal') {\n          return typeof child.expression.value === 'string' && child.expression.value.trim().length > 0\n        }\n        return false\n      })\n    }\n\n    /**\n     * Check className for specific patterns\n     */\n    function getClassNameInfo(node: any): { isGradientText: boolean; isVisualElement: boolean; hasTextClasses: boolean } {\n      let isGradientText = false\n      let isVisualElement = false\n      let hasTextClasses = false\n\n      node.attributes?.forEach((attr: any) => {\n        if (attr.type === 'JSXAttribute' && attr.name?.name === 'className') {\n          const value = attr.value?.value || ''\n          // Skip gradient text (bg-clip-text is used for gradient text effects)\n          if (/\\bbg-clip-text\\b/.test(value)) {\n            isGradientText = true\n          }\n          // Visual elements (dots, decorative elements with w-/h- but no text classes)\n          if (/\\b(w-\\d|h-\\d|rounded-full)\\b/.test(value) && !/\\btext-/.test(value)) {\n            isVisualElement = true\n          }\n          // Has text-related classes\n          if (/\\b(text-|font-|leading-|tracking-)/.test(value)) {\n            hasTextClasses = true\n          }\n        }\n      })\n\n      return { isGradientText, isVisualElement, hasTextClasses }\n    }\n\n    return {\n      JSXOpeningElement(node: any) {\n        const elementName = node.name?.name\n\n        // Check <p> tags\n        if (elementName === 'p') {\n          context.report({\n            node,\n            messageId: 'useParagraph',\n          })\n          return\n        }\n\n        // Check <blockquote> tags\n        if (elementName === 'blockquote') {\n          context.report({\n            node,\n            messageId: 'useQuote',\n          })\n          return\n        }\n\n        // Check <span> tags\n        if (elementName === 'span') {\n          if (isInsideTypographyComponent(node)) {\n            return\n          }\n\n          const { isGradientText, isVisualElement } = getClassNameInfo(node)\n\n          if (isGradientText || isVisualElement) {\n            return\n          }\n\n          if (hasDirectTextContent(node)) {\n            context.report({\n              node,\n              messageId: 'useSpan',\n            })\n          }\n          return\n        }\n\n        // Check <div> tags with text content and text styling\n        if (elementName === 'div') {\n          if (isInsideTypographyComponent(node)) {\n            return\n          }\n\n          const { hasTextClasses } = getClassNameInfo(node)\n\n          // Only flag divs that have both text content AND text-related classes\n          // This indicates it's being used for typography rather than layout\n          if (hasTextClasses && hasDirectTextContent(node)) {\n            context.report({\n              node,\n              messageId: 'useTypographyForDiv',\n            })\n          }\n        }\n      },\n    }\n  },\n}\n\nexport default rule\n"]}
|
|
150
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"prefer-typography-components.js","sourceRoot":"","sources":["../../../src/eslint/rules/prefer-typography-components.ts"],"names":[],"mappings":"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","sourcesContent":["import type { Rule } from 'eslint'\nimport { getCanonUrl, getCanonPattern } from '../utils/canon.js'\n\nconst RULE_NAME = 'prefer-typography-components'\nconst pattern = getCanonPattern(RULE_NAME)\n\nconst rule: Rule.RuleModule = {\n  meta: {\n    type: 'suggestion',\n    docs: {\n      description: pattern?.summary || 'Use Paragraph/Span, not raw tags',\n      recommended: true,\n      url: getCanonUrl(RULE_NAME),\n    },\n    messages: {\n      useHeading: `[Canon ${pattern?.id || '003'}] Use the Heading component instead of <{{tag}}>. Import: import { Heading } from \"@/components/heading\"`,\n      useParagraph: `[Canon ${pattern?.id || '003'}] Use the Paragraph component instead of <p>. Import: import { Paragraph } from \"@/components/paragraph\"`,\n      useSpan: `[Canon ${pattern?.id || '003'}] Use the Span component instead of <span> for text content. Import: import { Span } from \"@/components/span\"`,\n      useQuote: `[Canon ${pattern?.id || '003'}] Use the Quote component instead of <blockquote>. Import: import { Quote } from \"@/components/quote\"`,\n      useTypographyForDiv: `[Canon ${pattern?.id || '003'}] Use a typography component (Heading, Paragraph, Label, etc.) instead of <div> with text content.`,\n    },\n    schema: [],\n  },\n\n  create(context) {\n    const filename = context.filename || context.getFilename()\n\n    // Only apply to block files\n    if (!filename.includes('/blocks/')) {\n      return {}\n    }\n\n    const typographyComponents = ['Heading', 'Paragraph', 'Label', 'Span', 'Quote', 'Subheading', 'Accent']\n\n    /**\n     * Check if element is inside a typography component\n     */\n    function isInsideTypographyComponent(node: any): boolean {\n      let parent = node.parent\n      while (parent) {\n        if (\n          parent.type === 'JSXElement' &&\n          parent.openingElement?.name?.name &&\n          typographyComponents.includes(parent.openingElement.name.name)\n        ) {\n          return true\n        }\n        parent = parent.parent\n      }\n      return false\n    }\n\n    /**\n     * Check if element has direct text content\n     */\n    function hasDirectTextContent(node: any): boolean {\n      const jsxElement = node.parent\n      if (jsxElement?.type !== 'JSXElement') return false\n\n      const children = jsxElement.children || []\n      return children.some((child: any) => {\n        // Check for direct text content\n        if (child.type === 'JSXText') {\n          return child.value.trim().length > 0\n        }\n        // Check for expression with literal string\n        if (child.type === 'JSXExpressionContainer' && child.expression?.type === 'Literal') {\n          return typeof child.expression.value === 'string' && child.expression.value.trim().length > 0\n        }\n        return false\n      })\n    }\n\n    /**\n     * Check className for specific patterns\n     */\n    function getClassNameInfo(node: any): { isGradientText: boolean; isVisualElement: boolean; hasTextClasses: boolean } {\n      let isGradientText = false\n      let isVisualElement = false\n      let hasTextClasses = false\n\n      node.attributes?.forEach((attr: any) => {\n        if (attr.type === 'JSXAttribute' && attr.name?.name === 'className') {\n          const value = attr.value?.value || ''\n          // Skip gradient text (bg-clip-text is used for gradient text effects)\n          if (/\\bbg-clip-text\\b/.test(value)) {\n            isGradientText = true\n          }\n          // Visual elements (dots, decorative elements with w-/h- but no text classes)\n          if (/\\b(w-\\d|h-\\d|rounded-full)\\b/.test(value) && !/\\btext-/.test(value)) {\n            isVisualElement = true\n          }\n          // Has text-related classes\n          if (/\\b(text-|font-|leading-|tracking-)/.test(value)) {\n            hasTextClasses = true\n          }\n        }\n      })\n\n      return { isGradientText, isVisualElement, hasTextClasses }\n    }\n\n    return {\n      JSXOpeningElement(node: any) {\n        const elementName = node.name?.name\n\n        // Check <h1>–<h6> tags\n        if (['h1', 'h2', 'h3', 'h4', 'h5', 'h6'].includes(elementName)) {\n          context.report({ node, messageId: 'useHeading', data: { tag: elementName } })\n          return\n        }\n\n        // Check <p> tags\n        if (elementName === 'p') {\n          context.report({\n            node,\n            messageId: 'useParagraph',\n          })\n          return\n        }\n\n        // Check <blockquote> tags\n        if (elementName === 'blockquote') {\n          context.report({\n            node,\n            messageId: 'useQuote',\n          })\n          return\n        }\n\n        // Check <span> tags\n        if (elementName === 'span') {\n          if (isInsideTypographyComponent(node)) {\n            return\n          }\n\n          const { isGradientText, isVisualElement } = getClassNameInfo(node)\n\n          if (isGradientText || isVisualElement) {\n            return\n          }\n\n          if (hasDirectTextContent(node)) {\n            context.report({\n              node,\n              messageId: 'useSpan',\n            })\n          }\n          return\n        }\n\n        // Check <div> tags with text content and text styling\n        if (elementName === 'div') {\n          if (isInsideTypographyComponent(node)) {\n            return\n          }\n\n          const { hasTextClasses } = getClassNameInfo(node)\n\n          // Only flag divs that have both text content AND text-related classes\n          // This indicates it's being used for typography rather than layout\n          if (hasTextClasses && hasDirectTextContent(node)) {\n            context.report({\n              node,\n              messageId: 'useTypographyForDiv',\n            })\n          }\n        }\n      },\n    }\n  },\n}\n\nexport default rule\n"]}
|
|
@@ -21,6 +21,10 @@ const REQUIRED_SCRIPTS = {
|
|
|
21
21
|
contains: 'eslint',
|
|
22
22
|
definition: '"lint": "eslint src/"',
|
|
23
23
|
},
|
|
24
|
+
'lint:gallop': {
|
|
25
|
+
contains: 'eslint src/blocks/',
|
|
26
|
+
definition: `"lint:gallop": "eslint src/blocks/ --rule 'gallop/no-client-blocks: warn' --rule 'gallop/no-container-in-section: warn' --rule 'gallop/prefer-component-props: warn'"`,
|
|
27
|
+
},
|
|
24
28
|
ts: {
|
|
25
29
|
contains: 'tsc',
|
|
26
30
|
definition: '"ts": "tsc --noEmit"',
|
|
@@ -133,4 +137,4 @@ export function resetReported() {
|
|
|
133
137
|
hasReported = false;
|
|
134
138
|
}
|
|
135
139
|
export default rule;
|
|
136
|
-
//# 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,uHAAuH;AACvH,MAAM,gBAAgB,GAA6D;IACjF,MAAM,EAAE;QACN,QAAQ,EAAE,MAAM;QAChB,UAAU,EAAE,kBAAkB;KAC/B;IACD,KAAK,EAAE;QACL,QAAQ,EAAE,SAAS;QACnB,UAAU,EAAE,yDAAyD;KACtE;IACD,IAAI,EAAE;QACJ,QAAQ,EAAE,QAAQ;QAClB,UAAU,EAAE,uBAAuB;KACpC;IACD,EAAE,EAAE;QACF,QAAQ,EAAE,KAAK;QACf,UAAU,EAAE,sBAAsB;KACnC;IACD,KAAK,EAAE;QACL,QAAQ,EAAE,cAAc;QACxB,UAAU,EAAE,yBAAyB;KACtC;IACD,cAAc,EAAE;QACd,QAAQ,EAAE,cAAc;QACxB,UAAU,EAAE,yCAAyC;KACtD;IACD,YAAY,EAAE;QACZ,QAAQ,EAAE,cAAc;QACxB,UAAU,EAAE,qCAAqC;KAClD;IACD,mBAAmB,EAAE;QACnB,QAAQ,EAAE,iBAAiB;QAC3B,UAAU,EAAE,wGAAwG;KACrH;IACD,cAAc,EAAE;QACd,QAAQ,EAAE,wBAAwB;QAClC,UAAU,EAAE,qDAAqD;KAClE;CACF,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,+FAA+F;YAC9G,aAAa,EAAE,qFAAqF;SACrG;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,EAAE,QAAQ,EAAE,UAAU,EAAE,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,gBAAgB,CAAC,EAAE,CAAC;wBACtF,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,UAAU,EAAE;6BACzC,CAAC,CAAA;wBACJ,CAAC;6BAAM,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC,QAAQ,CAAC,QAAQ,CAAC,EAAE,CAAC;4BACnD,OAAO,CAAC,MAAM,CAAC;gCACb,IAAI;gCACJ,SAAS,EAAE,eAAe;gCAC1B,IAAI,EAAE,EAAE,MAAM,EAAE,UAAU,EAAE,QAAQ,EAAE,QAAQ,EAAE,UAAU,EAAE;6BAC7D,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 = { contains: string to check for, definition: exact script to add })\nconst REQUIRED_SCRIPTS: Record<string, { contains: string; definition: string }> = {\n  unused: {\n    contains: 'knip',\n    definition: '\"unused\": \"knip\"',\n  },\n  check: {\n    contains: 'npm run',\n    definition: '\"check\": \"npm run lint && npm run ts && npm run unused\"',\n  },\n  lint: {\n    contains: 'eslint',\n    definition: '\"lint\": \"eslint src/\"',\n  },\n  ts: {\n    contains: 'tsc',\n    definition: '\"ts\": \"tsc --noEmit\"',\n  },\n  audit: {\n    contains: 'gallop audit',\n    definition: '\"audit\": \"gallop audit\"',\n  },\n  'audit:strict': {\n    contains: 'gallop audit',\n    definition: '\"audit:strict\": \"gallop audit --strict\"',\n  },\n  'audit:json': {\n    contains: 'gallop audit',\n    definition: '\"audit:json\": \"gallop audit --json\"',\n  },\n  'generate:ai-rules': {\n    contains: 'gallop generate',\n    definition: '\"generate:ai-rules\": \"gallop generate .cursorrules && gallop generate .github/copilot-instructions.md\"',\n  },\n  'update:canon': {\n    contains: '@gallop.software/canon',\n    definition: '\"update:canon\": \"npm update @gallop.software/canon\"',\n  },\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: {{definition}}`,\n      invalidScript: `[Canon] Script \"{{script}}\" should contain \"{{expected}}\". Expected: {{definition}}`,\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, { contains, definition }] of Object.entries(REQUIRED_SCRIPTS)) {\n            if (!scripts[scriptName]) {\n              context.report({\n                node,\n                messageId: 'missingScript',\n                data: { script: scriptName, definition },\n              })\n            } else if (!scripts[scriptName].includes(contains)) {\n              context.report({\n                node,\n                messageId: 'invalidScript',\n                data: { script: scriptName, expected: contains, definition },\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"]}
|
|
140
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"require-canon-setup.js","sourceRoot":"","sources":["../../../src/eslint/rules/require-canon-setup.ts"],"names":[],"mappings":"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","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 = { contains: string to check for, definition: exact script to add })\nconst REQUIRED_SCRIPTS: Record<string, { contains: string; definition: string }> = {\n  unused: {\n    contains: 'knip',\n    definition: '\"unused\": \"knip\"',\n  },\n  check: {\n    contains: 'npm run',\n    definition: '\"check\": \"npm run lint && npm run ts && npm run unused\"',\n  },\n  lint: {\n    contains: 'eslint',\n    definition: '\"lint\": \"eslint src/\"',\n  },\n  'lint:gallop': {\n    contains: 'eslint src/blocks/',\n    definition: `\"lint:gallop\": \"eslint src/blocks/ --rule 'gallop/no-client-blocks: warn' --rule 'gallop/no-container-in-section: warn' --rule 'gallop/prefer-component-props: warn'\"`,\n  },\n  ts: {\n    contains: 'tsc',\n    definition: '\"ts\": \"tsc --noEmit\"',\n  },\n  audit: {\n    contains: 'gallop audit',\n    definition: '\"audit\": \"gallop audit\"',\n  },\n  'audit:strict': {\n    contains: 'gallop audit',\n    definition: '\"audit:strict\": \"gallop audit --strict\"',\n  },\n  'audit:json': {\n    contains: 'gallop audit',\n    definition: '\"audit:json\": \"gallop audit --json\"',\n  },\n  'generate:ai-rules': {\n    contains: 'gallop generate',\n    definition: '\"generate:ai-rules\": \"gallop generate .cursorrules && gallop generate .github/copilot-instructions.md\"',\n  },\n  'update:canon': {\n    contains: '@gallop.software/canon',\n    definition: '\"update:canon\": \"npm update @gallop.software/canon\"',\n  },\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: {{definition}}`,\n      invalidScript: `[Canon] Script \"{{script}}\" should contain \"{{expected}}\". Expected: {{definition}}`,\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, { contains, definition }] of Object.entries(REQUIRED_SCRIPTS)) {\n            if (!scripts[scriptName]) {\n              context.report({\n                node,\n                messageId: 'missingScript',\n                data: { script: scriptName, definition },\n              })\n            } else if (!scripts[scriptName].includes(contains)) {\n              context.report({\n                node,\n                messageId: 'invalidScript',\n                data: { script: scriptName, expected: contains, definition },\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/dist/index.d.ts
CHANGED
package/dist/index.js
CHANGED
|
@@ -99,4 +99,4 @@ export default {
|
|
|
99
99
|
isValidPattern,
|
|
100
100
|
getEnforcementStats,
|
|
101
101
|
};
|
|
102
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
102
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAEH,OAAO,MAAM,MAAM,gBAAgB,CAAC,OAAO,IAAI,EAAE,MAAM,EAAE,CAAA;AAiDzD,yBAAyB;AACzB,MAAM,CAAC,MAAM,KAAK,GAAgB,MAAqB,CAAA;AAEvD,iBAAiB;AACjB,MAAM,CAAC,MAAM,OAAO,GAAG,MAAM,CAAC,OAAO,CAAA;AAErC,wBAAwB;AACxB,MAAM,CAAC,MAAM,QAAQ,GAAc,MAAM,CAAC,QAAqB,CAAA;AAE/D,0BAA0B;AAC1B,MAAM,CAAC,MAAM,UAAU,GAAe,MAAM,CAAC,UAAU,CAAA;AAEvD,0BAA0B;AAC1B,MAAM,CAAC,MAAM,UAAU,GAAgB,MAAM,CAAC,UAAyB,CAAA;AAEvE;;;;GAIG;AACH,MAAM,UAAU,UAAU,CAAC,EAAU;IACnC,OAAO,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,EAAE,CAAC,CAAA;AAC1C,CAAC;AAED;;;;GAIG;AACH,MAAM,UAAU,qBAAqB,CAAC,QAAyB;IAC7D,OAAO,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,KAAK,QAAQ,CAAC,CAAA;AACxD,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,mBAAmB;IACjC,OAAO,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,KAAK,QAAQ,CAAC,CAAA;AAC3D,CAAC;AAED;;;;GAIG;AACH,MAAM,UAAU,iBAAiB,CAAC,IAAY;IAC5C,OAAO,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,IAAI,CAAC,CAAA;AAChD,CAAC;AAED;;;;GAIG;AACH,MAAM,UAAU,YAAY,CAAC,EAAU;IACrC,OAAO,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,EAAE,CAAC,CAAA;AAC5C,CAAC;AAED;;;;GAIG;AACH,MAAM,UAAU,uBAAuB,CAAC,WAAmB;IACzD,MAAM,SAAS,GAAG,YAAY,CAAC,WAAW,CAAC,CAAA;IAC3C,IAAI,CAAC,SAAS;QAAE,OAAO,EAAE,CAAA;IACzB,OAAO,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,SAAS,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAA;AAClE,CAAC;AAED;;;;GAIG;AACH,MAAM,UAAU,cAAc,CAAC,EAAU;IACvC,OAAO,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,EAAE,CAAC,CAAA;AAC1C,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,mBAAmB;IACjC,MAAM,KAAK,GAA2B,EAAE,CAAA;IACxC,KAAK,MAAM,OAAO,IAAI,QAAQ,EAAE,CAAC;QAC/B,KAAK,CAAC,OAAO,CAAC,WAAW,CAAC,GAAG,CAAC,KAAK,CAAC,OAAO,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAA;IACpE,CAAC;IACD,OAAO,KAAK,CAAA;AACd,CAAC;AAED,iBAAiB;AACjB,eAAe;IACb,OAAO;IACP,QAAQ;IACR,UAAU;IACV,UAAU;IACV,UAAU;IACV,qBAAqB;IACrB,mBAAmB;IACnB,iBAAiB;IACjB,YAAY;IACZ,uBAAuB;IACvB,cAAc;IACd,mBAAmB;CACpB,CAAA","sourcesContent":["/**\n * @gallop/canon\n *\n * Gallop Canon\n * Versioned, AI-compatible, auditable web architecture patterns\n */\n\nimport schema from '../schema.json' with { type: 'json' }\n\n// Template config type\nexport type { CanonTemplateConfig } from './types/config.js'\n\n// Types\nexport interface Pattern {\n  id: string\n  title: string\n  file: string\n  category: PatternCategory\n  status: 'stable' | 'proposed' | 'deprecated'\n  enforcement: 'eslint' | 'documentation' | 'ci'\n  rule: string | null\n  summary: string\n}\n\nexport interface Category {\n  id: string\n  name: string\n  description: string\n}\n\nexport interface Guarantee {\n  id: string\n  name: string\n  since: string\n  status: 'stable' | 'proposed' | 'deprecated'\n  patterns: string[]\n}\n\nexport type PatternCategory =\n  | 'rendering'\n  | 'layout'\n  | 'typography'\n  | 'structure'\n  | 'styling'\n  | 'components'\n  | 'seo'\n\nexport interface CanonSchema {\n  name: string\n  version: string\n  description: string\n  categories: Category[]\n  patterns: Pattern[]\n  guarantees: Guarantee[]\n}\n\n// Export the full schema\nexport const canon: CanonSchema = schema as CanonSchema\n\n// Export version\nexport const version = schema.version\n\n// Export patterns array\nexport const patterns: Pattern[] = schema.patterns as Pattern[]\n\n// Export categories array\nexport const categories: Category[] = schema.categories\n\n// Export guarantees array\nexport const guarantees: Guarantee[] = schema.guarantees as Guarantee[]\n\n/**\n * Get a pattern by ID\n * @param id - Pattern ID (e.g., \"001\", \"002\")\n * @returns Pattern object or undefined if not found\n */\nexport function getPattern(id: string): Pattern | undefined {\n  return patterns.find((p) => p.id === id)\n}\n\n/**\n * Get all patterns in a category\n * @param category - Category ID (e.g., \"rendering\", \"typography\")\n * @returns Array of patterns in that category\n */\nexport function getPatternsByCategory(category: PatternCategory): Pattern[] {\n  return patterns.filter((p) => p.category === category)\n}\n\n/**\n * Get all patterns enforced by ESLint\n * @returns Array of ESLint-enforced patterns\n */\nexport function getEnforcedPatterns(): Pattern[] {\n  return patterns.filter((p) => p.enforcement === 'eslint')\n}\n\n/**\n * Get all patterns with a specific ESLint rule\n * @param rule - ESLint rule name (e.g., \"gallop/no-client-blocks\")\n * @returns Array of patterns using that rule\n */\nexport function getPatternsByRule(rule: string): Pattern[] {\n  return patterns.filter((p) => p.rule === rule)\n}\n\n/**\n * Get a guarantee by ID\n * @param id - Guarantee ID (e.g., \"SEO_STABLE\")\n * @returns Guarantee object or undefined if not found\n */\nexport function getGuarantee(id: string): Guarantee | undefined {\n  return guarantees.find((g) => g.id === id)\n}\n\n/**\n * Get all patterns associated with a guarantee\n * @param guaranteeId - Guarantee ID (e.g., \"SEO_STABLE\")\n * @returns Array of patterns that support this guarantee\n */\nexport function getPatternsForGuarantee(guaranteeId: string): Pattern[] {\n  const guarantee = getGuarantee(guaranteeId)\n  if (!guarantee) return []\n  return patterns.filter((p) => guarantee.patterns.includes(p.id))\n}\n\n/**\n * Check if a pattern ID is valid\n * @param id - Pattern ID to check\n * @returns true if valid, false otherwise\n */\nexport function isValidPattern(id: string): boolean {\n  return patterns.some((p) => p.id === id)\n}\n\n/**\n * Get pattern count by enforcement type\n * @returns Object with counts per enforcement type\n */\nexport function getEnforcementStats(): Record<string, number> {\n  const stats: Record<string, number> = {}\n  for (const pattern of patterns) {\n    stats[pattern.enforcement] = (stats[pattern.enforcement] || 0) + 1\n  }\n  return stats\n}\n\n// Default export\nexport default {\n  version,\n  patterns,\n  categories,\n  guarantees,\n  getPattern,\n  getPatternsByCategory,\n  getEnforcedPatterns,\n  getPatternsByRule,\n  getGuarantee,\n  getPatternsForGuarantee,\n  isValidPattern,\n  getEnforcementStats,\n}\n"]}
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
export interface CanonTemplateConfig {
|
|
2
|
+
name: string;
|
|
3
|
+
generatedFiles?: {
|
|
4
|
+
path: string;
|
|
5
|
+
command: string;
|
|
6
|
+
trigger: string;
|
|
7
|
+
}[];
|
|
8
|
+
buildCommands?: {
|
|
9
|
+
script: string;
|
|
10
|
+
description: string;
|
|
11
|
+
}[];
|
|
12
|
+
colorTokens?: {
|
|
13
|
+
surface?: string[];
|
|
14
|
+
text?: string[];
|
|
15
|
+
overlay?: string[];
|
|
16
|
+
accents?: string[];
|
|
17
|
+
allowedRawClasses?: string[];
|
|
18
|
+
};
|
|
19
|
+
components?: {
|
|
20
|
+
name: string;
|
|
21
|
+
props?: string[];
|
|
22
|
+
notes?: string;
|
|
23
|
+
}[];
|
|
24
|
+
state?: {
|
|
25
|
+
library: string;
|
|
26
|
+
file: string;
|
|
27
|
+
readPattern: string;
|
|
28
|
+
writePattern: string;
|
|
29
|
+
properties?: string[];
|
|
30
|
+
};
|
|
31
|
+
rules?: {
|
|
32
|
+
type: 'do' | 'doNot';
|
|
33
|
+
rule: string;
|
|
34
|
+
}[];
|
|
35
|
+
verification?: string[];
|
|
36
|
+
}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uZmlnLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3R5cGVzL2NvbmZpZy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGludGVyZmFjZSBDYW5vblRlbXBsYXRlQ29uZmlnIHtcbiAgbmFtZTogc3RyaW5nXG4gIGdlbmVyYXRlZEZpbGVzPzogeyBwYXRoOiBzdHJpbmc7IGNvbW1hbmQ6IHN0cmluZzsgdHJpZ2dlcjogc3RyaW5nIH1bXVxuICBidWlsZENvbW1hbmRzPzogeyBzY3JpcHQ6IHN0cmluZzsgZGVzY3JpcHRpb246IHN0cmluZyB9W11cbiAgY29sb3JUb2tlbnM/OiB7XG4gICAgc3VyZmFjZT86IHN0cmluZ1tdXG4gICAgdGV4dD86IHN0cmluZ1tdXG4gICAgb3ZlcmxheT86IHN0cmluZ1tdXG4gICAgYWNjZW50cz86IHN0cmluZ1tdXG4gICAgYWxsb3dlZFJhd0NsYXNzZXM/OiBzdHJpbmdbXVxuICB9XG4gIGNvbXBvbmVudHM/OiB7IG5hbWU6IHN0cmluZzsgcHJvcHM/OiBzdHJpbmdbXTsgbm90ZXM/OiBzdHJpbmcgfVtdXG4gIHN0YXRlPzoge1xuICAgIGxpYnJhcnk6IHN0cmluZ1xuICAgIGZpbGU6IHN0cmluZ1xuICAgIHJlYWRQYXR0ZXJuOiBzdHJpbmdcbiAgICB3cml0ZVBhdHRlcm46IHN0cmluZ1xuICAgIHByb3BlcnRpZXM/OiBzdHJpbmdbXVxuICB9XG4gIHJ1bGVzPzogeyB0eXBlOiAnZG8nIHwgJ2RvTm90JzsgcnVsZTogc3RyaW5nIH1bXVxuICB2ZXJpZmljYXRpb24/OiBzdHJpbmdbXVxufVxuIl19
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@gallop.software/canon",
|
|
3
|
-
"version": "2.
|
|
3
|
+
"version": "2.29.0",
|
|
4
4
|
"type": "module",
|
|
5
5
|
"description": "Gallop Canon - Architecture patterns, ESLint plugin, and CLI for template governance",
|
|
6
6
|
"main": "dist/index.js",
|
|
@@ -13,6 +13,10 @@
|
|
|
13
13
|
"./eslint": {
|
|
14
14
|
"types": "./dist/eslint/index.d.ts",
|
|
15
15
|
"import": "./dist/eslint/index.js"
|
|
16
|
+
},
|
|
17
|
+
"./config": {
|
|
18
|
+
"types": "./dist/types/config.d.ts",
|
|
19
|
+
"import": "./dist/types/config.js"
|
|
16
20
|
}
|
|
17
21
|
},
|
|
18
22
|
"bin": {
|
|
@@ -3,7 +3,7 @@
|
|
|
3
3
|
**Canon Version:** 1.0
|
|
4
4
|
**Status:** Stable
|
|
5
5
|
**Category:** Structure
|
|
6
|
-
**Enforcement:**
|
|
6
|
+
**Enforcement:** ESLint (`gallop/prefer-alias-imports`)
|
|
7
7
|
|
|
8
8
|
## Decision
|
|
9
9
|
|
|
@@ -132,8 +132,9 @@ Barrel exports (`src/components/index.ts`) were previously used but removed beca
|
|
|
132
132
|
|
|
133
133
|
## Enforcement
|
|
134
134
|
|
|
135
|
-
- **
|
|
136
|
-
-
|
|
135
|
+
- **ESLint Rule:** `gallop/prefer-alias-imports`
|
|
136
|
+
- Flags relative imports with 2+ `../` segments targeting Canon zones
|
|
137
|
+
- Same-directory `./` imports are allowed
|
|
137
138
|
|
|
138
139
|
## References
|
|
139
140
|
|
|
@@ -3,7 +3,7 @@
|
|
|
3
3
|
**Canon Version:** 1.0
|
|
4
4
|
**Status:** Stable
|
|
5
5
|
**Category:** Styling
|
|
6
|
-
**Enforcement:**
|
|
6
|
+
**Enforcement:** ESLint (`gallop/no-raw-colors`)
|
|
7
7
|
|
|
8
8
|
## Decision
|
|
9
9
|
|
|
@@ -105,8 +105,9 @@ Raw Tailwind colors are acceptable for:
|
|
|
105
105
|
|
|
106
106
|
## Enforcement
|
|
107
107
|
|
|
108
|
-
- **
|
|
109
|
-
-
|
|
108
|
+
- **ESLint Rule:** `gallop/no-raw-colors`
|
|
109
|
+
- Flags raw Tailwind color classes (e.g. `text-white`, `bg-gray-500`)
|
|
110
|
+
- Accepts `allowedClasses` option for escape hatches (brand colors, etc.)
|
|
110
111
|
|
|
111
112
|
## References
|
|
112
113
|
|
|
@@ -3,7 +3,7 @@
|
|
|
3
3
|
**Canon Version:** 1.0
|
|
4
4
|
**Status:** Stable
|
|
5
5
|
**Category:** Components
|
|
6
|
-
**Enforcement:**
|
|
6
|
+
**Enforcement:** ESLint (`gallop/no-inline-svg`)
|
|
7
7
|
|
|
8
8
|
## Decision
|
|
9
9
|
|
|
@@ -110,8 +110,9 @@ import { ArrowRight } from 'react-icons/hi'
|
|
|
110
110
|
|
|
111
111
|
## Enforcement
|
|
112
112
|
|
|
113
|
-
- **
|
|
114
|
-
-
|
|
113
|
+
- **ESLint Rule:** `gallop/no-inline-svg`
|
|
114
|
+
- Flags `<svg>` elements in block files
|
|
115
|
+
- Use the Icon component with Iconify icons instead
|
|
115
116
|
|
|
116
117
|
## References
|
|
117
118
|
|
|
@@ -3,7 +3,7 @@
|
|
|
3
3
|
**Canon Version:** 1.0
|
|
4
4
|
**Status:** Stable
|
|
5
5
|
**Category:** Styling
|
|
6
|
-
**Enforcement:**
|
|
6
|
+
**Enforcement:** ESLint (`gallop/no-classnames-package`)
|
|
7
7
|
|
|
8
8
|
## Decision
|
|
9
9
|
|
|
@@ -119,8 +119,8 @@ The API is identical — just change the import.
|
|
|
119
119
|
|
|
120
120
|
## Enforcement
|
|
121
121
|
|
|
122
|
-
- **
|
|
123
|
-
-
|
|
122
|
+
- **ESLint Rule:** `gallop/no-classnames-package`
|
|
123
|
+
- Flags `import` or `require()` of `classnames`, `classnames/bind`, or `classnames/dedupe`
|
|
124
124
|
|
|
125
125
|
## References
|
|
126
126
|
|
package/schema.json
CHANGED
|
@@ -107,8 +107,8 @@
|
|
|
107
107
|
"file": "patterns/007-import-paths.md",
|
|
108
108
|
"category": "structure",
|
|
109
109
|
"status": "stable",
|
|
110
|
-
"enforcement": "
|
|
111
|
-
"rule":
|
|
110
|
+
"enforcement": "eslint",
|
|
111
|
+
"rule": "gallop/prefer-alias-imports",
|
|
112
112
|
"summary": "@/ aliases, direct file imports"
|
|
113
113
|
},
|
|
114
114
|
{
|
|
@@ -127,8 +127,8 @@
|
|
|
127
127
|
"file": "patterns/009-color-tokens.md",
|
|
128
128
|
"category": "styling",
|
|
129
129
|
"status": "stable",
|
|
130
|
-
"enforcement": "
|
|
131
|
-
"rule":
|
|
130
|
+
"enforcement": "eslint",
|
|
131
|
+
"rule": "gallop/no-raw-colors",
|
|
132
132
|
"summary": "Use semantic color tokens"
|
|
133
133
|
},
|
|
134
134
|
{
|
|
@@ -157,8 +157,8 @@
|
|
|
157
157
|
"file": "patterns/012-icon-system.md",
|
|
158
158
|
"category": "components",
|
|
159
159
|
"status": "stable",
|
|
160
|
-
"enforcement": "
|
|
161
|
-
"rule":
|
|
160
|
+
"enforcement": "eslint",
|
|
161
|
+
"rule": "gallop/no-inline-svg",
|
|
162
162
|
"summary": "Iconify with Icon component"
|
|
163
163
|
},
|
|
164
164
|
{
|
|
@@ -177,8 +177,8 @@
|
|
|
177
177
|
"file": "patterns/014-clsx-not-classnames.md",
|
|
178
178
|
"category": "styling",
|
|
179
179
|
"status": "stable",
|
|
180
|
-
"enforcement": "
|
|
181
|
-
"rule":
|
|
180
|
+
"enforcement": "eslint",
|
|
181
|
+
"rule": "gallop/no-classnames-package",
|
|
182
182
|
"summary": "Use clsx, never classnames package"
|
|
183
183
|
},
|
|
184
184
|
{
|