|
|
|
|
% container-signature(5) Container signature format
|
|
|
|
|
% Miloslav Trmač
|
|
|
|
|
% March 2017
|
|
|
|
|
|
|
|
|
|
# Container signature format
|
|
|
|
|
|
|
|
|
|
This document describes the format of container signatures,
|
|
|
|
|
as implemented by the `github.com/containers/image/signature` package.
|
|
|
|
|
|
|
|
|
|
Most users should be able to consume these signatures by using the `github.com/containers/image/signature` package
|
|
|
|
|
(preferably through the higher-level `signature.PolicyContext` interface)
|
|
|
|
|
without having to care about the details of the format described below.
|
|
|
|
|
This documentation exists primarily for maintainers of the package
|
|
|
|
|
and to allow independent reimplementations.
|
|
|
|
|
|
|
|
|
|
## High-level overview
|
|
|
|
|
|
|
|
|
|
The signature provides an end-to-end authenticated claim that a container image
|
|
|
|
|
has been approved by a specific party (e.g. the creator of the image as their work,
|
|
|
|
|
an automated build system as a result of an automated build,
|
|
|
|
|
a company IT department approving the image for production) under a specified _identity_
|
|
|
|
|
(e.g. an OS base image / specific application, with a specific version).
|
|
|
|
|
|
|
|
|
|
A container signature consists of a cryptographic signature which identifies
|
|
|
|
|
and authenticates who signed the image, and carries as a signed payload a JSON document.
|
|
|
|
|
The JSON document identifies the image being signed, claims a specific identity of the
|
|
|
|
|
image and if applicable, contains other information about the image.
|
|
|
|
|
|
|
|
|
|
The signatures do not modify the container image (the layers, configuration, manifest, …);
|
|
|
|
|
e.g. their presence does not change the manifest digest used to identify the image in
|
|
|
|
|
docker/distribution servers; rather, the signatures are associated with an immutable image.
|
|
|
|
|
An image can have any number of signatures so signature distribution systems SHOULD support
|
|
|
|
|
associating more than one signature with an image.
|
|
|
|
|
|
|
|
|
|
## The cryptographic signature
|
|
|
|
|
|
|
|
|
|
As distributed, the container signature is a blob which contains a cryptographic signature
|
|
|
|
|
in an industry-standard format, carrying a signed JSON payload (i.e. the blob contains both the
|
|
|
|
|
JSON document and a signature of the JSON document; it is not a “detached signature” with
|
|
|
|
|
independent blobs containing the JSON document and a cryptographic signature).
|
|
|
|
|
|
|
|
|
|
Currently the only defined cryptographic signature format is an OpenPGP signature (RFC 4880),
|
|
|
|
|
but others may be added in the future. (The blob does not contain metadata identifying the
|
|
|
|
|
cryptographic signature format. It is expected that most formats are sufficiently self-describing
|
|
|
|
|
that this is not necessary and the configured expected public key provides another indication
|
|
|
|
|
of the expected cryptographic signature format. Such metadata may be added in the future for
|
|
|
|
|
newly added cryptographic signature formats, if necessary.)
|
|
|
|
|
|
|
|
|
|
Consumers of container signatures SHOULD verify the cryptographic signature
|
|
|
|
|
against one or more trusted public keys
|
|
|
|
|
(e.g. defined in a [policy.json signature verification policy file](containers-policy.json.5.md))
|
|
|
|
|
before parsing or processing the JSON payload in _any_ way,
|
|
|
|
|
in particular they SHOULD stop processing the container signature
|
|
|
|
|
if the cryptographic signature verification fails, without even starting to process the JSON payload.
|
|
|
|
|
|
|
|
|
|
(Consumers MAY extract identification of the signing key and other metadata from the cryptographic signature,
|
|
|
|
|
and the JSON payload, without verifying the signature, if the purpose is to allow managing the signature blobs,
|
|
|
|
|
e.g. to list the authors and image identities of signatures associated with a single container image;
|
|
|
|
|
if so, they SHOULD design the output of such processing to minimize the risk of users considering the output trusted
|
|
|
|
|
or in any way usable for making policy decisions about the image.)
|
|
|
|
|
|
|
|
|
|
### OpenPGP signature verification
|
|
|
|
|
|
|
|
|
|
When verifying a cryptographic signature in the OpenPGP format,
|
|
|
|
|
the consumer MUST verify at least the following aspects of the signature
|
|
|
|
|
(like the `github.com/containers/image/signature` package does):
|
|
|
|
|
|
|
|
|
|
- The blob MUST be a “Signed Message” as defined RFC 4880 section 11.3.
|
|
|
|
|
(e.g. it MUST NOT be an unsigned “Literal Message”, or any other non-signature format).
|
|
|
|
|
- The signature MUST have been made by an expected key trusted for the purpose (and the specific container image).
|
|
|
|
|
- The signature MUST be correctly formed and pass the cryptographic validation.
|
|
|
|
|
- The signature MUST correctly authenticate the included JSON payload
|
|
|
|
|
(in particular, the parsing of the JSON payload MUST NOT start before the complete payload has been cryptographically authenticated).
|
|
|
|
|
- The signature MUST NOT be expired.
|
|
|
|
|
|
|
|
|
|
The consumer SHOULD have tests for its verification code which verify that signatures failing any of the above are rejected.
|
|
|
|
|
|
|
|
|
|
## JSON processing and forward compatibility
|
|
|
|
|
|
|
|
|
|
The payload of the cryptographic signature is a JSON document (RFC 7159).
|
|
|
|
|
Consumers SHOULD parse it very strictly,
|
|
|
|
|
refusing any signature which violates the expected format (e.g. missing members, incorrect member types)
|
|
|
|
|
or can be interpreted ambiguously (e.g. a duplicated member in a JSON object).
|
|
|
|
|
|
|
|
|
|
Any violations of the JSON format or of other requirements in this document MAY be accepted if the JSON document can be recognized
|
|
|
|
|
to have been created by a known-incorrect implementation (see [`optional.creator`](#optionalcreator) below)
|
|
|
|
|
and if the semantics of the invalid document, as created by such an implementation, is clear.
|
|
|
|
|
|
|
|
|
|
The top-level value of the JSON document MUST be a JSON object with exactly two members, `critical` and `optional`,
|
|
|
|
|
each a JSON object.
|
|
|
|
|
|
|
|
|
|
The `critical` object MUST contain a `type` member identifying the document as a container signature
|
|
|
|
|
(as defined [below](#criticaltype))
|
|
|
|
|
and signature consumers MUST reject signatures which do not have this member or in which this member does not have the expected value.
|
|
|
|
|
|
|
|
|
|
To ensure forward compatibility (allowing older signature consumers to correctly
|
|
|
|
|
accept or reject signatures created at a later date, with possible extensions to this format),
|
|
|
|
|
consumers MUST reject the signature if the `critical` object, or _any_ of its subobjects,
|
|
|
|
|
contain _any_ member or data value which is unrecognized, unsupported, invalid, or in any other way unexpected.
|
|
|
|
|
At a minimum, this includes unrecognized members in a JSON object, or incorrect types of expected members.
|
|
|
|
|
|
|
|
|
|
For the same reason, consumers SHOULD accept any members with unrecognized names in the `optional` object,
|
|
|
|
|
and MAY accept signatures where the object member is recognized but unsupported, or the value of the member is unsupported.
|
|
|
|
|
Consumers still SHOULD reject signatures where a member of an `optional` object is supported but the value is recognized as invalid.
|
|
|
|
|
|
|
|
|
|
## JSON data format
|
|
|
|
|
|
|
|
|
|
An example of the full format follows, with detailed description below.
|
|
|
|
|
To reiterate, consumers of the signature SHOULD perform successful cryptographic verification,
|
|
|
|
|
and MUST reject unexpected data in the `critical` object, or in the top-level object, as described above.
|
|
|
|
|
|
|
|
|
|
```json
|
|
|
|
|
{
|
|
|
|
|
"critical": {
|
|
|
|
|
"type": "atomic container signature",
|
|
|
|
|
"image": {
|
|
|
|
|
"docker-manifest-digest": "sha256:817a12c32a39bbe394944ba49de563e085f1d3c5266eb8e9723256bc4448680e"
|
|
|
|
|
},
|
|
|
|
|
"identity": {
|
|
|
|
|
"docker-reference": "docker.io/library/busybox:latest"
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
"optional": {
|
|
|
|
|
"creator": "some software package v1.0.1-35",
|
|
|
|
|
"timestamp": 1483228800,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
### `critical`
|
|
|
|
|
|
|
|
|
|
This MUST be a JSON object which contains data critical to correctly evaluating the validity of a signature.
|
|
|
|
|
|
|
|
|
|
Consumers MUST reject any signature where the `critical` object contains any unrecognized, unsupported, invalid or in any other way unexpected member or data.
|
|
|
|
|
|
|
|
|
|
### `critical.type`
|
|
|
|
|
|
|
|
|
|
This MUST be a string with a string value exactly equal to `atomic container signature` (three words, including the spaces).
|
|
|
|
|
|
|
|
|
|
Signature consumers MUST reject signatures which do not have this member or this member does not have exactly the expected value.
|
|
|
|
|
|
|
|
|
|
(The consumers MAY support signatures with a different value of the `type` member, if any is defined in the future;
|
|
|
|
|
if so, the rest of the JSON document is interpreted according to rules defining that value of `critical.type`,
|
|
|
|
|
not by this document.)
|
|
|
|
|
|
|
|
|
|
### `critical.image`
|
|
|
|
|
|
|
|
|
|
This MUST be a JSON object which identifies the container image this signature applies to.
|
|
|
|
|
|
|
|
|
|
Consumers MUST reject any signature where the `critical.image` object contains any unrecognized, unsupported, invalid or in any other way unexpected member or data.
|
|
|
|
|
|
|
|
|
|
(Currently only the `docker-manifest-digest` way of identifying a container image is defined;
|
|
|
|
|
alternatives to this may be defined in the future,
|
|
|
|
|
but existing consumers are required to reject signatures which use formats they do not support.)
|
|
|
|
|
|
|
|
|
|
### `critical.image.docker-manifest-digest`
|
|
|
|
|
|
|
|
|
|
This MUST be a JSON string, in the `github.com/opencontainers/go-digest.Digest` string format.
|
|
|
|
|
|
|
|
|
|
The value of this member MUST match the manifest of the signed container image, as implemented in the docker/distribution manifest addressing system.
|
|
|
|
|
|
|
|
|
|
The consumer of the signature SHOULD verify the manifest digest against a fully verified signature before processing the contents of the image manifest in any other way
|
|
|
|
|
(e.g. parsing the manifest further or downloading layers of the image).
|
|
|
|
|
|
|
|
|
|
Implementation notes:
|
|
|
|
|
* A single container image manifest may have several valid manifest digest values, using different algorithms.
|
|
|
|
|
* For “signed” [docker/distribution schema 1](https://github.com/docker/distribution/blob/master/docs/spec/manifest-v2-1.md) manifests,
|
|
|
|
|
the manifest digest applies to the payload of the JSON web signature, not to the raw manifest blob.
|
|
|
|
|
|
|
|
|
|
### `critical.identity`
|
|
|
|
|
|
|
|
|
|
This MUST be a JSON object which identifies the claimed identity of the image (usually the purpose of the image, or the application, along with a version information),
|
|
|
|
|
as asserted by the author of the signature.
|
|
|
|
|
|
|
|
|
|
Consumers MUST reject any signature where the `critical.identity` object contains any unrecognized, unsupported, invalid or in any other way unexpected member or data.
|
|
|
|
|
|
|
|
|
|
(Currently only the `docker-reference` way of claiming an image identity/purpose is defined;
|
|
|
|
|
alternatives to this may be defined in the future,
|
|
|
|
|
but existing consumers are required to reject signatures which use formats they do not support.)
|
|
|
|
|
|
|
|
|
|
### `critical.identity.docker-reference`
|
|
|
|
|
|
|
|
|
|
This MUST be a JSON string, in the `github.com/docker/distribution/reference` string format,
|
|
|
|
|
and using the same normalization semantics (where e.g. `busybox:latest` is equivalent to `docker.io/library/busybox:latest`).
|
|
|
|
|
If the normalization semantics allows multiple string representations of the claimed identity with equivalent meaning,
|
|
|
|
|
the `critical.identity.docker-reference` member SHOULD use the fully explicit form (including the full host name and namespaces).
|
|
|
|
|
|
|
|
|
|
The value of this member MUST match the image identity/purpose expected by the consumer of the image signature and the image
|
|
|
|
|
(again, accounting for the `docker/distribution/reference` normalization semantics).
|
|
|
|
|
|
|
|
|
|
In the most common case, this means that the `critical.identity.docker-reference` value must be equal to the docker/distribution reference used to refer to or download the image.
|
|
|
|
|
However, depending on the specific application, users or system administrators may accept less specific matches
|
|
|
|
|
(e.g. ignoring the tag value in the signature when pulling the `:latest` tag or when referencing an image by digest),
|
|
|
|
|
or they may require `critical.identity.docker-reference` values with a completely different namespace to the reference used to refer to/download the image
|
|
|
|
|
(e.g. requiring a `critical.identity.docker-reference` value which identifies the image as coming from a supplier when fetching it from a company-internal mirror of approved images).
|
|
|
|
|
The software performing this verification SHOULD allow the users to define such a policy using the [policy.json signature verification policy file format](containers-policy.json.5.md).
|
|
|
|
|
|
|
|
|
|
The `critical.identity.docker-reference` value SHOULD contain either a tag or digest;
|
|
|
|
|
in most cases, it SHOULD use a tag rather than a digest. (See also the default [`matchRepoDigestOrExact` matching semantics in `policy.json`](containers-policy.json.5.md#signedby).)
|
|
|
|
|
|
|
|
|
|
### `optional`
|
|
|
|
|
|
|
|
|
|
This MUST be a JSON object.
|
|
|
|
|
|
|
|
|
|
Consumers SHOULD accept any members with unrecognized names in the `optional` object,
|
|
|
|
|
and MAY accept a signature where the object member is recognized but unsupported, or the value of the member is valid but unsupported.
|
|
|
|
|
Consumers still SHOULD reject any signature where a member of an `optional` object is supported but the value is recognized as invalid.
|
|
|
|
|
|
|
|
|
|
### `optional.creator`
|
|
|
|
|
|
|
|
|
|
If present, this MUST be a JSON string, identifying the name and version of the software which has created the signature.
|
|
|
|
|
|
|
|
|
|
The contents of this string is not defined in detail; however each implementation creating container signatures:
|
|
|
|
|
|
|
|
|
|
- SHOULD define the contents to unambiguously define the software in practice (e.g. it SHOULD contain the name of the software, not only the version number)
|
|
|
|
|
- SHOULD use a build and versioning process which ensures that the contents of this string (e.g. an included version number)
|
|
|
|
|
changes whenever the format or semantics of the generated signature changes in any way;
|
|
|
|
|
it SHOULD not be possible for two implementations which use a different format or semantics to have the same `optional.creator` value
|
|
|
|
|
- SHOULD use a format which is reasonably easy to parse in software (perhaps using a regexp),
|
|
|
|
|
and which makes it easy enough to recognize a range of versions of a specific implementation
|
|
|
|
|
(e.g. the version of the implementation SHOULD NOT be only a git hash, because they don’t have an easily defined ordering;
|
|
|
|
|
the string should contain a version number, or at least a date of the commit).
|
|
|
|
|
|
|
|
|
|
Consumers of container signatures MAY recognize specific values or sets of values of `optional.creator`
|
|
|
|
|
(perhaps augmented with `optional.timestamp`),
|
|
|
|
|
and MAY change their processing of the signature based on these values
|
|
|
|
|
(usually to accommodate violations of this specification in past versions of the signing software which cannot be fixed retroactively),
|
|
|
|
|
as long as the semantics of the invalid document, as created by such an implementation, is clear.
|
|
|
|
|
|
|
|
|
|
If consumers of signatures do change their behavior based on the `optional.creator` value,
|
|
|
|
|
they SHOULD take care that the way they process the signatures is not inconsistent with
|
|
|
|
|
strictly validating signature consumers.
|
|
|
|
|
(I.e. it is acceptable for a consumer to accept a signature based on a specific `optional.creator` value
|
|
|
|
|
if other implementations would completely reject the signature,
|
|
|
|
|
but it would be very undesirable for the two kinds of implementations to accept the signature in different
|
|
|
|
|
and inconsistent situations.)
|
|
|
|
|
|
|
|
|
|
### `optional.timestamp`
|
|
|
|
|
|
|
|
|
|
If present, this MUST be a JSON number, which is representable as a 64-bit integer, and identifies the time when the signature was created
|
|
|
|
|
as the number of seconds since the UNIX epoch (Jan 1 1970 00:00 UTC).
|