# Authors: # Rob Crittenden # # Copyright (C) 2010 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, either version 3 of the License, or # (at your option) any later version. # # 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, see . """ Test `ipa-getkeytab` """ import os import shutil from cmdline import cmdline_test from ipalib import api from ipalib import errors import tempfile from ipapython import ipautil, ipaldap import nose import tempfile import gssapi from ipaserver.plugins.ldap2 import ldap2 from ipapython.dn import DN import pytest def use_keytab(principal, keytab): try: tmpdir = tempfile.mkdtemp(prefix = "tmp-") ccache_file = 'FILE:%s/ccache' % tmpdir name = gssapi.Name(principal, gssapi.NameType.kerberos_principal) store = {'ccache': ccache_file, 'client_keytab': keytab} os.environ['KRB5CCNAME'] = ccache_file gssapi.Credentials(name=name, usage='initiate', store=store) conn = ldap2(api) conn.connect(autobind=ipaldap.AUTOBIND_DISABLED) conn.disconnect() except gssapi.exceptions.GSSError as e: raise Exception('Unable to bind to LDAP. Error initializing principal %s in %s: %s' % (principal, keytab, str(e))) finally: os.environ.pop('KRB5CCNAME', None) if tmpdir: shutil.rmtree(tmpdir) @pytest.mark.tier0 class test_ipagetkeytab(cmdline_test): """ Test `ipa-getkeytab`. """ command = "ipa-getkeytab" host_fqdn = u'ipatest.%s' % api.env.domain service_princ = u'test/%s@%s' % (host_fqdn, api.env.realm) [keytabfd, keytabname] = tempfile.mkstemp() os.close(keytabfd) def test_0_setup(self): """ Create a host to test against. """ # Create the service try: api.Command['host_add'](self.host_fqdn, force=True) except errors.DuplicateEntry: # it already exists, no problem pass def test_1_run(self): """ Create a keytab with `ipa-getkeytab` for a non-existent service. """ new_args = [self.command, "-s", api.env.host, "-p", "test/notfound.example.com", "-k", self.keytabname, ] result = ipautil.run(new_args, stdin=None, raiseonerr=False, capture_error=True) err = result.error_output assert 'Failed to parse result: PrincipalName not found.\n' in err, err rc = result.returncode assert rc > 0, rc def test_2_run(self): """ Create a keytab with `ipa-getkeytab` for an existing service. """ # Create the service try: api.Command['service_add'](self.service_princ, force=True) except errors.DuplicateEntry: # it already exists, no problem pass os.unlink(self.keytabname) new_args = [self.command, "-s", api.env.host, "-p", self.service_princ, "-k", self.keytabname, ] try: result = ipautil.run(new_args, None, capture_error=True) expected = 'Keytab successfully retrieved and stored in: %s\n' % ( self.keytabname) assert expected in result.error_output, ( 'Success message not in output:\n%s' % result.error_output) except ipautil.CalledProcessError as e: assert (False) def test_3_use(self): """ Try to use the service keytab. """ use_keytab(self.service_princ, self.keytabname) def test_4_disable(self): """ Disable a kerberos principal """ # Verify that it has a principal key entry = api.Command['service_show'](self.service_princ)['result'] assert(entry['has_keytab'] == True) # Disable it api.Command['service_disable'](self.service_princ) # Verify that it looks disabled entry = api.Command['service_show'](self.service_princ)['result'] assert(entry['has_keytab'] == False) def test_5_use_disabled(self): """ Try to use the disabled keytab """ try: use_keytab(self.service_princ, self.keytabname) except Exception as errmsg: assert('Unable to bind to LDAP. Error initializing principal' in str(errmsg)) def test_9_cleanup(self): """ Clean up test data """ # First create the host that will use this policy os.unlink(self.keytabname) api.Command['host_del'](self.host_fqdn)