A library to convert between Sigstore Bundles and PEP-740 Attestation objects
Project description
pypi-attestations
A library to generate and convert between Sigstore Bundles and PEP 740 Attestation objects.
[!IMPORTANT] This library is an implementation detail within the reference implementation of PEP 740. Most users should not need to interact with it directly; see the PyPI documentation for full details.
Installation
python -m pip install pypi-attestations
Usage as a library
See the full API documentation here.
Signing and verification
Use these APIs to create a PEP 740-compliant Attestation object by signing a Python artifact
(i.e: sdist or wheel files), and to verify an Attestation object against a Python artifact.
from pathlib import Path
from pypi_attestations import Attestation, Distribution
from sigstore.oidc import Issuer
from sigstore.models import ClientTrustConfig
from sigstore.sign import SigningContext
from sigstore.verify import Verifier, policy
dist = Distribution.from_file(Path("test_package-0.0.1-py3-none-any.whl"))
# Sign a Python artifact
trust_config = ClientTrustConfig.production()
issuer: Issuer = Issuer(trust_config.signing_config.get_oidc_url())
signing_ctx = SigningContext.from_trust_config(trust_config)
with signing_ctx.signer(issuer.identity_token(), cache=True) as signer:
attestation = Attestation.sign(signer, dist)
print(attestation.model_dump_json())
# Verify an attestation against a Python artifact
attestation_path = Path("test_package-0.0.1-py3-none-any.whl.attestation")
attestation = Attestation.model_validate_json(attestation_path.read_bytes())
identity = policy.Identity(identity="example@gmail.com", issuer="https://accounts.google.com")
attestation.verify(identity=identity, dist=dist)
Low-level model conversions
These conversions assume that any Sigstore Bundle used as an input was created by signing a distribution file.
from pathlib import Path
from pypi_attestations import Attestation
from sigstore.models import Bundle
# Sigstore Bundle -> PEP 740 Attestation object
bundle_path = Path("test_package-0.0.1-py3-none-any.whl.sigstore")
with bundle_path.open("rb") as f:
sigstore_bundle = Bundle.from_json(f.read())
attestation_object = Attestation.from_bundle(sigstore_bundle)
print(attestation_object.model_dump_json())
# PEP 740 Attestation object -> Sigstore Bundle
attestation_path = Path("attestation.json")
with attestation_path.open("rb") as f:
attestation = Attestation.model_validate_json(f.read())
bundle = attestation.to_bundle()
print(bundle.to_json())
Usage as a command line tool
[!IMPORTANT] The
pypi-attestationsCLI is intended primarily for experimentation, and is not considered a stable interface for generating or verifying attestations. Users are encouraged to generate attestations using the official PyPA publishing action or via this package's public Python APIs.
pypi-attestations --help
usage: pypi-attestation [-h] [-v] [-V] COMMAND ...
Sign, inspect or verify PEP 740 attestations
positional arguments:
COMMAND The operation to perform
sign Sign one or more inputs
verify Verify one or more inputs
inspect Inspect one or more inputs
convert Convert a Sigstore bundle into a PEP 740 attestation
options:
-h, --help show this help message and exit
-v, --verbose run with additional debug logging; supply multiple times to
increase verbosity (default: 0)
-V, --version show program's version number and exit
Signing a package
[!NOTE] If run locally (i.e. not within GitHub Actions or another source of ambient OIDC credentials), this will open a browser window to perform the Sigstore OAuth flow.
# Generate a whl file
make package
pypi-attestations sign dist/pypi_attestations-*.whl
Inspecting a PEP 740 Attestation
[!WARNING] Inspecting does not mean verifying. It only prints the structure of the attestation.
pypi-attestations inspect dist/pypi_attestations-*.whl.publish.attestation
Verifying a PEP 740 Attestation
pypi-attestations verify attestation \
--identity https://github.com/trailofbits/pypi-attestations/.github/workflows/release.yml@refs/tags/v0.0.19 \
test/assets/pypi_attestations-0.0.19.tar.gz
Verifying a PyPI package
[!NOTE] The package to verify can be passed either as a path to a local file, a
pypi:prefixed filename (e.g: 'pypi:sampleproject-1.0.0-py3-none-any.whl'), or as a direct URL to the artifact hosted by PyPI.
# Download the artifact (and its provenance) from PyPI and verify it
pypi-attestations verify pypi --repository https://github.com/sigstore/sigstore-python \
pypi:sigstore-3.6.1-py3-none-any.whl
# or alternatively, using the direct URL:
pypi-attestations verify pypi --repository https://github.com/sigstore/sigstore-python \
https://files.pythonhosted.org/packages/70/f5/324edb6a802438e97e289992a41f81bb7a58a1cda2e49439e7e48896649e/sigstore-3.6.1-py3-none-any.whl
# Verify the artifact and its provenance using local files
pypi-attestations verify pypi --repository https://github.com/sigstore/sigstore-python \
--provenance-file ~/Downloads/sigstore-3.6.1-py3-none-any.whl.provenance \
~/Downloads/sigstore-3.6.1-py3-none-any.whl
This command downloads the artifact and its provenance from PyPI. The artifact is then verified against the provenance, while also checking that the provenance's signing identity matches the repository specified by the user.
Converting a Sigstore bundle into a PEP 740 Attestation
pypi-attestations convert --output-file /tmp/rfc8785-0.1.2-py3-none-any.whl.publish.attestation \
test/assets/rfc8785-0.1.2-py3-none-any.whl.sigstore
License
Copyright 2024 Trail of Bits
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file pypi_attestations-0.0.29.tar.gz.
File metadata
- Download URL: pypi_attestations-0.0.29.tar.gz
- Upload date:
- Size: 129.0 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
2daf3ec46ff4c7123184ec892852b4d4599b78128f01f742a44406a73200c5df
|
|
| MD5 |
5e588395455cf02623495cc3a5d80cc7
|
|
| BLAKE2b-256 |
8ba920169d7df89a79bfb00045c7d42d81ccc612008474306e1858984990e165
|
Provenance
The following attestation bundles were made for pypi_attestations-0.0.29.tar.gz:
Publisher:
release.yml on pypi/pypi-attestations
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
pypi_attestations-0.0.29.tar.gz -
Subject digest:
2daf3ec46ff4c7123184ec892852b4d4599b78128f01f742a44406a73200c5df - Sigstore transparency entry: 760314291
- Sigstore integration time:
-
Permalink:
pypi/pypi-attestations@fb35ffdce810566fdf7aa22f1b517f60be0c727f -
Branch / Tag:
refs/tags/v0.0.29 - Owner: https://github.com/pypi
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@fb35ffdce810566fdf7aa22f1b517f60be0c727f -
Trigger Event:
release
-
Statement type:
File details
Details for the file pypi_attestations-0.0.29-py3-none-any.whl.
File metadata
- Download URL: pypi_attestations-0.0.29-py3-none-any.whl
- Upload date:
- Size: 21.9 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
278a28d741b57d62973c00d453ec9b9bb30456464d69296c6780474cd0bf098e
|
|
| MD5 |
4d324c53fa91d59546081e7210f2fa11
|
|
| BLAKE2b-256 |
c84ca008365b5bdeab52f023945af9aff2baf7f99de5784adca314b55872d6be
|
Provenance
The following attestation bundles were made for pypi_attestations-0.0.29-py3-none-any.whl:
Publisher:
release.yml on pypi/pypi-attestations
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
pypi_attestations-0.0.29-py3-none-any.whl -
Subject digest:
278a28d741b57d62973c00d453ec9b9bb30456464d69296c6780474cd0bf098e - Sigstore transparency entry: 760314297
- Sigstore integration time:
-
Permalink:
pypi/pypi-attestations@fb35ffdce810566fdf7aa22f1b517f60be0c727f -
Branch / Tag:
refs/tags/v0.0.29 - Owner: https://github.com/pypi
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@fb35ffdce810566fdf7aa22f1b517f60be0c727f -
Trigger Event:
release
-
Statement type: