This repository has been archived by the owner on Apr 26, 2024. It is now read-only.
-
-
Notifications
You must be signed in to change notification settings - Fork 2.1k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
David Robertson
committed
Feb 25, 2022
1 parent
e5d73cb
commit 3ec1214
Showing
1 changed file
with
91 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,91 @@ | ||
from contextlib import contextmanager | ||
from typing import Optional, ContextManager | ||
from unittest.mock import patch | ||
|
||
from synapse.util.check_dependencies import ( | ||
check_requirements, | ||
DependencyException, | ||
metadata, | ||
) | ||
from tests.unittest import TestCase | ||
|
||
|
||
class DummyDistribution(metadata.Distribution): | ||
def __init__(self, version: str): | ||
self._version = version | ||
|
||
@property | ||
def version(self): | ||
return self._version | ||
|
||
|
||
old = DummyDistribution("0.1.2") | ||
new = DummyDistribution("1.2.3") | ||
|
||
# could probably use stdlib TestCase --- no need for twisted here | ||
|
||
|
||
class TestDependencyChecker(TestCase): | ||
@contextmanager | ||
def mock_installed_package( | ||
self, distribution: Optional[DummyDistribution] | ||
) -> ContextManager[None]: | ||
"""Pretend that looking up any distribution yields the given `distribution`.""" | ||
|
||
def mock_distribution(name: str): | ||
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_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.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() | ||
|