Botan 3.5.0
Crypto and TLS for C&
Botan::TLS::Callbacks Class Referenceabstract

#include <tls_callbacks.h>

Public Member Functions

virtual void tls_alert (Alert alert)=0
 
virtual std::chrono::system_clock::time_point tls_current_timestamp ()
 
virtual void tls_emit_data (std::span< const uint8_t > data)=0
 
virtual secure_vector< uint8_t > tls_ephemeral_key_agreement (const std::variant< TLS::Group_Params, DL_Group > &group, const PK_Key_Agreement_Key &private_key, const std::vector< uint8_t > &public_value, RandomNumberGenerator &rng, const Policy &policy)
 
virtual void tls_examine_extensions (const Extensions &extn, Connection_Side which_side, Handshake_Type which_message)
 
virtual std::unique_ptr< PK_Key_Agreement_Keytls_generate_ephemeral_key (const std::variant< TLS::Group_Params, DL_Group > &group, RandomNumberGenerator &rng)
 
virtual void tls_inspect_handshake_msg (const Handshake_Message &message)
 
virtual secure_vector< uint8_t > tls_kem_decapsulate (TLS::Group_Params group, const Private_Key &private_key, const std::vector< uint8_t > &encapsulated_bytes, RandomNumberGenerator &rng, const Policy &policy)
 
virtual KEM_Encapsulation tls_kem_encapsulate (TLS::Group_Params group, const std::vector< uint8_t > &encoded_public_key, RandomNumberGenerator &rng, const Policy &policy)
 
virtual std::unique_ptr< Private_Keytls_kem_generate_key (TLS::Group_Params group, RandomNumberGenerator &rng)
 
virtual void tls_log_debug (const char *what)
 
virtual void tls_log_debug_bin (const char *descr, const uint8_t val[], size_t val_len)
 
virtual void tls_log_error (const char *err)
 
virtual void tls_modify_extensions (Extensions &extn, Connection_Side which_side, Handshake_Type which_message)
 
virtual std::optional< OCSP::Responsetls_parse_ocsp_response (const std::vector< uint8_t > &raw_response)
 
virtual bool tls_peer_closed_connection ()
 
virtual std::string tls_peer_network_identity ()
 
virtual std::vector< std::vector< uint8_t > > tls_provide_cert_chain_status (const std::vector< X509_Certificate > &chain, const Certificate_Status_Request &csr)
 
virtual std::vector< uint8_t > tls_provide_cert_status (const std::vector< X509_Certificate > &chain, const Certificate_Status_Request &csr)
 
virtual void tls_record_received (uint64_t seq_no, std::span< const uint8_t > data)=0
 
virtual std::string tls_server_choose_app_protocol (const std::vector< std::string > &client_protos)
 
virtual void tls_session_activated ()
 
virtual void tls_session_established (const Session_Summary &session)
 
virtual bool tls_should_persist_resumption_information (const Session &session)
 
virtual std::vector< uint8_t > tls_sign_message (const Private_Key &key, RandomNumberGenerator &rng, std::string_view padding, Signature_Format format, const std::vector< uint8_t > &msg)
 
virtual void tls_ssl_key_log_data (std::string_view label, std::span< const uint8_t > client_random, std::span< const uint8_t > secret) const
 
virtual void tls_verify_cert_chain (const std::vector< X509_Certificate > &cert_chain, const std::vector< std::optional< OCSP::Response > > &ocsp_responses, const std::vector< Certificate_Store * > &trusted_roots, Usage_Type usage, std::string_view hostname, const TLS::Policy &policy)
 
virtual std::chrono::milliseconds tls_verify_cert_chain_ocsp_timeout () const
 
virtual bool tls_verify_message (const Public_Key &key, std::string_view padding, Signature_Format format, const std::vector< uint8_t > &msg, const std::vector< uint8_t > &sig)
 
virtual void tls_verify_raw_public_key (const Public_Key &raw_public_key, Usage_Type usage, std::string_view hostname, const TLS::Policy &policy)
 
virtual ~Callbacks ()=default
 

Detailed Description

Encapsulates the callbacks that a TLS channel will make which are due to channel specific operations.

Definition at line 44 of file tls_callbacks.h.

Constructor & Destructor Documentation

◆ ~Callbacks()

virtual Botan::TLS::Callbacks::~Callbacks ( )
virtualdefault

Member Function Documentation

◆ tls_alert()

virtual void Botan::TLS::Callbacks::tls_alert ( Alert alert)
pure virtual

Mandatory callback: alert received Called when an alert is received from the peer If fatal, the connection is closing. If not fatal, the connection may still be closing (depending on the error and the peer).

Parameters
alertthe source of the alert

◆ tls_current_timestamp()

std::chrono::system_clock::time_point Botan::TLS::Callbacks::tls_current_timestamp ( )
virtual

Optional callback: return a custom time stamp value

This allows the library user to specify a custom "now" timestamp when needed. By default it will use the current system clock time.

Note that typical usages will not need to override this callback but it is useful for testing purposes to allow for deterministic test outcomes.

Definition at line 59 of file tls_callbacks.cpp.

