| 
									
										
										
										
											2020-02-11 14:56:09 +09:00
										 |  |  | // 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 <memory>
 | 
					
						
							|  |  |  | #include <set>
 | 
					
						
							|  |  |  | #include <string>
 | 
					
						
							|  |  |  | #include <vector>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "base/optional.h"
 | 
					
						
							|  |  |  | #include "components/keyed_service/core/keyed_service_shutdown_notifier.h"
 | 
					
						
							| 
									
										
										
										
											2020-04-28 23:16:10 -07:00
										 |  |  | #include "content/public/browser/content_browser_client.h"
 | 
					
						
							| 
									
										
										
										
											2020-02-11 14:56:09 +09:00
										 |  |  | #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; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-28 23:16:10 -07:00
										 |  |  |   // AuthRequiredResponse indicates how an OnAuthRequired call is handled.
 | 
					
						
							|  |  |  |   enum class AuthRequiredResponse { | 
					
						
							| 
									
										
										
										
											2020-10-13 10:25:21 -07:00
										 |  |  |     // No credentials were provided.
 | 
					
						
							| 
									
										
										
										
											2020-04-28 23:16:10 -07:00
										 |  |  |     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, | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-11 14:56:09 +09:00
										 |  |  |   ProxyingWebSocket( | 
					
						
							|  |  |  |       WebRequestAPI* web_request_api, | 
					
						
							|  |  |  |       WebSocketFactory factory, | 
					
						
							|  |  |  |       const network::ResourceRequest& request, | 
					
						
							|  |  |  |       mojo::PendingRemote<network::mojom::WebSocketHandshakeClient> | 
					
						
							|  |  |  |           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<network::mojom::WebSocket> websocket, | 
					
						
							|  |  |  |       mojo::PendingReceiver<network::mojom::WebSocketClient> client_receiver, | 
					
						
							|  |  |  |       network::mojom::WebSocketHandshakeResponsePtr response, | 
					
						
							| 
									
										
										
										
											2020-04-06 13:09:52 -07:00
										 |  |  |       mojo::ScopedDataPipeConsumerHandle readable, | 
					
						
							|  |  |  |       mojo::ScopedDataPipeProducerHandle writable) override; | 
					
						
							| 
									
										
										
										
											2020-02-11 14:56:09 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // network::mojom::AuthenticationHandler method:
 | 
					
						
							|  |  |  |   void OnAuthRequired(const net::AuthChallengeInfo& auth_info, | 
					
						
							|  |  |  |                       const scoped_refptr<net::HttpResponseHeaders>& 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<std::string>& user_agent, | 
					
						
							|  |  |  |       mojo::PendingRemote<network::mojom::WebSocketHandshakeClient> | 
					
						
							|  |  |  |           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<std::string>& removed_headers, | 
					
						
							|  |  |  |                                    const std::set<std::string>& set_headers, | 
					
						
							|  |  |  |                                    int error_code); | 
					
						
							|  |  |  |   void ContinueToStartRequest(int error_code); | 
					
						
							|  |  |  |   void OnHeadersReceivedComplete(int error_code); | 
					
						
							|  |  |  |   void ContinueToHeadersReceived(); | 
					
						
							| 
									
										
										
										
											2020-04-28 23:16:10 -07:00
										 |  |  |   void OnAuthRequiredComplete(AuthRequiredResponse rv); | 
					
						
							| 
									
										
										
										
											2020-02-11 14:56:09 +09:00
										 |  |  |   void OnHeadersReceivedCompleteForAuth(const net::AuthChallengeInfo& auth_info, | 
					
						
							|  |  |  |                                         int rv); | 
					
						
							|  |  |  |   void ContinueToCompleted(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   void PauseIncomingMethodCallProcessing(); | 
					
						
							|  |  |  |   void ResumeIncomingMethodCallProcessing(); | 
					
						
							| 
									
										
										
										
											2020-10-27 08:18:36 -07:00
										 |  |  |   void OnError(int error_code); | 
					
						
							| 
									
										
										
										
											2020-02-11 14:56:09 +09:00
										 |  |  |   // 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<network::mojom::WebSocketHandshakeClient> | 
					
						
							|  |  |  |       forwarding_handshake_client_; | 
					
						
							|  |  |  |   mojo::Receiver<network::mojom::WebSocketHandshakeClient> | 
					
						
							|  |  |  |       receiver_as_handshake_client_{this}; | 
					
						
							|  |  |  |   mojo::Receiver<network::mojom::AuthenticationHandler> | 
					
						
							|  |  |  |       receiver_as_auth_handler_{this}; | 
					
						
							|  |  |  |   mojo::Receiver<network::mojom::TrustedHeaderClient> | 
					
						
							|  |  |  |       receiver_as_header_client_{this}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   net::HttpRequestHeaders request_headers_; | 
					
						
							|  |  |  |   network::mojom::URLResponseHeadPtr response_; | 
					
						
							|  |  |  |   net::AuthCredentials auth_credentials_; | 
					
						
							|  |  |  |   OnAuthRequiredCallback auth_required_callback_; | 
					
						
							|  |  |  |   scoped_refptr<net::HttpResponseHeaders> override_headers_; | 
					
						
							|  |  |  |   std::vector<network::mojom::HttpHeaderPtr> 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<network::mojom::WebSocket> websocket_; | 
					
						
							|  |  |  |   mojo::PendingReceiver<network::mojom::WebSocketClient> client_receiver_; | 
					
						
							|  |  |  |   network::mojom::WebSocketHandshakeResponsePtr handshake_response_ = nullptr; | 
					
						
							|  |  |  |   mojo::ScopedDataPipeConsumerHandle readable_; | 
					
						
							| 
									
										
										
										
											2020-04-06 13:09:52 -07:00
										 |  |  |   mojo::ScopedDataPipeProducerHandle writable_; | 
					
						
							| 
									
										
										
										
											2020-02-11 14:56:09 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  |   extensions::WebRequestInfo info_; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Notifies the proxy that the browser context has been shutdown.
 | 
					
						
							|  |  |  |   std::unique_ptr<KeyedServiceShutdownNotifier::Subscription> | 
					
						
							|  |  |  |       shutdown_notifier_; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   base::WeakPtrFactory<ProxyingWebSocket> weak_factory_{this}; | 
					
						
							|  |  |  |   DISALLOW_COPY_AND_ASSIGN(ProxyingWebSocket); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }  // namespace electron
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif  // SHELL_BROWSER_NET_PROXYING_WEBSOCKET_H_
 |