@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.
Files changed (172) hide show
  1. package/.turbo/turbo-build.log +4 -0
  2. package/.turbo/turbo-lint.log +498 -0
  3. package/.turbo/turbo-test.log +21 -0
  4. package/.turbo/turbo-typecheck.log +4 -0
  5. package/dist/__tests__/compatibility-scorer.test.d.ts +2 -0
  6. package/dist/__tests__/compatibility-scorer.test.d.ts.map +1 -0
  7. package/dist/__tests__/compatibility-scorer.test.js +226 -0
  8. package/dist/__tests__/compatibility-scorer.test.js.map +1 -0
  9. package/dist/__tests__/rules-engine.test.d.ts +2 -0
  10. package/dist/__tests__/rules-engine.test.d.ts.map +1 -0
  11. package/dist/__tests__/rules-engine.test.js +161 -0
  12. package/dist/__tests__/rules-engine.test.js.map +1 -0
  13. package/dist/__tests__/scaffold-orchestrator.test.d.ts +2 -0
  14. package/dist/__tests__/scaffold-orchestrator.test.d.ts.map +1 -0
  15. package/dist/__tests__/scaffold-orchestrator.test.js +149 -0
  16. package/dist/__tests__/scaffold-orchestrator.test.js.map +1 -0
  17. package/dist/__tests__/stack-engine.test.d.ts +2 -0
  18. package/dist/__tests__/stack-engine.test.d.ts.map +1 -0
  19. package/dist/__tests__/stack-engine.test.js +278 -0
  20. package/dist/__tests__/stack-engine.test.js.map +1 -0
  21. package/dist/db/database.d.ts +9 -0
  22. package/dist/db/database.d.ts.map +1 -0
  23. package/dist/db/database.js +106 -0
  24. package/dist/db/database.js.map +1 -0
  25. package/dist/db/index.d.ts +2 -0
  26. package/dist/db/index.d.ts.map +1 -0
  27. package/dist/db/index.js +2 -0
  28. package/dist/db/index.js.map +1 -0
  29. package/dist/engine/compatibility-scorer.d.ts +37 -0
  30. package/dist/engine/compatibility-scorer.d.ts.map +1 -0
  31. package/dist/engine/compatibility-scorer.js +178 -0
  32. package/dist/engine/compatibility-scorer.js.map +1 -0
  33. package/dist/engine/compose-generator.d.ts +35 -0
  34. package/dist/engine/compose-generator.d.ts.map +1 -0
  35. package/dist/engine/compose-generator.js +95 -0
  36. package/dist/engine/compose-generator.js.map +1 -0
  37. package/dist/engine/cost-estimator.d.ts +22 -0
  38. package/dist/engine/cost-estimator.d.ts.map +1 -0
  39. package/dist/engine/cost-estimator.js +451 -0
  40. package/dist/engine/cost-estimator.js.map +1 -0
  41. package/dist/engine/env-analyzer.d.ts +36 -0
  42. package/dist/engine/env-analyzer.d.ts.map +1 -0
  43. package/dist/engine/env-analyzer.js +111 -0
  44. package/dist/engine/env-analyzer.js.map +1 -0
  45. package/dist/engine/health-checker.d.ts +20 -0
  46. package/dist/engine/health-checker.d.ts.map +1 -0
  47. package/dist/engine/health-checker.js +377 -0
  48. package/dist/engine/health-checker.js.map +1 -0
  49. package/dist/engine/index.d.ts +11 -0
  50. package/dist/engine/index.d.ts.map +1 -0
  51. package/dist/engine/index.js +7 -0
  52. package/dist/engine/index.js.map +1 -0
  53. package/dist/engine/infra-generator.d.ts +26 -0
  54. package/dist/engine/infra-generator.d.ts.map +1 -0
  55. package/dist/engine/infra-generator.js +751 -0
  56. package/dist/engine/infra-generator.js.map +1 -0
  57. package/dist/engine/migration-planner.d.ts +34 -0
  58. package/dist/engine/migration-planner.d.ts.map +1 -0
  59. package/dist/engine/migration-planner.js +427 -0
  60. package/dist/engine/migration-planner.js.map +1 -0
  61. package/dist/engine/performance-profiler.d.ts +22 -0
  62. package/dist/engine/performance-profiler.d.ts.map +1 -0
  63. package/dist/engine/performance-profiler.js +292 -0
  64. package/dist/engine/performance-profiler.js.map +1 -0
  65. package/dist/engine/plugin-loader.d.ts +36 -0
  66. package/dist/engine/plugin-loader.d.ts.map +1 -0
  67. package/dist/engine/plugin-loader.js +157 -0
  68. package/dist/engine/plugin-loader.js.map +1 -0
  69. package/dist/engine/preferences.d.ts +24 -0
  70. package/dist/engine/preferences.d.ts.map +1 -0
  71. package/dist/engine/preferences.js +62 -0
  72. package/dist/engine/preferences.js.map +1 -0
  73. package/dist/engine/rules-engine.d.ts +31 -0
  74. package/dist/engine/rules-engine.d.ts.map +1 -0
  75. package/dist/engine/rules-engine.js +179 -0
  76. package/dist/engine/rules-engine.js.map +1 -0
  77. package/dist/engine/runtime-manager.d.ts +65 -0
  78. package/dist/engine/runtime-manager.d.ts.map +1 -0
  79. package/dist/engine/runtime-manager.js +181 -0
  80. package/dist/engine/runtime-manager.js.map +1 -0
  81. package/dist/engine/scaffold-orchestrator.d.ts +103 -0
  82. package/dist/engine/scaffold-orchestrator.d.ts.map +1 -0
  83. package/dist/engine/scaffold-orchestrator.js +934 -0
  84. package/dist/engine/scaffold-orchestrator.js.map +1 -0
  85. package/dist/engine/stack-detector.d.ts +21 -0
  86. package/dist/engine/stack-detector.d.ts.map +1 -0
  87. package/dist/engine/stack-detector.js +313 -0
  88. package/dist/engine/stack-detector.js.map +1 -0
  89. package/dist/engine/stack-differ.d.ts +26 -0
  90. package/dist/engine/stack-differ.d.ts.map +1 -0
  91. package/dist/engine/stack-differ.js +80 -0
  92. package/dist/engine/stack-differ.js.map +1 -0
  93. package/dist/engine/stack-engine.d.ts +54 -0
  94. package/dist/engine/stack-engine.d.ts.map +1 -0
  95. package/dist/engine/stack-engine.js +186 -0
  96. package/dist/engine/stack-engine.js.map +1 -0
  97. package/dist/engine/stack-serializer.d.ts +32 -0
  98. package/dist/engine/stack-serializer.d.ts.map +1 -0
  99. package/dist/engine/stack-serializer.js +75 -0
  100. package/dist/engine/stack-serializer.js.map +1 -0
  101. package/dist/engine/standards-linter.d.ts +34 -0
  102. package/dist/engine/standards-linter.d.ts.map +1 -0
  103. package/dist/engine/standards-linter.js +162 -0
  104. package/dist/engine/standards-linter.js.map +1 -0
  105. package/dist/engine/tech-installer.d.ts +37 -0
  106. package/dist/engine/tech-installer.d.ts.map +1 -0
  107. package/dist/engine/tech-installer.js +508 -0
  108. package/dist/engine/tech-installer.js.map +1 -0
  109. package/dist/index.d.ts +39 -0
  110. package/dist/index.d.ts.map +1 -0
  111. package/dist/index.js +25 -0
  112. package/dist/index.js.map +1 -0
  113. package/dist/types/index.d.ts +6 -0
  114. package/dist/types/index.d.ts.map +1 -0
  115. package/dist/types/index.js +2 -0
  116. package/dist/types/index.js.map +1 -0
  117. package/dist/types/project.d.ts +33 -0
  118. package/dist/types/project.d.ts.map +1 -0
  119. package/dist/types/project.js +6 -0
  120. package/dist/types/project.js.map +1 -0
  121. package/dist/types/stack.d.ts +29 -0
  122. package/dist/types/stack.d.ts.map +1 -0
  123. package/dist/types/stack.js +6 -0
  124. package/dist/types/stack.js.map +1 -0
  125. package/dist/types/technology.d.ts +47 -0
  126. package/dist/types/technology.d.ts.map +1 -0
  127. package/dist/types/technology.js +6 -0
  128. package/dist/types/technology.js.map +1 -0
  129. package/dist/types/template.d.ts +34 -0
  130. package/dist/types/template.d.ts.map +1 -0
  131. package/dist/types/template.js +6 -0
  132. package/dist/types/template.js.map +1 -0
  133. package/dist/types/validation.d.ts +20 -0
  134. package/dist/types/validation.d.ts.map +1 -0
  135. package/dist/types/validation.js +5 -0
  136. package/dist/types/validation.js.map +1 -0
  137. package/package.json +39 -0
  138. package/src/__tests__/compatibility-scorer.test.ts +264 -0
  139. package/src/__tests__/rules-engine.test.ts +170 -0
  140. package/src/__tests__/scaffold-orchestrator.test.ts +161 -0
  141. package/src/__tests__/stack-engine.test.ts +328 -0
  142. package/src/db/database.ts +112 -0
  143. package/src/db/index.ts +1 -0
  144. package/src/engine/compatibility-scorer.ts +222 -0
  145. package/src/engine/compose-generator.ts +134 -0
  146. package/src/engine/cost-estimator.ts +498 -0
  147. package/src/engine/env-analyzer.ts +156 -0
  148. package/src/engine/health-checker.ts +421 -0
  149. package/src/engine/index.ts +17 -0
  150. package/src/engine/infra-generator.ts +837 -0
  151. package/src/engine/migration-planner.ts +496 -0
  152. package/src/engine/performance-profiler.ts +354 -0
  153. package/src/engine/plugin-loader.ts +216 -0
  154. package/src/engine/preferences.ts +85 -0
  155. package/src/engine/rules-engine.ts +204 -0
  156. package/src/engine/runtime-manager.ts +207 -0
  157. package/src/engine/scaffold-orchestrator.ts +1052 -0
  158. package/src/engine/stack-detector.ts +345 -0
  159. package/src/engine/stack-differ.ts +118 -0
  160. package/src/engine/stack-engine.ts +258 -0
  161. package/src/engine/stack-serializer.ts +95 -0
  162. package/src/engine/standards-linter.ts +210 -0
  163. package/src/engine/tech-installer.ts +650 -0
  164. package/src/index.ts +78 -0
  165. package/src/types/index.ts +10 -0
  166. package/src/types/project.ts +36 -0
  167. package/src/types/stack.ts +32 -0
  168. package/src/types/technology.ts +58 -0
  169. package/src/types/template.ts +37 -0
  170. package/src/types/validation.ts +22 -0
  171. package/tsconfig.json +10 -0
  172. 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
+ }
@@ -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
+ }