MystBin
/5609e26466b88cb244 Created 6 days ago...
Raw
signup.tsx Hide Copy Raw
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> ); }
actions/auth.ts Hide Copy Raw
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> ); }
auth/config.ts Hide Copy Raw
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;
auth/index.ts Hide Copy Raw
1
2
3
4
5
6
7
8
9
10
11
12
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 };