Add rough notes on bluetooth console setup ; insecure ; you don't want this without thinking things through deeply
This commit is contained in:
parent
62e32bcdbc
commit
066bafcbcf
49
docs/bt_console/BtAutoPair.py
Normal file
49
docs/bt_console/BtAutoPair.py
Normal file
|
@ -0,0 +1,49 @@
|
||||||
|
#!/usr/bin/python3
|
||||||
|
# encoding=utf8
|
||||||
|
|
||||||
|
import sys
|
||||||
|
import time
|
||||||
|
import pexpect
|
||||||
|
import subprocess
|
||||||
|
|
||||||
|
class BtAutoPair:
|
||||||
|
"""Class to auto pair and trust with bluetooth."""
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
p = subprocess.Popen("/usr/local/bin/bt-auto-agent", shell = False)
|
||||||
|
out = subprocess.check_output("/usr/sbin/rfkill unblock bluetooth", shell = True)
|
||||||
|
self.child = pexpect.spawn("bluetoothctl", echo = False)
|
||||||
|
|
||||||
|
def get_output(self,command, pause = 0):
|
||||||
|
"""Run a command in bluetoothctl prompt, return output as a list of lines."""
|
||||||
|
self.child.send(command + "\n")
|
||||||
|
time.sleep(pause)
|
||||||
|
start_failed = self.child.expect(["bluetooth", pexpect.EOF])
|
||||||
|
|
||||||
|
if start_failed:
|
||||||
|
raise BluetoothctlError("Bluetoothctl failed after running " + command)
|
||||||
|
|
||||||
|
return self.child.before.split("\r\n")
|
||||||
|
|
||||||
|
def enable_pairing(self):
|
||||||
|
"""Make device visible to scanning and enable pairing."""
|
||||||
|
print("pairing enabled")
|
||||||
|
try:
|
||||||
|
out = self.get_output("power on")
|
||||||
|
out = self.get_output("discoverable on")
|
||||||
|
out = self.get_output("pairable on")
|
||||||
|
out = self.get_output("agent off")
|
||||||
|
|
||||||
|
except BluetoothctlError, e:
|
||||||
|
print(e)
|
||||||
|
return None
|
||||||
|
|
||||||
|
def disable_pairing(self):
|
||||||
|
"""Disable devices visibility and ability to pair."""
|
||||||
|
try:
|
||||||
|
out = self.get_output("discoverable off")
|
||||||
|
out = self.get_output("pairable off")
|
||||||
|
|
||||||
|
except BluetoothctlError, e:
|
||||||
|
print(e)
|
||||||
|
return None
|
5
docs/bt_console/README.md
Normal file
5
docs/bt_console/README.md
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
# UNSUPPORTED
|
||||||
|
|
||||||
|
The code and information in this folder is currently **UNSUPPORTED**.
|
||||||
|
|
||||||
|
Use this information at your own risk.
|
47
docs/bt_console/bluezutils.py
Normal file
47
docs/bt_console/bluezutils.py
Normal file
|
@ -0,0 +1,47 @@
|
||||||
|
import dbus
|
||||||
|
|
||||||
|
SERVICE_NAME = "org.bluez"
|
||||||
|
ADAPTER_INTERFACE = SERVICE_NAME + ".Adapter1"
|
||||||
|
DEVICE_INTERFACE = SERVICE_NAME + ".Device1"
|
||||||
|
|
||||||
|
def get_managed_objects():
|
||||||
|
bus = dbus.SystemBus()
|
||||||
|
manager = dbus.Interface(bus.get_object("org.bluez", "/"),
|
||||||
|
"org.freedesktop.DBus.ObjectManager")
|
||||||
|
return manager.GetManagedObjects()
|
||||||
|
|
||||||
|
def find_adapter(pattern=None):
|
||||||
|
return find_adapter_in_objects(get_managed_objects(), pattern)
|
||||||
|
|
||||||
|
def find_adapter_in_objects(objects, pattern=None):
|
||||||
|
bus = dbus.SystemBus()
|
||||||
|
for path, ifaces in objects.iteritems():
|
||||||
|
adapter = ifaces.get(ADAPTER_INTERFACE)
|
||||||
|
if adapter is None:
|
||||||
|
continue
|
||||||
|
if not pattern or pattern == adapter["Address"] or \
|
||||||
|
path.endswith(pattern):
|
||||||
|
obj = bus.get_object(SERVICE_NAME, path)
|
||||||
|
return dbus.Interface(obj, ADAPTER_INTERFACE)
|
||||||
|
raise Exception("Bluetooth adapter not found")
|
||||||
|
|
||||||
|
def find_device(device_address, adapter_pattern=None):
|
||||||
|
return find_device_in_objects(get_managed_objects(), device_address,
|
||||||
|
adapter_pattern)
|
||||||
|
|
||||||
|
def find_device_in_objects(objects, device_address, adapter_pattern=None):
|
||||||
|
bus = dbus.SystemBus()
|
||||||
|
path_prefix = ""
|
||||||
|
if adapter_pattern:
|
||||||
|
adapter = find_adapter_in_objects(objects, adapter_pattern)
|
||||||
|
path_prefix = adapter.object_path
|
||||||
|
for path, ifaces in objects.iteritems():
|
||||||
|
device = ifaces.get(DEVICE_INTERFACE)
|
||||||
|
if device is None:
|
||||||
|
continue
|
||||||
|
if (device["Address"] == device_address and
|
||||||
|
path.startswith(path_prefix)):
|
||||||
|
obj = bus.get_object(SERVICE_NAME, path)
|
||||||
|
return dbus.Interface(obj, DEVICE_INTERFACE)
|
||||||
|
|
||||||
|
raise Exception("Bluetooth device not found")
|
190
docs/bt_console/bt-auto-agent
Normal file
190
docs/bt_console/bt-auto-agent
Normal file
|
@ -0,0 +1,190 @@
|
||||||
|
|
||||||
|
#!/usr/bin/python2
|
||||||
|
|
||||||
|
from __future__ import absolute_import, print_function, unicode_literals
|
||||||
|
|
||||||
|
from optparse import OptionParser
|
||||||
|
import sys
|
||||||
|
import dbus
|
||||||
|
import dbus.service
|
||||||
|
import dbus.mainloop.glib
|
||||||
|
try:
|
||||||
|
from gi.repository import GObject
|
||||||
|
except ImportError:
|
||||||
|
import gobject as GObject
|
||||||
|
import bluezutils
|
||||||
|
|
||||||
|
BUS_NAME = 'org.bluez'
|
||||||
|
AGENT_INTERFACE = 'org.bluez.Agent1'
|
||||||
|
AGENT_PATH = "/test/agent"
|
||||||
|
|
||||||
|
bus = None
|
||||||
|
device_obj = None
|
||||||
|
dev_path = None
|
||||||
|
|
||||||
|
def ask(prompt):
|
||||||
|
try:
|
||||||
|
return raw_input(prompt)
|
||||||
|
except:
|
||||||
|
return input(prompt)
|
||||||
|
|
||||||
|
def set_trusted(path):
|
||||||
|
props = dbus.Interface(bus.get_object("org.bluez", path),
|
||||||
|
"org.freedesktop.DBus.Properties")
|
||||||
|
props.Set("org.bluez.Device1", "Trusted", True)
|
||||||
|
|
||||||
|
def dev_connect(path):
|
||||||
|
dev = dbus.Interface(bus.get_object("org.bluez", path),
|
||||||
|
"org.bluez.Device1")
|
||||||
|
dev.Connect()
|
||||||
|
|
||||||
|
class Rejected(dbus.DBusException):
|
||||||
|
_dbus_error_name = "org.bluez.Error.Rejected"
|
||||||
|
|
||||||
|
class Agent(dbus.service.Object):
|
||||||
|
exit_on_release = True
|
||||||
|
|
||||||
|
def set_exit_on_release(self, exit_on_release):
|
||||||
|
self.exit_on_release = exit_on_release
|
||||||
|
|
||||||
|
@dbus.service.method(AGENT_INTERFACE,
|
||||||
|
in_signature="", out_signature="")
|
||||||
|
def Release(self):
|
||||||
|
print("Release")
|
||||||
|
if self.exit_on_release:
|
||||||
|
mainloop.quit()
|
||||||
|
|
||||||
|
@dbus.service.method(AGENT_INTERFACE,
|
||||||
|
in_signature="os", out_signature="")
|
||||||
|
def AuthorizeService(self, device, uuid):
|
||||||
|
print("AuthorizeService (%s, %s)" % (device, uuid))
|
||||||
|
return # automatically authorize connection
|
||||||
|
authorize = ask("Authorize connection (yes/no): ")
|
||||||
|
if (authorize == "yes"):
|
||||||
|
return
|
||||||
|
raise Rejected("Connection rejected by user")
|
||||||
|
|
||||||
|
@dbus.service.method(AGENT_INTERFACE,
|
||||||
|
in_signature="o", out_signature="s")
|
||||||
|
def RequestPinCode(self, device):
|
||||||
|
print("RequestPinCode (%s)" % (device))
|
||||||
|
set_trusted(device)
|
||||||
|
#return ask("Enter PIN Code: ")
|
||||||
|
return "0000" # return default PIN Code of 0000
|
||||||
|
|
||||||
|
@dbus.service.method(AGENT_INTERFACE,
|
||||||
|
in_signature="o", out_signature="u")
|
||||||
|
def RequestPasskey(self, device):
|
||||||
|
print("RequestPasskey (%s)" % (device))
|
||||||
|
set_trusted(device)
|
||||||
|
#passkey = ask("Enter passkey: ")
|
||||||
|
passkey = "0000" # return default passkey of 0000
|
||||||
|
return dbus.UInt32(passkey)
|
||||||
|
|
||||||
|
@dbus.service.method(AGENT_INTERFACE,
|
||||||
|
in_signature="ouq", out_signature="")
|
||||||
|
def DisplayPasskey(self, device, passkey, entered):
|
||||||
|
print("DisplayPasskey (%s, %06u entered %u)" %
|
||||||
|
(device, passkey, entered))
|
||||||
|
|
||||||
|
@dbus.service.method(AGENT_INTERFACE,
|
||||||
|
in_signature="os", out_signature="")
|
||||||
|
def DisplayPinCode(self, device, pincode):
|
||||||
|
print("DisplayPinCode (%s, %s)" % (device, pincode))
|
||||||
|
|
||||||
|
@dbus.service.method(AGENT_INTERFACE,
|
||||||
|
in_signature="ou", out_signature="")
|
||||||
|
def RequestConfirmation(self, device, passkey):
|
||||||
|
print("RequestConfirmation (%s, %06d)" % (device, passkey))
|
||||||
|
set_trusted(device)
|
||||||
|
return # automatically trust
|
||||||
|
confirm = ask("Confirm passkey (yes/no): ")
|
||||||
|
if (confirm == "yes"):
|
||||||
|
set_trusted(device)
|
||||||
|
return
|
||||||
|
raise Rejected("Passkey doesn't match")
|
||||||
|
|
||||||
|
@dbus.service.method(AGENT_INTERFACE,
|
||||||
|
in_signature="o", out_signature="")
|
||||||
|
def RequestAuthorization(self, device):
|
||||||
|
print("RequestAuthorization (%s)" % (device))
|
||||||
|
return # automatically authorize
|
||||||
|
auth = ask("Authorize? (yes/no): ")
|
||||||
|
if (auth == "yes"):
|
||||||
|
return
|
||||||
|
raise Rejected("Pairing rejected")
|
||||||
|
|
||||||
|
@dbus.service.method(AGENT_INTERFACE,
|
||||||
|
in_signature="", out_signature="")
|
||||||
|
def Cancel(self):
|
||||||
|
print("Cancel")
|
||||||
|
|
||||||
|
def pair_reply():
|
||||||
|
print("Device paired")
|
||||||
|
set_trusted(dev_path)
|
||||||
|
dev_connect(dev_path)
|
||||||
|
mainloop.quit()
|
||||||
|
|
||||||
|
def pair_error(error):
|
||||||
|
err_name = error.get_dbus_name()
|
||||||
|
if err_name == "org.freedesktop.DBus.Error.NoReply" and device_obj:
|
||||||
|
print("Timed out. Cancelling pairing")
|
||||||
|
device_obj.CancelPairing()
|
||||||
|
else:
|
||||||
|
print("Creating device failed: %s" % (error))
|
||||||
|
|
||||||
|
|
||||||
|
mainloop.quit()
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
dbus.mainloop.glib.DBusGMainLoop(set_as_default=True)
|
||||||
|
|
||||||
|
bus = dbus.SystemBus()
|
||||||
|
|
||||||
|
capability = "KeyboardDisplay"
|
||||||
|
|
||||||
|
parser = OptionParser()
|
||||||
|
parser.add_option("-i", "--adapter", action="store",
|
||||||
|
type="string",
|
||||||
|
dest="adapter_pattern",
|
||||||
|
default=None)
|
||||||
|
parser.add_option("-c", "--capability", action="store",
|
||||||
|
type="string", dest="capability")
|
||||||
|
parser.add_option("-t", "--timeout", action="store",
|
||||||
|
type="int", dest="timeout",
|
||||||
|
default=60000)
|
||||||
|
(options, args) = parser.parse_args()
|
||||||
|
if options.capability:
|
||||||
|
capability = options.capability
|
||||||
|
|
||||||
|
path = "/test/agent"
|
||||||
|
agent = Agent(bus, path)
|
||||||
|
|
||||||
|
mainloop = GObject.MainLoop()
|
||||||
|
|
||||||
|
obj = bus.get_object(BUS_NAME, "/org/bluez");
|
||||||
|
manager = dbus.Interface(obj, "org.bluez.AgentManager1")
|
||||||
|
manager.RegisterAgent(path, capability)
|
||||||
|
|
||||||
|
print("Agent registered")
|
||||||
|
|
||||||
|
# Fix-up old style invocation (BlueZ 4)
|
||||||
|
if len(args) > 0 and args[0].startswith("hci"):
|
||||||
|
options.adapter_pattern = args[0]
|
||||||
|
del args[:1]
|
||||||
|
|
||||||
|
if len(args) > 0:
|
||||||
|
device = bluezutils.find_device(args[0],
|
||||||
|
options.adapter_pattern)
|
||||||
|
dev_path = device.object_path
|
||||||
|
agent.set_exit_on_release(False)
|
||||||
|
device.Pair(reply_handler=pair_reply, error_handler=pair_error,
|
||||||
|
timeout=60000)
|
||||||
|
device_obj = device
|
||||||
|
else:
|
||||||
|
manager.RequestDefaultAgent(path)
|
||||||
|
|
||||||
|
mainloop.run()
|
||||||
|
|
||||||
|
#adapter.UnregisterAgent(path)
|
||||||
|
#print("Agent unregistered")
|
70
docs/bt_console/bt_serial_console.md
Normal file
70
docs/bt_console/bt_serial_console.md
Normal file
|
@ -0,0 +1,70 @@
|
||||||
|
# Bluetooth Serial Console
|
||||||
|
|
||||||
|
The below information will help you setup bluetooth as a serial console on the Raspberry Pi. This is **WILDLY INSECURE** and should **NEVER** be trusted.
|
||||||
|
|
||||||
|
This is here for folks who don't want to add a serial adapter to their builds and/or need a quick and dirty way to get into a PiFrame from their phone.
|
||||||
|
|
||||||
|
**DO NOT USE THIS IF YOU'RE UNSURE OF THE SECURITY IMPLICATIONS**
|
||||||
|
|
||||||
|
``` sh
|
||||||
|
|
||||||
|
|
||||||
|
https://www.historiantech.com/create-a-bluetooth-wireless-console-server-with-raspberry-pi-zero-w/
|
||||||
|
https://www.raspberrypi.org/forums/viewtopic.php?t=272419
|
||||||
|
https://www.raspberrypi.org/forums/viewtopic.php?t=170353
|
||||||
|
https://play.google.com/store/apps/details?id=de.kai_morich.serial_bluetooth_terminal
|
||||||
|
|
||||||
|
|
||||||
|
apt install pi-bluetooth python-pexpect python-dbus rfkill
|
||||||
|
systemctl reboot
|
||||||
|
|
||||||
|
|
||||||
|
echo "PRETTY_HOSTNAME=piframe" > /etc/machine-info
|
||||||
|
|
||||||
|
|
||||||
|
/usr/local/bin/bt-auto-agent
|
||||||
|
/usr/local/bin/bluezutils.py
|
||||||
|
/usr/local/bin/BtAutoPair.py
|
||||||
|
|
||||||
|
|
||||||
|
sed -i 's/#DiscoverableTimeout = 0/DiscoverableTimeout = 0/g' /etc/bluetooth/main.conf
|
||||||
|
|
||||||
|
nano -w /usr/local/bin/bt-config.sh
|
||||||
|
#!/usr/bin/python2
|
||||||
|
import BtAutoPair
|
||||||
|
autopair = BtAutoPair.BtAutoPair()
|
||||||
|
autopair.enable_pairing()
|
||||||
|
|
||||||
|
chmod a+x /usr/local/bin/bt-config.sh
|
||||||
|
|
||||||
|
mkdir /etc/systemd/system/bluetooth.service.d/
|
||||||
|
cat > /etc/systemd/system/bluetooth.service.d/piframe.conf <<EOF
|
||||||
|
[Service]
|
||||||
|
ExecStartPre=/usr/bin/hciconfig hci0 up
|
||||||
|
ExecStart=
|
||||||
|
ExecStart=/usr/lib/bluetooth/bluetoothd -C
|
||||||
|
ExecStartPost=/usr/bin/sdptool add SP
|
||||||
|
ExecStartPost=/bin/hciconfig hci0 piscan
|
||||||
|
ExecStartPost=/usr/local/bin/bt-config.sh
|
||||||
|
EOF
|
||||||
|
|
||||||
|
cat > /etc/systemd/system/rfcomm.service <<EOF
|
||||||
|
[Unit]
|
||||||
|
Description=RFCOMM service
|
||||||
|
After=bluetooth.service
|
||||||
|
Requires=bluetooth.service
|
||||||
|
|
||||||
|
[Service]
|
||||||
|
ExecStart=/usr/bin/rfcomm watch hci0 1 getty rfcomm0 115200 vt100
|
||||||
|
|
||||||
|
[Install]
|
||||||
|
WantedBy=multi-user.target
|
||||||
|
EOF
|
||||||
|
|
||||||
|
|
||||||
|
systemctl daemon-reload
|
||||||
|
systemctl enable --now bluetooth.service
|
||||||
|
systemctl restart bluetooth.service
|
||||||
|
systemctl enable --now rfcomm
|
||||||
|
|
||||||
|
```
|
Loading…
Reference in a new issue