Files
better-auth/examples/nextjs-example/lib/auth.ts
2024-10-29 23:35:21 +03:00

171 lines
4.1 KiB
TypeScript

import Database from "better-sqlite3";
import { betterAuth } from "better-auth";
import {
bearer,
organization,
passkey,
twoFactor,
admin,
multiSession,
username,
createAuthMiddleware,
} from "better-auth/plugins";
import { reactInvitationEmail } from "./email/invitation";
import { reactResetPasswordEmail } from "./email/rest-password";
import { resend } from "./email/resend";
import { z } from "zod";
const from = process.env.BETTER_AUTH_EMAIL || "delivered@resend.dev";
const to = process.env.TEST_EMAIL || "";
const database = new Database("better-auth.sqlite");
export const auth = betterAuth({
database,
emailVerification: {
async sendVerificationEmail(user, url) {
console.log("Sending verification email to", user.email);
// const res = await resend.emails.send({
// from,
// to: to || user.email,
// subject: "Verify your email address",
// html: `<a href="${url}">Verify your email address</a>`,
// });
// console.log(res, user.email);
},
sendOnSignUp: true,
},
account: {
accountLinking: {
trustedProviders: ["google", "github"],
},
},
emailAndPassword: {
enabled: true,
async sendResetPassword(user, url) {
await resend.emails.send({
from,
to: user.email,
subject: "Reset your password",
react: reactResetPasswordEmail({
username: user.email,
resetLink: url,
}),
});
},
},
socialProviders: {
github: {
clientId: process.env.GITHUB_CLIENT_ID || "",
clientSecret: process.env.GITHUB_CLIENT_SECRET || "",
},
google: {
clientId: process.env.GOOGLE_CLIENT_ID || "",
clientSecret: process.env.GOOGLE_CLIENT_SECRET || "",
accessType: "offline",
prompt: "select_account",
},
discord: {
clientId: process.env.DISCORD_CLIENT_ID || "",
clientSecret: process.env.DISCORD_CLIENT_SECRET || "",
},
microsoft: {
clientId: process.env.MICROSOFT_CLIENT_ID || "",
clientSecret: process.env.MICROSOFT_CLIENT_SECRET || "",
},
twitch: {
clientId: process.env.TWITCH_CLIENT_ID || "",
clientSecret: process.env.TWITCH_CLIENT_SECRET || "",
},
},
plugins: [
organization({
async sendInvitationEmail(data) {
const res = await resend.emails.send({
from,
to: data.email,
subject: "You've been invited to join an organization",
react: reactInvitationEmail({
username: data.email,
invitedByUsername: data.inviter.user.name,
invitedByEmail: data.inviter.user.email,
teamName: data.organization.name,
inviteLink:
process.env.NODE_ENV === "development"
? `http://localhost:3000/accept-invitation/${data.id}`
: `https://${
process.env.NEXT_PUBLIC_APP_URL ||
process.env.VERCEL_URL ||
process.env.BETTER_AUTH_URL
}/accept-invitation/${data.id}`,
}),
});
console.log(res, data.email);
},
}),
twoFactor({
otpOptions: {
async sendOTP(user, otp) {
await resend.emails.send({
from,
to: user.email,
subject: "Your OTP",
html: `Your OTP is ${otp}`,
});
},
},
}),
passkey(),
bearer(),
admin(),
multiSession(),
username(),
{
id: "expo",
hooks: {
after: [
{
matcher(context) {
return context.path?.startsWith("/callback");
},
handler: createAuthMiddleware(async (ctx) => {
const response = ctx.context.returned as Response;
if (response.status === 302) {
const location = response.headers.get("location");
if (!location) {
return;
}
const trustedOrigins = ctx.context.trustedOrigins.filter(
(origin) => !origin.startsWith("http"),
);
const isTrustedOrigin = trustedOrigins.some((origin) =>
location?.startsWith(origin),
);
if (!isTrustedOrigin) {
return;
}
const cookie = response.headers.get("set-cookie");
if (!cookie) {
return;
}
const url = new URL(location);
url.searchParams.set("cookie", cookie);
response.headers.set("location", url.toString());
return {
response,
};
}
}),
},
],
},
},
],
trustedOrigins: ["expo"],
});