summaryrefslogtreecommitdiffstats
path: root/scripts/automation
diff options
context:
space:
mode:
Diffstat (limited to 'scripts/automation')
-rw-r--r--scripts/automation/trex_control_plane/stl/console/trex_tui.py6
-rwxr-xr-xscripts/automation/trex_control_plane/stl/trex_stl_lib/trex_stl_client.py12
-rw-r--r--scripts/automation/trex_control_plane/stl/trex_stl_lib/trex_stl_stats.py333
-rw-r--r--scripts/automation/trex_control_plane/stl/trex_stl_lib/utils/text_opts.py17
4 files changed, 181 insertions, 187 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 0c69f029..95dc654b 100644
--- a/scripts/automation/trex_control_plane/stl/console/trex_tui.py
+++ b/scripts/automation/trex_control_plane/stl/console/trex_tui.py
@@ -216,7 +216,11 @@ class TrexTUILatencyStats(TrexTUIPanel):
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)
+ if stat_type == 'latency_statistics':
+ untouched_header = ' (usec)'
+ else:
+ untouched_header = ''
+ text_tables.print_table_with_header(stat_data.text_table, stat_type, untouched_header = untouched_header)
def get_key_actions (self):
return self.key_actions
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 195ba5ff..fa24574f 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
@@ -857,7 +857,7 @@ class STLClient(object):
# clear stats
- def __clear_stats(self, port_id_list, clear_global, clear_flow_stats):
+ def __clear_stats(self, port_id_list, clear_global, clear_flow_stats, clear_latency_stats):
# we must be sync with the server
self.async_client.barrier()
@@ -871,6 +871,9 @@ class STLClient(object):
if clear_flow_stats:
self.flow_stats.clear_stats()
+ if clear_latency_stats:
+ self.latency_stats.clear_stats()
+
self.logger.log_cmd("Clearing stats on port(s) {0}:".format(port_id_list))
return RC
@@ -2071,7 +2074,7 @@ class STLClient(object):
@__api_check(False)
- def clear_stats (self, ports = None, clear_global = True, clear_flow_stats = True):
+ def clear_stats (self, ports = None, clear_global = True, clear_flow_stats = True, clear_latency_stats = True):
"""
Clear stats on port(s)
@@ -2085,6 +2088,9 @@ class STLClient(object):
clear_flow_stats : bool
Clear the flow stats
+ clear_latency_stats : bool
+ Clear the latency stats
+
:raises:
+ :exc:`STLError`
@@ -2097,7 +2103,7 @@ class STLClient(object):
if not type(clear_global) is bool:
raise STLArgumentError('clear_global', clear_global)
- rc = self.__clear_stats(ports, clear_global, clear_flow_stats)
+ rc = self.__clear_stats(ports, clear_global, clear_flow_stats, clear_latency_stats)
if not rc:
raise STLError(rc)
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 7c5a23cb..4ead255d 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
@@ -2,7 +2,7 @@
from .utils import text_tables
from .utils.text_opts import format_text, format_threshold, format_num
-from .trex_stl_types import StatNotAvailable
+from .trex_stl_types import StatNotAvailable, is_integer
from collections import namedtuple, OrderedDict, deque
import sys
@@ -33,6 +33,12 @@ ExportableStats = namedtuple('ExportableStats', ['raw_data', 'text_table'])
def round_float (f):
return float("%.2f" % f) if type(f) is float else f
+def try_int(i):
+ try:
+ return int(i)
+ except:
+ return i
+
# deep mrege of dicts dst = src + dst
def deep_merge_dicts (dst, src):
for k, v in src.items():
@@ -175,7 +181,53 @@ class CTRexInfoGenerator(object):
return return_data
def _generate_global_stats(self):
- stats_data = self._global_stats.generate_stats()
+ global_stats = self._global_stats
+ #if is_integer(global_stats.max_global_latency):
+ # max_global_latency_str = ', Max latency: %s' % format_num(global_stats.max_global_latency,
+ # suffix = 'usec',
+ # compact = False,
+ # opts = 'green' if global_stats.max_global_latency < 1000 else 'red')
+ #else:
+ # max_global_latency_str = ''
+ stats_data = OrderedDict([("connection", "{host}, Port {port}".format(host=global_stats.connection_info.get("server"),
+ port=global_stats.connection_info.get("sync_port"))),
+ ("version", "{ver}, UUID: {uuid}".format(ver=global_stats.server_version.get("version", "N/A"),
+ uuid="N/A")),
+
+ ("cpu_util", "{0}% {1}".format( format_threshold(round_float(global_stats.get("m_cpu_util")), [85, 100], [0, 85]),
+ global_stats.get_trend_gui("m_cpu_util", use_raw = True))),
+
+ ("rx_cpu_util", "{0}% {1}".format( format_threshold(round_float(global_stats.get("m_rx_cpu_util")), [85, 100], [0, 85]),
+ global_stats.get_trend_gui("m_rx_cpu_util", use_raw = True))),
+
+ (" ", ""),
+
+ ("total_tx_L2", "{0} {1}".format( global_stats.get("m_tx_bps", format=True, suffix="b/sec"),
+ global_stats.get_trend_gui("m_tx_bps"))),
+
+ ("total_tx_L1", "{0} {1}".format( global_stats.get("m_tx_bps_L1", format=True, suffix="b/sec"),
+ global_stats.get_trend_gui("m_tx_bps_L1"))),
+
+ ("total_rx", "{0} {1}".format( global_stats.get("m_rx_bps", format=True, suffix="b/sec"),
+ global_stats.get_trend_gui("m_rx_bps"))),
+
+ ("total_pps", "{0} {1}".format( global_stats.get("m_tx_pps", format=True, suffix="pkt/sec"),
+ global_stats.get_trend_gui("m_tx_pps"))),
+
+ (" ", ""),
+
+ ("drop_rate", "{0}".format( format_num(global_stats.get("m_rx_drop_bps"),
+ suffix = 'b/sec',
+ opts = 'green' if (global_stats.get("m_rx_drop_bps")== 0) else 'red'),
+ )),
+
+ ("queue_full", "{0}".format( format_num(global_stats.get_rel("m_total_queue_full"),
+ suffix = 'pkts',
+ compact = False,
+ opts = 'green' if (global_stats.get_rel("m_total_queue_full")== 0) else 'red'))),
+
+ ]
+ )
# build table representation
stats_table = text_tables.TRexTextInfo()
@@ -188,9 +240,51 @@ class CTRexInfoGenerator(object):
return {"global_statistics": ExportableStats(stats_data, stats_table)}
def _generate_streams_stats (self):
-
- streams_keys, sstats_data = self._rx_stats_ref.generate_stats()
- stream_count = len(streams_keys)
+ flow_stats = self._rx_stats_ref
+ # for TUI - maximum 4
+ pg_ids = list(filter(is_intable, flow_stats.latest_stats.keys()))[:4]
+ stream_count = len(pg_ids)
+
+ sstats_data = OrderedDict([ ('Tx pps', []),
+ ('Tx bps L2', []),
+ ('Tx bps L1', []),
+ ('---', [''] * stream_count),
+ ('Rx pps', []),
+ ('Rx bps', []),
+ ('----', [''] * stream_count),
+ ('opackets', []),
+ ('ipackets', []),
+ ('obytes', []),
+ ('ibytes', []),
+ ('-----', [''] * stream_count),
+ ('tx_pkts', []),
+ ('rx_pkts', []),
+ ('tx_bytes', []),
+ ('rx_bytes', [])
+ ])
+
+
+
+ # maximum 4
+ for pg_id in pg_ids:
+
+ sstats_data['Tx pps'].append(flow_stats.get([pg_id, 'tx_pps_lpf', 'total'], format = True, suffix = "pps"))
+ sstats_data['Tx bps L2'].append(flow_stats.get([pg_id, 'tx_bps_lpf', 'total'], format = True, suffix = "bps"))
+
+ sstats_data['Tx bps L1'].append(flow_stats.get([pg_id, 'tx_bps_L1_lpf', 'total'], format = True, suffix = "bps"))
+
+ sstats_data['Rx pps'].append(flow_stats.get([pg_id, 'rx_pps_lpf', 'total'], format = True, suffix = "pps"))
+ sstats_data['Rx bps'].append(flow_stats.get([pg_id, 'rx_bps_lpf', 'total'], format = True, suffix = "bps"))
+
+ sstats_data['opackets'].append(flow_stats.get_rel([pg_id, 'tx_pkts', 'total']))
+ sstats_data['ipackets'].append(flow_stats.get_rel([pg_id, 'rx_pkts', 'total']))
+ sstats_data['obytes'].append(flow_stats.get_rel([pg_id, 'tx_bytes', 'total']))
+ sstats_data['ibytes'].append(flow_stats.get_rel([pg_id, 'rx_bytes', 'total']))
+ sstats_data['tx_bytes'].append(flow_stats.get_rel([pg_id, 'tx_bytes', 'total'], format = True, suffix = "B"))
+ sstats_data['rx_bytes'].append(flow_stats.get_rel([pg_id, 'rx_bytes', 'total'], format = True, suffix = "B"))
+ sstats_data['tx_pkts'].append(flow_stats.get_rel([pg_id, 'tx_pkts', 'total'], format = True, suffix = "pkts"))
+ sstats_data['rx_pkts'].append(flow_stats.get_rel([pg_id, 'rx_pkts', 'total'], format = True, suffix = "pkts"))
+
stats_table = text_tables.TRexTextTable()
stats_table.set_cols_align(["l"] + ["r"] * stream_count)
@@ -201,14 +295,54 @@ class CTRexInfoGenerator(object):
for k, v in sstats_data.items()],
header=False)
- header = ["PG ID"] + [key for key in streams_keys]
+ header = ["PG ID"] + [key for key in pg_ids]
stats_table.header(header)
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)
+ def _generate_latency_stats(self):
+ lat_stats = self._latency_stats_ref
+ latency_window_size = 10
+
+ # for TUI - maximum 5
+ pg_ids = list(filter(is_intable, lat_stats.latest_stats.keys()))[:5]
+ stream_count = len(pg_ids)
+ lstats_data = OrderedDict([#('TX pkts', []),
+ #('RX pkts', []),
+ ('Max latency', []),
+ ('Avg latency', []),
+ ('-- Window --', [''] * stream_count),
+ ('Last (max)', []),
+ ] + [('Last-%s' % i, []) for i in range(1, latency_window_size)] + [
+ ('---', [''] * stream_count),
+ ('Jitter', []),
+ ('----', [''] * stream_count),
+ ('Errors', []),
+ ])
+
+ with lat_stats.lock:
+ history = [x for x in lat_stats.history]
+ flow_stats = self._rx_stats_ref.get_stats()
+ for pg_id in pg_ids:
+ #lstats_data['TX pkts'].append(flow_stats[pg_id]['tx_pkts']['total'] if pg_id in flow_stats else '')
+ #lstats_data['RX pkts'].append(flow_stats[pg_id]['rx_pkts']['total'] if pg_id in flow_stats else '')
+ lstats_data['Avg latency'].append(try_int(lat_stats.get([pg_id, 'latency', 'average'])))
+ lstats_data['Max latency'].append(try_int(lat_stats.get([pg_id, 'latency', 'total_max'])))
+ lstats_data['Last (max)'].append(try_int(lat_stats.get([pg_id, 'latency', 'last_max'])))
+ for i in range(1, latency_window_size):
+ val = history[-i - 1].get(pg_id, {}).get('latency', {}).get('last_max', '') if len(history) > i else ''
+ lstats_data['Last-%s' % i].append(try_int(val))
+ lstats_data['Jitter'].append(try_int(lat_stats.get([pg_id, 'latency', 'jitter'])))
+ errors = 0
+ seq_too_low = lat_stats.get([pg_id, 'err_cntrs', 'seq_too_low'])
+ if is_integer(seq_too_low):
+ errors += seq_too_low
+ seq_too_high = lat_stats.get([pg_id, 'err_cntrs', 'seq_too_high'])
+ if is_integer(seq_too_high):
+ errors += seq_too_high
+ lstats_data['Errors'].append(format_num(errors,
+ opts = 'green' if errors == 0 else 'red'))
+
stats_table = text_tables.TRexTextTable()
stats_table.set_cols_align(["l"] + ["r"] * stream_count)
@@ -218,7 +352,7 @@ class CTRexInfoGenerator(object):
for k, v in lstats_data.items()],
header=False)
- header = ["PG ID"] + [key for key in streams_keys]
+ header = ["PG ID"] + [key for key in pg_ids]
stats_table.header(header)
return {"latency_statistics": ExportableStats(lstats_data, stats_table)}
@@ -347,7 +481,7 @@ class CTRexInfoGenerator(object):
("status", []),
("promiscuous", []),
("--", []),
- ("HW src mac", []),
+ ("HW src mac", []),
("SW src mac", []),
("SW dst mac", []),
("---", []),
@@ -445,8 +579,8 @@ class CTRexStats(object):
self.reference_stats = {}
self.latest_stats = {}
self.last_update_ts = time.time()
- self.__history = deque(maxlen = 47)
- self.lock = threading.RLock()
+ self.history = deque(maxlen = 47)
+ self.lock = threading.Lock()
self.has_baseline = False
######## abstract methods ##########
@@ -483,24 +617,13 @@ 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 = val
+ self.history.append(self.latest_stats)
def clear_stats(self):
self.reference_stats = copy.deepcopy(self.latest_stats)
self.history.clear()
- self.history.append(self.latest_stats)
def invalidate (self):
@@ -561,7 +684,8 @@ class CTRexStats(object):
return 0
# must lock, deque is not thread-safe for iteration
- field_samples = [sample[field] for sample in list(self.history)[-5:]]
+ with self.lock:
+ field_samples = [sample[field] for sample in list(self.history)[-5:]]
if use_raw:
return calculate_diff_raw(field_samples)
@@ -663,46 +787,6 @@ class CGlobalStats(CTRexStats):
return True
- def generate_stats(self):
- return OrderedDict([("connection", "{host}, Port {port}".format(host=self.connection_info.get("server"),
- port=self.connection_info.get("sync_port"))),
- ("version", "{ver}, UUID: {uuid}".format(ver=self.server_version.get("version", "N/A"),
- uuid="N/A")),
-
- ("cpu_util", "{0}% {1}".format( format_threshold(round_float(self.get("m_cpu_util")), [85, 100], [0, 85]),
- self.get_trend_gui("m_cpu_util", use_raw = True))),
-
- ("rx_cpu_util", "{0}% {1}".format( format_threshold(round_float(self.get("m_rx_cpu_util")), [85, 100], [0, 85]),
- self.get_trend_gui("m_rx_cpu_util", use_raw = True))),
-
- (" ", ""),
-
- ("total_tx_L2", "{0} {1}".format( self.get("m_tx_bps", format=True, suffix="b/sec"),
- self.get_trend_gui("m_tx_bps"))),
-
- ("total_tx_L1", "{0} {1}".format( self.get("m_tx_bps_L1", format=True, suffix="b/sec"),
- self.get_trend_gui("m_tx_bps_L1"))),
-
- ("total_rx", "{0} {1}".format( self.get("m_rx_bps", format=True, suffix="b/sec"),
- self.get_trend_gui("m_rx_bps"))),
-
- ("total_pps", "{0} {1}".format( self.get("m_tx_pps", format=True, suffix="pkt/sec"),
- self.get_trend_gui("m_tx_pps"))),
-
- (" ", ""),
-
- ("drop_rate", "{0}".format( format_num(self.get("m_rx_drop_bps"),
- suffix = 'b/sec',
- opts = 'green' if (self.get("m_rx_drop_bps")== 0) else 'red'))),
-
- ("queue_full", "{0}".format( format_num(self.get_rel("m_total_queue_full"),
- suffix = 'pkts',
- compact = False,
- opts = 'green' if (self.get_rel("m_total_queue_full")== 0) else 'red'))),
-
- ]
- )
-
class CPortStats(CTRexStats):
def __init__(self, port_obj):
@@ -735,11 +819,13 @@ class CPortStats(CTRexStats):
else:
self.__merge_dicts(self.reference_stats, x.reference_stats)
- 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)
+ # 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)
return self
@@ -866,17 +952,16 @@ class CLatencyStats(CTRexStats):
def __init__(self, ports):
super(CLatencyStats, self).__init__()
+
# for API
def get_stats (self):
- ret = copy.deepcopy(self.latest_stats)
- return ret
+ return copy.deepcopy(self.latest_stats)
def _update(self, snapshot):
- if not snapshot:
- return
+ if snapshot is None:
+ snapshot = {}
output = {}
- #print snapshot
# we care only about the current active keys
pg_ids = list(filter(is_intable, snapshot.keys()))
@@ -896,54 +981,10 @@ class CLatencyStats(CTRexStats):
output[int_pg_id]['latency']['histogram'] = current_pg['latency']['h']['histogram']
zero_count = current_pg['latency']['h']['cnt'] - current_pg['latency']['h']['high_cnt']
output[int_pg_id]['latency']['histogram'].append({'key':0, 'val':zero_count})
-
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, 'latency', '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):
@@ -1152,7 +1193,7 @@ class CRxStats(CTRexStats):
return True
-
+
# for API
@@ -1185,56 +1226,6 @@ class CRxStats(CTRexStats):
return stats
- # for Console
- def generate_stats (self):
-
- # for TUI - maximum 4
- pg_ids = list(filter(is_intable, self.latest_stats.keys()))[:4]
- cnt = len(pg_ids)
-
- formatted_stats = OrderedDict([ ('Tx pps', []),
- ('Tx bps L2', []),
- ('Tx bps L1', []),
- ('---', [''] * cnt),
- ('Rx pps', []),
- ('Rx bps', []),
- ('----', [''] * cnt),
- ('opackets', []),
- ('ipackets', []),
- ('obytes', []),
- ('ibytes', []),
- ('-----', [''] * cnt),
- ('tx_pkts', []),
- ('rx_pkts', []),
- ('tx_bytes', []),
- ('rx_bytes', [])
- ])
-
-
-
- # maximum 4
- for pg_id in pg_ids:
-
- formatted_stats['Tx pps'].append(self.get([pg_id, 'tx_pps_lpf', 'total'], format = True, suffix = "pps"))
- formatted_stats['Tx bps L2'].append(self.get([pg_id, 'tx_bps_lpf', 'total'], format = True, suffix = "bps"))
-
- formatted_stats['Tx bps L1'].append(self.get([pg_id, 'tx_bps_L1_lpf', 'total'], format = True, suffix = "bps"))
-
- formatted_stats['Rx pps'].append(self.get([pg_id, 'rx_pps_lpf', 'total'], format = True, suffix = "pps"))
- formatted_stats['Rx bps'].append(self.get([pg_id, 'rx_bps_lpf', 'total'], format = True, suffix = "bps"))
-
- formatted_stats['opackets'].append(self.get_rel([pg_id, 'tx_pkts', 'total']))
- formatted_stats['ipackets'].append(self.get_rel([pg_id, 'rx_pkts', 'total']))
- formatted_stats['obytes'].append(self.get_rel([pg_id, 'tx_bytes', 'total']))
- formatted_stats['ibytes'].append(self.get_rel([pg_id, 'rx_bytes', 'total']))
- formatted_stats['tx_bytes'].append(self.get_rel([pg_id, 'tx_bytes', 'total'], format = True, suffix = "B"))
- formatted_stats['rx_bytes'].append(self.get_rel([pg_id, 'rx_bytes', 'total'], format = True, suffix = "B"))
- formatted_stats['tx_pkts'].append(self.get_rel([pg_id, 'tx_pkts', 'total'], format = True, suffix = "pkts"))
- formatted_stats['rx_pkts'].append(self.get_rel([pg_id, 'rx_pkts', 'total'], format = True, suffix = "pkts"))
-
-
-
- return pg_ids, formatted_stats
if __name__ == "__main__":
pass
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 72be7c29..7e0bf9e4 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,18 +36,11 @@ def format_num (size, suffix = "", compact = True, opts = ()):
u = ''
if compact:
- 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
+ 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)