-
Notifications
You must be signed in to change notification settings - Fork 21
Expand file tree
/
Copy pathruntime.py
More file actions
145 lines (101 loc) · 3.18 KB
/
runtime.py
File metadata and controls
145 lines (101 loc) · 3.18 KB
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
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
import time
import sys
import re
import os
import signal
from functools import wraps
try:
import resource
except ImportError:
pass
class runtime_info(object):
OS_LINUX = (sys.platform.startswith('linux'))
OS_DARWIN = (sys.platform == 'darwin')
OS_WIN = (sys.platform == 'win32')
PYTHON_2 = (sys.version_info.major == 2)
PYTHON_3 = (sys.version_info.major == 3)
GEVENT = False
try:
import gevent
if hasattr(gevent, '_threading'):
runtime_info.GEVENT = True
except ImportError:
pass
VM_RSS_REGEXP = re.compile('VmRSS:\s+(\d+)\s+kB')
VM_SIZE_REGEXP = re.compile('VmSize:\s+(\d+)\s+kB')
def min_version(major, minor=0):
return (sys.version_info.major == major and sys.version_info.minor >= minor)
def read_cpu_time():
rusage = resource.getrusage(resource.RUSAGE_SELF)
return int((rusage.ru_utime + rusage.ru_stime) * 1e9) # nanoseconds
def read_max_rss():
rusage = resource.getrusage(resource.RUSAGE_SELF)
if runtime_info.OS_DARWIN:
return int(rusage.ru_maxrss / 1000) # KB
else:
return rusage.ru_maxrss # KB
def read_current_rss():
pid = os.getpid()
output = None
try:
f = open('/proc/{0}/status'.format(os.getpid()))
output = f.read()
f.close()
except Exception:
return None
match = VM_RSS_REGEXP.search(output)
if match:
return int(float(match.group(1)))
return None
def read_vm_size():
pid = os.getpid()
output = None
try:
f = open('/proc/{0}/status'.format(os.getpid()))
output = f.read()
f.close()
except Exception:
return None
match = VM_SIZE_REGEXP.search(output)
if match:
return int(float(match.group(1)))
return None
def patch(obj, func_name, before_func, after_func):
if not hasattr(obj, func_name):
return False
target_func = getattr(obj, func_name)
# already patched
if hasattr(target_func, '__stackimpact_orig__'):
return True
@wraps(target_func)
def wrapper(*args, **kwargs):
data = None
if before_func:
args, kwargs, data = before_func(args, kwargs)
ret = target_func(*args, **kwargs)
if after_func:
after_func(args, kwargs, ret, data)
return ret
wrapper.__orig__ = target_func
setattr(obj, func_name, wrapper)
return True
def unpatch(obj, func_name):
if not hasattr(obj, func_name):
return
wrapper = getattr(obj, func_name)
if not hasattr(wrapper, '__stackimpact_orig__'):
return
setattr(obj, func_name, getattr(wrapper, '__stackimpact_orig__'))
def register_signal(signal_number, handler_func, once=False):
prev_handler = None
def _handler(signum, frame):
skip_prev = handler_func(signum, frame)
if not skip_prev:
if callable(prev_handler):
if once:
signal.signal(signum, prev_handler)
prev_handler(signum, frame)
elif prev_handler == signal.SIG_DFL and once:
signal.signal(signum, signal.SIG_DFL)
os.kill(os.getpid(), signum)
prev_handler = signal.signal(signal_number, _handler)