-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathBTServer2.py
216 lines (170 loc) · 6.24 KB
/
BTServer2.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
import time
import binascii
import pygatt
class BTServer:
"""A server class supporting Bluetooth."""
# Bluetooth addresses
AGV_1_LILA = "34:15:13:1C:AF:0B"
AGV_2_GREEN = "34:15:13:1C:6C:E6"
# Bluetooth messages and commands
AGV1_AT_P10 = 0
AGV1_AT_P11 = 1
COMMAND_MOVE_AGV1 = bytearray([0x32])
AGV1_MOVING = 3
AGV2_AT_P20 = 4
AGV2_AT_P21 = 5
COMMAND_MOVE_AGV2 = bytearray([0x36])
AGV2_MOVING = 7
# Bluetooth address type
ADDRESS_TYPE = pygatt.BLEAddressType.public
def __init__(self, debug=True):
# State variables
self.stateAGV1 = -1
self.stateAGV2 = -1
# Connection variables
self.adapter = 0
self.device = 0
self.msgreceived = False
# BT connection state variables
self.BTconnected = 0
self.BTsubscribed = 0
# Debug variable
self.d = debug
def updateState(self, AGV_id):
""" Creates a connection with the selected AGV and reads the latest state it published """
# 1 - Select which one to connect
if AGV_id == 1:
AGVaddress = self.AGV_1_LILA
elif AGV_id == 2:
AGVaddress = self.AGV_2_GREEN
else:
error("Unknown AGV_id: must be 1 or 2")
# 2 - If we are not connected: connect
if self.BTconnected != AGV_id:
# 2.1 - If there is another connection, close it
if self.BTconnected != 0:
self.adapter.stop()
if self.d: print("[BT %d]: Disconnected." %(self.BTconnected))
self.BTconnected = 0
self.BTsubscribed = 0
# 2.2 - Create the connection
self.adapter = pygatt.GATTToolBackend()
self.adapter.start()
while self.BTconnected != AGV_id:
try:
if self.d: print("[BT %d]: Connecting." % (AGV_id))
self.device = self.adapter.connect(AGVaddress, address_type=self.ADDRESS_TYPE)
if self.d: print("[BT %d]: Connected." %(AGV_id))
self.BTconnected = AGV_id
except:
if self.d: print("[BT %d]: Unable to connect, trying again." % (AGV_id))
time.sleep(0.2)
# 3 - If we are not subscribed: subscribe
if self.BTsubscribed != AGV_id:
# 3.1 - Subscribe to the device to listen for info
while self.BTsubscribed != AGV_id:
try:
if self.d: print("[BT %d]: Subscribing for info." %(AGV_id))
if AGV_id == 1:
self.device.subscribe("0000ffe1-0000-1000-8000-00805f9b34fb", callback=self.handle_msg_1)
else:
self.device.subscribe("0000ffe1-0000-1000-8000-00805f9b34fb", callback=self.handle_msg_2)
if self.d: print("[BT %d]: Subscribed to." %(AGV_id))
self.BTsubscribed = AGV_id
self.msgreceived = False
except:
if self.d: print("[BT %d]: Unable to subscribe, trying again." %(AGV_id))
time.sleep(0.2)
# 4 - Wait until we receive at least one message
while not self.msgreceived:
time.sleep(0.1)
continue
def handle_msg_1(self, handle, value):
""" Handle and incoming message for AGV1: get the last message """
##
##if self.d: print("[BT 1]: Received data: %s" % (value.decode("utf-8")))
##
# 1 - Get the data as ASCII and take the last value
data_str = value.decode("utf-8")
newest_value = int(data_str[-1])
self.stateAGV1 = newest_value # update state
self.msgreceived = True # we have received a message
def handle_msg_2(self, handle, value):
""" Handle and incoming message for AGV1: get the last message """
##
##if self.d: print("[BT 2]: Received data: %s" % (value.decode("utf-8")))
##
# 1 - Get the data as ASCII and take the last value
data_str = value.decode("utf-8")
newest_value = int(data_str[-1])
self.stateAGV2 = newest_value # update state
self.msgreceived = True # we have received a message
def sendMoveMessage(self, AGV_id, subscribeToState=False):
""" Creates a connection with the selected AGV and reads the latest state it published """
# 1 - Select which one to connect
if AGV_id == 1:
AGVaddress = self.AGV_1_LILA
msg = self.COMMAND_MOVE_AGV1
elif AGV_id == 2:
AGVaddress = self.AGV_2_GREEN
msg = self.COMMAND_MOVE_AGV2
else:
error("Unknown AGV_id: must be 1 or 2")
# 2 - If we are not connected: connect
if self.BTconnected != AGV_id:
# 2.1 - If there is another connection, close it
if self.BTconnected != 0:
self.adapter.stop()
if self.d: print("[BT %d]: Disconnected." %(self.BTconnected))
self.BTconnected = 0
self.BTsubscribed = 0
# 2.2 - Create the connection
self.adapter = pygatt.GATTToolBackend()
self.adapter.start()
while self.BTconnected != AGV_id:
try:
if self.d: print("[BT %d]: Connecting." % (AGV_id))
self.device = self.adapter.connect(AGVaddress, address_type=self.ADDRESS_TYPE)
if self.d: print("[BT %d]: Connected." %(AGV_id))
self.BTconnected = AGV_id
except:
if self.d: print("[BT %d]: Unable to connect, trying again." % (AGV_id))
time.sleep(0.2)
# 3 - Send the message
sent = False
while not sent:
try:
if self.d: print("[BT %d]: Sending data." %(AGV_id))
self.device.char_write("0000ffe1-0000-1000-8000-00805f9b34fb", msg)
if self.d: print("[BT %d] Sent data: %s" % (AGV_id, msg.decode("utf-8")))
sent = True
except:
if self.d: print("[BT %d]: Unable to send, trying again." %(AGV_id))
# 4 - If we want to subscribe
if subscribeToState:
# 4.1 - If we are not subscribed: subscribe
if self.BTsubscribed != AGV_id:
# 4.1.1 - Subscribe to the device to listen for info
while self.BTsubscribed != AGV_id:
try:
if self.d: print("[BT %d]: Subscribing for info." %(AGV_id))
if AGV_id == 1:
self.device.subscribe("0000ffe1-0000-1000-8000-00805f9b34fb", callback=self.handle_msg_1)
else:
self.device.subscribe("0000ffe1-0000-1000-8000-00805f9b34fb", callback=self.handle_msg_2)
if self.d: print("[BT %d]: Subscribed to." %(AGV_id))
self.BTsubscribed = AGV_id
self.msgreceived = False
except:
if self.d: print("[BT %d]: Unable to subscribe, trying again." %(AGV_id))
time.sleep(0.2)
def disconnect(self):
""" Stop the current adapter: disconnect from device """
if self.BTconnected:
self.adapter.stop()
if self.d: print("[BT %d]: Disconnected." %(self.BTconnected))
self.BTconnected = 0
self.BTsubscribed = 0
self.msgreceived = False
# device.char_write('0000ffe1-0000-1000-8000-00805f9b34fb', COMMAND_MOVE_AGV1)
# print("Sent data: (HEX) %s, ASCII %s" % (binascii.hexlify(COMMAND_MOVE_AGV1), COMMAND_MOVE_AGV1.decode("utf-8")))