@supatype/cli 0.1.0-alpha.6 → 0.1.0-alpha.8

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 (350) hide show
  1. package/.turbo/turbo-build.log +1 -1
  2. package/.turbo/turbo-test.log +208 -1
  3. package/.turbo/turbo-typecheck.log +1 -1
  4. package/dist/app/proxy-dev-app.d.ts +13 -0
  5. package/dist/app/proxy-dev-app.d.ts.map +1 -0
  6. package/dist/app/proxy-dev-app.js +53 -0
  7. package/dist/app/proxy-dev-app.js.map +1 -0
  8. package/dist/app-config.d.ts +7 -0
  9. package/dist/app-config.d.ts.map +1 -0
  10. package/dist/app-config.js +113 -0
  11. package/dist/app-config.js.map +1 -0
  12. package/dist/augmentation-generator.d.ts +2 -0
  13. package/dist/augmentation-generator.d.ts.map +1 -0
  14. package/dist/augmentation-generator.js +111 -0
  15. package/dist/augmentation-generator.js.map +1 -0
  16. package/dist/binary-cache.d.ts +94 -0
  17. package/dist/binary-cache.d.ts.map +1 -0
  18. package/dist/binary-cache.js +669 -0
  19. package/dist/binary-cache.js.map +1 -0
  20. package/dist/cli.d.ts.map +1 -1
  21. package/dist/cli.js +13 -7
  22. package/dist/cli.js.map +1 -1
  23. package/dist/commands/admin.d.ts.map +1 -1
  24. package/dist/commands/admin.js +4 -3
  25. package/dist/commands/admin.js.map +1 -1
  26. package/dist/commands/app.d.ts.map +1 -1
  27. package/dist/commands/app.js +56 -209
  28. package/dist/commands/app.js.map +1 -1
  29. package/dist/commands/cache.d.ts +6 -0
  30. package/dist/commands/cache.d.ts.map +1 -0
  31. package/dist/commands/cache.js +105 -0
  32. package/dist/commands/cache.js.map +1 -0
  33. package/dist/commands/cloud.d.ts +20 -0
  34. package/dist/commands/cloud.d.ts.map +1 -1
  35. package/dist/commands/cloud.js +50 -52
  36. package/dist/commands/cloud.js.map +1 -1
  37. package/dist/commands/db.d.ts.map +1 -1
  38. package/dist/commands/db.js +47 -54
  39. package/dist/commands/db.js.map +1 -1
  40. package/dist/commands/deploy.d.ts +2 -1
  41. package/dist/commands/deploy.d.ts.map +1 -1
  42. package/dist/commands/deploy.js +79 -52
  43. package/dist/commands/deploy.js.map +1 -1
  44. package/dist/commands/dev.d.ts +11 -0
  45. package/dist/commands/dev.d.ts.map +1 -1
  46. package/dist/commands/dev.js +759 -385
  47. package/dist/commands/dev.js.map +1 -1
  48. package/dist/commands/diff.d.ts.map +1 -1
  49. package/dist/commands/diff.js +30 -15
  50. package/dist/commands/diff.js.map +1 -1
  51. package/dist/commands/engine.d.ts +1 -3
  52. package/dist/commands/engine.d.ts.map +1 -1
  53. package/dist/commands/engine.js +13 -85
  54. package/dist/commands/engine.js.map +1 -1
  55. package/dist/commands/functions.d.ts.map +1 -1
  56. package/dist/commands/functions.js +92 -105
  57. package/dist/commands/functions.js.map +1 -1
  58. package/dist/commands/generate.d.ts.map +1 -1
  59. package/dist/commands/generate.js +22 -12
  60. package/dist/commands/generate.js.map +1 -1
  61. package/dist/commands/init.d.ts +1 -1
  62. package/dist/commands/init.d.ts.map +1 -1
  63. package/dist/commands/init.js +137 -410
  64. package/dist/commands/init.js.map +1 -1
  65. package/dist/commands/migrate-from-v1.d.ts +5 -0
  66. package/dist/commands/migrate-from-v1.d.ts.map +1 -0
  67. package/dist/commands/migrate-from-v1.js +125 -0
  68. package/dist/commands/migrate-from-v1.js.map +1 -0
  69. package/dist/commands/migrate.d.ts.map +1 -1
  70. package/dist/commands/migrate.js +27 -23
  71. package/dist/commands/migrate.js.map +1 -1
  72. package/dist/commands/pg.d.ts +8 -0
  73. package/dist/commands/pg.d.ts.map +1 -0
  74. package/dist/commands/pg.js +102 -0
  75. package/dist/commands/pg.js.map +1 -0
  76. package/dist/commands/pull.d.ts.map +1 -1
  77. package/dist/commands/pull.js +5 -66
  78. package/dist/commands/pull.js.map +1 -1
  79. package/dist/commands/push.d.ts.map +1 -1
  80. package/dist/commands/push.js +128 -38
  81. package/dist/commands/push.js.map +1 -1
  82. package/dist/commands/seed.d.ts +2 -0
  83. package/dist/commands/seed.d.ts.map +1 -1
  84. package/dist/commands/seed.js +44 -11
  85. package/dist/commands/seed.js.map +1 -1
  86. package/dist/commands/self-host.d.ts +7 -1
  87. package/dist/commands/self-host.d.ts.map +1 -1
  88. package/dist/commands/self-host.js +272 -758
  89. package/dist/commands/self-host.js.map +1 -1
  90. package/dist/commands/self-update.d.ts +9 -0
  91. package/dist/commands/self-update.d.ts.map +1 -0
  92. package/dist/commands/self-update.js +33 -0
  93. package/dist/commands/self-update.js.map +1 -0
  94. package/dist/commands/status.d.ts.map +1 -1
  95. package/dist/commands/status.js +4 -3
  96. package/dist/commands/status.js.map +1 -1
  97. package/dist/commands/types.d.ts +3 -0
  98. package/dist/commands/types.d.ts.map +1 -0
  99. package/dist/commands/types.js +62 -0
  100. package/dist/commands/types.js.map +1 -0
  101. package/dist/commands/update.d.ts +7 -0
  102. package/dist/commands/update.d.ts.map +1 -0
  103. package/dist/commands/update.js +93 -0
  104. package/dist/commands/update.js.map +1 -0
  105. package/dist/components.d.ts +5 -0
  106. package/dist/components.d.ts.map +1 -0
  107. package/dist/components.js +3 -0
  108. package/dist/components.js.map +1 -0
  109. package/dist/config.d.ts +10 -51
  110. package/dist/config.d.ts.map +1 -1
  111. package/dist/config.js +101 -33
  112. package/dist/config.js.map +1 -1
  113. package/dist/dev-compose.d.ts +17 -0
  114. package/dist/dev-compose.d.ts.map +1 -0
  115. package/dist/dev-compose.js +374 -0
  116. package/dist/dev-compose.js.map +1 -0
  117. package/dist/diff-output.d.ts +4 -0
  118. package/dist/diff-output.d.ts.map +1 -0
  119. package/dist/diff-output.js +12 -0
  120. package/dist/diff-output.js.map +1 -0
  121. package/dist/docker-postgres.d.ts +57 -0
  122. package/dist/docker-postgres.d.ts.map +1 -0
  123. package/dist/docker-postgres.js +208 -0
  124. package/dist/docker-postgres.js.map +1 -0
  125. package/dist/engine-client.d.ts +69 -0
  126. package/dist/engine-client.d.ts.map +1 -0
  127. package/dist/engine-client.js +157 -0
  128. package/dist/engine-client.js.map +1 -0
  129. package/dist/ensure-binary.d.ts +7 -0
  130. package/dist/ensure-binary.d.ts.map +1 -0
  131. package/dist/ensure-binary.js +17 -0
  132. package/dist/ensure-binary.js.map +1 -0
  133. package/dist/functions-router-gen.d.ts +14 -0
  134. package/dist/functions-router-gen.d.ts.map +1 -0
  135. package/dist/functions-router-gen.js +199 -0
  136. package/dist/functions-router-gen.js.map +1 -0
  137. package/dist/index.d.ts +4 -5
  138. package/dist/index.d.ts.map +1 -1
  139. package/dist/index.js +2 -3
  140. package/dist/index.js.map +1 -1
  141. package/dist/kong-config.d.ts +25 -0
  142. package/dist/kong-config.d.ts.map +1 -0
  143. package/dist/kong-config.js +71 -0
  144. package/dist/kong-config.js.map +1 -0
  145. package/dist/local-gateway.d.ts +7 -0
  146. package/dist/local-gateway.d.ts.map +1 -0
  147. package/dist/local-gateway.js +9 -0
  148. package/dist/local-gateway.js.map +1 -0
  149. package/dist/local-storage.d.ts +8 -0
  150. package/dist/local-storage.d.ts.map +1 -0
  151. package/dist/local-storage.js +14 -0
  152. package/dist/local-storage.js.map +1 -0
  153. package/dist/pgbouncer-userlist.d.ts +5 -0
  154. package/dist/pgbouncer-userlist.d.ts.map +1 -0
  155. package/dist/pgbouncer-userlist.js +14 -0
  156. package/dist/pgbouncer-userlist.js.map +1 -0
  157. package/dist/postgres-ctl.d.ts +44 -0
  158. package/dist/postgres-ctl.d.ts.map +1 -0
  159. package/dist/postgres-ctl.js +137 -0
  160. package/dist/postgres-ctl.js.map +1 -0
  161. package/dist/process-manager.d.ts +43 -0
  162. package/dist/process-manager.d.ts.map +1 -0
  163. package/dist/process-manager.js +135 -0
  164. package/dist/process-manager.js.map +1 -0
  165. package/dist/project-config.d.ts +235 -0
  166. package/dist/project-config.d.ts.map +1 -0
  167. package/dist/project-config.js +160 -0
  168. package/dist/project-config.js.map +1 -0
  169. package/dist/pull-utils.d.ts +15 -0
  170. package/dist/pull-utils.d.ts.map +1 -1
  171. package/dist/pull-utils.js +12 -0
  172. package/dist/pull-utils.js.map +1 -1
  173. package/dist/release-pins.d.ts +7 -0
  174. package/dist/release-pins.d.ts.map +1 -0
  175. package/dist/release-pins.js +27 -0
  176. package/dist/release-pins.js.map +1 -0
  177. package/dist/release-public-key.d.ts +8 -0
  178. package/dist/release-public-key.d.ts.map +1 -0
  179. package/dist/release-public-key.js +13 -0
  180. package/dist/release-public-key.js.map +1 -0
  181. package/dist/runtime-routes.d.ts +34 -0
  182. package/dist/runtime-routes.d.ts.map +1 -0
  183. package/dist/runtime-routes.js +252 -0
  184. package/dist/runtime-routes.js.map +1 -0
  185. package/dist/schema-ast-v2.d.ts +127 -0
  186. package/dist/schema-ast-v2.d.ts.map +1 -0
  187. package/dist/schema-ast-v2.js +226 -0
  188. package/dist/schema-ast-v2.js.map +1 -0
  189. package/dist/scripts/postinstall.d.ts +5 -6
  190. package/dist/scripts/postinstall.d.ts.map +1 -1
  191. package/dist/scripts/postinstall.js +36 -20
  192. package/dist/scripts/postinstall.js.map +1 -1
  193. package/dist/self-host-compose.d.ts +22 -0
  194. package/dist/self-host-compose.d.ts.map +1 -0
  195. package/dist/self-host-compose.js +347 -0
  196. package/dist/self-host-compose.js.map +1 -0
  197. package/dist/storage-provision.d.ts +24 -0
  198. package/dist/storage-provision.d.ts.map +1 -0
  199. package/dist/storage-provision.js +44 -0
  200. package/dist/storage-provision.js.map +1 -0
  201. package/dist/studio-admin-roles.d.ts +7 -0
  202. package/dist/studio-admin-roles.d.ts.map +1 -0
  203. package/dist/studio-admin-roles.js +14 -0
  204. package/dist/studio-admin-roles.js.map +1 -0
  205. package/dist/studio-dev-server.d.ts +22 -0
  206. package/dist/studio-dev-server.d.ts.map +1 -0
  207. package/dist/studio-dev-server.js +28 -0
  208. package/dist/studio-dev-server.js.map +1 -0
  209. package/dist/systemd.d.ts +26 -0
  210. package/dist/systemd.d.ts.map +1 -0
  211. package/dist/systemd.js +102 -0
  212. package/dist/systemd.js.map +1 -0
  213. package/dist/tsx-runner.d.ts.map +1 -1
  214. package/dist/tsx-runner.js +9 -2
  215. package/dist/tsx-runner.js.map +1 -1
  216. package/dist/type-extractor.d.ts +4 -0
  217. package/dist/type-extractor.d.ts.map +1 -0
  218. package/dist/type-extractor.js +1213 -0
  219. package/dist/type-extractor.js.map +1 -0
  220. package/dist/type-resolver.d.ts +33 -0
  221. package/dist/type-resolver.d.ts.map +1 -0
  222. package/dist/type-resolver.js +338 -0
  223. package/dist/type-resolver.js.map +1 -0
  224. package/package.json +4 -3
  225. package/releases/deno/VERSION +1 -0
  226. package/scripts/mirror-deno-release.sh +76 -0
  227. package/src/TYPE-RESOLUTION.md +294 -0
  228. package/src/app/proxy-dev-app.ts +67 -0
  229. package/src/app-config.ts +128 -0
  230. package/src/augmentation-generator.ts +126 -0
  231. package/src/binary-cache.ts +822 -0
  232. package/src/cli.ts +13 -8
  233. package/src/commands/admin.ts +4 -3
  234. package/src/commands/app.ts +67 -231
  235. package/src/commands/cache.ts +117 -0
  236. package/src/commands/cloud.ts +63 -64
  237. package/src/commands/db.ts +54 -63
  238. package/src/commands/deploy.ts +96 -62
  239. package/src/commands/dev.ts +933 -405
  240. package/src/commands/diff.ts +31 -29
  241. package/src/commands/engine.ts +13 -116
  242. package/src/commands/functions.ts +97 -115
  243. package/src/commands/generate.ts +23 -10
  244. package/src/commands/init.ts +149 -414
  245. package/src/commands/migrate-from-v1.ts +131 -0
  246. package/src/commands/migrate.ts +27 -23
  247. package/src/commands/pg.ts +133 -0
  248. package/src/commands/pull.ts +6 -85
  249. package/src/commands/push.ts +161 -56
  250. package/src/commands/seed.ts +54 -12
  251. package/src/commands/self-host.ts +312 -880
  252. package/src/commands/self-update.ts +45 -0
  253. package/src/commands/status.ts +4 -3
  254. package/src/commands/types.ts +76 -0
  255. package/src/commands/update.ts +109 -0
  256. package/src/components.ts +6 -0
  257. package/src/config.ts +127 -94
  258. package/src/dev-compose.ts +455 -0
  259. package/src/diff-output.ts +12 -0
  260. package/src/docker-postgres.ts +295 -0
  261. package/src/engine-client.ts +236 -0
  262. package/src/ensure-binary.ts +28 -0
  263. package/src/functions-router-gen.ts +224 -0
  264. package/src/index.ts +4 -12
  265. package/src/kong-config.ts +93 -0
  266. package/src/local-gateway.ts +9 -0
  267. package/src/local-storage.ts +14 -0
  268. package/src/pgbouncer-userlist.ts +15 -0
  269. package/src/postgres-ctl.ts +171 -0
  270. package/src/process-manager.ts +168 -0
  271. package/src/project-config.ts +386 -0
  272. package/src/pull-utils.ts +24 -0
  273. package/src/release-pins.ts +31 -0
  274. package/src/release-public-key.ts +12 -0
  275. package/src/runtime-routes.ts +291 -0
  276. package/src/schema-ast-v2.ts +324 -0
  277. package/src/scripts/postinstall.ts +36 -25
  278. package/src/self-host-compose.ts +389 -0
  279. package/src/storage-provision.ts +58 -0
  280. package/src/studio-admin-roles.ts +16 -0
  281. package/src/studio-dev-server.ts +53 -0
  282. package/src/systemd.ts +137 -0
  283. package/src/tsx-runner.ts +11 -1
  284. package/src/type-extractor.ts +1479 -0
  285. package/src/type-resolver.ts +457 -0
  286. package/tests/app-command.test.ts +54 -0
  287. package/tests/augmentation-generator.test.ts +59 -0
  288. package/tests/binary-cache-cloud-overrides.test.ts +123 -0
  289. package/tests/cached-artifact-format.test.ts +84 -0
  290. package/tests/cli-help.test.ts +40 -14
  291. package/tests/config.test.ts +171 -37
  292. package/tests/docker-postgres.test.ts +39 -0
  293. package/tests/engine-distribution.test.ts +3 -3
  294. package/tests/ensure-binary.test.ts +59 -0
  295. package/tests/init.test.ts +28 -86
  296. package/tests/migrate-from-v1.test.ts +29 -0
  297. package/tests/normalize-admin-config.test.ts +48 -0
  298. package/tests/pg-spawn-env.test.ts +18 -0
  299. package/tests/postgres-archive-tag.test.ts +9 -0
  300. package/tests/proxy-dev-app.test.ts +33 -0
  301. package/tests/pull-utils.test.ts +36 -1
  302. package/tests/release-pins.test.ts +28 -0
  303. package/tests/runtime-contract.test.ts +351 -0
  304. package/tests/seed-discover.test.ts +31 -0
  305. package/tests/studio-admin-roles.test.ts +27 -0
  306. package/tests/tsconfig.json +9 -0
  307. package/tests/type-extractor.test.ts +985 -0
  308. package/tests/type-resolver.test.ts +59 -0
  309. package/tsconfig.tsbuildinfo +1 -1
  310. package/vitest.config.ts +12 -0
  311. package/dist/engine/cache.d.ts +0 -37
  312. package/dist/engine/cache.d.ts.map +0 -1
  313. package/dist/engine/cache.js +0 -121
  314. package/dist/engine/cache.js.map +0 -1
  315. package/dist/engine/download.d.ts +0 -19
  316. package/dist/engine/download.d.ts.map +0 -1
  317. package/dist/engine/download.js +0 -108
  318. package/dist/engine/download.js.map +0 -1
  319. package/dist/engine/platform.d.ts +0 -24
  320. package/dist/engine/platform.d.ts.map +0 -1
  321. package/dist/engine/platform.js +0 -50
  322. package/dist/engine/platform.js.map +0 -1
  323. package/dist/engine/resolve.d.ts +0 -37
  324. package/dist/engine/resolve.d.ts.map +0 -1
  325. package/dist/engine/resolve.js +0 -133
  326. package/dist/engine/resolve.js.map +0 -1
  327. package/dist/engine/update-notify.d.ts +0 -11
  328. package/dist/engine/update-notify.d.ts.map +0 -1
  329. package/dist/engine/update-notify.js +0 -43
  330. package/dist/engine/update-notify.js.map +0 -1
  331. package/dist/engine/verify.d.ts +0 -50
  332. package/dist/engine/verify.d.ts.map +0 -1
  333. package/dist/engine/verify.js +0 -161
  334. package/dist/engine/verify.js.map +0 -1
  335. package/dist/engine-version.d.ts +0 -35
  336. package/dist/engine-version.d.ts.map +0 -1
  337. package/dist/engine-version.js +0 -35
  338. package/dist/engine-version.js.map +0 -1
  339. package/dist/engine.d.ts +0 -34
  340. package/dist/engine.d.ts.map +0 -1
  341. package/dist/engine.js +0 -76
  342. package/dist/engine.js.map +0 -1
  343. package/src/engine/cache.ts +0 -135
  344. package/src/engine/download.ts +0 -143
  345. package/src/engine/platform.ts +0 -66
  346. package/src/engine/resolve.ts +0 -197
  347. package/src/engine/update-notify.ts +0 -50
  348. package/src/engine/verify.ts +0 -206
  349. package/src/engine-version.ts +0 -39
  350. package/src/engine.ts +0 -99
