summaryrefslogtreecommitdiffstats
path: root/scripts
diff options
context:
space:
mode:
authorYaroslav Brustinov <ybrustin@cisco.com>2016-05-17 02:55:54 +0300
committerIdo Barnea <ibarnea@cisco.com>2016-05-18 19:23:47 +0300
commit3e024d8bdcebdd3100851f28b5724a2ecbfc923a (patch)
tree08c62dc02a86845f1d2ba643bd5fdb85860e79ce /scripts
parent9ba34d62a9faed8fb782062c58c3d746ce56a107 (diff)
add latency panel to TUI
Diffstat (limited to 'scripts')
-rw-r--r--scripts/automation/trex_control_plane/stl/console/trex_tui.py34
-rwxr-xr-xscripts/automation/trex_control_plane/stl/trex_stl_lib/trex_stl_client.py3
-rw-r--r--scripts/automation/trex_control_plane/stl/trex_stl_lib/trex_stl_stats.py124
-rw-r--r--scripts/automation/trex_control_plane/stl/trex_stl_lib/utils/text_opts.py17
4 files changed, 142 insertions, 36 deletions
diff --git a/scripts/automation/trex_control_plane/stl/console/trex_tui.py b/scripts/automation/trex_control_plane/stl/console/trex_tui.py
index 0c3ea8d6..0c69f029 100644
--- a/scripts/automation/trex_control_plane/stl/console/trex_tui.py
+++ b/scripts/automation/trex_control_plane/stl/console/trex_tui.py
@@ -204,6 +204,27 @@ class TrexTUIStreamsStats(TrexTUIPanel):
return ""
+# latency stats
+class TrexTUILatencyStats(TrexTUIPanel):
+ def __init__ (self, mng):
+ super(TrexTUILatencyStats, self).__init__(mng, "lstats")
+ self.key_actions = OrderedDict()
+ self.key_actions['c'] = {'action': self.action_clear, 'legend': 'clear', 'show': True}
+
+
+ def show (self):
+ stats = self.stateless_client._get_formatted_stats(port_id_list = None, stats_mask = trex_stl_stats.LS_COMPAT)
+ # print stats to screen
+ for stat_type, stat_data in stats.items():
+ text_tables.print_table_with_header(stat_data.text_table, stat_type)
+
+ def get_key_actions (self):
+ return self.key_actions
+
+ def action_clear (self):
+ self.stateless_client.latency_stats.clear_stats()
+ return ""
+
# log
class TrexTUILog():
def __init__ (self):
@@ -235,11 +256,13 @@ class TrexTUIPanelManager():
self.panels = {}
self.panels['dashboard'] = TrexTUIDashBoard(self)
self.panels['sstats'] = TrexTUIStreamsStats(self)
+ self.panels['lstats'] = TrexTUILatencyStats(self)
self.key_actions = OrderedDict()
self.key_actions['q'] = {'action': self.action_quit, 'legend': 'quit', 'show': True}
- self.key_actions['g'] = {'action': self.action_show_dash, 'legend': 'dashboard', 'show': True}
- self.key_actions['s'] = {'action': self.action_show_sstats, 'legend': 'streams stats', 'show': True}
+ self.key_actions['d'] = {'action': self.action_show_dash, 'legend': 'dashboard', 'show': True}
+ self.key_actions['s'] = {'action': self.action_show_sstats, 'legend': 'streams', 'show': True}
+ self.key_actions['l'] = {'action': self.action_show_lstats, 'legend': 'latency', 'show': True}
# start with dashboard
self.main_panel = self.panels['dashboard']
@@ -345,12 +368,17 @@ class TrexTUIPanelManager():
return action_show_port_x
-
def action_show_sstats (self):
self.main_panel = self.panels['sstats']
self.init(self.show_log)
return ""
+
+ def action_show_lstats (self):
+ self.main_panel = self.panels['lstats']
+ self.init(self.show_log)
+ return ""
+
# shows a textual top style window
class TrexTUI():
diff --git a/scripts/automation/trex_control_plane/stl/trex_stl_lib/trex_stl_client.py b/scripts/automation/trex_control_plane/stl/trex_stl_lib/trex_stl_client.py
index 88c64e3c..195ba5ff 100755
--- a/scripts/automation/trex_control_plane/stl/trex_stl_lib/trex_stl_client.py
+++ b/scripts/automation/trex_control_plane/stl/trex_stl_lib/trex_stl_client.py
@@ -557,7 +557,8 @@ class STLClient(object):
self.stats_generator = trex_stl_stats.CTRexInfoGenerator(self.global_stats,
self.ports,
- self.flow_stats)
+ self.flow_stats,
+ self.latency_stats)
diff --git a/scripts/automation/trex_control_plane/stl/trex_stl_lib/trex_stl_stats.py b/scripts/automation/trex_control_plane/stl/trex_stl_lib/trex_stl_stats.py
index a65ef9e0..173832a1 100644
--- a/scripts/automation/trex_control_plane/stl/trex_stl_lib/trex_stl_stats.py
+++ b/scripts/automation/trex_control_plane/stl/trex_stl_lib/trex_stl_stats.py
@@ -20,11 +20,13 @@ PORT_STATS = 'p'
PORT_GRAPH = 'pg'
PORT_STATUS = 'ps'
STREAMS_STATS = 's'
+LATENCY_STATS = 'l'
-ALL_STATS_OPTS = [GLOBAL_STATS, PORT_STATS, PORT_STATUS, STREAMS_STATS, PORT_GRAPH]
+ALL_STATS_OPTS = [GLOBAL_STATS, PORT_STATS, PORT_STATUS, STREAMS_STATS, LATENCY_STATS, PORT_GRAPH]
COMPACT = [GLOBAL_STATS, PORT_STATS]
GRAPH_PORT_COMPACT = [GLOBAL_STATS, PORT_GRAPH]
-SS_COMPAT = [GLOBAL_STATS, STREAMS_STATS]
+SS_COMPAT = [GLOBAL_STATS, STREAMS_STATS] # stream stats
+LS_COMPAT = [GLOBAL_STATS, LATENCY_STATS] # latency stats
ExportableStats = namedtuple('ExportableStats', ['raw_data', 'text_table'])
@@ -128,10 +130,11 @@ class CTRexInfoGenerator(object):
STLClient and the ports.
"""
- def __init__(self, global_stats_ref, ports_dict_ref, rx_stats_ref):
+ def __init__(self, global_stats_ref, ports_dict_ref, rx_stats_ref, latency_stats_ref):
self._global_stats = global_stats_ref
self._ports_dict = ports_dict_ref
self._rx_stats_ref = rx_stats_ref
+ self._latency_stats_ref = latency_stats_ref
def generate_single_statistic(self, port_id_list, statistic_type):
if statistic_type == GLOBAL_STATS:
@@ -149,6 +152,9 @@ class CTRexInfoGenerator(object):
elif statistic_type == STREAMS_STATS:
return self._generate_streams_stats()
+ elif statistic_type == LATENCY_STATS:
+ return self._generate_latency_stats()
+
else:
# ignore by returning empty object
return {}
@@ -200,6 +206,23 @@ class CTRexInfoGenerator(object):
return {"streams_statistics": ExportableStats(sstats_data, stats_table)}
+ def _generate_latency_stats (self):
+ streams_keys, lstats_data = self._latency_stats_ref.generate_stats()
+ stream_count = len(streams_keys)
+
+ stats_table = text_tables.TRexTextTable()
+ stats_table.set_cols_align(["l"] + ["r"] * stream_count)
+ stats_table.set_cols_width([12] + [14] * stream_count)
+ stats_table.set_cols_dtype(['t'] + ['t'] * stream_count)
+ stats_table.add_rows([[k] + v
+ for k, v in lstats_data.items()],
+ header=False)
+
+ header = ["PG ID"] + [key for key in streams_keys]
+ stats_table.header(header)
+
+ return {"latency_statistics": ExportableStats(lstats_data, stats_table)}
+
@staticmethod
def _get_rational_block_char(value, range_start, interval):
# in Konsole, utf-8 is sometimes printed with artifacts, return ascii for now
@@ -422,8 +445,8 @@ class CTRexStats(object):
self.reference_stats = {}
self.latest_stats = {}
self.last_update_ts = time.time()
- self.history = deque(maxlen = 47)
- self.lock = threading.Lock()
+ self.__history = deque(maxlen = 47)
+ self.lock = threading.RLock()
self.has_baseline = False
######## abstract methods ##########
@@ -460,13 +483,24 @@ class CTRexStats(object):
self.has_baseline = True
# save history
+ self.history.append(self.latest_stats)
+
+
+ @property
+ def history(self):
+ with self.lock:
+ return self.__history
+
+ @history.setter
+ def history(self, val):
with self.lock:
- self.history.append(self.latest_stats)
+ self.__history = val
def clear_stats(self):
self.reference_stats = copy.deepcopy(self.latest_stats)
self.history.clear()
+ self.history.append(self.latest_stats)
def invalidate (self):
@@ -527,8 +561,7 @@ class CTRexStats(object):
return 0
# must lock, deque is not thread-safe for iteration
- with self.lock:
- field_samples = [sample[field] for sample in list(self.history)[-5:]]
+ field_samples = [sample[field] for sample in list(self.history)[-5:]]
if use_raw:
return calculate_diff_raw(field_samples)
@@ -702,13 +735,11 @@ class CPortStats(CTRexStats):
else:
self.__merge_dicts(self.reference_stats, x.reference_stats)
- # history - should be traverse with a lock
- with self.lock, x.lock:
- if not self.history:
- self.history = copy.deepcopy(x.history)
- else:
- for h1, h2 in zip(self.history, x.history):
- self.__merge_dicts(h1, h2)
+ if not self.history:
+ self.history = copy.deepcopy(x.history)
+ else:
+ for h1, h2 in zip(self.history, x.history):
+ self.__merge_dicts(h1, h2)
return self
@@ -839,31 +870,70 @@ class CLatencyStats(CTRexStats):
def get_stats (self):
return self.latest_stats
- def process_snapshot (self, current):
+
+ def _update(self, snapshot):
+ if not snapshot:
+ return
output = {}
+ #print snapshot
# we care only about the current active keys
- pg_ids = list(filter(is_intable, current.keys()))
+ pg_ids = list(filter(is_intable, snapshot.keys()))
for pg_id in pg_ids:
- current_pg = current.get(pg_id)
+ current_pg = snapshot.get(pg_id)
int_pg_id = int(pg_id)
output[int_pg_id] = {}
for field in ['err_cntrs', 'jitter', 'latency']:
output[int_pg_id][field] = current_pg[field]
- return output
-
- def update (self, snapshot, baseline):
- # generate a new snapshot
- if (snapshot is not None):
- new_snapshot = self.process_snapshot(snapshot)
- else:
- return
- self.latest_stats = new_snapshot
+ self.latest_stats = output
return True
+ def generate_stats (self):
+ latency_window_size = 10
+
+ # for TUI - maximum 5
+ pg_ids = list(filter(is_intable, self.latest_stats.keys()))[:5]
+ cnt = len(pg_ids)
+ formatted_stats = OrderedDict([
+ ('Max latency', []),
+ ('Avg latency', []),
+ ('Min latency', []),
+ ('Last 0.5s', []),
+ ] + [
+ ('Last-%s' % i, []) for i in range(1, latency_window_size)
+ ] + [
+ ('---', [''] * cnt),
+ ('Jitter', []),
+ ('----', [''] * cnt),
+ ('Out of order', []),
+ ])
+
+ history = self.history # get it once with the lock, not per each index
+ for pg_id in pg_ids:
+ latency_dict = self.get([pg_id, 'latency'])
+
+ formatted_stats['Max latency'].append('%s usec' % self.get([pg_id, 'latency', 'max_usec']))
+ formatted_stats['Avg latency'].append('%s usec' % self.get([pg_id, 'latency', 's_avg']))
+ formatted_stats['Min latency'].append('%s usec' % self.get([pg_id, 'latency', 'min_usec']))
+ formatted_stats['Last 0.5s'].append('%s usec' % int(self.get([pg_id, 'latency', 'last_max'])))
+ for i in range(1, latency_window_size):
+ val = '%s usec' % int(history[-i - 1][pg_id]['latency']['last_max']) if len(history) > i else ''
+ formatted_stats['Last-%s' % i].append(val)
+ formatted_stats['Jitter'].append('%g usec' % round(self.get([pg_id, 'jitter']), 1))
+
+ #formatted_stats['Dropped'].append(format_num(self.get([pg_id, 'err_cntrs', 'dropped'], format = True, suffix = "pkts"))
+ #compact = False,
+ #opts = 'green' if self.get([pg_id, 'err_cntrs', 'dropped']) == 0 else 'red')
+ #)
+ formatted_stats['Out of order'].append(self.get([pg_id, 'err_cntrs', 'out_of_order'], format = True, suffix = "pkts"))
+
+ return pg_ids, formatted_stats
+
+
+
# RX stats objects - COMPLEX :-(
class CRxStats(CTRexStats):
def __init__(self, ports):
diff --git a/scripts/automation/trex_control_plane/stl/trex_stl_lib/utils/text_opts.py b/scripts/automation/trex_control_plane/stl/trex_stl_lib/utils/text_opts.py
index 7e0bf9e4..72be7c29 100644
--- a/scripts/automation/trex_control_plane/stl/trex_stl_lib/utils/text_opts.py
+++ b/scripts/automation/trex_control_plane/stl/trex_stl_lib/utils/text_opts.py
@@ -36,11 +36,18 @@ def format_num (size, suffix = "", compact = True, opts = ()):
u = ''
if compact:
- for unit in ['','K','M','G','T','P']:
- if abs(size) < 1000.0:
- u = unit
- break
- size /= 1000.0
+ if 0 < abs(size) < 1:
+ for unit in ['m','u','n','p']:
+ size *= 1000
+ if abs(size) >= 1:
+ u = unit
+ break
+ else:
+ for unit in ['','K','M','G','T','P']:
+ if abs(size) < 1000.0:
+ u = unit
+ break
+ size /= 1000.0
if isinstance(size, float):
txt = "%3.2f" % (size)