#!/usr/bin/env python
# cardinal_pythonlib/timing.py
"""
===============================================================================
Original code copyright (C) 2009-2022 Rudolf Cardinal (rudolf@pobox.com).
This file is part of cardinal_pythonlib.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
https://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
===============================================================================
**Timers for performance tuning.**
"""
from collections import OrderedDict
import datetime
from cardinal_pythonlib.datetimefunc import get_now_utc_pendulum
from cardinal_pythonlib.logs import get_brace_style_log_with_null_handler
log = get_brace_style_log_with_null_handler(__name__)
[docs]class MultiTimer(object):
"""
Mutually exclusive timing of a set of events.
Maintains a start count and times for a set of named timers.
See example in :class:`MultiTimerContext`.
"""
def __init__(self, start: bool = True) -> None:
"""
Args:
start: start the timer immediately?
"""
self._timing = start
self._overallstart = get_now_utc_pendulum()
self._starttimes = OrderedDict() # name: start time
self._totaldurations = OrderedDict() # name: duration
self._count = OrderedDict() # name: count
self._stack = [] # list of names
[docs] def reset(self) -> None:
"""
Reset the timers.
"""
self._overallstart = get_now_utc_pendulum()
self._starttimes.clear()
self._totaldurations.clear()
self._count.clear()
self._stack.clear()
[docs] def set_timing(self, timing: bool, reset: bool = False) -> None:
"""
Manually set the ``timing`` parameter, and optionally reset the timers.
Args:
timing: should we be timing?
reset: reset the timers?
"""
self._timing = timing
if reset:
self.reset()
[docs] def start(self, name: str, increment_count: bool = True) -> None:
"""
Start a named timer.
Args:
name: name of the timer
increment_count: increment the start count for this timer
"""
if not self._timing:
return
now = get_now_utc_pendulum()
# If we were already timing something else, pause that.
if self._stack:
last = self._stack[-1]
self._totaldurations[last] += now - self._starttimes[last]
# Start timing our new thing
if name not in self._starttimes:
self._totaldurations[name] = datetime.timedelta()
self._count[name] = 0
self._starttimes[name] = now
if increment_count:
self._count[name] += 1
self._stack.append(name)
[docs] def stop(self, name: str) -> None:
"""
Stop a named timer.
Args:
name: timer to stop
"""
if not self._timing:
return
now = get_now_utc_pendulum()
# Validity check
if not self._stack:
raise AssertionError("MultiTimer.stop() when nothing running")
if self._stack[-1] != name:
raise AssertionError(
f"MultiTimer.stop({name!r}) when "
f"{self._stack[-1]!r} is running"
)
# Finish what we were asked to
self._totaldurations[name] += now - self._starttimes[name]
self._stack.pop()
# Now, if we were timing something else before we started "name",
# resume...
if self._stack:
last = self._stack[-1]
self._starttimes[last] = now
[docs] def report(self) -> None:
"""
Finish and report to the log.
"""
while self._stack:
self.stop(self._stack[-1])
now = get_now_utc_pendulum()
grand_total = datetime.timedelta()
overall_duration = now - self._overallstart
for name, duration in self._totaldurations.items():
grand_total += duration
log.info("Timing summary:")
summaries = []
for name, duration in self._totaldurations.items():
n = self._count[name]
total_sec = duration.total_seconds()
mean = total_sec / n if n > 0 else None
summaries.append(
{
"total": total_sec,
"description": (
f"- {name}: {total_sec:.3f} s "
f"({(100 * total_sec / grand_total.total_seconds()):.2f}%, " # noqa: E501
f"n={n}, mean={mean:.3f}s)"
),
}
)
summaries.sort(key=lambda x: x["total"], reverse=True)
for s in summaries:
# noinspection PyTypeChecker
log.info(s["description"])
if not self._totaldurations:
log.info("<no timings recorded>")
unmetered = overall_duration - grand_total
log.info(
"Unmetered time: {:.3f} s ({:.2f}%)",
unmetered.total_seconds(),
100 * unmetered.total_seconds() / overall_duration.total_seconds(),
)
log.info("Total time: {:.3f} s", grand_total.total_seconds())
[docs]class MultiTimerContext(object):
"""
Context manager for :class:`MultiTimer`.
Example:
.. code-block:: python
import logging
from time import sleep
from cardinal_pythonlib.timing import MultiTimer, MultiTimerContext
timer = MultiTimer(start=True)
def f2():
with MultiTimerContext(timer, "f2"):
print("starting f2")
sleep(1)
print("finishing f2")
def f1():
with MultiTimerContext(timer, "f1"):
print("starting f1")
sleep(0.25)
f2()
f2()
print("finishing f1")
logging.basicConfig(level=logging.DEBUG)
f1()
timer.report()
"""
def __init__(self, multitimer: MultiTimer, name: str) -> None:
"""
Args:
multitimer: :class:`MultiTimer` to use
name: name of timer to start as we enter, and stop as we exit
"""
self.timer = multitimer
self.name = name
def __enter__(self):
self.timer.start(self.name)
def __exit__(self, exc_type, exc_val, exc_tb):
self.timer.stop(self.name)
# Optional global instance, for convenience.
# In a multithreading environment, MAKE YOUR OWN INSTANCES of MultiTimer().
timer = MultiTimer(start=False)