blob: b5f5afd47a01d60643794550ffefa50563c1f313 [file] [log] [blame]
// Copyright 2014 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 "chrome/browser/extensions/api/bluetooth_low_energy/bluetooth_low_energy_event_router.h"
#include "base/bind.h"
#include "base/logging.h"
#include "base/values.h"
#include "chrome/browser/extensions/api/bluetooth_low_energy/bluetooth_low_energy_connection.h"
#include "chrome/browser/extensions/api/bluetooth_low_energy/bluetooth_low_energy_notify_session.h"
#include "chrome/browser/extensions/api/bluetooth_low_energy/utils.h"
#include "chrome/common/extensions/api/bluetooth/bluetooth_manifest_data.h"
#include "content/public/browser/browser_thread.h"
#include "device/bluetooth/bluetooth_adapter_factory.h"
#include "device/bluetooth/bluetooth_gatt_characteristic.h"
#include "device/bluetooth/bluetooth_gatt_connection.h"
#include "device/bluetooth/bluetooth_gatt_descriptor.h"
#include "extensions/browser/event_router.h"
#include "extensions/browser/extension_registry.h"
using content::BrowserThread;
using device::BluetoothAdapter;
using device::BluetoothAdapterFactory;
using device::BluetoothDevice;
using device::BluetoothGattCharacteristic;
using device::BluetoothGattConnection;
using device::BluetoothGattDescriptor;
using device::BluetoothGattService;
namespace apibtle = extensions::api::bluetooth_low_energy;
namespace {
void PopulateService(const BluetoothGattService* service,
apibtle::Service* out) {
DCHECK(out);
out->uuid = service->GetUUID().canonical_value();
out->is_primary = service->IsPrimary();
out->is_local = service->IsLocal();
out->instance_id.reset(new std::string(service->GetIdentifier()));
if (!service->GetDevice())
return;
out->device_address.reset(
new std::string(service->GetDevice()->GetAddress()));
}
void PopulateCharacteristicProperties(
BluetoothGattCharacteristic::Properties properties,
std::vector<apibtle::CharacteristicProperty>* api_properties) {
DCHECK(api_properties && api_properties->empty());
if (properties == BluetoothGattCharacteristic::kPropertyNone)
return;
if (properties & BluetoothGattCharacteristic::kPropertyBroadcast)
api_properties->push_back(apibtle::CHARACTERISTIC_PROPERTY_BROADCAST);
if (properties & BluetoothGattCharacteristic::kPropertyRead)
api_properties->push_back(apibtle::CHARACTERISTIC_PROPERTY_READ);
if (properties & BluetoothGattCharacteristic::kPropertyWriteWithoutResponse) {
api_properties->push_back(
apibtle::CHARACTERISTIC_PROPERTY_WRITEWITHOUTRESPONSE);
}
if (properties & BluetoothGattCharacteristic::kPropertyWrite)
api_properties->push_back(apibtle::CHARACTERISTIC_PROPERTY_WRITE);
if (properties & BluetoothGattCharacteristic::kPropertyNotify)
api_properties->push_back(apibtle::CHARACTERISTIC_PROPERTY_NOTIFY);
if (properties & BluetoothGattCharacteristic::kPropertyIndicate)
api_properties->push_back(apibtle::CHARACTERISTIC_PROPERTY_INDICATE);
if (properties &
BluetoothGattCharacteristic::kPropertyAuthenticatedSignedWrites) {
api_properties->push_back(
apibtle::CHARACTERISTIC_PROPERTY_AUTHENTICATEDSIGNEDWRITES);
}
if (properties & BluetoothGattCharacteristic::kPropertyExtendedProperties) {
api_properties->push_back(
apibtle::CHARACTERISTIC_PROPERTY_EXTENDEDPROPERTIES);
}
if (properties & BluetoothGattCharacteristic::kPropertyReliableWrite)
api_properties->push_back(apibtle::CHARACTERISTIC_PROPERTY_RELIABLEWRITE);
if (properties & BluetoothGattCharacteristic::kPropertyWritableAuxiliaries) {
api_properties->push_back(
apibtle::CHARACTERISTIC_PROPERTY_WRITABLEAUXILIARIES);
}
}
void PopulateCharacteristic(const BluetoothGattCharacteristic* characteristic,
apibtle::Characteristic* out) {
DCHECK(out);
out->uuid = characteristic->GetUUID().canonical_value();
out->is_local = characteristic->IsLocal();
out->instance_id.reset(new std::string(characteristic->GetIdentifier()));
PopulateService(characteristic->GetService(), &out->service);
PopulateCharacteristicProperties(characteristic->GetProperties(),
&out->properties);
const std::vector<uint8>& value = characteristic->GetValue();
if (value.empty())
return;
out->value.reset(new std::string(value.begin(), value.end()));
}
void PopulateDescriptor(const BluetoothGattDescriptor* descriptor,
apibtle::Descriptor* out) {
DCHECK(out);
out->uuid = descriptor->GetUUID().canonical_value();
out->is_local = descriptor->IsLocal();
out->instance_id.reset(new std::string(descriptor->GetIdentifier()));
PopulateCharacteristic(descriptor->GetCharacteristic(), &out->characteristic);
const std::vector<uint8>& value = descriptor->GetValue();
if (value.empty())
return;
out->value.reset(new std::string(value.begin(), value.end()));
}
typedef extensions::ApiResourceManager<extensions::BluetoothLowEnergyConnection>
ConnectionResourceManager;
ConnectionResourceManager* GetConnectionResourceManager(
content::BrowserContext* context) {
ConnectionResourceManager* manager = ConnectionResourceManager::Get(context);
DCHECK(manager)
<< "There is no Bluetooth low energy connection manager. "
"If this assertion is failing during a test, then it is likely that "
"TestExtensionSystem is failing to provide an instance of "
"ApiResourceManager<BluetoothLowEnergyConnection>.";
return manager;
}
typedef extensions::ApiResourceManager<
extensions::BluetoothLowEnergyNotifySession> NotifySessionResourceManager;
NotifySessionResourceManager* GetNotifySessionResourceManager(
content::BrowserContext* context) {
NotifySessionResourceManager* manager =
NotifySessionResourceManager::Get(context);
DCHECK(manager)
<< "There is no Bluetooth low energy value update session manager."
"If this assertion is failing during a test, then it is likely that "
"TestExtensionSystem is failing to provide an instance of "
"ApiResourceManager<BluetoothLowEnergyNotifySession>.";
return manager;
}
} // namespace
namespace extensions {
BluetoothLowEnergyEventRouter::BluetoothLowEnergyEventRouter(
content::BrowserContext* context)
: adapter_(NULL), browser_context_(context), weak_ptr_factory_(this) {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
DCHECK(browser_context_);
VLOG(1) << "Initializing BluetoothLowEnergyEventRouter.";
if (!IsBluetoothSupported()) {
VLOG(1) << "Bluetooth not supported on the current platform.";
return;
}
}
BluetoothLowEnergyEventRouter::~BluetoothLowEnergyEventRouter() {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
if (!adapter_.get())
return;
adapter_->RemoveObserver(this);
for (std::set<std::string>::const_iterator iter = observed_devices_.begin();
iter != observed_devices_.end();
++iter) {
BluetoothDevice* device = adapter_->GetDevice(*iter);
if (!device)
continue;
device->RemoveObserver(this);
}
for (std::set<std::string>::const_iterator iter =
observed_gatt_services_.begin();
iter != observed_gatt_services_.end();
++iter) {
BluetoothGattService* service = FindServiceById(*iter);
if (!service)
continue;
service->RemoveObserver(this);
}
adapter_ = NULL;
}
bool BluetoothLowEnergyEventRouter::IsBluetoothSupported() const {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
return adapter_.get() ||
BluetoothAdapterFactory::IsBluetoothAdapterAvailable();
}
bool BluetoothLowEnergyEventRouter::InitializeAdapterAndInvokeCallback(
const base::Closure& callback) {
if (!IsBluetoothSupported())
return false;
if (adapter_.get()) {
callback.Run();
return true;
}
BluetoothAdapterFactory::GetAdapter(
base::Bind(&BluetoothLowEnergyEventRouter::OnGetAdapter,
weak_ptr_factory_.GetWeakPtr(),
callback));
return true;
}
bool BluetoothLowEnergyEventRouter::HasAdapter() const {
return (adapter_.get() != NULL);
}
void BluetoothLowEnergyEventRouter::Connect(
bool persistent,
const Extension* extension,
const std::string& device_address,
const base::Closure& callback,
const ErrorCallback& error_callback) {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
if (!adapter_) {
VLOG(1) << "BluetoothAdapter not ready.";
error_callback.Run(kStatusErrorFailed);
return;
}
const std::string extension_id = extension->id();
const std::string connect_id = extension_id + device_address;
if (connecting_devices_.count(connect_id) != 0) {
error_callback.Run(kStatusErrorInProgress);
return;
}
BluetoothLowEnergyConnection* conn =
FindConnection(extension_id, device_address);
if (conn) {
if (conn->GetConnection()->IsConnected()) {
VLOG(1) << "Application already connected to device: " << device_address;
error_callback.Run(kStatusErrorAlreadyConnected);
return;
}
// There is a connection object but it's no longer active. Simply remove it.
RemoveConnection(extension_id, device_address);
}
BluetoothDevice* device = adapter_->GetDevice(device_address);
if (!device) {
VLOG(1) << "Bluetooth device not found: " << device_address;
error_callback.Run(kStatusErrorNotFound);
return;
}
connecting_devices_.insert(connect_id);
device->CreateGattConnection(
base::Bind(&BluetoothLowEnergyEventRouter::OnCreateGattConnection,
weak_ptr_factory_.GetWeakPtr(),
persistent,
extension_id,
device_address,
callback),
base::Bind(&BluetoothLowEnergyEventRouter::OnConnectError,
weak_ptr_factory_.GetWeakPtr(),
extension_id,
device_address,
error_callback));
}
void BluetoothLowEnergyEventRouter::Disconnect(
const Extension* extension,
const std::string& device_address,
const base::Closure& callback,
const ErrorCallback& error_callback) {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
if (!adapter_) {
VLOG(1) << "BluetoothAdapter not ready.";
error_callback.Run(kStatusErrorFailed);
return;
}
const std::string extension_id = extension->id();
const std::string disconnect_id = extension_id + device_address;
if (disconnecting_devices_.count(disconnect_id) != 0) {
error_callback.Run(kStatusErrorInProgress);
return;
}
BluetoothLowEnergyConnection* conn =
FindConnection(extension_id, device_address);
if (!conn || !conn->GetConnection()->IsConnected()) {
VLOG(1) << "Application not connected to device: " << device_address;
error_callback.Run(kStatusErrorNotConnected);
return;
}
disconnecting_devices_.insert(disconnect_id);
conn->GetConnection()->Disconnect(
base::Bind(&BluetoothLowEnergyEventRouter::OnDisconnect,
weak_ptr_factory_.GetWeakPtr(),
extension_id,
device_address,
callback));
}
bool BluetoothLowEnergyEventRouter::GetServices(
const std::string& device_address,
ServiceList* out_services) const {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
DCHECK(out_services);
if (!adapter_) {
VLOG(1) << "BluetoothAdapter not ready.";
return false;
}
BluetoothDevice* device = adapter_->GetDevice(device_address);
if (!device) {
VLOG(1) << "Bluetooth device not found: " << device_address;
return false;
}
out_services->clear();
const std::vector<BluetoothGattService*>& services =
device->GetGattServices();
for (std::vector<BluetoothGattService*>::const_iterator iter =
services.begin();
iter != services.end();
++iter) {
// Populate an API service and add it to the return value.
const BluetoothGattService* service = *iter;
linked_ptr<apibtle::Service> api_service(new apibtle::Service());
PopulateService(service, api_service.get());
out_services->push_back(api_service);
}
return true;
}
BluetoothLowEnergyEventRouter::Status BluetoothLowEnergyEventRouter::GetService(
const std::string& instance_id,
apibtle::Service* out_service) const {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
DCHECK(out_service);
if (!adapter_) {
VLOG(1) << "BluetoothAdapter not ready.";
return kStatusErrorFailed;
}
BluetoothGattService* gatt_service = FindServiceById(instance_id);
if (!gatt_service) {
VLOG(1) << "Service not found: " << instance_id;
return kStatusErrorNotFound;
}
PopulateService(gatt_service, out_service);
return kStatusSuccess;
}
BluetoothLowEnergyEventRouter::Status
BluetoothLowEnergyEventRouter::GetIncludedServices(
const std::string& instance_id,
ServiceList* out_services) const {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
DCHECK(out_services);
if (!adapter_) {
VLOG(1) << "BluetoothAdapter not ready.";
return kStatusErrorFailed;
}
BluetoothGattService* service = FindServiceById(instance_id);
if (!service) {
VLOG(1) << "Service not found: " << instance_id;
return kStatusErrorNotFound;
}
out_services->clear();
const std::vector<BluetoothGattService*>& includes =
service->GetIncludedServices();
for (std::vector<BluetoothGattService*>::const_iterator iter =
includes.begin();
iter != includes.end();
++iter) {
// Populate an API service and add it to the return value.
const BluetoothGattService* included = *iter;
linked_ptr<apibtle::Service> api_service(new apibtle::Service());
PopulateService(included, api_service.get());
out_services->push_back(api_service);
}
return kStatusSuccess;
}
BluetoothLowEnergyEventRouter::Status
BluetoothLowEnergyEventRouter::GetCharacteristics(
const Extension* extension,
const std::string& instance_id,
CharacteristicList* out_characteristics) const {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
DCHECK(extension);
DCHECK(out_characteristics);
if (!adapter_) {
VLOG(1) << "BlutoothAdapter not ready.";
return kStatusErrorFailed;
}
BluetoothGattService* service = FindServiceById(instance_id);
if (!service) {
VLOG(1) << "Service not found: " << instance_id;
return kStatusErrorNotFound;
}
BluetoothPermissionRequest request(service->GetUUID().value());
if (!BluetoothManifestData::CheckRequest(extension, request)) {
VLOG(1) << "App has no permission to access the characteristics of this "
<< "service: " << instance_id;
return kStatusErrorPermissionDenied;
}
out_characteristics->clear();
const std::vector<BluetoothGattCharacteristic*>& characteristics =
service->GetCharacteristics();
for (std::vector<BluetoothGattCharacteristic*>::const_iterator iter =
characteristics.begin();
iter != characteristics.end();
++iter) {
// Populate an API characteristic and add it to the return value.
const BluetoothGattCharacteristic* characteristic = *iter;
linked_ptr<apibtle::Characteristic> api_characteristic(
new apibtle::Characteristic());
PopulateCharacteristic(characteristic, api_characteristic.get());
out_characteristics->push_back(api_characteristic);
}
return kStatusSuccess;
}
BluetoothLowEnergyEventRouter::Status
BluetoothLowEnergyEventRouter::GetCharacteristic(
const Extension* extension,
const std::string& instance_id,
apibtle::Characteristic* out_characteristic) const {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
DCHECK(extension);
DCHECK(out_characteristic);
if (!adapter_) {
VLOG(1) << "BluetoothAdapter not ready.";
return kStatusErrorFailed;
}
BluetoothGattCharacteristic* characteristic =
FindCharacteristicById(instance_id);
if (!characteristic) {
VLOG(1) << "Characteristic not found: " << instance_id;
return kStatusErrorNotFound;
}
BluetoothPermissionRequest request(
characteristic->GetService()->GetUUID().value());
if (!BluetoothManifestData::CheckRequest(extension, request)) {
VLOG(1) << "App has no permission to access this characteristic: "
<< instance_id;
return kStatusErrorPermissionDenied;
}
PopulateCharacteristic(characteristic, out_characteristic);
return kStatusSuccess;
}
BluetoothLowEnergyEventRouter::Status
BluetoothLowEnergyEventRouter::GetDescriptors(
const Extension* extension,
const std::string& instance_id,
DescriptorList* out_descriptors) const {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
DCHECK(extension);
DCHECK(out_descriptors);
if (!adapter_) {
VLOG(1) << "BlutoothAdapter not ready.";
return kStatusErrorFailed;
}
BluetoothGattCharacteristic* characteristic =
FindCharacteristicById(instance_id);
if (!characteristic) {
VLOG(1) << "Characteristic not found: " << instance_id;
return kStatusErrorNotFound;
}
BluetoothPermissionRequest request(
characteristic->GetService()->GetUUID().value());
if (!BluetoothManifestData::CheckRequest(extension, request)) {
VLOG(1) << "App has no permission to access the descriptors of this "
<< "characteristic: " << instance_id;
return kStatusErrorPermissionDenied;
}
out_descriptors->clear();
const std::vector<BluetoothGattDescriptor*>& descriptors =
characteristic->GetDescriptors();
for (std::vector<BluetoothGattDescriptor*>::const_iterator iter =
descriptors.begin();
iter != descriptors.end();
++iter) {
// Populate an API descriptor and add it to the return value.
const BluetoothGattDescriptor* descriptor = *iter;
linked_ptr<apibtle::Descriptor> api_descriptor(new apibtle::Descriptor());
PopulateDescriptor(descriptor, api_descriptor.get());
out_descriptors->push_back(api_descriptor);
}
return kStatusSuccess;
}
BluetoothLowEnergyEventRouter::Status
BluetoothLowEnergyEventRouter::GetDescriptor(
const Extension* extension,
const std::string& instance_id,
api::bluetooth_low_energy::Descriptor* out_descriptor) const {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
DCHECK(extension);
DCHECK(out_descriptor);
if (!adapter_) {
VLOG(1) << "BluetoothAdapter not ready.";
return kStatusErrorFailed;
}
BluetoothGattDescriptor* descriptor = FindDescriptorById(instance_id);
if (!descriptor) {
VLOG(1) << "Descriptor not found: " << instance_id;
return kStatusErrorNotFound;
}
BluetoothPermissionRequest request(
descriptor->GetCharacteristic()->GetService()->GetUUID().value());
if (!BluetoothManifestData::CheckRequest(extension, request)) {
VLOG(1) << "App has no permission to access this descriptor: "
<< instance_id;
return kStatusErrorPermissionDenied;
}
PopulateDescriptor(descriptor, out_descriptor);
return kStatusSuccess;
}
void BluetoothLowEnergyEventRouter::ReadCharacteristicValue(
const Extension* extension,
const std::string& instance_id,
const base::Closure& callback,
const ErrorCallback& error_callback) {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
DCHECK(extension);
if (!adapter_) {
VLOG(1) << "BluetoothAdapter not ready.";
error_callback.Run(kStatusErrorFailed);
return;
}
BluetoothGattCharacteristic* characteristic =
FindCharacteristicById(instance_id);
if (!characteristic) {
VLOG(1) << "Characteristic not found: " << instance_id;
error_callback.Run(kStatusErrorNotFound);
return;
}
BluetoothPermissionRequest request(
characteristic->GetService()->GetUUID().value());
if (!BluetoothManifestData::CheckRequest(extension, request)) {
VLOG(1) << "App has no permission to access this characteristic: "
<< instance_id;
error_callback.Run(kStatusErrorPermissionDenied);
return;
}
characteristic->ReadRemoteCharacteristic(
base::Bind(&BluetoothLowEnergyEventRouter::OnValueSuccess,
weak_ptr_factory_.GetWeakPtr(),
callback),
base::Bind(&BluetoothLowEnergyEventRouter::OnError,
weak_ptr_factory_.GetWeakPtr(),
error_callback));
}
void BluetoothLowEnergyEventRouter::WriteCharacteristicValue(
const Extension* extension,
const std::string& instance_id,
const std::vector<uint8>& value,
const base::Closure& callback,
const ErrorCallback& error_callback) {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
DCHECK(extension);
if (!adapter_) {
VLOG(1) << "BluetoothAdapter not ready.";
error_callback.Run(kStatusErrorFailed);
return;
}
BluetoothGattCharacteristic* characteristic =
FindCharacteristicById(instance_id);
if (!characteristic) {
VLOG(1) << "Characteristic not found: " << instance_id;
error_callback.Run(kStatusErrorNotFound);
return;
}
BluetoothPermissionRequest request(
characteristic->GetService()->GetUUID().value());
if (!BluetoothManifestData::CheckRequest(extension, request)) {
VLOG(1) << "App has no permission to access this characteristic: "
<< instance_id;
error_callback.Run(kStatusErrorPermissionDenied);
return;
}
characteristic->WriteRemoteCharacteristic(
value,
callback,
base::Bind(&BluetoothLowEnergyEventRouter::OnError,
weak_ptr_factory_.GetWeakPtr(),
error_callback));
}
void BluetoothLowEnergyEventRouter::StartCharacteristicNotifications(
bool persistent,
const Extension* extension,
const std::string& instance_id,
const base::Closure& callback,
const ErrorCallback& error_callback) {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
if (!adapter_) {
VLOG(1) << "BluetoothAdapter not ready.";
error_callback.Run(kStatusErrorFailed);
return;
}
const std::string extension_id = extension->id();
const std::string session_id = extension_id + instance_id;
if (pending_session_calls_.count(session_id) != 0) {
error_callback.Run(kStatusErrorInProgress);
return;
}
BluetoothLowEnergyNotifySession* session =
FindNotifySession(extension_id, instance_id);
if (session) {
if (session->GetSession()->IsActive()) {
VLOG(1) << "Application has already enabled notifications from "
<< "characteristic: " << instance_id;
error_callback.Run(kStatusErrorAlreadyNotifying);
return;
}
RemoveNotifySession(extension_id, instance_id);
}
BluetoothGattCharacteristic* characteristic =
FindCharacteristicById(instance_id);
if (!characteristic) {
VLOG(1) << "Characteristic not found: " << instance_id;
error_callback.Run(kStatusErrorNotFound);
return;
}
BluetoothPermissionRequest request(
characteristic->GetService()->GetUUID().value());
if (!BluetoothManifestData::CheckRequest(extension, request)) {
VLOG(1) << "App has no permission to access this characteristic: "
<< instance_id;
error_callback.Run(kStatusErrorPermissionDenied);
return;
}
pending_session_calls_.insert(session_id);
characteristic->StartNotifySession(
base::Bind(&BluetoothLowEnergyEventRouter::OnStartNotifySession,
weak_ptr_factory_.GetWeakPtr(),
persistent,
extension_id,
instance_id,
callback),
base::Bind(&BluetoothLowEnergyEventRouter::OnStartNotifySessionError,
weak_ptr_factory_.GetWeakPtr(),
extension_id,
instance_id,
error_callback));
}
void BluetoothLowEnergyEventRouter::StopCharacteristicNotifications(
const Extension* extension,
const std::string& instance_id,
const base::Closure& callback,
const ErrorCallback& error_callback) {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
if (!adapter_) {
VLOG(1) << "BluetoothAdapter not ready.";
error_callback.Run(kStatusErrorFailed);
return;
}
const std::string extension_id = extension->id();
BluetoothLowEnergyNotifySession* session =
FindNotifySession(extension_id, instance_id);
if (!session || !session->GetSession()->IsActive()) {
VLOG(1) << "Application has not enabled notifications from "
<< "characteristic: " << instance_id;
error_callback.Run(kStatusErrorNotNotifying);
return;
}
session->GetSession()->Stop(
base::Bind(&BluetoothLowEnergyEventRouter::OnStopNotifySession,
weak_ptr_factory_.GetWeakPtr(),
extension_id,
instance_id,
callback));
}
void BluetoothLowEnergyEventRouter::ReadDescriptorValue(
const Extension* extension,
const std::string& instance_id,
const base::Closure& callback,
const ErrorCallback& error_callback) {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
DCHECK(extension);
if (!adapter_) {
VLOG(1) << "BluetoothAdapter not ready.";
error_callback.Run(kStatusErrorFailed);
return;
}
BluetoothGattDescriptor* descriptor = FindDescriptorById(instance_id);
if (!descriptor) {
VLOG(1) << "Descriptor not found: " << instance_id;
error_callback.Run(kStatusErrorNotFound);
return;
}
BluetoothPermissionRequest request(
descriptor->GetCharacteristic()->GetService()->GetUUID().value());
if (!BluetoothManifestData::CheckRequest(extension, request)) {
VLOG(1) << "App has no permission to access this descriptor: "
<< instance_id;
error_callback.Run(kStatusErrorPermissionDenied);
return;
}
descriptor->ReadRemoteDescriptor(
base::Bind(&BluetoothLowEnergyEventRouter::OnValueSuccess,
weak_ptr_factory_.GetWeakPtr(),
callback),
base::Bind(&BluetoothLowEnergyEventRouter::OnError,
weak_ptr_factory_.GetWeakPtr(),
error_callback));
}
void BluetoothLowEnergyEventRouter::WriteDescriptorValue(
const Extension* extension,
const std::string& instance_id,
const std::vector<uint8>& value,
const base::Closure& callback,
const ErrorCallback& error_callback) {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
DCHECK(extension);
if (!adapter_) {
VLOG(1) << "BluetoothAdapter not ready.";
error_callback.Run(kStatusErrorFailed);
return;
}
BluetoothGattDescriptor* descriptor = FindDescriptorById(instance_id);
if (!descriptor) {
VLOG(1) << "Descriptor not found: " << instance_id;
error_callback.Run(kStatusErrorNotFound);
return;
}
BluetoothPermissionRequest request(
descriptor->GetCharacteristic()->GetService()->GetUUID().value());
if (!BluetoothManifestData::CheckRequest(extension, request)) {
VLOG(1) << "App has no permission to access this descriptor: "
<< instance_id;
error_callback.Run(kStatusErrorPermissionDenied);
return;
}
descriptor->WriteRemoteDescriptor(
value,
callback,
base::Bind(&BluetoothLowEnergyEventRouter::OnError,
weak_ptr_factory_.GetWeakPtr(),
error_callback));
}
void BluetoothLowEnergyEventRouter::SetAdapterForTesting(
device::BluetoothAdapter* adapter) {
adapter_ = adapter;
InitializeIdentifierMappings();
}
void BluetoothLowEnergyEventRouter::DeviceAdded(BluetoothAdapter* adapter,
BluetoothDevice* device) {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
DCHECK(observed_devices_.find(device->GetAddress()) ==
observed_devices_.end());
device->AddObserver(this);
observed_devices_.insert(device->GetAddress());
}
void BluetoothLowEnergyEventRouter::DeviceRemoved(BluetoothAdapter* adapter,
BluetoothDevice* device) {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
DCHECK(observed_devices_.find(device->GetAddress()) !=
observed_devices_.end());
device->RemoveObserver(this);
observed_devices_.erase(device->GetAddress());
}
void BluetoothLowEnergyEventRouter::GattServiceAdded(
BluetoothDevice* device,
BluetoothGattService* service) {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
VLOG(2) << "GATT service added: " << service->GetIdentifier();
DCHECK(observed_gatt_services_.find(service->GetIdentifier()) ==
observed_gatt_services_.end());
DCHECK(service_id_to_device_address_.find(service->GetIdentifier()) ==
service_id_to_device_address_.end());
service->AddObserver(this);
const std::string& service_id = service->GetIdentifier();
observed_gatt_services_.insert(service_id);
service_id_to_device_address_[service_id] = device->GetAddress();
// Signal API event.
apibtle::Service api_service;
PopulateService(service, &api_service);
scoped_ptr<base::ListValue> args =
apibtle::OnServiceAdded::Create(api_service);
scoped_ptr<Event> event(
new Event(apibtle::OnServiceAdded::kEventName, args.Pass()));
EventRouter::Get(browser_context_)->BroadcastEvent(event.Pass());
}
void BluetoothLowEnergyEventRouter::GattServiceRemoved(
BluetoothDevice* device,
BluetoothGattService* service) {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
VLOG(2) << "GATT service removed: " << service->GetIdentifier();
DCHECK(observed_gatt_services_.find(service->GetIdentifier()) !=
observed_gatt_services_.end());
DCHECK(service_id_to_device_address_.find(service->GetIdentifier()) !=
service_id_to_device_address_.end());
service->RemoveObserver(this);
observed_gatt_services_.erase(service->GetIdentifier());
DCHECK(device->GetAddress() ==
service_id_to_device_address_[service->GetIdentifier()]);
service_id_to_device_address_.erase(service->GetIdentifier());
// Signal API event.
apibtle::Service api_service;
PopulateService(service, &api_service);
scoped_ptr<base::ListValue> args =
apibtle::OnServiceRemoved::Create(api_service);
scoped_ptr<Event> event(
new Event(apibtle::OnServiceRemoved::kEventName, args.Pass()));
EventRouter::Get(browser_context_)->BroadcastEvent(event.Pass());
}
void BluetoothLowEnergyEventRouter::GattServiceChanged(
BluetoothGattService* service) {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
VLOG(2) << "GATT service changed: " << service->GetIdentifier();
DCHECK(observed_gatt_services_.find(service->GetIdentifier()) !=
observed_gatt_services_.end());
DCHECK(service_id_to_device_address_.find(service->GetIdentifier()) !=
service_id_to_device_address_.end());
// Signal API event.
apibtle::Service api_service;
PopulateService(service, &api_service);
DispatchEventToExtensionsWithPermission(
apibtle::OnServiceChanged::kEventName,
service->GetUUID(),
"" /* characteristic_id */,
apibtle::OnServiceChanged::Create(api_service));
}
void BluetoothLowEnergyEventRouter::GattCharacteristicAdded(
BluetoothGattService* service,
BluetoothGattCharacteristic* characteristic) {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
VLOG(2) << "GATT characteristic added: " << characteristic->GetIdentifier();
DCHECK(chrc_id_to_service_id_.find(characteristic->GetIdentifier()) ==
chrc_id_to_service_id_.end());
DCHECK(service_id_to_device_address_.find(service->GetIdentifier()) !=
service_id_to_device_address_.end());
chrc_id_to_service_id_[characteristic->GetIdentifier()] =
service->GetIdentifier();
}
void BluetoothLowEnergyEventRouter::GattCharacteristicRemoved(
BluetoothGattService* service,
BluetoothGattCharacteristic* characteristic) {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
VLOG(2) << "GATT characteristic removed: " << characteristic->GetIdentifier();
DCHECK(chrc_id_to_service_id_.find(characteristic->GetIdentifier()) !=
chrc_id_to_service_id_.end());
DCHECK(service->GetIdentifier() ==
chrc_id_to_service_id_[characteristic->GetIdentifier()]);
chrc_id_to_service_id_.erase(characteristic->GetIdentifier());
}
void BluetoothLowEnergyEventRouter::GattDescriptorAdded(
BluetoothGattCharacteristic* characteristic,
BluetoothGattDescriptor* descriptor) {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
VLOG(2) << "GATT descriptor added: " << descriptor->GetIdentifier();
DCHECK(desc_id_to_chrc_id_.find(descriptor->GetIdentifier()) ==
desc_id_to_chrc_id_.end());
DCHECK(chrc_id_to_service_id_.find(characteristic->GetIdentifier()) !=
chrc_id_to_service_id_.end());
desc_id_to_chrc_id_[descriptor->GetIdentifier()] =
characteristic->GetIdentifier();
}
void BluetoothLowEnergyEventRouter::GattDescriptorRemoved(
BluetoothGattCharacteristic* characteristic,
BluetoothGattDescriptor* descriptor) {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
VLOG(2) << "GATT descriptor removed: " << descriptor->GetIdentifier();
DCHECK(desc_id_to_chrc_id_.find(descriptor->GetIdentifier()) !=
desc_id_to_chrc_id_.end());
DCHECK(characteristic->GetIdentifier() ==
desc_id_to_chrc_id_[descriptor->GetIdentifier()]);
desc_id_to_chrc_id_.erase(descriptor->GetIdentifier());
}
void BluetoothLowEnergyEventRouter::GattCharacteristicValueChanged(
BluetoothGattService* service,
BluetoothGattCharacteristic* characteristic,
const std::vector<uint8>& value) {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
VLOG(2) << "GATT characteristic value changed: "
<< characteristic->GetIdentifier();
DCHECK(observed_gatt_services_.find(service->GetIdentifier()) !=
observed_gatt_services_.end());
DCHECK(service_id_to_device_address_.find(service->GetIdentifier()) !=
service_id_to_device_address_.end());
DCHECK(chrc_id_to_service_id_.find(characteristic->GetIdentifier()) !=
chrc_id_to_service_id_.end());
DCHECK(chrc_id_to_service_id_[characteristic->GetIdentifier()] ==
service->GetIdentifier());
// Send the event; manually construct the arguments, instead of using
// apibtle::OnCharacteristicValueChanged::Create, as it doesn't convert
// lists of enums correctly.
apibtle::Characteristic api_characteristic;
PopulateCharacteristic(characteristic, &api_characteristic);
scoped_ptr<base::ListValue> args(new base::ListValue());
args->Append(apibtle::CharacteristicToValue(&api_characteristic).release());
DispatchEventToExtensionsWithPermission(
apibtle::OnCharacteristicValueChanged::kEventName,
service->GetUUID(),
characteristic->GetIdentifier(),
args.Pass());
}
void BluetoothLowEnergyEventRouter::GattDescriptorValueChanged(
BluetoothGattCharacteristic* characteristic,
BluetoothGattDescriptor* descriptor,
const std::vector<uint8>& value) {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
VLOG(2) << "GATT descriptor value changed: " << descriptor->GetIdentifier();
DCHECK(desc_id_to_chrc_id_.find(descriptor->GetIdentifier()) !=
desc_id_to_chrc_id_.end());
DCHECK(characteristic->GetIdentifier() ==
desc_id_to_chrc_id_[descriptor->GetIdentifier()]);
// Send the event; manually construct the arguments, instead of using
// apibtle::OnDescriptorValueChanged::Create, as it doesn't convert
// lists of enums correctly.
apibtle::Descriptor api_descriptor;
PopulateDescriptor(descriptor, &api_descriptor);
scoped_ptr<base::ListValue> args(new base::ListValue());
args->Append(apibtle::DescriptorToValue(&api_descriptor).release());
DispatchEventToExtensionsWithPermission(
apibtle::OnDescriptorValueChanged::kEventName,
characteristic->GetService()->GetUUID(),
"" /* characteristic_id */,
args.Pass());
}
void BluetoothLowEnergyEventRouter::OnGetAdapter(
const base::Closure& callback,
scoped_refptr<device::BluetoothAdapter> adapter) {
adapter_ = adapter;
// Initialize instance ID mappings for all discovered GATT objects and add
// observers.
InitializeIdentifierMappings();
adapter_->AddObserver(this);
callback.Run();
}
void BluetoothLowEnergyEventRouter::InitializeIdentifierMappings() {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
DCHECK(service_id_to_device_address_.empty());
DCHECK(chrc_id_to_service_id_.empty());
DCHECK(observed_devices_.empty());
DCHECK(observed_gatt_services_.empty());
// Devices
BluetoothAdapter::DeviceList devices = adapter_->GetDevices();
for (BluetoothAdapter::DeviceList::iterator iter = devices.begin();
iter != devices.end();
++iter) {
BluetoothDevice* device = *iter;
device->AddObserver(this);
observed_devices_.insert(device->GetAddress());
// Services
std::vector<BluetoothGattService*> services = device->GetGattServices();
for (std::vector<BluetoothGattService*>::iterator siter = services.begin();
siter != services.end();
++siter) {
BluetoothGattService* service = *siter;
service->AddObserver(this);
const std::string& service_id = service->GetIdentifier();
observed_gatt_services_.insert(service_id);
service_id_to_device_address_[service_id] = device->GetAddress();
// Characteristics
const std::vector<BluetoothGattCharacteristic*>& characteristics =
service->GetCharacteristics();
for (std::vector<BluetoothGattCharacteristic*>::const_iterator citer =
characteristics.begin();
citer != characteristics.end();
++citer) {
BluetoothGattCharacteristic* characteristic = *citer;
const std::string& chrc_id = characteristic->GetIdentifier();
chrc_id_to_service_id_[chrc_id] = service_id;
// Descriptors
const std::vector<BluetoothGattDescriptor*>& descriptors =
characteristic->GetDescriptors();
for (std::vector<BluetoothGattDescriptor*>::const_iterator diter =
descriptors.begin();
diter != descriptors.end();
++diter) {
BluetoothGattDescriptor* descriptor = *diter;
const std::string& desc_id = descriptor->GetIdentifier();
desc_id_to_chrc_id_[desc_id] = chrc_id;
}
}
}
}
}
void BluetoothLowEnergyEventRouter::DispatchEventToExtensionsWithPermission(
const std::string& event_name,
const device::BluetoothUUID& uuid,
const std::string& characteristic_id,
scoped_ptr<base::ListValue> args) {
// Obtain the listeners of |event_name|. The list can contain multiple
// entries for the same extension, so we keep track of the extensions that we
// already sent the event to, since we want the send an event to an extension
// only once.
BluetoothPermissionRequest request(uuid.value());
std::set<std::string> handled_extensions;
const EventListenerMap::ListenerList listeners =
EventRouter::Get(browser_context_)->listeners().GetEventListenersByName(
event_name);
for (EventListenerMap::ListenerList::const_iterator iter = listeners.begin();
iter != listeners.end();
++iter) {
const std::string extension_id = (*iter)->extension_id();
if (handled_extensions.find(extension_id) != handled_extensions.end())
continue;
handled_extensions.insert(extension_id);
const Extension* extension =
ExtensionRegistry::Get(browser_context_)
->GetExtensionById(extension_id, ExtensionRegistry::EVERYTHING);
// For all API methods, the "low_energy" permission check is handled by
// BluetoothLowEnergyExtensionFunction but for events we have to do the
// check here.
if (!BluetoothManifestData::CheckRequest(extension, request) ||
!BluetoothManifestData::CheckLowEnergyPermitted(extension))
continue;
// If |event_name| is "onCharacteristicValueChanged", then send the
// event only if the extension has requested notifications from the
// related characteristic.
if (event_name == apibtle::OnCharacteristicValueChanged::kEventName &&
!characteristic_id.empty() &&
!FindNotifySession(extension_id, characteristic_id))
continue;
// Send the event.
scoped_ptr<base::ListValue> args_copy(args->DeepCopy());
scoped_ptr<Event> event(new Event(event_name, args_copy.Pass()));
EventRouter::Get(browser_context_)->DispatchEventToExtension(
extension_id, event.Pass());
}
}
BluetoothGattService* BluetoothLowEnergyEventRouter::FindServiceById(
const std::string& instance_id) const {
InstanceIdMap::const_iterator iter =
service_id_to_device_address_.find(instance_id);
if (iter == service_id_to_device_address_.end()) {
VLOG(1) << "GATT service identifier unknown: " << instance_id;
return NULL;
}
const std::string& address = iter->second;
BluetoothDevice* device = adapter_->GetDevice(address);
if (!device) {
VLOG(1) << "Bluetooth device not found: " << address;
return NULL;
}
BluetoothGattService* service = device->GetGattService(instance_id);
if (!service) {
VLOG(1) << "GATT service with ID \"" << instance_id
<< "\" not found on device \"" << address << "\"";
return NULL;
}
return service;
}
BluetoothGattCharacteristic*
BluetoothLowEnergyEventRouter::FindCharacteristicById(
const std::string& instance_id) const {
InstanceIdMap::const_iterator iter = chrc_id_to_service_id_.find(instance_id);
if (iter == chrc_id_to_service_id_.end()) {
VLOG(1) << "GATT characteristic identifier unknown: " << instance_id;
return NULL;
}
const std::string& service_id = iter->second;
BluetoothGattService* service = FindServiceById(service_id);
if (!service) {
VLOG(1) << "Failed to obtain service for characteristic: " << instance_id;
return NULL;
}
BluetoothGattCharacteristic* characteristic =
service->GetCharacteristic(instance_id);
if (!characteristic) {
VLOG(1) << "GATT characteristic with ID \"" << instance_id
<< "\" not found on service \"" << service_id << "\"";
return NULL;
}
return characteristic;
}
BluetoothGattDescriptor* BluetoothLowEnergyEventRouter::FindDescriptorById(
const std::string& instance_id) const {
InstanceIdMap::const_iterator iter = desc_id_to_chrc_id_.find(instance_id);
if (iter == desc_id_to_chrc_id_.end()) {
VLOG(1) << "GATT descriptor identifier unknown: " << instance_id;
return NULL;
}
const std::string& chrc_id = iter->second;
BluetoothGattCharacteristic* chrc = FindCharacteristicById(chrc_id);
if (!chrc) {
VLOG(1) << "Failed to obtain characteristic for descriptor: "
<< instance_id;
return NULL;
}
BluetoothGattDescriptor* descriptor = chrc->GetDescriptor(instance_id);
if (!descriptor) {
VLOG(1) << "GATT descriptor with ID \"" << instance_id
<< "\" not found on characteristic \"" << chrc_id << "\"";
return NULL;
}
return descriptor;
}
void BluetoothLowEnergyEventRouter::OnValueSuccess(
const base::Closure& callback,
const std::vector<uint8>& value) {
VLOG(2) << "Remote characteristic/descriptor value read successful.";
callback.Run();
}
void BluetoothLowEnergyEventRouter::OnCreateGattConnection(
bool persistent,
const std::string& extension_id,
const std::string& device_address,
const base::Closure& callback,
scoped_ptr<BluetoothGattConnection> connection) {
VLOG(2) << "GATT connection created.";
DCHECK(connection.get());
DCHECK(!FindConnection(extension_id, device_address));
DCHECK_EQ(device_address, connection->GetDeviceAddress());
const std::string connect_id = extension_id + device_address;
DCHECK_NE(0U, connecting_devices_.count(connect_id));
BluetoothLowEnergyConnection* conn = new BluetoothLowEnergyConnection(
persistent, extension_id, connection.Pass());
ConnectionResourceManager* manager =
GetConnectionResourceManager(browser_context_);
manager->Add(conn);
connecting_devices_.erase(connect_id);
callback.Run();
}
void BluetoothLowEnergyEventRouter::OnDisconnect(
const std::string& extension_id,
const std::string& device_address,
const base::Closure& callback) {
VLOG(2) << "GATT connection terminated.";
const std::string disconnect_id = extension_id + device_address;
DCHECK_NE(0U, disconnecting_devices_.count(disconnect_id));
if (!RemoveConnection(extension_id, device_address)) {
VLOG(1) << "The connection was removed before disconnect completed, id: "
<< extension_id << ", device: " << device_address;
}
disconnecting_devices_.erase(disconnect_id);
callback.Run();
}
void BluetoothLowEnergyEventRouter::OnError(
const ErrorCallback& error_callback) {
VLOG(2) << "Remote characteristic/descriptor value read/write failed.";
error_callback.Run(kStatusErrorFailed);
}
void BluetoothLowEnergyEventRouter::OnConnectError(
const std::string& extension_id,
const std::string& device_address,
const ErrorCallback& error_callback,
BluetoothDevice::ConnectErrorCode error_code) {
VLOG(2) << "Failed to create GATT connection: " << error_code;
const std::string connect_id = extension_id + device_address;
DCHECK_NE(0U, connecting_devices_.count(connect_id));
connecting_devices_.erase(connect_id);
error_callback.Run(kStatusErrorFailed);
}
void BluetoothLowEnergyEventRouter::OnStartNotifySession(
bool persistent,
const std::string& extension_id,
const std::string& characteristic_id,
const base::Closure& callback,
scoped_ptr<device::BluetoothGattNotifySession> session) {
VLOG(2) << "Value update session created for characteristic: "
<< characteristic_id;
DCHECK(session.get());
DCHECK(!FindNotifySession(extension_id, characteristic_id));
DCHECK_EQ(characteristic_id, session->GetCharacteristicIdentifier());
const std::string session_id = extension_id + characteristic_id;
DCHECK_NE(0U, pending_session_calls_.count(session_id));
BluetoothLowEnergyNotifySession* resource =
new BluetoothLowEnergyNotifySession(
persistent, extension_id, session.Pass());
NotifySessionResourceManager* manager =
GetNotifySessionResourceManager(browser_context_);
manager->Add(resource);
pending_session_calls_.erase(session_id);
callback.Run();
}
void BluetoothLowEnergyEventRouter::OnStartNotifySessionError(
const std::string& extension_id,
const std::string& characteristic_id,
const ErrorCallback& error_callback) {
VLOG(2) << "Failed to create value update session for characteristic: "
<< characteristic_id;
const std::string session_id = extension_id + characteristic_id;
DCHECK_NE(0U, pending_session_calls_.count(session_id));
pending_session_calls_.erase(session_id);
error_callback.Run(kStatusErrorFailed);
}
void BluetoothLowEnergyEventRouter::OnStopNotifySession(
const std::string& extension_id,
const std::string& characteristic_id,
const base::Closure& callback) {
VLOG(2) << "Value update session terminated.";
if (!RemoveNotifySession(extension_id, characteristic_id)) {
VLOG(1) << "The value update session was removed before Stop completed, "
<< "id: " << extension_id
<< ", characteristic: " << characteristic_id;
}
callback.Run();
}
BluetoothLowEnergyConnection* BluetoothLowEnergyEventRouter::FindConnection(
const std::string& extension_id,
const std::string& device_address) {
ConnectionResourceManager* manager =
GetConnectionResourceManager(browser_context_);
base::hash_set<int>* connection_ids = manager->GetResourceIds(extension_id);
if (!connection_ids)
return NULL;
for (base::hash_set<int>::const_iterator iter = connection_ids->begin();
iter != connection_ids->end();
++iter) {
extensions::BluetoothLowEnergyConnection* conn =
manager->Get(extension_id, *iter);
if (!conn)
continue;
if (conn->GetConnection()->GetDeviceAddress() == device_address)
return conn;
}
return NULL;
}
bool BluetoothLowEnergyEventRouter::RemoveConnection(
const std::string& extension_id,
const std::string& device_address) {
ConnectionResourceManager* manager =
GetConnectionResourceManager(browser_context_);
base::hash_set<int>* connection_ids = manager->GetResourceIds(extension_id);
if (!connection_ids)
return false;
for (base::hash_set<int>::const_iterator iter = connection_ids->begin();
iter != connection_ids->end();
++iter) {
extensions::BluetoothLowEnergyConnection* conn =
manager->Get(extension_id, *iter);
if (!conn || conn->GetConnection()->GetDeviceAddress() != device_address)
continue;
manager->Remove(extension_id, *iter);
return true;
}
return false;
}
BluetoothLowEnergyNotifySession*
BluetoothLowEnergyEventRouter::FindNotifySession(
const std::string& extension_id,
const std::string& characteristic_id) {
NotifySessionResourceManager* manager =
GetNotifySessionResourceManager(browser_context_);
base::hash_set<int>* ids = manager->GetResourceIds(extension_id);
if (!ids)
return NULL;
for (base::hash_set<int>::const_iterator iter = ids->begin();
iter != ids->end();
++iter) {
BluetoothLowEnergyNotifySession* session =
manager->Get(extension_id, *iter);
if (!session)
continue;
if (session->GetSession()->GetCharacteristicIdentifier() ==
characteristic_id)
return session;
}
return NULL;
}
bool BluetoothLowEnergyEventRouter::RemoveNotifySession(
const std::string& extension_id,
const std::string& characteristic_id) {
NotifySessionResourceManager* manager =
GetNotifySessionResourceManager(browser_context_);
base::hash_set<int>* ids = manager->GetResourceIds(extension_id);
if (!ids)
return false;
for (base::hash_set<int>::const_iterator iter = ids->begin();
iter != ids->end();
++iter) {
BluetoothLowEnergyNotifySession* session =
manager->Get(extension_id, *iter);
if (!session ||
session->GetSession()->GetCharacteristicIdentifier() !=
characteristic_id)
continue;
manager->Remove(extension_id, *iter);
return true;
}
return false;
}
} // namespace extensions