diff --git a/src/wormhole/servers/cmd_usage.py b/src/wormhole/servers/cmd_usage.py index 202a75e..463e9ba 100644 --- a/src/wormhole/servers/cmd_usage.py +++ b/src/wormhole/servers/cmd_usage.py @@ -13,16 +13,44 @@ def abbrev(t): return "%.1fms" % (t*1e3) return "%.1fus" % (t*1e6) +def abbreviate_space(s, SI=True): + if s is None: + return "-" + if SI: + U = 1000.0 + isuffix = "B" + else: + U = 1024.0 + isuffix = "iB" + def r(count, suffix): + return "%.2f %s%s" % (count, suffix, isuffix) + + if s < 1024: # 1000-1023 get emitted as bytes, even in SI mode + return "%d B" % s + if s < U*U: + return r(s/U, "k") + if s < U*U*U: + return r(s/(U*U), "M") + if s < U*U*U*U: + return r(s/(U*U*U), "G") + if s < U*U*U*U*U: + return r(s/(U*U*U*U), "T") + if s < U*U*U*U*U*U: + return r(s/(U*U*U*U*U), "P") + return r(s/(U*U*U*U*U*U), "E") + def print_event(event): - started, result, waiting_time, total_time = event + event_type, started, result, total_bytes, waiting_time, total_time = event followthrough = None if waiting_time and total_time: followthrough = total_time - waiting_time - print("%s: %-6s total=%7s wait=%7s ft=%7s" % - (time.ctime(started), result, + print(" %16s: total=%7s wait=%7s ft=%7s size=%s (%s)" % + ("%s-%s" % (event_type, result), abbrev(total_time), abbrev(waiting_time), abbrev(followthrough), + abbreviate_space(total_bytes), + time.ctime(started), )) def show_usage(args): @@ -30,31 +58,58 @@ def show_usage(args): raise UsageError("cannot find relay.sqlite, please run from the server directory") oldest = None newest = None - counters = defaultdict(int) + rendezvous_counters = defaultdict(int) + transit_counters = defaultdict(int) + total_transit_bytes = 0 db = get_db("relay.sqlite") - c = db.execute("SELECT * FROM `usage` WHERE `type`=?" + c = db.execute("SELECT * FROM `usage`" " ORDER BY `started` ASC LIMIT ?", - (u"rendezvous", args.n)) + (args.n,)) for row in c.fetchall(): + if row["type"] == u"rendezvous": + counters = rendezvous_counters + elif row["type"] == u"transit": + counters = transit_counters + total_transit_bytes += row["total_bytes"] + else: + continue counters["total"] += 1 counters[row["result"]] += 1 if oldest is None or row["started"] < oldest: oldest = row["started"] if newest is None or row["started"] > newest: newest = row["started"] - event = (row["started"], row["result"], - row["waiting_time"], row["total_time"]) + event = (row["type"], row["started"], row["result"], + row["total_bytes"], row["waiting_time"], row["total_time"]) print_event(event) - total = counters["total"] - if total: + if rendezvous_counters["total"] or transit_counters["total"]: + print("---") print("(most recent started %s ago)" % abbrev(time.time() - newest)) + if rendezvous_counters["total"]: + print("rendezvous events:") + counters = rendezvous_counters elapsed = time.time() - oldest - print("%d events in %s (%.2f per hour)" % (total, abbrev(elapsed), - (3600 * total / elapsed))) - print(", ".join(["%s=%d (%d%%)" % - (k, counters[k], (100.0 * counters[k] / total)) - for k in sorted(counters) - if k != "total"])) + total = counters["total"] + print(" %d events in %s (%.2f per hour)" % (total, abbrev(elapsed), + (3600 * total / elapsed))) + print("", ", ".join(["%s=%d (%d%%)" % + (k, counters[k], (100.0 * counters[k] / total)) + for k in sorted(counters) + if k != "total"])) + if transit_counters["total"]: + print("transit events:") + counters = transit_counters + elapsed = time.time() - oldest + total = counters["total"] + print(" %d events in %s (%.2f per hour)" % (total, abbrev(elapsed), + (3600 * total / elapsed))) + rate = total_transit_bytes / elapsed + print(" %s total bytes, %sps" % (abbreviate_space(total_transit_bytes), + abbreviate_space(rate))) + print("", ", ".join(["%s=%d (%d%%)" % + (k, counters[k], (100.0 * counters[k] / total)) + for k in sorted(counters) + if k != "total"])) return 0 def tail_usage(args): @@ -64,16 +119,20 @@ def tail_usage(args): # we don't seem to have unique row IDs, so this is an inaccurate and # inefficient hack seen = set() - while True: - old = time.time() - 2*60*60 - c = db.execute("SELECT * FROM `usage`" - " WHERE `type`=? AND `started` > ?" - " ORDER BY `started` ASC", (u"rendezvous", old)) - for row in c.fetchall(): - event = (row["started"], row["result"], - row["waiting_time"], row["total_time"]) - if event not in seen: - print_event(event) - seen.add(event) - time.sleep(2) + try: + while True: + old = time.time() - 2*60*60 + c = db.execute("SELECT * FROM `usage`" + " WHERE `started` > ?" + " ORDER BY `started` ASC", (old,)) + for row in c.fetchall(): + event = (row["type"], row["started"], row["result"], + row["total_bytes"], row["waiting_time"], + row["total_time"]) + if event not in seen: + print_event(event) + seen.add(event) + time.sleep(2) + except KeyboardInterrupt: + return 0 return 0