mirror of
				https://salsa.debian.org/freeipa-team/freeipa.git
				synced 2025-02-25 18:55:28 -06:00 
			
		
		
		
	Add ignore lists to migrate-ds command
When user migrates users/groups from an old DS instance, the
migration may fail on unsupported object classes and/or
relevant LDAP object attributes.
This patch implements a support for object class and attribute
ignore lists that can be used to suppress these migration issues.
Additionally, a redundant "dev/null" file is removed from git repo
(originally added in 26b0e8fc98).
https://fedorahosted.org/freeipa/ticket/1266
			
			
This commit is contained in:
		
							
								
								
									
										6
									
								
								API.txt
									
									
									
									
									
								
							
							
						
						
									
										6
									
								
								API.txt
									
									
									
									
									
								
							| @@ -1523,7 +1523,7 @@ output: Output('summary', (<type 'unicode'>, <type 'NoneType'>), 'User-friendly | ||||
| output: Entry('result', <type 'dict'>, Gettext('A dictionary representing an LDAP entry', domain='ipa', localedir=None)) | ||||
| output: Output('value', <type 'unicode'>, "The primary_key value of the entry, e.g. 'jdoe' for a user") | ||||
| command: migrate_ds | ||||
| args: 2,9,3 | ||||
| args: 2,13,3 | ||||
| arg: Str('ldapuri', validate_ldapuri, cli_name='ldap_uri', label=Gettext('LDAP URI', domain='ipa', localedir=None)) | ||||
| arg: Password('bindpw', cli_name='password', label=Gettext('Password', domain='ipa', localedir=None)) | ||||
| option: Str('binddn?', autofill=True, cli_name='bind_dn', default=u'cn=directory manager', label=Gettext('Bind DN', domain='ipa', localedir=None)) | ||||
| @@ -1531,6 +1531,10 @@ option: Str('usercontainer?', autofill=True, cli_name='user_container', default= | ||||
| option: Str('groupcontainer?', autofill=True, cli_name='group_container', default=u'ou=groups', label=Gettext('Group container', domain='ipa', localedir=None)) | ||||
| option: List('userobjectclass?', autofill=True, cli_name='user_objectclass', default=(u'person',), label=Gettext('User object class', domain='ipa', localedir=None), multivalue=True) | ||||
| option: List('groupobjectclass?', autofill=True, cli_name='group_objectclass', default=(u'groupOfUniqueNames', u'groupOfNames'), label=Gettext('Group object class', domain='ipa', localedir=None), multivalue=True) | ||||
| option: List('userignoreobjectclass?', autofill=True, cli_name='user_ignore_objectclass', default=(), label=Gettext('Ignore user object class', domain='ipa', localedir=None), multivalue=True) | ||||
| option: List('userignoreattribute?', autofill=True, cli_name='user_ignore_attribute', default=(), label=Gettext('Ignore user attribute', domain='ipa', localedir=None), multivalue=True) | ||||
| option: List('groupignoreobjectclass?', autofill=True, cli_name='group_ignore_objectclass', default=(), label=Gettext('Ignore group object class', domain='ipa', localedir=None), multivalue=True) | ||||
| option: List('groupignoreattribute?', autofill=True, cli_name='group_ignore_attribute', default=(), label=Gettext('Ignore group attribute', domain='ipa', localedir=None), multivalue=True) | ||||
| option: StrEnum('schema?', autofill=True, cli_name='schema', default=u'RFC2307bis', label=Gettext('LDAP schema', domain='ipa', localedir=None), values=(u'RFC2307bis', u'RFC2307')) | ||||
| option: Flag('continue?', autofill=True, default=False) | ||||
| option: List('exclude_groups?', autofill=True, cli_name='exclude_groups', default=(), multivalue=True) | ||||
|   | ||||
							
								
								
									
										2
									
								
								VERSION
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								VERSION
									
									
									
									
									
								
							| @@ -79,4 +79,4 @@ IPA_DATA_VERSION=20100614120000 | ||||
| #                                                      # | ||||
| ######################################################## | ||||
| IPA_API_VERSION_MAJOR=2 | ||||
| IPA_API_VERSION_MINOR=4 | ||||
| IPA_API_VERSION_MINOR=5 | ||||
|   | ||||
| @@ -70,8 +70,6 @@ if api.env.in_server and api.env.context in ['lite', 'server']: | ||||
|     except StandardError, e: | ||||
|         raise e | ||||
| from ipalib import _ | ||||
| from ipalib.text import Gettext # FIXME: remove once the other Gettext FIXME is removed | ||||
|  | ||||
|  | ||||
| # USER MIGRATION CALLBACKS AND VARS | ||||
|  | ||||
| @@ -84,6 +82,7 @@ _supported_schemas = (u'RFC2307bis', u'RFC2307') | ||||
|  | ||||
| def _pre_migrate_user(ldap, pkey, dn, entry_attrs, failed, config, ctx, **kwargs): | ||||
|     attr_blacklist = ['krbprincipalkey','memberofindirect','memberindirect'] | ||||
|     attr_blacklist.extend(kwargs.get('attr_blacklist', [])) | ||||
|  | ||||
|     # get default primary group for new users | ||||
|     if 'def_group_dn' not in ctx: | ||||
| @@ -110,6 +109,14 @@ def _pre_migrate_user(ldap, pkey, dn, entry_attrs, failed, config, ctx, **kwargs | ||||
|         if attr in attr_blacklist: | ||||
|             del entry_attrs[attr] | ||||
|  | ||||
|     # do not migrate all object classes | ||||
|     if 'objectclass' in entry_attrs: | ||||
|         for object_class in kwargs.get('oc_blacklist', []): | ||||
|             try: | ||||
|                 entry_attrs['objectclass'].remove(object_class) | ||||
|             except ValueError:  # object class not present | ||||
|                 pass | ||||
|  | ||||
|     # generate a principal name and check if it isn't already taken | ||||
|     principal = u'%s@%s' % (pkey, api.env.realm) | ||||
|     try: | ||||
| @@ -186,6 +193,7 @@ def _pre_migrate_group(ldap, pkey, dn, entry_attrs, failed, config, ctx, **kwarg | ||||
|         entry_attrs['member'] = new_members | ||||
|  | ||||
|     attr_blacklist = ['memberofindirect','memberindirect'] | ||||
|     attr_blacklist.extend(kwargs.get('attr_blacklist', [])) | ||||
|  | ||||
|     schema = kwargs.get('schema', None) | ||||
|     entry_attrs['ipauniqueid'] = 'autogenerate' | ||||
| @@ -206,6 +214,14 @@ def _pre_migrate_group(ldap, pkey, dn, entry_attrs, failed, config, ctx, **kwarg | ||||
|         if attr in attr_blacklist: | ||||
|             del entry_attrs[attr] | ||||
|  | ||||
|     # do not migrate all object classes | ||||
|     if 'objectclass' in entry_attrs: | ||||
|         for object_class in kwargs.get('oc_blacklist', []): | ||||
|             try: | ||||
|                 entry_attrs['objectclass'].remove(object_class) | ||||
|             except ValueError:  # object class not present | ||||
|                 pass | ||||
|  | ||||
|     return dn | ||||
|  | ||||
|  | ||||
| @@ -249,12 +265,16 @@ class migrate_ds(Command): | ||||
|         'user': { | ||||
|             'filter_template' : '(&(|%s)(uid=*))', | ||||
|             'oc_option' : 'userobjectclass', | ||||
|             'oc_blacklist_option' : 'userignoreobjectclass', | ||||
|             'attr_blacklist_option' : 'userignoreattribute', | ||||
|             'pre_callback' : _pre_migrate_user, | ||||
|             'post_callback' : _post_migrate_user | ||||
|         }, | ||||
|         'group': { | ||||
|             'filter_template' : '(&(|%s)(cn=*))', | ||||
|             'oc_option' : 'groupobjectclass', | ||||
|             'oc_blacklist_option' : 'groupignoreobjectclass', | ||||
|             'attr_blacklist_option' : 'groupignoreattribute', | ||||
|             'pre_callback' : _pre_migrate_group, | ||||
|             'post_callback' : None | ||||
|         }, | ||||
| @@ -309,6 +329,34 @@ class migrate_ds(Command): | ||||
|             default=(u'groupOfUniqueNames', u'groupOfNames'), | ||||
|             autofill=True, | ||||
|         ), | ||||
|         List('userignoreobjectclass?', | ||||
|             cli_name='user_ignore_objectclass', | ||||
|             label=_('Ignore user object class'), | ||||
|             doc=_('Comma-separated list of objectclasses to be ignored for user entries in DS'), | ||||
|             default=tuple(), | ||||
|             autofill=True, | ||||
|         ), | ||||
|         List('userignoreattribute?', | ||||
|             cli_name='user_ignore_attribute', | ||||
|             label=_('Ignore user attribute'), | ||||
|             doc=_('Comma-separated list of attributes to be ignored for user entries in DS'), | ||||
|             default=tuple(), | ||||
|             autofill=True, | ||||
|         ), | ||||
|         List('groupignoreobjectclass?', | ||||
|             cli_name='group_ignore_objectclass', | ||||
|             label=_('Ignore group object class'), | ||||
|             doc=_('Comma-separated list of objectclasses to be ignored for group entries in DS'), | ||||
|             default=tuple(), | ||||
|             autofill=True, | ||||
|         ), | ||||
|         List('groupignoreattribute?', | ||||
|             cli_name='group_ignore_attribute', | ||||
|             label=_('Ignore group attribute'), | ||||
|             doc=_('Comma-separated list of attributes to be ignored for group entries in DS'), | ||||
|             default=tuple(), | ||||
|             autofill=True, | ||||
|         ), | ||||
|         StrEnum('schema?', | ||||
|             cli_name='schema', | ||||
|             label=_('LDAP schema'), | ||||
| @@ -365,8 +413,7 @@ can use their Kerberos accounts.''') | ||||
|         for ldap_obj_name in self.migrate_objects: | ||||
|             ldap_obj = self.api.Object[ldap_obj_name] | ||||
|             name = 'exclude_%ss' % to_cli(ldap_obj_name) | ||||
|             # FIXME: can't substitute strings static Gettext instance | ||||
|             doc = Gettext(self.exclude_doc % ldap_obj.object_name_plural) | ||||
|             doc = self.exclude_doc % ldap_obj.object_name_plural | ||||
|             yield List( | ||||
|                 '%s?' % name, cli_name=name, doc=doc, default=tuple(), | ||||
|                 autofill=True | ||||
| @@ -436,6 +483,14 @@ can use their Kerberos accounts.''') | ||||
|                     ) | ||||
|                 ) | ||||
|  | ||||
|             blacklists = {} | ||||
|             for blacklist in ('oc_blacklist', 'attr_blacklist'): | ||||
|                 blacklist_option = self.migrate_objects[ldap_obj_name][blacklist+'_option'] | ||||
|                 if blacklist_option is not None: | ||||
|                     blacklists[blacklist] = options.get(blacklist_option, tuple()) | ||||
|                 else: | ||||
|                     blacklists[blacklist] = tuple() | ||||
|  | ||||
|             for (dn, entry_attrs) in entries: | ||||
|                 if dn is None:  # LDAP search reference | ||||
|                     failed[ldap_obj_name][entry_attrs[0]] = unicode(_ref_err_msg) | ||||
| @@ -459,7 +514,8 @@ can use their Kerberos accounts.''') | ||||
|                     dn = callback( | ||||
|                         ldap, pkey, dn, entry_attrs, failed[ldap_obj_name], | ||||
|                         config, context, schema = options['schema'], | ||||
|                         search_bases = search_bases | ||||
|                         search_bases = search_bases, | ||||
|                         **blacklists | ||||
|                     ) | ||||
|                     if not dn: | ||||
|                         continue | ||||
|   | ||||
		Reference in New Issue
	
	Block a user