buncargo 1.0.29 → 3.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 (246) hide show
  1. package/dist/bin.d.ts +1 -12
  2. package/dist/bin.js +261 -253
  3. package/dist/cli/bin.d.ts +13 -0
  4. package/dist/cli/bin.js +317 -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 +30 -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 -83
  22. package/dist/core/docker.js +35 -32
  23. package/dist/core/index.d.ts +1 -1
  24. package/dist/core/index.js +123 -118
  25. package/dist/core/network.js +2 -2
  26. package/dist/core/ports.js +1 -1
  27. package/dist/core/process.js +1 -1
  28. package/dist/core/quick-tunnel/cloudflared-process.d.ts +10 -0
  29. package/dist/core/quick-tunnel/constants.d.ts +9 -0
  30. package/dist/core/quick-tunnel/index.d.ts +17 -0
  31. package/dist/core/quick-tunnel/install.d.ts +1 -0
  32. package/dist/core/tunnel.d.ts +34 -0
  33. package/dist/core/utils.js +2 -2
  34. package/dist/core/watchdog-runner.js +45 -42
  35. package/dist/core/watchdog.d.ts +1 -0
  36. package/dist/core/watchdog.js +4 -2
  37. package/dist/docker/index.d.ts +1 -0
  38. package/dist/docker/index.js +38 -0
  39. package/dist/docker/runtime.d.ts +87 -0
  40. package/dist/docker/runtime.js +37 -0
  41. package/dist/docker-compose/compose.d.ts +1 -0
  42. package/dist/docker-compose/generated-file.d.ts +7 -0
  43. package/dist/docker-compose/index.d.ts +3 -0
  44. package/dist/docker-compose/index.js +15 -0
  45. package/dist/docker-compose/model.d.ts +6 -0
  46. package/dist/docker-compose/services/clickhouse.d.ts +16 -0
  47. package/dist/docker-compose/services/define-docker-service.d.ts +41 -0
  48. package/dist/docker-compose/services/index.d.ts +23 -0
  49. package/dist/docker-compose/services/index.js +17 -0
  50. package/dist/docker-compose/services/postgres.d.ts +12 -0
  51. package/dist/docker-compose/services/redis.d.ts +12 -0
  52. package/dist/docker-compose/services/shared.d.ts +7 -0
  53. package/dist/docker-compose/yaml.d.ts +2 -0
  54. package/dist/environment/create-dev-environment.d.ts +23 -0
  55. package/dist/environment/index.d.ts +1 -0
  56. package/dist/environment/index.js +15 -0
  57. package/dist/environment/logging.d.ts +17 -0
  58. package/dist/environment/only-apps.d.ts +10 -0
  59. package/dist/environment/seeding.d.ts +9 -0
  60. package/dist/environment.d.ts +1 -23
  61. package/dist/environment.js +12 -14
  62. package/dist/index-045jksh5.js +147 -0
  63. package/dist/index-08wa79cs.js +125 -117
  64. package/dist/index-0kxnae3z.js +335 -0
  65. package/dist/index-1mdrf7nz.js +51 -43
  66. package/dist/index-1yvbwj4k.js +262 -242
  67. package/dist/index-23ev345g.js +475 -0
  68. package/dist/index-2ckr49sf.js +228 -0
  69. package/dist/index-2f47khe5.js +376 -369
  70. package/dist/index-2fr3g85b.js +220 -183
  71. package/dist/index-38xnzpa6.js +450 -0
  72. package/dist/index-3eyrdxw9.js +577 -0
  73. package/dist/index-3h3dhtf2.js +51 -43
  74. package/dist/index-42x95209.js +51 -43
  75. package/dist/index-4gp0az1g.js +145 -0
  76. package/dist/index-4xrxh8yv.js +72 -0
  77. package/dist/index-5aq985p4.js +250 -0
  78. package/dist/index-5gmws6ah.js +181 -0
  79. package/dist/index-5hka0tff.js +78 -76
  80. package/dist/index-5rfqps4b.js +3 -0
  81. package/dist/index-5t9jxqm0.js +428 -0
  82. package/dist/index-6c1w1xk5.js +101 -0
  83. package/dist/index-6cmex7m5.js +72 -0
  84. package/dist/index-6d6x175r.js +572 -0
  85. package/dist/index-6fm7mvwj.js +118 -97
  86. package/dist/index-6srpc523.js +127 -128
  87. package/dist/index-731rzzfp.js +157 -142
  88. package/dist/index-75y4cg2z.js +51 -43
  89. package/dist/index-7ja4ywyj.js +126 -127
  90. package/dist/index-7v19es2e.js +666 -0
  91. package/dist/index-8bw1cmz4.js +531 -0
  92. package/dist/index-8hbbj1mp.js +120 -121
  93. package/dist/index-8xj2p5n5.js +118 -97
  94. package/dist/index-9wyhzw0h.js +574 -0
  95. package/dist/index-ag90ry8t.js +576 -0
  96. package/dist/index-bj79tw5w.js +0 -0
  97. package/dist/index-bnk6nr0g.js +73 -0
  98. package/dist/index-brbbzyks.js +72 -0
  99. package/dist/index-byeqyjrz.js +72 -0
  100. package/dist/index-c0dr6mcv.js +123 -0
  101. package/dist/index-cty0bcry.js +235 -218
  102. package/dist/index-d8tyv5se.js +228 -0
  103. package/dist/index-d9efy0n4.js +176 -150
  104. package/dist/index-enj4zdma.js +574 -0
  105. package/dist/index-etfmqjjf.js +427 -0
  106. package/dist/index-fb29934k.js +172 -0
  107. package/dist/index-g50jw1yf.js +72 -0
  108. package/dist/index-g6eb5wdw.js +118 -117
  109. package/dist/index-ggq3yryx.js +99 -95
  110. package/dist/index-h70tce00.js +177 -0
  111. package/dist/index-hkxtfqtc.js +333 -0
  112. package/dist/index-k370bech.js +72 -0
  113. package/dist/index-kf3dhser.js +146 -143
  114. package/dist/index-ma6tgdb2.js +500 -0
  115. package/dist/index-mam0bcyz.js +123 -0
  116. package/dist/index-mm412dkp.js +274 -0
  117. package/dist/index-n8v18aeb.js +0 -0
  118. package/dist/index-ndnmnsej.js +378 -371
  119. package/dist/index-p8wty0e2.js +389 -379
  120. package/dist/index-qa8akv6y.js +666 -0
  121. package/dist/index-qfphr2fd.js +78 -76
  122. package/dist/index-qqmms8rs.js +51 -43
  123. package/dist/index-qw4093g2.js +51 -43
  124. package/dist/index-qzwpzjbx.js +121 -122
  125. package/dist/index-segbnm0h.js +146 -143
  126. package/dist/index-t0fj6gg1.js +112 -0
  127. package/dist/index-thdkwnv7.js +122 -0
  128. package/dist/index-tjbx2r2t.js +270 -0
  129. package/dist/index-tjqw9vtj.js +62 -54
  130. package/dist/index-vbpb89jy.js +248 -0
  131. package/dist/index-vg55rq0y.js +250 -0
  132. package/dist/index-vhs88xhe.js +99 -95
  133. package/dist/index-vs81yaks.js +244 -0
  134. package/dist/index-w8zxnjka.js +249 -0
  135. package/dist/index-wk2na3t9.js +385 -375
  136. package/dist/index-wz9x8g7z.js +383 -373
  137. package/dist/index-x249gyde.js +388 -378
  138. package/dist/index-x54nbgs7.js +355 -0
  139. package/dist/index-xkvd0nsd.js +187 -0
  140. package/dist/index-yedqxm1z.js +80 -0
  141. package/dist/index-yz4jfz7z.js +338 -0
  142. package/dist/index-zfjzzjkf.js +240 -199
  143. package/dist/index.d.ts +12 -8
  144. package/dist/index.js +56 -34
  145. package/dist/lint.d.ts +1 -46
  146. package/dist/lint.js +3 -7
  147. package/dist/loader/cache.d.ts +4 -0
  148. package/dist/loader/find-config-file.d.ts +2 -0
  149. package/dist/loader/index.d.ts +5 -0
  150. package/dist/loader/index.js +24 -0
  151. package/dist/loader/load-dev-env.d.ts +5 -0
  152. package/dist/loader/loader.d.ts +1 -0
  153. package/dist/loader.d.ts +1 -45
  154. package/dist/loader.js +22 -20
  155. package/dist/prisma/index.d.ts +1 -0
  156. package/dist/prisma/prisma.d.ts +29 -0
  157. package/dist/prisma.d.ts +1 -29
  158. package/dist/prisma.js +6 -10
  159. package/dist/src/bin.js +309 -0
  160. package/dist/src/cli.js +5 -0
  161. package/dist/src/config.js +15 -0
  162. package/dist/src/core/docker.js +38 -0
  163. package/dist/src/core/index.js +130 -0
  164. package/dist/src/core/network.js +9 -0
  165. package/dist/src/core/ports.js +23 -0
  166. package/dist/src/core/process.js +31 -0
  167. package/dist/src/core/utils.js +11 -0
  168. package/dist/src/core/watchdog-runner.js +69 -0
  169. package/dist/src/core/watchdog.js +28 -0
  170. package/dist/src/docker/runtime.js +37 -0
  171. package/dist/src/docker-compose/index.js +16 -0
  172. package/dist/src/docker-compose/services/index.js +17 -0
  173. package/dist/src/environment.js +12 -0
  174. package/dist/src/index.js +122 -0
  175. package/dist/src/lint.js +3 -0
  176. package/dist/src/loader.js +25 -0
  177. package/dist/src/prisma.js +6 -0
  178. package/dist/src/types.js +0 -0
  179. package/dist/typecheck/index.d.ts +1 -0
  180. package/dist/typecheck/index.js +7 -0
  181. package/dist/typecheck/typecheck.d.ts +46 -0
  182. package/dist/types/all-types.d.ts +544 -0
  183. package/dist/types/cli.d.ts +1 -0
  184. package/dist/types/config.d.ts +6 -0
  185. package/dist/types/docker.d.ts +15 -0
  186. package/dist/types/environment.d.ts +8 -0
  187. package/dist/types/hooks.d.ts +9 -0
  188. package/dist/types/index.d.ts +1 -0
  189. package/dist/types/index.js +0 -0
  190. package/dist/types/prisma.d.ts +1 -0
  191. package/dist/types.d.ts +1 -399
  192. package/package.json +55 -48
  193. package/readme.md +365 -109
  194. package/src/cli/bin.ts +77 -0
  195. package/src/cli/commands/help.ts +39 -0
  196. package/src/cli/commands/runtime.ts +72 -0
  197. package/src/cli/commands/version.ts +4 -0
  198. package/src/cli/index.ts +1 -0
  199. package/{cli.ts → src/cli/run-cli.ts} +114 -10
  200. package/src/config/define-config.ts +30 -0
  201. package/src/config/index.ts +3 -0
  202. package/src/config/merge-configs.ts +33 -0
  203. package/src/config/validate-config.ts +136 -0
  204. package/{core → src/core}/index.ts +2 -2
  205. package/{core → src/core}/ports.ts +5 -2
  206. package/{core → src/core}/process.ts +6 -2
  207. package/src/core/quick-tunnel/cloudflared-process.ts +83 -0
  208. package/src/core/quick-tunnel/constants.ts +31 -0
  209. package/src/core/quick-tunnel/index.ts +96 -0
  210. package/src/core/quick-tunnel/install.ts +160 -0
  211. package/src/core/tunnel.ts +165 -0
  212. package/{core → src/core}/utils.ts +1 -0
  213. package/{core → src/core}/watchdog.ts +5 -1
  214. package/src/docker/index.ts +1 -0
  215. package/{core/docker.ts → src/docker/runtime.ts} +11 -4
  216. package/src/docker-compose/generated-file.ts +45 -0
  217. package/src/docker-compose/index.ts +7 -0
  218. package/src/docker-compose/model.ts +197 -0
  219. package/src/docker-compose/services/clickhouse.ts +79 -0
  220. package/src/docker-compose/services/define-docker-service.ts +109 -0
  221. package/src/docker-compose/services/index.ts +67 -0
  222. package/src/docker-compose/services/postgres.ts +60 -0
  223. package/src/docker-compose/services/redis.ts +48 -0
  224. package/src/docker-compose/services/shared.ts +79 -0
  225. package/src/docker-compose/yaml.ts +88 -0
  226. package/{environment.ts → src/environment/create-dev-environment.ts} +214 -141
  227. package/src/environment/index.ts +1 -0
  228. package/src/environment/logging.ts +115 -0
  229. package/src/environment/only-apps.ts +34 -0
  230. package/src/environment/seeding.ts +57 -0
  231. package/{index.ts → src/index.ts} +52 -20
  232. package/src/loader/cache.ts +23 -0
  233. package/src/loader/find-config-file.ts +29 -0
  234. package/src/loader/index.ts +17 -0
  235. package/src/loader/load-dev-env.ts +38 -0
  236. package/src/prisma/index.ts +1 -0
  237. package/{prisma.ts → src/prisma/prisma.ts} +4 -2
  238. package/src/typecheck/index.ts +1 -0
  239. package/{types.ts → src/types/all-types.ts} +186 -8
  240. package/src/types/index.ts +1 -0
  241. package/bin.ts +0 -192
  242. package/config.ts +0 -194
  243. package/loader.ts +0 -126
  244. /package/{core → src/core}/network.ts +0 -0
  245. /package/{core → src/core}/watchdog-runner.ts +0 -0
  246. /package/{lint.ts → src/typecheck/typecheck.ts} +0 -0
@@ -1,58 +1,66 @@
1
- import {
2
- createDevEnvironment
3
- } from "./index-p8wty0e2.js";
4
-
5
1
  // loader.ts
6
2
  import { existsSync } from "node:fs";
7
3
  import { dirname, join } from "node:path";
4
+ import { createDevEnvironment } from "./index-p8wty0e2.js";
5
+
8
6
  var CONFIG_FILES = [
9
- "dev.config.ts",
10
- "dev.config.js",
11
- "dev-tools.config.ts",
12
- "dev-tools.config.js"
7
+ "dev.config.ts",
8
+ "dev.config.js",
9
+ "dev-tools.config.ts",
10
+ "dev-tools.config.js",
13
11
  ];
14
12
  function findConfigFile(startDir) {
15
- let currentDir = startDir;
16
- while (true) {
17
- for (const file of CONFIG_FILES) {
18
- const configPath = join(currentDir, file);
19
- if (existsSync(configPath)) {
20
- return configPath;
21
- }
22
- }
23
- const parentDir = dirname(currentDir);
24
- if (parentDir === currentDir) {
25
- return null;
26
- }
27
- currentDir = parentDir;
28
- }
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
+ }
29
27
  }
30
28
  var cachedEnv = null;
31
29
  async function loadDevEnv(options) {
32
- if (cachedEnv && !options?.reload) {
33
- return cachedEnv;
34
- }
35
- const cwd = options?.cwd ?? process.cwd();
36
- const configPath = findConfigFile(cwd);
37
- if (configPath) {
38
- const mod = await import(configPath);
39
- const config = mod.default;
40
- if (!config?.projectPrefix || !config?.services) {
41
- throw new Error(`Invalid config in "${configPath}". Use defineDevConfig() and export as default.`);
42
- }
43
- cachedEnv = createDevEnvironment(config);
44
- return cachedEnv;
45
- }
46
- throw new Error(`No config file found. Create dev.config.ts with: export default defineDevConfig({ ... })`);
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
+ );
47
49
  }
