@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.
- package/README.md +175 -0
- package/bin/forge.js +2 -0
- package/dist/addons/index.d.ts +25 -0
- package/dist/addons/index.js +139 -0
- package/dist/addons/index.js.map +1 -0
- package/dist/commands/add.d.ts +1 -0
- package/dist/commands/add.js +61 -0
- package/dist/commands/add.js.map +1 -0
- package/dist/commands/doctor.d.ts +1 -0
- package/dist/commands/doctor.js +177 -0
- package/dist/commands/doctor.js.map +1 -0
- package/dist/commands/ingest.d.ts +24 -0
- package/dist/commands/ingest.js +316 -0
- package/dist/commands/ingest.js.map +1 -0
- package/dist/commands/init.d.ts +8 -0
- package/dist/commands/init.js +557 -0
- package/dist/commands/init.js.map +1 -0
- package/dist/commands/remove.d.ts +1 -0
- package/dist/commands/remove.js +42 -0
- package/dist/commands/remove.js.map +1 -0
- package/dist/commands/status.d.ts +1 -0
- package/dist/commands/status.js +48 -0
- package/dist/commands/status.js.map +1 -0
- package/dist/commands/upgrade.d.ts +5 -0
- package/dist/commands/upgrade.js +190 -0
- package/dist/commands/upgrade.js.map +1 -0
- package/dist/detect/features.d.ts +10 -0
- package/dist/detect/features.js +33 -0
- package/dist/detect/features.js.map +1 -0
- package/dist/detect/go.d.ts +3 -0
- package/dist/detect/go.js +38 -0
- package/dist/detect/go.js.map +1 -0
- package/dist/detect/index.d.ts +25 -0
- package/dist/detect/index.js +32 -0
- package/dist/detect/index.js.map +1 -0
- package/dist/detect/node.d.ts +3 -0
- package/dist/detect/node.js +99 -0
- package/dist/detect/node.js.map +1 -0
- package/dist/detect/python.d.ts +3 -0
- package/dist/detect/python.js +86 -0
- package/dist/detect/python.js.map +1 -0
- package/dist/index.d.ts +1 -0
- package/dist/index.js +51 -0
- package/dist/index.js.map +1 -0
- package/dist/render/engine.d.ts +8 -0
- package/dist/render/engine.js +71 -0
- package/dist/render/engine.js.map +1 -0
- package/dist/render/merge.d.ts +5 -0
- package/dist/render/merge.js +33 -0
- package/dist/render/merge.js.map +1 -0
- package/dist/utils/fs.d.ts +8 -0
- package/dist/utils/fs.js +42 -0
- package/dist/utils/fs.js.map +1 -0
- package/dist/utils/git.d.ts +3 -0
- package/dist/utils/git.js +31 -0
- package/dist/utils/git.js.map +1 -0
- package/dist/utils/hash.d.ts +8 -0
- package/dist/utils/hash.js +22 -0
- package/dist/utils/hash.js.map +1 -0
- package/dist/utils/yaml.d.ts +3 -0
- package/dist/utils/yaml.js +12 -0
- package/dist/utils/yaml.js.map +1 -0
- package/package.json +53 -0
- package/templates/addons/beads-dolt-backend/files/dolt-setup.sh +267 -0
- package/templates/addons/beads-dolt-backend/manifest.yaml +13 -0
- package/templates/addons/browser-testing/files/browser-smoke.sh +196 -0
- package/templates/addons/browser-testing/files/visual-qa.md +103 -0
- package/templates/addons/browser-testing/manifest.yaml +20 -0
- package/templates/addons/compliance-hipaa/files/hipaa-checks.sh +184 -0
- package/templates/addons/compliance-hipaa/files/hipaa-context.md +91 -0
- package/templates/addons/compliance-hipaa/manifest.yaml +15 -0
- package/templates/addons/compliance-soc2/files/soc2-checks.sh +232 -0
- package/templates/addons/compliance-soc2/files/soc2-context.md +147 -0
- package/templates/addons/compliance-soc2/manifest.yaml +15 -0
- package/templates/core/CLAUDE.md.hbs +70 -0
- package/templates/core/agents/architect.md.hbs +68 -0
- package/templates/core/agents/backend.md.hbs +27 -0
- package/templates/core/agents/frontend.md.hbs +25 -0
- package/templates/core/agents/quality.md.hbs +40 -0
- package/templates/core/agents/security.md.hbs +53 -0
- package/templates/core/context/project.md.hbs +60 -0
- package/templates/core/forge.yaml.hbs +69 -0
- package/templates/core/hooks/post-edit.sh.hbs +8 -0
- package/templates/core/hooks/pre-edit.sh.hbs +41 -0
- package/templates/core/hooks/session-start.sh.hbs +34 -0
- package/templates/core/pipeline/classify.sh.hbs +159 -0
- package/templates/core/pipeline/decompose.md.hbs +100 -0
- package/templates/core/pipeline/deliver.sh.hbs +171 -0
- package/templates/core/pipeline/execute.md.hbs +138 -0
- package/templates/core/pipeline/intake.sh.hbs +152 -0
- package/templates/core/pipeline/orchestrator.sh.hbs +361 -0
- package/templates/core/pipeline/verify.sh.hbs +160 -0
- package/templates/core/settings.json.hbs +55 -0
- package/templates/core/skill-creator.md.hbs +151 -0
- package/templates/core/skill-deliver.md.hbs +46 -0
- package/templates/core/skill-ingest.md.hbs +245 -0
- package/templates/presets/go/stack.md.hbs +133 -0
- package/templates/presets/python-fastapi/stack.md.hbs +101 -0
- package/templates/presets/react-next-ts/stack.md.hbs +77 -0
- 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,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,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,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,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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"}
|
package/dist/index.d.ts
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|