blob: b576b0f9c31e07cf03837ea413fba930d63e5179 [file] [log] [blame]
// Copyright 2018 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef NET_WEBSOCKETS_WEBSOCKET_HTTP2_HANDSHAKE_STREAM_H_
#define NET_WEBSOCKETS_WEBSOCKET_HTTP2_HANDSHAKE_STREAM_H_
#include <stdint.h>
#include <memory>
#include <set>
#include <string>
#include <vector>
#include "base/memory/raw_ptr.h"
#include "base/memory/weak_ptr.h"
#include "base/strings/string_piece.h"
#include "net/base/completion_once_callback.h"
#include "net/base/io_buffer.h"
#include "net/base/net_export.h"
#include "net/base/request_priority.h"
#include "net/log/net_log_with_source.h"
#include "net/third_party/quiche/src/quiche/spdy/core/http2_header_block.h"
#include "net/websockets/websocket_basic_stream_adapters.h"
#include "net/websockets/websocket_handshake_stream_base.h"
#include "net/websockets/websocket_stream.h"
#include "third_party/abseil-cpp/absl/types/optional.h"
namespace net {
struct LoadTimingInfo;
class SSLInfo;
class IOBuffer;
class SSLCertRequestInfo;
class IPEndPoint;
class HttpNetworkSession;
struct NetErrorDetails;
class HttpStream;
class HttpResponseHeaders;
struct HttpRequestInfo;
class HttpResponseInfo;
class SpdySession;
struct AlternativeService;
class SpdyStreamRequest;
struct WebSocketExtensionParams;
class NET_EXPORT_PRIVATE WebSocketHttp2HandshakeStream
: public WebSocketHandshakeStreamBase,
public WebSocketSpdyStreamAdapter::Delegate {
public:
// |connect_delegate| and |request| must out-live this object.
WebSocketHttp2HandshakeStream(
base::WeakPtr<SpdySession> session,
WebSocketStream::ConnectDelegate* connect_delegate,
std::vector<std::string> requested_sub_protocols,
std::vector<std::string> requested_extensions,
WebSocketStreamRequestAPI* request,
std::set<std::string> dns_aliases);
WebSocketHttp2HandshakeStream(const WebSocketHttp2HandshakeStream&) = delete;
WebSocketHttp2HandshakeStream& operator=(
const WebSocketHttp2HandshakeStream&) = delete;
~WebSocketHttp2HandshakeStream() override;
// HttpStream methods.
void RegisterRequest(const HttpRequestInfo* request_info) override;
int InitializeStream(bool can_send_early,
RequestPriority priority,
const NetLogWithSource& net_log,
CompletionOnceCallback callback) override;
int SendRequest(const HttpRequestHeaders& request_headers,
HttpResponseInfo* response,
CompletionOnceCallback callback) override;
int ReadResponseHeaders(CompletionOnceCallback callback) override;
int ReadResponseBody(IOBuffer* buf,
int buf_len,
CompletionOnceCallback callback) override;
void Close(bool not_reusable) override;
bool IsResponseBodyComplete() const override;
bool IsConnectionReused() const override;
void SetConnectionReused() override;
bool CanReuseConnection() const override;
int64_t GetTotalReceivedBytes() const override;
int64_t GetTotalSentBytes() const override;
bool GetAlternativeService(
AlternativeService* alternative_service) const override;
bool GetLoadTimingInfo(LoadTimingInfo* load_timing_info) const override;
void GetSSLInfo(SSLInfo* ssl_info) override;
void GetSSLCertRequestInfo(SSLCertRequestInfo* cert_request_info) override;
int GetRemoteEndpoint(IPEndPoint* endpoint) override;
void Drain(HttpNetworkSession* session) override;
void SetPriority(RequestPriority priority) override;
void PopulateNetErrorDetails(NetErrorDetails* details) override;
std::unique_ptr<HttpStream> RenewStreamForAuth() override;
const std::set<std::string>& GetDnsAliases() const override;
base::StringPiece GetAcceptChViaAlps() const override;
// WebSocketHandshakeStreamBase methods.
// This is called from the top level once correct handshake response headers
// have been received. It creates an appropriate subclass of WebSocketStream
// depending on what extensions were negotiated. This object is unusable after
// Upgrade() has been called and should be disposed of as soon as possible.
std::unique_ptr<WebSocketStream> Upgrade() override;
bool CanReadFromStream() const override;
base::WeakPtr<WebSocketHandshakeStreamBase> GetWeakPtr() override;
// WebSocketSpdyStreamAdapter::Delegate methods.
void OnHeadersSent() override;
void OnHeadersReceived(
const spdy::Http2HeaderBlock& response_headers) override;
void OnClose(int status) override;
// Called by |spdy_stream_request_| when requested stream is ready.
void StartRequestCallback(int rv);
private:
// Validates the response and sends the finished handshake event.
int ValidateResponse();
// Check that the headers are well-formed and have a 200 status code,
// in which case returns OK, otherwise returns ERR_INVALID_RESPONSE.
int ValidateUpgradeResponse(const HttpResponseHeaders* headers);
void OnFailure(const std::string& message,
int net_error,
absl::optional<int> response_code);
HandshakeResult result_ = HandshakeResult::HTTP2_INCOMPLETE;
// The connection to open the Websocket stream on.
base::WeakPtr<SpdySession> session_;
// Owned by another object.
// |connect_delegate| will live during the lifetime of this object.
const raw_ptr<WebSocketStream::ConnectDelegate, DanglingUntriaged>
connect_delegate_;
raw_ptr<HttpResponseInfo> http_response_info_ = nullptr;
spdy::Http2HeaderBlock http2_request_headers_;
// The sub-protocols we requested.
std::vector<std::string> requested_sub_protocols_;
// The extensions we requested.
std::vector<std::string> requested_extensions_;
const raw_ptr<WebSocketStreamRequestAPI, DanglingUntriaged> stream_request_;
raw_ptr<const HttpRequestInfo, DanglingUntriaged> request_info_ = nullptr;
RequestPriority priority_;
NetLogWithSource net_log_;
// SpdyStreamRequest that will create the stream.
std::unique_ptr<SpdyStreamRequest> spdy_stream_request_;
// SpdyStream corresponding to the request.
base::WeakPtr<SpdyStream> stream_;
// WebSocketSpdyStreamAdapter holding a WeakPtr to |stream_|.
// This can be passed on to WebSocketBasicStream when created.
std::unique_ptr<WebSocketSpdyStreamAdapter> stream_adapter_;
// True if |stream_| has been created then closed.
bool stream_closed_ = false;
// The error code corresponding to the reason for closing the stream.
// Only meaningful if |stream_closed_| is true.
int stream_error_ = OK;
// True if complete response headers have been received.
bool response_headers_complete_ = false;
// Save callback provided in asynchronous HttpStream methods.
CompletionOnceCallback callback_;
// The sub-protocol selected by the server.
std::string sub_protocol_;
// The extension(s) selected by the server.
std::string extensions_;
// The extension parameters. The class is defined in the implementation file
// to avoid including extension-related header files here.
std::unique_ptr<WebSocketExtensionParams> extension_params_;
// Stores any DNS aliases for the remote endpoint. Includes all known aliases,
// e.g. from A, AAAA, or HTTPS, not just from the address used for the
// connection, in no particular order. These are stored in the stream instead
// of the session due to complications related to IP-pooling.
std::set<std::string> dns_aliases_;
base::WeakPtrFactory<WebSocketHttp2HandshakeStream> weak_ptr_factory_{this};
};
} // namespace net
#endif // NET_WEBSOCKETS_WEBSOCKET_HTTP2_HANDSHAKE_STREAM_H_