2016-04-29 21:27:29 +00:00
|
|
|
from __future__ import print_function, absolute_import
|
2016-03-02 01:13:33 +00:00
|
|
|
import json, time
|
|
|
|
|
2016-04-29 21:27:29 +00:00
|
|
|
class Event:
|
|
|
|
def __init__(self, name, when, **details):
|
|
|
|
# data fields that will be dumped to JSON later
|
|
|
|
self._start = time.time() if when is None else float(when)
|
|
|
|
self._server_sent = None
|
|
|
|
self._stop = None
|
|
|
|
self._name = name
|
|
|
|
self._details = details
|
|
|
|
|
|
|
|
def server_sent(self, when):
|
|
|
|
self._server_sent = when
|
|
|
|
|
|
|
|
def detail(self, **details):
|
|
|
|
self._details.update(details)
|
|
|
|
|
|
|
|
def finish(self, server_sent=None, **details):
|
|
|
|
self._stop = time.time()
|
|
|
|
if server_sent:
|
|
|
|
self.server_sent(server_sent)
|
|
|
|
self.detail(**details)
|
|
|
|
|
|
|
|
def __enter__(self):
|
|
|
|
return self
|
|
|
|
|
|
|
|
def __exit__(self, exc_type, exc_value, exc_tb):
|
|
|
|
if exc_type:
|
|
|
|
# inlineCallbacks uses a special exception (defer._DefGen_Return)
|
|
|
|
# to deliver returnValue(), so if returnValue is used inside our
|
|
|
|
# with: block, we'll mistakenly think it means something broke.
|
|
|
|
# I've moved all returnValue() calls outside the 'with
|
|
|
|
# timing.add()' blocks to avoid this, but if a new one
|
|
|
|
# accidentally pops up, it'll get marked as an error. I used to
|
|
|
|
# catch-and-release _DefGen_Return to avoid this, but removed it
|
|
|
|
# because it requires referencing defer.py's private class
|
|
|
|
self.finish(exception=str(exc_type))
|
|
|
|
else:
|
|
|
|
self.finish()
|
|
|
|
|
2016-03-02 01:13:33 +00:00
|
|
|
class DebugTiming:
|
|
|
|
def __init__(self):
|
2016-04-29 21:27:29 +00:00
|
|
|
self._events = []
|
|
|
|
|
|
|
|
def add(self, name, when=None, **details):
|
|
|
|
ev = Event(name, when, **details)
|
|
|
|
self._events.append(ev)
|
|
|
|
return ev
|
|
|
|
|
2016-03-02 01:13:33 +00:00
|
|
|
def write(self, fn, stderr):
|
|
|
|
with open(fn, "wb") as f:
|
2016-04-29 21:27:29 +00:00
|
|
|
data = [ [e._start, e._server_sent, e._stop, e._name, e._details]
|
|
|
|
for e in self._events ]
|
|
|
|
json.dump(data, f)
|
2016-03-02 01:13:33 +00:00
|
|
|
f.write("\n")
|
|
|
|
print("Timing data written to %s" % fn, file=stderr)
|