48
50
  function getDevEnv() {
49
- if (!cachedEnv) {
50
- throw new Error("Dev environment not loaded. Call loadDevEnv() first.");
51
- }
52
- return cachedEnv;
51
+ if (!cachedEnv) {
52
+ throw new Error("Dev environment not loaded. Call loadDevEnv() first.");
53
+ }
54
+ return cachedEnv;
53
55
  }
54
56
  function clearDevEnvCache() {
55
- cachedEnv = null;
57
+ cachedEnv = null;
56
58
  }
57
59
 
58
- export { CONFIG_FILES, findConfigFile, loadDevEnv, getDevEnv, clearDevEnvCache };
60
+ export {
61
+ CONFIG_FILES,
62
+ findConfigFile,
63
+ loadDevEnv,
64
+ getDevEnv,
65
+ clearDevEnvCache,
66
+ };
@@ -0,0 +1,145 @@
1
+ // src/core/network.ts
2
+ import { networkInterfaces } from "node:os";
3
+ import {
4
+ calculatePortOffset,
5
+ computePorts,
6
+ computeUrls,
7
+ } from "./index-xkvd0nsd.js";
8
+
9
+ // src/core/utils.ts
10
+ function sleep(ms) {
11
+ return new Promise((resolve) => setTimeout(resolve, ms));
12
+ }
13
+ function isCI() {
14
+ return (
15
+ process.env.CI === "true" ||
16
+ process.env.CI === "1" ||
17
+ process.env.GITHUB_ACTIONS === "true" ||
18
+ process.env.GITLAB_CI === "true" ||
19
+ process.env.CIRCLECI === "true" ||
20
+ process.env.JENKINS_URL !== undefined
21
+ );
22
+ }
23
+ function logFrontendPort(port) {
24
+ console.log(`using_frontend_port:${port}`);
25
+ }
26
+ function getEnvVar(config, name, options = {}) {
27
+ const { log = true } = options;
28
+ const offset = calculatePortOffset();
29
+ const localIp = getLocalIp();
30
+ const ports = computePorts(config.services, config.apps, offset);
31
+ const urls = computeUrls(config.services, config.apps, ports, localIp);
32
+ const envVars = config.envVars?.(ports, urls, {
33
+ projectName: config.projectPrefix,
34
+ localIp,
35
+ portOffset: offset,
36
+ });
37
+ const value = envVars?.[name];
38
+ if (log && name === "VITE_PORT" && typeof value === "number") {
39
+ logFrontendPort(value);
40
+ }
41
+ return value;
42
+ }
43
+ function logApiUrl(url) {
44
+ console.log(`using_api_url:${url}`);
45
+ }
46
+ function logExpoApiUrl(url) {
47
+ console.log(`using_expo_api_url:${url}`);
48
+ }
49
+
50
+ // src/core/network.ts
51
+ function getLocalIp() {
52
+ const interfaces = networkInterfaces();
53
+ for (const name of Object.keys(interfaces)) {
54
+ const nets = interfaces[name];
55
+ if (!nets) continue;
56
+ for (const net of nets) {
57
+ if (net.family === "IPv4" && !net.internal) {
58
+ return net.address;
59
+ }
60
+ }
61
+ }
62
+ return "127.0.0.1";
63
+ }
64
+ async function waitForServer(url, options = {}) {
65
+ const { timeout = 30000, interval = 2000, verbose = false } = options;
66
+ const start = Date.now();
67
+ let attempts = 0;
68
+ while (Date.now() - start < timeout) {
69
+ attempts++;
70
+ const controller = new AbortController();
71
+ const timeoutId = setTimeout(() => controller.abort(), 5000);
72
+ try {
73
+ const response = await fetch(url, {
74
+ signal: controller.signal,
75
+ });
76
+ clearTimeout(timeoutId);
77
+ if (response.ok || response.status === 404) {
78
+ if (verbose) {
79
+ console.log(` ✓ ${url} ready after ${attempts} attempts`);
80
+ }
81
+ return;
82
+ }
83
+ } catch {
84
+ clearTimeout(timeoutId);
85
+ if (verbose && attempts % 5 === 0) {
86
+ console.log(
87
+ ` ⏳ Waiting for ${url}... (${Math.round((Date.now() - start) / 1000)}s)`,
88
+ );
89
+ }
90
+ }
91
+ await sleep(interval);
92
+ }
93
+ throw new Error(
94
+ `Server at ${url} did not respond within ${timeout}ms after ${attempts} attempts`,
95
+ );
96
+ }
97
+ async function waitForDevServers(apps, ports, options = {}) {
98
+ const { timeout = 60000, verbose = true } = options;
99
+ if (verbose) console.log("⏳ Waiting for servers to be ready...");
100
+ const promises = [];
101
+ for (const [name, config] of Object.entries(apps)) {
102
+ const port = ports[name];
103
+ const healthPath = config.healthEndpoint ?? "/";
104
+ const url = `http://localhost:${port}${healthPath}`;
105
+ const appTimeout = config.healthTimeout ?? timeout;
106
+ promises.push(waitForServer(url, { timeout: appTimeout, verbose }));
107
+ }
108
+ await Promise.all(promises);
109
+ if (verbose) console.log("✓ All servers ready");
110
+ }
111
+ async function isPortAvailable(port) {
112
+ const controller = new AbortController();
113
+ const timeoutId = setTimeout(() => controller.abort(), 500);
114
+ try {
115
+ const _response = await fetch(`http://localhost:${port}/`, {
116
+ signal: controller.signal,
117
+ });
118
+ clearTimeout(timeoutId);
119
+ return false;
120
+ } catch (error) {
121
+ clearTimeout(timeoutId);
122
+ if (error instanceof Error) {
123
+ if (
124
+ error.message.includes("ECONNREFUSED") ||
125
+ error.message.includes("fetch failed")
126
+ ) {
127
+ return true;
128
+ }
129
+ }
130
+ return true;
131
+ }
132
+ }
133
+
134
+ export {
135
+ getLocalIp,
136
+ waitForServer,
137
+ waitForDevServers,
138
+ isPortAvailable,
139
+ sleep,
140
+ isCI,
141
+ logFrontendPort,
142
+ getEnvVar,
143
+ logApiUrl,
144
+ logExpoApiUrl,
145
+ };
@@ -0,0 +1,72 @@
1
+ import {
2
+ createDevEnvironment
3
+ } from "./index-ma6tgdb2.js";
4
+
5
+ // src/loader/cache.ts
6
+ var cachedEnv = null;
7
+ function setCachedDevEnv(env) {
8
+ cachedEnv = env;
9
+ }
10
+ function getCachedDevEnv() {
11
+ return cachedEnv;
12
+ }
13
+ function clearDevEnvCache() {
14
+ cachedEnv = null;
15
+ }
16
+ // src/loader/find-config-file.ts
17
+ import { existsSync } from "node:fs";
18
+ import { dirname, join } from "node:path";
19
+ var CONFIG_FILES = [
20
+ "dev.config.ts",
21
+ "dev.config.js",
22
+ "dev-tools.config.ts",
23
+ "dev-tools.config.js"
24
+ ];
25
+ function findConfigFile(startDir) {
26
+ let currentDir = startDir;
27
+ while (true) {
28
+ for (const file of CONFIG_FILES) {
29
+ const configPath = join(currentDir, file);
30
+ if (existsSync(configPath)) {
31
+ return configPath;
32
+ }
33
+ }
34
+ const parentDir = dirname(currentDir);
35
+ if (parentDir === currentDir) {
36
+ return null;
37
+ }
38
+ currentDir = parentDir;
39
+ }
40
+ }
41
+ // src/loader/load-dev-env.ts
42
+ async function loadDevEnv(options) {
43
+ if (!options?.reload) {
44
+ const cached = getCachedDevEnv();
45
+ if (cached)
46
+ return cached;
47
+ }
48
+ const cwd = options?.cwd ?? process.cwd();
49
+ const configPath = findConfigFile(cwd);
50
+ if (configPath) {
51
+ const mod = await import(configPath);
52
+ const config = mod.default;
53
+ if (!config?.projectPrefix || !config?.services) {
54
+ throw new Error(`Invalid config in "${configPath}". Use defineDevConfig() and export as default.`);
55
+ }
56
+ const env = createDevEnvironment(config);
57
+ setCachedDevEnv(env);
58
+ return env;
59
+ }
60
+ throw new Error("No config file found. Create dev.config.ts with: export default defineDevConfig({ ... })");
61
+ }
62
+
63
+ // src/loader/index.ts
64
+ function getDevEnv() {
65
+ const env = getCachedDevEnv();
66
+ if (!env) {
67
+ throw new Error("Dev environment not loaded. Call loadDevEnv() first.");
68
+ }
69
+ return env;
70
+ }
71
+
72
+ export { clearDevEnvCache, CONFIG_FILES, findConfigFile, loadDevEnv, getDevEnv };
@@ -0,0 +1,250 @@
1
+ import {
2
+ resolveExposeTargets,
3
+ startPublicTunnels,
4
+ stopPublicTunnels
5
+ } from "./index-x54nbgs7.js";
6
+ import {
7
+ spawnWatchdog,
8
+ startHeartbeat,
9
+ stopHeartbeat
10
+ } from "./index-mam0bcyz.js";
11
+ import {
12
+ killProcessesOnAppPorts
13
+ } from "./index-mm412dkp.js";
14
+
15
+ // src/cli/run-cli.ts
16
+ import { spawn } from "node:child_process";
17
+ var ACCEPTED_FLAGS = [
18
+ "--help",
19
+ "--down",
20
+ "--reset",
21
+ "--migrate",
22
+ "--seed",
23
+ "--up-only",
24
+ "--expose"
25
+ ];
26
+ function printHelp() {
27
+ console.log(`
28
+ Usage: buncargo dev [options]
29
+
30
+ Options:
31
+ --help Show this help message
32
+ --down Stop all containers
33
+ --reset Stop containers and remove volumes (fresh start)
34
+ --migrate Run migrations and exit
35
+ --seed Run migrations and seeders, then exit
36
+ --up-only Start containers and run migrations, then exit (no dev servers)
37
+ --expose Expose configured targets via public quick tunnels
38
+
39
+ Examples:
40
+ bun dev Start dev environment with all services
41
+ bun dev --seed Run migrations and seed the database
42
+ bun dev --down Stop all containers
43
+ bun dev --reset Stop containers and remove all data
44
+ bun dev --expose Expose all targets with expose: true
45
+ bun dev --expose=api,web Expose specific targets
46
+ `);
47
+ }
48
+ function getUnknownFlags(args) {
49
+ return args.filter((arg) => arg.startsWith("--") && !ACCEPTED_FLAGS.includes(arg.includes("=") ? arg.split("=")[0] : arg));
50
+ }
51
+ async function runCli(env, options = {}) {
52
+ const {
53
+ args = process.argv.slice(2),
54
+ watchdog = true,
55
+ watchdogTimeout = 10,
56
+ devServersCommand,
57
+ cliTestTunnel
58
+ } = options;
59
+ const tunnelApi = cliTestTunnel ?? {
60
+ resolveExposeTargets,
61
+ startPublicTunnels,
62
+ stopPublicTunnels
63
+ };
64
+ const exposeRequested = hasFlag(args, "--expose");
65
+ const exposeValue = getFlagValue(args, "--expose");
66
+ let tunnels = [];
67
+ async function cleanupTunnels() {
68
+ env.clearPublicUrls();
69
+ if (tunnels.length === 0)
70
+ return;
71
+ await tunnelApi.stopPublicTunnels(tunnels);
72
+ tunnels = [];
73
+ }
74
+ if (args.includes("--help")) {
75
+ printHelp();
76
+ process.exit(0);
77
+ }
78
+ const unknownFlags = getUnknownFlags(args);
79
+ if (unknownFlags.length > 0) {
80
+ console.error(`❌ Unknown flag${unknownFlags.length > 1 ? "s" : ""}: ${unknownFlags.join(", ")}`);
81
+ console.error("");
82
+ printHelp();
83
+ process.exit(1);
84
+ }
85
+ if (args.includes("--down")) {
86
+ env.logInfo();
87
+ await cleanupTunnels();
88
+ await env.stop();
89
+ process.exit(0);
90
+ }
91
+ if (args.includes("--reset")) {
92
+ env.logInfo();
93
+ await cleanupTunnels();
94
+ await env.stop({ removeVolumes: true });
95
+ process.exit(0);
96
+ }
97
+ const skipSeed = args.includes("--seed");
98
+ await env.start({
99
+ startServers: false,
100
+ wait: true,
101
+ skipSeed,
102
+ skipEnvironmentLog: exposeRequested
103
+ });
104
+ if (exposeRequested) {
105
+ const { targets, unknownNames, notEnabledNames } = tunnelApi.resolveExposeTargets(env, exposeValue);
106
+ if (unknownNames.length > 0) {
107
+ console.error(`❌ Unknown expose target${unknownNames.length > 1 ? "s" : ""}: ${unknownNames.join(", ")}`);
108
+ await cleanupTunnels();
109
+ process.exit(1);
110
+ }
111
+ if (notEnabledNames.length > 0) {
112
+ console.error(`❌ Target${notEnabledNames.length > 1 ? "s" : ""} missing expose: true: ${notEnabledNames.join(", ")}`);
113
+ console.error(" Mark these in dev.config.ts with expose: true or remove them from --expose.");
114
+ await cleanupTunnels();
115
+ process.exit(1);
116
+ }
117
+ if (targets.length === 0) {
118
+ console.error("❌ No expose targets selected. Add expose: true to services/apps or pass names with --expose=<name>.");
119
+ await cleanupTunnels();
120
+ process.exit(1);
121
+ }
122
+ tunnels = await tunnelApi.startPublicTunnels(targets);
123
+ env.setPublicUrls(Object.fromEntries(tunnels.map((tunnel) => [tunnel.name, tunnel.publicUrl])));
124
+ env.logInfo("Dev Environment", tunnels);
125
+ }
126
+ if (args.includes("--migrate")) {
127
+ console.log("");
128
+ console.log("✅ Migrations applied successfully");
129
+ await cleanupTunnels();
130
+ process.exit(0);
131
+ }
132
+ if (args.includes("--seed")) {
133
+ console.log("\uD83C\uDF31 Running seeders...");
134
+ const result = await env.exec("bun run run:seeder", {
135
+ throwOnError: false
136
+ });
137
+ if (result.exitCode !== 0) {
138
+ console.error("❌ Seeding failed");
139
+ if (result.stderr) {
140
+ console.error(result.stderr);
141
+ }
142
+ if (result.stdout) {
143
+ console.error(result.stdout);
144
+ }
145
+ await cleanupTunnels();
146
+ process.exit(1);
147
+ }
148
+ console.log("");
149
+ console.log("✅ Seeding complete");
150
+ await cleanupTunnels();
151
+ process.exit(0);
152
+ }
153
+ if (args.includes("--up-only")) {
154
+ console.log("");
155
+ console.log("✅ Containers started. Environment ready.");
156
+ console.log("");
157
+ await cleanupTunnels();
158
+ process.exit(0);
159
+ }
160
+ if (watchdog) {
161
+ await spawnWatchdog(env.projectName, env.root, {
162
+ timeoutMinutes: watchdogTimeout,
163
+ verbose: true,
164
+ composeFile: env.composeFile
165
+ });
166
+ startHeartbeat(env.projectName);
167
+ }
168
+ const command = devServersCommand ?? buildDevServersCommand(env.apps);
169
+ if (!command) {
170
+ console.log("✅ Containers ready. No apps configured.");
171
+ await new Promise(() => {});
172
+ await cleanupTunnels();
173
+ return;
174
+ }
175
+ await killProcessesOnAppPorts(env.apps, env.ports);
176
+ console.log("");
177
+ console.log("\uD83D\uDD27 Starting dev servers...");
178
+ console.log("");
179
+ await runCommand(command, env.root, env.buildEnvVars(), {
180
+ onSignal: async () => {
181
+ await cleanupTunnels();
182
+ stopHeartbeat();
183
+ }
184
+ });
185
+ stopHeartbeat();
186
+ await cleanupTunnels();
187
+ }
188
+ function buildDevServersCommand(apps) {
189
+ const appEntries = Object.entries(apps);
190
+ if (appEntries.length === 0)
191
+ return null;
192
+ const commands = [];
193
+ const names = [];
194
+ const colors = ["blue", "green", "yellow", "magenta", "cyan", "red"];
195
+ for (const [name, config] of appEntries) {
196
+ names.push(name);
197
+ const cwdPart = config.cwd ? `--cwd ${config.cwd}` : "";
198
+ commands.push(`"bun run ${cwdPart} ${config.devCommand}"`.replace(/\s+/g, " ").trim());
199
+ }
200
+ const namesArg = `-n ${names.join(",")}`;
201
+ const colorsArg = `-c ${colors.slice(0, names.length).join(",")}`;
202
+ const commandsArg = commands.join(" ");
203
+ return `bun concurrently ${namesArg} ${colorsArg} ${commandsArg}`;
204
+ }
205
+ function runCommand(command, cwd, envVars, options = {}) {
206
+ const { onSignal } = options;
207
+ return new Promise((resolve, reject) => {
208
+ const proc = spawn(command, [], {
209
+ cwd,
210
+ env: { ...process.env, ...envVars },
211
+ stdio: "inherit",
212
+ shell: true
213
+ });
214
+ proc.on("close", (code) => {
215
+ if (code === 0 || code === null) {
216
+ resolve();
217
+ } else {
218
+ reject(new Error(`Command exited with code ${code}`));
219
+ }
220
+ });
221
+ proc.on("error", reject);
222
+ const cleanup = () => {
223
+ if (onSignal) {
224
+ onSignal();
225
+ }
226
+ proc.kill("SIGTERM");
227
+ };
228
+ process.on("SIGINT", cleanup);
229
+ process.on("SIGTERM", cleanup);
230
+ });
231
+ }
232
+ function hasFlag(args, flag) {
233
+ return args.some((arg) => arg === flag || arg.startsWith(`${flag}=`));
234
+ }
235
+ function getFlagValue(args, flag) {
236
+ const prefixed = args.find((arg) => arg.startsWith(`${flag}=`));
237
+ if (prefixed) {
238
+ return prefixed.split("=")[1];
239
+ }
240
+ const index = args.indexOf(flag);
241
+ if (index !== -1 && index + 1 < args.length) {
242
+ const nextArg = args[index + 1];
243
+ if (nextArg !== undefined && !nextArg.startsWith("-")) {
244
+ return nextArg;
245
+ }
246
+ }
247
+ return;
248
+ }
249
+
250
+ export { runCli, hasFlag, getFlagValue };