@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.
- package/.turbo/turbo-build.log +1 -1
- package/.turbo/turbo-test.log +208 -1
- package/.turbo/turbo-typecheck.log +1 -1
- package/dist/app/proxy-dev-app.d.ts +13 -0
- package/dist/app/proxy-dev-app.d.ts.map +1 -0
- package/dist/app/proxy-dev-app.js +53 -0
- package/dist/app/proxy-dev-app.js.map +1 -0
- package/dist/app-config.d.ts +7 -0
- package/dist/app-config.d.ts.map +1 -0
- package/dist/app-config.js +113 -0
- package/dist/app-config.js.map +1 -0
- package/dist/augmentation-generator.d.ts +2 -0
- package/dist/augmentation-generator.d.ts.map +1 -0
- package/dist/augmentation-generator.js +111 -0
- package/dist/augmentation-generator.js.map +1 -0
- package/dist/binary-cache.d.ts +94 -0
- package/dist/binary-cache.d.ts.map +1 -0
- package/dist/binary-cache.js +669 -0
- package/dist/binary-cache.js.map +1 -0
- package/dist/cli.d.ts.map +1 -1
- package/dist/cli.js +13 -7
- package/dist/cli.js.map +1 -1
- package/dist/commands/admin.d.ts.map +1 -1
- package/dist/commands/admin.js +4 -3
- package/dist/commands/admin.js.map +1 -1
- package/dist/commands/app.d.ts.map +1 -1
- package/dist/commands/app.js +56 -209
- package/dist/commands/app.js.map +1 -1
- package/dist/commands/cache.d.ts +6 -0
- package/dist/commands/cache.d.ts.map +1 -0
- package/dist/commands/cache.js +105 -0
- package/dist/commands/cache.js.map +1 -0
- package/dist/commands/cloud.d.ts +20 -0
- package/dist/commands/cloud.d.ts.map +1 -1
- package/dist/commands/cloud.js +50 -52
- package/dist/commands/cloud.js.map +1 -1
- package/dist/commands/db.d.ts.map +1 -1
- package/dist/commands/db.js +47 -54
- package/dist/commands/db.js.map +1 -1
- package/dist/commands/deploy.d.ts +2 -1
- package/dist/commands/deploy.d.ts.map +1 -1
- package/dist/commands/deploy.js +79 -52
- package/dist/commands/deploy.js.map +1 -1
- package/dist/commands/dev.d.ts +11 -0
- package/dist/commands/dev.d.ts.map +1 -1
- package/dist/commands/dev.js +759 -385
- package/dist/commands/dev.js.map +1 -1
- package/dist/commands/diff.d.ts.map +1 -1
- package/dist/commands/diff.js +30 -15
- package/dist/commands/diff.js.map +1 -1
- package/dist/commands/engine.d.ts +1 -3
- package/dist/commands/engine.d.ts.map +1 -1
- package/dist/commands/engine.js +13 -85
- package/dist/commands/engine.js.map +1 -1
- package/dist/commands/functions.d.ts.map +1 -1
- package/dist/commands/functions.js +92 -105
- package/dist/commands/functions.js.map +1 -1
- package/dist/commands/generate.d.ts.map +1 -1
- package/dist/commands/generate.js +22 -12
- package/dist/commands/generate.js.map +1 -1
- package/dist/commands/init.d.ts +1 -1
- package/dist/commands/init.d.ts.map +1 -1
- package/dist/commands/init.js +137 -410
- package/dist/commands/init.js.map +1 -1
- package/dist/commands/migrate-from-v1.d.ts +5 -0
- package/dist/commands/migrate-from-v1.d.ts.map +1 -0
- package/dist/commands/migrate-from-v1.js +125 -0
- package/dist/commands/migrate-from-v1.js.map +1 -0
- package/dist/commands/migrate.d.ts.map +1 -1
- package/dist/commands/migrate.js +27 -23
- package/dist/commands/migrate.js.map +1 -1
- package/dist/commands/pg.d.ts +8 -0
- package/dist/commands/pg.d.ts.map +1 -0
- package/dist/commands/pg.js +102 -0
- package/dist/commands/pg.js.map +1 -0
- package/dist/commands/pull.d.ts.map +1 -1
- package/dist/commands/pull.js +5 -66
- package/dist/commands/pull.js.map +1 -1
- package/dist/commands/push.d.ts.map +1 -1
- package/dist/commands/push.js +128 -38
- package/dist/commands/push.js.map +1 -1
- package/dist/commands/seed.d.ts +2 -0
- package/dist/commands/seed.d.ts.map +1 -1
- package/dist/commands/seed.js +44 -11
- package/dist/commands/seed.js.map +1 -1
- package/dist/commands/self-host.d.ts +7 -1
- package/dist/commands/self-host.d.ts.map +1 -1
- package/dist/commands/self-host.js +272 -758
- package/dist/commands/self-host.js.map +1 -1
- package/dist/commands/self-update.d.ts +9 -0
- package/dist/commands/self-update.d.ts.map +1 -0
- package/dist/commands/self-update.js +33 -0
- package/dist/commands/self-update.js.map +1 -0
- package/dist/commands/status.d.ts.map +1 -1
- package/dist/commands/status.js +4 -3
- package/dist/commands/status.js.map +1 -1
- package/dist/commands/types.d.ts +3 -0
- package/dist/commands/types.d.ts.map +1 -0
- package/dist/commands/types.js +62 -0
- package/dist/commands/types.js.map +1 -0
- package/dist/commands/update.d.ts +7 -0
- package/dist/commands/update.d.ts.map +1 -0
- package/dist/commands/update.js +93 -0
- package/dist/commands/update.js.map +1 -0
- package/dist/components.d.ts +5 -0
- package/dist/components.d.ts.map +1 -0
- package/dist/components.js +3 -0
- package/dist/components.js.map +1 -0
- package/dist/config.d.ts +10 -51
- package/dist/config.d.ts.map +1 -1
- package/dist/config.js +101 -33
- package/dist/config.js.map +1 -1
- package/dist/dev-compose.d.ts +17 -0
- package/dist/dev-compose.d.ts.map +1 -0
- package/dist/dev-compose.js +374 -0
- package/dist/dev-compose.js.map +1 -0
- package/dist/diff-output.d.ts +4 -0
- package/dist/diff-output.d.ts.map +1 -0
- package/dist/diff-output.js +12 -0
- package/dist/diff-output.js.map +1 -0
- package/dist/docker-postgres.d.ts +57 -0
- package/dist/docker-postgres.d.ts.map +1 -0
- package/dist/docker-postgres.js +208 -0
- package/dist/docker-postgres.js.map +1 -0
- package/dist/engine-client.d.ts +69 -0
- package/dist/engine-client.d.ts.map +1 -0
- package/dist/engine-client.js +157 -0
- package/dist/engine-client.js.map +1 -0
- package/dist/ensure-binary.d.ts +7 -0
- package/dist/ensure-binary.d.ts.map +1 -0
- package/dist/ensure-binary.js +17 -0
- package/dist/ensure-binary.js.map +1 -0
- package/dist/functions-router-gen.d.ts +14 -0
- package/dist/functions-router-gen.d.ts.map +1 -0
- package/dist/functions-router-gen.js +199 -0
- package/dist/functions-router-gen.js.map +1 -0
- package/dist/index.d.ts +4 -5
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +2 -3
- package/dist/index.js.map +1 -1
- package/dist/kong-config.d.ts +25 -0
- package/dist/kong-config.d.ts.map +1 -0
- package/dist/kong-config.js +71 -0
- package/dist/kong-config.js.map +1 -0
- package/dist/local-gateway.d.ts +7 -0
- package/dist/local-gateway.d.ts.map +1 -0
- package/dist/local-gateway.js +9 -0
- package/dist/local-gateway.js.map +1 -0
- package/dist/local-storage.d.ts +8 -0
- package/dist/local-storage.d.ts.map +1 -0
- package/dist/local-storage.js +14 -0
- package/dist/local-storage.js.map +1 -0
- package/dist/pgbouncer-userlist.d.ts +5 -0
- package/dist/pgbouncer-userlist.d.ts.map +1 -0
- package/dist/pgbouncer-userlist.js +14 -0
- package/dist/pgbouncer-userlist.js.map +1 -0
- package/dist/postgres-ctl.d.ts +44 -0
- package/dist/postgres-ctl.d.ts.map +1 -0
- package/dist/postgres-ctl.js +137 -0
- package/dist/postgres-ctl.js.map +1 -0
- package/dist/process-manager.d.ts +43 -0
- package/dist/process-manager.d.ts.map +1 -0
- package/dist/process-manager.js +135 -0
- package/dist/process-manager.js.map +1 -0
- package/dist/project-config.d.ts +235 -0
- package/dist/project-config.d.ts.map +1 -0
- package/dist/project-config.js +160 -0
- package/dist/project-config.js.map +1 -0
- package/dist/pull-utils.d.ts +15 -0
- package/dist/pull-utils.d.ts.map +1 -1
- package/dist/pull-utils.js +12 -0
- package/dist/pull-utils.js.map +1 -1
- package/dist/release-pins.d.ts +7 -0
- package/dist/release-pins.d.ts.map +1 -0
- package/dist/release-pins.js +27 -0
- package/dist/release-pins.js.map +1 -0
- package/dist/release-public-key.d.ts +8 -0
- package/dist/release-public-key.d.ts.map +1 -0
- package/dist/release-public-key.js +13 -0
- package/dist/release-public-key.js.map +1 -0
- package/dist/runtime-routes.d.ts +34 -0
- package/dist/runtime-routes.d.ts.map +1 -0
- package/dist/runtime-routes.js +252 -0
- package/dist/runtime-routes.js.map +1 -0
- package/dist/schema-ast-v2.d.ts +127 -0
- package/dist/schema-ast-v2.d.ts.map +1 -0
- package/dist/schema-ast-v2.js +226 -0
- package/dist/schema-ast-v2.js.map +1 -0
- package/dist/scripts/postinstall.d.ts +5 -6
- package/dist/scripts/postinstall.d.ts.map +1 -1
- package/dist/scripts/postinstall.js +36 -20
- package/dist/scripts/postinstall.js.map +1 -1
- package/dist/self-host-compose.d.ts +22 -0
- package/dist/self-host-compose.d.ts.map +1 -0
- package/dist/self-host-compose.js +347 -0
- package/dist/self-host-compose.js.map +1 -0
- package/dist/storage-provision.d.ts +24 -0
- package/dist/storage-provision.d.ts.map +1 -0
- package/dist/storage-provision.js +44 -0
- package/dist/storage-provision.js.map +1 -0
- package/dist/studio-admin-roles.d.ts +7 -0
- package/dist/studio-admin-roles.d.ts.map +1 -0
- package/dist/studio-admin-roles.js +14 -0
- package/dist/studio-admin-roles.js.map +1 -0
- package/dist/studio-dev-server.d.ts +22 -0
- package/dist/studio-dev-server.d.ts.map +1 -0
- package/dist/studio-dev-server.js +28 -0
- package/dist/studio-dev-server.js.map +1 -0
- package/dist/systemd.d.ts +26 -0
- package/dist/systemd.d.ts.map +1 -0
- package/dist/systemd.js +102 -0
- package/dist/systemd.js.map +1 -0
- package/dist/tsx-runner.d.ts.map +1 -1
- package/dist/tsx-runner.js +9 -2
- package/dist/tsx-runner.js.map +1 -1
- package/dist/type-extractor.d.ts +4 -0
- package/dist/type-extractor.d.ts.map +1 -0
- package/dist/type-extractor.js +1213 -0
- package/dist/type-extractor.js.map +1 -0
- package/dist/type-resolver.d.ts +33 -0
- package/dist/type-resolver.d.ts.map +1 -0
- package/dist/type-resolver.js +338 -0
- package/dist/type-resolver.js.map +1 -0
- package/package.json +4 -3
- package/releases/deno/VERSION +1 -0
- package/scripts/mirror-deno-release.sh +76 -0
- package/src/TYPE-RESOLUTION.md +294 -0
- package/src/app/proxy-dev-app.ts +67 -0
- package/src/app-config.ts +128 -0
- package/src/augmentation-generator.ts +126 -0
- package/src/binary-cache.ts +822 -0
- package/src/cli.ts +13 -8
- package/src/commands/admin.ts +4 -3
- package/src/commands/app.ts +67 -231
- package/src/commands/cache.ts +117 -0
- package/src/commands/cloud.ts +63 -64
- package/src/commands/db.ts +54 -63
- package/src/commands/deploy.ts +96 -62
- package/src/commands/dev.ts +933 -405
- package/src/commands/diff.ts +31 -29
- package/src/commands/engine.ts +13 -116
- package/src/commands/functions.ts +97 -115
- package/src/commands/generate.ts +23 -10
- package/src/commands/init.ts +149 -414
- package/src/commands/migrate-from-v1.ts +131 -0
- package/src/commands/migrate.ts +27 -23
- package/src/commands/pg.ts +133 -0
- package/src/commands/pull.ts +6 -85
- package/src/commands/push.ts +161 -56
- package/src/commands/seed.ts +54 -12
- package/src/commands/self-host.ts +312 -880
- package/src/commands/self-update.ts +45 -0
- package/src/commands/status.ts +4 -3
- package/src/commands/types.ts +76 -0
- package/src/commands/update.ts +109 -0
- package/src/components.ts +6 -0
- package/src/config.ts +127 -94
- package/src/dev-compose.ts +455 -0
- package/src/diff-output.ts +12 -0
- package/src/docker-postgres.ts +295 -0
- package/src/engine-client.ts +236 -0
- package/src/ensure-binary.ts +28 -0
- package/src/functions-router-gen.ts +224 -0
- package/src/index.ts +4 -12
- package/src/kong-config.ts +93 -0
- package/src/local-gateway.ts +9 -0
- package/src/local-storage.ts +14 -0
- package/src/pgbouncer-userlist.ts +15 -0
- package/src/postgres-ctl.ts +171 -0
- package/src/process-manager.ts +168 -0
- package/src/project-config.ts +386 -0
- package/src/pull-utils.ts +24 -0
- package/src/release-pins.ts +31 -0
- package/src/release-public-key.ts +12 -0
- package/src/runtime-routes.ts +291 -0
- package/src/schema-ast-v2.ts +324 -0
- package/src/scripts/postinstall.ts +36 -25
- package/src/self-host-compose.ts +389 -0
- package/src/storage-provision.ts +58 -0
- package/src/studio-admin-roles.ts +16 -0
- package/src/studio-dev-server.ts +53 -0
- package/src/systemd.ts +137 -0
- package/src/tsx-runner.ts +11 -1
- package/src/type-extractor.ts +1479 -0
- package/src/type-resolver.ts +457 -0
- package/tests/app-command.test.ts +54 -0
- package/tests/augmentation-generator.test.ts +59 -0
- package/tests/binary-cache-cloud-overrides.test.ts +123 -0
- package/tests/cached-artifact-format.test.ts +84 -0
- package/tests/cli-help.test.ts +40 -14
- package/tests/config.test.ts +171 -37
- package/tests/docker-postgres.test.ts +39 -0
- package/tests/engine-distribution.test.ts +3 -3
- package/tests/ensure-binary.test.ts +59 -0
- package/tests/init.test.ts +28 -86
- package/tests/migrate-from-v1.test.ts +29 -0
- package/tests/normalize-admin-config.test.ts +48 -0
- package/tests/pg-spawn-env.test.ts +18 -0
- package/tests/postgres-archive-tag.test.ts +9 -0
- package/tests/proxy-dev-app.test.ts +33 -0
- package/tests/pull-utils.test.ts +36 -1
- package/tests/release-pins.test.ts +28 -0
- package/tests/runtime-contract.test.ts +351 -0
- package/tests/seed-discover.test.ts +31 -0
- package/tests/studio-admin-roles.test.ts +27 -0
- package/tests/tsconfig.json +9 -0
- package/tests/type-extractor.test.ts +985 -0
- package/tests/type-resolver.test.ts +59 -0
- package/tsconfig.tsbuildinfo +1 -1
- package/vitest.config.ts +12 -0
- package/dist/engine/cache.d.ts +0 -37
- package/dist/engine/cache.d.ts.map +0 -1
- package/dist/engine/cache.js +0 -121
- package/dist/engine/cache.js.map +0 -1
- package/dist/engine/download.d.ts +0 -19
- package/dist/engine/download.d.ts.map +0 -1
- package/dist/engine/download.js +0 -108
- package/dist/engine/download.js.map +0 -1
- package/dist/engine/platform.d.ts +0 -24
- package/dist/engine/platform.d.ts.map +0 -1
- package/dist/engine/platform.js +0 -50
- package/dist/engine/platform.js.map +0 -1
- package/dist/engine/resolve.d.ts +0 -37
- package/dist/engine/resolve.d.ts.map +0 -1
- package/dist/engine/resolve.js +0 -133
- package/dist/engine/resolve.js.map +0 -1
- package/dist/engine/update-notify.d.ts +0 -11
- package/dist/engine/update-notify.d.ts.map +0 -1
- package/dist/engine/update-notify.js +0 -43
- package/dist/engine/update-notify.js.map +0 -1
- package/dist/engine/verify.d.ts +0 -50
- package/dist/engine/verify.d.ts.map +0 -1
- package/dist/engine/verify.js +0 -161
- package/dist/engine/verify.js.map +0 -1
- package/dist/engine-version.d.ts +0 -35
- package/dist/engine-version.d.ts.map +0 -1
- package/dist/engine-version.js +0 -35
- package/dist/engine-version.js.map +0 -1
- package/dist/engine.d.ts +0 -34
- package/dist/engine.d.ts.map +0 -1
- package/dist/engine.js +0 -76
- package/dist/engine.js.map +0 -1
- package/src/engine/cache.ts +0 -135
- package/src/engine/download.ts +0 -143
- package/src/engine/platform.ts +0 -66
- package/src/engine/resolve.ts +0 -197
- package/src/engine/update-notify.ts +0 -50
- package/src/engine/verify.ts +0 -206
- package/src/engine-version.ts +0 -39
- 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 {
|
|
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 {
|
|
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
|
+
}
|