@samahlstrom/forge-cli 0.1.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.
Files changed (100) hide show
  1. package/README.md +175 -0
  2. package/bin/forge.js +2 -0
  3. package/dist/addons/index.d.ts +25 -0
  4. package/dist/addons/index.js +139 -0
  5. package/dist/addons/index.js.map +1 -0
  6. package/dist/commands/add.d.ts +1 -0
  7. package/dist/commands/add.js +61 -0
  8. package/dist/commands/add.js.map +1 -0
  9. package/dist/commands/doctor.d.ts +1 -0
  10. package/dist/commands/doctor.js +177 -0
  11. package/dist/commands/doctor.js.map +1 -0
  12. package/dist/commands/ingest.d.ts +24 -0
  13. package/dist/commands/ingest.js +316 -0
  14. package/dist/commands/ingest.js.map +1 -0
  15. package/dist/commands/init.d.ts +8 -0
  16. package/dist/commands/init.js +557 -0
  17. package/dist/commands/init.js.map +1 -0
  18. package/dist/commands/remove.d.ts +1 -0
  19. package/dist/commands/remove.js +42 -0
  20. package/dist/commands/remove.js.map +1 -0
  21. package/dist/commands/status.d.ts +1 -0
  22. package/dist/commands/status.js +48 -0
  23. package/dist/commands/status.js.map +1 -0
  24. package/dist/commands/upgrade.d.ts +5 -0
  25. package/dist/commands/upgrade.js +190 -0
  26. package/dist/commands/upgrade.js.map +1 -0
  27. package/dist/detect/features.d.ts +10 -0
  28. package/dist/detect/features.js +33 -0
  29. package/dist/detect/features.js.map +1 -0
  30. package/dist/detect/go.d.ts +3 -0
  31. package/dist/detect/go.js +38 -0
  32. package/dist/detect/go.js.map +1 -0
  33. package/dist/detect/index.d.ts +25 -0
  34. package/dist/detect/index.js +32 -0
  35. package/dist/detect/index.js.map +1 -0
  36. package/dist/detect/node.d.ts +3 -0
  37. package/dist/detect/node.js +99 -0
  38. package/dist/detect/node.js.map +1 -0
  39. package/dist/detect/python.d.ts +3 -0
  40. package/dist/detect/python.js +86 -0
  41. package/dist/detect/python.js.map +1 -0
  42. package/dist/index.d.ts +1 -0
  43. package/dist/index.js +51 -0
  44. package/dist/index.js.map +1 -0
  45. package/dist/render/engine.d.ts +8 -0
  46. package/dist/render/engine.js +71 -0
  47. package/dist/render/engine.js.map +1 -0
  48. package/dist/render/merge.d.ts +5 -0
  49. package/dist/render/merge.js +33 -0
  50. package/dist/render/merge.js.map +1 -0
  51. package/dist/utils/fs.d.ts +8 -0
  52. package/dist/utils/fs.js +42 -0
  53. package/dist/utils/fs.js.map +1 -0
  54. package/dist/utils/git.d.ts +3 -0
  55. package/dist/utils/git.js +31 -0
  56. package/dist/utils/git.js.map +1 -0
  57. package/dist/utils/hash.d.ts +8 -0
  58. package/dist/utils/hash.js +22 -0
  59. package/dist/utils/hash.js.map +1 -0
  60. package/dist/utils/yaml.d.ts +3 -0
  61. package/dist/utils/yaml.js +12 -0
  62. package/dist/utils/yaml.js.map +1 -0
  63. package/package.json +53 -0
  64. package/templates/addons/beads-dolt-backend/files/dolt-setup.sh +267 -0
  65. package/templates/addons/beads-dolt-backend/manifest.yaml +13 -0
  66. package/templates/addons/browser-testing/files/browser-smoke.sh +196 -0
  67. package/templates/addons/browser-testing/files/visual-qa.md +103 -0
  68. package/templates/addons/browser-testing/manifest.yaml +20 -0
  69. package/templates/addons/compliance-hipaa/files/hipaa-checks.sh +184 -0
  70. package/templates/addons/compliance-hipaa/files/hipaa-context.md +91 -0
  71. package/templates/addons/compliance-hipaa/manifest.yaml +15 -0
  72. package/templates/addons/compliance-soc2/files/soc2-checks.sh +232 -0
  73. package/templates/addons/compliance-soc2/files/soc2-context.md +147 -0
  74. package/templates/addons/compliance-soc2/manifest.yaml +15 -0
  75. package/templates/core/CLAUDE.md.hbs +70 -0
  76. package/templates/core/agents/architect.md.hbs +68 -0
  77. package/templates/core/agents/backend.md.hbs +27 -0
  78. package/templates/core/agents/frontend.md.hbs +25 -0
  79. package/templates/core/agents/quality.md.hbs +40 -0
  80. package/templates/core/agents/security.md.hbs +53 -0
  81. package/templates/core/context/project.md.hbs +60 -0
  82. package/templates/core/forge.yaml.hbs +69 -0
  83. package/templates/core/hooks/post-edit.sh.hbs +8 -0
  84. package/templates/core/hooks/pre-edit.sh.hbs +41 -0
  85. package/templates/core/hooks/session-start.sh.hbs +34 -0
  86. package/templates/core/pipeline/classify.sh.hbs +159 -0
  87. package/templates/core/pipeline/decompose.md.hbs +100 -0
  88. package/templates/core/pipeline/deliver.sh.hbs +171 -0
  89. package/templates/core/pipeline/execute.md.hbs +138 -0
  90. package/templates/core/pipeline/intake.sh.hbs +152 -0
  91. package/templates/core/pipeline/orchestrator.sh.hbs +361 -0
  92. package/templates/core/pipeline/verify.sh.hbs +160 -0
  93. package/templates/core/settings.json.hbs +55 -0
  94. package/templates/core/skill-creator.md.hbs +151 -0
  95. package/templates/core/skill-deliver.md.hbs +46 -0
  96. package/templates/core/skill-ingest.md.hbs +245 -0
  97. package/templates/presets/go/stack.md.hbs +133 -0
  98. package/templates/presets/python-fastapi/stack.md.hbs +101 -0
  99. package/templates/presets/react-next-ts/stack.md.hbs +77 -0
  100. package/templates/presets/sveltekit-ts/stack.md.hbs +116 -0
