I am using Android 4.4.2 on a Nexus 7. I have a bluetooth low energy peripheral whose services change when it is rebooted. The android app calls BluetoothGatt.discoverServices(). However Android only queries the peripheral once to discover services, subsequent calls to discoverServices() result in the cached data from the first call, even between disconnections. If I disable/enable the Android bt adapter then discoverServices() refreshes the cache by querying the peripheral. Is there a programmatic way to force Android to refresh its' ble services cache without disabling/enabling the adapter?
5 Answers
I just had the same problem. If you see the source code of BluetoothGatt.java you can see that there is a method called refresh()
/**
* Clears the internal cache and forces a refresh of the services from the
* remote device.
* @hide
*/
public boolean refresh() {
if (DBG) Log.d(TAG, "refresh() - device: " + mDevice.getAddress());
if (mService == null || mClientIf == 0) return false;
try {
mService.refreshDevice(mClientIf, mDevice.getAddress());
} catch (RemoteException e) {
Log.e(TAG,"",e);
return false;
}
return true;
}
This method does actually clear the cache from a bluetooth device. But the problem is that we don't have access to it. But in java we have reflection, so we can access this method. Here is my code to connect a bluetooth device refreshing the cache.
private boolean refreshDeviceCache(BluetoothGatt gatt){
try {
BluetoothGatt localBluetoothGatt = gatt;
Method localMethod = localBluetoothGatt.getClass().getMethod("refresh", new Class[0]);
if (localMethod != null) {
boolean bool = ((Boolean) localMethod.invoke(localBluetoothGatt, new Object[0])).booleanValue();
return bool;
}
}
catch (Exception localException) {
Log.e(TAG, "An exception occurred while refreshing device");
}
return false;
}
public boolean connect(final String address) {
if (mBluetoothAdapter == null || address == null) {
Log.w(TAG,"BluetoothAdapter not initialized or unspecified address.");
return false;
}
// Previously connected device. Try to reconnect.
if (mBluetoothGatt != null) {
Log.d(TAG,"Trying to use an existing mBluetoothGatt for connection.");
if (mBluetoothGatt.connect()) {
return true;
} else {
return false;
}
}
final BluetoothDevice device = mBluetoothAdapter
.getRemoteDevice(address);
if (device == null) {
Log.w(TAG, "Device not found. Unable to connect.");
return false;
}
// We want to directly connect to the device, so we are setting the
// autoConnect
// parameter to false.
mBluetoothGatt = device.connectGatt(MyApp.getContext(), false, mGattCallback));
refreshDeviceCache(mBluetoothGatt);
Log.d(TAG, "Trying to create a new connection.");
return true;
}
Indeed Miguel's answer works. To use refreshDeviceCache, I'm successful with this calling order:
// Attempt GATT connection
public void connectGatt(MyBleDevice found) {
BluetoothDevice device = found.getDevice();
gatt = device.connectGatt(mActivity, false, mGattCallback);
refreshDeviceCache(gatt);
}
This works for OS 4.3 through 5.0 tested with Android and iPhone Peripherals.
In Some Devices, Even you disconnect the socket the connection wont end because of the cache. You need to disconnect the remote device by using the BluetoothGatt Class. As Below
BluetoothGatt mBluetoothGatt = device.connectGatt(appContext, false, new BluetoothGattCallback() {
};);
mBluetoothGatt.disconnect();
Note : This logic worked for me in china based devices
Here is the Kotlin version with RxAndroidBle for refresh:
class CustomRefresh: RxBleRadioOperationCustom<Boolean> {
@Throws(Throwable::class)
override fun asObservable(bluetoothGatt: BluetoothGatt,
rxBleGattCallback: RxBleGattCallback,
scheduler: Scheduler): Observable<Boolean> {
return Observable.fromCallable<Boolean> { refreshDeviceCache(bluetoothGatt) }
.delay(500, TimeUnit.MILLISECONDS, Schedulers.computation())
.subscribeOn(scheduler)
}
private fun refreshDeviceCache(gatt: BluetoothGatt): Boolean {
var isRefreshed = false
try {
val localMethod = gatt.javaClass.getMethod("refresh")
if (localMethod != null) {
isRefreshed = (localMethod.invoke(gatt) as Boolean)
Timber.i("Gatt cache refresh successful: [%b]", isRefreshed)
}
} catch (localException: Exception) {
Timber.e("An exception occured while refreshing device" + localException.toString())
}
return isRefreshed
}
}
Actual call:
Observable.just(rxBleConnection)
.flatMap { rxBleConnection -> rxBleConnection.queue(CustomRefresh()) }
.observeOn(Schedulers.io())
.doOnComplete{
switchToDFUmode()
}
.subscribe({ isSuccess ->
// check
},
{ throwable ->
Timber.d(throwable)
}).also {
refreshDisposable.add(it)
}