@gallop.software/canon 2.7.1 → 2.8.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/README.md CHANGED
@@ -1,10 +1,10 @@
1
- # @gallop/canon
1
+ # @gallop.software/canon
2
2
 
3
- **Gallop Enterprise Architecture Canon** — A versioned, AI-compatible, auditable system of approved web architecture patterns.
3
+ **Gallop Canon** — A versioned, AI-compatible, auditable system of approved web architecture patterns.
4
4
 
5
5
  ## What is the Canon?
6
6
 
7
- The Canon is a closed set of authoritative patterns that define how serious web applications should be built. Each pattern is:
7
+ The Canon is a closed set of authoritative patterns that define how web applications should be built. Each pattern is:
8
8
 
9
9
  - **Versioned** — Pin to a specific version, upgrade deliberately
10
10
  - **Documented** — Decision-level documentation, not just code comments
@@ -14,28 +14,98 @@ The Canon is a closed set of authoritative patterns that define how serious web
14
14
  ## Installation
15
15
 
16
16
  ```bash
17
- npm install @gallop/canon
17
+ npm install @gallop.software/canon
18
18
  ```
19
19
 
20
- ## Usage
20
+ ## ESLint Setup
21
+
22
+ Add Canon rules to your `eslint.config.mjs`:
23
+
24
+ ```javascript
25
+ import nextConfig from 'eslint-config-next'
26
+ import gallop from '@gallop.software/canon/eslint'
27
+
28
+ export default [
29
+ ...nextConfig,
30
+ {
31
+ files: ['src/blocks/**/*.tsx', 'src/components/**/*.tsx'],
32
+ plugins: {
33
+ gallop,
34
+ },
35
+ rules: {
36
+ ...gallop.recommended,
37
+ },
38
+ },
39
+ ]
40
+ ```
21
41
 
22
- ### Access Pattern Metadata
42
+ ### Opting Out of Rules
23
43
 
