Skip to content

Commit

Permalink
[BUG] Accepting prereleases as valid python version (#389)
Browse files Browse the repository at this point in the history
#### Reference Issues/PRs
Fixes: sktime/sktime#7517


Make "rc" python version recognizable during python version checking.
Introducing a parameter `prereleases=True`
The following code snipped now returns true:
```python
"3.11.0rc1" in SpecifierSet(">3.8", prereleases=True) 
```
  • Loading branch information
Abelarm authored Dec 29, 2024
1 parent 35e4953 commit c7a75ba
Show file tree
Hide file tree
Showing 3 changed files with 76 additions and 3 deletions.
9 changes: 9 additions & 0 deletions .all-contributorsrc
Original file line number Diff line number Diff line change
Expand Up @@ -103,6 +103,15 @@
"contributions": [
"maintenance"
]
},
{
"login": "Abelarm",
"name": "Luigi Giugliano",
"avatar_url": "https://avatars.githubusercontent.com/u/6976921?v=4",
"profile": "https://github.com/Abelarm",
"contributions": [
"code"
]
}
],
"projectName": "skbase",
Expand Down
16 changes: 14 additions & 2 deletions skbase/utils/dependencies/_dependencies.py
Original file line number Diff line number Diff line change
Expand Up @@ -236,7 +236,9 @@ def _get_pkg_version(package_name):
return pkg_env_version


def _check_python_version(obj, package=None, msg=None, severity="error"):
def _check_python_version(
obj, package=None, msg=None, severity="error", prereleases=True
):
"""Check if system python version is compatible with requirements of obj.
Parameters
Expand All @@ -259,6 +261,13 @@ def _check_python_version(obj, package=None, msg=None, severity="error"):
* "none" - does not raise exception or warning
function returns False if one of packages is not installed, otherwise True
prereleases: str, default = True
Whether prerelease versions are considered compatible.
If True, allows prerelease versions to be considered compatible.
If False, always considers prerelease versions as incompatible, i.e., always
raises error, warning, or returns False, if the system python version is a
prerelease.
Returns
-------
compatible : bool, whether obj is compatible with system python version
Expand All @@ -276,7 +285,7 @@ def _check_python_version(obj, package=None, msg=None, severity="error"):
return True

try:
est_specifier = SpecifierSet(est_specifier_tag)
est_specifier = SpecifierSet(est_specifier_tag, prereleases=prereleases)
except InvalidSpecifier:
msg_version = (
f"wrong format for python_version tag, "
Expand All @@ -303,6 +312,9 @@ def _check_python_version(obj, package=None, msg=None, severity="error"):
f" but system python version is {sys.version}."
)

if "rc" in sys_version:
msg += " This is due to the release candidate status of your system Python."

if package is not None:
msg += (
f" This is due to python version requirements of the {package} package."
Expand Down
54 changes: 53 additions & 1 deletion skbase/utils/dependencies/tests/test_check_dependencies.py
Original file line number Diff line number Diff line change
@@ -1,9 +1,11 @@
# -*- coding: utf-8 -*-
"""Tests for _check_soft_dependencies utility."""
from unittest.mock import patch

import pytest
from packaging.requirements import InvalidRequirement

from skbase.utils.dependencies._dependencies import _check_soft_dependencies
from skbase.utils.dependencies import _check_python_version, _check_soft_dependencies


def test_check_soft_deps():
Expand Down Expand Up @@ -47,3 +49,53 @@ def test_check_soft_deps():
assert _check_soft_dependencies(
("pytest", "!!numpy<~><>0.1.0"), severity="none"
)


@patch("skbase.utils.dependencies._dependencies.sys")
@pytest.mark.parametrize(
"mock_release_version, prereleases, expect_exception",
[
(True, True, False),
(True, False, True),
(False, False, False),
(False, True, False),
],
)
def test_check_python_version(
mock_sys, mock_release_version, prereleases, expect_exception
):
from skbase.base import BaseObject

if mock_release_version:
mock_sys.version = "3.8.1rc"
else:
mock_sys.version = "3.8.1"

class DummyObjectClass(BaseObject):
_tags = {
"python_version": ">=3.7.1", # PEP 440 version specifier, e.g., ">=3.7"
"python_dependencies": None, # PEP 440 dependency strs, e.g., "pandas>=1.0"
"env_marker": None, # PEP 508 environment marker, e.g., "os_name=='posix'"
}
"""Define dummy class to test set_tags."""

dummy_object_instance = DummyObjectClass()

try:
_check_python_version(dummy_object_instance, prereleases=prereleases)
except ModuleNotFoundError as exception:
expected_msg = (
f"{type(dummy_object_instance).__name__} requires python version "
f"to be {dummy_object_instance.get_tags()['python_version']}, "
f"but system python version is {mock_sys.version}. "
"This is due to the release candidate status of your system Python."
)

if not expect_exception or exception.msg != expected_msg:
# Throw Error since exception is not expected or has not the correct message
raise AssertionError(
"ModuleNotFoundError should be NOT raised by:",
f"\n\t - mock_release_version: {mock_release_version},",
f"\n\t - prereleases: {prereleases},",
f"\nERROR MESSAGE: {exception.msg}",
) from exception

0 comments on commit c7a75ba

Please sign in to comment.