59 {
60 return std::chrono::system_clock::now();
61}

Referenced by Botan::TLS::make_hello_random(), Botan::TLS::PSK::PSK(), and Botan::TLS::Session_Manager::retrieve().

◆ tls_emit_data()

virtual void Botan::TLS::Callbacks::tls_emit_data ( std::span< const uint8_t > data)
pure virtual

Mandatory callback: output function The channel will call this with data which needs to be sent to the peer (eg, over a socket or some other form of IPC). The array will be overwritten when the function returns so a copy must be made if the data cannot be sent immediately.

Parameters
dataa contiguous data buffer to send

◆ tls_ephemeral_key_agreement()

secure_vector< uint8_t > Botan::TLS::Callbacks::tls_ephemeral_key_agreement ( const std::variant< TLS::Group_Params, DL_Group > & group,
const PK_Key_Agreement_Key & private_key,
const std::vector< uint8_t > & public_value,
RandomNumberGenerator & rng,
const Policy & policy )
virtual

Agree on a shared secret with the peer's ephemeral public key for the TLS handshake.

Applications may use this to add custom groups, curves or entirely different ephemeral key agreement mechanisms to the TLS handshake. Note that this callback must be used in conjunction with Callbacks::tls_generate_ephemeral_key.

Typical use cases of the library don't need to do that and serious security risks are associated with customizing TLS's key exchange mechanism.

Parameters
groupthe TLS group identifier to be used TLS 1.2 allows for specifying custom discrete logarithm parameters as part of the protocol. Hence the variant<>.
private_keythe private key (generated ahead in tls_generate_ephemeral_key)
public_valuethe public key exchange information received by the peer
rnga random number generator
policya TLS policy object
Returns
the shared secret derived from public_value and private_key

Definition at line 294 of file tls_callbacks.cpp.

299 {
300 auto agree = [&](const PK_Key_Agreement_Key& sk, const auto& pk) {
301 PK_Key_Agreement ka(sk, rng, "Raw");
302 return ka.derive_key(0, pk.public_value()).bits_of();
303 };
304
305 if(is_dh_group(group)) {
306 // TLS 1.2 allows specifying arbitrary DL_Group parameters in-lieu of
307 // a standardized DH group identifier.
308 const auto dl_group = get_dl_group(group);
309
310 auto Y = BigInt::from_bytes(public_value);
311
312 /*
313 * A basic check for key validity. As we do not know q here we
314 * cannot check that Y is in the right subgroup. However since
315 * our key is ephemeral there does not seem to be any
316 * advantage to bogus keys anyway.
317 */
318 if(Y <= 1 || Y >= dl_group.get_p() - 1) {
319 throw TLS_Exception(Alert::IllegalParameter, "Server sent bad DH key for DHE exchange");
320 }
321
322 DH_PublicKey peer_key(dl_group, Y);
323 policy.check_peer_key_acceptable(peer_key);
324
325 return agree(private_key, peer_key);
326 }
327
328 BOTAN_ASSERT_NOMSG(std::holds_alternative<TLS::Group_Params>(group));
329 const auto group_params = std::get<TLS::Group_Params>(group);
330
331 if(group_params.is_ecdh_named_curve()) {
332 const auto ec_group = EC_Group::from_name(group_params.to_string().value());
333 ECDH_PublicKey peer_key(ec_group, ec_group.OS2ECP(public_value));
334 policy.check_peer_key_acceptable(peer_key);
335
336 return agree(private_key, peer_key);
337 }
338
339#if defined(BOTAN_HAS_X25519)
340 if(group_params.is_x25519()) {
341 if(public_value.size() != 32) {
342 throw TLS_Exception(Alert::HandshakeFailure, "Invalid X25519 key size");
343 }
344
345 X25519_PublicKey peer_key(public_value);
346 policy.check_peer_key_acceptable(peer_key);
347
348 return agree(private_key, peer_key);
349 }
350#endif
351
352#if defined(BOTAN_HAS_X448)
353 if(group_params.is_x448()) {
354 if(public_value.size() != 56) {
355 throw TLS_Exception(Alert::HandshakeFailure, "Invalid X448 key size");
356 }
357
358 X448_PublicKey peer_key(public_value);
359 policy.check_peer_key_acceptable(peer_key);
360
361 return agree(private_key, peer_key);
362 }
363#endif
364
365 throw TLS_Exception(Alert::IllegalParameter, "Did not recognize the key exchange group");
366}
#define BOTAN_ASSERT_NOMSG(expr)
Definition assert.h:59
static BigInt from_bytes(std::span< const uint8_t > bytes)
Definition bigint.cpp:95
static EC_Group from_name(std::string_view name)
Definition ec_group.cpp:420
FE_25519 Y
Definition ge.cpp:26

References Botan::OctetString::bits_of(), BOTAN_ASSERT_NOMSG, Botan::TLS::Policy::check_peer_key_acceptable(), Botan::PK_Key_Agreement::derive_key(), Botan::BigInt::from_bytes(), Botan::EC_Group::from_name(), and Y.

Referenced by Botan::TLS::Client_Key_Exchange::Client_Key_Exchange(), and Botan::TLS::Client_Key_Exchange::Client_Key_Exchange().

