initial
This commit is contained in:
207
ftpReceiver/demonize_ftpd.py
Normal file
207
ftpReceiver/demonize_ftpd.py
Normal file
@@ -0,0 +1,207 @@
|
||||
#!/usr/bin/env python3
|
||||
|
||||
# Copyright (C) 2007 Giampaolo Rodola' <g.rodola@gmail.com>.
|
||||
# Use of this source code is governed by MIT license that can be
|
||||
# found in the LICENSE file.
|
||||
|
||||
"""A basic unix daemon using the python-daemon library:
|
||||
http://pypi.python.org/pypi/python-daemon
|
||||
|
||||
Example usages:
|
||||
|
||||
$ python unix_daemon.py start
|
||||
$ python unix_daemon.py stop
|
||||
$ python unix_daemon.py status
|
||||
$ python unix_daemon.py # foreground (no daemon)
|
||||
$ python unix_daemon.py --logfile /var/log/ftpd.log start
|
||||
$ python unix_daemon.py --pidfile /var/run/ftpd.pid start
|
||||
|
||||
This is just a proof of concept which demonstrates how to daemonize
|
||||
the FTP server.
|
||||
You might want to use this as an example and provide the necessary
|
||||
customizations.
|
||||
|
||||
Parts you might want to customize are:
|
||||
- UMASK, WORKDIR, HOST, PORT constants
|
||||
- get_server() function (to define users and customize FTP handler)
|
||||
|
||||
Authors:
|
||||
- Ben Timby - btimby <at> gmail.com
|
||||
- Giampaolo Rodola' - g.rodola <at> gmail.com
|
||||
|
||||
"""
|
||||
|
||||
import atexit
|
||||
import errno
|
||||
import optparse
|
||||
import os
|
||||
import signal
|
||||
import sys
|
||||
import time
|
||||
|
||||
from pyftpdlib.authorizers import UnixAuthorizer
|
||||
from pyftpdlib.filesystems import UnixFilesystem
|
||||
from pyftpdlib.handlers import FTPHandler
|
||||
from pyftpdlib.servers import FTPServer
|
||||
|
||||
|
||||
# overridable options
|
||||
HOST = ""
|
||||
PORT = 21
|
||||
PID_FILE = "/var/run/pyftpdlib.pid"
|
||||
LOG_FILE = "/var/log/pyftpdlib.log"
|
||||
WORKDIR = os.getcwd()
|
||||
UMASK = 0
|
||||
|
||||
|
||||
def pid_exists(pid):
|
||||
"""Return True if a process with the given PID is currently running."""
|
||||
try:
|
||||
os.kill(pid, 0)
|
||||
except OSError as err:
|
||||
return err.errno == errno.EPERM
|
||||
else:
|
||||
return True
|
||||
|
||||
|
||||
def get_pid():
|
||||
"""Return the PID saved in the pid file if possible, else None."""
|
||||
try:
|
||||
with open(PID_FILE) as f:
|
||||
return int(f.read().strip())
|
||||
except IOError as err:
|
||||
if err.errno != errno.ENOENT:
|
||||
raise
|
||||
|
||||
|
||||
def stop():
|
||||
"""Keep attempting to stop the daemon for 5 seconds, first using
|
||||
SIGTERM, then using SIGKILL.
|
||||
"""
|
||||
pid = get_pid()
|
||||
if not pid or not pid_exists(pid):
|
||||
sys.exit("daemon not running")
|
||||
sig = signal.SIGTERM
|
||||
i = 0
|
||||
while True:
|
||||
sys.stdout.write('.')
|
||||
sys.stdout.flush()
|
||||
try:
|
||||
os.kill(pid, sig)
|
||||
except OSError as err:
|
||||
if err.errno == errno.ESRCH:
|
||||
print("\nstopped (pid %s)" % pid)
|
||||
return
|
||||
else:
|
||||
raise
|
||||
i += 1
|
||||
if i == 25:
|
||||
sig = signal.SIGKILL
|
||||
elif i == 50:
|
||||
sys.exit("\ncould not kill daemon (pid %s)" % pid)
|
||||
time.sleep(0.1)
|
||||
|
||||
|
||||
def status():
|
||||
"""Print daemon status and exit."""
|
||||
pid = get_pid()
|
||||
if not pid or not pid_exists(pid):
|
||||
print("daemon not running")
|
||||
else:
|
||||
print("daemon running with pid %s" % pid)
|
||||
sys.exit(0)
|
||||
|
||||
|
||||
def get_server():
|
||||
"""Return a pre-configured FTP server instance."""
|
||||
handler = FTPHandler
|
||||
handler.authorizer = UnixAuthorizer()
|
||||
handler.abstracted_fs = UnixFilesystem
|
||||
server = FTPServer((HOST, PORT), handler)
|
||||
return server
|
||||
|
||||
|
||||
def daemonize():
|
||||
"""A wrapper around python-daemonize context manager."""
|
||||
def _daemonize():
|
||||
pid = os.fork()
|
||||
if pid > 0:
|
||||
# exit first parent
|
||||
sys.exit(0)
|
||||
|
||||
# decouple from parent environment
|
||||
os.chdir(WORKDIR)
|
||||
os.setsid()
|
||||
os.umask(0)
|
||||
|
||||
# do second fork
|
||||
pid = os.fork()
|
||||
if pid > 0:
|
||||
# exit from second parent
|
||||
sys.exit(0)
|
||||
|
||||
# redirect standard file descriptors
|
||||
sys.stdout.flush()
|
||||
sys.stderr.flush()
|
||||
si = open(LOG_FILE, 'r')
|
||||
so = open(LOG_FILE, 'a+')
|
||||
se = open(LOG_FILE, 'a+', 0)
|
||||
os.dup2(si.fileno(), sys.stdin.fileno())
|
||||
os.dup2(so.fileno(), sys.stdout.fileno())
|
||||
os.dup2(se.fileno(), sys.stderr.fileno())
|
||||
|
||||
# write pidfile
|
||||
pid = str(os.getpid())
|
||||
with open(PID_FILE, 'w') as f:
|
||||
f.write("%s\n" % pid)
|
||||
atexit.register(lambda: os.remove(PID_FILE))
|
||||
|
||||
pid = get_pid()
|
||||
if pid and pid_exists(pid):
|
||||
sys.exit('daemon already running (pid %s)' % pid)
|
||||
# instance FTPd before daemonizing, so that in case of problems we
|
||||
# get an exception here and exit immediately
|
||||
server = get_server()
|
||||
_daemonize()
|
||||
server.serve_forever()
|
||||
|
||||
|
||||
def main():
|
||||
global PID_FILE, LOG_FILE
|
||||
USAGE = "python [-p PIDFILE] [-l LOGFILE]\n\n" \
|
||||
"Commands:\n - start\n - stop\n - status"
|
||||
parser = optparse.OptionParser(usage=USAGE)
|
||||
parser.add_option('-l', '--logfile', dest='logfile',
|
||||
help='the log file location')
|
||||
parser.add_option('-p', '--pidfile', dest='pidfile', default=PID_FILE,
|
||||
help='file to store/retreive daemon pid')
|
||||
options, args = parser.parse_args()
|
||||
|
||||
if options.pidfile:
|
||||
PID_FILE = options.pidfile
|
||||
if options.logfile:
|
||||
LOG_FILE = options.logfile
|
||||
|
||||
if not args:
|
||||
server = get_server()
|
||||
server.serve_forever()
|
||||
else:
|
||||
if len(args) != 1:
|
||||
sys.exit('too many commands')
|
||||
elif args[0] == 'start':
|
||||
daemonize()
|
||||
elif args[0] == 'stop':
|
||||
stop()
|
||||
elif args[0] == 'restart':
|
||||
try:
|
||||
stop()
|
||||
finally:
|
||||
daemonize()
|
||||
elif args[0] == 'status':
|
||||
status()
|
||||
else:
|
||||
sys.exit('invalid command')
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
sys.exit(main())
|
||||
Reference in New Issue
Block a user