pythonandroidlinuxbluetooth

How can I automate pairing RPi and Android with bluetooth Batch script


I am working on a project that connects an Android device with a Raspberry Pi. The RPi needs to be treated like a deployable device that the user never needs to touch. For this reason, I am trying to write a startup batch script on the RPi that will allow the user to pair their Android with the PI.

My idea is that when you startup, this script will run, the user on their phone will try and connect to the RPi, and the RPi will automatically accept this connection.

Here is what I have so far

#!/bin/bash
bluetoothctl -- discoverable on
bluetoothctl -- pairable on
bluetoothctl -- agent on
bluetoothctl -- default-agent

The issue is, when I do it this way I don't get into the [bluetoothctl] prompt that I need to communicate with the Android.

When I run these commands (Without batch script) and try and pair with my Android I get

Request confirmation
[agent] Confirm passkey 861797 (yes/no): yes

And from here I simply need to input yes to instantiate the connection. The issue I'm seeing is 1: I don't know how to stay in the [bluetoothctl] prompt within the command line to communicate with the device and 2: I don't know how to send "Yes" to the prompt.

Again, the important thing for me is that the user never needs to do anything more to the RPi than start it up for deployment purposes. Is there a fix for my problem or perhaps a better way to do it all together?

For those interested, the bluetooth startup connection is in place so that I can send network information to the RPi and it can automatically connect itself to the network so that the main application communication will take place that way.

Here is the desired result of the script which I was able to do manually.

enter image description here


Solution

  • Using bluetoothctl in that manner can be problematic as it is not designed to be interactive in that way. As you have put Python as one of the tags, the intended way of accessing this functionality from Python (and other languages) is through the D-Bus API.

    These are documented at: https://git.kernel.org/pub/scm/bluetooth/bluez.git/tree/doc

    And there are examples at: https://git.kernel.org/pub/scm/bluetooth/bluez.git/tree/test

    The confirmation is the RequestConfirmation in the agent API. You can also set discoverable and pairable with the adapter API. Using the API will also allow you to stop discoverable from timing out.

    Once the phone has connected, you typically want to mark it as trusted so that it doesn't need to pair again. This is done with the device API.

    Below is an example of setting these properties on the adapter with Python. I have left all of the Agent functions in although it is only the RequestConfirmation that is used. I have set it to always agree to whatever code it is sent which is what you asked for in your question.

    This example Python script would replace your batch script

    import dbus
    import dbus.service
    import dbus.mainloop.glib
    from gi.repository import GLib
    
    BUS_NAME = 'org.bluez'
    ADAPTER_IFACE = 'org.bluez.Adapter1'
    ADAPTER_ROOT = '/org/bluez/hci'
    AGENT_IFACE = 'org.bluez.Agent1'
    AGNT_MNGR_IFACE = 'org.bluez.AgentManager1'
    AGENT_PATH = '/my/app/agent'
    AGNT_MNGR_PATH = '/org/bluez'
    CAPABILITY = 'KeyboardDisplay'
    DEVICE_IFACE = 'org.bluez.Device1'
    dbus.mainloop.glib.DBusGMainLoop(set_as_default=True)
    bus = dbus.SystemBus()
    
    def set_trusted(path):
        props = dbus.Interface(bus.get_object(BUS_NAME, path), dbus.PROPERTIES_IFACE)
        props.Set(DEVICE_IFACE, "Trusted", True)
    
    class Agent(dbus.service.Object):
    
        @dbus.service.method(AGENT_IFACE,
                             in_signature="", out_signature="")
        def Release(self):
            print("Release")
    
        @dbus.service.method(AGENT_IFACE,
                             in_signature='o', out_signature='s')
        def RequestPinCode(self, device):
            print(f'RequestPinCode {device}')
            return '0000'
    
        @dbus.service.method(AGENT_IFACE,
                             in_signature="ou", out_signature="")
        def RequestConfirmation(self, device, passkey):
            print("RequestConfirmation (%s, %06d)" % (device, passkey))
            set_trusted(device)
            return
    
        @dbus.service.method(AGENT_IFACE,
                             in_signature="o", out_signature="")
        def RequestAuthorization(self, device):
            print("RequestAuthorization (%s)" % (device))
            auth = input("Authorize? (yes/no): ")
            if (auth == "yes"):
                return
            raise Rejected("Pairing rejected")
    
        @dbus.service.method(AGENT_IFACE,
                             in_signature="o", out_signature="u")
        def RequestPasskey(self, device):
            print("RequestPasskey (%s)" % (device))
            set_trusted(device)
            passkey = input("Enter passkey: ")
            return dbus.UInt32(passkey)
    
        @dbus.service.method(AGENT_IFACE,
                             in_signature="ouq", out_signature="")
        def DisplayPasskey(self, device, passkey, entered):
            print("DisplayPasskey (%s, %06u entered %u)" %
                  (device, passkey, entered))
    
        @dbus.service.method(AGENT_IFACE,
                             in_signature="os", out_signature="")
        def DisplayPinCode(self, device, pincode):
            print("DisplayPinCode (%s, %s)" % (device, pincode))
    
    
    class Adapter:
        def __init__(self, idx=0):
            bus = dbus.SystemBus()
            self.path = f'{ADAPTER_ROOT}{idx}'
            self.adapter_object = bus.get_object(BUS_NAME, self.path)
            self.adapter_props = dbus.Interface(self.adapter_object,
                                                dbus.PROPERTIES_IFACE)
            self.adapter_props.Set(ADAPTER_IFACE,
                                   'DiscoverableTimeout', dbus.UInt32(0))
            self.adapter_props.Set(ADAPTER_IFACE,
                                   'Discoverable', True)
            self.adapter_props.Set(ADAPTER_IFACE,
                                   'PairableTimeout', dbus.UInt32(0))
            self.adapter_props.Set(ADAPTER_IFACE,
                                   'Pairable', True)
    
    
    if __name__ == '__main__':
        agent = Agent(bus, AGENT_PATH)
        agnt_mngr = dbus.Interface(bus.get_object(BUS_NAME, AGNT_MNGR_PATH),
                                   AGNT_MNGR_IFACE)
        agnt_mngr.RegisterAgent(AGENT_PATH, CAPABILITY)
        agnt_mngr.RequestDefaultAgent(AGENT_PATH)
    
        adapter = Adapter()
        mainloop = GLib.MainLoop()
        try:
            mainloop.run()
        except KeyboardInterrupt:
            agnt_mngr.UnregisterAgent(AGENT_PATH)
            mainloop.quit()