1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
36 |
37 |
38 |
39 |
40 |
41 |
42 |
43 |
44 |
45 |
46 |
47 |
48 |
49 |
50 |
51 |
52 |
53 |
54 |
55 |
56 |
57 |
58 |
59 |
60 |
61 |
62 |
63 |
64 |
65 |
66 |
67 |
68 |
69 |
70 |
71 |
72 |
73 |
74 |
75 |
76 |
77 |
78 |
79 |
80 |
81 |
"use client";
import { getInputProps, useForm, type FieldMetadata } from "@conform-to/react";
import { getZodConstraint, parseWithZod } from "@conform-to/zod";
import { signup } from "~/app/actions/auth";
import { signupSchema } from "~/lib/schema/auth";
import { useActionState } from "react";
import { type z } from "zod";
import { Button } from "./ui/button";
import { Input } from "./ui/input";
import { Label } from "./ui/label";
export function SignupForm() {
const [lastResult, action, isSubmitting] = useActionState(signup, undefined);
const [form, fields] = useForm({
id: "signup",
lastResult,
constraint: getZodConstraint(signupSchema),
shouldValidate: "onBlur",
onValidate({ formData }) {
return parseWithZod(formData, { schema: signupSchema });
},
});
return (
<form
className="flex flex-col gap-2"
id={form.id}
onSubmit={form.onSubmit}
action={action}
noValidate
>
<Field
field={fields.firstName}
label="First Name"
placeholder="First Name"
type="text"
/>
<Field
field={fields.lastName}
label="Last Name"
placeholder="Last Name"
type="text"
/>
<Field
field={fields.email}
label="Email"
placeholder="[email protected]"
type="email"
/>
<Field field={fields.password} label="Password" type="password" />
<Button className="mt-2" type="submit" disabled={isSubmitting}>
Sign up
</Button>
</form>
);
}
function Field({
field,
label,
placeholder,
type,
}: {
field: FieldMetadata<string, z.infer<typeof signupSchema>, string[]>;
label: string;
placeholder?: string;
type: "email" | "password" | "text";
}) {
const { key, ...inputProps } = getInputProps(field, { type });
return (
<div>
<Label htmlFor={inputProps.id}>{label}:</Label>
<Input placeholder={placeholder} key={key} {...inputProps} />
<p>{field.errors?.join(", ")}</p>
</div>
);
}
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
36 |
37 |
38 |
39 |
40 |
41 |
42 |
43 |
44 |
45 |
46 |
47 |
48 |
49 |
50 |
51 |
52 |
53 |
54 |
55 |
56 |
57 |
58 |
59 |
60 |
61 |
62 |
63 |
64 |
65 |
66 |
67 |
68 |
69 |
70 |
71 |
72 |
73 |
74 |
75 |
76 |
77 |
78 |
79 |
80 |
81 |
"use client";
import { getInputProps, useForm, type FieldMetadata } from "@conform-to/react";
import { getZodConstraint, parseWithZod } from "@conform-to/zod";
import { signup } from "~/app/actions/auth";
import { signupSchema } from "~/lib/schema/auth";
import { useActionState } from "react";
import { type z } from "zod";
import { Button } from "./ui/button";
import { Input } from "./ui/input";
import { Label } from "./ui/label";
export function SignupForm() {
const [lastResult, action, isSubmitting] = useActionState(signup, undefined);
const [form, fields] = useForm({
id: "signup",
lastResult,
constraint: getZodConstraint(signupSchema),
shouldValidate: "onBlur",
onValidate({ formData }) {
return parseWithZod(formData, { schema: signupSchema });
},
});
return (
<form
className="flex flex-col gap-2"
id={form.id}
onSubmit={form.onSubmit}
action={action}
noValidate
>
<Field
field={fields.firstName}
label="First Name"
placeholder="First Name"
type="text"
/>
<Field
field={fields.lastName}
label="Last Name"
placeholder="Last Name"
type="text"
/>
<Field
field={fields.email}
label="Email"
placeholder="[email protected]"
type="email"
/>
<Field field={fields.password} label="Password" type="password" />
<Button className="mt-2" type="submit" disabled={isSubmitting}>
Sign up
</Button>
</form>
);
}
function Field({
field,
label,
placeholder,
type,
}: {
field: FieldMetadata<string, z.infer<typeof signupSchema>, string[]>;
label: string;
placeholder?: string;
type: "email" | "password" | "text";
}) {
const { key, ...inputProps } = getInputProps(field, { type });
return (
<div>
<Label htmlFor={inputProps.id}>{label}:</Label>
<Input placeholder={placeholder} key={key} {...inputProps} />
<p>{field.errors?.join(", ")}</p>
</div>
);
}
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
36 |
37 |
38 |
39 |
40 |
41 |
42 |
43 |
44 |
45 |
46 |
47 |
48 |
49 |
50 |
51 |
52 |
53 |
54 |
55 |
56 |
57 |
58 |
59 |
60 |
61 |
62 |
63 |
64 |
65 |
66 |
67 |
68 |
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 = {
baseURL: env.URL,
appName: "Kapil Jewels",
advanced: {
generateId: false,
useSecureCookies: true,
defaultCookieAttributes: {
secure: true,
httpOnly: true,
sameSite: "lax",
},
},
database: drizzleAdapter(db, {
provider: "pg",
schema: {
user,
account,
session,
verification,
rateLimit,
},
}),
emailVerification: {
expiresIn: 60 * 10,
sendOnSignUp: true,
async sendVerificationEmail(data) {
console.log(data);
// await api.resend.verify({ email: data.user.email, url: data.url });
},
},
emailAndPassword: {
enabled: true,
autoSignIn: true,
resetPasswordTokenExpiresIn: 60 * 10,
async sendResetPassword(data) {
console.log(data);
// await api.resend.resetPassword({ email: data.user.email, url: data.url });
},
},
rateLimit: {
storage: "database",
window: 60,
max: 10,
},
session: {
expiresIn: 30 * 24 * 60 * 60,
updateAge: 15 * 24 * 60 * 60,
cookieCache: {
enabled: true,
maxAge: 60 * 10,
},
},
plugins: [nextCookies()],
} satisfies BetterAuthOptions;
import { betterAuth } from "better-auth";
import { cache } from "react";
import { authConfig } from "./config";
const {
handler,
api: { getSession: uncachedGetSession, signInEmail, signOut, signUpEmail },
} = betterAuth(authConfig);
const getSession = cache(uncachedGetSession);
export { handler, getSession, signInEmail, signOut, signUpEmail };