// Copyright (c) 2018 GitHub, Inc. // Use of this source code is governed by the MIT license that can be // found in the LICENSE file. #include "atom/browser/net/resolve_proxy_helper.h" #include "atom/browser/atom_browser_context.h" #include "base/bind.h" #include "base/threading/thread_task_runner_handle.h" #include "net/url_request/url_request_context.h" #include "net/url_request/url_request_context_getter.h" namespace atom { ResolveProxyHelper::ResolveProxyHelper(AtomBrowserContext* browser_context) : context_getter_(browser_context->GetRequestContext()), original_thread_(base::ThreadTaskRunnerHandle::Get()) {} ResolveProxyHelper::~ResolveProxyHelper() { // Clear all pending requests if the ProxyService is still alive. pending_requests_.clear(); } void ResolveProxyHelper::ResolveProxy(const GURL& url, const ResolveProxyCallback& callback) { // Enqueue the pending request. pending_requests_.push_back(PendingRequest(url, callback)); // If nothing is in progress, start. if (pending_requests_.size() == 1) StartPendingRequest(); } void ResolveProxyHelper::StartPendingRequest() { auto& pending_request = pending_requests_.front(); context_getter_->GetNetworkTaskRunner()->PostTask( FROM_HERE, base::BindOnce(&ResolveProxyHelper::StartPendingRequestInIO, base::Unretained(this), pending_request.url)); } void ResolveProxyHelper::StartPendingRequestInIO(const GURL& url) { auto* proxy_service = context_getter_->GetURLRequestContext()->proxy_resolution_service(); // Start the request. int result = proxy_service->ResolveProxy( url, std::string(), &proxy_info_, base::Bind(&ResolveProxyHelper::OnProxyResolveComplete, base::RetainedRef(this)), nullptr, nullptr, net::NetLogWithSource()); // Completed synchronously. if (result != net::ERR_IO_PENDING) OnProxyResolveComplete(result); } void ResolveProxyHelper::OnProxyResolveComplete(int result) { DCHECK(!pending_requests_.empty()); std::string proxy; if (result == net::OK) proxy = proxy_info_.ToPacString(); original_thread_->PostTask( FROM_HERE, base::BindOnce(&ResolveProxyHelper::SendProxyResult, base::RetainedRef(this), proxy)); } void ResolveProxyHelper::SendProxyResult(const std::string& proxy) { DCHECK(!pending_requests_.empty()); const auto& completed_request = pending_requests_.front(); if (!completed_request.callback.is_null()) completed_request.callback.Run(proxy); // Clear the current (completed) request. pending_requests_.pop_front(); // Start the next request. if (!pending_requests_.empty()) StartPendingRequest(); } ResolveProxyHelper::PendingRequest::PendingRequest( const GURL& url, const ResolveProxyCallback& callback) : url(url), callback(callback) {} ResolveProxyHelper::PendingRequest::PendingRequest( ResolveProxyHelper::PendingRequest&& pending_request) = default; ResolveProxyHelper::PendingRequest::~PendingRequest() noexcept = default; ResolveProxyHelper::PendingRequest& ResolveProxyHelper::PendingRequest:: operator=(ResolveProxyHelper::PendingRequest&& pending_request) noexcept = default; } // namespace atom