mirror of
https://github.com/supabase/supabase.git
synced 2026-06-28 03:19:09 -04:00
9eab4f8fbf
**Stack 1/6** of the TanStack Start migration (#46424), split into reviewable, independently-mergeable PRs. > [!IMPORTANT] > **Next stays the default and only active framework after this PR.** This wires up the Vite/TanStack-Start build pipeline behind the `STUDIO_FRAMEWORK` flag, but there are no TanStack routes yet — so the TanStack build isn't functional or tested until later PRs in the stack. Nothing about the Next build, dev, or deploy changes behaviourally here. ## What's in this PR - **Dispatch:** `dev`/`build`/`start` now go through `scripts/dispatch.js`, which runs the Next variant unless `STUDIO_FRAMEWORK=tanstack`. The original commands are preserved as `dev:next`/`build:next`/`start:next`. - **Build pipeline:** `vite.config.ts`, `serve.js`, `smoke-server.mjs`, vite/tanstack deps, `turbo.jsonc`. - **`tsconfig.json`:** `jsx: react-jsx`, `moduleResolution: Bundler`, `target: ES2022`. Because `include` is `**/*.ts(x)`, this re-typechecks the whole app, so the companion adaptations below land with it. - **Shared adaptations (companions to the tsconfig change):** `BufferSource` casts, `packages/ui` unused-`React` import removals, etc. - **Routing/middleware plumbing:** `next.config.ts` + `redirects.shared.ts` (redirect rules now shared with `vercel.ts`), `proxy.ts`/`start.ts` middleware + `hosted-api-allowlist.ts`. ## Verification Run locally off `master`: frozen install ✓, `studio` typecheck ✓, **Next build ✓** (compiles + generates all routes), lint ratchet ✓ ("some rules improved"), prettier ✓. <!-- This is an auto-generated comment: release notes by coderabbit.ai --> ## Summary by CodeRabbit * **New Features** * Added a hosted API endpoint allowlist to return 404 for non-supported `/api/*` routes. * Introduced a TanStack route-migration checklist and expanded TanStack Start routing support. * **Improvements** * Enhanced deployment refresh/detection by tightening cookie handling for “latest deployment” updates. * Centralized redirect/maintenance-mode rules for consistent platform vs self-hosted behavior. * Improved production serving with a dedicated static + proxy server and a post-build smoke test. * **Dependencies** * Updated TanStack-related packages and React Table/query tooling versions. * **Documentation / Chores** * Updated formatting and tooling config; added shared build environment parsing utilities. <!-- end of auto-generated comment: release notes by coderabbit.ai --> --------- Co-authored-by: Alaister Young <[email protected]> Co-authored-by: Ivan Vasilov <[email protected]>
184 lines
6.7 KiB
JavaScript
184 lines
6.7 KiB
JavaScript
#!/usr/bin/env node
|
|
// Standalone Node HTTP server that hosts the production studio build.
|
|
//
|
|
// We export the fetch-handler shape from `dist/server/server.js` because
|
|
// Vercel consumes it directly (see `apps/studio/api/server.js`). For
|
|
// self-hosted / e2e, we need an HTTP listener of our own — this is that
|
|
// listener.
|
|
//
|
|
// Responsibilities:
|
|
// - Load env files in vite preview's order so non-NEXT_PUBLIC_* values
|
|
// (POSTGRES_PASSWORD, PG_META_CRYPTO_KEY, etc.) are in process.env
|
|
// at request time. NEXT_PUBLIC_* are already inlined into the bundle
|
|
// at build time and don't need to be re-loaded.
|
|
// - Serve static client assets from `dist/client/` directly with the
|
|
// right MIME types and cache headers.
|
|
// - Forward everything else to the TanStack Start handler exported
|
|
// from `dist/server/server.js`.
|
|
import { createReadStream } from 'node:fs'
|
|
import { stat } from 'node:fs/promises'
|
|
import { createServer } from 'node:http'
|
|
import path from 'node:path'
|
|
import { Readable } from 'node:stream'
|
|
import { fileURLToPath } from 'node:url'
|
|
|
|
import { readEnvFiles } from './lib/env.js'
|
|
|
|
const studioRoot = path.resolve(path.dirname(fileURLToPath(import.meta.url)), '..')
|
|
const clientDir = path.join(studioRoot, 'dist/client')
|
|
const mode = process.env.MODE || 'production'
|
|
|
|
const envFiles = ['.env', '.env.local', `.env.${mode}`, `.env.${mode}.local`]
|
|
const parsed = readEnvFiles(studioRoot, envFiles)
|
|
// Don't clobber values the shell already provides — match `vite preview`.
|
|
for (const [k, v] of Object.entries(parsed)) {
|
|
if (process.env[k] !== undefined) continue
|
|
process.env[k] = v.replace(
|
|
/\$\{?([A-Za-z_][A-Za-z0-9_]*)\}?/g,
|
|
(_, name) => process.env[name] ?? parsed[name] ?? ''
|
|
)
|
|
}
|
|
|
|
const { default: handler } = await import(path.join(studioRoot, 'dist/server/server.js'))
|
|
|
|
const mimeByExt = new Map([
|
|
['.js', 'application/javascript; charset=utf-8'],
|
|
['.mjs', 'application/javascript; charset=utf-8'],
|
|
['.css', 'text/css; charset=utf-8'],
|
|
['.html', 'text/html; charset=utf-8'],
|
|
['.json', 'application/json; charset=utf-8'],
|
|
['.map', 'application/json; charset=utf-8'],
|
|
['.png', 'image/png'],
|
|
['.jpg', 'image/jpeg'],
|
|
['.jpeg', 'image/jpeg'],
|
|
['.gif', 'image/gif'],
|
|
['.svg', 'image/svg+xml'],
|
|
['.ico', 'image/x-icon'],
|
|
['.woff', 'font/woff'],
|
|
['.woff2', 'font/woff2'],
|
|
['.txt', 'text/plain; charset=utf-8'],
|
|
['.webmanifest', 'application/manifest+json'],
|
|
])
|
|
|
|
// Vite emits hashed filenames (e.g. `index-DB4J79t9.js`) for everything
|
|
// it bundles. Those are content-addressed so we serve them immutable.
|
|
const HASHED_RE = /-[A-Za-z0-9_-]{6,}\.[a-z0-9]+$/
|
|
|
|
async function serveStatic(req, res) {
|
|
let pathname
|
|
try {
|
|
pathname = new URL(req.url, 'http://localhost').pathname
|
|
} catch {
|
|
return false
|
|
}
|
|
if (pathname === '/' || pathname.endsWith('/')) return false
|
|
if (pathname.includes('..') || pathname.includes('\\')) return false
|
|
const filePath = path.join(clientDir, pathname)
|
|
if (!filePath.startsWith(clientDir + path.sep)) return false
|
|
|
|
let st
|
|
try {
|
|
st = await stat(filePath)
|
|
} catch {
|
|
return false
|
|
}
|
|
if (!st.isFile()) return false
|
|
|
|
res.statusCode = 200
|
|
res.setHeader(
|
|
'content-type',
|
|
mimeByExt.get(path.extname(filePath).toLowerCase()) ?? 'application/octet-stream'
|
|
)
|
|
res.setHeader('content-length', String(st.size))
|
|
res.setHeader(
|
|
'cache-control',
|
|
HASHED_RE.test(pathname) ? 'public, max-age=31536000, immutable' : 'no-cache'
|
|
)
|
|
await new Promise((resolve, reject) => {
|
|
const stream = createReadStream(filePath)
|
|
stream.on('error', reject)
|
|
stream.on('end', resolve)
|
|
stream.pipe(res)
|
|
})
|
|
return true
|
|
}
|
|
|
|
function toWebRequest(req) {
|
|
const protocol = req.socket.encrypted ? 'https' : 'http'
|
|
const url = `${protocol}://${req.headers.host ?? 'localhost'}${req.url}`
|
|
const headers = new Headers()
|
|
for (const [k, v] of Object.entries(req.headers)) {
|
|
if (k.startsWith(':')) continue
|
|
if (Array.isArray(v)) for (const vv of v) headers.append(k, vv)
|
|
else if (v !== undefined) headers.set(k, v)
|
|
}
|
|
const init = { method: req.method, headers }
|
|
// Only attach a body for methods that can carry one AND that actually
|
|
// have body bytes coming. Wrapping `req` in `Readable.toWeb(req)` for
|
|
// requests where Node has nothing to deliver leaves undici's
|
|
// `extractBody` looking at an already-consumed stream and throwing
|
|
// `TypeError: Response body object should not be disturbed or locked`
|
|
// at the `new Request(...)` call below.
|
|
const contentLength = Number(req.headers['content-length'] ?? '0')
|
|
const hasBody =
|
|
req.method !== 'GET' &&
|
|
req.method !== 'HEAD' &&
|
|
(contentLength > 0 || req.headers['transfer-encoding'] === 'chunked')
|
|
if (hasBody) {
|
|
init.body = Readable.toWeb(req)
|
|
init.duplex = 'half'
|
|
}
|
|
return new Request(url, init)
|
|
}
|
|
|
|
async function pipeWebResponse(response, res) {
|
|
res.statusCode = response.status
|
|
// The Headers iterator collapses duplicate keys, and for `set-cookie` it joins
|
|
// every cookie into one comma-separated value — which corrupts auth/session
|
|
// cookies. Pull the cookies out separately via getSetCookie() and set them as
|
|
// an array so each one becomes its own header.
|
|
const setCookies =
|
|
typeof response.headers.getSetCookie === 'function' ? response.headers.getSetCookie() : []
|
|
for (const [k, v] of response.headers) {
|
|
if (k.toLowerCase() === 'set-cookie') continue
|
|
res.setHeader(k, v)
|
|
}
|
|
if (setCookies.length > 0) res.setHeader('set-cookie', setCookies)
|
|
if (!response.body) {
|
|
res.end()
|
|
return
|
|
}
|
|
// Pipe via Readable.fromWeb so the underlying stream gets proper backpressure
|
|
// and gets released cleanly. `for await (chunk of response.body)` works in
|
|
// simple cases but can leave the body in a "disturbed / locked" state when
|
|
// the handler internally peeks at it — surfacing as
|
|
// `TypeError: Response body object should not be disturbed or locked` on a
|
|
// subsequent request.
|
|
await new Promise((resolve, reject) => {
|
|
const readable = Readable.fromWeb(response.body)
|
|
readable.on('error', reject)
|
|
res.on('error', reject)
|
|
res.on('close', resolve)
|
|
res.on('finish', resolve)
|
|
readable.pipe(res)
|
|
})
|
|
}
|
|
|
|
const port = Number(process.env.PORT || 8082)
|
|
createServer(async (req, res) => {
|
|
try {
|
|
if (await serveStatic(req, res)) return
|
|
const response = await handler.fetch(toWebRequest(req))
|
|
await pipeWebResponse(response, res)
|
|
} catch (err) {
|
|
console.error('[serve] request failed:', err)
|
|
if (!res.headersSent) {
|
|
res.statusCode = 500
|
|
res.setHeader('content-type', 'text/plain; charset=utf-8')
|
|
}
|
|
res.end('Internal Server Error')
|
|
}
|
|
}).listen(port, () => {
|
|
console.log(`Studio listening on http://localhost:${port} (mode=${mode})`)
|
|
})
|