feat: turbostarter boilerplate

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>
This commit is contained in:
Alejandro Gutiérrez
2026-02-02 17:29:12 +00:00
commit 3527e732d4
1618 changed files with 338230 additions and 0 deletions

View File

@@ -0,0 +1 @@
export * from "./resend/env";

View File

@@ -0,0 +1 @@
export { send } from "./resend";

View File

@@ -0,0 +1,24 @@
import { defineEnv } from "envin";
import * as z from "zod";
import { envConfig } from "@turbostarter/shared/constants";
import { sharedPreset } from "../../utils/env";
import type { Preset } from "envin/types";
export const preset = {
id: "nodemailer",
server: {
NODEMAILER_HOST: z.string(),
NODEMAILER_PORT: z.coerce.number(),
NODEMAILER_USER: z.string(),
NODEMAILER_PASSWORD: z.string(),
},
extends: [sharedPreset],
} as const satisfies Preset;
export const env = defineEnv({
...envConfig,
...preset,
});

View File

@@ -0,0 +1,28 @@
import nodemailer from "nodemailer";
import { env } from "./env";
import type { EmailProviderStrategy } from "../types";
const from = env.EMAIL_FROM;
export const { send } = {
send: async ({ to, subject, html, text }) => {
const transporter = nodemailer.createTransport({
host: env.NODEMAILER_HOST,
port: env.NODEMAILER_PORT,
auth: {
user: env.NODEMAILER_USER,
pass: env.NODEMAILER_PASSWORD,
},
});
await transporter.sendMail({
from,
to,
subject,
html,
text,
});
},
} satisfies EmailProviderStrategy;

View File

@@ -0,0 +1,21 @@
import { defineEnv } from "envin";
import * as z from "zod";
import { envConfig } from "@turbostarter/shared/constants";
import { sharedPreset } from "../../utils/env";
import type { Preset } from "envin/types";
export const preset = {
id: "plunk",
server: {
PLUNK_API_KEY: z.string(),
},
extends: [sharedPreset],
} as const satisfies Preset;
export const env = defineEnv({
...envConfig,
...preset,
});

View File

@@ -0,0 +1,31 @@
import { logger } from "@turbostarter/shared/logger";
import { env } from "./env";
import type { EmailProviderStrategy } from "../types";
const from = env.EMAIL_FROM;
export const { send } = {
send: async ({ to, subject, html, text }) => {
const response = await fetch("https://api.useplunk.com/v1/send", {
method: "POST",
headers: {
"Content-Type": "application/json",
Authorization: `Bearer ${env.PLUNK_API_KEY}`,
},
body: JSON.stringify({
to,
from,
subject,
body: html,
text,
}),
});
if (!response.ok) {
logger.error(await response.json());
throw new Error("Could not send email!");
}
},
} satisfies EmailProviderStrategy;

View File

@@ -0,0 +1,21 @@
import { defineEnv } from "envin";
import * as z from "zod";
import { envConfig } from "@turbostarter/shared/constants";
import { sharedPreset } from "../../utils/env";
import type { Preset } from "envin/types";
export const preset = {
id: "postmark",
server: {
POSTMARK_API_KEY: z.string(),
},
extends: [sharedPreset],
} as const satisfies Preset;
export const env = defineEnv({
...envConfig,
...preset,
});

View File

@@ -0,0 +1,31 @@
import { logger } from "@turbostarter/shared/logger";
import { env } from "./env";
import type { EmailProviderStrategy } from "../types";
const from = env.EMAIL_FROM;
export const { send } = {
send: async ({ to, subject, html, text }) => {
const response = await fetch("https://api.postmarkapp.com/email", {
method: "POST",
headers: {
"Content-Type": "application/json",
"X-Postmark-Server-Token": env.POSTMARK_API_KEY,
},
body: JSON.stringify({
From: from,
To: to,
Subject: subject,
HtmlBody: html,
TextBody: text,
}),
});
if (!response.ok) {
logger.error(await response.json());
throw new Error("Could not send email!");
}
},
} satisfies EmailProviderStrategy;

View File

@@ -0,0 +1,21 @@
import { defineEnv } from "envin";
import * as z from "zod";
import { envConfig } from "@turbostarter/shared/constants";
import { sharedPreset } from "../../utils/env";
import type { Preset } from "envin/types";
export const preset = {
id: "resend",
server: {
RESEND_API_KEY: z.string().optional(),
},
extends: [sharedPreset],
} as const satisfies Preset;
export const env = defineEnv({
...envConfig,
...preset,
});

View File

@@ -0,0 +1,31 @@
import { logger } from "@turbostarter/shared/logger";
import { env } from "./env";
import type { EmailProviderStrategy } from "../types";
const from = env.EMAIL_FROM;
export const { send } = {
send: async ({ to, subject, html, text }) => {
const response = await fetch("https://api.resend.com/emails", {
method: "POST",
headers: {
"Content-Type": "application/json",
Authorization: `Bearer ${env.RESEND_API_KEY}`,
},
body: JSON.stringify({
from,
to,
subject,
html,
text,
}),
});
if (!response.ok) {
logger.error(await response.json());
throw new Error("Could not send email!");
}
},
} satisfies EmailProviderStrategy;

View File

@@ -0,0 +1,21 @@
import { defineEnv } from "envin";
import * as z from "zod";
import { envConfig } from "@turbostarter/shared/constants";
import { sharedPreset } from "../../utils/env";
import type { Preset } from "envin/types";
export const preset = {
id: "sendgrid",
server: {
SENDGRID_API_KEY: z.string(),
},
extends: [sharedPreset],
} as const satisfies Preset;
export const env = defineEnv({
...envConfig,
...preset,
});

View File

@@ -0,0 +1,43 @@
import { logger } from "@turbostarter/shared/logger";
import { env } from "./env";
import type { EmailProviderStrategy } from "../types";
const from = env.EMAIL_FROM;
export const { send } = {
send: async ({ to, subject, html, text }) => {
const response = await fetch("https://api.sendgrid.com/v3/mail/send", {
method: "POST",
headers: {
"Content-Type": "application/json",
Authorization: `Bearer ${env.SENDGRID_API_KEY}`,
},
body: JSON.stringify({
from: { email: from },
personalizations: [
{
to: [{ email: to }],
},
],
subject,
content: [
{
type: "text/plain",
value: text,
},
{
type: "text/html",
value: html,
},
],
}),
});
if (!response.ok) {
logger.error(await response.json());
throw new Error("Could not send email!");
}
},
} satisfies EmailProviderStrategy;

View File

@@ -0,0 +1,8 @@
export interface EmailProviderStrategy {
send: (args: {
to: string;
subject: string;
text: string;
html?: string;
}) => Promise<void>;
}