forked from al5681/Sphero_Bolt_Multiplatform_Python_Bleak
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsphero_bolt.py
255 lines (223 loc) · 9.93 KB
/
sphero_bolt.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
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
import struct
from sphero_constants import *
from bleak import BleakClient
from bleak import BleakScanner
class BoltScan:
def __init__(self):
self.bolts = []
async def scan(self, name=None):
print(f'[SCAN] Scanning for Sphero Bolt with name {name}')
devices = await BleakScanner.discover()
for de in devices:
#print(de)
try:
if de.name.startswith('SB-'):
if name == None:
return de.address
else:
if name == de.name:
return de.address
except Exception as e:
print(e)
async def scanAll(self):
print(f'[SCAN] Scanning for all nearby Sphero Bolts')
d = []
devices = await BleakScanner.discover()
for de in devices:
try:
if de.name.startswith('SB-'):
d += [de]
except:
pass
print('[SCAN] {} Spheros Detected'.format(len(d)))
return d
class SpheroBolt:
def __init__(self, address):
self.sequence = 0
self.address = address
self.notificationPacket = []
async def connect(self):
"""
Connects to a Sphero Bolt of a specified MAC address if it can find it.
"""
self.client = BleakClient(self.address)
await self.client.connect()
print("[BOLT] Connected: {0}".format(self.client.is_connected))
# cancel if not connected
if not self.client.is_connected:
return False
# get device name
try:
DEVICE_NAME_UUID = "00002A00-0000-1000-8000-00805f9b34fb"
device_name = await self.client.read_gatt_char(DEVICE_NAME_UUID)
print("Device Name: {0}".format("".join(map(chr, device_name))))
except Exception:
pass
self.API_V2_characteristic = "00010002-574f-4f20-5370-6865726f2121"
AntiDOS_characteristic = "00020005-574f-4f20-5370-6865726f2121"
# Unlock code: prevent the sphero mini from going to sleep again after 10 seconds
print("[INIT] Writing AntiDOS characteristic unlock code")
try:
await self.client.write_gatt_char(AntiDOS_characteristic, b"usetheforce...band", response=True)
except:
return False
print("[INIT] Initialization complete\n")
return True
async def disconnect(self):
"""
Disconnects the Sphero Bolt
"""
return await self.client.disconnect()
async def send(self, characteristic=None, devID=None, commID=None, targetId=None, data=[]):
"""
Generate databytes of command using input dictionary
This protocol copied completely from JS library
Messages are represented as:
[start flags targetID sourceID deviceID commandID seqNum data
checksum end]
The flags byte indicates which fields are populated.
The checksum is the ~sum(message[1:-2]) | 0xff.
"""
try:
self.sequence = (self.sequence + 1) % 256
running_sum = 0
command = []
command.append(SendPacketConstants["StartOfPacket"])
if targetId is None:
cmdflg = Flags["requestsResponse"] | \
Flags["resetsInactivityTimeout"] | 0
command.append(cmdflg)
running_sum += cmdflg
else:
cmdflg = Flags["requestsResponse"] | \
Flags["resetsInactivityTimeout"] | targetId
command.append(cmdflg)
running_sum += cmdflg
command.append(targetId)
running_sum += targetId
command.append(devID)
running_sum += devID
command.append(commID)
running_sum += commID
command.append(self.sequence)
running_sum += self.sequence
if data is not None:
for datum in data:
command.append(datum)
running_sum += datum
checksum = (~running_sum) & 0xff
command.append(checksum)
command.append(SendPacketConstants["EndOfPacket"])
await self.client.write_gatt_char(characteristic, command)
except Exception as e:
#print(e)
return
async def wake(self):
"""
Bring device out of sleep mode (can only be done if device was in sleep, not deep sleep).
If in deep sleep, the device should be connected to USB power to wake.
"""
print("[SEND {}] Waking".format(self.sequence))
while True:
try:
await self.send(
characteristic=self.API_V2_characteristic,
devID=DeviceID["powerInfo"],
commID=PowerCommandIDs["wake"],
data=[]) # empty payload
return
except Exception as e:
print('Error waking retrying', e)
async def setBothLEDColors(self, red=None, green=None, blue=None):
"""
Set device LED color based on RGB vales (each can range between 0 and 0xFF).
"""
print("[SEND {}] Setting front LED colour to [{}, {}, {}]".format(self.sequence, red, green, blue))
await self.send(characteristic=self.API_V2_characteristic,
devID=DeviceID["userIO"],
commID=UserIOCommandIDs["allLEDs"],
data=[0x3f, red, green, blue, red, green, blue])
async def setMatrixPix(self, x=0, y=0, red=None, green=None, blue=None):
"""
Set the LED matrix pixels based on RBG values.
"""
print("[SEND {}] Setting matrix Pixel colour to [{}, {}] [{}, {}, {}]".format(self.sequence, x, y, red, green, blue))
await self.send(characteristic=self.API_V2_characteristic,
devID=DeviceID["userIO"],
commID=UserIOCommandIDs["matrixPix"],
targetId=0x012,
data=[x, y, red, green, blue])
async def calibrateToNorth(self):
"""
Calibates Sphero to north
"""
print("[SEND {}] Calibrating to North".format(self.sequence))
await self.send(characteristic=self.API_V2_characteristic,
devID=DeviceID["userIO"],
commID=SensorCommandIds["calibrateToNorth"],
targetId=0x12,
data=[])
async def setFrontLEDColor(self, red=None, green=None, blue=None):
"""
Set device front LED color based on RGB vales (each can range between 0 and 0xFF).
"""
print("[SEND {}] Setting front LED colour to [{}, {}, {}]".format(self.sequence, red, green, blue))
await self.send(characteristic=self.API_V2_characteristic,
devID=DeviceID["userIO"],
commID=UserIOCommandIDs["allLEDs"],
data=[0x07, red, green, blue])
async def setBackLEDColor(self, red=None, green=None, blue=None):
"""
Set device back LED color based on RGB vales (each can range between 0 and 0xFF).
"""
print("[SEND {}] Setting back LED colour to [{}, {}, {}]".format(self.sequence, red, green, blue))
await self.send(characteristic=self.API_V2_characteristic,
devID=DeviceID["userIO"],
commID=UserIOCommandIDs["allLEDs"],
data=[0x38, red, green, blue])
async def setMatrixLED(self, red=None, green=None, blue=None):
"""
Set the LED matrix based on RBG values.
"""
print("[SEND {}] Setting matrix LED colour to [{}, {}, {}]".format(self.sequence, red, green, blue))
await self.send(characteristic=self.API_V2_characteristic,
devID=DeviceID["userIO"],
commID=UserIOCommandIDs["matrixColor"],
targetId=0x012,
data=[red, green, blue])
async def setMatrixLEDChar(self, char=None, red=None, green=None, blue=None):
"""
Write a character to the LED matrix, with a colour based on RGB values.
"""
print("[SEND {}] Setting matrix char to ' {} ' LED colour to [{}, {}, {}]".format(self.sequence, char, red, green, blue))
await self.send(characteristic=self.API_V2_characteristic,
devID=DeviceID["userIO"],
commID=UserIOCommandIDs["printChar"],
targetId=0x012,
data=[red, green, blue, ord(char)])
async def roll(self, speed=None, heading=None):
"""
Rolls the device at a specified speed (int between 0 and 255)
and heading (int between 0 and 359).
Data is format [speed, heading byte 1, heading byte 2, direction (0-forward, 1-back)].
"""
print("[SEND {}] Rolling with speed {} and heading {}".format(self.sequence, speed, heading))
await self.send(characteristic=self.API_V2_characteristic,
devID=DeviceID["driving"],
commID=DrivingCommands["driveWithHeading"],
targetId=0x012,
data=[speed, (heading >> 8) & 0xff, heading & 0xff, 0])
async def resetYaw(self):
print("[SEND {}] Resetting yaw".format(self.sequence))
await self.send(characteristic=self.API_V2_characteristic,
devID=DeviceID["driving"],
commID=DrivingCommands["resetYaw"],
data=[])
def bitsToNum(self, bits):
"""
This helper function decodes bytes from sensor packets into single precision floats. Encoding follows the
the IEEE-754 standard.
"""
num = int(bits, 2).to_bytes(len(bits) // 8, byteorder='little')
num = struct.unpack('f', num)[0]
return num