Re-subimission of https://codereview.chromium.org/1041213003/
[chromium-blink-merge.git] / extensions / browser / api / sockets_udp / sockets_udp_api.cc
blobb269aa8d750fcca474b2102a9051569f944a7f1f
1 // Copyright 2014 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.
5 #include "extensions/browser/api/sockets_udp/sockets_udp_api.h"
7 #include "content/public/common/socket_permission_request.h"
8 #include "extensions/browser/api/socket/udp_socket.h"
9 #include "extensions/browser/api/sockets_udp/udp_socket_event_dispatcher.h"
10 #include "extensions/common/api/sockets/sockets_manifest_data.h"
11 #include "net/base/net_errors.h"
13 namespace extensions {
14 namespace core_api {
16 using content::SocketPermissionRequest;
18 const char kSocketNotFoundError[] = "Socket not found";
19 const char kPermissionError[] = "App does not have permission";
20 const char kWildcardAddress[] = "*";
21 const int kWildcardPort = 0;
23 UDPSocketAsyncApiFunction::~UDPSocketAsyncApiFunction() {}
25 scoped_ptr<SocketResourceManagerInterface>
26 UDPSocketAsyncApiFunction::CreateSocketResourceManager() {
27 return scoped_ptr<SocketResourceManagerInterface>(
28 new SocketResourceManager<ResumableUDPSocket>()).Pass();
31 ResumableUDPSocket* UDPSocketAsyncApiFunction::GetUdpSocket(int socket_id) {
32 return static_cast<ResumableUDPSocket*>(GetSocket(socket_id));
35 UDPSocketExtensionWithDnsLookupFunction::
36 ~UDPSocketExtensionWithDnsLookupFunction() {}
38 scoped_ptr<SocketResourceManagerInterface>
39 UDPSocketExtensionWithDnsLookupFunction::CreateSocketResourceManager() {
40 return scoped_ptr<SocketResourceManagerInterface>(
41 new SocketResourceManager<ResumableUDPSocket>()).Pass();
44 ResumableUDPSocket* UDPSocketExtensionWithDnsLookupFunction::GetUdpSocket(
45 int socket_id) {
46 return static_cast<ResumableUDPSocket*>(GetSocket(socket_id));
49 linked_ptr<sockets_udp::SocketInfo> CreateSocketInfo(
50 int socket_id,
51 ResumableUDPSocket* socket) {
52 linked_ptr<sockets_udp::SocketInfo> socket_info(
53 new sockets_udp::SocketInfo());
54 // This represents what we know about the socket, and does not call through
55 // to the system.
56 socket_info->socket_id = socket_id;
57 if (!socket->name().empty()) {
58 socket_info->name.reset(new std::string(socket->name()));
60 socket_info->persistent = socket->persistent();
61 if (socket->buffer_size() > 0) {
62 socket_info->buffer_size.reset(new int(socket->buffer_size()));
64 socket_info->paused = socket->paused();
66 // Grab the local address as known by the OS.
67 net::IPEndPoint localAddress;
68 if (socket->GetLocalAddress(&localAddress)) {
69 socket_info->local_address.reset(
70 new std::string(localAddress.ToStringWithoutPort()));
71 socket_info->local_port.reset(new int(localAddress.port()));
74 return socket_info;
77 void SetSocketProperties(ResumableUDPSocket* socket,
78 sockets_udp::SocketProperties* properties) {
79 if (properties->name.get()) {
80 socket->set_name(*properties->name.get());
82 if (properties->persistent.get()) {
83 socket->set_persistent(*properties->persistent.get());
85 if (properties->buffer_size.get()) {
86 socket->set_buffer_size(*properties->buffer_size.get());
90 SocketsUdpCreateFunction::SocketsUdpCreateFunction() {}
92 SocketsUdpCreateFunction::~SocketsUdpCreateFunction() {}
94 bool SocketsUdpCreateFunction::Prepare() {
95 params_ = sockets_udp::Create::Params::Create(*args_);
96 EXTENSION_FUNCTION_VALIDATE(params_.get());
97 return true;
100 void SocketsUdpCreateFunction::Work() {
101 ResumableUDPSocket* socket = new ResumableUDPSocket(extension_->id());
103 sockets_udp::SocketProperties* properties = params_.get()->properties.get();
104 if (properties) {
105 SetSocketProperties(socket, properties);
108 sockets_udp::CreateInfo create_info;
109 create_info.socket_id = AddSocket(socket);
110 results_ = sockets_udp::Create::Results::Create(create_info);
113 SocketsUdpUpdateFunction::SocketsUdpUpdateFunction() {}
115 SocketsUdpUpdateFunction::~SocketsUdpUpdateFunction() {}
117 bool SocketsUdpUpdateFunction::Prepare() {
118 params_ = sockets_udp::Update::Params::Create(*args_);
119 EXTENSION_FUNCTION_VALIDATE(params_.get());
120 return true;
123 void SocketsUdpUpdateFunction::Work() {
124 ResumableUDPSocket* socket = GetUdpSocket(params_->socket_id);
125 if (!socket) {
126 error_ = kSocketNotFoundError;
127 return;
130 SetSocketProperties(socket, &params_.get()->properties);
131 results_ = sockets_udp::Update::Results::Create();
134 SocketsUdpSetPausedFunction::SocketsUdpSetPausedFunction()
135 : socket_event_dispatcher_(NULL) {}
137 SocketsUdpSetPausedFunction::~SocketsUdpSetPausedFunction() {}
139 bool SocketsUdpSetPausedFunction::Prepare() {
140 params_ = core_api::sockets_udp::SetPaused::Params::Create(*args_);
141 EXTENSION_FUNCTION_VALIDATE(params_.get());
143 socket_event_dispatcher_ = UDPSocketEventDispatcher::Get(browser_context());
144 DCHECK(socket_event_dispatcher_)
145 << "There is no socket event dispatcher. "
146 "If this assertion is failing during a test, then it is likely that "
147 "TestExtensionSystem is failing to provide an instance of "
148 "UDPSocketEventDispatcher.";
149 return socket_event_dispatcher_ != NULL;
152 void SocketsUdpSetPausedFunction::Work() {
153 ResumableUDPSocket* socket = GetUdpSocket(params_->socket_id);
154 if (!socket) {
155 error_ = kSocketNotFoundError;
156 return;
159 if (socket->paused() != params_->paused) {
160 socket->set_paused(params_->paused);
161 if (socket->IsBound() && !params_->paused) {
162 socket_event_dispatcher_->OnSocketResume(extension_->id(),
163 params_->socket_id);
167 results_ = sockets_udp::SetPaused::Results::Create();
170 SocketsUdpBindFunction::SocketsUdpBindFunction()
171 : socket_event_dispatcher_(NULL) {}
173 SocketsUdpBindFunction::~SocketsUdpBindFunction() {}
175 bool SocketsUdpBindFunction::Prepare() {
176 params_ = sockets_udp::Bind::Params::Create(*args_);
177 EXTENSION_FUNCTION_VALIDATE(params_.get());
179 socket_event_dispatcher_ = UDPSocketEventDispatcher::Get(browser_context());
180 DCHECK(socket_event_dispatcher_)
181 << "There is no socket event dispatcher. "
182 "If this assertion is failing during a test, then it is likely that "
183 "TestExtensionSystem is failing to provide an instance of "
184 "UDPSocketEventDispatcher.";
185 return socket_event_dispatcher_ != NULL;
188 void SocketsUdpBindFunction::AsyncWorkStart() {
189 ResumableUDPSocket* socket = GetUdpSocket(params_->socket_id);
190 if (!socket) {
191 error_ = kSocketNotFoundError;
192 AsyncWorkCompleted();
193 return;
196 content::SocketPermissionRequest param(
197 SocketPermissionRequest::UDP_BIND, params_->address, params_->port);
198 if (!SocketsManifestData::CheckRequest(extension(), param)) {
199 error_ = kPermissionError;
200 AsyncWorkCompleted();
201 return;
204 int net_result = socket->Bind(params_->address, params_->port);
205 results_ = sockets_udp::Bind::Results::Create(net_result);
206 if (net_result == net::OK) {
207 socket_event_dispatcher_->OnSocketBind(extension_->id(),
208 params_->socket_id);
209 } else {
210 error_ = net::ErrorToString(net_result);
211 AsyncWorkCompleted();
212 return;
215 OpenFirewallHole(params_->address, params_->socket_id, socket);
218 SocketsUdpSendFunction::SocketsUdpSendFunction() : io_buffer_size_(0) {}
220 SocketsUdpSendFunction::~SocketsUdpSendFunction() {}
222 bool SocketsUdpSendFunction::Prepare() {
223 params_ = sockets_udp::Send::Params::Create(*args_);
224 EXTENSION_FUNCTION_VALIDATE(params_.get());
225 io_buffer_size_ = params_->data.size();
226 io_buffer_ = new net::WrappedIOBuffer(params_->data.data());
228 return true;
231 void SocketsUdpSendFunction::AsyncWorkStart() {
232 ResumableUDPSocket* socket = GetUdpSocket(params_->socket_id);
233 if (!socket) {
234 error_ = kSocketNotFoundError;
235 AsyncWorkCompleted();
236 return;
239 content::SocketPermissionRequest param(
240 SocketPermissionRequest::UDP_SEND_TO, params_->address, params_->port);
241 if (!SocketsManifestData::CheckRequest(extension(), param)) {
242 error_ = kPermissionError;
243 AsyncWorkCompleted();
244 return;
247 StartDnsLookup(params_->address);
250 void SocketsUdpSendFunction::AfterDnsLookup(int lookup_result) {
251 if (lookup_result == net::OK) {
252 StartSendTo();
253 } else {
254 SetSendResult(lookup_result, -1);
258 void SocketsUdpSendFunction::StartSendTo() {
259 ResumableUDPSocket* socket = GetUdpSocket(params_->socket_id);
260 if (!socket) {
261 error_ = kSocketNotFoundError;
262 AsyncWorkCompleted();
263 return;
266 socket->SendTo(io_buffer_,
267 io_buffer_size_,
268 resolved_address_,
269 params_->port,
270 base::Bind(&SocketsUdpSendFunction::OnCompleted, this));
273 void SocketsUdpSendFunction::OnCompleted(int net_result) {
274 if (net_result >= net::OK) {
275 SetSendResult(net::OK, net_result);
276 } else {
277 SetSendResult(net_result, -1);
281 void SocketsUdpSendFunction::SetSendResult(int net_result, int bytes_sent) {
282 CHECK(net_result <= net::OK) << "Network status code must be < 0";
284 sockets_udp::SendInfo send_info;
285 send_info.result_code = net_result;
286 if (net_result == net::OK) {
287 send_info.bytes_sent.reset(new int(bytes_sent));
290 if (net_result != net::OK)
291 error_ = net::ErrorToString(net_result);
292 results_ = sockets_udp::Send::Results::Create(send_info);
293 AsyncWorkCompleted();
296 SocketsUdpCloseFunction::SocketsUdpCloseFunction() {}
298 SocketsUdpCloseFunction::~SocketsUdpCloseFunction() {}
300 bool SocketsUdpCloseFunction::Prepare() {
301 params_ = sockets_udp::Close::Params::Create(*args_);
302 EXTENSION_FUNCTION_VALIDATE(params_.get());
303 return true;
306 void SocketsUdpCloseFunction::Work() {
307 ResumableUDPSocket* socket = GetUdpSocket(params_->socket_id);
308 if (!socket) {
309 error_ = kSocketNotFoundError;
310 return;
313 socket->Disconnect();
314 RemoveSocket(params_->socket_id);
315 results_ = sockets_udp::Close::Results::Create();
318 SocketsUdpGetInfoFunction::SocketsUdpGetInfoFunction() {}
320 SocketsUdpGetInfoFunction::~SocketsUdpGetInfoFunction() {}
322 bool SocketsUdpGetInfoFunction::Prepare() {
323 params_ = sockets_udp::GetInfo::Params::Create(*args_);
324 EXTENSION_FUNCTION_VALIDATE(params_.get());
325 return true;
328 void SocketsUdpGetInfoFunction::Work() {
329 ResumableUDPSocket* socket = GetUdpSocket(params_->socket_id);
330 if (!socket) {
331 error_ = kSocketNotFoundError;
332 return;
335 linked_ptr<sockets_udp::SocketInfo> socket_info =
336 CreateSocketInfo(params_->socket_id, socket);
337 results_ = sockets_udp::GetInfo::Results::Create(*socket_info);
340 SocketsUdpGetSocketsFunction::SocketsUdpGetSocketsFunction() {}
342 SocketsUdpGetSocketsFunction::~SocketsUdpGetSocketsFunction() {}
344 bool SocketsUdpGetSocketsFunction::Prepare() { return true; }
346 void SocketsUdpGetSocketsFunction::Work() {
347 std::vector<linked_ptr<sockets_udp::SocketInfo> > socket_infos;
348 base::hash_set<int>* resource_ids = GetSocketIds();
349 if (resource_ids != NULL) {
350 for (base::hash_set<int>::iterator it = resource_ids->begin();
351 it != resource_ids->end();
352 ++it) {
353 int socket_id = *it;
354 ResumableUDPSocket* socket = GetUdpSocket(socket_id);
355 if (socket) {
356 socket_infos.push_back(CreateSocketInfo(socket_id, socket));
360 results_ = sockets_udp::GetSockets::Results::Create(socket_infos);
363 SocketsUdpJoinGroupFunction::SocketsUdpJoinGroupFunction() {}
365 SocketsUdpJoinGroupFunction::~SocketsUdpJoinGroupFunction() {}
367 bool SocketsUdpJoinGroupFunction::Prepare() {
368 params_ = sockets_udp::JoinGroup::Params::Create(*args_);
369 EXTENSION_FUNCTION_VALIDATE(params_.get());
370 return true;
373 void SocketsUdpJoinGroupFunction::Work() {
374 ResumableUDPSocket* socket = GetUdpSocket(params_->socket_id);
375 if (!socket) {
376 error_ = kSocketNotFoundError;
377 return;
380 content::SocketPermissionRequest param(
381 SocketPermissionRequest::UDP_MULTICAST_MEMBERSHIP,
382 kWildcardAddress,
383 kWildcardPort);
384 if (!SocketsManifestData::CheckRequest(extension(), param)) {
385 error_ = kPermissionError;
386 return;
389 int net_result = socket->JoinGroup(params_->address);
390 if (net_result != net::OK)
391 error_ = net::ErrorToString(net_result);
392 results_ = sockets_udp::JoinGroup::Results::Create(net_result);
395 SocketsUdpLeaveGroupFunction::SocketsUdpLeaveGroupFunction() {}
397 SocketsUdpLeaveGroupFunction::~SocketsUdpLeaveGroupFunction() {}
399 bool SocketsUdpLeaveGroupFunction::Prepare() {
400 params_ = core_api::sockets_udp::LeaveGroup::Params::Create(*args_);
401 EXTENSION_FUNCTION_VALIDATE(params_.get());
402 return true;
405 void SocketsUdpLeaveGroupFunction::Work() {
406 ResumableUDPSocket* socket = GetUdpSocket(params_->socket_id);
407 if (!socket) {
408 error_ = kSocketNotFoundError;
409 return;
412 content::SocketPermissionRequest param(
413 SocketPermissionRequest::UDP_MULTICAST_MEMBERSHIP,
414 kWildcardAddress,
415 kWildcardPort);
416 if (!SocketsManifestData::CheckRequest(extension(), param)) {
417 error_ = kPermissionError;
418 return;
421 int net_result = socket->LeaveGroup(params_->address);
422 if (net_result != net::OK)
423 error_ = net::ErrorToString(net_result);
424 results_ = sockets_udp::LeaveGroup::Results::Create(net_result);
427 SocketsUdpSetMulticastTimeToLiveFunction::
428 SocketsUdpSetMulticastTimeToLiveFunction() {}
430 SocketsUdpSetMulticastTimeToLiveFunction::
431 ~SocketsUdpSetMulticastTimeToLiveFunction() {}
433 bool SocketsUdpSetMulticastTimeToLiveFunction::Prepare() {
434 params_ =
435 core_api::sockets_udp::SetMulticastTimeToLive::Params::Create(*args_);
436 EXTENSION_FUNCTION_VALIDATE(params_.get());
437 return true;
440 void SocketsUdpSetMulticastTimeToLiveFunction::Work() {
441 ResumableUDPSocket* socket = GetUdpSocket(params_->socket_id);
442 if (!socket) {
443 error_ = kSocketNotFoundError;
444 return;
447 int net_result = socket->SetMulticastTimeToLive(params_->ttl);
448 if (net_result != net::OK)
449 error_ = net::ErrorToString(net_result);
450 results_ = sockets_udp::SetMulticastTimeToLive::Results::Create(net_result);
453 SocketsUdpSetMulticastLoopbackModeFunction::
454 SocketsUdpSetMulticastLoopbackModeFunction() {}
456 SocketsUdpSetMulticastLoopbackModeFunction::
457 ~SocketsUdpSetMulticastLoopbackModeFunction() {}
459 bool SocketsUdpSetMulticastLoopbackModeFunction::Prepare() {
460 params_ =
461 core_api::sockets_udp::SetMulticastLoopbackMode::Params::Create(*args_);
462 EXTENSION_FUNCTION_VALIDATE(params_.get());
463 return true;
466 void SocketsUdpSetMulticastLoopbackModeFunction::Work() {
467 ResumableUDPSocket* socket = GetUdpSocket(params_->socket_id);
468 if (!socket) {
469 error_ = kSocketNotFoundError;
470 return;
473 int net_result = socket->SetMulticastLoopbackMode(params_->enabled);
474 if (net_result != net::OK)
475 error_ = net::ErrorToString(net_result);
476 results_ = sockets_udp::SetMulticastLoopbackMode::Results::Create(net_result);
479 SocketsUdpGetJoinedGroupsFunction::SocketsUdpGetJoinedGroupsFunction() {}
481 SocketsUdpGetJoinedGroupsFunction::~SocketsUdpGetJoinedGroupsFunction() {}
483 bool SocketsUdpGetJoinedGroupsFunction::Prepare() {
484 params_ = core_api::sockets_udp::GetJoinedGroups::Params::Create(*args_);
485 EXTENSION_FUNCTION_VALIDATE(params_.get());
486 return true;
489 void SocketsUdpGetJoinedGroupsFunction::Work() {
490 ResumableUDPSocket* socket = GetUdpSocket(params_->socket_id);
491 if (!socket) {
492 error_ = kSocketNotFoundError;
493 return;
496 content::SocketPermissionRequest param(
497 SocketPermissionRequest::UDP_MULTICAST_MEMBERSHIP,
498 kWildcardAddress,
499 kWildcardPort);
500 if (!SocketsManifestData::CheckRequest(extension(), param)) {
501 error_ = kPermissionError;
502 return;
505 const std::vector<std::string>& groups = socket->GetJoinedGroups();
506 results_ = sockets_udp::GetJoinedGroups::Results::Create(groups);
509 } // namespace core_api
510 } // namespace extensions