From 04bb61ef7acd42de7b71963ef024d831893864bb Mon Sep 17 00:00:00 2001 From: Vladimir Sedmik Date: Thu, 17 Oct 2024 18:16:27 +0200 Subject: [PATCH 1/2] Add a test case for capsule content counts granularity --- tests/foreman/cli/test_capsulecontent.py | 178 +++++++++++++++++++++++ 1 file changed, 178 insertions(+) diff --git a/tests/foreman/cli/test_capsulecontent.py b/tests/foreman/cli/test_capsulecontent.py index c231d2a69ea..c676bf694e0 100644 --- a/tests/foreman/cli/test_capsulecontent.py +++ b/tests/foreman/cli/test_capsulecontent.py @@ -22,6 +22,7 @@ from robottelo.constants import ( CONTAINER_REGISTRY_HUB, CONTAINER_UPSTREAM_NAME, + FAKE_0_CUSTOM_PACKAGE_NAME, PULP_EXPORT_DIR, ) from robottelo.constants.repos import ANSIBLE_GALAXY, CUSTOM_FILE_REPO @@ -124,6 +125,32 @@ def module_capsule_artifact_cleanup( ) +def _dictionarize_counts(data): + """From CLI content counts usually come as a dict with several nested dicts using numbers as + keys which is not practical for assertions. This helper refactors the dict with entity names + for keys and grabs just the content counts. + """ + refactored_data = {} + + for lce_value in data['lifecycle-environments'].values(): + lce_name = lce_value['name'] + cvs_dict = {} + + for cv_value in lce_value['content-views'].values(): + cv_name = cv_value['name']['name'] + repo_dict = {} + + for repo_value in cv_value['repositories'].values(): + repo_name = repo_value['repository-name'] + repo_dict[repo_name] = repo_value['content-counts'] + + cvs_dict[cv_name] = repo_dict + + refactored_data[lce_name] = cvs_dict + + return refactored_data + + @pytest.mark.parametrize( 'repos_collection', [ @@ -309,6 +336,157 @@ def test_positive_update_counts(target_sat, module_capsule_configured): ) +@pytest.mark.parametrize('setting_update', ['automatic_content_count_updates=True'], indirect=True) +def test_positive_content_counts_granularity( + request, + module_target_sat, + module_capsule_configured, + setting_update, + function_org, + function_product, +): + """Verify the Capsule content counts are updated separately for each content view when it is + promoted to the Capsule's LCEs. + + :id: 14f9dbba-6d12-4faa-b33f-7ba1f328f8b6 + + :parametrized: yes + + :setup: + 1. Satellite with registered external Capsule. + + :steps: + 1. Create two LCEs, assign them to the Capsule. + 2. Create and sync a repo, publish it in two CVs, promote both CVs to both LCEs. + 3. Ensure full counts were calculated for both CVs, both LCEs. + 4. Create a filter for both CVs to change the counts of next version. + 5. Publish new version of the first CV, promote it to the first LCE. + 6. Ensure counts of the first CV were updated in the first LCE only, nothing else. + 7. Promote the first CV to the second LCE. + 8. Ensure counts for the first CV were updated in the second LCE, no changes for second CV. + + :expectedresults: + 1. Content counts are updated only for particular CVs promoted to particular LCEs. + """ + wait_query = ( + 'label = Actions::Katello::ContentView::CapsuleSync OR ' + 'Actions::Katello::CapsuleContent::UpdateContentCounts' + ) + + # Create two LCEs, assign them to the Capsule. + lce1 = module_target_sat.api.LifecycleEnvironment(organization=function_org).create() + lce2 = module_target_sat.api.LifecycleEnvironment( + organization=function_org, prior=lce1 + ).create() + module_capsule_configured.nailgun_capsule.content_add_lifecycle_environment( + data={'environment_id': [lce1.id, lce2.id]} + ) + + # Create and sync a repo, publish it in two CVs, promote both CVs to both LCEs. + repo = module_target_sat.api.Repository( + product=function_product, + ).create() + repo.sync() + repo = repo.read() + cvs = [] + for _ in range(2): + cv = module_target_sat.api.ContentView( + organization=function_org, repository=[repo] + ).create() + cv.publish() + cv = cv.read() + cv.version[0].promote(data={'environment_ids': [lce1.id, lce2.id]}) + cvs.append(cv.read()) + cv1, cv2 = cvs + module_target_sat.wait_for_tasks(search_query=wait_query, search_rate=5, max_tries=5) + + # Ensure full counts were calculated for both CVs, both LCEs. + info = module_target_sat.cli.Capsule.content_info( + {'id': module_capsule_configured.nailgun_capsule.id, 'organization-id': function_org.id} + ) + info = _dictionarize_counts(info) + + for lce in [lce1, lce2]: + for cv in cvs: + assert int(info[lce.name][cv.name][repo.name]['packages']) == repo.content_counts['rpm'] + assert ( + int(info[lce.name][cv.name][repo.name]['package-groups']) + == repo.content_counts['package_group'] + ) + assert ( + int(info[lce.name][cv.name][repo.name]['errata']) == repo.content_counts['erratum'] + ) + + # Create a filter for both CVs to change the counts of next version. + for cv in cvs: + cvf = module_target_sat.cli_factory.make_content_view_filter( + { + 'content-view-id': cv.id, + 'inclusion': 'true', + 'type': 'rpm', + }, + ) + module_target_sat.cli_factory.content_view_filter_rule( + { + 'content-view-filter-id': cvf['filter-id'], + 'name': FAKE_0_CUSTOM_PACKAGE_NAME, + } + ) + + # Publish new version of the first CV, promote it to the first LCE. + cv1.publish() + cv1 = cv1.read() + cv1.version[0].promote(data={'environment_ids': lce1.id}) + cv1 = cv1.read() + module_target_sat.wait_for_tasks(search_query=wait_query, search_rate=5, max_tries=5) + + # Ensure counts of the first CV were updated in the first LCE only, nothing else. + info = module_target_sat.cli.Capsule.content_info( + {'id': module_capsule_configured.nailgun_capsule.id, 'organization-id': function_org.id} + ) + info = _dictionarize_counts(info) + + assert int(info[lce1.name][cv1.name][repo.name]['packages']) == 1 + assert int(info[lce1.name][cv1.name][repo.name]['package-groups']) == 1 + assert int(info[lce1.name][cv1.name][repo.name]['errata']) == 1 + + for lce in [lce1, lce2]: + for cv in cvs: + if cv.name == cv1.name and lce.name == lce1.name: + continue + assert int(info[lce.name][cv.name][repo.name]['packages']) == repo.content_counts['rpm'] + assert ( + int(info[lce.name][cv.name][repo.name]['package-groups']) + == repo.content_counts['package_group'] + ) + assert ( + int(info[lce.name][cv.name][repo.name]['errata']) == repo.content_counts['erratum'] + ) + + # Promote the first CV to the second LCE. + cv1.version[0].promote(data={'environment_ids': lce2.id}) + cv1 = cv1.read() + module_target_sat.wait_for_tasks(search_query=wait_query, search_rate=5, max_tries=5) + + # Ensure counts for the first CV were updated in the second LCE, no changes for second CV. + info = module_target_sat.cli.Capsule.content_info( + {'id': module_capsule_configured.nailgun_capsule.id, 'organization-id': function_org.id} + ) + info = _dictionarize_counts(info) + + assert int(info[lce2.name][cv1.name][repo.name]['packages']) == 1 + assert int(info[lce2.name][cv1.name][repo.name]['package-groups']) == 1 + assert int(info[lce2.name][cv1.name][repo.name]['errata']) == 1 + + for lce in [lce1, lce2]: + assert int(info[lce.name][cv2.name][repo.name]['packages']) == repo.content_counts['rpm'] + assert ( + int(info[lce.name][cv2.name][repo.name]['package-groups']) + == repo.content_counts['package_group'] + ) + assert int(info[lce.name][cv2.name][repo.name]['errata']) == repo.content_counts['erratum'] + + @pytest.mark.tier4 @pytest.mark.skip_if_not_set('capsule') def test_positive_exported_imported_content_sync( From 557dd89ad6a3c8830edc0dd76edae368425f8e96 Mon Sep 17 00:00:00 2001 From: Vladimir Sedmik Date: Fri, 18 Oct 2024 10:43:59 +0200 Subject: [PATCH 2/2] Extend the test case by real changes in publications at Caps side --- tests/foreman/cli/test_capsulecontent.py | 94 ++++++++++++++++-------- 1 file changed, 64 insertions(+), 30 deletions(-) diff --git a/tests/foreman/cli/test_capsulecontent.py b/tests/foreman/cli/test_capsulecontent.py index c676bf694e0..caafc20ac55 100644 --- a/tests/foreman/cli/test_capsulecontent.py +++ b/tests/foreman/cli/test_capsulecontent.py @@ -352,6 +352,8 @@ def test_positive_content_counts_granularity( :parametrized: yes + :verifies: SAT-28337 + :setup: 1. Satellite with registered external Capsule. @@ -359,18 +361,24 @@ def test_positive_content_counts_granularity( 1. Create two LCEs, assign them to the Capsule. 2. Create and sync a repo, publish it in two CVs, promote both CVs to both LCEs. 3. Ensure full counts were calculated for both CVs, both LCEs. - 4. Create a filter for both CVs to change the counts of next version. - 5. Publish new version of the first CV, promote it to the first LCE. - 6. Ensure counts of the first CV were updated in the first LCE only, nothing else. - 7. Promote the first CV to the second LCE. - 8. Ensure counts for the first CV were updated in the second LCE, no changes for second CV. + 4. Disable automatic content counts update. + 5. Create a filter for both CVs to change the counts of next version. + 6. Publish new version of both CVs and promote to both LCEs. + After capsule sync completes we should have changes in all publictions. + 7. Ensure the counts were invalidated in both CVs, both LCEs. + 8. Enable automatic content counts update. + 9. Publish new version of the first CV, promote it to the first LCE. + 10. Ensure counts of the first CV were updated in the first LCE only, nothing else. + 11. Promote the first CV to the second LCE. + 12. Ensure counts for the first CV were updated in the second LCE, no changes for second CV. :expectedresults: 1. Content counts are updated only for particular CVs promoted to particular LCEs. """ wait_query = ( 'label = Actions::Katello::ContentView::CapsuleSync OR ' - 'Actions::Katello::CapsuleContent::UpdateContentCounts' + 'Actions::Katello::CapsuleContent::UpdateContentCounts ' + 'AND started_at >= "{}"' ) # Create two LCEs, assign them to the Capsule. @@ -398,7 +406,11 @@ def test_positive_content_counts_granularity( cv.version[0].promote(data={'environment_ids': [lce1.id, lce2.id]}) cvs.append(cv.read()) cv1, cv2 = cvs - module_target_sat.wait_for_tasks(search_query=wait_query, search_rate=5, max_tries=5) + module_target_sat.wait_for_tasks( + search_query=wait_query.format(datetime.utcnow().replace(microsecond=0)), + search_rate=5, + max_tries=5, + ) # Ensure full counts were calculated for both CVs, both LCEs. info = module_target_sat.cli.Capsule.content_info( @@ -417,6 +429,10 @@ def test_positive_content_counts_granularity( int(info[lce.name][cv.name][repo.name]['errata']) == repo.content_counts['erratum'] ) + # Disable automatic content counts update. + setting_update.value = False + setting_update = setting_update.update({'value'}) + # Create a filter for both CVs to change the counts of next version. for cv in cvs: cvf = module_target_sat.cli_factory.make_content_view_filter( @@ -433,12 +449,40 @@ def test_positive_content_counts_granularity( } ) + # Publish new version of both CVs and promote to both LCEs. + # After capsule sync completes we should have changes in all publictions. + for cv in cvs: + cv.publish() + cv = cv.read() + cv.version[0].promote(data={'environment_ids': [lce1.id, lce2.id]}) + cv = cv.read() + module_target_sat.wait_for_tasks( + search_query=wait_query.format(datetime.utcnow().replace(microsecond=0)), + search_rate=5, + max_tries=5, + ) + + # Ensure the counts were invalidated in both CVs, both LCEs. + info = module_target_sat.cli.Capsule.content_info( + {'id': module_capsule_configured.nailgun_capsule.id, 'organization-id': function_org.id} + ) + info = _dictionarize_counts(info) + assert all(info[lce.name][cv.name][repo.name] == {} for cv in cvs for lce in [lce1, lce2]) + + # Enable automatic content counts update. + setting_update.value = True + setting_update = setting_update.update({'value'}) + # Publish new version of the first CV, promote it to the first LCE. cv1.publish() cv1 = cv1.read() cv1.version[0].promote(data={'environment_ids': lce1.id}) cv1 = cv1.read() - module_target_sat.wait_for_tasks(search_query=wait_query, search_rate=5, max_tries=5) + module_target_sat.wait_for_tasks( + search_query=wait_query.format(datetime.utcnow().replace(microsecond=0)), + search_rate=5, + max_tries=5, + ) # Ensure counts of the first CV were updated in the first LCE only, nothing else. info = module_target_sat.cli.Capsule.content_info( @@ -449,24 +493,21 @@ def test_positive_content_counts_granularity( assert int(info[lce1.name][cv1.name][repo.name]['packages']) == 1 assert int(info[lce1.name][cv1.name][repo.name]['package-groups']) == 1 assert int(info[lce1.name][cv1.name][repo.name]['errata']) == 1 - - for lce in [lce1, lce2]: - for cv in cvs: - if cv.name == cv1.name and lce.name == lce1.name: - continue - assert int(info[lce.name][cv.name][repo.name]['packages']) == repo.content_counts['rpm'] - assert ( - int(info[lce.name][cv.name][repo.name]['package-groups']) - == repo.content_counts['package_group'] - ) - assert ( - int(info[lce.name][cv.name][repo.name]['errata']) == repo.content_counts['erratum'] - ) + assert all( + info[lce.name][cv.name][repo.name] == {} + for cv in cvs + for lce in [lce1, lce2] + if (cv.name != cv1.name or lce.name != lce1.name) + ) # Promote the first CV to the second LCE. cv1.version[0].promote(data={'environment_ids': lce2.id}) cv1 = cv1.read() - module_target_sat.wait_for_tasks(search_query=wait_query, search_rate=5, max_tries=5) + module_target_sat.wait_for_tasks( + search_query=wait_query.format(datetime.utcnow().replace(microsecond=0)), + search_rate=5, + max_tries=5, + ) # Ensure counts for the first CV were updated in the second LCE, no changes for second CV. info = module_target_sat.cli.Capsule.content_info( @@ -477,14 +518,7 @@ def test_positive_content_counts_granularity( assert int(info[lce2.name][cv1.name][repo.name]['packages']) == 1 assert int(info[lce2.name][cv1.name][repo.name]['package-groups']) == 1 assert int(info[lce2.name][cv1.name][repo.name]['errata']) == 1 - - for lce in [lce1, lce2]: - assert int(info[lce.name][cv2.name][repo.name]['packages']) == repo.content_counts['rpm'] - assert ( - int(info[lce.name][cv2.name][repo.name]['package-groups']) - == repo.content_counts['package_group'] - ) - assert int(info[lce.name][cv2.name][repo.name]['errata']) == repo.content_counts['erratum'] + assert all(info[lce.name][cv2.name][repo.name] == {} for lce in [lce1, lce2]) @pytest.mark.tier4