diff --git a/dirsrvtests/tests/suites/backups/backup_test.py b/dirsrvtests/tests/suites/backups/backup_test.py index 7d60cf44c7..1246469a33 100644 --- a/dirsrvtests/tests/suites/backups/backup_test.py +++ b/dirsrvtests/tests/suites/backups/backup_test.py @@ -7,21 +7,30 @@ # --- END COPYRIGHT BLOCK --- # import logging -import pytest import os import shutil +import time +import glob +import subprocess +import pytest +import ldap from datetime import datetime -from lib389._constants import DEFAULT_SUFFIX, INSTALL_LATEST_CONFIG +from lib389._constants import DN_DM, PASSWORD, DEFAULT_SUFFIX, INSTALL_LATEST_CONFIG from lib389.properties import BACKEND_SAMPLE_ENTRIES, TASK_WAIT -from lib389.topologies import topology_st as topo, topology_m2 as topo_m2 -from lib389.backend import Backend +from lib389.topologies import topology_st as topo, topology_m2 as topo_m2, set_timeout +from lib389.backend import Backends, Backend +from lib389.dbgen import dbgen_users from lib389.tasks import BackupTask, RestoreTask from lib389.config import BDB_LDBMConfig +from lib389.idm.nscontainer import nsContainers from lib389 import DSEldif from lib389.utils import ds_is_older, get_default_db_lib from lib389.replica import ReplicationManager +from threading import Thread, Event import tempfile + + pytestmark = pytest.mark.tier1 DEBUGGING = os.getenv("DEBUGGING", default=False) @@ -31,6 +40,86 @@ logging.getLogger(__name__).setLevel(logging.INFO) log = logging.getLogger(__name__) +# test_online_backup_and_dse_write may hang if 6372 is not fixed +# so lets use a shorter timeout +set_timeout(30*60) + +event = Event() + +BESTRUCT = [ + { "bename" : "be1", "suffix": "dc=be1", "nbusers": 1000 }, + { "bename" : "be2", "suffix": "dc=be2", "nbusers": 1000 }, + { "bename" : "be3", "suffix": "dc=be3", "nbusers": 1000 }, +] + +class DseConfigContextManager: + """Change a config parameter is dse.ldif and restore it afterwards.""" + + def __init__(self, inst, dn, attr, value): + self.inst = inst + self.dn = dn + self.attr = attr + self.value = value + self.oldvalue = None + self.dseldif = DSEldif(inst) + + def __enter__(self): + self.inst.stop() + self.oldvalue = self.dseldif.get(self.dn, self.attr, single=True) + self.dseldif.replace(self.dn, self.attr, self.value) + log.info(f"Switching {self.dn}:{self.attr} to {self.value}") + self.inst.start() + + def __exit__(self, exc_type, exc_value, exc_tb): + self.inst.stop() + log.info(f"Switching {self.dn}:{self.attr} to {self.oldvalue}") + self.dseldif.replace(self.dn, self.attr, self.oldvalue) + self.inst.start() + + +@pytest.fixture(scope="function") +def mytopo(topo, request): + bes = [] + + def fin(): + for be in bes: + be.delete() + for bak_dir in glob.glob(f'{inst.ds_paths.backup_dir}/*'): + shutil.rmtree(bak_dir) + + if not DEBUGGING: + request.addfinalizer(fin) + + inst = topo.standalone + + ldif_files = {} + for d in BESTRUCT: + bename = d['bename'] + suffix = d['suffix'] + nbusers = d['nbusers'] + log.info(f'Adding suffix: {suffix} and backend: {bename}...') + backends = Backends(inst) + try: + be = backends.create(properties={'nsslapd-suffix': suffix, 'name': bename}) + # Insert at list head so that children backends get deleted before parent one. + bes.insert(0, be) + except ldap.UNWILLING_TO_PERFORM as e: + if str(e) == "Mapping tree for this suffix exists!": + pass + else: + raise e + + ldif_dir = inst.get_ldif_dir() + ldif_files[bename] = os.path.join(ldif_dir, f'default_{bename}.ldif') + dbgen_users(inst, nbusers, ldif_files[bename], suffix) + inst.stop() + for d in BESTRUCT: + bename = d['bename'] + inst.ldif2db(bename, None, None, None, ldif_files[bename]) + inst.start() + return topo + + def test_missing_backend(topo): """Test that an error is returned when a restore is performed for a backend that is no longer present. @@ -80,8 +169,6 @@ def test_missing_backend(topo): assert restore_task.get_exit_code() != 0 -@pytest.mark.bz1851967 -@pytest.mark.ds4112 @pytest.mark.skipif(ds_is_older('1.4.1'), reason="Not implemented") @pytest.mark.skipif(get_default_db_lib() == "mdb", reason="Not supported over mdb") def test_db_home_dir_online_backup(topo): @@ -99,14 +186,16 @@ def test_db_home_dir_online_backup(topo): 2. Failure 3. Success """ - bdb_ldbmconfig = BDB_LDBMConfig(topo.standalone) - dseldif = DSEldif(topo.standalone) - topo.standalone.stop() - with tempfile.TemporaryDirectory() as backup_dir: - dseldif.replace(bdb_ldbmconfig.dn, 'nsslapd-db-home-directory', f'{backup_dir}') - topo.standalone.start() - topo.standalone.tasks.db2bak(backup_dir=f'{backup_dir}', args={TASK_WAIT: True}) - assert topo.standalone.ds_error_log.match(f".*Failed renaming {backup_dir}.bak back to {backup_dir}") + inst = topo.standalone + dn = BDB_LDBMConfig(inst).dn + attr = 'nsslapd-db-home-directory' + with tempfile.TemporaryDirectory() as dbhome_dir: + with DseConfigContextManager(inst, dn, attr, dbhome_dir): + backup_dir = str(dbhome_dir) + inst.tasks.db2bak(backup_dir=backup_dir, args={TASK_WAIT: True}) + assert inst.ds_error_log.match(f".*Failed renaming {backup_dir}.bak back to {backup_dir}") + + def test_replication(topo_m2): """Test that if the dbhome directory is set causing an online backup to fail, @@ -168,6 +257,138 @@ def test_replication(topo_m2): repl.wait_for_replication(S1, S2) +def test_after_db_log_rotation(topo): + """Test that off line backup restore works as expected. + + :id: 8a091d92-a1cf-11ef-823a-482ae39447e5 + :setup: One standalone instance + :steps: + 1. Stop instance + 2. Perform off line backup on instance + 3. Start instance + 4. Perform modify operation until db log file rotates + 5. Stop instance + 6. Restore instance from backup + 7. Start instance + :expectedresults: + 1. Success + 2. Success + 3. Success + 4. Success + 5. Success + 6. Success + 7. Success + """ + inst = topo.standalone + with tempfile.TemporaryDirectory(dir=inst.ds_paths.backup_dir) as backup_dir: + # repl.wait_for_replication perform some changes and wait until they get replicated + inst.stop() + assert inst.db2bak(backup_dir) + inst.start() + cmd = [ 'ldclt', '-h', 'localhost', '-b', DEFAULT_SUFFIX, + '-p', str(inst.port), '-t', '60', '-N', '2', + '-D', DN_DM, '-w', PASSWORD, '-f', "ou=People", + '-e', 'attreplace=description:XXXXXX' ] + log.info(f'Running {cmd}') + # Perform modify operations until log file rolls + result = subprocess.run(cmd, capture_output=True, text=True, check=True) + log.info(f'STDOUT: {result.stdout}') + log.info(f'STDERR: {result.stderr}') + if get_default_db_lib() == 'bdb': + while os.path.isfile(f'{inst.ds_paths.db_dir}/log.0000000001'): + subprocess.run(cmd, capture_output=True, text=True, check=True) + inst.stop() + assert inst.bak2db(backup_dir) + inst.start() + + +def test_backup_task_after_failure(mytopo): + """Test that new backup task is successful after a failure. + backend that is no longer present. + + :id: a6c24898-2cd9-11ef-8c09-482ae39447e5 + :setup: Standalone Instance with multiple backends + :steps: + 1. Cleanup + 2. Perform a back up + 3. Rename the backup directory while waiting for backup completion. + 4. Check that backup failed. + 5. Perform a back up + 6. Check that backup succeed. + :expectedresults: + 1. Success + 2. Success + 3. Success + 4. Backup should fail + 5. Success + 6. Backup should succeed + """ + + inst = mytopo.standalone + tasks = inst.tasks + archive_dir1 = f'{inst.ds_paths.backup_dir}/bak1' + archive_dir1b = f'{inst.ds_paths.backup_dir}/bak1b' + archive_dir2 = f'{inst.ds_paths.backup_dir}/bak2' + + # Sometime the backup complete too fast, so lets retry if first + # backup is successful + for retry in range(50): + # Step 1. Perform cleanup + for dir in glob.glob(f'{inst.ds_paths.backup_dir}/*'): + shutil.rmtree(dir) + # Step 2. Perform a backup + tasks.db2bak(backup_dir=archive_dir1) + # Step 3. Wait until task is completed, trying to rename backup directory + done,exitCode,warningCode = (False, None, None) + while not done: + if os.path.isdir(archive_dir1): + os.rename(archive_dir1, archive_dir1b) + done,exitCode,warningCode = tasks.checkTask(tasks.entry) + time.sleep(0.01) + if exitCode != 0: + break + # Step 4. Check that backup failed. + # If next assert fails too often, that means that the backup is too fast + # A fix would would probably be to add more backends within mytopo + assert exitCode != 0, "Backup did not fail as expected." + # Step 5. Perform a seconf backup after backup failure + exitCode = tasks.db2bak(backup_dir=archive_dir2, args={TASK_WAIT: True}) + # Step 6. Check it is successful + assert exitCode == 0, "Backup failed. Issue #6229 may not be fixed." + + +def load_dse(inst): + conts = nsContainers(inst, 'cn=config') + while not event.is_set(): + cont = conts.create(properties={'cn': 'test_online_backup_and_dse_write'}) + cont.delete() + + +def test_online_backup_and_dse_write(topo): + """Test online backup while attempting to add/delete entries in dse.ldif. + + :id: 4a1edd2c-be15-11ef-8bc8-482ae39447e5 + :setup: One standalone instance + :steps: + 1. Start a thread that loops adding then removing in the dse.ldif + 2. Perform 10 online backups + 3. Stop the thread + :expectedresults: + 1. Success + 2. Success (or timeout if issue #6372 is not fixed.) + 3. Success + """ + inst = topo.standalone + t = Thread(target=load_dse, args=[inst]) + t.start() + for x in range(10): + with tempfile.TemporaryDirectory() as backup_dir: + assert inst.tasks.db2bak(backup_dir=f'{backup_dir}', args={TASK_WAIT: True}) == 0 + event.set() + t.join() + event.clear() + + if __name__ == '__main__': # Run isolated # -s for DEBUG mode diff --git a/ldap/servers/slapd/dse.c b/ldap/servers/slapd/dse.c index 2edc3f1f60..e3157c1ce5 100644 --- a/ldap/servers/slapd/dse.c +++ b/ldap/servers/slapd/dse.c @@ -41,6 +41,7 @@ #include /* Needed to access read_config_dse */ #include "proto-slap.h" +#include #include /* provides fsync/close */ @@ -73,9 +74,6 @@ #define DSE_USE_LOCK 1 #define DSE_NO_LOCK 0 -/* Global lock used during backups */ -static PRLock *backup_lock = NULL; - struct dse_callback { int operation; @@ -104,6 +102,8 @@ struct dse /* initialize the dse */ int dse_is_updateable; /* if non-zero, this DSE can be written to */ int dse_readonly_error_reported; /* used to ensure that read-only errors are logged only once */ + pthread_mutex_t dse_backup_lock; /* used to block write when online backup is in progress */ + bool dse_backup_in_progress; /* tell that online backup is in progress (protected by dse_rwlock) */ }; struct dse_node @@ -155,6 +155,91 @@ static int dse_write_entry(caddr_t data, caddr_t arg); static int ldif_record_end(char *p); static int dse_call_callback(struct dse *pdse, Slapi_PBlock *pb, int operation, int flags, Slapi_Entry *entryBefore, Slapi_Entry *entryAfter, int *returncode, char *returntext); +/* Lock the dse in read mode */ +INLINE_DIRECTIVE static void +dse_lock_read(struct dse *pdse, int use_lock) +{ + if (use_lock == DSE_USE_LOCK && pdse->dse_rwlock) { + slapi_rwlock_rdlock(pdse->dse_rwlock); + } +} + +/* Lock the dse in write mode and wait until the */ +INLINE_DIRECTIVE static void +dse_lock_write(struct dse *pdse, int use_lock) +{ + if (use_lock != DSE_USE_LOCK || !pdse->dse_rwlock) { + return; + } + slapi_rwlock_wrlock(pdse->dse_rwlock); + while (pdse->dse_backup_in_progress) { + slapi_rwlock_unlock(pdse->dse_rwlock); + /* Wait util dse_backup_lock is unlocked */ + pthread_mutex_lock(&pdse->dse_backup_lock); + pthread_mutex_unlock(&pdse->dse_backup_lock); + slapi_rwlock_wrlock(pdse->dse_rwlock); + } +} + +/* release the dse lock */ +INLINE_DIRECTIVE static void +dse_lock_unlock(struct dse *pdse, int use_lock) +{ + if (use_lock == DSE_USE_LOCK && pdse->dse_rwlock) { + slapi_rwlock_unlock(pdse->dse_rwlock); + } +} + +/* Call cb(pdse) */ +INLINE_DIRECTIVE static void +dse_call_cb(void (*cb)(struct dse*)) +{ + Slapi_Backend *be = slapi_be_select_by_instance_name("DSE"); + if (be) { + struct dse *pdse = NULL; + slapi_be_Rlock(be); + pdse = be->be_database->plg_private; + if (pdse) { + cb(pdse); + } + slapi_be_Unlock(be); + } +} + +/* Helper for dse_backup_lock() */ +static void +dse_backup_lock_cb(struct dse *pdse) +{ + pthread_mutex_lock(&pdse->dse_backup_lock); + slapi_rwlock_wrlock(pdse->dse_rwlock); + pdse->dse_backup_in_progress = true; + slapi_rwlock_unlock(pdse->dse_rwlock); +} + +/* Helper for dse_backup_unlock() */ +static void +dse_backup_unlock_cb(struct dse *pdse) +{ + slapi_rwlock_wrlock(pdse->dse_rwlock); + pdse->dse_backup_in_progress = false; + slapi_rwlock_unlock(pdse->dse_rwlock); + pthread_mutex_unlock(&pdse->dse_backup_lock); +} + +/* Tells that a backup thread is starting */ +void +dse_backup_lock() +{ + dse_call_cb(dse_backup_lock_cb); +} + +/* Tells that a backup thread is ending */ +void +dse_backup_unlock() +{ + dse_call_cb(dse_backup_unlock_cb); +} + /* * Map a DN onto a dse_node. * Returns NULL if not found. @@ -192,18 +277,12 @@ dse_get_entry_copy(struct dse *pdse, const Slapi_DN *dn, int use_lock) Slapi_Entry *e = NULL; struct dse_node *n; - if (use_lock == DSE_USE_LOCK && pdse->dse_rwlock) { - slapi_rwlock_rdlock(pdse->dse_rwlock); - } - + dse_lock_read(pdse, use_lock); n = dse_find_node(pdse, dn); if (n != NULL) { e = slapi_entry_dup(n->entry); } - - if (use_lock == DSE_USE_LOCK && pdse->dse_rwlock) { - slapi_rwlock_unlock(pdse->dse_rwlock); - } + dse_lock_unlock(pdse, use_lock); return e; } @@ -393,6 +472,7 @@ dse_new(char *filename, char *tmpfilename, char *backfilename, char *startokfile pdse->dse_callback = NULL; pdse->dse_is_updateable = dse_permission_to_write(pdse, SLAPI_LOG_TRACE); + pthread_mutex_init(&pdse->dse_backup_lock, NULL); } slapi_ch_free((void **)&realconfigdir); } @@ -429,8 +509,7 @@ dse_destroy(struct dse *pdse) if (NULL == pdse) { return 0; /* no one checks this return value */ } - if (pdse->dse_rwlock) - slapi_rwlock_wrlock(pdse->dse_rwlock); + dse_lock_write(pdse, DSE_USE_LOCK); slapi_ch_free((void **)&(pdse->dse_filename)); slapi_ch_free((void **)&(pdse->dse_tmpfile)); slapi_ch_free((void **)&(pdse->dse_fileback)); @@ -439,8 +518,8 @@ dse_destroy(struct dse *pdse) dse_callback_deletelist(&pdse->dse_callback); charray_free(pdse->dse_filelist); nentries = avl_free(pdse->dse_tree, dse_internal_delete_entry); + dse_lock_unlock(pdse, DSE_USE_LOCK); if (pdse->dse_rwlock) { - slapi_rwlock_unlock(pdse->dse_rwlock); slapi_destroy_rwlock(pdse->dse_rwlock); } slapi_ch_free((void **)&pdse); @@ -928,9 +1007,7 @@ dse_check_for_readonly_error(Slapi_PBlock *pb, struct dse *pdse) { int rc = 0; /* default: no error */ - if (pdse->dse_rwlock) - slapi_rwlock_rdlock(pdse->dse_rwlock); - + dse_lock_read(pdse, DSE_USE_LOCK); if (!pdse->dse_is_updateable) { if (!pdse->dse_readonly_error_reported) { if (NULL != pdse->dse_filename) { @@ -944,9 +1021,7 @@ dse_check_for_readonly_error(Slapi_PBlock *pb, struct dse *pdse) } rc = 1; /* return an error to the client */ } - - if (pdse->dse_rwlock) - slapi_rwlock_unlock(pdse->dse_rwlock); + dse_lock_unlock(pdse, DSE_USE_LOCK); if (rc != 0) { slapi_send_ldap_result(pb, LDAP_UNWILLING_TO_PERFORM, NULL, @@ -973,8 +1048,6 @@ dse_write_file_nolock(struct dse *pdse) fpw.fpw_rc = 0; fpw.fpw_prfd = NULL; - dse_backup_lock(); - if (NULL != pdse->dse_filename) { if ((fpw.fpw_prfd = PR_Open(pdse->dse_tmpfile, PR_RDWR | PR_CREATE_FILE | PR_TRUNCATE, SLAPD_DEFAULT_DSE_FILE_MODE)) == NULL) { rc = PR_GetOSError(); @@ -1107,8 +1180,7 @@ dse_add_entry_pb(struct dse *pdse, Slapi_Entry *e, Slapi_PBlock *pb) slapi_pblock_get(pb, SLAPI_DSE_MERGE_WHEN_ADDING, &merge); /* keep write lock during both tree update and file write operations */ - if (pdse->dse_rwlock) - slapi_rwlock_wrlock(pdse->dse_rwlock); + dse_lock_write(pdse, DSE_USE_LOCK); if (merge) { rc = avl_insert(&(pdse->dse_tree), n, entry_dn_cmp, dupentry_merge); } else { @@ -1131,8 +1203,7 @@ dse_add_entry_pb(struct dse *pdse, Slapi_Entry *e, Slapi_PBlock *pb) } else { /* duplicate entry ignored */ dse_node_delete(&n); /* This also deletes the contained entry */ } - if (pdse->dse_rwlock) - slapi_rwlock_unlock(pdse->dse_rwlock); + dse_lock_unlock(pdse, DSE_USE_LOCK); if (rc == -1) { /* duplicate entry ignored */ @@ -1299,8 +1370,7 @@ dse_replace_entry(struct dse *pdse, Slapi_Entry *e, int write_file, int use_lock int rc = -1; if (NULL != e) { struct dse_node *n = dse_node_new(e); - if (use_lock && pdse->dse_rwlock) - slapi_rwlock_wrlock(pdse->dse_rwlock); + dse_lock_write(pdse, use_lock); rc = avl_insert(&(pdse->dse_tree), n, entry_dn_cmp, dupentry_replace); if (write_file) dse_write_file_nolock(pdse); @@ -1310,8 +1380,7 @@ dse_replace_entry(struct dse *pdse, Slapi_Entry *e, int write_file, int use_lock dse_node_delete(&n); rc = 0; /* for return to caller */ } - if (use_lock && pdse->dse_rwlock) - slapi_rwlock_unlock(pdse->dse_rwlock); + dse_lock_unlock(pdse, use_lock); } return rc; } @@ -1398,8 +1467,7 @@ dse_delete_entry(struct dse *pdse, Slapi_PBlock *pb, const Slapi_Entry *e) slapi_pblock_get(pb, SLAPI_DSE_DONT_WRITE_WHEN_ADDING, &dont_write_file); /* keep write lock for both tree deleting and file writing */ - if (pdse->dse_rwlock) - slapi_rwlock_wrlock(pdse->dse_rwlock); + dse_lock_write(pdse, DSE_USE_LOCK); if ((deleted_node = (struct dse_node *)avl_delete(&pdse->dse_tree, n, entry_dn_cmp))) { dse_node_delete(&deleted_node); } @@ -1411,8 +1479,7 @@ dse_delete_entry(struct dse *pdse, Slapi_PBlock *pb, const Slapi_Entry *e) SLAPI_OPERATION_DELETE); dse_write_file_nolock(pdse); } - if (pdse->dse_rwlock) - slapi_rwlock_unlock(pdse->dse_rwlock); + dse_lock_unlock(pdse, DSE_USE_LOCK); return 1; } @@ -1574,11 +1641,9 @@ do_dse_search(struct dse *pdse, Slapi_PBlock *pb, int scope, const Slapi_DN *bas * entries that change, we skip looking through the DSE entries. */ if (pb_op == NULL || !operation_is_flag_set(pb_op, OP_FLAG_PS_CHANGESONLY)) { - if (pdse->dse_rwlock) - slapi_rwlock_rdlock(pdse->dse_rwlock); + dse_lock_read(pdse, DSE_USE_LOCK); dse_apply_nolock(pdse, dse_search_filter_entry, (caddr_t)&stuff); - if (pdse->dse_rwlock) - slapi_rwlock_unlock(pdse->dse_rwlock); + dse_lock_unlock(pdse, DSE_USE_LOCK); } if (stuff.ss) /* something was found which matched our criteria */ @@ -2925,32 +2990,3 @@ dse_next_search_entry(Slapi_PBlock *pb) return 0; } -/* When a backup is occurring we can not allow the writing the dse.ldif file */ -void -dse_init_backup_lock() -{ - backup_lock = PR_NewLock(); -} - -void -dse_destroy_backup_lock() -{ - PR_DestroyLock(backup_lock); - backup_lock = NULL; -} - -void -dse_backup_lock() -{ - if (backup_lock) { - PR_Lock(backup_lock); - } -} - -void -dse_backup_unlock() -{ - if (backup_lock) { - PR_Unlock(backup_lock); - } -} diff --git a/ldap/servers/slapd/libglobs.c b/ldap/servers/slapd/libglobs.c index 0f4e6c88a3..c6f64c6980 100644 --- a/ldap/servers/slapd/libglobs.c +++ b/ldap/servers/slapd/libglobs.c @@ -2023,9 +2023,6 @@ FrontendConfig_init(void) /* Done, unlock! */ CFG_UNLOCK_WRITE(cfg); - /* init the dse file backup lock */ - dse_init_backup_lock(); - init_config_get_and_set(); } diff --git a/ldap/servers/slapd/main.c b/ldap/servers/slapd/main.c index a5352e298a..23e753bf3f 100644 --- a/ldap/servers/slapd/main.c +++ b/ldap/servers/slapd/main.c @@ -1158,7 +1158,6 @@ main(int argc, char **argv) slapd_ssl_destroy(); ndn_cache_destroy(); NSS_Shutdown(); - dse_destroy_backup_lock(); /* * Server has stopped, lets force everything to disk: logs diff --git a/ldap/servers/slapd/slapi-private.h b/ldap/servers/slapd/slapi-private.h index 0ee5cc0694..a61daab755 100644 --- a/ldap/servers/slapd/slapi-private.h +++ b/ldap/servers/slapd/slapi-private.h @@ -1414,8 +1414,6 @@ void modify_update_last_modified_attr(Slapi_PBlock *pb, Slapi_Mods *smods); void add_internal_modifiersname(Slapi_PBlock *pb, Slapi_Entry *e); /* dse.c */ -void dse_init_backup_lock(void); -void dse_destroy_backup_lock(void); void dse_backup_lock(void); void dse_backup_unlock(void); diff --git a/src/lib389/lib389/tasks.py b/src/lib389/lib389/tasks.py index 193805780a..6c2adb5b22 100644 --- a/src/lib389/lib389/tasks.py +++ b/src/lib389/lib389/tasks.py @@ -525,7 +525,7 @@ def importLDIF(self, suffix=None, benamebase=None, input_file=None, entry.setValues('nsIncludeSuffix', suffix) # start the task and possibly wait for task completion - self.conn.add_s(entry) + self.conn.add_s(entry, escapehatch='i am sure') exitCode = 0 warningCode = 0 @@ -598,7 +598,7 @@ def exportLDIF(self, suffix=None, benamebase=None, output_file=None, entry.setValues('nsExportReplica', 'true') # start the task and possibly wait for task completion - self.conn.add_s(entry) + self.conn.add_s(entry, escapehatch='i am sure') exitCode = 0 warningCode = 0 if args and args.get(TASK_WAIT, False): @@ -649,7 +649,7 @@ def db2bak(self, backup_dir=None, args=None): # start the task and possibly wait for task completion try: - self.conn.add_s(entry) + self.conn.add_s(entry, escapehatch='i am sure') except ldap.ALREADY_EXISTS: self.log.error("Fail to add the backup task (%s)", dn) return -1 @@ -706,7 +706,7 @@ def bak2db(self, backup_dir=None, args=None): # start the task and possibly wait for task completion try: - self.conn.add_s(entry) + self.conn.add_s(entry, escapehatch='i am sure') except ldap.ALREADY_EXISTS: self.log.error("Fail to add the backup task (%s)", dn) return -1 @@ -834,7 +834,7 @@ def reindex(self, suffix=None, benamebase=None, attrname=None, args=None, vlv=Fa # start the task and possibly wait for task completion try: - self.conn.add_s(entry) + self.conn.add_s(entry, escapehatch='i am sure') except ldap.ALREADY_EXISTS: self.log.error("Fail to add the index task for %s", attrname) return -1 @@ -915,7 +915,7 @@ def fixupMemberOf(self, suffix=None, benamebase=None, filt=None, # start the task and possibly wait for task completion try: - self.conn.add_s(entry) + self.conn.add_s(entry, escapehatch='i am sure') except ldap.ALREADY_EXISTS: self.log.error("Fail to add the memberOf fixup task") return -1 @@ -977,7 +977,7 @@ def fixupTombstones(self, bename=None, args=None): # start the task and possibly wait for task completion try: - self.conn.add_s(entry) + self.conn.add_s(entry, escapehatch='i am sure') except ldap.ALREADY_EXISTS: self.log.error("Fail to add the fixup tombstone task") return -1 @@ -1033,7 +1033,7 @@ def automemberRebuild(self, suffix=DEFAULT_SUFFIX, scope='sub', # start the task and possibly wait for task completion try: - self.conn.add_s(entry) + self.conn.add_s(entry, escapehatch='i am sure') except ldap.ALREADY_EXISTS: self.log.error("Fail to add Automember Rebuild Membership task") return -1 @@ -1089,7 +1089,7 @@ def automemberExport(self, suffix=DEFAULT_SUFFIX, scope='sub', # start the task and possibly wait for task completion try: - self.conn.add_s(entry) + self.conn.add_s(entry, escapehatch='i am sure') except ldap.ALREADY_EXISTS: self.log.error("Fail to add Automember Export Updates task") return -1 @@ -1140,7 +1140,7 @@ def automemberMap(self, ldif_in=None, ldif_out=None, args=None): # start the task and possibly wait for task completion try: - self.conn.add_s(entry) + self.conn.add_s(entry, escapehatch='i am sure') except ldap.ALREADY_EXISTS: self.log.error("Fail to add Automember Map Updates task") return -1 @@ -1185,7 +1185,7 @@ def fixupLinkedAttrs(self, linkdn=None, args=None): # start the task and possibly wait for task completion try: - self.conn.add_s(entry) + self.conn.add_s(entry, escapehatch='i am sure') except ldap.ALREADY_EXISTS: self.log.error("Fail to add Fixup Linked Attributes task") return -1 @@ -1229,7 +1229,7 @@ def schemaReload(self, schemadir=None, args=None): # start the task and possibly wait for task completion try: - self.conn.add_s(entry) + self.conn.add_s(entry, escapehatch='i am sure') except ldap.ALREADY_EXISTS: self.log.error("Fail to add Schema Reload task") return -1 @@ -1274,7 +1274,7 @@ def fixupWinsyncMembers(self, suffix=DEFAULT_SUFFIX, # start the task and possibly wait for task completion try: - self.conn.add_s(entry) + self.conn.add_s(entry, escapehatch='i am sure') except ldap.ALREADY_EXISTS: self.log.error("Fail to add fixupWinsyncMembers 'memberuid task'") return -1 @@ -1321,7 +1321,7 @@ def syntaxValidate(self, suffix=DEFAULT_SUFFIX, fstr='objectclass=top', # start the task and possibly wait for task completion try: - self.conn.add_s(entry) + self.conn.add_s(entry, escapehatch='i am sure') except ldap.ALREADY_EXISTS: self.log.error("Fail to add Syntax Validate task") return -1 @@ -1372,7 +1372,7 @@ def usnTombstoneCleanup(self, suffix=DEFAULT_SUFFIX, bename=None, # start the task and possibly wait for task completion try: - self.conn.add_s(entry) + self.conn.add_s(entry, escapehatch='i am sure') except ldap.ALREADY_EXISTS: self.log.error("Fail to add USN tombstone cleanup task") return -1 @@ -1423,7 +1423,7 @@ def sysconfigReload(self, configfile=None, logchanges=None, args=None): entry.setValues('logchanges', logchanges) # start the task and possibly wait for task completion try: - self.conn.add_s(entry) + self.conn.add_s(entry, escapehatch='i am sure') except ldap.ALREADY_EXISTS: self.log.error("Fail to add Sysconfig Reload task") return -1 @@ -1475,7 +1475,7 @@ def cleanAllRUV(self, suffix=None, replicaid=None, force=None, args=None): entry.setValues('replica-force-cleaning', 'yes') # start the task and possibly wait for task completion try: - self.conn.add_s(entry) + self.conn.add_s(entry, escapehatch='i am sure') except ldap.ALREADY_EXISTS: self.log.error("Fail to add cleanAllRUV task") return (dn, -1) @@ -1530,7 +1530,7 @@ def abortCleanAllRUV(self, suffix=None, replicaid=None, certify=None, # start the task and possibly wait for task completion try: - self.conn.add_s(entry) + self.conn.add_s(entry, escapehatch='i am sure') except ldap.ALREADY_EXISTS: self.log.error("Fail to add Abort cleanAllRUV task") return (dn, -1) @@ -1584,7 +1584,7 @@ def upgradeDB(self, nsArchiveDir=None, nsDatabaseType=None, # start the task and possibly wait for task completion try: - self.conn.add_s(entry) + self.conn.add_s(entry, escapehatch='i am sure') except ldap.ALREADY_EXISTS: self.log.error("Fail to add upgradedb task") return -1