@gallop.software/canon 2.15.2 → 2.16.1
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/eslint/index.d.ts
CHANGED
|
@@ -7,6 +7,9 @@ declare const plugin: {
|
|
|
7
7
|
'no-client-blocks': import("@typescript-eslint/utils/ts-eslint").RuleModule<"noClientBlocks", [], unknown, import("@typescript-eslint/utils/ts-eslint").RuleListener> & {
|
|
8
8
|
name: string;
|
|
9
9
|
};
|
|
10
|
+
'block-naming-convention': import("@typescript-eslint/utils/ts-eslint").RuleModule<"blockNamingMismatch" | "blockNamingNoNumber", [], unknown, import("@typescript-eslint/utils/ts-eslint").RuleListener> & {
|
|
11
|
+
name: string;
|
|
12
|
+
};
|
|
10
13
|
'no-container-in-section': import("@typescript-eslint/utils/ts-eslint").RuleModule<"noContainerInSection", [], unknown, import("@typescript-eslint/utils/ts-eslint").RuleListener> & {
|
|
11
14
|
name: string;
|
|
12
15
|
};
|
|
@@ -39,6 +42,7 @@ declare const plugin: {
|
|
|
39
42
|
*/
|
|
40
43
|
recommended: {
|
|
41
44
|
readonly 'gallop/no-client-blocks': "warn";
|
|
45
|
+
readonly 'gallop/block-naming-convention': "warn";
|
|
42
46
|
readonly 'gallop/no-container-in-section': "warn";
|
|
43
47
|
readonly 'gallop/prefer-component-props': "warn";
|
|
44
48
|
readonly 'gallop/prefer-typography-components': "warn";
|
package/dist/eslint/index.js
CHANGED
|
@@ -12,11 +12,13 @@ import noNativeIntersectionObserver from './rules/no-native-intersection-observe
|
|
|
12
12
|
import noComponentInBlocks from './rules/no-component-in-blocks.js';
|
|
13
13
|
import preferListComponents from './rules/prefer-list-components.js';
|
|
14
14
|
import noNativeDate from './rules/no-native-date.js';
|
|
15
|
+
import blockNamingConvention from './rules/block-naming-convention.js';
|
|
15
16
|
/**
|
|
16
17
|
* All Canon ESLint rules with recommended severity levels
|
|
17
18
|
*/
|
|
18
19
|
const recommended = {
|
|
19
20
|
'gallop/no-client-blocks': 'warn',
|
|
21
|
+
'gallop/block-naming-convention': 'warn',
|
|
20
22
|
'gallop/no-container-in-section': 'warn',
|
|
21
23
|
'gallop/prefer-component-props': 'warn',
|
|
22
24
|
'gallop/prefer-typography-components': 'warn',
|
|
@@ -38,6 +40,7 @@ const plugin = {
|
|
|
38
40
|
},
|
|
39
41
|
rules: {
|
|
40
42
|
'no-client-blocks': noClientBlocks,
|
|
43
|
+
'block-naming-convention': blockNamingConvention,
|
|
41
44
|
'no-container-in-section': noContainerInSection,
|
|
42
45
|
'prefer-component-props': preferComponentProps,
|
|
43
46
|
'prefer-typography-components': preferTypographyComponents,
|
|
@@ -59,4 +62,4 @@ const plugin = {
|
|
|
59
62
|
recommended,
|
|
60
63
|
};
|
|
61
64
|
export default plugin;
|
|
62
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
65
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvZXNsaW50L2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sY0FBYyxNQUFNLDZCQUE2QixDQUFBO0FBQ3hELE9BQU8sb0JBQW9CLE1BQU0sb0NBQW9DLENBQUE7QUFDckUsT0FBTyxvQkFBb0IsTUFBTSxtQ0FBbUMsQ0FBQTtBQUNwRSxPQUFPLDBCQUEwQixNQUFNLHlDQUF5QyxDQUFBO0FBQ2hGLE9BQU8sc0JBQXNCLE1BQU0scUNBQXFDLENBQUE7QUFDeEUsT0FBTyxzQkFBc0IsTUFBTSxxQ0FBcUMsQ0FBQTtBQUN4RSxPQUFPLGNBQWMsTUFBTSw2QkFBNkIsQ0FBQTtBQUN4RCxPQUFPLGlCQUFpQixNQUFNLGdDQUFnQyxDQUFBO0FBQzlELE9BQU8sa0JBQWtCLE1BQU0sa0NBQWtDLENBQUE7QUFDakUsT0FBTyxhQUFhLE1BQU0sNEJBQTRCLENBQUE7QUFDdEQsT0FBTyw0QkFBNEIsTUFBTSw0Q0FBNEMsQ0FBQTtBQUNyRixPQUFPLG1CQUFtQixNQUFNLG1DQUFtQyxDQUFBO0FBQ25FLE9BQU8sb0JBQW9CLE1BQU0sbUNBQW1DLENBQUE7QUFDcEUsT0FBTyxZQUFZLE1BQU0sMkJBQTJCLENBQUE7QUFDcEQsT0FBTyxxQkFBcUIsTUFBTSxvQ0FBb0MsQ0FBQTtBQUV0RTs7R0FFRztBQUNILE1BQU0sV0FBVyxHQUFHO0lBQ2xCLHlCQUF5QixFQUFFLE1BQU07SUFDakMsZ0NBQWdDLEVBQUUsTUFBTTtJQUN4QyxnQ0FBZ0MsRUFBRSxNQUFNO0lBQ3hDLCtCQUErQixFQUFFLE1BQU07SUFDdkMscUNBQXFDLEVBQUUsTUFBTTtJQUM3QyxpQ0FBaUMsRUFBRSxNQUFNO0lBQ3pDLGlDQUFpQyxFQUFFLE1BQU07SUFDekMseUJBQXlCLEVBQUUsTUFBTTtJQUNqQyw0QkFBNEIsRUFBRSxNQUFNO0lBQ3BDLDhCQUE4QixFQUFFLE1BQU07SUFDdEMsd0JBQXdCLEVBQUUsTUFBTTtJQUNoQyx3Q0FBd0MsRUFBRSxNQUFNO0lBQ2hELCtCQUErQixFQUFFLE1BQU07SUFDdkMsK0JBQStCLEVBQUUsTUFBTTtJQUN2Qyx1QkFBdUIsRUFBRSxNQUFNO0NBQ3ZCLENBQUE7QUFFVixNQUFNLE1BQU0sR0FBRztJQUNiLElBQUksRUFBRTtRQUNKLElBQUksRUFBRSxzQkFBc0I7UUFDNUIsT0FBTyxFQUFFLFFBQVE7S0FDbEI7SUFDRCxLQUFLLEVBQUU7UUFDTCxrQkFBa0IsRUFBRSxjQUFjO1FBQ2xDLHlCQUF5QixFQUFFLHFCQUFxQjtRQUNoRCx5QkFBeUIsRUFBRSxvQkFBb0I7UUFDL0Msd0JBQXdCLEVBQUUsb0JBQW9CO1FBQzlDLDhCQUE4QixFQUFFLDBCQUEwQjtRQUMxRCwwQkFBMEIsRUFBRSxzQkFBc0I7UUFDbEQsMEJBQTBCLEVBQUUsc0JBQXNCO1FBQ2xELGtCQUFrQixFQUFFLGNBQWM7UUFDbEMscUJBQXFCLEVBQUUsaUJBQWlCO1FBQ3hDLHVCQUF1QixFQUFFLGtCQUFrQjtRQUMzQyxpQkFBaUIsRUFBRSxhQUFhO1FBQ2hDLGlDQUFpQyxFQUFFLDRCQUE0QjtRQUMvRCx3QkFBd0IsRUFBRSxtQkFBbUI7UUFDN0Msd0JBQXdCLEVBQUUsb0JBQW9CO1FBQzlDLGdCQUFnQixFQUFFLFlBQVk7S0FDL0I7SUFDRDs7O09BR0c7SUFDSCxXQUFXO0NBQ1osQ0FBQTtBQUVELGVBQWUsTUFBTSxDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IG5vQ2xpZW50QmxvY2tzIGZyb20gJy4vcnVsZXMvbm8tY2xpZW50LWJsb2Nrcy5qcydcbmltcG9ydCBub0NvbnRhaW5lckluU2VjdGlvbiBmcm9tICcuL3J1bGVzL25vLWNvbnRhaW5lci1pbi1zZWN0aW9uLmpzJ1xuaW1wb3J0IHByZWZlckNvbXBvbmVudFByb3BzIGZyb20gJy4vcnVsZXMvcHJlZmVyLWNvbXBvbmVudC1wcm9wcy5qcydcbmltcG9ydCBwcmVmZXJUeXBvZ3JhcGh5Q29tcG9uZW50cyBmcm9tICcuL3J1bGVzL3ByZWZlci10eXBvZ3JhcGh5LWNvbXBvbmVudHMuanMnXG5pbXBvcnQgcHJlZmVyTGF5b3V0Q29tcG9uZW50cyBmcm9tICcuL3J1bGVzL3ByZWZlci1sYXlvdXQtY29tcG9uZW50cy5qcydcbmltcG9ydCBiYWNrZ3JvdW5kSW1hZ2VSb3VuZGVkIGZyb20gJy4vcnVsZXMvYmFja2dyb3VuZC1pbWFnZS1yb3VuZGVkLmpzJ1xuaW1wb3J0IG5vSW5saW5lU3R5bGVzIGZyb20gJy4vcnVsZXMvbm8taW5saW5lLXN0eWxlcy5qcydcbmltcG9ydCBub0FyYml0cmFyeUNvbG9ycyBmcm9tICcuL3J1bGVzL25vLWFyYml0cmFyeS1jb2xvcnMuanMnXG5pbXBvcnQgbm9Dcm9zc1pvbmVJbXBvcnRzIGZyb20gJy4vcnVsZXMvbm8tY3Jvc3Mtem9uZS1pbXBvcnRzLmpzJ1xuaW1wb3J0IG5vRGF0YUltcG9ydHMgZnJvbSAnLi9ydWxlcy9uby1kYXRhLWltcG9ydHMuanMnXG5pbXBvcnQgbm9OYXRpdmVJbnRlcnNlY3Rpb25PYnNlcnZlciBmcm9tICcuL3J1bGVzL25vLW5hdGl2ZS1pbnRlcnNlY3Rpb24tb2JzZXJ2ZXIuanMnXG5pbXBvcnQgbm9Db21wb25lbnRJbkJsb2NrcyBmcm9tICcuL3J1bGVzL25vLWNvbXBvbmVudC1pbi1ibG9ja3MuanMnXG5pbXBvcnQgcHJlZmVyTGlzdENvbXBvbmVudHMgZnJvbSAnLi9ydWxlcy9wcmVmZXItbGlzdC1jb21wb25lbnRzLmpzJ1xuaW1wb3J0IG5vTmF0aXZlRGF0ZSBmcm9tICcuL3J1bGVzL25vLW5hdGl2ZS1kYXRlLmpzJ1xuaW1wb3J0IGJsb2NrTmFtaW5nQ29udmVudGlvbiBmcm9tICcuL3J1bGVzL2Jsb2NrLW5hbWluZy1jb252ZW50aW9uLmpzJ1xuXG4vKipcbiAqIEFsbCBDYW5vbiBFU0xpbnQgcnVsZXMgd2l0aCByZWNvbW1lbmRlZCBzZXZlcml0eSBsZXZlbHNcbiAqL1xuY29uc3QgcmVjb21tZW5kZWQgPSB7XG4gICdnYWxsb3Avbm8tY2xpZW50LWJsb2Nrcyc6ICd3YXJuJyxcbiAgJ2dhbGxvcC9ibG9jay1uYW1pbmctY29udmVudGlvbic6ICd3YXJuJyxcbiAgJ2dhbGxvcC9uby1jb250YWluZXItaW4tc2VjdGlvbic6ICd3YXJuJyxcbiAgJ2dhbGxvcC9wcmVmZXItY29tcG9uZW50LXByb3BzJzogJ3dhcm4nLFxuICAnZ2FsbG9wL3ByZWZlci10eXBvZ3JhcGh5LWNvbXBvbmVudHMnOiAnd2FybicsXG4gICdnYWxsb3AvcHJlZmVyLWxheW91dC1jb21wb25lbnRzJzogJ3dhcm4nLFxuICAnZ2FsbG9wL2JhY2tncm91bmQtaW1hZ2Utcm91bmRlZCc6ICd3YXJuJyxcbiAgJ2dhbGxvcC9uby1pbmxpbmUtc3R5bGVzJzogJ3dhcm4nLFxuICAnZ2FsbG9wL25vLWFyYml0cmFyeS1jb2xvcnMnOiAnd2FybicsXG4gICdnYWxsb3Avbm8tY3Jvc3Mtem9uZS1pbXBvcnRzJzogJ3dhcm4nLFxuICAnZ2FsbG9wL25vLWRhdGEtaW1wb3J0cyc6ICd3YXJuJyxcbiAgJ2dhbGxvcC9uby1uYXRpdmUtaW50ZXJzZWN0aW9uLW9ic2VydmVyJzogJ3dhcm4nLFxuICAnZ2FsbG9wL25vLWNvbXBvbmVudC1pbi1ibG9ja3MnOiAnd2FybicsXG4gICdnYWxsb3AvcHJlZmVyLWxpc3QtY29tcG9uZW50cyc6ICd3YXJuJyxcbiAgJ2dhbGxvcC9uby1uYXRpdmUtZGF0ZSc6ICd3YXJuJyxcbn0gYXMgY29uc3RcblxuY29uc3QgcGx1Z2luID0ge1xuICBtZXRhOiB7XG4gICAgbmFtZTogJ2VzbGludC1wbHVnaW4tZ2FsbG9wJyxcbiAgICB2ZXJzaW9uOiAnMi4xMi4wJyxcbiAgfSxcbiAgcnVsZXM6IHtcbiAgICAnbm8tY2xpZW50LWJsb2Nrcyc6IG5vQ2xpZW50QmxvY2tzLFxuICAgICdibG9jay1uYW1pbmctY29udmVudGlvbic6IGJsb2NrTmFtaW5nQ29udmVudGlvbixcbiAgICAnbm8tY29udGFpbmVyLWluLXNlY3Rpb24nOiBub0NvbnRhaW5lckluU2VjdGlvbixcbiAgICAncHJlZmVyLWNvbXBvbmVudC1wcm9wcyc6IHByZWZlckNvbXBvbmVudFByb3BzLFxuICAgICdwcmVmZXItdHlwb2dyYXBoeS1jb21wb25lbnRzJzogcHJlZmVyVHlwb2dyYXBoeUNvbXBvbmVudHMsXG4gICAgJ3ByZWZlci1sYXlvdXQtY29tcG9uZW50cyc6IHByZWZlckxheW91dENvbXBvbmVudHMsXG4gICAgJ2JhY2tncm91bmQtaW1hZ2Utcm91bmRlZCc6IGJhY2tncm91bmRJbWFnZVJvdW5kZWQsXG4gICAgJ25vLWlubGluZS1zdHlsZXMnOiBub0lubGluZVN0eWxlcyxcbiAgICAnbm8tYXJiaXRyYXJ5LWNvbG9ycyc6IG5vQXJiaXRyYXJ5Q29sb3JzLFxuICAgICduby1jcm9zcy16b25lLWltcG9ydHMnOiBub0Nyb3NzWm9uZUltcG9ydHMsXG4gICAgJ25vLWRhdGEtaW1wb3J0cyc6IG5vRGF0YUltcG9ydHMsXG4gICAgJ25vLW5hdGl2ZS1pbnRlcnNlY3Rpb24tb2JzZXJ2ZXInOiBub05hdGl2ZUludGVyc2VjdGlvbk9ic2VydmVyLFxuICAgICduby1jb21wb25lbnQtaW4tYmxvY2tzJzogbm9Db21wb25lbnRJbkJsb2NrcyxcbiAgICAncHJlZmVyLWxpc3QtY29tcG9uZW50cyc6IHByZWZlckxpc3RDb21wb25lbnRzLFxuICAgICduby1uYXRpdmUtZGF0ZSc6IG5vTmF0aXZlRGF0ZSxcbiAgfSxcbiAgLyoqXG4gICAqIFJlY29tbWVuZGVkIHJ1bGUgY29uZmlndXJhdGlvbnMgLSBzcHJlYWQgaW50byB5b3VyIEVTTGludCBjb25maWdcbiAgICogQGV4YW1wbGUgcnVsZXM6IHsgLi4uZ2FsbG9wLnJlY29tbWVuZGVkIH1cbiAgICovXG4gIHJlY29tbWVuZGVkLFxufVxuXG5leHBvcnQgZGVmYXVsdCBwbHVnaW5cbiJdfQ==
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
import { ESLintUtils } from '@typescript-eslint/utils';
|
|
2
|
+
type MessageIds = 'blockNamingMismatch' | 'blockNamingNoNumber';
|
|
3
|
+
declare const _default: ESLintUtils.RuleModule<MessageIds, [], unknown, ESLintUtils.RuleListener> & {
|
|
4
|
+
name: string;
|
|
5
|
+
};
|
|
6
|
+
export default _default;
|
|
@@ -0,0 +1,107 @@
|
|
|
1
|
+
import { ESLintUtils } from '@typescript-eslint/utils';
|
|
2
|
+
import { getCanonUrl, getCanonPattern } from '../utils/canon.js';
|
|
3
|
+
const RULE_NAME = 'block-naming-convention';
|
|
4
|
+
const pattern = getCanonPattern(RULE_NAME);
|
|
5
|
+
const createRule = ESLintUtils.RuleCreator(() => getCanonUrl(RULE_NAME));
|
|
6
|
+
/**
|
|
7
|
+
* Converts a block filename to its expected PascalCase export name
|
|
8
|
+
* e.g., "hero-5" -> "Hero5", "section-10" -> "Section10", "content-39" -> "Content39"
|
|
9
|
+
*/
|
|
10
|
+
function filenameToPascalCase(filename) {
|
|
11
|
+
// Remove extension and split by hyphens
|
|
12
|
+
const baseName = filename.replace(/\.tsx?$/, '');
|
|
13
|
+
// Split by hyphens and convert each part
|
|
14
|
+
return baseName
|
|
15
|
+
.split('-')
|
|
16
|
+
.map((part) => {
|
|
17
|
+
// Capitalize first letter of each part
|
|
18
|
+
return part.charAt(0).toUpperCase() + part.slice(1);
|
|
19
|
+
})
|
|
20
|
+
.join('');
|
|
21
|
+
}
|
|
22
|
+
/**
|
|
23
|
+
* Converts a PascalCase export name to kebab-case filename
|
|
24
|
+
* e.g., "Content" -> "content", "Hero5" -> "hero-5", "CallToAction1" -> "call-to-action-1"
|
|
25
|
+
*/
|
|
26
|
+
function pascalCaseToFilename(name) {
|
|
27
|
+
return name
|
|
28
|
+
// Insert hyphen before uppercase letters (but not at start)
|
|
29
|
+
.replace(/([a-z])([A-Z])/g, '$1-$2')
|
|
30
|
+
// Insert hyphen before numbers
|
|
31
|
+
.replace(/([a-zA-Z])(\d)/g, '$1-$2')
|
|
32
|
+
.toLowerCase();
|
|
33
|
+
}
|
|
34
|
+
/**
|
|
35
|
+
* Check if a name ends with a number
|
|
36
|
+
*/
|
|
37
|
+
function hasTrailingNumber(name) {
|
|
38
|
+
return /\d+$/.test(name);
|
|
39
|
+
}
|
|
40
|
+
export default createRule({
|
|
41
|
+
name: RULE_NAME,
|
|
42
|
+
meta: {
|
|
43
|
+
type: 'suggestion',
|
|
44
|
+
docs: {
|
|
45
|
+
description: pattern?.summary || 'Block export names must match filename pattern',
|
|
46
|
+
},
|
|
47
|
+
messages: {
|
|
48
|
+
blockNamingMismatch: `[Canon ${pattern?.id || '006'}] Block export "{{actual}}" should be "{{expected}}" to match the filename "{{filename}}". Or rename the file to "{{suggestedFilename}}.tsx". See: ${pattern?.title || 'Block Naming'} pattern.`,
|
|
49
|
+
blockNamingNoNumber: `[Canon ${pattern?.id || '006'}] Block export "{{actual}}" must end with a number (e.g., "{{actual}}1"). Rename to "{{suggested}}" or rename file to "{{suggestedFilename}}-{n}.tsx". See: ${pattern?.title || 'Block Naming'} pattern.`,
|
|
50
|
+
},
|
|
51
|
+
schema: [],
|
|
52
|
+
},
|
|
53
|
+
defaultOptions: [],
|
|
54
|
+
create(context) {
|
|
55
|
+
const filename = context.filename || context.getFilename();
|
|
56
|
+
// Only check files in src/blocks/
|
|
57
|
+
if (!filename.includes('/blocks/') && !filename.includes('\\blocks\\')) {
|
|
58
|
+
return {};
|
|
59
|
+
}
|
|
60
|
+
// Extract just the filename (e.g., "hero-5.tsx")
|
|
61
|
+
const match = filename.match(/([^/\\]+\.tsx?)$/);
|
|
62
|
+
if (!match) {
|
|
63
|
+
return {};
|
|
64
|
+
}
|
|
65
|
+
const blockFilename = match[1];
|
|
66
|
+
const expectedName = filenameToPascalCase(blockFilename);
|
|
67
|
+
return {
|
|
68
|
+
// Check export default function declarations
|
|
69
|
+
ExportDefaultDeclaration(node) {
|
|
70
|
+
if (node.declaration.type === 'FunctionDeclaration' && node.declaration.id) {
|
|
71
|
+
const actualName = node.declaration.id.name;
|
|
72
|
+
if (actualName !== expectedName) {
|
|
73
|
+
// Check if the export name has a trailing number
|
|
74
|
+
if (!hasTrailingNumber(actualName)) {
|
|
75
|
+
// No number - suggest adding one
|
|
76
|
+
const suggestedFilename = pascalCaseToFilename(actualName);
|
|
77
|
+
context.report({
|
|
78
|
+
node: node.declaration.id,
|
|
79
|
+
messageId: 'blockNamingNoNumber',
|
|
80
|
+
data: {
|
|
81
|
+
actual: actualName,
|
|
82
|
+
suggested: `${actualName}1`,
|
|
83
|
+
suggestedFilename,
|
|
84
|
+
},
|
|
85
|
+
});
|
|
86
|
+
}
|
|
87
|
+
else {
|
|
88
|
+
// Has number but doesn't match filename
|
|
89
|
+
const suggestedFilename = pascalCaseToFilename(actualName);
|
|
90
|
+
context.report({
|
|
91
|
+
node: node.declaration.id,
|
|
92
|
+
messageId: 'blockNamingMismatch',
|
|
93
|
+
data: {
|
|
94
|
+
actual: actualName,
|
|
95
|
+
expected: expectedName,
|
|
96
|
+
filename: blockFilename,
|
|
97
|
+
suggestedFilename,
|
|
98
|
+
},
|
|
99
|
+
});
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
},
|
|
104
|
+
};
|
|
105
|
+
},
|
|
106
|
+
});
|
|
107
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"block-naming-convention.js","sourceRoot":"","sources":["../../../src/eslint/rules/block-naming-convention.ts"],"names":[],"mappings":"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","sourcesContent":["import { ESLintUtils } from '@typescript-eslint/utils'\nimport { getCanonUrl, getCanonPattern } from '../utils/canon.js'\n\nconst RULE_NAME = 'block-naming-convention'\nconst pattern = getCanonPattern(RULE_NAME)\n\nconst createRule = ESLintUtils.RuleCreator(() => getCanonUrl(RULE_NAME))\n\ntype MessageIds = 'blockNamingMismatch' | 'blockNamingNoNumber'\n\n/**\n * Converts a block filename to its expected PascalCase export name\n * e.g., \"hero-5\" -> \"Hero5\", \"section-10\" -> \"Section10\", \"content-39\" -> \"Content39\"\n */\nfunction filenameToPascalCase(filename: string): string {\n  // Remove extension and split by hyphens\n  const baseName = filename.replace(/\\.tsx?$/, '')\n  \n  // Split by hyphens and convert each part\n  return baseName\n    .split('-')\n    .map((part) => {\n      // Capitalize first letter of each part\n      return part.charAt(0).toUpperCase() + part.slice(1)\n    })\n    .join('')\n}\n\n/**\n * Converts a PascalCase export name to kebab-case filename\n * e.g., \"Content\" -> \"content\", \"Hero5\" -> \"hero-5\", \"CallToAction1\" -> \"call-to-action-1\"\n */\nfunction pascalCaseToFilename(name: string): string {\n  return name\n    // Insert hyphen before uppercase letters (but not at start)\n    .replace(/([a-z])([A-Z])/g, '$1-$2')\n    // Insert hyphen before numbers\n    .replace(/([a-zA-Z])(\\d)/g, '$1-$2')\n    .toLowerCase()\n}\n\n/**\n * Check if a name ends with a number\n */\nfunction hasTrailingNumber(name: string): boolean {\n  return /\\d+$/.test(name)\n}\n\nexport default createRule<[], MessageIds>({\n  name: RULE_NAME,\n  meta: {\n    type: 'suggestion',\n    docs: {\n      description: pattern?.summary || 'Block export names must match filename pattern',\n    },\n    messages: {\n      blockNamingMismatch: `[Canon ${pattern?.id || '006'}] Block export \"{{actual}}\" should be \"{{expected}}\" to match the filename \"{{filename}}\". Or rename the file to \"{{suggestedFilename}}.tsx\". See: ${pattern?.title || 'Block Naming'} pattern.`,\n      blockNamingNoNumber: `[Canon ${pattern?.id || '006'}] Block export \"{{actual}}\" must end with a number (e.g., \"{{actual}}1\"). Rename to \"{{suggested}}\" or rename file to \"{{suggestedFilename}}-{n}.tsx\". See: ${pattern?.title || 'Block Naming'} pattern.`,\n    },\n    schema: [],\n  },\n  defaultOptions: [],\n  create(context) {\n    const filename = context.filename || context.getFilename()\n\n    // Only check files in src/blocks/\n    if (!filename.includes('/blocks/') && !filename.includes('\\\\blocks\\\\')) {\n      return {}\n    }\n\n    // Extract just the filename (e.g., \"hero-5.tsx\")\n    const match = filename.match(/([^/\\\\]+\\.tsx?)$/)\n    if (!match) {\n      return {}\n    }\n\n    const blockFilename = match[1]\n    const expectedName = filenameToPascalCase(blockFilename)\n\n    return {\n      // Check export default function declarations\n      ExportDefaultDeclaration(node) {\n        if (node.declaration.type === 'FunctionDeclaration' && node.declaration.id) {\n          const actualName = node.declaration.id.name\n          \n          if (actualName !== expectedName) {\n            // Check if the export name has a trailing number\n            if (!hasTrailingNumber(actualName)) {\n              // No number - suggest adding one\n              const suggestedFilename = pascalCaseToFilename(actualName)\n              context.report({\n                node: node.declaration.id,\n                messageId: 'blockNamingNoNumber',\n                data: {\n                  actual: actualName,\n                  suggested: `${actualName}1`,\n                  suggestedFilename,\n                },\n              })\n            } else {\n              // Has number but doesn't match filename\n              const suggestedFilename = pascalCaseToFilename(actualName)\n              context.report({\n                node: node.declaration.id,\n                messageId: 'blockNamingMismatch',\n                data: {\n                  actual: actualName,\n                  expected: expectedName,\n                  filename: blockFilename,\n                  suggestedFilename,\n                },\n              })\n            }\n          }\n        }\n      },\n    }\n  },\n})\n"]}
|
package/package.json
CHANGED
package/schema.json
CHANGED
|
@@ -97,8 +97,8 @@
|
|
|
97
97
|
"file": "patterns/006-block-naming.md",
|
|
98
98
|
"category": "structure",
|
|
99
99
|
"status": "stable",
|
|
100
|
-
"enforcement": "
|
|
101
|
-
"rule":
|
|
100
|
+
"enforcement": "eslint",
|
|
101
|
+
"rule": "gallop/block-naming-convention",
|
|
102
102
|
"summary": "{type}-{n}.tsx naming, PascalCase exports"
|
|
103
103
|
},
|
|
104
104
|
{
|