# -*- coding: utf-8 -*- # SPDX-License-Identifier: MIT from __future__ import absolute_import from mock import call, patch, PropertyMock, Mock import pytest from sqlalchemy import func from module_build_service.common.config import conf import module_build_service.common.config from module_build_service.common.models import BUILD_STATES, ModuleBuild from module_build_service.scheduler.consumer import MBSConsumer from module_build_service.scheduler.db_session import db_session from module_build_service.scheduler.handlers.greenwave import ( decision_update, get_corresponding_module_build ) from tests import make_module_in_db @pytest.mark.usefixtures("require_empty_database") class TestGetCorrespondingModuleBuild: """Test get_corresponding_module_build""" @patch("koji.ClientSession") def test_module_build_nvr_does_not_exist_in_koji(self, ClientSession): ClientSession.return_value.getBuild.return_value = None assert get_corresponding_module_build("n-v-r") is None @pytest.mark.parametrize( "build_info", [ # Build info does not have key extra {"id": 1000, "name": "ed"}, # Build info contains key extra, but it is not for the module build {"extra": {"submitter": "osbs", "image": {}}}, # Key module_build_service_id is missing {"extra": {"typeinfo": {"module": {}}}}, ], ) @patch("koji.ClientSession") def test_cannot_find_module_build_id_from_build_info(self, ClientSession, build_info): ClientSession.return_value.getBuild.return_value = build_info assert get_corresponding_module_build("n-v-r") is None @patch("koji.ClientSession") def test_corresponding_module_build_id_does_not_exist_in_db(self, ClientSession, require_platform_and_default_arch): fake_module_build_id, = db_session.query(func.max(ModuleBuild.id)).first() ClientSession.return_value.getBuild.return_value = { "extra": {"typeinfo": {"module": {"module_build_service_id": fake_module_build_id + 1}}} } assert get_corresponding_module_build("n-v-r") is None @patch("koji.ClientSession") def test_find_the_module_build(self, ClientSession, require_platform_and_default_arch): expected_module_build = ( db_session.query(ModuleBuild).filter(ModuleBuild.name == "platform").first() ) ClientSession.return_value.getBuild.return_value = { "extra": {"typeinfo": {"module": {"module_build_service_id": expected_module_build.id}}} } build = get_corresponding_module_build("n-v-r") assert expected_module_build.id == build.id assert expected_module_build.name == build.name class TestDecisionUpdateHandler: """Test handler decision_update""" def setup_method(self, test_method): self.patch_config_broker = patch.object( module_build_service.common.config.Config, "celery_broker_url", create=True, new_callable=PropertyMock, return_value=False, ) self.patch_config_broker.start() def teardown_method(self, test_method): self.patch_config_broker.stop() @patch("module_build_service.scheduler.handlers.greenwave.log") def test_decision_context_is_not_match(self, log): decision_update( msg_id="msg-id-1", decision_context="bodhi_update_push_testing", policies_satisfied=True, subject_identifier="xxx", ) log.debug.assert_called_once_with( 'Skip Greenwave message %s as MBS only handles messages with the decision context "%s"', "msg-id-1", "test_dec_context" ) @patch("module_build_service.scheduler.handlers.greenwave.log") def test_not_satisfy_policies(self, log): subject_identifier = "pkg-0.1-1.c1" decision_update( msg_id="msg-id-1", decision_context="test_dec_context", policies_satisfied=False, subject_identifier=subject_identifier) log.debug.assert_called_once_with( "Skip to handle module build %s because it has not satisfied Greenwave policies.", subject_identifier, ) @patch("module_build_service.common.messaging.publish") @patch("koji.ClientSession") def test_transform_from_done_to_ready(self, ClientSession, publish, require_empty_database): # This build should be queried and transformed to ready state module_build = make_module_in_db( "pkg:0.1:1:c1", [ { "requires": {"platform": ["el8"]}, "buildrequires": {"platform": ["el8"]}, } ], ) module_build.transition( db_session, conf, BUILD_STATES["done"], "Move to done directly for running test." ) db_session.commit() # Assert this call below first_publish_call = call( "module.state.change", module_build.json(db_session, show_tasks=False), conf, "mbs", ) ClientSession.return_value.getBuild.return_value = { "extra": {"typeinfo": {"module": {"module_build_service_id": module_build.id}}} } msg = { "msg_id": "msg-id-1", "topic": "org.fedoraproject.prod.greenwave.decision.update", "msg": { "decision_context": "test_dec_context", "policies_satisfied": True, "subject_identifier": "pkg-0.1-1.c1", }, } hub = Mock(config={"validate_signatures": False}) consumer = MBSConsumer(hub) consumer.consume(msg) db_session.add(module_build) # Load module build again to check its state is moved correctly db_session.refresh(module_build) assert BUILD_STATES["ready"] == module_build.state publish.assert_has_calls([ first_publish_call, call( "module.state.change", module_build.json(db_session, show_tasks=False), conf, "mbs" ), ])