mirror of
				git://sourceware.org/git/lvm2.git
				synced 2025-10-25 03:33:16 +03:00 
			
		
		
		
	The thread does not contain field "damon" but "daemon". Actually found with codespell. Signed-off-by: Tobias Stoeckmann <tobias@stoeckmann.org>
		
			
				
	
	
		
			243 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			243 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| # Copyright (C) 2015-2016 Red Hat, Inc. All rights reserved.
 | |
| #
 | |
| # This copyrighted material is made available to anyone wishing to use,
 | |
| # modify, copy, or redistribute it subject to the terms and conditions
 | |
| # of the GNU General Public License v.2.
 | |
| #
 | |
| # You should have received a copy of the GNU General Public License
 | |
| # along with this program. If not, see <http://www.gnu.org/licenses/>.
 | |
| 
 | |
| from . import cfg
 | |
| from . import objectmanager
 | |
| from . import utils
 | |
| from .cfg import BUS_NAME, BASE_INTERFACE, BASE_OBJ_PATH, MANAGER_OBJ_PATH
 | |
| import threading
 | |
| from . import cmdhandler
 | |
| import time
 | |
| import signal
 | |
| import dbus
 | |
| import dbus.mainloop.glib
 | |
| from . import lvmdb
 | |
| # noinspection PyUnresolvedReferences
 | |
| from gi.repository import GLib
 | |
| from .fetch import StateUpdate
 | |
| from .manager import Manager
 | |
| import queue
 | |
| from . import udevwatch
 | |
| from .utils import log_debug, log_error, log_msg, DebugMessages
 | |
| import argparse
 | |
| import os
 | |
| import sys
 | |
| from .cmdhandler import LvmFlightRecorder, supports_vdo, supports_json
 | |
| from .request import RequestEntry
 | |
| 
 | |
| 
 | |
| class Lvm(objectmanager.ObjectManager):
 | |
| 	def __init__(self, object_path):
 | |
| 		super(Lvm, self).__init__(object_path, BASE_INTERFACE)
 | |
| 
 | |
| 
 | |
| def process_request():
 | |
| 	while cfg.run.value != 0:
 | |
| 		# noinspection PyBroadException
 | |
| 		try:
 | |
| 			req = cfg.worker_q.get(True, cfg.G_LOOP_TMO)
 | |
| 			log_debug(
 | |
| 				"Method start: %s with args %s (callback = %s)" %
 | |
| 				(str(req.method), str(req.arguments), str(req.cb)))
 | |
| 			req.run_cmd()
 | |
| 			log_debug("Method complete: %s" % str(req.method))
 | |
| 		except queue.Empty:
 | |
| 			pass
 | |
| 		except SystemExit:
 | |
| 			break
 | |
| 		except Exception as e:
 | |
| 			st = utils.extract_stack_trace(e)
 | |
| 			utils.log_error("process_request exception: \n%s" % st)
 | |
| 	log_debug("process_request thread exiting!")
 | |
| 
 | |
| 
 | |
| def check_fr_size(value):
 | |
| 	v = int(value)
 | |
| 	if v < 0:
 | |
| 		raise argparse.ArgumentTypeError(
 | |
| 			"positive integers only ('%s' invalid)" % value)
 | |
| 	return v
 | |
| 
 | |
| 
 | |
| def install_signal_handlers():
 | |
| 	# Because of the glib main loop stuff the python signal handler code is
 | |
| 	# apparently not usable, and we need to use the glib calls instead
 | |
| 	signal_add = None
 | |
| 
 | |
| 	if hasattr(GLib, 'unix_signal_add'):
 | |
| 		signal_add = GLib.unix_signal_add
 | |
| 	elif hasattr(GLib, 'unix_signal_add_full'):
 | |
| 		signal_add = GLib.unix_signal_add_full
 | |
| 
 | |
| 	if signal_add:
 | |
| 		signal_add(GLib.PRIORITY_HIGH, signal.SIGHUP, utils.handler, signal.SIGHUP)
 | |
| 		signal_add(GLib.PRIORITY_HIGH, signal.SIGINT, utils.handler, signal.SIGINT)
 | |
| 		signal_add(GLib.PRIORITY_HIGH, signal.SIGUSR1, utils.handler, signal.SIGUSR1)
 | |
| 		signal_add(GLib.PRIORITY_HIGH, signal.SIGUSR2, utils.handler, signal.SIGUSR2)
 | |
| 		signal_add(GLib.PRIORITY_HIGH, signal.SIGTERM, utils.handler, signal.SIGTERM)
 | |
| 	else:
 | |
| 		log_error("GLib.unix_signal_[add|add_full] are NOT available!")
 | |
| 
 | |
| 
 | |
| def process_args():
 | |
| 	parser = argparse.ArgumentParser()
 | |
| 	parser.add_argument(
 | |
| 		"--udev", action='store_true',
 | |
| 		help="Use udev for updating state",
 | |
| 		default=False,
 | |
| 		dest='use_udev')
 | |
| 	parser.add_argument(
 | |
| 		"--debug", action='store_true',
 | |
| 		help="Dump debug messages", default=False,
 | |
| 		dest='debug')
 | |
| 	parser.add_argument(
 | |
| 		"--nojson", action='store_false',
 | |
| 		help="Do not use LVM JSON output (disables lvmshell)", default=True,
 | |
| 		dest='use_json')
 | |
| 	parser.add_argument(
 | |
| 		"--lvmshell", action='store_true',
 | |
| 		help="Use the lvm shell, not fork & exec lvm",
 | |
| 		default=False,
 | |
| 		dest='use_lvm_shell')
 | |
| 	parser.add_argument(
 | |
| 		"--frsize",
 | |
| 		help="Size of the flight recorder (num. entries), 0 to disable (signal 12 to dump)",
 | |
| 		default=10,
 | |
| 		type=check_fr_size,
 | |
| 		dest='fr_size')
 | |
| 
 | |
| 	args = parser.parse_args()
 | |
| 
 | |
| 	if not args.use_json:
 | |
| 		log_error("Daemon no longer supports lvm without JSON support, exiting!")
 | |
| 		sys.exit(1)
 | |
| 	else:
 | |
| 		if not supports_json():
 | |
| 			log_error("Un-supported version of LVM, daemon requires JSON output, exiting!")
 | |
| 			sys.exit(1)
 | |
| 
 | |
| 	# Add udev watching
 | |
| 	if args.use_udev:
 | |
| 		# Make sure this msg ends up in the journal, so we know
 | |
| 		log_msg('The --udev option is no longer supported,'
 | |
| 				'the daemon always uses a combination of dbus notify from lvm tools and udev')
 | |
| 
 | |
| 	return args
 | |
| 
 | |
| 
 | |
| def running_under_systemd():
 | |
| 	""""
 | |
| 	Checks to see if we are running under systemd, by checking daemon fd 0, 1
 | |
| 	systemd sets stdin to /dev/null and 1 & 2 are a socket
 | |
| 	"""
 | |
| 	base = "/proc/self/fd"
 | |
| 	stdout = os.readlink("%s/0" % base)
 | |
| 	if stdout == "/dev/null":
 | |
| 		stdout = os.readlink("%s/1" % base)
 | |
| 		if "socket" in stdout:
 | |
| 			return True
 | |
| 	return False
 | |
| 
 | |
| 
 | |
| def main():
 | |
| 	start = time.time()
 | |
| 	use_session = os.getenv('LVM_DBUSD_USE_SESSION', False)
 | |
| 
 | |
| 	# Ensure that we get consistent output for parsing stdout/stderr and that we
 | |
| 	# are using the lvmdbusd profile.
 | |
| 	os.environ["LC_ALL"] = "C"
 | |
| 	os.environ["LVM_COMMAND_PROFILE"] = "lvmdbusd"
 | |
| 
 | |
| 	# Indicator if we are running under systemd
 | |
| 	cfg.systemd = running_under_systemd()
 | |
| 
 | |
| 	# Add simple command line handling
 | |