24
- ```typescript
25
- import { patterns, getPattern, getPatternsByCategory } from '@gallop/canon'
44
+ To disable specific rules, override them after spreading `recommended`:
26
45
 
27
- // Get all patterns
28
- console.log(patterns)
46
+ ```javascript
47
+ rules: {
48
+ ...gallop.recommended,
49
+ // Disable specific rules
50
+ 'gallop/no-inline-styles': 'off',
51
+ 'gallop/no-arbitrary-colors': 'off',
52
+ }
53
+ ```
29
54
 
30
- // Get a specific pattern
31
- const pattern = getPattern('001')
32
- console.log(pattern.title) // "Server-First Blocks"
55
+ To change a rule from warning to error:
33
56
 
34
- // Get patterns by category
35
- const renderingPatterns = getPatternsByCategory('rendering')
57
+ ```javascript
58
+ rules: {
59
+ ...gallop.recommended,
60
+ 'gallop/no-cross-zone-imports': 'error',
61
+ }
36
62
  ```
37
63
 
38
- ### Pattern Categories
64
+ ### Available Rules
65
+
66
+ | Rule | Description |
67
+ |------|-------------|
68
+ | `gallop/no-client-blocks` | Blocks must be server components |
69
+ | `gallop/no-container-in-section` | No Container inside Section |
70
+ | `gallop/prefer-component-props` | Use props over className for styles |
71
+ | `gallop/prefer-typography-components` | Use Paragraph/Span, not raw tags |
72
+ | `gallop/prefer-layout-components` | Use Grid/Columns, not raw div |
73
+ | `gallop/background-image-rounded` | Background images need rounded prop |
74
+ | `gallop/no-inline-styles` | No style attribute, use Tailwind |
75
+ | `gallop/no-arbitrary-colors` | Use color tokens, not arbitrary values |
76
+ | `gallop/no-cross-zone-imports` | Enforce import boundaries |
77
+ | `gallop/no-data-imports` | No direct _data/ imports in runtime |
78
+
79
+ ## CLI Commands
80
+
81
+ ### Generate AI Rules
82
+
83
+ Generate `.cursorrules` or Copilot instructions from Canon:
84
+
85
+ ```bash
86
+ npx gallop generate .cursorrules
87
+ npx gallop generate .github/copilot-instructions.md
88
+ ```
89
+
90
+ ### Validate Project Structure
91
+
92
+ Check folder structure compliance:
93
+
94
+ ```bash
95
+ npx gallop validate
96
+ npx gallop validate --strict # Exit code 1 on violations
97
+ ```
98
+
99
+ ### Audit Code
100
+
101
+ Check code compliance:
102
+
103
+ ```bash
104
+ npx gallop audit
105
+ npx gallop audit src/blocks/ --strict
106
+ ```
107
+
108
+ ## Pattern Categories
39
109
 
40
110
  | Category | Description |
41
111
  |----------|-------------|
@@ -51,29 +121,6 @@ const renderingPatterns = getPatternsByCategory('rendering')
51
121
 
52
122
  See the [patterns/](./patterns) directory for full documentation of each pattern.
53
123
 
54
- ### Enforced by ESLint
55
-
56
- - **001** Server-First Blocks
57
- - **002** Layout Hierarchy
58
- - **003** Typography Components
59
- - **004** Component Props
60
-
61
- ### Documentation-Only
62
-
63
- - **005** Page Structure
64
- - **006** Block Naming
65
- - **007** Import Paths
66
- - **008** Tailwind Only
67
- - **009** Color Tokens
68
- - **010** Spacing System
69
- - **011** Responsive Mobile-First
70
- - **012** Icon System
71
- - **013** New Component Pattern
72
- - **014** clsx Not classnames
73
- - **015** No Inline Hover Styles
74
- - **016** Client Extraction
75
- - **017** SEO Metadata
76
-
77
124
  ## Guarantees
78
125
 
79
126
  See [guarantees.md](./guarantees.md) for version-specific promises.
@@ -122,6 +122,7 @@ function generateCursorrules() {
122
122
  lines.push('- `Paragraph` - props: `color`, `margin`, `fontSize`, `lineHeight`, `textAlign`');
123
123
  lines.push('- `Span` - props: `color`, `margin`, `fontSize` (inline text, mb-0 default)');
124
124
  lines.push('- `Label` - props: `color`, `margin`, `fontSize`, `fontWeight`, `textAlign`');
125
+ lines.push('- `Quote` - props: `variant`, `color`, `margin`, `fontSize`, `fontWeight`, `textAlign`');
125
126
  lines.push('');
126
127
  lines.push('### Layout');
127
128
  lines.push('- `Section` - semantic section wrapper');
@@ -243,4 +244,4 @@ export async function generate(options) {
243
244
  console.log(` ${patterns.length} patterns, ${guarantees.length} guarantees`);
244
245
  }
245
246
  }
246
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"generate.js","sourceRoot":"","sources":["../../../src/cli/commands/generate.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs'\nimport * as path from 'path'\nimport { version, patterns, categories, guarantees } from '../../index.js'\n\ninterface GenerateOptions {\n  output: string\n  format: 'cursorrules' | 'markdown'\n}\n\n/**\n * Read a pattern markdown file and extract sections\n */\nfunction readPatternFile(patternFile: string): string | null {\n  // Try to find the canon package patterns directory\n  const possiblePaths = [\n    path.join(process.cwd(), 'node_modules/@gallop.software/canon', patternFile),\n    // Fallback for development\n    path.join(import.meta.dirname, '../../../', patternFile),\n  ]\n\n  for (const filePath of possiblePaths) {\n    if (fs.existsSync(filePath)) {\n      return fs.readFileSync(filePath, 'utf-8')\n    }\n  }\n\n  return null\n}\n\n/**\n * Extract examples from pattern markdown\n */\nfunction extractExamples(content: string): { good: string[]; bad: string[] } {\n  const good: string[] = []\n  const bad: string[] = []\n\n  // Find code blocks after \"### Good\" or \"### Bad\" headers\n  const goodMatch = content.match(/### Good\\s*\\n```[\\s\\S]*?```/g)\n  const badMatch = content.match(/### Bad\\s*\\n```[\\s\\S]*?```/g)\n\n  if (goodMatch) {\n    for (const match of goodMatch) {\n      const code = match.replace(/### Good\\s*\\n/, '').trim()\n      good.push(code)\n    }\n  }\n\n  if (badMatch) {\n    for (const match of badMatch) {\n      const code = match.replace(/### Bad\\s*\\n/, '').trim()\n      bad.push(code)\n    }\n  }\n\n  return { good, bad }\n}\n\n/**\n * Generate .cursorrules content from Canon\n */\nfunction generateCursorrules(): string {\n  const lines: string[] = []\n\n  // Header\n  lines.push(`# Gallop Canon v${version} - AI Rules`)\n  lines.push('')\n  lines.push('This file is auto-generated from @gallop.software/canon. Do not edit manually.')\n  lines.push('Regenerate with: npm run generate:ai-rules')\n  lines.push('')\n\n  // Tech stack (Canon-compatible templates)\n  lines.push('## Tech Stack')\n  lines.push('')\n  lines.push('- Next.js 16 with App Router')\n  lines.push('- React 19')\n  lines.push('- TypeScript')\n  lines.push('- Tailwind CSS v4')\n  lines.push('- clsx for conditional class names')\n  lines.push('')\n\n  // Enforced patterns (ESLint rules)\n  lines.push('## Enforced Patterns (ESLint)')\n  lines.push('')\n  lines.push('These patterns are enforced by `@gallop.software/canon/eslint`. Violations will be flagged.')\n  lines.push('')\n\n  const enforcedPatterns = patterns.filter(p => p.enforcement === 'eslint' && p.rule)\n  for (const pattern of enforcedPatterns) {\n    lines.push(`### ${pattern.id}: ${pattern.title}`)\n    lines.push('')\n    lines.push(pattern.summary)\n    lines.push('')\n    lines.push(`- **ESLint Rule:** \\`${pattern.rule}\\``)\n    lines.push(`- **Category:** ${pattern.category}`)\n    lines.push('')\n\n    // Try to read and include examples\n    const content = readPatternFile(pattern.file)\n    if (content) {\n      const { good, bad } = extractExamples(content)\n      if (bad.length > 0) {\n        lines.push('**Bad:**')\n        lines.push('')\n        lines.push(bad[0])\n        lines.push('')\n      }\n      if (good.length > 0) {\n        lines.push('**Good:**')\n        lines.push('')\n        lines.push(good[0])\n        lines.push('')\n      }\n    }\n  }\n\n  // Documentation patterns\n  lines.push('## Documentation Patterns')\n  lines.push('')\n  lines.push('These patterns are not enforced by ESLint but should be followed.')\n  lines.push('')\n\n  const docPatterns = patterns.filter(p => p.enforcement === 'documentation')\n  for (const pattern of docPatterns) {\n    lines.push(`### ${pattern.id}: ${pattern.title}`)\n    lines.push('')\n    lines.push(pattern.summary)\n    lines.push('')\n  }\n\n  // Guarantees\n  lines.push('## Canon Guarantees')\n  lines.push('')\n  lines.push('Following these patterns provides these guarantees:')\n  lines.push('')\n\n  for (const guarantee of guarantees) {\n    lines.push(`- **${guarantee.name}** (${guarantee.id}): Patterns ${guarantee.patterns.join(', ')}`)\n  }\n  lines.push('')\n\n  // Quick reference for components\n  lines.push('## Component Quick Reference')\n  lines.push('')\n  lines.push('### Typography')\n  lines.push('- `Heading` - props: `as`, `color`, `margin`, `fontSize`, `fontWeight`, `textAlign`')\n  lines.push('- `Paragraph` - props: `color`, `margin`, `fontSize`, `lineHeight`, `textAlign`')\n  lines.push('- `Span` - props: `color`, `margin`, `fontSize` (inline text, mb-0 default)')\n  lines.push('- `Label` - props: `color`, `margin`, `fontSize`, `fontWeight`, `textAlign`')\n  lines.push('')\n  lines.push('### Layout')\n  lines.push('- `Section` - semantic section wrapper')\n  lines.push('- `Columns` - grid layout, props: `cols`, `gap`, `align`')\n  lines.push('- `Column` - column child')\n  lines.push('')\n  lines.push('### Interactive')\n  lines.push('- `Button` - props: `href`, `variant`, `icon`, `iconPlacement`, `margin`')\n  lines.push('- `Icon` - Iconify icon wrapper')\n  lines.push('')\n\n  // Do NOT section\n  lines.push('## Do NOT')\n  lines.push('')\n  lines.push('- Use `\\'use client\\'` in blocks - extract to components')\n  lines.push('- Use raw `<p>` or `<span>` - use Paragraph/Span components')\n  lines.push('- Use className for margin/color/fontSize when component has props')\n  lines.push('- Use Container inside Section - Section already provides containment')\n  lines.push('- Use `classnames` package - use `clsx` instead')\n  lines.push('- Use inline styles for hover states - use Tailwind classes')\n  lines.push('')\n\n  // File & Folder Authority section\n  lines.push('## File & Folder Authority')\n  lines.push('')\n  lines.push('These rules govern what AI is allowed and forbidden to do when creating, moving, or modifying files and folders.')\n  lines.push('')\n\n  lines.push('### Defined `/src` Structure')\n  lines.push('')\n  lines.push('```')\n  lines.push('src/')\n  lines.push('├── app/          # Routes, layouts, metadata (Next.js App Router)')\n  lines.push('├── blocks/       # Page-level content sections')\n  lines.push('├── blog/         # Blog content (archive content type)')\n  lines.push('├── components/   # Reusable UI primitives')\n  lines.push('├── hooks/        # Custom React hooks')\n  lines.push('├── styles/       # CSS, Tailwind, fonts')\n  lines.push('├── template/     # Template-level components')\n  lines.push('├── tools/        # Utility tools')\n  lines.push('├── types/        # TypeScript types')\n  lines.push('├── utils/        # Utility functions')\n  lines.push('└── state.ts      # Global state')\n  lines.push('```')\n  lines.push('')\n\n  lines.push('### App Router Structure')\n  lines.push('')\n  lines.push('Routes must use Next.js route groups. At minimum, `(default)` must exist:')\n  lines.push('')\n  lines.push('```')\n  lines.push('src/app/')\n  lines.push('├── (default)/        # Required - default layout group')\n  lines.push('│   ├── layout.tsx')\n  lines.push('│   └── {routes}/')\n  lines.push('├── (hero)/           # Optional - hero layout variant')\n  lines.push('├── api/              # API routes (exception - no grouping)')\n  lines.push('├── layout.tsx        # Root layout')\n  lines.push('└── metadata.tsx      # Shared metadata')\n  lines.push('```')\n  lines.push('')\n  lines.push('- All page routes must be inside a route group (parentheses folder)')\n  lines.push('- Never create routes directly under `src/app/` (except `api/`, root files)')\n  lines.push('- New route groups are allowed freely when a new layout variant is needed')\n  lines.push('')\n\n  lines.push('### File Structure Rules')\n  lines.push('')\n  lines.push('**Blocks:**')\n  lines.push('- Always single files directly in `src/blocks/`')\n  lines.push('- Never create folders inside `src/blocks/`')\n  lines.push('- Example: `src/blocks/hero-1.tsx`, `src/blocks/testimonial-3.tsx`')\n  lines.push('')\n  lines.push('**Components:**')\n  lines.push('- Simple components: Single file in `src/components/`')\n  lines.push('- Complex components: Folder with `index.tsx`')\n  lines.push('- Use folders when component has multiple sub-files')\n  lines.push('')\n\n  lines.push('### DO - What AI IS Allowed To Do')\n  lines.push('')\n  lines.push('- Create files only inside existing Canon-defined zones')\n  lines.push('- Place new files in the zone that matches their architectural role')\n  lines.push('- Follow existing folder conventions within a zone')\n  lines.push('- Reuse existing folders when possible')\n  lines.push('- Create new route groups in `src/app/` when new layouts are needed')\n  lines.push('- Create new archive content folders (like `blog/`, `portfolio/`) in `/src`')\n  lines.push('- Create dotfiles/directories at project root (`.github/`, `.cursor/`, etc.)')\n  lines.push('- Ask for confirmation if the correct zone is ambiguous')\n  lines.push('')\n\n  lines.push('### DO NOT - What AI Is Forbidden To Do')\n  lines.push('')\n  lines.push('- Create new top-level directories (except dotfiles)')\n  lines.push('- Create new folders in `/src` (except archive content or route groups)')\n  lines.push('- Place files outside Canon-defined zones')\n  lines.push('- Mix responsibilities across zones (components importing blocks, etc.)')\n  lines.push('- Reorganize or move folders without explicit instruction')\n  lines.push('- Invent new organizational conventions')\n  lines.push('- Create placeholder or speculative files')\n  lines.push('- Import from `_scripts/` or `_data/` in runtime code')\n  lines.push('- Manually edit files in `_data/` (generated only)')\n  lines.push('')\n\n  // Post-edit verification\n  lines.push('## Post-Edit Verification')\n  lines.push('')\n  lines.push('After editing files:')\n  lines.push('1. Run `npm run lint` to check for errors')\n  lines.push('2. Fix any violations before committing')\n  lines.push('')\n  lines.push('Note: Only lint files you edited, not the entire codebase.')\n  lines.push('')\n\n  return lines.join('\\n')\n}\n\nexport async function generate(options: GenerateOptions): Promise<void> {\n  const content = generateCursorrules()\n\n  if (options.output === '-') {\n    // Output to stdout\n    console.log(content)\n  } else {\n    // Write to file\n    const outputPath = path.resolve(process.cwd(), options.output)\n    fs.writeFileSync(outputPath, content, 'utf-8')\n    console.log(`✓ Generated ${options.output} from Canon v${version}`)\n    console.log(`  ${patterns.length} patterns, ${guarantees.length} guarantees`)\n  }\n}\n"]}
247
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"generate.js","sourceRoot":"","sources":["../../../src/cli/commands/generate.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs'\nimport * as path from 'path'\nimport { version, patterns, categories, guarantees } from '../../index.js'\n\ninterface GenerateOptions {\n  output: string\n  format: 'cursorrules' | 'markdown'\n}\n\n/**\n * Read a pattern markdown file and extract sections\n */\nfunction readPatternFile(patternFile: string): string | null {\n  // Try to find the canon package patterns directory\n  const possiblePaths = [\n    path.join(process.cwd(), 'node_modules/@gallop.software/canon', patternFile),\n    // Fallback for development\n    path.join(import.meta.dirname, '../../../', patternFile),\n  ]\n\n  for (const filePath of possiblePaths) {\n    if (fs.existsSync(filePath)) {\n      return fs.readFileSync(filePath, 'utf-8')\n    }\n  }\n\n  return null\n}\n\n/**\n * Extract examples from pattern markdown\n */\nfunction extractExamples(content: string): { good: string[]; bad: string[] } {\n  const good: string[] = []\n  const bad: string[] = []\n\n  // Find code blocks after \"### Good\" or \"### Bad\" headers\n  const goodMatch = content.match(/### Good\\s*\\n```[\\s\\S]*?```/g)\n  const badMatch = content.match(/### Bad\\s*\\n```[\\s\\S]*?```/g)\n\n  if (goodMatch) {\n    for (const match of goodMatch) {\n      const code = match.replace(/### Good\\s*\\n/, '').trim()\n      good.push(code)\n    }\n  }\n\n  if (badMatch) {\n    for (const match of badMatch) {\n      const code = match.replace(/### Bad\\s*\\n/, '').trim()\n      bad.push(code)\n    }\n  }\n\n  return { good, bad }\n}\n\n/**\n * Generate .cursorrules content from Canon\n */\nfunction generateCursorrules(): string {\n  const lines: string[] = []\n\n  // Header\n  lines.push(`# Gallop Canon v${version} - AI Rules`)\n  lines.push('')\n  lines.push('This file is auto-generated from @gallop.software/canon. Do not edit manually.')\n  lines.push('Regenerate with: npm run generate:ai-rules')\n  lines.push('')\n\n  // Tech stack (Canon-compatible templates)\n  lines.push('## Tech Stack')\n  lines.push('')\n  lines.push('- Next.js 16 with App Router')\n  lines.push('- React 19')\n  lines.push('- TypeScript')\n  lines.push('- Tailwind CSS v4')\n  lines.push('- clsx for conditional class names')\n  lines.push('')\n\n  // Enforced patterns (ESLint rules)\n  lines.push('## Enforced Patterns (ESLint)')\n  lines.push('')\n  lines.push('These patterns are enforced by `@gallop.software/canon/eslint`. Violations will be flagged.')\n  lines.push('')\n\n  const enforcedPatterns = patterns.filter(p => p.enforcement === 'eslint' && p.rule)\n  for (const pattern of enforcedPatterns) {\n    lines.push(`### ${pattern.id}: ${pattern.title}`)\n    lines.push('')\n    lines.push(pattern.summary)\n    lines.push('')\n    lines.push(`- **ESLint Rule:** \\`${pattern.rule}\\``)\n    lines.push(`- **Category:** ${pattern.category}`)\n    lines.push('')\n\n    // Try to read and include examples\n    const content = readPatternFile(pattern.file)\n    if (content) {\n      const { good, bad } = extractExamples(content)\n      if (bad.length > 0) {\n        lines.push('**Bad:**')\n        lines.push('')\n        lines.push(bad[0])\n        lines.push('')\n      }\n      if (good.length > 0) {\n        lines.push('**Good:**')\n        lines.push('')\n        lines.push(good[0])\n        lines.push('')\n      }\n    }\n  }\n\n  // Documentation patterns\n  lines.push('## Documentation Patterns')\n  lines.push('')\n  lines.push('These patterns are not enforced by ESLint but should be followed.')\n  lines.push('')\n\n  const docPatterns = patterns.filter(p => p.enforcement === 'documentation')\n  for (const pattern of docPatterns) {\n    lines.push(`### ${pattern.id}: ${pattern.title}`)\n    lines.push('')\n    lines.push(pattern.summary)\n    lines.push('')\n  }\n\n  // Guarantees\n  lines.push('## Canon Guarantees')\n  lines.push('')\n  lines.push('Following these patterns provides these guarantees:')\n  lines.push('')\n\n  for (const guarantee of guarantees) {\n    lines.push(`- **${guarantee.name}** (${guarantee.id}): Patterns ${guarantee.patterns.join(', ')}`)\n  }\n  lines.push('')\n\n  // Quick reference for components\n  lines.push('## Component Quick Reference')\n  lines.push('')\n  lines.push('### Typography')\n  lines.push('- `Heading` - props: `as`, `color`, `margin`, `fontSize`, `fontWeight`, `textAlign`')\n  lines.push('- `Paragraph` - props: `color`, `margin`, `fontSize`, `lineHeight`, `textAlign`')\n  lines.push('- `Span` - props: `color`, `margin`, `fontSize` (inline text, mb-0 default)')\n  lines.push('- `Label` - props: `color`, `margin`, `fontSize`, `fontWeight`, `textAlign`')\n  lines.push('- `Quote` - props: `variant`, `color`, `margin`, `fontSize`, `fontWeight`, `textAlign`')\n  lines.push('')\n  lines.push('### Layout')\n  lines.push('- `Section` - semantic section wrapper')\n  lines.push('- `Columns` - grid layout, props: `cols`, `gap`, `align`')\n  lines.push('- `Column` - column child')\n  lines.push('')\n  lines.push('### Interactive')\n  lines.push('- `Button` - props: `href`, `variant`, `icon`, `iconPlacement`, `margin`')\n  lines.push('- `Icon` - Iconify icon wrapper')\n  lines.push('')\n\n  // Do NOT section\n  lines.push('## Do NOT')\n  lines.push('')\n  lines.push('- Use `\\'use client\\'` in blocks - extract to components')\n  lines.push('- Use raw `<p>` or `<span>` - use Paragraph/Span components')\n  lines.push('- Use className for margin/color/fontSize when component has props')\n  lines.push('- Use Container inside Section - Section already provides containment')\n  lines.push('- Use `classnames` package - use `clsx` instead')\n  lines.push('- Use inline styles for hover states - use Tailwind classes')\n  lines.push('')\n\n  // File & Folder Authority section\n  lines.push('## File & Folder Authority')\n  lines.push('')\n  lines.push('These rules govern what AI is allowed and forbidden to do when creating, moving, or modifying files and folders.')\n  lines.push('')\n\n  lines.push('### Defined `/src` Structure')\n  lines.push('')\n  lines.push('```')\n  lines.push('src/')\n  lines.push('├── app/          # Routes, layouts, metadata (Next.js App Router)')\n  lines.push('├── blocks/       # Page-level content sections')\n  lines.push('├── blog/         # Blog content (archive content type)')\n  lines.push('├── components/   # Reusable UI primitives')\n  lines.push('├── hooks/        # Custom React hooks')\n  lines.push('├── styles/       # CSS, Tailwind, fonts')\n  lines.push('├── template/     # Template-level components')\n  lines.push('├── tools/        # Utility tools')\n  lines.push('├── types/        # TypeScript types')\n  lines.push('├── utils/        # Utility functions')\n  lines.push('└── state.ts      # Global state')\n  lines.push('```')\n  lines.push('')\n\n  lines.push('### App Router Structure')\n  lines.push('')\n  lines.push('Routes must use Next.js route groups. At minimum, `(default)` must exist:')\n  lines.push('')\n  lines.push('```')\n  lines.push('src/app/')\n  lines.push('├── (default)/        # Required - default layout group')\n  lines.push('│   ├── layout.tsx')\n  lines.push('│   └── {routes}/')\n  lines.push('├── (hero)/           # Optional - hero layout variant')\n  lines.push('├── api/              # API routes (exception - no grouping)')\n  lines.push('├── layout.tsx        # Root layout')\n  lines.push('└── metadata.tsx      # Shared metadata')\n  lines.push('```')\n  lines.push('')\n  lines.push('- All page routes must be inside a route group (parentheses folder)')\n  lines.push('- Never create routes directly under `src/app/` (except `api/`, root files)')\n  lines.push('- New route groups are allowed freely when a new layout variant is needed')\n  lines.push('')\n\n  lines.push('### File Structure Rules')\n  lines.push('')\n  lines.push('**Blocks:**')\n  lines.push('- Always single files directly in `src/blocks/`')\n  lines.push('- Never create folders inside `src/blocks/`')\n  lines.push('- Example: `src/blocks/hero-1.tsx`, `src/blocks/testimonial-3.tsx`')\n  lines.push('')\n  lines.push('**Components:**')\n  lines.push('- Simple components: Single file in `src/components/`')\n  lines.push('- Complex components: Folder with `index.tsx`')\n  lines.push('- Use folders when component has multiple sub-files')\n  lines.push('')\n\n  lines.push('### DO - What AI IS Allowed To Do')\n  lines.push('')\n  lines.push('- Create files only inside existing Canon-defined zones')\n  lines.push('- Place new files in the zone that matches their architectural role')\n  lines.push('- Follow existing folder conventions within a zone')\n  lines.push('- Reuse existing folders when possible')\n  lines.push('- Create new route groups in `src/app/` when new layouts are needed')\n  lines.push('- Create new archive content folders (like `blog/`, `portfolio/`) in `/src`')\n  lines.push('- Create dotfiles/directories at project root (`.github/`, `.cursor/`, etc.)')\n  lines.push('- Ask for confirmation if the correct zone is ambiguous')\n  lines.push('')\n\n  lines.push('### DO NOT - What AI Is Forbidden To Do')\n  lines.push('')\n  lines.push('- Create new top-level directories (except dotfiles)')\n  lines.push('- Create new folders in `/src` (except archive content or route groups)')\n  lines.push('- Place files outside Canon-defined zones')\n  lines.push('- Mix responsibilities across zones (components importing blocks, etc.)')\n  lines.push('- Reorganize or move folders without explicit instruction')\n  lines.push('- Invent new organizational conventions')\n  lines.push('- Create placeholder or speculative files')\n  lines.push('- Import from `_scripts/` or `_data/` in runtime code')\n  lines.push('- Manually edit files in `_data/` (generated only)')\n  lines.push('')\n\n  // Post-edit verification\n  lines.push('## Post-Edit Verification')\n  lines.push('')\n  lines.push('After editing files:')\n  lines.push('1. Run `npm run lint` to check for errors')\n  lines.push('2. Fix any violations before committing')\n  lines.push('')\n  lines.push('Note: Only lint files you edited, not the entire codebase.')\n  lines.push('')\n\n  return lines.join('\\n')\n}\n\nexport async function generate(options: GenerateOptions): Promise<void> {\n  const content = generateCursorrules()\n\n  if (options.output === '-') {\n    // Output to stdout\n    console.log(content)\n  } else {\n    // Write to file\n    const outputPath = path.resolve(process.cwd(), options.output)\n    fs.writeFileSync(outputPath, content, 'utf-8')\n    console.log(`✓ Generated ${options.output} from Canon v${version}`)\n    console.log(`  ${patterns.length} patterns, ${guarantees.length} guarantees`)\n  }\n}\n"]}
@@ -34,6 +34,13 @@ const componentPropMappings = {
34
34
  margin: /^m([by])?-/, // m- (all), mb- (bottom), my- (y-axis) - all affect bottom margin
35
35
  color: /^text-(body|contrast|accent|white|black)/,
36
36
  },
