Fix daemon log being clobbered by running another instance with same config dir

Also includes small fixes and code cleanup
This commit is contained in:
Calum Lind 2013-05-19 22:21:32 +01:00
parent 86ac98b9f9
commit f8fbda97cd
2 changed files with 108 additions and 106 deletions

View File

@ -31,100 +31,98 @@
# this exception statement from your version. If you delete this exception # this exception statement from your version. If you delete this exception
# statement from all source files in the program, then also delete it here. # statement from all source files in the program, then also delete it here.
# #
import os import os
import logging import logging
from twisted.internet import reactor from twisted.internet import reactor
import deluge.component as component import deluge.component as component
import deluge.configmanager from deluge.configmanager import get_config_dir
import deluge.common from deluge.common import get_version, windows_check
from deluge.core.rpcserver import RPCServer, export from deluge.core.rpcserver import RPCServer, export
from deluge.error import DaemonRunningError from deluge.error import DaemonRunningError
from deluge.core.core import Core
if windows_check():
from win32api import SetConsoleCtrlHandler
from win32con import CTRL_CLOSE_EVENT, CTRL_SHUTDOWN_EVENT
log = logging.getLogger(__name__) log = logging.getLogger(__name__)
def check_running_daemon(pid_file):
"""Check for another running instance of the daemon using the same pid file"""
if os.path.isfile(pid_file):
# Get the PID and the port of the supposedly running daemon
with open(pid_file) as _file:
(pid, port) = _file.readline().strip().split(";")
try:
pid, port = int(pid), int(port)
except ValueError:
pid, port = None, None
def process_running(pid):
if windows_check():
from win32process import EnumProcesses
return pid in EnumProcesses()
else:
# We can just use os.kill on UNIX to test if the process is running
try:
os.kill(pid, 0)
except OSError:
return False
else:
return True
if pid is not None and process_running(pid):
# Ensure it's a deluged process by trying to open a socket to it's port.
import socket
_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
try:
_socket.connect(("127.0.0.1", port))
except socket.error:
# Can't connect, so it must not be a deluged process..
pass
else:
# This is a deluged!
_socket.close()
raise DaemonRunningError("Deluge daemon already running with this config directory!")
class Daemon(object): class Daemon(object):
def __init__(self, options=None, args=None, classic=False): def __init__(self, options=None, args=None, classic=False):
# Check for another running instance of the daemon log.info("Deluge daemon %s", get_version())
pid_file = deluge.configmanager.get_config_dir("deluged.pid") log.debug("options: %s", options)
if os.path.isfile(pid_file): log.debug("args: %s", args)
# Get the PID and the port of the supposedly running daemon
with open(pid_file) as _file:
(pid, port) = _file.readline().strip().split(";")
try:
pid, port = int(pid), int(port)
except ValueError:
pid, port = None, None
def process_running(pid): pid_file = get_config_dir("deluged.pid")
if deluge.common.windows_check(): check_running_daemon(pid_file)
import win32process
return pid in win32process.EnumProcesses()
else:
# We can just use os.kill on UNIX to test if the process is running
try:
os.kill(pid, 0)
except OSError:
return False
else:
return True
if pid is not None and process_running(pid):
# Ensure it's a deluged process by trying to open a socket to it's port.
import socket
_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
try:
_socket.connect(("127.0.0.1", port))
except socket.error:
# Can't connect, so it must not be a deluged process..
pass
else:
# This is a deluged!
_socket.close()
raise DaemonRunningError("Deluge daemon already running with this config directory!")
# Twisted catches signals to terminate, so just have it call the shutdown method. # Twisted catches signals to terminate, so just have it call the shutdown method.
reactor.addSystemEventTrigger("before", "shutdown", self._shutdown) reactor.addSystemEventTrigger("before", "shutdown", self._shutdown)
# Catch some Windows specific signals # Catch some Windows specific signals
if deluge.common.windows_check(): if windows_check():
from win32api import SetConsoleCtrlHandler
from win32con import CTRL_CLOSE_EVENT, CTRL_SHUTDOWN_EVENT
def win_handler(ctrl_type): def win_handler(ctrl_type):
log.debug("ctrl_type: %s", ctrl_type) log.debug("windows handler ctrl_type: %s", ctrl_type)
if ctrl_type == CTRL_CLOSE_EVENT or ctrl_type == CTRL_SHUTDOWN_EVENT: if ctrl_type == CTRL_CLOSE_EVENT or ctrl_type == CTRL_SHUTDOWN_EVENT:
self._shutdown() self._shutdown()
return 1 return 1
SetConsoleCtrlHandler(win_handler) SetConsoleCtrlHandler(win_handler)
version = deluge.common.get_version() listen_interface = None
log.info("Deluge daemon %s", version)
log.debug("options: %s", options)
log.debug("args: %s", args)
# Set the config directory
if options and options.config:
deluge.configmanager.set_config_dir(options.config)
if options and options.listen_interface: if options and options.listen_interface:
listen_interface = options.listen_interface listen_interface = options.listen_interface
else:
listen_interface = ""
from deluge.core.core import Core
# Start the core as a thread and join it until it's done # Start the core as a thread and join it until it's done
self.core = Core(listen_interface=listen_interface) self.core = Core(listen_interface=listen_interface)
port = self.core.config["daemon_port"] port = self.core.config["daemon_port"]
if options and options.port: if options and options.port:
port = options.port port = options.port
interface = None
if options and options.ui_interface: if options and options.ui_interface:
interface = options.ui_interface interface = options.ui_interface
else:
interface = ""
self.rpcserver = RPCServer( self.rpcserver = RPCServer(
port=port, port=port,
@ -141,31 +139,31 @@ class Daemon(object):
component.start("PreferencesManager") component.start("PreferencesManager")
if not classic: if not classic:
log.info("Deluge daemon starting...")
# Create pid file to track if deluged is running, also includes the port number. # Create pid file to track if deluged is running, also includes the port number.
log.debug("Creating pid file: %s", pid_file) pid = os.getpid()
open(pid_file, "wb").write("%s;%s\n" % (os.getpid(), port)) log.debug("Storing pid %s & port %s in: %s", pid, port, pid_file)
with open(pid_file, "wb") as _file:
_file.write("%s;%s\n" % (pid, port))
component.start() component.start()
try: try:
log.info("Deluge daemon starting...")
reactor.run() reactor.run()
finally: finally:
try: log.debug("Remove pid file: %s", pid_file)
log.debug("Removing pid file: %s", pid_file) os.remove(pid_file)
os.remove(pid_file)
except OSError, ex:
log.error("Error removing pid file: %s", ex)
log.info("Deluge daemon shutdown successfully") log.info("Deluge daemon shutdown successfully")
@export() @export()
def shutdown(self, *args, **kwargs): def shutdown(self, *args, **kwargs):
log.debug("Delgue daemon shutdown requested...") log.debug("Deluge daemon shutdown requested...")
reactor.callLater(0, reactor.stop) reactor.callLater(0, reactor.stop)
def _shutdown(self, *args, **kwargs): def _shutdown(self, *args, **kwargs):
log.info("Deluge daemon shutting down, waiting for components to shutdown...") log.info("Deluge daemon shutting down, waiting for components to shutdown...")
d = component.shutdown() return component.shutdown()
return d
@export() @export()
def get_method_list(self): def get_method_list(self):
@ -187,4 +185,4 @@ class Daemon(object):
return False return False
auth_level = self.rpcserver.get_session_auth_level() auth_level = self.rpcserver.get_session_auth_level()
return auth_level >= self.rpcserver.get_rpc_auth_level() return auth_level >= self.rpcserver.get_rpc_auth_level(rpc)

View File

@ -48,6 +48,9 @@ from errno import EEXIST
from deluge.log import setupLogger from deluge.log import setupLogger
import deluge.error import deluge.error
import deluge.common
import deluge.configmanager
def version_callback(option, opt_str, value, parser): def version_callback(option, opt_str, value, parser):
print os.path.basename(sys.argv[0]) + ": " + deluge.common.get_version() print os.path.basename(sys.argv[0]) + ": " + deluge.common.get_version()
@ -58,9 +61,9 @@ def version_callback(option, opt_str, value, parser):
pass pass
raise SystemExit raise SystemExit
def start_ui(): def start_ui():
"""Entry point for ui script""" """Entry point for ui script"""
import deluge.common
deluge.common.setup_translations() deluge.common.setup_translations()
# Setup the argument parser # Setup the argument parser
@ -106,7 +109,7 @@ def start_ui():
# Try to create the config folder if it doesn't exist # Try to create the config folder if it doesn't exist
try: try:
os.makedirs(options.config) os.makedirs(options.config)
except Exception, e: except OSError:
pass pass
elif not os.path.isdir(options.config): elif not os.path.isdir(options.config):
log.error("Config option needs to be a directory!") log.error("Config option needs to be a directory!")
@ -116,7 +119,6 @@ def start_ui():
os.makedirs(deluge.common.get_default_config_dir()) os.makedirs(deluge.common.get_default_config_dir())
if options.default_ui: if options.default_ui:
import deluge.configmanager
if options.config: if options.config:
deluge.configmanager.set_config_dir(options.config) deluge.configmanager.set_config_dir(options.config)
@ -137,9 +139,9 @@ def start_ui():
log.info("Starting ui..") log.info("Starting ui..")
UI(options, args, options.args) UI(options, args, options.args)
def start_daemon(): def start_daemon():
"""Entry point for daemon script""" """Entry point for daemon script"""
import deluge.common
deluge.common.setup_translations() deluge.common.setup_translations()
if 'dev' not in deluge.common.get_version(): if 'dev' not in deluge.common.get_version():
@ -185,6 +187,21 @@ this should be an IP address", metavar="IFACE",
# Get the options and args from the OptionParser # Get the options and args from the OptionParser
(options, args) = parser.parse_args() (options, args) = parser.parse_args()
if options.config:
if not deluge.configmanager.set_config_dir(options.config):
print "There was an error setting the config directory! Exiting..."
sys.exit(1)
# Check for any daemons running with this same config
from deluge.core.daemon import check_running_daemon
pid_file = deluge.configmanager.get_config_dir("deluged.pid")
try:
check_running_daemon(pid_file)
except deluge.error.DaemonRunningError:
print "You cannot run multiple daemons with the same config directory set."
print "If you believe this is an error, you can force a start by deleting: %s" % pid_file
sys.exit(1)
# Setup the logger # Setup the logger
if options.quiet: if options.quiet:
options.loglevel = "none" options.loglevel = "none"
@ -192,36 +209,25 @@ this should be an IP address", metavar="IFACE",
# Try to create the logfile's directory if it doesn't exist # Try to create the logfile's directory if it doesn't exist
try: try:
os.makedirs(os.path.abspath(os.path.dirname(options.logfile))) os.makedirs(os.path.abspath(os.path.dirname(options.logfile)))
except OSError, e: except OSError as ex:
if e.errno != EEXIST: if ex.errno != EEXIST:
print "There was an error creating the log directory, exiting... (%s)" % e print "There was an error creating the log directory, exiting... (%s)" % ex
sys.exit(1) sys.exit(1)
logfile_mode = 'w' logfile_mode = 'w'
if options.rotate_logs: if options.rotate_logs:
logfile_mode = 'a' logfile_mode = 'a'
setupLogger(level=options.loglevel, filename=options.logfile, filemode=logfile_mode) setupLogger(level=options.loglevel, filename=options.logfile, filemode=logfile_mode)
log = getLogger(__name__) log = getLogger(__name__)
import deluge.configmanager # If no logfile specified add logging to default location (as well as stdout)
if options.config: if not options.logfile:
if not deluge.configmanager.set_config_dir(options.config): options.logfile = deluge.configmanager.get_config_dir("deluged.log")
log.error("There was an error setting the config directory! Exiting...") file_handler = FileHandler(options.logfile)
sys.exit(1) log.addHandler(file_handler)
# Sets the options.logfile to point to the default location
def open_logfile():
if not options.logfile:
options.logfile = deluge.configmanager.get_config_dir("deluged.log")
file_handler = FileHandler(options.logfile)
log.addHandler(file_handler)
# Writes out a pidfile if necessary
def write_pidfile():
if options.pidfile:
open(options.pidfile, "wb").write("%s\n" % os.getpid())
# If the donot daemonize is set, then we just skip the forking # If the donot daemonize is set, then we just skip the forking
if not (deluge.common.windows_check() or deluge.common.osx_check() or options.donot): if not (options.donot or deluge.common.windows_check() or deluge.common.osx_check()):
if os.fork(): if os.fork():
# We've forked and this is now the parent process, so die! # We've forked and this is now the parent process, so die!
os._exit(0) os._exit(0)
@ -231,7 +237,9 @@ this should be an IP address", metavar="IFACE",
os._exit(0) os._exit(0)
# Write pid file before chuid # Write pid file before chuid
write_pidfile() if options.pidfile:
with open(options.pidfile, "wb") as _file:
_file.write("%s\n" % os.getpid())
if not deluge.common.windows_check(): if not deluge.common.windows_check():
if options.user: if options.user:
@ -245,20 +253,16 @@ this should be an IP address", metavar="IFACE",
options.group = grp.getgrnam(options.group)[2] options.group = grp.getgrnam(options.group)[2]
os.setuid(options.group) os.setuid(options.group)
open_logfile()
def run_daemon(options, args): def run_daemon(options, args):
from deluge.core.daemon import Daemon
try: try:
from deluge.core.daemon import Daemon
Daemon(options, args) Daemon(options, args)
except deluge.error.DaemonRunningError, e: except Exception as ex:
log.error(e) log.exception(ex)
log.error("You cannot run multiple daemons with the same config directory set.")
log.error("If you believe this is an error, you can force a start by deleting %s.", deluge.configmanager.get_config_dir("deluged.pid"))
sys.exit(1)
except Exception, e:
log.exception(e)
sys.exit(1) sys.exit(1)
finally:
if options.pidfile:
os.remove(options.pidfile)
if options.profile: if options.profile:
import cProfile import cProfile