blob: 75d314e6403b493e3fcbba41201b88c5be969490 [file] [log] [blame]
// Copyright 2020 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef THIRD_PARTY_BLINK_RENDERER_MODULES_DIRECT_SOCKETS_TCP_SOCKET_H_
#define THIRD_PARTY_BLINK_RENDERER_MODULES_DIRECT_SOCKETS_TCP_SOCKET_H_
#include <optional>
#include "base/gtest_prod_util.h"
#include "mojo/public/cpp/bindings/pending_receiver.h"
#include "mojo/public/cpp/bindings/pending_remote.h"
#include "mojo/public/cpp/bindings/receiver.h"
#include "mojo/public/cpp/bindings/remote.h"
#include "mojo/public/cpp/system/data_pipe.h"
#include "services/network/public/mojom/tcp_socket.mojom-blink.h"
#include "third_party/blink/renderer/bindings/core/v8/active_script_wrappable.h"
#include "third_party/blink/renderer/bindings/core/v8/script_promise_property.h"
#include "third_party/blink/renderer/core/execution_context/execution_context.h"
#include "third_party/blink/renderer/modules/direct_sockets/socket.h"
#include "third_party/blink/renderer/modules/direct_sockets/tcp_readable_stream_wrapper.h"
#include "third_party/blink/renderer/modules/direct_sockets/tcp_writable_stream_wrapper.h"
#include "third_party/blink/renderer/modules/modules_export.h"
#include "third_party/blink/renderer/platform/bindings/exception_code.h"
#include "third_party/blink/renderer/platform/bindings/exception_state.h"
#include "third_party/blink/renderer/platform/bindings/script_wrappable.h"
#include "third_party/blink/renderer/platform/bindings/trace_wrapper_v8_reference.h"
#include "third_party/blink/renderer/platform/heap/garbage_collected.h"
#include "third_party/blink/renderer/platform/heap/member.h"
#include "third_party/blink/renderer/platform/loader/fetch/unique_identifier.h"
#include "third_party/blink/renderer/platform/mojo/heap_mojo_receiver.h"
#include "third_party/blink/renderer/platform/mojo/heap_mojo_remote.h"
#include "third_party/blink/renderer/platform/scheduler/public/frame_or_worker_scheduler.h"
namespace net {
class IPEndPoint;
} // namespace net
namespace blink {
class SocketCloseOptions;
class TCPSocketOpenInfo;
class TCPSocketOptions;
// TCPSocket interface from tcp_socket.idl
class MODULES_EXPORT TCPSocket final
: public ScriptWrappable,
public Socket,
public ActiveScriptWrappable<TCPSocket>,
public network::mojom::blink::SocketObserver {
DEFINE_WRAPPERTYPEINFO();
public:
// IDL definitions
static TCPSocket* Create(ScriptState*,
const String& remote_address,
const uint16_t remote_port,
const TCPSocketOptions*,
ExceptionState&);
// Socket:
ScriptPromise<TCPSocketOpenInfo> opened(ScriptState*) const;
ScriptPromise<IDLUndefined> close(ScriptState*, ExceptionState&) override;
public:
// Constructor for use from TCPServerSocket.
static TCPSocket* CreateFromAcceptedConnection(
ScriptState*,
mojo::PendingRemote<network::mojom::blink::TCPConnectedSocket>,
mojo::PendingReceiver<network::mojom::blink::SocketObserver>,
const net::IPEndPoint& remote_addr,
mojo::ScopedDataPipeConsumerHandle receive_stream,
mojo::ScopedDataPipeProducerHandle send_stream);
explicit TCPSocket(ScriptState*);
~TCPSocket() override;
// Validates options and calls
// DirectSocketsServiceMojoRemote::OpenTCPSocket(...).
bool Open(const String& remote_address,
const uint16_t remote_port,
const TCPSocketOptions*,
ExceptionState&);
// On net::OK initializes readable/writable streams and resolves opened
// promise. Otherwise rejects the opened promise.
void OnTCPSocketOpened(
mojo::PendingRemote<network::mojom::blink::TCPConnectedSocket>,
mojo::PendingReceiver<network::mojom::blink::SocketObserver>,
int32_t result,
const std::optional<net::IPEndPoint>& local_addr,
const std::optional<net::IPEndPoint>& peer_addr,
mojo::ScopedDataPipeConsumerHandle receive_stream,
mojo::ScopedDataPipeProducerHandle send_stream);
void Trace(Visitor*) const override;
// ActiveScriptWrappable:
bool HasPendingActivity() const override;
// ExecutionContextLifecycleStateObserver:
void ContextDestroyed() override;
// Socket:
void SetState(State state) override;
private:
void FinishOpenOrAccept(
mojo::PendingRemote<network::mojom::blink::TCPConnectedSocket>,
mojo::PendingReceiver<network::mojom::blink::SocketObserver>,
const net::IPEndPoint& peer_addr,
const std::optional<net::IPEndPoint>& local_addr,
mojo::ScopedDataPipeConsumerHandle receive_stream,
mojo::ScopedDataPipeProducerHandle send_stream);
// Invoked if mojo pipe for |service_| breaks.
void OnServiceConnectionError() override;
// Invoked if mojo pipe for |socket_observer_| breaks.
void OnSocketConnectionError();
// Resets mojo resources held by this class.
void ReleaseResources();
// network::mojom::blink::SocketObserver:
void OnReadError(int32_t net_error) override;
void OnWriteError(int32_t net_error) override;
// Invoked when one of the streams (readable or writable) closes.
// `exception` is non-empty iff the stream closed with an error.
void OnStreamClosed(v8::Local<v8::Value> exception, int net_error);
void OnBothStreamsClosed();
HeapMojoRemote<network::mojom::blink::TCPConnectedSocket> tcp_socket_;
HeapMojoReceiver<network::mojom::blink::SocketObserver, TCPSocket>
socket_observer_;
Member<ScriptPromiseProperty<TCPSocketOpenInfo, DOMException>> opened_;
Member<TCPReadableStreamWrapper> readable_stream_wrapper_;
Member<TCPWritableStreamWrapper> writable_stream_wrapper_;
// Always less or equal to 2 (readable + writable).
int streams_closed_count_ = 0;
// Stores the first encountered stream error to be reported after both streams
// close.
TraceWrapperV8Reference<v8::Value> stream_error_;
// Stores the net error when the socket is aborted
int abort_net_error_;
// Unique id for devtools inspector_network_agent.
uint64_t inspector_id_ = CreateUniqueIdentifier();
FRIEND_TEST_ALL_PREFIXES(TCPSocketTest, OnSocketObserverConnectionError);
FRIEND_TEST_ALL_PREFIXES(TCPSocketCloseTest, OnErrorOrClose);
};
} // namespace blink
#endif // THIRD_PARTY_BLINK_RENDERER_MODULES_DIRECT_SOCKETS_TCP_SOCKET_H_