@polymorphism-tech/morph-spec 2.4.0 → 3.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/CLAUDE.md +158 -26
- package/LICENSE +72 -72
- package/bin/detect-agents.js +225 -225
- package/bin/morph-spec.js +8 -0
- package/bin/render-template.js +302 -302
- package/bin/semantic-detect-agents.js +246 -246
- package/bin/validate-agents-skills.js +251 -251
- package/bin/validate-agents.js +69 -69
- package/bin/validate-phase.js +263 -263
- package/content/.azure/README.md +293 -293
- package/content/.azure/docs/azure-devops-setup.md +454 -454
- package/content/.azure/docs/branch-strategy.md +398 -398
- package/content/.azure/docs/local-development.md +515 -515
- package/content/.azure/pipelines/pipeline-variables.yml +34 -34
- package/content/.azure/pipelines/prod-pipeline.yml +319 -319
- package/content/.azure/pipelines/staging-pipeline.yml +234 -234
- package/content/.azure/pipelines/templates/build-dotnet.yml +75 -75
- package/content/.azure/pipelines/templates/deploy-app-service.yml +94 -94
- package/content/.azure/pipelines/templates/deploy-container-app.yml +120 -120
- package/content/.azure/pipelines/templates/infra-deploy.yml +90 -90
- package/content/.claude/commands/morph-archive.md +79 -79
- package/content/.claude/commands/morph-deploy.md +529 -0
- package/content/.claude/commands/morph-infra.md +209 -209
- package/content/.claude/commands/morph-preflight.md +227 -227
- package/content/.claude/commands/morph-troubleshoot.md +122 -122
- package/content/.claude/settings.local.json +15 -15
- package/content/.claude/skills/infra/azure-deploy-specialist.md +699 -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/{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/.claude/skills/specialists/prompt-engineer.md +189 -0
- package/content/.claude/skills/specialists/seo-growth-hacker.md +320 -0
- package/content/.morph/.morphversion +5 -5
- package/content/.morph/archive/.gitkeep +25 -25
- package/content/.morph/config/agents.json +742 -358
- package/content/.morph/config/config.template.json +33 -0
- package/content/.morph/docs/STORY-DRIVEN-DEVELOPMENT.md +392 -392
- package/content/.morph/docs/workflows/enforcement-pipeline.md +668 -0
- package/content/.morph/examples/api-nextjs/README.md +241 -241
- package/content/.morph/examples/api-nextjs/contracts.ts +307 -307
- package/content/.morph/examples/api-nextjs/spec.md +399 -399
- package/content/.morph/examples/api-nextjs/tasks.md +168 -168
- package/content/.morph/examples/micro-saas/README.md +125 -125
- package/content/.morph/examples/micro-saas/contracts.cs +358 -358
- package/content/.morph/examples/micro-saas/decisions.md +246 -246
- package/content/.morph/examples/micro-saas/spec.md +236 -236
- package/content/.morph/examples/micro-saas/tasks.md +150 -150
- package/content/.morph/examples/multi-agent/README.md +309 -309
- package/content/.morph/examples/multi-agent/contracts.cs +433 -433
- package/content/.morph/examples/multi-agent/spec.md +479 -479
- package/content/.morph/examples/multi-agent/tasks.md +185 -185
- 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/examples/state-v3.json +188 -188
- package/content/.morph/features/.gitkeep +25 -25
- package/content/.morph/hooks/README.md +158 -0
- package/content/.morph/hooks/pre-commit-all.sh +48 -48
- package/content/.morph/hooks/pre-commit-specs.sh +49 -49
- package/content/.morph/hooks/pre-commit-tests.sh +60 -60
- package/content/.morph/hooks/task-completed.js +73 -0
- package/content/.morph/hooks/teammate-idle.js +68 -0
- package/content/.morph/project.md +160 -160
- package/content/.morph/schemas/agent.schema.json +296 -296
- package/content/.morph/schemas/tasks.schema.json +220 -220
- package/content/.morph/specs/.gitkeep +20 -20
- package/content/.morph/standards/agent-teams-workflow.md +474 -0
- package/content/.morph/standards/coding.md +377 -377
- package/content/.morph/standards/fluent-ui-setup.md +590 -590
- package/content/.morph/standards/migration-guide.md +514 -514
- package/content/.morph/standards/passkeys-auth.md +423 -423
- package/content/.morph/standards/vector-search-rag.md +536 -536
- package/content/.morph/state.json +17 -17
- package/content/.morph/templates/CONTEXT-FEATURE.md +276 -0
- package/content/.morph/templates/CONTEXT.md +170 -0
- package/content/.morph/templates/FluentDesignTheme.cs +149 -149
- package/content/.morph/templates/MudTheme.cs +281 -281
- package/content/.morph/templates/clarify-questions.md +159 -159
- package/content/.morph/templates/component.razor +239 -239
- 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/contracts.cs +217 -217
- package/content/.morph/templates/design-system.css +226 -226
- package/content/.morph/templates/infra/.dockerignore.example +89 -89
- package/content/.morph/templates/infra/Dockerfile.example +82 -82
- package/content/.morph/templates/infra/README.md +286 -286
- package/content/.morph/templates/infra/app-insights.bicep +63 -63
- package/content/.morph/templates/infra/app-service.bicep +164 -164
- package/content/.morph/templates/infra/azure-pipelines-deploy.yml +480 -0
- package/content/.morph/templates/infra/container-app-env.bicep +49 -49
- package/content/.morph/templates/infra/container-app.bicep +156 -156
- package/content/.morph/templates/infra/deploy-checklist.md +426 -426
- package/content/.morph/templates/infra/deploy.ps1 +229 -229
- package/content/.morph/templates/infra/deploy.sh +208 -208
- package/content/.morph/templates/infra/key-vault.bicep +91 -91
- package/content/.morph/templates/infra/main.bicep +189 -189
- package/content/.morph/templates/infra/parameters.dev.json +29 -29
- package/content/.morph/templates/infra/parameters.prod.json +29 -29
- package/content/.morph/templates/infra/parameters.staging.json +29 -29
- package/content/.morph/templates/infra/sql-database.bicep +103 -103
- package/content/.morph/templates/infra/storage.bicep +106 -106
- package/content/.morph/templates/integrations/asaas-client.cs +387 -387
- package/content/.morph/templates/integrations/asaas-webhook.cs +351 -351
- package/content/.morph/templates/integrations/azure-identity-config.cs +288 -288
- package/content/.morph/templates/integrations/clerk-config.cs +258 -258
- package/content/.morph/templates/job.cs +171 -171
- package/content/.morph/templates/migration.cs +83 -83
- package/content/.morph/templates/repository.cs +141 -141
- package/content/.morph/templates/saas/subscription.cs +347 -347
- package/content/.morph/templates/saas/tenant.cs +338 -338
- package/content/.morph/templates/service.cs +139 -139
- package/content/.morph/templates/sprint-status.yaml +68 -68
- package/content/.morph/templates/story.md +143 -143
- package/content/.morph/templates/test.cs +239 -239
- package/content/.morph/templates/ui-design-system.md +286 -286
- package/content/.morph/templates/ui-flows.md +336 -336
- package/content/.morph/templates/ui-mockups.md +133 -133
- package/content/.morph/test-infra/example.bicep +59 -59
- package/content/README.md +79 -79
- package/detectors/config-detector.js +223 -223
- package/detectors/conversation-analyzer.js +163 -163
- package/detectors/index.js +84 -84
- package/detectors/standards-generator.js +275 -275
- package/docs/api/fonts/Source-Sans-Pro/sourcesanspro-light-webfont.svg +977 -977
- package/docs/api/fonts/Source-Sans-Pro/sourcesanspro-regular-webfont.svg +1048 -1048
- package/docs/api/scripts/collapse.js +38 -38
- package/docs/api/scripts/commonNav.js +28 -28
- package/docs/api/scripts/linenumber.js +25 -25
- package/docs/api/scripts/nav.js +12 -12
- package/docs/api/scripts/polyfill.js +3 -3
- package/docs/api/scripts/prettify/Apache-License-2.0.txt +202 -202
- package/docs/api/scripts/prettify/lang-css.js +2 -2
- package/docs/api/scripts/prettify/prettify.js +28 -28
- package/docs/api/scripts/search.js +98 -98
- package/docs/api/styles/jsdoc.css +776 -776
- package/docs/api/styles/prettify.css +80 -80
- package/docs/examples.md +328 -328
- package/docs/templates.md +418 -418
- package/package.json +1 -1
- package/scripts/postinstall.js +132 -132
- package/src/commands/advance-phase.js +83 -0
- package/src/commands/analyze-blazor-concurrency.js +193 -193
- package/src/commands/create-story.js +351 -351
- package/src/commands/deploy.js +780 -0
- package/src/commands/detect-agents.js +34 -6
- package/src/commands/detect.js +104 -104
- package/src/commands/generate-context.js +40 -0
- package/src/commands/generate.js +149 -149
- package/src/commands/lint-fluent.js +352 -352
- package/src/commands/rollback-phase.js +185 -185
- package/src/commands/session-summary.js +291 -291
- package/src/commands/shard-spec.js +224 -224
- package/src/commands/sprint-status.js +250 -250
- package/src/commands/state.js +333 -333
- package/src/commands/sync.js +167 -167
- package/src/commands/troubleshoot.js +222 -222
- package/src/commands/validate-blazor-state.js +210 -210
- package/src/commands/validate-blazor.js +156 -156
- package/src/commands/validate-css.js +84 -84
- package/src/commands/validate-phase.js +221 -221
- package/src/lib/blazor-concurrency-analyzer.js +288 -288
- package/src/lib/blazor-state-validator.js +291 -291
- package/src/lib/blazor-validator.js +374 -374
- package/src/lib/context-generator.js +513 -0
- package/src/lib/css-validator.js +352 -352
- package/src/lib/design-system-detector.js +187 -0
- package/src/lib/design-system-generator.js +298 -298
- package/src/lib/design-system-scaffolder.js +299 -0
- package/src/lib/hook-executor.js +256 -0
- package/src/lib/learning-system.js +520 -520
- package/src/lib/mockup-generator.js +366 -366
- package/src/lib/spec-validator.js +258 -0
- package/src/lib/standards-context-injector.js +287 -0
- package/src/lib/team-orchestrator.js +322 -0
- package/src/lib/troubleshoot-grep.js +194 -194
- package/src/lib/troubleshoot-index.js +144 -144
- package/src/lib/ui-detector.js +350 -350
- package/src/lib/validation-runner.js +65 -13
- package/src/lib/validators/architecture-validator.js +387 -387
- package/src/lib/validators/design-system-validator.js +231 -0
- package/src/lib/validators/package-validator.js +360 -360
- package/src/lib/validators/ui-contrast-validator.js +422 -422
- package/src/utils/file-copier.js +9 -1
- package/src/utils/logger.js +32 -32
- package/src/utils/version-checker.js +175 -175
- /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
|
@@ -5,6 +5,7 @@ import chalk from 'chalk';
|
|
|
5
5
|
import { logger } from '../utils/logger.js';
|
|
6
6
|
import { analyzeRequestComplexity } from '../lib/complexity-analyzer.js';
|
|
7
7
|
import { LearningSystem } from '../lib/learning-system.js';
|
|
8
|
+
import { loadStandardsForAgent, getStandardsListForAgent } from '../lib/standards-context-injector.js';
|
|
8
9
|
|
|
9
10
|
const __dirname = dirname(fileURLToPath(import.meta.url));
|
|
10
11
|
const AGENTS_CONFIG_PATH = join(__dirname, '../../content/.morph/config/agents.json');
|
|
@@ -21,14 +22,24 @@ function loadAgentsConfig() {
|
|
|
21
22
|
|
|
22
23
|
function detectAgents(userInput, config) {
|
|
23
24
|
const input = userInput.toLowerCase();
|
|
24
|
-
const coreAgents = config.agents.core || [];
|
|
25
|
-
const specialists = config.agents.specialists || [];
|
|
26
25
|
|
|
26
|
+
// MORPH 3.0 Hierarchical: agents is a flat object, not arrays
|
|
27
|
+
const allAgents = Object.entries(config.agents || {})
|
|
28
|
+
.filter(([id]) => !id.startsWith('_comment'))
|
|
29
|
+
.map(([id, agent]) => ({ id, ...agent }));
|
|
30
|
+
|
|
31
|
+
// Core agents = always_active agents (typically Tier 1)
|
|
32
|
+
const coreAgents = allAgents.filter(a => a.always_active === true);
|
|
27
33
|
const coreIds = coreAgents.map(a => a.id);
|
|
34
|
+
|
|
35
|
+
// Specialist agents = keyword-matched agents
|
|
28
36
|
const specialistMatches = [];
|
|
29
37
|
|
|
30
|
-
for (const agent of
|
|
31
|
-
|
|
38
|
+
for (const agent of allAgents) {
|
|
39
|
+
if (agent.always_active) continue; // Skip core agents
|
|
40
|
+
|
|
41
|
+
// MORPH 3.0: keywords is directly on agent, not nested in autoActivation
|
|
42
|
+
const keywords = agent.keywords || [];
|
|
32
43
|
const matchedKeywords = [];
|
|
33
44
|
|
|
34
45
|
for (const keyword of keywords) {
|
|
@@ -40,8 +51,8 @@ function detectAgents(userInput, config) {
|
|
|
40
51
|
if (matchedKeywords.length > 0) {
|
|
41
52
|
specialistMatches.push({
|
|
42
53
|
id: agent.id,
|
|
43
|
-
name: agent.name
|
|
44
|
-
emoji: agent.
|
|
54
|
+
name: agent.title || agent.id, // Use title as name
|
|
55
|
+
emoji: agent.teammate?.icon || '🤖',
|
|
45
56
|
matchedKeywords,
|
|
46
57
|
matchCount: matchedKeywords.length
|
|
47
58
|
});
|
|
@@ -127,6 +138,23 @@ export function detectAgentsCommand(input, options) {
|
|
|
127
138
|
})
|
|
128
139
|
.filter(Boolean);
|
|
129
140
|
|
|
141
|
+
// Enrich with standards context when JSON output requested
|
|
142
|
+
if (options.json) {
|
|
143
|
+
const projectPath = process.cwd();
|
|
144
|
+
result.standardsSummary = {};
|
|
145
|
+
|
|
146
|
+
for (const agentId of result.all) {
|
|
147
|
+
const standardsList = getStandardsListForAgent(agentId);
|
|
148
|
+
const { fullContent } = loadStandardsForAgent(agentId, projectPath);
|
|
149
|
+
|
|
150
|
+
result.standardsSummary[agentId] = {
|
|
151
|
+
standards: standardsList,
|
|
152
|
+
fullContent: fullContent,
|
|
153
|
+
count: standardsList.length
|
|
154
|
+
};
|
|
155
|
+
}
|
|
156
|
+
}
|
|
157
|
+
|
|
130
158
|
if (options.json) {
|
|
131
159
|
console.log(JSON.stringify(result, null, 2));
|
|
132
160
|
} else if (options.verbose) {
|
package/src/commands/detect.js
CHANGED
|
@@ -1,104 +1,104 @@
|
|
|
1
|
-
import { join } from 'path';
|
|
2
|
-
import ora from 'ora';
|
|
3
|
-
import chalk from 'chalk';
|
|
4
|
-
import { logger } from '../utils/logger.js';
|
|
5
|
-
import { detectProject, getDetectionSummary } from '../../detectors/index.js';
|
|
6
|
-
import { ensureDir, writeFile } from '../utils/file-copier.js';
|
|
7
|
-
|
|
8
|
-
export async function detectCommand(options) {
|
|
9
|
-
const targetPath = options.path || process.cwd();
|
|
10
|
-
|
|
11
|
-
logger.header('MORPH-SPEC Project Detection');
|
|
12
|
-
logger.dim(`Analyzing: ${targetPath}`);
|
|
13
|
-
logger.blank();
|
|
14
|
-
|
|
15
|
-
const spinner = ora('Detecting project structure...').start();
|
|
16
|
-
|
|
17
|
-
try {
|
|
18
|
-
// Run detection
|
|
19
|
-
const results = await detectProject(targetPath, {
|
|
20
|
-
structure: true,
|
|
21
|
-
config: true,
|
|
22
|
-
conversation: true,
|
|
23
|
-
generateStandards: true
|
|
24
|
-
});
|
|
25
|
-
|
|
26
|
-
spinner.succeed('Detection complete!');
|
|
27
|
-
logger.blank();
|
|
28
|
-
|
|
29
|
-
// Display summary
|
|
30
|
-
logger.header('Detection Results');
|
|
31
|
-
logger.blank();
|
|
32
|
-
|
|
33
|
-
// Stack
|
|
34
|
-
logger.info(`Stack: ${chalk.cyan(results.structure.stack)}`);
|
|
35
|
-
logger.info(`Architecture: ${chalk.cyan(results.structure.architecture)}`);
|
|
36
|
-
if (results.structure.uiLibrary) {
|
|
37
|
-
logger.info(`UI Library: ${chalk.cyan(results.structure.uiLibrary)}`);
|
|
38
|
-
}
|
|
39
|
-
logger.blank();
|
|
40
|
-
|
|
41
|
-
// Technologies
|
|
42
|
-
if (results.config.technologies.length > 0) {
|
|
43
|
-
logger.header('Technologies:');
|
|
44
|
-
results.config.technologies.forEach(tech => {
|
|
45
|
-
logger.dim(` - ${tech}`);
|
|
46
|
-
});
|
|
47
|
-
logger.blank();
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
// Patterns
|
|
51
|
-
if (results.structure.patterns.length > 0) {
|
|
52
|
-
logger.header('Patterns:');
|
|
53
|
-
results.structure.patterns.forEach(pattern => {
|
|
54
|
-
logger.dim(` - ${pattern}`);
|
|
55
|
-
});
|
|
56
|
-
logger.blank();
|
|
57
|
-
}
|
|
58
|
-
|
|
59
|
-
// Recommendations
|
|
60
|
-
if (results.inferred.recommendations.length > 0) {
|
|
61
|
-
logger.header('Recommendations:');
|
|
62
|
-
results.inferred.recommendations.forEach(rec => {
|
|
63
|
-
logger.warn(` ⚠ ${rec}`);
|
|
64
|
-
});
|
|
65
|
-
logger.blank();
|
|
66
|
-
}
|
|
67
|
-
|
|
68
|
-
// Save results if requested
|
|
69
|
-
if (options.save !== false) {
|
|
70
|
-
spinner.start('Saving detection results...');
|
|
71
|
-
|
|
72
|
-
const outputDir = join(targetPath, '.morph', 'project', 'context');
|
|
73
|
-
await ensureDir(outputDir);
|
|
74
|
-
|
|
75
|
-
// Save detection log
|
|
76
|
-
const logPath = join(outputDir, 'detection-log.md');
|
|
77
|
-
const summary = getDetectionSummary(results);
|
|
78
|
-
await writeFile(logPath, summary);
|
|
79
|
-
|
|
80
|
-
// Save inferred standards
|
|
81
|
-
const standardsDir = join(targetPath, '.morph', 'project', 'standards');
|
|
82
|
-
await ensureDir(standardsDir);
|
|
83
|
-
|
|
84
|
-
const standardsPath = join(standardsDir, 'inferred.md');
|
|
85
|
-
await writeFile(standardsPath, results.inferred.markdown);
|
|
86
|
-
|
|
87
|
-
spinner.succeed('Results saved!');
|
|
88
|
-
logger.dim(` - ${logPath}`);
|
|
89
|
-
logger.dim(` - ${standardsPath}`);
|
|
90
|
-
}
|
|
91
|
-
|
|
92
|
-
// Verbose output
|
|
93
|
-
if (options.verbose) {
|
|
94
|
-
logger.blank();
|
|
95
|
-
logger.header('Detailed Results (JSON):');
|
|
96
|
-
console.log(JSON.stringify(results, null, 2));
|
|
97
|
-
}
|
|
98
|
-
|
|
99
|
-
} catch (error) {
|
|
100
|
-
spinner.fail('Detection failed');
|
|
101
|
-
logger.error(error.message);
|
|
102
|
-
process.exit(1);
|
|
103
|
-
}
|
|
104
|
-
}
|
|
1
|
+
import { join } from 'path';
|
|
2
|
+
import ora from 'ora';
|
|
3
|
+
import chalk from 'chalk';
|
|
4
|
+
import { logger } from '../utils/logger.js';
|
|
5
|
+
import { detectProject, getDetectionSummary } from '../../detectors/index.js';
|
|
6
|
+
import { ensureDir, writeFile } from '../utils/file-copier.js';
|
|
7
|
+
|
|
8
|
+
export async function detectCommand(options) {
|
|
9
|
+
const targetPath = options.path || process.cwd();
|
|
10
|
+
|
|
11
|
+
logger.header('MORPH-SPEC Project Detection');
|
|
12
|
+
logger.dim(`Analyzing: ${targetPath}`);
|
|
13
|
+
logger.blank();
|
|
14
|
+
|
|
15
|
+
const spinner = ora('Detecting project structure...').start();
|
|
16
|
+
|
|
17
|
+
try {
|
|
18
|
+
// Run detection
|
|
19
|
+
const results = await detectProject(targetPath, {
|
|
20
|
+
structure: true,
|
|
21
|
+
config: true,
|
|
22
|
+
conversation: true,
|
|
23
|
+
generateStandards: true
|
|
24
|
+
});
|
|
25
|
+
|
|
26
|
+
spinner.succeed('Detection complete!');
|
|
27
|
+
logger.blank();
|
|
28
|
+
|
|
29
|
+
// Display summary
|
|
30
|
+
logger.header('Detection Results');
|
|
31
|
+
logger.blank();
|
|
32
|
+
|
|
33
|
+
// Stack
|
|
34
|
+
logger.info(`Stack: ${chalk.cyan(results.structure.stack)}`);
|
|
35
|
+
logger.info(`Architecture: ${chalk.cyan(results.structure.architecture)}`);
|
|
36
|
+
if (results.structure.uiLibrary) {
|
|
37
|
+
logger.info(`UI Library: ${chalk.cyan(results.structure.uiLibrary)}`);
|
|
38
|
+
}
|
|
39
|
+
logger.blank();
|
|
40
|
+
|
|
41
|
+
// Technologies
|
|
42
|
+
if (results.config.technologies.length > 0) {
|
|
43
|
+
logger.header('Technologies:');
|
|
44
|
+
results.config.technologies.forEach(tech => {
|
|
45
|
+
logger.dim(` - ${tech}`);
|
|
46
|
+
});
|
|
47
|
+
logger.blank();
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
// Patterns
|
|
51
|
+
if (results.structure.patterns.length > 0) {
|
|
52
|
+
logger.header('Patterns:');
|
|
53
|
+
results.structure.patterns.forEach(pattern => {
|
|
54
|
+
logger.dim(` - ${pattern}`);
|
|
55
|
+
});
|
|
56
|
+
logger.blank();
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
// Recommendations
|
|
60
|
+
if (results.inferred.recommendations.length > 0) {
|
|
61
|
+
logger.header('Recommendations:');
|
|
62
|
+
results.inferred.recommendations.forEach(rec => {
|
|
63
|
+
logger.warn(` ⚠ ${rec}`);
|
|
64
|
+
});
|
|
65
|
+
logger.blank();
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
// Save results if requested
|
|
69
|
+
if (options.save !== false) {
|
|
70
|
+
spinner.start('Saving detection results...');
|
|
71
|
+
|
|
72
|
+
const outputDir = join(targetPath, '.morph', 'project', 'context');
|
|
73
|
+
await ensureDir(outputDir);
|
|
74
|
+
|
|
75
|
+
// Save detection log
|
|
76
|
+
const logPath = join(outputDir, 'detection-log.md');
|
|
77
|
+
const summary = getDetectionSummary(results);
|
|
78
|
+
await writeFile(logPath, summary);
|
|
79
|
+
|
|
80
|
+
// Save inferred standards
|
|
81
|
+
const standardsDir = join(targetPath, '.morph', 'project', 'standards');
|
|
82
|
+
await ensureDir(standardsDir);
|
|
83
|
+
|
|
84
|
+
const standardsPath = join(standardsDir, 'inferred.md');
|
|
85
|
+
await writeFile(standardsPath, results.inferred.markdown);
|
|
86
|
+
|
|
87
|
+
spinner.succeed('Results saved!');
|
|
88
|
+
logger.dim(` - ${logPath}`);
|
|
89
|
+
logger.dim(` - ${standardsPath}`);
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
// Verbose output
|
|
93
|
+
if (options.verbose) {
|
|
94
|
+
logger.blank();
|
|
95
|
+
logger.header('Detailed Results (JSON):');
|
|
96
|
+
console.log(JSON.stringify(results, null, 2));
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
} catch (error) {
|
|
100
|
+
spinner.fail('Detection failed');
|
|
101
|
+
logger.error(error.message);
|
|
102
|
+
process.exit(1);
|
|
103
|
+
}
|
|
104
|
+
}
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @fileoverview Generate Context Command
|
|
3
|
+
*
|
|
4
|
+
* CLI: morph-spec generate context [feature]
|
|
5
|
+
*
|
|
6
|
+
* Generates CONTEXT.md (project-level) or CONTEXT-{feature}.md (feature-level)
|
|
7
|
+
* with all relevant project data for Claude's consumption.
|
|
8
|
+
*
|
|
9
|
+
* @module generate-context
|
|
10
|
+
*/
|
|
11
|
+
|
|
12
|
+
import { generateContext } from '../lib/context-generator.js';
|
|
13
|
+
|
|
14
|
+
/**
|
|
15
|
+
* Generate context files
|
|
16
|
+
* @param {string} projectPath - Root path of the project
|
|
17
|
+
* @param {string} featureName - Optional feature name
|
|
18
|
+
* @returns {Promise<void>}
|
|
19
|
+
*/
|
|
20
|
+
export async function generateContextCommand(projectPath, featureName = null) {
|
|
21
|
+
console.log('📄 Generating CONTEXT files...\n');
|
|
22
|
+
|
|
23
|
+
try {
|
|
24
|
+
const results = await generateContext(projectPath, featureName);
|
|
25
|
+
|
|
26
|
+
console.log('✅ Generated:');
|
|
27
|
+
console.log(` - ${results.projectContext}`);
|
|
28
|
+
|
|
29
|
+
if (results.featureContext) {
|
|
30
|
+
console.log(` - ${results.featureContext}`);
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
console.log('\n💡 Use these files to provide full context to Claude Code.');
|
|
34
|
+
|
|
35
|
+
} catch (err) {
|
|
36
|
+
console.error('❌ Error generating context:');
|
|
37
|
+
console.error(` ${err.message}`);
|
|
38
|
+
process.exit(1);
|
|
39
|
+
}
|
|
40
|
+
}
|
package/src/commands/generate.js
CHANGED
|
@@ -1,149 +1,149 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* MORPH-SPEC Generate Command
|
|
3
|
-
* CLI wrapper for code generation operations
|
|
4
|
-
*/
|
|
5
|
-
|
|
6
|
-
import { mkdirSync, writeFileSync } from 'fs';
|
|
7
|
-
import { dirname } from 'path';
|
|
8
|
-
import ora from 'ora';
|
|
9
|
-
import chalk from 'chalk';
|
|
10
|
-
import { logger } from '../utils/logger.js';
|
|
11
|
-
import * as DesignSystemGenerator from '../lib/design-system-generator.js';
|
|
12
|
-
|
|
13
|
-
// ============================================================================
|
|
14
|
-
// Design System Subcommand
|
|
15
|
-
// ============================================================================
|
|
16
|
-
|
|
17
|
-
/**
|
|
18
|
-
* Generate design system files (CSS + themes) from ui-design-system.md
|
|
19
|
-
* @param {string} designSystemPath - Path to ui-design-system.md
|
|
20
|
-
* @param {Object} options - CLI options
|
|
21
|
-
*/
|
|
22
|
-
export async function generateDesignSystemCommand(designSystemPath, options) {
|
|
23
|
-
if (!designSystemPath) {
|
|
24
|
-
logger.error('Design system file path required');
|
|
25
|
-
logger.dim(' Usage: morph-spec generate design-system <ui-design-system.md>');
|
|
26
|
-
logger.blank();
|
|
27
|
-
logger.dim(' Example:');
|
|
28
|
-
logger.dim(' morph-spec generate design-system .morph/project/outputs/my-feature/ui-design-system.md');
|
|
29
|
-
process.exit(1);
|
|
30
|
-
}
|
|
31
|
-
|
|
32
|
-
logger.header('MORPH-SPEC Design System Generator');
|
|
33
|
-
logger.blank();
|
|
34
|
-
|
|
35
|
-
const spinner = ora('Parsing design system...').start();
|
|
36
|
-
|
|
37
|
-
try {
|
|
38
|
-
// Determine mode
|
|
39
|
-
let mode = 'both';
|
|
40
|
-
if (options.fluent && !options.mud) mode = 'fluent';
|
|
41
|
-
if (options.mud && !options.fluent) mode = 'mud';
|
|
42
|
-
|
|
43
|
-
// Generate design system
|
|
44
|
-
const generated = DesignSystemGenerator.generateDesignSystem(designSystemPath, {
|
|
45
|
-
mode,
|
|
46
|
-
namespace: options.namespace || 'YourProject.Themes'
|
|
47
|
-
});
|
|
48
|
-
|
|
49
|
-
spinner.succeed('Design system parsed!');
|
|
50
|
-
logger.blank();
|
|
51
|
-
|
|
52
|
-
// Display stats
|
|
53
|
-
logger.header('Parsed Design System:');
|
|
54
|
-
logger.info(`Primary Colors: ${chalk.cyan(generated.stats.primaryColors)}`);
|
|
55
|
-
logger.info(`Neutral Colors: ${chalk.cyan(generated.stats.neutralColors)}`);
|
|
56
|
-
logger.info(`Semantic Colors: ${chalk.cyan(generated.stats.semanticColors)}`);
|
|
57
|
-
logger.info(`Font Sizes: ${chalk.cyan(generated.stats.fontSizes)}`);
|
|
58
|
-
logger.info(`Spacing Values: ${chalk.cyan(generated.stats.spacingValues)}`);
|
|
59
|
-
logger.blank();
|
|
60
|
-
|
|
61
|
-
if (options.dryRun) {
|
|
62
|
-
logger.warn('Dry run - files not written');
|
|
63
|
-
logger.blank();
|
|
64
|
-
logger.header('Would generate:');
|
|
65
|
-
logger.dim(' - wwwroot/css/design-system.css');
|
|
66
|
-
if (generated.fluentTheme) logger.dim(' - Themes/FluentDesignTheme.cs');
|
|
67
|
-
if (generated.mudTheme) logger.dim(' - Themes/MudDesignTheme.cs');
|
|
68
|
-
logger.blank();
|
|
69
|
-
return;
|
|
70
|
-
}
|
|
71
|
-
|
|
72
|
-
// Write files
|
|
73
|
-
const writeSpinner = ora('Writing files...').start();
|
|
74
|
-
|
|
75
|
-
// Write CSS
|
|
76
|
-
const cssPath = options.cssOutput || 'wwwroot/css/design-system.css';
|
|
77
|
-
mkdirSync(dirname(cssPath), { recursive: true });
|
|
78
|
-
writeFileSync(cssPath, generated.css);
|
|
79
|
-
writeSpinner.text = `Created ${cssPath}`;
|
|
80
|
-
|
|
81
|
-
// Write Fluent theme
|
|
82
|
-
if (generated.fluentTheme) {
|
|
83
|
-
const fluentPath = options.fluentOutput || 'Themes/FluentDesignTheme.cs';
|
|
84
|
-
mkdirSync(dirname(fluentPath), { recursive: true });
|
|
85
|
-
writeFileSync(fluentPath, generated.fluentTheme);
|
|
86
|
-
writeSpinner.text = `Created ${fluentPath}`;
|
|
87
|
-
}
|
|
88
|
-
|
|
89
|
-
// Write MudBlazor theme
|
|
90
|
-
if (generated.mudTheme) {
|
|
91
|
-
const mudPath = options.mudOutput || 'Themes/MudDesignTheme.cs';
|
|
92
|
-
mkdirSync(dirname(mudPath), { recursive: true });
|
|
93
|
-
writeFileSync(mudPath, generated.mudTheme);
|
|
94
|
-
writeSpinner.text = `Created ${mudPath}`;
|
|
95
|
-
}
|
|
96
|
-
|
|
97
|
-
writeSpinner.succeed('Files generated!');
|
|
98
|
-
logger.blank();
|
|
99
|
-
|
|
100
|
-
// List generated files
|
|
101
|
-
logger.header('Generated Files:');
|
|
102
|
-
logger.success(` ✓ ${cssPath}`);
|
|
103
|
-
if (generated.fluentTheme) logger.success(` ✓ ${options.fluentOutput || 'Themes/FluentDesignTheme.cs'}`);
|
|
104
|
-
if (generated.mudTheme) logger.success(` ✓ ${options.mudOutput || 'Themes/MudDesignTheme.cs'}`);
|
|
105
|
-
logger.blank();
|
|
106
|
-
|
|
107
|
-
// Next steps
|
|
108
|
-
logger.header('Next Steps:');
|
|
109
|
-
logger.dim(' 1. Reference design-system.css in your layout');
|
|
110
|
-
logger.dim(' 2. Register theme in Program.cs');
|
|
111
|
-
logger.dim(' 3. Apply theme to your components');
|
|
112
|
-
logger.blank();
|
|
113
|
-
|
|
114
|
-
} catch (error) {
|
|
115
|
-
spinner.fail('Generation failed');
|
|
116
|
-
logger.error(error.message);
|
|
117
|
-
process.exit(1);
|
|
118
|
-
}
|
|
119
|
-
}
|
|
120
|
-
|
|
121
|
-
// ============================================================================
|
|
122
|
-
// Main Generate Command (router for future subcommands)
|
|
123
|
-
// ============================================================================
|
|
124
|
-
|
|
125
|
-
/**
|
|
126
|
-
* Main generate command router
|
|
127
|
-
* Future: Could add more generation commands here
|
|
128
|
-
*/
|
|
129
|
-
export async function generateCommand(subcommand, args, options) {
|
|
130
|
-
switch (subcommand) {
|
|
131
|
-
case 'design-system':
|
|
132
|
-
await generateDesignSystemCommand(args[0], options);
|
|
133
|
-
break;
|
|
134
|
-
|
|
135
|
-
// Future: Add more generation commands
|
|
136
|
-
// case 'component':
|
|
137
|
-
// await generateComponentCommand(args[0], options);
|
|
138
|
-
// break;
|
|
139
|
-
|
|
140
|
-
default:
|
|
141
|
-
logger.error(`Unknown subcommand: ${subcommand}`);
|
|
142
|
-
logger.blank();
|
|
143
|
-
logger.info('Available subcommands:');
|
|
144
|
-
logger.dim(' design-system Generate CSS + theme files from ui-design-system.md');
|
|
145
|
-
logger.blank();
|
|
146
|
-
logger.dim('Run "morph-spec generate --help" for more information');
|
|
147
|
-
process.exit(1);
|
|
148
|
-
}
|
|
149
|
-
}
|
|
1
|
+
/**
|
|
2
|
+
* MORPH-SPEC Generate Command
|
|
3
|
+
* CLI wrapper for code generation operations
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import { mkdirSync, writeFileSync } from 'fs';
|
|
7
|
+
import { dirname } from 'path';
|
|
8
|
+
import ora from 'ora';
|
|
9
|
+
import chalk from 'chalk';
|
|
10
|
+
import { logger } from '../utils/logger.js';
|
|
11
|
+
import * as DesignSystemGenerator from '../lib/design-system-generator.js';
|
|
12
|
+
|
|
13
|
+
// ============================================================================
|
|
14
|
+
// Design System Subcommand
|
|
15
|
+
// ============================================================================
|
|
16
|
+
|
|
17
|
+
/**
|
|
18
|
+
* Generate design system files (CSS + themes) from ui-design-system.md
|
|
19
|
+
* @param {string} designSystemPath - Path to ui-design-system.md
|
|
20
|
+
* @param {Object} options - CLI options
|
|
21
|
+
*/
|
|
22
|
+
export async function generateDesignSystemCommand(designSystemPath, options) {
|
|
23
|
+
if (!designSystemPath) {
|
|
24
|
+
logger.error('Design system file path required');
|
|
25
|
+
logger.dim(' Usage: morph-spec generate design-system <ui-design-system.md>');
|
|
26
|
+
logger.blank();
|
|
27
|
+
logger.dim(' Example:');
|
|
28
|
+
logger.dim(' morph-spec generate design-system .morph/project/outputs/my-feature/ui-design-system.md');
|
|
29
|
+
process.exit(1);
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
logger.header('MORPH-SPEC Design System Generator');
|
|
33
|
+
logger.blank();
|
|
34
|
+
|
|
35
|
+
const spinner = ora('Parsing design system...').start();
|
|
36
|
+
|
|
37
|
+
try {
|
|
38
|
+
// Determine mode
|
|
39
|
+
let mode = 'both';
|
|
40
|
+
if (options.fluent && !options.mud) mode = 'fluent';
|
|
41
|
+
if (options.mud && !options.fluent) mode = 'mud';
|
|
42
|
+
|
|
43
|
+
// Generate design system
|
|
44
|
+
const generated = DesignSystemGenerator.generateDesignSystem(designSystemPath, {
|
|
45
|
+
mode,
|
|
46
|
+
namespace: options.namespace || 'YourProject.Themes'
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
spinner.succeed('Design system parsed!');
|
|
50
|
+
logger.blank();
|
|
51
|
+
|
|
52
|
+
// Display stats
|
|
53
|
+
logger.header('Parsed Design System:');
|
|
54
|
+
logger.info(`Primary Colors: ${chalk.cyan(generated.stats.primaryColors)}`);
|
|
55
|
+
logger.info(`Neutral Colors: ${chalk.cyan(generated.stats.neutralColors)}`);
|
|
56
|
+
logger.info(`Semantic Colors: ${chalk.cyan(generated.stats.semanticColors)}`);
|
|
57
|
+
logger.info(`Font Sizes: ${chalk.cyan(generated.stats.fontSizes)}`);
|
|
58
|
+
logger.info(`Spacing Values: ${chalk.cyan(generated.stats.spacingValues)}`);
|
|
59
|
+
logger.blank();
|
|
60
|
+
|
|
61
|
+
if (options.dryRun) {
|
|
62
|
+
logger.warn('Dry run - files not written');
|
|
63
|
+
logger.blank();
|
|
64
|
+
logger.header('Would generate:');
|
|
65
|
+
logger.dim(' - wwwroot/css/design-system.css');
|
|
66
|
+
if (generated.fluentTheme) logger.dim(' - Themes/FluentDesignTheme.cs');
|
|
67
|
+
if (generated.mudTheme) logger.dim(' - Themes/MudDesignTheme.cs');
|
|
68
|
+
logger.blank();
|
|
69
|
+
return;
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
// Write files
|
|
73
|
+
const writeSpinner = ora('Writing files...').start();
|
|
74
|
+
|
|
75
|
+
// Write CSS
|
|
76
|
+
const cssPath = options.cssOutput || 'wwwroot/css/design-system.css';
|
|
77
|
+
mkdirSync(dirname(cssPath), { recursive: true });
|
|
78
|
+
writeFileSync(cssPath, generated.css);
|
|
79
|
+
writeSpinner.text = `Created ${cssPath}`;
|
|
80
|
+
|
|
81
|
+
// Write Fluent theme
|
|
82
|
+
if (generated.fluentTheme) {
|
|
83
|
+
const fluentPath = options.fluentOutput || 'Themes/FluentDesignTheme.cs';
|
|
84
|
+
mkdirSync(dirname(fluentPath), { recursive: true });
|
|
85
|
+
writeFileSync(fluentPath, generated.fluentTheme);
|
|
86
|
+
writeSpinner.text = `Created ${fluentPath}`;
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
// Write MudBlazor theme
|
|
90
|
+
if (generated.mudTheme) {
|
|
91
|
+
const mudPath = options.mudOutput || 'Themes/MudDesignTheme.cs';
|
|
92
|
+
mkdirSync(dirname(mudPath), { recursive: true });
|
|
93
|
+
writeFileSync(mudPath, generated.mudTheme);
|
|
94
|
+
writeSpinner.text = `Created ${mudPath}`;
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
writeSpinner.succeed('Files generated!');
|
|
98
|
+
logger.blank();
|
|
99
|
+
|
|
100
|
+
// List generated files
|
|
101
|
+
logger.header('Generated Files:');
|
|
102
|
+
logger.success(` ✓ ${cssPath}`);
|
|
103
|
+
if (generated.fluentTheme) logger.success(` ✓ ${options.fluentOutput || 'Themes/FluentDesignTheme.cs'}`);
|
|
104
|
+
if (generated.mudTheme) logger.success(` ✓ ${options.mudOutput || 'Themes/MudDesignTheme.cs'}`);
|
|
105
|
+
logger.blank();
|
|
106
|
+
|
|
107
|
+
// Next steps
|
|
108
|
+
logger.header('Next Steps:');
|
|
109
|
+
logger.dim(' 1. Reference design-system.css in your layout');
|
|
110
|
+
logger.dim(' 2. Register theme in Program.cs');
|
|
111
|
+
logger.dim(' 3. Apply theme to your components');
|
|
112
|
+
logger.blank();
|
|
113
|
+
|
|
114
|
+
} catch (error) {
|
|
115
|
+
spinner.fail('Generation failed');
|
|
116
|
+
logger.error(error.message);
|
|
117
|
+
process.exit(1);
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
// ============================================================================
|
|
122
|
+
// Main Generate Command (router for future subcommands)
|
|
123
|
+
// ============================================================================
|
|
124
|
+
|
|
125
|
+
/**
|
|
126
|
+
* Main generate command router
|
|
127
|
+
* Future: Could add more generation commands here
|
|
128
|
+
*/
|
|
129
|
+
export async function generateCommand(subcommand, args, options) {
|
|
130
|
+
switch (subcommand) {
|
|
131
|
+
case 'design-system':
|
|
132
|
+
await generateDesignSystemCommand(args[0], options);
|
|
133
|
+
break;
|
|
134
|
+
|
|
135
|
+
// Future: Add more generation commands
|
|
136
|
+
// case 'component':
|
|
137
|
+
// await generateComponentCommand(args[0], options);
|
|
138
|
+
// break;
|
|
139
|
+
|
|
140
|
+
default:
|
|
141
|
+
logger.error(`Unknown subcommand: ${subcommand}`);
|
|
142
|
+
logger.blank();
|
|
143
|
+
logger.info('Available subcommands:');
|
|
144
|
+
logger.dim(' design-system Generate CSS + theme files from ui-design-system.md');
|
|
145
|
+
logger.blank();
|
|
146
|
+
logger.dim('Run "morph-spec generate --help" for more information');
|
|
147
|
+
process.exit(1);
|
|
148
|
+
}
|
|
149
|
+
}
|