| 	cfg.args = process_args()
 | |
| 
 | |
| 	cfg.create_request_entry = RequestEntry
 | |
| 
 | |
| 	# We create a flight recorder in cmdhandler too, but we replace it here
 | |
| 	# as the user may be specifying a different size.  The default one in
 | |
| 	# cmdhandler is for when we are running other code with a different main.
 | |
| 	cfg.flightrecorder = LvmFlightRecorder(cfg.args.fr_size)
 | |
| 
 | |
| 	# Create a circular buffer for debug logs
 | |
| 	cfg.debug = DebugMessages()
 | |
| 
 | |
| 	log_debug("Using lvm binary: %s" % cfg.LVM_CMD)
 | |
| 
 | |
| 	# We will dynamically add interfaces which support vdo if it
 | |
| 	# exists.
 | |
| 	cfg.vdo_support = supports_vdo()
 | |
| 
 | |
| 	# List of threads that we start up
 | |
| 	thread_list = []
 | |
| 
 | |
| 	install_signal_handlers()
 | |
| 
 | |
| 	with utils.LockFile(cfg.LOCK_FILE):
 | |
| 		dbus.mainloop.glib.DBusGMainLoop(set_as_default=True)
 | |
| 		dbus.mainloop.glib.threads_init()
 | |
| 
 | |
| 		cmdhandler.set_execution(cfg.args.use_lvm_shell)
 | |
| 
 | |
| 		if use_session:
 | |
| 			cfg.bus = dbus.SessionBus()
 | |
| 		else:
 | |
| 			cfg.bus = dbus.SystemBus()
 | |
| 		# The base name variable needs to exist for things to work.
 | |
| 		# noinspection PyUnusedLocal
 | |
| 		base_name = dbus.service.BusName(BUS_NAME, cfg.bus)
 | |
| 		cfg.om = Lvm(BASE_OBJ_PATH)
 | |
| 		cfg.om.register_object(Manager(MANAGER_OBJ_PATH))
 | |
| 
 | |
| 		cfg.db = lvmdb.DataStore(vdo_support=cfg.vdo_support)
 | |
| 
 | |
| 		# Using a thread to process requests, we cannot hang the dbus library
 | |
| 		# thread that is handling the dbus interface
 | |
| 		thread_list.append(
 | |
| 			threading.Thread(target=process_request, name='process_request'))
 | |
| 
 | |
| 		# Have a single thread handling updating lvm and the dbus model, so we
 | |
| 		# don't have multiple threads doing this as the same time
 | |
| 		updater = StateUpdate()
 | |
| 		thread_list.append(updater.thread)
 | |
| 
 | |
| 		cfg.load = updater.load
 | |
| 
 | |
| 		cfg.loop = GLib.MainLoop()
 | |
| 
 | |
| 		for thread in thread_list:
 | |
| 			thread.daemon = True
 | |
| 			thread.start()
 | |
| 
 | |
| 		# In all cases we are going to monitor for udev until we get an
 | |
| 		# ExternalEvent.  In the case where we get an external event and the user
 | |
| 		# didn't specify --udev we will stop monitoring udev
 | |
| 		udevwatch.add()
 | |
| 
 | |
| 		end = time.time()
 | |
| 		log_debug(
 | |
| 			'Service ready! total time= %.4f, lvm time= %.4f count= %d' %
 | |
| 			(end - start, cmdhandler.total_time, cmdhandler.total_count),
 | |
| 			'bg_black', 'fg_light_green')
 | |
| 
 | |
| 		try:
 | |
| 			if cfg.run.value != 0:
 | |
| 				cfg.loop.run()
 | |
| 				udevwatch.remove()
 | |
| 
 | |
| 				for thread in thread_list:
 | |
| 					thread.join()
 | |
| 		except KeyboardInterrupt:
 | |
| 			# If we are unable to register signal handler, we will end up here when
 | |
| 			# the service gets a ^C or a kill -2 <parent pid>
 | |
| 			utils.handler(signal.SIGINT)
 | |
| 	return 0
 |