| Internet-Draft | Shared OpenPGP Certificate Directory | May 2022 | 
| Widdecke & Winter | Expires 2 December 2022 | [Page] | 
This document defines a generic OpenPGP certificate store that can be shared between implementations. It also defines a way to root trust, and a way to associate petnames with certificates. Sharing certificates and trust decisions increases security by enabling more applications to take advantage of OpenPGP. It also improves privacy by reducing the required certificate discoveries that go out to the network.¶
This note is to be removed before publishing as an RFC.¶
The latest revision of this draft can be found at https://sequoia-pgp.gitlab.io/pgp-cert-d. Status information for this document may be found at https://datatracker.ietf.org/doc/draft-nwjw-openpgp-cert-d/.¶
Discussion of this document takes place on the OpenPGP Working Group mailing list (mailto:openpgp@ietf.org), which is archived at https://mailarchive.ietf.org/arch/browse/openpgp/.¶
Source for this draft and an issue tracker can be found at https://gitlab.com/sequoia-pgp/pgp-cert-d.¶
This Internet-Draft is submitted in full conformance with the provisions of BCP 78 and BCP 79.¶
Internet-Drafts are working documents of the Internet Engineering Task Force (IETF). Note that other groups may also distribute working documents as Internet-Drafts. The list of current Internet-Drafts is at https://datatracker.ietf.org/drafts/current/.¶
Internet-Drafts are draft documents valid for a maximum of six months and may be updated, replaced, or obsoleted by other documents at any time. It is inappropriate to use Internet-Drafts as reference material or to cite them other than as "work in progress."¶
This Internet-Draft will expire on 2 December 2022.¶
Copyright (c) 2022 IETF Trust and the persons identified as the document authors. All rights reserved.¶
This document is subject to BCP 78 and the IETF Trust's Legal Provisions Relating to IETF Documents (https://trustee.ietf.org/license-info) in effect on the date of publication of this document. Please review these documents carefully, as they describe your rights and restrictions with respect to this document. Code Components extracted from this document must include Revised BSD License text as described in Section 4.e of the Trust Legal Provisions and are provided without warranty as described in the Revised BSD License.¶
Using OpenPGP for encryption requires a certificate for each communication partner. Likewise, verification of an OpenPGP signature requires the signer's certificate.¶
An OpenPGP certificate must be discovered before it can be used. There are a number of ways to do that, for example via [keys.openpgp.org] or [I-D.draft-koch-openpgp-webkey-service-12].¶
Furthermore, an OpenPGP certificate evolves over time. The certificate itself or one of its components may be revoked; a User ID may be added; certificate subkeys may be rotated, and meta-data stored on signatures updated. Crucially, the security of OpenPGP depends on distributing each update to every involved party. A certificate update may be passively collected (e.g. by consuming an [Autocrypt] header), or actively sought out using the key discovery options mentioned above.¶
However, actively reaching out to a network source leaks information about the expected communication partner or partners, so requests should be kept to a minimum. Now, if a user has more than one application supporting OpenPGP, then every application has to discover certificates and updates, increasing the meta-data leakage. The obvious solution here is to provide a way to share the certificates instead. This is the purpose of this specification.¶
Looking at X.509, we can see that on most systems, there is a shared store of root certificates. Now, this root certificate store solves a different problem: X.509 certificates do not need to be discovered. Instead, the shared store ensures that every application uses the same set of trust roots, which is also desirable for OpenPGP. The important aspect we want to point out is that the store is shared across different applications and TLS implementations. We will come back to the differences later in this text.¶
The key words "MUST", "MUST NOT", "REQUIRED", "SHALL", "SHALL NOT", "SHOULD", "SHOULD NOT", "RECOMMENDED", "NOT RECOMMENDED", "MAY", and "OPTIONAL" in this document are to be interpreted as described in BCP 14 [RFC2119] [RFC8174] when, and only when, they appear in all capitals, as shown here.¶
This specification is motivated by the following requirements:¶
Furthermore, the following requirements are required for secure and ergonomic use of OpenPGP. Since any application using OpenPGP needs to behave consistently so as not to jeopardize security and ergonomics, this information needs to be shared well. Hence the ideal place is the certificate store:¶
We also like to have some non-functional properties:¶
Conceptually, the cert store is a name-value store. We use cert fingerprints as names, as well as a set of special names. This is accomplished by mapping names to paths, then relying on the filesystem for efficient lookups.¶
One certificate in the store is used to root trust. It is used for mapping petnames to certs (see Section 2.3), and to designate trusted introducers (see Section 2.4).¶
Petnames span a namespace that is secure and human-meaningful, but not distributed. A common example of a petname scheme are address books in mobile phones that securely map human-meaningful names to numbers (which are secure and distributed, but not human-meaningful). See [Zookos-Triangle] for a more in-depth discussion.¶
Using petnames, we can securely map human-meaningful names, like "Mom" or "juliett@example.org", to OpenPGP certificates. In contrast to many other trust models, this is a concept that most users are already familiar with. Therefore, it should be easy to train users, increasing the chance that they will use it in a secure manner.¶
The petname mapping can also be used to integrate into existing address book-like functionality provided by the platform.¶
To improve the ergonomics of public-key systems, users often delegate questions about the identity of a communications partner to some set of trusted third parties.¶
In X.509, these decisions are delegated to a fixed set of vendor-specified trusted third parties known as root certification authorities (see [X509-PKI]). These trusted third parties then usually certify intermediate certification authorities, which in turn certify the binding between a peer's key and its identity. The trust relation forms a polyforest (i.e., a directed graph with multiple roots).¶
Using this trust relation as a client during the [TLS] handshake is straightforward: The server presents its certificate along with the chain of all intermediate certificates up to the root. The client simply checks if all links in the chain are valid, and whether the terminal certificate is in its set of root certification authorities. If so, the server's certificate is authenticated.¶
In contrast, the trust relation in OpenPGP forms a directed graph. Any certificate can certify that a cert belongs to an identity. Furthermore, the user is expected to provide not only the set of trust roots (the equivalent of X.509's root certification authorities), but also to identify acceptable intermediate authorities, which are known as "trusted introducers" in OpenPGP parlance.¶
Traditionally, OpenPGP implementations have used idiosyncratic mechanisms to configure both the trust roots and the trusted introducers. That has the downside of being a proprietary mechanism that cannot easily be shared between implementations. In contrast, this specification uses a single distinguished certificate as a trust root that delegates authority to the trusted introducers.¶
This section describes in detail how to interact with a certificate store. Note that we also provide a library that abstracts this away behind a simple-to-use API.¶
If not explicitly requested otherwise, an application SHOULD use the default store. The location is platform specific, see Section 3.8 for details.¶
The default store may be overridden by the user by setting the environment variable PGP_CERT_D.¶
The application may explicitly choose to use a different location entirely. Note, however, that this should be done only with good reasons, because it jeopardizes security, privacy, and ergonomics.¶
The location of the store MUST be a directory. If it does not exist, it MAY be created on demand.¶
Names are either fingerprints or special names.¶
The store is indexed by fingerprint. This is achieved by using the file system as a dictionary, storing each certificate using a path derived from the cert's fingerprint.¶
To compute the path to the certificate file:¶
For example, the certificate with the fingerprint eb85bb5fa33a75e15e944e63f231550c4f47e38e will be stored at ${BASEPATH}/eb/85bb5fa33a75e15e944e63f231550c4f47e38e.¶
There is a set of special names that can be used to address certificates in the store. The names map to fixed locations in the store.¶
| Special name | Location | 
|---|---|
| trust-root | trust-root | 
Before a cert can be inserted or updated, you MUST acquire an exclusive lock on the store. Note that this lock only synchronizes writers: Concurrent readers can continue to use the store, and will always see consistent certs.¶
The lock to the store is represented by a file located at ${BASEPATH}/writelock which does not contain any data.
If that file does not exist, the store SHOULD be assumed unlocked and the file MUST be created before any locking operation.
The locking is achieved with file descriptors using platform specific means, see Section 3.8 for details.¶
The following procedure MUST be followed to ensure that concurrent readers are not disturbed:¶
rename(2) fails if the rename crosses filesystem boundaries).¶
rename(2) on POSIX).¶
If a certificate is stored using a fingerprint as name, the name MUST match the certificate's fingerprint.¶
The trust root is an OpenPGP certificate that is stored under the special name trust-root.¶
The certificate:¶
If the certificate has a secret key, then any conforming OpenPGP implementation can use it to add a petname or a trusted introducer. Otherwise, only an implementation with access to the secret key material can do so.¶
To add a petname to a certificate, create a User ID with the desired petname, and bind it to the target certificate using the trust root. The binding signature SHOULD be marked as non-exportable.¶
To remove a petname from a certificate, revoke the User ID using the trust root. The revocation signature SHOULD be marked as non-exportable.¶
To look up certificates by petname, iterate over the store returning all certificates that contain the petname as User ID bound by the trust root.¶
This lookup SHOULD be facilitated using an index data structure. Currently, we do not define such an index structure, but we define an extension mechanism so that the index can be stored in the store (see Section 3.6).¶
To mark a certificate as trusted introducer, create a direct key signature for the trusted introducer using the trust root, with a subpacket marking it as trust signature. The trust signature MAY be scoped. The signature SHOULD be marked as non-exportable.¶
To rescind a trust delegation, create a new direct key signature for the trusted introducer using the trust root, without a subpacket marking it as trust signature. The signature SHOULD be marked as non-exportable.¶
The trust root can be used in conjunction with the default OpenPGP trust model to authenticate nicknames attached to certificates. To look up certificates by nickname, explore the trust relation of certs in the store starting with the trust root. Return all certificates that contain the desired nickname as User ID which are corroborated by a path from the root to the certificate.¶
This lookup should be facilitated using an index data structure. Currently, we do not define such an index structure, but we define an extension mechanism so that the index can be stored in the store (see Section 3.6).¶
Files or directories in the toplevel directory starting with an underscore (_) may be freely used for proprietary and experimental extensions.
Please use a unique and descriptive prefix to minimize the chance of collisions, e.g. _foopgp_subkey_map.sqlite.¶
Unknown or unsupported extensions MUST be ignored.¶
Any files or directories in the toplevel directory other than¶
_) (see Section 3.6)¶
are reserved for future extensions and MUST be ignored.¶
| Platform | Default store location | Locking mechanism | 
|---|---|---|
| POSIX | $XDG_DATA_HOME/pgp.cert.d | flock(2)with LOCK_EX | 
| macOS | $HOME/Library/Application Support/pgp.cert.d | flock(2)with LOCK_EX | 
| Windows | {FOLDERID_RoamingAppData}/pgp.cert.d | LockFile (fileapi.h) | 
Importing the certificates described [I-D.draft-bre-openpgp-samples-00] yields the following certificate store:¶
$ export PGP_CERT_D=$(mktemp -d) $ pgp-cert-d import < alice.pgp $ (cd $PGP_CERT_D ; find -type f) ./eb/85bb5fa33a75e15e944e63f231550c4f47e38e $ pgp-cert-d import < bob.pgp $ (cd $PGP_CERT_D ; find -type f) ./eb/85bb5fa33a75e15e944e63f231550c4f47e38e ./d1/a66e1a23b182c9980f788cfbfcc82a015e7330¶
We provide a reference implementation in the form of a library implemented in Rust (see [reference-implementation-api]). This library also has a C API, so it is easy to use from other languages.¶
The library deals with the low-level mechanics of accessing the store, and computing the fingerprints of inserted certs. It does not concern itself with emergent features like petname and authenticated nickname lookups.¶
There are two ways to open a store. The first one uses the default location, the second takes a path to the store's location.¶
function new() -> Store; function open(Path) -> Store;¶
Looking up a certificate returns the certs data and a tag if the certificate exists in the store, or a special value indicating that the cert was not found.¶
The tag can be used in subsequent lookups to quickly check if the cert has actually changed. This can be used to efficiently update index data structures.¶
Usually, this function returns a CERT (Section 5.5.3), but if NAME (Section 5.5.1) is a special name, it may return a KEY (Section 5.5.4).¶
function Store::get(NAME) -> Maybe(TAG, CERT-or-KEY); function Store::get_if_changed(TAG, NAME) -> Maybe(TAG, CERT-or-KEY);¶
Inserting or updating a cert requires the CERT (Section 5.5.3) and a callback function.¶
The callback is invoked with the existing cert data (if any), and SHOULD merge the two copies of the certificate together. The function MAY decide to omit (parts of) the existing data, but this should be done with great care as not to lose any vital information.¶
The insertion method returns the merged certificate data and the tag for the new state.¶
Locking is handled by the library.¶
function Store::insert(CERT, Merge) -> (TAG, CERT)
    where Merge is
        function(CERT, Maybe(CERT)) -> CERT;
