Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Allow to use constraints files for new-ansible and prepare #546

Merged
merged 6 commits into from
Sep 27, 2023
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions changelogs/fragments/546-constraints.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
minor_changes:
- "Support a constraints file that allows to fix dependencies for the ``new-ansible`` and ``prepare`` subcommands (https://github.com/ansible-community/antsibull/pull/546)."
177 changes: 70 additions & 107 deletions src/antsibull/build_ansible_commands.py
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,7 @@
from .dep_closure import check_collection_dependencies
from .tagging import get_collections_tags
from .utils.get_pkg_data import get_antsibull_data
from .versions import feature_freeze_version, load_constraints_if_exists

if TYPE_CHECKING:
from _typeshed import StrPath
Expand Down Expand Up @@ -84,11 +85,66 @@ async def get_latest_ansible_core_version(
return max(newer_versions) if newer_versions else None


async def get_latest_collection_version(
felixfontein marked this conversation as resolved.
Show resolved Hide resolved
client: GalaxyClient,
collection: str,
version_spec: str,
pre: bool = False,
constraint: SemVerSpec | None = None,
) -> SemVer:
"""
Get the latest version of a collection that matches a specification.

:arg collection: Namespace.collection identifying a collection.
:arg version_spec: String specifying the allowable versions.
:kwarg pre: If True, allow prereleases (versions which have the form X.Y.Z.SOMETHING).
This is **not** for excluding 0.Y.Z versions. non-pre-releases are still
preferred over pre-releases (for instance, with version_spec='>2.0.0-a1,<3.0.0'
and pre=True, if the available versions are 2.0.0-a1 and 2.0.0-a2, then 2.0.0-a2
will be returned. If the available versions are 2.0.0 and 2.1.0-b2, 2.0.0 will be
returned since non-pre-releases are preferred. The default is False
:kwarg constraint: If provided, only consider versions that match this specification.
:returns: :obj:`semantic_version.Version` of the latest collection version that satisfied
the specification.

.. seealso:: For the format of the version_spec, see the documentation
of :obj:`semantic_version.SimpleSpec`
"""
versions = await client.get_versions(collection)
sem_versions = [SemVer(v) for v in versions]
sem_versions.sort(reverse=True)

spec = SemVerSpec(version_spec)
prereleases = []
for version in (v for v in sem_versions if v in spec):
# Ignore all versions that do not match the constraints
if constraint is not None and version not in constraint:
continue
# If this is a pre-release, first check if there's a non-pre-release that
# will satisfy the version_spec.
if version.prerelease:
prereleases.append(version)
continue
return version

# We did not find a stable version that satisies the version_spec. If we
# allow prereleases, return the latest of those here.
if pre and prereleases:
return prereleases[0]

# No matching versions were found
constraint_clause = "" if constraint is None else f" (with constraint {constraint})"
raise ValueError(
f"{version_spec}{constraint_clause} did not match with any version of {collection}."
)


async def get_collection_and_core_versions(
deps: Mapping[str, str],
ansible_core_version: PypiVer | None,
galaxy_url: str,
ansible_core_allow_prerelease: bool = False,
constraints: dict[str, SemVerSpec] | None = None,
) -> tuple[dict[str, SemVer], PypiVer | None]:
"""
Retrieve the latest version of each collection.
Expand All @@ -101,15 +157,21 @@ async def get_collection_and_core_versions(
:returns: Tuple consisting of a dict mapping collection name to latest version, and of the
ansible-core version if it was provided.
"""
constraints = constraints or {}

requestors = {}
async with aiohttp.ClientSession() as aio_session:
lib_ctx = app_context.lib_ctx.get()
async with asyncio_pool.AioPool(size=lib_ctx.thread_max) as pool:
client = GalaxyClient(aio_session, galaxy_server=galaxy_url)
for collection_name, version_spec in deps.items():
requestors[collection_name] = await pool.spawn(
client.get_latest_matching_version(
collection_name, version_spec, pre=True
get_latest_collection_version(
client,
collection_name,
version_spec,
pre=True,
constraint=constraints.get(collection_name),
)
)
if ansible_core_version:
Expand Down Expand Up @@ -421,111 +483,6 @@ def _extract_python_requires(
)


class _FeatureFreezeVersion:
"""
Helper for making semantic version range specification valid for feature freeze.
"""

def __init__(self, spec: str, collection_name: str):
self.potential_clauses: list = []
self.spec = spec
self.collection_name = collection_name
self.upper_operator: str | None = None
self.upper_version: SemVer | None = None
self.min_version: SemVer | None = None
self.pinned = False

spec_obj = SemVerSpec(spec)

# If there is a single clause, it's available as spec_obj.clause;
# multiple ones are available as spec_obj.clause.clauses.
try:
clauses = spec_obj.clause.clauses
except AttributeError:
clauses = [spec_obj.clause]

self.clauses = clauses
for clause in clauses:
self._process_clause(clause)

def _process_clause(self, clause) -> None:
"""
Process one clause of the version range specification.
"""
if clause.operator in ("<", "<="):
if self.upper_operator is not None:
raise ValueError(
f"Multiple upper version limits specified for {self.collection_name}:"
f" {self.spec}"
)
self.upper_operator = clause.operator
self.upper_version = clause.target
# Omit the upper bound as we're replacing it
return

if clause.operator == ">=":
# Save the lower bound so we can write out a new compatible version
if self.min_version is not None:
raise ValueError(
f"Multiple minimum versions specified for {self.collection_name}: {self.spec}"
)
self.min_version = clause.target

if clause.operator == ">":
raise ValueError(
f"Strict lower bound specified for {self.collection_name}: {self.spec}"
)

if clause.operator == "==":
self.pinned = True

self.potential_clauses.append(clause)

def compute_new_spec(self) -> str:
"""
Compute a new version range specification that only allows newer patch releases that also
match the original range specification.
"""
if self.pinned:
if len(self.clauses) > 1:
raise ValueError(
f"Pin combined with other clauses for {self.collection_name}: {self.spec}"
)
return self.spec

upper_operator = self.upper_operator
upper_version = self.upper_version
if upper_operator is None or upper_version is None:
raise ValueError(
f"No upper version limit specified for {self.collection_name}: {self.spec}"
)

min_version = self.min_version
if min_version is None:
raise ValueError(
f"No minimum version specified for {self.collection_name}: {self.spec}"
)

if min_version.next_minor() <= upper_version:
upper_operator = "<"
upper_version = min_version.next_minor()

new_clauses = sorted(
str(clause)
for clause in self.potential_clauses
if clause.target < upper_version
)
new_clauses.append(f"{upper_operator}{upper_version}")
return ",".join(new_clauses)


def feature_freeze_version(spec: str, collection_name: str) -> str:
"""
Make semantic version range specification valid for feature freeze.
"""
return _FeatureFreezeVersion(spec, collection_name).compute_new_spec()


def prepare_command() -> int:
app_ctx = app_context.app_ctx.get()

Expand All @@ -537,6 +494,11 @@ def prepare_command() -> int:
ansible_core_version_obj = PypiVer(ansible_core_version)
python_requires = _extract_python_requires(ansible_core_version_obj, deps)

constraints_filename = os.path.join(
app_ctx.extra["data_dir"], app_ctx.extra["constraints_file"]
)
constraints = load_constraints_if_exists(constraints_filename)

# If we're building a feature frozen release (betas and rcs) then we need to
# change the upper version limit to not include new features.
if app_ctx.extra["feature_frozen"]:
Expand All @@ -550,6 +512,7 @@ def prepare_command() -> int:
ansible_core_version_obj,
app_ctx.galaxy_url,
ansible_core_allow_prerelease=_is_alpha(app_ctx.extra["ansible_version"]),
constraints=constraints,
)
)
if new_ansible_core_version:
Expand Down
38 changes: 31 additions & 7 deletions src/antsibull/cli/antsibull_build.py
Original file line number Diff line number Diff line change
Expand Up @@ -120,15 +120,19 @@ def _normalize_new_release_options(args: argparse.Namespace) -> None:
" per line"
)

