@polymorphism-tech/morph-spec 2.2.0 → 2.4.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 +314 -1673
- package/LICENSE +72 -72
- package/README.md +515 -516
- package/bin/detect-agents.js +225 -225
- package/bin/morph-spec.js +358 -173
- package/bin/render-template.js +302 -302
- package/bin/semantic-detect-agents.js +246 -246
- package/bin/task-manager.js +429 -0
- package/bin/validate-agents-skills.js +251 -251
- package/bin/validate-agents.js +69 -69
- package/bin/validate-phase.js +263 -263
- package/bin/validate.js +369 -0
- 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-apply.md +221 -158
- package/content/.claude/commands/morph-archive.md +79 -79
- package/content/.claude/commands/morph-infra.md +209 -209
- package/content/.claude/commands/morph-preflight.md +227 -0
- package/content/.claude/commands/morph-proposal.md +122 -101
- package/content/.claude/commands/morph-status.md +86 -86
- package/content/.claude/commands/morph-troubleshoot.md +122 -0
- package/content/.claude/settings.local.json +15 -15
- package/content/.claude/skills/checklists/code-review.md +226 -0
- package/content/.claude/skills/checklists/morph-checklist.md +117 -0
- package/content/.claude/skills/checklists/simulation-checklist.md +77 -0
- package/content/.claude/skills/infra/bicep-architect.md +126 -419
- package/content/.claude/skills/infra/container-specialist.md +131 -437
- package/content/.claude/skills/infra/devops-engineer.md +119 -405
- package/content/.claude/skills/integrations/asaas-financial.md +130 -333
- package/content/.claude/skills/integrations/azure-identity.md +142 -309
- package/content/.claude/skills/integrations/clerk-auth.md +108 -290
- package/content/.claude/skills/integrations/resend-email.md +119 -0
- package/content/.claude/skills/specialists/ai-system-architect.md +192 -604
- package/content/.claude/skills/specialists/azure-architect.md +142 -142
- package/content/.claude/skills/specialists/code-analyzer.md +235 -0
- package/content/.claude/skills/specialists/dotnet-senior.md +287 -0
- package/content/.claude/skills/specialists/ef-modeler.md +113 -200
- package/content/.claude/skills/specialists/hangfire-orchestrator.md +126 -245
- package/content/.claude/skills/specialists/ms-agent-expert.md +109 -263
- package/content/.claude/skills/specialists/po-pm-advisor.md +197 -197
- package/content/.claude/skills/specialists/standards-architect.md +156 -78
- package/content/.claude/skills/specialists/testing-specialist.md +126 -0
- package/content/.claude/skills/specialists/ui-ux-designer.md +191 -1060
- package/content/.claude/skills/stacks/dotnet-blazor.md +210 -588
- package/content/.claude/skills/stacks/dotnet-nextjs.md +154 -402
- package/content/.claude/skills/workflows/morph-replicate.md +213 -0
- package/content/.claude/{commands/morph-clarify.md → skills/workflows/phase-clarify.md} +5 -58
- package/content/.claude/{commands/morph-design.md → skills/workflows/phase-design.md} +16 -86
- package/content/.claude/{commands/morph-setup.md → skills/workflows/phase-setup.md} +9 -17
- package/content/.claude/skills/workflows/phase-tasks.md +164 -0
- package/content/.claude/{commands/morph-uiux.md → skills/workflows/phase-uiux.md} +15 -88
- package/content/.morph/.morphversion +5 -5
- package/content/.morph/archive/.gitkeep +25 -25
- package/content/.morph/config/agents.json +378 -242
- package/content/.morph/config/config.template.json +89 -108
- package/content/.morph/docs/STORY-DRIVEN-DEVELOPMENT.md +392 -392
- package/content/.morph/docs/workflows/design-impl.md +37 -0
- package/content/.morph/docs/workflows/fast-track.md +29 -0
- package/content/.morph/docs/workflows/full-morph.md +76 -0
- package/content/.morph/docs/workflows/standard.md +44 -0
- package/content/.morph/docs/workflows/ui-refresh.md +39 -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 -0
- package/content/.morph/examples/scheduled-reports/proposal.md +95 -0
- package/content/.morph/examples/scheduled-reports/spec.md +267 -0
- package/content/.morph/examples/state-v3.json +188 -0
- package/content/.morph/features/.gitkeep +25 -25
- package/content/.morph/hooks/README.md +190 -239
- package/content/.morph/hooks/pre-commit-agents.sh +24 -24
- 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/project.md +160 -160
- package/content/.morph/schemas/agent.schema.json +296 -296
- package/content/.morph/schemas/tasks.schema.json +220 -0
- package/content/.morph/specs/.gitkeep +20 -20
- package/content/.morph/standards/agent-framework-blazor-ui.md +359 -0
- package/content/.morph/standards/agent-framework-production.md +410 -0
- package/content/.morph/standards/agent-framework-setup.md +413 -453
- package/content/.morph/standards/agent-framework-workflows.md +349 -0
- package/content/.morph/standards/architecture.md +325 -325
- package/content/.morph/standards/azure.md +605 -379
- package/content/.morph/standards/coding.md +377 -377
- package/content/.morph/standards/dotnet10-migration.md +520 -494
- 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/FluentDesignTheme.cs +149 -149
- package/content/.morph/templates/MudTheme.cs +281 -281
- package/content/.morph/templates/agent.cs +163 -172
- package/content/.morph/templates/clarify-questions.md +159 -0
- package/content/.morph/templates/component.razor +239 -239
- package/content/.morph/templates/contracts/Commands.cs +74 -0
- package/content/.morph/templates/contracts/Entities.cs +25 -0
- package/content/.morph/templates/contracts/Queries.cs +74 -0
- package/content/.morph/templates/contracts/README.md +74 -0
- package/content/.morph/templates/contracts.cs +217 -217
- package/content/.morph/templates/decisions.md +123 -106
- 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/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 -0
- 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/proposal.md +141 -155
- package/content/.morph/templates/recap.md +94 -105
- 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/simulation.md +353 -0
- package/content/.morph/templates/spec.md +149 -148
- package/content/.morph/templates/sprint-status.yaml +68 -68
- package/content/.morph/templates/state.template.json +222 -222
- package/content/.morph/templates/story.md +143 -143
- package/content/.morph/templates/tasks.md +257 -235
- package/content/.morph/templates/test.cs +239 -239
- package/content/.morph/templates/ui-components.md +362 -276
- 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/CLAUDE.md +150 -442
- 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/detectors/structure-detector.js +245 -250
- package/docs/README.md +144 -149
- 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/getting-started.md +301 -302
- package/docs/installation.md +361 -361
- package/docs/templates.md +418 -418
- package/docs/validation-checklist.md +265 -266
- package/package.json +80 -80
- package/scripts/postinstall.js +132 -132
- package/src/commands/advance-phase.js +183 -0
- package/src/commands/analyze-blazor-concurrency.js +193 -0
- package/src/commands/create-story.js +351 -351
- package/src/commands/detect-agents.js +139 -0
- package/src/commands/detect.js +104 -104
- package/src/commands/doctor.js +356 -280
- package/src/commands/generate.js +149 -149
- package/src/commands/init.js +258 -245
- package/src/commands/lint-fluent.js +352 -0
- package/src/commands/rollback-phase.js +185 -0
- package/src/commands/session-summary.js +291 -0
- 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/task.js +78 -0
- package/src/commands/troubleshoot.js +222 -0
- package/src/commands/update.js +192 -159
- package/src/commands/validate-blazor-state.js +210 -0
- package/src/commands/validate-blazor.js +156 -0
- package/src/commands/validate-css.js +84 -0
- package/src/commands/validate-phase.js +221 -0
- package/src/lib/blazor-concurrency-analyzer.js +288 -0
- package/src/lib/blazor-state-validator.js +291 -0
- package/src/lib/blazor-validator.js +374 -0
- package/src/lib/complexity-analyzer.js +441 -292
- package/src/lib/continuous-validator.js +421 -0
- package/src/lib/css-validator.js +352 -0
- package/src/lib/decision-constraint-loader.js +109 -0
- package/src/lib/design-system-generator.js +298 -298
- package/src/lib/learning-system.js +520 -0
- package/src/lib/mockup-generator.js +366 -0
- package/src/lib/recap-generator.js +205 -0
- package/src/lib/state-manager.js +397 -340
- package/src/lib/troubleshoot-grep.js +194 -0
- package/src/lib/troubleshoot-index.js +144 -0
- package/src/lib/ui-detector.js +350 -0
- package/src/lib/validation-runner.js +231 -0
- package/src/lib/validators/architecture-validator.js +387 -0
- package/src/lib/validators/contract-compliance-validator.js +273 -0
- package/src/lib/validators/package-validator.js +360 -0
- package/src/lib/validators/ui-contrast-validator.js +422 -0
- package/src/utils/file-copier.js +179 -139
- package/src/utils/logger.js +32 -32
- package/src/utils/version-checker.js +175 -175
- package/content/.claude/commands/morph-costs.md +0 -206
- package/content/.claude/commands/morph-tasks.md +0 -319
- package/content/.claude/skills/specialists/cost-guardian.md +0 -110
- package/content/.claude/skills/stacks/shopify.md +0 -445
- package/content/.morph/config/azure-pricing.json +0 -70
- package/content/.morph/config/azure-pricing.schema.json +0 -50
- package/content/.morph/hooks/pre-commit-costs.sh +0 -91
- package/docs/api/cost-calculator.js.html +0 -513
- package/docs/api/design-system-generator.js.html +0 -382
- package/docs/api/global.html +0 -5263
- package/docs/api/index.html +0 -96
- package/docs/api/state-manager.js.html +0 -423
- package/src/commands/cost.js +0 -181
- package/src/commands/update-pricing.js +0 -206
- package/src/lib/cost-calculator.js +0 -429
|
@@ -1,175 +1,175 @@
|
|
|
1
|
-
import { readFileSync } from 'fs';
|
|
2
|
-
import { join, dirname } from 'path';
|
|
3
|
-
import { fileURLToPath } from 'url';
|
|
4
|
-
import { pathExists, readJson, writeJson } from './file-copier.js';
|
|
5
|
-
|
|
6
|
-
const __dirname = dirname(fileURLToPath(import.meta.url));
|
|
7
|
-
|
|
8
|
-
/**
|
|
9
|
-
* Get the current CLI version from package.json
|
|
10
|
-
* @returns {string} Version string (e.g., "1.0.4")
|
|
11
|
-
*/
|
|
12
|
-
export function getInstalledCLIVersion() {
|
|
13
|
-
try {
|
|
14
|
-
const pkgPath = join(__dirname, '..', '..', 'package.json');
|
|
15
|
-
const pkg = JSON.parse(readFileSync(pkgPath, 'utf8'));
|
|
16
|
-
return pkg.version;
|
|
17
|
-
} catch (error) {
|
|
18
|
-
return 'unknown';
|
|
19
|
-
}
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
/**
|
|
23
|
-
* Get the latest version from npm registry
|
|
24
|
-
* @param {string} packageName - The package name (default: @polymorphism-tech/morph-spec)
|
|
25
|
-
* @returns {Promise<string|null>} Latest version or null if failed
|
|
26
|
-
*/
|
|
27
|
-
export async function getLatestNPMVersion(packageName = '@polymorphism-tech/morph-spec') {
|
|
28
|
-
try {
|
|
29
|
-
const response = await fetch(`https://registry.npmjs.org/${packageName}/latest`);
|
|
30
|
-
if (!response.ok) return null;
|
|
31
|
-
const data = await response.json();
|
|
32
|
-
return data.version;
|
|
33
|
-
} catch (error) {
|
|
34
|
-
return null;
|
|
35
|
-
}
|
|
36
|
-
}
|
|
37
|
-
|
|
38
|
-
/**
|
|
39
|
-
* Get the MORPH version installed in the project
|
|
40
|
-
* @param {string} projectPath - Path to the project
|
|
41
|
-
* @returns {Promise<object|null>} Version info object or null if not found
|
|
42
|
-
*/
|
|
43
|
-
export async function getProjectMorphVersion(projectPath) {
|
|
44
|
-
try {
|
|
45
|
-
const versionPath = join(projectPath, '.morph', '.morphversion');
|
|
46
|
-
if (!(await pathExists(versionPath))) {
|
|
47
|
-
return null;
|
|
48
|
-
}
|
|
49
|
-
return await readJson(versionPath);
|
|
50
|
-
} catch (error) {
|
|
51
|
-
return null;
|
|
52
|
-
}
|
|
53
|
-
}
|
|
54
|
-
|
|
55
|
-
/**
|
|
56
|
-
* Save the MORPH version to the project
|
|
57
|
-
* @param {string} projectPath - Path to the project
|
|
58
|
-
* @param {string} version - Version to save
|
|
59
|
-
* @returns {Promise<void>}
|
|
60
|
-
*/
|
|
61
|
-
export async function saveProjectMorphVersion(projectPath, version) {
|
|
62
|
-
const versionPath = join(projectPath, '.morph', '.morphversion');
|
|
63
|
-
const versionInfo = {
|
|
64
|
-
morphVersion: version,
|
|
65
|
-
installedAt: new Date().toISOString(),
|
|
66
|
-
cliVersion: version
|
|
67
|
-
};
|
|
68
|
-
await writeJson(versionPath, versionInfo);
|
|
69
|
-
}
|
|
70
|
-
|
|
71
|
-
/**
|
|
72
|
-
* Compare two semver versions
|
|
73
|
-
* @param {string} v1 - First version
|
|
74
|
-
* @param {string} v2 - Second version
|
|
75
|
-
* @returns {number} -1 if v1 < v2, 0 if equal, 1 if v1 > v2
|
|
76
|
-
*/
|
|
77
|
-
export function compareVersions(v1, v2) {
|
|
78
|
-
if (!v1 || !v2) return 0;
|
|
79
|
-
|
|
80
|
-
const parts1 = v1.split('.').map(Number);
|
|
81
|
-
const parts2 = v2.split('.').map(Number);
|
|
82
|
-
|
|
83
|
-
for (let i = 0; i < Math.max(parts1.length, parts2.length); i++) {
|
|
84
|
-
const num1 = parts1[i] || 0;
|
|
85
|
-
const num2 = parts2[i] || 0;
|
|
86
|
-
|
|
87
|
-
if (num1 < num2) return -1;
|
|
88
|
-
if (num1 > num2) return 1;
|
|
89
|
-
}
|
|
90
|
-
|
|
91
|
-
return 0;
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
/**
|
|
95
|
-
* Detect how the CLI was installed
|
|
96
|
-
* @returns {'global'|'npx'|'unknown'}
|
|
97
|
-
*/
|
|
98
|
-
export function detectInstallMethod() {
|
|
99
|
-
// Check if running via npx
|
|
100
|
-
if (process.env.npm_execpath && process.env.npm_execpath.includes('npx')) {
|
|
101
|
-
return 'npx';
|
|
102
|
-
}
|
|
103
|
-
|
|
104
|
-
// Check if installed globally
|
|
105
|
-
const isGlobal = process.env.npm_config_global === 'true' ||
|
|
106
|
-
process.execPath.includes('node_modules');
|
|
107
|
-
|
|
108
|
-
if (isGlobal) {
|
|
109
|
-
return 'global';
|
|
110
|
-
}
|
|
111
|
-
|
|
112
|
-
return 'unknown';
|
|
113
|
-
}
|
|
114
|
-
|
|
115
|
-
/**
|
|
116
|
-
* Get update instructions based on install method
|
|
117
|
-
* @param {string} method - Install method ('global', 'npx', 'unknown')
|
|
118
|
-
* @returns {string[]} Array of instruction strings
|
|
119
|
-
*/
|
|
120
|
-
export function getUpdateInstructions(method) {
|
|
121
|
-
switch (method) {
|
|
122
|
-
case 'global':
|
|
123
|
-
return [
|
|
124
|
-
'npm install -g @polymorphism-tech/morph-spec@latest',
|
|
125
|
-
'morph-spec update'
|
|
126
|
-
];
|
|
127
|
-
case 'npx':
|
|
128
|
-
return [
|
|
129
|
-
'npx @polymorphism-tech/morph-spec@latest update'
|
|
130
|
-
];
|
|
131
|
-
default:
|
|
132
|
-
return [
|
|
133
|
-
'npm install -g @polymorphism-tech/morph-spec@latest',
|
|
134
|
-
'morph-spec update',
|
|
135
|
-
'',
|
|
136
|
-
'Or use npx:',
|
|
137
|
-
'npx @polymorphism-tech/morph-spec@latest update'
|
|
138
|
-
];
|
|
139
|
-
}
|
|
140
|
-
}
|
|
141
|
-
|
|
142
|
-
/**
|
|
143
|
-
* Check if CLI version is outdated
|
|
144
|
-
* @returns {Promise<{isOutdated: boolean, current: string, latest: string|null}>}
|
|
145
|
-
*/
|
|
146
|
-
export async function checkCLIOutdated() {
|
|
147
|
-
const current = getInstalledCLIVersion();
|
|
148
|
-
const latest = await getLatestNPMVersion();
|
|
149
|
-
|
|
150
|
-
if (!latest) {
|
|
151
|
-
return { isOutdated: false, current, latest: null };
|
|
152
|
-
}
|
|
153
|
-
|
|
154
|
-
const isOutdated = compareVersions(current, latest) < 0;
|
|
155
|
-
|
|
156
|
-
return { isOutdated, current, latest };
|
|
157
|
-
}
|
|
158
|
-
|
|
159
|
-
/**
|
|
160
|
-
* Check if project MORPH version is outdated
|
|
161
|
-
* @param {string} projectPath - Path to the project
|
|
162
|
-
* @returns {Promise<{isOutdated: boolean, current: string|null, cliVersion: string}>}
|
|
163
|
-
*/
|
|
164
|
-
export async function checkProjectOutdated(projectPath) {
|
|
165
|
-
const cliVersion = getInstalledCLIVersion();
|
|
166
|
-
const projectVersion = await getProjectMorphVersion(projectPath);
|
|
167
|
-
|
|
168
|
-
if (!projectVersion) {
|
|
169
|
-
return { isOutdated: false, current: null, cliVersion };
|
|
170
|
-
}
|
|
171
|
-
|
|
172
|
-
const isOutdated = compareVersions(projectVersion.morphVersion, cliVersion) < 0;
|
|
173
|
-
|
|
174
|
-
return { isOutdated, current: projectVersion.morphVersion, cliVersion };
|
|
175
|
-
}
|
|
1
|
+
import { readFileSync } from 'fs';
|
|
2
|
+
import { join, dirname } from 'path';
|
|
3
|
+
import { fileURLToPath } from 'url';
|
|
4
|
+
import { pathExists, readJson, writeJson } from './file-copier.js';
|
|
5
|
+
|
|
6
|
+
const __dirname = dirname(fileURLToPath(import.meta.url));
|
|
7
|
+
|
|
8
|
+
/**
|
|
9
|
+
* Get the current CLI version from package.json
|
|
10
|
+
* @returns {string} Version string (e.g., "1.0.4")
|
|
11
|
+
*/
|
|
12
|
+
export function getInstalledCLIVersion() {
|
|
13
|
+
try {
|
|
14
|
+
const pkgPath = join(__dirname, '..', '..', 'package.json');
|
|
15
|
+
const pkg = JSON.parse(readFileSync(pkgPath, 'utf8'));
|
|
16
|
+
return pkg.version;
|
|
17
|
+
} catch (error) {
|
|
18
|
+
return 'unknown';
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
/**
|
|
23
|
+
* Get the latest version from npm registry
|
|
24
|
+
* @param {string} packageName - The package name (default: @polymorphism-tech/morph-spec)
|
|
25
|
+
* @returns {Promise<string|null>} Latest version or null if failed
|
|
26
|
+
*/
|
|
27
|
+
export async function getLatestNPMVersion(packageName = '@polymorphism-tech/morph-spec') {
|
|
28
|
+
try {
|
|
29
|
+
const response = await fetch(`https://registry.npmjs.org/${packageName}/latest`);
|
|
30
|
+
if (!response.ok) return null;
|
|
31
|
+
const data = await response.json();
|
|
32
|
+
return data.version;
|
|
33
|
+
} catch (error) {
|
|
34
|
+
return null;
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
/**
|
|
39
|
+
* Get the MORPH version installed in the project
|
|
40
|
+
* @param {string} projectPath - Path to the project
|
|
41
|
+
* @returns {Promise<object|null>} Version info object or null if not found
|
|
42
|
+
*/
|
|
43
|
+
export async function getProjectMorphVersion(projectPath) {
|
|
44
|
+
try {
|
|
45
|
+
const versionPath = join(projectPath, '.morph', '.morphversion');
|
|
46
|
+
if (!(await pathExists(versionPath))) {
|
|
47
|
+
return null;
|
|
48
|
+
}
|
|
49
|
+
return await readJson(versionPath);
|
|
50
|
+
} catch (error) {
|
|
51
|
+
return null;
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
/**
|
|
56
|
+
* Save the MORPH version to the project
|
|
57
|
+
* @param {string} projectPath - Path to the project
|
|
58
|
+
* @param {string} version - Version to save
|
|
59
|
+
* @returns {Promise<void>}
|
|
60
|
+
*/
|
|
61
|
+
export async function saveProjectMorphVersion(projectPath, version) {
|
|
62
|
+
const versionPath = join(projectPath, '.morph', '.morphversion');
|
|
63
|
+
const versionInfo = {
|
|
64
|
+
morphVersion: version,
|
|
65
|
+
installedAt: new Date().toISOString(),
|
|
66
|
+
cliVersion: version
|
|
67
|
+
};
|
|
68
|
+
await writeJson(versionPath, versionInfo);
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
/**
|
|
72
|
+
* Compare two semver versions
|
|
73
|
+
* @param {string} v1 - First version
|
|
74
|
+
* @param {string} v2 - Second version
|
|
75
|
+
* @returns {number} -1 if v1 < v2, 0 if equal, 1 if v1 > v2
|
|
76
|
+
*/
|
|
77
|
+
export function compareVersions(v1, v2) {
|
|
78
|
+
if (!v1 || !v2) return 0;
|
|
79
|
+
|
|
80
|
+
const parts1 = v1.split('.').map(Number);
|
|
81
|
+
const parts2 = v2.split('.').map(Number);
|
|
82
|
+
|
|
83
|
+
for (let i = 0; i < Math.max(parts1.length, parts2.length); i++) {
|
|
84
|
+
const num1 = parts1[i] || 0;
|
|
85
|
+
const num2 = parts2[i] || 0;
|
|
86
|
+
|
|
87
|
+
if (num1 < num2) return -1;
|
|
88
|
+
if (num1 > num2) return 1;
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
return 0;
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
/**
|
|
95
|
+
* Detect how the CLI was installed
|
|
96
|
+
* @returns {'global'|'npx'|'unknown'}
|
|
97
|
+
*/
|
|
98
|
+
export function detectInstallMethod() {
|
|
99
|
+
// Check if running via npx
|
|
100
|
+
if (process.env.npm_execpath && process.env.npm_execpath.includes('npx')) {
|
|
101
|
+
return 'npx';
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
// Check if installed globally
|
|
105
|
+
const isGlobal = process.env.npm_config_global === 'true' ||
|
|
106
|
+
process.execPath.includes('node_modules');
|
|
107
|
+
|
|
108
|
+
if (isGlobal) {
|
|
109
|
+
return 'global';
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
return 'unknown';
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
/**
|
|
116
|
+
* Get update instructions based on install method
|
|
117
|
+
* @param {string} method - Install method ('global', 'npx', 'unknown')
|
|
118
|
+
* @returns {string[]} Array of instruction strings
|
|
119
|
+
*/
|
|
120
|
+
export function getUpdateInstructions(method) {
|
|
121
|
+
switch (method) {
|
|
122
|
+
case 'global':
|
|
123
|
+
return [
|
|
124
|
+
'npm install -g @polymorphism-tech/morph-spec@latest',
|
|
125
|
+
'morph-spec update'
|
|
126
|
+
];
|
|
127
|
+
case 'npx':
|
|
128
|
+
return [
|
|
129
|
+
'npx @polymorphism-tech/morph-spec@latest update'
|
|
130
|
+
];
|
|
131
|
+
default:
|
|
132
|
+
return [
|
|
133
|
+
'npm install -g @polymorphism-tech/morph-spec@latest',
|
|
134
|
+
'morph-spec update',
|
|
135
|
+
'',
|
|
136
|
+
'Or use npx:',
|
|
137
|
+
'npx @polymorphism-tech/morph-spec@latest update'
|
|
138
|
+
];
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
/**
|
|
143
|
+
* Check if CLI version is outdated
|
|
144
|
+
* @returns {Promise<{isOutdated: boolean, current: string, latest: string|null}>}
|
|
145
|
+
*/
|
|
146
|
+
export async function checkCLIOutdated() {
|
|
147
|
+
const current = getInstalledCLIVersion();
|
|
148
|
+
const latest = await getLatestNPMVersion();
|
|
149
|
+
|
|
150
|
+
if (!latest) {
|
|
151
|
+
return { isOutdated: false, current, latest: null };
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
const isOutdated = compareVersions(current, latest) < 0;
|
|
155
|
+
|
|
156
|
+
return { isOutdated, current, latest };
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
/**
|
|
160
|
+
* Check if project MORPH version is outdated
|
|
161
|
+
* @param {string} projectPath - Path to the project
|
|
162
|
+
* @returns {Promise<{isOutdated: boolean, current: string|null, cliVersion: string}>}
|
|
163
|
+
*/
|
|
164
|
+
export async function checkProjectOutdated(projectPath) {
|
|
165
|
+
const cliVersion = getInstalledCLIVersion();
|
|
166
|
+
const projectVersion = await getProjectMorphVersion(projectPath);
|
|
167
|
+
|
|
168
|
+
if (!projectVersion) {
|
|
169
|
+
return { isOutdated: false, current: null, cliVersion };
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
const isOutdated = compareVersions(projectVersion.morphVersion, cliVersion) < 0;
|
|
173
|
+
|
|
174
|
+
return { isOutdated, current: projectVersion.morphVersion, cliVersion };
|
|
175
|
+
}
|
|
@@ -1,206 +0,0 @@
|
|
|
1
|
-
# /morph-costs - Cost Estimation and Reporting
|
|
2
|
-
|
|
3
|
-
Estima e reporta custos de recursos Azure.
|
|
4
|
-
|
|
5
|
-
## Uso
|
|
6
|
-
|
|
7
|
-
```
|
|
8
|
-
/morph-costs [action]
|
|
9
|
-
```
|
|
10
|
-
|
|
11
|
-
### Actions
|
|
12
|
-
|
|
13
|
-
| Action | Descrição |
|
|
14
|
-
|--------|-----------|
|
|
15
|
-
| `estimate` | Estima custos da feature atual |
|
|
16
|
-
| `report` | Gera relatório de custos do projeto |
|
|
17
|
-
| `compare` | Compara custos entre ambientes |
|
|
18
|
-
|
|
19
|
-
---
|
|
20
|
-
|
|
21
|
-
## Workflow
|
|
22
|
-
|
|
23
|
-
### 1. ESTIMATE - Estimar Custos
|
|
24
|
-
|
|
25
|
-
Quando o usuário solicitar `/morph-costs estimate`:
|
|
26
|
-
|
|
27
|
-
1. Ler spec.md da feature atual para identificar recursos
|
|
28
|
-
|
|
29
|
-
2. Para cada recurso Azure, calcular custo mensal:
|
|
30
|
-
|
|
31
|
-
| Recurso | SKU | Custo/Mês |
|
|
32
|
-
|---------|-----|-----------|
|
|
33
|
-
| Container Apps | Consumption (scale-to-zero) | ~$0 |
|
|
34
|
-
| Azure SQL | Free (32GB) | $0 |
|
|
35
|
-
| Azure SQL | Basic (5 DTU) | ~$5 |
|
|
36
|
-
| Storage | Standard LRS (1GB) | ~$0.02 |
|
|
37
|
-
| Key Vault | Standard | ~$0.03/10k ops |
|
|
38
|
-
| App Insights | Free (5GB/mês) | $0 |
|
|
39
|
-
| Service Bus | Basic | ~$0.05 |
|
|
40
|
-
|
|
41
|
-
3. Apresentar estimativa:
|
|
42
|
-
```markdown
|
|
43
|
-
## Estimativa de Custos - {Feature}
|
|
44
|
-
|
|
45
|
-
| Recurso | SKU | Custo/Mês |
|
|
46
|
-
|---------|-----|-----------|
|
|
47
|
-
| Container Apps | Consumption | ~$0 |
|
|
48
|
-
| SQL Database | Free | $0 |
|
|
49
|
-
| Storage | LRS 1GB | $0.02 |
|
|
50
|
-
| **TOTAL** | | **$0.02/mês** |
|
|
51
|
-
|
|
52
|
-
### Aprovação
|
|
53
|
-
✅ Free tier - nenhuma aprovação necessária
|
|
54
|
-
```
|
|
55
|
-
|
|
56
|
-
4. Verificar contra limites do Cost Guardian:
|
|
57
|
-
- Free tier apenas: ✅ Aprovado automaticamente
|
|
58
|
-
- Até $10/mês: ⚠️ Requer confirmação
|
|
59
|
-
- Acima de $10: ❌ Requer ADR
|
|
60
|
-
|
|
61
|
-
---
|
|
62
|
-
|
|
63
|
-
### 2. REPORT - Relatório de Custos
|
|
64
|
-
|
|
65
|
-
Quando o usuário solicitar `/morph-costs report`:
|
|
66
|
-
|
|
67
|
-
1. Analisar todos os recursos definidos em `infra/`
|
|
68
|
-
|
|
69
|
-
2. Gerar relatório consolidado:
|
|
70
|
-
|
|
71
|
-
```markdown
|
|
72
|
-
## Relatório de Custos - {Projeto}
|
|
73
|
-
|
|
74
|
-
### Por Ambiente
|
|
75
|
-
|
|
76
|
-
| Ambiente | Recursos | Custo/Mês |
|
|
77
|
-
|----------|----------|-----------|
|
|
78
|
-
| dev | 6 | ~$0 |
|
|
79
|
-
| prod | 6 | ~$15 |
|
|
80
|
-
| **Total** | | **~$15/mês** |
|
|
81
|
-
|
|
82
|
-
### Por Tipo de Recurso
|
|
83
|
-
|
|
84
|
-
| Recurso | Dev | Prod |
|
|
85
|
-
|---------|-----|------|
|
|
86
|
-
| Container Apps | $0 | $5 |
|
|
87
|
-
| SQL Database | $0 | $5 |
|
|
88
|
-
| Storage | $0.02 | $2 |
|
|
89
|
-
| Key Vault | $0 | $0.50 |
|
|
90
|
-
| App Insights | $0 | $2.50 |
|
|
91
|
-
|
|
92
|
-
### Recomendações
|
|
93
|
-
1. ✅ Dev usa free tier corretamente
|
|
94
|
-
2. ⚠️ Considerar Reserved Capacity para prod (-30%)
|
|
95
|
-
```
|
|
96
|
-
|
|
97
|
-
3. Salvar relatório em `.morph/outputs/costs-report.md`
|
|
98
|
-
|
|
99
|
-
---
|
|
100
|
-
|
|
101
|
-
### 3. COMPARE - Comparar Ambientes
|
|
102
|
-
|
|
103
|
-
Quando o usuário solicitar `/morph-costs compare`:
|
|
104
|
-
|
|
105
|
-
1. Analisar `parameters.dev.json` e `parameters.prod.json`
|
|
106
|
-
|
|
107
|
-
2. Gerar comparativo:
|
|
108
|
-
|
|
109
|
-
```markdown
|
|
110
|
-
## Comparativo de Custos
|
|
111
|
-
|
|
112
|
-
| Recurso | Dev | Prod | Diferença |
|
|
113
|
-
|---------|-----|------|-----------|
|
|
114
|
-
| SQL SKU | Free | Basic | +$5 |
|
|
115
|
-
| Storage SKU | LRS | GRS | +$1 |
|
|
116
|
-
| Min Replicas | 0 | 1 | +$3 |
|
|
117
|
-
| **Total** | $0 | $15 | +$15 |
|
|
118
|
-
|
|
119
|
-
### Por que prod custa mais?
|
|
120
|
-
- SQL Basic: melhor performance, SLA 99.99%
|
|
121
|
-
- Storage GRS: redundância geográfica
|
|
122
|
-
- Min replicas 1: sem cold start
|
|
123
|
-
```
|
|
124
|
-
|
|
125
|
-
---
|
|
126
|
-
|
|
127
|
-
## Pricing Reference
|
|
128
|
-
|
|
129
|
-
### Free Tier Resources
|
|
130
|
-
|
|
131
|
-
| Recurso | Limite Grátis |
|
|
132
|
-
|---------|---------------|
|
|
133
|
-
| Azure SQL | 32GB, 100k vCore seconds |
|
|
134
|
-
| Container Apps | Scale-to-zero, 180k vCPU-sec |
|
|
135
|
-
| Storage | 5GB LRS |
|
|
136
|
-
| App Insights | 5GB/mês |
|
|
137
|
-
| Cosmos DB | 1000 RU/s, 25GB |
|
|
138
|
-
| Functions | 1M execuções |
|
|
139
|
-
|
|
140
|
-
### Costs by Region (Brazil South)
|
|
141
|
-
|
|
142
|
-
| Recurso | SKU | Preço |
|
|
143
|
-
|---------|-----|-------|
|
|
144
|
-
| Container Apps | Consumption | $0.000012/vCPU-sec |
|
|
145
|
-
| SQL | Basic | $4.99/mês |
|
|
146
|
-
| SQL | S0 | $14.72/mês |
|
|
147
|
-
| Storage | LRS | $0.0184/GB |
|
|
148
|
-
| Storage | GRS | $0.046/GB |
|
|
149
|
-
|
|
150
|
-
---
|
|
151
|
-
|
|
152
|
-
## Cost Guardian Rules
|
|
153
|
-
|
|
154
|
-
### Aprovação Automática (Free Tier)
|
|
155
|
-
- Container Apps com scale-to-zero
|
|
156
|
-
- SQL Free tier
|
|
157
|
-
- Storage Standard LRS até 5GB
|
|
158
|
-
- App Insights free tier
|
|
159
|
-
|
|
160
|
-
### Requer Confirmação (até $10/mês)
|
|
161
|
-
- SQL Basic
|
|
162
|
-
- Storage adicional
|
|
163
|
-
- Service Bus Basic
|
|
164
|
-
|
|
165
|
-
### Requer ADR (acima de $10/mês)
|
|
166
|
-
- Recursos Premium
|
|
167
|
-
- Multiple replicas
|
|
168
|
-
- Reserved capacity
|
|
169
|
-
- Qualquer recurso não essencial
|
|
170
|
-
|
|
171
|
-
---
|
|
172
|
-
|
|
173
|
-
## Exemplo de Uso
|
|
174
|
-
|
|
175
|
-
```
|
|
176
|
-
Usuário: /morph-costs estimate
|
|
177
|
-
Claude: Analisando feature "scheduled-reports"...
|
|
178
|
-
|
|
179
|
-
## Estimativa de Custos
|
|
180
|
-
|
|
181
|
-
| Recurso | SKU | Custo/Mês |
|
|
182
|
-
|---------|-----|-----------|
|
|
183
|
-
| Container Apps | Consumption | ~$0 |
|
|
184
|
-
| SQL Database | Free | $0 |
|
|
185
|
-
| Storage | LRS 1GB | $0.02 |
|
|
186
|
-
| Hangfire | In-process | $0 |
|
|
187
|
-
|
|
188
|
-
**TOTAL: ~$0.02/mês**
|
|
189
|
-
|
|
190
|
-
✅ Dentro do free tier - aprovado automaticamente!
|
|
191
|
-
|
|
192
|
-
Usuário: /morph-costs report
|
|
193
|
-
Claude: ## Relatório de Custos - MyProject
|
|
194
|
-
|
|
195
|
-
| Ambiente | Custo/Mês |
|
|
196
|
-
|----------|-----------|
|
|
197
|
-
| dev | $0.02 |
|
|
198
|
-
| prod | $12.50 |
|
|
199
|
-
|
|
200
|
-
⚠️ Prod excede $10/mês.
|
|
201
|
-
Deseja criar um ADR para justificar? (sim/não)
|
|
202
|
-
```
|
|
203
|
-
|
|
204
|
-
---
|
|
205
|
-
|
|
206
|
-
*MORPH-SPEC by Polymorphism Tech*
|