motan: Add support for graphing arbitrary get_status fields
Signed-off-by: Kevin O'Connor <kevin@koconnor.net>
This commit is contained in:
parent
1e4041a96b
commit
5e968db355
|
@ -16,6 +16,33 @@ class error(Exception):
|
||||||
# Log data handlers: {name: class, ...}
|
# Log data handlers: {name: class, ...}
|
||||||
LogHandlers = {}
|
LogHandlers = {}
|
||||||
|
|
||||||
|
# Extract requested position, velocity, and accel from a trapq log
|
||||||
|
class HandleStatusField:
|
||||||
|
SubscriptionIdParts = 0
|
||||||
|
ParametersMin = ParametersMax = 1
|
||||||
|
DataSets = [
|
||||||
|
('status(<field>)', 'A get_status field name (separate by periods)'),
|
||||||
|
]
|
||||||
|
def __init__(self, lmanager, name, name_parts):
|
||||||
|
self.status_tracker = lmanager.get_status_tracker()
|
||||||
|
self.field_name = name_parts[1]
|
||||||
|
self.field_parts = name_parts[1].split('.')
|
||||||
|
self.next_update_time = 0.
|
||||||
|
self.result = None
|
||||||
|
def get_label(self):
|
||||||
|
label = '%s field' % (self.field_name,)
|
||||||
|
return {'label': label, 'units': 'Unknown'}
|
||||||
|
def pull_data(self, req_time):
|
||||||
|
if req_time < self.next_update_time:
|
||||||
|
return self.result
|
||||||
|
db, next_update_time = self.status_tracker.pull_status(req_time)
|
||||||
|
for fp in self.field_parts[:-1]:
|
||||||
|
db = db.get(fp, {})
|
||||||
|
self.result = db.get(self.field_parts[-1], 0.)
|
||||||
|
self.next_update_time = next_update_time
|
||||||
|
return self.result
|
||||||
|
LogHandlers["status"] = HandleStatusField
|
||||||
|
|
||||||
# Extract requested position, velocity, and accel from a trapq log
|
# Extract requested position, velocity, and accel from a trapq log
|
||||||
class HandleTrapQ:
|
class HandleTrapQ:
|
||||||
SubscriptionIdParts = 2
|
SubscriptionIdParts = 2
|
||||||
|
@ -275,6 +302,30 @@ class JsonDispatcher:
|
||||||
# Dataset and log tracking
|
# Dataset and log tracking
|
||||||
######################################################################
|
######################################################################
|
||||||
|
|
||||||
|
# Tracking of get_status messages
|
||||||
|
class TrackStatus:
|
||||||
|
def __init__(self, lmanager, name, start_status):
|
||||||
|
self.name = name
|
||||||
|
self.jdispatch = lmanager.get_jdispatch()
|
||||||
|
self.next_status_time = 0.
|
||||||
|
self.status = dict(start_status)
|
||||||
|
self.next_update = {}
|
||||||
|
def pull_status(self, req_time):
|
||||||
|
status = self.status
|
||||||
|
while 1:
|
||||||
|
if req_time < self.next_status_time:
|
||||||
|
return status, self.next_status_time
|
||||||
|
for k, v in self.next_update.items():
|
||||||
|
status.setdefault(k, {}).update(v)
|
||||||
|
jmsg = self.jdispatch.pull_msg(req_time, self.name)
|
||||||
|
if jmsg is None:
|
||||||
|
self.next_status_time = req_time + 0.100
|
||||||
|
self.next_update = {}
|
||||||
|
return status, self.next_status_time
|
||||||
|
self.next_update = jmsg['status']
|
||||||
|
th = self.next_update.get('toolhead', {})
|
||||||
|
self.next_status_time = th.get('estimated_print_time', 0.)
|
||||||
|
|
||||||
# Split a string by commas while keeping parenthesis intact
|
# Split a string by commas while keeping parenthesis intact
|
||||||
def param_split(line):
|
def param_split(line):
|
||||||
out = []
|
out = []
|
||||||
|
@ -313,12 +364,13 @@ class LogManager:
|
||||||
self.initial_start_time = self.start_time = 0.
|
self.initial_start_time = self.start_time = 0.
|
||||||
self.datasets = {}
|
self.datasets = {}
|
||||||
self.initial_status = {}
|
self.initial_status = {}
|
||||||
|
self.start_status = {}
|
||||||
self.log_subscriptions = {}
|
self.log_subscriptions = {}
|
||||||
self.status = {}
|
self.status_tracker = None
|
||||||
def setup_index(self):
|
def setup_index(self):
|
||||||
fmsg = self.index_reader.pull_msg()
|
fmsg = self.index_reader.pull_msg()
|
||||||
self.initial_status = status = fmsg['status']
|
self.initial_status = status = fmsg['status']
|
||||||
self.status = dict(status)
|
self.start_status = dict(status)
|
||||||
start_time = status['toolhead']['estimated_print_time']
|
start_time = status['toolhead']['estimated_print_time']
|
||||||
self.initial_start_time = self.start_time = start_time
|
self.initial_start_time = self.start_time = start_time
|
||||||
self.log_subscriptions = fmsg.get('subscriptions', {})
|
self.log_subscriptions = fmsg.get('subscriptions', {})
|
||||||
|
@ -330,6 +382,7 @@ class LogManager:
|
||||||
return self.jdispatch
|
return self.jdispatch
|
||||||
def seek_time(self, req_time):
|
def seek_time(self, req_time):
|
||||||
self.start_time = req_start_time = self.initial_start_time + req_time
|
self.start_time = req_start_time = self.initial_start_time + req_time
|
||||||
|
start_status = self.start_status
|
||||||
seek_time = max(self.initial_start_time, req_start_time - 1.)
|
seek_time = max(self.initial_start_time, req_start_time - 1.)
|
||||||
file_position = 0
|
file_position = 0
|
||||||
while 1:
|
while 1:
|
||||||
|
@ -340,6 +393,8 @@ class LogManager:
|
||||||
ptime = max(th['estimated_print_time'], th.get('print_time', 0.))
|
ptime = max(th['estimated_print_time'], th.get('print_time', 0.))
|
||||||
if ptime > seek_time:
|
if ptime > seek_time:
|
||||||
break
|
break
|
||||||
|
for k, v in fmsg["status"].items():
|
||||||
|
start_status.setdefault(k, {}).update(v)
|
||||||
file_position = fmsg['file_position']
|
file_position = fmsg['file_position']
|
||||||
if file_position:
|
if file_position:
|
||||||
self.jdispatch.log_reader.seek(file_position)
|
self.jdispatch.log_reader.seek(file_position)
|
||||||
|
@ -347,6 +402,11 @@ class LogManager:
|
||||||
return self.initial_start_time
|
return self.initial_start_time
|
||||||
def get_start_time(self):
|
def get_start_time(self):
|
||||||
return self.start_time
|
return self.start_time
|
||||||
|
def get_status_tracker(self):
|
||||||
|
if self.status_tracker is None:
|
||||||
|
self.status_tracker = TrackStatus(self, "status", self.start_status)
|
||||||
|
self.jdispatch.add_handler("status", "status")
|
||||||
|
return self.status_tracker
|
||||||
def setup_dataset(self, name):
|
def setup_dataset(self, name):
|
||||||
if name in self.datasets:
|
if name in self.datasets:
|
||||||
return self.datasets[name]
|
return self.datasets[name]
|
||||||
|
@ -357,9 +417,10 @@ class LogManager:
|
||||||
len_pp = len(name_parts) - 1
|
len_pp = len(name_parts) - 1
|
||||||
if len_pp < cls.ParametersMin or len_pp > cls.ParametersMax:
|
if len_pp < cls.ParametersMin or len_pp > cls.ParametersMax:
|
||||||
raise error("Invalid number of parameters for '%s'" % (name,))
|
raise error("Invalid number of parameters for '%s'" % (name,))
|
||||||
|
if cls.SubscriptionIdParts:
|
||||||
subscription_id = ":".join(name_parts[:cls.SubscriptionIdParts])
|
subscription_id = ":".join(name_parts[:cls.SubscriptionIdParts])
|
||||||
if subscription_id not in self.log_subscriptions:
|
if subscription_id not in self.log_subscriptions:
|
||||||
raise error("Dataset '%s' not in capture" % (subscription_id,))
|
raise error("Dataset '%s' not in capture" % (subscription_id,))
|
||||||
self.datasets[name] = hdl = cls(self, name, name_parts)
|
|
||||||
self.jdispatch.add_handler(name, subscription_id)
|
self.jdispatch.add_handler(name, subscription_id)
|
||||||
|
self.datasets[name] = hdl = cls(self, name, name_parts)
|
||||||
return hdl
|
return hdl
|
||||||
|
|
Loading…
Reference in New Issue