◆ tls_examine_extensions()

void Botan::TLS::Callbacks::tls_examine_extensions ( const Extensions & extn,
Connection_Side which_side,
Handshake_Type which_message )
virtual

Optional callback: examine peer extensions.

Both client and server will call this callback with the Extensions object after receiving it from the peer. This allows examining the Extensions, for example to implement a custom extension. It also allows an application to require that a particular extension be implemented; throw an exception from this function to abort the handshake.

Default implementation does nothing.

Parameters
extnthe extensions
which_sidewill be Connection_Side::Client if these are are the clients extensions (ie we are the server) or Connection_Side::Server if these are the server extensions (we are the client).
which_messagewill state the handshake message type containing the extensions

Definition at line 67 of file tls_callbacks.cpp.

69 {}

Referenced by Botan::TLS::Certificate_13::validate_extensions().

◆ tls_generate_ephemeral_key()

std::unique_ptr< PK_Key_Agreement_Key > Botan::TLS::Callbacks::tls_generate_ephemeral_key ( const std::variant< TLS::Group_Params, DL_Group > & group,
RandomNumberGenerator & rng )
virtual

Generate an ephemeral key pair for the TLS handshake.

Applications may use this to add custom groups, curves or entirely different ephemeral key agreement mechanisms to the TLS handshake. Note that this callback must be used in conjunction with Callbacks::tls_ephemeral_key_agreement.

Typical use cases of the library don't need to do that and serious security risks are associated with customizing TLS's key exchange mechanism.

Exceptions
TLS_Exception(Alert::DecodeError)if the group is not known.
Parameters
groupthe group identifier to generate an ephemeral keypair for TLS 1.2 allows for specifying custom discrete logarithm parameters as part of the protocol. Hence the variant<>.
rnga random number generator
Returns
a private key of an algorithm usable for key agreement

Definition at line 260 of file tls_callbacks.cpp.

261 {
262 if(is_dh_group(group)) {
263 const DL_Group dl_group = get_dl_group(group);
264 return std::make_unique<DH_PrivateKey>(rng, dl_group);
265 }
266
267 BOTAN_ASSERT_NOMSG(std::holds_alternative<TLS::Group_Params>(group));
268 const auto group_params = std::get<TLS::Group_Params>(group);
269
270 if(group_params.is_ecdh_named_curve()) {
271 const auto ec_group = EC_Group::from_name(group_params.to_string().value());
272 return std::make_unique<ECDH_PrivateKey>(rng, ec_group);
273 }
274
275#if defined(BOTAN_HAS_X25519)
276 if(group_params.is_x25519()) {
277 return std::make_unique<X25519_PrivateKey>(rng);
278 }
279#endif
280
281#if defined(BOTAN_HAS_X448)
282 if(group_params.is_x448()) {
283 return std::make_unique<X448_PrivateKey>(rng);
284 }
285#endif
286
287 if(group_params.is_kem()) {
288 throw TLS_Exception(Alert::IllegalParameter, "cannot generate an ephemeral KEX key for a KEM");
289 }
290
291 throw TLS_Exception(Alert::DecodeError, "cannot create a key offering without a group definition");
292}

References BOTAN_ASSERT_NOMSG, and Botan::EC_Group::from_name().

Referenced by Botan::TLS::Client_Key_Exchange::Client_Key_Exchange(), and Botan::TLS::Server_Key_Exchange::Server_Key_Exchange().

◆ tls_inspect_handshake_msg()

void Botan::TLS::Callbacks::tls_inspect_handshake_msg ( const Handshake_Message & message)
virtual

Optional callback: inspect handshake message Throw an exception to abort the handshake. Default simply ignores the message.

Note: On connections that negotiated TLS 1.3 this callback is also invoked for post-handshake messages.

Parameters
messagethe handshake message

Definition at line 47 of file tls_callbacks.cpp.

47 {
48 // default is no op
49}

Referenced by Botan::TLS::Handshake_State::note_message().

◆ tls_kem_decapsulate()

secure_vector< uint8_t > Botan::TLS::Callbacks::tls_kem_decapsulate ( TLS::Group_Params group,
const Private_Key & private_key,
const std::vector< uint8_t > & encapsulated_bytes,
RandomNumberGenerator & rng,
const Policy & policy )
virtual

Performs a key decapsulation operation (used for TLS 1.3 clients).

Applications may use this to add custom KEM algorithms or entirely different key exchange schemes to the TLS 1.3 handshake. For instance, this could provide an entry point to implement a hybrid key exchange with both a traditional algorithm like ECDH and a quantum-secure KEM. Typical use cases of the library don't need to do that and serious security risks are associated with customizing TLS's key encapsulation mechanism.

Note that the KEM interface is usable for TLS 1.3 handshakes, only.

The default implementation simply delegates this to the tls_ephemeral_key_agreement() callback to obtain the shared secret.

Parameters
groupthe group identifier of the KEM/KEX algorithm
private_keythe private key used for decapsulation/KEX
encapsulated_bytesthe content to decapsulate (or the public key share)
rnga random number generator
policya TLS policy object
Returns
the plaintext shared secret from encapsulated_bytes after decapsulation with private_key.

