I'm trying to communicate with a BLE data logger/sensor using rxBleAndroid running on both an Android phone as well as the Raspberry Pi using Android Things.
I'm currently having an issue, however, where up to about 5 of the first notifications are never received by my app.
I have verified that the BLE device is actually successfully sending all the expected notifications. I have done that through the nRF Connect app and everything works as expected through there.
When I do it through the nRF Connect app, these are the steps I take:
- Write to password characteristic to unlock device
- Write to mode characteristic to put device in correct mode
- Subscribe to notifications (and notifications immediately start working)
When doing it through RxAndroidBle, I suspect it may be that the .subscribe() is not being setup fast enough.
Is there maybe some way to do setupNotification(), and then write the characteristics to tell the device to start sending notifications?
Here is my current code:
rxBleClient = RxBleClient.create(this);
RxBleDevice device = rxBleClient.getBleDevice(mac_address);
device.establishConnection(false)
.flatMap(rxBleConnection -> rxBleConnection.writeCharacteristic(pword_uuid, pword)
.flatMap(ignored1 -> rxBleConnection.writeCharacteristic(mode_uuid, mode))
.flatMap(ignored2 -> rxBleConnection.setupNotification(log_uuid))
)
.flatMap(notificationObservable -> notificationObservable)
.subscribe(
bytes -> {
System.out.println(">>> data from device " + bytesToHex(bytes));
},
throwable -> {
System.out.println("error");
System.out.println(throwable);
});
Most actions that one can do over BLE are asynchronous and take some time to finish. Setting up the notifications is no exception—it is a two-step procedure:
If your peripheral is first set to send notifications before the notifications are ready to be received by the central then some of the data may get lost during the notifications setup procedure.
Of course (this is how usually similar scenarios are handled)—there are multiple possible implementations. One of them could look like this:
Edit:
As it was mentioned in the comments below—the peripheral does not allow for any BLE interactions before setting the mode and writing the password. As I have written above setting the notifications is a two step-step procedure with a local step and remote (executed on peripheral) one which is executed before the mode/password in the above code snippet. It is possible to separate those two steps by using the
NotificationSetupMode.COMPAT
mode and writing theClient Characteristic Configuration Descriptor
manually later:rxBleConnection.discoverServices()
call may be omitted if we would know the Log Characteristic ServiceUUID
and userxBleConnection.writeDescriptor(UUID serviceUuid, UUID characteristicUuid, UUID descriptorUuid
function.Edit 2:
Since version
1.8.0
there is a newNotificationSetupMode.QUICK_SETUP
which first turns on internal notifications and then writes the CCC descriptor value.Pros:
Observable<byte[]>
is emitted before the descriptor is written allowing for notification observation right from the beginning (if the beginning is writing the descriptor)Cons: