2024-02-29 11:59:05 -08:00
|
|
|
# `actions/attest-sbom`
|
2024-02-20 11:28:19 -08:00
|
|
|
|
2024-02-29 11:59:05 -08:00
|
|
|
Generate signed SBOM attestations for workflow artifacts. Internally powered by
|
2024-02-29 16:30:33 -08:00
|
|
|
the [@actions/attest][1] package.
|
2024-02-29 11:59:05 -08:00
|
|
|
|
|
|
|
|
Attestations bind some subject (a named artifact along with its digest) to a a
|
|
|
|
|
Software Bill of Materials (SBOM) using the [in-toto][2] format. The action
|
2024-04-22 09:22:27 -07:00
|
|
|
accepts SBOMs which have been generated by external tools. Provided SBOMs must
|
|
|
|
|
be in either the [SPDX][4] or [CycloneDX][5] JSON-serialized format.
|
2024-02-29 11:59:05 -08:00
|
|
|
|
|
|
|
|
A verifiable signature is generated for the attestation using a short-lived
|
|
|
|
|
[Sigstore][6]-issued signing certificate. If the repository initiating the
|
|
|
|
|
GitHub Actions workflow is public, the public-good instance of Sigstore will be
|
|
|
|
|
used to generate the attestation signature. If the repository is
|
|
|
|
|
private/internal, it will use the GitHub private Sigstore instance.
|
|
|
|
|
|
|
|
|
|
Once the attestation has been created and signed, it will be uploaded to the GH
|
|
|
|
|
attestations API and associated with the repository from which the workflow was
|
|
|
|
|
initiated.
|
|
|
|
|
|
|
|
|
|
Attestations can be verified using the `attestation` command in the [GitHub
|
|
|
|
|
CLI][7].
|
2024-02-20 11:28:19 -08:00
|
|
|
|
2024-02-28 12:43:56 -08:00
|
|
|
## Usage
|
2024-02-20 11:28:19 -08:00
|
|
|
|
2024-02-28 12:43:56 -08:00
|
|
|
Within the GitHub Actions workflow which builds some artifact you would like to
|
2024-02-29 11:59:05 -08:00
|
|
|
attest:
|
2024-02-20 11:28:19 -08:00
|
|
|
|
2024-02-28 12:43:56 -08:00
|
|
|
1. Ensure that the following permissions are set:
|
2024-02-20 11:28:19 -08:00
|
|
|
|
2024-02-28 12:43:56 -08:00
|
|
|
```yaml
|
|
|
|
|
permissions:
|
|
|
|
|
id-token: write
|
2024-04-23 12:33:57 -04:00
|
|
|
attestations: write
|
2024-02-28 12:43:56 -08:00
|
|
|
```
|
2024-02-20 11:28:19 -08:00
|
|
|
|
2024-02-28 12:43:56 -08:00
|
|
|
The `id-token` permission gives the action the ability to mint the OIDC token
|
2024-04-23 12:33:57 -04:00
|
|
|
necessary to request a Sigstore signing certificate. The `attestations`
|
2024-02-28 12:43:56 -08:00
|
|
|
permission is necessary to persist the attestation.
|
2024-02-20 11:28:19 -08:00
|
|
|
|
2024-02-29 11:59:05 -08:00
|
|
|
1. Add the following to your workflow after your artifact has been built:
|
2024-02-20 11:28:19 -08:00
|
|
|
|
2024-02-28 12:43:56 -08:00
|
|
|
```yaml
|
2024-02-29 11:59:05 -08:00
|
|
|
- uses: actions/attest-sbom@v1
|
2024-02-28 12:43:56 -08:00
|
|
|
with:
|
2024-02-29 11:59:05 -08:00
|
|
|
subject-path: '<PATH TO ARTIFACT>'
|
2024-04-22 09:22:27 -07:00
|
|
|
sbom-path: '<PATH TO SBOM>'
|
2024-02-28 12:43:56 -08:00
|
|
|
```
|
2024-02-20 11:28:19 -08:00
|
|
|
|
2024-02-28 12:43:56 -08:00
|
|
|
The `subject-path` parameter should identity the artifact for which you want
|
2024-04-22 09:22:27 -07:00
|
|
|
to generate an SBOM attestation. The `sbom-path` parameter should identify
|
|
|
|
|
the SBOM document to be associated with the subject.
|
2024-02-20 11:28:19 -08:00
|
|
|
|
2024-02-28 12:43:56 -08:00
|
|
|
### Inputs
|
2024-02-20 11:28:19 -08:00
|
|
|
|
2024-02-29 11:59:05 -08:00
|
|
|
See [action.yml](action.yml)
|
2024-02-20 11:28:19 -08:00
|
|
|
|
2024-02-29 11:59:05 -08:00
|
|
|
```yaml
|
|
|
|
|
- uses: actions/attest@v1
|
|
|
|
|
with:
|
|
|
|
|
# Path to the artifact serving as the subject of the attestation. Must
|
|
|
|
|
# specify exactly one of "subject-path" or "subject-digest".
|
|
|
|
|
subject-path:
|
|
|
|
|
|
|
|
|
|
# SHA256 digest of the subject for for the attestation. Must be in the form
|
|
|
|
|
# "sha256:hex_digest" (e.g. "sha256:abc123..."). Must specify exactly one
|
|
|
|
|
# of "subject-path" or "subject-digest".
|
|
|
|
|
subject-digest:
|
|
|
|
|
|
|
|
|
|
# Subject name as it should appear in the attestation. Required unless
|
|
|
|
|
# "subject-path" is specified, in which case it will be inferred from the
|
|
|
|
|
# path.
|
|
|
|
|
subject-name:
|
|
|
|
|
|
|
|
|
|
# Path to the JSON-formatted SBOM file to attest. When specified, the
|
|
|
|
|
# "scan-path" and "sbom-format" inputs are ignored.
|
|
|
|
|
sbom-path:
|
|
|
|
|
|
|
|
|
|
# Whether to push the attestation to the image registry. Requires that the
|
|
|
|
|
# "subject-name" parameter specify the fully-qualified image name and that
|
|
|
|
|
# the "subject-digest" parameter be specified. Defaults to false.
|
|
|
|
|
push-to-registry:
|
|
|
|
|
|
|
|
|
|
# The GitHub token used to make authenticated API requests. Default is
|
|
|
|
|
# ${{ github.token }}
|
|
|
|
|
github-token:
|
|
|
|
|
```
|
2024-02-20 11:28:19 -08:00
|
|
|
|
2024-02-29 11:59:05 -08:00
|
|
|
### Outputs
|
2024-02-20 11:28:19 -08:00
|
|
|
|
2024-02-29 11:59:05 -08:00
|
|
|
<!-- markdownlint-disable MD013 -->
|
2024-02-20 11:28:19 -08:00
|
|
|
|
2024-02-29 11:59:05 -08:00
|
|
|
| Name | Description | Example |
|
|
|
|
|
| ------------- | -------------------------------------------------------------- | ----------------------- |
|
|
|
|
|
| `bundle-path` | Absolute path to the file containing the generated attestation | `/tmp/attestaion.jsonl` |
|
2024-02-20 11:28:19 -08:00
|
|
|
|
2024-02-29 11:59:05 -08:00
|
|
|
<!-- markdownlint-enable MD013 -->
|
2024-02-20 11:28:19 -08:00
|
|
|
|
2024-02-29 11:59:05 -08:00
|
|
|
Attestations are saved in the JSON-serialized [Sigstore bundle][8] format.
|
2024-02-20 11:28:19 -08:00
|
|
|
|
2024-02-29 11:59:05 -08:00
|
|
|
If multiple subjects are being attested at the same time, each attestation will
|
|
|
|
|
be written to the output file on a separate line (using the [JSON Lines][9]
|
|
|
|
|
format).
|
2024-02-20 11:28:19 -08:00
|
|
|
|
2024-02-29 11:59:05 -08:00
|
|
|
## Examples
|
2024-02-20 11:28:19 -08:00
|
|
|
|
2024-02-29 11:59:05 -08:00
|
|
|
### Identify Subject and SBOM by Path
|
2024-02-20 11:28:19 -08:00
|
|
|
|
2024-02-29 11:59:05 -08:00
|
|
|
For the basic use case, simply add the `attest-sbom` action to your workflow and
|
|
|
|
|
supply the path to the artifact and SBOM for which you want to generate
|
|
|
|
|
attestation.
|
2024-02-20 11:28:19 -08:00
|
|
|
|
2024-02-28 12:43:56 -08:00
|
|
|
```yaml
|
2024-02-29 11:59:05 -08:00
|
|
|
name: build-attest
|
2024-02-28 12:43:56 -08:00
|
|
|
|
|
|
|
|
on:
|
|
|
|
|
workflow_dispatch:
|
|
|
|
|
|
|
|
|
|
jobs:
|
|
|
|
|
build:
|
|
|
|
|
permissions:
|
|
|
|
|
id-token: write
|
2024-04-23 12:33:57 -04:00
|
|
|
contents: read
|
|
|
|
|
attestations: write
|
2024-02-28 12:43:56 -08:00
|
|
|
|
|
|
|
|
steps:
|
|
|
|
|
- name: Checkout
|
|
|
|
|
uses: actions/checkout@v4
|
|
|
|
|
- name: Build artifact
|
2024-02-29 11:59:05 -08:00
|
|
|
run: make my-app
|
|
|
|
|
- name: Generate SBOM
|
|
|
|
|
run: make sbom
|
|
|
|
|
- name: Attest
|
|
|
|
|
uses: actions/attest-sbom@v1
|
2024-02-28 12:43:56 -08:00
|
|
|
with:
|
2024-02-29 11:59:05 -08:00
|
|
|
subject-path: '${{ github.workspace }}/my-app'
|
|
|
|
|
sbom-path: '${{ github.workspace }}/my-app.sbom.spdx.json'
|
2024-02-28 12:43:56 -08:00
|
|
|
```
|
|
|
|
|
|
2024-02-29 11:59:05 -08:00
|
|
|
### Identify Subjects by Wildcard
|
2024-02-20 11:28:19 -08:00
|
|
|
|
2024-02-29 11:59:05 -08:00
|
|
|
If you are generating multiple artifacts, you can generate an attestation for
|
|
|
|
|
each by using a wildcard in the `subject-path` input.
|
2024-02-20 11:28:19 -08:00
|
|
|
|
|
|
|
|
```yaml
|
2024-02-29 11:59:05 -08:00
|
|
|
- uses: actions/attest-sbom@v1
|
|
|
|
|
with:
|
|
|
|
|
subject-path: 'dist/**/my-bin-*'
|
|
|
|
|
sbom-path: '${{ github.workspace }}/my-bin.sbom.spdx.json'
|
2024-02-20 11:28:19 -08:00
|
|
|
```
|
|
|
|
|
|
2024-02-28 12:43:56 -08:00
|
|
|
For supported wildcards along with behavior and documentation, see
|
2024-02-29 11:59:05 -08:00
|
|
|
[@actions/glob][10] which is used internally to search for files.
|
2024-02-20 11:28:19 -08:00
|
|
|
|
2024-02-28 12:43:56 -08:00
|
|
|
### Container Image
|
|
|
|
|
|
2024-02-29 11:59:05 -08:00
|
|
|
When working with container images you can invoke the action with the
|
|
|
|
|
`subject-name` and `subject-digest` inputs.
|
2024-02-20 11:28:19 -08:00
|
|
|
|
2024-02-28 12:43:56 -08:00
|
|
|
If you want to publish the attestation to the container registry with the
|
|
|
|
|
`push-to-registry` option, it is important that the `subject-name` specify the
|
|
|
|
|
fully-qualified image name (e.g. "ghcr.io/user/app" or
|
|
|
|
|
"acme.azurecr.io/user/app"). Do NOT include a tag as part of the image name --
|
|
|
|
|
the specific image being attested is identified by the supplied digest.
|
2024-02-20 11:28:19 -08:00
|
|
|
|
2024-02-28 12:43:56 -08:00
|
|
|
> **NOTE**: When pushing to Docker Hub, please use "index.docker.io" as the
|
|
|
|
|
> registry portion of the image name.
|
2024-02-20 11:28:19 -08:00
|
|
|
|
|
|
|
|
```yaml
|
2024-02-29 11:59:05 -08:00
|
|
|
name: build-attested-image
|
2024-02-28 12:43:56 -08:00
|
|
|
|
|
|
|
|
on:
|
|
|
|
|
push:
|
|
|
|
|
branches: [main]
|
|
|
|
|
|
|
|
|
|
jobs:
|
|
|
|
|
build:
|
|
|
|
|
runs-on: ubuntu-latest
|
|
|
|
|
permissions:
|
|
|
|
|
id-token: write
|
|
|
|
|
packages: write
|
2024-04-23 12:33:57 -04:00
|
|
|
contents: read
|
|
|
|
|
attestations: write
|
2024-02-28 12:43:56 -08:00
|
|
|
env:
|
|
|
|
|
REGISTRY: ghcr.io
|
|
|
|
|
IMAGE_NAME: ${{ github.repository }}
|
|
|
|
|
|
|
|
|
|
steps:
|
|
|
|
|
- name: Checkout
|
|
|
|
|
uses: actions/checkout@v4
|
|
|
|
|
- name: Login to GitHub Container Registry
|
|
|
|
|
uses: docker/login-action@v3
|
|
|
|
|
with:
|
|
|
|
|
registry: ${{ env.REGISTRY }}
|
|
|
|
|
username: ${{ github.actor }}
|
|
|
|
|
password: ${{ secrets.GITHUB_TOKEN }}
|
|
|
|
|
- name: Build and push image
|
|
|
|
|
id: push
|
|
|
|
|
uses: docker/build-push-action@v5.0.0
|
|
|
|
|
with:
|
|
|
|
|
context: .
|
|
|
|
|
push: true
|
|
|
|
|
tags: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}:latest
|
2024-02-29 11:59:05 -08:00
|
|
|
- name: Generate SBOM
|
|
|
|
|
run: make sbom
|
|
|
|
|
- name: Attest
|
|
|
|
|
uses: actions/attest-sbom@v1
|
|
|
|
|
id: attest
|
2024-02-28 12:43:56 -08:00
|
|
|
with:
|
|
|
|
|
subject-name: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}
|
|
|
|
|
subject-digest: ${{ steps.push.outputs.digest }}
|
2024-02-29 11:59:05 -08:00
|
|
|
sbom-path: 'sbom.cyclonedx.json'
|
|
|
|
|
push-to-registry: true
|
2024-02-20 11:28:19 -08:00
|
|
|
```
|
|
|
|
|
|
2024-02-29 11:59:05 -08:00
|
|
|
[1]: https://github.com/actions/toolkit/tree/main/packages/attest
|
|
|
|
|
[2]: https://github.com/in-toto/attestation/tree/main/spec/v1
|
|
|
|
|
[4]: https://spdx.dev/
|
|
|
|
|
[5]: https://cyclonedx.org/
|
|
|
|
|
[6]: https://www.sigstore.dev/
|
|
|
|
|
[7]: https://cli.github.com/
|
|
|
|
|
[8]:
|
|
|
|
|
https://github.com/sigstore/protobuf-specs/blob/main/protos/sigstore_bundle.proto
|
|
|
|
|
[9]: https://jsonlines.org/
|
|
|
|
|
[10]: https://github.com/actions/toolkit/tree/main/packages/glob#patterns
|