Definition at line 222 of file tls_callbacks.cpp.

226 {
227 if(group.is_kem()) {
228 PK_KEM_Decryptor kemdec(private_key, rng, "Raw");
229 return kemdec.decrypt(encapsulated_bytes, 0, {});
230 }
231
232 try {
233 auto& key_agreement_key = dynamic_cast<const PK_Key_Agreement_Key&>(private_key);
234 return tls_ephemeral_key_agreement(group, key_agreement_key, encapsulated_bytes, rng, policy);
235 } catch(const std::bad_cast&) {
236 throw Invalid_Argument("provided ephemeral key is not a PK_Key_Agreement_Key");
237 }
238}
virtual secure_vector< uint8_t > tls_ephemeral_key_agreement(const std::variant< TLS::Group_Params, DL_Group > &group, const PK_Key_Agreement_Key &private_key, const std::vector< uint8_t > &public_value, RandomNumberGenerator &rng, const Policy &policy)

References Botan::PK_KEM_Decryptor::decrypt(), and Botan::TLS::Group_Params::is_kem().

◆ tls_kem_encapsulate()

KEM_Encapsulation Botan::TLS::Callbacks::tls_kem_encapsulate ( TLS::Group_Params group,
const std::vector< uint8_t > & encoded_public_key,
RandomNumberGenerator & rng,
const Policy & policy )
virtual

Performs a key encapsulation operation (used for TLS 1.3 servers)

Applications may use this to add custom KEM algorithms or entirely different key exchange schemes to the TLS 1.3 handshake. For instance, this could provide an entry point to implement a hybrid key exchange with both a traditional algorithm like ECDH and a quantum-secure KEM. Typical use cases of the library don't need to do that and serious security risks are associated with customizing TLS's key encapsulation mechanism.

Note that the KEM interface is usable for TLS 1.3 handshakes, only.

The default implementation implements this key encapsulation as a combination of tls_generate_ephemeral_key() followed by tls_ephemeral_key_agreement() with the provided encoded_public_key. The just-generated ephemeral private key is destroyed immediately.

Parameters
groupthe group identifier of the KEM/KEX algorithm
encoded_public_keythe public key used for encapsulation/KEX
rnga random number generator
policya TLS policy object
Returns
the shared secret both in plaintext and encapsulated with encoded_public_key.

Definition at line 183 of file tls_callbacks.cpp.

186 {
187 if(group.is_kem()) {
188 auto kem_pub_key = [&]() -> std::unique_ptr<Public_Key> {
189
190#if defined(BOTAN_HAS_TLS_13_PQC)
191 if(group.is_pqc_hybrid()) {
192 return Hybrid_KEM_PublicKey::load_for_group(group, encoded_public_key);
193 }
194#endif
195
196#if defined(BOTAN_HAS_KYBER)
197 if(group.is_pure_kyber()) {
198 return std::make_unique<Kyber_PublicKey>(encoded_public_key, KyberMode(group.to_string().value()));
199 }
200#endif
201
202#if defined(BOTAN_HAS_FRODOKEM)
203 if(group.is_pure_frodokem()) {
204 return std::make_unique<FrodoKEM_PublicKey>(encoded_public_key, FrodoKEMMode(group.to_string().value()));
205 }
206#endif
207
208 throw TLS_Exception(Alert::IllegalParameter, "KEM is not supported");
209 }();
210
211 return PK_KEM_Encryptor(*kem_pub_key, "Raw").encrypt(rng);
212 }
213
214 // TODO: We could use the KEX_to_KEM_Adapter to remove the case distinction
215 // of KEM and KEX. However, the workarounds in this adapter class
216 // should first be addressed.
217 auto ephemeral_keypair = tls_generate_ephemeral_key(group, rng);
218 return KEM_Encapsulation(ephemeral_keypair->public_value(),
219 tls_ephemeral_key_agreement(group, *ephemeral_keypair, encoded_public_key, rng, policy));
220}
virtual std::unique_ptr< PK_Key_Agreement_Key > tls_generate_ephemeral_key(const std::variant< TLS::Group_Params, DL_Group > &group, RandomNumberGenerator &rng)
static std::unique_ptr< Hybrid_KEM_PublicKey > load_for_group(Group_Params group, std::span< const uint8_t > concatenated_public_values)

References Botan::PK_KEM_Encryptor::encrypt(), Botan::TLS::Group_Params::is_kem(), Botan::TLS::Group_Params::is_pqc_hybrid(), Botan::TLS::Group_Params::is_pure_frodokem(), Botan::TLS::Group_Params::is_pure_kyber(), Botan::TLS::Hybrid_KEM_PublicKey::load_for_group(), and Botan::TLS::Group_Params::to_string().

◆ tls_kem_generate_key()

std::unique_ptr< Private_Key > Botan::TLS::Callbacks::tls_kem_generate_key ( TLS::Group_Params group,
RandomNumberGenerator & rng )
virtual

Generate an ephemeral KEM key for a TLS 1.3 handshake

