blob: 418b91ddafaa97a3d32e924c0e6b511f3b9dba2a [file] [log] [blame]
// Copyright 2014 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef CONTENT_BROWSER_DEVTOOLS_PROTOCOL_NETWORK_HANDLER_H_
#define CONTENT_BROWSER_DEVTOOLS_PROTOCOL_NETWORK_HANDLER_H_
#include <memory>
#include <optional>
#include <vector>
#include "base/containers/flat_set.h"
#include "base/containers/unique_ptr_adapters.h"
#include "base/memory/raw_ptr.h"
#include "base/memory/weak_ptr.h"
#include "base/unguessable_token.h"
#include "content/browser/devtools/protocol/devtools_domain_handler.h"
#include "content/browser/devtools/protocol/network.h"
#include "mojo/public/cpp/bindings/receiver.h"
#include "mojo/public/cpp/system/data_pipe.h"
#include "net/base/net_errors.h"
#include "net/cookies/canonical_cookie.h"
#include "net/filter/source_stream.h"
#include "net/net_buildflags.h"
#include "services/network/public/mojom/devtools_observer.mojom-forward.h"
#include "services/network/public/mojom/http_raw_headers.mojom-forward.h"
#include "services/network/public/mojom/network_context.mojom.h"
#include "services/network/public/mojom/url_response_head.mojom.h"
#include "third_party/blink/public/mojom/loader/resource_load_info.mojom-shared.h"
#if BUILDFLAG(ENABLE_REPORTING)
#include "services/network/public/mojom/reporting_service.mojom.h"
#endif // BUILDFLAG(ENABLE_REPORTING)
namespace net {
class HttpRequestHeaders;
class SSLInfo;
class X509Certificate;
} // namespace net
namespace network {
struct CorsErrorStatus;
struct ResourceRequest;
struct URLLoaderCompletionStatus;
namespace mojom {
class URLLoaderFactoryOverride;
}
} // namespace network
namespace content {
class BrowserContext;
class DevToolsAgentHostImpl;
class DevToolsIOContext;
class DevToolsURLLoaderInterceptor;
class RenderFrameHostImpl;
class NavigationRequest;
class SignedExchangeEnvelope;
class StoragePartition;
struct InterceptedRequestInfo;
struct SignedExchangeError;
namespace protocol {
class BackgroundSyncRestorer;
class DevToolsNetworkResourceLoader;
class NetworkHandler : public DevToolsDomainHandler,
#if BUILDFLAG(ENABLE_REPORTING)
public network::mojom::ReportingApiObserver,
#endif // BUILDFLAG(ENABLE_REPORTING)
public Network::Backend {
public:
NetworkHandler(const std::string& host_id,
const base::UnguessableToken& devtools_token,
DevToolsIOContext* io_context,
base::RepeatingClosure update_loader_factories_callback,
bool allow_file_access,
bool client_is_trusted);
NetworkHandler(const NetworkHandler&) = delete;
NetworkHandler& operator=(const NetworkHandler&) = delete;
~NetworkHandler() override;
static std::vector<NetworkHandler*> ForAgentHost(DevToolsAgentHostImpl* host);
// static helpers used by other agents that depend on types defined
// in network domain.
static net::Error NetErrorFromString(const std::string& error, bool* ok);
static std::string NetErrorToString(int net_error);
static const char* ResourceTypeToString(
blink::mojom::ResourceType resource_type);
static bool AddInterceptedResourceType(
const std::string& resource_type,
base::flat_set<blink::mojom::ResourceType>* intercepted_resource_types);
static std::unique_ptr<Array<Network::Cookie>> BuildCookieArray(
const std::vector<net::CanonicalCookie>& cookie_list);
static void SetCookies(
StoragePartition* storage_partition,
std::unique_ptr<protocol::Array<Network::CookieParam>> cookies,
base::OnceCallback<void(bool)> callback);
void Wire(UberDispatcher* dispatcher) override;
void SetRenderer(int render_process_id,
RenderFrameHostImpl* frame_host) override;
Response Enable(Maybe<int> max_total_size,
Maybe<int> max_resource_size,
Maybe<int> max_post_data_size) override;
Response Disable() override;
#if BUILDFLAG(ENABLE_REPORTING)
void OnReportAdded(const net::ReportingReport& report) override;
void OnReportUpdated(const net::ReportingReport& report) override;
void OnEndpointsUpdatedForOrigin(
const std::vector<net::ReportingEndpoint>& endpoints) override;
std::unique_ptr<protocol::Network::ReportingApiReport> BuildProtocolReport(
const net::ReportingReport& report);
std::unique_ptr<protocol::Network::ReportingApiEndpoint>
BuildProtocolEndpoint(const net::ReportingEndpoint& endpoint);
#endif // BUILDFLAG(ENABLE_REPORTING)
Response EnableReportingApi(bool enable) override;
Response SetCacheDisabled(bool cache_disabled) override;
Response SetAcceptedEncodings(
std::unique_ptr<Array<Network::ContentEncoding>> encodings) override;
Response ClearAcceptedEncodingsOverride() override;
void ClearBrowserCache(
std::unique_ptr<ClearBrowserCacheCallback> callback) override;
void ClearBrowserCookies(
std::unique_ptr<ClearBrowserCookiesCallback> callback) override;
void GetCookies(Maybe<protocol::Array<String>> urls,
std::unique_ptr<GetCookiesCallback> callback) override;
void GetAllCookies(std::unique_ptr<GetAllCookiesCallback> callback) override;
void DeleteCookies(const std::string& name,
Maybe<std::string> url,
Maybe<std::string> domain,
Maybe<std::string> path,
Maybe<std::string> partition_key,
std::unique_ptr<DeleteCookiesCallback> callback) override;
void SetCookie(const std::string& name,
const std::string& value,
Maybe<std::string> url,
Maybe<std::string> domain,
Maybe<std::string> path,
Maybe<bool> secure,
Maybe<bool> http_only,
Maybe<std::string> same_site,
Maybe<double> expires,
Maybe<std::string> priority,
Maybe<bool> same_party,
Maybe<std::string> source_scheme,
Maybe<int> source_port,
Maybe<std::string> partition_key,
std::unique_ptr<SetCookieCallback> callback) override;
void SetCookies(
std::unique_ptr<protocol::Array<Network::CookieParam>> cookies,
std::unique_ptr<SetCookiesCallback> callback) override;
Response SetExtraHTTPHeaders(
std::unique_ptr<Network::Headers> headers) override;
Response CanEmulateNetworkConditions(bool* result) override;
Response EmulateNetworkConditions(
bool offline,
double latency,
double download_throughput,
double upload_throughput,
Maybe<protocol::Network::ConnectionType> connection_type,
Maybe<double> packet_loss,
Maybe<int> packet_queue_length,
Maybe<bool> packet_reordering) override;
Response SetBypassServiceWorker(bool bypass) override;
DispatchResponse SetRequestInterception(
std::unique_ptr<protocol::Array<protocol::Network::RequestPattern>>
patterns) override;
void ContinueInterceptedRequest(
const std::string& request_id,
Maybe<std::string> error_reason,
Maybe<protocol::Binary> raw_response,
Maybe<std::string> url,
Maybe<std::string> method,
Maybe<std::string> post_data,
Maybe<protocol::Network::Headers> headers,
Maybe<protocol::Network::AuthChallengeResponse> auth_challenge_response,
std::unique_ptr<ContinueInterceptedRequestCallback> callback) override;
void GetResponseBodyForInterception(
const String& interception_id,
std::unique_ptr<GetResponseBodyForInterceptionCallback> callback)
override;
void GetResponseBody(
const String& request_id,
std::unique_ptr<GetResponseBodyCallback> callback) override;
void TakeResponseBodyForInterceptionAsStream(
const String& interception_id,
std::unique_ptr<TakeResponseBodyForInterceptionAsStreamCallback> callback)
override;
// Note that |frame_token| below is for the frame that is associated with the
// factory being created, and is therefore not necessarily the same as one
// associated with the NetworkHandler itself (which is the token of the local
// root frame).
bool MaybeCreateProxyForInterception(
int process_id,
StoragePartition* storage_partition,
const base::UnguessableToken& frame_token,
bool is_navigation,
bool is_download,
network::mojom::URLLoaderFactoryOverride* intercepting_factory);
void ApplyOverrides(net::HttpRequestHeaders* headers,
bool* skip_service_worker,
bool* disable_cache,
std::optional<std::vector<net::SourceStream::SourceType>>*
accepted_stream_types);
void PrefetchRequestWillBeSent(
const std::string& request_id,
const network::ResourceRequest& request,
const GURL& initiator_url,
Maybe<std::string> frame_token,
base::TimeTicks timestamp,
std::optional<
std::pair<const GURL&,
const network::mojom::URLResponseHeadDevToolsInfo&>>
redirect_info);
void NavigationRequestWillBeSent(const NavigationRequest& nav_request,
base::TimeTicks timestamp);
void RequestSent(const std::string& request_id,
const std::string& loader_id,
const net::HttpRequestHeaders& request_headers,
const network::mojom::URLRequestDevToolsInfo& request_info,
const char* initiator_type,
const std::optional<GURL>& initiator_url,
const std::string& initiator_devtools_request_id,
base::TimeTicks timestamp);
void ResponseReceived(const std::string& request_id,
const std::string& loader_id,
const GURL& url,
const char* resource_type,
const network::mojom::URLResponseHeadDevToolsInfo& head,
Maybe<std::string> frame_id);
void LoadingComplete(
const std::string& request_id,
const char* resource_type,
const network::URLLoaderCompletionStatus& completion_status);
void FetchKeepAliveRequestWillBeSent(
const std::string& request_id,
const network::ResourceRequest& request,
const GURL& initiator_url,
Maybe<std::string> frame_token,
base::TimeTicks timestamp,
std::optional<
std::pair<const GURL&,
const network::mojom::URLResponseHeadDevToolsInfo&>>
redirect_info);
void OnSignedExchangeReceived(
std::optional<const base::UnguessableToken> devtools_navigation_token,
const GURL& outer_request_url,
const network::mojom::URLResponseHead& outer_response,
const std::optional<SignedExchangeEnvelope>& header,
const scoped_refptr<net::X509Certificate>& certificate,
const std::optional<net::SSLInfo>& ssl_info,
const std::vector<SignedExchangeError>& errors);
DispatchResponse GetSecurityIsolationStatus(
Maybe<String> in_frameId,
std::unique_ptr<protocol::Network::SecurityIsolationStatus>* out_info)
override;
void OnRequestWillBeSentExtraInfo(
const std::string& devtools_request_id,
const net::CookieAccessResultList& request_cookie_list,
const std::vector<network::mojom::HttpRawHeaderPairPtr>& request_headers,
const base::TimeTicks timestamp,
const network::mojom::ClientSecurityStatePtr& security_state,
const network::mojom::OtherPartitionInfoPtr& other_partition_info);
void OnResponseReceivedExtraInfo(
const std::string& devtools_request_id,
const net::CookieAndLineAccessResultList& response_cookie_list,
const std::vector<network::mojom::HttpRawHeaderPairPtr>& response_headers,
const std::optional<std::string>& response_headers_text,
network::mojom::IPAddressSpace resource_address_space,
int32_t http_status_code,
const std::optional<net::CookiePartitionKey>& cookie_partition_key);
void OnTrustTokenOperationDone(
const std::string& devtools_request_id,
const network::mojom::TrustTokenOperationResult& result);
void OnSubresourceWebBundleMetadata(const std::string& devtools_request_id,
const std::vector<GURL>& urls);
void OnSubresourceWebBundleMetadataError(
const std::string& devtools_request_id,
const std::string& error_message);
void OnSubresourceWebBundleInnerResponse(
const std::string& inner_request_devtools_id,
const GURL& url,
const std::optional<std::string>& bundle_request_devtools_id);
void OnSubresourceWebBundleInnerResponseError(
const std::string& inner_request_devtools_id,
const GURL& url,
const std::string& error_message,
const std::optional<std::string>& bundle_request_devtools_id);
bool enabled() const { return enabled_; }
Network::Frontend* frontend() const { return frontend_.get(); }
static std::string ExtractFragment(const GURL& url, std::string* fragment);
static std::unique_ptr<Network::Request> CreateRequestFromResourceRequest(
const network::ResourceRequest& request,
const std::string& cookie_line);
void LoadNetworkResource(
Maybe<content::protocol::String> frameId,
const String& url,
std::unique_ptr<protocol::Network::LoadNetworkResourceOptions> options,
std::unique_ptr<LoadNetworkResourceCallback> callback) override;
// Protocol builders.
static String BuildPrivateNetworkRequestPolicy(
network::mojom::PrivateNetworkRequestPolicy policy);
static protocol::Network::IPAddressSpace BuildIpAddressSpace(
network::mojom::IPAddressSpace space);
static std::unique_ptr<protocol::Network::ClientSecurityState>
MaybeBuildClientSecurityState(
const network::mojom::ClientSecurityStatePtr& state);
static std::unique_ptr<protocol::Network::CorsErrorStatus>
BuildCorsErrorStatus(const network::CorsErrorStatus& status);
void BodyDataReceived(const String& request_id,
const String& body,
bool is_base64_encoded);
private:
void OnLoadNetworkResourceFinished(DevToolsNetworkResourceLoader* loader,
const net::HttpResponseHeaders* rh,
bool success,
int net_error,
std::string content);
void RequestIntercepted(std::unique_ptr<InterceptedRequestInfo> request_info);
void SetNetworkConditions(network::mojom::NetworkConditionsPtr conditions);
void OnResponseBodyPipeTaken(
std::unique_ptr<TakeResponseBodyForInterceptionAsStreamCallback> callback,
Response response,
mojo::ScopedDataPipeConsumerHandle pipe,
const std::string& mime_type);
// TODO(dgozman): Remove this.
const std::string host_id_;
const base::UnguessableToken devtools_token_;
DevToolsIOContext* const io_context_;
const bool allow_file_access_;
const bool client_is_trusted_;
std::unique_ptr<Network::Frontend> frontend_;
raw_ptr<BrowserContext> browser_context_;
StoragePartition* storage_partition_;
RenderFrameHostImpl* host_;
bool enabled_;
#if BUILDFLAG(ENABLE_REPORTING)
mojo::Receiver<network::mojom::ReportingApiObserver> reporting_receiver_;
#endif // BUILDFLAG(ENABLE_REPORTING)
std::vector<std::pair<std::string, std::string>> extra_headers_;
std::unique_ptr<DevToolsURLLoaderInterceptor> url_loader_interceptor_;
bool bypass_service_worker_;
bool cache_disabled_;
std::unique_ptr<BackgroundSyncRestorer> background_sync_restorer_;
base::RepeatingClosure update_loader_factories_callback_;
std::map<std::unique_ptr<DevToolsNetworkResourceLoader>,
std::unique_ptr<LoadNetworkResourceCallback>,
base::UniquePtrComparator>
loaders_;
std::optional<std::set<net::SourceStream::SourceType>> accepted_stream_types_;
std::unordered_map<String, std::pair<String, bool>> received_body_data_;
base::WeakPtrFactory<NetworkHandler> weak_factory_{this};
};
} // namespace protocol
} // namespace content
#endif // CONTENT_BROWSER_DEVTOOLS_PROTOCOL_NETWORK_HANDLER_H_