Files
claudemesh/apps/web/env.config.ts
Alejandro Gutiérrez d3163a5bff feat(db): mesh data model — meshes, members, invites, audit log
- pgSchema "mesh" with 4 tables isolating the peer mesh domain
- Enums: visibility, transport, tier, role
- audit_log is metadata-only (E2E encryption enforced at broker/client)
- Cascade on mesh delete, soft-delete via archivedAt/revokedAt

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-04 21:19:32 +01:00

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,
},
});