2008-10-22 16:54:04 -05:00
|
|
|
# Authors:
|
|
|
|
# Rob Crittenden <rcritten@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
|
|
|
|
|
|
|
|
"""
|
|
|
|
Frontend plugins for host/machine Identity.
|
|
|
|
"""
|
|
|
|
|
2009-01-28 22:47:21 -06:00
|
|
|
from ipalib import api, crud, errors2, util
|
2009-01-14 22:55:04 -06:00
|
|
|
from ipalib import Object # Plugin base class
|
|
|
|
from ipalib import Str, Flag # Parameter types
|
2008-10-22 16:54:04 -05:00
|
|
|
|
|
|
|
|
|
|
|
def get_host(hostname):
|
|
|
|
"""
|
|
|
|
Try to get the hostname as fully-qualified first, then fall back to
|
|
|
|
just a host name search.
|
|
|
|
"""
|
|
|
|
ldap = api.Backend.ldap
|
|
|
|
|
|
|
|
# Strip off trailing dot
|
|
|
|
if hostname.endswith('.'):
|
|
|
|
hostname = hostname[:-1]
|
|
|
|
try:
|
|
|
|
dn = ldap.find_entry_dn("cn", hostname, "ipaHost")
|
2009-01-28 22:47:21 -06:00
|
|
|
except errors2.NotFound:
|
2008-10-22 16:54:04 -05:00
|
|
|
dn = ldap.find_entry_dn("serverhostname", hostname, "ipaHost")
|
|
|
|
return dn
|
|
|
|
|
2009-01-16 09:19:08 -06:00
|
|
|
def validate_host(ugettext, cn):
|
2008-10-22 16:54:04 -05:00
|
|
|
"""
|
|
|
|
Require at least one dot in the hostname (to support localhost.localdomain)
|
|
|
|
"""
|
|
|
|
dots = len(cn.split('.'))
|
|
|
|
if dots < 2:
|
|
|
|
return 'Fully-qualified hostname required'
|
|
|
|
return None
|
|
|
|
|
2008-11-04 13:02:42 -06:00
|
|
|
default_attributes = ['cn','description','localityname','nshostlocation','nshardwareplatform','nsosversion']
|
2008-10-22 16:54:04 -05:00
|
|
|
|
2009-01-14 22:55:04 -06:00
|
|
|
class host(Object):
|
2008-10-22 16:54:04 -05:00
|
|
|
"""
|
|
|
|
Host object.
|
|
|
|
"""
|
|
|
|
takes_params = (
|
2009-01-14 22:55:04 -06:00
|
|
|
Str('cn', validate_host,
|
2008-10-22 16:54:04 -05:00
|
|
|
cli_name='hostname',
|
|
|
|
primary_key=True,
|
2009-01-14 22:55:04 -06:00
|
|
|
normalizer=lambda value: value.lower(),
|
2008-10-22 16:54:04 -05:00
|
|
|
),
|
2009-01-14 22:55:04 -06:00
|
|
|
Str('description?',
|
2008-10-22 16:54:04 -05:00
|
|
|
doc='Description of the host',
|
|
|
|
),
|
2009-01-14 22:55:04 -06:00
|
|
|
Str('localityname?',
|
2008-10-22 16:54:04 -05:00
|
|
|
cli_name='locality',
|
|
|
|
doc='Locality of this host (Baltimore, MD)',
|
|
|
|
),
|
2009-01-14 22:55:04 -06:00
|
|
|
Str('nshostlocation?',
|
2008-10-22 16:54:04 -05:00
|
|
|
cli_name='location',
|
|
|
|
doc='Location of this host (e.g. Lab 2)',
|
|
|
|
),
|
2009-01-14 22:55:04 -06:00
|
|
|
Str('nshardwareplatform?',
|
2008-10-22 16:54:04 -05:00
|
|
|
cli_name='platform',
|
|
|
|
doc='Hardware platform of this host (e.g. Lenovo T61)',
|
|
|
|
),
|
2009-01-14 22:55:04 -06:00
|
|
|
Str('nsosversion?',
|
2008-10-22 16:54:04 -05:00
|
|
|
cli_name='os',
|
|
|
|
doc='Operating System and version on this host (e.g. Fedora 9)',
|
|
|
|
),
|
2009-01-14 22:55:04 -06:00
|
|
|
Str('userpassword?',
|
2008-10-22 16:54:04 -05:00
|
|
|
cli_name='password',
|
|
|
|
doc='Set a password to be used in bulk enrollment',
|
|
|
|
),
|
|
|
|
)
|
|
|
|
api.register(host)
|
|
|
|
|
|
|
|
|
|
|
|
class host_add(crud.Add):
|
|
|
|
'Add a new host.'
|
|
|
|
def execute(self, hostname, **kw):
|
|
|
|
"""
|
|
|
|
Execute the host-add operation.
|
|
|
|
|
|
|
|
The dn should not be passed as a keyword argument as it is constructed
|
|
|
|
by this method.
|
|
|
|
|
2008-10-24 10:39:47 -05:00
|
|
|
If password is set then this is considered a 'bulk' host so we
|
|
|
|
do not create a kerberos service principal.
|
|
|
|
|
2008-10-22 16:54:04 -05:00
|
|
|
Returns the entry as it will be created in LDAP.
|
|
|
|
|
|
|
|
:param hostname: The name of the host being added.
|
|
|
|
:param kw: Keyword arguments for the other LDAP attributes.
|
|
|
|
"""
|
|
|
|
assert 'cn' not in kw
|
|
|
|
assert 'dn' not in kw
|
2008-10-24 10:39:47 -05:00
|
|
|
assert 'krbprincipalname' not in kw
|
2008-10-22 16:54:04 -05:00
|
|
|
ldap = self.api.Backend.ldap
|
|
|
|
|
|
|
|
kw['cn'] = hostname
|
|
|
|
kw['serverhostname'] = hostname.split('.',1)[0]
|
|
|
|
kw['dn'] = ldap.make_host_dn(hostname)
|
|
|
|
|
|
|
|
# FIXME: do a DNS lookup to ensure host exists
|
|
|
|
|
|
|
|
current = util.get_current_principal()
|
|
|
|
if not current:
|
2009-01-28 22:47:21 -06:00
|
|
|
raise errors2.NotFound('Unable to determine current user')
|
2008-10-24 10:39:47 -05:00
|
|
|
kw['enrolledby'] = ldap.find_entry_dn("krbPrincipalName", current, "posixAccount")
|
2008-10-22 16:54:04 -05:00
|
|
|
|
|
|
|
# Get our configuration
|
|
|
|
config = ldap.get_ipa_config()
|
|
|
|
|
|
|
|
# some required objectclasses
|
|
|
|
# FIXME: add this attribute to cn=ipaconfig
|
2008-10-24 10:39:47 -05:00
|
|
|
#kw['objectclass'] = config.get('ipahostobjectclasses')
|
2009-01-16 09:19:08 -06:00
|
|
|
kw['objectclass'] = ['nsHost', 'ipaHost', 'pkiUser']
|
2008-10-24 10:39:47 -05:00
|
|
|
|
|
|
|
# Ensure the list of objectclasses is lower-case
|
|
|
|
kw['objectclass'] = map(lambda z: z.lower(), kw.get('objectclass'))
|
|
|
|
|
|
|
|
if not kw.get('userpassword', False):
|
|
|
|
kw['krbprincipalname'] = "host/%s@%s" % (hostname, self.api.env.realm)
|
|
|
|
|
|
|
|
if 'krbprincipalaux' not in kw.get('objectclass'):
|
|
|
|
kw['objectclass'].append('krbprincipalaux')
|
|
|
|
else:
|
|
|
|
if 'krbprincipalaux' in kw.get('objectclass'):
|
|
|
|
kw['objectclass'].remove('krbprincipalaux')
|
2008-10-22 16:54:04 -05:00
|
|
|
|
|
|
|
return ldap.create(**kw)
|
2008-12-10 15:42:45 -06:00
|
|
|
def output_for_cli(self, textui, result, *args, **options):
|
2008-10-22 16:54:04 -05:00
|
|
|
"""
|
|
|
|
Output result of this command to command line interface.
|
|
|
|
"""
|
2008-12-10 15:42:45 -06:00
|
|
|
textui.print_plain("Host added")
|
2008-10-22 16:54:04 -05:00
|
|
|
|
|
|
|
api.register(host_add)
|
|
|
|
|
|
|
|
|
|
|
|
class host_del(crud.Del):
|
|
|
|
'Delete an existing host.'
|
|
|
|
def execute(self, hostname, **kw):
|
|
|
|
"""Delete a host.
|
|
|
|
|
|
|
|
hostname is the name of the host to delete
|
|
|
|
|
|
|
|
:param hostname: The name of the host being removed.
|
|
|
|
:param kw: Not used.
|
|
|
|
"""
|
|
|
|
ldap = self.api.Backend.ldap
|
|
|
|
dn = get_host(hostname)
|
|
|
|
return ldap.delete(dn)
|
2008-12-10 15:42:45 -06:00
|
|
|
def output_for_cli(self, textui, result, *args, **options):
|
2008-10-22 16:54:04 -05:00
|
|
|
"""
|
|
|
|
Output result of this command to command line interface.
|
|
|
|
"""
|
2008-12-10 15:42:45 -06:00
|
|
|
textui.print_plain("Host deleted")
|
2008-10-22 16:54:04 -05:00
|
|
|
|
|
|
|
api.register(host_del)
|
|
|
|
|
|
|
|
|
|
|
|
class host_mod(crud.Mod):
|
|
|
|
'Edit an existing host.'
|
|
|
|
def execute(self, hostname, **kw):
|
|
|
|
"""
|
|
|
|
Execute the host-mod operation.
|
|
|
|
|
|
|
|
The dn should not be passed as a keyword argument as it is constructed
|
|
|
|
by this method.
|
|
|
|
|
|
|
|
Returns the entry
|
|
|
|
|
|
|
|
:param hostname: The name of the host to retrieve.
|
|
|
|
:param kw: Keyword arguments for the other LDAP attributes.
|
|
|
|
"""
|
|
|
|
assert 'cn' not in kw
|
|
|
|
assert 'dn' not in kw
|
|
|
|
ldap = self.api.Backend.ldap
|
|
|
|
dn = get_host(hostname)
|
|
|
|
return ldap.update(dn, **kw)
|
|
|
|
|
2008-12-10 15:42:45 -06:00
|
|
|
def output_for_cli(self, textui, result, *args, **options):
|
2008-10-22 16:54:04 -05:00
|
|
|
"""
|
|
|
|
Output result of this command to command line interface.
|
|
|
|
"""
|
2008-12-10 15:42:45 -06:00
|
|
|
textui.print_plain("Host updated")
|
2008-10-22 16:54:04 -05:00
|
|
|
|
|
|
|
api.register(host_mod)
|
|
|
|
|
|
|
|
|
|
|
|
class host_find(crud.Find):
|
|
|
|
'Search the hosts.'
|
2009-01-14 23:59:44 -06:00
|
|
|
|
2008-11-04 13:02:42 -06:00
|
|
|
takes_options = (
|
2009-01-14 22:55:04 -06:00
|
|
|
Flag('all', doc='Retrieve all attributes'),
|
2008-11-04 13:02:42 -06:00
|
|
|
)
|
2009-01-14 23:59:44 -06:00
|
|
|
|
|
|
|
# FIXME: This should no longer be needed with the Param.query kwarg.
|
|
|
|
# def get_args(self):
|
|
|
|
# """
|
|
|
|
# Override Find.get_args() so we can exclude the validation rules
|
|
|
|
# """
|
|
|
|
# yield self.obj.primary_key.__clone__(rules=tuple())
|
|
|
|
|
2008-10-22 16:54:04 -05:00
|
|
|
def execute(self, term, **kw):
|
|
|
|
ldap = self.api.Backend.ldap
|
|
|
|
|
|
|
|
# Pull the list of searchable attributes out of the configuration.
|
|
|
|
#config = ldap.get_ipa_config()
|
|
|
|
# FIXME: add this attribute to cn=ipaconfig
|
|
|
|
#search_fields_conf_str = config.get('ipahostsearchfields')
|
|
|
|
#search_fields = search_fields_conf_str.split(",")
|
|
|
|
search_fields = ['cn','serverhostname','description','localityname','nshostlocation','nshardwareplatform','nsosversion']
|
|
|
|
|
2008-12-10 15:45:07 -06:00
|
|
|
search_kw = {}
|
2008-10-22 16:54:04 -05:00
|
|
|
for s in search_fields:
|
2008-12-10 15:45:07 -06:00
|
|
|
search_kw[s] = term
|
2008-10-22 16:54:04 -05:00
|
|
|
|
|
|
|
# Can't use ldap.get_object_type() since cn is also used for group dns
|
2008-12-10 15:45:07 -06:00
|
|
|
search_kw['objectclass'] = "ipaHost"
|
2008-11-04 13:02:42 -06:00
|
|
|
if kw.get('all', False):
|
2008-12-10 15:45:07 -06:00
|
|
|
search_kw['attributes'] = ['*']
|
2008-11-04 13:02:42 -06:00
|
|
|
else:
|
2008-12-10 15:45:07 -06:00
|
|
|
search_kw['attributes'] = default_attributes
|
|
|
|
return ldap.search(**search_kw)
|
2008-12-10 15:42:45 -06:00
|
|
|
def output_for_cli(self, textui, result, *args, **options):
|
|
|
|
counter = result[0]
|
|
|
|
hosts = result[1:]
|
2008-10-22 16:54:04 -05:00
|
|
|
if counter == 0:
|
2008-12-10 15:42:45 -06:00
|
|
|
textui.print_plain("No entries found")
|
2008-10-22 16:54:04 -05:00
|
|
|
return
|
|
|
|
|
|
|
|
for h in hosts:
|
2008-12-10 15:42:45 -06:00
|
|
|
textui.print_entry(h)
|
|
|
|
if counter == -1:
|
|
|
|
textui.print_plain("These results are truncated.")
|
|
|
|
textui.print_plain("Please refine your search and try again.")
|
2008-10-22 16:54:04 -05:00
|
|
|
api.register(host_find)
|
|
|
|
|
|
|
|
|
|
|
|
class host_show(crud.Get):
|
|
|
|
'Examine an existing host.'
|
|
|
|
takes_options = (
|
2009-01-14 22:55:04 -06:00
|
|
|
Flag('all', doc='Display all host attributes'),
|
2008-10-22 16:54:04 -05:00
|
|
|
)
|
|
|
|
def execute(self, hostname, **kw):
|
|
|
|
"""
|
|
|
|
Execute the host-show operation.
|
|
|
|
|
|
|
|
The dn should not be passed as a keyword argument as it is constructed
|
|
|
|
by this method.
|
|
|
|
|
|
|
|
Returns the entry
|
|
|
|
|
|
|
|
:param hostname: The login name of the host to retrieve.
|
|
|
|
:param kw: "all" set to True = return all attributes
|
|
|
|
"""
|
|
|
|
ldap = self.api.Backend.ldap
|
|
|
|
dn = get_host(hostname)
|
|
|
|
# FIXME: should kw contain the list of attributes to display?
|
|
|
|
if kw.get('all', False):
|
|
|
|
return ldap.retrieve(dn)
|
|
|
|
else:
|
2008-11-04 13:02:42 -06:00
|
|
|
value = ldap.retrieve(dn, default_attributes)
|
2008-10-22 16:54:04 -05:00
|
|
|
del value['dn']
|
|
|
|
return value
|
2008-12-10 15:42:45 -06:00
|
|
|
def output_for_cli(self, textui, result, *args, **options):
|
|
|
|
textui.print_entry(result)
|
2008-10-22 16:54:04 -05:00
|
|
|
|
|
|
|
api.register(host_show)
|