Commit defdcec1 authored by Jim Fulton's avatar Jim Fulton

Added logging support.

parent eb311a70
...@@ -16,6 +16,7 @@ ...@@ -16,6 +16,7 @@
$Id$ $Id$
""" """
import logging
import md5 import md5
import os import os
import pprint import pprint
...@@ -64,15 +65,17 @@ class Buildout(dict): ...@@ -64,15 +65,17 @@ class Buildout(dict):
super(Buildout, self).__init__() super(Buildout, self).__init__()
# default options # default options
data = dict(buildout={'directory': os.path.dirname(config_file), data = dict(buildout={
'directory': os.path.dirname(config_file),
'eggs-directory': 'eggs', 'eggs-directory': 'eggs',
'bin-directory': 'bin', 'bin-directory': 'bin',
'parts-directory': 'parts', 'parts-directory': 'parts',
'installed': '.installed.cfg', 'installed': '.installed.cfg',
'python': 'buildout', 'python': 'buildout',
'executable': sys.executable, 'executable': sys.executable,
}, 'log-level': 'WARNING',
) 'log-format': '%(name)s: %(message)s',
})
# load user defaults, which override defaults # load user defaults, which override defaults
if 'HOME' in os.environ: if 'HOME' in os.environ:
...@@ -116,8 +119,6 @@ class Buildout(dict): ...@@ -116,8 +119,6 @@ class Buildout(dict):
for name in ('bin', 'parts', 'eggs'): for name in ('bin', 'parts', 'eggs'):
d = self._buildout_path(options[name+'-directory']) d = self._buildout_path(options[name+'-directory'])
options[name+'-directory'] = d options[name+'-directory'] = d
if not os.path.exists(d):
os.mkdir(d)
options['installed'] = os.path.join(options['directory'], options['installed'] = os.path.join(options['directory'],
options['installed']) options['installed'])
...@@ -163,6 +164,15 @@ class Buildout(dict): ...@@ -163,6 +164,15 @@ class Buildout(dict):
return os.path.join(self._buildout_dir, *names) return os.path.join(self._buildout_dir, *names)
def install(self, install_parts): def install(self, install_parts):
# Create buildout directories
for name in ('bin', 'parts', 'eggs'):
d = self['buildout'][name+'-directory']
if not os.path.exists(d):
self._logger.info('Creating directory %s', d)
os.mkdir(d)
# Build develop eggs
self._develop() self._develop()
# load installed data # load installed data
...@@ -181,7 +191,7 @@ class Buildout(dict): ...@@ -181,7 +191,7 @@ class Buildout(dict):
if install_parts: if install_parts:
extra = [p for p in install_parts if p not in conf_parts] extra = [p for p in install_parts if p not in conf_parts]
if extra: if extra:
error('Invalid install parts:', *extra) self._error('Invalid install parts:', *extra)
uninstall_missing = False uninstall_missing = False
else: else:
install_parts = conf_parts install_parts = conf_parts
...@@ -206,12 +216,14 @@ class Buildout(dict): ...@@ -206,12 +216,14 @@ class Buildout(dict):
continue continue
# ununstall part # ununstall part
self._logger.info('Uninstalling %s', part)
self._uninstall( self._uninstall(
installed_part_options[part]['__buildout_installed__']) installed_part_options[part]['__buildout_installed__'])
installed_parts = [p for p in installed_parts if p != part] installed_parts = [p for p in installed_parts if p != part]
# install new parts # install new parts
for part in install_parts: for part in install_parts:
self._logger.info('Installing %s', part)
installed_part_options[part] = self[part].copy() installed_part_options[part] = self[part].copy()
del self[part]['__buildout_signature__'] del self[part]['__buildout_signature__']
installed_files = recipes[part].install() or () installed_files = recipes[part].install() or ()
...@@ -241,7 +253,7 @@ class Buildout(dict): ...@@ -241,7 +253,7 @@ class Buildout(dict):
setup = self._buildout_path(setup) setup = self._buildout_path(setup)
if os.path.isdir(setup): if os.path.isdir(setup):
setup = os.path.join(setup, 'setup.py') setup = os.path.join(setup, 'setup.py')
self._logger.info("Running %s -q develop ...", setup)
os.chdir(os.path.dirname(setup)) os.chdir(os.path.dirname(setup))
os.spawnle( os.spawnle(
os.P_WAIT, sys.executable, sys.executable, os.P_WAIT, sys.executable, sys.executable,
...@@ -349,6 +361,33 @@ class Buildout(dict): ...@@ -349,6 +361,33 @@ class Buildout(dict):
_save_options(part, installed_options[part], f) _save_options(part, installed_options[part], f)
f.close() f.close()
def _error(self, message, *args, **kw):
self._logger.error(message, *args, **kw)
sys.exit(1)
def _setup_logging(self):
root_logger = logging.getLogger()
handler = logging.StreamHandler(sys.stdout)
handler.setFormatter(logging.Formatter(self['buildout']['log-format']))
root_logger.addHandler(handler)
self._logger = logging.getLogger('buildout')
level = self['buildout']['log-level']
if level in ('DEBUG', 'INFO', 'WARNING', 'ERROR', 'CRITICAL'):
level = getattr(logging, level)
else:
try:
level = int(level)
except ValueError:
self._error("Invalid logging level %s", level)
verbosity = self['buildout'].get('verbosity', 0)
try:
verbosity = int(verbosity)
except ValueError:
self._error("Invalid verbosity %s", verbosity)
root_logger.setLevel(level-verbosity)
def _save_options(section, options, f): def _save_options(section, options, f):
print >>f, '[%s]' % section print >>f, '[%s]' % section
items = options.items() items = options.items()
...@@ -435,23 +474,47 @@ def _error(*message): ...@@ -435,23 +474,47 @@ def _error(*message):
def main(args=None): def main(args=None):
if args is None: if args is None:
args = sys.argv[1:] args = sys.argv[1:]
if args and args[0] == '-c':
args.pop(0)
if not args:
_error("No configuration file specified,")
config_file = args.pop(0)
else:
config_file = 'buildout.cfg'
config_file = 'buildout.cfg'
verbosity = 0
options = [] options = []
while args and '=' in args[0]: while args:
if args[0][0] == '-':
op = orig_op = args.pop(0)
op = op[1:]
while op and op[0] in 'vq':
if op[0] == 'v':
verbosity += 10
else:
verbosity -= 10
op = op[1:]
if op[:1] == 'c':
op = op[1:]
if op:
config_file = op
else:
if args:
config_file = args.pop(0)
else:
_error("No file name specified for option", orig_op)
elif op:
_error("Invalid option", '-'+op[0])
elif '=' in args[0]:
option, value = args.pop(0).split('=', 1) option, value = args.pop(0).split('=', 1)
if len(option.split(':')) != 2: if len(option.split(':')) != 2:
_error('Invalid option:', option) _error('Invalid option:', option)
section, option = option.split(':') section, option = option.split(':')
options.append((section.strip(), option.strip(), value.strip())) options.append((section.strip(), option.strip(), value.strip()))
else:
# We've run out of command-line options and option assignnemnts
# The rest should be commands, so we'll stop here
break
if verbosity:
options.append(('buildout', 'verbosity', str(verbosity)))
buildout = Buildout(config_file, options) buildout = Buildout(config_file, options)
buildout._setup_logging()
if args: if args:
command = args.pop(0) command = args.pop(0)
...@@ -460,7 +523,10 @@ def main(args=None): ...@@ -460,7 +523,10 @@ def main(args=None):
else: else:
command = 'install' command = 'install'
try:
getattr(buildout, command)(args) getattr(buildout, command)(args)
finally:
logging.shutdown()
if sys.version_info[:2] < (2, 4): if sys.version_info[:2] < (2, 4):
def reversed(iterable): def reversed(iterable):
......
...@@ -102,7 +102,7 @@ and then we'll create a source file for our mkdir recipe: ...@@ -102,7 +102,7 @@ and then we'll create a source file for our mkdir recipe:
>>> write(sample_buildout, 'recipes', 'mkdir.py', >>> write(sample_buildout, 'recipes', 'mkdir.py',
... """ ... """
... import os ... import logging, os
... ...
... class Mkdir: ... class Mkdir:
... ...
...@@ -118,7 +118,8 @@ and then we'll create a source file for our mkdir recipe: ...@@ -118,7 +118,8 @@ and then we'll create a source file for our mkdir recipe:
... def install(self): ... def install(self):
... path = self.options['path'] ... path = self.options['path']
... if not os.path.isdir(path): ... if not os.path.isdir(path):
... print 'Creating directory', os.path.basename(path) ... logging.getLogger(self.name).info(
... 'Creating directory %s', os.path.basename(path))
... os.mkdir(path) ... os.mkdir(path)
... return path ... return path
... """) ... """)
...@@ -151,8 +152,7 @@ The install method is responsible for creating the part. In this ...@@ -151,8 +152,7 @@ The install method is responsible for creating the part. In this
case, we need the path of the directory to create. We'll use a case, we need the path of the directory to create. We'll use a
path option from our options dictionary. path option from our options dictionary.
We made the method chatty so that we can observe what it's doing. The install method logs what it's doing using the Python logging call.
XXX use python logging module!
We return the path that we installed. If the part is unistalled or We return the path that we installed. If the part is unistalled or
reinstalled, then the path returned will be removed by the buildout reinstalled, then the path returned will be removed by the buildout
...@@ -203,6 +203,7 @@ Now let's update our buildout.cfg: ...@@ -203,6 +203,7 @@ Now let's update our buildout.cfg:
... [buildout] ... [buildout]
... develop = recipes ... develop = recipes
... parts = data_dir ... parts = data_dir
... log-level = INFO
... ...
... [data_dir] ... [data_dir]
... recipe = recipes:mkdir ... recipe = recipes:mkdir
...@@ -230,6 +231,14 @@ Here we've named a part to be "built". We can use any name we want ...@@ -230,6 +231,14 @@ Here we've named a part to be "built". We can use any name we want
except that different part names must be unique and recipes will often except that different part names must be unique and recipes will often
use the part name to decide what to do. use the part name to decide what to do.
::
log-level = INFO
The default level is WARNING, which is fairly quite. In this example,
we set the level to INFO so we can see more details about what the
buildout and recipes are doing.
:: ::
[data_dir] [data_dir]
...@@ -247,7 +256,9 @@ buildout: ...@@ -247,7 +256,9 @@ buildout:
>>> import os >>> import os
>>> os.chdir(sample_buildout) >>> os.chdir(sample_buildout)
>>> print system(os.path.join(sample_buildout, 'bin', 'buildout')), >>> print system(os.path.join(sample_buildout, 'bin', 'buildout')),
Creating directory mystuff buildout: Running /tmp/sample-buildout/recipes/setup.py -q develop ...
buildout: Installing data_dir
data_dir: Creating directory mystuff
We see that the recipe created the directory, as expected: We see that the recipe created the directory, as expected:
...@@ -285,6 +296,7 @@ we'll see that the directory gets removed and recreated: ...@@ -285,6 +296,7 @@ we'll see that the directory gets removed and recreated:
... [buildout] ... [buildout]
... develop = recipes ... develop = recipes
... parts = data_dir ... parts = data_dir
... log-level = INFO
... ...
... [data_dir] ... [data_dir]
... recipe = recipes:mkdir ... recipe = recipes:mkdir
...@@ -292,7 +304,10 @@ we'll see that the directory gets removed and recreated: ...@@ -292,7 +304,10 @@ we'll see that the directory gets removed and recreated:
... """) ... """)
>>> print system(os.path.join(sample_buildout, 'bin', 'buildout')), >>> print system(os.path.join(sample_buildout, 'bin', 'buildout')),
Creating directory mydata buildout: Running /tmp/sample-buildout/recipes/setup.py -q develop ...
buildout: Uninstalling data_dir
buildout: Installing data_dir
data_dir: Creating directory mydata
>>> ls(sample_buildout) >>> ls(sample_buildout)
- .installed.cfg - .installed.cfg
...@@ -358,6 +373,7 @@ examples: ...@@ -358,6 +373,7 @@ examples:
... [buildout] ... [buildout]
... develop = recipes ... develop = recipes
... parts = data_dir debug ... parts = data_dir debug
... log-level = INFO
... ...
... [debug] ... [debug]
... recipe = recipes:debug ... recipe = recipes:debug
...@@ -387,7 +403,11 @@ Now, if we run the buildout, we'll see the options with the values ...@@ -387,7 +403,11 @@ Now, if we run the buildout, we'll see the options with the values
substituted. substituted.
>>> print system(os.path.join(sample_buildout, 'bin', 'buildout')), >>> print system(os.path.join(sample_buildout, 'bin', 'buildout')),
Creating directory mydata buildout: Running /tmp/sample-buildout/recipes/setup.py -q develop ...
buildout: Uninstalling data_dir
buildout: Installing data_dir
data_dir: Creating directory mydata
buildout: Installing debug
base var base var
file1 mydata/file file1 mydata/file
file2 mydata/file.out file2 mydata/file.out
...@@ -401,6 +421,9 @@ recipe, so it assumed it could and reinstalled mydata. If we rerun ...@@ -401,6 +421,9 @@ recipe, so it assumed it could and reinstalled mydata. If we rerun
the buildout: the buildout:
>>> print system(os.path.join(sample_buildout, 'bin', 'buildout')), >>> print system(os.path.join(sample_buildout, 'bin', 'buildout')),
buildout: Running /tmp/sample-buildout/recipes/setup.py -q develop ...
buildout: Installing data_dir
buildout: Installing debug
base var base var
file1 mydata/file file1 mydata/file
file2 mydata/file.out file2 mydata/file.out
...@@ -409,6 +432,9 @@ the buildout: ...@@ -409,6 +432,9 @@ the buildout:
We can see that mydata was not recreated. We can see that mydata was not recreated.
Note that, in this vase, we didn't specify a log level, so
we didn't get output about what the buildout was doing.
Multiple configuration files Multiple configuration files
---------------------------- ----------------------------
...@@ -615,14 +641,23 @@ Command-line usage ...@@ -615,14 +641,23 @@ Command-line usage
A number of arguments can be given on the buildout command line. The A number of arguments can be given on the buildout command line. The
command usage is:: command usage is::
buildout [-c file] [options] [command [command arguments]] buildout [-c file] [-q] [-v] [assignments] [command [command arguments]]
The -c option can be used to specify a configuration file, rather than The -c option can be used to specify a configuration file, rather than
buildout.cfg in the current durectory. Options are of the form:: buildout.cfg in the current durectory.
The -q and -v decrement and incremement the verbosity by 10. The
verbosity is used to adjust the logging level. The verbosity is
subtracted from the numeric value of the log-level option specified in
the configuration file.
Assignments are of the form::
section_name:option_name=value section_name:option_name=value
for example: Options and assignments can be given in any order.
Here's an example:
>>> write(sample_buildout, 'other.cfg', >>> write(sample_buildout, 'other.cfg',
... """ ... """
...@@ -640,12 +675,33 @@ Note that we used the installed buildout option to specify an ...@@ -640,12 +675,33 @@ Note that we used the installed buildout option to specify an
alternate file to store information about installed parts. alternate file to store information about installed parts.
>>> print system(os.path.join(sample_buildout, 'bin', 'buildout') >>> print system(os.path.join(sample_buildout, 'bin', 'buildout')
... + ' -c other.cfg debug:op1=foo'), ... + ' -c other.cfg debug:op1=foo -v'),
buildout: Running /tmp/sample-buildout/recipes/setup.py -q develop ...
buildout: Installing debug
name other
op1 foo
op7 7
recipe recipes:debug
Here we used the -c option to specify an alternate configuration file,
and the -v option to increase the level of logging from the default,
WARNING.
Options can also be combined in the usual Unix way, as in:
>>> print system(os.path.join(sample_buildout, 'bin', 'buildout')
... + ' -vcother.cfg debug:op1=foo'),
buildout: Running /tmp/sample-buildout/recipes/setup.py -q develop ...
buildout: Installing debug
name other name other
op1 foo op1 foo
op7 7 op7 7
recipe recipes:debug recipe recipes:debug
Here we combined the -v and -c options with the configuration file
name. Note that the -c option has to be last, because it takes an
argument.
>>> os.remove(os.path.join(sample_buildout, 'other.cfg')) >>> os.remove(os.path.join(sample_buildout, 'other.cfg'))
>>> os.remove(os.path.join(sample_buildout, '.other.cfg')) >>> os.remove(os.path.join(sample_buildout, '.other.cfg'))
...@@ -677,13 +733,19 @@ the buildout in the usual way: ...@@ -677,13 +733,19 @@ the buildout in the usual way:
... recipe = recipes:debug ... recipe = recipes:debug
... """) ... """)
>>> print system(os.path.join(sample_buildout, 'bin', 'buildout')), >>> print system(os.path.join(sample_buildout, 'bin', 'buildout') + ' -v'),
buildout: Running /sample-buildout/recipes/setup.py -q develop ...
buildout: Uninstalling debug
buildout: Installing debug
op1 1 op1 1
op7 7 op7 7
recipe recipes:debug recipe recipes:debug
Creating directory d1 buildout: Installing d1
Creating directory d2 d1: Creating directory d1
Creating directory d3 buildout: Installing d2
d2: Creating directory d2
buildout: Installing d3
d3: Creating directory d3
>>> ls(sample_buildout) >>> ls(sample_buildout)
- .installed.cfg - .installed.cfg
...@@ -756,10 +818,14 @@ Now we'll update our configuration file: ...@@ -756,10 +818,14 @@ Now we'll update our configuration file:
and run the buildout specifying just d2 and d3: and run the buildout specifying just d2 and d3:
>>> print system(os.path.join(sample_buildout, 'bin', 'buildout') >>> print system(os.path.join(sample_buildout, 'bin', 'buildout' + ' -v')
... + ' install d3 d4'), ... + ' install d3 d4'),
Creating directory data3 buildout: Running /sample-buildout/recipes/setup.py -q develop ...
Creating directory data4 buildout: Uninstalling d3
buildout: Installing d3
d3: Creating directory data3
buildout: Installing d4
d4: Creating directory data4
>>> ls(sample_buildout) >>> ls(sample_buildout)
- .installed.cfg - .installed.cfg
...@@ -821,14 +887,22 @@ Note that the installed data for debug, d1, and d2 haven't changed, ...@@ -821,14 +887,22 @@ Note that the installed data for debug, d1, and d2 haven't changed,
because we didn't install those parts and that the d1 and d2 because we didn't install those parts and that the d1 and d2
directories are still there. directories are still there.
Now, if we run the buildout without arguments: Now, if we run the buildout without the install command:
>>> print system(os.path.join(sample_buildout, 'bin', 'buildout')), >>> print system(os.path.join(sample_buildout, 'bin', 'buildout') + ' -v'),
buildout: Running /sample-buildout/recipes/setup.py -q develop ...
buildout: Uninstalling d1
buildout: Uninstalling d2
buildout: Uninstalling debug
buildout: Installing debug
op1 1 op1 1
op7 7 op7 7
recipe recipes:debug recipe recipes:debug
x 1 x 1
Creating directory data2 buildout: Installing d2
d2: Creating directory data2
buildout: Installing d3
buildout: Installing d4
We see the output of the debug recipe and that data2 was created. We We see the output of the debug recipe and that data2 was created. We
also see that d1 and d2 have gone away: also see that d1 and d2 have gone away:
...@@ -855,7 +929,7 @@ The buildout normally puts the bin, eggs, and parts directories in the ...@@ -855,7 +929,7 @@ The buildout normally puts the bin, eggs, and parts directories in the
directory in the directory containing the configuration file. You can directory in the directory containing the configuration file. You can
provide alternate locations, and even names for these directories. provide alternate locations, and even names for these directories.
>>> alt = tempfile.mkdtemp() >>> alt = tempfile.mkdtemp('sample-alt')
>>> write(sample_buildout, 'buildout.cfg', >>> write(sample_buildout, 'buildout.cfg',
... """ ... """
...@@ -871,7 +945,15 @@ provide alternate locations, and even names for these directories. ...@@ -871,7 +945,15 @@ provide alternate locations, and even names for these directories.
... work = os.path.join(alt, 'work'), ... work = os.path.join(alt, 'work'),
... )) ... ))
>>> print system(os.path.join(sample_buildout, 'bin', 'buildout')), >>> print system(os.path.join(sample_buildout, 'bin', 'buildout') + ' -v'),
buildout: Creating directory /tmp/sample-alt/scripts
buildout: Creating directory /tmp/sample-alt/work
buildout: Creating directory /tmp/sample-alt/basket
buildout: Running /tmp/sample-buildout/recipes/setup.py -q develop ...
buildout: Uninstalling d4
buildout: Uninstalling d3
buildout: Uninstalling d2
buildout: Uninstalling debug
>>> ls(alt) >>> ls(alt)
d basket d basket
...@@ -886,7 +968,7 @@ provide alternate locations, and even names for these directories. ...@@ -886,7 +968,7 @@ provide alternate locations, and even names for these directories.
You can also specify an alternate buildout directory: You can also specify an alternate buildout directory:
>>> alt = tempfile.mkdtemp() >>> alt = tempfile.mkdtemp('sample-alt')
>>> write(sample_buildout, 'buildout.cfg', >>> write(sample_buildout, 'buildout.cfg',
... """ ... """
...@@ -899,7 +981,11 @@ You can also specify an alternate buildout directory: ...@@ -899,7 +981,11 @@ You can also specify an alternate buildout directory:
... recipes=os.path.join(sample_buildout, 'recipes'), ... recipes=os.path.join(sample_buildout, 'recipes'),
... )) ... ))
>>> print system(os.path.join(sample_buildout, 'bin', 'buildout')), >>> print system(os.path.join(sample_buildout, 'bin', 'buildout') + ' -v'),
buildout: Creating directory /tmp/sample-alt/bin
buildout: Creating directory /tmp/sample-alt/parts
buildout: Creating directory /tmp/sample-alt/eggs
buildout: Running /tmp/sample-buildout/recipes/setup.py -q develop ...
>>> ls(alt) >>> ls(alt)
- .installed.cfg - .installed.cfg
...@@ -912,3 +998,41 @@ You can also specify an alternate buildout directory: ...@@ -912,3 +998,41 @@ You can also specify an alternate buildout directory:
>>> import shutil >>> import shutil
>>> shutil.rmtree(alt) >>> shutil.rmtree(alt)
Logging control
---------------
Three buildout options are used to control logging:
log-level
specifies the log level
verbosity
adjusts the log level
log-format
allows an alternate logging for mat to be specified
We've already seen the log level and verbosity. Let's look at an example
of changing the format:
>>> write(sample_buildout, 'buildout.cfg',
... """
... [buildout]
... develop = recipes
... parts =
... log-level = 25
... verbosity = 5
... log-format = %%(levelname)s %%(message)s
... """)
Here, we've changed the format to include the log-level name, rather
than the logger name.
We've also illustrated, with a contrived example, that the log level
can be a numeric value and that the verbosity can be specified in the
configuration file. Because the verbosoty is subtracted from the log
level, we get a final log level of 20, which is the INFO level.
>>> print system(os.path.join(sample_buildout, 'bin', 'buildout')),
INFO Running /tmp/sample-buildout/recipes/setup.py -q develop ...
...@@ -24,8 +24,6 @@ ...@@ -24,8 +24,6 @@
- Local download cache - Local download cache
- Logging
- Some way to freeze versions so we can have reproducable buildouts. - Some way to freeze versions so we can have reproducable buildouts.
- Part dependencies - Part dependencies
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment