# MystBin ! - signin-form.tsx
"use client";
import { getInputProps, useForm, type FieldMetadata } from "@conform-to/react";
import { getZodConstraint, parseWithZod } from "@conform-to/zod";
import { Label } from "@radix-ui/react-label";
import { signIn } from "~/actions/auth";
import { signInSchema as schema } from "~/schema/auth";
import Form from "next/form";
import { useActionState } from "react";
import { type z } from "zod";
import { CheckboxConform } from "./conform-inputs/checkbox";
import { Button } from "./ui/button";
import { Input } from "./ui/input";
export function SignInForm() {
const [lastResult, action, isSubmitting] = useActionState(signIn, undefined);
const [form, fields] = useForm({
id: "signin",
lastResult,
constraint: getZodConstraint(schema),
shouldValidate: "onBlur",
onValidate({ formData }) {
return parseWithZod(formData, { schema });
},
});
return (
);
}
function Field({
meta,
label,
placeholder,
type,
}: {
meta: FieldMetadata<
string,
Omit, "rememberMe">,
string[]
>;
label: string;
placeholder?: string;
type: "email" | "password";
}) {
const { key: _key, ...inputProps } = getInputProps(meta, { type });
return (
);
}
# MystBin ! - actions/auth.ts
"use server";
import { parseWithZod } from "@conform-to/zod";
import { signInSchema } from "~/schema/auth";
import { signInEmail } from "~/server/auth";
import { APIError } from "better-auth/api";
export async function signIn(prevState: unknown, formData: FormData) {
const submission = parseWithZod(formData, { schema: signInSchema });
if (submission.status !== "success") {
return submission.reply();
}
try {
await signInEmail({
body: {
...submission.value,
callbackURL: "/",
},
});
} catch (error) {
if (error instanceof APIError) {
if (error.status === "TOO_MANY_REQUESTS") {
return submission.reply({
formErrors: [error.message],
});
}
if (error.status === "FORBIDDEN") {
return submission.reply({
formErrors: ["Verify your email before siging in"],
});
}
if (error.status === "UNAUTHORIZED") {
return submission.reply({
fieldErrors: {
email: ["Invalid Credentials"],
password: ["Invalid Credentials"],
},
});
}
throw error;
}
throw error;
}
}
# MystBin ! - auth-config.ts
import { env } from "~/env";
import { db } from "~/server/db";
import {
account,
rateLimit,
session,
user,
verification,
} from "~/server/db/schema";
import { api } from "~/trpc/server";
import { type BetterAuthOptions } from "better-auth";
import { drizzleAdapter } from "better-auth/adapters/drizzle";
import { nextCookies } from "better-auth/next-js";
export const authConfig = {
appName: "kapil-jewels",
baseURL: env.URL,
advanced: {
generateId: false,
},
database: drizzleAdapter(db, {
provider: "pg",
schema: {
user,
account,
session,
verification,
rateLimit,
},
}),
emailVerification: {
expiresIn: 60 * 10,
sendOnSignUp: true,
async sendVerificationEmail({ url, user: { email } }) {
await api.resend.verify({ email, url });
},
},
emailAndPassword: {
enabled: true,
autoSignIn: false,
requireEmailVerification: true,
resetPasswordTokenExpiresIn: 60 * 10,
async sendResetPassword(data) {
await api.resend.resetPassword({ email: data.user.email, url: data.url });
},
},
rateLimit: {
enabled: true,
storage: "database",
window: 60,
max: 2,
},
session: {
expiresIn: 30 * 24 * 60 * 60,
updateAge: 15 * 24 * 60 * 60,
cookieCache: {
enabled: true,
maxAge: 60 * 10,
},
},
plugins: [nextCookies()],
} satisfies BetterAuthOptions;