37
+ Quote: {
38
+ margin: /^m([by])?-/, // m- (all), mb- (bottom), my- (y-axis) - all affect bottom margin
39
+ color: /^text-(body|contrast|accent|white|black)/,
40
+ fontSize: /^text-(xs|sm|base|lg|xl|2xl|3xl|4xl|5xl|6xl|7xl|8xl|9xl)$/,
41
+ fontWeight: /^font-(thin|extralight|light|normal|medium|semibold|bold|extrabold|black)$/,
42
+ textAlign: /^text-(left|center|right|justify)$/,
43
+ },
37
44
  };
38
45
  export default createRule({
39
46
  name: RULE_NAME,
@@ -97,4 +104,4 @@ export default createRule({
97
104
  };
98
105
  },
99
106
  });
100
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"prefer-component-props.js","sourceRoot":"","sources":["../../../src/eslint/rules/prefer-component-props.ts"],"names":[],"mappings":"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","sourcesContent":["import { ESLintUtils, TSESTree } from '@typescript-eslint/utils'\nimport { getCanonUrl, getCanonPattern } from '../utils/canon.js'\n\nconst RULE_NAME = 'prefer-component-props'\nconst pattern = getCanonPattern(RULE_NAME)\n\nconst createRule = ESLintUtils.RuleCreator(() => getCanonUrl(RULE_NAME))\n\ntype MessageIds = 'preferComponentProps'\n\n// Map of component names to their style props and corresponding Tailwind patterns\nconst componentPropMappings: Record<string, Record<string, RegExp>> = {\n  Paragraph: {\n    margin: /^m([by])?-/, // m- (all), mb- (bottom), my- (y-axis) - all affect bottom margin\n    color: /^text-(body|contrast|accent|white|black)/,\n    fontSize: /^text-(xs|sm|base|lg|xl|2xl|3xl|4xl|5xl|6xl|7xl|8xl|9xl)$/,\n    lineHeight: /^leading-/,\n    textAlign: /^text-(left|center|right|justify)$/,\n    fontWeight: /^font-(thin|extralight|light|normal|medium|semibold|bold|extrabold|black)$/,\n  },\n  Heading: {\n    margin: /^m([by])?-/, // m- (all), mb- (bottom), my- (y-axis) - all affect bottom margin\n    color: /^text-(body|contrast|accent|white|black)/,\n    fontSize: /^text-(xs|sm|base|lg|xl|2xl|3xl|4xl|5xl|6xl|7xl|8xl|9xl)$/,\n    lineHeight: /^leading-/,\n    textAlign: /^text-(left|center|right|justify)$/,\n    fontWeight: /^font-(thin|extralight|light|normal|medium|semibold|bold|extrabold|black)$/,\n  },\n  Accent: {\n    margin: /^m([by])?-/, // m- (all), mb- (bottom), my- (y-axis) - all affect bottom margin\n    color: /^text-(body|contrast|accent|white|black)/,\n    size: /^text-(xs|sm|base|lg|xl|2xl|3xl|4xl|5xl|6xl|7xl|8xl|9xl)$/,\n    textAlign: /^text-(left|center|right|justify)$/,\n  },\n  Button: {\n    margin: /^m([by])?-/, // m- (all), mb- (bottom), my- (y-axis) - all affect bottom margin\n  },\n  Label: {\n    margin: /^m([by])?-/, // m- (all), mb- (bottom), my- (y-axis) - all affect bottom margin\n    color: /^text-(body|contrast|accent|white|black)/,\n  },\n}\n\nexport default createRule<[], MessageIds>({\n  name: RULE_NAME,\n  meta: {\n    type: 'suggestion',\n    docs: {\n      description: pattern?.summary || 'Use props over className for supported styles',\n    },\n    messages: {\n      preferComponentProps: `[Canon ${pattern?.id || '004'}] \"{{className}}\" in className should use the \"{{propName}}\" prop instead. Replace className=\"{{className}}\" with {{propName}}=\"{{className}}\".`,\n    },\n    schema: [],\n  },\n  defaultOptions: [],\n  create(context) {\n    return {\n      JSXOpeningElement(node) {\n        // Get the component name\n        if (node.name.type !== 'JSXIdentifier') return\n        const componentName = node.name.name\n\n        // Check if this component has prop mappings\n        const propMappings = componentPropMappings[componentName]\n        if (!propMappings) return\n\n        // Find the className attribute\n        const classNameAttr = node.attributes.find(\n          (attr): attr is TSESTree.JSXAttribute =>\n            attr.type === 'JSXAttribute' &&\n            attr.name.type === 'JSXIdentifier' &&\n            attr.name.name === 'className'\n        )\n\n        if (!classNameAttr || !classNameAttr.value) return\n\n        // Extract class string value\n        let classValue: string | null = null\n\n        if (classNameAttr.value.type === 'Literal') {\n          classValue = String(classNameAttr.value.value)\n        } else if (\n          classNameAttr.value.type === 'JSXExpressionContainer' &&\n          classNameAttr.value.expression.type === 'Literal'\n        ) {\n          classValue = String(classNameAttr.value.expression.value)\n        }\n\n        if (!classValue) return\n\n        // Split into individual classes and check each\n        const classes = classValue.split(/\\s+/).filter(Boolean)\n\n        for (const cls of classes) {\n          for (const [propName, pattern] of Object.entries(propMappings)) {\n            if (pattern.test(cls)) {\n              context.report({\n                node: classNameAttr,\n                messageId: 'preferComponentProps',\n                data: {\n                  className: cls,\n                  propName,\n                },\n              })\n              break // Only report once per class\n            }\n          }\n        }\n      },\n    }\n  },\n})\n"]}
107
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"prefer-component-props.js","sourceRoot":"","sources":["../../../src/eslint/rules/prefer-component-props.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAY,MAAM,0BAA0B,CAAA;AAChE,OAAO,EAAE,WAAW,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAA;AAEhE,MAAM,SAAS,GAAG,wBAAwB,CAAA;AAC1C,MAAM,OAAO,GAAG,eAAe,CAAC,SAAS,CAAC,CAAA;AAE1C,MAAM,UAAU,GAAG,WAAW,CAAC,WAAW,CAAC,GAAG,EAAE,CAAC,WAAW,CAAC,SAAS,CAAC,CAAC,CAAA;AAIxE,kFAAkF;AAClF,MAAM,qBAAqB,GAA2C;IACpE,SAAS,EAAE;QACT,MAAM,EAAE,YAAY,EAAE,kEAAkE;QACxF,KAAK,EAAE,0CAA0C;QACjD,QAAQ,EAAE,2DAA2D;QACrE,UAAU,EAAE,WAAW;QACvB,SAAS,EAAE,oCAAoC;QAC/C,UAAU,EAAE,4EAA4E;KACzF;IACD,OAAO,EAAE;QACP,MAAM,EAAE,YAAY,EAAE,kEAAkE;QACxF,KAAK,EAAE,0CAA0C;QACjD,QAAQ,EAAE,2DAA2D;QACrE,UAAU,EAAE,WAAW;QACvB,SAAS,EAAE,oCAAoC;QAC/C,UAAU,EAAE,4EAA4E;KACzF;IACD,MAAM,EAAE;QACN,MAAM,EAAE,YAAY,EAAE,kEAAkE;QACxF,KAAK,EAAE,0CAA0C;QACjD,IAAI,EAAE,2DAA2D;QACjE,SAAS,EAAE,oCAAoC;KAChD;IACD,MAAM,EAAE;QACN,MAAM,EAAE,YAAY,EAAE,kEAAkE;KACzF;IACD,KAAK,EAAE;QACL,MAAM,EAAE,YAAY,EAAE,kEAAkE;QACxF,KAAK,EAAE,0CAA0C;KAClD;IACD,KAAK,EAAE;QACL,MAAM,EAAE,YAAY,EAAE,kEAAkE;QACxF,KAAK,EAAE,0CAA0C;QACjD,QAAQ,EAAE,2DAA2D;QACrE,UAAU,EAAE,4EAA4E;QACxF,SAAS,EAAE,oCAAoC;KAChD;CACF,CAAA;AAED,eAAe,UAAU,CAAiB;IACxC,IAAI,EAAE,SAAS;IACf,IAAI,EAAE;QACJ,IAAI,EAAE,YAAY;QAClB,IAAI,EAAE;YACJ,WAAW,EAAE,OAAO,EAAE,OAAO,IAAI,+CAA+C;SACjF;QACD,QAAQ,EAAE;YACR,oBAAoB,EAAE,UAAU,OAAO,EAAE,EAAE,IAAI,KAAK,iJAAiJ;SACtM;QACD,MAAM,EAAE,EAAE;KACX;IACD,cAAc,EAAE,EAAE;IAClB,MAAM,CAAC,OAAO;QACZ,OAAO;YACL,iBAAiB,CAAC,IAAI;gBACpB,yBAAyB;gBACzB,IAAI,IAAI,CAAC,IAAI,CAAC,IAAI,KAAK,eAAe;oBAAE,OAAM;gBAC9C,MAAM,aAAa,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,CAAA;gBAEpC,4CAA4C;gBAC5C,MAAM,YAAY,GAAG,qBAAqB,CAAC,aAAa,CAAC,CAAA;gBACzD,IAAI,CAAC,YAAY;oBAAE,OAAM;gBAEzB,+BAA+B;gBAC/B,MAAM,aAAa,GAAG,IAAI,CAAC,UAAU,CAAC,IAAI,CACxC,CAAC,IAAI,EAAiC,EAAE,CACtC,IAAI,CAAC,IAAI,KAAK,cAAc;oBAC5B,IAAI,CAAC,IAAI,CAAC,IAAI,KAAK,eAAe;oBAClC,IAAI,CAAC,IAAI,CAAC,IAAI,KAAK,WAAW,CACjC,CAAA;gBAED,IAAI,CAAC,aAAa,IAAI,CAAC,aAAa,CAAC,KAAK;oBAAE,OAAM;gBAElD,6BAA6B;gBAC7B,IAAI,UAAU,GAAkB,IAAI,CAAA;gBAEpC,IAAI,aAAa,CAAC,KAAK,CAAC,IAAI,KAAK,SAAS,EAAE,CAAC;oBAC3C,UAAU,GAAG,MAAM,CAAC,aAAa,CAAC,KAAK,CAAC,KAAK,CAAC,CAAA;gBAChD,CAAC;qBAAM,IACL,aAAa,CAAC,KAAK,CAAC,IAAI,KAAK,wBAAwB;oBACrD,aAAa,CAAC,KAAK,CAAC,UAAU,CAAC,IAAI,KAAK,SAAS,EACjD,CAAC;oBACD,UAAU,GAAG,MAAM,CAAC,aAAa,CAAC,KAAK,CAAC,UAAU,CAAC,KAAK,CAAC,CAAA;gBAC3D,CAAC;gBAED,IAAI,CAAC,UAAU;oBAAE,OAAM;gBAEvB,+CAA+C;gBAC/C,MAAM,OAAO,GAAG,UAAU,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAA;gBAEvD,KAAK,MAAM,GAAG,IAAI,OAAO,EAAE,CAAC;oBAC1B,KAAK,MAAM,CAAC,QAAQ,EAAE,OAAO,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,YAAY,CAAC,EAAE,CAAC;wBAC/D,IAAI,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC;4BACtB,OAAO,CAAC,MAAM,CAAC;gCACb,IAAI,EAAE,aAAa;gCACnB,SAAS,EAAE,sBAAsB;gCACjC,IAAI,EAAE;oCACJ,SAAS,EAAE,GAAG;oCACd,QAAQ;iCACT;6BACF,CAAC,CAAA;4BACF,MAAK,CAAC,6BAA6B;wBACrC,CAAC;oBACH,CAAC;gBACH,CAAC;YACH,CAAC;SACF,CAAA;IACH,CAAC;CACF,CAAC,CAAA","sourcesContent":["import { ESLintUtils, TSESTree } from '@typescript-eslint/utils'\nimport { getCanonUrl, getCanonPattern } from '../utils/canon.js'\n\nconst RULE_NAME = 'prefer-component-props'\nconst pattern = getCanonPattern(RULE_NAME)\n\nconst createRule = ESLintUtils.RuleCreator(() => getCanonUrl(RULE_NAME))\n\ntype MessageIds = 'preferComponentProps'\n\n// Map of component names to their style props and corresponding Tailwind patterns\nconst componentPropMappings: Record<string, Record<string, RegExp>> = {\n  Paragraph: {\n    margin: /^m([by])?-/, // m- (all), mb- (bottom), my- (y-axis) - all affect bottom margin\n    color: /^text-(body|contrast|accent|white|black)/,\n    fontSize: /^text-(xs|sm|base|lg|xl|2xl|3xl|4xl|5xl|6xl|7xl|8xl|9xl)$/,\n    lineHeight: /^leading-/,\n    textAlign: /^text-(left|center|right|justify)$/,\n    fontWeight: /^font-(thin|extralight|light|normal|medium|semibold|bold|extrabold|black)$/,\n  },\n  Heading: {\n    margin: /^m([by])?-/, // m- (all), mb- (bottom), my- (y-axis) - all affect bottom margin\n    color: /^text-(body|contrast|accent|white|black)/,\n    fontSize: /^text-(xs|sm|base|lg|xl|2xl|3xl|4xl|5xl|6xl|7xl|8xl|9xl)$/,\n    lineHeight: /^leading-/,\n    textAlign: /^text-(left|center|right|justify)$/,\n    fontWeight: /^font-(thin|extralight|light|normal|medium|semibold|bold|extrabold|black)$/,\n  },\n  Accent: {\n    margin: /^m([by])?-/, // m- (all), mb- (bottom), my- (y-axis) - all affect bottom margin\n    color: /^text-(body|contrast|accent|white|black)/,\n    size: /^text-(xs|sm|base|lg|xl|2xl|3xl|4xl|5xl|6xl|7xl|8xl|9xl)$/,\n    textAlign: /^text-(left|center|right|justify)$/,\n  },\n  Button: {\n    margin: /^m([by])?-/, // m- (all), mb- (bottom), my- (y-axis) - all affect bottom margin\n  },\n  Label: {\n    margin: /^m([by])?-/, // m- (all), mb- (bottom), my- (y-axis) - all affect bottom margin\n    color: /^text-(body|contrast|accent|white|black)/,\n  },\n  Quote: {\n    margin: /^m([by])?-/, // m- (all), mb- (bottom), my- (y-axis) - all affect bottom margin\n    color: /^text-(body|contrast|accent|white|black)/,\n    fontSize: /^text-(xs|sm|base|lg|xl|2xl|3xl|4xl|5xl|6xl|7xl|8xl|9xl)$/,\n    fontWeight: /^font-(thin|extralight|light|normal|medium|semibold|bold|extrabold|black)$/,\n    textAlign: /^text-(left|center|right|justify)$/,\n  },\n}\n\nexport default createRule<[], MessageIds>({\n  name: RULE_NAME,\n  meta: {\n    type: 'suggestion',\n    docs: {\n      description: pattern?.summary || 'Use props over className for supported styles',\n    },\n    messages: {\n      preferComponentProps: `[Canon ${pattern?.id || '004'}] \"{{className}}\" in className should use the \"{{propName}}\" prop instead. Replace className=\"{{className}}\" with {{propName}}=\"{{className}}\".`,\n    },\n    schema: [],\n  },\n  defaultOptions: [],\n  create(context) {\n    return {\n      JSXOpeningElement(node) {\n        // Get the component name\n        if (node.name.type !== 'JSXIdentifier') return\n        const componentName = node.name.name\n\n        // Check if this component has prop mappings\n        const propMappings = componentPropMappings[componentName]\n        if (!propMappings) return\n\n        // Find the className attribute\n        const classNameAttr = node.attributes.find(\n          (attr): attr is TSESTree.JSXAttribute =>\n            attr.type === 'JSXAttribute' &&\n            attr.name.type === 'JSXIdentifier' &&\n            attr.name.name === 'className'\n        )\n\n        if (!classNameAttr || !classNameAttr.value) return\n\n        // Extract class string value\n        let classValue: string | null = null\n\n        if (classNameAttr.value.type === 'Literal') {\n          classValue = String(classNameAttr.value.value)\n        } else if (\n          classNameAttr.value.type === 'JSXExpressionContainer' &&\n          classNameAttr.value.expression.type === 'Literal'\n        ) {\n          classValue = String(classNameAttr.value.expression.value)\n        }\n\n        if (!classValue) return\n\n        // Split into individual classes and check each\n        const classes = classValue.split(/\\s+/).filter(Boolean)\n\n        for (const cls of classes) {\n          for (const [propName, pattern] of Object.entries(propMappings)) {\n            if (pattern.test(cls)) {\n              context.report({\n                node: classNameAttr,\n                messageId: 'preferComponentProps',\n                data: {\n                  className: cls,\n                  propName,\n                },\n              })\n              break // Only report once per class\n            }\n          }\n        }\n      },\n    }\n  },\n})\n"]}
@@ -12,6 +12,7 @@ const rule = {
12
12
  messages: {
13
13
  useParagraph: `[Canon ${pattern?.id || '003'}] Use the Paragraph component instead of <p>. Import: import { Paragraph } from "@/components"`,
14
14
  useSpan: `[Canon ${pattern?.id || '003'}] Use the Span component instead of <span> for text content. Import: import { Span } from "@/components"`,
15
+ useQuote: `[Canon ${pattern?.id || '003'}] Use the Quote component instead of <blockquote>. Import: import { Quote } from "@/components"`,
15
16
  useTypographyForDiv: `[Canon ${pattern?.id || '003'}] Use a typography component (Heading, Paragraph, Label, etc.) instead of <div> with text content.`,
16
17
  },
17
18
  schema: [],
@@ -95,6 +96,14 @@ const rule = {
95
96
  });
