Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(2055)

Unified Diff: content/renderer/pepper/pepper_websocket_host.cc

Issue 10944005: Pepper WebSocket API: Implement new design Chrome IPC (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: fix nits Created 8 years, 2 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « content/renderer/pepper/pepper_websocket_host.h ('k') | content/renderer/pepper/renderer_ppapi_host_impl.h » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: content/renderer/pepper/pepper_websocket_host.cc
diff --git a/content/renderer/pepper/pepper_websocket_host.cc b/content/renderer/pepper/pepper_websocket_host.cc
new file mode 100644
index 0000000000000000000000000000000000000000..ec6341a327169e0a66836c88b458db643f38b9fc
--- /dev/null
+++ b/content/renderer/pepper/pepper_websocket_host.cc
@@ -0,0 +1,290 @@
+// 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/renderer/pepper/pepper_websocket_host.h"
+
+#include <string>
+
+#include "content/public/renderer/renderer_ppapi_host.h"
+#include "net/base/net_util.h"
+#include "ppapi/c/pp_errors.h"
+#include "ppapi/c/ppb_websocket.h"
+#include "ppapi/host/dispatch_host_message.h"
+#include "ppapi/host/host_message_context.h"
+#include "ppapi/host/ppapi_host.h"
+#include "ppapi/proxy/ppapi_messages.h"
+#include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebString.h"
+#include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebURL.h"
+#include "third_party/WebKit/Source/WebKit/chromium/public/WebArrayBuffer.h"
+#include "third_party/WebKit/Source/WebKit/chromium/public/WebDocument.h"
+#include "third_party/WebKit/Source/WebKit/chromium/public/WebElement.h"
+#include "third_party/WebKit/Source/WebKit/chromium/public/WebPluginContainer.h"
+#include "third_party/WebKit/Source/WebKit/chromium/public/WebSocket.h"
+
+using WebKit::WebArrayBuffer;
+using WebKit::WebDocument;
+using WebKit::WebString;
+using WebKit::WebSocket;
+using WebKit::WebURL;
+
+namespace content {
+
+PepperWebSocketHost::PepperWebSocketHost(
+ RendererPpapiHost* host,
+ PP_Instance instance,
+ PP_Resource resource)
+ : ResourceHost(host->GetPpapiHost(), instance, resource),
+ renderer_ppapi_host_(host),
+ connecting_(false),
+ initiating_close_(false),
+ accepting_close_(false),
+ error_was_received_(false) {
+}
+
+PepperWebSocketHost::~PepperWebSocketHost() {
+ if (websocket_.get())
+ websocket_->disconnect();
+}
+
+int32_t PepperWebSocketHost::OnResourceMessageReceived(
+ const IPC::Message& msg,
+ ppapi::host::HostMessageContext* context) {
+ IPC_BEGIN_MESSAGE_MAP(PepperWebSocketHost, msg)
+ PPAPI_DISPATCH_HOST_RESOURCE_CALL(PpapiHostMsg_WebSocket_Connect,
+ OnHostMsgConnect)
+ PPAPI_DISPATCH_HOST_RESOURCE_CALL(PpapiHostMsg_WebSocket_Close,
+ OnHostMsgClose)
+ PPAPI_DISPATCH_HOST_RESOURCE_CALL(PpapiHostMsg_WebSocket_SendText,
+ OnHostMsgSendText)
+ PPAPI_DISPATCH_HOST_RESOURCE_CALL(PpapiHostMsg_WebSocket_SendBinary,
+ OnHostMsgSendBinary)
+ PPAPI_DISPATCH_HOST_RESOURCE_CALL(PpapiHostMsg_WebSocket_Fail,
+ OnHostMsgFail)
+ IPC_END_MESSAGE_MAP()
+ return PP_ERROR_FAILED;
+}
+
+void PepperWebSocketHost::didConnect() {
+ std::string protocol;
+ if (websocket_.get())
+ protocol = websocket_->subprotocol().utf8();
+ connecting_ = false;
+ connect_reply_.params.set_result(PP_OK);
+ host()->SendReply(connect_reply_,
+ PpapiPluginMsg_WebSocket_ConnectReply(
+ url_,
+ protocol));
+}
+
+void PepperWebSocketHost::didReceiveMessage(const WebKit::WebString& message) {
+ // Dispose packets after receiving an error.
+ if (error_was_received_)
+ return;
+
+ // Send an IPC to transport received data.
+ std::string string_message = message.utf8();
+ host()->SendUnsolicitedReply(pp_resource(),
+ PpapiPluginMsg_WebSocket_ReceiveTextReply(
+ string_message));
+}
+
+void PepperWebSocketHost::didReceiveArrayBuffer(
+ const WebKit::WebArrayBuffer& binaryData) {
+ // Dispose packets after receiving an error.
+ if (error_was_received_)
+ return;
+
+ // Send an IPC to transport received data.
+ uint8_t* data = static_cast<uint8_t*>(binaryData.data());
+ std::vector<uint8_t> array_message(data, data + binaryData.byteLength());
+ host()->SendUnsolicitedReply(pp_resource(),
+ PpapiPluginMsg_WebSocket_ReceiveBinaryReply(
+ array_message));
+}
+
+void PepperWebSocketHost::didReceiveMessageError() {
+ // Records the error, then stops receiving any frames after this error.
+ // The error must be notified after all queued messages are read.
+ error_was_received_ = true;
+
+ // Send an IPC to report the error. After this IPC, ReceiveTextReply and
+ // ReceiveBinaryReply IPC are not sent anymore because |error_was_received_|
+ // blocks.
+ host()->SendUnsolicitedReply(pp_resource(),
+ PpapiPluginMsg_WebSocket_ErrorReply());
+}
+
+void PepperWebSocketHost::didUpdateBufferedAmount(
+ unsigned long buffered_amount) {
+ // Send an IPC to update buffered amount.
+ host()->SendUnsolicitedReply(pp_resource(),
+ PpapiPluginMsg_WebSocket_BufferedAmountReply(
+ buffered_amount));
+}
+
+void PepperWebSocketHost::didStartClosingHandshake() {
+ accepting_close_ = true;
+
+ // Send an IPC to notice that server starts closing handshake.
+ host()->SendUnsolicitedReply(pp_resource(),
+ PpapiPluginMsg_WebSocket_StateReply(
+ PP_WEBSOCKETREADYSTATE_CLOSING));
+}
+
+void PepperWebSocketHost::didClose(unsigned long unhandled_buffered_amount,
+ ClosingHandshakeCompletionStatus status,
+ unsigned short code,
+ const WebKit::WebString& reason) {
+ if (connecting_) {
+ connecting_ = false;
+ connect_reply_.params.set_result(PP_ERROR_FAILED);
+ host()->SendReply(
+ connect_reply_,
+ PpapiPluginMsg_WebSocket_ConnectReply(url_, std::string()));
+ }
+
+ // Set close_was_clean_.
+ bool was_clean =
+ (initiating_close_ || accepting_close_) &&
+ !unhandled_buffered_amount &&
+ status == WebSocketClient::ClosingHandshakeComplete;
+
+ if (initiating_close_) {
+ initiating_close_ = false;
+ close_reply_.params.set_result(PP_OK);
+ host()->SendReply(close_reply_, PpapiPluginMsg_WebSocket_CloseReply(
+ unhandled_buffered_amount,
+ was_clean,
+ code,
+ reason.utf8()));
+ } else {
+ accepting_close_ = false;
+ host()->SendUnsolicitedReply(pp_resource(),
+ PpapiPluginMsg_WebSocket_ClosedReply(
+ unhandled_buffered_amount,
+ was_clean,
+ code,
+ reason.utf8()));
+ }
+
+ // Disconnect.
+ if (websocket_.get())
+ websocket_->disconnect();
+}
+
+int32_t PepperWebSocketHost::OnHostMsgConnect(
+ ppapi::host::HostMessageContext* context,
+ const std::string& url,
+ const std::vector<std::string>& protocols) {
+ // Validate url and convert it to WebURL.
+ GURL gurl(url);
+ url_ = gurl.spec();
+ if (!gurl.is_valid())
+ return PP_ERROR_BADARGUMENT;
+ if (!gurl.SchemeIs("ws") && !gurl.SchemeIs("wss"))
+ return PP_ERROR_BADARGUMENT;
+ if (gurl.has_ref())
+ return PP_ERROR_BADARGUMENT;
+ if (!net::IsPortAllowedByDefault(gurl.IntPort()))
+ return PP_ERROR_BADARGUMENT;
+ WebURL web_url(gurl);
+
+ // Validate protocols.
+ std::string protocol_string;
+ for (std::vector<std::string>::const_iterator vector_it = protocols.begin();
+ vector_it != protocols.end();
+ ++vector_it) {
+
+ // Check containing characters.
+ for (std::string::const_iterator string_it = vector_it->begin();
+ string_it != vector_it->end();
+ ++string_it) {
+ uint8_t character = *string_it;
+ // WebSocket specification says "(Subprotocol string must consist of)
+ // characters in the range U+0021 to U+007E not including separator
+ // characters as defined in [RFC2616]."
+ const uint8_t minimumProtocolCharacter = '!'; // U+0021.
+ const uint8_t maximumProtocolCharacter = '~'; // U+007E.
+ if (character < minimumProtocolCharacter ||
+ character > maximumProtocolCharacter ||
+ character == '"' || character == '(' || character == ')' ||
+ character == ',' || character == '/' ||
+ (character >= ':' && character <= '@') || // U+003A - U+0040
+ (character >= '[' && character <= ']') || // U+005B - u+005D
+ character == '{' || character == '}')
+ return PP_ERROR_BADARGUMENT;
+ }
+ // Join protocols with the comma separator.
+ if (vector_it != protocols.begin())
+ protocol_string.append(",");
+ protocol_string.append(*vector_it);
+ }
+
+ // Convert protocols to WebString.
+ WebString web_protocols = WebString::fromUTF8(protocol_string);
+
+ // Create WebKit::WebSocket object and connect.
+ WebKit::WebPluginContainer* container =
+ renderer_ppapi_host_->GetContainerForInstance(pp_instance());
+ if (!container)
+ return PP_ERROR_BADARGUMENT;
+ // TODO(toyoshim) Remove following WebDocument object copy.
+ WebDocument document = container->element().document();
+ websocket_.reset(WebSocket::create(document, this));
+ DCHECK(websocket_.get());
+ if (!websocket_.get())
+ return PP_ERROR_NOTSUPPORTED;
+
+ // Set receiving binary object type.
+ websocket_->setBinaryType(WebSocket::BinaryTypeArrayBuffer);
+ websocket_->connect(web_url, web_protocols);
+
+ connect_reply_ = context->MakeReplyMessageContext();
+ connecting_ = true;
+ return PP_OK_COMPLETIONPENDING;
+}
+
+int32_t PepperWebSocketHost::OnHostMsgClose(
+ ppapi::host::HostMessageContext* context,
+ int32_t code,
+ const std::string& reason) {
+ if (!websocket_.get())
+ return PP_ERROR_FAILED;
+ close_reply_ = context->MakeReplyMessageContext();
+ initiating_close_ = true;
+ WebString web_reason = WebString::fromUTF8(reason);
+ websocket_->close(code, web_reason);
+ return PP_OK_COMPLETIONPENDING;
+}
+
+int32_t PepperWebSocketHost::OnHostMsgSendText(
+ ppapi::host::HostMessageContext* context,
+ const std::string& message) {
+ if (websocket_.get()) {
+ WebString web_message = WebString::fromUTF8(message);
+ websocket_->sendText(web_message);
+ }
+ return PP_OK;
+}
+
+int32_t PepperWebSocketHost::OnHostMsgSendBinary(
+ ppapi::host::HostMessageContext* context,
+ const std::vector<uint8_t>& message) {
+ if (websocket_.get()) {
+ WebArrayBuffer web_message = WebArrayBuffer::create(message.size(), 1);
+ memcpy(web_message.data(), &message.front(), message.size());
+ websocket_->sendArrayBuffer(web_message);
+ }
+ return PP_OK;
+}
+
+int32_t PepperWebSocketHost::OnHostMsgFail(
+ ppapi::host::HostMessageContext* context,
+ const std::string& message) {
+ if (websocket_.get())
+ websocket_->fail(WebString::fromUTF8(message));
+ return PP_OK;
+}
+
+} // namespace content
« no previous file with comments | « content/renderer/pepper/pepper_websocket_host.h ('k') | content/renderer/pepper/renderer_ppapi_host_impl.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698