Applications may use this to add custom KEM algorithms or entirely different key exchange schemes to the TLS 1.3 handshake. For instance, this could provide an entry point to implement a hybrid key exchange with both a traditional algorithm like ECDH and a quantum-secure KEM. Typical use cases of the library don't need to do that and serious security risks are associated with customizing TLS's key encapsulation mechanism.

Note that the KEM interface is usable for TLS 1.3 handshakes, only.

The default implementation simply delegates this to the tls_generate_ephemeral_key() call when appropriate.

Parameters
groupthe group identifier to generate an ephemeral keypair for
rnga random number generator
Returns
a keypair whose public key will be provided to the peer and the private key will be provided to tls_kem_decapsulate later in the handshake.

Definition at line 161 of file tls_callbacks.cpp.

161 {
162#if defined(BOTAN_HAS_KYBER)
163 if(group.is_pure_kyber()) {
164 return std::make_unique<Kyber_PrivateKey>(rng, KyberMode(group.to_string().value()));
165 }
166#endif
167
168#if defined(BOTAN_HAS_FRODOKEM)
169 if(group.is_pure_frodokem()) {
170 return std::make_unique<FrodoKEM_PrivateKey>(rng, FrodoKEMMode(group.to_string().value()));
171 }
172#endif
173
174#if defined(BOTAN_HAS_TLS_13_PQC)
175 if(group.is_pqc_hybrid()) {
177 }
178#endif
179
180 return tls_generate_ephemeral_key(group, rng);
181}
static std::unique_ptr< Hybrid_KEM_PrivateKey > generate_from_group(Group_Params group, RandomNumberGenerator &rng)

References Botan::TLS::Hybrid_KEM_PrivateKey::generate_from_group(), Botan::TLS::Group_Params::is_pqc_hybrid(), Botan::TLS::Group_Params::is_pure_frodokem(), Botan::TLS::Group_Params::is_pure_kyber(), and Botan::TLS::Group_Params::to_string().

◆ tls_log_debug()

virtual void Botan::TLS::Callbacks::tls_log_debug ( const char * what)
inlinevirtual

Optional callback: debug logging. (not currently called)

Parameters
whatSome hopefully informative string

Definition at line 552 of file tls_callbacks.h.

552{ BOTAN_UNUSED(what); }
#define BOTAN_UNUSED
Definition assert.h:118

References BOTAN_UNUSED.

◆ tls_log_debug_bin()

virtual void Botan::TLS::Callbacks::tls_log_debug_bin ( const char * descr,
const uint8_t val[],
size_t val_len )
inlinevirtual

Optional callback: debug logging taking a buffer. (not currently called)

Parameters
descrWhat this buffer is
valthe bytes
val_lenlength of val

Definition at line 560 of file tls_callbacks.h.

560 {
561 BOTAN_UNUSED(descr, val, val_len);
562 }

References BOTAN_UNUSED.

◆ tls_log_error()

virtual void Botan::TLS::Callbacks::tls_log_error ( const char * err)
inlinevirtual

Optional callback: error logging. (not currently called)

Parameters
errAn error message related to this connection.

Definition at line 546 of file tls_callbacks.h.

546{ BOTAN_UNUSED(err); }

References BOTAN_UNUSED.

◆ tls_modify_extensions()

void Botan::TLS::Callbacks::tls_modify_extensions ( Extensions & extn,
Connection_Side which_side,
Handshake_Type which_message )
virtual

Optional callback: examine/modify Extensions before sending.

Both client and server will call this callback on the Extensions object before serializing it in the specific handshake message. This allows an application to modify which extensions are sent during the handshake.

Default implementation does nothing.

Parameters
extnthe extensions
which_sidewill be Connection_Side::Client or Connection_Side::Server which is the current applications role in the exchange.
which_messagewill state the handshake message type containing the extensions

Definition at line 63 of file tls_callbacks.cpp.

65 {}

Referenced by Botan::TLS::Client_Hello_12::Client_Hello_12(), Botan::TLS::Client_Hello_12::Client_Hello_12(), Botan::TLS::Client_Hello_13::Client_Hello_13(), Botan::TLS::Encrypted_Extensions::Encrypted_Extensions(), Botan::TLS::Hello_Retry_Request::Hello_Retry_Request(), Botan::TLS::New_Session_Ticket_13::New_Session_Ticket_13(), Botan::TLS::Client_Hello_13::retry(), Botan::TLS::Server_Hello_12::Server_Hello_12(), and Botan::TLS::Server_Hello_13::Server_Hello_13().

◆ tls_parse_ocsp_response()

std::optional< OCSP::Response > Botan::TLS::Callbacks::tls_parse_ocsp_response ( const std::vector< uint8_t > & raw_response)
virtual

Optional callback: parse a single OCSP Response

Note: Typically a user of the library would not want to override this callback. We provide this callback to be able to support OCSP related tests from BoringSSL's BoGo tests that provide unparsable responses.

Default implementation tries to parse the provided raw OCSP response.

This function should not throw an exception but return a std::nullopt if the OCSP response cannot be parsed.

Parameters
raw_responseraw OCSP response buffer
Returns
the parsed OCSP response or std::nullopt on error

