Skip to content

Commit

Permalink
Implement conditions on labels for PR events
Browse files Browse the repository at this point in the history
For PR events, check whether the labels on PR match the labels
configuration (require.label.present and require.label.absent)
if present when getting matching jobs for events.
Fixes #2269
  • Loading branch information
lbarcziova committed Feb 6, 2024
1 parent 3e6737a commit 1aabf24
Show file tree
Hide file tree
Showing 6 changed files with 159 additions and 15 deletions.
33 changes: 32 additions & 1 deletion packit_service/utils.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,8 +6,9 @@
from io import StringIO
from logging import StreamHandler
from re import search
from typing import List, Tuple
from typing import List, Tuple, Optional

from ogr.abstract import PullRequest
from packit.config import JobConfig, PackageConfig
from packit.schema import JobConfigSchema, PackageConfigSchema
from packit.utils import PackitFormatter
Expand Down Expand Up @@ -222,3 +223,33 @@ def get_koji_task_id_and_url_from_stdout(stdout: str):
task_url = task_url_match.group(0)

return task_id, task_url


def pr_labels_match_configuration(
pull_request: Optional[PullRequest],
configured_labels_present: list[str],
configured_labels_absent: list[str],
) -> bool:
"""
Do the PR labels match the configuration of the labels?
"""
if not pull_request:
logger.debug("No PR to check the labels on.")
return True

logger.info(
f"About to check whether PR labels in PR {pull_request.id} "
f"match to the labels configuration "
f"(label.present: {configured_labels_present}, label.absent: {configured_labels_absent})"
)

pr_labels = pull_request.labels
logger.info(f"Labels on PR: {pr_labels}")

return (
not configured_labels_present
or any(label in pr_labels for label in configured_labels_present)
) and (
not configured_labels_absent
or all(label not in pr_labels for label in configured_labels_absent)
)
11 changes: 1 addition & 10 deletions packit_service/worker/events/comment.py
Original file line number Diff line number Diff line change
Expand Up @@ -7,8 +7,7 @@
from logging import getLogger
from typing import Dict, Optional, Set

from ogr.abstract import Comment, PullRequest, Issue

