-
Notifications
You must be signed in to change notification settings - Fork 3
/
keyble.py
executable file
·165 lines (140 loc) · 6 KB
/
keyble.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
#!/usr/bin/env python3
#
# 2019 Alexander 'lynxis' Couzens <lynxis@fe80.eu>
# GPLv3
import argparse
import binascii
import logging
import re
import sys
import os
import threading
import time
import traceback
from bluepy.btle import Scanner, DefaultDelegate
from fsm import Device
# exit on any exception
def global_exception_hook(ex_type, ex, trace):
traceback.print_exception(ex_type, ex, trace)
os._exit(1)
sys.excepthook = global_exception_hook
def filter_keyble(devices):
""" return only keyble locks """
keyble = []
for dev in devices:
for (adtype, desc, value) in dev.getScanData():
if desc == "Complete Local Name" and value == "KEY-BLE":
keyble.append(dev)
return keyble
def scan():
""" scan via BLE for locks """
scanner = Scanner()
devices = scanner.scan(10.0)
return filter_keyble(devices)
def ui_scan():
devices = scan()
if not devices:
print("No devices found")
return
print("Found keyble devices")
for dev in devices:
print("{}".format(dev.addr))
def ui_discover(device, userid=1):
device = Device(device, userid=userid)
infos = device.discover()
print(infos)
def ui_pair(device, userid, userkey, cardkey):
if userid == None:
userid = 0xff
# TODO: check userkey, cardkey, userid
_userkey = binascii.unhexlify(userkey)
if len(_userkey) != 16:
raise RuntimeError("Userkey is too short or too long. Expecting 16 byte encode as hex (32 characters)")
_cardkey = binascii.unhexlify(cardkey)
if len(_cardkey) != 16:
raise RuntimeError("Cardkey is too short or too long. Expecting 16 byte encode as hex (32 characters)")
device = Device(device, userid=userid)
device.pair(_userkey, _cardkey)
def ui_command(device, userid, userkey, command):
_userkey = binascii.unhexlify(userkey)
if len(_userkey) != 16:
raise RuntimeError("Userkey is too short or too long. Expecting 16 byte encode as hex (32 characters)")
device = Device(device, userid=userid, userkey=_userkey)
if command == "open":
device.open()
elif command == "unlock":
device.unlock()
elif command == "lock":
device.lock()
print("device %s" % str(command))
os._exit(0)
def ui_status(device, userid, userkey):
_userkey = binascii.unhexlify(userkey)
if len(_userkey) != 16:
raise RuntimeError("Userkey is too short or too long. Expecting 16 byte encode as hex (32 characters)")
device = Device(device, userid=userid, userkey=_userkey)
status = device.status()
if not status:
raise RuntimeError("Can not get the status")
print("device status = %s" % str(status))
def set_timeout(timeout):
""" exit after timeout seconds """
def _timeouter():
time.sleep(timeout)
print("Operation timed out! Exit 2", file=sys.stderr)
os._exit(2)
thr = threading.Thread(target=_timeouter)
thr.start()
def main():
parser = argparse.ArgumentParser(description='keybtle')
parser.add_argument('--scan', dest='scan', action='store_true', help='Scan for KeyBLEs')
parser.add_argument('--device', dest='device', help='Device MAC address')
parser.add_argument('--discover', dest='discover', action='store_true', help='Ask the bootloader/app version')
parser.add_argument('--user-id', dest='userid', help='The user id', type=int)
parser.add_argument('--user-key', dest='userkey', help='The user key (a rsa key generated when registering the user)')
parser.add_argument('--status', dest='status', action='store_true', help='Shows the status. Require --user-id --user-key --device.')
parser.add_argument('--open', dest='open', action='store_true', help='Unlock and Open. Require --user-id --user-key --device.')
parser.add_argument('--lock', dest='lock', action='store_true', help='Lock. Require --user-id --user-key --device.')
parser.add_argument('--unlock', dest='unlock', action='store_true', help='Unlock. Require --user-id --user-key --device.')
parser.add_argument('--register', dest='register', action='store_true', help='Register a new user. Require --qrdata, optional --user-name')
parser.add_argument('--user-name', dest='username', help='The administrator will see this name when listing all users')
parser.add_argument('--qrdata', dest='qrdata', help='The QR Code as data. This contains the mac,cardkey,serial.')
parser.add_argument('--verbose', dest='verbose', action='store_true', help='Enable debug logging.')
parser.add_argument('--timeout', dest='timeout', help='Exit after x seconds even when the operation hasn\'t finished.', type=float)
args = parser.parse_args()
if args.verbose:
logging.basicConfig(format="%(asctime)-15s %(levelname)-8s %(name)-22s %(message)s", level=logging.DEBUG)
else:
logging.basicConfig(format="%(asctime)-15s %(levelname)-8s %(name)-22s %(message)s", level=logging.ERROR)
if args.timeout:
set_timeout(args.timeout)
if args.scan:
ui_scan()
if args.status:
ui_status(args.device, args.userid, args.userkey)
if args.open:
ui_command(args.device, args.userid, args.userkey, "open")
if args.lock:
ui_command(args.device, args.userid, args.userkey, "lock")
if args.unlock:
ui_command(args.device, args.userid, args.userkey, "unlock")
if args.discover:
ui_discover(args.device)
if args.register:
if not args.qrdata:
raise RuntimeError("You need to specify --qrdata")
# M001234556678K01234567890ABCDEF023456789ABCDEF0123456789
rex = re.compile(r'^M([0-9A-F]{12})K([0-9A-F]{32})([0-9A-Z]{10})$')
match = rex.match(args.qrdata)
if not match:
raise RuntimeError("Invalid QR Data")
smac, cardkey, serial = match.groups()
mac = ""
for i in range(len(smac) >> 1):
mac += smac[i*2]
mac += smac[i*2+1]
mac += ":"
mac = mac[0:-1]
ui_pair(mac, args.userid, args.userkey, cardkey)
if __name__ == '__main__':
main()