Definition at line 123 of file tls_callbacks.cpp.

123 {
124 try {
125 return OCSP::Response(raw_response);
126 } catch(const Decoding_Error&) {
127 // ignore parsing errors and just ignore the broken OCSP response
128 return std::nullopt;
129 }
130}

◆ tls_peer_closed_connection()

virtual bool Botan::TLS::Callbacks::tls_peer_closed_connection ( )
inlinevirtual

Optional callback: peer closed connection (sent a "close_notify" alert)

The peer signaled that it wishes to shut down the connection. The application should not expect to receive any more data from the peer and may tear down the underlying transport socket.

Prior to TLS 1.3 it was required that peers discard pending writes and immediately respond with their own "close_notify". With TLS 1.3, applications can continue to send data despite the peer having already signaled their wish to shut down.

Returning true will cause the TLS 1.3 implementation to write all pending data and then also signal a connection shut down. Otherwise the application is responsible to call the Channel::close() method.

For TLS 1.2 the return value has no effect.

Returns
true causes the implementation to respond with a "close_notify"

Definition at line 115 of file tls_callbacks.h.

115{ return true; }

◆ tls_peer_network_identity()

std::string Botan::TLS::Callbacks::tls_peer_network_identity ( )
virtual

Optional callback: return peer network identity

There is no expected or specified format. The only expectation is this function will return a unique value. For example returning the peer host IP and port.

This is used to bind the DTLS cookie to a particular network identity. It is only called if the dtls-cookie-secret PSK is also defined.

Definition at line 55 of file tls_callbacks.cpp.

55 {
56 return "";
57}

◆ tls_provide_cert_chain_status()

std::vector< std::vector< uint8_t > > Botan::TLS::Callbacks::tls_provide_cert_chain_status ( const std::vector< X509_Certificate > & chain,
const Certificate_Status_Request & csr )
virtual

Called by TLS 1.3 client or server whenever the peer indicated that OCSP stapling is supported. In contrast to tls_provide_cert_status, this allows providing OCSP responses for each certificate in the chain.

The default implementation invokes tls_provide_cert_status assuming that no OCSP responses for intermediate certificates are available.

Returns
a vector of OCSP response buffers. An empty buffer indicates that no OCSP response should be provided for the respective certificate (at the same list index). The returned vector MUST be exactly the same length as the incoming chain.

Definition at line 132 of file tls_callbacks.cpp.

133 {
134 std::vector<std::vector<uint8_t>> result(chain.size());
135 if(!chain.empty()) {
136 result[0] = tls_provide_cert_status(chain, csr);
137 }
138 return result;
139}
virtual std::vector< uint8_t > tls_provide_cert_status(const std::vector< X509_Certificate > &chain, const Certificate_Status_Request &csr)

◆ tls_provide_cert_status()

virtual std::vector< uint8_t > Botan::TLS::Callbacks::tls_provide_cert_status ( const std::vector< X509_Certificate > & chain,
const Certificate_Status_Request & csr )
inlinevirtual

Called by the TLS server whenever the client included the status_request extension (see RFC 6066, a.k.a OCSP stapling) in the ClientHello.

Returns
the encoded OCSP response to be sent to the client which indicates the revocation status of the server certificate. Return an empty vector to indicate that no response is available, and thus suppress the Certificate_Status message.

Definition at line 226 of file tls_callbacks.h.

227 {
228 BOTAN_UNUSED(chain);
229 BOTAN_UNUSED(csr);
230 return std::vector<uint8_t>();
231 }

References BOTAN_UNUSED.

◆ tls_record_received()

virtual void Botan::TLS::Callbacks::tls_record_received ( uint64_t seq_no,
std::span< const uint8_t > data )
pure virtual

Mandatory callback: process application data Called when application data record is received from the peer. Again the array is overwritten immediately after the function returns.

Parameters
seq_nothe underlying TLS/DTLS record sequence number
dataa contiguous data buffer containing the received record

Referenced by Botan::TLS::Channel_Impl_13::from_peer().

◆ tls_server_choose_app_protocol()

std::string Botan::TLS::Callbacks::tls_server_choose_app_protocol ( const std::vector< std::string > & client_protos)
virtual

Optional callback for server: choose ALPN protocol

ALPN (RFC 7301) works by the client sending a list of application protocols it is willing to negotiate. The server then selects which protocol to use. RFC 7301 requires that if the server does not support any protocols offered by the client, then it should close the connection with an alert of no_application_protocol. Within this callback this would be done by throwing a TLS_Exception(Alert::NoApplicationProtocol)

Parameters
client_protosthe vector of protocols the client is willing to negotiate
Returns
the protocol selected by the server; if the empty string is returned, the server does not reply to the client ALPN extension.

The default implementation returns the empty string, causing client ALPN to be ignored.

It is highly recommended to support ALPN whenever possible to avoid cross-protocol attacks.

Definition at line 51 of file tls_callbacks.cpp.

51 {
52 return "";
53}

Referenced by Botan::TLS::Encrypted_Extensions::Encrypted_Extensions().

◆ tls_session_activated()

