1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
|
#!/usr/bin/python2.5
#
# Copyright 2009 Canonical Ltd. This software is licensed under the
# GNU Affero General Public License version 3 (see the file LICENSE).
"""Create lazr.config schema and confs from ZConfig data."""
__metatype__ = type
# Scripts may have relative imports.
# pylint: disable-msg=W0403
import _pythonpath
import os
import sys
from optparse import OptionParser
from operator import attrgetter
from textwrap import dedent
import canonical.config
from lazr.config import ImplicitTypeSchema
_schema_dir = os.path.abspath(os.path.dirname(canonical.config.__file__))
_root = os.path.dirname(os.path.dirname(os.path.dirname(_schema_dir)))
class Configuration:
"""A lazr.config configuration."""
_schema_path = os.path.join(_schema_dir, 'schema-lazr.conf')
def __init__(self, config):
self.config = config
@classmethod
def load(cls, conf_path, schema_path=None):
"""Initialise the Configuration.
:conf_path: The path to the lazr.config conf file.
:schema_path: The path to the lazr.config schema that defines
the configuration.
"""
if schema_path is None:
schema_path = cls._schema_path
schema = ImplicitTypeSchema(schema_path)
return cls(schema.load(conf_path))
def config_file_for_value(self, section, key):
"""Return the local path to the file that sets the section key."""
conf_file_name = self.config.schema.filename
value = section[key]
previous_config_data = self.config.data
# Walk the stack of config_data until a change is found.
for config_data in self.config.overlays:
if (section.name in config_data
and config_data[section.name][key] != value):
conf_file_name = previous_config_data.filename
break
previous_config_data = config_data
conf_path = os.path.abspath(conf_file_name)
return conf_path[len(_root) + 1:]
def list_config(self, verbose=False, section_name=None):
"""Print all the sections and keys in a configuration.
Print the final state of configuration after all the conf files
are loaded.
:param verbose: If True, each key has a comment stating where it
was defined.
:param section_name: Only print the named section.
"""
print '# This configuration derives from:'
for config_data in self.config.overlays:
print '# %s' % config_data.filename
print
name_key = attrgetter('name')
for count, section in enumerate(sorted(self.config, key=name_key)):
if section_name is not None and section_name != section.name:
continue
if count > 0:
# Separate sections by a blank line, or two when verbose.
print
print '[%s]' % section.name
if verbose and section.optional:
print '# This section is optional.\n'
for count, key in enumerate(sorted(section)):
if verbose:
if count > 0:
# Separate keys by a blank line.
print
conf_file_name = self.config_file_for_value(section, key)
print '# Defined in: %s' % conf_file_name
print '%s: %s' % (key, section[key])
def get_option_parser():
"""Return the option parser for this program."""
usage = dedent(""" %prog [options] lazr-config.conf
List all the sections and keys in an environment's lazr configuration.
The configuration is assembled from the schema and conf files. Verbose
annotates each key with the location of the file that set its value.
The 'section' option limits the list to just the named section.""")
parser = OptionParser(usage=usage)
parser.add_option(
"-l", "--schema", dest="schema_path",
help="the path to the lazr.config schema file")
parser.add_option(
"-v", "--verbose", action="store_true",
help="explain where the section and keys are set")
parser.add_option(
"-s", "--section", dest="section_name",
help="restrict the listing to the section")
parser.add_option(
'-i', "--instance", dest="instance_name",
help="the configuration instance to use")
return parser
def main(argv=None):
"""Run the command line operations."""
if argv is None:
argv = sys.argv
parser = get_option_parser()
(options, arguments) = parser.parse_args(args=argv[1:])
if len(arguments) == 0:
canonical_config = canonical.config.config
if options.instance_name:
canonical_config.setInstance(options.instance_name)
canonical_config._getConfig()
configuration = Configuration(canonical_config._config)
elif len(arguments) == 1:
conf_path = arguments[0]
configuration = Configuration.load(conf_path, options.schema_path)
else:
parser.error('Too many arguments.')
# Does not return.
configuration.list_config(
verbose=options.verbose, section_name=options.section_name)
if __name__ == '__main__':
sys.exit(main())
|