forked from iplayfast/zoidberg
-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathkicker.py
137 lines (111 loc) · 4.23 KB
/
kicker.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
import logging
import threading
from time import time, sleep
from typing import Optional
import yaml
import uavcan
from uavcan import UAVCANException
from serial_wrapper import find_serial
logger = logging.getLogger("canbus")
class CanBusMotor:
# TODO: this has the ability to not constantly update the speed, we should use that
def __init__(self, kill=True) -> None:
self.last_raw = ""
self.last_msg = {}
self.last_rpm = 0
self.rpm = []
self._speed = 0
self.last_edit = time()
self.kill = kill
self.node: Optional[uavcan.node.Node] = None
self.thread = threading.Thread(target=self.run, daemon=True)
self.thread.start()
def run(self):
self.reconnect()
while True:
try:
self.node.spin(1) # Spin forever or until an exception is thrown
except (Exception, UAVCANException) as ex:
logger.error('Node error: %s', ex)
if self.node:
logger.error('Closing node')
self.node.close()
self.node: Optional[uavcan.node.Node] = None
sleep(2)
self.reconnect()
def reconnect(self):
while True:
zubax = set(find_serial('zubax').keys())
if len(zubax) != 1:
logger.error(f"Zubax controller not determined, {zubax}")
sleep(1)
continue
serial_device = next(iter(zubax), None)
print(serial_device)
logger.info(f"Zubax controller determined, {zubax}")
try:
self.node: Optional[uavcan.node.Node] = uavcan.make_node(
serial_device,
node_id=10,
bitrate=1000000,
)
# setup
node_monitor = uavcan.app.node_monitor.NodeMonitor(self.node)
dynamic_node_id_allocator = uavcan.app.dynamic_node_id.CentralizedServer(self.node, node_monitor)
# Waiting for at least one other node to appear online (our local node is already online).
while len(dynamic_node_id_allocator.get_allocation_table()) <= 1:
logger.error('Waiting for other nodes to become online...')
self.node.spin(timeout=1)
# how fast can we blast this?
self.node.periodic(0.05, self.update)
self.node.add_handler(uavcan.equipment.esc.Status, self.listen)
logger.info('New zubax node: %s', serial_device)
return
except Exception as e:
logger.error("Opening zubax failed %s", e)
sleep(1)
continue
@property
def speed(self):
return self._speed
@speed.setter
def speed(self, speed):
speed = abs(speed)
speed = min(15000, speed)
self._speed = int(speed)
self.last_edit = time()
def update(self):
if time() - self.last_edit > 0.8:
if self._speed:
pass
if self.kill:
self._speed = 0
else:
self.last_edit = time()
min_speed = 3000
kick_speed = max(int(self._speed), min_speed)
message = uavcan.equipment.esc.RPMCommand(rpm=[kick_speed, 3000])
self.node.broadcast(message)
def listen(self, msg):
"""
Transfer(
id=4, source_node_id=125, dest_node_id=None, transfer_priority=7,
payload=uavcan.equipment.esc.Status(
error_count=0, voltage=12.9296875,
current=-0.0, temperature=307.0, rpm=0, power_rating_pct=0, esc_index=0)
)
"""
self.last_raw = uavcan.to_yaml(msg)
self.last_msg = yaml.load(self.last_raw)
if self.last_msg.get('esc_index', None) == 0:
self.rpm = (self.rpm + [self.last_msg.get('rpm', 0)])[-10:]
self.last_rpm = round(sum(self.rpm) / len(self.rpm))
if __name__ == '__main__':
kicker = CanBusMotor(kill=False)
while True:
try:
speed = int(input("speed?: "))
kicker.speed = speed
except ValueError:
pass
print(kicker.last_raw)