Spaces:
Runtime error
Runtime error
import { | |
BarElement, | |
CategoryScale, | |
Chart as ChartJS, | |
ChartOptions, | |
Legend, | |
LinearScale, | |
Title, | |
Tooltip, | |
} from 'chart.js'; | |
import annotationPlugin from 'chartjs-plugin-annotation'; | |
import React, { useEffect, useRef, useState } from 'react'; | |
import { Bar } from 'react-chartjs-2'; | |
import { useParams } from 'react-router-dom'; | |
import { cvssStringToScore } from '@/lib/utils'; | |
import { getAuditById } from '@/services/audits'; | |
import { getAuditsByClientName } from '@/services/clients'; | |
ChartJS.register( | |
CategoryScale, | |
LinearScale, | |
BarElement, | |
Title, | |
Tooltip, | |
Legend, | |
annotationPlugin, | |
); | |
type AverageCVSSProps = { | |
auditId?: string; | |
clientName?: string; | |
}; | |
const AverageCVSS: React.FC<AverageCVSSProps> = ({ auditId, clientName }) => { | |
const paramId = useParams().auditId; | |
const auditIdRef = useRef(auditId ?? paramId); | |
const [averageCVSS, setAverageCVSS] = useState(0); | |
const [data, setData] = useState({ | |
labels: [''], | |
datasets: [ | |
{ | |
data: [0], | |
backgroundColor: '#3498db', | |
}, | |
], | |
}); | |
useEffect(() => { | |
if (auditIdRef.current === undefined) { | |
auditIdRef.current = paramId; | |
} | |
if (clientName === undefined) { | |
getAuditById(auditIdRef.current) | |
.then(audit => { | |
setAverageCVSS( | |
Math.round( | |
(audit.datas.findings.reduce( | |
(acc, finding) => acc + cvssStringToScore(finding.cvssv3 ?? ''), | |
0, | |
) / | |
audit.datas.findings.length) * | |
10, | |
) / 10, | |
); | |
setData({ | |
labels: audit.datas.findings.map(finding => finding.title), | |
datasets: [ | |
{ | |
data: audit.datas.findings.map(finding => | |
cvssStringToScore(finding.cvssv3 ?? ''), | |
), | |
// @ts-expect-error component accepts string[] to put multiple colors, but the type is string | |
backgroundColor: audit.datas.findings.map(finding => | |
cvssStringToScore(finding.cvssv3 ?? '') >= 9 | |
? '#FF4136' | |
: cvssStringToScore(finding.cvssv3 ?? '') >= 7 | |
? '#FF851B' | |
: cvssStringToScore(finding.cvssv3 ?? '') >= 4 | |
? '#FFDC00' | |
: '#2ECC40', | |
), | |
}, | |
], | |
}); | |
}) | |
.catch(console.error); | |
} else { | |
getAuditsByClientName(clientName) | |
.then(audits => { | |
setAverageCVSS( | |
Math.round( | |
(audits.reduce( | |
(acc, audit) => acc + cvssStringToScore(audit.cvssv3), | |
0, | |
) / | |
audits.length) * | |
10, | |
) / 10, | |
); | |
setData({ | |
labels: audits.map(audit => audit.name), | |
datasets: [ | |
{ | |
data: audits.map(audit => cvssStringToScore(audit.cvssv3)), | |
// @ts-expect-error component accepts string[] to put multiple colors, but the type is string | |
backgroundColor: audits.map(audit => | |
cvssStringToScore(audit.cvssv3) >= 9 | |
? '#FF4136' | |
: cvssStringToScore(audit.cvssv3) >= 7 | |
? '#FF851B' | |
: cvssStringToScore(audit.cvssv3) >= 4 | |
? '#FFDC00' | |
: '#2ECC40', | |
), | |
}, | |
], | |
}); | |
}) | |
.catch(console.error); | |
} | |
}, [auditId, averageCVSS, clientName, paramId]); | |
const options: ChartOptions<'bar'> = { | |
indexAxis: 'y' as const, | |
responsive: true, | |
maintainAspectRatio: false, | |
layout: { | |
padding: { | |
top: 30, | |
}, | |
}, | |
scales: { | |
x: { | |
beginAtZero: true, | |
max: 10, | |
ticks: { | |
stepSize: 2, | |
color: 'white' as const, | |
}, | |
grid: { | |
color: 'rgba(255, 255, 255, 0.1)' as const, | |
}, | |
}, | |
y: { | |
ticks: { | |
color: 'white' as const, | |
}, | |
grid: { | |
display: false, | |
}, | |
}, | |
}, | |
plugins: { | |
legend: { | |
display: false, | |
}, | |
datalabels: { | |
formatter: () => '', | |
}, | |
annotation: { | |
annotations: { | |
line1: { | |
type: 'line' as const, | |
xMin: averageCVSS, | |
xMax: averageCVSS, | |
borderColor: '#2ecc71' as const, | |
borderWidth: 2, | |
borderDash: [5, 5], | |
}, | |
}, | |
}, | |
}, | |
}; | |
return ( | |
<div className="bg-gray-800 rounded-lg p-6"> | |
<div className="relative"> | |
<div className="absolute top-0 left-0 w-full text-right pr-4 text-green-400 text-sm"> | |
Average CVSS: {averageCVSS} | |
</div> | |
<div className="chart-container" style={{ height: '400px' }}> | |
<Bar data={data} options={options} /> | |
</div> | |
</div> | |
</div> | |
); | |
}; | |
export default AverageCVSS; | |