@gallop.software/canon 2.16.0 → 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.
@@ -7,7 +7,7 @@ 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", [], unknown, import("@typescript-eslint/utils/ts-eslint").RuleListener> & {
10
+ 'block-naming-convention': import("@typescript-eslint/utils/ts-eslint").RuleModule<"blockNamingMismatch" | "blockNamingNoNumber", [], unknown, import("@typescript-eslint/utils/ts-eslint").RuleListener> & {
11
11
  name: string;
12
12
  };
13
13
  'no-container-in-section': import("@typescript-eslint/utils/ts-eslint").RuleModule<"noContainerInSection", [], unknown, import("@typescript-eslint/utils/ts-eslint").RuleListener> & {
@@ -1,5 +1,6 @@
1
1
  import { ESLintUtils } from '@typescript-eslint/utils';
2
- declare const _default: ESLintUtils.RuleModule<"blockNamingMismatch", [], unknown, ESLintUtils.RuleListener> & {
2
+ type MessageIds = 'blockNamingMismatch' | 'blockNamingNoNumber';
3
+ declare const _default: ESLintUtils.RuleModule<MessageIds, [], unknown, ESLintUtils.RuleListener> & {
3
4
  name: string;
4
5
  };
5
6
  export default _default;
@@ -19,6 +19,24 @@ function filenameToPascalCase(filename) {
19
19
  })
20
20
  .join('');
21
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
+ }
22
40
  export default createRule({
23
41
  name: RULE_NAME,
24
42
  meta: {
@@ -27,7 +45,8 @@ export default createRule({
27
45
  description: pattern?.summary || 'Block export names must match filename pattern',
28
46
  },
29
47
  messages: {
30
- blockNamingMismatch: `[Canon ${pattern?.id || '006'}] Block export "{{actual}}" should be "{{expected}}" to match the filename "{{filename}}". See: ${pattern?.title || 'Block Naming'} pattern.`,
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.`,
31
50
  },
32
51
  schema: [],
33
52
  },
@@ -51,19 +70,38 @@ export default createRule({
51
70
  if (node.declaration.type === 'FunctionDeclaration' && node.declaration.id) {
52
71
  const actualName = node.declaration.id.name;
53
72
  if (actualName !== expectedName) {
54
- context.report({
55
- node: node.declaration.id,
56
- messageId: 'blockNamingMismatch',
57
- data: {
58
- actual: actualName,
59
- expected: expectedName,
60
- filename: blockFilename,
61
- },
62
- });
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
+ }
63
101
  }
64
102
  }
65
103
  },
66
104
  };
67
105
  },
68
106
  });
69
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmxvY2stbmFtaW5nLWNvbnZlbnRpb24uanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvZXNsaW50L3J1bGVzL2Jsb2NrLW5hbWluZy1jb252ZW50aW9uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSwwQkFBMEIsQ0FBQTtBQUN0RCxPQUFPLEVBQUUsV0FBVyxFQUFFLGVBQWUsRUFBRSxNQUFNLG1CQUFtQixDQUFBO0FBRWhFLE1BQU0sU0FBUyxHQUFHLHlCQUF5QixDQUFBO0FBQzNDLE1BQU0sT0FBTyxHQUFHLGVBQWUsQ0FBQyxTQUFTLENBQUMsQ0FBQTtBQUUxQyxNQUFNLFVBQVUsR0FBRyxXQUFXLENBQUMsV0FBVyxDQUFDLEdBQUcsRUFBRSxDQUFDLFdBQVcsQ0FBQyxTQUFTLENBQUMsQ0FBQyxDQUFBO0FBSXhFOzs7R0FHRztBQUNILFNBQVMsb0JBQW9CLENBQUMsUUFBZ0I7SUFDNUMsd0NBQXdDO0lBQ3hDLE1BQU0sUUFBUSxHQUFHLFFBQVEsQ0FBQyxPQUFPLENBQUMsU0FBUyxFQUFFLEVBQUUsQ0FBQyxDQUFBO0lBRWhELHlDQUF5QztJQUN6QyxPQUFPLFFBQVE7U0FDWixLQUFLLENBQUMsR0FBRyxDQUFDO1NBQ1YsR0FBRyxDQUFDLENBQUMsSUFBSSxFQUFFLEVBQUU7UUFDWix1Q0FBdUM7UUFDdkMsT0FBTyxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxDQUFDLFdBQVcsRUFBRSxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUE7SUFDckQsQ0FBQyxDQUFDO1NBQ0QsSUFBSSxDQUFDLEVBQUUsQ0FBQyxDQUFBO0FBQ2IsQ0FBQztBQUVELGVBQWUsVUFBVSxDQUFpQjtJQUN4QyxJQUFJLEVBQUUsU0FBUztJQUNmLElBQUksRUFBRTtRQUNKLElBQUksRUFBRSxZQUFZO1FBQ2xCLElBQUksRUFBRTtZQUNKLFdBQVcsRUFBRSxPQUFPLEVBQUUsT0FBTyxJQUFJLGdEQUFnRDtTQUNsRjtRQUNELFFBQVEsRUFBRTtZQUNSLG1CQUFtQixFQUFFLFVBQVUsT0FBTyxFQUFFLEVBQUUsSUFBSSxLQUFLLG1HQUFtRyxPQUFPLEVBQUUsS0FBSyxJQUFJLGNBQWMsV0FBVztTQUNsTTtRQUNELE1BQU0sRUFBRSxFQUFFO0tBQ1g7SUFDRCxjQUFjLEVBQUUsRUFBRTtJQUNsQixNQUFNLENBQUMsT0FBTztRQUNaLE1BQU0sUUFBUSxHQUFHLE9BQU8sQ0FBQyxRQUFRLElBQUksT0FBTyxDQUFDLFdBQVcsRUFBRSxDQUFBO1FBRTFELGtDQUFrQztRQUNsQyxJQUFJLENBQUMsUUFBUSxDQUFDLFFBQVEsQ0FBQyxVQUFVLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxRQUFRLENBQUMsWUFBWSxDQUFDLEVBQUUsQ0FBQztZQUN2RSxPQUFPLEVBQUUsQ0FBQTtRQUNYLENBQUM7UUFFRCxpREFBaUQ7UUFDakQsTUFBTSxLQUFLLEdBQUcsUUFBUSxDQUFDLEtBQUssQ0FBQyxrQkFBa0IsQ0FBQyxDQUFBO1FBQ2hELElBQUksQ0FBQyxLQUFLLEVBQUUsQ0FBQztZQUNYLE9BQU8sRUFBRSxDQUFBO1FBQ1gsQ0FBQztRQUVELE1BQU0sYUFBYSxHQUFHLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQTtRQUM5QixNQUFNLFlBQVksR0FBRyxvQkFBb0IsQ0FBQyxhQUFhLENBQUMsQ0FBQTtRQUV4RCxPQUFPO1lBQ0wsNkNBQTZDO1lBQzdDLHdCQUF3QixDQUFDLElBQUk7Z0JBQzNCLElBQUksSUFBSSxDQUFDLFdBQVcsQ0FBQyxJQUFJLEtBQUsscUJBQXFCLElBQUksSUFBSSxDQUFDLFdBQVcsQ0FBQyxFQUFFLEVBQUUsQ0FBQztvQkFDM0UsTUFBTSxVQUFVLEdBQUcsSUFBSSxDQUFDLFdBQVcsQ0FBQyxFQUFFLENBQUMsSUFBSSxDQUFBO29CQUMzQyxJQUFJLFVBQVUsS0FBSyxZQUFZLEVBQUUsQ0FBQzt3QkFDaEMsT0FBTyxDQUFDLE1BQU0sQ0FBQzs0QkFDYixJQUFJLEVBQUUsSUFBSSxDQUFDLFdBQVcsQ0FBQyxFQUFFOzRCQUN6QixTQUFTLEVBQUUscUJBQXFCOzRCQUNoQyxJQUFJLEVBQUU7Z0NBQ0osTUFBTSxFQUFFLFVBQVU7Z0NBQ2xCLFFBQVEsRUFBRSxZQUFZO2dDQUN0QixRQUFRLEVBQUUsYUFBYTs2QkFDeEI7eUJBQ0YsQ0FBQyxDQUFBO29CQUNKLENBQUM7Z0JBQ0gsQ0FBQztZQUNILENBQUM7U0FDRixDQUFBO0lBQ0gsQ0FBQztDQUNGLENBQUMsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEVTTGludFV0aWxzIH0gZnJvbSAnQHR5cGVzY3JpcHQtZXNsaW50L3V0aWxzJ1xuaW1wb3J0IHsgZ2V0Q2Fub25VcmwsIGdldENhbm9uUGF0dGVybiB9IGZyb20gJy4uL3V0aWxzL2Nhbm9uLmpzJ1xuXG5jb25zdCBSVUxFX05BTUUgPSAnYmxvY2stbmFtaW5nLWNvbnZlbnRpb24nXG5jb25zdCBwYXR0ZXJuID0gZ2V0Q2Fub25QYXR0ZXJuKFJVTEVfTkFNRSlcblxuY29uc3QgY3JlYXRlUnVsZSA9IEVTTGludFV0aWxzLlJ1bGVDcmVhdG9yKCgpID0+IGdldENhbm9uVXJsKFJVTEVfTkFNRSkpXG5cbnR5cGUgTWVzc2FnZUlkcyA9ICdibG9ja05hbWluZ01pc21hdGNoJ1xuXG4vKipcbiAqIENvbnZlcnRzIGEgYmxvY2sgZmlsZW5hbWUgdG8gaXRzIGV4cGVjdGVkIFBhc2NhbENhc2UgZXhwb3J0IG5hbWVcbiAqIGUuZy4sIFwiaGVyby01XCIgLT4gXCJIZXJvNVwiLCBcInNlY3Rpb24tMTBcIiAtPiBcIlNlY3Rpb24xMFwiLCBcImNvbnRlbnQtMzlcIiAtPiBcIkNvbnRlbnQzOVwiXG4gKi9cbmZ1bmN0aW9uIGZpbGVuYW1lVG9QYXNjYWxDYXNlKGZpbGVuYW1lOiBzdHJpbmcpOiBzdHJpbmcge1xuICAvLyBSZW1vdmUgZXh0ZW5zaW9uIGFuZCBzcGxpdCBieSBoeXBoZW5zXG4gIGNvbnN0IGJhc2VOYW1lID0gZmlsZW5hbWUucmVwbGFjZSgvXFwudHN4PyQvLCAnJylcbiAgXG4gIC8vIFNwbGl0IGJ5IGh5cGhlbnMgYW5kIGNvbnZlcnQgZWFjaCBwYXJ0XG4gIHJldHVybiBiYXNlTmFtZVxuICAgIC5zcGxpdCgnLScpXG4gICAgLm1hcCgocGFydCkgPT4ge1xuICAgICAgLy8gQ2FwaXRhbGl6ZSBmaXJzdCBsZXR0ZXIgb2YgZWFjaCBwYXJ0XG4gICAgICByZXR1cm4gcGFydC5jaGFyQXQoMCkudG9VcHBlckNhc2UoKSArIHBhcnQuc2xpY2UoMSlcbiAgICB9KVxuICAgIC5qb2luKCcnKVxufVxuXG5leHBvcnQgZGVmYXVsdCBjcmVhdGVSdWxlPFtdLCBNZXNzYWdlSWRzPih7XG4gIG5hbWU6IFJVTEVfTkFNRSxcbiAgbWV0YToge1xuICAgIHR5cGU6ICdzdWdnZXN0aW9uJyxcbiAgICBkb2NzOiB7XG4gICAgICBkZXNjcmlwdGlvbjogcGF0dGVybj8uc3VtbWFyeSB8fCAnQmxvY2sgZXhwb3J0IG5hbWVzIG11c3QgbWF0Y2ggZmlsZW5hbWUgcGF0dGVybicsXG4gICAgfSxcbiAgICBtZXNzYWdlczoge1xuICAgICAgYmxvY2tOYW1pbmdNaXNtYXRjaDogYFtDYW5vbiAke3BhdHRlcm4/LmlkIHx8ICcwMDYnfV0gQmxvY2sgZXhwb3J0IFwie3thY3R1YWx9fVwiIHNob3VsZCBiZSBcInt7ZXhwZWN0ZWR9fVwiIHRvIG1hdGNoIHRoZSBmaWxlbmFtZSBcInt7ZmlsZW5hbWV9fVwiLiBTZWU6ICR7cGF0dGVybj8udGl0bGUgfHwgJ0Jsb2NrIE5hbWluZyd9IHBhdHRlcm4uYCxcbiAgICB9LFxuICAgIHNjaGVtYTogW10sXG4gIH0sXG4gIGRlZmF1bHRPcHRpb25zOiBbXSxcbiAgY3JlYXRlKGNvbnRleHQpIHtcbiAgICBjb25zdCBmaWxlbmFtZSA9IGNvbnRleHQuZmlsZW5hbWUgfHwgY29udGV4dC5nZXRGaWxlbmFtZSgpXG5cbiAgICAvLyBPbmx5IGNoZWNrIGZpbGVzIGluIHNyYy9ibG9ja3MvXG4gICAgaWYgKCFmaWxlbmFtZS5pbmNsdWRlcygnL2Jsb2Nrcy8nKSAmJiAhZmlsZW5hbWUuaW5jbHVkZXMoJ1xcXFxibG9ja3NcXFxcJykpIHtcbiAgICAgIHJldHVybiB7fVxuICAgIH1cblxuICAgIC8vIEV4dHJhY3QganVzdCB0aGUgZmlsZW5hbWUgKGUuZy4sIFwiaGVyby01LnRzeFwiKVxuICAgIGNvbnN0IG1hdGNoID0gZmlsZW5hbWUubWF0Y2goLyhbXi9cXFxcXStcXC50c3g/KSQvKVxuICAgIGlmICghbWF0Y2gpIHtcbiAgICAgIHJldHVybiB7fVxuICAgIH1cblxuICAgIGNvbnN0IGJsb2NrRmlsZW5hbWUgPSBtYXRjaFsxXVxuICAgIGNvbnN0IGV4cGVjdGVkTmFtZSA9IGZpbGVuYW1lVG9QYXNjYWxDYXNlKGJsb2NrRmlsZW5hbWUpXG5cbiAgICByZXR1cm4ge1xuICAgICAgLy8gQ2hlY2sgZXhwb3J0IGRlZmF1bHQgZnVuY3Rpb24gZGVjbGFyYXRpb25zXG4gICAgICBFeHBvcnREZWZhdWx0RGVjbGFyYXRpb24obm9kZSkge1xuICAgICAgICBpZiAobm9kZS5kZWNsYXJhdGlvbi50eXBlID09PSAnRnVuY3Rpb25EZWNsYXJhdGlvbicgJiYgbm9kZS5kZWNsYXJhdGlvbi5pZCkge1xuICAgICAgICAgIGNvbnN0IGFjdHVhbE5hbWUgPSBub2RlLmRlY2xhcmF0aW9uLmlkLm5hbWVcbiAgICAgICAgICBpZiAoYWN0dWFsTmFtZSAhPT0gZXhwZWN0ZWROYW1lKSB7XG4gICAgICAgICAgICBjb250ZXh0LnJlcG9ydCh7XG4gICAgICAgICAgICAgIG5vZGU6IG5vZGUuZGVjbGFyYXRpb24uaWQsXG4gICAgICAgICAgICAgIG1lc3NhZ2VJZDogJ2Jsb2NrTmFtaW5nTWlzbWF0Y2gnLFxuICAgICAgICAgICAgICBkYXRhOiB7XG4gICAgICAgICAgICAgICAgYWN0dWFsOiBhY3R1YWxOYW1lLFxuICAgICAgICAgICAgICAgIGV4cGVjdGVkOiBleHBlY3RlZE5hbWUsXG4gICAgICAgICAgICAgICAgZmlsZW5hbWU6IGJsb2NrRmlsZW5hbWUsXG4gICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICB9KVxuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfSxcbiAgICB9XG4gIH0sXG59KVxuIl19
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
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@gallop.software/canon",
3
- "version": "2.16.0",
3
+ "version": "2.16.1",
4
4
  "type": "module",
5
5
  "description": "Gallop Canon - Architecture patterns, ESLint plugin, and CLI for template governance",
6
6
  "main": "dist/index.js",