How to use bluetoothctl like hcitool lescan to report repeated proximity beacons

前端 未结 2 1186
暖寄归人
暖寄归人 2021-02-02 13:50

I can use hcitool lescan with the --duplicates flag to capture periodic LE Advertising Reports (proximity beacons) from two nearby BLE devices:

$ sudo hcitool le         


        
2条回答
  •  太阳男子
    2021-02-02 13:56

    Thanks Florian for your suggestion.

    Since my original post, I have migrated to an Ubuntu 16.04.4 embedded Linux distribution which includes bluez 5.42. Unfortunately, bluetoothctl with this version does not recognize 'menu scan' or 'clear':

    [bluetooth]# menu scan
    Invalid command
    
    [bluetooth]# clear
    Invalid command
    

    However, encouraged by your mention of 'default scan filter active that blocks most advertisements', I experimented with the commands available in my version of bluetoothctl as seen in the --help output, and got something working:

    root@iot:~# bluetoothctl
    [NEW] Controller 00:1A:7D:DA:71:13 iot #1 [default]
    [NEW] Controller 70:2C:1F:31:F4:AF iot 
    
    [bluetooth]# set-scan-filter-clear
    SetDiscoveryFilter success
    
    [bluetooth]# set-scan-filter-transport le
    SetDiscoveryFilter success
    
    [bluetooth]# scan on
    Discovery started
    
    [CHG] Controller 00:1A:7D:DA:71:13 Discovering: yes
    [NEW] Device 0F:64:64:EE:E7:C4 0F-64-64-EE-E7-C4
    [NEW] Device 0D:6F:45:77:87:F3 0D-6F-45-77-87-F3
    [NEW] Device 40:CB:C0:F2:96:27 40-CB-C0-F2-96-27
    [CHG] Device 0D:6F:45:77:87:F3 RSSI: -71
    [CHG] Device FC:F1:36:73:77:B3 RSSI: -57
    [CHG] Device 0F:64:64:EE:E7:C4 RSSI: -49
    

    It takes a bit of typing in bluetoothctl to get it configured the way I want, and that typing is obscured by bluetoothctl logging activity rapidly in our beacon rich environment. So I put together a bash script that uses a heredoc and expect to feed the commands to bluetoothctl and sed/grep/perl massages the output:

    $ cat beacon-scan.sh 
    #!/bin/bash
    
    # beacon-scan.sh
    # Displays beacons including duplicates in real time.
    # Uses expect to automate interaction with bluetoothctl.
    # Uses sed to remove bluetoothctl colorization escape characters.
    # Uses grep to filter out beacon manufacturer data logging.
    # Uses perl to prefix each beacon with a timestamp.
    
    if [ "$(id -u)" != "0" ]; then
        echo "ERROR: must run as root"
        exit 1
    fi
    
    (cat <<'END' | /usr/bin/expect
    
        set prompt "#"
        set timeout -1
    
        spawn bluetoothctl
    
        expect -re $prompt
        send "scan off\r"
    
        expect -re $prompt
        send "remove *\r"
    
        expect -re $prompt
        send "set-scan-filter-clear\r"
    
        expect -re $prompt
        send "set-scan-filter-transport le\r"
    
        expect -re $prompt
        send "scan on\r"
    
        trap {
            expect -re $prompt
            send "scan off\r"
    
            expect -re $prompt
            send "remove *\r"
    
            expect -re $prompt
            send "quit\r"
        } SIGINT
    
        expect eof
    
    END
    ) | sed --unbuffered --quiet --expression 's/^.*Device //p' \
      | grep --line-buffered -v ManufacturerData \
      | perl -nle 'print scalar(localtime), " ", $_'
    

    It works:

    $ sudo ./beacon-scan.sh 
    Wed Aug 22 19:34:07 2018 0F:64:64:EE:E7:C4 RSSI: -59
    Wed Aug 22 19:34:07 2018 03:46:00:1D:E9:91 03-46-00-1D-E9-91
    Wed Aug 22 19:34:07 2018 4E:20:6B:C7:68:D0 RSSI: -55
    Wed Aug 22 19:34:07 2018 76:F1:1A:B9:ED:28 RSSI: -57
    Wed Aug 22 19:34:07 2018 32:5D:8C:6A:72:C2 32-5D-8C-6A-72-C2
    ^C
    

    Beacons that repeat are now reported by bluetoothctl, similar to when hcitool lescan is run with the duplicates flag.

    I will say that bluetoothctl would be easier to use if it could be configured from the command line without needing to configure it interactively or having to resort to more complex scripting.

    Thanks Florian for your help.

提交回复
热议问题