123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175 |
- # Copyright 2022 The Matrix.org Foundation C.I.C.
- #
- # Licensed under the Apache License, Version 2.0 (the "License");
- # you may not use this file except in compliance with the License.
- # You may obtain a copy of the License at
- #
- # http://www.apache.org/licenses/LICENSE-2.0
- #
- # Unless required by applicable law or agreed to in writing, software
- # distributed under the License is distributed on an "AS IS" BASIS,
- # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- # See the License for the specific language governing permissions and
- # limitations under the License.
- from contextlib import contextmanager
- from os import PathLike
- from typing import Generator, Optional, Union
- from unittest.mock import patch
- from synapse.util.check_dependencies import (
- DependencyException,
- check_requirements,
- metadata,
- )
- from tests.unittest import TestCase
- class DummyDistribution(metadata.Distribution):
- def __init__(self, version: str):
- self._version = version
- @property
- def version(self) -> str:
- return self._version
- def locate_file(self, path: Union[str, PathLike]) -> PathLike:
- raise NotImplementedError()
- def read_text(self, filename: str) -> None:
- raise NotImplementedError()
- old = DummyDistribution("0.1.2")
- old_release_candidate = DummyDistribution("0.1.2rc3")
- new = DummyDistribution("1.2.3")
- new_release_candidate = DummyDistribution("1.2.3rc4")
- distribution_with_no_version = DummyDistribution(None) # type: ignore[arg-type]
- # could probably use stdlib TestCase --- no need for twisted here
- class TestDependencyChecker(TestCase):
- @contextmanager
- def mock_installed_package(
- self, distribution: Optional[DummyDistribution]
- ) -> Generator[None, None, None]:
- """Pretend that looking up any package yields the given `distribution`.
- If `distribution = None`, we pretend that the package is not installed.
- """
- def mock_distribution(name: str) -> DummyDistribution:
- if distribution is None:
- raise metadata.PackageNotFoundError
- else:
- return distribution
- with patch(
- "synapse.util.check_dependencies.metadata.distribution",
- mock_distribution,
- ):
- yield
- def test_mandatory_dependency(self) -> None:
- """Complain if a required package is missing or old."""
- with patch(
- "synapse.util.check_dependencies.metadata.requires",
- return_value=["dummypkg >= 1"],
- ):
- with self.mock_installed_package(None):
- self.assertRaises(DependencyException, check_requirements)
- with self.mock_installed_package(old):
- self.assertRaises(DependencyException, check_requirements)
- with self.mock_installed_package(new):
- # should not raise
- check_requirements()
- def test_version_reported_as_none(self) -> None:
- """Complain if importlib.metadata.version() returns None.
- This shouldn't normally happen, but it was seen in the wild (#12223).
- """
- with patch(
- "synapse.util.check_dependencies.metadata.requires",
- return_value=["dummypkg >= 1"],
- ):
- with self.mock_installed_package(distribution_with_no_version):
- self.assertRaises(DependencyException, check_requirements)
- def test_checks_ignore_dev_dependencies(self) -> None:
- """Both generic and per-extra checks should ignore dev dependencies."""
- with patch(
- "synapse.util.check_dependencies.metadata.requires",
- return_value=["dummypkg >= 1; extra == 'mypy'"],
- ), patch("synapse.util.check_dependencies.RUNTIME_EXTRAS", {"cool-extra"}):
- # We're testing that none of these calls raise.
- with self.mock_installed_package(None):
- check_requirements()
- check_requirements("cool-extra")
- with self.mock_installed_package(old):
- check_requirements()
- check_requirements("cool-extra")
- with self.mock_installed_package(new):
- check_requirements()
- check_requirements("cool-extra")
- def test_generic_check_of_optional_dependency(self) -> None:
- """Complain if an optional package is old."""
- with patch(
- "synapse.util.check_dependencies.metadata.requires",
- return_value=["dummypkg >= 1; extra == 'cool-extra'"],
- ):
- with self.mock_installed_package(None):
- # should not raise
- check_requirements()
- with self.mock_installed_package(old):
- self.assertRaises(DependencyException, check_requirements)
- with self.mock_installed_package(new):
- # should not raise
- check_requirements()
- def test_check_for_extra_dependencies(self) -> None:
- """Complain if a package required for an extra is missing or old."""
- with patch(
- "synapse.util.check_dependencies.metadata.requires",
- return_value=["dummypkg >= 1; extra == 'cool-extra'"],
- ), patch("synapse.util.check_dependencies.RUNTIME_EXTRAS", {"cool-extra"}):
- with self.mock_installed_package(None):
- self.assertRaises(DependencyException, check_requirements, "cool-extra")
- with self.mock_installed_package(old):
- self.assertRaises(DependencyException, check_requirements, "cool-extra")
- with self.mock_installed_package(new):
- # should not raise
- check_requirements("cool-extra")
- def test_release_candidates_satisfy_dependency(self) -> None:
- """
- Tests that release candidates count as far as satisfying a dependency
- is concerned.
- (Regression test, see #12176.)
- """
- with patch(
- "synapse.util.check_dependencies.metadata.requires",
- return_value=["dummypkg >= 1"],
- ):
- with self.mock_installed_package(old_release_candidate):
- self.assertRaises(DependencyException, check_requirements)
- with self.mock_installed_package(new_release_candidate):
- # should not raise
- check_requirements()
- def test_setuptools_rust_ignored(self) -> None:
- """Test a workaround for a `poetry build` problem. Reproduces #13926."""
- with patch(
- "synapse.util.check_dependencies.metadata.requires",
- return_value=["setuptools_rust >= 1.3"],
- ):
- with self.mock_installed_package(None):
- # should not raise, even if setuptools_rust is not installed
- check_requirements()
- with self.mock_installed_package(old):
- # We also ignore old versions of setuptools_rust
- check_requirements()
|