forked from python/mypy
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathgclogger.py
46 lines (39 loc) · 1.59 KB
/
gclogger.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
import gc
import time
from typing import Mapping, Optional
class GcLogger:
"""Context manager to log GC stats and overall time."""
def __enter__(self) -> 'GcLogger':
self.gc_start_time: Optional[float] = None
self.gc_time = 0.0
self.gc_calls = 0
self.gc_collected = 0
self.gc_uncollectable = 0
gc.callbacks.append(self.gc_callback)
self.start_time = time.time()
return self
def gc_callback(self, phase: str, info: Mapping[str, int]) -> None:
if phase == 'start':
assert self.gc_start_time is None, "Start phase out of sequence"
self.gc_start_time = time.time()
elif phase == 'stop':
assert self.gc_start_time is not None, "Stop phase out of sequence"
self.gc_calls += 1
self.gc_time += time.time() - self.gc_start_time
self.gc_start_time = None
self.gc_collected += info['collected']
self.gc_uncollectable += info['uncollectable']
else:
assert False, "Unrecognized gc phase (%r)" % (phase,)
def __exit__(self, *args: object) -> None:
while self.gc_callback in gc.callbacks:
gc.callbacks.remove(self.gc_callback)
def get_stats(self) -> Mapping[str, float]:
end_time = time.time()
result = {}
result['gc_time'] = self.gc_time
result['gc_calls'] = self.gc_calls
result['gc_collected'] = self.gc_collected
result['gc_uncollectable'] = self.gc_uncollectable
result['build_time'] = end_time - self.start_time
return result