96
97
  return;
97
98
  }
99
+ // Check <blockquote> tags
100
+ if (elementName === 'blockquote') {
101
+ context.report({
102
+ node,
103
+ messageId: 'useQuote',
104
+ });
105
+ return;
106
+ }
98
107
  // Check <span> tags
99
108
  if (elementName === 'span') {
100
109
  if (isInsideTypographyComponent(node)) {
@@ -132,4 +141,4 @@ const rule = {
132
141
  },
133
142
  };
134
143
  export default rule;
135
- //# 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\"`,\n      useSpan: `[Canon ${pattern?.id || '003'}] Use the Span component instead of <span> for text content. Import: import { Span } from \"@/components\"`,\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 <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"]}
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\"`,\n      useSpan: `[Canon ${pattern?.id || '003'}] Use the Span component instead of <span> for text content. Import: import { Span } from \"@/components\"`,\n      useQuote: `[Canon ${pattern?.id || '003'}] Use the Quote component instead of <blockquote>. Import: import { Quote } from \"@/components\"`,\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"]}
package/dist/index.d.ts CHANGED
@@ -1,7 +1,7 @@
1
1
  /**
2
2
  * @gallop/canon
3
3
  *
4
- * Gallop Enterprise Architecture Canon
4
+ * Gallop Canon
5
5
  * Versioned, AI-compatible, auditable web architecture patterns
6
6
  */
