diff --git a/playwright.config.js b/playwright.config.js index 294d54e4..c88309c9 100644 --- a/playwright.config.js +++ b/playwright.config.js @@ -42,7 +42,8 @@ const config = { /* Collect trace when retrying the failed test. See https://playwright.dev/docs/trace-viewer */ trace: 'on-first-retry', - ignoreHTTPSErrors: true + ignoreHTTPSErrors: true, + screenshot: 'only-on-failure' }, /* Configure projects for major browsers */ @@ -101,7 +102,7 @@ const config = { ], /* Folder for test artifacts such as screenshots, videos, traces, etc. */ - // outputDir: 'test-results/', + outputDir: 'test-results/', /* Run your local dev server before starting the tests */ // webServer: { diff --git a/src/__tests__/TagPage/VulnerabilitiesDetails.test.js b/src/__tests__/TagPage/VulnerabilitiesDetails.test.js index 9b015276..3202b747 100644 --- a/src/__tests__/TagPage/VulnerabilitiesDetails.test.js +++ b/src/__tests__/TagPage/VulnerabilitiesDetails.test.js @@ -22,6 +22,14 @@ const mockCVEList = { CVEListForImage: { Tag: '', Page: { ItemCount: 20, TotalCount: 20 }, + Summary: { + Count: 5, + UnknownCount: 1, + LowCount: 1, + MediumCount: 1, + HighCount: 1, + CriticalCount: 1, + }, CVEList: [ { Id: 'CVE-2020-16156', @@ -499,6 +507,7 @@ describe('Vulnerabilties page', () => { jest.spyOn(api, 'get').mockResolvedValue({ status: 200, data: { data: mockCVEList } }); render(); await waitFor(() => expect(screen.getAllByText('Vulnerabilities')).toHaveLength(1)); + await waitFor(() => expect(screen.getAllByText('Total 5')).toHaveLength(1)); await waitFor(() => expect(screen.getAllByText(/fixed in/i)).toHaveLength(20)); }); @@ -515,7 +524,7 @@ describe('Vulnerabilties page', () => { it('renders no vulnerabilities if there are not any', async () => { jest.spyOn(api, 'get').mockResolvedValue({ status: 200, - data: { data: { CVEListForImage: { Tag: '', Page: {}, CVEList: [] } } } + data: { data: { CVEListForImage: { Tag: '', Page: {}, CVEList: [], Summary: {} } } } }); render(); await waitFor(() => expect(screen.getAllByText('No Vulnerabilities')).toHaveLength(1)); diff --git a/src/api.js b/src/api.js index 0596edf9..9ea5687c 100644 --- a/src/api.js +++ b/src/api.js @@ -97,7 +97,7 @@ const endpoints = { if (!isEmpty(searchTerm)) { query += `, searchedCVE: "${searchTerm}"`; } - return `${query}){Tag Page {TotalCount ItemCount} CVEList {Id Title Description Severity PackageList {Name InstalledVersion FixedVersion}}}}`; + return `${query}){Tag Page {TotalCount ItemCount} CVEList {Id Title Description Severity PackageList {Name InstalledVersion FixedVersion}} Summary {Count UnknownCount LowCount MediumCount HighCount CriticalCount}}}`; }, allVulnerabilitiesForRepo: (name) => `/v2/_zot/ext/search?query={CVEListForImage(image: "${name}"){Tag Page {TotalCount ItemCount} CVEList {Id Title Description Severity Reference PackageList {Name InstalledVersion FixedVersion}}}}`, diff --git a/src/components/Shared/VulnerabilityCountCard.jsx b/src/components/Shared/VulnerabilityCountCard.jsx new file mode 100644 index 00000000..c478136c --- /dev/null +++ b/src/components/Shared/VulnerabilityCountCard.jsx @@ -0,0 +1,92 @@ +import React from 'react'; + +import makeStyles from '@mui/styles/makeStyles'; +import { Stack, Tooltip } from '@mui/material'; + +const criticalColor = '#ff5c74'; +const criticalBorderColor = '#f9546d'; + +const highColor = '#ff6840'; +const highBorderColor = '#ee6b49'; + +const mediumColor = '#ffa052'; +const mediumBorderColor = '#f19d5b'; + +const lowColor = '#f9f486'; +const lowBorderColor = '#f0ed94'; + +const unknownColor = '#f2ffdd'; +const unknownBorderColor = '#e9f4d7'; + +const fontSize = '0.75rem'; + +const useStyles = makeStyles((theme) => ({ + cveCountCard: { + display: 'flex', + alignItems: 'center', + paddingLeft: '0.5rem', + paddingRight: '0.5rem', + color: theme.palette.primary.main, + fontSize: fontSize, + fontWeight: '600', + borderRadius: '3px', + marginBottom: '0' + }, + severityList: { + fontSize: fontSize, + display: 'flex', + flexWrap: 'wrap', + alignItems: 'center', + gap: '0.5em' + }, + criticalSeverity: { + backgroundColor: criticalColor, + border: '1px solid ' + criticalBorderColor + }, + highSeverity: { + backgroundColor: highColor, + border: '1px solid ' + highBorderColor + }, + mediumSeverity: { + backgroundColor: mediumColor, + border: '1px solid ' + mediumBorderColor + }, + lowSeverity: { + backgroundColor: lowColor, + border: '1px solid ' + lowBorderColor + }, + unknownSeverity: { + backgroundColor: unknownColor, + border: '1px solid ' + unknownBorderColor + } +})); + +function VulnerabilitiyCountCard(props) { + const classes = useStyles(); + const { total, critical, high, medium, low, unknown } = props; + + return ( + +
Total {total}
+
+ +
C {critical}
+
+ +
H {high}
+
+ +
M {medium}
+
+ +
L {low}
+
+ +
U {unknown}
+
+
+
+ ); +} + +export default VulnerabilitiyCountCard; diff --git a/src/components/Tag/Tabs/VulnerabilitiesDetails.jsx b/src/components/Tag/Tabs/VulnerabilitiesDetails.jsx index f80612e2..30a68328 100644 --- a/src/components/Tag/Tabs/VulnerabilitiesDetails.jsx +++ b/src/components/Tag/Tabs/VulnerabilitiesDetails.jsx @@ -31,14 +31,25 @@ import ViewHeadlineIcon from '@mui/icons-material/ViewHeadline'; import ViewAgendaIcon from '@mui/icons-material/ViewAgenda'; import VulnerabilitiyCard from '../../Shared/VulnerabilityCard'; +import VulnerabilityCountCard from '../../Shared/VulnerabilityCountCard'; const useStyles = makeStyles((theme) => ({ + searchAndDisplayBar: { + display: 'flex', + justifyContent: 'space-between' + }, title: { color: theme.palette.primary.main, fontSize: '1.5rem', fontWeight: '600', marginBottom: '0' }, + cveCountSummary: { + color: theme.palette.primary.main, + fontSize: '1.5rem', + fontWeight: '600', + marginBottom: '0' + }, cveId: { color: theme.palette.primary.main, fontSize: '1rem', @@ -67,6 +78,7 @@ const useStyles = makeStyles((theme) => ({ search: { position: 'relative', maxWidth: '100%', + flex: 0.95, flexDirection: 'row', alignItems: 'center', justifyContent: 'space-between', @@ -74,15 +86,18 @@ const useStyles = makeStyles((theme) => ({ border: '0.063rem solid #E7E7E7', borderRadius: '0.625rem' }, + expandableSearchInput: { + flexGrow: 0.95 + }, view: { alignContent: 'right', variant: 'outlined' }, viewModes: { position: 'relative', + alignItems: 'baseline', maxWidth: '100%', flexDirection: 'row', - alignItems: 'right', justifyContent: 'right' }, searchIcon: { @@ -114,6 +129,7 @@ function VulnerabilitiesDetails(props) { const classes = useStyles(); const [cveData, setCveData] = useState([]); const [allCveData, setAllCveData] = useState([]); + const [cveSummary, setCVESummary] = useState({}); const [isLoading, setIsLoading] = useState(true); const [isLoadingAllCve, setIsLoadingAllCve] = useState(true); const abortController = useMemo(() => new AbortController(), []); @@ -147,9 +163,23 @@ function VulnerabilitiesDetails(props) { .then((response) => { if (response.data && response.data.data) { let cveInfo = response.data.data.CVEListForImage?.CVEList; + let summary = response.data.data.CVEListForImage?.Summary; let cveListData = mapCVEInfo(cveInfo); setCveData((previousState) => (pageNumber === 1 ? cveListData : [...previousState, ...cveListData])); setIsEndOfList(response.data.data.CVEListForImage.Page?.ItemCount < EXPLORE_PAGE_SIZE); + setCVESummary((previousState) => { + if (isEmpty(summary)) { + return previousState; + } + return { + Count: summary.Count, + UnknownCount: summary.UnknownCount, + LowCount: summary.LowCount, + MediumCount: summary.MediumCount, + HighCount: summary.HighCount, + CriticalCount: summary.CriticalCount + }; + }); } else if (response.data.errors) { setIsEndOfList(true); } @@ -159,6 +189,7 @@ function VulnerabilitiesDetails(props) { console.error(e); setIsLoading(false); setCveData([]); + setCVESummary(() => {}); setIsEndOfList(true); }); }; @@ -283,6 +314,27 @@ function VulnerabilitiesDetails(props) { ); }; + const renderCVESummary = () => { + if (cveSummary === undefined) { + return; + } + + console.log('Test'); + + return !isEmpty(cveSummary) ? ( + + ) : ( + <> + ); + }; + const renderListBottom = () => { if (isLoading) { return ; @@ -364,6 +416,7 @@ function VulnerabilitiesDetails(props) { + {renderCVESummary()}