24 Commits

Author SHA1 Message Date
5bf7d5652c feat: add cve summary in vulnerability tab (#416)
Signed-off-by: Laurentiu Niculae <niculae.laurentiu1@gmail.com>
2024-01-18 22:00:19 +02:00
12f9229320 fix(export vuln): change sheet name and download options name (#417)
Signed-off-by: Andreea-Lupu <andreealupu1470@yahoo.com>
2024-01-18 15:41:18 +02:00
df19fa811c feat: add expand/collapse view list buttons for vulnerabilities (#409)
Signed-off-by: Andreea-Lupu <andreealupu1470@yahoo.com>
2024-01-17 15:38:39 +02:00
6cda89c710 fix: change 'csv' to 'CSV' in the vulnerabilities download options list (#413)
Signed-off-by: Andreea-Lupu <andreealupu1470@yahoo.com>
2024-01-17 14:16:46 +02:00
12b474e126 fix: update zot documentation urls (#411)
resolves #410

Signed-off-by: Andrei Aaron <aaaron@luxoft.com>
2024-01-12 13:47:16 +02:00
a9db66bd34 feat: add freebsd as an OS filter (#407) (#408)
Signed-off-by: Doug Rabson <dfr@rabson.org>
2023-12-28 14:52:27 +02:00
f4600b8b79 feat: export vulnerabilities list
Signed-off-by: Andreea-Lupu <andreealupu1470@yahoo.com>
2023-12-20 09:23:47 -08:00
c375c0697a feat: added button to delete tag
Signed-off-by: Petu Eusebiu <peusebiu@cisco.com>
2023-12-15 15:32:28 -08:00
2e1e2e92b7 fix: show a loading message while waiting for a response
Signed-off-by: Andreea-Lupu <andreealupu1470@yahoo.com>
2023-12-11 13:14:30 -08:00
d9370fb9c1 feat: starred repos implementation (#399)
Signed-off-by: Andreea-Lupu <andreealupu1470@yahoo.com>
2023-12-07 16:36:54 +02:00
e97e04eee5 ci: dco job should run only on PRs (#396)
See also message in b919279eef

Signed-off-by: Andrei Aaron <aaaron@luxoft.com>
2023-11-28 11:23:15 +02:00
a288523a3f feat: vulnerability chips - show icon before string (#392)
Signed-off-by: Andreea-Lupu <andreealupu1470@yahoo.com>
2023-11-28 10:51:23 +02:00
fad5572db4 feat: add prefix zot to /auth urls (#389)
See: https://github.com/project-zot/zot/issues/1883

Signed-off-by: Andrei Aaron <aaaron@luxoft.com>
2023-10-20 13:17:24 +03:00
19e366ee1f fix: use the official icon
Signed-off-by: Ramkumar Chinchani <rchincha@cisco.com>
2023-10-11 19:56:02 -07:00
b41fb2f841 patch: update nodata display on homepage
Signed-off-by: Raul-Cristian Kele <raulkeleblk@gmail.com>
2023-10-02 14:16:15 -07:00
b787273b84 fix: fixed display of new signature tooltips in some cases (#379)
Signed-off-by: Raul-Cristian Kele <raulkeleblk@gmail.com>
2023-08-29 19:46:13 +03:00
9ecd46e4d0 ci(end-to-end): Fix a few issues with the workflow (#380)
- free up disk space before running tests
- remove uneeded call to /v2/_catalog.
- add a check to make sure the images are scanned for CVEs before tests start

Signed-off-by: Andrei Aaron <aaaron@luxoft.com>
2023-08-29 19:09:52 +03:00
845726cd08 feat: Update signature integration to display extra info (#378)
Signed-off-by: Raul-Cristian Kele <raulkeleblk@gmail.com>
2023-08-28 18:03:32 +03:00
ac84c375c0 feat: add customizable generic oidc login button
Rebased and modified to reflect https://github.com/project-zot/zot/pull/1691 conclusion

Signed-off-by: Damien Degois <damien@degois.info>
2023-08-28 15:15:17 +03:00
96008d67be feat: Implement no data component
- Implement customizeable component for no data display
- Added component to homepage

Signed-off-by: Raul-Cristian Kele <raulkeleblk@gmail.com>
2023-08-18 20:09:28 +03:00
087b42693f patch: update integration tests
Signed-off-by: raulkele <raulkeleblk@gmail.com>
Signed-off-by: Raul-Cristian Kele <raulkeleblk@gmail.com>
2023-08-15 20:12:30 +03:00
8f4c23bf40 fix: Update tooltip for vulnerability chips
Signed-off-by: Raul-Cristian Kele <raulkeleblk@gmail.com>
2023-08-08 21:33:22 +03:00
54c764c996 patch: update cve api usage
- updated CVEListForImage api calls
- updated ImageListWithCVEFixed api calls
- now cves are shown for specific tag
- fixed tags now only shows tags that match platform with current digest
- moved platform selector on tagdetails page

Signed-off-by: Raul-Cristian Kele <raulkeleblk@gmail.com>
2023-07-27 08:51:27 +03:00
44289c751f fix: fixed login page refresh bug
Signed-off-by: Raul-Cristian Kele <raulkeleblk@gmail.com>
2023-07-19 17:44:44 +03:00
45 changed files with 1293 additions and 311 deletions

View File

@ -2,16 +2,18 @@
name: DCO name: DCO
on: on:
pull_request: pull_request:
push:
branches: branches:
- main - main
permissions: read-all
jobs: jobs:
check: check:
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- uses: actions/checkout@v2 - uses: actions/checkout@v4
- name: Set up Python 3.x - name: Set up Python 3.x
uses: actions/setup-python@v1 uses: actions/setup-python@v4
with: with:
python-version: '3.x' python-version: '3.x'
- name: Check DCO - name: Check DCO

View File

@ -23,6 +23,14 @@ jobs:
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- name: Cleanup disk space
run: |
# To free up ~15 GB of disk space
sudo rm -rf /opt/ghc
sudo rm -rf /usr/local/share/boost
sudo rm -rf /usr/local/lib/android
sudo rm -rf /usr/share/dotnet
- name: Checkout zui repository - name: Checkout zui repository
uses: actions/checkout@v3 uses: actions/checkout@v3
with: with:
@ -86,7 +94,7 @@ jobs:
- name: Build zot - name: Build zot
run: | run: |
cd $GITHUB_WORKSPACE/zot cd $GITHUB_WORKSPACE/zot
make binary make binary ZUI_BUILD_PATH=$GITHUB_WORKSPACE/build
ls -l bin/ ls -l bin/
- name: Bringup zot server - name: Bringup zot server
@ -116,6 +124,11 @@ jobs:
cd $GITHUB_WORKSPACE cd $GITHUB_WORKSPACE
make playwright-browsers make playwright-browsers
- name: Trigger CVE scanning
run: |
# trigger CVE scanning for all images before running the tests
curl -X POST -H "Content-Type: application/json" -m 600 --data '{ "query": "{ ImageListForCVE (id:\"CVE-2021-43616\") { Results { RepoName Tag } } }" }' http://$REGISTRY_HOST:$REGISTRY_PORT/v2/_zot/ext/search
- name: Run integration tests - name: Run integration tests
run: | run: |
cd $GITHUB_WORKSPACE cd $GITHUB_WORKSPACE

103
package-lock.json generated
View File

@ -19,6 +19,7 @@
"@testing-library/user-event": "^13.5.0", "@testing-library/user-event": "^13.5.0",
"axios": "^0.24.0", "axios": "^0.24.0",
"downshift": "^6.1.12", "downshift": "^6.1.12",
"export-from-json": "^1.7.3",
"lodash": "^4.17.21", "lodash": "^4.17.21",
"luxon": "^2.5.2", "luxon": "^2.5.2",
"markdown-to-jsx": "^7.1.7", "markdown-to-jsx": "^7.1.7",
@ -26,7 +27,8 @@
"react-dom": "^17.0.2", "react-dom": "^17.0.2",
"react-router-dom": "^6.2.1", "react-router-dom": "^6.2.1",
"react-sticky-el": "^2.0.9", "react-sticky-el": "^2.0.9",
"web-vitals": "^2.1.3" "web-vitals": "^2.1.3",
"xlsx": "^0.18.5"
}, },
"devDependencies": { "devDependencies": {
"@babel/plugin-proposal-private-property-in-object": "^7.16.7", "@babel/plugin-proposal-private-property-in-object": "^7.16.7",
@ -5592,6 +5594,14 @@
"node": ">=8.9" "node": ">=8.9"
} }
}, },
"node_modules/adler-32": {
"version": "1.3.1",
"resolved": "https://registry.npmjs.org/adler-32/-/adler-32-1.3.1.tgz",
"integrity": "sha512-ynZ4w/nUUv5rrsR8UUGoe1VC9hZj6V5hU9Qw1HlMDJGEJw5S7TfTErWTjMys6M7vr0YWcPqs3qAr4ss0nDfP+A==",
"engines": {
"node": ">=0.8"
}
},
"node_modules/agent-base": { "node_modules/agent-base": {
"version": "6.0.2", "version": "6.0.2",
"resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz", "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz",
@ -6547,6 +6557,18 @@
"node": ">=4" "node": ">=4"
} }
}, },
"node_modules/cfb": {
"version": "1.2.2",
"resolved": "https://registry.npmjs.org/cfb/-/cfb-1.2.2.tgz",
"integrity": "sha512-KfdUZsSOw19/ObEWasvBP/Ac4reZvAGauZhs6S/gqNhXhI7cKwvlH7ulj+dOEYnca4bm4SGo8C1bTAQvnTjgQA==",
"dependencies": {
"adler-32": "~1.3.0",
"crc-32": "~1.2.0"
},
"engines": {
"node": ">=0.8"
}
},
"node_modules/chalk": { "node_modules/chalk": {
"version": "4.1.2", "version": "4.1.2",
"resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz",
@ -6780,6 +6802,14 @@
"node": ">=4" "node": ">=4"
} }
}, },
"node_modules/codepage": {
"version": "1.15.0",
"resolved": "https://registry.npmjs.org/codepage/-/codepage-1.15.0.tgz",
"integrity": "sha512-3g6NUTPd/YtuuGrhMnOMRjFc+LJw/bnMp3+0r/Wcz3IXUuCosKRJvMphm5+Q+bvTVGcJJuRvVLuYba+WojaFaA==",
"engines": {
"node": ">=0.8"
}
},
"node_modules/collect-v8-coverage": { "node_modules/collect-v8-coverage": {
"version": "1.0.1", "version": "1.0.1",
"resolved": "https://registry.npmjs.org/collect-v8-coverage/-/collect-v8-coverage-1.0.1.tgz", "resolved": "https://registry.npmjs.org/collect-v8-coverage/-/collect-v8-coverage-1.0.1.tgz",
@ -7030,6 +7060,17 @@
"node": ">=10" "node": ">=10"
} }
}, },
"node_modules/crc-32": {
"version": "1.2.2",
"resolved": "https://registry.npmjs.org/crc-32/-/crc-32-1.2.2.tgz",
"integrity": "sha512-ROmzCKrTnOwybPcJApAA6WBWij23HVfGVNKqqrZpuyZOHqK2CwHSvpGuyt/UNNvaIjEd8X5IFGp4Mh+Ie1IHJQ==",
"bin": {
"crc32": "bin/crc32.njs"
},
"engines": {
"node": ">=0.8"
}
},
"node_modules/cross-spawn": { "node_modules/cross-spawn": {
"version": "7.0.3", "version": "7.0.3",
"resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz",
@ -8926,6 +8967,11 @@
"node": "^14.15.0 || ^16.10.0 || >=18.0.0" "node": "^14.15.0 || ^16.10.0 || >=18.0.0"
} }
}, },
"node_modules/export-from-json": {
"version": "1.7.3",
"resolved": "https://registry.npmjs.org/export-from-json/-/export-from-json-1.7.3.tgz",
"integrity": "sha512-Xg0L0saYz+CBz2MnaZvSEAHr17hWtHAfFWXw/frllG9t6aijuQukiU40ElOeM9nDTrtQPhLJMLN0q8lo897FYg=="
},
"node_modules/express": { "node_modules/express": {
"version": "4.18.2", "version": "4.18.2",
"resolved": "https://registry.npmjs.org/express/-/express-4.18.2.tgz", "resolved": "https://registry.npmjs.org/express/-/express-4.18.2.tgz",
@ -9423,6 +9469,14 @@
"node": ">= 0.6" "node": ">= 0.6"
} }
}, },
"node_modules/frac": {
"version": "1.1.2",
"resolved": "https://registry.npmjs.org/frac/-/frac-1.1.2.tgz",
"integrity": "sha512-w/XBfkibaTl3YDqASwfDUqkna4Z2p9cFSr1aHDt0WoMTECnRfBOv2WArlZILlqgWlmdIlALXGpM2AOhEk5W3IA==",
"engines": {
"node": ">=0.8"
}
},
"node_modules/fraction.js": { "node_modules/fraction.js": {
"version": "4.2.0", "version": "4.2.0",
"resolved": "https://registry.npmjs.org/fraction.js/-/fraction.js-4.2.0.tgz", "resolved": "https://registry.npmjs.org/fraction.js/-/fraction.js-4.2.0.tgz",
@ -16997,6 +17051,17 @@
"integrity": "sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==", "integrity": "sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==",
"dev": true "dev": true
}, },
"node_modules/ssf": {
"version": "0.11.2",
"resolved": "https://registry.npmjs.org/ssf/-/ssf-0.11.2.tgz",
"integrity": "sha512-+idbmIXoYET47hH+d7dfm2epdOMUDjqcB4648sTZ+t2JwoyBFL/insLfB/racrDmsKB3diwsDA696pZMieAC5g==",
"dependencies": {
"frac": "~1.1.2"
},
"engines": {
"node": ">=0.8"
}
},
"node_modules/stable": { "node_modules/stable": {
"version": "0.1.8", "version": "0.1.8",
"resolved": "https://registry.npmjs.org/stable/-/stable-0.1.8.tgz", "resolved": "https://registry.npmjs.org/stable/-/stable-0.1.8.tgz",
@ -18746,6 +18811,22 @@
"url": "https://github.com/sponsors/ljharb" "url": "https://github.com/sponsors/ljharb"
} }
}, },
"node_modules/wmf": {
"version": "1.0.2",
"resolved": "https://registry.npmjs.org/wmf/-/wmf-1.0.2.tgz",
"integrity": "sha512-/p9K7bEh0Dj6WbXg4JG0xvLQmIadrner1bi45VMJTfnbVHsc7yIajZyoSoK60/dtVBs12Fm6WkUI5/3WAVsNMw==",
"engines": {
"node": ">=0.8"
}
},
"node_modules/word": {
"version": "0.3.0",
"resolved": "https://registry.npmjs.org/word/-/word-0.3.0.tgz",
"integrity": "sha512-OELeY0Q61OXpdUfTp+oweA/vtLVg5VDOXh+3he3PNzLGG/y0oylSOC1xRVj0+l4vQ3tj/bB1HVHv1ocXkQceFA==",
"engines": {
"node": ">=0.8"
}
},
"node_modules/word-wrap": { "node_modules/word-wrap": {
"version": "1.2.3", "version": "1.2.3",
"resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.3.tgz", "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.3.tgz",
@ -19133,6 +19214,26 @@
} }
} }
}, },
"node_modules/xlsx": {
"version": "0.18.5",
"resolved": "https://registry.npmjs.org/xlsx/-/xlsx-0.18.5.tgz",
"integrity": "sha512-dmg3LCjBPHZnQp5/F/+nnTa+miPJxUXB6vtk42YjBBKayDNagxGEeIdWApkYPOf3Z3pm3k62Knjzp7lMeTEtFQ==",
"dependencies": {
"adler-32": "~1.3.0",
"cfb": "~1.2.1",
"codepage": "~1.15.0",
"crc-32": "~1.2.1",
"ssf": "~0.11.2",
"wmf": "~1.0.1",
"word": "~0.3.0"
},
"bin": {
"xlsx": "bin/xlsx.njs"
},
"engines": {
"node": ">=0.8"
}
},
"node_modules/xml-name-validator": { "node_modules/xml-name-validator": {
"version": "3.0.0", "version": "3.0.0",
"resolved": "https://registry.npmjs.org/xml-name-validator/-/xml-name-validator-3.0.0.tgz", "resolved": "https://registry.npmjs.org/xml-name-validator/-/xml-name-validator-3.0.0.tgz",

View File

@ -14,6 +14,7 @@
"@testing-library/user-event": "^13.5.0", "@testing-library/user-event": "^13.5.0",
"axios": "^0.24.0", "axios": "^0.24.0",
"downshift": "^6.1.12", "downshift": "^6.1.12",
"export-from-json": "^1.7.3",
"lodash": "^4.17.21", "lodash": "^4.17.21",
"luxon": "^2.5.2", "luxon": "^2.5.2",
"markdown-to-jsx": "^7.1.7", "markdown-to-jsx": "^7.1.7",
@ -21,17 +22,18 @@
"react-dom": "^17.0.2", "react-dom": "^17.0.2",
"react-router-dom": "^6.2.1", "react-router-dom": "^6.2.1",
"react-sticky-el": "^2.0.9", "react-sticky-el": "^2.0.9",
"web-vitals": "^2.1.3" "web-vitals": "^2.1.3",
"xlsx": "^0.18.5"
}, },
"devDependencies": { "devDependencies": {
"@babel/plugin-proposal-private-property-in-object": "^7.16.7",
"@playwright/test": "^1.28.1", "@playwright/test": "^1.28.1",
"eslint": "^8.23.1", "eslint": "^8.23.1",
"eslint-config-prettier": "^8.5.0", "eslint-config-prettier": "^8.5.0",
"eslint-plugin-prettier": "^4.2.1", "eslint-plugin-prettier": "^4.2.1",
"eslint-plugin-react": "^7.31.8", "eslint-plugin-react": "^7.31.8",
"prettier": "^2.7.1", "prettier": "^2.7.1",
"react-scripts": "^5.0.1", "react-scripts": "^5.0.1"
"@babel/plugin-proposal-private-property-in-object": "^7.16.7"
}, },
"scripts": { "scripts": {
"start": "react-scripts start", "start": "react-scripts start",

View File

@ -42,7 +42,8 @@ const config = {
/* Collect trace when retrying the failed test. See https://playwright.dev/docs/trace-viewer */ /* Collect trace when retrying the failed test. See https://playwright.dev/docs/trace-viewer */
trace: 'on-first-retry', trace: 'on-first-retry',
ignoreHTTPSErrors: true ignoreHTTPSErrors: true,
screenshot: 'only-on-failure'
}, },
/* Configure projects for major browsers */ /* Configure projects for major browsers */
@ -101,7 +102,7 @@ const config = {
], ],
/* Folder for test artifacts such as screenshots, videos, traces, etc. */ /* 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 */ /* Run your local dev server before starting the tests */
// webServer: { // webServer: {

