mirror of
https://salsa.debian.org/freeipa-team/freeipa.git
synced 2025-01-11 08:41:55 -06:00
851f6d48ac
Pylint 2.9 introduced new check: > New checker consider-using-dict-items. Emitted when iterating over dictionary keys and then indexing the same dictionary with the key within loop body. Fixes: https://pagure.io/freeipa/issue/9117 Signed-off-by: Stanislav Levin <slev@altlinux.org> Reviewed-By: Rob Crittenden <rcritten@redhat.com>
551 lines
20 KiB
Python
551 lines
20 KiB
Python
#
|
|
# Copyright (C) 2015 FreeIPA Contributors see COPYING for license
|
|
#
|
|
|
|
from ipalib import api, errors
|
|
from ipaplatform.constants import constants as platformconstants
|
|
from ipapython.dn import DN
|
|
|
|
import six
|
|
|
|
from ipatests.util import assert_deepequal, get_group_dn
|
|
from ipatests.test_xmlrpc import objectclasses
|
|
from ipatests.test_xmlrpc.xmlrpc_test import (
|
|
fuzzy_set_optional_oc,
|
|
fuzzy_digits, fuzzy_uuid, fuzzy_user_or_group_sid, raises_exact)
|
|
from ipatests.test_xmlrpc.tracker.base import Tracker
|
|
from ipatests.test_xmlrpc.tracker.kerberos_aliases import KerberosAliasMixin
|
|
from ipatests.test_xmlrpc.tracker.certmapdata import CertmapdataMixin
|
|
|
|
if six.PY3:
|
|
unicode = str
|
|
|
|
|
|
class UserTracker(CertmapdataMixin, KerberosAliasMixin, Tracker):
|
|
""" Class for host plugin like tests """
|
|
|
|
retrieve_keys = {
|
|
u'dn', u'uid', u'givenname', u'sn', u'homedirectory', u'loginshell',
|
|
u'uidnumber', u'gidnumber', u'mail', u'ou',
|
|
u'telephonenumber', u'title', u'memberof', u'nsaccountlock',
|
|
u'memberofindirect', u'ipauserauthtype', u'userclass',
|
|
u'ipatokenradiusconfiglink', u'ipatokenradiususername',
|
|
u'krbprincipalexpiration', u'usercertificate;binary',
|
|
u'has_keytab', u'has_password', u'memberof_group', u'sshpubkeyfp',
|
|
u'krbcanonicalname', 'krbprincipalname'
|
|
}
|
|
|
|
retrieve_all_keys = retrieve_keys | {
|
|
u'usercertificate', u'street', u'postalcode',
|
|
u'facsimiletelephonenumber', u'carlicense', u'ipasshpubkey',
|
|
u'l', u'mobile', u'krbextradata', u'krblastpwdchange',
|
|
u'krbpasswordexpiration', u'pager', u'st', u'manager', u'cn',
|
|
u'ipauniqueid', u'objectclass', u'mepmanagedentry',
|
|
u'displayname', u'gecos', u'initials', u'preserved',
|
|
'ipantsecurityidentifier'}
|
|
|
|
retrieve_preserved_keys = (retrieve_keys - {u'memberof_group'}) | {
|
|
u'preserved'}
|
|
retrieve_preserved_all_keys = retrieve_all_keys - {u'memberof_group'}
|
|
|
|
create_keys = retrieve_all_keys | {
|
|
u'krbextradata', u'krbpasswordexpiration', u'krblastpwdchange',
|
|
u'krbprincipalkey', u'userpassword', u'randompassword'}
|
|
create_keys = create_keys - {u'nsaccountlock'}
|
|
create_keys = create_keys - {'ipantsecurityidentifier'}
|
|
|
|
update_keys = retrieve_keys - {u'dn'}
|
|
activate_keys = retrieve_keys
|
|
|
|
find_keys = retrieve_keys - {
|
|
u'mepmanagedentry', u'memberof_group', u'has_keytab', u'has_password',
|
|
u'manager',
|
|
}
|
|
find_all_keys = retrieve_all_keys - {
|
|
u'has_keytab', u'has_password'
|
|
}
|
|
|
|
primary_keys = {u'uid', u'dn'}
|
|
|
|
def __init__(self, name=None, givenname=None, sn=None, **kwargs):
|
|
""" Check for non-empty unicode string for the required attributes
|
|
in the init method """
|
|
|
|
if not (isinstance(givenname, str) and givenname):
|
|
raise ValueError(
|
|
"Invalid first name provided: {!r}".format(givenname)
|
|
)
|
|
if not (isinstance(sn, str) and sn):
|
|
raise ValueError("Invalid second name provided: {!r}".format(sn))
|
|
|
|
super(UserTracker, self).__init__(default_version=None)
|
|
self.uid = unicode(name)
|
|
self.givenname = unicode(givenname)
|
|
self.sn = unicode(sn)
|
|
self.dn = DN(('uid', self.uid), api.env.container_user, api.env.basedn)
|
|
|
|
self.kwargs = kwargs
|
|
|
|
def make_create_command(self, force=None):
|
|
|
|
""" Make function that creates a user using user-add
|
|
with all set of attributes and with minimal values,
|
|
where uid is not specified """
|
|
|
|
if self.uid is not None:
|
|
return self.make_command(
|
|
'user_add', self.uid,
|
|
givenname=self.givenname,
|
|
sn=self.sn, **self.kwargs
|
|
)
|
|
else:
|
|
return self.make_command(
|
|
'user_add', givenname=self.givenname,
|
|
sn=self.sn, **self.kwargs
|
|
)
|
|
|
|
def make_delete_command(self, no_preserve=True, preserve=False):
|
|
""" Make function that deletes a user using user-del
|
|
|
|
Arguments 'preserve' and 'no_preserve' represent implemented
|
|
options --preserve and --no-preserve of user-del command,
|
|
which are mutually exclusive.
|
|
If --preserve=True and --no-preserve=False, the user is moved
|
|
to deleted container.
|
|
If --preserve=True and --no-preserve=True, an error is raised.
|
|
If --preserve=False and --no-preserver=True, user is deleted.
|
|
"""
|
|
|
|
if preserve and not no_preserve:
|
|
# --preserve=True and --no-preserve=False - user is moved to
|
|
# another container, hence it is necessary to change some user
|
|
# attributes
|
|
self.attrs[u'dn'] = DN(
|
|
('uid', self.uid),
|
|
api.env.container_deleteuser,
|
|
api.env.basedn
|
|
)
|
|
self.attrs[u'objectclass'] = objectclasses.user_base \
|
|
+ ['ipantuserattrs']
|
|
|
|
return self.make_command(
|
|
'user_del', self.uid,
|
|
no_preserve=no_preserve,
|
|
preserve=preserve
|
|
)
|
|
|
|
def make_retrieve_command(self, all=False, raw=False):
|
|
""" Make function that retrieves a user using user-show """
|
|
return self.make_command('user_show', self.uid, all=all)
|
|
|
|
def make_find_command(self, *args, **kwargs):
|
|
""" Make function that finds user using user-find """
|
|
return self.make_command('user_find', *args, **kwargs)
|
|
|
|
def make_update_command(self, updates):
|
|
""" Make function that updates user using user-mod """
|
|
return self.make_command('user_mod', self.uid, **updates)
|
|
|
|
def make_undelete_command(self):
|
|
""" Make function that activates preserved user using user-undel """
|
|
return self.make_command('user_undel', self.uid)
|
|
|
|
def make_enable_command(self):
|
|
""" Make function that enables user using user-enable """
|
|
return self.make_command('user_enable', self.uid)
|
|
|
|
def make_disable_command(self):
|
|
""" Make function that disables user using user-disable """
|
|
return self.make_command('user_disable', self.uid)
|
|
|
|
def make_stage_command(self):
|
|
""" Make function that restores preserved user by moving it to
|
|
staged container """
|
|
return self.make_command('user_stage', self.uid)
|
|
|
|
def make_group_add_member_command(self, *args, **kwargs):
|
|
return self.make_command('group_add_member', *args, **kwargs)
|
|
|
|
def track_create(self):
|
|
""" Update expected state for user creation """
|
|
self.attrs = dict(
|
|
dn=self.dn,
|
|
uid=[self.uid],
|
|
givenname=[self.givenname],
|
|
sn=[self.sn],
|
|
homedirectory=[u'/home/%s' % self.uid],
|
|
displayname=[u'%s %s' % (self.givenname, self.sn)],
|
|
cn=[u'%s %s' % (self.givenname, self.sn)],
|
|
initials=[u'%s%s' % (self.givenname[0], self.sn[0])],
|
|
objectclass=fuzzy_set_optional_oc(
|
|
objectclasses.user, 'ipantuserattrs'),
|
|
description=[u'__no_upg__'],
|
|
ipauniqueid=[fuzzy_uuid],
|
|
uidnumber=[fuzzy_digits],
|
|
gidnumber=[fuzzy_digits],
|
|
krbprincipalname=[u'%s@%s' % (self.uid, self.api.env.realm)],
|
|
krbcanonicalname=[u'%s@%s' % (self.uid, self.api.env.realm)],
|
|
mail=[u'%s@%s' % (self.uid, self.api.env.domain)],
|
|
gecos=[u'%s %s' % (self.givenname, self.sn)],
|
|
loginshell=[platformconstants.DEFAULT_SHELL],
|
|
has_keytab=False,
|
|
has_password=False,
|
|
mepmanagedentry=[get_group_dn(self.uid)],
|
|
memberof_group=[u'ipausers'],
|
|
nsaccountlock=[u'false'],
|
|
ipantsecurityidentifier=[fuzzy_user_or_group_sid],
|
|
)
|
|
|
|
for key, value in self.kwargs.items():
|
|
if key == "krbprincipalname":
|
|
try:
|
|
princ_splitted = value.split("@", maxsplit=1)
|
|
self.attrs[key] = [
|
|
"{}@{}".format(
|
|
princ_splitted[0].lower(),
|
|
princ_splitted[1],
|
|
)
|
|
]
|
|
except IndexError:
|
|
# we can provide just principal part
|
|
self.attrs[key] = [
|
|
"{}@{}".format(value.lower(), self.api.env.realm)
|
|
]
|
|
else:
|
|
if not isinstance(value, list):
|
|
self.attrs[key] = [value]
|
|
else:
|
|
self.attrs[key] = value
|
|
|
|
self.exists = True
|
|
|
|
def update(self, updates, expected_updates=None):
|
|
"""Helper function to update this user and check the result
|
|
|
|
Overriding Tracker method for setting self.attrs correctly;
|
|
* most attributes stores its value in list
|
|
* the rest can be overridden by expected_updates
|
|
* allow deleting parameters if update value is None
|
|
"""
|
|
if expected_updates is None:
|
|
expected_updates = {}
|
|
|
|
self.ensure_exists()
|
|
command = self.make_update_command(updates)
|
|
result = command()
|
|
|
|
for key, value in updates.items():
|
|
if value is None or value == '':
|
|
del self.attrs[key]
|
|
elif key == 'rename':
|
|
new_principal = u'{0}@{1}'.format(value, self.api.env.realm)
|
|
self.attrs['uid'] = [value]
|
|
self.attrs['krbcanonicalname'] = [new_principal]
|
|
if new_principal not in self.attrs['krbprincipalname']:
|
|
self.attrs['krbprincipalname'].append(new_principal)
|
|
else:
|
|
if type(value) is list:
|
|
self.attrs[key] = value
|
|
else:
|
|
self.attrs[key] = [value]
|
|
for key, value in expected_updates.items():
|
|
if value is None or value == '':
|
|
del self.attrs[key]
|
|
else:
|
|
self.attrs[key] = value
|
|
|
|
self.check_update(
|
|
result,
|
|
extra_keys=set(updates.keys()) | set(expected_updates.keys())
|
|
)
|
|
|
|
if 'rename' in updates:
|
|
self.uid = self.attrs['uid'][0]
|
|
|
|
def check_create(self, result, extra_keys=()):
|
|
""" Check 'user-add' command result """
|
|
expected = self.filter_attrs(self.create_keys | set(extra_keys))
|
|
assert_deepequal(dict(
|
|
value=self.uid,
|
|
summary=u'Added user "%s"' % self.uid,
|
|
result=self.filter_attrs(expected),
|
|
), result)
|
|
|
|
def check_delete(self, result):
|
|
""" Check 'user-del' command result """
|
|
assert_deepequal(dict(
|
|
value=[self.uid],
|
|
summary=u'Deleted user "%s"' % self.uid,
|
|
result=dict(failed=[]),
|
|
), result)
|
|
|
|
def check_retrieve(self, result, all=False, raw=False):
|
|
""" Check 'user-show' command result """
|
|
if u'preserved' in self.attrs and self.attrs[u'preserved']:
|
|
self.retrieve_all_keys = self.retrieve_preserved_all_keys
|
|
self.retrieve_keys = self.retrieve_preserved_keys
|
|
elif u'preserved' not in self.attrs and all:
|
|
self.attrs[u'preserved'] = False
|
|
|
|
if all:
|
|
expected = self.filter_attrs(self.retrieve_all_keys)
|
|
else:
|
|
expected = self.filter_attrs(self.retrieve_keys)
|
|
|
|
# small override because stageuser-find returns different type
|
|
# of nsaccountlock value than DS, but overall the value fits
|
|
# expected result
|
|
if u'nsaccountlock' in expected:
|
|
if expected[u'nsaccountlock'] == [u'true']:
|
|
expected[u'nsaccountlock'] = True
|
|
elif expected[u'nsaccountlock'] == [u'false']:
|
|
expected[u'nsaccountlock'] = False
|
|
|
|
assert_deepequal(dict(
|
|
value=self.uid,
|
|
summary=None,
|
|
result=expected,
|
|
), result)
|
|
|
|
def check_find(self, result, all=False, pkey_only=False, raw=False,
|
|
expected_override=None):
|
|
""" Check 'user-find' command result """
|
|
if all:
|
|
if u'preserved' not in self.attrs:
|
|
self.attrs.update(preserved=False)
|
|
expected = self.filter_attrs(self.find_all_keys)
|
|
elif pkey_only:
|
|
expected = self.filter_attrs(self.primary_keys)
|
|
else:
|
|
expected = self.filter_attrs(self.find_keys)
|
|
|
|
if all and self.attrs[u'preserved']:
|
|
del expected[u'mepmanagedentry']
|
|
|
|
if u'nsaccountlock' in expected:
|
|
if expected[u'nsaccountlock'] == [u'true']:
|
|
expected[u'nsaccountlock'] = True
|
|
elif expected[u'nsaccountlock'] == [u'false']:
|
|
expected[u'nsaccountlock'] = False
|
|
|
|
if expected_override:
|
|
assert isinstance(expected_override, dict)
|
|
expected.update(expected_override)
|
|
|
|
assert_deepequal(dict(
|
|
count=1,
|
|
truncated=False,
|
|
summary=u'1 user matched',
|
|
result=[expected],
|
|
), result)
|
|
|
|
def check_find_nomatch(self, result):
|
|
""" Check 'user-find' command result when no user should be found """
|
|
assert_deepequal(dict(
|
|
count=0,
|
|
truncated=False,
|
|
summary=u'0 users matched',
|
|
result=[],
|
|
), result)
|
|
|
|
def check_update(self, result, extra_keys=()):
|
|
""" Check 'user-mod' command result """
|
|
expected = self.filter_attrs(self.update_keys | set(extra_keys))
|
|
if expected[u'nsaccountlock'] == [u'true']:
|
|
expected[u'nsaccountlock'] = True
|
|
elif expected[u'nsaccountlock'] == [u'false']:
|
|
expected[u'nsaccountlock'] = False
|
|
|
|
assert_deepequal(dict(
|
|
value=self.uid,
|
|
summary=u'Modified user "%s"' % self.uid,
|
|
result=expected
|
|
), result)
|
|
|
|
def check_enable(self, result):
|
|
""" Check result of enable user operation """
|
|
assert_deepequal(dict(
|
|
value=self.name,
|
|
summary=u'Enabled user account "%s"' % self.name,
|
|
result=True
|
|
), result)
|
|
|
|
def check_disable(self, result):
|
|
""" Check result of disable user operation """
|
|
assert_deepequal(dict(
|
|
value=self.name,
|
|
summary=u'Disabled user account "%s"' % self.name,
|
|
result=True
|
|
), result)
|
|
|
|
def create_from_staged(self, stageduser):
|
|
""" Copies attributes from staged user - helper function for
|
|
activation tests """
|
|
self.attrs = stageduser.attrs
|
|
self.uid = stageduser.uid
|
|
self.givenname = stageduser.givenname
|
|
self.sn = stageduser.sn
|
|
|
|
self.attrs[u'mepmanagedentry'] = None
|
|
self.attrs[u'dn'] = self.dn
|
|
self.attrs[u'ipauniqueid'] = [fuzzy_uuid]
|
|
self.attrs[u'memberof_group'] = [u'ipausers']
|
|
self.attrs[u'mepmanagedentry'] = [u'cn=%s,%s,%s' % (
|
|
self.uid, api.env.container_group, api.env.basedn
|
|
)]
|
|
self.attrs[u'objectclass'] = objectclasses.user
|
|
if self.attrs[u'gidnumber'] == [u'-1']:
|
|
self.attrs[u'gidnumber'] = [fuzzy_digits]
|
|
if self.attrs[u'uidnumber'] == [u'-1']:
|
|
self.attrs[u'uidnumber'] = [fuzzy_digits]
|
|
|
|
if u'ipasshpubkey' in self.kwargs:
|
|
self.attrs[u'ipasshpubkey'] = [str(
|
|
self.kwargs[u'ipasshpubkey']
|
|
)]
|
|
self.attrs[u'nsaccountlock'] = [u'false']
|
|
|
|
def check_activate(self, result):
|
|
""" Check 'stageuser-activate' command result """
|
|
expected = dict(
|
|
value=self.uid,
|
|
summary=u'Stage user %s activated' % self.uid,
|
|
result=self.filter_attrs(self.activate_keys))
|
|
|
|
# small override because stageuser-find returns different
|
|
# type of nsaccountlock value than DS, but overall the value
|
|
# fits expected result
|
|
if expected['result'][u'nsaccountlock'] == [u'true']:
|
|
expected['result'][u'nsaccountlock'] = True
|
|
elif expected['result'][u'nsaccountlock'] == [u'false']:
|
|
expected['result'][u'nsaccountlock'] = False
|
|
|
|
assert_deepequal(expected, result)
|
|
|
|
self.exists = True
|
|
|
|
def check_undel(self, result):
|
|
""" Check 'user-undel' command result """
|
|
assert_deepequal(dict(
|
|
value=self.uid,
|
|
summary=u'Undeleted user account "%s"' % self.uid,
|
|
result=True
|
|
), result)
|
|
|
|
def enable(self):
|
|
""" Enable user account if it was disabled """
|
|
if (self.attrs['nsaccountlock'] is True or
|
|
self.attrs['nsaccountlock'] == [u'true']):
|
|
self.attrs.update(nsaccountlock=False)
|
|
result = self.make_enable_command()()
|
|
self.check_enable(result)
|
|
|
|
def disable(self):
|
|
""" Disable user account if it was enabled """
|
|
if (self.attrs['nsaccountlock'] is False or
|
|
self.attrs['nsaccountlock'] == [u'false']):
|
|
self.attrs.update(nsaccountlock=True)
|
|
result = self.make_disable_command()()
|
|
self.check_disable(result)
|
|
|
|
def track_delete(self, preserve=False):
|
|
"""Update expected state for host deletion"""
|
|
if preserve:
|
|
self.exists = True
|
|
if u'memberof_group' in self.attrs:
|
|
del self.attrs[u'memberof_group']
|
|
self.attrs[u'nsaccountlock'] = True
|
|
self.attrs[u'preserved'] = True
|
|
else:
|
|
self.exists = False
|
|
self.attrs = {}
|
|
|
|
def make_preserved_user(self):
|
|
""" 'Creates' a preserved user necessary for some tests """
|
|
self.ensure_exists()
|
|
self.track_delete(preserve=True)
|
|
command = self.make_delete_command(no_preserve=False, preserve=True)
|
|
result = command()
|
|
self.check_delete(result)
|
|
|
|
def check_attr_preservation(self, expected):
|
|
""" Verifies that ipaUniqueID, uidNumber and gidNumber are
|
|
preserved upon reactivation. Also verifies that resulting
|
|
active user is a member of ipausers group only."""
|
|
command = self.make_retrieve_command(all=True)
|
|
result = command()
|
|
|
|
assert_deepequal(dict(
|
|
ipauniqueid=result[u'result'][u'ipauniqueid'],
|
|
uidnumber=result[u'result'][u'uidnumber'],
|
|
gidnumber=result[u'result'][u'gidnumber']
|
|
), expected)
|
|
|
|
if (u'memberof_group' not in result[u'result'] or
|
|
result[u'result'][u'memberof_group'] != (u'ipausers',)):
|
|
assert False
|
|
|
|
def make_fixture_restore(self, request):
|
|
"""Make a pytest fixture for a preserved user that is to be moved to
|
|
staged area.
|
|
|
|
The fixture ensures the plugin entry does not exist before
|
|
and after the tests that use it. It takes into account
|
|
that the preserved user no longer exists after restoring it,
|
|
therefore the fixture verifies after the tests
|
|
that the preserved user doesn't exist instead of deleting it.
|
|
"""
|
|
del_command = self.make_delete_command()
|
|
try:
|
|
del_command()
|
|
except errors.NotFound:
|
|
pass
|
|
|
|
def finish():
|
|
with raises_exact(errors.NotFound(
|
|
reason=u'%s: user not found' % self.uid)):
|
|
del_command()
|
|
|
|
request.addfinalizer(finish)
|
|
|
|
return self
|
|
|
|
def make_admin(self, admin_group=u'admins'):
|
|
""" Add user to the administrator's group """
|
|
result = self.run_command('group_show', admin_group)
|
|
admin_group_content = result[u'result'][u'member_user']
|
|
admin_group_expected = list(admin_group_content) + [self.name]
|
|
|
|
command = self.make_group_add_member_command(
|
|
admin_group, **dict(user=self.name)
|
|
)
|
|
result = command()
|
|
assert_deepequal(dict(
|
|
completed=1,
|
|
failed=dict(
|
|
member=dict(group=tuple(), user=tuple(),
|
|
service=tuple(), idoverrideuser=tuple())
|
|
),
|
|
result={
|
|
'dn': get_group_dn(admin_group),
|
|
'member_user': admin_group_expected,
|
|
'gidnumber': [fuzzy_digits],
|
|
'cn': [admin_group],
|
|
'description': [u'Account administrators group'],
|
|
},
|
|
), result)
|
|
|
|
# Kerberos aliases methods
|
|
def _make_add_alias_cmd(self):
|
|
return self.make_command('user_add_principal', self.name)
|
|
|
|
def _make_remove_alias_cmd(self):
|
|
return self.make_command('user_remove_principal', self.name)
|
|
|
|
# Certificate identity mapping methods
|
|
def _make_add_certmap(self):
|
|
return self.make_command('user_add_certmapdata', self.name)
|
|
|
|
def _make_remove_certmap(self):
|
|
return self.make_command('user_remove_certmapdata', self.name)
|