## Introduction
Aarogya Setu is a mobile application developed by the Government of India to connect essential health services with the people of India in our combined fight against COVID-19.Do visit this link for more details.
AarogyaSetu started with an idea of automatic contact tracing. Proximity logging within an app addresses a key limitation of manual contact tracing: it is dependent on a person’s memory and is therefore limited to the contacts that a person is acquainted with and remembers having met.
For contact tracing, we had two options to go with: Bluetooth or GPS.
Bluetooth was chosen because it is able to classify close contacts with a significantly lower false-positive rate than GPS.
Given that GPS accuracy decreases in indoor environments, entire shopping malls or skyscrapers would be within the margin of error of a single GPS point.
So, with an idea that is more scalable, accurate and less resource-intensive, we created an app which:-
Can help us in tracing the nearby contacts through Bluetooth.
-
Consumes low battery power.
-
Performs continuous.
-
Is compatible with both Android and iOS.
We have used Bluetooth Low Energy(BLE) technology that has built-in platform support in the central role and provides APIs that apps can use to discover devices, query for services, and transmit information since Android 4.3(API level 18).
In contrast to Classic Bluetooth, BLE is designed to provide significantly lower power consumption.
"android.permission.BLUETOOTH"
"android.permission.BLUETOOTH_ADMIN"
"android.permission.ACCESS_COARSE_LOCATION"
Note: Some of the devices (above 6.0) requires location permission to use Bluetooth. You can read more here.
BLE works with advertisement and scanning. The device in the central role scans, looking for advertisement, and the device in the peripheral role makes the advertisement.
We have BluetoothLeAdvertiser class for starting and stopping advertising.
An advertiser(Added in API level 21) can broadcast up to 31 bytes of advertisement data represented by AdvertiseData.
The AdvertiseSettings provide a way to adjust advertising preferences for each Bluetooth LE advertisement instance.
-
setTxPowerLevel: We have set the lowest transmission (TX) power level to restrict the visibility range of advertising packets.
-
setAdvertiseMode: We are setting the advertising power and latency based on the number of unique scans/interval and changing it accordingly to provide lower power consumption.
It can only be one of AdvertiseSettings#ADVERTISE_MODE_LOW_POWER, AdvertiseSettings#ADVERTISE_MODE_BALANCED, or AdvertiseSettings#ADVERTISE_MODE_LOW_LATENCY.
- setConnectable: We have used it for making our advertisement connectable(GATT connection for iOS, will explain it later).This flag takes 1 byte+2 bytes header.
This is the data to be advertised as well as the scan response data for active scans.
-
setIncludeDeviceName: We are setting an 8 bytes(+2 bytes header) unique id as Bluetooth name for distinguishing a packet during the scan. So, we have made this flag true.
-
addServiceUuid: Setting a 16 bytes(+2 bytes header) UUID that is used for identifying our packets when they are advertised.
So now, we know how all 31 bytes are getting consumed. For more clarification on byte structure, please check this class.
We need to connect to the GATT server on the device to make devices talk to each other.
After opening a GATT server for connection, we have added one service with 2 characteristics (UniqueID & Pinger characteristics).
As per the iOS limitation, iOS is not able to scan if it goes in background.
So, we found that making advertisement connectable through GATT can actually minimize this limitation by pinging the devices for connection and getting the unique id from there so that the app remains active and the connection breaks only if the device goes too far.
Furthermore, you don't need to worry about multiple connections at the same time.GATT itself handles it for a good number of connections.
We have BluetoothLeScanner(Added in API level 21) to perform scan related operations for Bluetooth LE devices. An application can scan for a particular type of Bluetooth LE devices using ScanFilter.
A ScanFilter allows clients to restrict scan results to only those that are of interest to them.
- setServiceUUID: We are filtering using the Service UUID which we sent in our advertisement packet.
Bluetooth LE scan settings are passed to BluetoothLeScanner#startScan to define the parameters for the scan.
- setScanMode: We are setting the scan mode based on the number of unique scans/interval and changing it accordingly to provide lower power consumption.
The scan mode can be one of ScanSettings#SCAN_MODE_LOW_POWER, ScanSettings#SCAN_MODE_BALANCED or ScanSettings#SCAN_MODE_LOW_LATENCY.
-
setMatchMode(Added in API level 23): We have set it to MATCH_MODE_STICKY (For sticky mode, a higher threshold of signal strength and sightings is required before reporting by h/w)
-
setPhy(Added in API level 26): Setting the Physical Layer to use during this scan. This is used only if ScanSettings.Builder#setLegacy is set to false. To improve scan performance, we have used PHY_LE_1M channel for scanning. Read more about it here.
Scan results are reported using these callbacks.
In onScanResult,
We get out scanResult from which we parse the device name, RSSI and txPower(Added in API 26).
- Change the scan and advertisement mode for the device accordingly.
- Store the scanResult in our local storage i.e database.
Also, we are checking the duplicacy of a scan record within a few seconds before we add it to our local storage.
Android 7.0 introduced a BLE scan timeout, where any scan running for 30 minutes or more is effectively stopped automatically and can only resume “opportunistically”. So we are starting the scan again after an interval.
We have implemented adaptive scanning through which we are changing the scan and advertisement mode of the scanning device based on the number of unique scans/interval.
As scan and advertisement modes are directly related to power consumption, this has helped us in adjusting the consumption based on the number of close contacts.
So, you can now worry even less for your battery power usage. Cheers !!
To make our app scan continuously, we are doing all this in a Service and to cater to the background service restrictions in Android API 26 or higher, we are using startForegroundService which is similar to startService(android.content.Intent)(android.content.Intent), but with an implicit promise that the Service will call startForeground(int, android.app.Notification) to make the service run in the foreground.
We use this method if killing your service would be disruptive to the user. Read more about the foreground mechanism here.
There are vulnerabilities in Bluetooth technology that have to be patched at the operating system-level, and we, therefore, urge users to ensure that their operating systems are regularly patched.
We have tried to make the most out of the Bluetooth and we are constantly improving.
Check out my talk on Understanding the BLE in integration in AarogyaSetu:
This is the link for slides.
Also, You can check the source code.
Thank You!!
I am fortunate enough to be a part of the magnificently electrifying @SetuAarogya team. Thanks to the Government Of India for giving me this opportunity to work on a project that has an impact on human lives. Stay Safe!!