forked from aws/aws-cli
-
Notifications
You must be signed in to change notification settings - Fork 7
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
The plan is to build out this test suite to test against the AWS CLI's dependencies to help facilitate dependency upgrades. To start, this test suite contains the following new test cases to better monitor the overall dependency closure of the awscli package: * Assert expected packages in runtime closure. This will alert us if a dependency introduces a new transitive depenency to the AWS CLI closure. * Assert expected unbounded dependencies in runtime closure. Specifically these are dependencies that do not have a version ceiling. This will alert us if a new unbounded dependency is introduced into the AWS CLI runtime dependency closure. See additional implementation notes below: * These tests were broken into a separate test suite (i.e. instead of adding them to the unit and functional test suite) to allow more granularity when running them. Specifically, it is useful for: 1. Avoiding the main unit and functional CI test suite from failing if a dependency changes from underneath of us (e.g. a new build dependency is added that we cannot control). 2. For individuals that package the awscli, they generally will not want to run this test suite as it is fairly specific to how pip installs dependencies. * To determine the runtime dependency closure, the Package and DependencyClosure utilities traverse the dist-info METADATA files of the packages installed in the current site packages to build the runtime graph. This approach was chosen because: 1. Since pip already installed the package, this logic avoids having to reconstruct the logic of how pip decides to resolve dependencies to figure out how to traverse the runtime graph. Any custom logic may deviate from how pip behaves which is what most users will be using to install the awscli as a Python package 2. It's faster. The runtime closure test cases do not require downloading or installing any additional packages.
- Loading branch information
Showing
4 changed files
with
217 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,27 @@ | ||
name: Run dependency tests | ||
|
||
on: | ||
push: | ||
pull_request: | ||
branches-ignore: [ master ] | ||
|
||
jobs: | ||
build: | ||
|
||
runs-on: ${{ matrix.os }} | ||
strategy: | ||
fail-fast: false | ||
matrix: | ||
python-version: ["3.8", "3.9", "3.10", "3.11", "3.12"] | ||
os: [ubuntu-latest, macOS-latest, windows-latest] | ||
|
||
steps: | ||
- uses: actions/checkout@v4 | ||
- name: Set up Python ${{ matrix.python-version }} | ||
uses: actions/setup-python@v5 | ||
with: | ||
python-version: ${{ matrix.python-version }} | ||
- name: Install dependencies | ||
run: python scripts/ci/install | ||
- name: Run tests | ||
run: python scripts/ci/run-dep-tests |
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,35 @@ | ||
#!/usr/bin/env python | ||
# Don't run tests from the root repo dir. | ||
# We want to ensure we're importing from the installed | ||
# binary package not from the CWD. | ||
|
||
import os | ||
import sys | ||
from contextlib import contextmanager | ||
from subprocess import check_call | ||
|
||
_dname = os.path.dirname | ||
|
||
REPO_ROOT = _dname(_dname(_dname(os.path.abspath(__file__)))) | ||
|
||
|
||
@contextmanager | ||
def cd(path): | ||
"""Change directory while inside context manager.""" | ||
cwd = os.getcwd() | ||
try: | ||
os.chdir(path) | ||
yield | ||
finally: | ||
os.chdir(cwd) | ||
|
||
|
||
def run(command): | ||
env = os.environ.copy() | ||
env['TESTS_REMOVE_REPO_ROOT_FROM_PATH'] = 'true' | ||
return check_call(command, shell=True, env=env) | ||
|
||
|
||
if __name__ == "__main__": | ||
with cd(os.path.join(REPO_ROOT, "tests")): | ||
run(f"{sys.executable} {REPO_ROOT}/scripts/ci/run-tests dependencies") |
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,12 @@ | ||
# Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved. | ||
# | ||
# Licensed under the Apache License, Version 2.0 (the "License"). You | ||
# may not use this file except in compliance with the License. A copy of | ||
# the License is located at | ||
# | ||
# http://aws.amazon.com/apache2.0/ | ||
# | ||
# or in the "license" file accompanying this file. This file 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. |
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,143 @@ | ||
# Copyright 2024 Amazon.com, Inc. or its affiliates. All Rights Reserved. | ||
# | ||
# Licensed under the Apache License, Version 2.0 (the "License"). You | ||
# may not use this file except in compliance with the License. A copy of | ||
# the License is located at | ||
# | ||
# http://aws.amazon.com/apache2.0/ | ||
# | ||
# or in the "license" file accompanying this file. This file 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. | ||
import functools | ||
import importlib.metadata | ||
import json | ||
from typing import Dict, Iterator, List, Tuple | ||
|
||
import pytest | ||
from packaging.requirements import Requirement | ||
|
||
_NESTED_STR_DICT = Dict[str, "_NESTED_STR_DICT"] | ||
|
||
|
||
@pytest.fixture(scope="module") | ||
def awscli_package(): | ||
yield Package(name="awscli") | ||
|
||
|
||
class Package: | ||
def __init__(self, name: str) -> None: | ||
self.name = name | ||
|
||
@functools.cached_property | ||
def runtime_dependencies(self) -> "DependencyClosure": | ||
return self._get_runtime_closure() | ||
|
||
def _get_runtime_closure(self) -> "DependencyClosure": | ||
closure = DependencyClosure() | ||
for requirement in self._get_runtime_requirements(): | ||
if self._requirement_applies_to_environment(requirement): | ||
closure[requirement] = Package(name=requirement.name) | ||
return closure | ||
|
||
def _get_runtime_requirements(self) -> List[Requirement]: | ||
req_strings = importlib.metadata.distribution(self.name).requires | ||
if req_strings is None: | ||
return [] | ||
return [Requirement(req_string) for req_string in req_strings] | ||
|
||
def _requirement_applies_to_environment( | ||
self, requirement: Requirement | ||
) -> bool: | ||
# Do not include any requirements defined as extras as currently | ||
# our dependency closure does not use any extras | ||
if requirement.extras: | ||
return False | ||
# Only include requirements where the markers apply to the current | ||
# environment. | ||
if requirement.marker and not requirement.marker.evaluate(): | ||
return False | ||
return True | ||
|
||
|
||
class DependencyClosure: | ||
def __init__(self) -> None: | ||
self._req_to_package: Dict[Requirement, Package] = {} | ||
|
||
def __setitem__(self, key: Requirement, value: Package) -> None: | ||
self._req_to_package[key] = value | ||
|
||
def __getitem__(self, key: Requirement) -> Package: | ||
return self._req_to_package[key] | ||
|
||
def __delitem__(self, key: Requirement) -> None: | ||
del self._req_to_package[key] | ||
|
||
def __iter__(self) -> Iterator[Requirement]: | ||
return iter(self._req_to_package) | ||
|
||
def __len__(self) -> int: | ||
return len(self._req_to_package) | ||
|
||
def walk(self) -> Iterator[Tuple[Requirement, Package]]: | ||
for req, package in self._req_to_package.items(): | ||
yield req, package | ||
yield from package.runtime_dependencies.walk() | ||
|
||
def to_dict(self) -> _NESTED_STR_DICT: | ||
reqs = {} | ||
for req, package in self._req_to_package.items(): | ||
reqs[str(req)] = package.runtime_dependencies.to_dict() | ||
return reqs | ||
|
||
|
||
class TestDependencyClosure: | ||
def _is_bounded_version_requirement( | ||
self, requirement: Requirement | ||
) -> bool: | ||
for specifier in requirement.specifier: | ||
if specifier.operator in ["==", "=<", "<"]: | ||
return True | ||
return False | ||
|
||
def _pformat_closure(self, closure: DependencyClosure) -> str: | ||
return json.dumps(closure.to_dict(), sort_keys=True, indent=2) | ||
|
||
def test_expected_runtime_dependencies(self, awscli_package): | ||
expected_dependencies = { | ||
"botocore", | ||
"colorama", | ||
"docutils", | ||
"jmespath", | ||
"pyasn1", | ||
"python-dateutil", | ||
"PyYAML", | ||
"rsa", | ||
"s3transfer", | ||
"six", | ||
"urllib3", | ||
} | ||
actual_dependencies = set() | ||
for _, package in awscli_package.runtime_dependencies.walk(): | ||
actual_dependencies.add(package.name) | ||
assert actual_dependencies == expected_dependencies, ( | ||
f"Unexpected dependency found in runtime closure: " | ||
f"{self._pformat_closure(awscli_package.runtime_dependencies)}" | ||
) | ||
|
||
def test_expected_unbounded_runtime_dependencies(self, awscli_package): | ||
expected_unbounded_dependencies = { | ||
"pyasn1", # Transitive dependency from rsa | ||
"six", # Transitive dependency from python-dateutil | ||
} | ||
actual_unbounded_dependencies = set() | ||
for req, package in awscli_package.runtime_dependencies.walk(): | ||
if not self._is_bounded_version_requirement(req): | ||
actual_unbounded_dependencies.add(package.name) | ||
assert ( | ||
actual_unbounded_dependencies == expected_unbounded_dependencies | ||
), ( | ||
f"Unexpected unbounded dependency found in runtime closure: " | ||
f"{self._pformat_closure(awscli_package.runtime_dependencies)}" | ||
) |