compat_version_part = (
f"{args.ansible_version.major}"
if args.ansible_version.major > 2
else f"{args.ansible_version.major}.{args.ansible_version.minor}"
)
felixfontein marked this conversation as resolved.
Show resolved Hide resolved

if args.build_file is None:
basename = os.path.basename(os.path.splitext(args.pieces_file)[0])
if args.ansible_version.major > 2:
args.build_file = f"{basename}-{args.ansible_version.major}.build"
else:
args.build_file = (
f"{basename}-{args.ansible_version.major}"
f".{args.ansible_version.minor}.build"
)
args.build_file = f"{basename}-{compat_version_part}.build"

if args.constraints_file is None:
basename = os.path.basename(os.path.splitext(args.pieces_file)[0])
args.constraints_file = f"{basename}-{compat_version_part}.constraints"


def _check_release_build_directories(args: argparse.Namespace) -> None:
Expand Down Expand Up @@ -170,6 +174,10 @@ def _normalize_release_build_options(args: argparse.Namespace) -> None:
" of versions per line"
)

if args.constraints_file is None:
basename = os.path.basename(os.path.splitext(args.build_file)[0])
args.constraints_file = f"{basename}.constraints"

if args.deps_file is None:
version_suffix = f"-{compat_version_part}"
basename = os.path.basename(os.path.splitext(args.build_file)[0])
Expand Down Expand Up @@ -296,6 +304,14 @@ def parse_args(program_name: str, args: list[str]) -> argparse.Namespace:
" --build-data-dir. The default is"
" $BASENAME_OF_BUILD_FILE-X.Y.Z.deps",
)
build_step_parser.add_argument(
"--constraints-file",
default=None,
help="File containing a list of constraints for collections"
" included in Ansible. This is considered to be relative to"
" --build-data-dir. The default is"
" $BASENAME_OF_BUILD_FILE-X.Y.constraints",
Comment on lines +307 to +313
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can we create a constraints_parser that contains this argument that build_step_parserandnew_parser` inherit instead of defining this argument twice?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The descriptions for the two options differ w.r.t. the default value. One uses the basename of the build file, while the other uses the basename of the pieces file. That's also why --build-file both shows up in the same two parsers.

)

feature_freeze_parser = argparse.ArgumentParser(add_help=False)
feature_freeze_parser.add_argument(
Expand Down Expand Up @@ -357,6 +373,14 @@ def parse_args(program_name: str, args: list[str]) -> argparse.Namespace:
default=False,
help="Allow prereleases of collections to be included in the build" " file",
)
new_parser.add_argument(
"--constraints-file",
default=None,
help="File containing a list of constraints for collections"
" included in Ansible. This is considered to be relative to"
" --build-data-dir. The default is"
" $BASENAME_OF_PIECES_FILE-X.Y.constraints",
)

prepare_parser = subparsers.add_parser(
"prepare",
Expand Down
Loading