Skip to content

Commit

Permalink
Merge pull request #19792 from michael-redpanda/CORE-2426-Support-OSS…
Browse files Browse the repository at this point in the history
…L-Cipher-Strings

CORE-2426 support ossl cipher strings
  • Loading branch information
michael-redpanda authored Jun 12, 2024
2 parents b9d66b7 + e578efe commit 5ec1849
Show file tree
Hide file tree
Showing 7 changed files with 227 additions and 151 deletions.
10 changes: 10 additions & 0 deletions src/v/cloud_storage_clients/configuration.cc
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@

#include "cloud_storage_clients/logger.h"
#include "config/configuration.h"
#include "config/tls_config.h"
#include "net/tls.h"
#include "net/tls_certificate_probe.h"

Expand All @@ -25,11 +26,20 @@ build_tls_credentials(
std::optional<cloud_storage_clients::ca_trust_file> trust_file,
ss::logger& log) {
ss::tls::credentials_builder cred_builder;
#ifdef SEASTAR_WITH_TLS_OSSL
cred_builder.set_cipher_string(
{config::tlsv1_2_cipher_string.data(),
config::tlsv1_2_cipher_string.size()});
cred_builder.set_ciphersuites(
{config::tlsv1_3_ciphersuites.data(),
config::tlsv1_3_ciphersuites.size()});
#else
// NOTE: this is a pre-defined gnutls priority string that
// picks the ciphersuites with 128-bit ciphers which
// leads to up to 10x improvement in upload speed, compared
// to 256-bit ciphers
cred_builder.set_priority_string("PERFORMANCE");
#endif
if (trust_file.has_value()) {
auto file = trust_file.value();
vlog(log.info, "Use non-default trust file {}", file());
Expand Down
1 change: 1 addition & 0 deletions src/v/config/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ v_cc_library(
rjson_serialization.cc
validators.cc
throughput_control_group.cc
tls_config.cc
DEPS
v::json
v::model
Expand Down
180 changes: 180 additions & 0 deletions src/v/config/tls_config.cc
Original file line number Diff line number Diff line change
@@ -0,0 +1,180 @@
/*
* Copyright 2024 Redpanda Data, Inc.
*
* Use of this software is governed by the Business Source License
* included in the file licenses/BSL.md
*
* As of the Change Date specified in that file, in accordance with
* the Business Source License, use of this software will be governed
* by the Apache License, Version 2.0
*/

#include "tls_config.h"

#include "config/convert.h"
#include "utils/to_string.h"

#include <seastar/core/do_with.hh>
#include <seastar/core/shared_ptr.hh>

#include <boost/filesystem.hpp>

namespace config {
ss::future<std::optional<ss::tls::credentials_builder>>
tls_config::get_credentials_builder() const& {
if (_enabled) {
return ss::do_with(
ss::tls::credentials_builder{},
[this](ss::tls::credentials_builder& builder) {
#ifdef SEASTAR_WITH_TLS_OSSL
builder.enable_server_precedence();
builder.set_cipher_string(
{tlsv1_2_cipher_string.data(), tlsv1_2_cipher_string.size()});
builder.set_ciphersuites(
{tlsv1_3_ciphersuites.data(), tlsv1_3_ciphersuites.size()});
#else
builder.set_priority_string("PERFORMANCE:%SERVER_PRECEDENCE");
#endif
builder.set_dh_level(ss::tls::dh_params::level::MEDIUM);
if (_require_client_auth) {
builder.set_client_auth(ss::tls::client_auth::REQUIRE);
}

auto f = _truststore_file ? builder.set_x509_trust_file(
*_truststore_file, ss::tls::x509_crt_format::PEM)
: builder.set_system_trust();

if (_crl_file) {
f = f.then([this, &builder] {
return builder.set_x509_crl_file(
*_crl_file, ss::tls::x509_crt_format::PEM);
});
}

if (_key_cert) {
f = f.then([this, &builder] {
return builder.set_x509_key_file(
(*_key_cert).cert_file,
(*_key_cert).key_file,
ss::tls::x509_crt_format::PEM);
});
}

return f.then([&builder]() {
return std::make_optional(std::move(builder));
});
});
}

return ss::make_ready_future<std::optional<ss::tls::credentials_builder>>(
std::nullopt);
}

ss::future<std::optional<ss::tls::credentials_builder>>
tls_config::get_credentials_builder() && {
auto ptr = ss::make_lw_shared(std::move(*this));
return ptr->get_credentials_builder().finally([ptr] {});
}

std::optional<ss::sstring> tls_config::validate(const tls_config& c) {
if (c.get_require_client_auth() && !c.get_truststore_file()) {
return "Trust store is required when client authentication is "
"enabled";
}

return std::nullopt;
}

std::ostream& operator<<(std::ostream& o, const config::key_cert& c) {
o << "{ "
<< "key_file: " << c.key_file << " "
<< "cert_file: " << c.cert_file << " }";
return o;
}

std::ostream& operator<<(std::ostream& o, const config::tls_config& c) {
o << "{ "
<< "enabled: " << c.is_enabled() << " "
<< "key/cert files: " << c.get_key_cert_files() << " "
<< "ca file: " << c.get_truststore_file() << " "
<< "crl file: " << c.get_crl_file() << " "
<< "client_auth_required: " << c.get_require_client_auth() << ""
<< " }";
return o;
}
} // namespace config

namespace YAML {

inline ss::sstring to_absolute(const ss::sstring& path) {
namespace fs = std::filesystem;
if (path.empty()) {
return path;
}
return fs::absolute(fs::path(path)).native();
}

inline std::optional<ss::sstring>
to_absolute(const std::optional<ss::sstring>& path) {
if (path) {
return to_absolute(*path);
}
return std::nullopt;
}

Node convert<config::tls_config>::encode(const config::tls_config& rhs) {
Node node;

node["enabled"] = rhs.is_enabled();
node["require_client_auth"] = rhs.get_require_client_auth();

if (rhs.get_key_cert_files()) {
node["cert_file"] = (*rhs.get_key_cert_files()).key_file;
node["key_file"] = (*rhs.get_key_cert_files()).cert_file;
}

if (rhs.get_truststore_file()) {
node["truststore_file"] = *rhs.get_truststore_file();
}

return node;
}

std::optional<ss::sstring> convert<config::tls_config>::read_optional(
const Node& node, const ss::sstring& key) {
if (node[key]) {
return node[key].as<ss::sstring>();
}
return std::nullopt;
}

bool convert<config::tls_config>::decode(
const Node& node, config::tls_config& rhs) {
// either both true or both false
if (
static_cast<bool>(node["key_file"])
^ static_cast<bool>(node["cert_file"])) {
return false;
}
auto enabled = node["enabled"] && node["enabled"].as<bool>();
if (!enabled) {
rhs = config::tls_config(
false, std::nullopt, std::nullopt, std::nullopt, false);
} else {
auto key_cert = node["key_file"] ? std::make_optional<config::key_cert>(
config::key_cert{
to_absolute(node["key_file"].as<ss::sstring>()),
to_absolute(node["cert_file"].as<ss::sstring>())})
: std::nullopt;
rhs = config::tls_config(
enabled,
key_cert,
to_absolute(read_optional(node, "truststore_file")),
to_absolute(read_optional(node, "crl_file")),
node["require_client_auth"]
&& node["require_client_auth"].as<bool>());
}
return true;
}

} // namespace YAML
Loading

0 comments on commit 5ec1849

Please sign in to comment.