@stackweld/core 0.2.1 → 0.3.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 (108) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +54 -0
  3. package/package.json +16 -10
  4. package/.turbo/turbo-build.log +0 -4
  5. package/.turbo/turbo-lint.log +0 -498
  6. package/.turbo/turbo-test.log +0 -21
  7. package/.turbo/turbo-typecheck.log +0 -4
  8. package/dist/__tests__/compatibility-scorer.test.d.ts.map +0 -1
  9. package/dist/__tests__/compatibility-scorer.test.js.map +0 -1
  10. package/dist/__tests__/rules-engine.test.d.ts.map +0 -1
  11. package/dist/__tests__/rules-engine.test.js.map +0 -1
  12. package/dist/__tests__/scaffold-orchestrator.test.d.ts.map +0 -1
  13. package/dist/__tests__/scaffold-orchestrator.test.js.map +0 -1
  14. package/dist/__tests__/stack-engine.test.d.ts.map +0 -1
  15. package/dist/__tests__/stack-engine.test.js.map +0 -1
  16. package/dist/db/database.d.ts.map +0 -1
  17. package/dist/db/database.js.map +0 -1
  18. package/dist/db/index.d.ts.map +0 -1
  19. package/dist/db/index.js.map +0 -1
  20. package/dist/engine/compatibility-scorer.d.ts.map +0 -1
  21. package/dist/engine/compatibility-scorer.js.map +0 -1
  22. package/dist/engine/compose-generator.d.ts.map +0 -1
  23. package/dist/engine/compose-generator.js.map +0 -1
  24. package/dist/engine/cost-estimator.d.ts.map +0 -1
  25. package/dist/engine/cost-estimator.js.map +0 -1
  26. package/dist/engine/env-analyzer.d.ts.map +0 -1
  27. package/dist/engine/env-analyzer.js.map +0 -1
  28. package/dist/engine/health-checker.d.ts.map +0 -1
  29. package/dist/engine/health-checker.js.map +0 -1
  30. package/dist/engine/index.d.ts.map +0 -1
  31. package/dist/engine/index.js.map +0 -1
  32. package/dist/engine/infra-generator.d.ts.map +0 -1
  33. package/dist/engine/infra-generator.js.map +0 -1
  34. package/dist/engine/migration-planner.d.ts.map +0 -1
  35. package/dist/engine/migration-planner.js.map +0 -1
  36. package/dist/engine/performance-profiler.d.ts.map +0 -1
  37. package/dist/engine/performance-profiler.js.map +0 -1
  38. package/dist/engine/plugin-loader.d.ts.map +0 -1
  39. package/dist/engine/plugin-loader.js.map +0 -1
  40. package/dist/engine/preferences.d.ts.map +0 -1
  41. package/dist/engine/preferences.js.map +0 -1
  42. package/dist/engine/rules-engine.d.ts.map +0 -1
  43. package/dist/engine/rules-engine.js.map +0 -1
  44. package/dist/engine/runtime-manager.d.ts.map +0 -1
  45. package/dist/engine/runtime-manager.js.map +0 -1
  46. package/dist/engine/scaffold-orchestrator.d.ts.map +0 -1
  47. package/dist/engine/scaffold-orchestrator.js.map +0 -1
  48. package/dist/engine/stack-detector.d.ts.map +0 -1
  49. package/dist/engine/stack-detector.js.map +0 -1
  50. package/dist/engine/stack-differ.d.ts.map +0 -1
  51. package/dist/engine/stack-differ.js.map +0 -1
  52. package/dist/engine/stack-engine.d.ts.map +0 -1
  53. package/dist/engine/stack-engine.js.map +0 -1
  54. package/dist/engine/stack-serializer.d.ts.map +0 -1
  55. package/dist/engine/stack-serializer.js.map +0 -1
  56. package/dist/engine/standards-linter.d.ts.map +0 -1
  57. package/dist/engine/standards-linter.js.map +0 -1
  58. package/dist/engine/tech-installer.d.ts.map +0 -1
  59. package/dist/engine/tech-installer.js.map +0 -1
  60. package/dist/index.d.ts.map +0 -1
  61. package/dist/index.js.map +0 -1
  62. package/dist/types/index.d.ts.map +0 -1
  63. package/dist/types/index.js.map +0 -1
  64. package/dist/types/project.d.ts.map +0 -1
  65. package/dist/types/project.js.map +0 -1
  66. package/dist/types/stack.d.ts.map +0 -1
  67. package/dist/types/stack.js.map +0 -1
  68. package/dist/types/technology.d.ts.map +0 -1
  69. package/dist/types/technology.js.map +0 -1
  70. package/dist/types/template.d.ts.map +0 -1
  71. package/dist/types/template.js.map +0 -1
  72. package/dist/types/validation.d.ts.map +0 -1
  73. package/dist/types/validation.js.map +0 -1
  74. package/src/__tests__/compatibility-scorer.test.ts +0 -264
  75. package/src/__tests__/rules-engine.test.ts +0 -170
  76. package/src/__tests__/scaffold-orchestrator.test.ts +0 -161
  77. package/src/__tests__/stack-engine.test.ts +0 -328
  78. package/src/db/database.ts +0 -112
  79. package/src/db/index.ts +0 -1
  80. package/src/engine/compatibility-scorer.ts +0 -222
  81. package/src/engine/compose-generator.ts +0 -134
  82. package/src/engine/cost-estimator.ts +0 -498
  83. package/src/engine/env-analyzer.ts +0 -156
  84. package/src/engine/health-checker.ts +0 -421
  85. package/src/engine/index.ts +0 -17
  86. package/src/engine/infra-generator.ts +0 -837
  87. package/src/engine/migration-planner.ts +0 -496
  88. package/src/engine/performance-profiler.ts +0 -354
  89. package/src/engine/plugin-loader.ts +0 -216
  90. package/src/engine/preferences.ts +0 -85
  91. package/src/engine/rules-engine.ts +0 -204
  92. package/src/engine/runtime-manager.ts +0 -207
  93. package/src/engine/scaffold-orchestrator.ts +0 -1052
  94. package/src/engine/stack-detector.ts +0 -345
  95. package/src/engine/stack-differ.ts +0 -118
  96. package/src/engine/stack-engine.ts +0 -258
  97. package/src/engine/stack-serializer.ts +0 -95
  98. package/src/engine/standards-linter.ts +0 -210
  99. package/src/engine/tech-installer.ts +0 -650
  100. package/src/index.ts +0 -78
  101. package/src/types/index.ts +0 -10
  102. package/src/types/project.ts +0 -36
  103. package/src/types/stack.ts +0 -32
  104. package/src/types/technology.ts +0 -58
  105. package/src/types/template.ts +0 -37
  106. package/src/types/validation.ts +0 -22
  107. package/tsconfig.json +0 -10
  108. 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";