mirror of
https://github.com/stack-auth/stack.git
synced 2026-06-04 21:04:37 +08:00
Some checks failed
all-good: Did all the other checks pass? / all-good (push) Has been cancelled
Ensure Prisma migrations are in sync with the schema / check_prisma_migrations (22.x) (push) Has been cancelled
DB migration compat / Check if migrations changed (push) Has been cancelled
Docker Server Build and Push / Docker Build and Push Server (push) Has been cancelled
Docker Server Build and Run / docker (push) Has been cancelled
Runs E2E API Tests (Local Emulator) / E2E Tests (Local Emulator, Node ${{ matrix.node-version }}) (22.x) (push) Has been cancelled
Runs E2E API Tests / E2E Tests (Node ${{ matrix.node-version }}, Freestyle ${{ matrix.freestyle-mode }}) (mock, 22.x) (push) Has been cancelled
Runs E2E API Tests / E2E Tests (Node ${{ matrix.node-version }}, Freestyle ${{ matrix.freestyle-mode }}) (prod, 22.x) (push) Has been cancelled
Runs E2E API Tests with custom port prefix / build (22.x) (push) Has been cancelled
Runs E2E Fallback Tests / E2E Fallback Tests (Node ${{ matrix.node-version }}) (22.x) (push) Has been cancelled
Lint & build / lint_and_build (24) (push) Has been cancelled
TOC Generator / TOC Generator (push) Has been cancelled
DB migration compat / Back-compat — Current branch migrations with ${{ needs.check-migrations-changed.outputs.base_branch }} branch code (push) Has been cancelled
DB migration compat / Forward-compat — Current branch code with ${{ needs.check-migrations-changed.outputs.base_branch }} branch migrations (push) Has been cancelled
DB migration compat / No migration changes (skipped) (push) Has been cancelled
## Summary **Stacked on [#1475](https://github.com/hexclave/stack-auth/pull/1475)** (`cl/hexclave-pr1`, the invisible compatibility layer). Diff vs that base = the actual PR 2 code. This is **PR 2 of the Stack Auth → Hexclave rebrand: the visible flip**. Old wire identifiers (cookies, request/response headers, Bearer prefix, JWT issuers, MCP tool name) keep working indefinitely via PR 1's dual-accept. This PR flips every user-visible surface — package names taught in docs, SDK class names in code examples, dashboard setup snippets, page titles, error messages, email content, CLI binary, default base URLs, GitHub repo slug, contributor guidance — to the Hexclave brand. See [`RENAME-TO-HEXCLAVE.md`](./RENAME-TO-HEXCLAVE.md) → *"PR 2: Rebrand to Hexclave (visible)"* for the full per-work-area spec. ## What's implemented (per the plan's PR 2 scope) - **SDK base URLs** flipped: `defaultBaseUrl` and `defaultAnalyticsBaseUrl` in [common.ts](packages/template/src/lib/stack-app/apps/implementations/common.ts:127) → `https://api.hexclave.com` / `https://r.hexclave.com`. PR 1's [`getHardcodedFallbackUrls`](packages/stack-shared/src/utils/urls.tsx:199) table now keys on the Hexclave domain. - **Domain inventory sweep** (16 subdomains from the plan): every `api/app/docs/discord/demo/mcp/skill/feedback/test/preview/r/api2/api.staging/idp-jwk-audience/built-with.stack-auth.com` reference in production code, docs-mintlify, examples, READMEs, and contributor guidance flipped to `*.hexclave.com`. Carve-outs: PR 1's intentional JWT issuer dual-accept table in [tokens.tsx](apps/backend/src/lib/tokens.tsx), the legacy `./docs/` folder, the `unified-docs-widget` allowlist (deliberately accepts both during DNS transition), and `url-targets.ts` hosted-component default (baked into existing customer deploys). - **`@deprecated` JSDoc** on every `Stack*` public export ([packages/template/src/lib/stack-app/index.ts](packages/template/src/lib/stack-app/index.ts) + [packages/template/src/index.ts](packages/template/src/index.ts)) — `StackClientApp`, `StackServerApp`, `StackAdminApp` + every constructor/options/JSON type, `StackHandler`, `StackProvider`, `StackTheme`, `useStackApp`, `defineStackConfig`, `StackConfig`. Hexclave\* aliases are now canonical. - **Runtime `console.warn`** ([packages/template/src/internal/deprecation-warning.ts](packages/template/src/internal/deprecation-warning.ts)) — once-per-process when the SDK is loaded from a `@stackframe/*` artifact. Detection uses the existing `STACK_COMPILE_TIME_CLIENT_PACKAGE_VERSION_SENTINEL` (rewritten at build time to e.g. `js @stackframe/stack@2.8.92` or `js @hexclave/next@1.0.0`); `@hexclave/*` mirror artifacts short-circuit the warning. - **Tier 3 data migration**: new idempotent SQL migration [`20260523000000_rename_internal_project_to_hexclave`](apps/backend/prisma/migrations/20260523000000_rename_internal_project_to_hexclave/migration.sql) — updates the internal Project `displayName` 'Stack Dashboard' → 'Hexclave Dashboard' and `description` only if both still hold the pre-rebrand defaults. Operator-renamed projects untouched, missing row no-ops, re-runs are no-ops. [`seed.ts`](apps/backend/prisma/seed.ts:87) default flipped. `getSharedEmailConfig("Stack Auth")` → `("Hexclave")`. - **Tier 4 brand strings** (mechanical sweep, ~340 files): - Page + OpenAPI titles (Hexclave API / Dashboard / REST API / Webhooks API / Documentation). OpenAPI `info.description` documents `X-Hexclave-*` headers as canonical with compat note on `X-Stack-*`. - `HexclaveAssertionError` message text ([errors.tsx:71](packages/stack-shared/src/utils/errors.tsx:71)) — "an error in Stack." → "an error in Hexclave." - Known-error message templates ([known-errors.tsx](packages/stack-shared/src/known-errors.tsx)) flipped to lead with `x-hexclave-*` + the new `docs.hexclave.com` URL; legacy `x-stack-*` mentioned as compat aliases. **25 e2e test files updated in lockstep**. - Email content: failed-emails-digest body, sendTestEmail recipient (now `sent-with-hexclave.com`), test-email-recipient default. - `CHANGELOG.md` title → "Hexclave Changelog". - `AGENTS.md` env var convention: new vars prefix `HEXCLAVE_` / `NEXT_PUBLIC_HEXCLAVE_` for Category A/B; legacy `STACK_*` explicitly noted as accepted via PR 1's dual-read. - **CLI / init wizard**: - Every dashboard setup snippet, init-stack template, and docs-mintlify page teaches `npx @hexclave/cli@latest init` (was `@stackframe/stack-cli`). [setup-page.tsx](apps/dashboard/src/app/(main)/(protected)/projects/[projectId]/(overview)/setup-page.tsx) + [link-existing-onboarding](apps/dashboard/src/app/(main)/(protected)/(outside-dashboard)/new-project/page-client-parts/link-existing-onboarding.tsx). - [init-stack](packages/init-stack/src/index.ts:634) `STACK_*_INSTALL_PACKAGE_NAME_OVERRIDE` defaults flipped to `@hexclave/*`. - Generated `stack/client.ts` / `stack/server.ts` import from `@hexclave/next` and reference `HexclaveClientApp` / `HexclaveServerApp`. - Internal `StackAuthKeys` dashboard component renamed to `HexclaveKeys`. - **docs-mintlify rewrite** (legacy `./docs/` intentionally untouched per scoping decision): - **78 MDX files swept**. `@stackframe/{react,stack,js,tanstack-start,...}` → `@hexclave/{react,stack,js,...}` in install snippets and code blocks; `Stack*` SDK class names → `Hexclave*` in all code examples; 'Stack Auth' brand phrase → 'Hexclave'. - `openapi/{server,admin,client,webhooks}.json` titles → 'Hexclave REST API' / 'Hexclave Webhooks API'. - **Generators flipped before regeneration**: - [`packages/stack-shared/src/helpers/init-prompt.ts`](packages/stack-shared/src/helpers/init-prompt.ts), [`/ai/prompts.ts`](packages/stack-shared/src/ai/prompts.ts), [`apps/backend/src/lib/ai/prompts.ts`](apps/backend/src/lib/ai/prompts.ts), [`apps/backend/src/lib/ai/tools/create-email-{template,draft}.ts`](apps/backend/src/lib/ai/tools/create-email-template.ts), [`apps/skills/src/app/route.ts`](apps/skills/src/app/route.ts) (taught MCP tool → `ask_hexclave` with compat note; CLI binary teach → `hexclave`), [`docs-mintlify/snippets/home-prompt-island.jsx`](docs-mintlify/snippets/home-prompt-island.jsx), [`packages/template/README.md`](packages/template/README.md) + integrations/convex/component/README.md. - `generate-sdks` propagated changes to `packages/{react,stack,js}`. - **OpenAPI dual-documentation**: [`apps/backend/src/app/api/latest/route.ts`](apps/backend/src/app/api/latest/route.ts) now lists `X-Hexclave-*` headers as primary documented schemas with `X-Stack-*` duplicates marked `.optional()` (both accepted at runtime by PR 1's normalize-at-proxy shim). - **`@stackframe/emails` virtual module**: dual-aliased to `@hexclave/emails` at the bundler boundary ([email-rendering.tsx:89](apps/backend/src/lib/email-rendering.tsx:89)). Stored email templates continue to import from either name; new AI-generated templates and the system prompt teach `@hexclave/emails`. - **Tier 2 mirror-publish wiring** (new this PR, lays the groundwork for `@hexclave/*` first publish): - [`scripts/rewrite-packages-to-hexclave.ts`](scripts/rewrite-packages-to-hexclave.ts) — rewrites 9 publishable `@stackframe/*` → `@hexclave/*` `package.json` files (reads `HEXCLAVE_VERSION` env or `--version=` flag), pins cross-deps to the shared `@hexclave` version, registers `hexclave` bin alongside `stack` for `@hexclave/cli`. - [`.github/workflows/npm-publish.yaml`](.github/workflows/npm-publish.yaml) appended with rewrite-then-republish step. `pnpm publish` skips already-on-npm versions so reruns are safe. - **Sender email domain**: `noreply@stackframe.co` → `noreply@sent-with-hexclave.com` (the dedicated transactional-sender domain split per the plan, to isolate bulk deliverability from `hexclave.com` reputation); `security@` / `team@stack-auth.com` inbound mailboxes → `@hexclave.com`. - **Self-host docs**: docker network / container names in the bash examples flipped from `stack-auth` to `hexclave` (`hexclave-postgres`, `hexclave-clickhouse`, `hexclave.env`). The docker image tag `stackauth/server:latest` stays per the plan's locked decision. - **GitHub repo slug**: `hexclave/stack-auth` → `hexclave/hexclave` in every `package.json` `repository` field, README link, CHANGELOG raw-asset URL. ## Carve-outs (deliberately untouched) - **[`apps/backend/src/lib/tokens.tsx`](apps/backend/src/lib/tokens.tsx)** JWT issuer dual-accept table — PR 1 intentional infrastructure, kept indefinitely. - **Legacy `./docs/` folder** — per scoping decision (only `docs-mintlify/` rewritten). - **`unified-docs-widget` hostname allowlist** — accepts both `.hexclave.com` (canonical) and `.stack-auth.com` (transition window) for DNS rollout. - **`url-targets.ts`** hosted-domain default `.built-with-stack-auth.com` — wire identifier baked into existing customer deploys; indefinite read-fallback. - **Binary visual assets** (logos, favicons, OG images, README screenshots) — out of scope for this PR. Need design work; tracked separately. ## Verification - **`pnpm typecheck`** on `packages/{template,stack-shared,react,stack,js}` + `apps/dashboard`: **all green**. The remaining backend / e-commerce-demo typecheck errors are pre-existing (Prisma codegen output + `./generated/api-versions.json` not present in fresh worktrees without `pnpm run codegen-prisma` + a live DB) and unrelated to this diff. - **`pnpm lint`** on the same 6 packages: all green. - **Final grep** for residual `Stack Auth` / `stack-auth.com` / `@stackframe/stack-cli@latest` references: zero outside the intentional carve-outs above. - **25 e2e test files updated in lockstep** with the known-error message changes (asserted strings flipped to match the new x-hexclave-* + compat-note messages). ## Deploy blockers (ops sequencing before this rebrand goes live) This PR is code-complete, but the rebrand's visible surfaces (SDK default URLs, dashboard links, npm READMEs, REST error messages, runtime deprecation warning) all point at `*.hexclave.com` / `@hexclave/*` resources that don't exist yet. None of these are fixable from a PR — they're ops/registrar/npm work that has to be sequenced before merging this to a release tag. Suggested ordering, hardest blockers first: ### Tier 1 — required before customer-facing deploy (everything below this line *will visibly break customers on day 1* if skipped) 1. **DNS + TLS for `api.hexclave.com` + `api1./api2.hexclave.com`** → must point at the same backend that serves `api.stack-auth.com` (or a backend that mirrors PR 1's dual-accept). The SDK's new `defaultBaseUrl` is `https://api.hexclave.com`; every customer that relied on the old default and upgrades to a post-PR2 SDK build sends API requests here. Until this resolves, every default-config customer's API call NXDOMAINs. 2. **DNS for `app.hexclave.com`** → the dashboard. Referenced in the SDK's default-error messages ("Please create a project on the Hexclave dashboard at https://app.hexclave.com"), the init-stack flow's `wizard-congrats` redirect, and the OAuth dashboard handoff. 3. **DNS for `docs.hexclave.com`** + Mintlify deploy → the SDK runtime deprecation warning (`https://docs.hexclave.com/migration`), every README, every "Learn more" link in the dashboard, and every REST API error body (`/api/overview#authentication`) points here. The MDX is in this PR; the docs build target needs DNS. 4. **DNS for `mcp.hexclave.com`** → the MCP server endpoint that every taught agent integration (`claude mcp add ...`, `cursor`, `codex`, `vscode`) registers. Until this resolves, every `npx @hexclave/cli@latest init` MCP-registration step fails. 5. **Reserve the `@hexclave` npm scope + set repo variable `HEXCLAVE_VERSION`** → the mirror-publish step in `.github/workflows/npm-publish.yaml` is gated on this variable. Without it, the entire taught onboarding command `npx @hexclave/cli@latest init` 404s from the npm registry, *and* every README that says "install `@hexclave/next`" leads to install failure. Pick the initial version intentionally (`1.0.0` or aligned to `@stackframe/stack`); don't accept a silent default. ### Tier 2 — required before announcing the rebrand publicly (lookalike or low-traffic surfaces, but visibly broken) 6. **DNS for `r.hexclave.com`** → the analytics beacon `defaultAnalyticsBaseUrl`. Silent failure if missing (analytics drops), but should land alongside Tier 1. 7. **Register `sent-with-hexclave.com` + full email auth (SPF / DKIM / DMARC)** → the new default sender domain for shared-sender transactional emails. Without it the dashboard "send test email" path emits bounces, and shared-sender flows (`getSharedEmailConfig("Hexclave")`) deliver to spam at best. 8. **MX + SPF / DMARC for `hexclave.com`** → `team@hexclave.com` and `security@hexclave.com` mailboxes. The security disclosure mailbox is referenced in [`.github/SECURITY.md`](.github/SECURITY.md); `team@hexclave.com` is the actual recipient of internal feedback emails sent at runtime by [`apps/backend/src/lib/internal-feedback-emails.tsx`](apps/backend/src/lib/internal-feedback-emails.tsx). Today, every runtime feedback email bounces. 9. **DNS for `skill.hexclave.com`** → the canonical AI-agent skill fetch URL (the agent bootstrap pivot). Without it, the entire "agent downloads `SKILL.md` from a known URL" flow taught in [`packages/stack-shared/src/helpers/init-prompt.ts`](packages/stack-shared/src/helpers/init-prompt.ts) fails. 10. **Create `github.com/hexclave/hexclave` as a public repo** (even as a redirect to `hexclave/stack-auth`) **OR** rewrite every `package.json` `"repository"` field + dashboard footer "view on GitHub" link to point at `hexclave/stack-auth` (which already exists). Currently every npm package page's "Repository" link is dead, and the dashboard's GitHub button + dev-tool repo link are dead. ### Tier 3 — broken but low-visibility / low-traffic 11. **DNS for `discord.hexclave.com`** → Discord invite redirect, used in every README's chip and the dashboard footer. 12. **DNS for `demo.hexclave.com`** → "✨ Demo" badge in every npm package README. Broken-image badge on the package page. 13. **DNS + TLS for `built-with-hexclave.com`** → optional hosted-handler domain (the default reverted to `.built-with-stack-auth.com` in this PR's carve-outs, so this only matters for projects that manually flip). ## Other follow-ups (not deploy-blocking) - **E2E snapshot regen across the full suite** for the dual-emitted `x-hexclave-*` response headers (PR 1 follow-up; `vitest -u` in CI absorbs). - **Binary visual assets** — logos, favicons, OG images, README screenshots; need design pass. - **Backend OpenAPI fumadocs regen** in CI flow — the JSON files in `docs-mintlify/openapi/` are committed but regen runs in CI. Verify the workflow that does this still works against the post-PR2 source. - **Backend typecheck infra debt** — needs `codegen-prisma` + `codegen-route-info` to clear; pre-existing, unaffected by this PR. ## Test plan - [ ] CI runs full e2e suite (with `vitest -u` to absorb residual snapshot deltas, then committed back). - [ ] Spot-check: new `@hexclave/cli init` (once published) generates `hexclave.config.ts` and works against a fresh project. - [ ] Spot-check: existing customer with `@stackframe/stack` import sees the once-per-process `console.warn` recommending `@hexclave/next` on SDK init. - [ ] Manual: dashboard setup page renders the `npx @hexclave/cli@latest init` snippet and the `x-hexclave-publishable-client-key` API header in the curl example. - [ ] Manual: a fresh `pnpm run prisma migrate` against a clean DB sets the internal project displayName to 'Hexclave Dashboard'. --------- Co-authored-by: Konstantin Wohlwend <n2d4xc@gmail.com>
291 lines
12 KiB
Docker
291 lines
12 KiB
Docker
# Hexclave Local Emulator — All-in-One Image
|
|
# Packages: PostgreSQL 16, Redis 7, Inbucket, Svix, ClickHouse, MinIO, QStash
|
|
# + built Hexclave backend and dashboard
|
|
|
|
ARG NODE_VERSION=22.21.1
|
|
|
|
# ── Node.js build stages ──────────────────────────────────────────────────────
|
|
|
|
FROM node:${NODE_VERSION} AS node-base
|
|
|
|
WORKDIR /app
|
|
|
|
RUN apt-get update && \
|
|
apt-get upgrade -y && \
|
|
rm -rf /var/lib/apt/lists
|
|
|
|
ENV PNPM_HOME=/pnpm
|
|
ENV PATH=$PNPM_HOME:$PATH
|
|
|
|
RUN corepack enable
|
|
RUN corepack prepare pnpm@10.23.0 --activate
|
|
RUN pnpm add -g turbo
|
|
RUN pnpm add -g tsx
|
|
|
|
|
|
FROM node-base AS pruner
|
|
|
|
COPY . .
|
|
|
|
RUN tsx ./scripts/generate-sdks.ts
|
|
|
|
# https://turbo.build/repo/docs/guides/tools/docker
|
|
RUN turbo prune --scope=@stackframe/backend --scope=@stackframe/dashboard --docker
|
|
|
|
|
|
FROM node-base AS builder
|
|
|
|
# copy over package.json files and install dependencies
|
|
COPY --from=pruner /app/out/json/ .
|
|
COPY --from=pruner /app/out/pnpm-lock.yaml .
|
|
COPY .gitignore .
|
|
COPY pnpm-workspace.yaml .
|
|
COPY turbo.json .
|
|
COPY configs ./configs
|
|
COPY --from=pruner /app/scripts/postinstall-patch-next-async-debug-info.mjs ./scripts/
|
|
RUN --mount=type=cache,id=pnpm,target=/pnpm/store STACK_SKIP_TEMPLATE_GENERATION=true pnpm install --frozen-lockfile
|
|
|
|
# copy over the rest of the code for the build
|
|
COPY --from=pruner /app/out/full/ .
|
|
|
|
# docs are currently required for the NextJS backend build, but won't exist in the final image
|
|
COPY docs ./docs
|
|
|
|
# https://nextjs.org/docs/pages/api-reference/next-config-js/output
|
|
ENV NEXT_CONFIG_OUTPUT=standalone
|
|
ENV NEXT_PUBLIC_STACK_STRIPE_PUBLISHABLE_KEY=pk_test_mock_publishable_key_for_local_emulator
|
|
|
|
# Build the backend NextJS app
|
|
RUN pnpm turbo run docker-build --filter=@stackframe/backend... --filter=@stackframe/dashboard...
|
|
|
|
# Build the self-host seed script.
|
|
# tsdown -> rolldown is multi-threaded Rust; under qemu-user (cross-arch
|
|
# arm64-on-amd64) its futex emulation occasionally deadlocks and the build
|
|
# hangs forever. Bound each attempt and retry to ride out the race.
|
|
RUN cd apps/backend && \
|
|
attempt=1; \
|
|
while :; do \
|
|
timeout --kill-after=30s 600s pnpm build-self-host-migration-script && break; \
|
|
rc=$?; \
|
|
if [ "$attempt" -ge 3 ]; then \
|
|
echo "build-self-host-migration-script failed after $attempt attempts (last rc=$rc)" >&2; \
|
|
exit "$rc"; \
|
|
fi; \
|
|
echo "build-self-host-migration-script attempt $attempt failed (rc=$rc); retrying..." >&2; \
|
|
attempt=$((attempt + 1)); \
|
|
done
|
|
|
|
|
|
# Prune node_modules for runtime: remove dev tools, heavy UI packages,
|
|
# duplicate framework copies, and native binaries not needed by the
|
|
# migration script or server at runtime.
|
|
FROM builder AS migration-pruner
|
|
RUN cp -a /app/node_modules /pruned-node_modules && \
|
|
cd /pruned-node_modules/.pnpm && \
|
|
rm -rf \
|
|
# Dev tools (never needed at runtime)
|
|
typescript@* eslint@* eslint-*@* @typescript-eslint+*@* \
|
|
prettier@* vitest@* jsdom@* turbo@* turbo-*@* \
|
|
tsdown@* @changesets+*@* codebuff@* \
|
|
@testing-library+*@* vite@* vite-*@* @vitejs+*@* \
|
|
# Heavy UI packages (already traced into Next.js standalone bundles)
|
|
monaco-editor@* \
|
|
three@* three-globe@* globe.gl@* react-globe*@* \
|
|
react-icons@* lucide-react@* @phosphor-icons+*@* \
|
|
# Large optional packages not needed by migration script
|
|
posthog-js@* \
|
|
@prisma+studio-core@* @prisma+dev@* @prisma+query-plan-executor@* \
|
|
convex@* @electric-sql+*@* \
|
|
next@14* @next+swc-*@14* \
|
|
# Native build binaries not needed at runtime
|
|
@esbuild+*@* esbuild@* @rolldown+*@* \
|
|
# Duplicate date-fns versions (keep v4 only)
|
|
date-fns@2* date-fns@3*
|
|
|
|
|
|
# ── Freestyle mock build ─────────────────────────────────────────────────────
|
|
|
|
FROM node-base AS freestyle-mock-builder
|
|
WORKDIR /freestyle-mock
|
|
COPY docker/dependencies/freestyle-mock/Dockerfile /tmp/freestyle-mock-dockerfile
|
|
# Extract the inline package.json and server.mjs from the Dockerfile's RUN cat commands,
|
|
# then install dependencies. This avoids duplicating the source.
|
|
RUN node -e " \
|
|
const fs = require('fs'); \
|
|
const df = fs.readFileSync('/tmp/freestyle-mock-dockerfile', 'utf8'); \
|
|
const pkgMatch = df.match(/cat <<'EOF' > package\\.json\\n([\\s\\S]*?)\\nEOF/); \
|
|
fs.writeFileSync('package.json', pkgMatch[1]); \
|
|
const srvMatch = df.match(/cat <<'EOF' > server\\.mjs\\n([\\s\\S]*?)\\nEOF/); \
|
|
let server = srvMatch[1]; \
|
|
server = server.replace( \
|
|
'from \"fs/promises\"', \
|
|
'from \"fs/promises\"; import { symlinkSync } from \"fs\"' \
|
|
); \
|
|
server = server.replace( \
|
|
'await mkdir(workDir, { recursive: true });', \
|
|
'await mkdir(workDir, { recursive: true }); try { symlinkSync(\"/app/freestyle-mock/node_modules\", join(workDir, \"node_modules\")); } catch {}' \
|
|
); \
|
|
fs.writeFileSync('server.mjs', server); \
|
|
"
|
|
RUN npm install
|
|
|
|
|
|
# ── Mock OAuth server build ───────────────────────────────────────────────────
|
|
|
|
FROM node-base AS mock-oauth-builder
|
|
WORKDIR /mock-oauth
|
|
COPY apps/mock-oauth-server/package.json .
|
|
RUN pnpm install && pnpm add esbuild --save-dev
|
|
COPY apps/mock-oauth-server/src ./src
|
|
RUN npx esbuild src/index.ts --bundle --platform=node --target=node22 --outfile=dist/index.cjs
|
|
|
|
|
|
# ── Service binary stages ─────────────────────────────────────────────────────
|
|
|
|
FROM stripe/stripe-mock:v0.195.0 AS stripe-mock-bin
|
|
FROM inbucket/inbucket:3.1.0 AS inbucket-bin
|
|
FROM svix/svix-server:v1.88.0 AS svix-bin
|
|
FROM clickhouse/clickhouse-server:25.10 AS clickhouse-bin
|
|
FROM minio/minio:RELEASE.2025-09-07T16-13-09Z AS minio-bin
|
|
FROM minio/mc:RELEASE.2025-02-21T16-00-46Z AS mc-bin
|
|
|
|
FROM bgodil/qstash:latest AS qstash-bin
|
|
RUN cp $(which qstash) /qstash-binary 2>/dev/null || \
|
|
cp $(find / -name 'qstash' -type f -executable 2>/dev/null | head -1) /qstash-binary || \
|
|
{ echo "ERROR: qstash binary not found" >&2; exit 1; }
|
|
|
|
|
|
# ── Strip / compress service binaries (parallel stages) ──────────────────────
|
|
|
|
FROM debian:trixie-slim AS upx-compress
|
|
RUN apt-get update && apt-get install -y --no-install-recommends upx-ucl binutils && \
|
|
rm -rf /var/lib/apt/lists/*
|
|
COPY --from=clickhouse-bin /usr/bin/clickhouse /out/clickhouse
|
|
COPY --from=svix-bin /usr/local/bin/svix-server /out/svix-server
|
|
COPY --from=minio-bin /usr/bin/minio /out/minio
|
|
COPY --from=mc-bin /usr/bin/mc /out/mc
|
|
COPY --from=qstash-bin /qstash-binary /out/qstash
|
|
RUN chmod u+w /out/* && \
|
|
# Intentionally NOT stripping /out/clickhouse. The clickhouse binary is a
|
|
# self-extracting compressed executable (a small loader with a ZSTD
|
|
# payload appended after the section table); strip rewrites the ELF and
|
|
# can invalidate the loader's "find my payload" lookup, causing the
|
|
# decompressor to spin on garbage with zero log output — the exact
|
|
# symptom seen on cross-arch TCG runs. Savings from stripping would be
|
|
# only the tiny bootstrap anyway since the payload isn't in any section.
|
|
strip --strip-all /out/minio /out/svix-server /out/mc /out/qstash && \
|
|
upx -9 /out/minio /out/svix-server /out/mc /out/qstash
|
|
|
|
|
|
# ── Final image ───────────────────────────────────────────────────────────────
|
|
|
|
FROM debian:trixie-slim
|
|
|
|
ENV DEBIAN_FRONTEND=noninteractive
|
|
|
|
RUN apt-get update && \
|
|
apt-get install -y --no-install-recommends \
|
|
gnupg2 \
|
|
lsb-release \
|
|
curl \
|
|
ca-certificates \
|
|
&& echo "deb http://apt.postgresql.org/pub/repos/apt $(lsb_release -cs)-pgdg main" \
|
|
> /etc/apt/sources.list.d/pgdg.list \
|
|
&& curl -fsSL https://www.postgresql.org/media/keys/ACCC4CF8.asc \
|
|
| gpg --dearmor -o /etc/apt/trusted.gpg.d/postgresql.gpg \
|
|
&& apt-get update \
|
|
&& apt-get install -y --no-install-recommends \
|
|
postgresql-16 \
|
|
postgresql-client-16 \
|
|
redis-server \
|
|
supervisor \
|
|
gosu \
|
|
procps \
|
|
libssl3 \
|
|
openssl \
|
|
socat \
|
|
&& apt-get purge -y --auto-remove gnupg2 lsb-release \
|
|
&& rm -rf /var/lib/apt/lists/* /usr/share/doc /usr/share/man /usr/share/i18n
|
|
|
|
# Node.js runtime (binary only — app bundles include all JS dependencies)
|
|
COPY --from=node-base /usr/local/bin/node /usr/local/bin/node
|
|
|
|
# Inbucket
|
|
COPY --from=inbucket-bin /opt/inbucket /opt/inbucket
|
|
|
|
# Stripe mock
|
|
COPY --from=stripe-mock-bin /bin/stripe-mock /usr/local/bin/stripe-mock
|
|
|
|
# Svix (UPX-compressed)
|
|
COPY --from=upx-compress /out/svix-server /usr/local/bin/svix-server
|
|
|
|
# ClickHouse (stripped only)
|
|
COPY --from=upx-compress /out/clickhouse /usr/bin/clickhouse
|
|
RUN ln -sf /usr/bin/clickhouse /usr/bin/clickhouse-server && \
|
|
ln -sf /usr/bin/clickhouse /usr/bin/clickhouse-client
|
|
|
|
# MinIO (UPX-compressed)
|
|
COPY --from=upx-compress /out/minio /usr/local/bin/minio
|
|
COPY --from=upx-compress /out/mc /usr/local/bin/mc
|
|
|
|
# QStash (UPX-compressed)
|
|
COPY --from=upx-compress --chmod=755 /out/qstash /usr/local/bin/qstash
|
|
|
|
# App
|
|
WORKDIR /app
|
|
COPY --from=builder /app/apps/backend/.next/standalone ./
|
|
COPY --from=builder /app/apps/backend/.next/static ./apps/backend/.next/static
|
|
COPY --from=builder /app/apps/backend/prisma ./apps/backend/prisma
|
|
COPY --from=builder /app/apps/backend/dist ./apps/backend/dist
|
|
COPY --from=builder /app/apps/backend/node_modules ./apps/backend/node_modules
|
|
COPY --from=builder /app/apps/dashboard/.next/standalone ./
|
|
COPY --from=builder /app/apps/dashboard/.next/static ./apps/dashboard/.next/static
|
|
COPY --from=builder /app/apps/dashboard/public ./apps/dashboard/public
|
|
# Save the standalone-traced node_modules (runtime deps only) before the full
|
|
# migration-pruner copy overwrites it. The slim-docker-image step in the QEMU
|
|
# build restores this after migrations are baked in.
|
|
RUN cp -a /app/node_modules /app/node_modules.standalone 2>/dev/null || mkdir -p /app/node_modules.standalone
|
|
COPY --from=migration-pruner /pruned-node_modules ./node_modules
|
|
COPY --from=builder /app/packages ./packages
|
|
|
|
# Mock OAuth server (bundled single file)
|
|
COPY --from=mock-oauth-builder /mock-oauth/dist/index.cjs /app/mock-oauth-server/index.cjs
|
|
|
|
# Freestyle mock (JS execution for email rendering)
|
|
COPY --from=freestyle-mock-builder /freestyle-mock /app/freestyle-mock
|
|
COPY --from=node-base /usr/local/bin/npm /usr/local/bin/npm
|
|
COPY --from=node-base /usr/local/lib/node_modules/npm /usr/local/lib/node_modules/npm
|
|
|
|
RUN mkdir -p \
|
|
/data/postgres \
|
|
/data/redis \
|
|
/data/clickhouse \
|
|
/data/clickhouse/access \
|
|
/data/clickhouse/tmp \
|
|
/data/clickhouse/user_files \
|
|
/data/clickhouse/format_schemas \
|
|
/data/minio \
|
|
/data/inbucket \
|
|
/var/log/supervisor \
|
|
/var/log/clickhouse \
|
|
/etc/clickhouse-server \
|
|
&& chown -R postgres:postgres /data/postgres
|
|
|
|
COPY docker/local-emulator/supervisord.conf /etc/supervisor/conf.d/supervisord.conf
|
|
COPY docker/local-emulator/run-cron-jobs.sh /run-cron-jobs.sh
|
|
COPY docker/local-emulator/entrypoint.sh /entrypoint.sh
|
|
COPY docker/local-emulator/init-services.sh /init-services.sh
|
|
COPY docker/local-emulator/start-app.sh /start-app.sh
|
|
COPY docker/local-emulator/rotate-secrets.sh /usr/local/bin/rotate-secrets
|
|
COPY docker/local-emulator/clickhouse-config.xml /etc/clickhouse-server/config.xml
|
|
COPY docker/local-emulator/clickhouse-users.xml /etc/clickhouse-server/users.xml
|
|
COPY docker/server/entrypoint.sh /app-entrypoint.sh
|
|
RUN chmod +x /entrypoint.sh /init-services.sh /start-app.sh /app-entrypoint.sh /run-cron-jobs.sh /usr/local/bin/rotate-secrets
|
|
|
|
# PostgreSQL: 5432, Redis: 6379, Inbucket: 2500/9001/1100,
|
|
# Svix: 8071, ClickHouse: 8123/9009, MinIO: 9090, QStash: 8080
|
|
# Backend: 8102, Dashboard: 8101, Mock OAuth: 8114
|
|
EXPOSE 5432 6379 2500 9001 1100 8071 8123 9009 9090 8080 8101 8102 8114
|
|
|
|
ENTRYPOINT ["/entrypoint.sh"]
|