Verification of SLSA provenance
Overview
What is SLSA?
Supply chain Levels for Software Artifacts, or SLSA (salsa),
is a security framework, a check-list of standards and controls to prevent
tampering, improve integrity, and secure packages and infrastructure in your
projects, businesses or enterprises.
SLSA defines an incrementially adoptable set of levels which are defined in
terms of increasing compliance and assurance. SLSA levels are like a common
language to talk about how secure software, supply chains and their component
parts really are.
What is provenance?
Provenance is information, or metadata, about how a software artifact was
created. This could include information about what source code, build system,
and build steps were used, as well as who and why the build was initiated.
Provenance can be used to determine the authenticity and trustworthiness of
software artifacts that you use.
As part of the framework, SLSA defines a
provenance format which can be used hold this
metadata.
What is slsa-verifier?
slsa-verifier is a tool for verifying
SLSA provenance that was generated by CI/CD
builders. slsa-verifier verifies the provenance by verifying the cryptographic
signatures on provenance to make sure it was created by the expected builder.
It then verifies that various values such as the builder id, source code
repository, ref (branch or tag) matches the expected values.
It currently supports verifying provenance generated by:
- SLSA generator
- Google Cloud Build (GCB).
Installation
Available options
Verification for GitHub builders
Verification for Google Cloud Build
Known Issues
Technical design
Installation
You have two options to install the verifier.
Compilation from source
Option 1: Install via go
$ go install github.com/slsa-framework/slsa-verifier/cli/slsa-verifier@v1.3.2
$ slsa-verifier <options>
Option 2: Compile manually
$ git clone git@github.com:slsa-framework/slsa-verifier.git
$ cd slsa-verifier && git checkout v2.0.0
$ go run ./cli/slsa-verifier <options>
Download the binary
Download the binary from the latest release at https://github.com/slsa-framework/slsa-verifier/releases/tag/v2.0.0
Download the SHA256SUM.md.
Verify the checksum:
$ sha256sum -c --strict SHA256SUM.md
slsa-verifier-linux-amd64: OK
Available options
We currently support artifact verification (for binary blobs) and container images.
Option list
Below is a list of options currently supported for binary blobs and container images. Note that signature verification is handled seamlessly without the need for developers to manipulate public keys. See Available options for details on the options exposed to validate the provenance.
$ git clone git@github.com:slsa-framework/slsa-verifier.git
$ go run ./cli/slsa-verifier/ verify-artifact --help
Verifies SLSA provenance on an artifact blob
Usage:
slsa-verifier verify-artifact [flags]
Flags:
--build-workflow-input map[] [optional] a workflow input provided by a user at trigger time in the format 'key=value'. (Only for 'workflow_dispatch' events). (default map[])
--builder-id string [optional] the unique builder ID who created the provenance
-h, --help help for verify-artifact
--print-provenance [optional] print the verified provenance to stdout
--provenance-path string path to a provenance file
--source-branch string [optional] expected branch the binary was compiled from
--source-tag string [optional] expected tag the binary was compiled from
--source-uri string expected source repository that should have produced the binary, e.g. github.com/some/repo
--source-versioned-tag string [optional] expected version the binary was compiled from. Uses semantic version to match the tag
Option details
The following options are available:
Option |
Description |
Support |
source-uri |
Expects a source, for e.g. github.com/org/repo . |
All builders |
source-branch |
Expects a branch like main or dev . Not supported for all GitHub Workflow triggers. |
GitHub builders |
source-tag |
Expects a tag like v0.0.1 . Verifies exact tag used to create the binary. Supported for new tag and release triggers. |
GitHub builders |
source-versioned-tag |
Like tag , but verifies using semantic versioning. |
GitHub builders |
build-workflow-input |
Expects key-value pairs like key=value to match against inputs for GitHub Actions workflow_dispatch triggers. |
GitHub builders |
Verification for GitHub builders
Artifacts
To verify an artifact, run the following command:
$ slsa-verifier verify-artifact slsa-test-linux-amd64 \
--provenance-path slsa-test-linux-amd64.intoto.jsonl \
--source-uri github.com/slsa-framework/slsa-test \
--source-tag v1.0.3
Verified signature against tlog entry index 3189970 at URL: https://rekor.sigstore.dev/api/v1/log/entries/206071d5ca7a2346e4db4dcb19a648c7f13b4957e655f4382b735894059bd199
Verified build using builder https://github.com/slsa-framework/slsa-github-generator/.github/workflows/builder_go_slsa3.yml@refs/tags/v1.2.0 at commit 5bb13ef508b2b8ded49f9264d7712f1316830d10
PASSED: Verified SLSA provenance
The verified in-toto statement may be written to stdout with the --print-provenance
flag to pipe into policy engines.
Only GitHub URIs are supported with the --source-uri
flag. A tag should not be specified, even if the provenance was built at some tag. If you intend to do source versioning validation, use --print-provenance
and inspect the commit SHA of the config source or materials.
Containers
This is WIP and currently not supported.
Verification for Google Cloud Build
Artifacts
This is WIP and currently not supported.
Containers
To verify a container image, you need to pass a container image name that is immutable by providing its digest, in order to avoid TOCTOU attacks.
Run the commands below:
$ IMAGE=laurentsimon/slsa-gcb-v0.3:test
Download the provenance:
$ gcloud artifacts docker images describe $IMAGE --format json --show-provenance > provenance.json
Get the digest for your container without pulling it using the crane command:
$ IMAGE="${IMAGE}@"$(crane digest "${IMAGE}")
Verify the image:
$ slsa-verifier verify-image "$IMAGE" \
--provenance-path provenance.json \
--source-uri github.com/laurentsimon/gcb-tests \
--builder-id=https://cloudbuild.googleapis.com/GoogleHostedWorker
PASSED: Verified SLSA provenance
The verified in-toto statement may be written to stdout with the --print-provenance
flag to pipe into policy engines.
Note that --source-uri
supports GitHub repository URIs like github.com/$OWNER/$REPO
when the build was enabled with a Cloud Build GitHub trigger. Otherwise, the build provenance will contain the name of the Cloud Storage bucket used to host the source files, usually of the form gs://[PROJECT_ID]_cloudbuild/source
(see Running build). We recommend using GitHub triggers in order to preserve the source provenance and valiate that the source came from an expected, version-controlled repository. You may match on the fully-qualified tar like gs://[PROJECT_ID]_cloudbuild/source/1665165360.279777-955d1904741e4bbeb3461080299e929a.tgz
.
Known Issues
tuf: invalid key
This will occur only when verifying provenance generated with GitHub Actions.
Affected versions: v1.3.0-v1.3.1, v1.2.0-v1.2.1, v1.1.0-v1.1.2, v1.0.0-v1.0.4
slsa-verifier
will fail with the following error:
FAILED: SLSA verification failed: could not find a matching valid signature entry: got unexpected errors unable to initialize client, local cache may be corrupt: tuf: invalid key: unable to fetch Rekor public keys from TUF repository
This issue is tracked by issue #325. You must update to the newest patch versions of each minor release to fix this issue.
Technical design
Blog post
Find our blog post series here.
Specifications
For a more in-depth technical dive, read the SPECIFICATIONS.md.
TOCTOU attacks
As explained on Wikipedia, a "time-of-check to time-of-use (TOCTOU) is a class of software bugs caused by a race condition involving the checking of the state of a part of a system and the use of the results of that check".
In the context of provenance verification, imagine you verify a container refered to via a mutable image image:tag
. The verification succeeds and verifies the corresponding hash is sha256:abcdef...
. After verification, you pull and run the image using docker run image:tag
. An attacker could have altered the image between the verification step and the run step. To mitigate this attack, we ask users to always pass an immutable reference to the artifact they verify.