@gallop.software/canon 1.0.0 → 2.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cli/commands/audit.d.ts +7 -0
- package/dist/cli/commands/audit.js +172 -0
- package/dist/cli/commands/generate.d.ts +6 -0
- package/dist/cli/commands/generate.js +170 -0
- package/dist/cli/index.d.ts +2 -0
- package/dist/cli/index.js +104 -0
- package/dist/eslint/configs/recommended.d.ts +13 -0
- package/dist/eslint/configs/recommended.js +15 -0
- package/dist/eslint/configs/speedwell.d.ts +13 -0
- package/dist/eslint/configs/speedwell.js +17 -0
- package/dist/eslint/index.d.ts +37 -0
- package/dist/eslint/index.js +24 -0
- package/dist/eslint/rules/no-client-blocks.d.ts +5 -0
- package/dist/eslint/rules/no-client-blocks.js +45 -0
- package/dist/eslint/rules/no-container-in-section.d.ts +5 -0
- package/dist/eslint/rules/no-container-in-section.js +50 -0
- package/dist/eslint/rules/prefer-component-props.d.ts +5 -0
- package/dist/eslint/rules/prefer-component-props.js +100 -0
- package/dist/eslint/rules/prefer-typography-components.d.ts +3 -0
- package/dist/eslint/rules/prefer-typography-components.js +76 -0
- package/dist/eslint/utils/canon.d.ts +25 -0
- package/dist/eslint/utils/canon.js +50 -0
- package/dist/index.js +28 -42
- package/package.json +33 -3
|
@@ -0,0 +1,172 @@
|
|
|
1
|
+
import { spawn } from 'child_process';
|
|
2
|
+
import { version, patterns, getPattern } from '../../index.js';
|
|
3
|
+
// Colors for terminal output
|
|
4
|
+
const c = {
|
|
5
|
+
reset: '\x1b[0m',
|
|
6
|
+
bold: '\x1b[1m',
|
|
7
|
+
dim: '\x1b[2m',
|
|
8
|
+
red: '\x1b[31m',
|
|
9
|
+
green: '\x1b[32m',
|
|
10
|
+
yellow: '\x1b[33m',
|
|
11
|
+
blue: '\x1b[34m',
|
|
12
|
+
cyan: '\x1b[36m',
|
|
13
|
+
};
|
|
14
|
+
// Build rule-to-pattern mapping from Canon schema (single source of truth)
|
|
15
|
+
const ruleToPattern = {};
|
|
16
|
+
for (const pattern of patterns) {
|
|
17
|
+
if (pattern.rule) {
|
|
18
|
+
ruleToPattern[pattern.rule] = pattern.id;
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
function parseCanonFromMessage(message) {
|
|
22
|
+
const match = message.match(/\[Canon (\d{3})\]/);
|
|
23
|
+
return match ? match[1] : null;
|
|
24
|
+
}
|
|
25
|
+
function extractViolations(eslintOutput) {
|
|
26
|
+
const violations = [];
|
|
27
|
+
const lines = eslintOutput.split('\n');
|
|
28
|
+
let currentFile = '';
|
|
29
|
+
for (const line of lines) {
|
|
30
|
+
// File path line (starts with /)
|
|
31
|
+
if (line.startsWith('/') && !line.includes(':')) {
|
|
32
|
+
currentFile = line.trim();
|
|
33
|
+
continue;
|
|
34
|
+
}
|
|
35
|
+
// Violation line (starts with spaces, has line:col pattern)
|
|
36
|
+
const match = line.match(/^\s+(\d+):(\d+)\s+(warning|error)\s+(.+?)\s{2,}([\w/-]+)$/);
|
|
37
|
+
if (match && currentFile) {
|
|
38
|
+
const [, lineNum, column, , message, ruleId] = match;
|
|
39
|
+
const canonPattern = ruleToPattern[ruleId] || parseCanonFromMessage(message);
|
|
40
|
+
const pattern = canonPattern ? getPattern(canonPattern) : null;
|
|
41
|
+
violations.push({
|
|
42
|
+
file: currentFile,
|
|
43
|
+
line: parseInt(lineNum, 10),
|
|
44
|
+
column: parseInt(column, 10),
|
|
45
|
+
message: message.trim(),
|
|
46
|
+
ruleId,
|
|
47
|
+
canonPattern,
|
|
48
|
+
patternTitle: pattern?.title || null,
|
|
49
|
+
});
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
return violations;
|
|
53
|
+
}
|
|
54
|
+
function countByPattern(violations) {
|
|
55
|
+
const counts = {};
|
|
56
|
+
for (const v of violations) {
|
|
57
|
+
const key = v.canonPattern || 'other';
|
|
58
|
+
counts[key] = (counts[key] || 0) + 1;
|
|
59
|
+
}
|
|
60
|
+
return counts;
|
|
61
|
+
}
|
|
62
|
+
function groupByFile(violations) {
|
|
63
|
+
const groups = new Map();
|
|
64
|
+
for (const v of violations) {
|
|
65
|
+
const existing = groups.get(v.file) || [];
|
|
66
|
+
existing.push(v);
|
|
67
|
+
groups.set(v.file, existing);
|
|
68
|
+
}
|
|
69
|
+
return groups;
|
|
70
|
+
}
|
|
71
|
+
function printReport(result, options) {
|
|
72
|
+
if (options.json) {
|
|
73
|
+
console.log(JSON.stringify(result, null, 2));
|
|
74
|
+
return;
|
|
75
|
+
}
|
|
76
|
+
// Header
|
|
77
|
+
console.log('');
|
|
78
|
+
console.log(`${c.bold}╔════════════════════════════════════════════════════════════╗${c.reset}`);
|
|
79
|
+
console.log(`${c.bold}║${c.reset} ${c.cyan}Gallop Canon${c.reset} ${c.dim}v${result.version}${c.reset} ${c.bold}Compliance Report${c.reset} ${c.bold}║${c.reset}`);
|
|
80
|
+
console.log(`${c.bold}╚════════════════════════════════════════════════════════════╝${c.reset}`);
|
|
81
|
+
console.log('');
|
|
82
|
+
if (result.violations.length === 0) {
|
|
83
|
+
console.log(` ${c.green}✓${c.reset} ${c.bold}All files pass Canon compliance checks${c.reset}`);
|
|
84
|
+
console.log(` ${c.dim}Path: ${result.path}${c.reset}`);
|
|
85
|
+
console.log('');
|
|
86
|
+
return;
|
|
87
|
+
}
|
|
88
|
+
// Summary
|
|
89
|
+
const filesWithViolations = new Set(result.violations.map(v => v.file)).size;
|
|
90
|
+
console.log(` ${c.red}✗${c.reset} ${c.bold}${result.violations.length} violation${result.violations.length === 1 ? '' : 's'}${c.reset} in ${filesWithViolations} file${filesWithViolations === 1 ? '' : 's'}`);
|
|
91
|
+
console.log(` ${c.dim}Path: ${result.path}${c.reset}`);
|
|
92
|
+
console.log('');
|
|
93
|
+
// Violations by pattern
|
|
94
|
+
console.log(`${c.bold} Violations by Pattern:${c.reset}`);
|
|
95
|
+
for (const [patternId, count] of Object.entries(result.summary.byPattern).sort()) {
|
|
96
|
+
const pattern = getPattern(patternId);
|
|
97
|
+
const title = pattern?.title || 'Other';
|
|
98
|
+
console.log(` ${c.yellow}${patternId}${c.reset} ${title}: ${c.bold}${count}${c.reset}`);
|
|
99
|
+
}
|
|
100
|
+
console.log('');
|
|
101
|
+
// Detailed violations by file
|
|
102
|
+
console.log(`${c.bold} Details:${c.reset}`);
|
|
103
|
+
console.log('');
|
|
104
|
+
const byFile = groupByFile(result.violations);
|
|
105
|
+
for (const [file, violations] of byFile) {
|
|
106
|
+
// Shorten file path for display
|
|
107
|
+
const shortPath = file.replace(process.cwd() + '/', '');
|
|
108
|
+
console.log(` ${c.cyan}${shortPath}${c.reset}`);
|
|
109
|
+
for (const v of violations) {
|
|
110
|
+
const patternLabel = v.canonPattern
|
|
111
|
+
? `${c.yellow}[${v.canonPattern}]${c.reset}`
|
|
112
|
+
: `${c.dim}[---]${c.reset}`;
|
|
113
|
+
const location = `${c.dim}${v.line}:${v.column}${c.reset}`;
|
|
114
|
+
// Clean up message (remove [Canon XXX] prefix since we show it separately)
|
|
115
|
+
const cleanMessage = v.message.replace(/\[Canon \d{3}\]\s*/, '');
|
|
116
|
+
console.log(` ${location} ${patternLabel} ${cleanMessage}`);
|
|
117
|
+
}
|
|
118
|
+
console.log('');
|
|
119
|
+
}
|
|
120
|
+
// Footer
|
|
121
|
+
console.log(`${c.dim} ─────────────────────────────────────────────────────────${c.reset}`);
|
|
122
|
+
console.log(` ${c.dim}Fix violations manually following the Canon patterns${c.reset}`);
|
|
123
|
+
console.log('');
|
|
124
|
+
}
|
|
125
|
+
export async function audit(path, options) {
|
|
126
|
+
return new Promise((resolve, reject) => {
|
|
127
|
+
const args = [
|
|
128
|
+
'eslint',
|
|
129
|
+
path,
|
|
130
|
+
'--format', 'stylish',
|
|
131
|
+
];
|
|
132
|
+
if (options.fix) {
|
|
133
|
+
args.push('--fix');
|
|
134
|
+
}
|
|
135
|
+
const eslint = spawn('npx', args, {
|
|
136
|
+
cwd: process.cwd(),
|
|
137
|
+
stdio: ['inherit', 'pipe', 'pipe'],
|
|
138
|
+
shell: true,
|
|
139
|
+
});
|
|
140
|
+
let stdout = '';
|
|
141
|
+
let stderr = '';
|
|
142
|
+
eslint.stdout?.on('data', (data) => {
|
|
143
|
+
stdout += data.toString();
|
|
144
|
+
});
|
|
145
|
+
eslint.stderr?.on('data', (data) => {
|
|
146
|
+
stderr += data.toString();
|
|
147
|
+
});
|
|
148
|
+
eslint.on('close', (code) => {
|
|
149
|
+
const violations = extractViolations(stdout);
|
|
150
|
+
const result = {
|
|
151
|
+
version,
|
|
152
|
+
timestamp: new Date().toISOString(),
|
|
153
|
+
path,
|
|
154
|
+
totalFiles: 0, // ESLint stylish format doesn't give us this easily
|
|
155
|
+
violations,
|
|
156
|
+
summary: {
|
|
157
|
+
total: violations.length,
|
|
158
|
+
byPattern: countByPattern(violations),
|
|
159
|
+
},
|
|
160
|
+
};
|
|
161
|
+
printReport(result, options);
|
|
162
|
+
if (options.strict && violations.length > 0) {
|
|
163
|
+
process.exit(1);
|
|
164
|
+
}
|
|
165
|
+
resolve();
|
|
166
|
+
});
|
|
167
|
+
eslint.on('error', (error) => {
|
|
168
|
+
reject(error);
|
|
169
|
+
});
|
|
170
|
+
});
|
|
171
|
+
}
|
|
172
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"audit.js","sourceRoot":"","sources":["../../../src/cli/commands/audit.ts"],"names":[],"mappings":"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","sourcesContent":["import { spawn } from 'child_process'\nimport { version, patterns, getPattern } from '../../index.js'\n\ninterface AuditOptions {\n  strict: boolean\n  json: boolean\n  fix: boolean\n}\n\ninterface Violation {\n  file: string\n  line: number\n  column: number\n  message: string\n  ruleId: string\n  canonPattern: string | null\n  patternTitle: string | null\n}\n\ninterface AuditResult {\n  version: string\n  timestamp: string\n  path: string\n  totalFiles: number\n  violations: Violation[]\n  summary: {\n    total: number\n    byPattern: Record<string, number>\n  }\n}\n\n// Colors for terminal output\nconst c = {\n  reset: '\\x1b[0m',\n  bold: '\\x1b[1m',\n  dim: '\\x1b[2m',\n  red: '\\x1b[31m',\n  green: '\\x1b[32m',\n  yellow: '\\x1b[33m',\n  blue: '\\x1b[34m',\n  cyan: '\\x1b[36m',\n}\n\n// Build rule-to-pattern mapping from Canon schema (single source of truth)\nconst ruleToPattern: Record<string, string> = {}\nfor (const pattern of patterns) {\n  if (pattern.rule) {\n    ruleToPattern[pattern.rule] = pattern.id\n  }\n}\n\nfunction parseCanonFromMessage(message: string): string | null {\n  const match = message.match(/\\[Canon (\\d{3})\\]/)\n  return match ? match[1] : null\n}\n\nfunction extractViolations(eslintOutput: string): Violation[] {\n  const violations: Violation[] = []\n  const lines = eslintOutput.split('\\n')\n  \n  let currentFile = ''\n  \n  for (const line of lines) {\n    // File path line (starts with /)\n    if (line.startsWith('/') && !line.includes(':')) {\n      currentFile = line.trim()\n      continue\n    }\n    \n    // Violation line (starts with spaces, has line:col pattern)\n    const match = line.match(/^\\s+(\\d+):(\\d+)\\s+(warning|error)\\s+(.+?)\\s{2,}([\\w/-]+)$/)\n    if (match && currentFile) {\n      const [, lineNum, column, , message, ruleId] = match\n      const canonPattern = ruleToPattern[ruleId] || parseCanonFromMessage(message)\n      const pattern = canonPattern ? getPattern(canonPattern) : null\n      \n      violations.push({\n        file: currentFile,\n        line: parseInt(lineNum, 10),\n        column: parseInt(column, 10),\n        message: message.trim(),\n        ruleId,\n        canonPattern,\n        patternTitle: pattern?.title || null,\n      })\n    }\n  }\n  \n  return violations\n}\n\nfunction countByPattern(violations: Violation[]): Record<string, number> {\n  const counts: Record<string, number> = {}\n  for (const v of violations) {\n    const key = v.canonPattern || 'other'\n    counts[key] = (counts[key] || 0) + 1\n  }\n  return counts\n}\n\nfunction groupByFile(violations: Violation[]): Map<string, Violation[]> {\n  const groups = new Map<string, Violation[]>()\n  for (const v of violations) {\n    const existing = groups.get(v.file) || []\n    existing.push(v)\n    groups.set(v.file, existing)\n  }\n  return groups\n}\n\nfunction printReport(result: AuditResult, options: AuditOptions) {\n  if (options.json) {\n    console.log(JSON.stringify(result, null, 2))\n    return\n  }\n\n  // Header\n  console.log('')\n  console.log(`${c.bold}╔════════════════════════════════════════════════════════════╗${c.reset}`)\n  console.log(`${c.bold}║${c.reset}           ${c.cyan}Gallop Canon${c.reset} ${c.dim}v${result.version}${c.reset} ${c.bold}Compliance Report${c.reset}           ${c.bold}║${c.reset}`)\n  console.log(`${c.bold}╚════════════════════════════════════════════════════════════╝${c.reset}`)\n  console.log('')\n\n  if (result.violations.length === 0) {\n    console.log(`  ${c.green}✓${c.reset} ${c.bold}All files pass Canon compliance checks${c.reset}`)\n    console.log(`  ${c.dim}Path: ${result.path}${c.reset}`)\n    console.log('')\n    return\n  }\n\n  // Summary\n  const filesWithViolations = new Set(result.violations.map(v => v.file)).size\n  console.log(`  ${c.red}✗${c.reset} ${c.bold}${result.violations.length} violation${result.violations.length === 1 ? '' : 's'}${c.reset} in ${filesWithViolations} file${filesWithViolations === 1 ? '' : 's'}`)\n  console.log(`  ${c.dim}Path: ${result.path}${c.reset}`)\n  console.log('')\n\n  // Violations by pattern\n  console.log(`${c.bold}  Violations by Pattern:${c.reset}`)\n  for (const [patternId, count] of Object.entries(result.summary.byPattern).sort()) {\n    const pattern = getPattern(patternId)\n    const title = pattern?.title || 'Other'\n    console.log(`    ${c.yellow}${patternId}${c.reset} ${title}: ${c.bold}${count}${c.reset}`)\n  }\n  console.log('')\n\n  // Detailed violations by file\n  console.log(`${c.bold}  Details:${c.reset}`)\n  console.log('')\n\n  const byFile = groupByFile(result.violations)\n  for (const [file, violations] of byFile) {\n    // Shorten file path for display\n    const shortPath = file.replace(process.cwd() + '/', '')\n    console.log(`  ${c.cyan}${shortPath}${c.reset}`)\n    \n    for (const v of violations) {\n      const patternLabel = v.canonPattern \n        ? `${c.yellow}[${v.canonPattern}]${c.reset}` \n        : `${c.dim}[---]${c.reset}`\n      const location = `${c.dim}${v.line}:${v.column}${c.reset}`\n      \n      // Clean up message (remove [Canon XXX] prefix since we show it separately)\n      const cleanMessage = v.message.replace(/\\[Canon \\d{3}\\]\\s*/, '')\n      \n      console.log(`    ${location} ${patternLabel} ${cleanMessage}`)\n    }\n    console.log('')\n  }\n\n  // Footer\n  console.log(`${c.dim}  ─────────────────────────────────────────────────────────${c.reset}`)\n  console.log(`  ${c.dim}Fix violations manually following the Canon patterns${c.reset}`)\n  console.log('')\n}\n\nexport async function audit(path: string, options: AuditOptions): Promise<void> {\n  return new Promise((resolve, reject) => {\n    const args = [\n      'eslint',\n      path,\n      '--format', 'stylish',\n    ]\n\n    if (options.fix) {\n      args.push('--fix')\n    }\n\n    const eslint = spawn('npx', args, {\n      cwd: process.cwd(),\n      stdio: ['inherit', 'pipe', 'pipe'],\n      shell: true,\n    })\n\n    let stdout = ''\n    let stderr = ''\n\n    eslint.stdout?.on('data', (data) => {\n      stdout += data.toString()\n    })\n\n    eslint.stderr?.on('data', (data) => {\n      stderr += data.toString()\n    })\n\n    eslint.on('close', (code) => {\n      const violations = extractViolations(stdout)\n      \n      const result: AuditResult = {\n        version,\n        timestamp: new Date().toISOString(),\n        path,\n        totalFiles: 0, // ESLint stylish format doesn't give us this easily\n        violations,\n        summary: {\n          total: violations.length,\n          byPattern: countByPattern(violations),\n        },\n      }\n\n      printReport(result, options)\n\n      if (options.strict && violations.length > 0) {\n        process.exit(1)\n      }\n\n      resolve()\n    })\n\n    eslint.on('error', (error) => {\n      reject(error)\n    })\n  })\n}\n"]}
|
|
@@ -0,0 +1,170 @@
|
|
|
1
|
+
import * as fs from 'fs';
|
|
2
|
+
import * as path from 'path';
|
|
3
|
+
import { version, patterns, guarantees } from '../../index.js';
|
|
4
|
+
/**
|
|
5
|
+
* Read a pattern markdown file and extract sections
|
|
6
|
+
*/
|
|
7
|
+
function readPatternFile(patternFile) {
|
|
8
|
+
// Try to find the canon package patterns directory
|
|
9
|
+
const possiblePaths = [
|
|
10
|
+
path.join(process.cwd(), 'node_modules/@gallop.software/canon', patternFile),
|
|
11
|
+
// Fallback for development
|
|
12
|
+
path.join(import.meta.dirname, '../../../', patternFile),
|
|
13
|
+
];
|
|
14
|
+
for (const filePath of possiblePaths) {
|
|
15
|
+
if (fs.existsSync(filePath)) {
|
|
16
|
+
return fs.readFileSync(filePath, 'utf-8');
|
|
17
|
+
}
|
|
18
|
+
}
|
|
19
|
+
return null;
|
|
20
|
+
}
|
|
21
|
+
/**
|
|
22
|
+
* Extract examples from pattern markdown
|
|
23
|
+
*/
|
|
24
|
+
function extractExamples(content) {
|
|
25
|
+
const good = [];
|
|
26
|
+
const bad = [];
|
|
27
|
+
// Find code blocks after "### Good" or "### Bad" headers
|
|
28
|
+
const goodMatch = content.match(/### Good\s*\n```[\s\S]*?```/g);
|
|
29
|
+
const badMatch = content.match(/### Bad\s*\n```[\s\S]*?```/g);
|
|
30
|
+
if (goodMatch) {
|
|
31
|
+
for (const match of goodMatch) {
|
|
32
|
+
const code = match.replace(/### Good\s*\n/, '').trim();
|
|
33
|
+
good.push(code);
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
if (badMatch) {
|
|
37
|
+
for (const match of badMatch) {
|
|
38
|
+
const code = match.replace(/### Bad\s*\n/, '').trim();
|
|
39
|
+
bad.push(code);
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
return { good, bad };
|
|
43
|
+
}
|
|
44
|
+
/**
|
|
45
|
+
* Generate .cursorrules content from Canon
|
|
46
|
+
*/
|
|
47
|
+
function generateCursorrules() {
|
|
48
|
+
const lines = [];
|
|
49
|
+
// Header
|
|
50
|
+
lines.push(`# Gallop Canon v${version} - AI Rules`);
|
|
51
|
+
lines.push('');
|
|
52
|
+
lines.push('This file is auto-generated from @gallop.software/canon. Do not edit manually.');
|
|
53
|
+
lines.push('Regenerate with: npm run generate:ai-rules');
|
|
54
|
+
lines.push('');
|
|
55
|
+
// Tech stack (from speedwell context)
|
|
56
|
+
lines.push('## Tech Stack');
|
|
57
|
+
lines.push('');
|
|
58
|
+
lines.push('- Next.js 16 with App Router');
|
|
59
|
+
lines.push('- React 19');
|
|
60
|
+
lines.push('- TypeScript');
|
|
61
|
+
lines.push('- Tailwind CSS v4');
|
|
62
|
+
lines.push('- clsx for conditional class names');
|
|
63
|
+
lines.push('');
|
|
64
|
+
// Enforced patterns (ESLint rules)
|
|
65
|
+
lines.push('## Enforced Patterns (ESLint)');
|
|
66
|
+
lines.push('');
|
|
67
|
+
lines.push('These patterns are enforced by `@gallop.software/canon/eslint`. Violations will be flagged.');
|
|
68
|
+
lines.push('');
|
|
69
|
+
const enforcedPatterns = patterns.filter(p => p.enforcement === 'eslint' && p.rule);
|
|
70
|
+
for (const pattern of enforcedPatterns) {
|
|
71
|
+
lines.push(`### ${pattern.id}: ${pattern.title}`);
|
|
72
|
+
lines.push('');
|
|
73
|
+
lines.push(pattern.summary);
|
|
74
|
+
lines.push('');
|
|
75
|
+
lines.push(`- **ESLint Rule:** \`${pattern.rule}\``);
|
|
76
|
+
lines.push(`- **Category:** ${pattern.category}`);
|
|
77
|
+
lines.push('');
|
|
78
|
+
// Try to read and include examples
|
|
79
|
+
const content = readPatternFile(pattern.file);
|
|
80
|
+
if (content) {
|
|
81
|
+
const { good, bad } = extractExamples(content);
|
|
82
|
+
if (bad.length > 0) {
|
|
83
|
+
lines.push('**Bad:**');
|
|
84
|
+
lines.push('');
|
|
85
|
+
lines.push(bad[0]);
|
|
86
|
+
lines.push('');
|
|
87
|
+
}
|
|
88
|
+
if (good.length > 0) {
|
|
89
|
+
lines.push('**Good:**');
|
|
90
|
+
lines.push('');
|
|
91
|
+
lines.push(good[0]);
|
|
92
|
+
lines.push('');
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
// Documentation patterns
|
|
97
|
+
lines.push('## Documentation Patterns');
|
|
98
|
+
lines.push('');
|
|
99
|
+
lines.push('These patterns are not enforced by ESLint but should be followed.');
|
|
100
|
+
lines.push('');
|
|
101
|
+
const docPatterns = patterns.filter(p => p.enforcement === 'documentation');
|
|
102
|
+
for (const pattern of docPatterns) {
|
|
103
|
+
lines.push(`### ${pattern.id}: ${pattern.title}`);
|
|
104
|
+
lines.push('');
|
|
105
|
+
lines.push(pattern.summary);
|
|
106
|
+
lines.push('');
|
|
107
|
+
}
|
|
108
|
+
// Guarantees
|
|
109
|
+
lines.push('## Canon Guarantees');
|
|
110
|
+
lines.push('');
|
|
111
|
+
lines.push('Following these patterns provides these guarantees:');
|
|
112
|
+
lines.push('');
|
|
113
|
+
for (const guarantee of guarantees) {
|
|
114
|
+
lines.push(`- **${guarantee.name}** (${guarantee.id}): Patterns ${guarantee.patterns.join(', ')}`);
|
|
115
|
+
}
|
|
116
|
+
lines.push('');
|
|
117
|
+
// Quick reference for components
|
|
118
|
+
lines.push('## Component Quick Reference');
|
|
119
|
+
lines.push('');
|
|
120
|
+
lines.push('### Typography');
|
|
121
|
+
lines.push('- `Heading` - props: `as`, `color`, `margin`, `fontSize`, `fontWeight`, `textAlign`');
|
|
122
|
+
lines.push('- `Paragraph` - props: `color`, `margin`, `fontSize`, `lineHeight`, `textAlign`');
|
|
123
|
+
lines.push('- `Span` - props: `color`, `margin`, `fontSize` (inline text, mb-0 default)');
|
|
124
|
+
lines.push('- `Label` - props: `color`, `margin`, `fontSize`, `fontWeight`, `textAlign`');
|
|
125
|
+
lines.push('');
|
|
126
|
+
lines.push('### Layout');
|
|
127
|
+
lines.push('- `Section` - semantic section wrapper');
|
|
128
|
+
lines.push('- `Columns` - grid layout, props: `cols`, `gap`, `align`');
|
|
129
|
+
lines.push('- `Column` - column child');
|
|
130
|
+
lines.push('');
|
|
131
|
+
lines.push('### Interactive');
|
|
132
|
+
lines.push('- `Button` - props: `href`, `variant`, `icon`, `iconPlacement`, `margin`');
|
|
133
|
+
lines.push('- `Icon` - Iconify icon wrapper');
|
|
134
|
+
lines.push('');
|
|
135
|
+
// Do NOT section
|
|
136
|
+
lines.push('## Do NOT');
|
|
137
|
+
lines.push('');
|
|
138
|
+
lines.push('- Use `\'use client\'` in blocks - extract to components');
|
|
139
|
+
lines.push('- Use raw `<p>` or `<span>` - use Paragraph/Span components');
|
|
140
|
+
lines.push('- Use className for margin/color/fontSize when component has props');
|
|
141
|
+
lines.push('- Use Container inside Section - Section already provides containment');
|
|
142
|
+
lines.push('- Use `classnames` package - use `clsx` instead');
|
|
143
|
+
lines.push('- Use inline styles for hover states - use Tailwind classes');
|
|
144
|
+
lines.push('');
|
|
145
|
+
// Post-edit verification
|
|
146
|
+
lines.push('## Post-Edit Verification');
|
|
147
|
+
lines.push('');
|
|
148
|
+
lines.push('After editing files:');
|
|
149
|
+
lines.push('1. Run `npm run lint` to check for errors');
|
|
150
|
+
lines.push('2. Fix any violations before committing');
|
|
151
|
+
lines.push('');
|
|
152
|
+
lines.push('Note: Only lint files you edited, not the entire codebase.');
|
|
153
|
+
lines.push('');
|
|
154
|
+
return lines.join('\n');
|
|
155
|
+
}
|
|
156
|
+
export async function generate(options) {
|
|
157
|
+
const content = generateCursorrules();
|
|
158
|
+
if (options.output === '-') {
|
|
159
|
+
// Output to stdout
|
|
160
|
+
console.log(content);
|
|
161
|
+
}
|
|
162
|
+
else {
|
|
163
|
+
// Write to file
|
|
164
|
+
const outputPath = path.resolve(process.cwd(), options.output);
|
|
165
|
+
fs.writeFileSync(outputPath, content, 'utf-8');
|
|
166
|
+
console.log(`✓ Generated ${options.output} from Canon v${version}`);
|
|
167
|
+
console.log(` ${patterns.length} patterns, ${guarantees.length} guarantees`);
|
|
168
|
+
}
|
|
169
|
+
}
|
|
170
|
+
//# 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 (from speedwell context)\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  // 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"]}
|
|
@@ -0,0 +1,104 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
import { audit } from './commands/audit.js';
|
|
3
|
+
import { generate } from './commands/generate.js';
|
|
4
|
+
import { version } from '../index.js';
|
|
5
|
+
const args = process.argv.slice(2);
|
|
6
|
+
const command = args[0];
|
|
7
|
+
// Colors for terminal output
|
|
8
|
+
const colors = {
|
|
9
|
+
reset: '\x1b[0m',
|
|
10
|
+
bold: '\x1b[1m',
|
|
11
|
+
dim: '\x1b[2m',
|
|
12
|
+
red: '\x1b[31m',
|
|
13
|
+
green: '\x1b[32m',
|
|
14
|
+
yellow: '\x1b[33m',
|
|
15
|
+
blue: '\x1b[34m',
|
|
16
|
+
magenta: '\x1b[35m',
|
|
17
|
+
cyan: '\x1b[36m',
|
|
18
|
+
};
|
|
19
|
+
function showHelp() {
|
|
20
|
+
console.log(`
|
|
21
|
+
${colors.bold}Gallop CLI${colors.reset} - Canon Compliance Tooling
|
|
22
|
+
${colors.dim}Canon Version: ${version}${colors.reset}
|
|
23
|
+
|
|
24
|
+
${colors.bold}Usage:${colors.reset}
|
|
25
|
+
gallop <command> [options]
|
|
26
|
+
|
|
27
|
+
${colors.bold}Commands:${colors.reset}
|
|
28
|
+
audit [path] Check Canon compliance (default: src/blocks/)
|
|
29
|
+
generate [output] Generate AI rules from Canon (default: .cursorrules)
|
|
30
|
+
version Show version information
|
|
31
|
+
help Show this help message
|
|
32
|
+
|
|
33
|
+
${colors.bold}Audit Options:${colors.reset}
|
|
34
|
+
--strict Exit with error code on violations
|
|
35
|
+
--json Output as JSON
|
|
36
|
+
|
|
37
|
+
${colors.bold}Generate Options:${colors.reset}
|
|
38
|
+
--output, -o Output file path (default: .cursorrules)
|
|
39
|
+
|
|
40
|
+
${colors.bold}Examples:${colors.reset}
|
|
41
|
+
gallop audit
|
|
42
|
+
gallop audit src/blocks/ --strict
|
|
43
|
+
gallop generate
|
|
44
|
+
gallop generate .cursorrules
|
|
45
|
+
gallop generate --output .github/copilot-instructions.md
|
|
46
|
+
`);
|
|
47
|
+
}
|
|
48
|
+
function showVersion() {
|
|
49
|
+
console.log(`Gallop CLI v1.0.0`);
|
|
50
|
+
console.log(`Canon v${version}`);
|
|
51
|
+
}
|
|
52
|
+
async function main() {
|
|
53
|
+
switch (command) {
|
|
54
|
+
case 'audit':
|
|
55
|
+
const auditPath = args[1] && !args[1].startsWith('--') ? args[1] : 'src/blocks/';
|
|
56
|
+
const auditOptions = {
|
|
57
|
+
strict: args.includes('--strict'),
|
|
58
|
+
json: args.includes('--json'),
|
|
59
|
+
fix: args.includes('--fix'),
|
|
60
|
+
};
|
|
61
|
+
await audit(auditPath, auditOptions);
|
|
62
|
+
break;
|
|
63
|
+
case 'generate':
|
|
64
|
+
// Find output path from args
|
|
65
|
+
let outputPath = '.cursorrules';
|
|
66
|
+
const outputIndex = args.indexOf('--output');
|
|
67
|
+
const outputIndexShort = args.indexOf('-o');
|
|
68
|
+
if (outputIndex !== -1 && args[outputIndex + 1]) {
|
|
69
|
+
outputPath = args[outputIndex + 1];
|
|
70
|
+
}
|
|
71
|
+
else if (outputIndexShort !== -1 && args[outputIndexShort + 1]) {
|
|
72
|
+
outputPath = args[outputIndexShort + 1];
|
|
73
|
+
}
|
|
74
|
+
else if (args[1] && !args[1].startsWith('--')) {
|
|
75
|
+
outputPath = args[1];
|
|
76
|
+
}
|
|
77
|
+
const generateOptions = {
|
|
78
|
+
output: outputPath,
|
|
79
|
+
format: 'cursorrules',
|
|
80
|
+
};
|
|
81
|
+
await generate(generateOptions);
|
|
82
|
+
break;
|
|
83
|
+
case 'version':
|
|
84
|
+
case '-v':
|
|
85
|
+
case '--version':
|
|
86
|
+
showVersion();
|
|
87
|
+
break;
|
|
88
|
+
case 'help':
|
|
89
|
+
case '-h':
|
|
90
|
+
case '--help':
|
|
91
|
+
case undefined:
|
|
92
|
+
showHelp();
|
|
93
|
+
break;
|
|
94
|
+
default:
|
|
95
|
+
console.error(`Unknown command: ${command}`);
|
|
96
|
+
console.error(`Run 'gallop help' for usage information.`);
|
|
97
|
+
process.exit(1);
|
|
98
|
+
}
|
|
99
|
+
}
|
|
100
|
+
main().catch((error) => {
|
|
101
|
+
console.error('Error:', error.message);
|
|
102
|
+
process.exit(1);
|
|
103
|
+
});
|
|
104
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY2xpL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFFQSxPQUFPLEVBQUUsS0FBSyxFQUFFLE1BQU0scUJBQXFCLENBQUE7QUFDM0MsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLHdCQUF3QixDQUFBO0FBQ2pELE9BQU8sRUFBRSxPQUFPLEVBQUUsTUFBTSxhQUFhLENBQUE7QUFFckMsTUFBTSxJQUFJLEdBQUcsT0FBTyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUE7QUFDbEMsTUFBTSxPQUFPLEdBQUcsSUFBSSxDQUFDLENBQUMsQ0FBQyxDQUFBO0FBRXZCLDZCQUE2QjtBQUM3QixNQUFNLE1BQU0sR0FBRztJQUNiLEtBQUssRUFBRSxTQUFTO0lBQ2hCLElBQUksRUFBRSxTQUFTO0lBQ2YsR0FBRyxFQUFFLFNBQVM7SUFDZCxHQUFHLEVBQUUsVUFBVTtJQUNmLEtBQUssRUFBRSxVQUFVO0lBQ2pCLE1BQU0sRUFBRSxVQUFVO0lBQ2xCLElBQUksRUFBRSxVQUFVO0lBQ2hCLE9BQU8sRUFBRSxVQUFVO0lBQ25CLElBQUksRUFBRSxVQUFVO0NBQ2pCLENBQUE7QUFFRCxTQUFTLFFBQVE7SUFDZixPQUFPLENBQUMsR0FBRyxDQUFDO0VBQ1osTUFBTSxDQUFDLElBQUksYUFBYSxNQUFNLENBQUMsS0FBSztFQUNwQyxNQUFNLENBQUMsR0FBRyxrQkFBa0IsT0FBTyxHQUFHLE1BQU0sQ0FBQyxLQUFLOztFQUVsRCxNQUFNLENBQUMsSUFBSSxTQUFTLE1BQU0sQ0FBQyxLQUFLOzs7RUFHaEMsTUFBTSxDQUFDLElBQUksWUFBWSxNQUFNLENBQUMsS0FBSzs7Ozs7O0VBTW5DLE1BQU0sQ0FBQyxJQUFJLGlCQUFpQixNQUFNLENBQUMsS0FBSzs7OztFQUl4QyxNQUFNLENBQUMsSUFBSSxvQkFBb0IsTUFBTSxDQUFDLEtBQUs7OztFQUczQyxNQUFN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QUMsT0FBTyxDQUFDLENBQUE7SUFDdEMsT0FBTyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQTtBQUNqQixDQUFDLENBQUMsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbIiMhL3Vzci9iaW4vZW52IG5vZGVcblxuaW1wb3J0IHsgYXVkaXQgfSBmcm9tICcuL2NvbW1hbmRzL2F1ZGl0LmpzJ1xuaW1wb3J0IHsgZ2VuZXJhdGUgfSBmcm9tICcuL2NvbW1hbmRzL2dlbmVyYXRlLmpzJ1xuaW1wb3J0IHsgdmVyc2lvbiB9IGZyb20gJy4uL2luZGV4LmpzJ1xuXG5jb25zdCBhcmdzID0gcHJvY2Vzcy5hcmd2LnNsaWNlKDIpXG5jb25zdCBjb21tYW5kID0gYXJnc1swXVxuXG4vLyBDb2xvcnMgZm9yIHRlcm1pbmFsIG91dHB1dFxuY29uc3QgY29sb3JzID0ge1xuICByZXNldDogJ1xceDFiWzBtJyxcbiAgYm9sZDogJ1xceDFiWzFtJyxcbiAgZGltOiAnXFx4MWJbMm0nLFxuICByZWQ6ICdcXHgxYlszMW0nLFxuICBncmVlbjogJ1xceDFiWzMybScsXG4gIHllbGxvdzogJ1xceDFiWzMzbScsXG4gIGJsdWU6ICdcXHgxYlszNG0nLFxuICBtYWdlbnRhOiAnXFx4MWJbMzVtJyxcbiAgY3lhbjogJ1xceDFiWzM2bScsXG59XG5cbmZ1bmN0aW9uIHNob3dIZWxwKCkge1xuICBjb25zb2xlLmxvZyhgXG4ke2NvbG9ycy5ib2xkfUdhbGxvcCBDTEkke2NvbG9ycy5yZXNldH0gLSBDYW5vbiBDb21wbGlhbmNlIFRvb2xpbmdcbiR7Y29sb3JzLmRpbX1DYW5vbiBWZXJzaW9uOiAke3ZlcnNpb259JHtjb2xvcnMucmVzZXR9XG5cbiR7Y29sb3JzLmJvbGR9VXNhZ2U6JHtjb2xvcnMucmVzZXR9XG4gIGdhbGxvcCA8Y29tbWFuZD4gW29wdGlvbnNdXG5cbiR7Y29sb3JzLmJvbGR9Q29tbWFuZHM6JHtjb2xvcnMucmVzZXR9XG4gIGF1ZGl0IFtwYXRoXSAgICAgICBDaGVjayBDYW5vbiBjb21wbGlhbmNlIChkZWZhdWx0OiBzcmMvYmxvY2tzLylcbiAgZ2VuZXJhdGUgW291dHB1dF0gIEdlbmVyYXRlIEFJIHJ1bGVzIGZyb20gQ2Fub24gKGRlZmF1bHQ6IC5jdXJzb3JydWxlcylcbiAgdmVyc2lvbiAgICAgICAgICAgIFNob3cgdmVyc2lvbiBpbmZvcm1hdGlvblxuICBoZWxwICAgICAgICAgICAgICAgU2hvdyB0aGlzIGhlbHAgbWVzc2FnZVxuXG4ke2NvbG9ycy5ib2xkfUF1ZGl0IE9wdGlvbnM6JHtjb2xvcnMucmVzZXR9XG4gIC0tc3RyaWN0ICAgICAgICAgICBFeGl0IHdpdGggZXJyb3IgY29kZSBvbiB2aW9sYXRpb25zXG4gIC0tanNvbiAgICAgICAgICAgICBPdXRwdXQgYXMgSlNPTlxuXG4ke2NvbG9ycy5ib2xkfUdlbmVyYXRlIE9wdGlvbnM6JHtjb2xvcnMucmVzZXR9XG4gIC0tb3V0cHV0LCAtbyAgICAgICBPdXRwdXQgZmlsZSBwYXRoIChkZWZhdWx0OiAuY3Vyc29ycnVsZXMpXG5cbiR7Y29sb3JzLmJvbGR9RXhhbXBsZXM6JHtjb2xvcnMucmVzZXR9XG4gIGdhbGxvcCBhdWRpdFxuICBnYWxsb3AgYXVkaXQgc3JjL2Jsb2Nrcy8gLS1zdHJpY3RcbiAgZ2FsbG9wIGdlbmVyYXRlXG4gIGdhbGxvcCBnZW5lcmF0ZSAuY3Vyc29ycnVsZXNcbiAgZ2FsbG9wIGdlbmVyYXRlIC0tb3V0cHV0IC5naXRodWIvY29waWxvdC1pbnN0cnVjdGlvbnMubWRcbmApXG59XG5cbmZ1bmN0aW9uIHNob3dWZXJzaW9uKCkge1xuICBjb25zb2xlLmxvZyhgR2FsbG9wIENMSSB2MS4wLjBgKVxuICBjb25zb2xlLmxvZyhgQ2Fub24gdiR7dmVyc2lvbn1gKVxufVxuXG5hc3luYyBmdW5jdGlvbiBtYWluKCkge1xuICBzd2l0Y2ggKGNvbW1hbmQpIHtcbiAgICBjYXNlICdhdWRpdCc6XG4gICAgICBjb25zdCBhdWRpdFBhdGggPVxuICAgICAgICBhcmdzWzFdICYmICFhcmdzWzFdLnN0YXJ0c1dpdGgoJy0tJykgPyBhcmdzWzFdIDogJ3NyYy9ibG9ja3MvJ1xuICAgICAgY29uc3QgYXVkaXRPcHRpb25zID0ge1xuICAgICAgICBzdHJpY3Q6IGFyZ3MuaW5jbHVkZXMoJy0tc3RyaWN0JyksXG4gICAgICAgIGpzb246IGFyZ3MuaW5jbHVkZXMoJy0tanNvbicpLFxuICAgICAgICBmaXg6IGFyZ3MuaW5jbHVkZXMoJy0tZml4JyksXG4gICAgICB9XG4gICAgICBhd2FpdCBhdWRpdChhdWRpdFBhdGgsIGF1ZGl0T3B0aW9ucylcbiAgICAgIGJyZWFrXG5cbiAgICBjYXNlICdnZW5lcmF0ZSc6XG4gICAgICAvLyBGaW5kIG91dHB1dCBwYXRoIGZyb20gYXJnc1xuICAgICAgbGV0IG91dHB1dFBhdGggPSAnLmN1cnNvcnJ1bGVzJ1xuICAgICAgY29uc3Qgb3V0cHV0SW5kZXggPSBhcmdzLmluZGV4T2YoJy0tb3V0cHV0JylcbiAgICAgIGNvbnN0IG91dHB1dEluZGV4U2hvcnQgPSBhcmdzLmluZGV4T2YoJy1vJylcbiAgICAgIGlmIChvdXRwdXRJbmRleCAhPT0gLTEgJiYgYXJnc1tvdXRwdXRJbmRleCArIDFdKSB7XG4gICAgICAgIG91dHB1dFBhdGggPSBhcmdzW291dHB1dEluZGV4ICsgMV1cbiAgICAgIH0gZWxzZSBpZiAob3V0cHV0SW5kZXhTaG9ydCAhPT0gLTEgJiYgYXJnc1tvdXRwdXRJbmRleFNob3J0ICsgMV0pIHtcbiAgICAgICAgb3V0cHV0UGF0aCA9IGFyZ3Nbb3V0cHV0SW5kZXhTaG9ydCArIDFdXG4gICAgICB9IGVsc2UgaWYgKGFyZ3NbMV0gJiYgIWFyZ3NbMV0uc3RhcnRzV2l0aCgnLS0nKSkge1xuICAgICAgICBvdXRwdXRQYXRoID0gYXJnc1sxXVxuICAgICAgfVxuICAgICAgY29uc3QgZ2VuZXJhdGVPcHRpb25zID0ge1xuICAgICAgICBvdXRwdXQ6IG91dHB1dFBhdGgsXG4gICAgICAgIGZvcm1hdDogJ2N1cnNvcnJ1bGVzJyBhcyBjb25zdCxcbiAgICAgIH1cbiAgICAgIGF3YWl0IGdlbmVyYXRlKGdlbmVyYXRlT3B0aW9ucylcbiAgICAgIGJyZWFrXG5cbiAgICBjYXNlICd2ZXJzaW9uJzpcbiAgICBjYXNlICctdic6XG4gICAgY2FzZSAnLS12ZXJzaW9uJzpcbiAgICAgIHNob3dWZXJzaW9uKClcbiAgICAgIGJyZWFrXG5cbiAgICBjYXNlICdoZWxwJzpcbiAgICBjYXNlICctaCc6XG4gICAgY2FzZSAnLS1oZWxwJzpcbiAgICBjYXNlIHVuZGVmaW5lZDpcbiAgICAgIHNob3dIZWxwKClcbiAgICAgIGJyZWFrXG5cbiAgICBkZWZhdWx0OlxuICAgICAgY29uc29sZS5lcnJvcihgVW5rbm93biBjb21tYW5kOiAke2NvbW1hbmR9YClcbiAgICAgIGNvbnNvbGUuZXJyb3IoYFJ1biAnZ2FsbG9wIGhlbHAnIGZvciB1c2FnZSBpbmZvcm1hdGlvbi5gKVxuICAgICAgcHJvY2Vzcy5leGl0KDEpXG4gIH1cbn1cblxubWFpbigpLmNhdGNoKChlcnJvcikgPT4ge1xuICBjb25zb2xlLmVycm9yKCdFcnJvcjonLCBlcnJvci5tZXNzYWdlKVxuICBwcm9jZXNzLmV4aXQoMSlcbn0pXG4iXX0=
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Recommended configuration
|
|
3
|
+
* A sensible default for any Gallop-based template
|
|
4
|
+
*/
|
|
5
|
+
declare const recommendedConfig: {
|
|
6
|
+
plugins: string[];
|
|
7
|
+
rules: {
|
|
8
|
+
'gallop/no-client-blocks': string;
|
|
9
|
+
'gallop/no-container-in-section': string;
|
|
10
|
+
'gallop/prefer-component-props': string;
|
|
11
|
+
};
|
|
12
|
+
};
|
|
13
|
+
export default recommendedConfig;
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Recommended configuration
|
|
3
|
+
* A sensible default for any Gallop-based template
|
|
4
|
+
*/
|
|
5
|
+
const recommendedConfig = {
|
|
6
|
+
plugins: ['gallop'],
|
|
7
|
+
rules: {
|
|
8
|
+
// Core rules that apply to most templates
|
|
9
|
+
'gallop/no-client-blocks': 'warn',
|
|
10
|
+
'gallop/no-container-in-section': 'warn',
|
|
11
|
+
'gallop/prefer-component-props': 'warn',
|
|
12
|
+
},
|
|
13
|
+
};
|
|
14
|
+
export default recommendedConfig;
|
|
15
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVjb21tZW5kZWQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvZXNsaW50L2NvbmZpZ3MvcmVjb21tZW5kZWQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7OztHQUdHO0FBQ0gsTUFBTSxpQkFBaUIsR0FBRztJQUN4QixPQUFPLEVBQUUsQ0FBQyxRQUFRLENBQUM7SUFDbkIsS0FBSyxFQUFFO1FBQ0wsMENBQTBDO1FBQzFDLHlCQUF5QixFQUFFLE1BQU07UUFDakMsZ0NBQWdDLEVBQUUsTUFBTTtRQUN4QywrQkFBK0IsRUFBRSxNQUFNO0tBQ3hDO0NBQ0YsQ0FBQTtBQUVELGVBQWUsaUJBQWlCLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFJlY29tbWVuZGVkIGNvbmZpZ3VyYXRpb25cbiAqIEEgc2Vuc2libGUgZGVmYXVsdCBmb3IgYW55IEdhbGxvcC1iYXNlZCB0ZW1wbGF0ZVxuICovXG5jb25zdCByZWNvbW1lbmRlZENvbmZpZyA9IHtcbiAgcGx1Z2luczogWydnYWxsb3AnXSxcbiAgcnVsZXM6IHtcbiAgICAvLyBDb3JlIHJ1bGVzIHRoYXQgYXBwbHkgdG8gbW9zdCB0ZW1wbGF0ZXNcbiAgICAnZ2FsbG9wL25vLWNsaWVudC1ibG9ja3MnOiAnd2FybicsXG4gICAgJ2dhbGxvcC9uby1jb250YWluZXItaW4tc2VjdGlvbic6ICd3YXJuJyxcbiAgICAnZ2FsbG9wL3ByZWZlci1jb21wb25lbnQtcHJvcHMnOiAnd2FybicsXG4gIH0sXG59XG5cbmV4cG9ydCBkZWZhdWx0IHJlY29tbWVuZGVkQ29uZmlnXG4iXX0=
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Speedwell template configuration
|
|
3
|
+
* Enables all Gallop rules relevant to the Speedwell architecture
|
|
4
|
+
*/
|
|
5
|
+
declare const speedwellConfig: {
|
|
6
|
+
plugins: string[];
|
|
7
|
+
rules: {
|
|
8
|
+
'gallop/no-client-blocks': string;
|
|
9
|
+
'gallop/no-container-in-section': string;
|
|
10
|
+
'gallop/prefer-component-props': string;
|
|
11
|
+
};
|
|
12
|
+
};
|
|
13
|
+
export default speedwellConfig;
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Speedwell template configuration
|
|
3
|
+
* Enables all Gallop rules relevant to the Speedwell architecture
|
|
4
|
+
*/
|
|
5
|
+
const speedwellConfig = {
|
|
6
|
+
plugins: ['gallop'],
|
|
7
|
+
rules: {
|
|
8
|
+
// Blocks should be server components - extract client logic to components
|
|
9
|
+
'gallop/no-client-blocks': 'warn',
|
|
10
|
+
// Section already provides containment
|
|
11
|
+
'gallop/no-container-in-section': 'warn',
|
|
12
|
+
// Use component props instead of className for style values
|
|
13
|
+
'gallop/prefer-component-props': 'warn',
|
|
14
|
+
},
|
|
15
|
+
};
|
|
16
|
+
export default speedwellConfig;
|
|
17
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3BlZWR3ZWxsLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2VzbGludC9jb25maWdzL3NwZWVkd2VsbC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFDSCxNQUFNLGVBQWUsR0FBRztJQUN0QixPQUFPLEVBQUUsQ0FBQyxRQUFRLENBQUM7SUFDbkIsS0FBSyxFQUFFO1FBQ0wsMEVBQTBFO1FBQzFFLHlCQUF5QixFQUFFLE1BQU07UUFFakMsdUNBQXVDO1FBQ3ZDLGdDQUFnQyxFQUFFLE1BQU07UUFFeEMsNERBQTREO1FBQzVELCtCQUErQixFQUFFLE1BQU07S0FDeEM7Q0FDRixDQUFBO0FBRUQsZUFBZSxlQUFlLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFNwZWVkd2VsbCB0ZW1wbGF0ZSBjb25maWd1cmF0aW9uXG4gKiBFbmFibGVzIGFsbCBHYWxsb3AgcnVsZXMgcmVsZXZhbnQgdG8gdGhlIFNwZWVkd2VsbCBhcmNoaXRlY3R1cmVcbiAqL1xuY29uc3Qgc3BlZWR3ZWxsQ29uZmlnID0ge1xuICBwbHVnaW5zOiBbJ2dhbGxvcCddLFxuICBydWxlczoge1xuICAgIC8vIEJsb2NrcyBzaG91bGQgYmUgc2VydmVyIGNvbXBvbmVudHMgLSBleHRyYWN0IGNsaWVudCBsb2dpYyB0byBjb21wb25lbnRzXG4gICAgJ2dhbGxvcC9uby1jbGllbnQtYmxvY2tzJzogJ3dhcm4nLFxuXG4gICAgLy8gU2VjdGlvbiBhbHJlYWR5IHByb3ZpZGVzIGNvbnRhaW5tZW50XG4gICAgJ2dhbGxvcC9uby1jb250YWluZXItaW4tc2VjdGlvbic6ICd3YXJuJyxcblxuICAgIC8vIFVzZSBjb21wb25lbnQgcHJvcHMgaW5zdGVhZCBvZiBjbGFzc05hbWUgZm9yIHN0eWxlIHZhbHVlc1xuICAgICdnYWxsb3AvcHJlZmVyLWNvbXBvbmVudC1wcm9wcyc6ICd3YXJuJyxcbiAgfSxcbn1cblxuZXhwb3J0IGRlZmF1bHQgc3BlZWR3ZWxsQ29uZmlnXG4iXX0=
|