virtual void Botan::TLS::Callbacks::tls_session_activated ( )
inlinevirtual

Optional callback: session activated Called when a session is active and can be written to

Definition at line 93 of file tls_callbacks.h.

93{}

Referenced by Botan::TLS::Channel_Impl_12::activate_session().

◆ tls_session_established()

virtual void Botan::TLS::Callbacks::tls_session_established ( const Session_Summary & session)
inlinevirtual

Optional callback: session established Called when a session is established. Throw an exception to abort the connection.

Parameters
sessionthe session descriptor

Definition at line 87 of file tls_callbacks.h.

87{ BOTAN_UNUSED(session); }

References BOTAN_UNUSED.

◆ tls_should_persist_resumption_information()

bool Botan::TLS::Callbacks::tls_should_persist_resumption_information ( const Session & session)
virtual

Optional callback: Resumption information was received/established

TLS 1.3 calls this when we sent or received a TLS 1.3 session ticket at any point after the initial handshake has finished.

TLS 1.2 calls this when a session was established successfully and its resumption information may be stored for later usage.

Note that for servers this is called as soon as resumption information is available and could be sent to the client. If this callback returns 'false', the information will neither be cached nor sent.

Parameters
sessionthe session descriptor
Returns
false to prevent the resumption information from being cached, and true to cache it in the configured Session_Manager

Definition at line 71 of file tls_callbacks.cpp.

71 {
72 // RFC 5077 3.3
73 // The ticket_lifetime_hint field contains a hint from the server about
74 // how long the ticket should be stored. A value of zero is reserved to
75 // indicate that the lifetime of the ticket is unspecified.
76 //
77 // RFC 8446 4.6.1
78 // [A ticket_lifetime] of zero indicates that the ticket should be discarded
79 // immediately.
80 //
81 // By default we opt to keep all sessions, except for TLS 1.3 with a lifetime
82 // hint of zero.
83 return session.lifetime_hint().count() > 0 || session.version().is_pre_tls_13();
84}

References Botan::TLS::Protocol_Version::is_pre_tls_13(), Botan::TLS::Session::lifetime_hint(), and Botan::TLS::Session_Base::version().

◆ tls_sign_message()

std::vector< uint8_t > Botan::TLS::Callbacks::tls_sign_message ( const Private_Key & key,
RandomNumberGenerator & rng,
std::string_view padding,
Signature_Format format,
const std::vector< uint8_t > & msg )
virtual

Optional callback with default impl: sign a message

Default implementation uses PK_Signer::sign_message(). Override to provide a different approach, e.g. using an external device.

Parameters
keythe private key of the signer
rnga random number generator
paddingthe encoding method to be applied to the message
formatthe signature format
msgthe input data for the signature
Returns
the signature

Definition at line 141 of file tls_callbacks.cpp.

145 {
146 PK_Signer signer(key, rng, padding, format);
147
148 return signer.sign_message(msg, rng);
149}

References Botan::PK_Signer::sign_message().

Referenced by Botan::TLS::Certificate_Verify_12::Certificate_Verify_12(), Botan::TLS::Certificate_Verify_13::Certificate_Verify_13(), and Botan::TLS::Server_Key_Exchange::Server_Key_Exchange().

◆ tls_ssl_key_log_data()

virtual void Botan::TLS::Callbacks::tls_ssl_key_log_data ( std::string_view label,
std::span< const uint8_t > client_random,
std::span< const uint8_t > secret ) const
inlinevirtual

Optional callback: Allows access to a connection's secret data

Useful to implement the SSLKEYLOGFILE for connection debugging as specified in ietf.org/archive/id/draft-thomson-tls-keylogfile-00.html

Invoked if Policy::allow_ssl_key_log_file returns true.

Default implementation simply ignores the inputs.

Parameters
labelIdentifies the reported secret type See draft-thomson-tls-keylogfile-00 Section 3.1 and 3.2
client_randomrandom value from ClientHello message acting as an identifier of the TLS sessions
secretthe actual secret value

Definition at line 580 of file tls_callbacks.h.

582 {
583 BOTAN_UNUSED(label, client_random, secret);
584 }

References BOTAN_UNUSED.

◆ tls_verify_cert_chain()

void Botan::TLS::Callbacks::tls_verify_cert_chain ( const std::vector< X509_Certificate > & cert_chain,
const std::vector< std::optional< OCSP::Response > > & ocsp_responses,
const std::vector< Certificate_Store * > & trusted_roots,
Usage_Type usage,
std::string_view hostname,
const TLS::Policy & policy )
virtual

Optional callback with default impl: verify cert chain

Default implementation performs a standard PKIX validation and initiates network OCSP request for end-entity cert. Override to provide different behavior.

Check the certificate chain is valid up to a trusted root, and optionally (if hostname != "") that the hostname given is consistent with the leaf certificate.

This function should throw an exception derived from std::exception with an informative what() result if the certificate chain cannot be verified.

