org.sputnikdev.bluetooth.manager.impl.BluetoothManagerImpl Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of bluetooth-manager Show documentation
Show all versions of bluetooth-manager Show documentation
A library/framework for managing bluetooth adapters, bluetooth devices, GATT services and characteristics
package org.sputnikdev.bluetooth.manager.impl;
/*-
* #%L
* org.sputnikdev:bluetooth-manager
* %%
* Copyright (C) 2017 Sputnik Dev
* %%
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
* #L%
*/
import com.google.common.collect.Lists;
import com.google.common.collect.Sets;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.sputnikdev.bluetooth.URL;
import org.sputnikdev.bluetooth.manager.*;
import org.sputnikdev.bluetooth.manager.transport.BluetoothObject;
import org.sputnikdev.bluetooth.manager.transport.BluetoothObjectFactory;
import java.util.*;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.ScheduledFuture;
import java.util.concurrent.TimeUnit;
/**
*
* @author Vlad Kolotov
*/
class BluetoothManagerImpl implements BluetoothManager {
private static final int REFRESH_RATE_SEC = 5;
private static final int DISCOVERY_RATE_SEC = 10;
private Logger logger = LoggerFactory.getLogger(BluetoothManagerImpl.class);
private final ScheduledExecutorService singleThreadScheduler = Executors.newSingleThreadScheduledExecutor();
private final ScheduledExecutorService scheduler = Executors.newScheduledThreadPool(5);
private final Set deviceDiscoveryListeners = new HashSet<>();
private final Set adapterDiscoveryListeners = new HashSet<>();
private final Map governors = new HashMap<>();
private ScheduledFuture discoveryFuture;
private final Map governorFutures = new HashMap<>();
private final Set discoveredDevices = new HashSet<>();
private final Set discoveredAdapters = new HashSet<>();
private final Map adapterToProtocolCache = Collections.synchronizedMap(new HashMap<>());
private boolean startDiscovering;
private int discoveryRate = DISCOVERY_RATE_SEC;
private int refreshRate = REFRESH_RATE_SEC;
private boolean rediscover = false;
@Override
public synchronized void start(boolean startDiscovering) {
if (discoveryFuture == null) {
this.startDiscovering = startDiscovering;
discoveryFuture = singleThreadScheduler.scheduleAtFixedRate(
new DiscoveryJob(), 0, discoveryRate, TimeUnit.SECONDS);
}
}
@Override
public synchronized void stop() {
if (discoveryFuture != null) {
discoveryFuture.cancel(true);
discoveryFuture = null;
}
}
@Override
public void addDeviceDiscoveryListener(DeviceDiscoveryListener deviceDiscoveryListener) {
deviceDiscoveryListeners.add(deviceDiscoveryListener);
}
@Override
public void removeDeviceDiscoveryListener(DeviceDiscoveryListener deviceDiscoveryListener) {
deviceDiscoveryListeners.remove(deviceDiscoveryListener);
}
@Override
public void addAdapterDiscoveryListener(AdapterDiscoveryListener adapterDiscoveryListener) {
adapterDiscoveryListeners.add(adapterDiscoveryListener);
}
@Override
public void removeAdapterDiscoveryListener(AdapterDiscoveryListener adapterDiscoveryListener) {
adapterDiscoveryListeners.remove(adapterDiscoveryListener);
}
@Override
public void disposeGovernor(URL url) {
synchronized (governors) {
if (governors.containsKey(url)) {
disposeGovernor(governors.get(url));
}
}
}
@Override
public void disposeDescendantGovernors(URL url) {
synchronized (governors) {
for (BluetoothObjectGovernor governor : new ArrayList<>(governors.values())) {
if (governor.getURL().isDescendant(url)) {
disposeGovernor(governor);
}
}
}
}
@Override
public DeviceGovernor getDeviceGovernor(URL url) {
return (DeviceGovernorImpl) getGovernor(url.getDeviceURL());
}
@Override
public AdapterGovernor getAdapterGovernor(URL url) {
return (AdapterGovernorImpl) getGovernor(url.getAdapterURL());
}
@Override
public CharacteristicGovernor getCharacteristicGovernor(URL url) {
return (CharacteristicGovernorImpl) getGovernor(url.getCharacteristicURL());
}
@Override
public void dispose() {
logger.info("Disposing Bluetooth manager");
singleThreadScheduler.shutdown();
scheduler.shutdown();
if (discoveryFuture != null) {
discoveryFuture.cancel(true);
}
for (ScheduledFuture future : Sets.newHashSet(governorFutures.values())) {
future.cancel(true);
}
deviceDiscoveryListeners.clear();
adapterDiscoveryListeners.clear();
synchronized (governors) {
for (BluetoothObjectGovernor governor : governors.values()) {
try {
governor.reset();
} catch (Exception ex) {
logger.error("Could not dispose governor: " + governor.getURL());
}
}
governors.clear();
}
logger.info("Bluetooth service has been disposed");
}
@Override
public Set getDiscoveredDevices() {
synchronized (discoveredDevices) {
return Collections.unmodifiableSet(discoveredDevices);
}
}
@Override
public Set getDiscoveredAdapters() {
synchronized (discoveredAdapters) {
return Collections.unmodifiableSet(discoveredAdapters);
}
}
@Override
public BluetoothGovernor getGovernor(URL url) {
synchronized (governors) {
if (!governors.containsKey(url)) {
BluetoothObjectGovernor governor = createGovernor(url);
governors.put(url, governor);
governorFutures.put(url,
scheduler.scheduleAtFixedRate((Runnable) () -> update(governor),
5, refreshRate, TimeUnit.SECONDS));
update(governor);
return governor;
}
return governors.get(url);
}
}
@Override
public void setDiscoveryRate(int seconds) {
this.discoveryRate = seconds;
}
@Override
public void setRediscover(boolean rediscover) {
this.rediscover = rediscover;
}
@Override
public void setRefreshRate(int refreshRate) {
this.refreshRate = refreshRate;
}
List getGovernors(List extends BluetoothObject> objects) {
List result = new ArrayList<>(objects.size());
synchronized (governors) {
for (BluetoothObject object : objects) {
result.add(getGovernor(object.getURL()));
}
}
return Collections.unmodifiableList(result);
}
void updateDescendants(URL parent) {
synchronized (governors) {
for (BluetoothObjectGovernor governor : governors.values()) {
if (governor.getURL().isDescendant(parent)) {
update(governor);
}
}
}
}
void resetDescendants(URL parent) {
synchronized (governors) {
for (BluetoothObjectGovernor governor : governors.values()) {
if (governor.getURL().isDescendant(parent)) {
reset(governor);
}
}
}
}
T getBluetoothObject(URL url) {
BluetoothObjectFactory factory = findFactory(url);
BluetoothObject bluetoothObject = null;
if (factory != null) {
if (url.isAdapter()) {
bluetoothObject = factory.getAdapter(url);
} else if (url.isDevice()) {
bluetoothObject = factory.getDevice(url);
} else if (url.isCharacteristic()) {
bluetoothObject = factory.getCharacteristic(url);
}
if (bluetoothObject != null && !adapterToProtocolCache.containsKey(url.getAdapterAddress())) {
adapterToProtocolCache.put(url.getAdapterAddress(), factory.getProtocolName());
}
}
return (T) bluetoothObject;
}
BluetoothObjectGovernor createGovernor(URL url) {
if (url.isAdapter()) {
AdapterGovernorImpl adapterGovernor = new AdapterGovernorImpl(this, url);
adapterGovernor.setDiscoveringControl(startDiscovering);
return adapterGovernor;
} else if (url.isDevice()) {
return new DeviceGovernorImpl(this, url);
} else if (url.isCharacteristic()) {
return new CharacteristicGovernorImpl(this, url);
}
throw new IllegalStateException("Unknown url");
}
private void disposeGovernor(BluetoothObjectGovernor governor) {
URL url = governor.getURL();
reset(governor);
synchronized (governorFutures) {
if (governorFutures.containsKey(url)) {
governorFutures.get(url).cancel(true);
governorFutures.remove(url);
}
}
governors.remove(url);
}
private BluetoothObjectFactory findFactory(URL url) {
String protocol = url.getProtocol();
String adapterAddress = url.getAdapterAddress();
if (url.getProtocol() != null) {
return BluetoothObjectFactoryProvider.getFactory(protocol);
} else if (adapterToProtocolCache.containsKey(adapterAddress)) {
return BluetoothObjectFactoryProvider.getFactory(adapterToProtocolCache.get(adapterAddress));
} else {
for (DiscoveredAdapter adapter : BluetoothObjectFactoryProvider.getAllDiscoveredAdapters()) {
if (adapter.getURL().getAdapterAddress().equals(adapterAddress)) {
adapterToProtocolCache.put(url.getAdapterAddress(), adapter.getURL().getProtocol());
return BluetoothObjectFactoryProvider.getFactory(adapter.getURL().getProtocol());
}
}
}
return null;
}
private void notifyDeviceDiscovered(DiscoveredDevice device) {
if (this.discoveredDevices.contains(device) && !this.rediscover) {
return;
}
for (DeviceDiscoveryListener deviceDiscoveryListener : Lists.newArrayList(deviceDiscoveryListeners)) {
try {
deviceDiscoveryListener.discovered(device);
} catch (Exception ex) {
logger.error("Discovery listener error (device)", ex);
}
}
}
private void notifyAdapterDiscovered(DiscoveredAdapter adapter) {
if (this.discoveredAdapters.contains(adapter) && !this.rediscover) {
return;
}
for (AdapterDiscoveryListener adapterDiscoveryListener : Lists.newArrayList(adapterDiscoveryListeners)) {
try {
adapterDiscoveryListener.discovered(adapter);
} catch (Exception ex) {
logger.error("Discovery listener error (adapter)", ex);
}
}
}
private void handleDeviceLost(URL url) {
logger.info("Device has been lost: " + url);
for (DeviceDiscoveryListener deviceDiscoveryListener : Lists.newArrayList(deviceDiscoveryListeners)) {
try {
deviceDiscoveryListener.deviceLost(url);
} catch (Throwable ex) {
logger.error("Device listener error", ex);
}
}
}
private void handleAdapterLost(URL url) {
logger.info("Adapter has been lost: " + url);
for (AdapterDiscoveryListener adapterDiscoveryListener : Lists.newArrayList(adapterDiscoveryListeners)) {
try {
adapterDiscoveryListener.adapterLost(url);
} catch (Throwable ex) {
logger.error("Adapter listener error", ex);
}
}
try {
((AdapterGovernorImpl) getAdapterGovernor(url)).reset();
} catch (Throwable ex) {
logger.warn("Could not reset adapter governor", ex);
}
}
void reset(BluetoothObjectGovernor governor) {
try {
governor.reset();
} catch (Exception ex) {
logger.error("Could not reset governor: " + governor, ex);
}
}
private void update(BluetoothObjectGovernor governor) {
try {
governor.update();
} catch (Exception ex) {
logger.error("Could not update governor: " + governor, ex);
}
}
private class DiscoveryJob implements Runnable {
@Override
public void run() {
discoverAdapters();
discoverDevices();
}
private void discoverDevices() {
try {
synchronized (discoveredDevices) {
List list = BluetoothObjectFactoryProvider.getAllDiscoveredDevices();
if (list == null) {
return;
}
Set newDiscovery = new HashSet<>();
for (DiscoveredDevice discoveredDevice : list) {
short rssi = discoveredDevice.getRSSI();
if (rssi == 0) {
continue;
}
notifyDeviceDiscovered(discoveredDevice);
newDiscovery.add(discoveredDevice);
}
for (DiscoveredDevice lost : Sets.difference(discoveredDevices, newDiscovery)) {
handleDeviceLost(lost.getURL());
}
discoveredDevices.clear();
discoveredDevices.addAll(newDiscovery);
}
} catch (Exception ex) {
logger.error("Device discovery job error", ex);
}
}
private void discoverAdapters() {
try {
synchronized (discoveredAdapters) {
Set newDiscovery = new HashSet<>();
for (DiscoveredAdapter discoveredAdapter :
BluetoothObjectFactoryProvider.getAllDiscoveredAdapters()) {
notifyAdapterDiscovered(discoveredAdapter);
newDiscovery.add(discoveredAdapter);
if (startDiscovering) {
// create (if not created before) adapter governor which will trigger its discovering status
// (by default when it is created "discovering" flag is set to true)
getAdapterGovernor(discoveredAdapter.getURL());
}
}
for (DiscoveredAdapter lost : Sets.difference(discoveredAdapters, newDiscovery)) {
handleAdapterLost(lost.getURL());
}
discoveredAdapters.clear();
discoveredAdapters.addAll(newDiscovery);
}
} catch (Exception ex) {
logger.error("Adapter discovery job error", ex);
}
}
}
}