@polymorphism-tech/morph-spec 2.4.0 → 3.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CLAUDE.md +75 -239
- package/bin/morph-spec.js +8 -0
- package/content/.claude/commands/morph-deploy.md +529 -0
- package/content/.claude/skills/level-0-meta/README.md +7 -0
- package/content/.claude/skills/{checklists → level-0-meta}/morph-checklist.md +117 -117
- package/content/.claude/skills/level-1-workflows/README.md +7 -0
- package/content/.claude/skills/{workflows → level-1-workflows}/morph-replicate.md +213 -213
- package/content/.claude/skills/{workflows → level-1-workflows}/phase-clarify.md +131 -131
- package/content/.claude/skills/{workflows → level-1-workflows}/phase-design.md +213 -205
- package/content/.claude/skills/{workflows → level-1-workflows}/phase-setup.md +106 -92
- package/content/.claude/skills/{workflows → level-1-workflows}/phase-tasks.md +164 -164
- package/content/.claude/skills/{workflows → level-1-workflows}/phase-uiux.md +169 -138
- package/content/.claude/skills/level-2-domains/README.md +14 -0
- package/content/.claude/skills/level-2-domains/architecture/prompt-engineer.md +189 -0
- package/content/.claude/skills/level-2-domains/architecture/seo-growth-hacker.md +320 -0
- package/content/.claude/skills/level-2-domains/infrastructure/azure-deploy-specialist.md +699 -0
- package/content/.claude/skills/{specialists → level-2-domains/quality}/testing-specialist.md +126 -126
- package/content/.claude/skills/level-3-technologies/README.md +7 -0
- package/content/.claude/skills/level-4-patterns/README.md +7 -0
- package/content/.morph/config/agents.json +744 -358
- package/content/.morph/config/config.template.json +33 -0
- package/content/.morph/docs/workflows/enforcement-pipeline.md +668 -0
- package/content/.morph/examples/scheduled-reports/decisions.md +158 -158
- package/content/.morph/examples/scheduled-reports/proposal.md +95 -95
- package/content/.morph/examples/scheduled-reports/spec.md +267 -267
- package/content/.morph/hooks/README.md +158 -0
- package/content/.morph/hooks/task-completed.js +73 -0
- package/content/.morph/hooks/teammate-idle.js +68 -0
- package/content/.morph/schemas/tasks.schema.json +220 -220
- package/content/.morph/standards/agent-teams-workflow.md +474 -0
- package/content/.morph/templates/CONTEXT-FEATURE.md +276 -0
- package/content/.morph/templates/CONTEXT.md +170 -0
- package/content/.morph/templates/clarify-questions.md +159 -159
- package/content/.morph/templates/contracts/Commands.cs +74 -74
- package/content/.morph/templates/contracts/Entities.cs +25 -25
- package/content/.morph/templates/contracts/Queries.cs +74 -74
- package/content/.morph/templates/contracts/README.md +74 -74
- package/content/.morph/templates/infra/azure-pipelines-deploy.yml +480 -0
- package/package.json +1 -2
- package/scripts/reorganize-skills.cjs +175 -0
- package/scripts/validate-agents-structure.cjs +52 -0
- package/scripts/validate-skills.cjs +180 -0
- package/src/commands/advance-phase.js +83 -0
- package/src/commands/deploy.js +780 -0
- package/src/commands/detect-agents.js +43 -6
- package/src/commands/detect.js +1 -1
- package/src/commands/generate-context.js +40 -0
- package/src/commands/state.js +2 -1
- package/src/commands/sync.js +1 -1
- package/src/commands/update.js +13 -1
- package/src/lib/context-generator.js +513 -0
- package/src/lib/design-system-detector.js +187 -0
- package/src/lib/design-system-scaffolder.js +299 -0
- package/src/lib/hook-executor.js +256 -0
- package/src/lib/spec-validator.js +258 -0
- package/src/lib/standards-context-injector.js +287 -0
- package/src/lib/state-manager.js +21 -4
- package/src/lib/team-orchestrator.js +322 -0
- package/src/lib/validation-runner.js +65 -13
- package/src/lib/validators/design-system-validator.js +231 -0
- package/src/utils/file-copier.js +9 -1
- /package/content/.claude/skills/{checklists → level-0-meta}/code-review.md +0 -0
- /package/content/.claude/skills/{checklists → level-0-meta}/simulation-checklist.md +0 -0
- /package/content/.claude/skills/{specialists → level-2-domains/ai-agents}/ai-system-architect.md +0 -0
- /package/content/.claude/skills/{specialists → level-2-domains/architecture}/po-pm-advisor.md +0 -0
- /package/content/.claude/skills/{specialists → level-2-domains/architecture}/standards-architect.md +0 -0
- /package/content/.claude/skills/{specialists → level-2-domains/backend}/dotnet-senior.md +0 -0
- /package/content/.claude/skills/{specialists → level-2-domains/backend}/ef-modeler.md +0 -0
- /package/content/.claude/skills/{specialists → level-2-domains/backend}/hangfire-orchestrator.md +0 -0
- /package/content/.claude/skills/{specialists → level-2-domains/backend}/ms-agent-expert.md +0 -0
- /package/content/.claude/skills/{stacks/dotnet-blazor.md → level-2-domains/frontend/blazor-builder.md} +0 -0
- /package/content/.claude/skills/{stacks/dotnet-nextjs.md → level-2-domains/frontend/nextjs-expert.md} +0 -0
- /package/content/.claude/skills/{specialists → level-2-domains/frontend}/ui-ux-designer.md +0 -0
- /package/content/.claude/skills/{specialists → level-2-domains/infrastructure}/azure-architect.md +0 -0
- /package/content/.claude/skills/{infra → level-2-domains/infrastructure}/bicep-architect.md +0 -0
- /package/content/.claude/skills/{infra → level-2-domains/infrastructure}/container-specialist.md +0 -0
- /package/content/.claude/skills/{infra → level-2-domains/infrastructure}/devops-engineer.md +0 -0
- /package/content/.claude/skills/{integrations → level-2-domains/integrations}/asaas-financial.md +0 -0
- /package/content/.claude/skills/{integrations → level-2-domains/integrations}/azure-identity.md +0 -0
- /package/content/.claude/skills/{integrations → level-2-domains/integrations}/clerk-auth.md +0 -0
- /package/content/.claude/skills/{integrations → level-2-domains/integrations}/resend-email.md +0 -0
- /package/content/.claude/skills/{specialists → level-2-domains/quality}/code-analyzer.md +0 -0
- /package/{detectors → src/lib/detectors}/config-detector.js +0 -0
- /package/{detectors → src/lib/detectors}/conversation-analyzer.js +0 -0
- /package/{detectors → src/lib/detectors}/index.js +0 -0
- /package/{detectors → src/lib/detectors}/standards-generator.js +0 -0
- /package/{detectors → src/lib/detectors}/structure-detector.js +0 -0
|
@@ -11,17 +11,32 @@ import { existsSync, readFileSync } from 'fs';
|
|
|
11
11
|
import { join } from 'path';
|
|
12
12
|
import chalk from 'chalk';
|
|
13
13
|
import { loadState } from './state-manager.js';
|
|
14
|
+
import { loadConstraints } from './decision-constraint-loader.js';
|
|
14
15
|
|
|
15
16
|
/**
|
|
16
|
-
*
|
|
17
|
+
* Load agent → validators map from agents.json (data-driven)
|
|
18
|
+
* @param {string} projectPath - Root path of the project
|
|
19
|
+
* @returns {Object} Map of agent IDs to validator arrays
|
|
17
20
|
*/
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
}
|
|
21
|
+
function loadAgentValidatorMap(projectPath) {
|
|
22
|
+
try {
|
|
23
|
+
const agentsPath = join(projectPath, 'content/.morph/config/agents.json');
|
|
24
|
+
const agentsConfig = JSON.parse(readFileSync(agentsPath, 'utf8'));
|
|
25
|
+
|
|
26
|
+
const map = {};
|
|
27
|
+
Object.entries(agentsConfig.agents || {}).forEach(([agentId, agent]) => {
|
|
28
|
+
if (!agentId.startsWith('_comment') && agent.validators && agent.validators.length > 0) {
|
|
29
|
+
map[agentId] = agent.validators;
|
|
30
|
+
}
|
|
31
|
+
});
|
|
32
|
+
|
|
33
|
+
return map;
|
|
34
|
+
} catch (err) {
|
|
35
|
+
console.warn(chalk.yellow('⚠️ Failed to load agents.json - no validators will run'));
|
|
36
|
+
console.warn(chalk.gray(` Run 'morph-spec init' to set up MORPH-SPEC configuration`));
|
|
37
|
+
return {}; // Empty map = honest failure
|
|
38
|
+
}
|
|
39
|
+
}
|
|
25
40
|
|
|
26
41
|
/**
|
|
27
42
|
* Run all relevant validators for a feature
|
|
@@ -39,11 +54,18 @@ export async function runValidation(projectPath, featureName, options = {}) {
|
|
|
39
54
|
errors: 0,
|
|
40
55
|
warnings: 0,
|
|
41
56
|
issues: [],
|
|
42
|
-
summary: ''
|
|
57
|
+
summary: '',
|
|
58
|
+
results: {} // Per-validator results for hooks integration
|
|
43
59
|
};
|
|
44
60
|
|
|
61
|
+
// Load decision constraints from decisions.md
|
|
62
|
+
const constraints = loadConstraints(projectPath, featureName);
|
|
63
|
+
|
|
45
64
|
// Determine which validators to run
|
|
46
|
-
const validatorIds = options.validators || detectValidators(featureName);
|
|
65
|
+
const validatorIds = options.validators || detectValidators(featureName, projectPath);
|
|
66
|
+
|
|
67
|
+
// Merge constraint-enabled validators
|
|
68
|
+
validatorIds.push(...constraints.enabledValidators);
|
|
47
69
|
|
|
48
70
|
if (validatorIds.length === 0) {
|
|
49
71
|
result.summary = 'No validators applicable for this feature';
|
|
@@ -52,6 +74,9 @@ export async function runValidation(projectPath, featureName, options = {}) {
|
|
|
52
74
|
|
|
53
75
|
if (options.verbose) {
|
|
54
76
|
console.log(chalk.gray(`\n Running validators: ${validatorIds.join(', ')}`));
|
|
77
|
+
if (constraints.enabledValidators.length > 0) {
|
|
78
|
+
console.log(chalk.gray(` (+ ${constraints.enabledValidators.length} from decisions)`));
|
|
79
|
+
}
|
|
55
80
|
}
|
|
56
81
|
|
|
57
82
|
// Always run contract compliance if contracts.cs exists
|
|
@@ -66,7 +91,13 @@ export async function runValidation(projectPath, featureName, options = {}) {
|
|
|
66
91
|
// Run each validator
|
|
67
92
|
for (const validatorId of uniqueValidators) {
|
|
68
93
|
try {
|
|
69
|
-
|
|
94
|
+
// Pass validator-specific constraints from decisions.md
|
|
95
|
+
const validatorOptions = {
|
|
96
|
+
...options,
|
|
97
|
+
constraints: constraints.validators[validatorId] || {}
|
|
98
|
+
};
|
|
99
|
+
|
|
100
|
+
const validatorResult = await runSingleValidator(validatorId, projectPath, featureName, validatorOptions);
|
|
70
101
|
|
|
71
102
|
if (validatorResult) {
|
|
72
103
|
result.errors += validatorResult.errors || 0;
|
|
@@ -75,11 +106,21 @@ export async function runValidation(projectPath, featureName, options = {}) {
|
|
|
75
106
|
if (validatorResult.issues) {
|
|
76
107
|
result.issues.push(...validatorResult.issues);
|
|
77
108
|
}
|
|
109
|
+
|
|
110
|
+
// Store per-validator results for hooks
|
|
111
|
+
result.results[validatorId] = {
|
|
112
|
+
passed: (validatorResult.errors || 0) === 0,
|
|
113
|
+
issues: validatorResult.issues || []
|
|
114
|
+
};
|
|
78
115
|
}
|
|
79
116
|
} catch (err) {
|
|
80
117
|
if (options.verbose) {
|
|
81
118
|
console.log(chalk.yellow(` ⚠ Validator '${validatorId}' failed: ${err.message}`));
|
|
82
119
|
}
|
|
120
|
+
result.results[validatorId] = {
|
|
121
|
+
passed: false,
|
|
122
|
+
error: err.message
|
|
123
|
+
};
|
|
83
124
|
}
|
|
84
125
|
}
|
|
85
126
|
|
|
@@ -93,8 +134,11 @@ export async function runValidation(projectPath, featureName, options = {}) {
|
|
|
93
134
|
|
|
94
135
|
/**
|
|
95
136
|
* Detect which validators to run based on feature's active agents
|
|
137
|
+
* @param {string} featureName - Feature name
|
|
138
|
+
* @param {string} projectPath - Project root path (default: current directory)
|
|
139
|
+
* @returns {string[]} Array of validator IDs to run
|
|
96
140
|
*/
|
|
97
|
-
function detectValidators(featureName) {
|
|
141
|
+
function detectValidators(featureName, projectPath = '.') {
|
|
98
142
|
const state = loadState(false);
|
|
99
143
|
if (!state || !state.features[featureName]) {
|
|
100
144
|
return ['architecture', 'packages']; // Default validators
|
|
@@ -104,8 +148,11 @@ function detectValidators(featureName) {
|
|
|
104
148
|
const agents = feature.activeAgents || [];
|
|
105
149
|
const validatorSet = new Set();
|
|
106
150
|
|
|
151
|
+
// Load agent → validators map from agents.json (data-driven)
|
|
152
|
+
const agentValidatorMap = loadAgentValidatorMap(projectPath);
|
|
153
|
+
|
|
107
154
|
for (const agentId of agents) {
|
|
108
|
-
const validators =
|
|
155
|
+
const validators = agentValidatorMap[agentId] || [];
|
|
109
156
|
validators.forEach(v => validatorSet.add(v));
|
|
110
157
|
}
|
|
111
158
|
|
|
@@ -177,6 +224,11 @@ async function runSingleValidator(validatorId, projectPath, featureName, options
|
|
|
177
224
|
return await validateContracts(projectPath, featureName, options);
|
|
178
225
|
}
|
|
179
226
|
|
|
227
|
+
case 'design-system': {
|
|
228
|
+
const { validateDesignSystem } = await import('./validators/design-system-validator.js');
|
|
229
|
+
return await validateDesignSystem(projectPath, featureName, options);
|
|
230
|
+
}
|
|
231
|
+
|
|
180
232
|
default:
|
|
181
233
|
if (options.verbose) {
|
|
182
234
|
console.log(chalk.gray(` Unknown validator: ${validatorId}`));
|
|
@@ -0,0 +1,231 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Design System Validator
|
|
3
|
+
*
|
|
4
|
+
* Validates that CSS and Razor files only use design system tokens
|
|
5
|
+
* instead of hardcoded colors, fonts, or spacing values.
|
|
6
|
+
*
|
|
7
|
+
* MORPH-SPEC 3.0 - Phase 3: Design System Enforcement
|
|
8
|
+
*/
|
|
9
|
+
|
|
10
|
+
import { existsSync, readFileSync, readdirSync, statSync } from 'fs';
|
|
11
|
+
import { join, extname } from 'path';
|
|
12
|
+
import { parseColors, parseTypography, parseSpacing } from '../design-system-generator.js';
|
|
13
|
+
import { detectDesignSystem } from '../design-system-detector.js';
|
|
14
|
+
|
|
15
|
+
/**
|
|
16
|
+
* Validate design system compliance
|
|
17
|
+
* @param {string} projectPath - Root path of the project
|
|
18
|
+
* @param {string} featureName - Feature name (optional)
|
|
19
|
+
* @param {Object} options - Validation options
|
|
20
|
+
* @returns {Object} { errors, warnings, issues }
|
|
21
|
+
*/
|
|
22
|
+
export async function validateDesignSystem(projectPath, featureName = null, options = {}) {
|
|
23
|
+
const result = {
|
|
24
|
+
errors: 0,
|
|
25
|
+
warnings: 0,
|
|
26
|
+
issues: []
|
|
27
|
+
};
|
|
28
|
+
|
|
29
|
+
// Detect design system
|
|
30
|
+
const dsDetection = detectDesignSystem(projectPath, featureName);
|
|
31
|
+
|
|
32
|
+
if (!dsDetection.hasDesignSystem) {
|
|
33
|
+
result.errors++;
|
|
34
|
+
result.issues.push({
|
|
35
|
+
level: 'error',
|
|
36
|
+
message: 'No design system found',
|
|
37
|
+
solution: 'Create a design system at .morph/project/design-system.md or run design system generator'
|
|
38
|
+
});
|
|
39
|
+
return result;
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
// Parse design system
|
|
43
|
+
let designSystem;
|
|
44
|
+
try {
|
|
45
|
+
const markdown = readFileSync(dsDetection.path, 'utf8');
|
|
46
|
+
designSystem = {
|
|
47
|
+
colors: parseColors(markdown),
|
|
48
|
+
typography: parseTypography(markdown),
|
|
49
|
+
spacing: parseSpacing(markdown)
|
|
50
|
+
};
|
|
51
|
+
} catch (err) {
|
|
52
|
+
result.errors++;
|
|
53
|
+
result.issues.push({
|
|
54
|
+
level: 'error',
|
|
55
|
+
message: `Failed to parse design system: ${err.message}`,
|
|
56
|
+
file: dsDetection.path
|
|
57
|
+
});
|
|
58
|
+
return result;
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
// Build allowed color set
|
|
62
|
+
const allowedColors = new Set();
|
|
63
|
+
Object.values(designSystem.colors.primary).forEach(hex => allowedColors.add(hex.toLowerCase()));
|
|
64
|
+
Object.values(designSystem.colors.secondary).forEach(hex => allowedColors.add(hex.toLowerCase()));
|
|
65
|
+
Object.values(designSystem.colors.neutral).forEach(hex => allowedColors.add(hex.toLowerCase()));
|
|
66
|
+
Object.values(designSystem.colors.semantic).forEach(hex => allowedColors.add(hex.toLowerCase()));
|
|
67
|
+
|
|
68
|
+
// Common system colors (allowed)
|
|
69
|
+
const systemColors = new Set([
|
|
70
|
+
'#ffffff', '#fff', '#000000', '#000', 'transparent', 'inherit', 'currentcolor'
|
|
71
|
+
]);
|
|
72
|
+
|
|
73
|
+
// Scan CSS files
|
|
74
|
+
const cssFiles = findFiles(projectPath, ['.css'], options.scanPaths || ['wwwroot/css', 'styles']);
|
|
75
|
+
for (const cssFile of cssFiles) {
|
|
76
|
+
const cssResult = validateCSSFile(cssFile, allowedColors, systemColors);
|
|
77
|
+
result.warnings += cssResult.warnings;
|
|
78
|
+
result.issues.push(...cssResult.issues);
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
// Scan Razor files
|
|
82
|
+
const razorFiles = findFiles(projectPath, ['.razor'], options.scanPaths || ['Components', 'Pages']);
|
|
83
|
+
for (const razorFile of razorFiles) {
|
|
84
|
+
const razorResult = validateRazorFile(razorFile, allowedColors, systemColors);
|
|
85
|
+
result.warnings += razorResult.warnings;
|
|
86
|
+
result.issues.push(...razorResult.issues);
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
return result;
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
/**
|
|
93
|
+
* Find files with specific extensions in directories
|
|
94
|
+
*/
|
|
95
|
+
function findFiles(projectPath, extensions, scanPaths) {
|
|
96
|
+
const files = [];
|
|
97
|
+
|
|
98
|
+
for (const scanPath of scanPaths) {
|
|
99
|
+
const fullPath = join(projectPath, scanPath);
|
|
100
|
+
if (!existsSync(fullPath)) continue;
|
|
101
|
+
|
|
102
|
+
try {
|
|
103
|
+
walkDirectory(fullPath, extensions, files);
|
|
104
|
+
} catch (err) {
|
|
105
|
+
// Ignore permission errors
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
return files;
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
/**
|
|
113
|
+
* Recursively walk directory
|
|
114
|
+
*/
|
|
115
|
+
function walkDirectory(dir, extensions, files) {
|
|
116
|
+
const items = readdirSync(dir);
|
|
117
|
+
|
|
118
|
+
for (const item of items) {
|
|
119
|
+
const fullPath = join(dir, item);
|
|
120
|
+
const stat = statSync(fullPath);
|
|
121
|
+
|
|
122
|
+
if (stat.isDirectory()) {
|
|
123
|
+
walkDirectory(fullPath, extensions, files);
|
|
124
|
+
} else if (stat.isFile()) {
|
|
125
|
+
const ext = extname(fullPath);
|
|
126
|
+
if (extensions.includes(ext)) {
|
|
127
|
+
files.push(fullPath);
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
/**
|
|
134
|
+
* Validate CSS file for hardcoded colors
|
|
135
|
+
*/
|
|
136
|
+
function validateCSSFile(filePath, allowedColors, systemColors) {
|
|
137
|
+
const result = {
|
|
138
|
+
warnings: 0,
|
|
139
|
+
issues: []
|
|
140
|
+
};
|
|
141
|
+
|
|
142
|
+
try {
|
|
143
|
+
const content = readFileSync(filePath, 'utf8');
|
|
144
|
+
|
|
145
|
+
// Match hex colors: #RGB or #RRGGBB
|
|
146
|
+
const hexRegex = /#([0-9A-Fa-f]{3}|[0-9A-Fa-f]{6})\b/g;
|
|
147
|
+
|
|
148
|
+
let match;
|
|
149
|
+
let lineNumber = 1;
|
|
150
|
+
let lastIndex = 0;
|
|
151
|
+
|
|
152
|
+
while ((match = hexRegex.exec(content)) !== null) {
|
|
153
|
+
const hex = match[0].toLowerCase();
|
|
154
|
+
|
|
155
|
+
// Count line number
|
|
156
|
+
const upToMatch = content.substring(lastIndex, match.index);
|
|
157
|
+
lineNumber += (upToMatch.match(/\n/g) || []).length;
|
|
158
|
+
lastIndex = match.index;
|
|
159
|
+
|
|
160
|
+
// Skip if in allowed or system colors
|
|
161
|
+
if (allowedColors.has(hex) || systemColors.has(hex)) {
|
|
162
|
+
continue;
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
result.warnings++;
|
|
166
|
+
result.issues.push({
|
|
167
|
+
level: 'warning',
|
|
168
|
+
message: `Hardcoded color ${hex} not in design system palette`,
|
|
169
|
+
file: filePath,
|
|
170
|
+
line: lineNumber,
|
|
171
|
+
solution: 'Use CSS variable like var(--color-primary-500) instead'
|
|
172
|
+
});
|
|
173
|
+
}
|
|
174
|
+
} catch (err) {
|
|
175
|
+
// Ignore read errors
|
|
176
|
+
}
|
|
177
|
+
|
|
178
|
+
return result;
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
/**
|
|
182
|
+
* Validate Razor file for inline styles with hardcoded colors
|
|
183
|
+
*/
|
|
184
|
+
function validateRazorFile(filePath, allowedColors, systemColors) {
|
|
185
|
+
const result = {
|
|
186
|
+
warnings: 0,
|
|
187
|
+
issues: []
|
|
188
|
+
};
|
|
189
|
+
|
|
190
|
+
try {
|
|
191
|
+
const content = readFileSync(filePath, 'utf8');
|
|
192
|
+
|
|
193
|
+
// Match style="" attributes with hex colors
|
|
194
|
+
const styleRegex = /style\s*=\s*"([^"]*)"/gi;
|
|
195
|
+
|
|
196
|
+
let match;
|
|
197
|
+
while ((match = styleRegex.exec(content)) !== null) {
|
|
198
|
+
const styleContent = match[1];
|
|
199
|
+
|
|
200
|
+
// Find hex colors in style attribute
|
|
201
|
+
const hexRegex = /#([0-9A-Fa-f]{3}|[0-9A-Fa-f]{6})\b/g;
|
|
202
|
+
let hexMatch;
|
|
203
|
+
|
|
204
|
+
while ((hexMatch = hexRegex.exec(styleContent)) !== null) {
|
|
205
|
+
const hex = hexMatch[0].toLowerCase();
|
|
206
|
+
|
|
207
|
+
// Skip if in allowed or system colors
|
|
208
|
+
if (allowedColors.has(hex) || systemColors.has(hex)) {
|
|
209
|
+
continue;
|
|
210
|
+
}
|
|
211
|
+
|
|
212
|
+
// Count line number
|
|
213
|
+
const upToMatch = content.substring(0, match.index);
|
|
214
|
+
const lineNumber = (upToMatch.match(/\n/g) || []).length + 1;
|
|
215
|
+
|
|
216
|
+
result.warnings++;
|
|
217
|
+
result.issues.push({
|
|
218
|
+
level: 'warning',
|
|
219
|
+
message: `Inline style with hardcoded color ${hex} not in design system`,
|
|
220
|
+
file: filePath,
|
|
221
|
+
line: lineNumber,
|
|
222
|
+
solution: 'Use CSS class with design system variables or component props instead of inline styles'
|
|
223
|
+
});
|
|
224
|
+
}
|
|
225
|
+
}
|
|
226
|
+
} catch (err) {
|
|
227
|
+
// Ignore read errors
|
|
228
|
+
}
|
|
229
|
+
|
|
230
|
+
return result;
|
|
231
|
+
}
|
package/src/utils/file-copier.js
CHANGED
|
@@ -1,14 +1,20 @@
|
|
|
1
1
|
import fs from 'fs-extra';
|
|
2
|
-
import { join, dirname } from 'path';
|
|
2
|
+
import { join, dirname, resolve } from 'path';
|
|
3
3
|
import { fileURLToPath } from 'url';
|
|
4
4
|
|
|
5
5
|
const __dirname = dirname(fileURLToPath(import.meta.url));
|
|
6
6
|
|
|
7
|
+
function isSamePath(a, b) {
|
|
8
|
+
return resolve(a) === resolve(b);
|
|
9
|
+
}
|
|
10
|
+
|
|
7
11
|
export function getContentDir() {
|
|
8
12
|
return join(__dirname, '..', '..', 'content');
|
|
9
13
|
}
|
|
10
14
|
|
|
11
15
|
export async function copyDirectory(src, dest, options = {}) {
|
|
16
|
+
if (isSamePath(src, dest)) return;
|
|
17
|
+
|
|
12
18
|
const { filter, overwrite = true } = options;
|
|
13
19
|
|
|
14
20
|
await fs.copy(src, dest, {
|
|
@@ -19,6 +25,8 @@ export async function copyDirectory(src, dest, options = {}) {
|
|
|
19
25
|
}
|
|
20
26
|
|
|
21
27
|
export async function copyFile(src, dest) {
|
|
28
|
+
if (isSamePath(src, dest)) return;
|
|
29
|
+
|
|
22
30
|
await fs.ensureDir(dirname(dest));
|
|
23
31
|
await fs.copy(src, dest, { overwrite: true });
|
|
24
32
|
}
|
|
File without changes
|
|
File without changes
|
/package/content/.claude/skills/{specialists → level-2-domains/ai-agents}/ai-system-architect.md
RENAMED
|
File without changes
|
/package/content/.claude/skills/{specialists → level-2-domains/architecture}/po-pm-advisor.md
RENAMED
|
File without changes
|
/package/content/.claude/skills/{specialists → level-2-domains/architecture}/standards-architect.md
RENAMED
|
File without changes
|
|
File without changes
|
|
File without changes
|
/package/content/.claude/skills/{specialists → level-2-domains/backend}/hangfire-orchestrator.md
RENAMED
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
/package/content/.claude/skills/{specialists → level-2-domains/infrastructure}/azure-architect.md
RENAMED
|
File without changes
|
|
File without changes
|
/package/content/.claude/skills/{infra → level-2-domains/infrastructure}/container-specialist.md
RENAMED
|
File without changes
|
|
File without changes
|
/package/content/.claude/skills/{integrations → level-2-domains/integrations}/asaas-financial.md
RENAMED
|
File without changes
|
/package/content/.claude/skills/{integrations → level-2-domains/integrations}/azure-identity.md
RENAMED
|
File without changes
|
|
File without changes
|
/package/content/.claude/skills/{integrations → level-2-domains/integrations}/resend-email.md
RENAMED
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|