buncargo 1.0.26 → 3.0.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 (222) hide show
  1. package/dist/bin.d.ts +1 -12
  2. package/dist/bin.js +261 -252
  3. package/dist/cli/bin.d.ts +13 -0
  4. package/dist/cli/bin.js +315 -0
  5. package/dist/cli/commands/help.d.ts +1 -0
  6. package/dist/cli/commands/runtime.d.ts +5 -0
  7. package/dist/cli/commands/version.d.ts +1 -0
  8. package/dist/cli/index.d.ts +1 -0
  9. package/dist/cli/index.js +14 -0
  10. package/dist/cli/run-cli.d.ts +22 -0
  11. package/dist/cli.d.ts +1 -22
  12. package/dist/cli.js +5 -13
  13. package/dist/config/config.d.ts +1 -0
  14. package/dist/config/define-config.d.ts +13 -0
  15. package/dist/config/index.d.ts +3 -0
  16. package/dist/config/index.js +15 -0
  17. package/dist/config/merge-configs.d.ts +3 -0
  18. package/dist/config/validate-config.d.ts +3 -0
  19. package/dist/config.d.ts +1 -72
  20. package/dist/config.js +12 -12
  21. package/dist/core/docker.d.ts +1 -74
  22. package/dist/core/docker.js +35 -26
  23. package/dist/core/index.d.ts +1 -1
  24. package/dist/core/index.js +123 -108
  25. package/dist/core/network.js +2 -2
  26. package/dist/core/ports.d.ts +22 -0
  27. package/dist/core/ports.js +5 -1
  28. package/dist/core/process.js +1 -1
  29. package/dist/core/tunnel.d.ts +33 -0
  30. package/dist/core/utils.js +2 -2
  31. package/dist/core/watchdog-runner.js +45 -42
  32. package/dist/core/watchdog.d.ts +1 -0
  33. package/dist/core/watchdog.js +4 -2
  34. package/dist/docker/index.d.ts +1 -0
  35. package/dist/docker/index.js +38 -0
  36. package/dist/docker/runtime.d.ts +87 -0
  37. package/dist/docker/runtime.js +37 -0
  38. package/dist/docker-compose/compose.d.ts +1 -0
  39. package/dist/docker-compose/generated-file.d.ts +7 -0
  40. package/dist/docker-compose/index.d.ts +3 -0
  41. package/dist/docker-compose/index.js +15 -0
  42. package/dist/docker-compose/model.d.ts +6 -0
  43. package/dist/docker-compose/services/clickhouse.d.ts +16 -0
  44. package/dist/docker-compose/services/define-docker-service.d.ts +41 -0
  45. package/dist/docker-compose/services/index.d.ts +23 -0
  46. package/dist/docker-compose/services/index.js +17 -0
  47. package/dist/docker-compose/services/postgres.d.ts +12 -0
  48. package/dist/docker-compose/services/redis.d.ts +12 -0
  49. package/dist/docker-compose/services/shared.d.ts +7 -0
  50. package/dist/docker-compose/yaml.d.ts +2 -0
  51. package/dist/environment/create-dev-environment.d.ts +23 -0
  52. package/dist/environment/index.d.ts +1 -0
  53. package/dist/environment/index.js +15 -0
  54. package/dist/environment/logging.d.ts +17 -0
  55. package/dist/environment/seeding.d.ts +9 -0
  56. package/dist/environment.d.ts +1 -23
  57. package/dist/environment.js +12 -14
  58. package/dist/index-045jksh5.js +147 -0
  59. package/dist/index-08wa79cs.js +125 -117
  60. package/dist/index-0kxnae3z.js +335 -0
  61. package/dist/index-1mdrf7nz.js +66 -0
  62. package/dist/index-1yvbwj4k.js +262 -242
  63. package/dist/index-23ev345g.js +475 -0
  64. package/dist/index-2ckr49sf.js +228 -0
  65. package/dist/index-2f47khe5.js +376 -369
  66. package/dist/index-2fr3g85b.js +220 -183
  67. package/dist/index-38xnzpa6.js +450 -0
  68. package/dist/index-3h3dhtf2.js +51 -43
  69. package/dist/index-42x95209.js +51 -43
  70. package/dist/index-4gp0az1g.js +145 -0
  71. package/dist/index-4xrxh8yv.js +72 -0
  72. package/dist/index-5gmws6ah.js +181 -0
  73. package/dist/index-5hka0tff.js +78 -76
  74. package/dist/index-5rfqps4b.js +3 -0
  75. package/dist/index-5t9jxqm0.js +428 -0
  76. package/dist/index-6c1w1xk5.js +101 -0
  77. package/dist/index-6fm7mvwj.js +118 -97
  78. package/dist/index-6srpc523.js +127 -128
  79. package/dist/index-731rzzfp.js +187 -0
  80. package/dist/index-75y4cg2z.js +51 -43
  81. package/dist/index-7ja4ywyj.js +126 -127
  82. package/dist/index-8bw1cmz4.js +531 -0
  83. package/dist/index-8hbbj1mp.js +120 -121
  84. package/dist/index-8xj2p5n5.js +145 -0
  85. package/dist/index-bj79tw5w.js +0 -0
  86. package/dist/index-bnk6nr0g.js +73 -0
  87. package/dist/index-brbbzyks.js +72 -0
  88. package/dist/index-c0dr6mcv.js +123 -0
  89. package/dist/index-cty0bcry.js +235 -218
  90. package/dist/index-d8tyv5se.js +228 -0
  91. package/dist/index-d9efy0n4.js +176 -150
  92. package/dist/index-etfmqjjf.js +427 -0
  93. package/dist/index-fb29934k.js +172 -0
  94. package/dist/index-g50jw1yf.js +72 -0
  95. package/dist/index-g6eb5wdw.js +118 -117
  96. package/dist/index-ggq3yryx.js +99 -95
  97. package/dist/index-h70tce00.js +177 -0
  98. package/dist/index-hkxtfqtc.js +333 -0
  99. package/dist/index-kf3dhser.js +146 -143
  100. package/dist/index-ma6tgdb2.js +500 -0
  101. package/dist/index-mam0bcyz.js +123 -0
  102. package/dist/index-mm412dkp.js +274 -0
  103. package/dist/index-n8v18aeb.js +0 -0
  104. package/dist/index-ndnmnsej.js +378 -371
  105. package/dist/index-p8wty0e2.js +389 -379
  106. package/dist/index-qfphr2fd.js +100 -0
  107. package/dist/index-qqmms8rs.js +51 -43
  108. package/dist/index-qw4093g2.js +51 -43
  109. package/dist/index-qzwpzjbx.js +121 -122
  110. package/dist/index-segbnm0h.js +146 -143
  111. package/dist/index-t0fj6gg1.js +112 -0
  112. package/dist/index-thdkwnv7.js +122 -0
  113. package/dist/index-tjbx2r2t.js +270 -0
  114. package/dist/index-tjqw9vtj.js +62 -54
  115. package/dist/index-vbpb89jy.js +248 -0
  116. package/dist/index-vhs88xhe.js +99 -95
  117. package/dist/index-w8zxnjka.js +249 -0
  118. package/dist/index-wk2na3t9.js +404 -0
  119. package/dist/index-wz9x8g7z.js +383 -373
  120. package/dist/index-x249gyde.js +388 -378
  121. package/dist/index-xkvd0nsd.js +187 -0
  122. package/dist/index-yedqxm1z.js +80 -0
  123. package/dist/index-zfjzzjkf.js +266 -0
  124. package/dist/index.d.ts +12 -8
  125. package/dist/index.js +66 -35
  126. package/dist/lint.d.ts +1 -46
  127. package/dist/lint.js +3 -7
  128. package/dist/loader/cache.d.ts +4 -0
  129. package/dist/loader/find-config-file.d.ts +2 -0
  130. package/dist/loader/index.d.ts +5 -0
  131. package/dist/loader/index.js +24 -0
  132. package/dist/loader/load-dev-env.d.ts +5 -0
  133. package/dist/loader/loader.d.ts +1 -0
  134. package/dist/loader.d.ts +1 -45
  135. package/dist/loader.js +22 -20
  136. package/dist/prisma/index.d.ts +1 -0
  137. package/dist/prisma/prisma.d.ts +29 -0
  138. package/dist/prisma.d.ts +1 -29
  139. package/dist/prisma.js +6 -10
  140. package/dist/src/bin.js +309 -0
  141. package/dist/src/cli.js +5 -0
  142. package/dist/src/config.js +15 -0
  143. package/dist/src/core/docker.js +38 -0
  144. package/dist/src/core/index.js +130 -0
  145. package/dist/src/core/network.js +9 -0
  146. package/dist/src/core/ports.js +23 -0
  147. package/dist/src/core/process.js +31 -0
  148. package/dist/src/core/utils.js +11 -0
  149. package/dist/src/core/watchdog-runner.js +69 -0
  150. package/dist/src/core/watchdog.js +28 -0
  151. package/dist/src/docker/runtime.js +37 -0
  152. package/dist/src/docker-compose/index.js +16 -0
  153. package/dist/src/docker-compose/services/index.js +17 -0
  154. package/dist/src/environment.js +12 -0
  155. package/dist/src/index.js +122 -0
  156. package/dist/src/lint.js +3 -0
  157. package/dist/src/loader.js +25 -0
  158. package/dist/src/prisma.js +6 -0
  159. package/dist/src/types.js +0 -0
  160. package/dist/typecheck/index.d.ts +1 -0
  161. package/dist/typecheck/index.js +7 -0
  162. package/dist/typecheck/typecheck.d.ts +46 -0
  163. package/dist/types/all-types.d.ts +501 -0
  164. package/dist/types/cli.d.ts +1 -0
  165. package/dist/types/config.d.ts +6 -0
  166. package/dist/types/docker.d.ts +15 -0
  167. package/dist/types/environment.d.ts +8 -0
  168. package/dist/types/hooks.d.ts +9 -0
  169. package/dist/types/index.d.ts +1 -0
  170. package/dist/types/index.js +0 -0
  171. package/dist/types/prisma.d.ts +1 -0
  172. package/dist/types.d.ts +1 -393
  173. package/package.json +145 -140
  174. package/readme.md +358 -105
  175. package/src/cli/bin.ts +77 -0
  176. package/src/cli/commands/help.ts +39 -0
  177. package/src/cli/commands/runtime.ts +72 -0
  178. package/src/cli/commands/version.ts +4 -0
  179. package/src/cli/index.ts +1 -0
  180. package/{cli.ts → src/cli/run-cli.ts} +95 -6
  181. package/src/config/define-config.ts +30 -0
  182. package/src/config/index.ts +3 -0
  183. package/src/config/merge-configs.ts +33 -0
  184. package/src/config/validate-config.ts +136 -0
  185. package/{core → src/core}/index.ts +2 -2
  186. package/{core → src/core}/ports.ts +68 -1
  187. package/{core → src/core}/process.ts +6 -2
  188. package/src/core/tunnel.ts +151 -0
  189. package/{core → src/core}/utils.ts +1 -0
  190. package/{core → src/core}/watchdog.ts +5 -1
  191. package/src/docker/index.ts +1 -0
  192. package/{core/docker.ts → src/docker/runtime.ts} +40 -4
  193. package/src/docker-compose/generated-file.ts +45 -0
  194. package/src/docker-compose/index.ts +7 -0
  195. package/src/docker-compose/model.ts +197 -0
  196. package/src/docker-compose/services/clickhouse.ts +79 -0
  197. package/src/docker-compose/services/define-docker-service.ts +109 -0
  198. package/src/docker-compose/services/index.ts +67 -0
  199. package/src/docker-compose/services/postgres.ts +60 -0
  200. package/src/docker-compose/services/redis.ts +48 -0
  201. package/src/docker-compose/services/shared.ts +79 -0
  202. package/src/docker-compose/yaml.ts +88 -0
  203. package/{environment.ts → src/environment/create-dev-environment.ts} +101 -146
  204. package/src/environment/index.ts +1 -0
  205. package/src/environment/logging.ts +101 -0
  206. package/src/environment/seeding.ts +57 -0
  207. package/{index.ts → src/index.ts} +49 -15
  208. package/src/loader/cache.ts +23 -0
  209. package/src/loader/find-config-file.ts +29 -0
  210. package/src/loader/index.ts +17 -0
  211. package/src/loader/load-dev-env.ts +38 -0
  212. package/src/prisma/index.ts +1 -0
  213. package/{prisma.ts → src/prisma/prisma.ts} +4 -2
  214. package/src/typecheck/index.ts +1 -0
  215. package/{types.ts → src/types/all-types.ts} +137 -6
  216. package/src/types/index.ts +1 -0
  217. package/bin.ts +0 -191
  218. package/config.ts +0 -194
  219. package/loader.ts +0 -126
  220. /package/{core → src/core}/network.ts +0 -0
  221. /package/{core → src/core}/watchdog-runner.ts +0 -0
  222. /package/{lint.ts → src/typecheck/typecheck.ts} +0 -0