Parameters
cert_chainspecifies a certificate chain leading to a trusted root CA certificate.
ocsp_responsesthe server may have provided some
trusted_rootsthe list of trusted certificates
usagewhat this cert chain is being used for Usage_Type::TLS_SERVER_AUTH for server chains, Usage_Type::TLS_CLIENT_AUTH for client chains, Usage_Type::UNSPECIFIED for other uses
hostnamewhen authenticating a server, this is the hostname the client requested (eg via SNI). When authenticating a client, this is the server name the client is authenticating to. Empty in other cases or if no hostname was used.
policythe TLS policy associated with the session being authenticated using the certificate chain

Definition at line 86 of file tls_callbacks.cpp.

91 {
92 if(cert_chain.empty()) {
93 throw Invalid_Argument("Certificate chain was empty");
94 }
95
96 Path_Validation_Restrictions restrictions(policy.require_cert_revocation_info(),
97 policy.minimum_signature_strength());
98
99 Path_Validation_Result result = x509_path_validate(cert_chain,
100 restrictions,
101 trusted_roots,
102 hostname,
103 usage,
106 ocsp_responses);
107
108 if(!result.successful_validation()) {
109 throw TLS_Exception(Alert::BadCertificate, "Certificate validation failure: " + result.result_string());
110 }
111}
virtual std::chrono::milliseconds tls_verify_cert_chain_ocsp_timeout() const
virtual std::chrono::system_clock::time_point tls_current_timestamp()
Path_Validation_Result x509_path_validate(const std::vector< X509_Certificate > &end_certs, const Path_Validation_Restrictions &restrictions, const std::vector< Certificate_Store * > &trusted_roots, std::string_view hostname, Usage_Type usage, std::chrono::system_clock::time_point ref_time, std::chrono::milliseconds ocsp_timeout, const std::vector< std::optional< OCSP::Response > > &ocsp_resp)
Definition x509path.cpp:882

References Botan::TLS::Policy::minimum_signature_strength(), Botan::TLS::Policy::require_cert_revocation_info(), Botan::Path_Validation_Result::result_string(), Botan::Path_Validation_Result::successful_validation(), and Botan::x509_path_validate().

◆ tls_verify_cert_chain_ocsp_timeout()

virtual std::chrono::milliseconds Botan::TLS::Callbacks::tls_verify_cert_chain_ocsp_timeout ( ) const
inlinevirtual

Called by default tls_verify_cert_chain to get the timeout to use for OCSP requests. Return 0 to disable online OCSP checks.

This function should not be "const" since the implementation might need to perform some side effecting operation to compute the result.

Definition at line 212 of file tls_callbacks.h.

212 {
213 return std::chrono::milliseconds(0);
214 }

◆ tls_verify_message()

bool Botan::TLS::Callbacks::tls_verify_message ( const Public_Key & key,
std::string_view padding,
Signature_Format format,
const std::vector< uint8_t > & msg,
const std::vector< uint8_t > & sig )
virtual

Optional callback with default impl: verify a message signature

Default implementation uses PK_Verifier::verify_message(). Override to provide a different approach, e.g. using an external device.

Parameters
keythe public key of the signer
paddingthe encoding method to be applied to the message
formatthe signature format
msgthe input data for the signature
sigthe signature to be checked
Returns
true if the signature is valid, false otherwise

Definition at line 151 of file tls_callbacks.cpp.

155 {
156 PK_Verifier verifier(key, padding, format);
157
158 return verifier.verify_message(msg, sig);
159}

References Botan::PK_Verifier::verify_message().

Referenced by Botan::TLS::Certificate_Verify_12::verify(), Botan::TLS::Certificate_Verify_13::verify(), and Botan::TLS::Server_Key_Exchange::verify().

◆ tls_verify_raw_public_key()

void Botan::TLS::Callbacks::tls_verify_raw_public_key ( const Public_Key & raw_public_key,
Usage_Type usage,
std::string_view hostname,
const TLS::Policy & policy )
virtual

Optional callback. Default impl always rejects.

This allows using raw public keys for authentication of peers as described in RFC 7250 and RFC 8446 4.2.2. Applications that wish to use raw public keys MUST override this callback to verify the authenticity of the received public keys.

Default implementation always rejects the raw public key.

This function should throw an exception derived from std::exception with an informative what() result if the raw public key cannot be verified.

Parameters
raw_public_keyspecifies the raw public key to be used for peer authentication
usagewhat this cert chain is being used for Usage_Type::TLS_SERVER_AUTH for server chains, Usage_Type::TLS_CLIENT_AUTH for client chains,
hostnamewhen authenticating a server, this is the hostname the client requested (eg via SNI). When authenticating a client, this is the server name the client is authenticating to. Empty in other cases or if no hostname was used.
policythe TLS policy associated with the session being authenticated using the raw public key

Definition at line 113 of file tls_callbacks.cpp.

116 {
117 BOTAN_UNUSED(raw_public_key, usage, hostname, policy);
118 // There is no good default implementation for authenticating raw public key.
119 // Applications that wish to use them for authentication, must override this.
120 throw TLS_Exception(Alert::CertificateUnknown, "Application did not provide a means to validate the raw public key");
121}

References BOTAN_UNUSED.

Referenced by Botan::TLS::Certificate_13::verify().


The documentation for this class was generated from the following files: