@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
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
|
}
|
package/src/utils/logger.js
CHANGED
|
@@ -1,32 +1,32 @@
|
|
|
1
|
-
import chalk from 'chalk';
|
|
2
|
-
|
|
3
|
-
export const logger = {
|
|
4
|
-
info: (msg) => console.log(chalk.cyan(` ${msg}`)),
|
|
5
|
-
success: (msg) => console.log(chalk.green(` ✓ ${msg}`)),
|
|
6
|
-
warn: (msg) => console.log(chalk.yellow(` ⚠ ${msg}`)),
|
|
7
|
-
error: (msg) => console.log(chalk.red(` ✗ ${msg}`)),
|
|
8
|
-
dim: (msg) => console.log(chalk.gray(` ${msg}`)),
|
|
9
|
-
|
|
10
|
-
header: (msg) => {
|
|
11
|
-
console.log('');
|
|
12
|
-
console.log(chalk.cyan.bold(` ${msg}`));
|
|
13
|
-
console.log(chalk.cyan(' ' + '─'.repeat(msg.length)));
|
|
14
|
-
console.log('');
|
|
15
|
-
},
|
|
16
|
-
|
|
17
|
-
step: (num, msg) => console.log(chalk.white(` ${num}. ${msg}`)),
|
|
18
|
-
|
|
19
|
-
blank: () => console.log(''),
|
|
20
|
-
|
|
21
|
-
box: (lines) => {
|
|
22
|
-
const maxLen = Math.max(...lines.map(l => l.length));
|
|
23
|
-
const border = '─'.repeat(maxLen + 2);
|
|
24
|
-
|
|
25
|
-
console.log(chalk.cyan(` ┌${border}┐`));
|
|
26
|
-
lines.forEach(line => {
|
|
27
|
-
const padded = line.padEnd(maxLen);
|
|
28
|
-
console.log(chalk.cyan(` │ ${padded} │`));
|
|
29
|
-
});
|
|
30
|
-
console.log(chalk.cyan(` └${border}┘`));
|
|
31
|
-
}
|
|
32
|
-
};
|
|
1
|
+
import chalk from 'chalk';
|
|
2
|
+
|
|
3
|
+
export const logger = {
|
|
4
|
+
info: (msg) => console.log(chalk.cyan(` ${msg}`)),
|
|
5
|
+
success: (msg) => console.log(chalk.green(` ✓ ${msg}`)),
|
|
6
|
+
warn: (msg) => console.log(chalk.yellow(` ⚠ ${msg}`)),
|
|
7
|
+
error: (msg) => console.log(chalk.red(` ✗ ${msg}`)),
|
|
8
|
+
dim: (msg) => console.log(chalk.gray(` ${msg}`)),
|
|
9
|
+
|
|
10
|
+
header: (msg) => {
|
|
11
|
+
console.log('');
|
|
12
|
+
console.log(chalk.cyan.bold(` ${msg}`));
|
|
13
|
+
console.log(chalk.cyan(' ' + '─'.repeat(msg.length)));
|
|
14
|
+
console.log('');
|
|
15
|
+
},
|
|
16
|
+
|
|
17
|
+
step: (num, msg) => console.log(chalk.white(` ${num}. ${msg}`)),
|
|
18
|
+
|
|
19
|
+
blank: () => console.log(''),
|
|
20
|
+
|
|
21
|
+
box: (lines) => {
|
|
22
|
+
const maxLen = Math.max(...lines.map(l => l.length));
|
|
23
|
+
const border = '─'.repeat(maxLen + 2);
|
|
24
|
+
|
|
25
|
+
console.log(chalk.cyan(` ┌${border}┐`));
|
|
26
|
+
lines.forEach(line => {
|
|
27
|
+
const padded = line.padEnd(maxLen);
|
|
28
|
+
console.log(chalk.cyan(` │ ${padded} │`));
|
|
29
|
+
});
|
|
30
|
+
console.log(chalk.cyan(` └${border}┘`));
|
|
31
|
+
}
|
|
32
|
+
};
|
|
@@ -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
|
+
}
|
|
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
|