@@ -1,141 +1,149 @@
1
1
  // core/ports.ts
2
2
  import { existsSync, readFileSync, statSync } from "node:fs";
3
3
  import { basename, dirname, resolve } from "node:path";
4
+
4
5
  function findMonorepoRoot(startDir) {
5
- let dir = startDir ?? process.cwd();
6
- while (dir !== "/") {
7
- try {
8
- const pkgPath = resolve(dir, "package.json");
9
- if (existsSync(pkgPath)) {
10
- const content = readFileSync(pkgPath, "utf-8");
11
- const pkg = JSON.parse(content);
12
- if (pkg.workspaces) {
13
- return dir;
14
- }
15
- }
16
- } catch {}
17
- dir = dirname(dir);
18
- }
19
- return process.cwd();
6
+ let dir = startDir ?? process.cwd();
7
+ while (dir !== "/") {
8
+ try {
9
+ const pkgPath = resolve(dir, "package.json");
10
+ if (existsSync(pkgPath)) {
11
+ const content = readFileSync(pkgPath, "utf-8");
12
+ const pkg = JSON.parse(content);
13
+ if (pkg.workspaces) {
14
+ return dir;
15
+ }
16
+ }
17
+ } catch {}
18
+ dir = dirname(dir);
19
+ }
20
+ return process.cwd();
20
21
  }
