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

Side by Side 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: (not for review: fix win build) 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 unified diff | Download patch | Annotate | Revision Log
OLDNEW
(Empty)
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "content/renderer/pepper/pepper_websocket_host.h"
6
7 #include <string>
8
9 #include "content/public/renderer/renderer_ppapi_host.h"
10 #include "net/base/net_util.h"
11 #include "ppapi/c/pp_errors.h"
12 #include "ppapi/c/ppb_websocket.h"
13 #include "ppapi/host/dispatch_host_message.h"
14 #include "ppapi/host/host_message_context.h"
15 #include "ppapi/host/ppapi_host.h"
16 #include "ppapi/proxy/ppapi_messages.h"
17 #include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebString.h"
18 #include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebURL.h"
19 #include "third_party/WebKit/Source/WebKit/chromium/public/WebArrayBuffer.h"
20 #include "third_party/WebKit/Source/WebKit/chromium/public/WebDocument.h"
21 #include "third_party/WebKit/Source/WebKit/chromium/public/WebElement.h"
22 #include "third_party/WebKit/Source/WebKit/chromium/public/WebPluginContainer.h"
23 #include "third_party/WebKit/Source/WebKit/chromium/public/WebSocket.h"
24
25 using WebKit::WebArrayBuffer;
26 using WebKit::WebDocument;
27 using WebKit::WebString;
28 using WebKit::WebSocket;
29 using WebKit::WebURL;
30
31 namespace content {
32
33 PepperWebSocketHost::PepperWebSocketHost(
34 RendererPpapiHost* host,
35 PP_Instance instance,
36 PP_Resource resource)
37 : ResourceHost(host->GetPpapiHost(), instance, resource),
38 renderer_ppapi_host_(host),
39 initiating_close_(false),
40 accepting_close_(false),
41 error_was_received_(false) {
42 }
43
44 PepperWebSocketHost::~PepperWebSocketHost() {
45 if (websocket_.get())
46 websocket_->disconnect();
47 }
48
49 int32_t PepperWebSocketHost::OnResourceMessageReceived(
50 const IPC::Message& msg,
51 ppapi::host::HostMessageContext* context) {
52 IPC_BEGIN_MESSAGE_MAP(PepperWebSocketHost, msg)
53 PPAPI_DISPATCH_HOST_RESOURCE_CALL(PpapiHostMsg_WebSocket_Connect,
54 OnHostMsgConnect)
55 PPAPI_DISPATCH_HOST_RESOURCE_CALL(PpapiHostMsg_WebSocket_Close,
56 OnHostMsgClose)
57 PPAPI_DISPATCH_HOST_RESOURCE_CALL(PpapiHostMsg_WebSocket_SendText,
58 OnHostMsgSendText)
59 PPAPI_DISPATCH_HOST_RESOURCE_CALL(PpapiHostMsg_WebSocket_SendBinary,
60 OnHostMsgSendBinary)
61 PPAPI_DISPATCH_HOST_RESOURCE_CALL(PpapiHostMsg_WebSocket_Fail,
62 OnHostMsgFail)
63 IPC_END_MESSAGE_MAP()
64 return PP_ERROR_FAILED;
65 }
66
67 void PepperWebSocketHost::didConnect() {
68 std::string protocol;
69 if (websocket_.get())
70 protocol = websocket_->subprotocol().utf8();
71 connect_reply_.params.set_result(PP_OK);
72 host()->SendReply(connect_reply_,
73 PpapiPluginMsg_WebSocket_ConnectReply(
74 std::string(),
75 protocol));
76 }
77
78 void PepperWebSocketHost::didReceiveMessage(const WebKit::WebString& message) {
79 // Dispose packets after receiving an error.
80 if (error_was_received_)
81 return;
82
83 // Send an IPC to transport received data.
84 std::string string_message = message.utf8();
85 host()->SendUnsolicitedReply(pp_resource(),
86 PpapiPluginMsg_WebSocket_ReceiveTextReply(
87 string_message));
88 }
89
90 void PepperWebSocketHost::didReceiveArrayBuffer(
91 const WebKit::WebArrayBuffer& binaryData) {
92 // Dispose packets after receiving an error.
93 if (error_was_received_)
94 return;
95
96 // Send an IPC to transport received data.
97 uint8_t* data = static_cast<uint8_t*>(binaryData.data());
98 std::vector<uint8_t> array_message(data, data + binaryData.byteLength());
99 host()->SendUnsolicitedReply(pp_resource(),
100 PpapiPluginMsg_WebSocket_ReceiveBinaryReply(
101 array_message));
102 }
103
104 void PepperWebSocketHost::didReceiveMessageError() {
105 error_was_received_ = true;
106
107 // Send an IPC to report error.
108 host()->SendUnsolicitedReply(pp_resource(),
109 PpapiPluginMsg_WebSocket_ErrorReply());
110 }
111
112 void PepperWebSocketHost::didUpdateBufferedAmount(
113 unsigned long buffered_amount) {
114 // Send an IPC to update buffered amount.
115 host()->SendUnsolicitedReply(pp_resource(),
116 PpapiPluginMsg_WebSocket_BufferedAmountReply(
117 buffered_amount));
118 }
119
120 void PepperWebSocketHost::didStartClosingHandshake() {
121 accepting_close_ = true;
122
123 // Send an IPC to notice that server starts closing handshake.
124 host()->SendUnsolicitedReply(pp_resource(),
125 PpapiPluginMsg_WebSocket_StateReply(
126 PP_WEBSOCKETREADYSTATE_CLOSING));
127 }
128
129 void PepperWebSocketHost::didClose(unsigned long unhandled_buffered_amount,
130 ClosingHandshakeCompletionStatus status,
131 unsigned short code,
132 const WebKit::WebString& reason) {
133 // Set close_was_clean_.
134 bool was_clean =
135 (initiating_close_ || accepting_close_) &&
136 !unhandled_buffered_amount &&
137 status == WebSocketClient::ClosingHandshakeComplete;
138
139 if (initiating_close_) {
140 initiating_close_ = false;
141 close_reply_.params.set_result(PP_OK);
142 host()->SendReply(close_reply_, PpapiPluginMsg_WebSocket_CloseReply(
143 unhandled_buffered_amount,
144 was_clean,
145 code,
146 reason.utf8()));
147 } else {
148 accepting_close_ = false;
149 host()->SendUnsolicitedReply(pp_resource(),
150 PpapiPluginMsg_WebSocket_ClosedReply(
151 unhandled_buffered_amount,
152 was_clean,
153 code,
154 reason.utf8()));
155 }
156
157 // Disconnect.
158 if (websocket_.get())
159 websocket_->disconnect();
160 }
161
162 int32_t PepperWebSocketHost::OnHostMsgConnect(
163 ppapi::host::HostMessageContext* context,
164 const std::string& url,
165 const std::vector<std::string>& protocols) {
166 std::string spec;
167 connect_reply_ = context->MakeReplyMessageContext();
168 int32_t result = DoConnect(url, protocols, &spec);
169 connect_reply_.params.set_result(result);
170 host()->SendReply(connect_reply_,
brettw 2012/10/03 20:30:24 As with the comment below, you shouldn't need to e
Takashi Toyoshima 2012/10/05 07:35:01 Done.
171 PpapiPluginMsg_WebSocket_ConnectReply(spec,
172 std::string()));
173 return result;
174 }
175
176 int32_t PepperWebSocketHost::OnHostMsgClose(
177 ppapi::host::HostMessageContext* context,
178 int32_t code,
179 const std::string& reason) {
180 close_reply_ = context->MakeReplyMessageContext();
181 if (!websocket_.get()) {
brettw 2012/10/03 20:30:24 In this error case, I believe you can just return
Takashi Toyoshima 2012/10/05 07:35:01 Done.
182 close_reply_.params.set_result(PP_ERROR_FAILED);
183 host()->SendReply(close_reply_,
184 PpapiPluginMsg_WebSocket_CloseReply(0,
185 false,
186 0,
187 std::string()));
188 }
189 initiating_close_ = true;
190 WebString web_reason = WebString::fromUTF8(reason);
191 websocket_->close(code, web_reason);
192 return PP_OK_COMPLETIONPENDING;
193 }
194
195 int32_t PepperWebSocketHost::OnHostMsgSendText(
196 ppapi::host::HostMessageContext* context,
197 const std::string& message) {
198 if (websocket_.get()) {
199 WebString web_message = WebString::fromUTF8(message);
200 websocket_->sendText(web_message);
201 }
202 return PP_OK;
203 }
204
205 int32_t PepperWebSocketHost::OnHostMsgSendBinary(
206 ppapi::host::HostMessageContext* context,
207 const std::vector<uint8_t>& message) {
208 if (websocket_.get()) {
209 WebArrayBuffer web_message = WebArrayBuffer::create(message.size(), 1);
210 memcpy(web_message.data(), &message.front(), message.size());
211 websocket_->sendArrayBuffer(web_message);
212 }
213 return PP_OK;
214 }
215
216 int32_t PepperWebSocketHost::OnHostMsgFail(
217 ppapi::host::HostMessageContext* context,
218 const std::string& message) {
219 if (websocket_.get())
220 websocket_->fail(WebString::fromUTF8(message));
221 return PP_OK;
222 }
223
224 int32_t PepperWebSocketHost::DoConnect(
225 const std::string& url,
226 const std::vector<std::string>& protocols,
227 std::string* spec) {
228 // Validate url and convert it to WebURL.
229 GURL gurl(url);
230 *spec = gurl.spec();
231 if (!gurl.is_valid())
232 return PP_ERROR_BADARGUMENT;
233 if (!gurl.SchemeIs("ws") && !gurl.SchemeIs("wss"))
234 return PP_ERROR_BADARGUMENT;
235 if (gurl.has_ref())
236 return PP_ERROR_BADARGUMENT;
237 if (!net::IsPortAllowedByDefault(gurl.IntPort()))
238 return PP_ERROR_BADARGUMENT;
239 WebURL web_url(gurl);
240
241 // Validate protocols.
242 std::string protocol_string;
243 for (std::vector<std::string>::const_iterator vector_it = protocols.begin();
244 vector_it != protocols.end();
245 ++vector_it) {
246 // Check containing characters.
247 for (std::string::const_iterator string_it = vector_it->begin();
248 string_it != vector_it->end();
249 ++string_it) {
250 uint8_t character = *string_it;
251 // WebSocket specification says "(Subprotocol string must consist of)
252 // characters in the range U+0021 to U+007E not including separator
253 // characters as defined in [RFC2616]."
254 const uint8_t minimumProtocolCharacter = '!'; // U+0021.
255 const uint8_t maximumProtocolCharacter = '~'; // U+007E.
256 if (character < minimumProtocolCharacter ||
257 character > maximumProtocolCharacter ||
258 character == '"' || character == '(' || character == ')' ||
259 character == ',' || character == '/' ||
260 (character >= ':' && character <= '@') || // U+003A - U+0040
261 (character >= '[' && character <= ']') || // U+005B - u+005D
262 character == '{' || character == '}')
263 return PP_ERROR_BADARGUMENT;
264 }
265 // Join protocols with the comma separator.
266 if (vector_it != protocols.begin())
267 protocol_string.append(",");
268 protocol_string.append(*vector_it);
269 }
270
271 // Convert protocols to WebString.
272 WebString web_protocols = WebString::fromUTF8(protocol_string);
273
274 // Create WebKit::WebSocket object and connect.
275 WebKit::WebPluginContainer* container =
276 renderer_ppapi_host_->GetContainerForInstance(pp_instance());
277 if (!container)
278 return PP_ERROR_BADARGUMENT;
279 // TODO(toyoshim) Remove following WebDocument object copy.
280 WebDocument document = container->element().document();
281 websocket_.reset(WebSocket::create(document, this));
282 DCHECK(websocket_.get());
283 if (!websocket_.get())
284 return PP_ERROR_NOTSUPPORTED;
285
286 // Set receiving binary object type.
287 websocket_->setBinaryType(WebSocket::BinaryTypeArrayBuffer);
288
289 websocket_->connect(web_url, web_protocols);
290 return PP_OK_COMPLETIONPENDING;
291 }
292
293 } // namespace content
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698