N-FIN-79: remove lucia authentication components
This commit is contained in:
parent
6ba9a8872b
commit
c4146a36a4
10 changed files with 0 additions and 347 deletions
|
@ -1,13 +0,0 @@
|
||||||
import React from 'react';
|
|
||||||
|
|
||||||
export default function AuthLayout({
|
|
||||||
children,
|
|
||||||
}: Readonly<{
|
|
||||||
children: React.ReactNode;
|
|
||||||
}>) {
|
|
||||||
return (
|
|
||||||
<div className="flex justify-center">
|
|
||||||
{children}
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
|
@ -1,25 +0,0 @@
|
||||||
import React from 'react';
|
|
||||||
import { Card, CardContent, CardDescription, CardFooter, CardHeader, CardTitle } from '@/components/ui/card';
|
|
||||||
import SignInForm from '@/components/form/signInForm';
|
|
||||||
import signIn from '@/lib/actions/signIn';
|
|
||||||
import Link from 'next/link';
|
|
||||||
import { URL_SIGN_UP } from '@/lib/constants';
|
|
||||||
|
|
||||||
export default async function SignInPage() {
|
|
||||||
return (
|
|
||||||
<Card className="w-full max-w-md mt-12">
|
|
||||||
<CardHeader>
|
|
||||||
<CardTitle>Sign in</CardTitle>
|
|
||||||
<CardDescription>Sign into your existing account</CardDescription>
|
|
||||||
</CardHeader>
|
|
||||||
<CardContent>
|
|
||||||
<SignInForm onSubmit={signIn}/>
|
|
||||||
</CardContent>
|
|
||||||
<CardFooter>
|
|
||||||
<Link href={URL_SIGN_UP}>
|
|
||||||
Don't have an account? Sign up
|
|
||||||
</Link>
|
|
||||||
</CardFooter>
|
|
||||||
</Card>
|
|
||||||
);
|
|
||||||
}
|
|
|
@ -1,25 +0,0 @@
|
||||||
import React from 'react';
|
|
||||||
import { Card, CardContent, CardDescription, CardFooter, CardHeader, CardTitle } from '@/components/ui/card';
|
|
||||||
import signUp from '@/lib/actions/signUp';
|
|
||||||
import SignUpForm from '@/components/form/signUpForm';
|
|
||||||
import Link from 'next/link';
|
|
||||||
import { URL_SIGN_IN } from '@/lib/constants';
|
|
||||||
|
|
||||||
export default async function SignUpPage() {
|
|
||||||
return (
|
|
||||||
<Card className="w-full max-w-md mt-12">
|
|
||||||
<CardHeader>
|
|
||||||
<CardTitle>Sign up</CardTitle>
|
|
||||||
<CardDescription>Create a new account.</CardDescription>
|
|
||||||
</CardHeader>
|
|
||||||
<CardContent>
|
|
||||||
<SignUpForm onSubmit={signUp}/>
|
|
||||||
</CardContent>
|
|
||||||
<CardFooter>
|
|
||||||
<Link href={URL_SIGN_IN}>
|
|
||||||
Already have an account? Sign in
|
|
||||||
</Link>
|
|
||||||
</CardFooter>
|
|
||||||
</Card>
|
|
||||||
);
|
|
||||||
}
|
|
|
@ -1,71 +0,0 @@
|
||||||
'use client';
|
|
||||||
|
|
||||||
import { zodResolver } from '@hookform/resolvers/zod';
|
|
||||||
import { useForm } from 'react-hook-form';
|
|
||||||
import { z } from 'zod';
|
|
||||||
import { Form, FormControl, FormField, FormItem, FormLabel, FormMessage } from '@/components/ui/form';
|
|
||||||
import { Input } from '@/components/ui/input';
|
|
||||||
import React from 'react';
|
|
||||||
import { Button } from '@/components/ui/button';
|
|
||||||
import { signInFormSchema } from '@/lib/form-schemas/signInFormSchema';
|
|
||||||
import { ActionResponse } from '@/lib/types/actionResponse';
|
|
||||||
import { useRouter } from 'next/navigation';
|
|
||||||
import { toast } from 'sonner';
|
|
||||||
import { sonnerContent } from '@/components/ui/sonner';
|
|
||||||
|
|
||||||
export default function SignInForm({onSubmit}: {
|
|
||||||
onSubmit: (data: z.infer<typeof signInFormSchema>) => Promise<ActionResponse>
|
|
||||||
}) {
|
|
||||||
|
|
||||||
const router = useRouter();
|
|
||||||
|
|
||||||
const form = useForm<z.infer<typeof signInFormSchema>>({
|
|
||||||
resolver: zodResolver(signInFormSchema),
|
|
||||||
defaultValues: {
|
|
||||||
username: '',
|
|
||||||
password: '',
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const handleSubmit = async (data: z.infer<typeof signInFormSchema>) => {
|
|
||||||
const response = await onSubmit(data);
|
|
||||||
toast(sonnerContent(response));
|
|
||||||
if (response.redirect) {
|
|
||||||
router.push(response.redirect);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Form {...form}>
|
|
||||||
<form onSubmit={form.handleSubmit(handleSubmit)} className="space-y-2">
|
|
||||||
<FormField
|
|
||||||
control={form.control}
|
|
||||||
name="username"
|
|
||||||
render={({field}) => (
|
|
||||||
<FormItem>
|
|
||||||
<FormLabel>Username</FormLabel>
|
|
||||||
<FormControl>
|
|
||||||
<Input placeholder="Username" {...field} />
|
|
||||||
</FormControl>
|
|
||||||
<FormMessage/>
|
|
||||||
</FormItem>
|
|
||||||
)}
|
|
||||||
/>
|
|
||||||
<FormField
|
|
||||||
control={form.control}
|
|
||||||
name="password"
|
|
||||||
render={({field}) => (
|
|
||||||
<FormItem>
|
|
||||||
<FormLabel>Password</FormLabel>
|
|
||||||
<FormControl>
|
|
||||||
<Input placeholder="••••••••" type="password" {...field} />
|
|
||||||
</FormControl>
|
|
||||||
<FormMessage/>
|
|
||||||
</FormItem>
|
|
||||||
)}
|
|
||||||
/>
|
|
||||||
<Button type="submit" className="w-full">Sign in</Button>
|
|
||||||
</form>
|
|
||||||
</Form>
|
|
||||||
);
|
|
||||||
}
|
|
|
@ -1,84 +0,0 @@
|
||||||
'use client';
|
|
||||||
|
|
||||||
import { zodResolver } from '@hookform/resolvers/zod';
|
|
||||||
import { useForm } from 'react-hook-form';
|
|
||||||
import { z } from 'zod';
|
|
||||||
import { Form, FormControl, FormField, FormItem, FormLabel, FormMessage } from '@/components/ui/form';
|
|
||||||
import { Input } from '@/components/ui/input';
|
|
||||||
import React from 'react';
|
|
||||||
import { Button } from '@/components/ui/button';
|
|
||||||
import { signUpFormSchema } from '@/lib/form-schemas/signUpFormSchema';
|
|
||||||
import { ActionResponse } from '@/lib/types/actionResponse';
|
|
||||||
import { useRouter } from 'next/navigation';
|
|
||||||
import { toast } from 'sonner';
|
|
||||||
import { sonnerContent } from '@/components/ui/sonner';
|
|
||||||
|
|
||||||
export default function SignUpForm({onSubmit}: {
|
|
||||||
onSubmit: (data: z.infer<typeof signUpFormSchema>) => Promise<ActionResponse>
|
|
||||||
}) {
|
|
||||||
|
|
||||||
const router = useRouter();
|
|
||||||
|
|
||||||
const form = useForm<z.infer<typeof signUpFormSchema>>({
|
|
||||||
resolver: zodResolver(signUpFormSchema),
|
|
||||||
defaultValues: {
|
|
||||||
username: '',
|
|
||||||
password: '',
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const handleSubmit = async (data: z.infer<typeof signUpFormSchema>) => {
|
|
||||||
const response = await onSubmit(data);
|
|
||||||
toast(sonnerContent(response));
|
|
||||||
if (response.redirect) {
|
|
||||||
router.push(response.redirect);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Form {...form}>
|
|
||||||
<form onSubmit={form.handleSubmit(handleSubmit)} className="space-y-2">
|
|
||||||
<FormField
|
|
||||||
control={form.control}
|
|
||||||
name="username"
|
|
||||||
render={({field}) => (
|
|
||||||
<FormItem>
|
|
||||||
<FormLabel>Username</FormLabel>
|
|
||||||
<FormControl>
|
|
||||||
<Input placeholder="Username" {...field} />
|
|
||||||
</FormControl>
|
|
||||||
<FormMessage/>
|
|
||||||
</FormItem>
|
|
||||||
)}
|
|
||||||
/>
|
|
||||||
<FormField
|
|
||||||
control={form.control}
|
|
||||||
name="password"
|
|
||||||
render={({field}) => (
|
|
||||||
<FormItem>
|
|
||||||
<FormLabel>Password</FormLabel>
|
|
||||||
<FormControl>
|
|
||||||
<Input placeholder="••••••••" type="password" {...field} />
|
|
||||||
</FormControl>
|
|
||||||
<FormMessage/>
|
|
||||||
</FormItem>
|
|
||||||
)}
|
|
||||||
/>
|
|
||||||
<FormField
|
|
||||||
control={form.control}
|
|
||||||
name="confirm"
|
|
||||||
render={({field}) => (
|
|
||||||
<FormItem>
|
|
||||||
<FormLabel>Confirm password</FormLabel>
|
|
||||||
<FormControl>
|
|
||||||
<Input placeholder="••••••••" type="password" {...field} />
|
|
||||||
</FormControl>
|
|
||||||
<FormMessage/>
|
|
||||||
</FormItem>
|
|
||||||
)}
|
|
||||||
/>
|
|
||||||
<Button type="submit" className="w-full">Create Account</Button>
|
|
||||||
</form>
|
|
||||||
</Form>
|
|
||||||
);
|
|
||||||
}
|
|
|
@ -1,41 +0,0 @@
|
||||||
import { z } from 'zod';
|
|
||||||
import { Argon2id } from 'oslo/password';
|
|
||||||
import { lucia } from '@/auth';
|
|
||||||
import { cookies } from 'next/headers';
|
|
||||||
import { signInFormSchema } from '@/lib/form-schemas/signInFormSchema';
|
|
||||||
import { ActionResponse } from '@/lib/types/actionResponse';
|
|
||||||
import { URL_HOME } from '@/lib/constants';
|
|
||||||
import prisma from '@/prisma';
|
|
||||||
|
|
||||||
export default async function signIn({username, password}: z.infer<typeof signInFormSchema>): Promise<ActionResponse> {
|
|
||||||
'use server';
|
|
||||||
|
|
||||||
const existingUser = await prisma.user.findFirst({
|
|
||||||
where: {
|
|
||||||
username: username.toLowerCase(),
|
|
||||||
},
|
|
||||||
});
|
|
||||||
if (!existingUser) {
|
|
||||||
return {
|
|
||||||
type: 'error',
|
|
||||||
message: 'Incorrect username or password',
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
const validPassword = await new Argon2id().verify(existingUser.password, password);
|
|
||||||
if (!validPassword) {
|
|
||||||
return {
|
|
||||||
type: 'error',
|
|
||||||
message: 'Incorrect username or password',
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
const session = await lucia.createSession(existingUser.id, {});
|
|
||||||
const sessionCookie = lucia.createSessionCookie(session.id);
|
|
||||||
cookies().set(sessionCookie.name, sessionCookie.value, sessionCookie.attributes);
|
|
||||||
return {
|
|
||||||
type: 'success',
|
|
||||||
message: 'Signed in successfully',
|
|
||||||
redirect: URL_HOME,
|
|
||||||
};
|
|
||||||
}
|
|
|
@ -1,26 +0,0 @@
|
||||||
import { getSession, lucia } from '@/auth';
|
|
||||||
import { cookies } from 'next/headers';
|
|
||||||
import { ActionResponse } from '@/lib/types/actionResponse';
|
|
||||||
import { URL_SIGN_IN } from '@/lib/constants';
|
|
||||||
|
|
||||||
export default async function signOut(): Promise<ActionResponse> {
|
|
||||||
'use server';
|
|
||||||
|
|
||||||
const session = await getSession();
|
|
||||||
if (!session) {
|
|
||||||
return {
|
|
||||||
type: 'error',
|
|
||||||
message: 'You aren\'t signed in',
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
await lucia.invalidateSession(session.id);
|
|
||||||
|
|
||||||
const sessionCookie = lucia.createBlankSessionCookie();
|
|
||||||
cookies().set(sessionCookie.name, sessionCookie.value, sessionCookie.attributes);
|
|
||||||
return {
|
|
||||||
type: 'success',
|
|
||||||
message: 'Signed out successfully',
|
|
||||||
redirect: URL_SIGN_IN,
|
|
||||||
};
|
|
||||||
}
|
|
|
@ -1,46 +0,0 @@
|
||||||
import { z } from 'zod';
|
|
||||||
import { Argon2id } from 'oslo/password';
|
|
||||||
import { generateId } from 'lucia';
|
|
||||||
import { lucia } from '@/auth';
|
|
||||||
import { cookies } from 'next/headers';
|
|
||||||
import { signUpFormSchema } from '@/lib/form-schemas/signUpFormSchema';
|
|
||||||
import { ActionResponse } from '@/lib/types/actionResponse';
|
|
||||||
import { URL_HOME } from '@/lib/constants';
|
|
||||||
import prisma from '@/prisma';
|
|
||||||
|
|
||||||
export default async function signUp({username, password}: z.infer<typeof signUpFormSchema>): Promise<ActionResponse> {
|
|
||||||
'use server';
|
|
||||||
|
|
||||||
const hashedPassword = await new Argon2id().hash(password);
|
|
||||||
const userId = generateId(15);
|
|
||||||
|
|
||||||
const existingUser = await prisma.user.findFirst({
|
|
||||||
where: {
|
|
||||||
username: username.toLowerCase(),
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
if (existingUser) {
|
|
||||||
return {
|
|
||||||
type: 'error',
|
|
||||||
message: 'Username already exists',
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
await prisma.user.create({
|
|
||||||
data: {
|
|
||||||
id: userId,
|
|
||||||
username: username,
|
|
||||||
password: hashedPassword,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const session = await lucia.createSession(userId, {});
|
|
||||||
const sessionCookie = lucia.createSessionCookie(session.id);
|
|
||||||
cookies().set(sessionCookie.name, sessionCookie.value, sessionCookie.attributes);
|
|
||||||
return {
|
|
||||||
type: 'success',
|
|
||||||
message: 'Signed up successfully',
|
|
||||||
redirect: URL_HOME,
|
|
||||||
};
|
|
||||||
}
|
|
|
@ -1,6 +0,0 @@
|
||||||
import { z } from 'zod';
|
|
||||||
|
|
||||||
export const signInFormSchema = z.object({
|
|
||||||
username: z.string().min(3).max(16),
|
|
||||||
password: z.string().min(8).max(255),
|
|
||||||
});
|
|
|
@ -1,10 +0,0 @@
|
||||||
import { z } from 'zod';
|
|
||||||
|
|
||||||
export const signUpFormSchema = z.object({
|
|
||||||
username: z.string().min(3).max(16),
|
|
||||||
password: z.string().min(8).max(255),
|
|
||||||
confirm: z.string().min(8).max(255),
|
|
||||||
}).refine(data => data.password === data.confirm, {
|
|
||||||
message: 'Passwords do not match',
|
|
||||||
path: ['confirm'],
|
|
||||||
});
|
|
Loading…
Add table
Reference in a new issue