@stackweld/core 0.2.1 → 0.3.1

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 (142) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +54 -0
  3. package/dist/__tests__/compatibility-scorer.test.d.ts +1 -1
  4. package/dist/__tests__/rules-engine.test.d.ts +1 -1
  5. package/dist/__tests__/scaffold-orchestrator.test.d.ts +1 -1
  6. package/dist/__tests__/stack-engine.test.d.ts +1 -1
  7. package/dist/db/database.d.ts +1 -1
  8. package/dist/engine/compatibility-scorer.d.ts +16 -19
  9. package/dist/engine/compose-generator.d.ts +21 -23
  10. package/dist/engine/cost-estimator.d.ts +13 -13
  11. package/dist/engine/env-analyzer.d.ts +14 -14
  12. package/dist/engine/health-checker.d.ts +12 -12
  13. package/dist/engine/health-checker.js +14 -6
  14. package/dist/engine/infra-generator.d.ts +14 -17
  15. package/dist/engine/migration-planner.d.ts +18 -22
  16. package/dist/engine/performance-profiler.d.ts +13 -13
  17. package/dist/engine/preferences.d.ts +7 -7
  18. package/dist/engine/runtime-manager.d.ts +46 -56
  19. package/dist/engine/runtime-manager.js +24 -10
  20. package/dist/engine/scaffold-orchestrator.js +42 -6
  21. package/dist/engine/stack-detector.d.ts +10 -10
  22. package/dist/engine/stack-differ.d.ts +15 -15
  23. package/dist/engine/stack-differ.js +76 -72
  24. package/dist/engine/stack-serializer.d.ts +8 -8
  25. package/dist/engine/tech-installer.d.ts +12 -12
  26. package/dist/engine/tech-installer.js +3 -2
  27. package/dist/types/project.d.ts +19 -19
  28. package/dist/types/project.js +1 -1
  29. package/dist/types/stack.d.ts +18 -18
  30. package/dist/types/stack.js +1 -1
  31. package/dist/types/technology.d.ts +28 -37
  32. package/dist/types/technology.js +1 -1
  33. package/dist/types/template.d.ts +22 -22
  34. package/dist/types/template.js +1 -1
  35. package/dist/types/validation.d.ts +12 -12
  36. package/dist/types/validation.js +1 -1
  37. package/package.json +16 -10
  38. package/.turbo/turbo-build.log +0 -4
  39. package/.turbo/turbo-lint.log +0 -498
  40. package/.turbo/turbo-test.log +0 -21
  41. package/.turbo/turbo-typecheck.log +0 -4
  42. package/dist/__tests__/compatibility-scorer.test.d.ts.map +0 -1
  43. package/dist/__tests__/compatibility-scorer.test.js.map +0 -1
  44. package/dist/__tests__/rules-engine.test.d.ts.map +0 -1
  45. package/dist/__tests__/rules-engine.test.js.map +0 -1
  46. package/dist/__tests__/scaffold-orchestrator.test.d.ts.map +0 -1
  47. package/dist/__tests__/scaffold-orchestrator.test.js.map +0 -1
  48. package/dist/__tests__/stack-engine.test.d.ts.map +0 -1
  49. package/dist/__tests__/stack-engine.test.js.map +0 -1
  50. package/dist/db/database.d.ts.map +0 -1
  51. package/dist/db/database.js.map +0 -1
  52. package/dist/db/index.d.ts.map +0 -1
  53. package/dist/db/index.js.map +0 -1
  54. package/dist/engine/compatibility-scorer.d.ts.map +0 -1
  55. package/dist/engine/compatibility-scorer.js.map +0 -1
  56. package/dist/engine/compose-generator.d.ts.map +0 -1
  57. package/dist/engine/compose-generator.js.map +0 -1
  58. package/dist/engine/cost-estimator.d.ts.map +0 -1
  59. package/dist/engine/cost-estimator.js.map +0 -1
  60. package/dist/engine/env-analyzer.d.ts.map +0 -1
  61. package/dist/engine/env-analyzer.js.map +0 -1
  62. package/dist/engine/health-checker.d.ts.map +0 -1
  63. package/dist/engine/health-checker.js.map +0 -1
  64. package/dist/engine/index.d.ts.map +0 -1
  65. package/dist/engine/index.js.map +0 -1
  66. package/dist/engine/infra-generator.d.ts.map +0 -1
  67. package/dist/engine/infra-generator.js.map +0 -1
  68. package/dist/engine/migration-planner.d.ts.map +0 -1
  69. package/dist/engine/migration-planner.js.map +0 -1
  70. package/dist/engine/performance-profiler.d.ts.map +0 -1
  71. package/dist/engine/performance-profiler.js.map +0 -1
  72. package/dist/engine/plugin-loader.d.ts.map +0 -1
  73. package/dist/engine/plugin-loader.js.map +0 -1
  74. package/dist/engine/preferences.d.ts.map +0 -1
  75. package/dist/engine/preferences.js.map +0 -1
  76. package/dist/engine/rules-engine.d.ts.map +0 -1
  77. package/dist/engine/rules-engine.js.map +0 -1
  78. package/dist/engine/runtime-manager.d.ts.map +0 -1
  79. package/dist/engine/runtime-manager.js.map +0 -1
  80. package/dist/engine/scaffold-orchestrator.d.ts.map +0 -1
  81. package/dist/engine/scaffold-orchestrator.js.map +0 -1
  82. package/dist/engine/stack-detector.d.ts.map +0 -1
  83. package/dist/engine/stack-detector.js.map +0 -1
  84. package/dist/engine/stack-differ.d.ts.map +0 -1
  85. package/dist/engine/stack-differ.js.map +0 -1
  86. package/dist/engine/stack-engine.d.ts.map +0 -1
  87. package/dist/engine/stack-engine.js.map +0 -1
  88. package/dist/engine/stack-serializer.d.ts.map +0 -1
  89. package/dist/engine/stack-serializer.js.map +0 -1
  90. package/dist/engine/standards-linter.d.ts.map +0 -1
  91. package/dist/engine/standards-linter.js.map +0 -1
  92. package/dist/engine/tech-installer.d.ts.map +0 -1
  93. package/dist/engine/tech-installer.js.map +0 -1
  94. package/dist/index.d.ts.map +0 -1
  95. package/dist/index.js.map +0 -1
  96. package/dist/types/index.d.ts.map +0 -1
  97. package/dist/types/index.js.map +0 -1
  98. package/dist/types/project.d.ts.map +0 -1
  99. package/dist/types/project.js.map +0 -1
  100. package/dist/types/stack.d.ts.map +0 -1
  101. package/dist/types/stack.js.map +0 -1
  102. package/dist/types/technology.d.ts.map +0 -1
  103. package/dist/types/technology.js.map +0 -1
  104. package/dist/types/template.d.ts.map +0 -1
  105. package/dist/types/template.js.map +0 -1
  106. package/dist/types/validation.d.ts.map +0 -1
  107. package/dist/types/validation.js.map +0 -1
  108. package/src/__tests__/compatibility-scorer.test.ts +0 -264
  109. package/src/__tests__/rules-engine.test.ts +0 -170
  110. package/src/__tests__/scaffold-orchestrator.test.ts +0 -161
  111. package/src/__tests__/stack-engine.test.ts +0 -328
  112. package/src/db/database.ts +0 -112
  113. package/src/db/index.ts +0 -1
  114. package/src/engine/compatibility-scorer.ts +0 -222
  115. package/src/engine/compose-generator.ts +0 -134
  116. package/src/engine/cost-estimator.ts +0 -498
  117. package/src/engine/env-analyzer.ts +0 -156
  118. package/src/engine/health-checker.ts +0 -421
  119. package/src/engine/index.ts +0 -17
  120. package/src/engine/infra-generator.ts +0 -837
  121. package/src/engine/migration-planner.ts +0 -496
  122. package/src/engine/performance-profiler.ts +0 -354
  123. package/src/engine/plugin-loader.ts +0 -216
  124. package/src/engine/preferences.ts +0 -85
  125. package/src/engine/rules-engine.ts +0 -204
  126. package/src/engine/runtime-manager.ts +0 -207
  127. package/src/engine/scaffold-orchestrator.ts +0 -1052
  128. package/src/engine/stack-detector.ts +0 -345
  129. package/src/engine/stack-differ.ts +0 -118
  130. package/src/engine/stack-engine.ts +0 -258
  131. package/src/engine/stack-serializer.ts +0 -95
  132. package/src/engine/standards-linter.ts +0 -210
  133. package/src/engine/tech-installer.ts +0 -650
  134. package/src/index.ts +0 -78
  135. package/src/types/index.ts +0 -10
  136. package/src/types/project.ts +0 -36
  137. package/src/types/stack.ts +0 -32
  138. package/src/types/technology.ts +0 -58
  139. package/src/types/template.ts +0 -37
  140. package/src/types/validation.ts +0 -22
  141. package/tsconfig.json +0 -10
  142. package/tsconfig.tsbuildinfo +0 -1
