Some checks failed
Test / pytest (pull_request) Failing after 8s
Implements scripts/dependency_freshness.py which compares installed dependencies against latest PyPI versions and flags packages that are more than 2 major versions behind. Includes comprehensive tests in scripts/test_dependency_freshness.py. Closes #161
180 lines
6.4 KiB
Python
180 lines
6.4 KiB
Python
#!/usr/bin/env python3
|
|
"""Tests for scripts/dependency_freshness.py — 9.7 Dependency Freshness."""
|
|
|
|
import json
|
|
import os
|
|
import sys
|
|
from unittest.mock import patch, MagicMock
|
|
|
|
# Import target module
|
|
sys.path.insert(0, os.path.dirname(__file__) or ".")
|
|
import importlib.util
|
|
spec = importlib.util.spec_from_file_location(
|
|
"dependency_freshness",
|
|
os.path.join(os.path.dirname(__file__) or ".", "dependency_freshness.py")
|
|
)
|
|
mod = importlib.util.module_from_spec(spec)
|
|
spec.loader.exec_module(mod)
|
|
|
|
parse_requirements = mod.parse_requirements
|
|
get_major_version = mod.get_major_version
|
|
is_more_than_two_majors_behind = mod.is_more_than_two_majors_behind
|
|
analyze_dependencies = mod.analyze_dependencies
|
|
|
|
|
|
def test_parse_requirements_simple():
|
|
"""Parse a simple package line."""
|
|
import tempfile
|
|
with tempfile.NamedTemporaryFile(mode='w', suffix='.txt', delete=False) as f:
|
|
f.write("requests\n")
|
|
tmp = f.name
|
|
try:
|
|
pkgs = parse_requirements(tmp)
|
|
assert pkgs == ["requests"], f"got {pkgs}"
|
|
print("PASS: test_parse_requirements_simple")
|
|
finally:
|
|
os.unlink(tmp)
|
|
|
|
|
|
def test_parse_requirements_with_specifiers():
|
|
"""Parse lines with version specifiers."""
|
|
import tempfile
|
|
with tempfile.NamedTemporaryFile(mode='w', suffix='.txt', delete=False) as f:
|
|
f.write("pytest>=8,<9\n")
|
|
f.write("aiohttp>=3.8\n")
|
|
tmp = f.name
|
|
try:
|
|
pkgs = parse_requirements(tmp)
|
|
assert pkgs == ["pytest", "aiohttp"], f"got {pkgs}"
|
|
print("PASS: test_parse_requirements_with_specifiers")
|
|
finally:
|
|
os.unlink(tmp)
|
|
|
|
|
|
def test_parse_requirements_ignores_comments_and_blanks():
|
|
"""Comments and blank lines are skipped."""
|
|
import tempfile
|
|
with tempfile.NamedTemporaryFile(mode='w', suffix='.txt', delete=False) as f:
|
|
f.write("# This is a comment\n")
|
|
f.write("\n")
|
|
f.write(" \n")
|
|
f.write("numpy\n")
|
|
f.write("# another comment\n")
|
|
tmp = f.name
|
|
try:
|
|
pkgs = parse_requirements(tmp)
|
|
assert pkgs == ["numpy"], f"got {pkgs}"
|
|
print("PASS: test_parse_requirements_ignores_comments_and_blanks")
|
|
finally:
|
|
os.unlink(tmp)
|
|
|
|
|
|
def test_get_major_version_normal():
|
|
"""Extract major version from typical semantic strings."""
|
|
assert get_major_version("1.2.3") == 1
|
|
assert get_major_version("3.4.5") == 3
|
|
assert get_major_version("0.11.0") == 0
|
|
print("PASS: test_get_major_version_normal")
|
|
|
|
|
|
def test_get_major_version_with_rc():
|
|
"""Prerelease versions still yield major number."""
|
|
assert get_major_version("2.0.0rc1") == 2
|
|
assert get_major_version("1.0.0a1") == 1
|
|
print("PASS: test_get_major_version_with_rc")
|
|
|
|
|
|
def test_is_more_than_two_majors_behind():
|
|
"""Difference >2 triggers True; <=2 triggers False."""
|
|
assert is_more_than_two_majors_behind("1.2.3", "4.0.0") is True
|
|
assert is_more_than_two_majors_behind("3.9.0", "4.0.0") is False
|
|
assert is_more_than_two_majors_behind("2.1.0", "5.2.0") is True
|
|
assert is_more_than_two_majors_behind("8.0.0", "9.0.0") is False
|
|
assert is_more_than_two_majors_behind("4.0.0", "4.0.0") is False
|
|
print("PASS: test_is_more_than_two_majors_behind")
|
|
|
|
|
|
def test_analyze_dependencies_very_outdated():
|
|
"""Flag packages more than 2 major versions behind."""
|
|
required = ["pkg_a", "pkg_b"]
|
|
installed = {"pkg_a": "1.0.0", "pkg_b": "3.5.2"}
|
|
outdated = {
|
|
"pkg_a": {"installed": "1.0.0", "latest": "4.0.0"},
|
|
"pkg_b": {"installed": "3.5.2", "latest": "4.0.0"},
|
|
}
|
|
very_out, missing, outdated_ok = analyze_dependencies(required, installed, outdated)
|
|
assert len(very_out) == 1 and very_out[0]["package"] == "pkg_a"
|
|
assert len(missing) == 0
|
|
assert len(outdated_ok) == 1 and outdated_ok[0]["package"] == "pkg_b"
|
|
print("PASS: test_analyze_dependencies_very_outdated")
|
|
|
|
|
|
def test_analyze_dependencies_missing():
|
|
"""Detect packages not installed at all."""
|
|
required = ["pkg_a", "pkg_missing"]
|
|
installed = {"pkg_a": "2.0.0"}
|
|
outdated = {"pkg_a": {"installed": "2.0.0", "latest": "3.0.0"}}
|
|
very_out, missing, outdated_ok = analyze_dependencies(required, installed, outdated)
|
|
assert "pkg_missing" in missing
|
|
assert len(very_out) == 0
|
|
assert len(outdated_ok) == 1
|
|
print("PASS: test_analyze_dependencies_missing")
|
|
|
|
|
|
def test_analyze_dependencies_up_to_date():
|
|
"""Packages up-to-date are not flagged."""
|
|
required = ["pkg_good"]
|
|
installed = {"pkg_good": "5.0.0"}
|
|
outdated = {}
|
|
very_out, missing, outdated_ok = analyze_dependencies(required, installed, outdated)
|
|
assert len(very_out) == 0
|
|
assert len(missing) == 0
|
|
assert len(outdated_ok) == 0
|
|
print("PASS: test_analyze_dependencies_up_to_date")
|
|
|
|
|
|
def test_generate_human_report_contains_very_outdated():
|
|
"""Human report includes very outdated packages."""
|
|
very_out = [
|
|
{"package": "oldpkg", "installed": "1.0", "latest": "4.0", "major_diff": 3}
|
|
]
|
|
missing = []
|
|
outdated_ok = []
|
|
report = mod.generate_human_report(very_out, missing, outdated_ok, "requirements.txt")
|
|
assert "oldpkg" in report
|
|
assert "Installed: 1.0" in report
|
|
assert "Latest: 4.0" in report
|
|
assert "Major diff: 3" in report
|
|
print("PASS: test_generate_human_report_contains_very_outdated")
|
|
|
|
|
|
def test_generate_json_report_structure():
|
|
"""JSON report contains required keys."""
|
|
very_out = [{"package": "oldpkg", "installed": "1.0", "latest": "4.0", "major_diff": 3}]
|
|
missing = ["missing_pkg"]
|
|
outdated_ok = []
|
|
report_json = mod.generate_json_report(very_out, missing, outdated_ok, "requirements.txt")
|
|
data = json.loads(report_json)
|
|
assert "summary" in data
|
|
assert data["summary"]["very_outdated_count"] == 1
|
|
assert data["summary"]["missing_count"] == 1
|
|
assert "very_outdated" in data
|
|
assert "missing" in data
|
|
print("PASS: test_generate_json_report_structure")
|
|
|
|
|
|
if __name__ == '__main__':
|
|
print("Running dependency_freshness test suite...")
|
|
test_parse_requirements_simple()
|
|
test_parse_requirements_with_specifiers()
|
|
test_parse_requirements_ignores_comments_and_blanks()
|
|
test_get_major_version_normal()
|
|
test_get_major_version_with_rc()
|
|
test_is_more_than_two_majors_behind()
|
|
test_analyze_dependencies_very_outdated()
|
|
test_analyze_dependencies_missing()
|
|
test_analyze_dependencies_up_to_date()
|
|
test_generate_human_report_contains_very_outdated()
|
|
test_generate_json_report_structure()
|
|
print("ALL TESTS PASSED.")
|