Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(KONFLUX-3935) Add pruning check to FBC pipeline #1744

Open
wants to merge 1 commit into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions CODEOWNERS
Validating CODEOWNERS rules …
Original file line number Diff line number Diff line change
Expand Up @@ -58,6 +58,7 @@
/task/fbc-fips-check @konflux-ci/integration-service-maintainers
/task/fbc-fips-check-oci-ta @konflux-ci/integration-service-maintainers
/task/fbc-related-image-check @konflux-ci/integration-service-maintainers
/task/fbc-target-index-pruning-check @konflux-ci/integration-service-maintainers
/task/fbc-validation @konflux-ci/integration-service-maintainers
/task/inspect-image @konflux-ci/integration-service-maintainers
/task/sbom-json-check @konflux-ci/integration-service-maintainers
Expand Down
16 changes: 14 additions & 2 deletions pipelines/fbc-builder/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -84,6 +84,13 @@ This pipeline is pushed as a Tekton bundle to [quay.io](https://quay.io/reposito
|IMAGE_URL| Fully qualified image name.| None| '$(tasks.build-image-index.results.IMAGE_URL)'|
|POLICY_DIR| Path to directory containing Conftest policies.| /project/repository/| |
|POLICY_NAMESPACE| Namespace for Conftest policy.| required_checks| |
### fbc-target-index-pruning-check:0.1 task parameters
|name|description|default value|already set by|
|---|---|---|---|
|IMAGE_DIGEST| Image digest.| None| '$(tasks.build-image-index.results.IMAGE_DIGEST)'|
|IMAGE_URL| Fully qualified image name.| None| '$(tasks.build-image-index.results.IMAGE_URL)'|
|OCP_VERSION| OCP version.| None| '$(tasks.validate-fbc.results.OCP_VERSION)'|
|TARGET_INDEX| Image name of target index, minus tag.| registry.redhat.io/redhat/redhat-operator-index| 'registry.redhat.io/redhat/redhat-operator-index'|
### git-clone-oci-ta:0.1 task parameters
|name|description|default value|already set by|
|---|---|---|---|
Expand Down Expand Up @@ -150,9 +157,9 @@ This pipeline is pushed as a Tekton bundle to [quay.io](https://quay.io/reposito
|name|description|used in params (taskname:taskrefversion:taskparam)
|---|---|---|
|IMAGES| List of all referenced image manifests| |
|IMAGE_DIGEST| Digest of the image just built| deprecated-base-image-check:0.4:IMAGE_DIGEST ; validate-fbc:0.1:IMAGE_DIGEST|
|IMAGE_DIGEST| Digest of the image just built| deprecated-base-image-check:0.4:IMAGE_DIGEST ; validate-fbc:0.1:IMAGE_DIGEST ; fbc-target-index-pruning-check:0.1:IMAGE_DIGEST|
|IMAGE_REF| Image reference of the built image containing both the repository and the digest| |
|IMAGE_URL| Image repository and tag where the built image was pushed| show-sbom:0.1:IMAGE_URL ; deprecated-base-image-check:0.4:IMAGE_URL ; apply-tags:0.1:IMAGE ; validate-fbc:0.1:IMAGE_URL|
|IMAGE_URL| Image repository and tag where the built image was pushed| show-sbom:0.1:IMAGE_URL ; deprecated-base-image-check:0.4:IMAGE_URL ; apply-tags:0.1:IMAGE ; validate-fbc:0.1:IMAGE_URL ; fbc-target-index-pruning-check:0.1:IMAGE_URL|
|SBOM_BLOB_URL| Reference of SBOM blob digest to enable digest-based verification from provenance| |
### buildah-remote-oci-ta:0.3 task results
|name|description|used in params (taskname:taskrefversion:taskparam)
Expand All @@ -166,6 +173,10 @@ This pipeline is pushed as a Tekton bundle to [quay.io](https://quay.io/reposito
|---|---|---|
|IMAGES_PROCESSED| Images processed in the task.| |
|TEST_OUTPUT| Tekton task test output.| |
### fbc-target-index-pruning-check:0.1 task results
|name|description|used in params (taskname:taskrefversion:taskparam)
|---|---|---|
|TEST_OUTPUT| Tekton task test output.| |
### git-clone-oci-ta:0.1 task results
|name|description|used in params (taskname:taskrefversion:taskparam)
|---|---|---|
Expand All @@ -189,6 +200,7 @@ This pipeline is pushed as a Tekton bundle to [quay.io](https://quay.io/reposito
|name|description|used in params (taskname:taskrefversion:taskparam)
|---|---|---|
|IMAGES_PROCESSED| Images processed in the task.| |
|OCP_VERSION| OCP version derived from base image.| fbc-target-index-pruning-check:0.1:OCP_VERSION|
|RELATED_IMAGES_DIGEST| Digest for attached json file containing related images| |
|RELATED_IMAGE_ARTIFACT| The Trusted Artifact URI pointing to the artifact with the related images for the FBC fragment.| |
|TEST_OUTPUT| Tekton task test output.| |
Expand Down
22 changes: 22 additions & 0 deletions pipelines/fbc-builder/patch.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -98,3 +98,25 @@
value: $(tasks.build-image-index.results.IMAGE_URL)
- name: IMAGE_DIGEST
value: $(tasks.build-image-index.results.IMAGE_DIGEST)
- op: add
path: /spec/tasks/-
value:
name: fbc-target-index-pruning-check
when:
- input: $(params.skip-checks)
operator: in
values: ["false"]
runAfter:
- validate-fbc
taskRef:
name: fbc-target-index-pruning-check
version: "0.1"
params:
- name: IMAGE_URL
value: $(tasks.build-image-index.results.IMAGE_URL)
- name: IMAGE_DIGEST
value: $(tasks.build-image-index.results.IMAGE_DIGEST)
- name: TARGET_INDEX
value: registry.redhat.io/redhat/redhat-operator-index
- name: OCP_VERSION
value: $(tasks.validate-fbc.results.OCP_VERSION)
1 change: 1 addition & 0 deletions renovate.json
Original file line number Diff line number Diff line change
Expand Up @@ -89,6 +89,7 @@
"task/fbc-fips-check-oci-ta/**",
"task/fbc-fips-check/**",
"task/fbc-related-image-check/**",
"task/fbc-target-index-pruning-check/**",
"task/fbc-validation/**",
"task/fips-operator-bundle-check-oci-ta/**",
"task/fips-operator-bundle-check/**",
Expand Down
33 changes: 33 additions & 0 deletions task/fbc-target-index-pruning-check/0.1/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
# fbc-target-index-pruning-check task

## Description:
This task ensures file-based catalog (FBC) components do not remove previously released versions of operators from a target catalog, specified in the `TARGET_INDEX` parameter, which by default points to the Red Hat production Index Image `registry.redhat.io/redhat/redhat-operator-index`. Image pull credentials are required for `registry.redhat.io` or the registry you specify in `TARGET_INDEX`.

### What this check does:
- Runs `opm render` on both FBC fragment and TARGET_INDEX:OCP_VERSION images.
- Compares the channel data of the FBC fragment and target index.
- Checks if the FBC fragment will remove channels or channel entries previously added to the target index.


## Params:

| name | description | default value |
|--------------|----------------------------------|---------|
| IMAGE_URL | Fully qualified image name. | |
| IMAGE_DIGEST | Image digest. | |
| TARGET_IMAGE | Image name of target index, minus tag. | `registry.redhat.io/redhat/redhat-operator-index` |
| OCP_VERSION | OCP version of FBC image. | |

## Results:

| name | description |
|--------------------|---------------------------|
| TEST_OUTPUT | Tekton task test output. |

## Source repository for image:
https://github.com/konflux-ci/konflux-test

## Additional links:
https://olm.operatorframework.io/docs/reference/file-based-catalogs/
https://github.com/containers/skopeo
https://docs.openshift.com/container-platform/4.12/cli_reference/opm/cli-opm-install.html
Original file line number Diff line number Diff line change
@@ -0,0 +1,199 @@
apiVersion: tekton.dev/v1
kind: Task
metadata:
labels:
app.kubernetes.io/version: "0.1"
annotations:
tekton.dev/pipelines.minVersion: "0.12.1"
tekton.dev/tags: "konflux"
name: fbc-target-index-pruning-check
spec:
description: >-
Ensures file-based catalog (FBC) components do not remove versions of operators already added to a released catalog.
params:
- name: IMAGE_URL
description: Fully qualified image name.
- name: IMAGE_DIGEST
description: Image digest.
- name: TARGET_INDEX
description: Image name of target index, minus tag.
default: registry.redhat.io/redhat/redhat-operator-index
- name: OCP_VERSION
description: OCP version.
results:
- name: TEST_OUTPUT
description: Tekton task test output.
steps:
- name: check-if-fragment-prunes-target-index
image: quay.io/redhat-appstudio/konflux-test:v1.4.9@sha256:eee855e60b437d9a55a30e63f2eb7f95d9fd6d3b111c32cac8730c9b7a071394
# per https://kubernetes.io/docs/concepts/containers/images/#imagepullpolicy-defaulting
# the cluster will set imagePullPolicy to IfNotPresent
workingDir: /var/workdir/fbc-pruning
env:
- name: IMAGE_URL
value: $(params.IMAGE_URL)
- name: IMAGE_DIGEST
value: $(params.IMAGE_DIGEST)
- name: TARGET_INDEX
value: $(params.TARGET_INDEX)
- name: OCP_VERSION
value: $(params.OCP_VERSION)
securityContext:
runAsUser: 0
capabilities:
add:
- SETFCAP
computeResources:
limits:
memory: 4Gi
requests:
memory: 512Mi
cpu: 10m
script: |
#!/usr/bin/env bash
set -euo pipefail
# shellcheck source=/dev/null
source /utils.sh
trap 'handle_error $(results.TEST_OUTPUT.path)' EXIT

IMAGE_URL="${IMAGE_URL}@${IMAGE_DIGEST}"
# Given a tag and a the digest in the IMAGE_URL we opt to use the digest alone
# this is because containers/image currently doesn't support image references
# that contain both. See https://github.com/containers/image/issues/1736
if [[ "${IMAGE_URL}" == *":"*"@"* ]]; then
IMAGE_URL="${IMAGE_URL/:*@/@}"
fi

### Check if TARGET_INDEX is defined
if [ -z "${TARGET_INDEX}" ]; then
echo "TARGET_INDEX is not defined."
note="Task $(context.task.name) failed: The TARGET_INDEX is not defined."
TEST_OUTPUT=$(make_result_json -r ERROR -t "$note")
echo "${TEST_OUTPUT}" | tee "$(results.TEST_OUTPUT.path)"
exit 0
fi

### Check if OCP_VERSION is defined
if [ -z "${OCP_VERSION}" ]; then
echo "OCP_VERSION is not defined."
note="Task $(context.task.name) failed: The OCP_VERSION is not defined."
TEST_OUTPUT=$(make_result_json -r ERROR -t "$note")
echo "${TEST_OUTPUT}" | tee "$(results.TEST_OUTPUT.path)"
exit 0
fi

### Run opm render for FBC fragment and target index
rendered_fbc_image=/tmp/opm-render-fbc-fragment.json
rendered_target_index=/tmp/opm-render-target-index.json

echo "Rendering FBC image: ${IMAGE_URL}"
opm render "${IMAGE_URL}" | tr -d '\000-\031' > "${rendered_fbc_image}"
if [[ ! -f "${rendered_fbc_image}" ]]; then
note="Task $(context.task.name) failed: Unable to render the fragment FBC image: ${IMAGE_URL}"
echo "${note}"
TEST_OUTPUT=$(make_result_json -r ERROR -t "${note}")
exit 0
fi

target_index_pullspec="${TARGET_INDEX}:v${OCP_VERSION}"
echo "Rendering target index: ${target_index_pullspec}"
opm render "${target_index_pullspec}" | tr -d '\000-\031' > "${rendered_target_index}"
if [[ ! -f "${rendered_target_index}" ]]; then
note="Task $(context.task.name) failed: Unable to render the fragment target index image: ${IMAGE_URL}"
echo "${note}"
TEST_OUTPUT=$(make_result_json -r ERROR -t "${note}")
exit 0
fi

failure_num=0
TESTPASSED=true

fbc_channels=/tmp/olm-channels-fbc-image.json
ndx_channels=/tmp/olm-channels-target-index.json

### Filter out channels and channel entries from FBC fragment render
jq -s 'map(select(.schema == "olm.channel")) | reduce .[] as $obj ([]; . += [{package: $obj.package, channel: $obj.name, entries: [$obj.entries[].name]}])' "${rendered_fbc_image}" > "${fbc_channels}"

echo ""
echo "Channels defined in FBC fragment:"
jq '.' "${fbc_channels}"
echo ""

### Filter out channels and channel entries from target index render
jq -s 'map(select(.schema == "olm.channel")) | reduce .[] as $obj ([]; . += [{package: $obj.package, channel: $obj.name, entries: [$obj.entries[].name]}])' "${rendered_target_index}" > "${ndx_channels}"

### Get the package(s) the fragment is configuring
mapfile -t fbc_pkgs < <(jq -r '.[].package ' "${fbc_channels}" | sort -u)
if (( ${#fbc_pkgs[@]} < 1 )); then
note="Task $(context.task.name) failed: No OLM packages detected in FBC fragment."
echo "${note}"
TEST_OUTPUT=$(make_result_json -r ERROR -t "${note}")
exit 0
fi

### Get packages in target index
mapfile -t ndx_pkgs < <(jq -r '.[].package ' "${ndx_channels}" | sort -u)
if [[ ${#ndx_pkgs[@]} -lt 1 ]]; then
note="Task $(context.task.name) failed: No OLM packages detected in target index."
echo "${note}"
TEST_OUTPUT=$(make_result_json -r ERROR -t "${note}")
exit 0
fi

### Test packages in the FBC fragment that already exist in the target index.
pkgs_to_test=()
for pkg in "${fbc_pkgs[@]}"; do
if echo "${ndx_pkgs[@]}" | grep -Fwq "${pkg}"; then
pkgs_to_test+=("${pkg}")
fi
done

if (( ${#pkgs_to_test[@]} > 0 )); then
for pkg in "${pkgs_to_test[@]}"; do
channels_to_test=()
mapfile -t fbc_channel_names < <(jq -r --arg p "${pkg}" '.[] | select(.package == $p) | .channel' ${fbc_channels})
mapfile -t ndx_channel_names < <(jq -r --arg p "${pkg}" '.[] | select(.package == $p) | .channel' ${ndx_channels})

### Check for removed channels
for chan in "${ndx_channel_names[@]}"; do
if echo "${fbc_channel_names[@]}" | grep -Fwq "${chan}"; then
channels_to_test+=("${chan}")
else
echo "!FAILURE! - FBC fragment prunes entire ${pkg}.${chan} channel."
TESTPASSED=false
failure_num=$((failure_num + 1))
fi
done

### Check each channel for removed entries
if (( ${#channels_to_test[@]} > 0 )); then
for chan in "${channels_to_test[@]}"; do
echo ""
echo "TARGET INDEX ${pkg}.${chan} channel:"
jq -r --arg p "${pkg}" --arg c "${chan}" '.[] | select(.package == $p and .channel == $c)' "${ndx_channels}"
echo ""
mapfile -t ndx_entries < <(jq -r --arg p "${pkg}" --arg c "${chan}" '.[] | select(.package == $p and .channel == $c) | .entries[]' "${ndx_channels}")
mapfile -t fbc_entries < <(jq -r --arg p "${pkg}" --arg c "${chan}" '.[] | select(.package == $p and .channel == $c) | .entries[]' "${fbc_channels}")

for entry in "${ndx_entries[@]}"; do
if ! echo "${fbc_entries[@]}" | grep -Fwq "${entry}"; then
echo "!FAILURE! - FBC fragment prunes ${entry} from ${pkg}.${chan} channel."
failure_num=$((failure_num + 1))
TESTPASSED=false
fi
done
done
fi
done
else
echo "FBC fragment is not modifying any existing channels in the target index."
fi

note="Task $(context.task.name) completed: Check result for task result."
if [[ $TESTPASSED == false ]]; then
ERROR_OUTPUT=$(make_result_json -r FAILURE -f $failure_num -s 0 -t "${note}")
echo "${ERROR_OUTPUT}" | tee "$(results.TEST_OUTPUT.path)"
else
TEST_OUTPUT=$(make_result_json -r SUCCESS -s 1 -t "${note}")
echo "${TEST_OUTPUT}" | tee "$(results.TEST_OUTPUT.path)"
fi
1 change: 1 addition & 0 deletions task/validate-fbc/0.1/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -16,4 +16,5 @@ Ensures file-based catalog (FBC) components are uniquely linted for proper const
|TEST_OUTPUT|Tekton task test output.|
|RELATED_IMAGES_DIGEST|Digest for attached json file containing related images|
|IMAGES_PROCESSED|Images processed in the task.|
|OCP_VERSION|OCP version of FBC image.|

3 changes: 3 additions & 0 deletions task/validate-fbc/0.1/validate-fbc.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,8 @@ spec:
description: Digest for attached json file containing related images
- name: IMAGES_PROCESSED
description: Images processed in the task.
- name: OCP_VERSION
description: OCP version derived from base image.
volumes:
- name: shared
emptyDir: {}
Expand Down Expand Up @@ -386,6 +388,7 @@ spec:
# extracts major digits and filters out any leading alphabetic characters, for e.g. 'v4' --> '4'
OCP_VER_MAJOR=$(echo "${OCP_VER_FROM_BASE}" | cut -d '.' -f 1 | sed "s/^[a-zA-Z]*//")
OCP_VER_MINOR=$(echo "${OCP_VER_FROM_BASE}" | cut -d '.' -f 2)
echo -n "${OCP_VER_MAJOR}.${OCP_VER_MINOR}" > "$(results.OCP_VERSION.path)"

RUN_OCP_VERSION_VALIDATION="false"
digits_regex='^[0-9]*$'
Expand Down
Loading