Spaces:
Runtime error
Runtime error
import { | |
Chart as ChartJS, | |
Filler, | |
Legend, | |
LineElement, | |
PointElement, | |
RadialLinearScale, | |
Tooltip, | |
} from 'chart.js'; | |
import { t } from 'i18next'; | |
import React, { useEffect, useState } from 'react'; | |
import { Radar } from 'react-chartjs-2'; | |
import { useParams } from 'react-router-dom'; | |
import { cvssStringToCIA } from '@/lib/utils'; | |
import { getAuditById } from '@/services/audits'; | |
ChartJS.register( | |
RadialLinearScale, | |
PointElement, | |
LineElement, | |
Filler, | |
Tooltip, | |
Legend, | |
); | |
type Dataset = { | |
label: string; | |
data: number[]; | |
backgroundColor: string; | |
borderColor: string; | |
borderWidth: number; | |
}; | |
type CIATriadProps = { | |
auditId?: string; | |
}; | |
const CIATriad: React.FC<CIATriadProps> = ({ auditId }) => { | |
const paramId = useParams().auditId; | |
if (!auditId) { | |
auditId = paramId; | |
} | |
const [data, setData] = useState<{ | |
labels: string[]; | |
datasets: Dataset[]; | |
}>({ | |
labels: ['Integrity', 'Availability', 'Confidentiality'] as const, | |
datasets: [], | |
}); | |
useEffect(() => { | |
getAuditById(auditId) | |
.then(audit => { | |
setData({ | |
labels: ['Integrity', 'Availability', 'Confidentiality'] as const, | |
datasets: audit.datas.findings.map(finding => { | |
if (!finding.cvssv3) { | |
return { | |
label: finding.title, | |
data: [0, 0, 0], | |
backgroundColor: `rgba(${Math.floor(Math.random() * 155 + 100)}, ${Math.floor(Math.random() * 155 + 100)}, ${Math.floor(Math.random() * 155 + 100)}, 0.2)`, | |
borderColor: 'rgba(255, 255, 255, 0.2)' as const, | |
borderWidth: 2, | |
}; | |
} | |
return { | |
label: finding.title, | |
data: [ | |
cvssStringToCIA('integrity', finding.cvssv3), | |
cvssStringToCIA('availability', finding.cvssv3), | |
cvssStringToCIA('confidentiality', finding.cvssv3), | |
], | |
backgroundColor: `rgba(${Math.floor(Math.random() * 155 + 100)}, ${Math.floor(Math.random() * 155 + 100)}, ${Math.floor(Math.random() * 155 + 100)}, 0.2)`, | |
borderColor: 'rgba(255, 255, 255, 0.2)' as const, | |
borderWidth: 2, | |
}; | |
}), | |
}); | |
}) | |
.catch(console.error); | |
}, [auditId]); | |
const options = { | |
responsive: true, | |
maintainAspectRatio: false, | |
scales: { | |
r: { | |
angleLines: { | |
display: true, | |
color: 'rgba(255, 255, 255, 0.1)' as const, | |
}, | |
suggestedMin: 0, | |
suggestedMax: 2, | |
ticks: { | |
stepSize: 1, | |
display: false, | |
}, | |
pointLabels: { | |
font: { | |
size: 14, | |
}, | |
color: 'white' as const, | |
}, | |
grid: { | |
color: 'rgba(255, 255, 255, 0.1)' as const, | |
}, | |
}, | |
}, | |
plugins: { | |
legend: { | |
position: 'bottom' as const, | |
title: { | |
display: true, | |
text: t('filters'), | |
color: 'white' as const, | |
font: { | |
weight: 'bold' as const, | |
size: 15, | |
}, | |
}, | |
labels: { | |
color: 'white' as const, | |
boxWidth: 20, | |
padding: 20, | |
}, | |
}, | |
datalabels: { | |
formatter: () => '', | |
}, | |
tooltip: { | |
backgroundColor: 'rgba(0, 0, 0, 0.8)' as const, | |
titleColor: 'white' as const, | |
bodyColor: 'white' as const, | |
borderColor: 'white' as const, | |
borderWidth: 1, | |
}, | |
}, | |
}; | |
return ( | |
<div className="bg-gray-800 rounded-lg p-6"> | |
<div | |
className="chart-container" | |
style={{ height: '400px', width: '100%' }} | |
> | |
<Radar data={data} options={options} /> | |
</div> | |
</div> | |
); | |
}; | |
export default CIATriad; | |