mindforge-cc 1.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/.agent/CLAUDE.md +462 -0
- package/.agent/forge/help.md +7 -0
- package/.agent/forge/init-project.md +32 -0
- package/.agent/forge/plan-phase.md +30 -0
- package/.agent/mindforge/approve.md +18 -0
- package/.agent/mindforge/audit.md +30 -0
- package/.agent/mindforge/benchmark.md +33 -0
- package/.agent/mindforge/complete-milestone.md +18 -0
- package/.agent/mindforge/debug.md +126 -0
- package/.agent/mindforge/discuss-phase.md +138 -0
- package/.agent/mindforge/execute-phase.md +165 -0
- package/.agent/mindforge/health.md +21 -0
- package/.agent/mindforge/help.md +23 -0
- package/.agent/mindforge/init-org.md +131 -0
- package/.agent/mindforge/init-project.md +155 -0
- package/.agent/mindforge/install-skill.md +15 -0
- package/.agent/mindforge/map-codebase.md +298 -0
- package/.agent/mindforge/metrics.md +22 -0
- package/.agent/mindforge/migrate.md +40 -0
- package/.agent/mindforge/milestone.md +12 -0
- package/.agent/mindforge/next.md +105 -0
- package/.agent/mindforge/plan-phase.md +125 -0
- package/.agent/mindforge/plugins.md +40 -0
- package/.agent/mindforge/pr-review.md +41 -0
- package/.agent/mindforge/profile-team.md +23 -0
- package/.agent/mindforge/publish-skill.md +19 -0
- package/.agent/mindforge/quick.md +135 -0
- package/.agent/mindforge/release.md +10 -0
- package/.agent/mindforge/retrospective.md +26 -0
- package/.agent/mindforge/review.md +157 -0
- package/.agent/mindforge/security-scan.md +233 -0
- package/.agent/mindforge/ship.md +100 -0
- package/.agent/mindforge/skills.md +141 -0
- package/.agent/mindforge/status.md +104 -0
- package/.agent/mindforge/sync-confluence.md +11 -0
- package/.agent/mindforge/sync-jira.md +12 -0
- package/.agent/mindforge/tokens.md +8 -0
- package/.agent/mindforge/update.md +42 -0
- package/.agent/mindforge/verify-phase.md +62 -0
- package/.agent/mindforge/workspace.md +29 -0
- package/.claude/CLAUDE.md +462 -0
- package/.claude/commands/forge/help.md +7 -0
- package/.claude/commands/forge/init-project.md +32 -0
- package/.claude/commands/forge/plan-phase.md +30 -0
- package/.claude/commands/mindforge/approve.md +18 -0
- package/.claude/commands/mindforge/audit.md +30 -0
- package/.claude/commands/mindforge/benchmark.md +33 -0
- package/.claude/commands/mindforge/complete-milestone.md +18 -0
- package/.claude/commands/mindforge/debug.md +126 -0
- package/.claude/commands/mindforge/discuss-phase.md +138 -0
- package/.claude/commands/mindforge/execute-phase.md +165 -0
- package/.claude/commands/mindforge/health.md +21 -0
- package/.claude/commands/mindforge/help.md +23 -0
- package/.claude/commands/mindforge/init-org.md +131 -0
- package/.claude/commands/mindforge/init-project.md +155 -0
- package/.claude/commands/mindforge/install-skill.md +15 -0
- package/.claude/commands/mindforge/map-codebase.md +298 -0
- package/.claude/commands/mindforge/metrics.md +22 -0
- package/.claude/commands/mindforge/migrate.md +40 -0
- package/.claude/commands/mindforge/milestone.md +12 -0
- package/.claude/commands/mindforge/next.md +105 -0
- package/.claude/commands/mindforge/plan-phase.md +125 -0
- package/.claude/commands/mindforge/plugins.md +40 -0
- package/.claude/commands/mindforge/pr-review.md +41 -0
- package/.claude/commands/mindforge/profile-team.md +23 -0
- package/.claude/commands/mindforge/publish-skill.md +19 -0
- package/.claude/commands/mindforge/quick.md +135 -0
- package/.claude/commands/mindforge/release.md +10 -0
- package/.claude/commands/mindforge/retrospective.md +26 -0
- package/.claude/commands/mindforge/review.md +157 -0
- package/.claude/commands/mindforge/security-scan.md +233 -0
- package/.claude/commands/mindforge/ship.md +100 -0
- package/.claude/commands/mindforge/skills.md +141 -0
- package/.claude/commands/mindforge/status.md +104 -0
- package/.claude/commands/mindforge/sync-confluence.md +11 -0
- package/.claude/commands/mindforge/sync-jira.md +12 -0
- package/.claude/commands/mindforge/tokens.md +8 -0
- package/.claude/commands/mindforge/update.md +42 -0
- package/.claude/commands/mindforge/verify-phase.md +62 -0
- package/.claude/commands/mindforge/workspace.md +29 -0
- package/.forge/org/CONVENTIONS.md +0 -0
- package/.forge/org/ORG.md +0 -0
- package/.forge/org/SECURITY.md +0 -0
- package/.forge/org/TOOLS.md +0 -0
- package/.forge/personas/analyst.md +0 -0
- package/.forge/personas/architect.md +0 -0
- package/.forge/personas/debug-specialist.md +0 -0
- package/.forge/personas/developer.md +26 -0
- package/.forge/personas/qa-engineer.md +0 -0
- package/.forge/personas/release-manager.md +0 -0
- package/.forge/personas/security-reviewer.md +33 -0
- package/.forge/personas/tech-writer.md +0 -0
- package/.forge/skills/api-design/SKILL.md +0 -0
- package/.forge/skills/code-quality/SKILL.md +0 -0
- package/.forge/skills/documentation/SKILL.md +0 -0
- package/.forge/skills/security-review/SKILL.md +23 -0
- package/.forge/skills/testing-standards/SKILL.md +27 -0
- package/.github/workflows/mindforge-ci.yml +224 -0
- package/.gitlab-ci-mindforge.yml +18 -0
- package/.mindforge/MINDFORGE-SCHEMA.json +165 -0
- package/.mindforge/audit/AUDIT-SCHEMA.md +451 -0
- package/.mindforge/ci/ci-config-schema.md +21 -0
- package/.mindforge/ci/ci-mode.md +179 -0
- package/.mindforge/ci/github-actions-adapter.md +224 -0
- package/.mindforge/ci/gitlab-ci-adapter.md +31 -0
- package/.mindforge/ci/jenkins-adapter.md +44 -0
- package/.mindforge/distribution/registry-client.md +166 -0
- package/.mindforge/distribution/registry-schema.md +96 -0
- package/.mindforge/distribution/skill-publisher.md +44 -0
- package/.mindforge/distribution/skill-validator.md +74 -0
- package/.mindforge/engine/compaction-protocol.md +182 -0
- package/.mindforge/engine/context-injector.md +128 -0
- package/.mindforge/engine/dependency-parser.md +113 -0
- package/.mindforge/engine/skills/conflict-resolver.md +69 -0
- package/.mindforge/engine/skills/loader.md +184 -0
- package/.mindforge/engine/skills/registry.md +98 -0
- package/.mindforge/engine/skills/versioning.md +75 -0
- package/.mindforge/engine/verification-pipeline.md +111 -0
- package/.mindforge/engine/wave-executor.md +235 -0
- package/.mindforge/governance/GOVERNANCE-CONFIG.md +17 -0
- package/.mindforge/governance/approval-workflow.md +37 -0
- package/.mindforge/governance/change-classifier.md +63 -0
- package/.mindforge/governance/compliance-gates.md +31 -0
- package/.mindforge/integrations/confluence.md +27 -0
- package/.mindforge/integrations/connection-manager.md +163 -0
- package/.mindforge/integrations/github.md +25 -0
- package/.mindforge/integrations/gitlab.md +13 -0
- package/.mindforge/integrations/jira.md +102 -0
- package/.mindforge/integrations/slack.md +41 -0
- package/.mindforge/intelligence/antipattern-detector.md +75 -0
- package/.mindforge/intelligence/difficulty-scorer.md +55 -0
- package/.mindforge/intelligence/health-engine.md +208 -0
- package/.mindforge/intelligence/skill-gap-analyser.md +40 -0
- package/.mindforge/intelligence/smart-compaction.md +71 -0
- package/.mindforge/metrics/METRICS-SCHEMA.md +42 -0
- package/.mindforge/metrics/quality-tracker.md +32 -0
- package/.mindforge/monorepo/cross-package-planner.md +114 -0
- package/.mindforge/monorepo/dependency-graph-builder.md +32 -0
- package/.mindforge/monorepo/workspace-detector.md +129 -0
- package/.mindforge/org/CONVENTIONS.md +62 -0
- package/.mindforge/org/ORG.md +51 -0
- package/.mindforge/org/SECURITY.md +50 -0
- package/.mindforge/org/TOOLS.md +53 -0
- package/.mindforge/org/integrations/INTEGRATIONS-CONFIG.md +58 -0
- package/.mindforge/org/skills/MANIFEST.md +38 -0
- package/.mindforge/personas/analyst.md +52 -0
- package/.mindforge/personas/architect.md +75 -0
- package/.mindforge/personas/debug-specialist.md +52 -0
- package/.mindforge/personas/developer.md +85 -0
- package/.mindforge/personas/overrides/README.md +85 -0
- package/.mindforge/personas/qa-engineer.md +61 -0
- package/.mindforge/personas/release-manager.md +76 -0
- package/.mindforge/personas/security-reviewer.md +91 -0
- package/.mindforge/personas/tech-writer.md +51 -0
- package/.mindforge/plugins/PLUGINS-MANIFEST.md +23 -0
- package/.mindforge/plugins/plugin-loader.md +93 -0
- package/.mindforge/plugins/plugin-registry.md +44 -0
- package/.mindforge/plugins/plugin-schema.md +68 -0
- package/.mindforge/pr-review/ai-reviewer.md +266 -0
- package/.mindforge/pr-review/finding-formatter.md +46 -0
- package/.mindforge/pr-review/review-prompt-templates.md +44 -0
- package/.mindforge/production/compatibility-layer.md +39 -0
- package/.mindforge/production/migration-engine.md +52 -0
- package/.mindforge/production/production-checklist.md +165 -0
- package/.mindforge/production/token-optimiser.md +68 -0
- package/.mindforge/skills/accessibility/SKILL.md +106 -0
- package/.mindforge/skills/api-design/SKILL.md +98 -0
- package/.mindforge/skills/code-quality/SKILL.md +88 -0
- package/.mindforge/skills/data-privacy/SKILL.md +126 -0
- package/.mindforge/skills/database-patterns/SKILL.md +192 -0
- package/.mindforge/skills/documentation/SKILL.md +91 -0
- package/.mindforge/skills/incident-response/SKILL.md +180 -0
- package/.mindforge/skills/performance/SKILL.md +120 -0
- package/.mindforge/skills/security-review/SKILL.md +83 -0
- package/.mindforge/skills/testing-standards/SKILL.md +97 -0
- package/.mindforge/team/TEAM-PROFILE.md +42 -0
- package/.mindforge/team/multi-handoff.md +23 -0
- package/.mindforge/team/profiles/README.md +13 -0
- package/.mindforge/team/session-merger.md +18 -0
- package/.planning/ARCHITECTURE.md +0 -0
- package/.planning/AUDIT.jsonl +0 -0
- package/.planning/HANDOFF.json +28 -0
- package/.planning/PROJECT.md +33 -0
- package/.planning/RELEASE-CHECKLIST.md +68 -0
- package/.planning/REQUIREMENTS.md +0 -0
- package/.planning/ROADMAP.md +0 -0
- package/.planning/STATE.md +31 -0
- package/.planning/approvals/.gitkeep +1 -0
- package/.planning/archive/.gitkeep +1 -0
- package/.planning/audit-archive/.gitkeep +1 -0
- package/.planning/decisions/.gitkeep +0 -0
- package/.planning/decisions/ADR-001-handoff-tracking.md +41 -0
- package/.planning/decisions/ADR-002-markdown-commands.md +46 -0
- package/.planning/decisions/ADR-003-skills-trigger-model.md +37 -0
- package/.planning/decisions/ADR-004-wave-parallelism-model.md +45 -0
- package/.planning/decisions/ADR-005-append-only-audit-log.md +51 -0
- package/.planning/decisions/ADR-006-tiered-skills-system.md +22 -0
- package/.planning/decisions/ADR-007-trigger-keyword-model.md +22 -0
- package/.planning/decisions/ADR-008-just-in-time-skill-loading.md +29 -0
- package/.planning/decisions/ADR-009-enterprise-integration-retry-policy.md +8 -0
- package/.planning/decisions/ADR-010-governance-tier-escalation.md +8 -0
- package/.planning/decisions/ADR-011-multi-developer-handoff-contract.md +8 -0
- package/.planning/decisions/ADR-012-intelligence-feedback-loops.md +19 -0
- package/.planning/decisions/ADR-013-mindforge-md-constitution.md +16 -0
- package/.planning/decisions/ADR-014-metrics-as-signals-not-evaluation.md +15 -0
- package/.planning/decisions/ADR-015-npm-based-skill-registry.md +26 -0
- package/.planning/decisions/ADR-016-ci-exit-code-0-on-timeout.md +27 -0
- package/.planning/decisions/ADR-017-sdk-localhost-only.md +28 -0
- package/.planning/decisions/ADR-018-installer-self-install-detection.md +15 -0
- package/.planning/decisions/ADR-019-self-update-scope-preservation.md +14 -0
- package/.planning/decisions/ADR-020-v1.0.0-stable-interface-contract.md +23 -0
- package/.planning/jira-sync.json +9 -0
- package/.planning/milestones/.gitkeep +1 -0
- package/.planning/phases/day1/REVIEW-DAY1.md +50 -0
- package/.planning/phases/day1/SECURITY-REVIEW-DAY1.md +15 -0
- package/.planning/phases/day2/REVIEW-DAY2.md +521 -0
- package/.planning/phases/day3/REVIEW-DAY3.md +234 -0
- package/.planning/slack-threads.json +6 -0
- package/CHANGELOG.md +175 -0
- package/LICENSE +21 -0
- package/MINDFORGE.md +76 -0
- package/README.md +182 -0
- package/RELEASENOTES.md +41 -0
- package/SECURITY.md +4 -0
- package/bin/install.js +120 -0
- package/bin/installer-core.js +292 -0
- package/bin/migrations/0.1.0-to-0.5.0.js +37 -0
- package/bin/migrations/0.5.0-to-0.6.0.js +17 -0
- package/bin/migrations/0.6.0-to-1.0.0.js +100 -0
- package/bin/migrations/migrate.js +151 -0
- package/bin/migrations/schema-versions.js +64 -0
- package/bin/updater/changelog-fetcher.js +62 -0
- package/bin/updater/self-update.js +169 -0
- package/bin/updater/version-comparator.js +68 -0
- package/bin/validate-config.js +92 -0
- package/bin/wizard/config-generator.js +112 -0
- package/bin/wizard/environment-detector.js +76 -0
- package/bin/wizard/setup-wizard.js +237 -0
- package/docs/Context/Master-Context.md +701 -0
- package/docs/architecture/README.md +35 -0
- package/docs/architecture/decision-records-index.md +26 -0
- package/docs/ci-cd-integration.md +30 -0
- package/docs/ci-quickstart.md +78 -0
- package/docs/commands-reference.md +11 -0
- package/docs/contributing/CONTRIBUTING.md +38 -0
- package/docs/contributing/plugin-authoring.md +50 -0
- package/docs/contributing/skill-authoring.md +41 -0
- package/docs/enterprise-setup.md +25 -0
- package/docs/faq.md +38 -0
- package/docs/getting-started.md +36 -0
- package/docs/governance-guide.md +23 -0
- package/docs/mindforge-md-reference.md +53 -0
- package/docs/monorepo-guide.md +26 -0
- package/docs/persona-customisation.md +56 -0
- package/docs/quick-verify.md +33 -0
- package/docs/reference/audit-events.md +53 -0
- package/docs/reference/commands.md +82 -0
- package/docs/reference/config-reference.md +64 -0
- package/docs/reference/sdk-api.md +48 -0
- package/docs/reference/skills-api.md +57 -0
- package/docs/release-checklist-guide.md +37 -0
- package/docs/requirements.md +29 -0
- package/docs/sdk-reference.md +27 -0
- package/docs/security/SECURITY.md +42 -0
- package/docs/security/penetration-test-results.md +31 -0
- package/docs/security/threat-model.md +142 -0
- package/docs/skills-authoring-guide.md +119 -0
- package/docs/skills-publishing-guide.md +21 -0
- package/docs/team-setup-guide.md +21 -0
- package/docs/troubleshooting.md +119 -0
- package/docs/tutorial.md +195 -0
- package/docs/upgrade.md +44 -0
- package/docs/user-guide.md +131 -0
- package/docs/usp-features.md +214 -0
- package/eslint.config.mjs +31 -0
- package/examples/starter-project/.planning/AUDIT.jsonl +1 -0
- package/examples/starter-project/.planning/HANDOFF.json +23 -0
- package/examples/starter-project/.planning/PROJECT.md +27 -0
- package/examples/starter-project/.planning/STATE.md +10 -0
- package/examples/starter-project/MINDFORGE.md +40 -0
- package/examples/starter-project/README.md +14 -0
- package/implementation-roadmap/day-1-imp/DAY1-HARDEN.md +823 -0
- package/implementation-roadmap/day-1-imp/DAY1-IMPLEMENT.md +2459 -0
- package/implementation-roadmap/day-1-imp/DAY1-REVIEW.md +288 -0
- package/implementation-roadmap/day-2-imp/DAY2-HARDEN.md +954 -0
- package/implementation-roadmap/day-2-imp/DAY2-IMPLEMENT.md +2347 -0
- package/implementation-roadmap/day-2-imp/DAY2-REVIEW.md +422 -0
- package/implementation-roadmap/day-3-imp/DAY3-HARDEN.md +870 -0
- package/implementation-roadmap/day-3-imp/DAY3-IMPLEMENT.md +2798 -0
- package/implementation-roadmap/day-3-imp/DAY3-REVIEW.md +484 -0
- package/implementation-roadmap/day-4-imp/DAY4-HARDEN.md +1087 -0
- package/implementation-roadmap/day-4-imp/DAY4-IMPLEMENT.md +2874 -0
- package/implementation-roadmap/day-4-imp/DAY4-REVIEW.md +386 -0
- package/implementation-roadmap/day-5-imp/DAY5-HARDEN.md +1078 -0
- package/implementation-roadmap/day-5-imp/DAY5-IMPLEMENT.md +3151 -0
- package/implementation-roadmap/day-5-imp/DAY5-REVIEW.md +345 -0
- package/implementation-roadmap/day-6-imp/DAY6-COMPLETE.md +3919 -0
- package/implementation-roadmap/day-7-imp-prod/DAY7-PRODUCTION-FINAL.md +4513 -0
- package/package.json +31 -0
- package/sdk/README.md +69 -0
- package/sdk/eslint.config.mjs +34 -0
- package/sdk/package-lock.json +1507 -0
- package/sdk/package.json +30 -0
- package/sdk/src/client.ts +133 -0
- package/sdk/src/commands.ts +63 -0
- package/sdk/src/events.ts +166 -0
- package/sdk/src/index.ts +22 -0
- package/sdk/src/types.ts +87 -0
- package/sdk/tsconfig.json +13 -0
- package/tests/audit.test.js +206 -0
- package/tests/ci-mode.test.js +162 -0
- package/tests/compaction.test.js +161 -0
- package/tests/distribution.test.js +205 -0
- package/tests/e2e.test.js +618 -0
- package/tests/governance.test.js +130 -0
- package/tests/install.test.js +209 -0
- package/tests/integrations.test.js +128 -0
- package/tests/intelligence.test.js +117 -0
- package/tests/metrics.test.js +96 -0
- package/tests/migration.test.js +309 -0
- package/tests/production.test.js +416 -0
- package/tests/sdk.test.js +200 -0
- package/tests/skills-platform.test.js +403 -0
- package/tests/wave-engine.test.js +338 -0
|
@@ -0,0 +1,112 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
const fs = require('fs');
|
|
4
|
+
const path = require('path');
|
|
5
|
+
|
|
6
|
+
function ensureDir(p) {
|
|
7
|
+
if (!fs.existsSync(p)) fs.mkdirSync(p, { recursive: true });
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
function replaceIfPlaceholder(content, placeholder, value, keyName) {
|
|
11
|
+
if (content.includes(placeholder)) {
|
|
12
|
+
return {
|
|
13
|
+
content: content.replace(placeholder, value),
|
|
14
|
+
changed: true,
|
|
15
|
+
msg: ` ✅ ${keyName} configured`,
|
|
16
|
+
};
|
|
17
|
+
}
|
|
18
|
+
if (!content.includes(value)) {
|
|
19
|
+
return {
|
|
20
|
+
content,
|
|
21
|
+
changed: false,
|
|
22
|
+
msg: ` ⏭️ ${keyName} already configured — not overwriting`,
|
|
23
|
+
};
|
|
24
|
+
}
|
|
25
|
+
return { content, changed: false, msg: null };
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
async function writeIntegrationsConfig(config) {
|
|
29
|
+
const configPath = '.mindforge/org/integrations/INTEGRATIONS-CONFIG.md';
|
|
30
|
+
ensureDir(path.dirname(configPath));
|
|
31
|
+
|
|
32
|
+
if (!fs.existsSync(configPath)) {
|
|
33
|
+
console.log(' ⏭️ INTEGRATIONS-CONFIG.md not found — skipping integration write');
|
|
34
|
+
return;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
let content = fs.readFileSync(configPath, 'utf8');
|
|
38
|
+
const logs = [];
|
|
39
|
+
|
|
40
|
+
if (config.jira) {
|
|
41
|
+
let out = replaceIfPlaceholder(
|
|
42
|
+
content,
|
|
43
|
+
'JIRA_BASE_URL=https://your-org.atlassian.net',
|
|
44
|
+
`JIRA_BASE_URL=${config.jira.baseUrl}`,
|
|
45
|
+
'JIRA_BASE_URL'
|
|
46
|
+
);
|
|
47
|
+
content = out.content; if (out.msg) logs.push(out.msg);
|
|
48
|
+
|
|
49
|
+
out = replaceIfPlaceholder(
|
|
50
|
+
content,
|
|
51
|
+
'JIRA_PROJECT_KEY=ENG',
|
|
52
|
+
`JIRA_PROJECT_KEY=${config.jira.projectKey}`,
|
|
53
|
+
'JIRA_PROJECT_KEY'
|
|
54
|
+
);
|
|
55
|
+
content = out.content; if (out.msg) logs.push(out.msg);
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
if (config.slack) {
|
|
59
|
+
const out = replaceIfPlaceholder(
|
|
60
|
+
content,
|
|
61
|
+
'SLACK_CHANNEL_ID=C01234ABCDE',
|
|
62
|
+
`SLACK_CHANNEL_ID=${config.slack.channelId}`,
|
|
63
|
+
'SLACK_CHANNEL_ID'
|
|
64
|
+
);
|
|
65
|
+
content = out.content; if (out.msg) logs.push(out.msg);
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
if (config.github) {
|
|
69
|
+
let out = replaceIfPlaceholder(
|
|
70
|
+
content,
|
|
71
|
+
'GITHUB_REPO=your-org/your-repo',
|
|
72
|
+
`GITHUB_REPO=${config.github.repo}`,
|
|
73
|
+
'GITHUB_REPO'
|
|
74
|
+
);
|
|
75
|
+
content = out.content; if (out.msg) logs.push(out.msg);
|
|
76
|
+
|
|
77
|
+
out = replaceIfPlaceholder(
|
|
78
|
+
content,
|
|
79
|
+
'GITHUB_DEFAULT_REVIEWERS=senior-engineer-1,senior-engineer-2',
|
|
80
|
+
`GITHUB_DEFAULT_REVIEWERS=${config.github.reviewers}`,
|
|
81
|
+
'GITHUB_DEFAULT_REVIEWERS'
|
|
82
|
+
);
|
|
83
|
+
content = out.content; if (out.msg) logs.push(out.msg);
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
fs.writeFileSync(configPath, content);
|
|
87
|
+
logs.forEach((l) => console.log(l));
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
async function writeGovernanceConfig(team) {
|
|
91
|
+
const configPath = '.mindforge/governance/GOVERNANCE-CONFIG.md';
|
|
92
|
+
ensureDir(path.dirname(configPath));
|
|
93
|
+
if (!fs.existsSync(configPath)) return;
|
|
94
|
+
|
|
95
|
+
let content = fs.readFileSync(configPath, 'utf8');
|
|
96
|
+
if (team.tier2Approvers && content.includes('TIER2_APPROVERS=senior-engineer-1,senior-engineer-2')) {
|
|
97
|
+
content = content.replace(
|
|
98
|
+
'TIER2_APPROVERS=senior-engineer-1,senior-engineer-2',
|
|
99
|
+
`TIER2_APPROVERS=${team.tier2Approvers}`
|
|
100
|
+
);
|
|
101
|
+
}
|
|
102
|
+
if (team.tier3Approvers && content.includes('TIER3_APPROVERS=security-officer,compliance-officer,cto')) {
|
|
103
|
+
content = content.replace(
|
|
104
|
+
'TIER3_APPROVERS=security-officer,compliance-officer,cto',
|
|
105
|
+
`TIER3_APPROVERS=${team.tier3Approvers}`
|
|
106
|
+
);
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
fs.writeFileSync(configPath, content);
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
module.exports = { writeIntegrationsConfig, writeGovernanceConfig };
|
|
@@ -0,0 +1,76 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
const fs = require('fs');
|
|
4
|
+
const path = require('path');
|
|
5
|
+
const { execSync } = require('child_process');
|
|
6
|
+
|
|
7
|
+
function safeExec(cmd) {
|
|
8
|
+
try {
|
|
9
|
+
return execSync(cmd, { encoding: 'utf8', stdio: ['ignore', 'pipe', 'ignore'] }).trim();
|
|
10
|
+
} catch {
|
|
11
|
+
return null;
|
|
12
|
+
}
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
function readJsonSafe(p) {
|
|
16
|
+
try {
|
|
17
|
+
return JSON.parse(fs.readFileSync(p, 'utf8'));
|
|
18
|
+
} catch {
|
|
19
|
+
return null;
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
async function detect() {
|
|
24
|
+
const home = process.env.HOME || process.env.USERPROFILE || '';
|
|
25
|
+
const cwd = process.cwd();
|
|
26
|
+
|
|
27
|
+
const runtimes = [];
|
|
28
|
+
if (fs.existsSync(path.join(home, '.claude')) || fs.existsSync(path.join(cwd, '.claude'))) runtimes.push('claude');
|
|
29
|
+
if (fs.existsSync(path.join(home, '.gemini', 'antigravity')) || fs.existsSync(path.join(cwd, '.agent'))) runtimes.push('antigravity');
|
|
30
|
+
|
|
31
|
+
let projectType = 'unknown';
|
|
32
|
+
const pkgPath = path.join(cwd, 'package.json');
|
|
33
|
+
if (fs.existsSync(pkgPath)) {
|
|
34
|
+
const pkg = readJsonSafe(pkgPath) || {};
|
|
35
|
+
const deps = { ...(pkg.dependencies || {}), ...(pkg.devDependencies || {}) };
|
|
36
|
+
if (deps.next || deps.react) projectType = 'Next.js / React';
|
|
37
|
+
else if (deps.fastify || deps.express) projectType = 'Node.js API';
|
|
38
|
+
else projectType = 'Node.js';
|
|
39
|
+
} else if (fs.existsSync(path.join(cwd, 'pyproject.toml'))) {
|
|
40
|
+
projectType = 'Python';
|
|
41
|
+
} else if (fs.existsSync(path.join(cwd, 'Cargo.toml'))) {
|
|
42
|
+
projectType = 'Rust';
|
|
43
|
+
} else if (fs.existsSync(path.join(cwd, 'go.mod'))) {
|
|
44
|
+
projectType = 'Go';
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
let packageManager = 'npm';
|
|
48
|
+
if (fs.existsSync(path.join(cwd, 'pnpm-lock.yaml'))) packageManager = 'pnpm';
|
|
49
|
+
else if (fs.existsSync(path.join(cwd, 'yarn.lock'))) packageManager = 'yarn';
|
|
50
|
+
else if (fs.existsSync(path.join(cwd, 'bun.lockb'))) packageManager = 'bun';
|
|
51
|
+
|
|
52
|
+
let existingInstall = false;
|
|
53
|
+
let existingVersion = null;
|
|
54
|
+
const claudePath = path.join(cwd, '.claude', 'CLAUDE.md');
|
|
55
|
+
if (fs.existsSync(claudePath)) {
|
|
56
|
+
const content = fs.readFileSync(claudePath, 'utf8');
|
|
57
|
+
if (content.includes('MindForge')) {
|
|
58
|
+
existingInstall = true;
|
|
59
|
+
const m = content.match(/MindForge v(\d+\.\d+\.\d+)/);
|
|
60
|
+
existingVersion = m ? m[1] : 'unknown';
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
return {
|
|
65
|
+
runtimes,
|
|
66
|
+
hasGit: !!safeExec('git rev-parse --is-inside-work-tree'),
|
|
67
|
+
projectType,
|
|
68
|
+
packageManager,
|
|
69
|
+
existingInstall,
|
|
70
|
+
existingVersion,
|
|
71
|
+
nodeVersion: process.versions.node,
|
|
72
|
+
platform: process.platform,
|
|
73
|
+
};
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
module.exports = { detect };
|
|
@@ -0,0 +1,237 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
|
|
3
|
+
'use strict';
|
|
4
|
+
|
|
5
|
+
const readline = require('readline');
|
|
6
|
+
const detector = require('./environment-detector');
|
|
7
|
+
const generator = require('./config-generator');
|
|
8
|
+
|
|
9
|
+
const VERSION = require('../../package.json').version;
|
|
10
|
+
const ARGS = process.argv.slice(2);
|
|
11
|
+
const IS_INTERACTIVE =
|
|
12
|
+
!ARGS.some((a) => ['--claude', '--antigravity', '--all', '--help'].includes(a)) &&
|
|
13
|
+
process.stdin.isTTY !== false;
|
|
14
|
+
|
|
15
|
+
const TTY = process.stdout.isTTY;
|
|
16
|
+
const c = {
|
|
17
|
+
bold: (s) => (TTY ? `\x1b[1m${s}\x1b[0m` : s),
|
|
18
|
+
cyan: (s) => (TTY ? `\x1b[36m${s}\x1b[0m` : s),
|
|
19
|
+
green: (s) => (TTY ? `\x1b[32m${s}\x1b[0m` : s),
|
|
20
|
+
yellow: (s) => (TTY ? `\x1b[33m${s}\x1b[0m` : s),
|
|
21
|
+
red: (s) => (TTY ? `\x1b[31m${s}\x1b[0m` : s),
|
|
22
|
+
dim: (s) => (TTY ? `\x1b[2m${s}\x1b[0m` : s),
|
|
23
|
+
};
|
|
24
|
+
|
|
25
|
+
function createReadline() {
|
|
26
|
+
return readline.createInterface({
|
|
27
|
+
input: process.stdin,
|
|
28
|
+
output: process.stdout,
|
|
29
|
+
terminal: TTY,
|
|
30
|
+
});
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
function ask(rl, q, def) {
|
|
34
|
+
return new Promise((resolve) => {
|
|
35
|
+
const prompt = def ? `${q} ${c.dim(`[${def}]`)}: ` : `${q}: `;
|
|
36
|
+
rl.question(prompt, (answer) => resolve(answer.trim() || def || ''));
|
|
37
|
+
});
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
function askChoice(rl, q, choices, defaultIdx = 0) {
|
|
41
|
+
console.log(`\n${q}`);
|
|
42
|
+
choices.forEach((choice, i) => console.log(` ${i + 1}) ${choice}`));
|
|
43
|
+
return new Promise((resolve) => {
|
|
44
|
+
rl.question(`Choice [${defaultIdx + 1}]: `, (answer) => {
|
|
45
|
+
const idx = Number.parseInt(answer.trim(), 10) - 1;
|
|
46
|
+
resolve(choices[Number.isInteger(idx) && idx >= 0 && idx < choices.length ? idx : defaultIdx]);
|
|
47
|
+
});
|
|
48
|
+
});
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
function askMultiChoice(rl, q, choices) {
|
|
52
|
+
console.log(`\n${q}`);
|
|
53
|
+
choices.forEach((choice, i) => console.log(` ${i + 1}) ${choice}`));
|
|
54
|
+
return new Promise((resolve) => {
|
|
55
|
+
rl.question('Select (comma-separated): ', (answer) => {
|
|
56
|
+
const selected = answer
|
|
57
|
+
.split(',')
|
|
58
|
+
.map((s) => Number.parseInt(s.trim(), 10) - 1)
|
|
59
|
+
.filter((i) => Number.isInteger(i) && i >= 0 && i < choices.length)
|
|
60
|
+
.map((i) => choices[i]);
|
|
61
|
+
resolve(selected.length ? selected : [choices[0]]);
|
|
62
|
+
});
|
|
63
|
+
});
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
function printBanner() {
|
|
67
|
+
console.log('');
|
|
68
|
+
console.log(c.bold(c.cyan(' MindForge Setup Wizard')));
|
|
69
|
+
console.log(c.dim(` v${VERSION}`));
|
|
70
|
+
console.log('');
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
async function detectEnvironment() {
|
|
74
|
+
console.log(c.bold(' Detecting environment...\n'));
|
|
75
|
+
const env = await detector.detect();
|
|
76
|
+
const rows = [
|
|
77
|
+
['Runtime(s)', env.runtimes.join(', ') || 'none'],
|
|
78
|
+
['Git repo', env.hasGit ? 'yes' : 'no'],
|
|
79
|
+
['Node.js', env.nodeVersion],
|
|
80
|
+
['Project type', env.projectType],
|
|
81
|
+
['Package manager', env.packageManager],
|
|
82
|
+
['Existing MindForge', env.existingInstall ? `yes (${env.existingVersion})` : 'no'],
|
|
83
|
+
];
|
|
84
|
+
rows.forEach(([k, v]) => console.log(` ${c.dim(k.padEnd(20))} ${v}`));
|
|
85
|
+
console.log('');
|
|
86
|
+
return env;
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
async function selectRuntime(rl, detected) {
|
|
90
|
+
const choices = ['Claude Code', 'Antigravity', 'Both'];
|
|
91
|
+
const pick = await askChoice(rl, 'Install runtime:', choices, detected.includes('claude') ? 0 : 2);
|
|
92
|
+
if (pick === 'Claude Code') return ['claude'];
|
|
93
|
+
if (pick === 'Antigravity') return ['antigravity'];
|
|
94
|
+
return ['claude', 'antigravity'];
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
async function selectScope(rl) {
|
|
98
|
+
const pick = await askChoice(rl, 'Install scope:', ['Global', 'Local'], 0);
|
|
99
|
+
return pick.toLowerCase();
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
async function configureFeatures(rl) {
|
|
103
|
+
const features = await askMultiChoice(rl, 'Configure optional integrations:', [
|
|
104
|
+
'Jira integration',
|
|
105
|
+
'Slack notifications',
|
|
106
|
+
'GitHub integration',
|
|
107
|
+
'None',
|
|
108
|
+
]);
|
|
109
|
+
|
|
110
|
+
const config = {};
|
|
111
|
+
const credGuidance = [];
|
|
112
|
+
|
|
113
|
+
if (features.includes('Jira integration')) {
|
|
114
|
+
config.jira = {
|
|
115
|
+
baseUrl: await ask(rl, ' Jira base URL', 'https://your-org.atlassian.net'),
|
|
116
|
+
projectKey: await ask(rl, ' Jira project key', 'ENG'),
|
|
117
|
+
};
|
|
118
|
+
credGuidance.push({
|
|
119
|
+
service: 'Jira / Confluence (Atlassian)',
|
|
120
|
+
envVar: 'JIRA_API_TOKEN',
|
|
121
|
+
url: 'https://id.atlassian.com/manage-profile/security/api-tokens',
|
|
122
|
+
instruction: 'Create token and export JIRA_API_TOKEN',
|
|
123
|
+
});
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
if (features.includes('Slack notifications')) {
|
|
127
|
+
config.slack = { channelId: await ask(rl, ' Slack channel ID', 'C01234ABCDE') };
|
|
128
|
+
credGuidance.push({
|
|
129
|
+
service: 'Slack',
|
|
130
|
+
envVar: 'SLACK_BOT_TOKEN',
|
|
131
|
+
url: 'https://api.slack.com/apps',
|
|
132
|
+
instruction: 'Create bot token and export SLACK_BOT_TOKEN',
|
|
133
|
+
});
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
if (features.includes('GitHub integration')) {
|
|
137
|
+
config.github = {
|
|
138
|
+
repo: await ask(rl, ' GitHub repo (owner/name)', ''),
|
|
139
|
+
reviewers: await ask(rl, ' Default reviewers', ''),
|
|
140
|
+
};
|
|
141
|
+
credGuidance.push({
|
|
142
|
+
service: 'GitHub',
|
|
143
|
+
envVar: 'GITHUB_TOKEN',
|
|
144
|
+
url: 'https://github.com/settings/tokens',
|
|
145
|
+
instruction: 'Create PAT and export GITHUB_TOKEN',
|
|
146
|
+
});
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
return { config, credGuidance };
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
async function install(runtimes, scope) {
|
|
153
|
+
const installer = require('../install');
|
|
154
|
+
if (!installer || typeof installer.install !== 'function') return;
|
|
155
|
+
for (const runtime of runtimes) {
|
|
156
|
+
await installer.install(runtime, scope);
|
|
157
|
+
}
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
function printNextSteps(runtimes, scope, credGuidance = []) {
|
|
161
|
+
console.log('');
|
|
162
|
+
console.log(c.bold(c.green(' Setup complete')));
|
|
163
|
+
console.log(` Runtime: ${runtimes.join(', ')}`);
|
|
164
|
+
console.log(` Scope: ${scope}`);
|
|
165
|
+
console.log(' Next: run /mindforge:health');
|
|
166
|
+
|
|
167
|
+
if (credGuidance.length > 0) {
|
|
168
|
+
console.log(c.bold('\n Configure credentials:\n'));
|
|
169
|
+
credGuidance.forEach((g) => {
|
|
170
|
+
console.log(` ${c.cyan(g.service)}`);
|
|
171
|
+
console.log(` ${g.instruction} (${g.envVar})`);
|
|
172
|
+
console.log(c.dim(` Docs: ${g.url}\n`));
|
|
173
|
+
});
|
|
174
|
+
}
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
function handleWizardError(err) {
|
|
178
|
+
console.error(c.red('\n Setup encountered an issue:\n'));
|
|
179
|
+
|
|
180
|
+
const COMMON_ERRORS = {
|
|
181
|
+
ENOENT: {
|
|
182
|
+
message: 'A required file was not found.',
|
|
183
|
+
action: 'Run from project root and verify install artifacts exist.',
|
|
184
|
+
},
|
|
185
|
+
EACCES: {
|
|
186
|
+
message: 'Permission denied writing to target path.',
|
|
187
|
+
action: 'Adjust permissions or run with appropriate access.',
|
|
188
|
+
},
|
|
189
|
+
MODULE_NOT_FOUND: {
|
|
190
|
+
message: 'Installer module is missing.',
|
|
191
|
+
action: 'Reinstall package and retry.',
|
|
192
|
+
},
|
|
193
|
+
ERR_INVALID_ARG_TYPE: {
|
|
194
|
+
message: 'Unexpected argument type while running setup.',
|
|
195
|
+
action: 'Run non-interactive mode as fallback.',
|
|
196
|
+
},
|
|
197
|
+
};
|
|
198
|
+
|
|
199
|
+
const known = COMMON_ERRORS[err.code] || COMMON_ERRORS.ERR_INVALID_ARG_TYPE;
|
|
200
|
+
console.error(` Problem: ${known.message}`);
|
|
201
|
+
console.error(` Action: ${known.action}`);
|
|
202
|
+
console.error(c.dim('\n Fallback examples:'));
|
|
203
|
+
console.error(c.dim(' npx mindforge-cc --claude --local'));
|
|
204
|
+
console.error(c.dim(' npx mindforge-cc --antigravity --local'));
|
|
205
|
+
console.error(c.dim(' npx mindforge-cc --all --global\n'));
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
async function main() {
|
|
209
|
+
printBanner();
|
|
210
|
+
|
|
211
|
+
if (!IS_INTERACTIVE) {
|
|
212
|
+
const installer = require('../install');
|
|
213
|
+
if (installer && typeof installer.runCli === 'function') {
|
|
214
|
+
installer.runCli();
|
|
215
|
+
}
|
|
216
|
+
return;
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
const rl = createReadline();
|
|
220
|
+
try {
|
|
221
|
+
const env = await detectEnvironment();
|
|
222
|
+
const runtimes = await selectRuntime(rl, env.runtimes);
|
|
223
|
+
const scope = await selectScope(rl);
|
|
224
|
+
const { config, credGuidance } = await configureFeatures(rl);
|
|
225
|
+
rl.close();
|
|
226
|
+
|
|
227
|
+
await install(runtimes, scope);
|
|
228
|
+
await generator.writeIntegrationsConfig(config);
|
|
229
|
+
printNextSteps(runtimes, scope, credGuidance);
|
|
230
|
+
} catch (err) {
|
|
231
|
+
rl.close();
|
|
232
|
+
handleWizardError(err);
|
|
233
|
+
process.exit(1);
|
|
234
|
+
}
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
main();
|