diff --git a/L46-core-tls-credential-API.md b/L46-core-tls-credential-API.md new file mode 100644 index 000000000..d5bab9753 --- /dev/null +++ b/L46-core-tls-credential-API.md @@ -0,0 +1,478 @@ +L46: C++: New TLS Credentials API +---- +* Author(s): gtcooke94, ZhenLian +* Approver: +* Status: Draft +* Implemented in: +* Last updated: June 11, 2024 + +## Abstract + +This proposal aims to provide a new TLS credentials API with the following features: + +1. Credential reloading: reload transport credentials periodically from various sources, e.g. from credential files, or users' customized Provider implementations, without restarting the process. +1. Customizable verification of the peer certificate: perform custom validation checks on the peer's certificate chain after it has been verified to chain up to a root of trust, or fully customize path building and cryptographic verification. +1. TLS version configuration: optionally configure the minimum and maximum of the TLS versions to be negotiated during the handshake. +1. Support for certificate revocation via Certificate Revocation Lists (CRLs). + +## Background + +The current TLS implementation (the SSL credentials API) in gRPC C core has some restrictions: +1. Lack of the credential reloading support. Once an application starts, it can never reload its private key, certificate chain conveying the application's identity, and root certificates without shutting down. +1. Inflexibility of how the peer certificate chain is verified. There is no API to add additional custom verification checks. +1. Inability to choose the maximum/minimum TLS version to be negotiated during the TLS handshake. +1. No support for certificate revocation. + +We propose an API that supports the above features and meets the following requirements: + +1. Flexible enough to support multiple use cases. Users should be able to write their own reloading or verification logic if the provided ones do not fit their needs. +1. The credential reloading should be efficient. We shall reload only when needed rather than reloading per TLS connection. +1. C core API needs to be simple enough, so that most of existing SSL credentials in wrapped languages can migrate to call this new API. + +### Related Proposals: + +1. [L46: Create a new TLS credential API that supports SPIFFE mutual TLS](https://github.com/grpc/proposal/pull/98) - this was subsumed into this proposal. +1. [A69: Crl Providers](https://github.com/grpc/proposal/pull/382) + +## Proposal +Until recently, the C-Core boundary APIs needed to be written in C89, then we had C++ wrappers of those APIs in `grpcpp`. Requirements have changed such that we can use C++ directly in Core. The rest of this proposal is written as C++ that will be implemented in core, and simple aliases to everything will be written in the `grpcpp`. + +### TLS Credentials and TLS Builder +This part of the proposal introduces the base builder class, called `TlsCredentialsBuilder`, that has a large set of options for configuring TLS connections. There are 2 derived classes of `TlsCredentialsBuilder`, called `TlsChannelCredentialsBuilder` and `TlsServerCredentialsBuilder`, that users create and can use to build channel credentials or server credentials instances, respectively. The API for building these channel/server credentials instances are called `TlsCredentials`/`TlsServerCredentials` respectively. + +```c++ +// Base class of TLS credentials builder. +class TlsCredentialsBuilder { + public: + ~TlsCredentialsBuilder(); + + // Copy constructor does a deep copy. No assignment permitted. + TlsCredentialsBuilder(const TlsCredentialsBuilder& other); + + TlsCredentialsBuilder& operator=(const TlsCredentialsBuilder& other) = delete; + + // ---- Setters for member fields ---- + // Sets the certificate provider which is used to store: + // 1. The root certificates used to (cryptographically) verify peer certificate chains. + // 2. The certificate chain conveying the application's identity and the corresponding private key. + void set_certificate_provider( + std::shared_ptr certificate_provider); + + // Watches the updates of root certificates with name |name|. If used in TLS + // credentials, setting this field is optional for both the client side and + // the server side. If this is not set on the client side, we will use the + // root certificates stored in the default system location, since client side + // must provide root certificates in TLS (no matter single-side TLS or mutual + // TLS). If this is not set on the server side, we will not watch any root + // certificate updates, and assume no root certificates are needed for the + // server (in the one-side TLS scenario, the server is not required to provide + // root certificates). + // + // @param name the name of root certificates being set. + void watch_root_certificates(absl::string_view name); + + + // Watches the updates of identity key-certificate pairs with name + // |name|. If used in TLS credentials, it is required to be set + // on the server side, and optional for the client side(in the one-side + // TLS scenario, the client is not required to provide identity certificates). + + // @param name the name of the identity key cert pairs to watch. + void watch_identity_key_cert_pairs(absl::string_view name); + + // WARNING: EXPERT USE ONLY. MISUSE CAN LEAD TO SIGNIFICANT SECURITY DEGRADATION. + // + // Sets the TLS session key logging file path. If not set, TLS + // session key logging is disabled. Note that this should be used only for + // debugging purposes. It should never be used in a production environment + // due to security concerns - anyone who can obtain the (logged) session key + // can decrypt all traffic on a connection. + // + // @param tls_session_key_log_file_path: Path where TLS session keys should + // be logged. + void set_tls_session_key_log_file_path_dangerous( + absl::string_view tls_session_key_log_file_path); + + // Sets the certificate verifier. The certificate verifier performs checks on + // the peer certificate chain after the chain has been (cryptographically) + // verified to chain up to a trusted root. + // If unset on the client-side, this will default to the + // `HostNameCertificateVerifier` detailed below. + // If unset on the server-side, this will default to a nullptr. + // If set to nulltpr, this will overwrite the host name verifier and lead you + // to not doing any checks (aside from the cryptographic ones). + void set_certificate_verifier( + std::shared_ptr certificate_verifier); + + // Sets the crl provider, see CrlProvider for more details. + void set_crl_provider(std::shared_ptr crl_provider); + + // Sets the minimum TLS version that will be negotiated during the TLS + // handshake. If not set, the underlying SSL library will default to TLS v1.2. + // @param tls_version: The minimum TLS version. + void set_min_tls_version(grpc_tls_version tls_version); + + // Sets the maximum TLS version that will be negotiated during the TLS + // handshake. If not set, the underlying SSL library will default to TLS v1.3. + // @param tls_version: The maximum TLS version. + void set_max_tls_version(grpc_tls_version tls_version); + + // WARNING: EXPERT USE ONLY. MISUSE CAN LEAD TO SIGNIFICANT SECURITY DEGRADATION. + // + // Sets a custom chain builder implementation that replaces the default chain + // building from the underlying SSL library. Fully replacing and implementing + // chain building is a complex task and has dangerous security implications if + // done wrong, thus this API is inteded for expert use only. + void set_custom_chain_builder_dangerous(std::shared_ptr + chain_builder); +} + +// Server-specific options for configuring TLS. +class TlsServerCredentialsBuilder final : public TlsCredentialsBuilder { + public: + // A certificate provider that provides identity credentials is required, + // because a server must always present identity credentials during any TLS + // handshake. The certificate provider may optionally provide root + // certificates, in case the server requests client certificates. + explicit TlsServerCredentialsBuilder( + std::shared_ptr certificate_provider) + : TlsCredentialsBuilder() { + set_certificate_provider(certificate_provider); + } + + // Builds a ServerCredentials instance that establishes TLS connections in the manner specified by options. + std::shared_ptr TlsServerCredentials( + const TlsServerCredentialsBuilder& options); + + + // Sets requirements for if client certificates are requested, if they + // are required, and if the client certificate must be trusted. The + // default is GRPC_SSL_DONT_REQUEST_CLIENT_CERTIFICATE, which represents + // normal TLS. + void set_cert_request_type( + grpc_ssl_client_certificate_request_type cert_request_type); +} + +// Client-specific options for configuring TLS. +// +// A client may optionally set a certificate provider. +// If there is no certificate provider, the system default root certificates will be used to +// verify server certificates. +// If a certificate provider is set and it provides root certifices that root +// will be used. +// If a certificate provider is set and it provides identity credentials those +// identity credentials will be used. +class TlsChannelCredentialsBuilder final : public TlsCredentialsBuilder { + public: + // WARNING: EXPERT USE ONLY. MISUSE CAN LEAD TO SIGNIFICANT SECURITY DEGRADATION. + // + // Sets the decision of whether to do a crypto check on the server certificates. + // Setting this to false is potentially dangerous as in most TLS setups the + // client verifies the server. + // The default is true. + void set_verify_server_certificates_dangerous(bool verify_server_certs); + + // Builds a ChannelCredentials instance that establishes TLS connections in the manner specified by options. + std::shared_ptr BuildTlsCredentials(); +} + +``` + +`TlsCredentials` features should represent a superset of `SslCredentials` +features, so we anticipate `TlsCredentials` being the API of choice now going +forward. To make migration from the `SslCredentials` API to the +`TlsCredentials` API easier, we will also have a method that takes +`SslCredentialsOptions` and creates a `TlsChannelCredentialsBuilder` with the +same settings. We will do similarly for `TlsServerCredentialsBuilder`. + +### Credential Reloading +Credential reloading will be implemented via the `CertificateProviderInterface`. +Note that there is a naming mismatch here - there exist credentials beyond +certificates that this _could_ theoretically support. This provider is +responsible for sourcing credential material and supplying it to the internal +stack via the `CertificateProvider`'s `SetRootCertificates` and +`SetIdentityChainAndPrivateKey` APIs. Two reference implementations of the +`CertificateProviderInterface` are provided - `StaticDataCertificateProvider` +and `FileWatcherCertificateProvider`. The `StaticDataCertificateProvider` +provides certificates from raw string data. The `FileWatcherCertificateProvider` +will periodically and asynchronously refresh certificate data from a file, +allowing changes to certificates without restarting the process. More details +about these reference implementations are provided below. These implementations +handle many common use cases. If these providers do not support the intricacies +for a specific use case, a user can provide their own implementations of the +`CertificateProviderInterface`. + +```c +/* Opaque types. */ +// A struct that stores the credential data presented to the peer in handshake +// to show local identity. The private key and certificate chain must be PEM +// encoded and the public key in the leaf certificate must correspond to the +// given private key. +struct IdentityKeyCertPair { + std::string private_key_pem; + std::string certificate_chain_pem; +}; + +/** ------------------------------------ Provider ------------------------------------ */ +/** Provides identity credentials and root certificates. +*/ +// This represents a potential decoupling of roots and identity chains, with +// further extension points for something like SPIFFE bundles +class CertificateProviderInterface { + public: + virtual ~CertificateProviderInterface() = default; + + // Starts the provider. Must be called before the provider is used for any TLS + // handshakes. Does important internal setup steps. + void Start(); + + // CredentialType represents the different types of credentials that the + // provider can provide. + enum CredentialType { + // The case of a SPIFFE trust bundle still falls into RootCertificates, it's + // just another way of representing root information + RootCertificates, + IdentityChainAndPrivateKey + } + +protected: + // Provider implementations MUST provide a OnWatchStarted callback that will + // be called by the internal stack. This will be invoked when a new + // certificate name is starting to be used internally. When being invoked for + // a new certificate, this callback should call SetRootCertificates or + // SetIdentityChainAndPrivateKey to do the initial population the certificate + // data in the internal stack. + // cert_name The name of the certificates being watched. + // type The type of certificates being watched. + virtual void OnWatchStarted(std::string name, CredentialType type) = 0; + + // Provider implementations MUST provide a OnWatchStarted callback that will + // be called by the internal stack. This will be invoked when a certificate + // name is no longer being used internally. + // cert_name The name of the certificates being watched. + // type The type of certificates being watched. + virtual void OnWatchStopped(std::string name, CredentialType type) = 0; + + // Sets the root certificates based on their name. + // This value is layered and represents the following. + // The top level `absl::StatusOr` represents setting an error or a value. If + // the input is a status, it will be propagated across the internal stack as + // an error. + // The next layer is an `absl::optional`. This allows the user to set a value + // or `absl::nullopt`, with `absl::nullopt` representing a deletion/un-setting + // of the root certificate data. + // The last layer is an `absl::variant`. This is an extension point for us to + // add other kinds of root information, for example SPIFFE trust bundles. + void SetRootCertificates(absl::string_view name, absl::StatusOr>> root_data); + + // Sets the identity chain and private key based on their name. + // This value is layered and represents the following. + // The top level `absl::StatusOr` represents setting an error or a value. If + // the input is a status, it will be propagated across the internal stack as + // an error. + // The next layer is an `absl::optional`. This allows the user to set a value + // or `absl::nullopt`, with `absl::nullopt` representing a deletion/un-setting + // of the identity chain and private key data. + void SetIdentityChainAndPrivateKey(absl::string_view name, absl::StatusOr>> pem_key_cert_pairs); + + private: + std::unique_ptr distributor_; +}; + +// A basic CertificateProviderInterface implementation that will load credential +// data from static string during initialization. This provider will always +// return the same certificate data for all cert names, and reloading is not supported. +class StaticDataCertificateProvider + : public CertificateProviderInterface { + public: + StaticDataCertificateProvider( + absl::string_view root_certificate, + const absl::Span& identity_key_cert_pairs); + + explicit StaticDataCertificateProvider(absl::string_view root_certificate); + + explicit StaticDataCertificateProvider( + const absl::Span& identity_key_cert_pairs); + } + +// A CertificateProviderInterface implementation that will watch the credential +// changes on the file system. This provider will always return the up-to-date +// certificate data for all the cert names callers set through |TlsCredentialsBuilder|. +// Several things to note: +// 1. This API only supports one key-certificate file and hence one set of identity +// key-certificate pair, so SNI(Server Name Indication) is not supported. +// 2. The private key and identity certificate should always match. This API +// guarantees atomic read, and it is the callers' responsibility to do atomic +// updates. There are many ways to atomically update the key and certificates in the +// file system. To name a few: +// 1) creating a new directory, renaming the old directory to a new name, and +// then renaming the new directory to the original name of the old directory. +// 2) using a symlink for the directory. When need to change, put new +// credential data in a new directory, and change symlink. +class FileWatcherCertificateProvider final + : public CertificateProviderInterface { + public: + // Constructor to get credential updates from root and identity file paths. + // + // @param private_key_path is the file path of the private key. + // @param identity_certificate_path is the file path of the identity + // certificate chain. + // @param root_cert_path is the file path to the root certificate bundle. + // @param refresh_interval_sec is the refreshing interval that we will check + // the files for updates. + FileWatcherCertificateProvider(absl::string_view private_key_path, + absl::string_view identity_certificate_path, + absl::string_view root_cert_path, + absl::Duration refresh_interval_sec); + + // Constructor to get credential updates from identity file paths only. + FileWatcherCertificateProvider(absl::string_view private_key_path, + absl::string_view identity_certificate_path, + absl::Duration refresh_interval_sec); + + // Constructor to get credential updates from root file path only. + FileWatcherCertificateProvider(absl::string_view root_cert_path, + absl::Duration refresh_interval_sec); + } + + +``` +### Customizable Verification of the Peer Certificate + +Users should be able to customize verification of the peer certificate, and we provide two distinct interfaces for how this can be done: + +1. Allow users to perform custom validation checks on the peer's certificate chain after it has been verified to chain up to a root of trust, but before the TLS handshake completes. This is a common customization that might be needed whenever a private PKI is involved, and does not require the user to have any significant domain knowledge. + +2. Allow users to fully customize path building and cryptographic verification. This is a relatively rare requirement from users and requires deep X.509 expertise to do correctly and without introducing significant security vulnerabilities. However, it is unavoidable in order to unblock some advanced use cases, e.g. SPIFFE federation. + +The next sections explain these interfaces in more detail. + +#### Custom Validation Checks on the Peer Certificate Chain +```c++ +// Contains the information from the (verified) peer certificate chain that can +// be used to perform custom validation checks. Users should not directly +// create or destroy this request object. +class VerifiedPeerCertificateChainInfo { + public: + // The target hostname that is expected to appear in the server leaf + // certicate, e.g. github.com. Empty on the server-side. + absl::string_view target_name() const; + + // The PEM-encoded leaf cert. + absl::string_view peer_cert() const; + + // The PEM-encoded unverified peer cert chain, ordered from leaf to root, but + // not including the root. + absl::string_view peer_cert_full_chain() const; + + // The common name string from the Subject extension in the peer's leaf + // certificate. + absl::string_view common_name() const; + + // The subject name of the root certificate used to verify the peer chain + // If verification fails or the peer certificate is self-signed, this will be an + // empty string. If verification is successful, it is a comma-separated list, + // where the entries are of the form "FIELD_ABBREVIATION=string" + // ex: "CN=testca,O=Internet Widgits Pty Ltd,ST=Some-State,C=AU" + // ex: "CN=GTS Root R1,O=Google Trust Services LLC,C=US" + absl::string_view verified_root_cert_subject() const; + + // The (possibly empty) list of URI names from the Subject Alternative Name + // extension in the peer's leaf certificate. + absl::Span uri_names() const; + + // The (possibly empty) list of DNS names from the Subject Alternative Name + // extension in the peer's leaf certificate. + absl::Span dns_names() const; + + // The (possibly empty) list of email names from the Subject Alternative Name + // extension in the peer's leaf certificate. + absl::Span email_names() const; + + // The (possibly empty) list of IP names from the Subject Alternative Name + // extension in the peer's leaf certificate. + absl::Span ip_names() const; +} + +// The base class of all verifier implementations. Note that custom +// verifier implementations can compose their functionality with existing +// implementations of this interface, such as HostNameVerifier, by delegating +// to an instance of that class. +class CertificateVerifierInterface { + public: + virtual ~CertificateVerifierInterface() = default; + + // Performs a custom verification check. + // Verification occurs asynchronously. gRPC calls `Verify` on this interface. + // The callback must be called when verification is complete and the + // verification result must be populated in the output parameter of the + // callback function. + // This method will only be called in the case that regular verification has succeeded. + // Custom verification WILL NOT be called on resumed handshakes. + // + // request: the verification information associated with this request + // callback: The function that gRPC will invoke when the verifier has already + // completed its asynchronous check. Callers can use this function + // to perform any additional checks. The input parameter of the + // any::Invocable indicates the status of the verifier check. + virtual void Verify(std::shared_ptr verified_chain_info, + absl::AnyInvocable callback) = 0; + + // Cancels a verification request previously started via Verify(). + // Used when the connection attempt times out or is cancelled while an async + // verification request is pending. + // + // request: the verification information associated with this request + virtual void Cancel(std::shared_ptr verified_chain_info) = 0; +}; + +// A CertificateVerifier that will perform hostname verification, to see if the +// target name set from the client side matches the identity information +// specified on the server's certificate. +// This will be used as the default verifier if no verification is otherwise +// set. +class HostNameCertificateVerifier : public CertificateVerifierInterface { + public: + HostNameCertificateVerifier(); +}; +``` + +#### Custom Chain Building +```c++ + + +class UnverifiedPeerCertificateChain { + absl::Span peer_cert_chain_der; +} + + +class CustomChainBuilderInterface { +public: + virtual ~CustomChainBuilderInterface() = default; + + // Performs custom chain building. When chain building is complete, the + // implementer must invoke the callback. The output parameter of the callback + // must be populated with the verified DER-encoded certificate chain, ordered + // from leaf to root. Both the leaf and the root must be included. + virtual void BuildAndVerifyChain(std::shared_ptr peer_chain, absl::AnyInvocable>> callback)) = 0; + + // Cancels a chain building request request previously started via + // BuildAndVerifyChain(). Used when the connection attempt times out or is + // cancelled while an async request is pending. + // + // request: the verification information associated with this request + virtual void Cancel(std::shared_ptr peer_chain) = 0; +} +``` + +The default behavior for chain building is to defer to the underlying SSL library, which has a built-in chain building API that has been hardened over many years of use with the web PKI. For example, [X509_verify_cert](https://www.openssl.org/docs/man1.1.1/man3/X509_verify_cert.html) is the implementation in OpenSSL that builds a chain and verifies that it is trusted by one of the root certificates. +One use case for the custom chain building feature is to enable [SPIFFE federation](https://spiffe.io/), where the set of root certificates to use is determined based on the SPIFFE trust domain in the peer's leaf certificate. + +```c++ + // Builds a trusted and validated certificate chain based on SPIFFE trust maps + // rather than the standard X509 approach. +class SpiffeFederationChainBuilder : public CustomChainBuilderInterface { +public: + SpiffeFederationChainBuilder(); +} +``` \ No newline at end of file