@@ -0,0 +1,48 @@
1
+ import * as p from '@clack/prompts';
2
+ import chalk from 'chalk';
3
+ import { join } from 'node:path';
4
+ import { exists } from '../utils/fs.js';
5
+ import { readYaml } from '../utils/yaml.js';
6
+ import { readHashes } from '../utils/hash.js';
7
+ export async function status() {
8
+ const cwd = process.cwd();
9
+ p.intro(chalk.bold('forge status'));
10
+ if (!(await exists(join(cwd, 'forge.yaml')))) {
11
+ p.log.error('No forge.yaml found. Run `forge init` first.');
12
+ process.exit(1);
13
+ }
14
+ const config = await readYaml(join(cwd, 'forge.yaml'));
15
+ const project = config.project;
16
+ const agents = config.agents ?? [];
17
+ const addons = config.addons ?? [];
18
+ const hashes = await readHashes(cwd);
19
+ // Get task counts from bd
20
+ let openCount = 0;
21
+ let closedCount = 0;
22
+ let bdAvailable = false;
23
+ try {
24
+ const { execSync } = await import('node:child_process');
25
+ const openResult = execSync('bd list --status open --json 2>/dev/null', { stdio: ['pipe', 'pipe', 'pipe'] }).toString().trim();
26
+ openCount = openResult && openResult !== '[]' ? JSON.parse(openResult).length : 0;
27
+ const closedResult = execSync('bd list --status closed --json 2>/dev/null', { stdio: ['pipe', 'pipe', 'pipe'] }).toString().trim();
28
+ closedCount = closedResult && closedResult !== '[]' ? JSON.parse(closedResult).length : 0;
29
+ bdAvailable = true;
30
+ }
31
+ catch {
32
+ // bd not available
33
+ }
34
+ const lines = [
35
+ `Project: ${chalk.cyan(project?.name ?? 'unknown')}`,
36
+ `Preset: ${chalk.cyan(project?.preset ?? 'unknown')}`,
37
+ `Version: ${chalk.cyan(hashes.version)}`,
38
+ `Agents: ${chalk.cyan(agents.join(', '))}`,
39
+ `Addons: ${addons.length > 0 ? chalk.cyan(addons.join(', ')) : chalk.dim('none')}`,
40
+ ``,
41
+ `Tracking: ${bdAvailable ? chalk.green('bd (beads)') : chalk.yellow('bd not installed')}`,
42
+ `Open tasks: ${openCount > 0 ? chalk.yellow(String(openCount)) : chalk.green('0')}`,
43
+ `Closed: ${chalk.dim(String(closedCount))}`,
44
+ ];
45
+ p.note(lines.join('\n'), 'Harness Status');
46
+ p.outro('');
47
+ }
48
+ //# sourceMappingURL=status.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"status.js","sourceRoot":"","sources":["../../src/commands/status.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ interface UpgradeOptions {
2
+ force?: boolean;
3
+ }
4
+ export declare function upgrade(options: UpgradeOptions): Promise<void>;
5
+ export {};
@@ -0,0 +1,190 @@
1
+ import * as p from '@clack/prompts';
2
+ import chalk from 'chalk';
3
+ import { join } from 'node:path';
4
+ import { exists, readText, writeText } from '../utils/fs.js';
5
+ import { readHashes, writeHashes, hashContent, hashFile } from '../utils/hash.js';
6
+ import { readYaml } from '../utils/yaml.js';
7
+ import { render } from '../render/engine.js';
8
+ import { mergeForgeYaml } from '../render/merge.js';
9
+ import { resolveTemplatePath } from '../utils/fs.js';
10
+ import { detect } from '../detect/index.js';
11
+ // Files that are ALWAYS overwritten (tool-owned)
12
+ const ALWAYS_OVERWRITE = ['.forge/context/stack.md'];
13
+ // Files that are NEVER touched (user-owned)
14
+ const NEVER_TOUCH = ['.forge/context/project.md'];
15
+ export async function upgrade(options) {
16
+ const cwd = process.cwd();
17
+ p.intro(chalk.bold('forge upgrade'));
18
+ if (!(await exists(join(cwd, 'forge.yaml')))) {
19
+ p.log.error('No forge.yaml found. Run `forge init` first.');
20
+ process.exit(1);
21
+ }
22
+ const hashes = await readHashes(cwd);
23
+ const currentVersion = hashes.version;
24
+ const newVersion = '0.1.0'; // TODO: read from package.json
25
+ p.log.info(`Current: v${currentVersion} → Available: v${newVersion}`);
26
+ const config = await readYaml(join(cwd, 'forge.yaml'));
27
+ const project = config.project;
28
+ const preset = project?.preset ?? 'sveltekit-ts';
29
+ // Rebuild template context
30
+ const detected = await detect(cwd);
31
+ const ctx = buildUpgradeContext(config, detected, preset);
32
+ let updated = 0;
33
+ let skipped = 0;
34
+ let prompted = 0;
35
+ const newHashes = { version: newVersion, files: {} };
36
+ // Get all template files that should exist
37
+ for (const [relativePath, installedHash] of Object.entries(hashes.files)) {
38
+ // Skip addon files — they're managed separately
39
+ if (relativePath.startsWith('.forge/addons/'))
40
+ continue;
41
+ // Never touch user-owned files
42
+ if (NEVER_TOUCH.includes(relativePath)) {
43
+ p.log.message(` ${chalk.dim('⊘')} ${relativePath} ${chalk.dim('— user-owned, skipped')}`);
44
+ skipped++;
45
+ // Keep the existing hash
46
+ newHashes.files[relativePath] = installedHash;
47
+ continue;
48
+ }
49
+ const filePath = join(cwd, relativePath);
50
+ if (!(await exists(filePath))) {
51
+ // File was deleted by user — skip
52
+ skipped++;
53
+ continue;
54
+ }
55
+ // Try to find and render the new template
56
+ const templatePath = findTemplatePath(relativePath, preset);
57
+ if (!templatePath) {
58
+ newHashes.files[relativePath] = installedHash;
59
+ continue;
60
+ }
61
+ let newContent;
62
+ try {
63
+ const template = await readText(resolveTemplatePath(templatePath));
64
+ newContent = render(template, ctx);
65
+ }
66
+ catch {
67
+ newHashes.files[relativePath] = installedHash;
68
+ continue;
69
+ }
70
+ const newContentHash = hashContent(newContent);
71
+ // Always overwrite tool-owned files
72
+ if (ALWAYS_OVERWRITE.includes(relativePath)) {
73
+ await writeText(filePath, newContent);
74
+ newHashes.files[relativePath] = newContentHash;
75
+ p.log.success(`${relativePath} ${chalk.dim('— tool-owned, updated')}`);
76
+ updated++;
77
+ continue;
78
+ }
79
+ // Check if user has modified the file
80
+ const currentHash = await hashFile(filePath);
81
+ const userModified = currentHash !== installedHash;
82
+ if (!userModified || options.force) {
83
+ // User hasn't modified — safe to overwrite
84
+ await writeText(filePath, newContent);
85
+ newHashes.files[relativePath] = newContentHash;
86
+ if (newContentHash !== installedHash) {
87
+ p.log.success(`${relativePath} ${chalk.dim('— updated')}`);
88
+ updated++;
89
+ }
90
+ }
91
+ else {
92
+ // User modified — ask
93
+ const action = await p.select({
94
+ message: `${relativePath} — you modified this. What to do?`,
95
+ options: [
96
+ { value: 'skip', label: 'Skip (keep your version)' },
97
+ { value: 'overwrite', label: 'Overwrite (use new version)' },
98
+ ],
99
+ });
100
+ if (p.isCancel(action)) {
101
+ p.cancel('Upgrade cancelled.');
102
+ process.exit(0);
103
+ }
104
+ if (action === 'overwrite') {
105
+ await writeText(filePath, newContent);
106
+ newHashes.files[relativePath] = newContentHash;
107
+ updated++;
108
+ }
109
+ else {
110
+ newHashes.files[relativePath] = currentHash;
111
+ skipped++;
112
+ }
113
+ prompted++;
114
+ }
115
+ }
116
+ // Merge forge.yaml — add new fields, preserve existing
117
+ try {
118
+ const templatePath = resolveTemplatePath('core', 'forge.yaml.hbs');
119
+ const template = await readText(templatePath);
120
+ const newForgeYaml = render(template, ctx);
121
+ const existingForgeYaml = await readText(join(cwd, 'forge.yaml'));
122
+ const merged = mergeForgeYaml(existingForgeYaml, newForgeYaml);
123
+ if (merged !== existingForgeYaml) {
124
+ await writeText(join(cwd, 'forge.yaml'), merged);
125
+ p.log.success('forge.yaml — merged new fields');
126
+ updated++;
127
+ }
128
+ }
129
+ catch {
130
+ // Merge failed — skip
131
+ }
132
+ await writeHashes(cwd, newHashes);
133
+ p.outro(`Updated to v${newVersion}. ${chalk.green(`${updated} updated`)}, ${chalk.dim(`${skipped} skipped`)}${prompted > 0 ? `, ${chalk.yellow(`${prompted} prompted`)}` : ''}`);
134
+ }
135
+ function findTemplatePath(relativePath, preset) {
136
+ // Map output paths back to template paths
137
+ const mappings = {
138
+ 'forge.yaml': 'core/forge.yaml.hbs',
139
+ 'CLAUDE.md': 'core/CLAUDE.md.hbs',
140
+ '.claude/settings.json': 'core/settings.json.hbs',
141
+ '.claude/skills/deliver/SKILL.md': 'core/skill-deliver.md.hbs',
142
+ '.forge/pipeline/orchestrator.sh': 'core/pipeline/orchestrator.sh.hbs',
143
+ '.forge/pipeline/intake.sh': 'core/pipeline/intake.sh.hbs',
144
+ '.forge/pipeline/classify.sh': 'core/pipeline/classify.sh.hbs',
145
+ '.forge/pipeline/decompose.md': 'core/pipeline/decompose.md.hbs',
146
+ '.forge/pipeline/execute.md': 'core/pipeline/execute.md.hbs',
147
+ '.forge/pipeline/verify.sh': 'core/pipeline/verify.sh.hbs',
148
+ '.forge/pipeline/deliver.sh': 'core/pipeline/deliver.sh.hbs',
149
+ '.forge/agents/architect.md': 'core/agents/architect.md.hbs',
150
+ '.forge/agents/quality.md': 'core/agents/quality.md.hbs',
151
+ '.forge/agents/security.md': 'core/agents/security.md.hbs',
152
+ '.forge/agents/frontend.md': 'core/agents/frontend.md.hbs',
153
+ '.forge/agents/backend.md': 'core/agents/backend.md.hbs',
154
+ '.forge/context/stack.md': `presets/${preset}/stack.md.hbs`,
155
+ '.forge/context/project.md': 'core/context/project.md.hbs',
156
+ '.forge/hooks/pre-edit.sh': 'core/hooks/pre-edit.sh.hbs',
157
+ '.forge/hooks/post-edit.sh': 'core/hooks/post-edit.sh.hbs',
158
+ '.forge/hooks/session-start.sh': 'core/hooks/session-start.sh.hbs',
159
+ };
160
+ return mappings[relativePath] ?? null;
161
+ }
162
+ function buildUpgradeContext(config, detected, preset) {
163
+ const project = config.project;
164
+ const commands = config.commands;
165
+ const agents = config.agents ?? [];
166
+ const pipeline = config.pipeline ?? {};
167
+ const isFrontendPreset = ['sveltekit-ts', 'react-next-ts', 'vue-nuxt-ts'].includes(preset);
168
+ return {
169
+ project: {
170
+ name: project?.name ?? 'my-app',
171
+ preset,
172
+ },
173
+ commands: commands ?? {},
174
+ agents,
175
+ has_frontend: isFrontendPreset,
176
+ has_format: Boolean(commands?.format),
177
+ auto_pr: pipeline?.auto_pr ?? true,
178
+ detected: {
179
+ language: detected.language,
180
+ framework: detected.framework,
181
+ features: detected.features,
182
+ },
183
+ preset,
184
+ is_sveltekit: preset === 'sveltekit-ts',
185
+ is_nextjs: preset === 'react-next-ts',
186
+ is_fastapi: preset === 'python-fastapi',
187
+ is_go: preset === 'go',
188
+ };
189
+ }
190
+ //# sourceMappingURL=upgrade.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"upgrade.js","sourceRoot":"","sources":["../../src/commands/upgrade.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,10 @@
1
+ export interface FeatureFlags {
2
+ git: boolean;
3
+ ci: 'github-actions' | 'gitlab-ci' | 'jenkins' | null;
4
+ docker: boolean;
5
+ playwright: boolean;
6
+ semgrep: boolean;
7
+ firebase: boolean;
8
+ vercel: boolean;
9
+ }
10
+ export declare function detectFeatures(cwd: string): Promise<FeatureFlags>;
@@ -0,0 +1,33 @@
1
+ import { join } from 'node:path';
2
+ import { exists } from '../utils/fs.js';
3
+ export async function detectFeatures(cwd) {
4
+ const [git, githubActions, gitlabCi, jenkinsfile, dockerfile, dockerCompose, playwrightConfig, semgrepConfig, firebase, vercel] = await Promise.all([
5
+ exists(join(cwd, '.git')),
6
+ exists(join(cwd, '.github', 'workflows')),
7
+ exists(join(cwd, '.gitlab-ci.yml')),
8
+ exists(join(cwd, 'Jenkinsfile')),
9
+ exists(join(cwd, 'Dockerfile')),
10
+ exists(join(cwd, 'docker-compose.yml')),
11
+ exists(join(cwd, 'playwright.config.ts')).then((e) => e || exists(join(cwd, 'playwright.config.js'))),
12
+ exists(join(cwd, '.semgrep.yml')),
13
+ exists(join(cwd, 'firebase.json')),
14
+ exists(join(cwd, 'vercel.json')),
15
+ ]);
16
+ let ci = null;
17
+ if (githubActions)
18
+ ci = 'github-actions';
19
+ else if (gitlabCi)
20
+ ci = 'gitlab-ci';
21
+ else if (jenkinsfile)
22
+ ci = 'jenkins';
23
+ return {
24
+ git,
25
+ ci,
26
+ docker: dockerfile || dockerCompose,
27
+ playwright: playwrightConfig,
28
+ semgrep: semgrepConfig,
29
+ firebase,
30
+ vercel,
31
+ };
32
+ }
33
+ //# sourceMappingURL=features.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"features.js","sourceRoot":"","sources":["../../src/detect/features.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,MAAM,EAAE,MAAM,gBAAgB,CAAC;AAYxC,MAAM,CAAC,KAAK,UAAU,cAAc,CAAC,GAAW;IAC/C,MAAM,CAAC,GAAG,EAAE,aAAa,EAAE,QAAQ,EAAE,WAAW,EAAE,UAAU,EAAE,aAAa,EAAE,gBAAgB,EAAE,aAAa,EAAE,QAAQ,EAAE,MAAM,CAAC,GAC9H,MAAM,OAAO,CAAC,GAAG,CAAC;QACjB,MAAM,CAAC,IAAI,CAAC,GAAG,EAAE,MAAM,CAAC,CAAC;QACzB,MAAM,CAAC,IAAI,CAAC,GAAG,EAAE,SAAS,EAAE,WAAW,CAAC,CAAC;QACzC,MAAM,CAAC,IAAI,CAAC,GAAG,EAAE,gBAAgB,CAAC,CAAC;QACnC,MAAM,CAAC,IAAI,CAAC,GAAG,EAAE,aAAa,CAAC,CAAC;QAChC,MAAM,CAAC,IAAI,CAAC,GAAG,EAAE,YAAY,CAAC,CAAC;QAC/B,MAAM,CAAC,IAAI,CAAC,GAAG,EAAE,oBAAoB,CAAC,CAAC;QACvC,MAAM,CAAC,IAAI,CAAC,GAAG,EAAE,sBAAsB,CAAC,CAAC,CAAC,IAAI,CAC7C,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,IAAI,MAAM,CAAC,IAAI,CAAC,GAAG,EAAE,sBAAsB,CAAC,CAAC,CACrD;QACD,MAAM,CAAC,IAAI,CAAC,GAAG,EAAE,cAAc,CAAC,CAAC;QACjC,MAAM,CAAC,IAAI,CAAC,GAAG,EAAE,eAAe,CAAC,CAAC;QAClC,MAAM,CAAC,IAAI,CAAC,GAAG,EAAE,aAAa,CAAC,CAAC;KAChC,CAAC,CAAC;IAEJ,IAAI,EAAE,GAAuB,IAAI,CAAC;IAClC,IAAI,aAAa;QAAE,EAAE,GAAG,gBAAgB,CAAC;SACpC,IAAI,QAAQ;QAAE,EAAE,GAAG,WAAW,CAAC;SAC/B,IAAI,WAAW;QAAE,EAAE,GAAG,SAAS,CAAC;IAErC,OAAO;QACN,GAAG;QACH,EAAE;QACF,MAAM,EAAE,UAAU,IAAI,aAAa;QACnC,UAAU,EAAE,gBAAgB;QAC5B,OAAO,EAAE,aAAa;QACtB,QAAQ;QACR,MAAM;KACN,CAAC;AACH,CAAC"}
@@ -0,0 +1,3 @@
1
+ type PartialStack = Omit<import('./index.js').DetectedStack, 'features'>;
2
+ export declare function detectGo(cwd: string): Promise<PartialStack | null>;
3
+ export {};
@@ -0,0 +1,38 @@
1
+ import { join } from 'node:path';
2
+ import { exists, readText } from '../utils/fs.js';
3
+ export async function detectGo(cwd) {
4
+ const goModPath = join(cwd, 'go.mod');
5
+ if (!(await exists(goModPath)))
6
+ return null;
7
+ const goMod = await readText(goModPath);
8
+ const result = {
9
+ language: 'go',
10
+ framework: null,
11
+ preset: 'go',
12
+ testRunner: { name: 'go test', command: 'go test ./...' },
13
+ linter: null,
14
+ typeChecker: { name: 'go vet', command: 'go vet ./...' },
15
+ formatter: { name: 'gofmt', command: 'gofmt -w .' },
16
+ packageManager: 'go modules',
17
+ };
18
+ // Framework detection from go.mod requires
19
+ if (goMod.includes('github.com/gin-gonic/gin')) {
20
+ result.framework = 'gin';
21
+ }
22
+ else if (goMod.includes('github.com/go-chi/chi')) {
23
+ result.framework = 'chi';
24
+ }
25
+ else if (goMod.includes('github.com/gofiber/fiber')) {
26
+ result.framework = 'fiber';
27
+ }
28
+ else if (goMod.includes('github.com/labstack/echo')) {
29
+ result.framework = 'echo';
30
+ }
31
+ // Linter — check if golangci-lint config exists
32
+ if ((await exists(join(cwd, '.golangci.yml'))) ||
33
+ (await exists(join(cwd, '.golangci.yaml')))) {
34
+ result.linter = { name: 'golangci-lint', command: 'golangci-lint run' };
35
+ }
36
+ return result;
37
+ }
38
+ //# sourceMappingURL=go.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"go.js","sourceRoot":"","sources":["../../src/detect/go.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,gBAAgB,CAAC;AAIlD,MAAM,CAAC,KAAK,UAAU,QAAQ,CAAC,GAAW;IACzC,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,QAAQ,CAAC,CAAC;IACtC,IAAI,CAAC,CAAC,MAAM,MAAM,CAAC,SAAS,CAAC,CAAC;QAAE,OAAO,IAAI,CAAC;IAE5C,MAAM,KAAK,GAAG,MAAM,QAAQ,CAAC,SAAS,CAAC,CAAC;IAExC,MAAM,MAAM,GAAiB;QAC5B,QAAQ,EAAE,IAAI;QACd,SAAS,EAAE,IAAI;QACf,MAAM,EAAE,IAAI;QACZ,UAAU,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,eAAe,EAAE;QACzD,MAAM,EAAE,IAAI;QACZ,WAAW,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,OAAO,EAAE,cAAc,EAAE;QACxD,SAAS,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,OAAO,EAAE,YAAY,EAAE;QACnD,cAAc,EAAE,YAAY;KAC5B,CAAC;IAEF,2CAA2C;IAC3C,IAAI,KAAK,CAAC,QAAQ,CAAC,0BAA0B,CAAC,EAAE,CAAC;QAChD,MAAM,CAAC,SAAS,GAAG,KAAK,CAAC;IAC1B,CAAC;SAAM,IAAI,KAAK,CAAC,QAAQ,CAAC,uBAAuB,CAAC,EAAE,CAAC;QACpD,MAAM,CAAC,SAAS,GAAG,KAAK,CAAC;IAC1B,CAAC;SAAM,IAAI,KAAK,CAAC,QAAQ,CAAC,0BAA0B,CAAC,EAAE,CAAC;QACvD,MAAM,CAAC,SAAS,GAAG,OAAO,CAAC;IAC5B,CAAC;SAAM,IAAI,KAAK,CAAC,QAAQ,CAAC,0BAA0B,CAAC,EAAE,CAAC;QACvD,MAAM,CAAC,SAAS,GAAG,MAAM,CAAC;IAC3B,CAAC;IAED,gDAAgD;IAChD,IACC,CAAC,MAAM,MAAM,CAAC,IAAI,CAAC,GAAG,EAAE,eAAe,CAAC,CAAC,CAAC;QAC1C,CAAC,MAAM,MAAM,CAAC,IAAI,CAAC,GAAG,EAAE,gBAAgB,CAAC,CAAC,CAAC,EAC1C,CAAC;QACF,MAAM,CAAC,MAAM,GAAG,EAAE,IAAI,EAAE,eAAe,EAAE,OAAO,EAAE,mBAAmB,EAAE,CAAC;IACzE,CAAC;IAED,OAAO,MAAM,CAAC;AACf,CAAC"}
@@ -0,0 +1,25 @@
1
+ import { type FeatureFlags } from './features.js';
2
+ export interface DetectedStack {
3
+ language: 'typescript' | 'javascript' | 'python' | 'go' | 'unknown';
4
+ framework: string | null;
5
+ preset: string | null;
6
+ testRunner: {
7
+ name: string;
8
+ command: string;
9
+ } | null;
10
+ linter: {
11
+ name: string;
12
+ command: string;
13
+ } | null;
14
+ typeChecker: {
15
+ name: string;
16
+ command: string;
17
+ } | null;
18
+ formatter: {
19
+ name: string;
20
+ command: string;
21
+ } | null;
22
+ packageManager: string | null;
23
+ features: FeatureFlags;
24
+ }
25
+ export declare function detect(cwd: string): Promise<DetectedStack>;
@@ -0,0 +1,32 @@
1
+ import { detectNode } from './node.js';
2
+ import { detectPython } from './python.js';
3
+ import { detectGo } from './go.js';
4
+ import { detectFeatures } from './features.js';
5
+ export async function detect(cwd) {
6
+ const features = await detectFeatures(cwd);
7
+ // Try each detector in order of likelihood
8
+ const nodeResult = await detectNode(cwd);
9
+ if (nodeResult) {
10
+ return { ...nodeResult, features };
11
+ }
12
+ const pythonResult = await detectPython(cwd);
13
+ if (pythonResult) {
14
+ return { ...pythonResult, features };
15
+ }
16
+ const goResult = await detectGo(cwd);
17
+ if (goResult) {
18
+ return { ...goResult, features };
19
+ }
20
+ return {
21
+ language: 'unknown',
22
+ framework: null,
23
+ preset: null,
24
+ testRunner: null,
25
+ linter: null,
26
+ typeChecker: null,
27
+ formatter: null,
28
+ packageManager: null,
29
+ features,
30
+ };
31
+ }
32
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/detect/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,WAAW,CAAC;AACvC,OAAO,EAAE,YAAY,EAAE,MAAM,aAAa,CAAC;AAC3C,OAAO,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;AACnC,OAAO,EAAE,cAAc,EAAqB,MAAM,eAAe,CAAC;AAclE,MAAM,CAAC,KAAK,UAAU,MAAM,CAAC,GAAW;IACvC,MAAM,QAAQ,GAAG,MAAM,cAAc,CAAC,GAAG,CAAC,CAAC;IAE3C,2CAA2C;IAC3C,MAAM,UAAU,GAAG,MAAM,UAAU,CAAC,GAAG,CAAC,CAAC;IACzC,IAAI,UAAU,EAAE,CAAC;QAChB,OAAO,EAAE,GAAG,UAAU,EAAE,QAAQ,EAAE,CAAC;IACpC,CAAC;IAED,MAAM,YAAY,GAAG,MAAM,YAAY,CAAC,GAAG,CAAC,CAAC;IAC7C,IAAI,YAAY,EAAE,CAAC;QAClB,OAAO,EAAE,GAAG,YAAY,EAAE,QAAQ,EAAE,CAAC;IACtC,CAAC;IAED,MAAM,QAAQ,GAAG,MAAM,QAAQ,CAAC,GAAG,CAAC,CAAC;IACrC,IAAI,QAAQ,EAAE,CAAC;QACd,OAAO,EAAE,GAAG,QAAQ,EAAE,QAAQ,EAAE,CAAC;IAClC,CAAC;IAED,OAAO;QACN,QAAQ,EAAE,SAAS;QACnB,SAAS,EAAE,IAAI;QACf,MAAM,EAAE,IAAI;QACZ,UAAU,EAAE,IAAI;QAChB,MAAM,EAAE,IAAI;QACZ,WAAW,EAAE,IAAI;QACjB,SAAS,EAAE,IAAI;QACf,cAAc,EAAE,IAAI;QACpB,QAAQ;KACR,CAAC;AACH,CAAC"}
@@ -0,0 +1,3 @@
1
+ type PartialStack = Omit<import('./index.js').DetectedStack, 'features'>;
2
+ export declare function detectNode(cwd: string): Promise<PartialStack | null>;
3
+ export {};
@@ -0,0 +1,99 @@
1
+ import { join } from 'node:path';
2
+ import { exists, readJson } from '../utils/fs.js';
3
+ function hasDep(pkg, name) {
4
+ return Boolean(pkg.dependencies?.[name] || pkg.devDependencies?.[name]);
5
+ }
6
+ export async function detectNode(cwd) {
7
+ const pkgPath = join(cwd, 'package.json');
8
+ if (!(await exists(pkgPath)))
9
+ return null;
10
+ const pkg = await readJson(pkgPath);
11
+ const result = {
12
+ language: hasDep(pkg, 'typescript') || (await exists(join(cwd, 'tsconfig.json')))
13
+ ? 'typescript'
14
+ : 'javascript',
15
+ framework: null,
16
+ preset: null,
17
+ testRunner: null,
18
+ linter: null,
19
+ typeChecker: null,
20
+ formatter: null,
21
+ packageManager: null,
22
+ };
23
+ // Framework detection
24
+ if (hasDep(pkg, '@sveltejs/kit')) {
25
+ result.framework = 'sveltekit';
26
+ result.preset = 'sveltekit-ts';
27
+ }
28
+ else if (hasDep(pkg, 'next')) {
29
+ result.framework = 'next';
30
+ result.preset = 'react-next-ts';
31
+ }
32
+ else if (hasDep(pkg, 'nuxt')) {
33
+ result.framework = 'nuxt';
34
+ result.preset = 'vue-nuxt-ts';
35
+ }
36
+ else if (hasDep(pkg, 'vue')) {
37
+ result.framework = 'vue';
38
+ result.preset = 'vue-nuxt-ts';
39
+ }
40
+ else if (hasDep(pkg, 'express')) {
41
+ result.framework = 'express';
42
+ result.preset = 'node-express';
43
+ }
44
+ else if (hasDep(pkg, 'fastify')) {
45
+ result.framework = 'fastify';
46
+ result.preset = 'node-express';
47
+ }
48
+ // Test runner
49
+ if (hasDep(pkg, 'vitest')) {
50
+ result.testRunner = { name: 'vitest', command: 'npx vitest run' };
51
+ }
52
+ else if (hasDep(pkg, 'jest')) {
53
+ result.testRunner = { name: 'jest', command: 'npx jest' };
54
+ }
55
+ else if (hasDep(pkg, 'mocha')) {
56
+ result.testRunner = { name: 'mocha', command: 'npx mocha' };
57
+ }
58
+ // Linter
59
+ if (hasDep(pkg, 'eslint')) {
60
+ result.linter = { name: 'eslint', command: 'npx eslint .' };
61
+ }
62
+ else if (hasDep(pkg, '@biomejs/biome')) {
63
+ result.linter = { name: 'biome', command: 'npx biome check .' };
64
+ }
65
+ // Type checker
66
+ if (await exists(join(cwd, 'tsconfig.json'))) {
67
+ if (result.framework === 'sveltekit') {
68
+ result.typeChecker = { name: 'svelte-check', command: 'npm run check' };
69
+ }
70
+ else if (result.framework === 'next') {
71
+ result.typeChecker = { name: 'tsc', command: 'npx tsc --noEmit' };
72
+ }
73
+ else {
74
+ result.typeChecker = { name: 'tsc', command: 'npx tsc --noEmit' };
75
+ }
76
+ }
77
+ // Formatter
78
+ if (hasDep(pkg, 'prettier')) {
79
+ result.formatter = { name: 'prettier', command: 'npx prettier --write' };
80
+ }
81
+ else if (hasDep(pkg, '@biomejs/biome')) {
82
+ result.formatter = { name: 'biome', command: 'npx biome format --write .' };
83
+ }
84
+ // Package manager
85
+ if (await exists(join(cwd, 'pnpm-lock.yaml'))) {
86
+ result.packageManager = 'pnpm';
87
+ }
88
+ else if (await exists(join(cwd, 'yarn.lock'))) {
89
+ result.packageManager = 'yarn';
90
+ }
91
+ else if (await exists(join(cwd, 'bun.lockb'))) {
92
+ result.packageManager = 'bun';
93
+ }
94
+ else {
95
+ result.packageManager = 'npm';
96
+ }
97
+ return result;
98
+ }
99
+ //# sourceMappingURL=node.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"node.js","sourceRoot":"","sources":["../../src/detect/node.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ type PartialStack = Omit<import('./index.js').DetectedStack, 'features'>;
2
+ export declare function detectPython(cwd: string): Promise<PartialStack | null>;
3
+ export {};
@@ -0,0 +1,86 @@
1
+ import { join } from 'node:path';
2
+ import { exists, readText } from '../utils/fs.js';
3
+ export async function detectPython(cwd) {
4
+ const hasPyproject = await exists(join(cwd, 'pyproject.toml'));
5
+ const hasRequirements = await exists(join(cwd, 'requirements.txt'));
6
+ if (!hasPyproject && !hasRequirements)
7
+ return null;
8
+ const result = {
9
+ language: 'python',
10
+ framework: null,
11
+ preset: null,
12
+ testRunner: null,
13
+ linter: null,
14
+ typeChecker: null,
15
+ formatter: null,
16
+ packageManager: null,
17
+ };
18
+ // Read deps from available files
19
+ let depsText = '';
20
+ if (hasPyproject) {
21
+ depsText = await readText(join(cwd, 'pyproject.toml'));
22
+ }
23
+ if (hasRequirements) {
24
+ depsText += '\n' + await readText(join(cwd, 'requirements.txt'));
25
+ }
26
+ const hasDep = (name) => depsText.includes(name);
27
+ // Framework
28
+ if (hasDep('fastapi')) {
29
+ result.framework = 'fastapi';
30
+ result.preset = 'python-fastapi';
31
+ }
32
+ else if (hasDep('django')) {
33
+ result.framework = 'django';
34
+ result.preset = 'python-django';
35
+ }
36
+ else if (hasDep('flask')) {
37
+ result.framework = 'flask';
38
+ result.preset = 'python-flask';
39
+ }
40
+ // Test runner
41
+ if (hasDep('pytest')) {
42
+ result.testRunner = { name: 'pytest', command: 'pytest' };
43
+ }
44
+ else {
45
+ result.testRunner = { name: 'unittest', command: 'python -m unittest discover' };
46
+ }
47
+ // Linter
48
+ if (hasDep('ruff')) {
49
+ result.linter = { name: 'ruff', command: 'ruff check .' };
50
+ }
51
+ else if (hasDep('flake8')) {
52
+ result.linter = { name: 'flake8', command: 'flake8' };
53
+ }
54
+ else if (hasDep('pylint')) {
55
+ result.linter = { name: 'pylint', command: 'pylint src/' };
56
+ }
57
+ // Type checker
58
+ if (hasDep('mypy')) {
59
+ result.typeChecker = { name: 'mypy', command: 'mypy .' };
60
+ }
61
+ else if (hasDep('pyright')) {
62
+ result.typeChecker = { name: 'pyright', command: 'pyright' };
63
+ }
64
+ // Formatter
65
+ if (hasDep('ruff')) {
66
+ result.formatter = { name: 'ruff', command: 'ruff format .' };
67
+ }
68
+ else if (hasDep('black')) {
69
+ result.formatter = { name: 'black', command: 'black .' };
70
+ }
71
+ // Package manager
72
+ if (await exists(join(cwd, 'poetry.lock'))) {
73
+ result.packageManager = 'poetry';
74
+ }
75
+ else if (await exists(join(cwd, 'pdm.lock'))) {
76
+ result.packageManager = 'pdm';
77
+ }
78
+ else if (await exists(join(cwd, 'uv.lock'))) {
79
+ result.packageManager = 'uv';
80
+ }
81
+ else {
82
+ result.packageManager = 'pip';
83
+ }
84
+ return result;
85
+ }
86
+ //# sourceMappingURL=python.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"python.js","sourceRoot":"","sources":["../../src/detect/python.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1 @@
1
+ export {};