| // Copyright (c) 2011 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/browser/renderer_host/socket_stream_dispatcher_host.h" |
| |
| #include "base/logging.h" |
| #include "chrome/browser/profiles/profile.h" |
| #include "chrome/common/render_messages.h" |
| #include "chrome/common/render_messages_params.h" |
| #include "chrome/common/net/socket_stream.h" |
| #include "chrome/common/net/url_request_context_getter.h" |
| #include "content/browser/renderer_host/socket_stream_host.h" |
| #include "net/websockets/websocket_job.h" |
| #include "net/websockets/websocket_throttle.h" |
| |
| SocketStreamDispatcherHost::SocketStreamDispatcherHost() { |
| net::WebSocketJob::EnsureInit(); |
| } |
| |
| SocketStreamDispatcherHost::~SocketStreamDispatcherHost() { |
| // TODO(ukai): Implement IDMap::RemoveAll(). |
| for (IDMap<SocketStreamHost>::const_iterator iter(&hosts_); |
| !iter.IsAtEnd(); |
| iter.Advance()) { |
| int socket_id = iter.GetCurrentKey(); |
| const SocketStreamHost* socket_stream_host = iter.GetCurrentValue(); |
| delete socket_stream_host; |
| hosts_.Remove(socket_id); |
| } |
| } |
| |
| bool SocketStreamDispatcherHost::OnMessageReceived(const IPC::Message& message, |
| bool* message_was_ok) { |
| bool handled = true; |
| IPC_BEGIN_MESSAGE_MAP_EX(SocketStreamDispatcherHost, message, *message_was_ok) |
| IPC_MESSAGE_HANDLER(ViewHostMsg_SocketStream_Connect, OnConnect) |
| IPC_MESSAGE_HANDLER(ViewHostMsg_SocketStream_SendData, OnSendData) |
| IPC_MESSAGE_HANDLER(ViewHostMsg_SocketStream_Close, OnCloseReq) |
| IPC_MESSAGE_UNHANDLED(handled = false) |
| IPC_END_MESSAGE_MAP_EX() |
| return handled; |
| } |
| |
| // SocketStream::Delegate methods implementations. |
| void SocketStreamDispatcherHost::OnConnected(net::SocketStream* socket, |
| int max_pending_send_allowed) { |
| int socket_id = SocketStreamHost::SocketIdFromSocketStream(socket); |
| DVLOG(1) << "SocketStreamDispatcherHost::OnConnected socket_id=" << socket_id |
| << " max_pending_send_allowed=" << max_pending_send_allowed; |
| if (socket_id == chrome_common_net::kNoSocketId) { |
| LOG(ERROR) << "NoSocketId in OnConnected"; |
| return; |
| } |
| if (!Send(new ViewMsg_SocketStream_Connected( |
| socket_id, max_pending_send_allowed))) { |
| LOG(ERROR) << "ViewMsg_SocketStream_Connected failed."; |
| DeleteSocketStreamHost(socket_id); |
| } |
| } |
| |
| void SocketStreamDispatcherHost::OnSentData(net::SocketStream* socket, |
| int amount_sent) { |
| int socket_id = SocketStreamHost::SocketIdFromSocketStream(socket); |
| DVLOG(1) << "SocketStreamDispatcherHost::OnSentData socket_id=" << socket_id |
| << " amount_sent=" << amount_sent; |
| if (socket_id == chrome_common_net::kNoSocketId) { |
| LOG(ERROR) << "NoSocketId in OnReceivedData"; |
| return; |
| } |
| if (!Send(new ViewMsg_SocketStream_SentData(socket_id, amount_sent))) { |
| LOG(ERROR) << "ViewMsg_SocketStream_SentData failed."; |
| DeleteSocketStreamHost(socket_id); |
| } |
| } |
| |
| void SocketStreamDispatcherHost::OnReceivedData( |
| net::SocketStream* socket, const char* data, int len) { |
| int socket_id = SocketStreamHost::SocketIdFromSocketStream(socket); |
| DVLOG(1) << "SocketStreamDispatcherHost::OnReceiveData socket_id=" |
| << socket_id; |
| if (socket_id == chrome_common_net::kNoSocketId) { |
| LOG(ERROR) << "NoSocketId in OnReceivedData"; |
| return; |
| } |
| if (!Send(new ViewMsg_SocketStream_ReceivedData( |
| socket_id, std::vector<char>(data, data + len)))) { |
| LOG(ERROR) << "ViewMsg_SocketStream_ReceivedData failed."; |
| DeleteSocketStreamHost(socket_id); |
| } |
| } |
| |
| void SocketStreamDispatcherHost::OnClose(net::SocketStream* socket) { |
| int socket_id = SocketStreamHost::SocketIdFromSocketStream(socket); |
| DVLOG(1) << "SocketStreamDispatcherHost::OnClosed socket_id=" << socket_id; |
| if (socket_id == chrome_common_net::kNoSocketId) { |
| LOG(ERROR) << "NoSocketId in OnClose"; |
| return; |
| } |
| DeleteSocketStreamHost(socket_id); |
| } |
| |
| // Message handlers called by OnMessageReceived. |
| void SocketStreamDispatcherHost::OnConnect(const GURL& url, int socket_id) { |
| DVLOG(1) << "SocketStreamDispatcherHost::OnConnect url=" << url |
| << " socket_id=" << socket_id; |
| DCHECK_NE(chrome_common_net::kNoSocketId, socket_id); |
| if (hosts_.Lookup(socket_id)) { |
| LOG(ERROR) << "socket_id=" << socket_id << " already registered."; |
| return; |
| } |
| SocketStreamHost* socket_stream_host = new SocketStreamHost(this, socket_id); |
| hosts_.AddWithID(socket_stream_host, socket_id); |
| socket_stream_host->Connect(url, GetURLRequestContext()); |
| DVLOG(1) << "SocketStreamDispatcherHost::OnConnect -> " << socket_id; |
| } |
| |
| void SocketStreamDispatcherHost::OnSendData( |
| int socket_id, const std::vector<char>& data) { |
| DVLOG(1) << "SocketStreamDispatcherHost::OnSendData socket_id=" << socket_id; |
| SocketStreamHost* socket_stream_host = hosts_.Lookup(socket_id); |
| if (!socket_stream_host) { |
| LOG(ERROR) << "socket_id=" << socket_id << " already closed."; |
| return; |
| } |
| if (!socket_stream_host->SendData(data)) { |
| // Cannot accept more data to send. |
| socket_stream_host->Close(); |
| } |
| } |
| |
| void SocketStreamDispatcherHost::OnCloseReq(int socket_id) { |
| DVLOG(1) << "SocketStreamDispatcherHost::OnCloseReq socket_id=" << socket_id; |
| SocketStreamHost* socket_stream_host = hosts_.Lookup(socket_id); |
| if (!socket_stream_host) |
| return; |
| socket_stream_host->Close(); |
| } |
| |
| void SocketStreamDispatcherHost::DeleteSocketStreamHost(int socket_id) { |
| SocketStreamHost* socket_stream_host = hosts_.Lookup(socket_id); |
| DCHECK(socket_stream_host); |
| delete socket_stream_host; |
| hosts_.Remove(socket_id); |
| if (!Send(new ViewMsg_SocketStream_Closed(socket_id))) { |
| LOG(ERROR) << "ViewMsg_SocketStream_Closed failed."; |
| } |
| } |
| |
| net::URLRequestContext* SocketStreamDispatcherHost::GetURLRequestContext() { |
| net::URLRequestContext* rv = NULL; |
| if (url_request_context_override_.get()) { |
| ViewHostMsg_Resource_Request request; |
| rv = url_request_context_override_->GetRequestContext(request); |
| } |
| if (!rv) { |
| URLRequestContextGetter* context_getter = |
| Profile::GetDefaultRequestContext(); |
| if (context_getter) |
| rv = context_getter->GetURLRequestContext(); |
| } |
| |
| return rv; |
| } |