Coverage for src/sensai/util/profiling.py: 29%
34 statements
« prev ^ index » next coverage.py v7.6.1, created at 2024-11-29 18:29 +0000
« prev ^ index » next coverage.py v7.6.1, created at 2024-11-29 18:29 +0000
1import functools
2import logging
3from typing import Optional, Callable
4from time import time
7log = logging.getLogger(__name__)
10def profiled(func: Optional[Callable] = None, *, sampling_interval_seconds: Optional[float] = None,
11 print_to_console=True, log_to_file=False, file_name: Optional[str] = None):
12 """
13 Function decorator for profiling the annotated/given function with the pyinstrument profiler
15 :param func: the function to be profiled
16 :param sampling_interval_seconds: sampling every <sampling_interval_seconds> seconds, if None use default parameter of pyinstrument
17 profiler
18 :param print_to_console: if, true print profiler output to console
19 :param log_to_file: if true, logs profiler output to html file
20 :param file_name: optional file name for log, if None defaults to file name derived from function name
21 :return: the wrapped function
22 """
23 from pyinstrument import Profiler
25 def wrapper(_func):
26 @functools.wraps(_func)
27 def _wrapper(*args, **kwargs):
28 _profiler = Profiler(sampling_interval_seconds) if sampling_interval_seconds is not None else Profiler()
29 _profiler.start()
30 result = _func(*args, **kwargs)
31 _profiler.stop()
32 if print_to_console:
33 print(_profiler.output_text(unicode=True, color=True))
34 if log_to_file:
35 name = file_name if file_name is not None else _func.__name__ + ".profile"
36 with open(f"{name}.html", "w") as f:
37 f.write(_profiler.output_html())
38 return result
39 return _wrapper
41 if func:
42 return wrapper(func)
44 return wrapper
47def timed(func):
48 """
49 Function decorator which logs execution times of the wrapped/annotated function at INFO level
50 :param func: the function whose execution time to log
51 :return: the wrapped function
52 """
53 @functools.wraps(func)
54 def _wrapper(*args, **kwargs):
55 start = int(round(time() * 1000))
56 try:
57 return func(*args, **kwargs)
58 finally:
59 end_ = int(round(time() * 1000)) - start
60 log.info(f"Execution time of {func.__qualname__}: {end_ if end_ > 0 else 0} ms")
61 return _wrapper