agnes-the-ai-analyst/tests/test_keboola_extractor_exit_codes.py
ZdenekSrotyr ef74ec010c
fix(ops): #81 Group B — Keboola partial-failure exit code 2 (squashed) (#99)
Closes M14 from issue #81. Keboola extractor exits 0/1/2
(success/full-fail/partial). sync.py interprets exit 2 as
PARTIAL FAILURE (data-quality alert, distinct from exit 1).

Tests: tests/test_keboola_extractor_exit_codes.py — 14 cases including
runtime mock subprocess (rc=0/1/2/124).

Refs #81 Group B.
2026-04-27 21:52:46 +02:00

159 lines
7 KiB
Python

"""Issue #81 Group B — Keboola extractor exit codes.
Three contracts:
- 0 = full success (every table OK)
- 1 = full failure (every table failed)
- 2 = partial (at least one OK + at least one failed)
Plus the sync.py interpretation: exit 2 must NOT be treated as a crash;
it logs a PARTIAL FAILURE notice and continues to the orchestrator
rebuild step (the orchestrator's per-table _meta machinery already
captures which tables succeeded).
"""
import subprocess as subprocess_real
import pytest
from connectors.keboola.extractor import compute_exit_code
class TestComputeExitCode:
@pytest.mark.parametrize(
"stats,total,expected",
[
# Full success
({"tables_extracted": 10, "tables_failed": 0}, 10, 0),
# Single-table full success
({"tables_extracted": 1, "tables_failed": 0}, 1, 0),
# No tables registered → 0 (vacuous success)
({"tables_extracted": 0, "tables_failed": 0}, 0, 0),
# Full failure
({"tables_extracted": 0, "tables_failed": 10}, 10, 1),
# Single-table full failure
({"tables_extracted": 0, "tables_failed": 1}, 1, 1),
# Partial — single failure in 10
({"tables_extracted": 9, "tables_failed": 1}, 10, 2),
# Partial — half-and-half
({"tables_extracted": 5, "tables_failed": 5}, 10, 2),
# Partial — only one succeeded
({"tables_extracted": 1, "tables_failed": 9}, 10, 2),
],
)
def test_exit_code_matrix(self, stats, total, expected):
assert compute_exit_code(stats, total) == expected
def test_missing_tables_failed_key_treated_as_zero(self):
"""Defensive — older stats dicts without `tables_failed` should
be treated as full success."""
assert compute_exit_code({"tables_extracted": 5}, 5) == 0
def test_failed_exceeds_total_still_full_failure(self):
"""If somehow `tables_failed > total` (counting bug, retries),
exit 1 — not 2 — so partial-failure alerting only fires on a
legitimate mixed outcome."""
assert compute_exit_code({"tables_failed": 11}, 10) == 1
class TestSyncApiPartialFailureHandling:
"""Runtime test: exit code from the extractor subprocess maps to the
correct [SYNC] log branch. Drives `_run_sync` with a mocked
`subprocess.run` and asserts the print() calls into stderr. This
catches inverted-comparison regressions (e.g. `if returncode == 1`
used for the partial branch) that a source-substring grep would
miss.
"""
def _drive_run_sync(self, monkeypatch, capsys, returncode):
"""Invoke `_run_sync` with the extractor subprocess returning
``returncode``, return the captured stderr as a single string.
sync.py does several `import` inside `_run_sync` (subprocess,
SyncOrchestrator, get_system_db). Stubs must target either the
global module (so the local import-from-cache picks them up)
or the runtime call sites via ``patch.object`` on the imported
names after the function has resolved them.
"""
from unittest.mock import MagicMock, patch
from app.api import sync as sync_mod
def fake_run(*args, **kwargs):
return MagicMock(
returncode=returncode, stdout="{}", stderr="",
)
# subprocess is imported locally inside _run_sync; patching the
# real module's run() works because Python's module cache means
# both call sites resolve to the same object.
monkeypatch.setattr(subprocess_real, "run", fake_run)
# SyncOrchestrator is imported as `from src.orchestrator import
# SyncOrchestrator` inside _run_sync, so patching sync_mod
# doesn't reach it. Patch the source module instead.
from src import orchestrator as orch_mod
monkeypatch.setattr(
orch_mod, "SyncOrchestrator",
lambda *a, **kw: MagicMock(rebuild=MagicMock(return_value={})),
raising=False,
)
# Pretend a Keboola token is configured so the inline subprocess
# cmd is built (don't enter the missing-credentials early-exit).
monkeypatch.setenv("KEBOOLA_STORAGE_TOKEN", "test-token")
monkeypatch.setenv("KEBOOLA_STACK_URL", "https://test.example")
# _run_sync calls TableRegistryRepository.list_local on a real
# system DB connection. Stub the registry method directly so we
# don't need a populated DB; also stub get_system_db /
# get_data_source_type to avoid filesystem-dependency on a
# configured instance.yaml in CI.
from src.repositories.table_registry import TableRegistryRepository
monkeypatch.setattr(
TableRegistryRepository, "list_local",
lambda self, *a, **kw: [
{"id": "x", "name": "x", "source_type": "keboola",
"bucket": "in.c-x", "source_table": "y",
"query_mode": "local"}
],
)
# Stub system DB + data-source-type. sync.py does
# `from src.db import get_system_db` and
# `from app.instance_config import get_data_source_type`
# **inside** _run_sync (not module top-level), so we must patch
# on the SOURCE modules — patching sync_mod is silently
# ineffective because the local imports re-bind the names.
fake_conn = MagicMock()
fake_conn.close = MagicMock()
from src import db as db_mod
from app import instance_config as ic_mod
monkeypatch.setattr(db_mod, "get_system_db", lambda: fake_conn)
monkeypatch.setattr(ic_mod, "get_data_source_type", lambda: "keboola")
monkeypatch.setattr(ic_mod, "get_value", lambda *a, **kw: "")
sync_mod._run_sync()
return capsys.readouterr().err
def test_exit_0_is_logged_as_ok(self, monkeypatch, capsys):
stderr = self._drive_run_sync(monkeypatch, capsys, returncode=0)
assert "[SYNC] Extractor OK" in stderr
assert "PARTIAL FAILURE" not in stderr
assert "Extractor FAILED" not in stderr
def test_exit_1_is_logged_as_failed(self, monkeypatch, capsys):
stderr = self._drive_run_sync(monkeypatch, capsys, returncode=1)
assert "[SYNC] Extractor FAILED (exit 1)" in stderr
assert "PARTIAL FAILURE" not in stderr
assert "Extractor OK" not in stderr
def test_exit_2_is_logged_as_partial(self, monkeypatch, capsys):
stderr = self._drive_run_sync(monkeypatch, capsys, returncode=2)
assert "[SYNC] Extractor PARTIAL FAILURE (exit 2)" in stderr
# The partial branch must NOT also log OK or FAILED.
assert "Extractor OK" not in stderr
assert "Extractor FAILED (exit" not in stderr
def test_exit_124_falls_through_to_failed(self, monkeypatch, capsys):
"""Timeouts (124), signal kills (-N), and other non-zero codes
all hit the catchall else branch and log FAILED."""
stderr = self._drive_run_sync(monkeypatch, capsys, returncode=124)
assert "[SYNC] Extractor FAILED (exit 124)" in stderr