21
22
  function getWorktreeName(root) {
22
- const monorepoRoot = root ?? findMonorepoRoot();
23
- const gitPath = resolve(monorepoRoot, ".git");
24
- try {
25
- if (!existsSync(gitPath) || !statSync(gitPath).isFile())
26
- return null;
27
- const content = readFileSync(gitPath, "utf-8").trim();
28
- const match = content.match(/^gitdir:\s*(.+)$/);
29
- if (!match?.[1])
30
- return null;
31
- return basename(match[1]);
32
- } catch {
33
- return null;
34
- }
23
+ const monorepoRoot = root ?? findMonorepoRoot();
24
+ const gitPath = resolve(monorepoRoot, ".git");
25
+ try {
26
+ if (!existsSync(gitPath) || !statSync(gitPath).isFile()) return null;
27
+ const content = readFileSync(gitPath, "utf-8").trim();
28
+ const match = content.match(/^gitdir:\s*(.+)$/);
29
+ if (!match?.[1]) return null;
30
+ return basename(match[1]);
31
+ } catch {
32
+ return null;
33
+ }
35
34
  }
36
35
  function isWorktree(root) {
37
- return getWorktreeName(root) !== null;
36
+ return getWorktreeName(root) !== null;
38
37
  }
39
38
  function simpleHash(str) {
40
- let hash = 0;
41
- for (let i = 0;i < str.length; i++) {
42
- const char = str.charCodeAt(i);
43
- hash = (hash << 5) - hash + char;
44
- hash = hash & hash;
45
- }
46
- return Math.abs(hash);
39
+ let hash = 0;
40
+ for (let i = 0; i < str.length; i++) {
41
+ const char = str.charCodeAt(i);
42
+ hash = (hash << 5) - hash + char;
43
+ hash = hash & hash;
44
+ }
45
+ return Math.abs(hash);
47
46
  }
