add client/admin pages, show info and created admin api and server actions
This commit is contained in:
parent
db66161d81
commit
d6b259d71c
14
actions/admin.ts
Normal file
14
actions/admin.ts
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
'use server'
|
||||||
|
|
||||||
|
import { currentRole } from '@/lib/auth'
|
||||||
|
import { UserRole } from '@prisma/client'
|
||||||
|
|
||||||
|
export const admin = async () => {
|
||||||
|
const role = await currentRole()
|
||||||
|
|
||||||
|
if (role === UserRole.ADMIN) {
|
||||||
|
return { success: `Allowed Server Action for ${UserRole.ADMIN}` }
|
||||||
|
}
|
||||||
|
|
||||||
|
return { error: `Forbidden Server Action for ${UserRole.ADMIN}` }
|
||||||
|
}
|
@ -3,45 +3,31 @@
|
|||||||
|
|
||||||
import { Button } from '@/components/ui/button'
|
import { Button } from '@/components/ui/button'
|
||||||
import Link from 'next/link'
|
import Link from 'next/link'
|
||||||
import { USER_PROFILE_URL } from '@/config/routes'
|
import { CABINET_ROUTES, USER_PROFILE_URL } from '@/config/routes'
|
||||||
import { usePathname } from 'next/navigation'
|
import { usePathname } from 'next/navigation'
|
||||||
import UserButton from '@/components/auth/user-button'
|
import UserButton from '@/components/auth/user-button'
|
||||||
import LocaleSwitcher from '@/components/locale-switcher'
|
import LocaleSwitcher from '@/components/locale-switcher'
|
||||||
|
|
||||||
const Navbar = () => {
|
export const Navbar = () => {
|
||||||
const pathname = usePathname()
|
const pathname = usePathname()
|
||||||
//
|
|
||||||
|
console.log(USER_PROFILE_URL)
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<nav className="bg-secondary flex justify-between items-center top-0 absolute px-6 py-4 w-full shadow-sm">
|
<nav className="bg-secondary flex justify-between items-center top-0 absolute px-6 py-4 w-full shadow-sm">
|
||||||
<div className="flex gap-x-4">
|
<div className="flex gap-x-4">
|
||||||
<Button asChild variant={pathname.match(/^\/(en\/|)server/) ? 'default' : 'outline'}>
|
{CABINET_ROUTES.map((route) => (
|
||||||
<Link href={'/server'}>
|
<Button asChild key={route} variant={pathname.endsWith(route) ? 'default' : 'outline'} className="border">
|
||||||
Server
|
<Link href={route}>
|
||||||
</Link>
|
{route[1]?.toUpperCase() + route.substring(2)}
|
||||||
</Button>
|
|
||||||
<Button asChild variant={pathname.match(/^\/(en\/|)client/) ? 'default' : 'outline'}>
|
|
||||||
<Link href={'/client'}>
|
|
||||||
Client
|
|
||||||
</Link>
|
|
||||||
</Button>
|
|
||||||
<Button asChild variant={pathname.match(/^\/(en\/|)admin/) ? 'default' : 'outline'}>
|
|
||||||
<Link href={'/admin'}>
|
|
||||||
Admin
|
|
||||||
</Link>
|
|
||||||
</Button>
|
|
||||||
<Button asChild variant={pathname.match(/^\/(en\/|)cabinet/) ? 'default' : 'outline'}>
|
|
||||||
<Link href={USER_PROFILE_URL}>
|
|
||||||
Cabinet
|
|
||||||
</Link>
|
</Link>
|
||||||
</Button>
|
</Button>
|
||||||
|
))}
|
||||||
</div>
|
</div>
|
||||||
<div className="flex gap-x-2">
|
<div className="flex gap-x-2">
|
||||||
<LocaleSwitcher/>
|
<LocaleSwitcher/>
|
||||||
<UserButton/>
|
<UserButton/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
</nav>
|
</nav>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
export default Navbar
|
|
||||||
|
68
app/[locale]/(protected)/cabinet/admin/page.tsx
Normal file
68
app/[locale]/(protected)/cabinet/admin/page.tsx
Normal file
@ -0,0 +1,68 @@
|
|||||||
|
'use client'
|
||||||
|
|
||||||
|
import { Card, CardContent, CardHeader } from '@/components/ui/card'
|
||||||
|
import { RoleGate } from '@/components/auth/role-gate'
|
||||||
|
import FormSuccess from '@/components/form-success'
|
||||||
|
import { UserRole } from '@prisma/client'
|
||||||
|
import { Button } from '@/components/ui/button'
|
||||||
|
import { toast } from 'sonner'
|
||||||
|
import { admin } from '@/actions/admin'
|
||||||
|
|
||||||
|
const AdminPage = () => {
|
||||||
|
const onServerActionClick = () => {
|
||||||
|
admin()
|
||||||
|
.then((data) => {
|
||||||
|
if (data.error) {
|
||||||
|
toast.error(data.error)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (data.success) {
|
||||||
|
toast.success(data.success)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const onApiRouteClick = () => {
|
||||||
|
fetch('/api/admin')
|
||||||
|
.then((response) => {
|
||||||
|
if (response.ok) {
|
||||||
|
toast.success('Allow API Route')
|
||||||
|
} else {
|
||||||
|
toast.error('Forbidden API Route')
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Card className="w-[600px]">
|
||||||
|
<CardHeader>
|
||||||
|
<p className="text-2xl font-semibold text-center">
|
||||||
|
🔑 Admin
|
||||||
|
</p>
|
||||||
|
</CardHeader>
|
||||||
|
<CardContent className="space-y-4">
|
||||||
|
<RoleGate allowedRole={UserRole.ADMIN}>
|
||||||
|
<FormSuccess message="You are allowed to see this content!"/>
|
||||||
|
</RoleGate>
|
||||||
|
<div className="flex flex-row justify-between items-center rounded-lg border p-3 shadow-md">
|
||||||
|
<p className="text-sm font-medium">
|
||||||
|
Admin-only API Route
|
||||||
|
</p>
|
||||||
|
<Button onClick={onApiRouteClick}>
|
||||||
|
Click to test
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
<div className="flex flex-row justify-between items-center rounded-lg border p-3 shadow-md">
|
||||||
|
<p className="text-sm font-medium">
|
||||||
|
Admin-only Server Action
|
||||||
|
</p>
|
||||||
|
<Button onClick={onServerActionClick}>
|
||||||
|
Click to test
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</CardContent>
|
||||||
|
</Card>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export default AdminPage
|
14
app/[locale]/(protected)/cabinet/client/page.tsx
Normal file
14
app/[locale]/(protected)/cabinet/client/page.tsx
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
'use client'
|
||||||
|
|
||||||
|
import { UserInfo } from '@/components/cabinet/user-info'
|
||||||
|
import { useCurrentUser } from '@/hooks/useCurrentUser'
|
||||||
|
|
||||||
|
const ClientPage = ({ params }: any) => {
|
||||||
|
const user = useCurrentUser()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<UserInfo user={user} label="💻 Client component"/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export default ClientPage
|
@ -1,10 +1,9 @@
|
|||||||
'use client'
|
'use client'
|
||||||
|
|
||||||
import { logout } from '@/actions/logout'
|
import { logout } from '@/actions/logout'
|
||||||
import { useCurrentUser } from '@/hooks/useCurrentUser'
|
|
||||||
|
|
||||||
const CabinetPage = () => {
|
const CabinetPage = ({ params }: any) => {
|
||||||
const user = useCurrentUser()
|
|
||||||
const btnOnClick = () => logout()
|
const btnOnClick = () => logout()
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
15
app/[locale]/(protected)/cabinet/server/page.tsx
Normal file
15
app/[locale]/(protected)/cabinet/server/page.tsx
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
'use server'
|
||||||
|
|
||||||
|
import { currentUser } from '@/lib/auth'
|
||||||
|
import { UserInfo } from '@/components/cabinet/user-info'
|
||||||
|
|
||||||
|
const ServerPage = async () => {
|
||||||
|
const user = await currentUser()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<UserInfo user={user} label="🗄️ Server component"/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export default ServerPage
|
||||||
|
|
@ -1,4 +1,4 @@
|
|||||||
import Navbar from '@/app/[locale]/(protected)/_components/navbar'
|
import { Navbar } from '@/app/[locale]/(protected)/_components/navbar'
|
||||||
|
|
||||||
interface ProtectedLayoutProps {
|
interface ProtectedLayoutProps {
|
||||||
children: React.ReactNode;
|
children: React.ReactNode;
|
||||||
|
@ -10,7 +10,6 @@ type Props = {
|
|||||||
const AuthLayout = ({ children }: Props) => {
|
const AuthLayout = ({ children }: Props) => {
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Navbar/>
|
|
||||||
<div
|
<div
|
||||||
className="h-full flex items-center justify-center bg-[radial-gradient(ellipse_at_top,_var(--tw-gradient-stops))] from-sky-400 to-blue-800">
|
className="h-full flex items-center justify-center bg-[radial-gradient(ellipse_at_top,_var(--tw-gradient-stops))] from-sky-400 to-blue-800">
|
||||||
{children}
|
{children}
|
||||||
|
@ -30,7 +30,7 @@ input[aria-invalid='false']:not(:placeholder-shown) {
|
|||||||
--card-foreground: 222.2 84% 4.9%;
|
--card-foreground: 222.2 84% 4.9%;
|
||||||
--popover: 0 0% 100%;
|
--popover: 0 0% 100%;
|
||||||
--popover-foreground: 222.2 84% 4.9%;
|
--popover-foreground: 222.2 84% 4.9%;
|
||||||
--primary: 221.2 83.2% 53.3%;
|
--primary: 200 98% 39%;
|
||||||
--primary-foreground: 210 40% 98%;
|
--primary-foreground: 210 40% 98%;
|
||||||
--secondary: 210 40% 96.1%;
|
--secondary: 210 40% 96.1%;
|
||||||
--secondary-foreground: 222.2 47.4% 11.2%;
|
--secondary-foreground: 222.2 47.4% 11.2%;
|
||||||
|
@ -6,6 +6,8 @@ import { lc } from '@/lib/utils'
|
|||||||
import './globals.css'
|
import './globals.css'
|
||||||
import { SessionProvider } from 'next-auth/react'
|
import { SessionProvider } from 'next-auth/react'
|
||||||
import { auth } from '@/config/auth'
|
import { auth } from '@/config/auth'
|
||||||
|
import Navbar from '@/components/auth/navbar'
|
||||||
|
import { Toaster } from '@/components/ui/sonner'
|
||||||
|
|
||||||
const inter = Inter({ subsets: ['cyrillic'] })
|
const inter = Inter({ subsets: ['cyrillic'] })
|
||||||
|
|
||||||
@ -24,6 +26,8 @@ export default async function RootLayout ({ params: { locale }, children }: Read
|
|||||||
<html lang={lc(locale)?.java}>
|
<html lang={lc(locale)?.java}>
|
||||||
<body className={inter.className}>
|
<body className={inter.className}>
|
||||||
<I18nProviderClient locale={locale} fallback="Loading...">
|
<I18nProviderClient locale={locale} fallback="Loading...">
|
||||||
|
<Navbar/>
|
||||||
|
<Toaster/>
|
||||||
{children}
|
{children}
|
||||||
</I18nProviderClient>
|
</I18nProviderClient>
|
||||||
</body>
|
</body>
|
||||||
|
10
app/api/admin/route.ts
Normal file
10
app/api/admin/route.ts
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
import { NextResponse } from 'next/server'
|
||||||
|
import { currentRole } from '@/lib/auth'
|
||||||
|
import { UserRole } from '@prisma/client'
|
||||||
|
|
||||||
|
export async function GET () {
|
||||||
|
const role = await currentRole()
|
||||||
|
const status: number = role === UserRole.ADMIN ? 200 : 403
|
||||||
|
|
||||||
|
return new NextResponse(null, { status })
|
||||||
|
}
|
@ -15,7 +15,7 @@ export default function robots (): MetadataRoute.Robots {
|
|||||||
{
|
{
|
||||||
userAgent: '*',
|
userAgent: '*',
|
||||||
allow: ['/'],
|
allow: ['/'],
|
||||||
disallow: ['/auth/', '/api/'],
|
disallow: ['/auth/', '/api/', '/en/auth/', '/en/api/'],
|
||||||
crawlDelay: 3,
|
crawlDelay: 3,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
|
@ -26,7 +26,7 @@ export const CardWrapper = ({
|
|||||||
}: Props) => {
|
}: Props) => {
|
||||||
return (
|
return (
|
||||||
<Card
|
<Card
|
||||||
className="shadow-2xl max-w-[430px] w-full sm:min-w-[430px]">
|
className={`shadow-2xl max-w-[430px] w-full sm:min-w-[430px]`}>
|
||||||
<CardHeader>
|
<CardHeader>
|
||||||
<Header label={headerLabel} title={headerTitle}/>
|
<Header label={headerLabel} title={headerTitle}/>
|
||||||
</CardHeader>
|
</CardHeader>
|
||||||
@ -34,7 +34,7 @@ export const CardWrapper = ({
|
|||||||
{children}
|
{children}
|
||||||
</CardContent>
|
</CardContent>
|
||||||
{showSocial && <CardFooter className="flex-wrap">
|
{showSocial && <CardFooter className="flex-wrap">
|
||||||
<div className="relative flex-none w-[100%] mb-4" style={{ background: 'block' }}>
|
<div className="relative flex-none w-[100%] mb-4">
|
||||||
<div className="absolute inset-0 flex items-center">
|
<div className="absolute inset-0 flex items-center">
|
||||||
<span className="w-full border-t"></span>
|
<span className="w-full border-t"></span>
|
||||||
</div>
|
</div>
|
||||||
|
@ -10,7 +10,7 @@ const LogoutButton = ({ children }: LogoutButtonProps) => {
|
|||||||
const onClick = () => logout()
|
const onClick = () => logout()
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<span onClick={onClick} className="cursor-pointer">
|
<span onClick={onClick}>
|
||||||
{children}
|
{children}
|
||||||
</span>
|
</span>
|
||||||
)
|
)
|
||||||
|
23
components/auth/role-gate.tsx
Normal file
23
components/auth/role-gate.tsx
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
'use client'
|
||||||
|
|
||||||
|
import { UserRole } from '@prisma/client'
|
||||||
|
import { useCurrentRole } from '@/hooks/useCurrentRole'
|
||||||
|
import FormError from '@/components/form-error'
|
||||||
|
|
||||||
|
interface RoleGateProps {
|
||||||
|
children: React.ReactNode
|
||||||
|
allowedRole: UserRole
|
||||||
|
}
|
||||||
|
|
||||||
|
export const RoleGate = ({
|
||||||
|
children,
|
||||||
|
allowedRole,
|
||||||
|
}: RoleGateProps) => {
|
||||||
|
const role = useCurrentRole()
|
||||||
|
|
||||||
|
if (role !== allowedRole) {
|
||||||
|
return <FormError message="You do not have permission to view this content!"/>
|
||||||
|
}
|
||||||
|
|
||||||
|
return <>{children}</>
|
||||||
|
}
|
@ -3,26 +3,29 @@
|
|||||||
import { useCurrentUser } from '@/hooks/useCurrentUser'
|
import { useCurrentUser } from '@/hooks/useCurrentUser'
|
||||||
import { DropdownMenu, DropdownMenuContent, DropdownMenuItem, DropdownMenuTrigger } from '@/components/ui/dropdown-menu'
|
import { DropdownMenu, DropdownMenuContent, DropdownMenuItem, DropdownMenuTrigger } from '@/components/ui/dropdown-menu'
|
||||||
import { Avatar, AvatarFallback, AvatarImage } from '@/components/ui/avatar'
|
import { Avatar, AvatarFallback, AvatarImage } from '@/components/ui/avatar'
|
||||||
import { FaUser } from 'react-icons/fa6'
|
|
||||||
import { IoExitOutline } from 'react-icons/io5'
|
import { IoExitOutline } from 'react-icons/io5'
|
||||||
import LogoutButton from '@/components/auth/logout-button'
|
import LogoutButton from '@/components/auth/logout-button'
|
||||||
|
|
||||||
|
const fallbackInitials = (name?: string | null | undefined): string => {
|
||||||
|
return (name ?? '').split(' ').map((w: string) => w[0]).join('').toUpperCase()
|
||||||
|
}
|
||||||
|
|
||||||
const UserButton = () => {
|
const UserButton = () => {
|
||||||
const user = useCurrentUser()
|
const user = useCurrentUser()
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<DropdownMenu>
|
<DropdownMenu>
|
||||||
<DropdownMenuTrigger>
|
<DropdownMenuTrigger className="outline-0">
|
||||||
<Avatar>
|
<Avatar>
|
||||||
<AvatarImage src={user?.image || ''} alt="User Avatar"/>
|
<AvatarImage src={user?.image || ''} alt="User Avatar"/>
|
||||||
<AvatarFallback className="bg-sky-400">
|
<AvatarFallback className="bg-sky-600 text-muted text-lg font-light">
|
||||||
<FaUser className="text-muted"/>
|
{fallbackInitials(user?.name)}
|
||||||
</AvatarFallback>
|
</AvatarFallback>
|
||||||
</Avatar>
|
</Avatar>
|
||||||
</DropdownMenuTrigger>
|
</DropdownMenuTrigger>
|
||||||
<DropdownMenuContent className="w-40" align="end">
|
<DropdownMenuContent className="w-28 p-0" align="end">
|
||||||
<LogoutButton>
|
<LogoutButton>
|
||||||
<DropdownMenuItem className="cursor-pointer">
|
<DropdownMenuItem className="cursor-pointer p-2">
|
||||||
<IoExitOutline className="w-4 h-4 mr-2"/>
|
<IoExitOutline className="w-4 h-4 mr-2"/>
|
||||||
Logout
|
Logout
|
||||||
</DropdownMenuItem>
|
</DropdownMenuItem>
|
||||||
|
44
components/cabinet/user-info.tsx
Normal file
44
components/cabinet/user-info.tsx
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
import { type ExtendedUser } from '@/config/auth'
|
||||||
|
import { Card, CardContent, CardHeader } from '@/components/ui/card'
|
||||||
|
import { Badge } from '@/components/ui/badge'
|
||||||
|
|
||||||
|
interface UserInfoProps {
|
||||||
|
user?: ExtendedUser
|
||||||
|
label: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export const UserInfo = ({ user, label }: UserInfoProps) => {
|
||||||
|
return (
|
||||||
|
<Card className={`max-w-[430px] w-full sm:min-w-[430px] shadow-md`}>
|
||||||
|
<CardHeader>
|
||||||
|
<p className="text-2xl font-semibold text-center">
|
||||||
|
{label}
|
||||||
|
</p>
|
||||||
|
</CardHeader>
|
||||||
|
<CardContent className="space-y-4">
|
||||||
|
<div className="flex flex-row justify-between items-center rounded-lg border p-3 shadow-sm">
|
||||||
|
<span className="text-sm font-medium">ID</span>
|
||||||
|
<span className="trancate text-xs max-w-[180px] font-mono p-1 bg-slate-100 rounded-md">{user?.id}</span>
|
||||||
|
</div>
|
||||||
|
<div className="flex flex-row justify-between items-center rounded-lg border p-3 shadow-sm">
|
||||||
|
<span className="text-sm font-medium">Name</span>
|
||||||
|
<span className="trancate text-xs max-w-[180px] font-mono p-1 bg-slate-100 rounded-md">{user?.name}</span>
|
||||||
|
</div>
|
||||||
|
<div className="flex flex-row justify-between items-center rounded-lg border p-3 shadow-sm">
|
||||||
|
<span className="text-sm font-medium">Email</span>
|
||||||
|
<span className="trancate text-xs max-w-[180px] font-mono p-1 bg-slate-100 rounded-md">{user?.email}</span>
|
||||||
|
</div>
|
||||||
|
<div className="flex flex-row justify-between items-center rounded-lg border p-3 shadow-sm">
|
||||||
|
<span className="text-sm font-medium">Role</span>
|
||||||
|
<span className="trancate text-xs max-w-[180px] font-mono p-1 bg-slate-100 rounded-md">{user?.role}</span>
|
||||||
|
</div>
|
||||||
|
<div className="flex flex-row justify-between items-center rounded-lg border p-3 shadow-sm">
|
||||||
|
<span className="text-sm font-medium">Two factor authentication</span>
|
||||||
|
<Badge variant={user?.isTwoFactorEnabled ? 'success' : 'destructive'}>
|
||||||
|
{user?.isTwoFactorEnabled ? 'ON' : 'OFF'}
|
||||||
|
</Badge>
|
||||||
|
</div>
|
||||||
|
</CardContent>
|
||||||
|
</Card>
|
||||||
|
)
|
||||||
|
}
|
@ -1,15 +1,14 @@
|
|||||||
import { TriangleAlert } from 'lucide-react'
|
import { TriangleAlert } from 'lucide-react'
|
||||||
|
|
||||||
type Props = {
|
type FormErrorProps = {
|
||||||
message?: string
|
message?: string
|
||||||
}
|
}
|
||||||
|
|
||||||
const FormError = ({ message }: Props) => {
|
const FormError = ({ message }: FormErrorProps) => {
|
||||||
if (!message) return null
|
if (!message) return null
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div className="bg-destructive/15 p-3 rounded-md flex items-center gap-x-2 text-sm text-destructive">
|
||||||
className="bg-destructive/15 p-3 rounded-md flex items-center gap-x-2 text-sm text-destructive">
|
|
||||||
<TriangleAlert className="w-4 h-4"/>
|
<TriangleAlert className="w-4 h-4"/>
|
||||||
<p>{message}</p>
|
<p>{message}</p>
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,15 +1,14 @@
|
|||||||
import { CircleCheck } from 'lucide-react'
|
import { CircleCheck } from 'lucide-react'
|
||||||
|
|
||||||
type Props = {
|
type FormSuccessProps = {
|
||||||
message?: string
|
message?: string
|
||||||
}
|
}
|
||||||
|
|
||||||
const FormSuccess = ({ message }: Props) => {
|
const FormSuccess = ({ message }: FormSuccessProps) => {
|
||||||
if (!message) return null
|
if (!message) return null
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div className="bg-lime-500/15 p-3 rounded-md flex items-center gap-x-2 text-sm text-lime-800">
|
||||||
className="bg-lime-500/15 p-3 rounded-md flex items-center gap-x-2 text-sm text-lime-800">
|
|
||||||
<CircleCheck className="w-4 h-4"/>
|
<CircleCheck className="w-4 h-4"/>
|
||||||
<p>{message}</p>
|
<p>{message}</p>
|
||||||
</div>
|
</div>
|
||||||
|
@ -7,15 +7,18 @@ import { ChangeEvent } from 'react'
|
|||||||
export default function LocaleSwitcher () {
|
export default function LocaleSwitcher () {
|
||||||
const changeLocale = useChangeLocale()
|
const changeLocale = useChangeLocale()
|
||||||
const locale = useCurrentLocale()
|
const locale = useCurrentLocale()
|
||||||
const selectHandler = (e: ChangeEvent<HTMLSelectElement>) => changeLocale(e.target.value as loc)
|
const selectHandler = (e: ChangeEvent<HTMLSelectElement>) =>
|
||||||
|
changeLocale(e.target.value as loc)
|
||||||
|
|
||||||
// {cn(styles['yo-locale-switcher'], 'pr-4')}
|
// {cn(styles['yo-locale-switcher'], 'pr-4')}
|
||||||
return (//@ts-ignore
|
return (//@ts-ignore
|
||||||
<select className="appearance-none bg-transparent block text-center text-xs text-sky-600 py-1 px-2 my-2 mx-0 outline-0"
|
<form><select
|
||||||
|
className="appearance-none bg-transparent block text-center text-xs text-sky-600 py-1 px-2 my-2 mx-0 outline-0"
|
||||||
aria-label="Switch locale"
|
aria-label="Switch locale"
|
||||||
defaultValue={locale} onChange={selectHandler}
|
defaultValue={locale} onChange={selectHandler}
|
||||||
>
|
>
|
||||||
{LC.map(item => (<option key={item.iso} value={item.code} className="pr-4">
|
{LC.map(item => (<option key={item.iso} value={item.code} className="pr-4">
|
||||||
{item.iso.toUpperCase()}
|
{item.iso.toUpperCase()}
|
||||||
</option>))}
|
</option>))}
|
||||||
</select>)
|
</select></form>)
|
||||||
}
|
}
|
38
components/ui/badge.tsx
Normal file
38
components/ui/badge.tsx
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
import * as React from 'react'
|
||||||
|
import { cva, type VariantProps } from 'class-variance-authority'
|
||||||
|
|
||||||
|
import { cn } from '@/lib/utils'
|
||||||
|
|
||||||
|
const badgeVariants = cva(
|
||||||
|
'inline-flex items-center rounded-full border px-2.5 py-0.5 text-xs font-semibold transition-colors focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2',
|
||||||
|
{
|
||||||
|
variants: {
|
||||||
|
variant: {
|
||||||
|
default:
|
||||||
|
'border-transparent bg-primary text-primary-foreground hover:bg-primary/80',
|
||||||
|
secondary:
|
||||||
|
'border-transparent bg-secondary text-secondary-foreground hover:bg-secondary/80',
|
||||||
|
destructive:
|
||||||
|
'border-transparent bg-destructive text-destructive-foreground hover:bg-destructive/80',
|
||||||
|
outline: 'text-foreground',
|
||||||
|
success:
|
||||||
|
'border-transparent bg-emerald-500 text-primary-foreground hover:bg-emerald-500/80',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
defaultVariants: {
|
||||||
|
variant: 'default',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
export interface BadgeProps
|
||||||
|
extends React.HTMLAttributes<HTMLDivElement>,
|
||||||
|
VariantProps<typeof badgeVariants> {}
|
||||||
|
|
||||||
|
function Badge ({ className, variant, ...props }: BadgeProps) {
|
||||||
|
return (
|
||||||
|
<div className={cn(badgeVariants({ variant }), className)} {...props} />
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export { Badge, badgeVariants }
|
31
components/ui/sonner.tsx
Normal file
31
components/ui/sonner.tsx
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
"use client"
|
||||||
|
|
||||||
|
import { useTheme } from "next-themes"
|
||||||
|
import { Toaster as Sonner } from "sonner"
|
||||||
|
|
||||||
|
type ToasterProps = React.ComponentProps<typeof Sonner>
|
||||||
|
|
||||||
|
const Toaster = ({ ...props }: ToasterProps) => {
|
||||||
|
const { theme = "system" } = useTheme()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Sonner
|
||||||
|
theme={theme as ToasterProps["theme"]}
|
||||||
|
className="toaster group"
|
||||||
|
toastOptions={{
|
||||||
|
classNames: {
|
||||||
|
toast:
|
||||||
|
"group toast group-[.toaster]:bg-background group-[.toaster]:text-foreground group-[.toaster]:border-border group-[.toaster]:shadow-lg",
|
||||||
|
description: "group-[.toast]:text-muted-foreground",
|
||||||
|
actionButton:
|
||||||
|
"group-[.toast]:bg-primary group-[.toast]:text-primary-foreground",
|
||||||
|
cancelButton:
|
||||||
|
"group-[.toast]:bg-muted group-[.toast]:text-muted-foreground",
|
||||||
|
},
|
||||||
|
}}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export { Toaster }
|
@ -1,4 +1,4 @@
|
|||||||
import NextAuth from 'next-auth'
|
import NextAuth, { type DefaultSession } from 'next-auth'
|
||||||
import { UserRole } from '@prisma/client'
|
import { UserRole } from '@prisma/client'
|
||||||
import { PrismaAdapter } from '@auth/prisma-adapter'
|
import { PrismaAdapter } from '@auth/prisma-adapter'
|
||||||
import db from '@/lib/db'
|
import db from '@/lib/db'
|
||||||
@ -9,9 +9,16 @@ import { getCurrentLocale } from '@/locales/server'
|
|||||||
import { type loc } from '@/config/locales'
|
import { type loc } from '@/config/locales'
|
||||||
import { getTwoFactorConfirmationByUserId } from '@/data/two-factor-confirmation'
|
import { getTwoFactorConfirmationByUserId } from '@/data/two-factor-confirmation'
|
||||||
|
|
||||||
|
export type ExtendedUser = DefaultSession['user'] & {
|
||||||
|
role: UserRole,
|
||||||
|
locale: loc,
|
||||||
|
isTwoFactorEnabled?: boolean,
|
||||||
|
image?: string
|
||||||
|
}
|
||||||
|
|
||||||
declare module 'next-auth' {
|
declare module 'next-auth' {
|
||||||
interface Session {
|
interface Session {
|
||||||
user: { role: UserRole, locale: loc, image?: string }
|
user: ExtendedUser
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -67,6 +74,10 @@ export const {
|
|||||||
session.user.role = token.role as UserRole
|
session.user.role = token.role as UserRole
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (session.user) {
|
||||||
|
session.user.isTwoFactorEnabled = token.isTwoFactorEnabled as boolean
|
||||||
|
}
|
||||||
|
|
||||||
session.user.locale = getCurrentLocale()
|
session.user.locale = getCurrentLocale()
|
||||||
|
|
||||||
return session
|
return session
|
||||||
@ -79,6 +90,7 @@ export const {
|
|||||||
if (!existingUser) return token
|
if (!existingUser) return token
|
||||||
|
|
||||||
token.role = existingUser.role
|
token.role = existingUser.role
|
||||||
|
token.isTwoFactorEnabled = existingUser.isTwoFactorEnabled
|
||||||
|
|
||||||
return token
|
return token
|
||||||
},
|
},
|
||||||
|
@ -37,4 +37,6 @@ const LC: Locale[] = [
|
|||||||
|
|
||||||
const locales: loc[] = LC.map((locale: Locale) => locale.code)
|
const locales: loc[] = LC.map((locale: Locale) => locale.code)
|
||||||
|
|
||||||
export { locales, defaultLocale, fallbackLocale, LC, importLocales, type loc }
|
const SKIP_I18N_URLS = '/api/'
|
||||||
|
|
||||||
|
export { locales, defaultLocale, fallbackLocale, LC, importLocales, type loc, SKIP_I18N_URLS }
|
@ -1,6 +1,9 @@
|
|||||||
import { UUID_V4_REGEX } from '@/config/validation'
|
import { UUID_V4_REGEX } from '@/config/validation'
|
||||||
|
|
||||||
export const USER_PROFILE_URL: string = '/cabinet'
|
export const USER_PROFILE_URL: string = '/cabinet'
|
||||||
|
export const USER_SERVER_URL: string = `${USER_PROFILE_URL}/server`
|
||||||
|
export const USER_CLIENT_URL: string = `${USER_PROFILE_URL}/client`
|
||||||
|
export const USER_ADMIN_URL: string = `${USER_PROFILE_URL}/admin`
|
||||||
export const AUTH_URL: string = '/auth/'
|
export const AUTH_URL: string = '/auth/'
|
||||||
export const AUTH_LOGIN_URL: string = `${AUTH_URL}login`
|
export const AUTH_LOGIN_URL: string = `${AUTH_URL}login`
|
||||||
export const AUTH_REGISTER_URL: string = `${AUTH_URL}register`
|
export const AUTH_REGISTER_URL: string = `${AUTH_URL}register`
|
||||||
@ -9,6 +12,8 @@ export const AUTH_ERROR_URL: string = `${AUTH_URL}error`
|
|||||||
export const AUTH_USER_VERIFICATION_URL: string = `${AUTH_URL}user-verification`
|
export const AUTH_USER_VERIFICATION_URL: string = `${AUTH_URL}user-verification`
|
||||||
export const AUTH_NEW_PASSWORD_URL: string = `${AUTH_URL}new-password`
|
export const AUTH_NEW_PASSWORD_URL: string = `${AUTH_URL}new-password`
|
||||||
|
|
||||||
|
export const CABINET_ROUTES: string[] = [USER_SERVER_URL, USER_CLIENT_URL, USER_ADMIN_URL, USER_PROFILE_URL] as const
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* An array of routes that accessible to the public.
|
* An array of routes that accessible to the public.
|
||||||
* These routes do not requite authentication.
|
* These routes do not requite authentication.
|
||||||
@ -31,10 +36,10 @@ export const authRoutesRegEx = [
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The prefix for API authentication routes.
|
* The prefix for API authentication routes.
|
||||||
* Routes that start with this prefix are used for API authentication purpose.
|
* Routes that start with this prefix are used for API authentication purposes.
|
||||||
* @type {string}
|
* @type {string}
|
||||||
*/
|
*/
|
||||||
export const apiAuthPrefix: string = '/api/auth'
|
export const apiAuthPrefixRegEx: string = '/api/(auth|admin)'
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The default redirect path after logging in.
|
* The default redirect path after logging in.
|
||||||
|
7
hooks/useCurrentRole.ts
Normal file
7
hooks/useCurrentRole.ts
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
import { useSession } from 'next-auth/react'
|
||||||
|
|
||||||
|
export const useCurrentRole = () => {
|
||||||
|
const session = useSession()
|
||||||
|
|
||||||
|
return session.data?.user?.role
|
||||||
|
}
|
60
lib/CSP.ts
Normal file
60
lib/CSP.ts
Normal file
@ -0,0 +1,60 @@
|
|||||||
|
// @Docs https://content-security-policy.com/
|
||||||
|
|
||||||
|
import { NextRequest, NextResponse } from 'next/server'
|
||||||
|
|
||||||
|
export class CSP {
|
||||||
|
private readonly request: any
|
||||||
|
private readonly on: boolean
|
||||||
|
private readonly request_headers?: Headers
|
||||||
|
private readonly csp_with_nonce?: string
|
||||||
|
private next_response?: NextResponse
|
||||||
|
|
||||||
|
constructor (request: any, on?: boolean) {
|
||||||
|
this.request = request
|
||||||
|
this.on = on ?? true
|
||||||
|
|
||||||
|
if (this.on) {
|
||||||
|
const nonce = Buffer.from(crypto.randomUUID()).toString('base64')
|
||||||
|
this.csp_with_nonce = CSP.contentSecurityPolicyHeaderValue(nonce)
|
||||||
|
this.request_headers = new Headers(this.request.headers)
|
||||||
|
this.request_headers.set('x-nonce', nonce)
|
||||||
|
this.request_headers.set('x-xxx', '123')
|
||||||
|
this.request_headers.set('Content-Security-Policy', this.csp_with_nonce)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static contentSecurityPolicyHeaderValue (nonce: string): string {
|
||||||
|
//style-src 'self' 'nonce-${nonce}';
|
||||||
|
return `
|
||||||
|
default-src 'self';
|
||||||
|
script-src 'self' 'nonce-${nonce}' 'strict-dynamic';
|
||||||
|
style-src 'self' 'unsafe-inline';
|
||||||
|
img-src 'self' blob: data: avatars.githubusercontent.com *.googleusercontent.com;
|
||||||
|
connect-src 'self';
|
||||||
|
font-src 'self';
|
||||||
|
object-src 'none';
|
||||||
|
base-uri 'self';
|
||||||
|
form-action 'self';
|
||||||
|
frame-ancestors 'none';
|
||||||
|
upgrade-insecure-requests`.replace(/\s{2,}/g, ' ').trim()
|
||||||
|
}
|
||||||
|
|
||||||
|
next (middleware?: (request: NextRequest) => NextResponse<unknown>): NextResponse<unknown> {
|
||||||
|
if (!this.on) {
|
||||||
|
return middleware ? middleware(this.request) : NextResponse.next()
|
||||||
|
}
|
||||||
|
|
||||||
|
if (middleware) {
|
||||||
|
const reqNext = new NextRequest(this.request, { headers: this.request_headers })
|
||||||
|
this.next_response = middleware(reqNext)
|
||||||
|
} else {
|
||||||
|
this.next_response = NextResponse.next({
|
||||||
|
request: { headers: this.request_headers },
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
this.next_response.headers.set('Content-Security-Policy', this.csp_with_nonce as string)
|
||||||
|
|
||||||
|
return this.next_response
|
||||||
|
}
|
||||||
|
}
|
13
lib/auth.ts
Normal file
13
lib/auth.ts
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
import { auth } from '@/config/auth'
|
||||||
|
|
||||||
|
export const currentUser = async () => {
|
||||||
|
const session = await auth()
|
||||||
|
|
||||||
|
return session?.user
|
||||||
|
}
|
||||||
|
|
||||||
|
export const currentRole = async () => {
|
||||||
|
const session = await auth()
|
||||||
|
|
||||||
|
return session?.user?.role
|
||||||
|
}
|
@ -21,6 +21,8 @@ export const testPathnameRegex = (
|
|||||||
const pattern: string = `^(/(${locales.join('|')}))?(${pages.flatMap(
|
const pattern: string = `^(/(${locales.join('|')}))?(${pages.flatMap(
|
||||||
(p) => (p === '/' ? ['', '/'] : p)).join('|')})/?$`
|
(p) => (p === '/' ? ['', '/'] : p)).join('|')})/?$`
|
||||||
|
|
||||||
|
//console.log(pattern)
|
||||||
|
|
||||||
return RegExp(pattern, 'is').test(pathName)
|
return RegExp(pattern, 'is').test(pathName)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3,64 +3,56 @@ import NextAuth from 'next-auth'
|
|||||||
import { NextRequest, NextResponse } from 'next/server'
|
import { NextRequest, NextResponse } from 'next/server'
|
||||||
import { defaultLocale, locales } from '@/config/locales'
|
import { defaultLocale, locales } from '@/config/locales'
|
||||||
import authConfig from '@/auth.config'
|
import authConfig from '@/auth.config'
|
||||||
import { apiAuthPrefix, AUTH_LOGIN_URL, authRoutesRegEx, DEFAULT_LOGIN_REDIRECT, publicRoutes } from '@/config/routes'
|
import { apiAuthPrefixRegEx, AUTH_LOGIN_URL, authRoutesRegEx, DEFAULT_LOGIN_REDIRECT, publicRoutes } from '@/config/routes'
|
||||||
import { testPathnameRegex } from '@/lib/utils'
|
import { testPathnameRegex } from '@/lib/utils'
|
||||||
import { createI18nMiddleware } from 'next-international/middleware'
|
import { createI18nMiddleware } from 'next-international/middleware'
|
||||||
|
import { CSP } from '@/lib/CSP'
|
||||||
|
|
||||||
interface AppRouteHandlerFnContext {
|
interface AppRouteHandlerFnContext {
|
||||||
params?: Record<string, string | string[]>;
|
params?: Record<string, string | string[]>;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const I18nMiddleware = createI18nMiddleware({
|
||||||
|
locales, defaultLocale, urlMappingStrategy: 'rewriteDefault',
|
||||||
|
})
|
||||||
|
|
||||||
|
const { auth } = NextAuth(authConfig)
|
||||||
|
|
||||||
export const middleware = (request: NextRequest, event: AppRouteHandlerFnContext): NextResponse | null => {
|
export const middleware = (request: NextRequest, event: AppRouteHandlerFnContext): NextResponse | null => {
|
||||||
return NextAuth(authConfig).auth((request): any => {
|
|
||||||
|
|
||||||
|
return auth((request): any => {
|
||||||
|
//const csp = new CSP(request, process.env.NODE_ENV === 'production')
|
||||||
|
const csp = new CSP(request, false)
|
||||||
const { nextUrl }: { nextUrl: NextURL } = request
|
const { nextUrl }: { nextUrl: NextURL } = request
|
||||||
const isLoggedIn: boolean = !!request.auth
|
|
||||||
const isApiAuthRoute: boolean = nextUrl.pathname.startsWith(apiAuthPrefix)
|
|
||||||
const isPublicRoute: boolean = testPathnameRegex(publicRoutes, nextUrl.pathname)
|
|
||||||
const isAuthRoute: boolean = testPathnameRegex(authRoutesRegEx, nextUrl.pathname)
|
|
||||||
|
|
||||||
if (isApiAuthRoute) {
|
if (nextUrl.pathname.match(apiAuthPrefixRegEx)) {
|
||||||
return null
|
return csp.next()
|
||||||
}
|
}
|
||||||
|
|
||||||
const I18nMiddleware = createI18nMiddleware({
|
const isLoggedIn: boolean = !!request.auth
|
||||||
locales, defaultLocale, urlMappingStrategy: 'rewriteDefault',
|
const isPublicRoute: boolean = testPathnameRegex(publicRoutes, nextUrl.pathname)
|
||||||
})
|
const isAuthRoute: boolean = testPathnameRegex(authRoutesRegEx, nextUrl.pathname)
|
||||||
|
|
||||||
if (isAuthRoute) {
|
if (isAuthRoute) {
|
||||||
if (isLoggedIn) {
|
if (isLoggedIn) {
|
||||||
return NextResponse.redirect(new URL(DEFAULT_LOGIN_REDIRECT, nextUrl))
|
return NextResponse.redirect(new URL(DEFAULT_LOGIN_REDIRECT, nextUrl))
|
||||||
}
|
}
|
||||||
return I18nMiddleware(request)
|
return csp.next(I18nMiddleware)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!isLoggedIn && !isPublicRoute) {
|
if (!isLoggedIn && !isPublicRoute) {
|
||||||
return NextResponse.redirect(new URL(AUTH_LOGIN_URL, nextUrl))
|
return NextResponse.redirect(new URL(AUTH_LOGIN_URL, nextUrl))
|
||||||
}
|
}
|
||||||
|
|
||||||
return I18nMiddleware(request)
|
return csp.next(I18nMiddleware)
|
||||||
|
|
||||||
})(request, event) as NextResponse
|
})(request, event) as NextResponse
|
||||||
}
|
}
|
||||||
|
|
||||||
// export const config = {
|
|
||||||
// matcher: [
|
|
||||||
// /*
|
|
||||||
// * Match all request paths except for the ones starting with:
|
|
||||||
// * - api (API routes)
|
|
||||||
// * - _next/static (static files)
|
|
||||||
// * - _next/image (image optimization files)
|
|
||||||
// * - favicon.ico (favicon file)
|
|
||||||
// */
|
|
||||||
// {
|
|
||||||
// source: '/((?!.+\\.[\\w]+$|api|_next/image|favicon.ico|robots.txt|trpc).*)', missing: [
|
|
||||||
// { type: 'header', key: 'next-router-prefetch' }, { type: 'header', key: 'purpose', value: 'prefetch' }],
|
|
||||||
// }],
|
|
||||||
// }
|
|
||||||
|
|
||||||
export const config = {
|
export const config = {
|
||||||
matcher: [
|
matcher: [
|
||||||
'/((?!.+\\.[\\w]+$|_next|_next/image|_next/static).*)', '/(api|trpc)(.*)',
|
'/((?!.+\\.[\\w]+$|_next|_next/image|_next/static|favicon.ico|robots.txt).*)',
|
||||||
|
'/',
|
||||||
|
'/(api|trpc)(.*)',
|
||||||
],
|
],
|
||||||
}
|
}
|
20
package-lock.json
generated
20
package-lock.json
generated
@ -24,6 +24,7 @@
|
|||||||
"next": "14.1.4",
|
"next": "14.1.4",
|
||||||
"next-auth": "^5.0.0-beta.16",
|
"next-auth": "^5.0.0-beta.16",
|
||||||
"next-international": "^1.2.4",
|
"next-international": "^1.2.4",
|
||||||
|
"next-themes": "^0.3.0",
|
||||||
"nodemailer": "^6.9.13",
|
"nodemailer": "^6.9.13",
|
||||||
"pino": "^9.0.0",
|
"pino": "^9.0.0",
|
||||||
"pino-http": "^9.0.0",
|
"pino-http": "^9.0.0",
|
||||||
@ -34,6 +35,7 @@
|
|||||||
"react-loader-spinner": "^6.1.6",
|
"react-loader-spinner": "^6.1.6",
|
||||||
"sharp": "^0.33.3",
|
"sharp": "^0.33.3",
|
||||||
"shart": "^0.0.4",
|
"shart": "^0.0.4",
|
||||||
|
"sonner": "^1.4.41",
|
||||||
"tailwind-merge": "^2.2.2",
|
"tailwind-merge": "^2.2.2",
|
||||||
"tailwindcss-animate": "^1.0.7",
|
"tailwindcss-animate": "^1.0.7",
|
||||||
"uuid": "^9.0.1",
|
"uuid": "^9.0.1",
|
||||||
@ -4985,6 +4987,15 @@
|
|||||||
"server-only": "^0.0.1"
|
"server-only": "^0.0.1"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/next-themes": {
|
||||||
|
"version": "0.3.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/next-themes/-/next-themes-0.3.0.tgz",
|
||||||
|
"integrity": "sha512-/QHIrsYpd6Kfk7xakK4svpDI5mmXP0gfvCoJdGpZQ2TOrQZmsW0QxjaiLn8wbIKjtm4BTSqLoix4lxYYOnLJ/w==",
|
||||||
|
"peerDependencies": {
|
||||||
|
"react": "^16.8 || ^17 || ^18",
|
||||||
|
"react-dom": "^16.8 || ^17 || ^18"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/next/node_modules/postcss": {
|
"node_modules/next/node_modules/postcss": {
|
||||||
"version": "8.4.31",
|
"version": "8.4.31",
|
||||||
"resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.31.tgz",
|
"resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.31.tgz",
|
||||||
@ -6333,6 +6344,15 @@
|
|||||||
"atomic-sleep": "^1.0.0"
|
"atomic-sleep": "^1.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/sonner": {
|
||||||
|
"version": "1.4.41",
|
||||||
|
"resolved": "https://registry.npmjs.org/sonner/-/sonner-1.4.41.tgz",
|
||||||
|
"integrity": "sha512-uG511ggnnsw6gcn/X+YKkWPo5ep9il9wYi3QJxHsYe7yTZ4+cOd1wuodOUmOpFuXL+/RE3R04LczdNCDygTDgQ==",
|
||||||
|
"peerDependencies": {
|
||||||
|
"react": "^18.0.0",
|
||||||
|
"react-dom": "^18.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/source-map": {
|
"node_modules/source-map": {
|
||||||
"version": "0.6.1",
|
"version": "0.6.1",
|
||||||
"resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz",
|
"resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz",
|
||||||
|
@ -3,9 +3,9 @@
|
|||||||
"version": "0.1.1",
|
"version": "0.1.1",
|
||||||
"private": true,
|
"private": true,
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "next dev",
|
"dev": "chcp 65001 && next dev",
|
||||||
"build": "next build",
|
"build": "next build",
|
||||||
"start": "next start",
|
"start": "chcp 65001 && next start",
|
||||||
"lint": "next lint",
|
"lint": "next lint",
|
||||||
"browserslist:update": "npx update-browserslist-db@latest",
|
"browserslist:update": "npx update-browserslist-db@latest",
|
||||||
"browserslist": "npx browserslist"
|
"browserslist": "npx browserslist"
|
||||||
@ -32,6 +32,7 @@
|
|||||||
"next": "14.1.4",
|
"next": "14.1.4",
|
||||||
"next-auth": "^5.0.0-beta.16",
|
"next-auth": "^5.0.0-beta.16",
|
||||||
"next-international": "^1.2.4",
|
"next-international": "^1.2.4",
|
||||||
|
"next-themes": "^0.3.0",
|
||||||
"nodemailer": "^6.9.13",
|
"nodemailer": "^6.9.13",
|
||||||
"pino": "^9.0.0",
|
"pino": "^9.0.0",
|
||||||
"pino-http": "^9.0.0",
|
"pino-http": "^9.0.0",
|
||||||
@ -42,6 +43,7 @@
|
|||||||
"react-loader-spinner": "^6.1.6",
|
"react-loader-spinner": "^6.1.6",
|
||||||
"sharp": "^0.33.3",
|
"sharp": "^0.33.3",
|
||||||
"shart": "^0.0.4",
|
"shart": "^0.0.4",
|
||||||
|
"sonner": "^1.4.41",
|
||||||
"tailwind-merge": "^2.2.2",
|
"tailwind-merge": "^2.2.2",
|
||||||
"tailwindcss-animate": "^1.0.7",
|
"tailwindcss-animate": "^1.0.7",
|
||||||
"uuid": "^9.0.1",
|
"uuid": "^9.0.1",
|
||||||
|
@ -29,6 +29,9 @@
|
|||||||
"@/config/*": [
|
"@/config/*": [
|
||||||
"./config/*"
|
"./config/*"
|
||||||
],
|
],
|
||||||
|
"@/hooks/*": [
|
||||||
|
"./hooks/*"
|
||||||
|
],
|
||||||
"@/data/*": [
|
"@/data/*": [
|
||||||
"./data/*"
|
"./data/*"
|
||||||
],
|
],
|
||||||
|
Loading…
Reference in New Issue
Block a user