// Copyright 2020 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef SHELL_BROWSER_NET_PROXYING_WEBSOCKET_H_ #define SHELL_BROWSER_NET_PROXYING_WEBSOCKET_H_ #include #include #include #include #include "base/optional.h" #include "components/keyed_service/core/keyed_service_shutdown_notifier.h" #include "content/public/browser/content_browser_client.h" #include "extensions/browser/api/web_request/web_request_info.h" #include "mojo/public/cpp/bindings/pending_receiver.h" #include "mojo/public/cpp/bindings/receiver.h" #include "mojo/public/cpp/bindings/remote.h" #include "services/network/public/cpp/resource_request.h" #include "services/network/public/mojom/network_context.mojom.h" #include "services/network/public/mojom/websocket.mojom.h" #include "shell/browser/net/web_request_api_interface.h" #include "url/gurl.h" #include "url/origin.h" namespace electron { // A ProxyingWebSocket proxies a WebSocket connection and dispatches // WebRequest API events. // // The code is referenced from the // extensions::WebRequestProxyingWebSocket class. class ProxyingWebSocket : public network::mojom::WebSocketHandshakeClient, public network::mojom::AuthenticationHandler, public network::mojom::TrustedHeaderClient { public: using WebSocketFactory = content::ContentBrowserClient::WebSocketFactory; // AuthRequiredResponse indicates how an OnAuthRequired call is handled. enum class AuthRequiredResponse { // No credentials were provided. AUTH_REQUIRED_RESPONSE_NO_ACTION, // AuthCredentials is filled in with a username and password, which should // be used in a response to the provided auth challenge. AUTH_REQUIRED_RESPONSE_SET_AUTH, // The request should be canceled. AUTH_REQUIRED_RESPONSE_CANCEL_AUTH, // The action will be decided asynchronously. |callback| will be invoked // when the decision is made, and one of the other AuthRequiredResponse // values will be passed in with the same semantics as described above. AUTH_REQUIRED_RESPONSE_IO_PENDING, }; ProxyingWebSocket( WebRequestAPI* web_request_api, WebSocketFactory factory, const network::ResourceRequest& request, mojo::PendingRemote handshake_client, bool has_extra_headers, int process_id, int render_frame_id, content::BrowserContext* browser_context, uint64_t* request_id_generator); ~ProxyingWebSocket() override; void Start(); // network::mojom::WebSocketHandshakeClient methods: void OnOpeningHandshakeStarted( network::mojom::WebSocketHandshakeRequestPtr request) override; void OnConnectionEstablished( mojo::PendingRemote websocket, mojo::PendingReceiver client_receiver, network::mojom::WebSocketHandshakeResponsePtr response, mojo::ScopedDataPipeConsumerHandle readable, mojo::ScopedDataPipeProducerHandle writable) override; // network::mojom::AuthenticationHandler method: void OnAuthRequired(const net::AuthChallengeInfo& auth_info, const scoped_refptr& headers, const net::IPEndPoint& remote_endpoint, OnAuthRequiredCallback callback) override; // network::mojom::TrustedHeaderClient methods: void OnBeforeSendHeaders(const net::HttpRequestHeaders& headers, OnBeforeSendHeadersCallback callback) override; void OnHeadersReceived(const std::string& headers, const net::IPEndPoint& endpoint, OnHeadersReceivedCallback callback) override; static void StartProxying( WebRequestAPI* web_request_api, WebSocketFactory factory, const GURL& url, const GURL& site_for_cookies, const base::Optional& user_agent, mojo::PendingRemote handshake_client, bool has_extra_headers, int process_id, int render_frame_id, const url::Origin& origin, content::BrowserContext* browser_context, uint64_t* request_id_generator); WebRequestAPI* web_request_api() { return web_request_api_; } private: void OnBeforeRequestComplete(int error_code); void OnBeforeSendHeadersComplete(const std::set& removed_headers, const std::set& set_headers, int error_code); void ContinueToStartRequest(int error_code); void OnHeadersReceivedComplete(int error_code); void ContinueToHeadersReceived(); void OnAuthRequiredComplete(AuthRequiredResponse rv); void OnHeadersReceivedCompleteForAuth(const net::AuthChallengeInfo& auth_info, int rv); void ContinueToCompleted(); void PauseIncomingMethodCallProcessing(); void ResumeIncomingMethodCallProcessing(); void OnError(int error_code); // This is used for detecting errors on mojo connection with the network // service. void OnMojoConnectionErrorWithCustomReason(uint32_t custom_reason, const std::string& description); // This is used for detecting errors on mojo connection with original client // (i.e., renderer). void OnMojoConnectionError(); // Passed from api::WebRequest. WebRequestAPI* web_request_api_; // Saved to feed the api::WebRequest. network::ResourceRequest request_; WebSocketFactory factory_; mojo::Remote forwarding_handshake_client_; mojo::Receiver receiver_as_handshake_client_{this}; mojo::Receiver receiver_as_auth_handler_{this}; mojo::Receiver receiver_as_header_client_{this}; net::HttpRequestHeaders request_headers_; network::mojom::URLResponseHeadPtr response_; net::AuthCredentials auth_credentials_; OnAuthRequiredCallback auth_required_callback_; scoped_refptr override_headers_; std::vector additional_headers_; OnBeforeSendHeadersCallback on_before_send_headers_callback_; OnHeadersReceivedCallback on_headers_received_callback_; GURL redirect_url_; bool is_done_ = false; bool has_extra_headers_; mojo::PendingRemote websocket_; mojo::PendingReceiver client_receiver_; network::mojom::WebSocketHandshakeResponsePtr handshake_response_ = nullptr; mojo::ScopedDataPipeConsumerHandle readable_; mojo::ScopedDataPipeProducerHandle writable_; extensions::WebRequestInfo info_; // Notifies the proxy that the browser context has been shutdown. std::unique_ptr shutdown_notifier_; base::WeakPtrFactory weak_factory_{this}; DISALLOW_COPY_AND_ASSIGN(ProxyingWebSocket); }; } // namespace electron #endif // SHELL_BROWSER_NET_PROXYING_WEBSOCKET_H_