|
10 | 10 | # copyright notice, and modified files need to carry a notice indicating
|
11 | 11 | # that they have been altered from the originals.
|
12 | 12 | """
|
13 |
| -Functions for checking dependency versions. |
| 13 | +Functions for checking and reporting installed package versions. |
14 | 14 | """
|
15 | 15 |
|
16 |
| -from importlib.metadata import version |
| 16 | +from __future__ import annotations |
17 | 17 |
|
| 18 | +import warnings |
| 19 | +from functools import lru_cache |
| 20 | +from importlib.metadata import version as metadata_version |
18 | 21 |
|
19 |
| -def numpy_version(): |
20 |
| - """Returns the current numpy version in (major, minor) form.""" |
21 |
| - return tuple(map(int, version("numpy").split(".")[:2])) |
| 22 | +from packaging.version import InvalidVersion, Version |
| 23 | + |
| 24 | +from qiskit.utils.lazy_tester import LazyImportTester |
| 25 | + |
| 26 | + |
| 27 | +__all__ = ["HAS_SKLEARN", "HAS_DYNAMICS", "qiskit_version", "version_is_at_least"] |
| 28 | + |
| 29 | + |
| 30 | +HAS_SKLEARN = LazyImportTester( |
| 31 | + { |
| 32 | + "sklearn.discriminant_analysis": ( |
| 33 | + "LinearDiscriminantAnalysis", |
| 34 | + "QuadraticDiscriminantAnalysis", |
| 35 | + ) |
| 36 | + }, |
| 37 | + name="scikit-learn", |
| 38 | + install="pip install scikit-learn", |
| 39 | +) |
| 40 | + |
| 41 | +HAS_DYNAMICS = LazyImportTester( |
| 42 | + "qiskit_dynamics", |
| 43 | + name="qiskit-dynamics", |
| 44 | + install="pip install qiskit-dynamics", |
| 45 | +) |
| 46 | + |
| 47 | + |
| 48 | +def qiskit_version() -> dict[str, str]: |
| 49 | + """Return a dict with Qiskit names and versions.""" |
| 50 | + return {p: metadata_version(p) for p in ("qiskit", "qiskit-experiments")} |
| 51 | + |
| 52 | + |
| 53 | +@lru_cache(maxsize=None) |
| 54 | +def version_is_at_least(package: str, version: str) -> bool: |
| 55 | + """Return True if the installed version of package greater than minimum version |
| 56 | +
|
| 57 | + Args: |
| 58 | + package: Name of the package |
| 59 | + version: Minimum version name as a string. This should just include |
| 60 | + major, minor, and micro parts. The function will add ``.dev0`` to |
| 61 | + also catch any pre-release versions (otherwise ``0.5.0a1`` would |
| 62 | + evaluate as less than ``0.5.0``). |
| 63 | +
|
| 64 | + Returns: |
| 65 | + True if installed version greater than ``version``. False if it is less |
| 66 | + or if the installed version of ``package`` can not be parsed using the |
| 67 | + specifications of PEP440. |
| 68 | +
|
| 69 | + Raises: |
| 70 | + PackageNotFoundError: |
| 71 | + If ``package`` is not installed. |
| 72 | + """ |
| 73 | + raw_installed_version = metadata_version(package) |
| 74 | + try: |
| 75 | + installed_version = Version(raw_installed_version) |
| 76 | + except InvalidVersion: |
| 77 | + warnings.warn( |
| 78 | + ( |
| 79 | + f"Version string of installed {package} does not match PyPA " |
| 80 | + f"specification. Treating as less than {version}." |
| 81 | + ), |
| 82 | + RuntimeWarning, |
| 83 | + ) |
| 84 | + return False |
| 85 | + return installed_version >= Version(f"{version}.dev0") |
0 commit comments