View File

@ -34,6 +34,7 @@ const mockImageList = {
Size: '2806985', Size: '2806985',
LastUpdated: '2022-08-09T17:19:53.274069586Z', LastUpdated: '2022-08-09T17:19:53.274069586Z',
IsBookmarked: false, IsBookmarked: false,
IsStarred: false,
NewestImage: { NewestImage: {
Tag: 'latest', Tag: 'latest',
Description: 'w', Description: 'w',
@ -58,6 +59,7 @@ const mockImageList = {
Size: '231383863', Size: '231383863',
LastUpdated: '2022-08-02T01:30:49.193203152Z', LastUpdated: '2022-08-02T01:30:49.193203152Z',
IsBookmarked: false, IsBookmarked: false,
IsStarred: false,
NewestImage: { NewestImage: {
Tag: 'latest', Tag: 'latest',
Description: '', Description: '',
@ -82,6 +84,7 @@ const mockImageList = {
Size: '369311301', Size: '369311301',
LastUpdated: '2022-08-23T00:20:40.144281895Z', LastUpdated: '2022-08-23T00:20:40.144281895Z',
IsBookmarked: false, IsBookmarked: false,
IsStarred: false,
NewestImage: { NewestImage: {
Tag: 'latest', Tag: 'latest',
Description: '', Description: '',
@ -106,6 +109,7 @@ const mockImageList = {
Size: '369311301', Size: '369311301',
LastUpdated: '2022-08-23T00:20:40.144281895Z', LastUpdated: '2022-08-23T00:20:40.144281895Z',
IsBookmarked: false, IsBookmarked: false,
IsStarred: false,
NewestImage: { NewestImage: {
Tag: 'latest', Tag: 'latest',
Description: '', Description: '',
@ -130,6 +134,7 @@ const mockImageList = {
Size: '369311301', Size: '369311301',
LastUpdated: '2022-08-23T00:20:40.144281895Z', LastUpdated: '2022-08-23T00:20:40.144281895Z',
IsBookmarked: false, IsBookmarked: false,
IsStarred: false,
NewestImage: { NewestImage: {
Tag: 'latest', Tag: 'latest',
Description: '', Description: '',
@ -158,6 +163,7 @@ const mockImageList = {
Size: '369311301', Size: '369311301',
LastUpdated: '2022-08-23T00:20:40.144281895Z', LastUpdated: '2022-08-23T00:20:40.144281895Z',
IsBookmarked: false, IsBookmarked: false,
IsStarred: false,
NewestImage: { NewestImage: {
Tag: 'latest', Tag: 'latest',
Description: '', Description: '',
@ -182,6 +188,7 @@ const mockImageList = {
Size: '369311301', Size: '369311301',
LastUpdated: '2022-08-23T00:20:40.144281895Z', LastUpdated: '2022-08-23T00:20:40.144281895Z',
IsBookmarked: false, IsBookmarked: false,
IsStarred: false,
NewestImage: { NewestImage: {
Tag: 'latest', Tag: 'latest',
Description: '', Description: '',
@ -338,4 +345,13 @@ describe('Explore component', () => {
await userEvent.click(bookmarkButton); await userEvent.click(bookmarkButton);
expect(await screen.findAllByTestId('bookmarked')).toHaveLength(1); expect(await screen.findAllByTestId('bookmarked')).toHaveLength(1);
}); });
it('should star a repo if star button is clicked', async () => {
jest.spyOn(api, 'get').mockResolvedValueOnce({ status: 200, data: { data: mockImageList } });
render(<StateExploreWrapper />);
const starButton = (await screen.findAllByTestId('star-button'))[0];
jest.spyOn(api, 'put').mockResolvedValueOnce({ status: 200, data: {} });
await userEvent.click(starButton);
expect(await screen.findAllByTestId('starred')).toHaveLength(1);
});
}); });

View File

@ -21,7 +21,7 @@ const StateFilterCardWrapper = () => {
describe('Filters components', () => { describe('Filters components', () => {
it('renders the filters cards', async () => { it('renders the filters cards', async () => {
render(<StateFilterCardWrapper />); render(<StateFilterCardWrapper />);
expect(screen.getAllByRole('checkbox')).toHaveLength(2); expect(screen.getAllByRole('checkbox')).toHaveLength(3);
const checkbox = screen.getAllByRole('checkbox'); const checkbox = screen.getAllByRole('checkbox');
expect(checkbox[0]).not.toBeChecked(); expect(checkbox[0]).not.toBeChecked();

View File

@ -164,6 +164,48 @@ const mockImageListBookmarks = {
} }
}; };
const mockImageListStars = {
GlobalSearch: {
Page: { TotalCount: 3, ItemCount: 2 },
Repos: [
{
Name: 'alpine',
Size: '2806985',
LastUpdated: '2022-08-09T17:19:53.274069586Z',
NewestImage: {
Tag: 'latest',
Description: 'w',
IsSigned: false,
Licenses: '',
Vendor: '',
Labels: '',
Vulnerabilities: {
MaxSeverity: 'LOW',
Count: 7
}
}
},
{
Name: 'mongo',
Size: '231383863',
LastUpdated: '2022-08-02T01:30:49.193203152Z',
NewestImage: {
Tag: 'latest',
Description: '',
IsSigned: true,
Licenses: '',
Vendor: '',
Labels: '',
Vulnerabilities: {
MaxSeverity: 'HIGH',
Count: 2
}
}
}
]
}
};
beforeEach(() => { beforeEach(() => {
window.scrollTo = jest.fn(); window.scrollTo = jest.fn();
}); });
@ -178,8 +220,8 @@ describe('Home component', () => {
jest.spyOn(api, 'get').mockResolvedValueOnce({ status: 200, data: { data: mockImageList } }); jest.spyOn(api, 'get').mockResolvedValueOnce({ status: 200, data: { data: mockImageList } });
jest.spyOn(api, 'get').mockResolvedValue({ status: 200, data: { data: mockImageListRecent } }); jest.spyOn(api, 'get').mockResolvedValue({ status: 200, data: { data: mockImageListRecent } });
render(<HomeWrapper />); render(<HomeWrapper />);
await waitFor(() => expect(screen.getAllByText(/alpine/i)).toHaveLength(3)); await waitFor(() => expect(screen.getAllByText(/alpine/i)).toHaveLength(4));
await waitFor(() => expect(screen.getAllByText(/mongo/i)).toHaveLength(3)); await waitFor(() => expect(screen.getAllByText(/mongo/i)).toHaveLength(4));
await waitFor(() => expect(screen.getAllByText(/node/i)).toHaveLength(1)); await waitFor(() => expect(screen.getAllByText(/node/i)).toHaveLength(1));
}); });
@ -187,16 +229,16 @@ describe('Home component', () => {
jest.spyOn(api, 'get').mockResolvedValueOnce({ status: 200, data: { data: mockImageList } }); jest.spyOn(api, 'get').mockResolvedValueOnce({ status: 200, data: { data: mockImageList } });
jest.spyOn(api, 'get').mockResolvedValue({ status: 200, data: { data: mockImageListRecent } }); jest.spyOn(api, 'get').mockResolvedValue({ status: 200, data: { data: mockImageListRecent } });
render(<HomeWrapper />); render(<HomeWrapper />);
expect(await screen.findAllByTestId('unverified-icon')).toHaveLength(3); expect(await screen.findAllByTestId('unverified-icon')).toHaveLength(4);
expect(await screen.findAllByTestId('verified-icon')).toHaveLength(4); expect(await screen.findAllByTestId('verified-icon')).toHaveLength(5);
}); });
it('renders vulnerability icons', async () => { it('renders vulnerability icons', async () => {
jest.spyOn(api, 'get').mockResolvedValueOnce({ status: 200, data: { data: mockImageList } }); jest.spyOn(api, 'get').mockResolvedValueOnce({ status: 200, data: { data: mockImageList } });
jest.spyOn(api, 'get').mockResolvedValue({ status: 200, data: { data: mockImageListRecent } }); jest.spyOn(api, 'get').mockResolvedValue({ status: 200, data: { data: mockImageListRecent } });
render(<HomeWrapper />); render(<HomeWrapper />);
expect(await screen.findAllByTestId('low-vulnerability-icon')).toHaveLength(3); expect(await screen.findAllByTestId('low-vulnerability-icon')).toHaveLength(4);
expect(await screen.findAllByTestId('high-vulnerability-icon')).toHaveLength(3); expect(await screen.findAllByTestId('high-vulnerability-icon')).toHaveLength(4);
expect(await screen.findAllByTestId('critical-vulnerability-icon')).toHaveLength(1); expect(await screen.findAllByTestId('critical-vulnerability-icon')).toHaveLength(1);
}); });
@ -204,16 +246,17 @@ describe('Home component', () => {
jest.spyOn(api, 'get').mockRejectedValue({ status: 500, data: {} }); jest.spyOn(api, 'get').mockRejectedValue({ status: 500, data: {} });
const error = jest.spyOn(console, 'error').mockImplementation(() => {}); const error = jest.spyOn(console, 'error').mockImplementation(() => {});
render(<HomeWrapper />); render(<HomeWrapper />);
await waitFor(() => expect(error).toBeCalledTimes(3)); await waitFor(() => expect(error).toBeCalledTimes(4));
}); });
it('should redirect to explore page when clicking view all popular', async () => { it('should redirect to explore page when clicking view all popular', async () => {
jest.spyOn(api, 'get').mockResolvedValueOnce({ status: 200, data: { data: mockImageList } }); jest.spyOn(api, 'get').mockResolvedValueOnce({ status: 200, data: { data: mockImageList } });
jest.spyOn(api, 'get').mockResolvedValueOnce({ status: 200, data: { data: mockImageListRecent } }); jest.spyOn(api, 'get').mockResolvedValueOnce({ status: 200, data: { data: mockImageListRecent } });
jest.spyOn(api, 'get').mockResolvedValueOnce({ status: 200, data: { data: mockImageListBookmarks } }); jest.spyOn(api, 'get').mockResolvedValueOnce({ status: 200, data: { data: mockImageListBookmarks } });
jest.spyOn(api, 'get').mockResolvedValueOnce({ status: 200, data: { data: mockImageListStars } });
render(<HomeWrapper />); render(<HomeWrapper />);
const viewAllButtons = await screen.findAllByText(/view all/i); const viewAllButtons = await screen.findAllByText(/view all/i);
expect(viewAllButtons).toHaveLength(3); expect(viewAllButtons).toHaveLength(4);
jest.spyOn(api, 'get').mockResolvedValue({ status: 200, data: { data: [] } }); jest.spyOn(api, 'get').mockResolvedValue({ status: 200, data: { data: [] } });
fireEvent.click(viewAllButtons[0]); fireEvent.click(viewAllButtons[0]);
expect(mockedUsedNavigate).toHaveBeenCalledWith({ expect(mockedUsedNavigate).toHaveBeenCalledWith({
@ -230,5 +273,10 @@ describe('Home component', () => {
pathname: `/explore`, pathname: `/explore`,
search: createSearchParams({ filter: 'IsBookmarked' }).toString() search: createSearchParams({ filter: 'IsBookmarked' }).toString()
}); });
fireEvent.click(viewAllButtons[3]);
expect(mockedUsedNavigate).toHaveBeenCalledWith({
pathname: `/explore`,
search: createSearchParams({ filter: 'IsStarred' }).toString()
});
}); });
}); });

View File

@ -47,6 +47,7 @@ const mockRepoDetailsData = {
Size: '451554070', Size: '451554070',
Vendors: ['[The Node.js Docker Team](https://github.com/nodejs/docker-node)\n'], Vendors: ['[The Node.js Docker Team](https://github.com/nodejs/docker-node)\n'],
IsBookmarked: false, IsBookmarked: false,
IsStarred: false,
NewestImage: { NewestImage: {
RepoName: 'mongo', RepoName: 'mongo',
IsSigned: true, IsSigned: true,
@ -316,4 +317,13 @@ describe('Repo details component', () => {
await userEvent.click(bookmarkButton); await userEvent.click(bookmarkButton);
expect(await screen.findByTestId('bookmarked')).toBeInTheDocument(); expect(await screen.findByTestId('bookmarked')).toBeInTheDocument();
}); });
it('should star a repo if star button is clicked', async () => {
jest.spyOn(api, 'get').mockResolvedValueOnce({ status: 200, data: { data: mockRepoDetailsData } });
render(<RepoDetailsThemeWrapper />);
const starButton = await screen.findByTestId('star-button');
jest.spyOn(api, 'put').mockResolvedValue({ status: 200, data: {} });
await userEvent.click(starButton);
expect(await screen.findByTestId('starred')).toBeInTheDocument();
});
}); });

View File

@ -22,6 +22,7 @@ const mockedTagsData = [
{ {
tag: 'latest', tag: 'latest',
vendor: 'test1', vendor: 'test1',
isDeletable: true,
manifests: [ manifests: [
{ {
lastUpdated: '2022-07-19T18:06:18.818788283Z', lastUpdated: '2022-07-19T18:06:18.818788283Z',
@ -37,6 +38,7 @@ const mockedTagsData = [
{ {
tag: 'bullseye', tag: 'bullseye',
vendor: 'test1', vendor: 'test1',
isDeletable: true,
manifests: [ manifests: [
{ {
digest: 'sha256:adca4815c494becc1bf053af0c4640b2d81ab1a779e6d649e1b8b92a75f1d559', digest: 'sha256:adca4815c494becc1bf053af0c4640b2d81ab1a779e6d649e1b8b92a75f1d559',
@ -52,6 +54,7 @@ const mockedTagsData = [
{ {
tag: '1.5.2', tag: '1.5.2',
vendor: 'test1', vendor: 'test1',
isDeletable: true,
manifests: [ manifests: [
{ {
lastUpdated: '2022-07-19T18:06:18.818788283Z', lastUpdated: '2022-07-19T18:06:18.818788283Z',
@ -76,6 +79,18 @@ describe('Tags component', () => {
await waitFor(() => expect(screen.queryByText(/OS\/ARCH/i)).not.toBeInTheDocument()); await waitFor(() => expect(screen.queryByText(/OS\/ARCH/i)).not.toBeInTheDocument());
}); });
it('should see delete tag button and its dialog', async () => {
render(<TagsThemeWrapper />);
const deleteBtn = await screen.findAllByTestId('DeleteIcon');
fireEvent.click(deleteBtn[0]);
expect(await screen.findByTestId('delete-dialog')).toBeInTheDocument();
const confirmBtn = await screen.findByTestId('confirm-delete');
expect(confirmBtn).toBeInTheDocument();
fireEvent.click(confirmBtn);
expect(await screen.findByTestId('confirm-delete')).toBeInTheDocument();
expect(await screen.findByTestId('cancel-delete')).toBeInTheDocument();
});
it('should navigate to tag page details when tag is clicked', async () => { it('should navigate to tag page details when tag is clicked', async () => {
render(<TagsThemeWrapper />); render(<TagsThemeWrapper />);
const tagLink = await screen.findByText('latest'); const tagLink = await screen.findByText('latest');

View File

@ -22,6 +22,29 @@ const mockImage = {
vendor: '', vendor: '',
size: '585', size: '585',
tags: '', tags: '',
isSigned: true,
signatureInfo: [
{
Tool: 'cosign',
IsTrusted: false,
Author: ''
},
{
Tool: 'cosign',
IsTrusted: false,
Author: ''
},
{
Tool: 'cosign',
IsTrusted: false,
Author: ''
},
{
Tool: 'cosign',
IsTrusted: false,
Author: ''
}
],
platforms: [{ Os: 'linux', Arch: 'amd64' }] platforms: [{ Os: 'linux', Arch: 'amd64' }]
}; };
@ -34,6 +57,8 @@ const RepoCardWrapper = (props) => {
version={image.latestVersion} version={image.latestVersion}
description={image.description} description={image.description}
vendor={image.vendor} vendor={image.vendor}
isSigned={image.isSigned}
signatureInfo={image.signatureInfo}
key={1} key={1}
lastUpdated={image.lastUpdated} lastUpdated={image.lastUpdated}
platforms={image.platforms} platforms={image.platforms}

View File

@ -6,6 +6,8 @@ import VulnerabilitiesDetails from 'components/Tag/Tabs/VulnerabilitiesDetails';
import React from 'react'; import React from 'react';
import { MemoryRouter } from 'react-router-dom'; import { MemoryRouter } from 'react-router-dom';
jest.mock('xlsx');
const StateVulnerabilitiesWrapper = () => { const StateVulnerabilitiesWrapper = () => {
return ( return (
<MockThemeProvider> <MockThemeProvider>
@ -20,6 +22,14 @@ const mockCVEList = {
CVEListForImage: { CVEListForImage: {
Tag: '', Tag: '',
Page: { ItemCount: 20, TotalCount: 20 }, Page: { ItemCount: 20, TotalCount: 20 },
Summary: {
Count: 5,
UnknownCount: 1,
LowCount: 1,
MediumCount: 1,
HighCount: 1,
CriticalCount: 1,
},
CVEList: [ CVEList: [
{ {
Id: 'CVE-2020-16156', Id: 'CVE-2020-16156',
@ -497,6 +507,7 @@ describe('Vulnerabilties page', () => {
jest.spyOn(api, 'get').mockResolvedValue({ status: 200, data: { data: mockCVEList } }); jest.spyOn(api, 'get').mockResolvedValue({ status: 200, data: { data: mockCVEList } });
render(<StateVulnerabilitiesWrapper />); render(<StateVulnerabilitiesWrapper />);
await waitFor(() => expect(screen.getAllByText('Vulnerabilities')).toHaveLength(1)); 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)); await waitFor(() => expect(screen.getAllByText(/fixed in/i)).toHaveLength(20));
}); });
@ -513,7 +524,7 @@ describe('Vulnerabilties page', () => {
it('renders no vulnerabilities if there are not any', async () => { it('renders no vulnerabilities if there are not any', async () => {
jest.spyOn(api, 'get').mockResolvedValue({ jest.spyOn(api, 'get').mockResolvedValue({
status: 200, status: 200,
data: { data: { CVEListForImage: { Tag: '', Page: {}, CVEList: [] } } } data: { data: { CVEListForImage: { Tag: '', Page: {}, CVEList: [], Summary: {} } } }
}); });
render(<StateVulnerabilitiesWrapper />); render(<StateVulnerabilitiesWrapper />);
await waitFor(() => expect(screen.getAllByText('No Vulnerabilities')).toHaveLength(1)); await waitFor(() => expect(screen.getAllByText('No Vulnerabilities')).toHaveLength(1));
@ -558,6 +569,48 @@ describe('Vulnerabilties page', () => {
expect(await screen.findByText('latest')).toBeInTheDocument(); expect(await screen.findByText('latest')).toBeInTheDocument();
}); });
it('should allow export of vulnerabilities list', async () => {
const xlsxMock = jest.createMockFromModule('xlsx');
xlsxMock.writeFile = jest.fn();
jest
.spyOn(api, 'get')
.mockResolvedValueOnce({ status: 200, data: { data: mockCVEList } })
.mockResolvedValueOnce({ status: 200, data: { data: mockCVEList } });
render(<StateVulnerabilitiesWrapper />);
await waitFor(() => expect(screen.getAllByText('Vulnerabilities')).toHaveLength(1));
const downloadBtn = await screen.findAllByTestId('DownloadIcon');
fireEvent.click(downloadBtn[0]);
expect(await screen.findByTestId('export-csv-menuItem')).toBeInTheDocument();
expect(await screen.findByTestId('export-excel-menuItem')).toBeInTheDocument();
const exportAsCSVBtn = screen.getByText(/csv/i);
expect(exportAsCSVBtn).toBeInTheDocument();
global.URL.createObjectURL = jest.fn();
await fireEvent.click(exportAsCSVBtn);
expect(await screen.findByTestId('export-csv-menuItem')).not.toBeInTheDocument();
fireEvent.click(downloadBtn[0]);
const exportAsExcelBtn = screen.getByText(/xlsx/i);
expect(exportAsExcelBtn).toBeInTheDocument();
await fireEvent.click(exportAsExcelBtn);
expect(await screen.findByTestId('export-excel-menuItem')).not.toBeInTheDocument();
});
it('should expand/collapse the list of CVEs', async () => {
jest
.spyOn(api, 'get')
.mockResolvedValueOnce({ status: 200, data: { data: mockCVEList } })
.mockResolvedValueOnce({ status: 200, data: { data: mockCVEList } });
render(<StateVulnerabilitiesWrapper />);
await waitFor(() => expect(screen.getAllByText('Vulnerabilities')).toHaveLength(1));
await waitFor(() => expect(screen.getAllByText('Fixed in')).toHaveLength(20));
const collapseListBtn = await screen.findAllByTestId('ViewHeadlineIcon');
fireEvent.click(collapseListBtn[0]);
expect(await screen.findByText('Fixed in')).not.toBeVisible();
const expandListBtn = await screen.findAllByTestId('ViewAgendaIcon');
fireEvent.click(expandListBtn[0]);
await waitFor(() => expect(screen.getAllByText('Fixed in')).toHaveLength(20));
});
it('should handle fixed CVE query errors', async () => { it('should handle fixed CVE query errors', async () => {
jest jest
.spyOn(api, 'get') .spyOn(api, 'get')

View File

@ -1,11 +1,11 @@
import axios from 'axios'; import axios from 'axios';
import { isEmpty } from 'lodash'; import { isEmpty } from 'lodash';
import { sortByCriteria } from 'utilities/sortCriteria'; import { sortByCriteria } from 'utilities/sortCriteria';
import { logoutUser } from 'utilities/authUtilities'; import { isAuthenticationEnabled, logoutUser } from 'utilities/authUtilities';
import { host } from 'host'; import { host } from 'host';
axios.interceptors.request.use((config) => { axios.interceptors.request.use((config) => {
if (config.url.includes(endpoints.authConfig)) { if (config.url.includes(endpoints.authConfig) || !isAuthenticationEnabled()) {
config.withCredentials = false; config.withCredentials = false;
} else { } else {
config.headers['X-ZOT-API-CLIENT'] = 'zot-ui'; config.headers['X-ZOT-API-CLIENT'] = 'zot-ui';
@ -19,6 +19,7 @@ axios.interceptors.response.use(
}, },
(error) => { (error) => {
if (error?.response?.status === 401) { if (error?.response?.status === 401) {
if (window.location.pathname.includes('/login')) return Promise.reject(error);
logoutUser(); logoutUser();
window.location.replace('/login'); window.location.replace('/login');
return Promise.reject(error); return Promise.reject(error);
@ -78,16 +79,17 @@ const api = {
const endpoints = { const endpoints = {
status: `/v2/`, status: `/v2/`,
authConfig: `/v2/_zot/ext/mgmt`, authConfig: `/v2/_zot/ext/mgmt`,
openidAuth: `/auth/login`, openidAuth: `/zot/auth/login`,
logout: `/auth/logout`, logout: `/zot/auth/logout`,
deleteImage: (name, tag) => `/v2/${name}/manifests/${tag}`,
repoList: ({ pageNumber = 1, pageSize = 15 } = {}) => repoList: ({ pageNumber = 1, pageSize = 15 } = {}) =>
`/v2/_zot/ext/search?query={RepoListWithNewestImage(requestedPage: {limit:${pageSize} offset:${ `/v2/_zot/ext/search?query={RepoListWithNewestImage(requestedPage: {limit:${pageSize} offset:${
(pageNumber - 1) * pageSize (pageNumber - 1) * pageSize
}}){Results {Name LastUpdated Size Platforms {Os Arch} NewestImage { Tag Vulnerabilities {MaxSeverity Count} Description Licenses Title Source IsSigned Documentation Vendor Labels} IsStarred IsBookmarked DownloadCount}}}`, }}){Results {Name LastUpdated Size Platforms {Os Arch} NewestImage { Tag Vulnerabilities {MaxSeverity Count} Description Licenses Title Source IsSigned SignatureInfo { Tool IsTrusted Author } Documentation Vendor Labels} IsStarred IsBookmarked StarCount DownloadCount}}}`,
detailedRepoInfo: (name) => detailedRepoInfo: (name) =>
`/v2/_zot/ext/search?query={ExpandedRepoInfo(repo:"${name}"){Images {Manifests {Digest Platform {Os Arch} Size} Vulnerabilities {MaxSeverity Count} Tag LastUpdated Vendor } Summary {Name LastUpdated Size Platforms {Os Arch} Vendors IsStarred IsBookmarked NewestImage {RepoName IsSigned Vulnerabilities {MaxSeverity Count} Manifests {Digest} Tag Vendor Title Documentation DownloadCount Source Description Licenses}}}}`, `/v2/_zot/ext/search?query={ExpandedRepoInfo(repo:"${name}"){Images {Manifests {Digest Platform {Os Arch} Size} Vulnerabilities {MaxSeverity Count} Tag LastUpdated Vendor IsDeletable } Summary {Name LastUpdated Size Platforms {Os Arch} Vendors IsStarred IsBookmarked NewestImage {RepoName IsSigned SignatureInfo { Tool IsTrusted Author } Vulnerabilities {MaxSeverity Count} Manifests {Digest} Tag Vendor Title Documentation DownloadCount Source Description Licenses}}}}`,
detailedImageInfo: (name, tag) => detailedImageInfo: (name, tag) =>
`/v2/_zot/ext/search?query={Image(image: "${name}:${tag}"){RepoName IsSigned Vulnerabilities {MaxSeverity Count} Referrers {MediaType ArtifactType Size Digest Annotations{Key Value}} Tag Manifests {History {Layer {Size Digest} HistoryDescription {CreatedBy EmptyLayer}} Digest ConfigDigest LastUpdated Size Platform {Os Arch}} Vendor Licenses }}`, `/v2/_zot/ext/search?query={Image(image: "${name}:${tag}"){RepoName IsSigned SignatureInfo { Tool IsTrusted Author } Vulnerabilities {MaxSeverity Count} Referrers {MediaType ArtifactType Size Digest Annotations{Key Value}} Tag Manifests {History {Layer {Size Digest} HistoryDescription {CreatedBy EmptyLayer}} Digest ConfigDigest LastUpdated Size Platform {Os Arch}} Vendor Licenses }}`,
vulnerabilitiesForRepo: (name, { pageNumber = 1, pageSize = 15 }, searchTerm = '') => { vulnerabilitiesForRepo: (name, { pageNumber = 1, pageSize = 15 }, searchTerm = '') => {
let query = `/v2/_zot/ext/search?query={CVEListForImage(image: "${name}", requestedPage: {limit:${pageSize} offset:${ let query = `/v2/_zot/ext/search?query={CVEListForImage(image: "${name}", requestedPage: {limit:${pageSize} offset:${
(pageNumber - 1) * pageSize (pageNumber - 1) * pageSize
@ -95,20 +97,30 @@ const endpoints = {
if (!isEmpty(searchTerm)) { if (!isEmpty(searchTerm)) {
query += `, searchedCVE: "${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}}}`;
}, },
imageListWithCVEFixed: (cveId, repoName, { pageNumber = 1, pageSize = 3 }) => allVulnerabilitiesForRepo: (name) =>
`/v2/_zot/ext/search?query={ImageListWithCVEFixed(id:"${cveId}", image:"${repoName}", requestedPage: {limit:${pageSize} offset:${ `/v2/_zot/ext/search?query={CVEListForImage(image: "${name}"){Tag Page {TotalCount ItemCount} CVEList {Id Title Description Severity Reference PackageList {Name InstalledVersion FixedVersion}}}}`,
imageListWithCVEFixed: (cveId, repoName, { pageNumber = 1, pageSize = 3 }, filter = {}) => {
let filterParam = '';
if (filter.Os || filter.Arch) {
filterParam = `,filter:{`;
if (filter.Os) filterParam += ` Os:${!isEmpty(filter.Os) ? `${JSON.stringify(filter.Os)}` : '""'}`;
if (filter.Arch) filterParam += ` Arch:${!isEmpty(filter.Arch) ? `${JSON.stringify(filter.Arch)}` : '""'}`;
filterParam += '}';
}
return `/v2/_zot/ext/search?query={ImageListWithCVEFixed(id:"${cveId}", image:"${repoName}", requestedPage: {limit:${pageSize} offset:${
(pageNumber - 1) * pageSize (pageNumber - 1) * pageSize
}}) {Page {TotalCount ItemCount} Results {Tag}}}`, }}${filterParam}) {Page {TotalCount ItemCount} Results {Tag}}}`;
},
dependsOnForImage: (name, { pageNumber = 1, pageSize = 15 } = {}) => dependsOnForImage: (name, { pageNumber = 1, pageSize = 15 } = {}) =>
`/v2/_zot/ext/search?query={BaseImageList(image: "${name}", requestedPage: {limit:${pageSize} offset:${ `/v2/_zot/ext/search?query={BaseImageList(image: "${name}", requestedPage: {limit:${pageSize} offset:${
(pageNumber - 1) * pageSize (pageNumber - 1) * pageSize
}}){Page {TotalCount ItemCount} Results { RepoName Tag Description Manifests {Digest Platform {Os Arch} Size} Vendor DownloadCount LastUpdated IsSigned Vulnerabilities {MaxSeverity Count}}}}`, }}){Page {TotalCount ItemCount} Results { RepoName Tag Description Manifests {Digest Platform {Os Arch} Size} Vendor DownloadCount LastUpdated IsSigned SignatureInfo { Tool IsTrusted Author } Vulnerabilities {MaxSeverity Count}}}}`,
isDependentOnForImage: (name, { pageNumber = 1, pageSize = 15 } = {}) => isDependentOnForImage: (name, { pageNumber = 1, pageSize = 15 } = {}) =>
`/v2/_zot/ext/search?query={DerivedImageList(image: "${name}", requestedPage: {limit:${pageSize} offset:${ `/v2/_zot/ext/search?query={DerivedImageList(image: "${name}", requestedPage: {limit:${pageSize} offset:${
(pageNumber - 1) * pageSize (pageNumber - 1) * pageSize
}}){Page {TotalCount ItemCount} Results {RepoName Tag Description Manifests {Digest Platform {Os Arch} Size} Vendor DownloadCount LastUpdated IsSigned Vulnerabilities {MaxSeverity Count}}}}`, }}){Page {TotalCount ItemCount} Results {RepoName Tag Description Manifests {Digest Platform {Os Arch} Size} Vendor DownloadCount LastUpdated IsSigned SignatureInfo { Tool IsTrusted Author } Vulnerabilities {MaxSeverity Count}}}}`,
globalSearch: ({ globalSearch: ({
searchQuery = '""', searchQuery = '""',
pageNumber = 1, pageNumber = 1,
@ -125,9 +137,10 @@ const endpoints = {
if (filter.Arch) filterParam += ` Arch:${!isEmpty(filter.Arch) ? `${JSON.stringify(filter.Arch)}` : '""'}`; if (filter.Arch) filterParam += ` Arch:${!isEmpty(filter.Arch) ? `${JSON.stringify(filter.Arch)}` : '""'}`;
if (filter.HasToBeSigned) filterParam += ` HasToBeSigned: ${filter.HasToBeSigned}`; if (filter.HasToBeSigned) filterParam += ` HasToBeSigned: ${filter.HasToBeSigned}`;
if (filter.IsBookmarked) filterParam += ` IsBookmarked: ${filter.IsBookmarked}`; if (filter.IsBookmarked) filterParam += ` IsBookmarked: ${filter.IsBookmarked}`;
if (filter.IsStarred) filterParam += ` IsStarred: ${filter.IsStarred}`;
filterParam += '}'; filterParam += '}';
if (Object.keys(filter).length === 0) filterParam = ''; if (Object.keys(filter).length === 0) filterParam = '';
return `/v2/_zot/ext/search?query={GlobalSearch(${searchParam}, ${paginationParam} ${filterParam}) {Page {TotalCount ItemCount} Repos {Name LastUpdated Size Platforms { Os Arch } IsStarred IsBookmarked NewestImage { Tag Vulnerabilities {MaxSeverity Count} Description IsSigned Licenses Vendor Labels } DownloadCount}}}`; return `/v2/_zot/ext/search?query={GlobalSearch(${searchParam}, ${paginationParam} ${filterParam}) {Page {TotalCount ItemCount} Repos {Name LastUpdated Size Platforms { Os Arch } IsStarred IsBookmarked NewestImage { Tag Vulnerabilities {MaxSeverity Count} Description IsSigned SignatureInfo { Tool IsTrusted Author } Licenses Vendor Labels } StarCount DownloadCount}}}`;
}, },
imageSuggestions: ({ searchQuery = '""', pageNumber = 1, pageSize = 15 }) => { imageSuggestions: ({ searchQuery = '""', pageNumber = 1, pageSize = 15 }) => {
const searchParam = searchQuery !== '' ? `query:"${searchQuery}"` : `query:""`; const searchParam = searchQuery !== '' ? `query:"${searchQuery}"` : `query:""`;
@ -136,7 +149,8 @@ const endpoints = {
}, },
referrers: ({ repo, digest, type = '' }) => referrers: ({ repo, digest, type = '' }) =>
`/v2/_zot/ext/search?query={Referrers(repo: "${repo}" digest: "${digest}" type: "${type}"){MediaType ArtifactType Size Digest Annotations{Key Value}}}`, `/v2/_zot/ext/search?query={Referrers(repo: "${repo}" digest: "${digest}" type: "${type}"){MediaType ArtifactType Size Digest Annotations{Key Value}}}`,
bookmarkToggle: (repo) => `/v2/_zot/ext/userprefs?repo=${repo}&action=toggleBookmark` bookmarkToggle: (repo) => `/v2/_zot/ext/userprefs?repo=${repo}&action=toggleBookmark`,
starToggle: (repo) => `/v2/_zot/ext/userprefs?repo=${repo}&action=toggleStar`
}; };
export { api, endpoints }; export { api, endpoints };

15
src/assets/noData.svg Normal file
View File

@ -0,0 +1,15 @@
<svg width="235" height="240" viewBox="0 0 235 240" fill="none" xmlns="http://www.w3.org/2000/svg">
<rect x="34.8711" y="16.2017" width="36" height="249.944" rx="18" transform="rotate(-26.7465 34.8711 16.2017)" fill="#F15527" fill-opacity="0.2"/>
<rect x="156.871" y="36.2017" width="36" height="100.235" rx="18" transform="rotate(-26.7465 156.871 36.2017)" fill="#F15527" fill-opacity="0.2"/>
<rect x="0.871094" y="138.037" width="26.745" height="74.4663" rx="13.3725" transform="rotate(-26.7465 0.871094 138.037)" fill="#F15527" fill-opacity="0.2"/>
<g clip-path="url(#clip0_2865_33046)">
<path d="M117.5 199C161.225 199 197 163.225 197 119.5C197 75.775 161.225 40 117.5 40C73.775 40 38 75.775 38 119.5C38 163.225 73.775 199 117.5 199ZM119.885 82.0555C121.475 80.6245 123.383 79.75 125.45 79.75C127.596 79.75 129.425 80.6245 131.094 82.0555C132.605 83.725 133.4 85.633 133.4 87.7C133.4 89.8465 132.605 91.675 131.094 93.3445C129.425 94.855 127.596 95.65 125.45 95.65C123.383 95.65 121.475 94.855 119.885 93.3445C118.375 91.675 117.5 89.8465 117.5 87.7C117.5 85.633 118.375 83.725 119.885 82.0555ZM100.01 119.261C100.01 119.261 117.261 105.588 123.542 105.031C129.425 104.554 128.232 111.311 127.676 114.809L127.597 115.286C126.484 119.5 125.132 124.588 123.78 129.438C120.759 140.488 117.818 151.3 118.534 153.287C119.328 155.991 124.257 152.572 127.835 150.187C128.312 149.869 128.709 149.551 129.107 149.312C129.107 149.312 129.743 148.676 130.379 149.551C130.538 149.789 130.697 150.028 130.856 150.187C131.572 151.3 131.969 151.697 131.015 152.333L130.697 152.492C128.948 153.685 121.475 158.932 118.454 160.84C115.194 162.987 102.713 170.142 104.621 156.229C106.291 146.451 108.517 138.023 110.266 131.425C113.525 119.5 114.956 114.094 107.642 118.785C104.701 120.534 102.951 121.646 101.918 122.362C101.043 122.998 100.964 122.998 100.408 121.965L100.169 121.487L99.7715 120.852C99.215 120.057 99.215 119.977 100.01 119.261Z" fill="#0F2139"/>
<path d="M119.885 82.0555C121.475 80.6245 123.383 79.75 125.45 79.75C127.596 79.75 129.425 80.6245 131.094 82.0555C132.605 83.725 133.4 85.633 133.4 87.7C133.4 89.8465 132.605 91.675 131.094 93.3445C129.425 94.855 127.596 95.65 125.45 95.65C123.383 95.65 121.475 94.855 119.885 93.3445C118.375 91.675 117.5 89.8465 117.5 87.7C117.5 85.633 118.375 83.725 119.885 82.0555Z" fill="white"/>
<path d="M100.01 119.261C100.01 119.261 117.261 105.588 123.542 105.031C129.425 104.554 128.232 111.311 127.676 114.809L127.597 115.286C126.484 119.5 125.132 124.588 123.78 129.438C120.759 140.488 117.818 151.3 118.534 153.287C119.328 155.991 124.257 152.572 127.835 150.187C128.312 149.869 128.709 149.551 129.107 149.312C129.107 149.312 129.743 148.676 130.379 149.551C130.538 149.789 130.697 150.028 130.856 150.187C131.572 151.3 131.969 151.697 131.015 152.333L130.697 152.492C128.948 153.685 121.475 158.932 118.454 160.84C115.194 162.987 102.713 170.142 104.621 156.229C106.291 146.451 108.517 138.023 110.266 131.425C113.525 119.5 114.956 114.094 107.642 118.785C104.701 120.534 102.951 121.646 101.918 122.362C101.043 122.998 100.964 122.998 100.408 121.965L100.169 121.487L99.7715 120.852C99.215 120.057 99.215 119.977 100.01 119.261Z" fill="white"/>
</g>
<defs>
<clipPath id="clip0_2865_33046">
<rect width="159" height="159" fill="white" transform="translate(38 40)"/>
</clipPath>
</defs>
</svg>

After

Width:  |  Height:  |  Size: 3.3 KiB

File diff suppressed because one or more lines are too long

Before

Width:  |  Height:  |  Size: 32 KiB

After

Width:  |  Height:  |  Size: 1.7 KiB

View File

@ -220,8 +220,11 @@ function Explore({ searchInputValue }) {
version={item.latestVersion} version={item.latestVersion}
description={item.description} description={item.description}
downloads={item.downloads} downloads={item.downloads}
stars={item.stars}
isSigned={item.isSigned} isSigned={item.isSigned}
signatureInfo={item.signatureInfo}
isBookmarked={item.isBookmarked} isBookmarked={item.isBookmarked}
isStarred={item.isStarred}
vendor={item.vendor} vendor={item.vendor}
platforms={item.platforms} platforms={item.platforms}
key={index} key={index}

View File

@ -149,14 +149,14 @@ function Header({ setSearchCurrentValue = () => {} }) {
</Link> </Link>
</Grid> </Grid>
<Grid item className={classes.headerLinkContainer}> <Grid item className={classes.headerLinkContainer}>
<a className={classes.link} href="https://zotregistry.io" target="_blank" rel="noreferrer"> <a className={classes.link} href="https://zotregistry.dev" target="_blank" rel="noreferrer">
Product Product
</a> </a>
</Grid> </Grid>
<Grid item className={classes.headerLinkContainer}> <Grid item className={classes.headerLinkContainer}>
<a <a
className={classes.link} className={classes.link}
href="https://zotregistry.io/v1.4.3/general/concepts/" href="https://zotregistry.dev/v2.0.0/general/concepts/"
target="_blank" target="_blank"
rel="noreferrer" rel="noreferrer"
> >

View File

@ -295,12 +295,26 @@ function SearchSuggestion({ setSearchCurrentValue = () => {} }) {
<List <List
{...getMenuProps()} {...getMenuProps()}
className={ className={
isOpen && !isLoading && !isFailedSearch isOpen && !isFailedSearch
? `${classes.resultsWrapper} ${isComponentFocused && classes.resultsWrapperFocused}` ? `${classes.resultsWrapper} ${isComponentFocused && classes.resultsWrapperFocused}`
: classes.resultsWrapperHidden : classes.resultsWrapperHidden
} }
> >
{isOpen && suggestionData?.length > 0 && renderSuggestions()} {isOpen && suggestionData?.length > 0 && renderSuggestions()}
{isOpen && isLoading && !isEmpty(searchQuery) && isEmpty(suggestionData) && (
<>
<ListItem
className={classes.searchItem}
style={{ color: '#52637A', fontSize: '1rem', textOverflow: 'ellipsis' }}
{...getItemProps({ item: '', index: 0 })}
spacing={2}
>
<Stack direction="row" spacing={2}>
<Typography>Loading...</Typography>
</Stack>
</ListItem>
</>
)}
{isOpen && isEmpty(searchQuery) && isEmpty(suggestionData) && ( {isOpen && isEmpty(searchQuery) && isEmpty(suggestionData) && (
<> <>
<ListItem <ListItem

View File

@ -8,8 +8,14 @@ import { mapToRepo } from 'utilities/objectModels';
import Loading from '../Shared/Loading'; import Loading from '../Shared/Loading';
import { useNavigate, createSearchParams } from 'react-router-dom'; import { useNavigate, createSearchParams } from 'react-router-dom';
import { sortByCriteria } from 'utilities/sortCriteria'; import { sortByCriteria } from 'utilities/sortCriteria';
import { HOME_POPULAR_PAGE_SIZE, HOME_RECENT_PAGE_SIZE, HOME_BOOKMARKS_PAGE_SIZE } from 'utilities/paginationConstants'; import {
HOME_POPULAR_PAGE_SIZE,
HOME_RECENT_PAGE_SIZE,
HOME_BOOKMARKS_PAGE_SIZE,
HOME_STARS_PAGE_SIZE
} from 'utilities/paginationConstants';
import { isEmpty } from 'lodash'; import { isEmpty } from 'lodash';
import NoDataComponent from 'components/Shared/NoDataComponent';
const useStyles = makeStyles((theme) => ({ const useStyles = makeStyles((theme) => ({
gridWrapper: { gridWrapper: {
@ -88,6 +94,8 @@ function Home() {
const [isLoadingRecent, setIsLoadingRecent] = useState(true); const [isLoadingRecent, setIsLoadingRecent] = useState(true);
const [bookmarkData, setBookmarkData] = useState([]); const [bookmarkData, setBookmarkData] = useState([]);
const [isLoadingBookmarks, setIsLoadingBookmarks] = useState(true); const [isLoadingBookmarks, setIsLoadingBookmarks] = useState(true);
const [starData, setStarData] = useState([]);
const [isLoadingStars, setIsLoadingStars] = useState(true);
const navigate = useNavigate(); const navigate = useNavigate();
const abortController = useMemo(() => new AbortController(), []); const abortController = useMemo(() => new AbortController(), []);
@ -184,12 +192,44 @@ function Home() {
}); });
}; };
const getStars = () => {
setIsLoadingStars(true);
api
.get(
`${host()}${endpoints.globalSearch({
searchQuery: '',
pageNumber: 1,
pageSize: HOME_STARS_PAGE_SIZE,
sortBy: sortByCriteria.relevance?.value,
filter: { IsStarred: true }
})}`,
abortController.signal
)
.then((response) => {
if (response.data && response.data.data) {
let repoList = response.data.data.GlobalSearch.Repos;
let repoData = repoList.map((responseRepo) => {
return mapToRepo(responseRepo);
});
setStarData(repoData);
setIsLoading(false);
setIsLoadingStars(false);
}
})
.catch((e) => {
setIsLoading(false);
setIsLoadingStars(false);
console.error(e);
});
};
useEffect(() => { useEffect(() => {
window.scrollTo(0, 0); window.scrollTo(0, 0);
setIsLoading(true); setIsLoading(true);
getPopularData(); getPopularData();
getRecentData(); getRecentData();
getBookmarks(); getBookmarks();
getStars();
return () => { return () => {
abortController.abort(); abortController.abort();
}; };
@ -199,6 +239,17 @@ function Home() {
navigate({ pathname: `/explore`, search: createSearchParams({ [type]: value }).toString() }); navigate({ pathname: `/explore`, search: createSearchParams({ [type]: value }).toString() });
}; };
const isNoData = () =>
!isLoading &&
!isLoadingBookmarks &&
!isLoadingStars &&
!isLoadingPopular &&
!isLoadingRecent &&
bookmarkData.length === 0 &&
starData.length === 0 &&
popularData.length === 0 &&
recentData.length === 0;
const renderCards = (cardArray) => { const renderCards = (cardArray) => {
return ( return (
cardArray && cardArray &&
@ -209,8 +260,11 @@ function Home() {
version={item.latestVersion} version={item.latestVersion}
description={item.description} description={item.description}
downloads={item.downloads} downloads={item.downloads}
stars={item.stars}
isSigned={item.isSigned} isSigned={item.isSigned}
signatureInfo={item.signatureInfo}
isBookmarked={item.isBookmarked} isBookmarked={item.isBookmarked}
isStarred={item.isStarred}
vendor={item.vendor} vendor={item.vendor}
platforms={item.platforms} platforms={item.platforms}
key={index} key={index}
@ -226,68 +280,89 @@ function Home() {
); );
}; };
return ( const renderContent = () => {
<> return isNoData() === true ? (
{isLoading ? ( <NoDataComponent text="No images" />
<Loading /> ) : (
) : ( <Stack alignItems="center" className={classes.gridWrapper}>
<Stack alignItems="center" className={classes.gridWrapper}> <Stack className={classes.sectionHeaderContainer} sx={{ paddingTop: '3rem' }}>
<Stack className={classes.sectionHeaderContainer} sx={{ paddingTop: '3rem' }}> <div>
<div> <Typography variant="h4" align="left" className={classes.sectionTitle}>
<Typography variant="h4" align="left" className={classes.sectionTitle}> Most popular images
Most popular images </Typography>
</Typography> </div>
</div> <div onClick={() => handleClickViewAll('sortby', sortByCriteria.downloads.value)}>
<div onClick={() => handleClickViewAll('sortby', sortByCriteria.downloads.value)}> <Typography variant="body2" className={classes.viewAll}>
<Typography variant="body2" className={classes.viewAll}> View all
View all </Typography>
</Typography> </div>
</div>
</Stack>
{isLoadingPopular ? <Loading /> : renderCards(popularData)}
{/* currently most popular will be by downloads until stars are implemented */}
<Stack className={classes.sectionHeaderContainer}>
<div>
<Typography variant="h4" align="left" className={classes.sectionTitle}>
Recently updated images
</Typography>
</div>
<div>
<Typography
variant="body2"
className={classes.viewAll}
onClick={() => handleClickViewAll('sortby', sortByCriteria.updateTime.value)}
>
View all
</Typography>
</div>
</Stack>
{isLoadingRecent ? <Loading /> : renderCards(recentData)}
{!isEmpty(bookmarkData) && (
<>
<Stack className={classes.sectionHeaderContainer}>
<div>
<Typography variant="h4" align="left" className={classes.sectionTitle}>
Bookmarks
</Typography>
</div>
<div>
<Typography
variant="body2"
className={classes.viewAll}
onClick={() => handleClickViewAll('filter', 'IsBookmarked')}
>
View all
</Typography>
</div>
</Stack>
{isLoadingBookmarks ? <Loading /> : renderCards(bookmarkData)}
</>
)}
</Stack> </Stack>
)} {isLoadingPopular ? <Loading /> : renderCards(popularData, isLoadingPopular)}
</> {/* currently most popular will be by downloads until stars are implemented */}
); <Stack className={classes.sectionHeaderContainer}>
<div>
<Typography variant="h4" align="left" className={classes.sectionTitle}>
Recently updated images
</Typography>
</div>
<div>
<Typography
variant="body2"
className={classes.viewAll}
onClick={() => handleClickViewAll('sortby', sortByCriteria.updateTime.value)}
>
View all
</Typography>
</div>
</Stack>
{isLoadingRecent ? <Loading /> : renderCards(recentData, isLoadingRecent)}
{!isEmpty(bookmarkData) && (
<>
<Stack className={classes.sectionHeaderContainer}>
<div>
<Typography variant="h4" align="left" className={classes.sectionTitle}>
Bookmarks
</Typography>
</div>
<div>
<Typography
variant="body2"
className={classes.viewAll}
onClick={() => handleClickViewAll('filter', 'IsBookmarked')}
>
View all
</Typography>
</div>
</Stack>
{isLoadingBookmarks ? <Loading /> : renderCards(bookmarkData, isLoadingBookmarks)}
</>
)}
{!isEmpty(starData) && (
<>
<Stack className={classes.sectionHeaderContainer}>
<div>
<Typography variant="h4" align="left" className={classes.sectionTitle}>
Stars
</Typography>
</div>
<div>
<Typography
variant="body2"
className={classes.viewAll}
onClick={() => handleClickViewAll('filter', 'IsStarred')}
>
View all
</Typography>
</div>
</Stack>
{isLoadingStars ? <Loading /> : renderCards(starData, isLoadingStars)}
</>
)}
</Stack>
);
};
return <>{isLoading ? <Loading /> : renderContent()}</>;
} }
export default Home; export default Home;

View File

@ -20,7 +20,7 @@ import Alert from '@mui/material/Alert';
import CircularProgress from '@mui/material/CircularProgress'; import CircularProgress from '@mui/material/CircularProgress';
import Loading from '../Shared/Loading'; import Loading from '../Shared/Loading';
import { GoogleLoginButton, GithubLoginButton, DexLoginButton } from './ThirdPartyLoginComponents'; import { GoogleLoginButton, GithubLoginButton, OIDCLoginButton } from './ThirdPartyLoginComponents';
// styling // styling
import { makeStyles } from '@mui/styles'; import { makeStyles } from '@mui/styles';
@ -284,14 +284,15 @@ export default function SignIn({ isLoggedIn, setIsLoggedIn, wrapperSetLoading =
let isGoogle = isObject(authMethods.openid?.providers?.google); let isGoogle = isObject(authMethods.openid?.providers?.google);
// let isGitlab = isObject(authMethods.openid?.providers?.gitlab); // let isGitlab = isObject(authMethods.openid?.providers?.gitlab);
let isGithub = isObject(authMethods.openid?.providers?.github); let isGithub = isObject(authMethods.openid?.providers?.github);
let isDex = isObject(authMethods.openid?.providers?.dex); let isOIDC = isObject(authMethods.openid?.providers?.oidc);
let oidcName = authMethods.openid?.providers?.oidc?.name;
return ( return (
<Stack direction="column" spacing="1rem" className={classes.thirdPartyLoginContainer}> <Stack direction="column" spacing="1rem" className={classes.thirdPartyLoginContainer}>
{isGithub && <GithubLoginButton handleClick={handleClickExternalLogin} />} {isGithub && <GithubLoginButton handleClick={handleClickExternalLogin} />}
{isGoogle && <GoogleLoginButton handleClick={handleClickExternalLogin} />} {isGoogle && <GoogleLoginButton handleClick={handleClickExternalLogin} />}
{/* {isGitlab && <GitlabLoginButton handleClick={handleClickExternalLogin} />} */} {/* {isGitlab && <GitlabLoginButton handleClick={handleClickExternalLogin} />} */}
{isDex && <DexLoginButton handleClick={handleClickExternalLogin} />} {isOIDC && <OIDCLoginButton handleClick={handleClickExternalLogin} oidcName={oidcName} />}
</Stack> </Stack>
); );
}; };
@ -308,7 +309,7 @@ export default function SignIn({ isLoggedIn, setIsLoggedIn, wrapperSetLoading =
Sign In Sign In
</Typography> </Typography>
<Typography align="left" className={classes.subtext} variant="body1" gutterBottom> <Typography align="left" className={classes.subtext} variant="body1" gutterBottom>
Welcome back! Please enter your details. Welcome back! Please login.
</Typography> </Typography>
{renderThirdPartyLoginMethods()} {renderThirdPartyLoginMethods()}
{Object.keys(authMethods).length > 1 && <Divider className={classes.divider}>or</Divider>} {Object.keys(authMethods).length > 1 && <Divider className={classes.divider}>or</Divider>}

View File

@ -1,37 +0,0 @@
import React from 'react';
import { makeStyles } from '@mui/styles';
import { Stack, Typography } from '@mui/material';
const useStyles = makeStyles(() => ({
subtext: {
color: '#52637A',
fontSize: '0.8125rem',
fontWeight: '400',
lineHeight: '154%',
letterSpacing: '0.025rem',
marginBottom: '0'
},
text: {
color: '#0F2139',
fontSize: '0.8125rem',
lineHeight: '154%',
fontWeight: '600',
letterSpacing: '0.025rem'
}
}));
export default function TermsOfService(props) {
const classes = useStyles();
return (
<Stack spacing={0}>
<Typography variant="caption" className={classes.subtext} align="justify" {...props} pb={6}>
By using zot UI, you agree to the Terms of Service. For more information about our privacy practices, see
zot&apos;s Privacy Policy.
</Typography>
<Typography variant="caption" className={classes.text} align="center" {...props}>
Privacy Policy | Terms of Service
</Typography>
</Stack>
);
}

View File

@ -80,14 +80,15 @@ function GitlabLoginButton({ handleClick }) {
); );
} }
function DexLoginButton({ handleClick }) { function OIDCLoginButton({ handleClick, oidcName }) {
const classes = useStyles(); const classes = useStyles();
const loginWithName = oidcName || 'OIDC';
return ( return (
<Button fullWidth variant="contained" className={classes.button} onClick={(e) => handleClick(e, 'dex')}> <Button fullWidth variant="contained" className={classes.button} onClick={(e) => handleClick(e, 'oidc')}>
Sign in with Dex Sign in with {loginWithName}
</Button> </Button>
); );
} }
export { GithubLoginButton, GoogleLoginButton, GitlabLoginButton, DexLoginButton }; export { GithubLoginButton, GoogleLoginButton, GitlabLoginButton, OIDCLoginButton };

View File

@ -14,6 +14,8 @@ import { useParams, useNavigate, createSearchParams } from 'react-router-dom';
import { Card, CardContent, CardMedia, Chip, Grid, Stack, Tooltip, Typography, IconButton } from '@mui/material'; import { Card, CardContent, CardMedia, Chip, Grid, Stack, Tooltip, Typography, IconButton } from '@mui/material';
import BookmarkIcon from '@mui/icons-material/Bookmark'; import BookmarkIcon from '@mui/icons-material/Bookmark';
import BookmarkBorderIcon from '@mui/icons-material/BookmarkBorder'; import BookmarkBorderIcon from '@mui/icons-material/BookmarkBorder';
import StarIcon from '@mui/icons-material/Star';
import StarBorderIcon from '@mui/icons-material/StarBorder';
import makeStyles from '@mui/styles/makeStyles'; import makeStyles from '@mui/styles/makeStyles';
// placeholder images // placeholder images
@ -195,6 +197,10 @@ function RepoDetails() {
}; };
}, [name]); }, [name]);
const handleDeleteTag = (removed) => {
setTags((prevState) => prevState.filter((tag) => tag.tag !== removed));
};
const handlePlatformChipClick = (event) => { const handlePlatformChipClick = (event) => {
const { textContent } = event.target; const { textContent } = event.target;
event.stopPropagation(); event.stopPropagation();
@ -221,7 +227,7 @@ function RepoDetails() {
const handleBookmarkClick = () => { const handleBookmarkClick = () => {
api.put(`${host()}${endpoints.bookmarkToggle(name)}`, abortController.signal).then((response) => { api.put(`${host()}${endpoints.bookmarkToggle(name)}`, abortController.signal).then((response) => {
if (response.status === 200) { if (response && response.status === 200) {
setRepoDetailData((prevState) => ({ setRepoDetailData((prevState) => ({
...prevState, ...prevState,
isBookmarked: !prevState.isBookmarked isBookmarked: !prevState.isBookmarked
@ -230,6 +236,17 @@ function RepoDetails() {
}); });
}; };
const handleStarClick = () => {
api.put(`${host()}${endpoints.starToggle(name)}`, abortController.signal).then((response) => {
if (response.status === 200) {
setRepoDetailData((prevState) => ({
...prevState,
isStarred: !prevState.isStarred
}));
}
});
};
const getVendor = () => { const getVendor = () => {
return `${repoDetailData.newestTag?.Vendor || 'Vendor not available'}`; return `${repoDetailData.newestTag?.Vendor || 'Vendor not available'}`;
}; };
@ -271,17 +288,31 @@ function RepoDetails() {
</Stack> </Stack>
<Stack alignItems="center" sx={{ width: { xs: '100%', md: 'auto' } }} direction="row" spacing={2}> <Stack alignItems="center" sx={{ width: { xs: '100%', md: 'auto' } }} direction="row" spacing={2}>
<VulnerabilityIconCheck vulnerabilitySeverity={repoDetailData?.vulnerabilitySeverity} /> <VulnerabilityIconCheck vulnerabilitySeverity={repoDetailData?.vulnerabilitySeverity} />
<SignatureIconCheck isSigned={repoDetailData.isSigned} /> <SignatureIconCheck
isSigned={repoDetailData.isSigned}
signatureInfo={repoDetailData.signatureInfo}
/>
</Stack>
<Stack alignItems="center" sx={{ width: { xs: '100%', md: 'auto' } }} direction="row" spacing={1}>
{isAuthenticated() && (
<IconButton component="span" onClick={handleStarClick} data-testid="star-button">
{repoDetailData?.isStarred ? (
<StarIcon data-testid="starred" />
) : (
<StarBorderIcon data-testid="not-starred" />
)}
</IconButton>
)}
{isAuthenticated() && (
<IconButton component="span" onClick={handleBookmarkClick} data-testid="bookmark-button">
{repoDetailData?.isBookmarked ? (
<BookmarkIcon data-testid="bookmarked" />
) : (
<BookmarkBorderIcon data-testid="not-bookmarked" />
)}
</IconButton>
)}
</Stack> </Stack>
{isAuthenticated() && (
<IconButton component="span" onClick={handleBookmarkClick} data-testid="bookmark-button">
{repoDetailData?.isBookmarked ? (
<BookmarkIcon data-testid="bookmarked" />
) : (
<BookmarkBorderIcon data-testid="not-bookmarked" />
)}
</IconButton>
)}
</Stack> </Stack>
<Typography gutterBottom className={classes.repoTitle}> <Typography gutterBottom className={classes.repoTitle}>
{repoDetailData?.title || 'Title not available'} {repoDetailData?.title || 'Title not available'}
@ -314,7 +345,7 @@ function RepoDetails() {
<Grid item xs={12} md={8} className={classes.tags}> <Grid item xs={12} md={8} className={classes.tags}>
<Card className={classes.cardRoot}> <Card className={classes.cardRoot}>
<CardContent className={classes.tagsContent}> <CardContent className={classes.tagsContent}>
<Tags tags={tags} /> <Tags tags={tags} repoName={name} onTagDelete={handleDeleteTag} />
</CardContent> </CardContent>
</Card> </Card>
</Grid> </Grid>

View File

@ -43,7 +43,7 @@ const useStyles = makeStyles(() => ({
export default function Tags(props) { export default function Tags(props) {
const classes = useStyles(); const classes = useStyles();
const { tags } = props; const { tags, repoName, onTagDelete } = props;
const [tagsFilter, setTagsFilter] = useState(''); const [tagsFilter, setTagsFilter] = useState('');
const [sortFilter, setSortFilter] = useState(tagsSortByCriteria.updateTimeDesc.value); const [sortFilter, setSortFilter] = useState(tagsSortByCriteria.updateTimeDesc.value);
@ -63,6 +63,9 @@ export default function Tags(props) {
lastUpdated={tag.lastUpdated} lastUpdated={tag.lastUpdated}
vendor={tag.vendor} vendor={tag.vendor}
manifests={tag.manifests} manifests={tag.manifests}
repo={repoName}
onTagDelete={onTagDelete}
isDeletable={tag.isDeletable}
/> />
); );
}) })

View File

@ -0,0 +1,54 @@
import React, { useState } from 'react';
import { IconButton } from '@mui/material';
import DeleteIcon from '@mui/icons-material/Delete';
// utility
import { api, endpoints } from '../../api';
// components
import DeleteTagConfirmDialog from 'components/Shared/DeleteTagConfirmDialog';
import { host } from '../../host';
export default function DeleteTag(props) {
const { repo, tag, onTagDelete } = props;
const [open, setOpen] = useState(false);
const handleClickOpen = () => {
setOpen(true);
};
const handleClose = () => {
setOpen(false);
};
const deleteTag = (repo, tag) => {
api
.delete(`${host()}${endpoints.deleteImage(repo, tag)}`)
.then((response) => {
if (response && response.status == 202) {
onTagDelete(tag);
}
})
.catch((err) => {
console.error(err);
});
};
const onConfirm = () => {
deleteTag(repo, tag);
};
return (
<React.Fragment>
<IconButton onClick={handleClickOpen}>
<DeleteIcon />
</IconButton>
<DeleteTagConfirmDialog
onClose={handleClose}
open={open}
title={`Permanently delete image ${repo}:${tag}?`}
onConfirm={onConfirm}
/>
</React.Fragment>
);
}

View File

@ -0,0 +1,30 @@
import React from 'react';
// components
import { Button, Dialog, DialogTitle, DialogActions } from '@mui/material';
export default function DeleteTagConfirmDialog(props) {
const { onClose, open, title, onConfirm } = props;
return (
<Dialog data-testid="delete-dialog" onClose={onClose} open={open} color="primary">
<DialogTitle> {title} </DialogTitle>
<DialogActions style={{ justifyContent: 'center' }}>
<Button data-testid="cancel-delete" variant="contained" onClick={onClose} color="primary">
Cancel
</Button>
<Button
data-testid="confirm-delete"
color="error"
variant="contained"
onClick={() => {
onConfirm();
onClose();
}}
>
Delete
</Button>
</DialogActions>
</Dialog>
);
}

View File

@ -0,0 +1,40 @@
// react global
import React from 'react';
// components
import { Stack, Typography } from '@mui/material';
//styling
import makeStyles from '@mui/styles/makeStyles';
import nodataImage from '../../assets/noData.svg';
const useStyles = makeStyles((theme) => ({
noDataContainer: {
display: 'flex',
justifyContent: 'center',
alignItems: 'center'
},
noDataImage: {
maxWidth: '233px',
maxHeight: '240px'
},
noDataText: {
fontSize: '1.5rem',
fontWeight: '600',
color: theme.palette.secondary.main
}
}));
function NoDataComponent({ text }) {
const classes = useStyles();
return (
<Stack className={classes.noDataContainer}>
<img src={nodataImage} className={classes.noDataImage} />
<Typography className={classes.noDataText}>{text ? text : 'No Data'}</Typography>
</Stack>
);
}
export default NoDataComponent;

View File

@ -28,6 +28,8 @@ import {
import makeStyles from '@mui/styles/makeStyles'; import makeStyles from '@mui/styles/makeStyles';
import BookmarkIcon from '@mui/icons-material/Bookmark'; import BookmarkIcon from '@mui/icons-material/Bookmark';
import BookmarkBorderIcon from '@mui/icons-material/BookmarkBorder'; import BookmarkBorderIcon from '@mui/icons-material/BookmarkBorder';
import StarIcon from '@mui/icons-material/Star';
import StarBorderIcon from '@mui/icons-material/StarBorder';
import { useTheme } from '@emotion/react'; import { useTheme } from '@emotion/react';
// placeholder images // placeholder images
@ -183,16 +185,24 @@ function RepoCard(props) {
platforms, platforms,
description, description,
downloads, downloads,
stars,
isSigned, isSigned,
signatureInfo,
lastUpdated, lastUpdated,
version, version,
vulnerabilityData, vulnerabilityData,
isBookmarked isBookmarked,
isStarred
} = props; } = props;
// keep a local bookmark state to display in the ui dynamically on updates // keep a local bookmark state to display in the ui dynamically on updates
const [currentBookmarkValue, setCurrentBookmarkValue] = useState(isBookmarked); const [currentBookmarkValue, setCurrentBookmarkValue] = useState(isBookmarked);
// keep a local star state to display in the ui dynamically on updates
const [currentStarValue, setCurrentStarValue] = useState(isStarred);
const [currentStarCount, setCurrentStarCount] = useState(stars);
const goToDetails = () => { const goToDetails = () => {
navigate(`/image/${encodeURIComponent(name)}`); navigate(`/image/${encodeURIComponent(name)}`);
}; };
@ -214,6 +224,23 @@ function RepoCard(props) {
}); });
}; };
const handleStarClick = (event) => {
event.stopPropagation();
event.preventDefault();
api.put(`${host()}${endpoints.starToggle(name)}`, abortController.signal).then((response) => {
if (response.status === 200) {
setCurrentStarValue((prevState) => !prevState);
currentStarValue
? setCurrentStarCount((prevState) => {
return !isNaN(prevState) ? prevState - 1 : prevState;
})
: setCurrentStarCount((prevState) => {
return !isNaN(prevState) ? prevState + 1 : prevState;
});
}
});
};
const platformChips = () => { const platformChips = () => {
const filteredPlatforms = uniq(platforms?.flatMap((platform) => [platform.Os, platform.Arch])); const filteredPlatforms = uniq(platforms?.flatMap((platform) => [platform.Os, platform.Arch]));
const hiddenChips = filteredPlatforms.length - MAX_PLATFORM_CHIPS; const hiddenChips = filteredPlatforms.length - MAX_PLATFORM_CHIPS;
@ -259,6 +286,16 @@ function RepoCard(props) {
); );
}; };
const renderStar = () => {
return (
isAuthenticated() && (
<IconButton component="span" onClick={handleStarClick} data-testid="star-button">
{currentStarValue ? <StarIcon data-testid="starred" /> : <StarBorderIcon data-testid="not-starred" />}
</IconButton>
)
);
};
return ( return (
<Card variant="outlined" className={classes.card} data-testid="repo-card"> <Card variant="outlined" className={classes.card} data-testid="repo-card">
<CardActionArea <CardActionArea
@ -290,7 +327,7 @@ function RepoCard(props) {
<VulnerabilityIconCheck {...vulnerabilityData} className="hide-on-mobile" /> <VulnerabilityIconCheck {...vulnerabilityData} className="hide-on-mobile" />
</div> </div>
<div className="hide-on-mobile"> <div className="hide-on-mobile">
<SignatureIconCheck isSigned={isSigned} className="hide-on-mobile" /> <SignatureIconCheck isSigned={isSigned} signatureInfo={signatureInfo} className="hide-on-mobile" />
</div> </div>
</Stack> </Stack>
<Tooltip title={description || 'Description not available'} placement="top"> <Tooltip title={description || 'Description not available'} placement="top">
@ -336,6 +373,15 @@ function RepoCard(props) {
#1 #1
</Typography> </Typography>
</Grid> */} </Grid> */}
<Grid item xs={12}>
{renderStar()}
<Typography variant="body2" component="span" className={classes.contentRightLabel}>
Stars
</Typography>
<Typography variant="body2" component="span" className={classes.contentRightValue}>
{!isNaN(currentStarCount) ? currentStarCount : `not available`}
</Typography>
</Grid>
<Grid container item xs={12} className={classes.contentRightActions}> <Grid container item xs={12} className={classes.contentRightActions}>
<Grid item>{renderBookmark()}</Grid> <Grid item>{renderBookmark()}</Grid>
</Grid> </Grid>

View File

@ -0,0 +1,21 @@
import React from 'react';
import { Typography, Stack } from '@mui/material';
import { isEmpty } from 'lodash';
function SignatureTooltip({ isSigned, signatureInfo }) {
const { tool, isTrusted, author } = !isEmpty(signatureInfo)
? signatureInfo[0]
: { tool: 'Unknown', isTrusted: 'Unknown', author: 'Unknown' };
return (
<Stack direction="column">
<Typography>{isSigned ? 'Verified Signature' : 'Unverified Signature'}</Typography>
<Typography>Tool: {tool}</Typography>
<Typography>Trusted: {!isEmpty(isTrusted) ? isTrusted : 'Unknown'}</Typography>
<Typography>Author: {!isEmpty(author) ? author : 'Unknown'}</Typography>
</Stack>
);
}
export default SignatureTooltip;

View File

@ -6,6 +6,7 @@ import { Markdown } from 'utilities/MarkdowntojsxWrapper';
import transform from 'utilities/transform'; import transform from 'utilities/transform';
import { DateTime } from 'luxon'; import { DateTime } from 'luxon';
import { KeyboardArrowDown, KeyboardArrowRight } from '@mui/icons-material'; import { KeyboardArrowDown, KeyboardArrowRight } from '@mui/icons-material';
import DeleteTag from 'components/Shared/DeleteTag';
const useStyles = makeStyles((theme) => ({ const useStyles = makeStyles((theme) => ({
card: { card: {
@ -78,9 +79,9 @@ const useStyles = makeStyles((theme) => ({
})); }));
export default function TagCard(props) { export default function TagCard(props) {
const { repoName, tag, lastUpdated, vendor, manifests } = props; const { repoName, tag, lastUpdated, vendor, manifests, repo, onTagDelete, isDeletable } = props;
const [open, setOpen] = useState(false); const [open, setOpen] = useState(false);
const classes = useStyles(); const classes = useStyles();
const lastDate = lastUpdated const lastDate = lastUpdated
@ -99,9 +100,12 @@ export default function TagCard(props) {
return ( return (
<Card className={classes.card} raised> <Card className={classes.card} raised>
<CardContent className={classes.content}> <CardContent className={classes.content}>
<Typography variant="body1" align="left" className={classes.tagHeading}> <Stack direction="row" spacing={2} justifyContent="space-between">
Tag <Typography variant="body1" align="left" className={classes.tagHeading}>
</Typography> Tag
</Typography>
{isDeletable && <DeleteTag repo={repo} tag={tag} onTagDelete={onTagDelete} />}
</Stack>
<Typography variant="body1" align="left" className={classes.tagName} onClick={() => goToTags()}> <Typography variant="body1" align="left" className={classes.tagName} onClick={() => goToTags()}>
{repoName && `${repoName}:`} {repoName && `${repoName}:`}
{tag} {tag}

View File

@ -66,13 +66,18 @@ const useStyles = makeStyles((theme) => ({
cursor: 'pointer', cursor: 'pointer',
textAlign: 'center' textAlign: 'center'
}, },
dropdownCVE: {
color: '#1479FF',
cursor: 'pointer'
},
vulnerabilityCardDivider: { vulnerabilityCardDivider: {
margin: '1rem 0' margin: '1rem 0'
} }
})); }));
function VulnerabilitiyCard(props) { function VulnerabilitiyCard(props) {
const classes = useStyles(); const classes = useStyles();
const { cve, name } = props; const { cve, name, platform, expand } = props;
const [openCVE, setOpenCVE] = useState(expand);
const [openDesc, setOpenDesc] = useState(false); const [openDesc, setOpenDesc] = useState(false);
const [openFixed, setOpenFixed] = useState(false); const [openFixed, setOpenFixed] = useState(false);
const [loadingFixed, setLoadingFixed] = useState(true); const [loadingFixed, setLoadingFixed] = useState(true);
@ -90,7 +95,12 @@ function VulnerabilitiyCard(props) {
setLoadingFixed(true); setLoadingFixed(true);
api api
.get( .get(
`${host()}${endpoints.imageListWithCVEFixed(cve.id, name, { pageNumber, pageSize: CVE_FIXEDIN_PAGE_SIZE })}`, `${host()}${endpoints.imageListWithCVEFixed(
cve.id,
name,
{ pageNumber, pageSize: CVE_FIXEDIN_PAGE_SIZE },
platform ? { Os: platform.Os, Arch: platform.Arch } : {}
)}`,
abortController.signal abortController.signal
) )
.then((response) => { .then((response) => {
@ -117,6 +127,10 @@ function VulnerabilitiyCard(props) {
}; };
}, [openFixed, pageNumber]); }, [openFixed, pageNumber]);
useEffect(() => {
setOpenCVE(expand);
}, [expand]);
const loadMore = () => { const loadMore = () => {
if (loadingFixed || isEndOfList) return; if (loadingFixed || isEndOfList) return;
setPageNumber((pageNumber) => pageNumber + 1); setPageNumber((pageNumber) => pageNumber + 1);
@ -161,49 +175,56 @@ function VulnerabilitiyCard(props) {
<Card className={classes.card} raised> <Card className={classes.card} raised>
<CardContent className={classes.content}> <CardContent className={classes.content}>
<Stack direction="row" spacing="1.25rem"> <Stack direction="row" spacing="1.25rem">
{!openCVE ? (
<KeyboardArrowRight className={classes.dropdownCVE} onClick={() => setOpenCVE(!openCVE)} />
) : (
<KeyboardArrowDown className={classes.dropdownCVE} onClick={() => setOpenCVE(!openCVE)} />
)}
<Typography variant="body1" align="left" className={classes.cveId}> <Typography variant="body1" align="left" className={classes.cveId}>
{cve.id} {cve.id}
</Typography> </Typography>
<VulnerabilityChipCheck vulnerabilitySeverity={cve.severity} /> <VulnerabilityChipCheck vulnerabilitySeverity={cve.severity} />
</Stack> </Stack>
<Typography variant="body1" align="left" className={classes.cveSummary}> <Collapse in={openCVE} timeout="auto" unmountOnExit>
{cve.title} <Typography variant="body1" align="left" className={classes.cveSummary}>
</Typography> {cve.title}
<Divider className={classes.vulnerabilityCardDivider} /> </Typography>
<Stack className={classes.dropdown} onClick={() => setOpenFixed(!openFixed)}> <Divider className={classes.vulnerabilityCardDivider} />
{!openFixed ? ( <Stack className={classes.dropdown} onClick={() => setOpenFixed(!openFixed)}>
<KeyboardArrowRight className={classes.dropdownText} /> {!openFixed ? (
) : ( <KeyboardArrowRight className={classes.dropdownText} />
<KeyboardArrowDown className={classes.dropdownText} />
)}
<Typography className={classes.dropdownText}>Fixed in</Typography>
</Stack>
<Collapse in={openFixed} timeout="auto" unmountOnExit>
<Box sx={{ width: '100%', padding: '0.5rem 0' }}>
{loadingFixed ? (
'Loading...'
) : ( ) : (
<Stack direction="row" sx={{ flexWrap: 'wrap' }}> <KeyboardArrowDown className={classes.dropdownText} />
{renderFixedVer()}
{renderLoadMore()}
</Stack>
)} )}
</Box> <Typography className={classes.dropdownText}>Fixed in</Typography>
</Collapse> </Stack>
<Stack className={classes.dropdown} onClick={() => setOpenDesc(!openDesc)}> <Collapse in={openFixed} timeout="auto" unmountOnExit>
{!openDesc ? ( <Box sx={{ width: '100%', padding: '0.5rem 0' }}>
<KeyboardArrowRight className={classes.dropdownText} /> {loadingFixed ? (
) : ( 'Loading...'
<KeyboardArrowDown className={classes.dropdownText} /> ) : (
)} <Stack direction="row" sx={{ flexWrap: 'wrap' }}>
<Typography className={classes.dropdownText}>Description</Typography> {renderFixedVer()}
</Stack> {renderLoadMore()}
<Collapse in={openDesc} timeout="auto" unmountOnExit> </Stack>
<Box sx={{ padding: '0.5rem 0' }}> )}
<Typography variant="body2" align="left" sx={{ color: '#0F2139', fontSize: '1rem' }}> </Box>
{cve.description} </Collapse>
</Typography> <Stack className={classes.dropdown} onClick={() => setOpenDesc(!openDesc)}>
</Box> {!openDesc ? (
<KeyboardArrowRight className={classes.dropdownText} />
) : (
<KeyboardArrowDown className={classes.dropdownText} />
)}
<Typography className={classes.dropdownText}>Description</Typography>
</Stack>
<Collapse in={openDesc} timeout="auto" unmountOnExit>
<Box sx={{ padding: '0.5rem 0' }}>
<Typography variant="body2" align="left" sx={{ color: '#0F2139', fontSize: '1rem' }}>
{cve.description}
</Typography>
</Box>
</Collapse>
</Collapse> </Collapse>
</CardContent> </CardContent>
</Card> </Card>

View File

@ -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 (
<Stack direction="row" spacing="0.5em">
<div className={[classes.cveCountCard].join(' ')}>Total {total}</div>
<div className={classes.severityList}>
<Tooltip title="Critical">
<div className={[classes.cveCountCard, classes.criticalSeverity].join(' ')}>C {critical}</div>
</Tooltip>
<Tooltip title="High">
<div className={[classes.cveCountCard, classes.highSeverity].join(' ')}>H {high}</div>
</Tooltip>
<Tooltip title="Medium">
<div className={[classes.cveCountCard, classes.mediumSeverity].join(' ')}>M {medium}</div>
</Tooltip>
<Tooltip title="Low">
<div className={[classes.cveCountCard, classes.lowSeverity].join(' ')}>L {low}</div>
</Tooltip>
<Tooltip title="Unknown">
<div className={[classes.cveCountCard, classes.unknownSeverity].join(' ')}>U {unknown}</div>
</Tooltip>
</div>
</Stack>
);
}
export default VulnerabilitiyCountCard;

View File

@ -4,24 +4,52 @@ import React, { useEffect, useMemo, useState, useRef } from 'react';
import { api, endpoints } from '../../../api'; import { api, endpoints } from '../../../api';
// components // components
import { Stack, Typography, InputBase } from '@mui/material'; import {
IconButton,
Stack,
Typography,
InputBase,
ToggleButton,
Menu,
MenuItem,
Divider,
Snackbar,
CircularProgress
} from '@mui/material';
import makeStyles from '@mui/styles/makeStyles'; import makeStyles from '@mui/styles/makeStyles';
import { host } from '../../../host'; import { host } from '../../../host';
import { debounce, isEmpty } from 'lodash'; import { debounce, isEmpty } from 'lodash';
import Loading from '../../Shared/Loading'; import Loading from '../../Shared/Loading';
import { mapCVEInfo } from 'utilities/objectModels'; import { mapCVEInfo, mapAllCVEInfo } from 'utilities/objectModels';
import { EXPLORE_PAGE_SIZE } from 'utilities/paginationConstants'; import { EXPLORE_PAGE_SIZE } from 'utilities/paginationConstants';
import SearchIcon from '@mui/icons-material/Search'; import SearchIcon from '@mui/icons-material/Search';
import DownloadIcon from '@mui/icons-material/Download';
import * as XLSX from 'xlsx';
import exportFromJSON from 'export-from-json';
import ViewHeadlineIcon from '@mui/icons-material/ViewHeadline';
import ViewAgendaIcon from '@mui/icons-material/ViewAgenda';
import VulnerabilitiyCard from '../../Shared/VulnerabilityCard'; import VulnerabilitiyCard from '../../Shared/VulnerabilityCard';
import VulnerabilityCountCard from '../../Shared/VulnerabilityCountCard';
const useStyles = makeStyles((theme) => ({ const useStyles = makeStyles((theme) => ({
searchAndDisplayBar: {
display: 'flex',
justifyContent: 'space-between'
},
title: { title: {
color: theme.palette.primary.main, color: theme.palette.primary.main,
fontSize: '1.5rem', fontSize: '1.5rem',
fontWeight: '600', fontWeight: '600',
marginBottom: '0' marginBottom: '0'
}, },
cveCountSummary: {
color: theme.palette.primary.main,
fontSize: '1.5rem',
fontWeight: '600',
marginBottom: '0'
},
cveId: { cveId: {
color: theme.palette.primary.main, color: theme.palette.primary.main,
fontSize: '1rem', fontSize: '1rem',
@ -40,9 +68,17 @@ const useStyles = makeStyles((theme) => ({
fontSize: '1.4rem', fontSize: '1.4rem',
fontWeight: '600' fontWeight: '600'
}, },
vulnerabilities: {
position: 'relative',
maxWidth: '100%',
flexDirection: 'row',
alignItems: 'center',
justifyContent: 'space-between'
},
search: { search: {
position: 'relative', position: 'relative',
maxWidth: '100%', maxWidth: '100%',
flex: 0.95,
flexDirection: 'row', flexDirection: 'row',
alignItems: 'center', alignItems: 'center',
justifyContent: 'space-between', justifyContent: 'space-between',
@ -50,6 +86,20 @@ const useStyles = makeStyles((theme) => ({
border: '0.063rem solid #E7E7E7', border: '0.063rem solid #E7E7E7',
borderRadius: '0.625rem' borderRadius: '0.625rem'
}, },
expandableSearchInput: {
flexGrow: 0.95
},
view: {
alignContent: 'right',
variant: 'outlined'
},
viewModes: {
position: 'relative',
alignItems: 'baseline',
maxWidth: '100%',
flexDirection: 'row',
justifyContent: 'right'
},
searchIcon: { searchIcon: {
color: '#52637A', color: '#52637A',
paddingRight: '3%' paddingRight: '3%'
@ -65,15 +115,25 @@ const useStyles = makeStyles((theme) => ({
'&::placeholder': { '&::placeholder': {
opacity: '1' opacity: '1'
} }
},
popper: {
width: '100%',
overflow: 'hidden',
padding: '0.3rem',
display: 'flex',
justifyContent: 'left'
} }
})); }));
function VulnerabilitiesDetails(props) { function VulnerabilitiesDetails(props) {
const classes = useStyles(); const classes = useStyles();
const [cveData, setCveData] = useState([]); const [cveData, setCveData] = useState([]);
const [allCveData, setAllCveData] = useState([]);
const [cveSummary, setCVESummary] = useState({});
const [isLoading, setIsLoading] = useState(true); const [isLoading, setIsLoading] = useState(true);
const [isLoadingAllCve, setIsLoadingAllCve] = useState(true);
const abortController = useMemo(() => new AbortController(), []); const abortController = useMemo(() => new AbortController(), []);
const { name, tag } = props; const { name, tag, digest, platform } = props;
// pagination props // pagination props
const [cveFilter, setCveFilter] = useState(''); const [cveFilter, setCveFilter] = useState('');
@ -81,11 +141,20 @@ function VulnerabilitiesDetails(props) {
const [isEndOfList, setIsEndOfList] = useState(false); const [isEndOfList, setIsEndOfList] = useState(false);
const listBottom = useRef(null); const listBottom = useRef(null);
const [anchorExport, setAnchorExport] = useState(null);
const openExport = Boolean(anchorExport);
const [selectedViewMore, setSelectedViewMore] = useState(true);
const getCVERequestName = () => {
return digest !== '' ? `${name}@${digest}` : `${name}:${tag}`;
};
const getPaginatedCVEs = () => { const getPaginatedCVEs = () => {
api api
.get( .get(
`${host()}${endpoints.vulnerabilitiesForRepo( `${host()}${endpoints.vulnerabilitiesForRepo(
`${name}:${tag}`, getCVERequestName(),
{ pageNumber, pageSize: EXPLORE_PAGE_SIZE }, { pageNumber, pageSize: EXPLORE_PAGE_SIZE },
cveFilter cveFilter
)}`, )}`,
@ -94,9 +163,23 @@ function VulnerabilitiesDetails(props) {
.then((response) => { .then((response) => {
if (response.data && response.data.data) { if (response.data && response.data.data) {
let cveInfo = response.data.data.CVEListForImage?.CVEList; let cveInfo = response.data.data.CVEListForImage?.CVEList;
let summary = response.data.data.CVEListForImage?.Summary;
let cveListData = mapCVEInfo(cveInfo); let cveListData = mapCVEInfo(cveInfo);
setCveData((previousState) => (pageNumber === 1 ? cveListData : [...previousState, ...cveListData])); setCveData((previousState) => (pageNumber === 1 ? cveListData : [...previousState, ...cveListData]));
setIsEndOfList(response.data.data.CVEListForImage.Page?.ItemCount < EXPLORE_PAGE_SIZE); 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) { } else if (response.data.errors) {
setIsEndOfList(true); setIsEndOfList(true);
} }
@ -106,10 +189,29 @@ function VulnerabilitiesDetails(props) {
console.error(e); console.error(e);
setIsLoading(false); setIsLoading(false);
setCveData([]); setCveData([]);
setCVESummary(() => {});
setIsEndOfList(true); setIsEndOfList(true);
}); });
}; };
const getAllCVEs = () => {
api
.get(`${host()}${endpoints.allVulnerabilitiesForRepo(getCVERequestName())}`, abortController.signal)
.then((response) => {
if (response.data && response.data.data) {
const cveInfo = response.data.data.CVEListForImage?.CVEList;
const cveListData = mapAllCVEInfo(cveInfo);
setAllCveData(cveListData);
}
setIsLoadingAllCve(false);
})
.catch((e) => {
console.error(e);
setAllCveData([]);
setIsLoadingAllCve(false);
});
};
const resetPagination = () => { const resetPagination = () => {
setIsLoading(true); setIsLoading(true);
setIsEndOfList(false); setIsEndOfList(false);
@ -120,11 +222,39 @@ function VulnerabilitiesDetails(props) {
} }
}; };
const handleOnExportExcel = () => {
const wb = XLSX.utils.book_new(),
ws = XLSX.utils.json_to_sheet(allCveData);
XLSX.utils.book_append_sheet(wb, ws, name.replaceAll('/', '_') + '_' + tag);
XLSX.writeFile(wb, `${name}:${tag}-vulnerabilities.xlsx`);
handleCloseExport();
};
const handleOnExportCSV = () => {
const fileName = `${name}:${tag}-vulnerabilities`;
const exportType = exportFromJSON.types.csv;
exportFromJSON({ data: allCveData, fileName, exportType });
handleCloseExport();
};
const handleCveFilterChange = (e) => { const handleCveFilterChange = (e) => {
const { value } = e.target; const { value } = e.target;
setCveFilter(value); setCveFilter(value);
}; };
const handleClickExport = (event) => {
setAnchorExport(event.currentTarget);
};
const handleCloseExport = () => {
setAnchorExport(null);
};
const debouncedChangeHandler = useMemo(() => debounce(handleCveFilterChange, 300)); const debouncedChangeHandler = useMemo(() => debounce(handleCveFilterChange, 300));
useEffect(() => { useEffect(() => {
@ -168,16 +298,43 @@ function VulnerabilitiesDetails(props) {
}; };
}, []); }, []);
useEffect(() => {
if (openExport && isEmpty(allCveData)) {
getAllCVEs();
}
}, [openExport]);
const renderCVEs = () => { const renderCVEs = () => {
return !isEmpty(cveData) ? ( return !isEmpty(cveData) ? (
cveData.map((cve, index) => { cveData.map((cve, index) => {
return <VulnerabilitiyCard key={index} cve={cve} name={name} />; return <VulnerabilitiyCard key={index} cve={cve} name={name} platform={platform} expand={selectedViewMore} />;
}) })
) : ( ) : (
<div>{!isLoading && <Typography className={classes.none}> No Vulnerabilities </Typography>}</div> <div>{!isLoading && <Typography className={classes.none}> No Vulnerabilities </Typography>}</div>
); );
}; };
const renderCVESummary = () => {
if (cveSummary === undefined) {
return;
}
console.log('Test');
return !isEmpty(cveSummary) ? (
<VulnerabilityCountCard
total={cveSummary.Count}
critical={cveSummary.CriticalCount}
high={cveSummary.HighCount}
medium={cveSummary.MediumCount}
low={cveSummary.LowCount}
unknown={cveSummary.UnknownCount}
/>
) : (
<></>
);
};
const renderListBottom = () => { const renderListBottom = () => {
if (isLoading) { if (isLoading) {
return <Loading />; return <Loading />;
@ -190,9 +347,76 @@ function VulnerabilitiesDetails(props) {
return ( return (
<Stack direction="column" spacing="1rem" data-testid="vulnerability-container"> <Stack direction="column" spacing="1rem" data-testid="vulnerability-container">
<Typography variant="h4" gutterBottom component="div" align="left" className={classes.title}> <Stack className={classes.vulnerabilities}>
Vulnerabilities <Typography variant="h4" gutterBottom component="div" align="left" className={classes.title}>
</Typography> Vulnerabilities
</Typography>
<Stack direction="row" spacing="1rem" className={classes.viewModes}>
<IconButton disableRipple onClick={handleClickExport}>
<DownloadIcon />
</IconButton>
<Snackbar
open={openExport && isLoadingAllCve}
message="Getting your data ready for export"
action={<CircularProgress size="2rem" sx={{ color: '#FFFFFF' }} />}
/>
<ToggleButton
value="viewLess"
title="Collapse list view"
size="small"
className={classes.view}
selected={!selectedViewMore}
onChange={() => setSelectedViewMore(false)}
>
<ViewHeadlineIcon />
</ToggleButton>
<ToggleButton
value="viewMore"
title="Expand list view"
size="small"
className={classes.view}
selected={selectedViewMore}
onChange={() => setSelectedViewMore(true)}
>
<ViewAgendaIcon />
</ToggleButton>
</Stack>
<Menu
anchorEl={anchorExport}
open={openExport}
onClose={handleCloseExport}
data-testid="export-dropdown"
anchorOrigin={{
vertical: 'bottom',
horizontal: 'center'
}}
transformOrigin={{
vertical: 'top',
horizontal: 'center'
}}
>
<MenuItem
onClick={handleOnExportCSV}
disableRipple
disabled={isLoadingAllCve}
className={classes.popper}
data-testid="export-csv-menuItem"
>
csv
</MenuItem>
<Divider sx={{ my: 0.5 }} />
<MenuItem
onClick={handleOnExportExcel}
disableRipple
disabled={isLoadingAllCve}
className={classes.popper}
data-testid="export-excel-menuItem"
>
xlsx
</MenuItem>
</Menu>
</Stack>
{renderCVESummary()}
<Stack className={classes.search}> <Stack className={classes.search}>
<InputBase <InputBase
placeholder={'Search'} placeholder={'Search'}

View File

@ -59,7 +59,6 @@ const useStyles = makeStyles((theme) => ({
fontSize: '1rem', fontSize: '1rem',
lineHeight: '1.5rem', lineHeight: '1.5rem',
color: '#52637A', color: '#52637A',
padding: '1rem 0 0 0',
maxWidth: '100%', maxWidth: '100%',
[theme.breakpoints.down('md')]: { [theme.breakpoints.down('md')]: {
padding: '0.5rem 0 0 0', padding: '0.5rem 0 0 0',
@ -209,7 +208,14 @@ function TagDetails() {
case 'IsDependentOn': case 'IsDependentOn':
return <IsDependentOn name={imageDetailData.name} digest={selectedManifest.digest} />; return <IsDependentOn name={imageDetailData.name} digest={selectedManifest.digest} />;
case 'Vulnerabilities': case 'Vulnerabilities':
return <VulnerabilitiesDetails name={reponame} tag={tag} />; return (
<VulnerabilitiesDetails
name={reponame}
tag={tag}
digest={selectedManifest?.digest}
platform={selectedManifest.platform}
/>
);
case 'ReferredBy': case 'ReferredBy':
return <ReferredBy referrers={imageDetailData.referrers} />; return <ReferredBy referrers={imageDetailData.referrers} />;
default: default:
@ -227,10 +233,10 @@ function TagDetails() {
<Card className={classes.cardRoot}> <Card className={classes.cardRoot}>
<CardContent className={classes.cardContent}> <CardContent className={classes.cardContent}>
<Grid container> <Grid container>
<Grid item xs={12} md={8} className={classes.header}> <Grid item xs={12} md={9} className={classes.header}>
<Stack <Stack
alignItems="center" alignItems="center"
sx={{ width: { xs: '100%', md: 'auto' } }} sx={{ width: { xs: '100%', md: 'auto' }, marginBottom: '1rem' }}
direction={{ xs: 'column', md: 'row' }} direction={{ xs: 'column', md: 'row' }}
spacing={1} spacing={1}
> >
@ -254,32 +260,34 @@ function TagDetails() {
vulnerabilitySeverity={imageDetailData.vulnerabiltySeverity} vulnerabilitySeverity={imageDetailData.vulnerabiltySeverity}
count={imageDetailData.vulnerabilityCount} count={imageDetailData.vulnerabilityCount}
/> />
<SignatureIconCheck isSigned={imageDetailData.isSigned} /> <SignatureIconCheck
</Stack> isSigned={imageDetailData.isSigned}
signatureInfo={imageDetailData.signatureInfo}
<Stack sx={{ width: { xs: '100%', md: 'auto' } }}> />
<FormControl sx={{ m: '1', minWidth: '4.6875rem' }} className={classes.sortForm} size="small">
<InputLabel>OS/Arch</InputLabel>
{!isEmpty(selectedManifest) && (
<Select
label="OS/Arch"
value={selectedManifest}
onChange={handleOSArchChange}
MenuProps={{ disableScrollLock: true }}
>
{imageDetailData.manifests.map((el) => (
<MenuItem key={el.digest} value={el}>
{`${el.platform?.Os}/${el.platform?.Arch}`}
</MenuItem>
))}
</Select>
)}
</FormControl>
</Stack> </Stack>
</Stack> </Stack>
<Typography gutterBottom className={classes.digest}> <Stack direction="row" alignItems="center" spacing="1rem">
Digest: {selectedManifest?.digest} <FormControl sx={{ m: '1', minWidth: '4.6875rem' }} className={classes.sortForm} size="small">
</Typography> <InputLabel>OS/Arch</InputLabel>
{!isEmpty(selectedManifest) && (
<Select
label="OS/Arch"
value={selectedManifest}
onChange={handleOSArchChange}
MenuProps={{ disableScrollLock: true }}
>
{imageDetailData.manifests.map((el) => (
<MenuItem key={el.digest} value={el}>
{`${el.platform?.Os}/${el.platform?.Arch}`}
</MenuItem>
))}
</Select>
)}
</FormControl>
<Typography gutterBottom className={classes.digest}>
Digest: {selectedManifest?.digest}
</Typography>
</Stack>
</Grid> </Grid>
</Grid> </Grid>
</CardContent> </CardContent>

View File

@ -6,6 +6,10 @@ const osFilters = [
{ {
label: 'linux', label: 'linux',
value: 'linux' value: 'linux'
},
{
label: 'freebsd',
value: 'freebsd'
} }
]; ];
@ -19,6 +23,11 @@ const imageFilters = [
label: 'Bookmarks', label: 'Bookmarks',
value: 'IsBookmarked', value: 'IsBookmarked',
type: 'boolean' type: 'boolean'
},
{
label: 'Starred Repositories',
value: 'IsStarred',
type: 'boolean'
} }
]; ];

View File

@ -5,6 +5,7 @@ const mapToRepo = (responseRepo) => {
tags: responseRepo.NewestImage?.Labels, tags: responseRepo.NewestImage?.Labels,
description: responseRepo.NewestImage?.Description, description: responseRepo.NewestImage?.Description,
isSigned: responseRepo.NewestImage?.IsSigned, isSigned: responseRepo.NewestImage?.IsSigned,
signatureInfo: responseRepo.NewestImage?.SignatureInfo?.map((sigInfo) => mapSignatureInfo(sigInfo)),
isBookmarked: responseRepo.IsBookmarked, isBookmarked: responseRepo.IsBookmarked,
isStarred: responseRepo.IsStarred, isStarred: responseRepo.IsStarred,
platforms: responseRepo.Platforms, platforms: responseRepo.Platforms,
@ -14,6 +15,7 @@ const mapToRepo = (responseRepo) => {
logo: responseRepo.NewestImage?.Logo, logo: responseRepo.NewestImage?.Logo,
lastUpdated: responseRepo.LastUpdated, lastUpdated: responseRepo.LastUpdated,
downloads: responseRepo.DownloadCount, downloads: responseRepo.DownloadCount,
stars: responseRepo.StarCount,
vulnerabiltySeverity: responseRepo.NewestImage?.Vulnerabilities?.MaxSeverity, vulnerabiltySeverity: responseRepo.NewestImage?.Vulnerabilities?.MaxSeverity,
vulnerabilityCount: responseRepo.NewestImage?.Vulnerabilities?.Count vulnerabilityCount: responseRepo.NewestImage?.Vulnerabilities?.Count
}; };
@ -32,11 +34,13 @@ const mapToRepoFromRepoInfo = (responseRepoInfo) => {
title: responseRepoInfo.Summary?.NewestImage?.Title, title: responseRepoInfo.Summary?.NewestImage?.Title,
source: responseRepoInfo.Summary?.NewestImage?.Source, source: responseRepoInfo.Summary?.NewestImage?.Source,
downloads: responseRepoInfo.Summary?.NewestImage?.DownloadCount, downloads: responseRepoInfo.Summary?.NewestImage?.DownloadCount,
stars: responseRepoInfo.Summary?.NewestImage?.StarCount,
overview: responseRepoInfo.Summary?.NewestImage?.Documentation, overview: responseRepoInfo.Summary?.NewestImage?.Documentation,
license: responseRepoInfo.Summary?.NewestImage?.Licenses, license: responseRepoInfo.Summary?.NewestImage?.Licenses,
vulnerabilitySeverity: responseRepoInfo.Summary?.NewestImage?.Vulnerabilities?.MaxSeverity, vulnerabilitySeverity: responseRepoInfo.Summary?.NewestImage?.Vulnerabilities?.MaxSeverity,
vulnerabilityCount: responseRepoInfo.Summary?.NewestImage?.Vulnerabilities?.Count, vulnerabilityCount: responseRepoInfo.Summary?.NewestImage?.Vulnerabilities?.Count,
isSigned: responseRepoInfo.Summary?.NewestImage?.IsSigned, isSigned: responseRepoInfo.Summary?.NewestImage?.IsSigned,
signatureInfo: responseRepoInfo.Summary?.NewestImage?.SignatureInfo?.map((sigInfo) => mapSignatureInfo(sigInfo)),
isBookmarked: responseRepoInfo.Summary?.IsBookmarked, isBookmarked: responseRepoInfo.Summary?.IsBookmarked,
isStarred: responseRepoInfo.Summary?.IsStarred, isStarred: responseRepoInfo.Summary?.IsStarred,
logo: responseRepoInfo.Summary?.NewestImage?.Logo logo: responseRepoInfo.Summary?.NewestImage?.Logo
@ -51,9 +55,11 @@ const mapToImage = (responseImage) => {
referrers: responseImage.Referrers, referrers: responseImage.Referrers,
size: responseImage.Size, size: responseImage.Size,
downloadCount: responseImage.DownloadCount, downloadCount: responseImage.DownloadCount,
starCount: responseImage.StarCount,
lastUpdated: responseImage.LastUpdated, lastUpdated: responseImage.LastUpdated,
description: responseImage.Description, description: responseImage.Description,
isSigned: responseImage.IsSigned, isSigned: responseImage.IsSigned,
signatureInfo: responseImage.SignatureInfo?.map((sigInfo) => mapSignatureInfo(sigInfo)),
license: responseImage.Licenses, license: responseImage.Licenses,
labels: responseImage.Labels, labels: responseImage.Labels,
title: responseImage.Title, title: responseImage.Title,
@ -63,6 +69,7 @@ const mapToImage = (responseImage) => {
authors: responseImage.Authors, authors: responseImage.Authors,
vulnerabiltySeverity: responseImage.Vulnerabilities?.MaxSeverity, vulnerabiltySeverity: responseImage.Vulnerabilities?.MaxSeverity,
vulnerabilityCount: responseImage.Vulnerabilities?.Count, vulnerabilityCount: responseImage.Vulnerabilities?.Count,
isDeletable: responseImage.IsDeletable,
// frontend only prop to increase interop with Repo objects and code reusability // frontend only prop to increase interop with Repo objects and code reusability
name: `${responseImage.RepoName}:${responseImage.Tag}` name: `${responseImage.RepoName}:${responseImage.Tag}`
}; };
@ -76,6 +83,7 @@ const mapToManifest = (responseManifest) => {
size: responseManifest.Size, size: responseManifest.Size,
platform: responseManifest.Platform, platform: responseManifest.Platform,
downloadCount: responseManifest.DownloadCount, downloadCount: responseManifest.DownloadCount,
starCount: responseManifest.StarCount,
layers: responseManifest.Layers, layers: responseManifest.Layers,
history: responseManifest.History, history: responseManifest.History,
vulnerabilities: responseManifest.Vulnerabilities vulnerabilities: responseManifest.Vulnerabilities
@ -94,6 +102,38 @@ const mapCVEInfo = (cveInfo) => {
return cveList; return cveList;
}; };
const mapAllCVEInfo = (cveInfo) => {
const cveList = cveInfo.flatMap((cve) => {
return cve.PackageList.map((packageInfo) => {
return {
id: cve.Id,
severity: cve.Severity,
title: cve.Title,
description: cve.Description,
reference: cve.Reference,
packageName: packageInfo.Name,
packageInstalledVersion: packageInfo.InstalledVersion,
packageFixedVersion: packageInfo.FixedVersion
};
});
});
return cveList;
};
const mapSignatureInfo = (signatureInfo) => {
return signatureInfo
? {
tool: signatureInfo.Tool,
isTrusted: signatureInfo.IsTrusted?.toString(),
author: signatureInfo.Author
}
: {
tool: 'Unknown',
isTrusted: 'Unknown',
author: 'Unknown'
};
};
const mapReferrer = (referrer) => ({ const mapReferrer = (referrer) => ({
mediaType: referrer.MediaType, mediaType: referrer.MediaType,
artifactType: referrer.ArtifactType, artifactType: referrer.ArtifactType,
@ -102,4 +142,4 @@ const mapReferrer = (referrer) => ({
annotations: referrer.Annotations?.map((annotation) => ({ key: annotation.Key, value: annotation.Value })) annotations: referrer.Annotations?.map((annotation) => ({ key: annotation.Key, value: annotation.Value }))
}); });
export { mapToRepo, mapToImage, mapToRepoFromRepoInfo, mapCVEInfo, mapReferrer, mapToManifest }; export { mapToRepo, mapToImage, mapToRepoFromRepoInfo, mapCVEInfo, mapAllCVEInfo, mapReferrer, mapToManifest };

View File

@ -4,6 +4,7 @@ const HOME_PAGE_SIZE = 10;
const HOME_POPULAR_PAGE_SIZE = 3; const HOME_POPULAR_PAGE_SIZE = 3;
const HOME_RECENT_PAGE_SIZE = 2; const HOME_RECENT_PAGE_SIZE = 2;
const HOME_BOOKMARKS_PAGE_SIZE = 2; const HOME_BOOKMARKS_PAGE_SIZE = 2;
const HOME_STARS_PAGE_SIZE = 2;
const CVE_FIXEDIN_PAGE_SIZE = 5; const CVE_FIXEDIN_PAGE_SIZE = 5;
export { export {
@ -13,5 +14,6 @@ export {
CVE_FIXEDIN_PAGE_SIZE, CVE_FIXEDIN_PAGE_SIZE,
HOME_POPULAR_PAGE_SIZE, HOME_POPULAR_PAGE_SIZE,
HOME_RECENT_PAGE_SIZE, HOME_RECENT_PAGE_SIZE,
HOME_BOOKMARKS_PAGE_SIZE HOME_BOOKMARKS_PAGE_SIZE,
HOME_STARS_PAGE_SIZE
}; };

View File

@ -84,11 +84,11 @@ const VulnerabilityChipCheck = ({ vulnerabilitySeverity }) => {
return result; return result;
}; };
const SignatureIconCheck = ({ isSigned }) => { const SignatureIconCheck = ({ isSigned, signatureInfo }) => {
if (isSigned) { if (isSigned) {
return <VerifiedSignatureIcon />; return <VerifiedSignatureIcon signatureInfo={signatureInfo} />;
} else { } else {
return <UnverifiedSignatureIcon />; return <UnverifiedSignatureIcon signatureInfo={signatureInfo} />;
} }
}; };

View File

@ -3,6 +3,7 @@ import { Chip, Tooltip } from '@mui/material';
import SvgIcon from '@mui/material/SvgIcon'; import SvgIcon from '@mui/material/SvgIcon';
import { ReactComponent as failedScanBug } from '../assets/failedScan.svg'; import { ReactComponent as failedScanBug } from '../assets/failedScan.svg';
import { createSvgIcon } from '@mui/material/utils'; import { createSvgIcon } from '@mui/material/utils';
import SignatureTooltip from 'components/Shared/SignatureTooltip';
const FilledBugIcon = createSvgIcon( const FilledBugIcon = createSvgIcon(
<path d="M17.0293 5.13093V6.1543H18.3828L21.2414 3.24068L22.2621 4.27812L19.5552 7.03876L19.5879 7.12668C20.1841 8.73695 20.4862 10.4449 20.4793 12.1662C20.4793 12.5064 20.4678 12.8466 20.4448 13.186L20.4397 13.2634H24V14.7334H20.2569L20.2466 14.7932C19.9431 16.4882 19.3517 18.0338 18.5466 19.335L18.4862 19.4335L21.9276 22.9608L20.9052 24L17.6121 20.6239L17.5138 20.7365C16.0259 22.4333 14.0983 23.4514 11.9983 23.4514C9.86724 23.4514 7.91207 22.4016 6.41552 20.6573L6.31552 20.5413L3.08966 23.833L2.06897 22.792L5.45345 19.3403L5.39483 19.2436C4.61897 17.9618 4.04655 16.4478 3.75 14.7932L3.73966 14.7334H0V13.2634H3.55862L3.55345 13.1843C3.53103 12.8502 3.51897 12.509 3.51897 12.1644C3.51202 10.4654 3.80581 8.77905 4.38621 7.18646L4.41897 7.1003L1.64138 4.2535L2.66379 3.21606L5.53103 6.1543H6.96724V5.13093C6.96724 3.77012 7.49729 2.46505 8.4408 1.50281C9.3843 0.540578 10.664 0 11.9983 0C13.3326 0 14.6123 0.540578 15.5558 1.50281C16.4993 2.46505 17.0293 3.77012 17.0293 5.13093Z" />, <path d="M17.0293 5.13093V6.1543H18.3828L21.2414 3.24068L22.2621 4.27812L19.5552 7.03876L19.5879 7.12668C20.1841 8.73695 20.4862 10.4449 20.4793 12.1662C20.4793 12.5064 20.4678 12.8466 20.4448 13.186L20.4397 13.2634H24V14.7334H20.2569L20.2466 14.7932C19.9431 16.4882 19.3517 18.0338 18.5466 19.335L18.4862 19.4335L21.9276 22.9608L20.9052 24L17.6121 20.6239L17.5138 20.7365C16.0259 22.4333 14.0983 23.4514 11.9983 23.4514C9.86724 23.4514 7.91207 22.4016 6.41552 20.6573L6.31552 20.5413L3.08966 23.833L2.06897 22.792L5.45345 19.3403L5.39483 19.2436C4.61897 17.9618 4.04655 16.4478 3.75 14.7932L3.73966 14.7334H0V13.2634H3.55862L3.55345 13.1843C3.53103 12.8502 3.51897 12.509 3.51897 12.1644C3.51202 10.4654 3.80581 8.77905 4.38621 7.18646L4.41897 7.1003L1.64138 4.2535L2.66379 3.21606L5.53103 6.1543H6.96724V5.13093C6.96724 3.77012 7.49729 2.46505 8.4408 1.50281C9.3843 0.540578 10.664 0 11.9983 0C13.3326 0 14.6123 0.540578 15.5558 1.50281C16.4993 2.46505 17.0293 3.77012 17.0293 5.13093Z" />,
@ -23,7 +24,7 @@ const VerifiedShieldIcon = createSvgIcon(
const NoneVulnerabilityIcon = ({ vulnerabilityStringTitle }) => { const NoneVulnerabilityIcon = ({ vulnerabilityStringTitle }) => {
return ( return (
<Tooltip title={`${vulnerabilityStringTitle} Vulnerability`} placement="top"> <Tooltip title={`${vulnerabilityStringTitle}`} placement="top">
<OutlinedBugIcon <OutlinedBugIcon
sx={{ sx={{
color: '#43A047!important', color: '#43A047!important',
@ -40,7 +41,7 @@ const NoneVulnerabilityIcon = ({ vulnerabilityStringTitle }) => {
}; };
const UnknownVulnerabilityIcon = ({ vulnerabilityStringTitle }) => { const UnknownVulnerabilityIcon = ({ vulnerabilityStringTitle }) => {
return ( return (
<Tooltip title={`${vulnerabilityStringTitle} Vulnerability`} placement="top"> <Tooltip title={`${vulnerabilityStringTitle}`} placement="top">
<OutlinedBugIcon <OutlinedBugIcon
sx={{ sx={{
color: '#52637A', color: '#52637A',
@ -75,7 +76,7 @@ const FailedScanIcon = () => {
}; };
const LowVulnerabilityIcon = ({ vulnerabilityStringTitle }) => { const LowVulnerabilityIcon = ({ vulnerabilityStringTitle }) => {
return ( return (
<Tooltip title={`${vulnerabilityStringTitle} Vulnerability`} placement="top"> <Tooltip title={`${vulnerabilityStringTitle}`} placement="top">
<OutlinedBugIcon <OutlinedBugIcon
sx={{ sx={{
color: '#FB8C00', color: '#FB8C00',
@ -92,7 +93,7 @@ const LowVulnerabilityIcon = ({ vulnerabilityStringTitle }) => {
}; };
const MediumVulnerabilityIcon = ({ vulnerabilityStringTitle }) => { const MediumVulnerabilityIcon = ({ vulnerabilityStringTitle }) => {
return ( return (
<Tooltip title={`${vulnerabilityStringTitle} Vulnerability`} placement="top"> <Tooltip title={`${vulnerabilityStringTitle}`} placement="top">
<FilledBugIcon <FilledBugIcon
sx={{ sx={{
color: '#FB8C00', color: '#FB8C00',
@ -109,7 +110,7 @@ const MediumVulnerabilityIcon = ({ vulnerabilityStringTitle }) => {
}; };
const HighVulnerabilityIcon = ({ vulnerabilityStringTitle }) => { const HighVulnerabilityIcon = ({ vulnerabilityStringTitle }) => {
return ( return (
<Tooltip title={`${vulnerabilityStringTitle} Vulnerability`} placement="top"> <Tooltip title={`${vulnerabilityStringTitle}`} placement="top">
<OutlinedBugIcon <OutlinedBugIcon
sx={{ sx={{
color: '#E53935', color: '#E53935',
@ -126,7 +127,7 @@ const HighVulnerabilityIcon = ({ vulnerabilityStringTitle }) => {
}; };
const CriticalVulnerabilityIcon = ({ vulnerabilityStringTitle }) => { const CriticalVulnerabilityIcon = ({ vulnerabilityStringTitle }) => {
return ( return (
<Tooltip title={`${vulnerabilityStringTitle} Vulnerability`} placement="top"> <Tooltip title={`${vulnerabilityStringTitle}`} placement="top">
<FilledBugIcon <FilledBugIcon
sx={{ sx={{
color: '#E53935', color: '#E53935',
@ -144,13 +145,10 @@ const CriticalVulnerabilityIcon = ({ vulnerabilityStringTitle }) => {
const NoneVulnerabilityChip = () => { const NoneVulnerabilityChip = () => {
return ( return (
<Chip <Chip
label="No Vulnerability" label="None"
sx={{ backgroundColor: '#E8F5E9', color: '#388E3C', fontSize: '0.8125rem' }} sx={{ backgroundColor: '#E8F5E9', color: '#388E3C', fontSize: '0.8125rem' }}
variant="filled" variant="filled"
onDelete={() => { icon={<OutlinedBugIcon sx={{ color: '#388E3C!important' }} />}
return;
}}
deleteIcon={<OutlinedBugIcon sx={{ color: '#388E3C!important' }} />}
data-testid="none-vulnerability-chip" data-testid="none-vulnerability-chip"
/> />
); );
@ -158,13 +156,10 @@ const NoneVulnerabilityChip = () => {
const UnknownVulnerabilityChip = () => { const UnknownVulnerabilityChip = () => {
return ( return (
<Chip <Chip
label="Unknown Vulnerability" label="Unknown"
sx={{ backgroundColor: '#ECEFF1', color: '#52637A', fontSize: '0.8125rem' }} sx={{ backgroundColor: '#ECEFF1', color: '#52637A', fontSize: '0.8125rem' }}
variant="filled" variant="filled"
onDelete={() => { icon={<OutlinedBugIcon sx={{ color: '#52637A!important' }} />}
return;
}}
deleteIcon={<OutlinedBugIcon sx={{ color: '#52637A!important' }} />}
data-testid="unknown-vulnerability-chip" data-testid="unknown-vulnerability-chip"
/> />
); );
@ -175,10 +170,7 @@ const FailedScanChip = () => {
label="Failed to scan" label="Failed to scan"
sx={{ backgroundColor: '#848484', color: '#F6F7F9', fontSize: '0.8125rem' }} sx={{ backgroundColor: '#848484', color: '#F6F7F9', fontSize: '0.8125rem' }}
variant="filled" variant="filled"
onDelete={() => { icon={<SvgIcon component={failedScanBug} sx={{ color: '#F6F7F9!important' }} />}
return;
}}
deleteIcon={<SvgIcon component={failedScanBug} sx={{ color: '#F6F7F9!important' }} />}
data-testid="failed-vulnerability-chip" data-testid="failed-vulnerability-chip"
/> />
); );
@ -186,13 +178,10 @@ const FailedScanChip = () => {
const LowVulnerabilityChip = () => { const LowVulnerabilityChip = () => {
return ( return (
<Chip <Chip
label="Low Vulnerability" label="Low"
sx={{ backgroundColor: '#FFF3E0', color: '#FB8C00', fontSize: '0.8125rem' }} sx={{ backgroundColor: '#FFF3E0', color: '#FB8C00', fontSize: '0.8125rem' }}
variant="filled" variant="filled"
onDelete={() => { icon={<OutlinedBugIcon sx={{ color: '#FB8C00!important' }} />}
return;
}}
deleteIcon={<OutlinedBugIcon sx={{ color: '#FB8C00!important' }} />}
data-testid="low-vulnerability-chip" data-testid="low-vulnerability-chip"
/> />
); );
@ -200,13 +189,10 @@ const LowVulnerabilityChip = () => {
const MediumVulnerabilityChip = () => { const MediumVulnerabilityChip = () => {
return ( return (
<Chip <Chip
label="Medium Vulnerability" label="Medium"
sx={{ backgroundColor: '#FFF3E0', color: '#FB8C00', fontSize: '0.8125rem' }} sx={{ backgroundColor: '#FFF3E0', color: '#FB8C00', fontSize: '0.8125rem' }}
variant="filled" variant="filled"
onDelete={() => { icon={<FilledBugIcon sx={{ color: '#FB8C00!important' }} />}
return;
}}
deleteIcon={<FilledBugIcon sx={{ color: '#FB8C00!important' }} />}
data-testid="medium-vulnerability-chip" data-testid="medium-vulnerability-chip"
/> />
); );
@ -214,13 +200,10 @@ const MediumVulnerabilityChip = () => {
const HighVulnerabilityChip = () => { const HighVulnerabilityChip = () => {
return ( return (
<Chip <Chip
label="High Vulnerability" label="High"
sx={{ backgroundColor: '#FEEBEE', color: '#E53935', fontSize: '0.8125rem' }} sx={{ backgroundColor: '#FEEBEE', color: '#E53935', fontSize: '0.8125rem' }}
variant="filled" variant="filled"
onDelete={() => { icon={<OutlinedBugIcon sx={{ color: '#E53935!important' }} />}
return;
}}
deleteIcon={<OutlinedBugIcon sx={{ color: '#E53935!important' }} />}
data-testid="high-vulnerability-chip" data-testid="high-vulnerability-chip"
/> />
); );
@ -228,21 +211,18 @@ const HighVulnerabilityChip = () => {
const CriticalVulnerabilityChip = () => { const CriticalVulnerabilityChip = () => {
return ( return (
<Chip <Chip
label="Critical Vulnerability" label="Critical"
sx={{ backgroundColor: '#FEEBEE', color: '#E53935', fontSize: '0.8125rem' }} sx={{ backgroundColor: '#FEEBEE', color: '#E53935', fontSize: '0.8125rem' }}
variant="filled" variant="filled"
onDelete={() => { icon={<FilledBugIcon sx={{ color: '#E53935!important' }} />}
return;
}}
deleteIcon={<FilledBugIcon sx={{ color: '#E53935!important' }} />}
data-testid="critical-vulnerability-chip" data-testid="critical-vulnerability-chip"
/> />
); );
}; };
const UnverifiedSignatureIcon = () => { const UnverifiedSignatureIcon = ({ signatureInfo }) => {
return ( return (
<Tooltip title="Unverified Signature" placement="top"> <Tooltip title={<SignatureTooltip isSigned={false} signatureInfo={signatureInfo} />} placement="top">
<UnverifiedShieldIcon <UnverifiedShieldIcon
sx={{ sx={{
color: '#E53935', color: '#E53935',
@ -257,9 +237,9 @@ const UnverifiedSignatureIcon = () => {
</Tooltip> </Tooltip>
); );
}; };
const VerifiedSignatureIcon = () => { const VerifiedSignatureIcon = ({ signatureInfo }) => {
return ( return (
<Tooltip title="Verified Signature" placement="top"> <Tooltip title={<SignatureTooltip isSigned={true} signatureInfo={signatureInfo} />} placement="top">
<VerifiedShieldIcon <VerifiedShieldIcon
viewBox="0 0 24 24" viewBox="0 0 24 24"
sx={{ sx={{

View File

@ -7,7 +7,7 @@ import { hosts, endpoints, sortCriteria } from './values/test-constants';
test.describe('explore page test', () => { test.describe('explore page test', () => {
test.beforeEach(async ({ page }) => { test.beforeEach(async ({ page }) => {
await page.addInitScript(() => { await page.addInitScript(() => {
window.localStorage.setItem('token', '-'); window.localStorage.setItem('authConfig', '{}');
}); });
}); });
@ -76,8 +76,14 @@ test.describe('explore page test', () => {
await expect(exploreFirst).toBeVisible({ timeout: 250000 }); await expect(exploreFirst).toBeVisible({ timeout: 250000 });
const windowsFilter = page.getByRole('checkbox', { name: 'windows' });
await linuxFilter.uncheck(); await linuxFilter.uncheck();
await page.getByRole('checkbox', { name: 'windows' }).check(); await windowsFilter.check();
await expect(exploreFirst).not.toBeVisible({ timeout: 250000 });
const freebsdFilter = page.getByRole('checkbox', { name: 'freebsd' });
await windowsFilter.uncheck();
await freebsdFilter.check();
await expect(exploreFirst).not.toBeVisible({ timeout: 250000 }); await expect(exploreFirst).not.toBeVisible({ timeout: 250000 });
}); });
}); });

View File

@ -5,7 +5,7 @@ import { hosts, endpoints, sortCriteria } from './values/test-constants';
test.describe('homepage test', () => { test.describe('homepage test', () => {
test.beforeEach(async ({ page }) => { test.beforeEach(async ({ page }) => {
await page.addInitScript(() => { await page.addInitScript(() => {
window.localStorage.setItem('token', '-'); window.localStorage.setItem('authConfig', '{}');
}); });
}); });

View File

@ -5,7 +5,7 @@ import { getRepoListOrderedAlpha } from './utils/test-data-parser';
test.describe('navbar test', () => { test.describe('navbar test', () => {
test.beforeEach(async ({ page }) => { test.beforeEach(async ({ page }) => {
await page.addInitScript(() => { await page.addInitScript(() => {
window.localStorage.setItem('token', '-'); window.localStorage.setItem('authConfig', '{}');
}); });
}); });

View File

@ -8,7 +8,7 @@ const testRepo = getMultiTagRepo();
test.describe('Repository page test', () => { test.describe('Repository page test', () => {
test.beforeEach(async ({ page }) => { test.beforeEach(async ({ page }) => {
await page.addInitScript(() => { await page.addInitScript(() => {
window.localStorage.setItem('token', '-'); window.localStorage.setItem('authConfig', '{}');
}); });
await page.goto(`${hosts.ui}/image/${testRepo.repo}`); await page.goto(`${hosts.ui}/image/${testRepo.repo}`);

View File

@ -5,7 +5,7 @@ import { hosts, pageSizes } from './values/test-constants';
test.describe('Tag page test', () => { test.describe('Tag page test', () => {
test.beforeEach(async ({ page }) => { test.beforeEach(async ({ page }) => {
await page.addInitScript(() => { await page.addInitScript(() => {
window.localStorage.setItem('token', '-'); window.localStorage.setItem('authConfig', '{}');
}); });
}); });
@ -37,6 +37,7 @@ test.describe('Tag page test', () => {
await page.goto(`${hosts.ui}/image/${tagWithVulnerabilities.title}/tag/${tagWithVulnerabilities.tag}`); await page.goto(`${hosts.ui}/image/${tagWithVulnerabilities.title}/tag/${tagWithVulnerabilities.tag}`);
await page.getByRole('tab', { name: 'Vulnerabilities' }).click(); await page.getByRole('tab', { name: 'Vulnerabilities' }).click();
await expect(page.getByTestId('vulnerability-container').locator('div').nth(1)).toBeVisible({ timeout: 100000 }); await expect(page.getByTestId('vulnerability-container').locator('div').nth(1)).toBeVisible({ timeout: 100000 });
await expect(page.getByText('CVE-').nth(0)).toBeVisible({ timeout: 100000 });
await expect(await page.getByText('CVE-').count()).toBeGreaterThan(0); await expect(await page.getByText('CVE-').count()).toBeGreaterThan(0);
await expect(await page.getByText('CVE-').count()).toBeLessThanOrEqual(pageSizes.EXPLORE); await expect(await page.getByText('CVE-').count()).toBeLessThanOrEqual(pageSizes.EXPLORE);
}); });

View File

@ -17,15 +17,15 @@ const pageSizes = {
}; };
const endpoints = { const endpoints = {
repoList: `/v2/_zot/ext/search?query={RepoListWithNewestImage(requestedPage:%20{limit:15%20offset:0}){Results%20{Name%20LastUpdated%20Size%20Platforms%20{Os%20Arch}%20%20NewestImage%20{%20Tag%20Vulnerabilities%20{MaxSeverity%20Count}%20Description%20%20Licenses%20Logo%20Title%20Source%20IsSigned%20Documentation%20Vendor%20Labels}%20DownloadCount}}}`, repoList: `/v2/_zot/ext/search?query={RepoListWithNewestImage(requestedPage:%20{limit:15%20offset:0}){Results%20{Name%20LastUpdated%20Size%20Platforms%20{Os%20Arch}%20%20NewestImage%20{%20Tag%20Vulnerabilities%20{MaxSeverity%20Count}%20Description%20%20Licenses%20Logo%20Title%20Source%20IsSigned%20Documentation%20Vendor%20Labels}%20StarCount%20DownloadCount}}}`,
detailedRepoInfo: (name) => detailedRepoInfo: (name) =>
`/v2/_zot/ext/search?query={ExpandedRepoInfo(repo:%22${name}%22){Images%20{Manifests%20{Digest%20Platform%20{Os%20Arch}%20Size}%20Vulnerabilities%20{MaxSeverity%20Count}%20Tag%20LastUpdated%20Vendor%20}%20Summary%20{Name%20LastUpdated%20Size%20Platforms%20{Os%20Arch}%20Vendors%20NewestImage%20{RepoName%20IsSigned%20Vulnerabilities%20{MaxSeverity%20Count}%20Manifests%20{Digest}%20Tag%20Title%20Documentation%20DownloadCount%20Source%20Description%20Licenses}}}}`, `/v2/_zot/ext/search?query={ExpandedRepoInfo(repo:%22${name}%22){Images%20{Manifests%20{Digest%20Platform%20{Os%20Arch}%20Size}%20Vulnerabilities%20{MaxSeverity%20Count}%20Tag%20LastUpdated%20Vendor%20IsDeletable%20}%20Summary%20{Name%20LastUpdated%20Size%20Platforms%20{Os%20Arch}%20Vendors%20NewestImage%20{RepoName%20IsSigned%20Vulnerabilities%20{MaxSeverity%20Count}%20Manifests%20{Digest}%20Tag%20Title%20Documentation%20DownloadCount%20Source%20Description%20Licenses}}}}`,
globalSearch: (searchTerm, sortCriteria, pageNumber = 1, pageSize = 10) => globalSearch: (searchTerm, sortCriteria, pageNumber = 1, pageSize = 10) =>
`/v2/_zot/ext/search?query={GlobalSearch(query:%22${searchTerm}%22,%20requestedPage:%20{limit:${pageSize}%20offset:${ `/v2/_zot/ext/search?query={GlobalSearch(query:%22${searchTerm}%22,%20requestedPage:%20{limit:${pageSize}%20offset:${
10 * (pageNumber - 1) 10 * (pageNumber - 1)
}%20sortBy:%20${sortCriteria}}%20)%20{Page%20{TotalCount%20ItemCount}%20Repos%20{Name%20LastUpdated%20Size%20Platforms%20{%20Os%20Arch%20}%20IsStarred%20IsBookmarked%20NewestImage%20{%20Tag%20Vulnerabilities%20{MaxSeverity%20Count}%20Description%20IsSigned%20Licenses%20Vendor%20Labels%20}%20DownloadCount}}}`, }%20sortBy:%20${sortCriteria}}%20)%20{Page%20{TotalCount%20ItemCount}%20Repos%20{Name%20LastUpdated%20Size%20Platforms%20{%20Os%20Arch%20}%20IsStarred%20IsBookmarked%20NewestImage%20{%20Tag%20Vulnerabilities%20{MaxSeverity%20Count}%20Description%20IsSigned%20SignatureInfo%20{%20Tool%20IsTrusted%20Author%20}%20Licenses%20Vendor%20Labels%20}%20StarCount%20DownloadCount}}}`,
image: (name) => image: (name) =>
`/v2/_zot/ext/search?query={Image(image:%20%22${name}%22){RepoName%20IsSigned%20Vulnerabilities%20{MaxSeverity%20Count}%20%20Referrers%20{MediaType%20ArtifactType%20Size%20Digest%20Annotations{Key%20Value}}%20Tag%20Manifests%20{History%20{Layer%20{Size%20Digest}%20HistoryDescription%20{CreatedBy%20EmptyLayer}}%20Digest%20ConfigDigest%20LastUpdated%20Size%20Platform%20{Os%20Arch}}%20Vendor%20Licenses%20}}` `/v2/_zot/ext/search?query={Image(image:%20%22${name}%22){RepoName%20IsSigned%20SignatureInfo%20{%20Tool%20IsTrusted%20Author%20}%20Vulnerabilities%20{MaxSeverity%20Count}%20%20Referrers%20{MediaType%20ArtifactType%20Size%20Digest%20Annotations{Key%20Value}}%20Tag%20Manifests%20{History%20{Layer%20{Size%20Digest}%20HistoryDescription%20{CreatedBy%20EmptyLayer}}%20Digest%20ConfigDigest%20LastUpdated%20Size%20Platform%20{Os%20Arch}}%20Vendor%20Licenses%20}}`
}; };
export { hosts, endpoints, sortCriteria, pageSizes }; export { hosts, endpoints, sortCriteria, pageSizes };