// Copyright (C) 2018 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only #include "client_cert_qt.h" #include "base/functional/bind.h" #include "base/functional/callback_forward.h" #include "content/public/browser/browser_thread.h" #include "content/public/browser/browser_task_traits.h" #include "crypto/crypto_buildflags.h" #include "net/ssl/client_cert_store.h" #include "net/ssl/ssl_cert_request_info.h" #include "net/ssl/ssl_private_key.h" #include "net/cert/x509_certificate.h" #include "third_party/boringssl/src/include/openssl/pem.h" #include "third_party/boringssl/src/include/openssl/err.h" #include "third_party/boringssl/src/include/openssl/evp.h" #include "client_cert_store_data.h" #include #if BUILDFLAG(USE_NSS_CERTS) #include "net/ssl/client_cert_store_nss.h" #endif #if defined(Q_OS_WIN) #include "net/ssl/client_cert_store_win.h" #endif #if BUILDFLAG(IS_MAC) #include "net/ssl/client_cert_store_mac.h" #endif namespace { class ClientCertIdentityQt : public net::ClientCertIdentity { public: ClientCertIdentityQt(scoped_refptr cert, scoped_refptr key) : net::ClientCertIdentity(std::move(cert)), m_key(std::move(key)) {} ~ClientCertIdentityQt() override = default; void AcquirePrivateKey(base::OnceCallback)> private_key_callback) override { std::move(private_key_callback).Run(m_key); } private: scoped_refptr m_key; }; } // namespace namespace QtWebEngineCore { ClientCertStoreQt::ClientCertStoreQt(ClientCertificateStoreData *storeData) : ClientCertStore() , m_storeData(storeData) , m_nativeStore(createNativeStore()) { } ClientCertStoreQt::~ClientCertStoreQt() = default; #if QT_CONFIG(ssl) net::ClientCertIdentityList ClientCertStoreQt::GetClientCertsOnUIThread(const net::SSLCertRequestInfo &cert_request_info) { DCHECK_CURRENTLY_ON(content::BrowserThread::UI); const auto &clientCertOverrideData = m_storeData->extraCerts; // Look for certificates in memory store net::ClientCertIdentityList selected_identities; for (int i = 0; i < clientCertOverrideData.length(); i++) { scoped_refptr cert = clientCertOverrideData[i]->certPtr; if (cert) { if (cert->HasExpired()) { qWarning() << "Expired certificate" << clientCertOverrideData[i]; continue; } if (cert_request_info.cert_authorities.empty() || cert->IsIssuedByEncoded(cert_request_info.cert_authorities)) { selected_identities.push_back(std::make_unique( cert, clientCertOverrideData[i]->keyPtr)); } } } return selected_identities; } void ClientCertStoreQt::GetClientCertsReturn(const net::SSLCertRequestInfo &cert_request_info, ClientCertListCallback callback, net::ClientCertIdentityList &&result) { // Continue with native cert store and append them after memory certificates if (m_nativeStore) { ClientCertListCallback callback2 = base::BindOnce( [](ClientCertStoreQt::ClientCertListCallback callback, net::ClientCertIdentityList result1, net::ClientCertIdentityList result2) { while (!result2.empty()) { result1.push_back(std::move(result2.back())); result2.pop_back(); } std::move(callback).Run(std::move(result1)); }, std::move(callback), std::move(result)); m_nativeStore->GetClientCerts(cert_request_info, std::move(callback2)); } else { std::move(callback).Run(std::move(result)); } } #endif // QT_CONFIG(ssl) void ClientCertStoreQt::GetClientCerts(const net::SSLCertRequestInfo &cert_request_info, ClientCertListCallback callback) { #if QT_CONFIG(ssl) // Access the user-provided data from the UI thread, but return on whatever thread this is. bool ok = content::GetUIThreadTaskRunner({})->PostTaskAndReplyWithResult( FROM_HERE, base::BindOnce(&ClientCertStoreQt::GetClientCertsOnUIThread, base::Unretained(this), std::cref(cert_request_info)), base::BindOnce(&ClientCertStoreQt::GetClientCertsReturn, base::Unretained(this), std::cref(cert_request_info), std::move(callback))); DCHECK(ok); // callback is already moved and we can't really recover here. #else if (m_nativeStore) m_nativeStore->GetClientCerts(cert_request_info, std::move(callback)); else std::move(callback).Run(net::ClientCertIdentityList()); #endif // QT_CONFIG(ssl) } // static std::unique_ptr ClientCertStoreQt::createNativeStore() { #if BUILDFLAG(USE_NSS_CERTS) return std::unique_ptr(new net::ClientCertStoreNSS(net::ClientCertStoreNSS::PasswordDelegateFactory())); #elif defined(Q_OS_WIN) return std::unique_ptr(new net::ClientCertStoreWin()); #elif BUILDFLAG(IS_MAC) return std::unique_ptr(new net::ClientCertStoreMac()); #else return nullptr; #endif } } // namespace QtWebEngineCore