48
47
  function calculatePortOffset(suffix, root) {
49
- const worktreeName = getWorktreeName(root);
50
- if (!worktreeName)
51
- return 0;
52
- const hashInput = suffix ? `${worktreeName}-${suffix}` : worktreeName;
53
- return 10 + simpleHash(hashInput) % 90;
48
+ const worktreeName = getWorktreeName(root);
49
+ if (!worktreeName) return 0;
50
+ const hashInput = suffix ? `${worktreeName}-${suffix}` : worktreeName;
51
+ return 10 + (simpleHash(hashInput) % 90);
54
52
  }
55
53
  function getProjectName(prefix, suffix, root) {
56
- const monorepoRoot = root ?? findMonorepoRoot();
57
- const dirName = basename(monorepoRoot);
58
- const baseName = `${prefix}-${dirName.toLowerCase().replace(/[^a-z0-9-]/g, "-")}`;
59
- return suffix ? `${baseName}-${suffix}` : baseName;
54
+ const monorepoRoot = root ?? findMonorepoRoot();
55
+ const dirName = basename(monorepoRoot);
56
+ const baseName = `${prefix}-${dirName.toLowerCase().replace(/[^a-z0-9-]/g, "-")}`;
57
+ return suffix ? `${baseName}-${suffix}` : baseName;
60
58
  }
61
59
  function computePorts(services, apps, offset) {
62
- const ports = {};
63
- for (const [name, config] of Object.entries(services)) {
64
- ports[name] = config.port + offset;
65
- if (config.secondaryPort) {
66
- ports[`${name}Secondary`] = config.secondaryPort + offset;
67
- }
68
- }
69
- if (apps) {
70
- for (const [name, config] of Object.entries(apps)) {
71
- ports[name] = config.port + offset;
72
- }
73
- }
74
- return ports;
60
+ const ports = {};
61
+ for (const [name, config] of Object.entries(services)) {
62
+ ports[name] = config.port + offset;
63
+ if (config.secondaryPort) {
64
+ ports[`${name}Secondary`] = config.secondaryPort + offset;
65
+ }
66
+ }
67
+ if (apps) {
68
+ for (const [name, config] of Object.entries(apps)) {
69
+ ports[name] = config.port + offset;
70
+ }
71
+ }
72
+ return ports;
75
73
  }
76
74
  var SERVICE_DEFAULTS = {
77
- postgres: { user: "postgres", password: "postgres", database: "postgres" },
78
- postgresql: { user: "postgres", password: "postgres", database: "postgres" },
79
- redis: { user: "", password: "", database: "" },
80
- clickhouse: { user: "default", password: "clickhouse", database: "default" },
81
- mysql: { user: "root", password: "root", database: "mysql" },
82
- mongodb: { user: "", password: "", database: "" }
75
+ postgres: { user: "postgres", password: "postgres", database: "postgres" },
76
+ postgresql: { user: "postgres", password: "postgres", database: "postgres" },
77
+ redis: { user: "", password: "", database: "" },
78
+ clickhouse: { user: "default", password: "clickhouse", database: "default" },
79
+ mysql: { user: "root", password: "root", database: "mysql" },
80
+ mongodb: { user: "", password: "", database: "" },
83
81
  };
84
82
  function buildServiceUrl(serviceName, ctx, config) {
85
- const defaults = SERVICE_DEFAULTS[serviceName];
86
- if (!defaults && !config.database)
87
- return null;
88
- const user = config.user ?? defaults?.user ?? "";
89
- const password = config.password ?? defaults?.password ?? "";
90
- const database = config.database ?? defaults?.database ?? "";
91
- switch (serviceName) {
92
- case "postgres":
93
- case "postgresql":
94
- return `postgresql://${user}:${password}@${ctx.host}:${ctx.port}/${database}`;
95
- case "redis":
96
- return `redis://${ctx.host}:${ctx.port}`;
97
- case "clickhouse":
98
- return `http://${user}:${password}@${ctx.host}:${ctx.port}/${database}`;
99
- case "mysql":
100
- return `mysql://${user}:${password}@${ctx.host}:${ctx.port}/${database}`;
101
- case "mongodb":
102
- return `mongodb://${ctx.host}:${ctx.port}/${database}`;
103
- default:
104
- return null;
105
- }
83
+ const defaults = SERVICE_DEFAULTS[serviceName];
84
+ if (!defaults && !config.database) return null;
85
+ const user = config.user ?? defaults?.user ?? "";
86
+ const password = config.password ?? defaults?.password ?? "";
87
+ const database = config.database ?? defaults?.database ?? "";
88
+ switch (serviceName) {
89
+ case "postgres":
90
+ case "postgresql":
91
+ return `postgresql://${user}:${password}@${ctx.host}:${ctx.port}/${database}`;
92
+ case "redis":
93
+ return `redis://${ctx.host}:${ctx.port}`;
94
+ case "clickhouse":
95
+ return `http://${user}:${password}@${ctx.host}:${ctx.port}/${database}`;
96
+ case "mysql":
97
+ return `mysql://${user}:${password}@${ctx.host}:${ctx.port}/${database}`;
98
+ case "mongodb":
99
+ return `mongodb://${ctx.host}:${ctx.port}/${database}`;
100
+ default:
101
+ return null;
102
+ }
106
103
  }
107
104
  function computeUrls(services, apps, ports, localIp) {
108
- const urls = {};
109
- const host = "localhost";
110
- for (const [name, config] of Object.entries(services)) {
111
- const port = ports[name];
112
- const secondaryPort = ports[`${name}Secondary`];
113
- if (port === undefined)
114
- continue;
115
- const ctx = { port, secondaryPort, host, localIp };
116
- if (config.urlTemplate) {
117
- urls[name] = config.urlTemplate(ctx);
118
- } else {
119
- const builtUrl = buildServiceUrl(name, { port, host }, {
120
- database: config.database,
121
- user: config.user,
122
- password: config.password
123
- });
124
- if (builtUrl) {
125
- urls[name] = builtUrl;
126
- } else {
127
- urls[name] = `http://${host}:${port}`;
128
- }
129
- }
130
- }
131
- if (apps) {
132
- for (const [name, _config] of Object.entries(apps)) {
133
- const port = ports[name];
134
- urls[name] = `http://${host}:${port}`;
135
- urls[`${name}Local`] = `http://${localIp}:${port}`;
136
- }
137
- }
138
- return urls;
105
+ const urls = {};
106
+ const host = "localhost";
107
+ for (const [name, config] of Object.entries(services)) {
108
+ const port = ports[name];
109
+ const secondaryPort = ports[`${name}Secondary`];
110
+ if (port === undefined) continue;
111
+ const ctx = { port, secondaryPort, host, localIp };
112
+ if (config.urlTemplate) {
113
+ urls[name] = config.urlTemplate(ctx);
114
+ } else {
115
+ const builtUrl = buildServiceUrl(
116
+ name,
117
+ { port, host },
118
+ {
119
+ database: config.database,
120
+ user: config.user,
121
+ password: config.password,
122
+ },
123
+ );
124
+ if (builtUrl) {
125
+ urls[name] = builtUrl;
126
+ } else {
127
+ urls[name] = `http://${host}:${port}`;
128
+ }
129
+ }
130
+ }
131
+ if (apps) {
132
+ for (const [name, _config] of Object.entries(apps)) {
133
+ const port = ports[name];
134
+ urls[name] = `http://${host}:${port}`;
135
+ urls[`${name}Local`] = `http://${localIp}:${port}`;
136
+ }
137
+ }
138
+ return urls;
139
139
  }
140
140
 
