-
Notifications
You must be signed in to change notification settings - Fork 140
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
OCI-attach reports from
clair-scan
0.2 Task
This attaches the "clair"-formatted output of the `clair-action` command to the scanned image. Contrary to the `sast-snyk-check` Task only a single variant of attachment method is supported, based on the registry support. For quay.io OCI Distribution 1.1. Referrers API will be used. The `clair-action` needs to be run twice as the Rego rules executed in the `conftest-vulnerabilities` require the "quay" format, which does not include any date information, whereas the (future) EC policy Rego rules require the "clair" format, which does. Resolves: https://issues.redhat.com/browse/EC-837
- Loading branch information
Showing
2 changed files
with
204 additions
and
3 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,156 @@ | ||
#!/bin/env bash | ||
|
||
set -o errexit | ||
set -o pipefail | ||
set -o nounset | ||
|
||
eval "$(shellspec - -c) exit 1" | ||
|
||
task_path=clair-scan.yaml | ||
|
||
if [[ -f "../${task_path}" ]]; then | ||
task_path="../${task_path}" | ||
fi | ||
|
||
|
||
extract_script() { | ||
script="$(mktemp --tmpdir script_XXXXXXXXXX.sh)" | ||
yq -r ".spec.steps[] | select(.name == \"$1\").script" "${task_path}" > "${script}" | ||
chmod +x "${script}" | ||
|
||
echo "${script}" | ||
} | ||
|
||
# array containing files/directories to remove on test exit | ||
cleanup=() | ||
trap 'rm -rf "${cleanup[@]}"' EXIT | ||
|
||
# Extract the get-vulnerabilities Step script so we can test it | ||
get_vulnerabilities_script="$(extract_script get-vulnerabilities)" | ||
cleanup+=("${get_vulnerabilities_script}") | ||
|
||
testdir() { | ||
testdir="$(mktemp -d)" && cleanup+=("${testdir}") && cd "${testdir}" | ||
|
||
AfterEach 'rm -rf "$testdir"' | ||
} | ||
|
||
Describe "get vulnerabilities" | ||
BeforeEach testdir | ||
|
||
export IMAGE_URL=registry.io/repository/image:tag | ||
export IMAGE_DIGEST=sha256:f0cacc1a | ||
|
||
It "generates reports and images-processed.json" | ||
Mock clair-action | ||
clair_action_args+=("$*") | ||
%preserve clair_action_args | ||
# expecting the --format parameter to be the last one | ||
echo "report in ${clair_action_args[-1]#*--format=} format" | ||
End | ||
echo "sha256:f0cacc1a" > image-manifest-amd64.sha | ||
|
||
When call "${get_vulnerabilities_script}" | ||
The output should eq "Running clair-action on amd64 image manifest. | ||
report in quay format" | ||
The contents of file "images-processed.json" should equal '{"image": {"pullspec": "registry.io/repository/image:tag", "digests": ["sha256:f0cacc1a"]}}' | ||
The contents of file "clair-result-amd64.json" should equal 'report in quay format' | ||
The contents of file "clair-report-amd64.json" should equal 'report in clair format' | ||
The variable clair_action_args[@] should eq "report --image-ref=registry.io/repository/image@sha256:f0cacc1a --db-path=/tmp/matcher.db --format=quay "\ | ||
"report --image-ref=registry.io/repository/image@sha256:f0cacc1a --db-path=/tmp/matcher.db --format=clair" | ||
End | ||
|
||
It "fails in clair-action quay report" | ||
Mock clair-action | ||
clair_action_args+=("$*") | ||
%preserve clair_action_args | ||
[[ "$*" == *--format=quay* ]] && echo "didn't work out" && exit 1 | ||
End | ||
echo "sha256:f0cacc1a" > image-manifest-amd64.sha | ||
|
||
When call "${get_vulnerabilities_script}" | ||
The output should eq "Running clair-action on amd64 image manifest. | ||
didn't work out" | ||
The contents of file "images-processed.json" should equal '{"image": {"pullspec": "registry.io/repository/image:tag", "digests": ["sha256:f0cacc1a"]}}' | ||
The contents of file "clair-result-amd64.json" should equal "didn't work out" | ||
The file "clair-report-amd64.json" should not exist | ||
The variable clair_action_args[@] should eq "report --image-ref=registry.io/repository/image@sha256:f0cacc1a --db-path=/tmp/matcher.db --format=quay" | ||
End | ||
End | ||
|
||
# Extract the oci-attach-report Step script so we can test it | ||
oci_attach_report_script="$(extract_script oci-attach-report)" | ||
cleanup+=("${oci_attach_report_script}") | ||
|
||
oras_attach() { | ||
echo "attach --no-tty --registry-config $HOME/auth.json --artifact-type application/vnd.redhat.clair-report+json registry.io/repository/image@$1 $2:application/vnd.redhat.clair-report+json" | ||
} | ||
|
||
Describe "OCI attach report" | ||
BeforeEach testdir | ||
|
||
export IMAGE_URL=registry.io/repository/image:tag | ||
|
||
It "skips attachments if no reports generated" | ||
Mock select-oci-auth | ||
echo select-oci-auth should not be called | ||
End | ||
|
||
Mock oras | ||
echo oras should not be called | ||
End | ||
|
||
When call "${oci_attach_report_script}" | ||
The output should eq "No Clair reports generated. Skipping upload." | ||
End | ||
|
||
It "attaches for single architecture" | ||
export HOME="${testdir}" | ||
|
||
Mock select-oci-auth | ||
echo selected auth | ||
End | ||
|
||
Mock oras | ||
oras_args+=("$*") | ||
%preserve oras_args | ||
End | ||
|
||
echo "sha256:f0cacc1a" > image-manifest-amd64.sha | ||
touch clair-report-amd64.json | ||
|
||
When call "${oci_attach_report_script}" | ||
The output should eq "Selecting auth | ||
Attaching clair-report-amd64.json to registry.io/repository/image@sha256:f0cacc1a" | ||
The contents of file "auth.json" should equal "selected auth" | ||
The variable oras_args[@] should eq "$(oras_attach sha256:f0cacc1a clair-report-amd64.json)" | ||
End | ||
|
||
It "attaches for multiple architecture" | ||
export HOME="${testdir}" | ||
|
||
Mock select-oci-auth | ||
echo selected auth | ||
End | ||
|
||
Mock oras | ||
oras_args+=("$*") | ||
%preserve oras_args | ||
End | ||
|
||
echo "sha256:f0cacc1a" > image-manifest-amd64.sha | ||
echo "sha256:cc1af0ca" > image-manifest-arm64.sha | ||
echo "sha256:f01acacc" > image-manifest-ppc64le.sha | ||
touch clair-report-{amd64,arm64,ppc64le}.json | ||
|
||
When call "${oci_attach_report_script}" | ||
The output should eq "Selecting auth | ||
Attaching clair-report-amd64.json to registry.io/repository/image@sha256:f0cacc1a | ||
Attaching clair-report-arm64.json to registry.io/repository/image@sha256:cc1af0ca | ||
Attaching clair-report-ppc64le.json to registry.io/repository/image@sha256:f01acacc" | ||
The contents of file "auth.json" should equal "selected auth" | ||
The variable oras_args[@] should eq "$(oras_attach sha256:f0cacc1a clair-report-amd64.json) "\ | ||
"$(oras_attach sha256:cc1af0ca clair-report-arm64.json) "\ | ||
"$(oras_attach sha256:f01acacc clair-report-ppc64le.json)" | ||
End | ||
End |