@stackweld/core 0.2.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/.turbo/turbo-build.log +4 -0
- package/.turbo/turbo-lint.log +498 -0
- package/.turbo/turbo-test.log +21 -0
- package/.turbo/turbo-typecheck.log +4 -0
- package/dist/__tests__/compatibility-scorer.test.d.ts +2 -0
- package/dist/__tests__/compatibility-scorer.test.d.ts.map +1 -0
- package/dist/__tests__/compatibility-scorer.test.js +226 -0
- package/dist/__tests__/compatibility-scorer.test.js.map +1 -0
- package/dist/__tests__/rules-engine.test.d.ts +2 -0
- package/dist/__tests__/rules-engine.test.d.ts.map +1 -0
- package/dist/__tests__/rules-engine.test.js +161 -0
- package/dist/__tests__/rules-engine.test.js.map +1 -0
- package/dist/__tests__/scaffold-orchestrator.test.d.ts +2 -0
- package/dist/__tests__/scaffold-orchestrator.test.d.ts.map +1 -0
- package/dist/__tests__/scaffold-orchestrator.test.js +149 -0
- package/dist/__tests__/scaffold-orchestrator.test.js.map +1 -0
- package/dist/__tests__/stack-engine.test.d.ts +2 -0
- package/dist/__tests__/stack-engine.test.d.ts.map +1 -0
- package/dist/__tests__/stack-engine.test.js +278 -0
- package/dist/__tests__/stack-engine.test.js.map +1 -0
- package/dist/db/database.d.ts +9 -0
- package/dist/db/database.d.ts.map +1 -0
- package/dist/db/database.js +106 -0
- package/dist/db/database.js.map +1 -0
- package/dist/db/index.d.ts +2 -0
- package/dist/db/index.d.ts.map +1 -0
- package/dist/db/index.js +2 -0
- package/dist/db/index.js.map +1 -0
- package/dist/engine/compatibility-scorer.d.ts +37 -0
- package/dist/engine/compatibility-scorer.d.ts.map +1 -0
- package/dist/engine/compatibility-scorer.js +178 -0
- package/dist/engine/compatibility-scorer.js.map +1 -0
- package/dist/engine/compose-generator.d.ts +35 -0
- package/dist/engine/compose-generator.d.ts.map +1 -0
- package/dist/engine/compose-generator.js +95 -0
- package/dist/engine/compose-generator.js.map +1 -0
- package/dist/engine/cost-estimator.d.ts +22 -0
- package/dist/engine/cost-estimator.d.ts.map +1 -0
- package/dist/engine/cost-estimator.js +451 -0
- package/dist/engine/cost-estimator.js.map +1 -0
- package/dist/engine/env-analyzer.d.ts +36 -0
- package/dist/engine/env-analyzer.d.ts.map +1 -0
- package/dist/engine/env-analyzer.js +111 -0
- package/dist/engine/env-analyzer.js.map +1 -0
- package/dist/engine/health-checker.d.ts +20 -0
- package/dist/engine/health-checker.d.ts.map +1 -0
- package/dist/engine/health-checker.js +377 -0
- package/dist/engine/health-checker.js.map +1 -0
- package/dist/engine/index.d.ts +11 -0
- package/dist/engine/index.d.ts.map +1 -0
- package/dist/engine/index.js +7 -0
- package/dist/engine/index.js.map +1 -0
- package/dist/engine/infra-generator.d.ts +26 -0
- package/dist/engine/infra-generator.d.ts.map +1 -0
- package/dist/engine/infra-generator.js +751 -0
- package/dist/engine/infra-generator.js.map +1 -0
- package/dist/engine/migration-planner.d.ts +34 -0
- package/dist/engine/migration-planner.d.ts.map +1 -0
- package/dist/engine/migration-planner.js +427 -0
- package/dist/engine/migration-planner.js.map +1 -0
- package/dist/engine/performance-profiler.d.ts +22 -0
- package/dist/engine/performance-profiler.d.ts.map +1 -0
- package/dist/engine/performance-profiler.js +292 -0
- package/dist/engine/performance-profiler.js.map +1 -0
- package/dist/engine/plugin-loader.d.ts +36 -0
- package/dist/engine/plugin-loader.d.ts.map +1 -0
- package/dist/engine/plugin-loader.js +157 -0
- package/dist/engine/plugin-loader.js.map +1 -0
- package/dist/engine/preferences.d.ts +24 -0
- package/dist/engine/preferences.d.ts.map +1 -0
- package/dist/engine/preferences.js +62 -0
- package/dist/engine/preferences.js.map +1 -0
- package/dist/engine/rules-engine.d.ts +31 -0
- package/dist/engine/rules-engine.d.ts.map +1 -0
- package/dist/engine/rules-engine.js +179 -0
- package/dist/engine/rules-engine.js.map +1 -0
- package/dist/engine/runtime-manager.d.ts +65 -0
- package/dist/engine/runtime-manager.d.ts.map +1 -0
- package/dist/engine/runtime-manager.js +181 -0
- package/dist/engine/runtime-manager.js.map +1 -0
- package/dist/engine/scaffold-orchestrator.d.ts +103 -0
- package/dist/engine/scaffold-orchestrator.d.ts.map +1 -0
- package/dist/engine/scaffold-orchestrator.js +934 -0
- package/dist/engine/scaffold-orchestrator.js.map +1 -0
- package/dist/engine/stack-detector.d.ts +21 -0
- package/dist/engine/stack-detector.d.ts.map +1 -0
- package/dist/engine/stack-detector.js +313 -0
- package/dist/engine/stack-detector.js.map +1 -0
- package/dist/engine/stack-differ.d.ts +26 -0
- package/dist/engine/stack-differ.d.ts.map +1 -0
- package/dist/engine/stack-differ.js +80 -0
- package/dist/engine/stack-differ.js.map +1 -0
- package/dist/engine/stack-engine.d.ts +54 -0
- package/dist/engine/stack-engine.d.ts.map +1 -0
- package/dist/engine/stack-engine.js +186 -0
- package/dist/engine/stack-engine.js.map +1 -0
- package/dist/engine/stack-serializer.d.ts +32 -0
- package/dist/engine/stack-serializer.d.ts.map +1 -0
- package/dist/engine/stack-serializer.js +75 -0
- package/dist/engine/stack-serializer.js.map +1 -0
- package/dist/engine/standards-linter.d.ts +34 -0
- package/dist/engine/standards-linter.d.ts.map +1 -0
- package/dist/engine/standards-linter.js +162 -0
- package/dist/engine/standards-linter.js.map +1 -0
- package/dist/engine/tech-installer.d.ts +37 -0
- package/dist/engine/tech-installer.d.ts.map +1 -0
- package/dist/engine/tech-installer.js +508 -0
- package/dist/engine/tech-installer.js.map +1 -0
- package/dist/index.d.ts +39 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +25 -0
- package/dist/index.js.map +1 -0
- package/dist/types/index.d.ts +6 -0
- package/dist/types/index.d.ts.map +1 -0
- package/dist/types/index.js +2 -0
- package/dist/types/index.js.map +1 -0
- package/dist/types/project.d.ts +33 -0
- package/dist/types/project.d.ts.map +1 -0
- package/dist/types/project.js +6 -0
- package/dist/types/project.js.map +1 -0
- package/dist/types/stack.d.ts +29 -0
- package/dist/types/stack.d.ts.map +1 -0
- package/dist/types/stack.js +6 -0
- package/dist/types/stack.js.map +1 -0
- package/dist/types/technology.d.ts +47 -0
- package/dist/types/technology.d.ts.map +1 -0
- package/dist/types/technology.js +6 -0
- package/dist/types/technology.js.map +1 -0
- package/dist/types/template.d.ts +34 -0
- package/dist/types/template.d.ts.map +1 -0
- package/dist/types/template.js +6 -0
- package/dist/types/template.js.map +1 -0
- package/dist/types/validation.d.ts +20 -0
- package/dist/types/validation.d.ts.map +1 -0
- package/dist/types/validation.js +5 -0
- package/dist/types/validation.js.map +1 -0
- package/package.json +39 -0
- package/src/__tests__/compatibility-scorer.test.ts +264 -0
- package/src/__tests__/rules-engine.test.ts +170 -0
- package/src/__tests__/scaffold-orchestrator.test.ts +161 -0
- package/src/__tests__/stack-engine.test.ts +328 -0
- package/src/db/database.ts +112 -0
- package/src/db/index.ts +1 -0
- package/src/engine/compatibility-scorer.ts +222 -0
- package/src/engine/compose-generator.ts +134 -0
- package/src/engine/cost-estimator.ts +498 -0
- package/src/engine/env-analyzer.ts +156 -0
- package/src/engine/health-checker.ts +421 -0
- package/src/engine/index.ts +17 -0
- package/src/engine/infra-generator.ts +837 -0
- package/src/engine/migration-planner.ts +496 -0
- package/src/engine/performance-profiler.ts +354 -0
- package/src/engine/plugin-loader.ts +216 -0
- package/src/engine/preferences.ts +85 -0
- package/src/engine/rules-engine.ts +204 -0
- package/src/engine/runtime-manager.ts +207 -0
- package/src/engine/scaffold-orchestrator.ts +1052 -0
- package/src/engine/stack-detector.ts +345 -0
- package/src/engine/stack-differ.ts +118 -0
- package/src/engine/stack-engine.ts +258 -0
- package/src/engine/stack-serializer.ts +95 -0
- package/src/engine/standards-linter.ts +210 -0
- package/src/engine/tech-installer.ts +650 -0
- package/src/index.ts +78 -0
- package/src/types/index.ts +10 -0
- package/src/types/project.ts +36 -0
- package/src/types/stack.ts +32 -0
- package/src/types/technology.ts +58 -0
- package/src/types/template.ts +37 -0
- package/src/types/validation.ts +22 -0
- package/tsconfig.json +10 -0
- package/tsconfig.tsbuildinfo +1 -0
|
@@ -0,0 +1,112 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Database — SQLite persistence layer for stacks, projects, and versions.
|
|
3
|
+
* Uses better-sqlite3 for synchronous, reliable access.
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import * as fs from "node:fs";
|
|
7
|
+
import * as path from "node:path";
|
|
8
|
+
import Database from "better-sqlite3";
|
|
9
|
+
|
|
10
|
+
const SCHEMA = `
|
|
11
|
+
CREATE TABLE IF NOT EXISTS stacks (
|
|
12
|
+
id TEXT PRIMARY KEY,
|
|
13
|
+
name TEXT NOT NULL UNIQUE,
|
|
14
|
+
description TEXT NOT NULL DEFAULT '',
|
|
15
|
+
profile TEXT NOT NULL DEFAULT 'standard',
|
|
16
|
+
version INTEGER NOT NULL DEFAULT 1,
|
|
17
|
+
created_at TEXT NOT NULL DEFAULT (datetime('now')),
|
|
18
|
+
updated_at TEXT NOT NULL DEFAULT (datetime('now')),
|
|
19
|
+
tags TEXT NOT NULL DEFAULT '[]'
|
|
20
|
+
);
|
|
21
|
+
|
|
22
|
+
CREATE TABLE IF NOT EXISTS stack_technologies (
|
|
23
|
+
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
24
|
+
stack_id TEXT NOT NULL REFERENCES stacks(id) ON DELETE CASCADE,
|
|
25
|
+
technology_id TEXT NOT NULL,
|
|
26
|
+
version TEXT NOT NULL,
|
|
27
|
+
port INTEGER,
|
|
28
|
+
config TEXT DEFAULT '{}',
|
|
29
|
+
UNIQUE(stack_id, technology_id)
|
|
30
|
+
);
|
|
31
|
+
|
|
32
|
+
CREATE TABLE IF NOT EXISTS stack_versions (
|
|
33
|
+
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
34
|
+
stack_id TEXT NOT NULL REFERENCES stacks(id) ON DELETE CASCADE,
|
|
35
|
+
version INTEGER NOT NULL,
|
|
36
|
+
changelog TEXT NOT NULL DEFAULT '',
|
|
37
|
+
snapshot TEXT NOT NULL,
|
|
38
|
+
created_at TEXT NOT NULL DEFAULT (datetime('now')),
|
|
39
|
+
UNIQUE(stack_id, version)
|
|
40
|
+
);
|
|
41
|
+
|
|
42
|
+
CREATE TABLE IF NOT EXISTS project_instances (
|
|
43
|
+
id TEXT PRIMARY KEY,
|
|
44
|
+
stack_id TEXT NOT NULL REFERENCES stacks(id),
|
|
45
|
+
name TEXT NOT NULL,
|
|
46
|
+
path TEXT NOT NULL,
|
|
47
|
+
template_id TEXT,
|
|
48
|
+
created_at TEXT NOT NULL DEFAULT (datetime('now')),
|
|
49
|
+
last_opened_at TEXT
|
|
50
|
+
);
|
|
51
|
+
|
|
52
|
+
CREATE TABLE IF NOT EXISTS project_services (
|
|
53
|
+
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
54
|
+
project_id TEXT NOT NULL REFERENCES project_instances(id) ON DELETE CASCADE,
|
|
55
|
+
service_name TEXT NOT NULL,
|
|
56
|
+
technology_id TEXT,
|
|
57
|
+
port INTEGER,
|
|
58
|
+
docker_container_id TEXT,
|
|
59
|
+
status TEXT DEFAULT 'not_started',
|
|
60
|
+
UNIQUE(project_id, service_name)
|
|
61
|
+
);
|
|
62
|
+
|
|
63
|
+
CREATE TABLE IF NOT EXISTS user_tech_notes (
|
|
64
|
+
tech_id TEXT PRIMARY KEY,
|
|
65
|
+
rating INTEGER CHECK(rating >= 1 AND rating <= 5),
|
|
66
|
+
notes TEXT DEFAULT '',
|
|
67
|
+
last_used_at TEXT
|
|
68
|
+
);
|
|
69
|
+
|
|
70
|
+
CREATE INDEX IF NOT EXISTS idx_stack_tech_stack ON stack_technologies(stack_id);
|
|
71
|
+
CREATE INDEX IF NOT EXISTS idx_stack_versions_stack ON stack_versions(stack_id);
|
|
72
|
+
CREATE INDEX IF NOT EXISTS idx_projects_stack ON project_instances(stack_id);
|
|
73
|
+
CREATE INDEX IF NOT EXISTS idx_project_services_project ON project_services(project_id);
|
|
74
|
+
|
|
75
|
+
CREATE TABLE IF NOT EXISTS custom_templates (
|
|
76
|
+
id TEXT PRIMARY KEY,
|
|
77
|
+
name TEXT NOT NULL UNIQUE,
|
|
78
|
+
description TEXT NOT NULL DEFAULT '',
|
|
79
|
+
profile TEXT NOT NULL DEFAULT 'standard',
|
|
80
|
+
technology_ids TEXT NOT NULL DEFAULT '[]',
|
|
81
|
+
created_at TEXT NOT NULL DEFAULT (datetime('now'))
|
|
82
|
+
);
|
|
83
|
+
`;
|
|
84
|
+
|
|
85
|
+
let db: Database.Database | null = null;
|
|
86
|
+
|
|
87
|
+
export function getDatabase(dbPath?: string): Database.Database {
|
|
88
|
+
if (db) return db;
|
|
89
|
+
|
|
90
|
+
const resolvedPath = dbPath || getDefaultDbPath();
|
|
91
|
+
const dir = path.dirname(resolvedPath);
|
|
92
|
+
if (!fs.existsSync(dir)) fs.mkdirSync(dir, { recursive: true });
|
|
93
|
+
|
|
94
|
+
db = new Database(resolvedPath);
|
|
95
|
+
db.pragma("journal_mode = WAL");
|
|
96
|
+
db.pragma("foreign_keys = ON");
|
|
97
|
+
db.exec(SCHEMA);
|
|
98
|
+
|
|
99
|
+
return db;
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
export function closeDatabase(): void {
|
|
103
|
+
if (db) {
|
|
104
|
+
db.close();
|
|
105
|
+
db = null;
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
export function getDefaultDbPath(): string {
|
|
110
|
+
const home = process.env.HOME || process.env.USERPROFILE || ".";
|
|
111
|
+
return path.join(home, ".stackweld", "stackweld.db");
|
|
112
|
+
}
|
package/src/db/index.ts
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export { closeDatabase, getDatabase, getDefaultDbPath } from "./database.js";
|
|
@@ -0,0 +1,222 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Compatibility Scorer — Calculates a 0-100 score for technology combinations.
|
|
3
|
+
* Goes beyond binary compatible/incompatible to show nuanced pairing quality.
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import type { Technology } from "../types/index.js";
|
|
7
|
+
|
|
8
|
+
export interface CompatibilityResult {
|
|
9
|
+
score: number; // 0-100
|
|
10
|
+
grade: "S" | "A" | "B" | "C" | "D" | "F";
|
|
11
|
+
factors: CompatibilityFactor[];
|
|
12
|
+
recommendation: string;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
export interface CompatibilityFactor {
|
|
16
|
+
label: string;
|
|
17
|
+
points: number;
|
|
18
|
+
description: string;
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
export interface StackScoreResult {
|
|
22
|
+
overall: number;
|
|
23
|
+
grade: string;
|
|
24
|
+
pairs: Array<{ a: string; b: string; score: number }>;
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
// ─── Complementary category pairs ─────────────────────
|
|
28
|
+
|
|
29
|
+
const COMPLEMENTARY_PAIRS: ReadonlySet<string> = new Set([
|
|
30
|
+
"frontend:backend",
|
|
31
|
+
"backend:frontend",
|
|
32
|
+
"database:orm",
|
|
33
|
+
"orm:database",
|
|
34
|
+
"frontend:styling",
|
|
35
|
+
"styling:frontend",
|
|
36
|
+
"frontend:orm",
|
|
37
|
+
"orm:frontend",
|
|
38
|
+
"backend:database",
|
|
39
|
+
"database:backend",
|
|
40
|
+
"runtime:frontend",
|
|
41
|
+
"frontend:runtime",
|
|
42
|
+
"runtime:backend",
|
|
43
|
+
"backend:runtime",
|
|
44
|
+
"backend:auth",
|
|
45
|
+
"auth:backend",
|
|
46
|
+
"frontend:auth",
|
|
47
|
+
"auth:frontend",
|
|
48
|
+
]);
|
|
49
|
+
|
|
50
|
+
// ─── Grade thresholds & recommendations ───────────────
|
|
51
|
+
|
|
52
|
+
function getGrade(score: number): "S" | "A" | "B" | "C" | "D" | "F" {
|
|
53
|
+
if (score >= 90) return "S";
|
|
54
|
+
if (score >= 75) return "A";
|
|
55
|
+
if (score >= 60) return "B";
|
|
56
|
+
if (score >= 45) return "C";
|
|
57
|
+
if (score >= 25) return "D";
|
|
58
|
+
return "F";
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
const RECOMMENDATIONS: Record<"S" | "A" | "B" | "C" | "D" | "F", string> = {
|
|
62
|
+
S: "Excellent combination — these technologies are designed to work together.",
|
|
63
|
+
A: "Strong pairing with good ecosystem support.",
|
|
64
|
+
B: "Compatible — works well with minor configuration needed.",
|
|
65
|
+
C: "Neutral — no known issues but limited synergy.",
|
|
66
|
+
D: "Weak pairing — consider alternatives.",
|
|
67
|
+
F: "Incompatible — these technologies conflict.",
|
|
68
|
+
};
|
|
69
|
+
|
|
70
|
+
// ─── Runtime detection ────────────────────────────────
|
|
71
|
+
|
|
72
|
+
function getRuntime(tech: Technology): string | null {
|
|
73
|
+
if (tech.category === "runtime") return tech.id;
|
|
74
|
+
for (const req of tech.requires) {
|
|
75
|
+
if (["nodejs", "python", "go", "rust", "php", "bun", "deno"].includes(req)) {
|
|
76
|
+
return req;
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
return null;
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
// ─── Core scoring ─────────────────────────────────────
|
|
83
|
+
|
|
84
|
+
/**
|
|
85
|
+
* Score the compatibility between two technologies (0-100).
|
|
86
|
+
*/
|
|
87
|
+
export function scoreCompatibility(techA: Technology, techB: Technology): CompatibilityResult {
|
|
88
|
+
const factors: CompatibilityFactor[] = [];
|
|
89
|
+
let score = 50; // neutral baseline
|
|
90
|
+
|
|
91
|
+
// +25 if suggested pairing (either direction)
|
|
92
|
+
const aSuggestsB = techA.suggestedWith.includes(techB.id);
|
|
93
|
+
const bSuggestsA = techB.suggestedWith.includes(techA.id);
|
|
94
|
+
if (aSuggestsB || bSuggestsA) {
|
|
95
|
+
factors.push({
|
|
96
|
+
label: "Suggested pairing",
|
|
97
|
+
points: 25,
|
|
98
|
+
description:
|
|
99
|
+
aSuggestsB && bSuggestsA
|
|
100
|
+
? `${techA.name} and ${techB.name} mutually recommend each other`
|
|
101
|
+
: aSuggestsB
|
|
102
|
+
? `${techA.name} suggests ${techB.name}`
|
|
103
|
+
: `${techB.name} suggests ${techA.name}`,
|
|
104
|
+
});
|
|
105
|
+
score += 25;
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
// +15 if shared runtime
|
|
109
|
+
const runtimeA = getRuntime(techA);
|
|
110
|
+
const runtimeB = getRuntime(techB);
|
|
111
|
+
if (runtimeA && runtimeB && runtimeA === runtimeB && techA.id !== techB.id) {
|
|
112
|
+
const runtimeName = runtimeA === "nodejs" ? "Node.js" : runtimeA;
|
|
113
|
+
factors.push({
|
|
114
|
+
label: "Shared runtime",
|
|
115
|
+
points: 15,
|
|
116
|
+
description: `Both use ${runtimeName}`,
|
|
117
|
+
});
|
|
118
|
+
score += 15;
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
// +10 if complementary categories
|
|
122
|
+
const catPair = `${techA.category}:${techB.category}`;
|
|
123
|
+
if (COMPLEMENTARY_PAIRS.has(catPair)) {
|
|
124
|
+
factors.push({
|
|
125
|
+
label: "Complementary categories",
|
|
126
|
+
points: 10,
|
|
127
|
+
description: `${techA.category} + ${techB.category}`,
|
|
128
|
+
});
|
|
129
|
+
score += 10;
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
// -50 if incompatible (either direction) — applied after positives, then clamp
|
|
133
|
+
const aIncompatB = techA.incompatibleWith.includes(techB.id);
|
|
134
|
+
const bIncompatA = techB.incompatibleWith.includes(techA.id);
|
|
135
|
+
if (aIncompatB || bIncompatA) {
|
|
136
|
+
factors.push({
|
|
137
|
+
label: "Incompatible",
|
|
138
|
+
points: -50,
|
|
139
|
+
description:
|
|
140
|
+
aIncompatB && bIncompatA
|
|
141
|
+
? `${techA.name} and ${techB.name} are mutually incompatible`
|
|
142
|
+
: aIncompatB
|
|
143
|
+
? `${techA.name} lists ${techB.name} as incompatible`
|
|
144
|
+
: `${techB.name} lists ${techA.name} as incompatible`,
|
|
145
|
+
});
|
|
146
|
+
score -= 50;
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
// -15 if same category AND same runtime (competing technologies)
|
|
150
|
+
if (
|
|
151
|
+
techA.category === techB.category &&
|
|
152
|
+
techA.id !== techB.id &&
|
|
153
|
+
runtimeA &&
|
|
154
|
+
runtimeB &&
|
|
155
|
+
runtimeA === runtimeB
|
|
156
|
+
) {
|
|
157
|
+
factors.push({
|
|
158
|
+
label: "Same category and runtime",
|
|
159
|
+
points: -15,
|
|
160
|
+
description: `Both are ${techA.category} technologies for ${runtimeA === "nodejs" ? "Node.js" : runtimeA}`,
|
|
161
|
+
});
|
|
162
|
+
score -= 15;
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
// -5 if same default port (minor conflict)
|
|
166
|
+
if (
|
|
167
|
+
techA.defaultPort &&
|
|
168
|
+
techB.defaultPort &&
|
|
169
|
+
techA.defaultPort === techB.defaultPort &&
|
|
170
|
+
techA.id !== techB.id
|
|
171
|
+
) {
|
|
172
|
+
factors.push({
|
|
173
|
+
label: "Port conflict",
|
|
174
|
+
points: -5,
|
|
175
|
+
description: `Both default to port ${techA.defaultPort}`,
|
|
176
|
+
});
|
|
177
|
+
score -= 5;
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
// Clamp
|
|
181
|
+
score = Math.max(0, Math.min(100, score));
|
|
182
|
+
|
|
183
|
+
const grade = getGrade(score);
|
|
184
|
+
|
|
185
|
+
return {
|
|
186
|
+
score,
|
|
187
|
+
grade,
|
|
188
|
+
factors,
|
|
189
|
+
recommendation: RECOMMENDATIONS[grade],
|
|
190
|
+
};
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
/**
|
|
194
|
+
* Score an entire stack by evaluating all unique pairs.
|
|
195
|
+
*/
|
|
196
|
+
export function scoreStack(technologies: Technology[]): StackScoreResult {
|
|
197
|
+
if (technologies.length < 2) {
|
|
198
|
+
return { overall: 100, grade: "S", pairs: [] };
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
const pairs: Array<{ a: string; b: string; score: number }> = [];
|
|
202
|
+
|
|
203
|
+
for (let i = 0; i < technologies.length; i++) {
|
|
204
|
+
for (let j = i + 1; j < technologies.length; j++) {
|
|
205
|
+
const result = scoreCompatibility(technologies[i], technologies[j]);
|
|
206
|
+
pairs.push({
|
|
207
|
+
a: technologies[i].id,
|
|
208
|
+
b: technologies[j].id,
|
|
209
|
+
score: result.score,
|
|
210
|
+
});
|
|
211
|
+
}
|
|
212
|
+
}
|
|
213
|
+
|
|
214
|
+
const overall =
|
|
215
|
+
pairs.length > 0 ? Math.round(pairs.reduce((sum, p) => sum + p.score, 0) / pairs.length) : 100;
|
|
216
|
+
|
|
217
|
+
return {
|
|
218
|
+
overall,
|
|
219
|
+
grade: getGrade(overall),
|
|
220
|
+
pairs,
|
|
221
|
+
};
|
|
222
|
+
}
|
|
@@ -0,0 +1,134 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Compose Generator — Pure function that generates docker-compose.yml content.
|
|
3
|
+
* No disk I/O — returns structured data for preview or writing.
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
export interface ComposePreviewResult {
|
|
7
|
+
yaml: string;
|
|
8
|
+
services: string[];
|
|
9
|
+
ports: Record<string, number>;
|
|
10
|
+
volumes: string[];
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
interface ComposeTechnology {
|
|
14
|
+
id: string;
|
|
15
|
+
name: string;
|
|
16
|
+
category?: string;
|
|
17
|
+
dockerImage?: string;
|
|
18
|
+
defaultPort?: number;
|
|
19
|
+
envVars?: Record<string, string>;
|
|
20
|
+
healthCheck?: {
|
|
21
|
+
endpoint?: string;
|
|
22
|
+
command?: string;
|
|
23
|
+
interval?: string;
|
|
24
|
+
timeout?: string;
|
|
25
|
+
retries?: number;
|
|
26
|
+
};
|
|
27
|
+
port?: number; // Override port from stack definition
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
const DATA_MOUNTS: Record<string, string> = {
|
|
31
|
+
postgresql: "/var/lib/postgresql/data",
|
|
32
|
+
mysql: "/var/lib/mysql",
|
|
33
|
+
mongodb: "/data/db",
|
|
34
|
+
redis: "/data",
|
|
35
|
+
};
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* Generate a docker-compose.yml preview from a list of technologies.
|
|
39
|
+
* Only includes technologies that have a `dockerImage`.
|
|
40
|
+
*/
|
|
41
|
+
export function generateComposePreview(
|
|
42
|
+
technologies: ComposeTechnology[],
|
|
43
|
+
projectName: string,
|
|
44
|
+
): ComposePreviewResult {
|
|
45
|
+
const services: string[] = [];
|
|
46
|
+
const ports: Record<string, number> = {};
|
|
47
|
+
const volumes: string[] = [];
|
|
48
|
+
const lines: string[] = ["services:"];
|
|
49
|
+
|
|
50
|
+
const networkName = `${projectName.toLowerCase().replace(/[^a-z0-9-]/g, "-")}_net`;
|
|
51
|
+
|
|
52
|
+
for (const tech of technologies) {
|
|
53
|
+
if (!tech.dockerImage) continue;
|
|
54
|
+
|
|
55
|
+
services.push(tech.id);
|
|
56
|
+
const port = tech.port ?? tech.defaultPort;
|
|
57
|
+
|
|
58
|
+
lines.push(` ${tech.id}:`);
|
|
59
|
+
lines.push(` image: ${tech.dockerImage}`);
|
|
60
|
+
lines.push(" restart: unless-stopped");
|
|
61
|
+
|
|
62
|
+
// Ports
|
|
63
|
+
if (port) {
|
|
64
|
+
lines.push(" ports:");
|
|
65
|
+
lines.push(` - "${port}:${port}"`);
|
|
66
|
+
ports[tech.id] = port;
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
// Environment variables
|
|
70
|
+
const envVars = tech.envVars ? Object.entries(tech.envVars) : [];
|
|
71
|
+
if (envVars.length > 0) {
|
|
72
|
+
lines.push(" environment:");
|
|
73
|
+
for (const [key, value] of envVars) {
|
|
74
|
+
lines.push(` ${key}: "${value}"`);
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
// Health check
|
|
79
|
+
if (tech.healthCheck) {
|
|
80
|
+
lines.push(" healthcheck:");
|
|
81
|
+
if (tech.healthCheck.command) {
|
|
82
|
+
lines.push(` test: ["CMD-SHELL", "${tech.healthCheck.command}"]`);
|
|
83
|
+
} else if (tech.healthCheck.endpoint) {
|
|
84
|
+
lines.push(` test: ["CMD-SHELL", "curl -f ${tech.healthCheck.endpoint} || exit 1"]`);
|
|
85
|
+
}
|
|
86
|
+
if (tech.healthCheck.interval) {
|
|
87
|
+
lines.push(` interval: ${tech.healthCheck.interval}`);
|
|
88
|
+
}
|
|
89
|
+
if (tech.healthCheck.timeout) {
|
|
90
|
+
lines.push(` timeout: ${tech.healthCheck.timeout}`);
|
|
91
|
+
}
|
|
92
|
+
if (tech.healthCheck.retries) {
|
|
93
|
+
lines.push(` retries: ${tech.healthCheck.retries}`);
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
// Volumes for databases
|
|
98
|
+
const mountPath = DATA_MOUNTS[tech.id];
|
|
99
|
+
if (mountPath && tech.category === "database") {
|
|
100
|
+
const volName = `${tech.id}_data`;
|
|
101
|
+
lines.push(" volumes:");
|
|
102
|
+
lines.push(` - ${volName}:${mountPath}`);
|
|
103
|
+
volumes.push(volName);
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
// Network
|
|
107
|
+
lines.push(" networks:");
|
|
108
|
+
lines.push(` - ${networkName}`);
|
|
109
|
+
|
|
110
|
+
lines.push("");
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
// Named volumes
|
|
114
|
+
if (volumes.length > 0) {
|
|
115
|
+
lines.push("volumes:");
|
|
116
|
+
for (const vol of volumes) {
|
|
117
|
+
lines.push(` ${vol}:`);
|
|
118
|
+
}
|
|
119
|
+
lines.push("");
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
// Network
|
|
123
|
+
lines.push("networks:");
|
|
124
|
+
lines.push(` ${networkName}:`);
|
|
125
|
+
lines.push(" driver: bridge");
|
|
126
|
+
lines.push("");
|
|
127
|
+
|
|
128
|
+
return {
|
|
129
|
+
yaml: lines.join("\n"),
|
|
130
|
+
services,
|
|
131
|
+
ports,
|
|
132
|
+
volumes,
|
|
133
|
+
};
|
|
134
|
+
}
|