141
- export { findMonorepoRoot, getWorktreeName, isWorktree, calculatePortOffset, getProjectName, computePorts, computeUrls };
141
+ export {
142
+ findMonorepoRoot,
143
+ getWorktreeName,
144
+ isWorktree,
145
+ calculatePortOffset,
146
+ getProjectName,
147
+ computePorts,
148
+ computeUrls,
149
+ };
@@ -0,0 +1,335 @@
1
+ import {
2
+ logPublicUrls
3
+ } from "./index-bnk6nr0g.js";
4
+ import {
5
+ spawnWatchdog,
6
+ startHeartbeat,
7
+ stopHeartbeat
8
+ } from "./index-mam0bcyz.js";
9
+ import {
10
+ killProcessesOnAppPorts
11
+ } from "./index-mm412dkp.js";
12
+
13
+ // src/core/tunnel.ts
14
+ import { startTunnel } from "untun";
15
+ function parseExposeNames(exposeValue) {
16
+ if (exposeValue === undefined)
17
+ return null;
18
+ const names = exposeValue.split(",").map((name) => name.trim()).filter(Boolean);
19
+ return new Set(names);
20
+ }
21
+ function asPublicUrl(tunnel) {
22
+ return tunnel.url ?? tunnel.publicUrl ?? tunnel.tunnelUrl ?? null;
23
+ }
24
+ function toCloseFn(tunnel) {
25
+ const close = tunnel.close ?? tunnel.stop ?? tunnel.destroy;
26
+ if (!close)
27
+ return async () => {};
28
+ return async () => {
29
+ await close();
30
+ };
31
+ }
32
+ function resolveExposeTargets(env, exposeValue) {
33
+ const requestedNames = parseExposeNames(exposeValue);
34
+ const knownTargets = new Map;
35
+ const enabledTargets = new Map;
36
+ for (const [name, config] of Object.entries(env.services)) {
37
+ const port = env.ports[name];
38
+ if (port === undefined)
39
+ continue;
40
+ const target = { kind: "service", name, port };
41
+ knownTargets.set(name, target);
42
+ if (config.expose === true) {
43
+ enabledTargets.set(name, target);
44
+ }
45
+ }
46
+ for (const [name, config] of Object.entries(env.apps)) {
47
+ const port = env.ports[name];
48
+ if (port === undefined)
49
+ continue;
50
+ const target = { kind: "app", name, port };
51
+ knownTargets.set(name, target);
52
+ if (config.expose === true) {
53
+ enabledTargets.set(name, target);
54
+ }
55
+ }
56
+ if (requestedNames === null) {
57
+ return {
58
+ targets: Array.from(enabledTargets.values()),
59
+ unknownNames: [],
60
+ notEnabledNames: []
61
+ };
62
+ }
63
+ const unknownNames = [];
64
+ const notEnabledNames = [];
65
+ const targets = [];
66
+ for (const name of requestedNames) {
67
+ if (!knownTargets.has(name)) {
68
+ unknownNames.push(name);
69
+ continue;
70
+ }
71
+ const enabledTarget = enabledTargets.get(name);
72
+ if (!enabledTarget) {
73
+ notEnabledNames.push(name);
74
+ continue;
75
+ }
76
+ targets.push(enabledTarget);
77
+ }
78
+ return { targets, unknownNames, notEnabledNames };
79
+ }
80
+ async function startPublicTunnels(targets, options = {}) {
81
+ const start = options.start ?? ((input) => startTunnel(input));
82
+ const tunnels = [];
83
+ try {
84
+ for (const target of targets) {
85
+ const localUrl = `http://localhost:${target.port}`;
86
+ const tunnel = await start({
87
+ url: localUrl
88
+ });
89
+ const publicUrl = asPublicUrl(tunnel);
90
+ if (!publicUrl) {
91
+ throw new Error(`Tunnel for "${target.name}" did not provide a public URL`);
92
+ }
93
+ tunnels.push({
94
+ kind: target.kind,
95
+ name: target.name,
96
+ localUrl,
97
+ publicUrl,
98
+ close: toCloseFn(tunnel)
99
+ });
100
+ }
101
+ return tunnels;
102
+ } catch (error) {
103
+ await stopPublicTunnels(tunnels);
104
+ throw error;
105
+ }
106
+ }
107
+ async function stopPublicTunnels(tunnels) {
108
+ await Promise.allSettled(tunnels.map((tunnel) => tunnel.close()));
109
+ }
110
+
111
+ // src/cli/run-cli.ts
112
+ import { spawn } from "node:child_process";
113
+ var ACCEPTED_FLAGS = [
114
+ "--help",
115
+ "--down",
116
+ "--reset",
117
+ "--migrate",
118
+ "--seed",
119
+ "--up-only",
120
+ "--expose"
121
+ ];
122
+ function printHelp() {
123
+ console.log(`
124
+ Usage: buncargo dev [options]
125
+
126
+ Options:
127
+ --help Show this help message
128
+ --down Stop all containers
129
+ --reset Stop containers and remove volumes (fresh start)
130
+ --migrate Run migrations and exit
131
+ --seed Run migrations and seeders, then exit
132
+ --up-only Start containers and run migrations, then exit (no dev servers)
133
+ --expose Expose configured targets via public quick tunnels
134
+
135
+ Examples:
136
+ bun dev Start dev environment with all services
137
+ bun dev --seed Run migrations and seed the database
138
+ bun dev --down Stop all containers
139
+ bun dev --reset Stop containers and remove all data
140
+ bun dev --expose Expose all targets with expose: true
141
+ bun dev --expose=api,web Expose specific targets
142
+ `);
143
+ }
144
+ function getUnknownFlags(args) {
145
+ return args.filter((arg) => arg.startsWith("--") && !ACCEPTED_FLAGS.includes(arg.includes("=") ? arg.split("=")[0] : arg));
146
+ }
147
+ async function runCli(env, options = {}) {
148
+ const {
149
+ args = process.argv.slice(2),
150
+ watchdog = true,
151
+ watchdogTimeout = 10,
152
+ devServersCommand
153
+ } = options;
154
+ const exposeRequested = hasFlag(args, "--expose");
155
+ const exposeValue = getFlagValue(args, "--expose");
156
+ let tunnels = [];
157
+ async function cleanupTunnels() {
158
+ env.clearPublicUrls();
159
+ if (tunnels.length === 0)
160
+ return;
161
+ await stopPublicTunnels(tunnels);
162
+ tunnels = [];
163
+ }
164
+ if (args.includes("--help")) {
165
+ printHelp();
166
+ process.exit(0);
167
+ }
168
+ const unknownFlags = getUnknownFlags(args);
169
+ if (unknownFlags.length > 0) {
170
+ console.error(`❌ Unknown flag${unknownFlags.length > 1 ? "s" : ""}: ${unknownFlags.join(", ")}`);
171
+ console.error("");
172
+ printHelp();
173
+ process.exit(1);
174
+ }
175
+ if (args.includes("--down")) {
176
+ env.logInfo();
177
+ await cleanupTunnels();
178
+ await env.stop();
179
+ process.exit(0);
180
+ }
181
+ if (args.includes("--reset")) {
182
+ env.logInfo();
183
+ await cleanupTunnels();
184
+ await env.stop({ removeVolumes: true });
185
+ process.exit(0);
186
+ }
187
+ const skipSeed = args.includes("--seed");
188
+ await env.start({ startServers: false, wait: true, skipSeed });
189
+ if (exposeRequested) {
190
+ const { targets, unknownNames, notEnabledNames } = resolveExposeTargets(env, exposeValue);
191
+ if (unknownNames.length > 0) {
192
+ console.error(`❌ Unknown expose target${unknownNames.length > 1 ? "s" : ""}: ${unknownNames.join(", ")}`);
193
+ await cleanupTunnels();
194
+ process.exit(1);
195
+ }
196
+ if (notEnabledNames.length > 0) {
197
+ console.error(`❌ Target${notEnabledNames.length > 1 ? "s" : ""} missing expose: true: ${notEnabledNames.join(", ")}`);
198
+ console.error(" Mark these in dev.config.ts with expose: true or remove them from --expose.");
199
+ await cleanupTunnels();
200
+ process.exit(1);
201
+ }
202
+ if (targets.length === 0) {
203
+ console.error("❌ No expose targets selected. Add expose: true to services/apps or pass names with --expose=<name>.");
204
+ await cleanupTunnels();
205
+ process.exit(1);
206
+ }
207
+ tunnels = await startPublicTunnels(targets);
208
+ env.setPublicUrls(Object.fromEntries(tunnels.map((tunnel) => [tunnel.name, tunnel.publicUrl])));
209
+ logPublicUrls(tunnels);
210
+ }
211
+ if (args.includes("--migrate")) {
212
+ console.log("");
213
+ console.log("✅ Migrations applied successfully");
214
+ await cleanupTunnels();
215
+ process.exit(0);
216
+ }
217
+ if (args.includes("--seed")) {
218
+ console.log("\uD83C\uDF31 Running seeders...");
219
+ const result = await env.exec("bun run run:seeder", {
220
+ throwOnError: false
221
+ });
222
+ if (result.exitCode !== 0) {
223
+ console.error("❌ Seeding failed");
224
+ if (result.stderr) {
225
+ console.error(result.stderr);
226
+ }
227
+ if (result.stdout) {
228
+ console.error(result.stdout);
229
+ }
230
+ await cleanupTunnels();
231
+ process.exit(1);
232
+ }
233
+ console.log("");
234
+ console.log("✅ Seeding complete");
235
+ await cleanupTunnels();
236
+ process.exit(0);
237
+ }
238
+ if (args.includes("--up-only")) {
239
+ console.log("");
240
+ console.log("✅ Containers started. Environment ready.");
241
+ console.log("");
242
+ await cleanupTunnels();
243
+ process.exit(0);
244
+ }
245
+ if (watchdog) {
246
+ await spawnWatchdog(env.projectName, env.root, {
247
+ timeoutMinutes: watchdogTimeout,
248
+ verbose: true,
249
+ composeFile: env.composeFile
250
+ });
251
+ startHeartbeat(env.projectName);
252
+ }
253
+ const command = devServersCommand ?? buildDevServersCommand(env.apps);
254
+ if (!command) {
255
+ console.log("✅ Containers ready. No apps configured.");
256
+ await new Promise(() => {});
257
+ await cleanupTunnels();
258
+ return;
259
+ }
260
+ await killProcessesOnAppPorts(env.apps, env.ports);
261
+ console.log("");
262
+ console.log("\uD83D\uDD27 Starting dev servers...");
263
+ console.log("");
264
+ await runCommand(command, env.root, env.buildEnvVars(), {
265
+ onSignal: async () => {
266
+ await cleanupTunnels();
267
+ stopHeartbeat();
268
+ }
269
+ });
270
+ stopHeartbeat();
271
+ await cleanupTunnels();
272
+ }
273
+ function buildDevServersCommand(apps) {
274
+ const appEntries = Object.entries(apps);
275
+ if (appEntries.length === 0)
276
+ return null;
277
+ const commands = [];
278
+ const names = [];
279
+ const colors = ["blue", "green", "yellow", "magenta", "cyan", "red"];
280
+ for (const [name, config] of appEntries) {
281
+ names.push(name);
282
+ const cwdPart = config.cwd ? `--cwd ${config.cwd}` : "";
283
+ commands.push(`"bun run ${cwdPart} ${config.devCommand}"`.replace(/\s+/g, " ").trim());
284
+ }
285
+ const namesArg = `-n ${names.join(",")}`;
286
+ const colorsArg = `-c ${colors.slice(0, names.length).join(",")}`;
287
+ const commandsArg = commands.join(" ");
288
+ return `bun concurrently ${namesArg} ${colorsArg} ${commandsArg}`;
289
+ }
290
+ function runCommand(command, cwd, envVars, options = {}) {
291
+ const { onSignal } = options;
292
+ return new Promise((resolve, reject) => {
293
+ const proc = spawn(command, [], {
294
+ cwd,
295
+ env: { ...process.env, ...envVars },
296
+ stdio: "inherit",
297
+ shell: true
298
+ });
299
+ proc.on("close", (code) => {
300
+ if (code === 0 || code === null) {
301
+ resolve();
302
+ } else {
303
+ reject(new Error(`Command exited with code ${code}`));
304
+ }
305
+ });
306
+ proc.on("error", reject);
307
+ const cleanup = () => {
308
+ if (onSignal) {
309
+ onSignal();
310
+ }
311
+ proc.kill("SIGTERM");
312
+ };
313
+ process.on("SIGINT", cleanup);
314
+ process.on("SIGTERM", cleanup);
315
+ });
316
+ }
317
+ function hasFlag(args, flag) {
318
+ return args.includes(flag);
319
+ }
320
+ function getFlagValue(args, flag) {
321
+ const prefixed = args.find((arg) => arg.startsWith(`${flag}=`));
322
+ if (prefixed) {
323
+ return prefixed.split("=")[1];
324
+ }
325
+ const index = args.indexOf(flag);
326
+ if (index !== -1 && index + 1 < args.length) {
327
+ const nextArg = args[index + 1];
328
+ if (nextArg !== undefined && !nextArg.startsWith("-")) {
329
+ return nextArg;
330
+ }
331
+ }
332
+ return;
333
+ }
334
+
335
+ export { resolveExposeTargets, startPublicTunnels, stopPublicTunnels, runCli, hasFlag, getFlagValue };
@@ -0,0 +1,66 @@
1
+ // loader.ts
2
+ import { existsSync } from "node:fs";
3
+ import { dirname, join } from "node:path";
4
+ import { createDevEnvironment } from "./index-wk2na3t9.js";
5
+
6
+ var CONFIG_FILES = [
7
+ "dev.config.ts",
8
+ "dev.config.js",
9
+ "dev-tools.config.ts",
10
+ "dev-tools.config.js",
11
+ ];
12
+ function findConfigFile(startDir) {
13
+ let currentDir = startDir;
14
+ while (true) {
15
+ for (const file of CONFIG_FILES) {
16
+ const configPath = join(currentDir, file);
17
+ if (existsSync(configPath)) {
18
+ return configPath;
19
+ }
20
+ }
21
+ const parentDir = dirname(currentDir);
22
+ if (parentDir === currentDir) {
23
+ return null;
24
+ }
25
+ currentDir = parentDir;
26
+ }
27
+ }
28
+ var cachedEnv = null;
29
+ async function loadDevEnv(options) {
30
+ if (cachedEnv && !options?.reload) {
31
+ return cachedEnv;
32
+ }
33
+ const cwd = options?.cwd ?? process.cwd();
34
+ const configPath = findConfigFile(cwd);
35
+ if (configPath) {
36
+ const mod = await import(configPath);
37
+ const config = mod.default;
38
+ if (!config?.projectPrefix || !config?.services) {
39
+ throw new Error(
40
+ `Invalid config in "${configPath}". Use defineDevConfig() and export as default.`,
41
+ );
42
+ }
43
+ cachedEnv = createDevEnvironment(config);
44
+ return cachedEnv;
45
+ }
46
+ throw new Error(
47
+ `No config file found. Create dev.config.ts with: export default defineDevConfig({ ... })`,
48
+ );
49
+ }
50
+ function getDevEnv() {
51
+ if (!cachedEnv) {
52
+ throw new Error("Dev environment not loaded. Call loadDevEnv() first.");
53
+ }
54
+ return cachedEnv;
55
+ }
56
+ function clearDevEnvCache() {
57
+ cachedEnv = null;
58
+ }
59
+
60
+ export {
61
+ CONFIG_FILES,
62
+ findConfigFile,
63
+ loadDevEnv,
64
+ getDevEnv,
65
+ clearDevEnvCache,
66
+ };