From c514cb4ad0ef1baea78bf6d618cdad591673f343 Mon Sep 17 00:00:00 2001 From: Gerard Segarra Date: Mon, 29 Apr 2024 11:58:45 +0200 Subject: [PATCH 1/4] Add job handler to process jobs --- .github/workflows/tests.yaml | 2 +- src/app.py | 7 ++++- src/jobs.py | 57 ++++++++++++++++++++++++++++++++++++ tests/test_jobs.py | 56 +++++++++++++++++++++++++++++++++++ 4 files changed, 120 insertions(+), 2 deletions(-) create mode 100644 src/jobs.py create mode 100644 tests/test_jobs.py diff --git a/.github/workflows/tests.yaml b/.github/workflows/tests.yaml index 98103c8..ba9d63d 100644 --- a/.github/workflows/tests.yaml +++ b/.github/workflows/tests.yaml @@ -29,4 +29,4 @@ jobs: flake8 - name: Test with pytest run: | - pytest --cov=src + pytest --cov=src tests/ diff --git a/src/app.py b/src/app.py index d170aa0..8aac49e 100644 --- a/src/app.py +++ b/src/app.py @@ -9,6 +9,7 @@ from const import GithubHeaders, LOGGING_CONFIG from github import GithubJob +from jobs import JobEventsHandler from utils import dict_to_logfmt dictConfig(LOGGING_CONFIG) @@ -26,6 +27,7 @@ logging.getLogger('apscheduler.executors.default').setLevel(logging.WARNING) jobs = dict() +job_handler = JobEventsHandler() # check all calls are valid @@ -50,7 +52,10 @@ def validate_origin_github(): def process_workflow_job(): - job = GithubJob(request.get_json()) + event = request.get_json() + job_handler.process_event(event) + + job = GithubJob(event) context_details = { "action": job.action, diff --git a/src/jobs.py b/src/jobs.py new file mode 100644 index 0000000..4e31e9b --- /dev/null +++ b/src/jobs.py @@ -0,0 +1,57 @@ +from github import GithubJob + + +class Job: + def __init__(self, github_job: GithubJob) -> None: + self.github_job = github_job + + +class JobEventsHandler: + def __init__(self) -> None: + self.queued = dict() + self.in_progress = dict() + + def process_event(self, event: dict): + status = event["action"] + + if status == "queued": + self._process_queued_event(event) + + elif status == "in_progress": + self._process_in_progress_event(event) + + elif status == "completed": + self._process_completed_event(event) + + else: + pass + + def _get_event_job_id(self, event: dict): + return event["workflow_job"]["id"] + + def _create_job(self, githubJob: GithubJob) -> Job: + return Job(github_job=githubJob) + + def _process_queued_event(self, event: dict): + job = self._create_job(GithubJob(event)) + self.queued[self._get_event_job_id(event)] = job + + def _process_in_progress_event(self, event: dict): + job_id = self._get_event_job_id(event) + job = self.queued.pop(job_id, None) + + if not job: + job = self._create_job(GithubJob(event)) + else: + # Update github job event from job + job.github_job = GithubJob(event) + + self.in_progress[job_id] = job + + # TODO send final time in queue + + def _process_completed_event(self, event: dict): + job_id = self._get_event_job_id(event) + self.in_progress.pop(job_id, None) + + # TODO send final time in progress diff --git a/tests/test_jobs.py b/tests/test_jobs.py new file mode 100644 index 0000000..90632a3 --- /dev/null +++ b/tests/test_jobs.py @@ -0,0 +1,56 @@ +import pytest + +from unittest.mock import Mock + +from jobs import JobEventsHandler + + +@pytest.fixture +def new_job_event(): + return {"workflow_job": {"id": "workflow_id"}, "action": "queued"} + + +@pytest.fixture +def in_progress_job_event(): + return {"workflow_job": {"id": "workflow_id"}, "action": "in_progress"} + + +@pytest.fixture +def completed_job_event(): + return {"workflow_job": {"id": "workflow_id"}, "action": "completed"} + + +def test_new_job(new_job_event): + handler = JobEventsHandler() + + handler.process_event(new_job_event) + + assert handler.queued.get("workflow_id") + + +def test_in_progress_job(in_progress_job_event): + handler = JobEventsHandler() + job = Mock() + handler.queued["workflow_id"] = job + + handler.process_event(in_progress_job_event) + + assert not handler.queued.get("workflow_id") + assert handler.in_progress.get("workflow_id") == job + + +def test_unprocessed_in_progress_job(in_progress_job_event): + handler = JobEventsHandler() + handler.process_event(in_progress_job_event) + + assert handler.in_progress.get("workflow_id") + + +def test_completed_job(completed_job_event): + handler = JobEventsHandler() + handler.in_progress["workflow_id"] = Mock() + + handler.process_event(completed_job_event) + + assert not handler.queued.get("workflow_id") + assert not handler.in_progress.get("workflow_id") From 671d66237ba8394338ed1c34fd8de19fd2396118 Mon Sep 17 00:00:00 2001 From: Gerard Segarra Date: Mon, 29 Apr 2024 11:59:12 +0200 Subject: [PATCH 2/4] Install black --- src/app.py | 12 ++++-------- src/github.py | 2 +- tests/test_jobs.py | 4 ++-- tests/tests.py | 4 ++-- 4 files changed, 9 insertions(+), 13 deletions(-) diff --git a/src/app.py b/src/app.py index 8aac49e..781f5b1 100644 --- a/src/app.py +++ b/src/app.py @@ -24,7 +24,7 @@ if hasattr(logging, loglevel_flask): loglevel_flask = getattr(logging, loglevel_flask) log.setLevel(loglevel_flask) -logging.getLogger('apscheduler.executors.default').setLevel(logging.WARNING) +logging.getLogger("apscheduler.executors.default").setLevel(logging.WARNING) jobs = dict() job_handler = JobEventsHandler() @@ -82,9 +82,7 @@ def process_workflow_job(): app.logger.error(f"Job {job.id} was in progress before being queued") del jobs[job.id] else: - time_to_start = ( - job.time_start - job_requested.time_start - ).seconds + time_to_start = (job.time_start - job_requested.time_start).seconds context_details = { **context_details, @@ -105,9 +103,7 @@ def process_workflow_job(): app.logger.warning(f"Job {job.id} is {job.action} but not stored!") time_to_finish = 0 else: - time_to_finish = ( - job.time_completed - job.time_start - ).seconds + time_to_finish = (job.time_completed - job.time_start).seconds # delete from memory del jobs[job.id] @@ -129,7 +125,7 @@ def process_workflow_job(): return True -@scheduler.task('interval', id='monitor_queued', seconds=30) +@scheduler.task("interval", id="monitor_queued", seconds=30) def monitor_queued_jobs(): """Return the job that has been queued and not starting for long time.""" app.logger.debug("Starting monitor_queued_jobs") diff --git a/src/github.py b/src/github.py index 78282c9..9d14a55 100644 --- a/src/github.py +++ b/src/github.py @@ -1,7 +1,7 @@ from utils import parse_datetime -class GithubJob(): +class GithubJob: def __init__(self, json_body: str): self.data = json_body diff --git a/tests/test_jobs.py b/tests/test_jobs.py index 90632a3..aef7e38 100644 --- a/tests/test_jobs.py +++ b/tests/test_jobs.py @@ -12,12 +12,12 @@ def new_job_event(): @pytest.fixture def in_progress_job_event(): - return {"workflow_job": {"id": "workflow_id"}, "action": "in_progress"} + return {"workflow_job": {"id": "workflow_id"}, "action": "in_progress"} @pytest.fixture def completed_job_event(): - return {"workflow_job": {"id": "workflow_id"}, "action": "completed"} + return {"workflow_job": {"id": "workflow_id"}, "action": "completed"} def test_new_job(new_job_event): diff --git a/tests/tests.py b/tests/tests.py index ca21f15..9cf34fa 100644 --- a/tests/tests.py +++ b/tests/tests.py @@ -41,7 +41,7 @@ def test_method_not_allowed(client): def test_headers_not_correct(client, caplog): - response = client.post(HOOK_ENDPOINT, headers={'User-Agent': 'foo'}) + response = client.post(HOOK_ENDPOINT, headers={"User-Agent": "foo"}) assert response.status_code == 401 assert caplog.messages == [ "User-Agent is foo", @@ -197,5 +197,5 @@ def test_line_break_in_job_name(client, caplog): assert caplog.messages == [ 'action=queued repository=foo/foo branch=new-feature-branch job_id=6 run_id=10 job_name="Build and push ' 'images (actions-runner-dind, NPROC=2 , runner-images/devops/actions-runner-dind, l..."' - ' workflow=CI requestor=testerbot' + " workflow=CI requestor=testerbot" ] From 0e68e1e84a398364bd66812f4c5406a05cf60188 Mon Sep 17 00:00:00 2001 From: Gerard Segarra Date: Mon, 29 Apr 2024 12:01:30 +0200 Subject: [PATCH 3/4] Pin versions --- .pre-commit-config.yaml | 4 ++++ tests/requirements.txt | 10 +++++----- 2 files changed, 9 insertions(+), 5 deletions(-) diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index 0bae781..7d81f5d 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -25,3 +25,7 @@ repos: rev: 6.0.0 hooks: - id: flake8 +- repo: https://github.com/psf/black + rev: stable + hooks: + - id: black diff --git a/tests/requirements.txt b/tests/requirements.txt index 4a98b0b..3de5755 100644 --- a/tests/requirements.txt +++ b/tests/requirements.txt @@ -1,5 +1,5 @@ -Flask -Flask-APScheduler==1.13.1 -pytest -pytest-cov -flake8 +-e . +pytest==8.2.0 +pytest-cov==5.0.0 +flake8==7.0.0 +black==24.4.2 From 595ec09603f2b4deb23ab83bf7c1efe353dda29c Mon Sep 17 00:00:00 2001 From: Gerard Segarra Date: Mon, 29 Apr 2024 12:02:02 +0200 Subject: [PATCH 4/4] fixup! Pin versions --- .pre-commit-config.yaml | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index 7d81f5d..1e3367f 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -1,6 +1,6 @@ repos: - repo: https://github.com/pre-commit/pre-commit-hooks - rev: v4.4.0 + rev: v4.6.0 hooks: - id: trailing-whitespace - id: check-docstring-first @@ -12,20 +12,20 @@ repos: - id: debug-statements - id: end-of-file-fixer - repo: https://github.com/myint/docformatter - rev: v1.5.1 + rev: v1.7.5 hooks: - id: docformatter args: [--in-place] - repo: https://github.com/asottile/pyupgrade - rev: v3.3.1 + rev: v3.15.2 hooks: - id: pyupgrade args: [--py38-plus] - repo: https://github.com/PyCQA/flake8 - rev: 6.0.0 + rev: 7.0.0 hooks: - id: flake8 - repo: https://github.com/psf/black - rev: stable + rev: 24.4.2 hooks: - id: black