mirror of
https://salsa.debian.org/freeipa-team/freeipa.git
synced 2024-12-26 17:01:14 -06:00
537 lines
16 KiB
Python
537 lines
16 KiB
Python
# Authors:
|
|
# Martin Nagy <mnagy@redhat.com>
|
|
#
|
|
# Copyright (C) 2008 Red Hat
|
|
# see file 'COPYING' for use and warranty information
|
|
#
|
|
# This program 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; version 2 only
|
|
#
|
|
# This program 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 this program; if not, write to the Free Software
|
|
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
"""
|
|
Basic configuration management.
|
|
|
|
This module handles the reading and representation of basic local settings.
|
|
It will also take care of settings that can be discovered by different
|
|
methods, such as DNS.
|
|
"""
|
|
|
|
from ConfigParser import SafeConfigParser, ParsingError, RawConfigParser
|
|
import types
|
|
import os
|
|
from os import path
|
|
import sys
|
|
from errors import check_isinstance, raise_TypeError
|
|
import constants
|
|
|
|
DEFAULT_CONF='/etc/ipa/ipa.conf'
|
|
|
|
|
|
class Environment(object):
|
|
"""
|
|
A mapping object used to store the environment variables.
|
|
"""
|
|
|
|
def __init__(self):
|
|
object.__setattr__(self, '_Environment__map', {})
|
|
|
|
def __getattr__(self, name):
|
|
"""
|
|
Return the attribute named ``name``.
|
|
"""
|
|
return self[name]
|
|
|
|
def __setattr__(self, name, value):
|
|
"""
|
|
Set the attribute named ``name`` to ``value``.
|
|
"""
|
|
self[name] = value
|
|
|
|
def __delattr__(self, name):
|
|
"""
|
|
Raise AttributeError (deletion is not allowed).
|
|
"""
|
|
raise AttributeError('cannot del %s.%s' %
|
|
(self.__class__.__name__, name)
|
|
)
|
|
|
|
def __getitem__(self, key):
|
|
"""
|
|
Return the value corresponding to ``key``.
|
|
"""
|
|
val = self.__map[key]
|
|
if hasattr(val, 'get_value'):
|
|
return val.get_value()
|
|
else:
|
|
return val
|
|
|
|
def __setitem__(self, key, value):
|
|
"""
|
|
Set the item at ``key`` to ``value``.
|
|
"""
|
|
if key in self or hasattr(self, key):
|
|
if hasattr(self.__map[key], 'set_value'):
|
|
self.__map[key].set_value(value)
|
|
else:
|
|
raise AttributeError('cannot overwrite %s.%s' %
|
|
(self.__class__.__name__, key)
|
|
)
|
|
else:
|
|
self.__map[key] = value
|
|
|
|
def __contains__(self, key):
|
|
"""
|
|
Return True if instance contains ``key``; otherwise return False.
|
|
"""
|
|
return key in self.__map
|
|
|
|
def __iter__(self):
|
|
"""
|
|
Iterate through keys in ascending order.
|
|
"""
|
|
for key in sorted(self.__map):
|
|
yield key
|
|
|
|
def update(self, new_vals, ignore_errors = False):
|
|
"""
|
|
Update variables using keys and values from ``new_vals``.
|
|
|
|
Error will occur if there is an attempt to override variable that was
|
|
already set, unless``ignore_errors`` is True.
|
|
"""
|
|
assert type(new_vals) == dict
|
|
for key, value in new_vals.iteritems():
|
|
if ignore_errors:
|
|
try:
|
|
self[key] = value
|
|
except (AttributeError, KeyError):
|
|
pass
|
|
else:
|
|
self[key] = value
|
|
|
|
def get(self, name, default=None):
|
|
"""
|
|
Return the value corresponding to ``key``. Defaults to ``default``.
|
|
"""
|
|
if name in self:
|
|
return self[name]
|
|
else:
|
|
return default
|
|
|
|
|
|
class Env(object):
|
|
"""
|
|
A mapping object used to store the environment variables.
|
|
"""
|
|
|
|
__locked = False
|
|
|
|
def __init__(self):
|
|
object.__setattr__(self, '_Env__d', {})
|
|
object.__setattr__(self, '_Env__done', set())
|
|
self.ipalib = path.dirname(path.abspath(__file__))
|
|
self.site_packages = path.dirname(self.ipalib)
|
|
self.script = path.abspath(sys.argv[0])
|
|
self.bin = path.dirname(self.script)
|
|
self.home = path.abspath(os.environ['HOME'])
|
|
self.dot_ipa = path.join(self.home, '.ipa')
|
|
|
|
def __doing(self, name):
|
|
if name in self.__done:
|
|
raise StandardError(
|
|
'%s.%s() already called' % (self.__class__.__name__, name)
|
|
)
|
|
self.__done.add(name)
|
|
|
|
def __do_if_not_done(self, name):
|
|
if name not in self.__done:
|
|
getattr(self, name)()
|
|
|
|
def _isdone(self, name):
|
|
return name in self.__done
|
|
|
|
def _bootstrap(self, **overrides):
|
|
"""
|
|
Initialize basic environment.
|
|
|
|
This method will initialize only enough environment information to
|
|
determine whether ipa is running in-tree, what the context is,
|
|
and the location of the configuration file.
|
|
"""
|
|
self.__doing('_bootstrap')
|
|
for (key, value) in overrides.iteritems():
|
|
self[key] = value
|
|
if 'in_tree' not in self:
|
|
if self.bin == self.site_packages and \
|
|
path.isfile(path.join(self.bin, 'setup.py')):
|
|
self.in_tree = True
|
|
else:
|
|
self.in_tree = False
|
|
if 'context' not in self:
|
|
self.context = 'default'
|
|
if self.in_tree:
|
|
base = self.dot_ipa
|
|
else:
|
|
base = path.join('/', 'etc', 'ipa')
|
|
if 'conf' not in self:
|
|
self.conf = path.join(base, '%s.conf' % self.context)
|
|
if 'conf_default' not in self:
|
|
self.conf_default = path.join(base, 'default.conf')
|
|
if 'conf_dir' not in self:
|
|
self.conf_dir = base
|
|
|
|
def _finalize_core(self, **defaults):
|
|
"""
|
|
Complete initialization of standard IPA environment.
|
|
|
|
After this method is called, the all environment variables
|
|
used by all the built-in plugins will be available.
|
|
|
|
This method should be called before loading any plugins. It will
|
|
automatically call `Env._bootstrap()` if it has not yet been called.
|
|
|
|
After this method has finished, the `Env` instance is still writable
|
|
so that third
|
|
"""
|
|
self.__doing('_finalize_core')
|
|
self.__do_if_not_done('_bootstrap')
|
|
if self.__d.get('mode', None) != 'dummy':
|
|
self._merge_config(self.conf)
|
|
self._merge_config(self.conf_default)
|
|
if 'in_server' not in self:
|
|
self.in_server = (self.context == 'server')
|
|
if 'log' not in self:
|
|
name = '%s.log' % self.context
|
|
if self.in_tree or self.context == 'cli':
|
|
self.log = path.join(self.dot_ipa, 'log', name)
|
|
else:
|
|
self.log = path.join('/', 'var', 'log', 'ipa', name)
|
|
for (key, value) in defaults.iteritems():
|
|
if key not in self:
|
|
self[key] = value
|
|
|
|
def _finalize(self, **lastchance):
|
|
"""
|
|
Finalize and lock environment.
|
|
|
|
This method should be called after all plugins have bean loaded and
|
|
after `plugable.API.finalize()` has been called.
|
|
"""
|
|
self.__doing('_finalize')
|
|
self.__do_if_not_done('_finalize_core')
|
|
for (key, value) in lastchance.iteritems():
|
|
if key not in self:
|
|
self[key] = value
|
|
self.__lock__()
|
|
|
|
def _merge_config(self, conf_file):
|
|
"""
|
|
Merge values from ``conf_file`` into this `Env`.
|
|
"""
|
|
section = constants.CONFIG_SECTION
|
|
if not path.isfile(conf_file):
|
|
return
|
|
parser = RawConfigParser()
|
|
try:
|
|
parser.read(conf_file)
|
|
except ParsingError:
|
|
return
|
|
if not parser.has_section(section):
|
|
parser.add_section(section)
|
|
items = parser.items(section)
|
|
if len(items) == 0:
|
|
return
|
|
i = 0
|
|
for (key, value) in items:
|
|
if key not in self:
|
|
self[key] = value
|
|
i += 1
|
|
return (i, len(items))
|
|
|
|
def __lock__(self):
|
|
"""
|
|
Prevent further changes to environment.
|
|
"""
|
|
if self.__locked is True:
|
|
raise StandardError(
|
|
'%s.__lock__() already called' % self.__class__.__name__
|
|
)
|
|
object.__setattr__(self, '_Env__locked', True)
|
|
|
|
def __islocked__(self):
|
|
return self.__locked
|
|
|
|
def __getattr__(self, name):
|
|
"""
|
|
Return the attribute named ``name``.
|
|
"""
|
|
if name in self.__d:
|
|
return self[name]
|
|
raise AttributeError('%s.%s' %
|
|
(self.__class__.__name__, name)
|
|
)
|
|
|
|
def __setattr__(self, name, value):
|
|
"""
|
|
Set the attribute named ``name`` to ``value``.
|
|
"""
|
|
self[name] = value
|
|
|
|
def __delattr__(self, name):
|
|
"""
|
|
Raise AttributeError (deletion is not allowed).
|
|
"""
|
|
raise AttributeError('cannot del %s.%s' %
|
|
(self.__class__.__name__, name)
|
|
)
|
|
|
|
def __getitem__(self, key):
|
|
"""
|
|
Return the value corresponding to ``key``.
|
|
"""
|
|
if key not in self.__d:
|
|
raise KeyError(key)
|
|
value = self.__d[key]
|
|
if callable(value):
|
|
return value()
|
|
return value
|
|
|
|
def __setitem__(self, key, value):
|
|
"""
|
|
Set ``key`` to ``value``.
|
|
"""
|
|
# FIXME: the key should be checked with check_name()
|
|
if self.__locked:
|
|
raise AttributeError('locked: cannot set %s.%s to %r' %
|
|
(self.__class__.__name__, key, value)
|
|
)
|
|
if key in self.__d or hasattr(self, key):
|
|
raise AttributeError('cannot overwrite %s.%s with %r' %
|
|
(self.__class__.__name__, key, value)
|
|
)
|
|
if not callable(value):
|
|
if isinstance(value, basestring):
|
|
value = str(value.strip())
|
|
if value.lower() == 'true':
|
|
value = True
|
|
elif value.lower() == 'false':
|
|
value = False
|
|
elif value.isdigit():
|
|
value = int(value)
|
|
assert type(value) in (str, int, bool)
|
|
object.__setattr__(self, key, value)
|
|
self.__d[key] = value
|
|
|
|
def __contains__(self, key):
|
|
"""
|
|
Return True if instance contains ``key``; otherwise return False.
|
|
"""
|
|
return key in self.__d
|
|
|
|
def __len__(self):
|
|
"""
|
|
Return number of variables currently set.
|
|
"""
|
|
return len(self.__d)
|
|
|
|
def __iter__(self):
|
|
"""
|
|
Iterate through keys in ascending order.
|
|
"""
|
|
for key in sorted(self.__d):
|
|
yield key
|
|
|
|
|
|
def set_default_env(env):
|
|
"""
|
|
Set default values for ``env``.
|
|
"""
|
|
assert isinstance(env, Environment)
|
|
|
|
default = dict(
|
|
basedn = EnvProp(basestring, 'dc=example,dc=com'),
|
|
container_accounts = EnvProp(basestring, 'cn=accounts'),
|
|
container_user = EnvProp(basestring, 'cn=users,cn=accounts'),
|
|
container_group = EnvProp(basestring, 'cn=groups,cn=accounts'),
|
|
container_service = EnvProp(basestring, 'cn=services,cn=accounts'),
|
|
container_host = EnvProp(basestring, 'cn=computers,cn=accounts'),
|
|
container_hostgroup = EnvProp(basestring, 'cn=hostgroups,cn=accounts'),
|
|
domain = LazyProp(basestring, get_domain),
|
|
interactive = EnvProp(bool, True),
|
|
query_dns = EnvProp(bool, True),
|
|
realm = LazyProp(basestring, get_realm),
|
|
server_context = EnvProp(bool, True),
|
|
server = LazyIter(basestring, get_servers),
|
|
verbose = EnvProp(bool, False),
|
|
ldaphost = EnvProp(basestring, 'localhost'),
|
|
ldapport = EnvProp(int, 389),
|
|
)
|
|
|
|
env.update(default)
|
|
|
|
|
|
class EnvProp(object):
|
|
"""
|
|
Environment set-once property with optional default value.
|
|
"""
|
|
def __init__(self, type_, default, multi_value=False):
|
|
"""
|
|
:param type_: Type of the property.
|
|
:param default: Default value.
|
|
:param multi_value: Allow multiple values.
|
|
"""
|
|
if multi_value:
|
|
if isinstance(default, tuple) and len(default):
|
|
check_isinstance(default[0], type_, allow_none=True)
|
|
self._type = type_
|
|
self._default = default
|
|
self._value = None
|
|
self._multi_value = multi_value
|
|
|
|
def get_value(self):
|
|
"""
|
|
Return the value if it was set.
|
|
|
|
If the value is not set return the default. Otherwise raise an
|
|
exception.
|
|
"""
|
|
if self._get() != None:
|
|
return self._get()
|
|
else:
|
|
raise KeyError, 'Value not set'
|
|
|
|
def set_value(self, value):
|
|
"""
|
|
Set the value.
|
|
"""
|
|
if self._value != None:
|
|
raise KeyError, 'Value already set'
|
|
self._value = self._validate(value)
|
|
|
|
def _get(self):
|
|
"""
|
|
Return value, default, or None.
|
|
"""
|
|
if self._value != None:
|
|
return self._value
|
|
elif self._default != None:
|
|
return self._default
|
|
else:
|
|
return None
|
|
|
|
def _validate(self, value):
|
|
"""
|
|
Make sure ``value`` is of the right type. Do conversions if necessary.
|
|
|
|
This will also handle multi value.
|
|
"""
|
|
if self._multi_value and isinstance(value, tuple):
|
|
converted = []
|
|
for val in value:
|
|
converted.append(self._validate_value(val))
|
|
return tuple(converted)
|
|
else:
|
|
return self._validate_value(value)
|
|
|
|
def _validate_value(self, value):
|
|
"""
|
|
Validate and convert a single value.
|
|
"""
|
|
bool_true = ('true', 'yes', 'on')
|
|
bool_false = ('false', 'no', 'off')
|
|
|
|
if self._type == bool and isinstance(value, basestring):
|
|
if value.lower() in bool_true:
|
|
return True
|
|
elif value.lower() in bool_false:
|
|
return False
|
|
else:
|
|
raise raise_TypeError(value, bool, 'value')
|
|
check_isinstance(value, self._type, 'value')
|
|
return value
|
|
|
|
|
|
class LazyProp(EnvProp):
|
|
def __init__(self, type_, func, default=None, multi_value=False):
|
|
check_isinstance(func, types.FunctionType, 'func')
|
|
self._func = func
|
|
EnvProp.__init__(self, type_, default, multi_value)
|
|
|
|
def get_value(self):
|
|
if self._get() != None:
|
|
return self._get()
|
|
else:
|
|
return self._func()
|
|
|
|
|
|
class LazyIter(LazyProp):
|
|
def __init__(self, type_, func, default=None):
|
|
LazyProp.__init__(self, type_, func, default, multi_value=True)
|
|
|
|
def get_value(self):
|
|
val = self._get()
|
|
if val != None:
|
|
if type(val) == tuple:
|
|
for item in val:
|
|
yield item
|
|
else:
|
|
yield val
|
|
for item in self._func():
|
|
if not val or item not in val:
|
|
yield item
|
|
|
|
|
|
# TODO: Make it possible to use var = 'foo, bar' without
|
|
# turning it into ("'foo", "bar'")
|
|
def read_config(config_file=None):
|
|
assert config_file == None or isinstance(config_file, (basestring, file))
|
|
|
|
parser = SafeConfigParser()
|
|
if config_file == None:
|
|
files = [DEFAULT_CONF, os.path.expanduser('~/.ipa.conf')]
|
|
else:
|
|
files = [config_file]
|
|
|
|
for f in files:
|
|
try:
|
|
if isinstance(f, file):
|
|
parser.readfp(f)
|
|
else:
|
|
parser.read(f)
|
|
except ParsingError:
|
|
print "Can't read %s" % f
|
|
|
|
ret = {}
|
|
if parser.has_section('defaults'):
|
|
for name, value in parser.items('defaults'):
|
|
value = tuple(elem.strip() for elem in value.split(','))
|
|
if len(value) == 1:
|
|
value = value[0]
|
|
ret[name] = value
|
|
|
|
return ret
|
|
|
|
|
|
# these functions are here just to "emulate" dns resolving for now
|
|
def get_domain():
|
|
return "ipatest.com"
|
|
|
|
|
|
def get_realm():
|
|
return "IPATEST.COM"
|
|
|
|
|
|
def get_servers():
|
|
yield "server.ipatest.com"
|
|
yield "backup.ipatest.com"
|
|
yield "fake.ipatest.com"
|