Production-ready Next.js boilerplate with: - Runtime env validation (fail-fast on missing vars) - Feature-gated config (S3, Stripe, email, OAuth) - Docker + Coolify deployment pipeline - PostgreSQL + pgvector, MinIO S3, Better Auth - TypeScript strict mode (no ignoreBuildErrors) - i18n (en/es), AI modules, billing, monitoring Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
91 lines
3.7 KiB
TypeScript
91 lines
3.7 KiB
TypeScript
import { defineEnv } from "envin";
|
|
import { vercel } from "envin/presets/zod";
|
|
import * as z from "zod";
|
|
|
|
import { preset as analytics } from "@turbostarter/analytics-web/env";
|
|
import { preset as api } from "@turbostarter/api/env";
|
|
import { preset as i18n } from "@turbostarter/i18n/env";
|
|
import { envConfig, NodeEnv } from "@turbostarter/shared/constants";
|
|
import { ThemeColor, ThemeMode } from "@turbostarter/ui";
|
|
|
|
const castStringToBool = z.preprocess((val) => {
|
|
if (typeof val === "string") {
|
|
if (["1", "true"].includes(val.toLowerCase())) return true;
|
|
if (["0", "false"].includes(val.toLowerCase())) return false;
|
|
}
|
|
return val;
|
|
}, z.coerce.boolean());
|
|
|
|
export default defineEnv({
|
|
...envConfig,
|
|
extends: [vercel, api, analytics, i18n],
|
|
shared: {
|
|
NODE_ENV: z.enum(NodeEnv).default(NodeEnv.DEVELOPMENT),
|
|
ANALYZE: castStringToBool.optional().default(false),
|
|
},
|
|
/**
|
|
* Specify your server-side environment variables schema here.
|
|
* This way you can ensure the app isn't built with invalid env vars.
|
|
*/
|
|
server: {
|
|
CONTACT_EMAIL: z.string().email().optional().default("contact@example.com"),
|
|
},
|
|
|
|
/**
|
|
* Specify your client-side environment variables schema here.
|
|
* For them to be exposed to the client, prefix them with `NEXT_PUBLIC_`.
|
|
*/
|
|
clientPrefix: "NEXT_PUBLIC_",
|
|
client: {
|
|
NEXT_PUBLIC_AUTH_PASSWORD: castStringToBool.optional().default(true),
|
|
NEXT_PUBLIC_AUTH_MAGIC_LINK: castStringToBool.optional().default(false),
|
|
NEXT_PUBLIC_AUTH_PASSKEY: castStringToBool.optional().default(true),
|
|
NEXT_PUBLIC_AUTH_ANONYMOUS: castStringToBool.optional().default(true),
|
|
|
|
NEXT_PUBLIC_PRODUCT_NAME: z.string().optional().default("TurboStarter"),
|
|
NEXT_PUBLIC_URL: z.string().url().optional().default("http://localhost:3000"),
|
|
NEXT_PUBLIC_DEFAULT_LOCALE: z.string().optional().default("en"),
|
|
NEXT_PUBLIC_THEME_MODE: z
|
|
.enum(ThemeMode)
|
|
.optional()
|
|
.default(ThemeMode.SYSTEM),
|
|
NEXT_PUBLIC_THEME_COLOR: z
|
|
.enum(ThemeColor)
|
|
.optional()
|
|
.default(ThemeColor.ORANGE),
|
|
},
|
|
/**
|
|
* Destructure all variables from `process.env` to make sure they aren't tree-shaken away.
|
|
*/
|
|
env: {
|
|
...process.env,
|
|
NEXT_PUBLIC_PRODUCT_NAME: process.env.NEXT_PUBLIC_PRODUCT_NAME,
|
|
NEXT_PUBLIC_URL: process.env.NEXT_PUBLIC_URL,
|
|
NEXT_PUBLIC_DEFAULT_LOCALE: process.env.NEXT_PUBLIC_DEFAULT_LOCALE,
|
|
NEXT_PUBLIC_THEME_MODE: process.env.NEXT_PUBLIC_THEME_MODE,
|
|
NEXT_PUBLIC_THEME_COLOR: process.env.NEXT_PUBLIC_THEME_COLOR,
|
|
|
|
NEXT_PUBLIC_AUTH_PASSWORD: process.env.NEXT_PUBLIC_AUTH_PASSWORD,
|
|
NEXT_PUBLIC_AUTH_MAGIC_LINK: process.env.NEXT_PUBLIC_AUTH_MAGIC_LINK,
|
|
NEXT_PUBLIC_AUTH_PASSKEY: process.env.NEXT_PUBLIC_AUTH_PASSKEY,
|
|
NEXT_PUBLIC_AUTH_ANONYMOUS: process.env.NEXT_PUBLIC_AUTH_ANONYMOUS,
|
|
|
|
NEXT_PUBLIC_SENTRY_DSN: process.env.NEXT_PUBLIC_SENTRY_DSN,
|
|
NEXT_PUBLIC_SENTRY_ENVIRONMENT: process.env.NEXT_PUBLIC_SENTRY_ENVIRONMENT,
|
|
|
|
NEXT_PUBLIC_POSTHOG_KEY: process.env.NEXT_PUBLIC_POSTHOG_KEY,
|
|
NEXT_PUBLIC_POSTHOG_HOST: process.env.NEXT_PUBLIC_POSTHOG_HOST,
|
|
NEXT_PUBLIC_OPEN_PANEL_CLIENT_ID:
|
|
process.env.NEXT_PUBLIC_OPEN_PANEL_CLIENT_ID,
|
|
NEXT_PUBLIC_GOOGLE_ANALYTICS_MEASUREMENT_ID:
|
|
process.env.NEXT_PUBLIC_GOOGLE_ANALYTICS_MEASUREMENT_ID,
|
|
NEXT_PUBLIC_UMAMI_HOST: process.env.NEXT_PUBLIC_UMAMI_HOST,
|
|
NEXT_PUBLIC_UMAMI_WEBSITE_ID: process.env.NEXT_PUBLIC_UMAMI_WEBSITE_ID,
|
|
NEXT_PUBLIC_PLAUSIBLE_HOST: process.env.NEXT_PUBLIC_PLAUSIBLE_HOST,
|
|
NEXT_PUBLIC_PLAUSIBLE_DOMAIN: process.env.NEXT_PUBLIC_PLAUSIBLE_DOMAIN,
|
|
NEXT_PUBLIC_MIXPANEL_TOKEN: process.env.NEXT_PUBLIC_MIXPANEL_TOKEN,
|
|
NEXT_PUBLIC_VEMETRIC_PROJECT_TOKEN:
|
|
process.env.NEXT_PUBLIC_VEMETRIC_PROJECT_TOKEN,
|
|
},
|
|
});
|