130 lines
5.8 KiB
TypeScript
130 lines
5.8 KiB
TypeScript
import { Button } from "@/components/ui/button"
|
|
import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/components/ui/card"
|
|
import { AlertCircle, CheckCircleIcon, XCircleIcon, Loader2, ShieldCheck, Search, Lightbulb } from "lucide-react"
|
|
import { useState } from "react"
|
|
import { SiAuthentik } from "react-icons/si"
|
|
import { type SecurityResults } from "@/app/api/users/security/route"
|
|
|
|
export const SecurityTab = () => {
|
|
const [scanning, setScanning] = useState(false)
|
|
const [scanResults, setScanResults] = useState<SecurityResults | null>(null)
|
|
const [error, setError] = useState<string | null>(null)
|
|
|
|
const scanAcc = async () => {
|
|
setScanning(true)
|
|
try {
|
|
const res = await fetch("/api/users/security")
|
|
const data = await res.json()
|
|
setScanResults(data)
|
|
setScanning(false)
|
|
} catch (error) {
|
|
console.error(error)
|
|
setError(error instanceof Error ? error.message : "An unknown error occurred")
|
|
setScanning(false)
|
|
}
|
|
}
|
|
|
|
// If password was changed over 3 months ago, this will be true
|
|
const shouldResetPass = scanResults?.authentik?.passwordChangeDate && scanResults?.authentik?.passwordChangeDate < new Date(Date.now() - 90 * 24 * 60 * 60 * 1000)
|
|
|
|
// If user has no 2FA methods setup, this will be true
|
|
const insufficient2FA = scanResults?.authentik?.authenticators.length === 0
|
|
|
|
return (
|
|
<div className="grid gap-6 grid-cols-1 md:grid-cols-2">
|
|
<Card>
|
|
<CardHeader>
|
|
<CardTitle className="flex items-center gap-1">
|
|
<ShieldCheck size={18} />
|
|
<span className="text-xl">Account Security Scan</span>
|
|
</CardTitle>
|
|
<CardDescription>Evaluate the security of your account with a simple button click!</CardDescription>
|
|
</CardHeader>
|
|
<CardContent>
|
|
{error ? (
|
|
<div className="text-red-500">
|
|
<div className="flex items-center gap-1 mb-2">
|
|
<AlertCircle className="h-4 w-4" />
|
|
<p className="font-bold">Error</p>
|
|
</div>
|
|
<p className="text-sm">{error}</p>
|
|
</div>
|
|
) : scanResults ? (
|
|
<>
|
|
<div className="flex items-center gap-2">
|
|
<SiAuthentik size={20} />
|
|
<h3 className="text-xl">Authentik</h3>
|
|
</div>
|
|
|
|
{shouldResetPass ? (
|
|
<div className="flex items-center gap-2 mt-2">
|
|
<XCircleIcon className="h-4 w-4 text-red-500" />
|
|
<p className="text-sm">Password last changed {"on " + (scanResults?.authentik?.passwordChangeDate ? new Date(scanResults.authentik.passwordChangeDate).toLocaleDateString() : "never")}</p>
|
|
</div>
|
|
) : (
|
|
<div className="flex items-center gap-2 mt-2">
|
|
<CheckCircleIcon className="h-4 w-4 text-green-500" />
|
|
<p className="text-sm">Password last changed {"on " + (scanResults?.authentik?.passwordChangeDate ? new Date(scanResults.authentik.passwordChangeDate).toLocaleDateString() : "never")}</p>
|
|
</div>
|
|
)}
|
|
|
|
{insufficient2FA ? (
|
|
<div className="flex items-center gap-2 mt-2">
|
|
<XCircleIcon className="h-4 w-4 text-red-500" />
|
|
<p className="text-sm">No 2FA methods setup</p>
|
|
</div>
|
|
) : (
|
|
<div className="flex items-center gap-2 mt-2">
|
|
<CheckCircleIcon className="h-4 w-4 text-green-500" />
|
|
<p className="text-sm"><span className="font-bold">{scanResults?.authentik?.authenticators.length}</span> two-factor authentication methods setup</p>
|
|
</div>
|
|
)}
|
|
</>
|
|
) : (
|
|
<>
|
|
{scanning ? (
|
|
<Button className="w-full cursor-pointer" disabled>
|
|
<Loader2 className="h-4 w-4 animate-spin" /> Scanning...
|
|
</Button>
|
|
) : (
|
|
<Button className="w-full cursor-pointer" onClick={scanAcc}>
|
|
<Search className="h-4 w-4" /> Scan my Account
|
|
</Button>
|
|
)}
|
|
</>
|
|
)}
|
|
</CardContent>
|
|
</Card>
|
|
<Card>
|
|
<CardHeader>
|
|
<CardTitle className="flex items-center gap-1">
|
|
<Lightbulb size={18} />
|
|
<span className="text-xl">Recommendations</span>
|
|
</CardTitle>
|
|
<CardDescription>Steps you can take to improve your account's security</CardDescription>
|
|
</CardHeader>
|
|
<CardContent>
|
|
<ul className="list-disc pl-5 space-y-2">
|
|
<li className="text-sm">Enable Two-Factor Authentication</li>
|
|
<li className="text-sm">Use a strong and unique password</li>
|
|
<li className="text-sm">Always make sure the URL matches <span className="font-bold">librecloud.cc</span></li>
|
|
<div className="border border-green-400 rounded-md p-1 w-full flex items-center justify-between">
|
|
<span className="text-sm flex items-center">
|
|
<Search size={16} className="mx-1" />
|
|
https://<span className="font-bold text-green-400">librecloud.cc</span>
|
|
</span>
|
|
<CheckCircleIcon size={16} className="text-green-500 mr-1" />
|
|
</div>
|
|
<div className="border border-red-400 rounded-md p-1 w-full flex items-center justify-between">
|
|
<span className="text-sm flex items-center">
|
|
<Search size={16} className="mx-1" />
|
|
https://<span className="font-bold text-red-400">libre-cloud-login.com</span>
|
|
</span>
|
|
<XCircleIcon size={16} className="text-red-500 mr-1" />
|
|
</div>
|
|
</ul>
|
|
</CardContent>
|
|
</Card>
|
|
</div>
|
|
)
|
|
} |