@@ -1,161 +0,0 @@
1
- import { describe, expect, it } from "vitest";
2
- import { ScaffoldOrchestrator } from "../engine/scaffold-orchestrator.js";
3
- import type { StackDefinition, Technology } from "../types/index.js";
4
-
5
- const mockTechs: Technology[] = [
6
- {
7
- id: "nodejs",
8
- name: "Node.js",
9
- category: "runtime",
10
- description: "JavaScript runtime",
11
- website: "https://nodejs.org",
12
- versions: [{ version: "22" }],
13
- defaultVersion: "22",
14
- defaultPort: 3000,
15
- requires: [],
16
- incompatibleWith: [],
17
- suggestedWith: [],
18
- envVars: { NODE_ENV: "development", PORT: "3000" },
19
- configFiles: ["package.json"],
20
- lastVerified: "2026-03-22",
21
- tags: ["nodejs"],
22
- },
23
- {
24
- id: "postgresql",
25
- name: "PostgreSQL",
26
- category: "database",
27
- description: "Relational database",
28
- website: "https://postgresql.org",
29
- versions: [{ version: "17" }],
30
- defaultVersion: "17",
31
- defaultPort: 5432,
32
- requires: [],
33
- incompatibleWith: [],
34
- suggestedWith: [],
35
- dockerImage: "postgres:17",
36
- healthCheck: {
37
- command: "pg_isready -U postgres",
38
- interval: "5s",
39
- timeout: "5s",
40
- retries: 5,
41
- },
42
- envVars: {
43
- POSTGRES_USER: "postgres",
44
- POSTGRES_PASSWORD: "postgres",
45
- POSTGRES_DB: "app",
46
- },
47
- configFiles: [],
48
- lastVerified: "2026-03-22",
49
- tags: ["postgresql"],
50
- },
51
- {
52
- id: "redis",
53
- name: "Redis",
54
- category: "database",
55
- description: "In-memory store",
56
- website: "https://redis.io",
57
- versions: [{ version: "7" }],
58
- defaultVersion: "7",
59
- defaultPort: 6379,
60
- requires: [],
61
- incompatibleWith: [],
62
- suggestedWith: [],
63
- dockerImage: "redis:7-alpine",
64
- healthCheck: {
65
- command: "redis-cli ping",
66
- interval: "5s",
67
- timeout: "5s",
68
- retries: 5,
69
- },
70
- envVars: { REDIS_URL: "redis://localhost:6379/0" },
71
- configFiles: [],
72
- lastVerified: "2026-03-22",
73
- tags: ["redis"],
74
- },
75
- ];
76
-
77
- const mockStack: StackDefinition = {
78
- id: "test-stack",
79
- name: "Test Stack",
80
- description: "A test stack",
81
- profile: "standard",
82
- technologies: [
83
- { technologyId: "nodejs", version: "22", port: 3000 },
84
- { technologyId: "postgresql", version: "17", port: 5432 },
85
- { technologyId: "redis", version: "7", port: 6379 },
86
- ],
87
- createdAt: "2026-03-22T00:00:00Z",
88
- updatedAt: "2026-03-22T00:00:00Z",
89
- version: 1,
90
- tags: ["test"],
91
- };
92
-
93
- describe("ScaffoldOrchestrator", () => {
94
- const orchestrator = new ScaffoldOrchestrator(mockTechs);
95
-
96
- it("generates all scaffold files", () => {
97
- const output = orchestrator.generate(mockStack);
98
- expect(output.dockerCompose).not.toBeNull();
99
- expect(output.envExample).toBeTruthy();
100
- expect(output.readme).toBeTruthy();
101
- expect(output.gitignore).toBeTruthy();
102
- expect(output.devcontainer).toBeTruthy();
103
- });
104
-
105
- it("generates docker-compose with correct services", () => {
106
- const compose = orchestrator.generateDockerCompose(mockStack, mockTechs);
107
- expect(compose).toContain("postgresql:");
108
- expect(compose).toContain("image: postgres:17");
109
- expect(compose).toContain("redis:");
110
- expect(compose).toContain("image: redis:7-alpine");
111
- expect(compose).toContain("5432:5432");
112
- expect(compose).toContain("6379:6379");
113
- expect(compose).toContain("pg_isready");
114
- expect(compose).toContain("redis-cli ping");
115
- expect(compose).toContain("volumes:");
116
- // Should NOT include nodejs (no docker image)
117
- expect(compose).not.toContain("nodejs:");
118
- });
119
-
120
- it("generates .env.example with all env vars", () => {
121
- const env = orchestrator.generateEnvExample(mockStack, mockTechs);
122
- expect(env).toContain("NODE_ENV=development");
123
- expect(env).toContain("POSTGRES_USER=postgres");
124
- expect(env).toContain("REDIS_URL=redis://localhost:6379/0");
125
- });
126
-
127
- it("generates README with stack table", () => {
128
- const readme = orchestrator.generateReadme(mockStack, mockTechs);
129
- expect(readme).toContain("# Test Stack");
130
- expect(readme).toContain("Node.js");
131
- expect(readme).toContain("PostgreSQL");
132
- expect(readme).toContain("docker compose up");
133
- });
134
-
135
- it("generates .gitignore with relevant patterns", () => {
136
- const gitignore = orchestrator.generateGitignore(mockTechs);
137
- expect(gitignore).toContain("node_modules/");
138
- expect(gitignore).toContain(".env");
139
- expect(gitignore).toContain("*.db");
140
- });
141
-
142
- it("generates devcontainer.json", () => {
143
- const devcontainer = JSON.parse(orchestrator.generateDevcontainer(mockStack, mockTechs));
144
- expect(devcontainer.name).toBe("Test Stack");
145
- expect(devcontainer.forwardPorts).toContain(3000);
146
- expect(devcontainer.forwardPorts).toContain(5432);
147
- expect(devcontainer.features).toHaveProperty("ghcr.io/devcontainers/features/node:1");
148
- expect(devcontainer.features).toHaveProperty(
149
- "ghcr.io/devcontainers/features/docker-in-docker:2",
150
- );
151
- });
152
-
153
- it("returns null docker-compose when no docker services", () => {
154
- const noDockerStack: StackDefinition = {
155
- ...mockStack,
156
- technologies: [{ technologyId: "nodejs", version: "22", port: 3000 }],
157
- };
158
- const output = orchestrator.generate(noDockerStack);
159
- expect(output.dockerCompose).toBeNull();
160
- });
161
- });
@@ -1,328 +0,0 @@
1
- import * as fs from "node:fs";
2
- import * as os from "node:os";
3
- import * as path from "node:path";
4
- import { afterEach, beforeEach, describe, expect, it } from "vitest";
5
- import { closeDatabase, getDatabase } from "../db/database.js";
6
- import { RulesEngine } from "../engine/rules-engine.js";
7
- import { StackEngine } from "../engine/stack-engine.js";
8
- import type { Technology } from "../types/technology.js";
9
-
10
- // Minimal technology fixtures for testing
11
- function makeTech(overrides: Partial<Technology> & { id: string; name: string }): Technology {
12
- return {
13
- category: "backend",
14
- description: "Test technology",
15
- website: "https://example.com",
16
- versions: [{ version: "1.0.0" }],
17
- defaultVersion: "1.0.0",
18
- requires: [],
19
- incompatibleWith: [],
20
- suggestedWith: [],
21
- envVars: {},
22
- configFiles: [],
23
- lastVerified: "2026-01-01",
24
- tags: [],
25
- ...overrides,
26
- };
27
- }
28
-
29
- const fakeTechs: Technology[] = [
30
- makeTech({ id: "nodejs", name: "Node.js", category: "runtime" }),
31
- makeTech({ id: "express", name: "Express", category: "backend", requires: ["nodejs"] }),
32
- makeTech({ id: "postgresql", name: "PostgreSQL", category: "database", defaultPort: 5432 }),
33
- makeTech({ id: "redis", name: "Redis", category: "service", defaultPort: 6379 }),
34
- ];
35
-
36
- let tmpDbPath: string;
37
- let engine: StackEngine;
38
-
39
- beforeEach(() => {
40
- // Use a temp file for the database so tests are isolated
41
- const tmpDir = fs.mkdtempSync(path.join(os.tmpdir(), "stackweld-test-"));
42
- tmpDbPath = path.join(tmpDir, "test.db");
43
- // Initialize the database at the temp path
44
- getDatabase(tmpDbPath);
45
- const rules = new RulesEngine(fakeTechs);
46
- engine = new StackEngine(rules);
47
- });
48
-
49
- afterEach(() => {
50
- closeDatabase();
51
- // Clean up temp file
52
- if (tmpDbPath && fs.existsSync(tmpDbPath)) {
53
- const dir = path.dirname(tmpDbPath);
54
- fs.rmSync(dir, { recursive: true, force: true });
55
- }
56
- });
57
-
58
- describe("StackEngine", () => {
59
- describe("create()", () => {
60
- it("creates a stack and persists it", () => {
61
- const { stack, validation } = engine.create({
62
- name: "test-stack",
63
- technologies: [{ technologyId: "nodejs", version: "1.0.0" }],
64
- });
65
-
66
- expect(validation.valid).toBe(true);
67
- expect(stack.name).toBe("test-stack");
68
- expect(stack.id).toBeTruthy();
69
- expect(stack.version).toBe(1);
70
-
71
- // Verify it's actually persisted
72
- const retrieved = engine.get(stack.id);
73
- expect(retrieved).not.toBeNull();
74
- expect(retrieved?.name).toBe("test-stack");
75
- });
76
-
77
- it("auto-resolves dependencies", () => {
78
- const { stack, validation } = engine.create({
79
- name: "express-stack",
80
- technologies: [{ technologyId: "express", version: "1.0.0" }],
81
- });
82
-
83
- expect(validation.valid).toBe(true);
84
- // express requires nodejs, which should be auto-resolved
85
- expect(validation.resolvedDependencies).toContain("nodejs");
86
- const techIds = stack.technologies.map((t) => t.technologyId);
87
- expect(techIds).toContain("express");
88
- expect(techIds).toContain("nodejs");
89
- });
90
-
91
- it("assigns ports to technologies with defaultPort", () => {
92
- const { stack, validation } = engine.create({
93
- name: "db-stack",
94
- technologies: [
95
- { technologyId: "postgresql", version: "1.0.0" },
96
- { technologyId: "redis", version: "1.0.0" },
97
- ],
98
- });
99
-
100
- expect(validation.valid).toBe(true);
101
- expect(validation.portAssignments.postgresql).toBe(5432);
102
- expect(validation.portAssignments.redis).toBe(6379);
103
- });
104
-
105
- it("sets default profile to standard", () => {
106
- const { stack } = engine.create({
107
- name: "default-profile",
108
- technologies: [{ technologyId: "nodejs", version: "1.0.0" }],
109
- });
110
-
111
- expect(stack.profile).toBe("standard");
112
- });
113
-
114
- it("sets tags when provided", () => {
115
- const { stack } = engine.create({
116
- name: "tagged-stack",
117
- technologies: [{ technologyId: "nodejs", version: "1.0.0" }],
118
- tags: ["test", "dev"],
119
- });
120
-
121
- expect(stack.tags).toEqual(["test", "dev"]);
122
- });
123
- });
124
-
125
- describe("list()", () => {
126
- it("returns empty array when no stacks exist", () => {
127
- const stacks = engine.list();
128
- expect(stacks).toEqual([]);
129
- });
130
-
131
- it("returns saved stacks", () => {
132
- engine.create({
133
- name: "stack-a",
134
- technologies: [{ technologyId: "nodejs", version: "1.0.0" }],
135
- });
136
- engine.create({
137
- name: "stack-b",
138
- technologies: [{ technologyId: "redis", version: "1.0.0" }],
139
- });
140
-
141
- const stacks = engine.list();
142
- expect(stacks.length).toBe(2);
143
- const names = stacks.map((s) => s.name);
144
- expect(names).toContain("stack-a");
145
- expect(names).toContain("stack-b");
146
- });
147
- });
148
-
149
- describe("get()", () => {
150
- it("returns correct stack by ID", () => {
151
- const { stack: created } = engine.create({
152
- name: "get-test",
153
- description: "Testing get",
154
- technologies: [{ technologyId: "nodejs", version: "1.0.0" }],
155
- });
156
-
157
- const retrieved = engine.get(created.id);
158
- expect(retrieved).not.toBeNull();
159
- expect(retrieved?.id).toBe(created.id);
160
- expect(retrieved?.name).toBe("get-test");
161
- expect(retrieved?.description).toBe("Testing get");
162
- });
163
-
164
- it("returns null for non-existent ID", () => {
165
- const result = engine.get("non-existent-uuid");
166
- expect(result).toBeNull();
167
- });
168
-
169
- it("includes technologies in the returned stack", () => {
170
- const { stack: created } = engine.create({
171
- name: "tech-check",
172
- technologies: [
173
- { technologyId: "postgresql", version: "1.0.0" },
174
- { technologyId: "redis", version: "1.0.0" },
175
- ],
176
- });
177
-
178
- const retrieved = engine.get(created.id);
179
- expect(retrieved?.technologies.length).toBe(2);
180
- const ids = retrieved?.technologies.map((t) => t.technologyId);
181
- expect(ids).toContain("postgresql");
182
- expect(ids).toContain("redis");
183
- });
184
- });
185
-
186
- describe("delete()", () => {
187
- it("removes a stack", () => {
188
- const { stack } = engine.create({
189
- name: "to-delete",
190
- technologies: [{ technologyId: "nodejs", version: "1.0.0" }],
191
- });
192
-
193
- const deleted = engine.delete(stack.id);
194
- expect(deleted).toBe(true);
195
-
196
- const retrieved = engine.get(stack.id);
197
- expect(retrieved).toBeNull();
198
- });
199
-
200
- it("returns false for non-existent stack", () => {
201
- const deleted = engine.delete("non-existent-uuid");
202
- expect(deleted).toBe(false);
203
- });
204
-
205
- it("does not affect other stacks", () => {
206
- const { stack: a } = engine.create({
207
- name: "keep-me",
208
- technologies: [{ technologyId: "nodejs", version: "1.0.0" }],
209
- });
210
- const { stack: b } = engine.create({
211
- name: "delete-me",
212
- technologies: [{ technologyId: "redis", version: "1.0.0" }],
213
- });
214
-
215
- engine.delete(b.id);
216
-
217
- expect(engine.get(a.id)).not.toBeNull();
218
- expect(engine.get(b.id)).toBeNull();
219
- });
220
- });
221
-
222
- describe("update()", () => {
223
- it("modifies stack fields", () => {
224
- const { stack } = engine.create({
225
- name: "original",
226
- description: "original desc",
227
- technologies: [{ technologyId: "nodejs", version: "1.0.0" }],
228
- });
229
-
230
- const result = engine.update(stack.id, {
231
- name: "updated-name",
232
- description: "new desc",
233
- });
234
-
235
- expect(result).not.toBeNull();
236
- expect(result?.stack.name).toBe("updated-name");
237
- expect(result?.stack.description).toBe("new desc");
238
- });
239
-
240
- it("increments version number on update", () => {
241
- const { stack } = engine.create({
242
- name: "versioned",
243
- technologies: [{ technologyId: "nodejs", version: "1.0.0" }],
244
- });
245
-
246
- expect(stack.version).toBe(1);
247
-
248
- const result = engine.update(stack.id, { name: "versioned-v2" });
249
- expect(result?.stack.version).toBe(2);
250
-
251
- const result2 = engine.update(stack.id, { name: "versioned-v3" });
252
- expect(result2?.stack.version).toBe(3);
253
- });
254
-
255
- it("returns null for non-existent stack", () => {
256
- const result = engine.update("non-existent", { name: "nope" });
257
- expect(result).toBeNull();
258
- });
259
-
260
- it("updates technologies", () => {
261
- const { stack } = engine.create({
262
- name: "tech-update",
263
- technologies: [{ technologyId: "nodejs", version: "1.0.0" }],
264
- });
265
-
266
- const result = engine.update(stack.id, {
267
- technologies: [
268
- { technologyId: "nodejs", version: "1.0.0" },
269
- { technologyId: "redis", version: "1.0.0" },
270
- ],
271
- });
272
-
273
- const techIds = result?.stack.technologies.map((t) => t.technologyId);
274
- expect(techIds).toContain("nodejs");
275
- expect(techIds).toContain("redis");
276
- });
277
- });
278
-
279
- describe("version management", () => {
280
- it("saves a version snapshot on create", () => {
281
- const { stack } = engine.create({
282
- name: "versioned-stack",
283
- technologies: [{ technologyId: "nodejs", version: "1.0.0" }],
284
- });
285
-
286
- const history = engine.getVersionHistory(stack.id);
287
- expect(history.length).toBe(1);
288
- expect(history[0].version).toBe(1);
289
- expect(history[0].changelog).toBe("Initial creation");
290
- expect(history[0].snapshot.name).toBe("versioned-stack");
291
- });
292
-
293
- it("saves a version snapshot on update", () => {
294
- const { stack } = engine.create({
295
- name: "multi-version",
296
- technologies: [{ technologyId: "nodejs", version: "1.0.0" }],
297
- });
298
-
299
- engine.update(stack.id, { name: "multi-version-v2" });
300
-
301
- const history = engine.getVersionHistory(stack.id);
302
- expect(history.length).toBe(2);
303
- // Ordered by version DESC
304
- expect(history[0].version).toBe(2);
305
- expect(history[1].version).toBe(1);
306
- });
307
-
308
- it("returns empty history for non-existent stack", () => {
309
- const history = engine.getVersionHistory("non-existent");
310
- expect(history).toEqual([]);
311
- });
312
-
313
- it("version snapshot contains full stack data", () => {
314
- const { stack } = engine.create({
315
- name: "snapshot-check",
316
- description: "checking snapshots",
317
- technologies: [{ technologyId: "postgresql", version: "1.0.0" }],
318
- tags: ["prod"],
319
- });
320
-
321
- const history = engine.getVersionHistory(stack.id);
322
- const snapshot = history[0].snapshot;
323
- expect(snapshot.name).toBe("snapshot-check");
324
- expect(snapshot.description).toBe("checking snapshots");
325
- expect(snapshot.tags).toEqual(["prod"]);
326
- });
327
- });
328
- });
@@ -1,112 +0,0 @@
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 DELETED
@@ -1 +0,0 @@
1
- export { closeDatabase, getDatabase, getDefaultDbPath } from "./database.js";