@@ -0,0 +1,224 @@
1
+ /**
2
+ * Generates a Deno router entrypoint so `deno run` (optionally `deno run --watch` in dev)
3
+ * can serve multiple handlers (same routing contract as `supatype functions serve`).
4
+ */
5
+ import {
6
+ basename,
7
+ join,
8
+ relative,
9
+ dirname,
10
+ resolve as pathResolve,
11
+ } from "node:path"
12
+ import { existsSync, readdirSync, statSync, mkdirSync, writeFileSync } from "node:fs"
13
+
14
+ export interface DiscoveredFunctionRoute {
15
+ name: string
16
+ entrypoint: string
17
+ }
18
+
19
+ /** Discover *.ts handlers and `{name}/index.ts` dirs; skips `_shared`, dotfiles. */
20
+ export function discoverTsFunctionsInDir(functionsDir: string): DiscoveredFunctionRoute[] {
21
+ if (!existsSync(functionsDir)) return []
22
+
23
+ const entries = readdirSync(functionsDir)
24
+ const out: DiscoveredFunctionRoute[] = []
25
+
26
+ for (const entry of entries) {
27
+ if (entry.startsWith("_") || entry.startsWith(".")) continue
28
+
29
+ const fullPath = join(functionsDir, entry)
30
+ const st = statSync(fullPath)
31
+
32
+ if (st.isDirectory()) {
33
+ const indexTs = join(fullPath, "index.ts")
34
+ if (existsSync(indexTs)) out.push({ name: entry, entrypoint: indexTs })
35
+ } else if (entry.endsWith(".ts") && !entry.endsWith(".d.ts")) {
36
+ const name = basename(entry, ".ts")
37
+ out.push({ name, entrypoint: fullPath })
38
+ }
39
+ }
40
+
41
+ return out.sort((a, b) => a.name.localeCompare(b.name))
42
+ }
43
+
44
+ function importSpecFromRouter(routerFilePath: string, entrypoint: string): string {
45
+ const rel = relative(dirname(routerFilePath), entrypoint).replace(/\\/g, "/")
46
+ if (!rel.startsWith(".")) return `./${rel}`
47
+ return rel
48
+ }
49
+
50
+ /** Router source rooted at routerFilePath (defines import paths). */
51
+ export function generateFunctionsRouterSource(
52
+ routerFilePath: string,
53
+ fns: DiscoveredFunctionRoute[],
54
+ ): string {
55
+ const imports = fns.map(
56
+ (fn, i) =>
57
+ `import handler_${i} from "${importSpecFromRouter(routerFilePath, fn.entrypoint)}"`,
58
+ )
59
+ const routes = fns.map((fn, i) => ` "${fn.name}": handler_${i},`)
60
+
61
+ return `// Auto-generated — do not edit (regenerated by supatype dev / functions serve)
62
+ ${imports.join("\n")}
63
+
64
+ const handlers: Record<string, (req: Request) => Response | Promise<Response>> = {
65
+ ${routes.join("\n")}
66
+ }
67
+
68
+ const port = parseInt(Deno.env.get("PORT") ?? "8001", 10)
69
+ const functionsDir = Deno.env.get("SUPATYPE_DENO_FUNCTIONS_DIR") ?? ""
70
+ const normalizedFunctionsDir = functionsDir.endsWith("/") ? functionsDir.slice(0, -1) : functionsDir
71
+ const sharedEnvPath = Deno.env.get("SUPATYPE_SHARED_ENV_FILE")
72
+ ?? (normalizedFunctionsDir ? normalizedFunctionsDir + "/.env.local" : "")
73
+
74
+ let envLock: Promise<void> = Promise.resolve()
75
+
76
+ async function withEnvLock<T>(run: () => Promise<T>): Promise<T> {
77
+ const prev = envLock
78
+ let release: () => void = () => {}
79
+ envLock = new Promise<void>(resolve => { release = resolve })
80
+ await prev
81
+ try {
82
+ return await run()
83
+ } finally {
84
+ release()
85
+ }
86
+ }
87
+
88
+ async function readEnvFile(path: string): Promise<Record<string, string>> {
89
+ if (!path) return {}
90
+ try {
91
+ const text = await Deno.readTextFile(path)
92
+ const out: Record<string, string> = {}
93
+ for (const line of text.split("\\n")) {
94
+ const trimmed = line.trim()
95
+ if (!trimmed || trimmed.startsWith("#")) continue
96
+ const eq = trimmed.indexOf("=")
97
+ if (eq <= 0) continue
98
+ out[trimmed.slice(0, eq)] = trimmed.slice(eq + 1)
99
+ }
100
+ return out
101
+ } catch {
102
+ return {}
103
+ }
104
+ }
105
+
106
+ async function scopedEnvForFunction(fnName: string): Promise<Record<string, string>> {
107
+ const shared = await readEnvFile(sharedEnvPath)
108
+ if (!normalizedFunctionsDir) return shared
109
+ const fnPath = normalizedFunctionsDir + "/.env." + fnName + ".local"
110
+ const fnVars = await readEnvFile(fnPath)
111
+ return { ...shared, ...fnVars }
112
+ }
113
+
114
+ async function runWithScopedEnv<T>(fnName: string, run: () => Promise<T>): Promise<T> {
115
+ return withEnvLock(async () => {
116
+ const scoped = await scopedEnvForFunction(fnName)
117
+ const prev = new Map<string, string | undefined>()
118
+ for (const [k, v] of Object.entries(scoped)) {
119
+ prev.set(k, Deno.env.get(k))
120
+ Deno.env.set(k, v)
121
+ }
122
+ try {
123
+ return await run()
124
+ } finally {
125
+ for (const k of Object.keys(scoped)) {
126
+ const old = prev.get(k)
127
+ if (old === undefined) Deno.env.delete(k)
128
+ else Deno.env.set(k, old)
129
+ }
130
+ }
131
+ })
132
+ }
133
+
134
+ Deno.serve({ port }, async (req: Request): Promise<Response> => {
135
+ const url = new URL(req.url)
136
+ const pathParts = url.pathname
137
+ .replace(/^\\/functions\\/v1\\/?/, "")
138
+ .split("/")
139
+ .filter(Boolean)
140
+ const fnName = pathParts[0] ?? ""
141
+
142
+ if (!fnName || !handlers[fnName]) {
143
+ return new Response(JSON.stringify({
144
+ error: "not_found",
145
+ message: fnName ? \`Function "\${fnName}" not found\` : "No function specified",
146
+ available: Object.keys(handlers),
147
+ }), { status: 404, headers: { "Content-Type": "application/json" } })
148
+ }
149
+
150
+ try {
151
+ const start = performance.now()
152
+ const response = await runWithScopedEnv(fnName, async () => {
153
+ const prev = new Map<string, string | undefined>()
154
+ const setScoped = (key: string, value: string | undefined) => {
155
+ if (value === undefined || value.length === 0) return
156
+ prev.set(key, Deno.env.get(key))
157
+ Deno.env.set(key, value)
158
+ }
159
+
160
+ // Supatype runtime defaults available to every invocation.
161
+ const supatypeUrl = Deno.env.get("SUPATYPE_URL")
162
+ const supatypeAnon = Deno.env.get("SUPATYPE_ANON_KEY")
163
+ const supatypeServiceRole = Deno.env.get("SUPATYPE_SERVICE_ROLE_KEY")
164
+ const supatypeDbUrl = Deno.env.get("SUPATYPE_DB_URL") ?? Deno.env.get("DATABASE_URL")
165
+ const supatypeJwks = Deno.env.get("SUPATYPE_JWKS")
166
+
167
+ setScoped("SUPATYPE_URL", supatypeUrl)
168
+ setScoped("SUPATYPE_ANON_KEY", supatypeAnon)
169
+ setScoped("SUPATYPE_SERVICE_ROLE_KEY", supatypeServiceRole)
170
+ setScoped("SUPATYPE_DB_URL", supatypeDbUrl)
171
+ setScoped("SUPATYPE_JWKS", supatypeJwks)
172
+ if (!Deno.env.get("SUPATYPE_PUBLISHABLE_KEYS") && supatypeAnon) {
173
+ setScoped("SUPATYPE_PUBLISHABLE_KEYS", JSON.stringify({ anon: supatypeAnon }))
174
+ }
175
+ if (!Deno.env.get("SUPATYPE_SECRET_KEYS") && supatypeServiceRole) {
176
+ setScoped("SUPATYPE_SECRET_KEYS", JSON.stringify({ service_role: supatypeServiceRole }))
177
+ }
178
+
179
+ setScoped("SUPATYPE_REGION", Deno.env.get("SUPATYPE_REGION") ?? "local")
180
+ setScoped("SUPATYPE_EXECUTION_ID", crypto.randomUUID())
181
+ setScoped("DENO_DEPLOYMENT_ID", Deno.env.get("DENO_DEPLOYMENT_ID") ?? "local-dev")
182
+
183
+ try {
184
+ return await handlers[fnName]!(req)
185
+ } finally {
186
+ for (const [key, old] of prev.entries()) {
187
+ if (old === undefined) Deno.env.delete(key)
188
+ else Deno.env.set(key, old)
189
+ }
190
+ }
191
+ })
192
+ const duration = (performance.now() - start).toFixed(1)
193
+ console.log(\`\${req.method} /functions/v1/\${fnName} → \${response.status} (\${duration}ms)\`)
194
+ return response
195
+ } catch (err) {
196
+ console.error(\`Error in function "\${fnName}":\`, err)
197
+ return new Response(JSON.stringify({
198
+ error: "function_error",
199
+ message: err instanceof Error ? err.message : "Unknown error",
200
+ }), { status: 500, headers: { "Content-Type": "application/json" } })
201
+ }
202
+ })
203
+ `
204
+ }
205
+
206
+ /**
207
+ * Writes `.supatype/functions-router.ts` under project cwd. Returns absolute path or null if no handlers.
208
+ * Pass `routes` when callers already scanned the dir to avoid duplicate I/O.
209
+ */
210
+ export function writeDevFunctionsRouter(
211
+ cwd: string,
212
+ functionsDir: string,
213
+ routes?: DiscoveredFunctionRoute[],
214
+ ): string | null {
215
+ const fns = routes ?? discoverTsFunctionsInDir(functionsDir)
216
+ if (fns.length === 0) return null
217
+
218
+ const supatypeDir = pathResolve(cwd, ".supatype")
219
+ mkdirSync(supatypeDir, { recursive: true })
220
+ const routerPath = join(supatypeDir, "functions-router.ts")
221
+ const src = generateFunctionsRouterSource(routerPath, fns)
222
+ writeFileSync(routerPath, src, "utf8")
223
+ return routerPath
224
+ }
package/src/index.ts CHANGED
@@ -4,16 +4,8 @@
4
4
  * For CLI usage: use the `supatype` binary.