7
7
  export interface Pattern {
package/dist/index.js CHANGED
@@ -1,7 +1,7 @@
1
1
  /**
2
2
  * @gallop/canon
3
3
  *
4
- * Gallop Enterprise Architecture Canon
4
+ * Gallop Canon
5
5
  * Versioned, AI-compatible, auditable web architecture patterns
6
6
  */
7
7
  import schema from '../schema.json' with { type: 'json' };
@@ -99,4 +99,4 @@ export default {
99
99
  isValidPattern,
100
100
  getEnforcementStats,
101
101
  };
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;AA8CzD,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 Enterprise Architecture Canon\n * Versioned, AI-compatible, auditable web architecture patterns\n */\n\nimport schema from '../schema.json' with { type: 'json' }\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"]}
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;AA8CzD,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// 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"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@gallop.software/canon",
3
- "version": "2.7.1",
3
+ "version": "2.8.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",
package/schema.json CHANGED
@@ -2,7 +2,7 @@
2
2
  "$schema": "https://json-schema.org/draft/2020-12/schema",
3
3
  "name": "Gallop Canon",
4
4
  "version": "1.0.0",
5
- "description": "Gallop Enterprise Architecture Canon - Versioned, AI-compatible, auditable web architecture patterns",
5
+ "description": "Gallop Canon - Versioned, AI-compatible, auditable web architecture patterns",
6
6
  "categories": [
7
7
  {
8
8
  "id": "rendering",