from ogr.abstract import Comment, Issue
from packit_service.models import TestingFarmResult, BuildStatus
from packit_service.service.db_project_events import (
AddIssueEventToDb,
Expand Down Expand Up @@ -70,7 +69,6 @@ def __init__(
self._comment_object = comment_object
self._build_targets_override = build_targets_override
self._tests_targets_override = tests_targets_override
self._pull_request_object = None

@property
def commit_sha(self) -> str: # type:ignore
Expand All @@ -79,12 +77,6 @@ def commit_sha(self) -> str: # type:ignore
self._commit_sha = self.project.get_pr(pr_id=self.pr_id).head_commit
return self._commit_sha

@property
def pull_request_object(self) -> PullRequest:
if not self._pull_request_object:
self._pull_request_object = self.project.get_pr(self.pr_id)
return self._pull_request_object

@property
def comment_object(self) -> Optional[Comment]:
if not self._comment_object:
Expand Down Expand Up @@ -121,7 +113,6 @@ def get_dict(self, default_dict: Optional[Dict] = None) -> dict:
result["commit_sha"] = self.commit_sha
result.pop("_build_targets_override")
result.pop("_tests_targets_override")
result.pop("_pull_request_object")
return result


Expand Down
14 changes: 13 additions & 1 deletion packit_service/worker/events/event.py
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@
from logging import getLogger
from typing import Dict, Optional, Type, Union, Set, List

from ogr.abstract import GitProject
from ogr.abstract import GitProject, PullRequest
from ogr.parsing import RepoUrl

from packit.config import JobConfigTriggerType, PackageConfig
Expand Down Expand Up @@ -461,6 +461,7 @@ def __init__(
self._pr_id = pr_id
self.fail_when_config_file_missing = False
self.actor = actor
self._pull_request_object = None

@property
def project(self):
Expand Down Expand Up @@ -491,6 +492,12 @@ def get_db_project_event(self) -> Optional[ProjectEventModel]:
def pr_id(self) -> Optional[int]:
return self._pr_id

@property
def pull_request_object(self) -> Optional[PullRequest]:
if not self._pull_request_object and self.pr_id:
self._pull_request_object = self.project.get_pr(self.pr_id)
return self._pull_request_object

def get_project(self) -> Optional[GitProject]:
if not (self.project_url or self.db_project_object):
return None
Expand Down Expand Up @@ -552,6 +559,11 @@ def get_all_build_targets_by_status(
statuses_to_filter_with=statuses_to_filter_with,
)

def get_dict(self, default_dict: Optional[Dict] = None) -> dict:
result = super().get_dict()
result.pop("_pull_request_object")
return result


class AbstractResultEvent(AbstractForgeIndependentEvent):
"""
Expand Down
21 changes: 19 additions & 2 deletions packit_service/worker/jobs.py
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,11 @@
COMMENT_REACTION,
PACKIT_VERIFY_FAS_COMMAND,
)
from packit_service.utils import get_packit_commands_from_comment, elapsed_seconds
from packit_service.utils import (
get_packit_commands_from_comment,
elapsed_seconds,
pr_labels_match_configuration,
)
from packit_service.worker.allowlist import Allowlist
from packit_service.worker.events import (
Event,
Expand All @@ -39,7 +43,10 @@
AbstractIssueCommentEvent,
AbstractPRCommentEvent,
)
from packit_service.worker.events.event import AbstractResultEvent
from packit_service.worker.events.event import (
AbstractResultEvent,
AbstractForgeIndependentEvent,
)
from packit_service.worker.handlers import (
CoprBuildHandler,
GithubAppInstallationHandler,
Expand Down Expand Up @@ -666,6 +673,16 @@ def get_jobs_matching_event(self) -> List[JobConfig]:
for event_type in MANUAL_OR_RESULT_EVENTS
)
)
and (
job.trigger != JobConfigTriggerType.pull_request
or not (job.require.label.present or job.require.label.absent)
or not isinstance(self.event, AbstractForgeIndependentEvent)
or pr_labels_match_configuration(
pull_request=self.event.pull_request_object,
configured_labels_absent=job.require.label.absent,
configured_labels_present=job.require.label.present,
)
)
):
jobs_matching_trigger.append(job)

Expand Down
7 changes: 7 additions & 0 deletions tests/unit/test_babysit_vm_image.py
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@
from flexmock import flexmock
from flexmock import Mock
from packit.config.job_config import JobConfigTriggerType, JobType
from packit.config.requirements import RequirementsConfig, LabelRequirementsConfig
from packit_service.config import ServiceConfig
from packit_service.models import (
VMImageBuildTargetModel,
Expand Down Expand Up @@ -117,6 +118,12 @@ def test_update_vm_image_build(stop_babysitting, build_status, vm_image_builder_
trigger=JobConfigTriggerType.pull_request,
type=JobType.vm_image_build,
manual_trigger=False,
require=RequirementsConfig(
LabelRequirementsConfig(
absent=[],
present=[],
)
),
)
],
)
Expand Down
88 changes: 87 additions & 1 deletion tests/unit/test_utils.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,9 @@
# Copyright Contributors to the Packit project.
# SPDX-License-Identifier: MIT
import pytest
from flexmock import flexmock

from packit_service.utils import only_once
from packit_service.utils import only_once, pr_labels_match_configuration


def test_only_once():
Expand Down Expand Up @@ -74,3 +76,87 @@ def f(one, two, three="something"):
assert counter == 1
f("b", "b", three="different")
assert counter == 1


@pytest.mark.parametrize(
"absent,present,pr_labels,should_pass",
[
pytest.param(
[],
["my-label"],
[],
False,
),
pytest.param(
[],
["my-label"],
["my-label"],
True,
),
pytest.param(
["skip-ci"],
["my-label"],
["my-label"],
True,
),
pytest.param(
["skip-ci"],
["my-label"],
["my-label", "skip-ci"],
False,
),
pytest.param(
["skip-ci"],
["my-label"],
["skip-ci"],
False,
),
pytest.param(
["skip-ci"],
[],
["skip-ci"],
False,
),
pytest.param(
["skip-ci"],
[],
[],
True,
),
pytest.param(
["skip-ci"],
["first", "second"],
["second"],
True,
),
pytest.param(
["skip-ci"],
["first", "second"],
["third"],
False,
),
pytest.param(
["skip-ci", "block-ci"],
["first", "second"],
["block-ci"],
False,
),
pytest.param(
[],
[],
[],
True,
),
pytest.param(
[],
[],
["some-label"],
True,
),
],
)
def test_pr_labels_match(absent, present, pr_labels, should_pass):
assert (
pr_labels_match_configuration(flexmock(labels=pr_labels, id=1), present, absent)
== should_pass
)

0 comments on commit 1aabf24

Please sign in to comment.