5
5
  */
6
6
 
7
- export { invokeEngine, getEnginePath, ensureEngine, getEnginePathAsync } from "./engine.js"
8
- export {
9
- ENGINE_VERSION,
10
- ENGINE_REPO,
11
- ENGINE_DOWNLOAD_BASE,
12
- CDN_BASE_URL,
13
- ENGINE_RELEASES_REPO,
14
- GITHUB_RELEASES_FALLBACK_URL,
15
- } from "./engine-version.js"
7
+ export { ensureEngine, engineRequest, engineHealth, EngineError } from "./engine-client.js"
8
+ export type { EngineResult, DiffResult, Operation, IntrospectResult } from "./engine-client.js"
16
9
  export { defineConfig, loadConfig, loadSchemaAst } from "./config.js"
17
- export type { SupatypeConfig } from "./config.js"
18
- export { detectPlatform } from "./engine/platform.js"
19
- export type { PlatformInfo } from "./engine/platform.js"
10
+ export type { SupatypeConfig, SupatypeProjectConfig } from "./config.js"
11
+ export { schemaPathFromProject, localDSN, connectionString, serverBaseUrl } from "./project-config.js"
@@ -0,0 +1,93 @@
1
+ /**
2
+ * Kong declarative config for local dev and self-hosted production.
3
+ * When `engineGatewayKey` is set, `/studio-config` and `/sql` require `apikey` (key-auth).
4
+ */
5
+
6
+ export interface KongDeclarativeOptions {
7
+ /**
8
+ * When non-empty, Kong `key-auth` is enabled on engine routes; clients must send
9
+ * header `apikey: <key>` (same convention as PostgREST). Omit for open local dev.
10
+ */
11
+ engineGatewayKey?: string | undefined
12
+ appUpstream?: string | undefined
13
+ staticAppServiceUrl?: string | undefined
14
+ functionsServiceUrl?: string | undefined
15
+ /** Self-host: route API paths through supatype-server (see runtime-routes). */
16
+ unifiedGateway?: boolean | undefined
17
+ /** Studio UI upstream (default: in-compose `studio:3002`). */
18
+ studioServiceUrl?: string | undefined
19
+ /** See {@link RuntimeRouteOptions.studioStripPath}. */
20
+ studioStripPath?: boolean | undefined
21
+ }
22
+
23
+ /** Escape a string for use inside YAML double quotes. */
24
+ function yamlQuotedString(s: string): string {
25
+ return JSON.stringify(s)
26
+ }
27
+
28
+ import { runtimeRouteSpec } from "./runtime-routes.js"
29
+
30
+ /**
31
+ * Build full `kong.yml` content. Single source of truth for CLI `dev` and `self-host setup`.
32
+ */
33
+ export function buildKongDeclarative(opts: KongDeclarativeOptions = {}): string {
34
+ const gatewayKey = opts.engineGatewayKey?.trim()
35
+ const secured = Boolean(gatewayKey)
36
+ const routes = runtimeRouteSpec({
37
+ ...(opts.unifiedGateway === true && { unifiedGateway: true }),
38
+ ...(opts.unifiedGateway !== true && opts.appUpstream !== undefined && { appUpstream: opts.appUpstream }),
39
+ ...(opts.unifiedGateway !== true &&
40
+ opts.staticAppServiceUrl !== undefined && { staticAppServiceUrl: opts.staticAppServiceUrl }),
41
+ ...(opts.functionsServiceUrl !== undefined && { functionsServiceUrl: opts.functionsServiceUrl }),
42
+ ...(opts.studioServiceUrl !== undefined && { studioServiceUrl: opts.studioServiceUrl }),
43
+ ...(opts.studioStripPath === false && { studioStripPath: false }),
44
+ })
45
+
46
+ const consumersBlock = secured
47
+ ? `
48
+ consumers:
49
+ - username: studio-engine-gateway
50
+ keyauth_credentials:
51
+ - key: ${yamlQuotedString(gatewayKey!)}
52
+ `
53
+ : ""
54
+
55
+ const servicesBlock = routes.map((route) => {
56
+ const routePlugins = route.engineProtected && secured
57
+ ? ` plugins:
58
+ - name: key-auth
59
+ config:
60
+ key_names:
61
+ - apikey
62
+ hide_credentials: true
63
+ `
64
+ : ""
65
+ const protocols = route.protocols && route.protocols.length > 0
66
+ ? ` protocols:\n${route.protocols.map((p) => ` - ${p}`).join("\n")}\n`
67
+ : ""
68
+ const stripPath = route.stripPath ?? false
69
+ const graphqlPlugins = route.graphqlPostgrest
70
+ ? ` plugins:
71
+ - name: request-transformer
72
+ config:
73
+ add:
74
+ headers:
75
+ - Content-Profile:graphql_public
76
+ `
77
+ : ""
78
+ return ` - name: ${route.serviceName}
79
+ url: ${route.serviceUrl}
80
+ routes:
81
+ - name: ${route.name}
82
+ strip_path: ${stripPath}
83
+ paths:
84
+ ${route.paths.map((path) => ` - ${path}`).join("\n")}
85
+ ${protocols}${routePlugins}${graphqlPlugins}`
86
+ }).join("\n")
87
+
88
+ return `_format_version: "3.0"
89
+ ${consumersBlock}
90
+ services:
91
+ ${servicesBlock}
92
+ `
93
+ }
@@ -0,0 +1,9 @@
1
+ /**
2
+ * Host port published for Kong in local Docker Compose. Kong still listens on
3
+ * 8000 inside the container; this avoids clashing with other tools on :8000.
4
+ */
5
+ export const LOCAL_KONG_HOST_PORT = 18473
6
+
7
+ export function localKongBaseUrl(): string {
8
+ return `http://localhost:${LOCAL_KONG_HOST_PORT}`
9
+ }
@@ -0,0 +1,14 @@
1
+ import { join } from "node:path"
2
+
3
+ /**
4
+ * Returns env vars that configure supatype-server to use local-disk storage.
5
+ * Spread into the server process env when config.storage?.provider !== "s3".
6
+ *
7
+ * @param stateDir Per-project state directory (e.g. ~/.supatype/projects/{name}/)
8
+ */
9
+ export function localStorageEnv(stateDir: string): Record<string, string> {
10
+ return {
11
+ STORAGE_PROVIDER: "local",
12
+ STORAGE_PATH: join(stateDir, "storage"),
13
+ }
14
+ }
@@ -0,0 +1,15 @@
1
+ import { createHash } from "node:crypto"
2
+
3
+ /** PgBouncer md5 auth: "md5" + md5(password + username) — same as supatype-cloud / self-host. */
4
+ export function pgbouncerMd5Hash(password: string, username: string): string {
5
+ return "md5" + createHash("md5").update(password + username).digest("hex")
6
+ }
7
+
8
+ /** Two roles used by local compose: superuser pools (GoTrue, engine) and PostgREST. */
9
+ export function pgbouncerUserlistContent(pgPassword: string): string {
10
+ return `# PgBouncer md5 userlist — matches supatype-cloud transaction pool (auth_type = md5).
11
+ # Regenerated by supatype dev --local / init from POSTGRES_PASSWORD.
12
+ "supatype_admin" "${pgbouncerMd5Hash(pgPassword, "supatype_admin")}"
13
+ "authenticator" "${pgbouncerMd5Hash(pgPassword, "authenticator")}"
14
+ `
15
+ }
@@ -0,0 +1,171 @@
1
+ /**
2
+ * postgres-ctl — wrappers around pg_ctl, initdb, and pg_isready for managing
3
+ * a native Postgres installation.
4
+ */
5
+
6
+ import { spawnSync } from "node:child_process"
7
+ import { existsSync, mkdirSync } from "node:fs"
8
+ import { dirname, join } from "node:path"
9
+
10
+ export interface PgOptions {
11
+ /** Absolute path to the directory containing pg_ctl, initdb, psql, etc. */
12
+ pgBinDir: string
13
+ /** Absolute path to the Postgres data directory (PGDATA). */
14
+ dataDir: string
15
+ /** Port Postgres should listen on. */
16
+ port: number
17
+ /** Path to write the postgres log file. */
18
+ logPath?: string
19
+ }
20
+
21
+ /**
22
+ * Native Postgres bundles are built with prefix /usr/local/supatype-pg; dyld/ld
23
+ * must load libpq and friends from the extracted lib/ next to bin/.
24
+ */
25
+ export function pgSpawnEnv(
26
+ pgBinDir: string,
27
+ platform: NodeJS.Platform = process.platform,
28
+ ): NodeJS.ProcessEnv {
29
+ const libDir = join(dirname(pgBinDir), "lib")
30
+ const env = { ...process.env } as NodeJS.ProcessEnv
31
+ if (platform === "darwin") {
32
+ const prev = env.DYLD_LIBRARY_PATH ?? ""
33
+ env.DYLD_LIBRARY_PATH = prev ? `${libDir}:${prev}` : libDir
34
+ } else if (platform === "linux") {
35
+ const prev = env.LD_LIBRARY_PATH ?? ""
36
+ env.LD_LIBRARY_PATH = prev ? `${libDir}:${prev}` : libDir
37
+ }
38
+ return env
39
+ }
40
+
41
+ // ---------------------------------------------------------------------------
42
+ // initdb
43
+ // ---------------------------------------------------------------------------
44
+
45
+ /**
46
+ * Initialise a Postgres data directory.
47
+ * Does nothing if the data directory already contains a PG_VERSION file.
48
+ */
49
+ export function initdb(opts: PgOptions): void {
50
+ const pgVersionFile = join(opts.dataDir, "PG_VERSION")
51
+ if (existsSync(pgVersionFile)) return // Already initialised.
52
+
53
+ mkdirSync(opts.dataDir, { recursive: true })
54
+
55
+ const bin = pgBin(opts.pgBinDir, "initdb")
56
+ const result = spawnSync(bin, ["-D", opts.dataDir, "--username", "postgres", "--auth", "trust"], {
57
+ stdio: "inherit",
58
+ encoding: "utf8",
59
+ env: pgSpawnEnv(opts.pgBinDir),
60
+ })
61
+
62
+ if (result.status !== 0) {
63
+ throw new Error(`initdb failed (exit ${result.status})`)
64
+ }
65
+ }
66
+
67
+ // ---------------------------------------------------------------------------
68
+ // start / stop
69
+ // ---------------------------------------------------------------------------
70
+
71
+ /**
72
+ * Start Postgres using pg_ctl.
73
+ * Returns immediately once pg_ctl has handed off to the server process.
74
+ */
75
+ export function start(opts: PgOptions): void {
76
+ const bin = pgBin(opts.pgBinDir, "pg_ctl")
77
+ const logPath = opts.logPath ?? join(opts.dataDir, "postgres.log")
78
+
79
+ const args = [
80
+ "start",
81
+ "-D", opts.dataDir,
82
+ "-l", logPath,
83
+ "-o", `-p ${opts.port}`,
84
+ "--wait",
85
+ ]
86
+
87
+ const result = spawnSync(bin, args, {
88
+ stdio: "inherit",
89
+ encoding: "utf8",
90
+ env: pgSpawnEnv(opts.pgBinDir),
91
+ })
92
+ if (result.status !== 0) {
93
+ throw new Error(`pg_ctl start failed (exit ${result.status})`)
94
+ }
95
+ }
96
+
97
+ /**
98
+ * Stop Postgres using pg_ctl (fast mode).
99
+ */
100
+ export function stop(opts: PgOptions): void {
101
+ const bin = pgBin(opts.pgBinDir, "pg_ctl")
102
+ const result = spawnSync(bin, ["stop", "-D", opts.dataDir, "-m", "fast", "--wait"], {
103
+ stdio: "inherit",
104
+ encoding: "utf8",
105
+ env: pgSpawnEnv(opts.pgBinDir),
106
+ })
107
+ // Ignore exit code — Postgres may already be stopped.
108
+ void result
109
+ }
110
+
111
+ // ---------------------------------------------------------------------------
112
+ // waitReady
113
+ // ---------------------------------------------------------------------------
114
+
115
+ /**
116
+ * Wait until Postgres is accepting connections.
117
+ * Polls pg_isready every 200ms up to timeoutMs.
118
+ * Throws if the timeout is exceeded.
119
+ */
120
+ export async function waitReady(opts: PgOptions, timeoutMs = 10_000): Promise<void> {
121
+ const bin = pgBin(opts.pgBinDir, "pg_isready")
122
+ const deadline = Date.now() + timeoutMs
123
+
124
+ while (Date.now() < deadline) {
125
+ const result = spawnSync(bin, ["-p", String(opts.port), "-q"], {
126
+ encoding: "utf8",
127
+ env: pgSpawnEnv(opts.pgBinDir),
128
+ })
129
+ if (result.status === 0) return
130
+
131
+ await sleep(200)
132
+ }
133
+
134
+ throw new Error(
135
+ `Postgres did not become ready within ${timeoutMs}ms on port ${opts.port}`,
136
+ )
137
+ }
138
+
139
+ // ---------------------------------------------------------------------------
140
+ // Port check
141
+ // ---------------------------------------------------------------------------
142
+
143
+ /**
144
+ * Returns true if a TCP listener is already bound to port on 127.0.0.1.
145
+ */
146
+ export async function isPortInUse(port: number): Promise<boolean> {
147
+ const { createServer } = await import("node:net")
148
+ return new Promise((resolve) => {
149
+ const server = createServer()
150
+ server.once("error", (err: NodeJS.ErrnoException) => {
151
+ resolve(err.code === "EADDRINUSE")
152
+ })
153
+ server.once("listening", () => {
154
+ server.close(() => resolve(false))
155
+ })
156
+ server.listen(port, "127.0.0.1")
157
+ })
158
+ }
159
+
160
+ // ---------------------------------------------------------------------------
161
+ // Helpers
162
+ // ---------------------------------------------------------------------------
163
+
164
+ function sleep(ms: number): Promise<void> {
165
+ return new Promise((resolve) => setTimeout(resolve, ms))
166
+ }
167
+
168
+ /** Returns the full path to a Postgres binary, appending .exe on Windows. */
169
+ function pgBin(binDir: string, name: string): string {
170
+ return join(binDir, process.platform === "win32" ? `${name}.exe` : name)
171
+ }