# Copyright (C) 2006-2012 by the Free Software Foundation, Inc. # # This file is part of GNU Mailman. # # GNU Mailman is free software: you can redistribute it and/or modify it under # the terms of the GNU General Public License as published by the Free # Software Foundation, either version 3 of the License, or (at your option) # any later version. # # GNU Mailman is distributed in the hope that it will be useful, but WITHOUT # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or # FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for # more details. # # You should have received a copy of the GNU General Public License along with # GNU Mailman. If not, see . """Initialize all global state. Every entrance into the Mailman system, be it by command line, mail program, or cgi, must call the initialize function here in order for the system's global state to be set up properly. Typically this is called after command line argument parsing, since some of the initialization behavior is controlled by the command line arguments. """ from __future__ import absolute_import, unicode_literals __metaclass__ = type __all__ = [ 'initialize', 'initialize_1', 'initialize_2', 'initialize_3', 'INHIBIT_CONFIG_FILE', ] import os import sys from pkg_resources import resource_string from zope.configuration import xmlconfig from zope.interface.verify import verifyObject import mailman.config.config import mailman.core.logging from mailman.interfaces.database import IDatabase from mailman.utilities.modules import call_name # The test infrastructure uses this to prevent the search and loading of any # existing configuration file. Otherwise the existence of say a # ~/.mailman.cfg file can break tests. INHIBIT_CONFIG_FILE = object() def search_for_configuration_file(): """Search the file system for a configuration file to use. This is only called if the -C command line argument was not given. """ config_path = os.getenv('MAILMAN_CONFIG_FILE') # Both None and the empty string are considered "missing". if config_path and os.path.exists(config_path): return os.path.abspath(config_path) # ./mailman.cfg config_path = os.path.abspath('mailman.cfg') if os.path.exists(config_path): return os.path.abspath(config_path) # ~/.mailman.cfg config_path = os.path.join(os.getenv('HOME'), '.mailman.cfg') if os.path.exists(config_path): return os.path.abspath(config_path) # /etc/mailman.cfg config_path = '/etc/mailman.cfg' if os.path.exists(config_path): return os.path.abspath(config_path) # $argv0/../../etc/mailman.cfg bindir = os.path.dirname(sys.argv[0]) parent = os.path.dirname(bindir) config_path = os.path.join(parent, 'etc', 'mailman.cfg') if os.path.exists(config_path): return os.path.abspath(config_path) # Are there any others we should search by default? return None # These initialization calls are separated for the testing framework, which # needs to do some internal calculations after config file loading and log # initialization, but before database initialization. Generally all other # code will just call initialize(). def initialize_1(config_path=None): """First initialization step. * Zope component architecture * The configuration system * Run-time directories :param config_path: The path to the configuration file. :type config_path: string """ zcml = resource_string('mailman.config', 'configure.zcml') xmlconfig.string(zcml) # By default, set the umask so that only owner and group can read and # write our files. Specifically we must have g+rw and we probably want # o-rwx although I think in most cases it doesn't hurt if other can read # or write the files. os.umask(007) # config_path will be set if the command line argument -C is given. That # case overrides all others. When not given on the command line, the # configuration file is searched for in the file system. if config_path is None: config_path = search_for_configuration_file() elif config_path is INHIBIT_CONFIG_FILE: # For the test suite, force this back to not using a config file. config_path = None mailman.config.config.load(config_path) def initialize_2(debug=False, propagate_logs=None): """Second initialization step. * Logging * Pre-hook * Rules * Chains * Pipelines * Commands :param debug: Should the database layer be put in debug mode? :type debug: boolean :param propagate_logs: Should the log output propagate to stderr? :type propagate_logs: boolean or None """ # Create the queue and log directories if they don't already exist. mailman.core.logging.initialize(propagate_logs) # Run the pre-hook if there is one. config = mailman.config.config if config.mailman.pre_hook: call_name(config.mailman.pre_hook) # Instantiate the database class, ensure that it's of the right type, and # initialize it. Then stash the object on our configuration object. database_class = config.database['class'] database = call_name(database_class) verifyObject(IDatabase, database) database.initialize(debug) config.db = database # Initialize the rules and chains. Do the imports here so as to avoid # circular imports. from mailman.app.commands import initialize as initialize_commands from mailman.app.events import initialize as initialize_events from mailman.core.chains import initialize as initialize_chains from mailman.core.pipelines import initialize as initialize_pipelines from mailman.core.rules import initialize as initialize_rules # Order here is somewhat important. initialize_rules() initialize_chains() initialize_pipelines() initialize_commands() initialize_events() def initialize_3(): """Third initialization step. * Post-hook """ # Run the post-hook if there is one. config = mailman.config.config if config.mailman.post_hook: call_name(config.mailman.post_hook) def initialize(config_path=None, propagate_logs=None): initialize_1(config_path) initialize_2(propagate_logs=propagate_logs) initialize_3()