// Copyright (c) 2012 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.
#include "content/child/socket_stream_dispatcher.h"
#include <vector>
#include "base/bind.h"
#include "base/id_map.h"
#include "base/lazy_instance.h"
#include "base/memory/ref_counted.h"
#include "base/message_loop/message_loop.h"
#include "base/strings/string16.h"
#include "base/strings/utf_string_conversions.h"
#include "content/child/child_thread.h"
#include "content/common/socket_stream.h"
#include "content/common/socket_stream_handle_data.h"
#include "content/common/socket_stream_messages.h"
#include "net/base/net_errors.h"
#include "url/gurl.h"
#include "webkit/child/websocketstreamhandle_bridge.h"
#include "webkit/child/websocketstreamhandle_delegate.h"
namespace content {
// IPCWebSocketStreamHandleBridge is owned by each SocketStreamHandle.
// It communicates with the main browser process via SocketStreamDispatcher.
class IPCWebSocketStreamHandleBridge
: public webkit_glue::WebSocketStreamHandleBridge {
public:
IPCWebSocketStreamHandleBridge(
blink::WebSocketStreamHandle* handle,
webkit_glue::WebSocketStreamHandleDelegate* delegate)
: socket_id_(kNoSocketId),
handle_(handle),
delegate_(delegate) {}
// Returns the handle having given id or NULL if there is no such handle.
static IPCWebSocketStreamHandleBridge* FromSocketId(int id);
// webkit_glue::WebSocketStreamHandleBridge methods.
virtual void Connect(const GURL& url) OVERRIDE;
virtual bool Send(const std::vector<char>& data) OVERRIDE;
virtual void Close() OVERRIDE;
// Called by SocketStreamDispatcher.
void OnConnected(int max_amount_send_allowed);
void OnSentData(int amount_sent);
void OnReceivedData(const std::vector<char>& data);
void OnClosed();
void OnFailed(int error_code, const char* error_msg);
private:
virtual ~IPCWebSocketStreamHandleBridge();
// The ID for this bridge and corresponding SocketStream instance in the
// browser process.
int socket_id_;
blink::WebSocketStreamHandle* handle_;
webkit_glue::WebSocketStreamHandleDelegate* delegate_;
// Map from ID to bridge instance.
static base::LazyInstance<IDMap<IPCWebSocketStreamHandleBridge> >::Leaky
all_bridges;
};
// static
base::LazyInstance<IDMap<IPCWebSocketStreamHandleBridge> >::Leaky
IPCWebSocketStreamHandleBridge::all_bridges = LAZY_INSTANCE_INITIALIZER;
/* static */
IPCWebSocketStreamHandleBridge* IPCWebSocketStreamHandleBridge::FromSocketId(
int id) {
return all_bridges.Get().Lookup(id);
}
IPCWebSocketStreamHandleBridge::~IPCWebSocketStreamHandleBridge() {
DVLOG(1) << "Bridge (" << this << ", socket_id_=" << socket_id_
<< ") Destructor";
if (socket_id_ == kNoSocketId)
return;
ChildThread::current()->Send(new SocketStreamHostMsg_Close(socket_id_));
socket_id_ = kNoSocketId;
}
void IPCWebSocketStreamHandleBridge::Connect(const GURL& url) {
DVLOG(1) << "Bridge (" << this << ") Connect (url=" << url << ")";
DCHECK_EQ(socket_id_, kNoSocketId);
if (delegate_)
delegate_->WillOpenStream(handle_, url);
socket_id_ = all_bridges.Get().Add(this);
DCHECK_NE(socket_id_, kNoSocketId);
int render_view_id = MSG_ROUTING_NONE;
const SocketStreamHandleData* data =
SocketStreamHandleData::ForHandle(handle_);
if (data)
render_view_id = data->render_view_id();
AddRef(); // Released in OnClosed().
ChildThread::current()->Send(
new SocketStreamHostMsg_Connect(render_view_id, url, socket_id_));
DVLOG(1) << "Bridge #" << socket_id_ << " sent IPC Connect";
// TODO(ukai): timeout to OnConnected.
}
bool IPCWebSocketStreamHandleBridge::Send(const std::vector<char>& data) {
DVLOG(1) << "Bridge #" << socket_id_ << " Send (" << data.size()
<< " bytes)";
ChildThread::current()->Send(
new SocketStreamHostMsg_SendData(socket_id_, data));
if (delegate_)
delegate_->WillSendData(handle_, &data[0], data.size());
return true;
}
void IPCWebSocketStreamHandleBridge::Close() {
DVLOG(1) << "Bridge #" << socket_id_ << " Close";
ChildThread::current()->Send(new SocketStreamHostMsg_Close(socket_id_));
}
void IPCWebSocketStreamHandleBridge::OnConnected(int max_pending_send_allowed) {
DVLOG(1) << "Bridge #" << socket_id_
<< " OnConnected (max_pending_send_allowed="
<< max_pending_send_allowed << ")";
if (delegate_)
delegate_->DidOpenStream(handle_, max_pending_send_allowed);
}
void IPCWebSocketStreamHandleBridge::OnSentData(int amount_sent) {
DVLOG(1) << "Bridge #" << socket_id_ << " OnSentData (" << amount_sent
<< " bytes)";
if (delegate_)
delegate_->DidSendData(handle_, amount_sent);
}
void IPCWebSocketStreamHandleBridge::OnReceivedData(
const std::vector<char>& data) {
DVLOG(1) << "Bridge #" << socket_id_ << " OnReceiveData (" << data.size()
<< " bytes)";
if (delegate_)
delegate_->DidReceiveData(handle_, &data[0], data.size());
}
void IPCWebSocketStreamHandleBridge::OnClosed() {
DVLOG(1) << "Bridge #" << socket_id_ << " OnClosed";
if (socket_id_ != kNoSocketId) {
all_bridges.Get().Remove(socket_id_);
socket_id_ = kNoSocketId;
}
if (delegate_)
delegate_->DidClose(handle_);
delegate_ = NULL;
Release();
}
void IPCWebSocketStreamHandleBridge::OnFailed(int error_code,
const char* error_msg) {
DVLOG(1) << "Bridge #" << socket_id_ << " OnFailed (error_code=" << error_code
<< ")";
if (delegate_)
delegate_->DidFail(handle_, error_code, ASCIIToUTF16(error_msg));
}
SocketStreamDispatcher::SocketStreamDispatcher() {
}
/* static */
webkit_glue::WebSocketStreamHandleBridge*
SocketStreamDispatcher::CreateBridge(
blink::WebSocketStreamHandle* handle,
webkit_glue::WebSocketStreamHandleDelegate* delegate) {
return new IPCWebSocketStreamHandleBridge(handle, delegate);
}
bool SocketStreamDispatcher::OnMessageReceived(const IPC::Message& msg) {
bool handled = true;
IPC_BEGIN_MESSAGE_MAP(SocketStreamDispatcher, msg)
IPC_MESSAGE_HANDLER(SocketStreamMsg_Connected, OnConnected)
IPC_MESSAGE_HANDLER(SocketStreamMsg_SentData, OnSentData)
IPC_MESSAGE_HANDLER(SocketStreamMsg_ReceivedData, OnReceivedData)
IPC_MESSAGE_HANDLER(SocketStreamMsg_Closed, OnClosed)
IPC_MESSAGE_HANDLER(SocketStreamMsg_Failed, OnFailed)
IPC_MESSAGE_UNHANDLED(handled = false)
IPC_END_MESSAGE_MAP()
return handled;
}
void SocketStreamDispatcher::OnConnected(int socket_id,
int max_pending_send_allowed) {
DVLOG(1) << "SocketStreamDispatcher::OnConnected (max_pending_send_allowed="
<< max_pending_send_allowed << ") to socket_id=" << socket_id;
IPCWebSocketStreamHandleBridge* bridge =
IPCWebSocketStreamHandleBridge::FromSocketId(socket_id);
if (bridge)
bridge->OnConnected(max_pending_send_allowed);
else
DLOG(ERROR) << "No bridge for socket_id=" << socket_id;
}
void SocketStreamDispatcher::OnSentData(int socket_id, int amount_sent) {
DVLOG(1) << "SocketStreamDispatcher::OnSentData (" << amount_sent
<< " bytes) to socket_id=" << socket_id;
IPCWebSocketStreamHandleBridge* bridge =
IPCWebSocketStreamHandleBridge::FromSocketId(socket_id);
if (bridge)
bridge->OnSentData(amount_sent);
else
DLOG(ERROR) << "No bridge for socket_id=" << socket_id;
}
void SocketStreamDispatcher::OnReceivedData(
int socket_id, const std::vector<char>& data) {
DVLOG(1) << "SocketStreamDispatcher::OnReceivedData (" << data.size()
<< " bytes) to socket_id=" << socket_id;
IPCWebSocketStreamHandleBridge* bridge =
IPCWebSocketStreamHandleBridge::FromSocketId(socket_id);
if (bridge)
bridge->OnReceivedData(data);
else
DLOG(ERROR) << "No bridge for socket_id=" << socket_id;
}
void SocketStreamDispatcher::OnClosed(int socket_id) {
DVLOG(1) << "SocketStreamDispatcher::OnClosed to socket_id=" << socket_id;
IPCWebSocketStreamHandleBridge* bridge =
IPCWebSocketStreamHandleBridge::FromSocketId(socket_id);
if (bridge)
bridge->OnClosed();
else
DLOG(ERROR) << "No bridge for socket_id=" << socket_id;
}
void SocketStreamDispatcher::OnFailed(int socket_id, int error_code) {
IPCWebSocketStreamHandleBridge* bridge =
IPCWebSocketStreamHandleBridge::FromSocketId(socket_id);
if (bridge)
bridge->OnFailed(error_code, net::ErrorToString(error_code));
else
DLOG(ERROR) << "No bridge for socket_id=" << socket_id;
}
} // namespace content