#include "services/cert_verifier/cert_verifier_service.h"
#include "base/functional/bind.h"
#include "base/logging.h"
#include "mojo/public/cpp/bindings/message.h"
#include "mojo/public/cpp/bindings/remote.h"
#include "net/base/completion_once_callback.h"
#include "net/cert/cert_verify_result.h"
#include "net/cert/crl_set.h"
#include "services/cert_verifier/cert_net_url_loader/cert_net_fetcher_url_loader.h"
#include "services/cert_verifier/cert_verifier_service_factory.h"
#include "services/network/public/mojom/cert_verifier_service.mojom.h"
namespace cert_verifier {
namespace internal {
namespace {
class CertVerifyResultHelper {
public:
CertVerifyResultHelper() = default;
void Initialize(mojo::PendingRemote<mojom::CertVerifierRequest> request,
std::unique_ptr<net::CertVerifyResult> result) {
request_.Bind(std::move(request));
result_ = std::move(result);
request_.set_disconnect_handler(base::BindOnce(
&CertVerifyResultHelper::DisconnectRequest, base::Unretained(this)));
}
void CompleteCertVerifierRequest(int net_error) {
DCHECK(request_.is_bound());
DCHECK(local_request_);
DCHECK(result_);
request_->Complete(*result_, net_error);
}
void DisconnectRequest() {
DCHECK(request_.is_bound());
DCHECK(local_request_);
DCHECK(result_);
local_request_.reset();
}
std::unique_ptr<net::CertVerifier::Request>* local_request() {
return &local_request_;
}
private:
mojo::Remote<mojom::CertVerifierRequest> request_;
std::unique_ptr<net::CertVerifyResult> result_;
std::unique_ptr<net::CertVerifier::Request> local_request_;
};
void ReconnectURLLoaderFactory(
mojo::Remote<mojom::URLLoaderFactoryConnector>* reconnector,
mojo::PendingReceiver<network::mojom::URLLoaderFactory> receiver) {
(*reconnector)->CreateURLLoaderFactory(std::move(receiver));
}
}
CertVerifierServiceImpl::CertVerifierServiceImpl(
std::unique_ptr<net::CertVerifierWithUpdatableProc> verifier,
mojo::PendingReceiver<mojom::CertVerifierService> receiver,
mojo::PendingRemote<mojom::CertVerifierServiceClient> client,
scoped_refptr<CertNetFetcherURLLoader> cert_net_fetcher)
: verifier_(std::move(verifier)),
receiver_(this, std::move(receiver)),
client_(std::move(client)),
cert_net_fetcher_(std::move(cert_net_fetcher)) {
receiver_.set_disconnect_handler(
base::BindRepeating(&CertVerifierServiceImpl::OnDisconnectFromService,
base::Unretained(this)));
verifier_->AddObserver(this);
}
CertVerifierServiceImpl::~CertVerifierServiceImpl() {
verifier_->RemoveObserver(this);
if (cert_net_fetcher_)
cert_net_fetcher_->Shutdown();
}
void CertVerifierServiceImpl::SetConfig(
const net::CertVerifier::Config& config) {
verifier_->SetConfig(config);
}
void CertVerifierServiceImpl::EnableNetworkAccess(
mojo::PendingRemote<network::mojom::URLLoaderFactory> url_loader_factory,
mojo::PendingRemote<mojom::URLLoaderFactoryConnector> reconnector) {
if (cert_net_fetcher_) {
auto reconnect_cb =
std::make_unique<mojo::Remote<mojom::URLLoaderFactoryConnector>>(
std::move(reconnector));
cert_net_fetcher_->SetURLLoaderFactoryAndReconnector(
std::move(url_loader_factory),
base::BindRepeating(&ReconnectURLLoaderFactory,
base::Owned(std::move(reconnect_cb))));
}
}
void CertVerifierServiceImpl::SetCertVerifierServiceFactory(
base::WeakPtr<cert_verifier::CertVerifierServiceFactoryImpl>
service_factory_impl) {
service_factory_impl_ = std::move(service_factory_impl);
}
void CertVerifierServiceImpl::UpdateVerifierData(
const net::CertVerifyProcFactory::ImplParams& impl_params) {
verifier_->UpdateVerifyProcData(cert_net_fetcher_, impl_params);
}
void CertVerifierServiceImpl::Verify(
const net::CertVerifier::RequestParams& params,
uint32_t netlog_source_type,
uint32_t netlog_source_id,
base::TimeTicks netlog_source_start_time,
mojo::PendingRemote<mojom::CertVerifierRequest> cert_verifier_request) {
DVLOG(3) << "Received certificate validation request for hostname: "
<< params.hostname();
auto result = std::make_unique<net::CertVerifyResult>();
auto result_helper = std::make_unique<CertVerifyResultHelper>();
CertVerifyResultHelper* result_helper_ptr = result_helper.get();
net::CompletionOnceCallback callback =
base::BindOnce(&CertVerifyResultHelper::CompleteCertVerifierRequest,
std::move(result_helper));
if (netlog_source_type >=
static_cast<uint32_t>(net::NetLogSourceType::COUNT)) {
receiver_.ReportBadMessage("invalid NetLogSource");
return;
}
int net_err = verifier_->Verify(
params, result.get(), std::move(callback),
result_helper_ptr->local_request(),
net::NetLogWithSource::Make(
net::NetLog::Get(),
net::NetLogSource(
static_cast<net::NetLogSourceType>(netlog_source_type),
netlog_source_id, netlog_source_start_time)));
if (net_err == net::ERR_IO_PENDING) {
result_helper_ptr->Initialize(std::move(cert_verifier_request),
std::move(result));
} else {
mojo::Remote<mojom::CertVerifierRequest> remote(
std::move(cert_verifier_request));
remote->Complete(*result, net_err);
}
}
void CertVerifierServiceImpl::OnCertVerifierChanged() {
client_->OnCertVerifierChanged();
}
void CertVerifierServiceImpl::OnDisconnectFromService() {
if (service_factory_impl_) {
service_factory_impl_->RemoveService(this);
}
delete this;
}
}
}