Adds documentation on configuring logging and a test that log_file works. It didn't, so this also inludes fixes for setting up log handling :)
This commit is contained in:
parent
02a3cd30c1
commit
75575b41c6
@ -47,7 +47,7 @@ def create_options(parser):
|
|||||||
:param parser: The option parser
|
:param parser: The option parser
|
||||||
"""
|
"""
|
||||||
config.add_common_options(parser)
|
config.add_common_options(parser)
|
||||||
config.add_log_options('glance-api', parser)
|
config.add_log_options(parser)
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
@ -57,7 +57,6 @@ if __name__ == '__main__':
|
|||||||
(options, args) = config.parse_options(oparser)
|
(options, args) = config.parse_options(oparser)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
config.setup_logging(options)
|
|
||||||
conf, app = config.load_paste_app('glance-api', options, args)
|
conf, app = config.load_paste_app('glance-api', options, args)
|
||||||
|
|
||||||
server = wsgi.Server()
|
server = wsgi.Server()
|
||||||
|
@ -47,7 +47,7 @@ def create_options(parser):
|
|||||||
:param parser: The option parser
|
:param parser: The option parser
|
||||||
"""
|
"""
|
||||||
config.add_common_options(parser)
|
config.add_common_options(parser)
|
||||||
config.add_log_options('glance-registry', parser)
|
config.add_log_options(parser)
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
@ -57,7 +57,6 @@ if __name__ == '__main__':
|
|||||||
(options, args) = config.parse_options(oparser)
|
(options, args) = config.parse_options(oparser)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
config.setup_logging(options)
|
|
||||||
conf, app = config.load_paste_app('glance-registry', options, args)
|
conf, app = config.load_paste_app('glance-registry', options, args)
|
||||||
|
|
||||||
server = wsgi.Server()
|
server = wsgi.Server()
|
||||||
|
@ -18,3 +18,49 @@ Configuring Glance
|
|||||||
==================
|
==================
|
||||||
|
|
||||||
.. todo:: Complete details of configuration with paste.deploy config files
|
.. todo:: Complete details of configuration with paste.deploy config files
|
||||||
|
|
||||||
|
Configuring Logging in Glance
|
||||||
|
-----------------------------
|
||||||
|
|
||||||
|
There are a number of configuration options in Glance that control how Glance
|
||||||
|
servers log messages. The configuration options are specified in the
|
||||||
|
``glance.conf`` config file.
|
||||||
|
|
||||||
|
* ``--log-config=PATH``
|
||||||
|
|
||||||
|
Optional. Default: ``None``
|
||||||
|
|
||||||
|
Specified on the command line only.
|
||||||
|
|
||||||
|
Takes a path to a configuration file to use for configuring logging.
|
||||||
|
|
||||||
|
* ``--log-format``
|
||||||
|
|
||||||
|
*Because of a bug in the PasteDeploy package, this option is only available
|
||||||
|
on the command line.*
|
||||||
|
|
||||||
|
Optional. Default: ``%(asctime)s %(levelname)8s [%(name)s] %(message)s``
|
||||||
|
|
||||||
|
The format of the log records. See the
|
||||||
|
`logging module <http://docs.python.org/library/logging.html>`_ documentation for
|
||||||
|
more information on setting this format string.
|
||||||
|
|
||||||
|
* ``log_file`` (``--log-file`` when specified on the command line)
|
||||||
|
|
||||||
|
The filepath of the file to use for logging messages from Glance's servers. If
|
||||||
|
missing, the default is to output messages to ``stdout``, so if you are running
|
||||||
|
Glance servers in a daemon mode (using ``glance-control``) you should make
|
||||||
|
sure that the ``log_file`` option is set appropriately.
|
||||||
|
|
||||||
|
* ``log_dir`` (``--log-dir`` when specified on the command line)
|
||||||
|
|
||||||
|
The filepath of the directory to use for log files. If not specified (the default)
|
||||||
|
the ``log_file`` is used as an absolute filepath.
|
||||||
|
|
||||||
|
* ``log_date_format`` (``--log-date-format`` when specified from the command line)
|
||||||
|
|
||||||
|
The format string for timestamps in the log output.
|
||||||
|
|
||||||
|
Defaults to ``%Y-%m-%d %H:%M:%S``. See the
|
||||||
|
`logging module <http://docs.python.org/library/logging.html>`_ documentation for
|
||||||
|
more information on setting this format string.
|
||||||
|
@ -35,8 +35,6 @@ import glance.common.exception as exception
|
|||||||
|
|
||||||
DEFAULT_LOG_FORMAT = "%(asctime)s %(levelname)8s [%(name)s] %(message)s"
|
DEFAULT_LOG_FORMAT = "%(asctime)s %(levelname)8s [%(name)s] %(message)s"
|
||||||
DEFAULT_LOG_DATE_FORMAT = "%Y-%m-%d %H:%M:%S"
|
DEFAULT_LOG_DATE_FORMAT = "%Y-%m-%d %H:%M:%S"
|
||||||
DEFAULT_LOG_HANDLER = 'stream'
|
|
||||||
LOGGING_HANDLER_CHOICES = ['syslog', 'file', 'stream']
|
|
||||||
|
|
||||||
|
|
||||||
def parse_options(parser, cli_args=None):
|
def parse_options(parser, cli_args=None):
|
||||||
@ -82,38 +80,7 @@ def add_common_options(parser):
|
|||||||
parser.add_option_group(group)
|
parser.add_option_group(group)
|
||||||
|
|
||||||
|
|
||||||
def add_daemon_options(parser):
|
def add_log_options(parser):
|
||||||
"""
|
|
||||||
Given a supplied optparse.OptionParser, adds an OptionGroup that
|
|
||||||
represents all the configuration options around daemonization.
|
|
||||||
|
|
||||||
:param parser: optparse.OptionParser
|
|
||||||
"""
|
|
||||||
help_text = "The following configuration options are specific to "\
|
|
||||||
"the daemonizing of this program."
|
|
||||||
|
|
||||||
group = optparse.OptionGroup(parser, "Daemon Options", help_text)
|
|
||||||
group.add_option('--config', default=None,
|
|
||||||
help="Configuration file to read when loading "
|
|
||||||
"application. If missing, the first argument is "
|
|
||||||
"used. If no arguments are found, then a set of "
|
|
||||||
"standard directories are searched for a config "
|
|
||||||
"file.")
|
|
||||||
group.add_option("--pid-file", default=None, metavar="PATH",
|
|
||||||
help="(Optional) Name of pid file for the server. "
|
|
||||||
"If not specified, the pid file will be named "
|
|
||||||
"/var/run/glance/<SERVER>.pid.")
|
|
||||||
group.add_option("--uid", type=int, default=os.getuid(),
|
|
||||||
help="uid under which to run. Default: %default")
|
|
||||||
group.add_option("--gid", type=int, default=os.getgid(),
|
|
||||||
help="gid under which to run. Default: %default")
|
|
||||||
group.add_option('--working-directory', '--working-dir',
|
|
||||||
default=os.path.abspath(os.getcwd()),
|
|
||||||
help="The working directory. Default: %default")
|
|
||||||
parser.add_option_group(group)
|
|
||||||
|
|
||||||
|
|
||||||
def add_log_options(prog_name, parser):
|
|
||||||
"""
|
"""
|
||||||
Given a supplied optparse.OptionParser, adds an OptionGroup that
|
Given a supplied optparse.OptionParser, adds an OptionGroup that
|
||||||
represents all the configuration options around logging.
|
represents all the configuration options around logging.
|
||||||
@ -130,29 +97,25 @@ def add_log_options(prog_name, parser):
|
|||||||
"any other logging options specified. Please see "
|
"any other logging options specified. Please see "
|
||||||
"the Python logging module documentation for "
|
"the Python logging module documentation for "
|
||||||
"details on logging configuration files.")
|
"details on logging configuration files.")
|
||||||
group.add_option('--log-handler', default=DEFAULT_LOG_HANDLER,
|
|
||||||
metavar="HANDLER",
|
|
||||||
choices=LOGGING_HANDLER_CHOICES,
|
|
||||||
help="What logging handler to use? "
|
|
||||||
"Default: %default")
|
|
||||||
group.add_option('--log-date-format', metavar="FORMAT",
|
group.add_option('--log-date-format', metavar="FORMAT",
|
||||||
default=DEFAULT_LOG_DATE_FORMAT,
|
default=DEFAULT_LOG_DATE_FORMAT,
|
||||||
help="Format string for %(asctime)s in log records. "
|
help="Format string for %(asctime)s in log records. "
|
||||||
"Default: %default")
|
"Default: %default")
|
||||||
group.add_option('--log-file', default="%s.log" % prog_name,
|
group.add_option('--log-file', default=None, metavar="PATH",
|
||||||
metavar="PATH",
|
help="(Optional) Name of log file to output to. "
|
||||||
help="(Optional) Name of log file to output to.")
|
"If not set, logging will go to stdout.")
|
||||||
group.add_option("--log-dir", default=None,
|
group.add_option("--log-dir", default=None,
|
||||||
help="(Optional) The directory to keep log files in "
|
help="(Optional) The directory to keep log files in "
|
||||||
"(will be prepended to --logfile)")
|
"(will be prepended to --logfile)")
|
||||||
parser.add_option_group(group)
|
parser.add_option_group(group)
|
||||||
|
|
||||||
|
|
||||||
def setup_logging(options):
|
def setup_logging(options, conf):
|
||||||
"""
|
"""
|
||||||
Sets up the logging options for a log with supplied name
|
Sets up the logging options for a log with supplied name
|
||||||
|
|
||||||
:param options: Mapping of typed option key/values
|
:param options: Mapping of typed option key/values
|
||||||
|
:param conf: Mapping of untyped key/values from config file
|
||||||
"""
|
"""
|
||||||
|
|
||||||
if options.get('log_config', None):
|
if options.get('log_config', None):
|
||||||
@ -182,27 +145,24 @@ def setup_logging(options):
|
|||||||
log_date_format = options.get('log_date_format', DEFAULT_LOG_DATE_FORMAT)
|
log_date_format = options.get('log_date_format', DEFAULT_LOG_DATE_FORMAT)
|
||||||
formatter = logging.Formatter(log_format, log_date_format)
|
formatter = logging.Formatter(log_format, log_date_format)
|
||||||
|
|
||||||
log_handler = options.get('log_handler', DEFAULT_LOG_HANDLER)
|
logfile = options.get('log_file')
|
||||||
if log_handler == 'syslog':
|
if not logfile:
|
||||||
syslog = logging.handlers.SysLogHandler(address='/dev/log')
|
logfile = conf.get('log_file')
|
||||||
syslog.setFormatter(formatter)
|
|
||||||
root_logger.addHandler(syslog)
|
if logfile:
|
||||||
elif log_handler == 'file':
|
logdir = options.get('log_dir')
|
||||||
logfile = options['log_file']
|
if not logdir:
|
||||||
logdir = options['log_dir']
|
logdir = conf.get('log_dir')
|
||||||
if logdir:
|
if logdir:
|
||||||
logfile = os.path.join(logdir, logfile)
|
logfile = os.path.join(logdir, logfile)
|
||||||
logfile = logging.FileHandler(logfile)
|
logfile = logging.FileHandler(logfile)
|
||||||
logfile.setFormatter(formatter)
|
logfile.setFormatter(formatter)
|
||||||
logfile.setFormatter(formatter)
|
logfile.setFormatter(formatter)
|
||||||
root_logger.addHandler(logfile)
|
root_logger.addHandler(logfile)
|
||||||
elif log_handler == 'stream':
|
else:
|
||||||
handler = logging.StreamHandler(sys.stdout)
|
handler = logging.StreamHandler(sys.stdout)
|
||||||
handler.setFormatter(formatter)
|
handler.setFormatter(formatter)
|
||||||
root_logger.addHandler(handler)
|
root_logger.addHandler(handler)
|
||||||
else:
|
|
||||||
raise exception.BadInputError(
|
|
||||||
"unrecognized log handler '%(log_handler)s'" % locals())
|
|
||||||
|
|
||||||
|
|
||||||
def find_config_file(options, args):
|
def find_config_file(options, args):
|
||||||
@ -270,6 +230,11 @@ def load_paste_app(app_name, options, args):
|
|||||||
"Cannot load application %s" % app_name)
|
"Cannot load application %s" % app_name)
|
||||||
try:
|
try:
|
||||||
conf = deploy.appconfig("config:%s" % conf_file, name=app_name)
|
conf = deploy.appconfig("config:%s" % conf_file, name=app_name)
|
||||||
|
|
||||||
|
# Setup logging early, supplying both the CLI options and the
|
||||||
|
# configuration mapping from the config file
|
||||||
|
setup_logging(options, conf)
|
||||||
|
|
||||||
# We only update the conf dict for the verbose and debug
|
# We only update the conf dict for the verbose and debug
|
||||||
# flags. Everything else must be set up in the conf file...
|
# flags. Everything else must be set up in the conf file...
|
||||||
conf['verbose'] = options['verbose']
|
conf['verbose'] = options['verbose']
|
||||||
|
@ -31,7 +31,7 @@ class TestMigrations(unittest.TestCase):
|
|||||||
os.unlink(self.db_path)
|
os.unlink(self.db_path)
|
||||||
self.options = dict(sql_connection="sqlite:///%s" % self.db_path,
|
self.options = dict(sql_connection="sqlite:///%s" % self.db_path,
|
||||||
verbose=False)
|
verbose=False)
|
||||||
config.setup_logging(self.options)
|
config.setup_logging(self.options, {})
|
||||||
|
|
||||||
def tearDown(self):
|
def tearDown(self):
|
||||||
if os.path.exists(self.db_path):
|
if os.path.exists(self.db_path):
|
||||||
|
@ -204,3 +204,148 @@ sql_idle_timeout = 3600
|
|||||||
cmd = "./bin/glance-control registry stop "\
|
cmd = "./bin/glance-control registry stop "\
|
||||||
"%s --pid-file=glance-registry.pid" % conf_file_name
|
"%s --pid-file=glance-registry.pid" % conf_file_name
|
||||||
ignored, out, err = execute(cmd)
|
ignored, out, err = execute(cmd)
|
||||||
|
|
||||||
|
|
||||||
|
# TODO(jaypipes): Move this to separate test file once
|
||||||
|
# LP Bug#731304 moves execute() out to a common file, etc
|
||||||
|
class TestLogging(unittest.TestCase):
|
||||||
|
|
||||||
|
"""Tests that logging can be configured correctly"""
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
self.logfiles = []
|
||||||
|
|
||||||
|
def tearDown(self):
|
||||||
|
self._cleanup_test_servers()
|
||||||
|
self._cleanup_log_files()
|
||||||
|
|
||||||
|
def _cleanup_test_servers(self):
|
||||||
|
# Clean up any leftover test servers...
|
||||||
|
pid_files = ('glance-api.pid', 'glance-registry.pid')
|
||||||
|
for pid_file in pid_files:
|
||||||
|
if os.path.exists(pid_file):
|
||||||
|
pid = int(open(pid_file).read().strip())
|
||||||
|
try:
|
||||||
|
os.killpg(pid, signal.SIGTERM)
|
||||||
|
except:
|
||||||
|
pass # Ignore if the process group is dead
|
||||||
|
os.unlink(pid_file)
|
||||||
|
|
||||||
|
def _cleanup_log_files(self):
|
||||||
|
for f in self.logfiles:
|
||||||
|
if os.path.exists(f):
|
||||||
|
os.unlink(f)
|
||||||
|
|
||||||
|
def test_logfile(self):
|
||||||
|
"""
|
||||||
|
A test that logging can be configured properly from the
|
||||||
|
glance.conf file with the log_file option.
|
||||||
|
|
||||||
|
We start both servers daemonized with a temporary config
|
||||||
|
file that has some logging options in it.
|
||||||
|
|
||||||
|
We then use curl to issue a few requests and verify that each server's
|
||||||
|
logging statements were logged to the one log file
|
||||||
|
"""
|
||||||
|
logfile = "/tmp/test_logfile.log"
|
||||||
|
self.logfiles.append(logfile)
|
||||||
|
|
||||||
|
if os.path.exists(logfile):
|
||||||
|
os.unlink(logfile)
|
||||||
|
|
||||||
|
self._cleanup_test_servers()
|
||||||
|
|
||||||
|
# Port numbers hopefully not used by anything...
|
||||||
|
api_port = 32001
|
||||||
|
reg_port = 32000
|
||||||
|
image_dir = "/tmp/test.images.%d" % api_port
|
||||||
|
if os.path.exists(image_dir):
|
||||||
|
shutil.rmtree(image_dir)
|
||||||
|
|
||||||
|
# A config file to use just for this test...we don't want
|
||||||
|
# to trample on currently-running Glance servers, now do we?
|
||||||
|
with tempfile.NamedTemporaryFile() as conf_file:
|
||||||
|
conf_contents = """[DEFAULT]
|
||||||
|
verbose = True
|
||||||
|
debug = True
|
||||||
|
log_file = %(logfile)s
|
||||||
|
|
||||||
|
[app:glance-api]
|
||||||
|
paste.app_factory = glance.server:app_factory
|
||||||
|
filesystem_store_datadir=%(image_dir)s
|
||||||
|
default_store = file
|
||||||
|
bind_host = 0.0.0.0
|
||||||
|
bind_port = %(api_port)s
|
||||||
|
registry_host = 0.0.0.0
|
||||||
|
registry_port = %(reg_port)s
|
||||||
|
|
||||||
|
[app:glance-registry]
|
||||||
|
paste.app_factory = glance.registry.server:app_factory
|
||||||
|
bind_host = 0.0.0.0
|
||||||
|
bind_port = %(reg_port)s
|
||||||
|
sql_connection = sqlite://
|
||||||
|
sql_idle_timeout = 3600
|
||||||
|
""" % locals()
|
||||||
|
conf_file.write(conf_contents)
|
||||||
|
conf_file.flush()
|
||||||
|
conf_file_name = conf_file.name
|
||||||
|
|
||||||
|
venv = ""
|
||||||
|
if 'VIRTUAL_ENV' in os.environ:
|
||||||
|
venv = "tools/with_venv.sh "
|
||||||
|
|
||||||
|
# Start up the API and default registry server
|
||||||
|
cmd = venv + "./bin/glance-control api start "\
|
||||||
|
"%s --pid-file=glance-api.pid" % conf_file_name
|
||||||
|
exitcode, out, err = execute(cmd)
|
||||||
|
|
||||||
|
self.assertEquals(0, exitcode)
|
||||||
|
self.assertTrue("Starting glance-api with" in out)
|
||||||
|
|
||||||
|
cmd = venv + "./bin/glance-control registry start "\
|
||||||
|
"%s --pid-file=glance-registry.pid" % conf_file_name
|
||||||
|
exitcode, out, err = execute(cmd)
|
||||||
|
|
||||||
|
self.assertEquals(0, exitcode)
|
||||||
|
self.assertTrue("Starting glance-registry with" in out)
|
||||||
|
|
||||||
|
time.sleep(2) # Gotta give some time for spin up...
|
||||||
|
|
||||||
|
cmd = "curl -X POST -H 'Content-Type: application/octet-stream' "\
|
||||||
|
"-H 'X-Image-Meta-Name: ImageName' "\
|
||||||
|
"-H 'X-Image-Meta-Disk-Format: Invalid' "\
|
||||||
|
"http://0.0.0.0:%d/images" % api_port
|
||||||
|
ignored, out, err = execute(cmd)
|
||||||
|
|
||||||
|
self.assertTrue('Invalid disk format' in out,
|
||||||
|
"Could not find 'Invalid disk format' "
|
||||||
|
"in output: %s" % out)
|
||||||
|
|
||||||
|
self.assertTrue(os.path.exists(logfile),
|
||||||
|
"Logfile %s does not exist!" % logfile)
|
||||||
|
|
||||||
|
logfile_contents = open(logfile, 'rb').read()
|
||||||
|
|
||||||
|
# Check that BOTH the glance API and registry server
|
||||||
|
# modules are logged to the file.
|
||||||
|
self.assertTrue('[glance.server]' in logfile_contents,
|
||||||
|
"Could not find '[glance.server]' "
|
||||||
|
"in logfile: %s" % logfile_contents)
|
||||||
|
self.assertTrue('[glance.registry.server]' in logfile_contents,
|
||||||
|
"Could not find '[glance.registry.server]' "
|
||||||
|
"in logfile: %s" % logfile_contents)
|
||||||
|
|
||||||
|
# Test that the error we caused above is in the log
|
||||||
|
self.assertTrue('Invalid disk format' in logfile_contents,
|
||||||
|
"Could not find 'Invalid disk format' "
|
||||||
|
"in logfile: %s" % logfile_contents)
|
||||||
|
|
||||||
|
# Check the log file for the log of the above error
|
||||||
|
|
||||||
|
# Spin down the API and default registry server
|
||||||
|
cmd = "./bin/glance-control api stop "\
|
||||||
|
"%s --pid-file=glance-api.pid" % conf_file_name
|
||||||
|
ignored, out, err = execute(cmd)
|
||||||
|
cmd = "./bin/glance-control registry stop "\
|
||||||
|
"%s --pid-file=glance-registry.pid" % conf_file_name
|
||||||
|
ignored, out, err = execute(cmd)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user