From c71f46713086b2a0591af7464544cff43fe0ee43 Mon Sep 17 00:00:00 2001 From: Viktor Ashirov Date: Mon, 16 Sep 2024 13:16:07 +0200 Subject: [PATCH] Bump version to 2.5.2-2 - Resolves: RHEL-55744 - ipahealthcheck.ds.backends.BackendsCheck.DSBLE0006: BDB is deprecated and should not be used as a backend --- ...anch-2.5-healthcheck-report-an-inval.patch | 60 +++++ ...reindex-is-broken-if-index-type-is-s.patch | 237 ++++++++++++++++++ 389-ds-base.spec | 9 +- 3 files changed, 305 insertions(+), 1 deletion(-) create mode 100644 0001-Issue-6312-In-branch-2.5-healthcheck-report-an-inval.patch create mode 100644 0002-Issue-6316-lmdb-reindex-is-broken-if-index-type-is-s.patch diff --git a/0001-Issue-6312-In-branch-2.5-healthcheck-report-an-inval.patch b/0001-Issue-6312-In-branch-2.5-healthcheck-report-an-inval.patch new file mode 100644 index 0000000..ff6da91 --- /dev/null +++ b/0001-Issue-6312-In-branch-2.5-healthcheck-report-an-inval.patch @@ -0,0 +1,60 @@ +From 0ff5aa641d619bdcc154c2c94f8f8180bcaec776 Mon Sep 17 00:00:00 2001 +From: tbordaz +Date: Thu, 29 Aug 2024 10:49:57 +0200 +Subject: [PATCH] Issue 6312 - In branch 2.5, healthcheck report an invalid + warning regarding BDB deprecation (#6313) + +Bug description: + during healthcheck, _lint_backend_implementation checks that + the instance is not running a BDB backend. + This check only applies for instance after 3.0.0 + +Fix description: + If the instance is newer than 3.0.0 the health check + just returns + +relates: #6312 + +Reviewed by: +--- + dirsrvtests/tests/suites/healthcheck/healthcheck_test.py | 1 + + src/lib389/lib389/backend.py | 4 ++-- + 2 files changed, 3 insertions(+), 2 deletions(-) + +diff --git a/dirsrvtests/tests/suites/healthcheck/healthcheck_test.py b/dirsrvtests/tests/suites/healthcheck/healthcheck_test.py +index 29cca187e..66cf3c7d3 100644 +--- a/dirsrvtests/tests/suites/healthcheck/healthcheck_test.py ++++ b/dirsrvtests/tests/suites/healthcheck/healthcheck_test.py +@@ -556,6 +556,7 @@ def test_lint_backend_implementation_wrong_files(topology_st): + + + @pytest.mark.skipif(get_default_db_lib() == "mdb", reason="Not needed for mdb") ++@pytest.mark.skipif(ds_is_older("3.0.0"), reason="mdb and bdb are both supported") + def test_lint_backend_implementation(topology_st): + """Test the lint for backend implementation mismatch + +diff --git a/src/lib389/lib389/backend.py b/src/lib389/lib389/backend.py +index caee88e6a..0ed00a4a7 100644 +--- a/src/lib389/lib389/backend.py ++++ b/src/lib389/lib389/backend.py +@@ -14,7 +14,7 @@ from lib389._constants import DN_LDBM, DN_CHAIN, DN_PLUGIN, DEFAULT_BENAME + from lib389.properties import BACKEND_OBJECTCLASS_VALUE, BACKEND_PROPNAME_TO_ATTRNAME, BACKEND_CHAIN_BIND_DN, \ + BACKEND_CHAIN_BIND_PW, BACKEND_CHAIN_URLS, BACKEND_PROPNAME_TO_ATTRNAME, BACKEND_NAME, \ + BACKEND_SUFFIX, BACKEND_SAMPLE_ENTRIES, TASK_WAIT +-from lib389.utils import normalizeDN, ensure_str, assert_c ++from lib389.utils import normalizeDN, ensure_str, assert_c, ds_is_newer + from lib389 import Entry + + # Need to fix this .... +@@ -513,7 +513,7 @@ class Backend(DSLdapObject): + + def _lint_backend_implementation(self): + backend_impl = self._instance.get_db_lib() +- if backend_impl == 'bdb': ++ if backend_impl == 'bdb' and ds_is_newer('3.0.0', instance=self._instance): + result = DSBLE0006 + result['items'] = [self.lint_uid()] + yield result +-- +2.46.0 + diff --git a/0002-Issue-6316-lmdb-reindex-is-broken-if-index-type-is-s.patch b/0002-Issue-6316-lmdb-reindex-is-broken-if-index-type-is-s.patch new file mode 100644 index 0000000..da27be5 --- /dev/null +++ b/0002-Issue-6316-lmdb-reindex-is-broken-if-index-type-is-s.patch @@ -0,0 +1,237 @@ +From af27f433ec14bcaf070108ab0b6af64ad1153a11 Mon Sep 17 00:00:00 2001 +From: progier389 +Date: Fri, 6 Sep 2024 18:07:17 +0200 +Subject: [PATCH] Issue 6316 - lmdb reindex is broken if index type is + specified (#6318) + +While reindexing using task or offline reindex, if the attribute name contains the index type (for example :eq,pres) +Then the attribute is not reindexed. Problem occurs when lmdb is used, things are working fine with bdb. +Solution: strip the index type in reindex as it is done in bdb case. +Anyway the reindex design requires that for a given attribute all the configured index types must be rebuild. + +Issue: #6316 + +Reviewed by: @tbordaz, @droideck (Thanks!) +--- + .../tests/suites/indexes/regression_test.py | 141 +++++++++++++++++- + .../slapd/back-ldbm/db-mdb/mdb_import.c | 10 +- + 2 files changed, 147 insertions(+), 4 deletions(-) + +diff --git a/dirsrvtests/tests/suites/indexes/regression_test.py b/dirsrvtests/tests/suites/indexes/regression_test.py +index c385f5ca4..b077b529a 100644 +--- a/dirsrvtests/tests/suites/indexes/regression_test.py ++++ b/dirsrvtests/tests/suites/indexes/regression_test.py +@@ -10,6 +10,9 @@ import time + import os + import pytest + import ldap ++import logging ++import glob ++import re + from lib389._constants import DEFAULT_BENAME, DEFAULT_SUFFIX + from lib389.backend import Backend, Backends, DatabaseConfig + from lib389.cos import CosClassicDefinition, CosClassicDefinitions, CosTemplate +@@ -31,6 +34,8 @@ SUFFIX2 = 'dc=example2,dc=com' + BENAME2 = 'be2' + + DEBUGGING = os.getenv("DEBUGGING", default=False) ++logging.getLogger(__name__).setLevel(logging.INFO) ++log = logging.getLogger(__name__) + + + @pytest.fixture(scope="function") +@@ -83,6 +88,7 @@ def add_a_group_with_users(request, topo): + 'cn': USER_NAME, + 'uidNumber': f'{num}', + 'gidNumber': f'{num}', ++ 'description': f'Description for {USER_NAME}', + 'homeDirectory': f'/home/{USER_NAME}' + }) + users_list.append(user) +@@ -95,9 +101,10 @@ def add_a_group_with_users(request, topo): + # If the server crashed, start it again to do the cleanup + if not topo.standalone.status(): + topo.standalone.start() +- for user in users_list: +- user.delete() +- group.delete() ++ if not DEBUGGING: ++ for user in users_list: ++ user.delete() ++ group.delete() + + request.addfinalizer(fin) + +@@ -124,6 +131,38 @@ def set_small_idlistscanlimit(request, topo): + + request.addfinalizer(fin) + ++ ++@pytest.fixture(scope="function") ++def set_description_index(request, topo, add_a_group_with_users): ++ """ ++ Set some description values and description index without reindexing. ++ """ ++ inst = topo.standalone ++ backends = Backends(inst) ++ backend = backends.get(DEFAULT_BENAME) ++ indexes = backend.get_indexes() ++ attr = 'description' ++ ++ def fin(always=False): ++ if always or not DEBUGGING: ++ try: ++ idx = indexes.get(attr) ++ idx.delete() ++ except ldap.NO_SUCH_OBJECT: ++ pass ++ ++ request.addfinalizer(fin) ++ fin(always=True) ++ index = indexes.create(properties={ ++ 'cn': attr, ++ 'nsSystemIndex': 'false', ++ 'nsIndexType': ['eq', 'pres', 'sub'] ++ }) ++ # Restart needed with lmdb (to open the dbi handle) ++ inst.restart() ++ return (indexes, attr) ++ ++ + #unstable or unstatus tests, skipped for now + @pytest.mark.flaky(max_runs=2, min_passes=1) + @pytest.mark.skipif(ds_is_older("1.4.4.4"), reason="Not implemented") +@@ -347,6 +386,102 @@ def test_task_status(topo): + assert reindex_task.get_exit_code() == 0 + + ++def count_keys(inst, bename, attr, prefix=''): ++ indexfile = os.path.join(inst.dbdir, bename, attr + '.db') ++ # (bdb - we should also accept a version number for .db suffix) ++ for f in glob.glob(f'{indexfile}*'): ++ indexfile = f ++ ++ inst.stop() ++ output = inst.dbscan(None, None, args=['-f', indexfile, '-A'], stopping=False).decode() ++ inst.start() ++ count = 0 ++ regexp = f'^KEY: {re.escape(prefix)}' ++ for match in re.finditer(regexp, output, flags=re.MULTILINE): ++ count += 1 ++ log.info(f"count_keys found {count} keys starting with '{prefix}' in {indexfile}") ++ return count ++ ++ ++def test_reindex_task_with_type(topo, set_description_index): ++ """Check that reindex task works as expected when index type is specified. ++ ++ :id: 0c7f2fda-69f6-11ef-9eb8-083a88554478 ++ :setup: Standalone instance ++ - with 100 users having description attribute ++ - with description:eq,pres,sub index entry but not yet reindexed ++ :steps: ++ 1. Set description in suffix entry ++ 2. Count number of equality keys in description index ++ 3. Start a Reindex task on description:eq,pres and wait for completion ++ 4. Check the task status and exit code ++ 5. Count the equality, presence and substring keys in description index ++ 6. Start a Reindex task on description and wait for completion ++ 7. Check the task status and exit code ++ 8. Count the equality, presence and substring keys in description index ++ ++ :expectedresults: ++ 1. Success ++ 2. Should be either no key (bdb) or a single one (lmdb) ++ 3. Success ++ 4. Success ++ 5. Should have: more equality keys than in step 2 ++ one presence key ++ some substrings keys ++ 6. Success ++ 7. Success ++ 8. Should have same counts than in step 5 ++ """ ++ (indexes, attr) = set_description_index ++ inst = topo.standalone ++ if not inst.is_dbi_supported(): ++ pytest.skip('This test requires that dbscan supports -A option') ++ # modify indexed value ++ Domain(inst, DEFAULT_SUFFIX).replace(attr, f'test_before_reindex') ++ ++ keys1 = count_keys(inst, DEFAULT_BENAME, attr, prefix='=') ++ assert keys1 <= 1 ++ ++ tasks = Tasks(topo.standalone) ++ # completed reindex tasks MUST have a status because freeipa check it. ++ ++ # Reindex attr with eq,pres types ++ log.info(f'Reindex {attr} with eq,pres types') ++ tasks.reindex( ++ suffix=DEFAULT_SUFFIX, ++ attrname=f'{attr}:eq,pres', ++ args={TASK_WAIT: True} ++ ) ++ reindex_task = Task(topo.standalone, tasks.dn) ++ assert reindex_task.status() ++ assert reindex_task.get_exit_code() == 0 ++ ++ keys2e = count_keys(inst, DEFAULT_BENAME, attr, prefix='=') ++ keys2p = count_keys(inst, DEFAULT_BENAME, attr, prefix='+') ++ keys2s = count_keys(inst, DEFAULT_BENAME, attr, prefix='*') ++ assert keys2e > keys1 ++ assert keys2p > 0 ++ assert keys2s > 0 ++ ++ # Reindex attr without types ++ log.info(f'Reindex {attr} without types') ++ tasks.reindex( ++ suffix=DEFAULT_SUFFIX, ++ attrname=attr, ++ args={TASK_WAIT: True} ++ ) ++ reindex_task = Task(topo.standalone, tasks.dn) ++ assert reindex_task.status() ++ assert reindex_task.get_exit_code() == 0 ++ ++ keys3e = count_keys(inst, DEFAULT_BENAME, attr, prefix='=') ++ keys3p = count_keys(inst, DEFAULT_BENAME, attr, prefix='+') ++ keys3s = count_keys(inst, DEFAULT_BENAME, attr, prefix='*') ++ assert keys3e == keys2e ++ assert keys3p == keys2p ++ assert keys3s == keys2s ++ ++ + def test_task_and_be(topo, add_backend_and_ldif_50K_users): + """Check that backend is writable after finishing a tasks + +diff --git a/ldap/servers/slapd/back-ldbm/db-mdb/mdb_import.c b/ldap/servers/slapd/back-ldbm/db-mdb/mdb_import.c +index d57146953..ce2151174 100644 +--- a/ldap/servers/slapd/back-ldbm/db-mdb/mdb_import.c ++++ b/ldap/servers/slapd/back-ldbm/db-mdb/mdb_import.c +@@ -1150,6 +1150,8 @@ process_db2index_attrs(Slapi_PBlock *pb, ImportCtx_t *ctx) + * TBD + */ + char **attrs = NULL; ++ char *attrname = NULL; ++ char *pt = NULL; + int i; + + slapi_pblock_get(pb, SLAPI_DB2INDEX_ATTRS, &attrs); +@@ -1157,7 +1159,13 @@ process_db2index_attrs(Slapi_PBlock *pb, ImportCtx_t *ctx) + for (i = 0; attrs && attrs[i]; i++) { + switch (attrs[i][0]) { + case 't': /* attribute type to index */ +- slapi_ch_array_add(&ctx->indexAttrs, slapi_ch_strdup(attrs[i] + 1)); ++ attrname = slapi_ch_strdup(attrs[i] + 1); ++ /* Strip index type */ ++ pt = strchr(attrname, ':'); ++ if (pt != NULL) { ++ *pt = '\0'; ++ } ++ slapi_ch_array_add(&ctx->indexAttrs, attrname); + break; + case 'T': /* VLV Search to index */ + slapi_ch_array_add(&ctx->indexVlvs, get_vlv_dbname(attrs[i] + 1)); +-- +2.46.0 + diff --git a/389-ds-base.spec b/389-ds-base.spec index 458e84e..938d441 100644 --- a/389-ds-base.spec +++ b/389-ds-base.spec @@ -47,7 +47,7 @@ ExcludeArch: i686 Summary: 389 Directory Server (base) Name: 389-ds-base Version: 2.5.2 -Release: 1%{?dist} +Release: 2%{?dist} License: GPL-3.0-or-later AND (0BSD OR Apache-2.0 OR MIT) AND (Apache-2.0 OR Apache-2.0 WITH LLVM-exception OR MIT) AND (Apache-2.0 OR BSL-1.0) AND (Apache-2.0 OR MIT OR Zlib) AND (Apache-2.0 OR MIT) AND (CC-BY-4.0 AND MIT) AND (MIT OR Apache-2.0) AND Unicode-DFS-2016 AND (MIT OR CC0-1.0) AND (MIT OR Unlicense) AND 0BSD AND Apache-2.0 AND BSD-2-Clause AND BSD-3-Clause AND ISC AND MIT AND MIT AND ISC AND MPL-2.0 AND PSF-2.0 URL: https://www.port389.org Conflicts: selinux-policy-base < 3.9.8 @@ -469,6 +469,9 @@ Source3: https://github.com/jemalloc/%{jemalloc_name}/releases/download %endif Source4: 389-ds-base.sysusers +Patch: 0001-Issue-6312-In-branch-2.5-healthcheck-report-an-inval.patch +Patch: 0002-Issue-6316-lmdb-reindex-is-broken-if-index-type-is-s.patch + %description 389 Directory Server is an LDAPv3 compliant server. The base package includes @@ -911,6 +914,10 @@ exit 0 %endif %changelog +* Mon Sep 16 2024 Viktor Ashirov - 2.5.2-2 +- Bump version to 2.5.2-2 +- Resolves: RHEL-55744 - ipahealthcheck.ds.backends.BackendsCheck.DSBLE0006: BDB is deprecated and should not be used as a backend + * Mon Aug 12 2024 Viktor Ashirov - 2.5.2-1 - Bump version to 2.5.2-1 - Resolves: RHEL-5108 - ns-slapd crash in referint_get_config