-
Notifications
You must be signed in to change notification settings - Fork 813
/
Copy pathdaemon.py
378 lines (324 loc) · 12.2 KB
/
daemon.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
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
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
"""
***
Modified generic daemon class
***
Author: http://www.jejik.com/articles/2007/02/a_simple_unix_linux_daemon_in_python/
www.boxedice.com
www.datadoghq.com
License: http://creativecommons.org/licenses/by-sa/3.0/
"""
# Core modules
from contextlib import nested
import atexit
import errno
import logging
import os
import signal
import sys
import tempfile
import time
# project
from utils.platform import Platform
from utils.process import is_my_process
from utils.subprocess_output import subprocess
from config import get_logging_config
log = logging.getLogger(__name__)
class AgentSupervisor(object):
''' A simple supervisor to keep a restart a child on expected auto-restarts
'''
RESTART_EXIT_STATUS = 5
@classmethod
def start(cls, parent_func, child_func=None):
''' `parent_func` is a function that's called every time the child
process dies.
`child_func` is a function that should be run by the forked child
that will auto-restart with the RESTART_EXIT_STATUS.
'''
# Allow the child process to die on SIGTERM
signal.signal(signal.SIGTERM, cls._handle_sigterm)
cls.need_stop = False
while True:
try:
if hasattr(cls, 'child_pid'):
delattr(cls, 'child_pid')
pid = os.fork()
if pid > 0:
# The parent waits on the child.
cls.child_pid = pid
while not cls.need_stop:
cpid, status = os.waitpid(pid, os.WNOHANG)
if (cpid, status) != (0, 0):
break
time.sleep(1)
if parent_func is not None:
parent_func()
if cls.need_stop:
break
else:
# The child will call our given function
if child_func is not None:
child_func()
else:
break
except OSError as e:
msg = "Agent fork failed: %d (%s)" % (e.errno, e.strerror)
logging.error(msg)
sys.stderr.write(msg + "\n")
sys.exit(1)
# Exit from the parent cleanly
if pid > 0:
sys.exit(0)
@classmethod
def _handle_sigterm(cls, signum, frame):
# in the parent
if hasattr(cls, 'child_pid'):
os.kill(cls.child_pid, signal.SIGTERM)
cls.need_stop = True
# in the child
else:
sys.exit(0)
class ProcessRunner(object):
def __init__(self):
self.logging_config = get_logging_config()
self._process = None
self._running = True
@property
def status(self):
"""
Get the status of the runner. Exits with 0 if running, 1 if not.
"""
if self._process and self._running:
return 0
return 1
def terminate(self):
if self._process:
self._process.terminate()
def _handle_sigterm(self, signum, frame):
# Terminate jmx process on SIGTERM signal
log.debug("Caught sigterm. Stopping subprocess.")
self.terminate()
def register_signal_handlers(self):
"""
Enable SIGTERM and SIGINT handlers
"""
try:
# Gracefully exit on sigterm
signal.signal(signal.SIGTERM, self._handle_sigterm)
# Handle Keyboard Interrupt
signal.signal(signal.SIGINT, self._handle_sigterm)
except ValueError:
log.exception("Unable to register signal handlers.")
def execute(self, process_args, redirect_std_streams=None, env=None):
try:
with nested(tempfile.TemporaryFile(), tempfile.TemporaryFile()) as (stdout_f, stderr_f):
process = subprocess.Popen(
process_args,
close_fds=not redirect_std_streams, # only set to True when the streams are not redirected, for WIN compatibility
stdout=stdout_f if redirect_std_streams else None,
stderr=stderr_f if redirect_std_streams else None,
env=env
)
self._process = process
self._running = True
# Register SIGINT and SIGTERM signal handlers
self.register_signal_handlers()
# Wait for process to return
self._process.wait()
self._running = False
if redirect_std_streams:
stderr_f.seek(0)
err = stderr_f.read()
stdout_f.seek(0)
out = stdout_f.read()
sys.stdout.write(out)
sys.stderr.write(err)
return self._process.returncode
except Exception:
log.exception("Could not launch process")
raise
class Daemon(object):
"""
A generic daemon class.
Usage: subclass the Daemon class and override the run() method
"""
def __init__(self, pidfile, stdin=os.devnull, stdout=os.devnull, stderr=os.devnull, autorestart=False):
self.autorestart = autorestart
self.stdin = stdin
self.stdout = stdout
self.stderr = stderr
self.pidfile = pidfile
def daemonize(self):
"""
Do the UNIX double-fork magic, see Stevens' "Advanced
Programming in the UNIX Environment" for details (ISBN 0201563177)
http://www.erlenstar.demon.co.uk/unix/faq_2.html#SEC16
"""
try:
pid = os.fork()
if pid > 0:
# Exit first parent
sys.exit(0)
except OSError as e:
msg = "fork #1 failed: %d (%s)" % (e.errno, e.strerror)
log.error(msg)
sys.stderr.write(msg + "\n")
sys.exit(1)
log.debug("Fork 1 ok")
# Decouple from parent environment
os.chdir("/")
os.setsid()
if self.autorestart:
# Set up the supervisor callbacks and put a fork in it.
logging.info('Running with auto-restart ON')
AgentSupervisor.start(parent_func=None, child_func=None)
else:
# Do second fork
try:
pid = os.fork()
if pid > 0:
# Exit from second parent
sys.exit(0)
except OSError as e:
msg = "fork #2 failed: %d (%s)" % (e.errno, e.strerror)
logging.error(msg)
sys.stderr.write(msg + "\n")
sys.exit(1)
if sys.platform != 'darwin': # This block breaks on OS X
# Redirect standard file descriptors
sys.stdout.flush()
sys.stderr.flush()
si = file(self.stdin, 'r')
so = file(self.stdout, 'a+')
se = file(self.stderr, 'a+', 0)
os.dup2(si.fileno(), sys.stdin.fileno())
os.dup2(so.fileno(), sys.stdout.fileno())
os.dup2(se.fileno(), sys.stderr.fileno())
log.info("Daemon started")
def start(self, foreground=False):
log.info("Starting")
if not Platform.is_windows():
pid = self.pid()
if pid:
# Check if the pid in the pidfile corresponds to a running process
# and if psutil is installed, check if it's a datadog-agent one
if is_my_process(pid):
log.error("Not starting, another instance is already running"
" (using pidfile {0})".format(self.pidfile))
sys.exit(1)
else:
log.warn("pidfile doesn't contain the pid of an agent process."
' Starting normally')
if not foreground:
self.daemonize()
self.write_pidfile()
else:
log.debug("Skipping pidfile check for Windows")
self.run()
def stop(self):
log.info("Stopping daemon")
pid = self.pid()
# Clear the pid file
if os.path.exists(self.pidfile):
os.remove(self.pidfile)
if pid > 1:
try:
if self.autorestart:
# Try killing the supervising process
try:
os.kill(os.getpgid(pid), signal.SIGTERM)
except OSError:
log.warn("Couldn't not kill parent pid %s. Killing pid." % os.getpgid(pid))
os.kill(pid, signal.SIGTERM)
else:
# No supervising process present
os.kill(pid, signal.SIGTERM)
log.info("Daemon is stopped")
except OSError as err:
if str(err).find("No such process") <= 0:
log.exception("Cannot kill Agent daemon at pid %s" % pid)
sys.stderr.write(str(err) + "\n")
else:
message = "Pidfile %s does not exist. Not running?\n" % self.pidfile
log.info(message)
sys.stderr.write(message)
# A ValueError might occur if the PID file is empty but does actually exist
if os.path.exists(self.pidfile):
os.remove(self.pidfile)
return # Not an error in a restart
def restart(self):
"Restart the daemon"
self.stop()
self.start()
def run(self):
"""
You should override this method when you subclass Daemon. It will be called after the process has been
daemonized by start() or restart().
"""
raise NotImplementedError
@classmethod
def info(cls):
"""
You should override this method when you subclass Daemon. It will be
called to provide information about the status of the process
"""
raise NotImplementedError
def status(self):
"""
Get the status of the daemon. Exits with 0 if running, 1 if not.
"""
pid = self.pid()
if pid < 0:
message = '%s is not running' % self.__class__.__name__
exit_code = 1
else:
# Check for the existence of a process with the pid
try:
# os.kill(pid, 0) will raise an OSError exception if the process
# does not exist, or if access to the process is denied (access denied will be an EPERM error).
# If we get an OSError that isn't an EPERM error, the process
# does not exist.
# (from http://stackoverflow.com/questions/568271/check-if-pid-is-not-in-use-in-python,
# Giampaolo's answer)
os.kill(pid, 0)
except OSError as e:
if e.errno != errno.EPERM:
message = '%s pidfile contains pid %s, but no running process could be found' % (self.__class__.__name__, pid)
else:
message = 'You do not have sufficient permissions'
exit_code = 1
else:
message = '%s is running with pid %s' % (self.__class__.__name__, pid)
exit_code = 0
log.info(message)
sys.stdout.write(message + "\n")
sys.exit(exit_code)
def pid(self):
# Get the pid from the pidfile
try:
pf = file(self.pidfile, 'r')
pid = int(pf.read().strip())
pf.close()
return pid
except IOError:
return None
except ValueError:
return None
def write_pidfile(self):
# Write pidfile
atexit.register(self.delpid) # Make sure pid file is removed if we quit
pid = str(os.getpid())
try:
fp = open(self.pidfile, 'w+')
fp.write(str(pid))
fp.close()
os.chmod(self.pidfile, 0644)
except Exception:
msg = "Unable to write pidfile: %s" % self.pidfile
log.exception(msg)
sys.stderr.write(msg + "\n")
sys.exit(1)
def delpid(self):
try:
os.remove(self.pidfile)
except OSError:
pass