¶
The user can iterate over all certificates in the store. The iterator returns tuples of fingerprints and tags, which can be used to efficiently update index data structures.¶
Note: The iterator does not return any special names like the trust root (see Section 3.2.2).¶
function Store::iter() -> Iterator over (NAME, TAG, CERT);¶
A string representing a fingerprint or a special name (see Section 3.2).¶
An opaque value corresponding to a cert in store. If the cert is updated, its tag will change. This can be used to quickly determine if an index data structure must be updated.¶
| Mnemonic | Meaning | 
|---|---|
| OK | Success | 
| BAD_NAME | The name was neither a valid fingerprint, nor a known special name | 
| NOT_A_STORE | The base directory cannot possibly contain a store | 
| BAD_DATA | The data was not valid OpenPGP cert or key in binary format | 
| IO_ERROR | Unspecified I/O error occurred | 
Despite the fact that this spec is designed with ease of implementation in mind, and we explicitly invite reimplementations, please consider using our reference implementation.¶
This is a list of implementation considerations that interoperating implementations need to follow:¶
This is a first draft that has not been published.¶
OpenPGP requires efficient lookup by subkey fingerprint and keyids. This is currently not provided by this spec, hence implementations need to build their own index on top of this store. Future revisions may specify a way to do this natively.¶
Collecting usage information for TOFU-like trust models creates a write-heavy workload during normal usage, and requires more complex data structures that are not easily expressed using file-system operations and OpenPGP data structures. Future revisions of this spec may define suitable mechanisms to keep a record of certificate uses.¶
This spec contains platform-specific conventions (see Section 3.8), like default store locations and locking mechanisms. Porting to new platforms requires amending the spec.¶