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 remove specific collection changelog entries from Ansible changelog #639

Merged
merged 3 commits into from
Nov 3, 2024
Merged
Show file tree
Hide file tree
Changes from all 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/639-changelog-cleanup.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
minor_changes:
- "Allow to remove collection changelog entries from the Ansible changelog (/~https://github.com/ansible-community/antsibull-build/pull/639)."
38 changes: 37 additions & 1 deletion src/antsibull_build/build_data_lint.py
Original file line number Diff line number Diff line change
Expand Up @@ -12,10 +12,43 @@

import os

import pydantic as p
from antsibull_changelog.lint import lint_changelog_yaml as _lint_changelog_yaml
from antsibull_core import app_context
from antsibull_core.collection_meta import lint_collection_meta as _lint_collection_meta
from antsibull_core.dependency_files import parse_pieces_file
from antsibull_core.pydantic import forbid_extras, get_formatted_error_messages
from semantic_version import Version as SemVer

from .changelog import RemoveCollectionChangelogEntries


def _lint_rcce(rcce: dict, errors: list[str]) -> None:
forbid_extras(RemoveCollectionChangelogEntries)
try:
rcce_obj = RemoveCollectionChangelogEntries.model_validate(rcce)
for collection_name, versions in rcce_obj.root.items():
for version in versions.root:
try:
SemVer(version)
except ValueError:
errors.append(
f"remove_collection_changelog_entries -> {collection_name}"
f" -> {version}: {version!r} is not a valid semantic version"
)
except p.ValidationError as e:
for msg in get_formatted_error_messages(e):
errors.append(f"remove_collection_changelog_entries -> {msg}")


def _lint_changelog_extra_data(data: dict, errors: list[str]) -> None:
rcce = data.pop("remove_collection_changelog_entries", None) # noqa: F841
if isinstance(rcce, dict):
_lint_rcce(rcce, errors)
elif rcce is not None:
errors.append(
"remove_collection_changelog_entries: Input should be a valid dictionary"
)


def lint_build_data() -> int:
Expand All @@ -38,7 +71,10 @@ def lint_build_data() -> int:
# Lint changelog.yaml
changelog_path = os.path.join(data_dir, "changelog.yaml")
for path, _, __, message in _lint_changelog_yaml(
changelog_path, no_semantic_versioning=True, strict=True
changelog_path,
no_semantic_versioning=True,
strict=True,
preprocess_data=lambda data: _lint_changelog_extra_data(data, errors),
):
errors.append(f"{path}: {message}")

Expand Down
127 changes: 124 additions & 3 deletions src/antsibull_build/changelog.py
Original file line number Diff line number Diff line change
Expand Up @@ -17,9 +17,11 @@
import tempfile
import typing as t
from collections import defaultdict
from collections.abc import Callable

import aiohttp
import asyncio_pool # type: ignore[import]
import pydantic as p
from antsibull_changelog.changes import ChangesData, add_release
from antsibull_changelog.config import (
ChangelogConfig,
Expand Down Expand Up @@ -51,6 +53,37 @@
mlog = log.fields(mod=__name__)


class RemoveCollectionVersionSchema(p.BaseModel):
changes: dict[str, list[str]]


RemoveCollectionVersionsSchema = p.RootModel[dict[str, RemoveCollectionVersionSchema]]

RemoveCollectionChangelogEntries = p.RootModel[
dict[str, RemoveCollectionVersionsSchema]
]


def _extract_extra_data(
data: dict,
store: Callable[[dict[str, dict[SemVer, RemoveCollectionVersionSchema]]], None],
) -> None:
try:
rcce_obj = RemoveCollectionChangelogEntries.model_validate(
data.get("remove_collection_changelog_entries") or {}
)
rcce = {
collection_name: {
SemVer(version): data for version, data in versions.root.items()
}
for collection_name, versions in rcce_obj.root.items()
}
except (p.ValidationError, ValueError):
# ignore error; linting should complain, not us
rcce = {}
store(rcce)


class ChangelogData:
"""
Data for a single changelog (for a collection, for ansible-core, for Ansible)
Expand All @@ -62,6 +95,10 @@ class ChangelogData:
generator: ChangelogGenerator
generator_flatmap: bool

remove_collection_changelog_entries: (
dict[str, dict[SemVer, RemoveCollectionVersionSchema]] | None
)

def __init__(
self,
paths: PathsConfig,
Expand All @@ -74,6 +111,7 @@ def __init__(
self.changes = changes
self.generator_flatmap = flatmap
self.generator = ChangelogGenerator(self.config, self.changes, flatmap=flatmap)
self.remove_collection_changelog_entries = None

@classmethod
def collection(
Expand Down Expand Up @@ -114,13 +152,28 @@ def ansible(
config.release_tag_re = r"""(v(?:[\d.ab\-]|rc)+)"""
config.pre_release_tag_re = r"""(?P<pre_release>(?:[ab]|rc)+\d*)$"""

remove_collection_changelog_entries = {}

def store_extra_data(
rcce: dict[str, dict[SemVer, RemoveCollectionVersionSchema]]
) -> None:
remove_collection_changelog_entries.update(rcce)

changelog_path = ""
if directory is not None:
changelog_path = os.path.join(directory, "changelog.yaml")
changes = ChangesData(config, changelog_path)
changes = ChangesData(
config,
changelog_path,
extra_data_extractor=lambda data: _extract_extra_data(
data, store_extra_data
),
)
if output_directory is not None:
changes.path = os.path.join(output_directory, "changelog.yaml")
return cls(paths, config, changes, flatmap=True)
result = cls(paths, config, changes, flatmap=True)
result.remove_collection_changelog_entries = remove_collection_changelog_entries
return result

@classmethod
def concatenate(cls, changelogs: list[ChangelogData]) -> ChangelogData:
Expand Down Expand Up @@ -159,7 +212,16 @@ def add_ansible_release(
release_date["changes"]["release_summary"] = release_summary

def save(self):
self.changes.save()
extra_data = {}
if self.remove_collection_changelog_entries is not None:
extra_data["remove_collection_changelog_entries"] = {
collection_name: {
str(version): changes.model_dump()
for version, changes in versions.items()
}
for collection_name, versions in self.remove_collection_changelog_entries.items()
}
self.changes.save(extra_data=extra_data or None)


def read_file(tarball_path: str, matcher: t.Callable[[str], bool]) -> bytes | None:
Expand Down Expand Up @@ -763,6 +825,63 @@ def _populate_ansible_changelog(
)


def _cleanup_collection_version(
collection_name: str,
collection_data: dict[SemVer, RemoveCollectionVersionSchema],
changelog: ChangelogData,
) -> None:
flog = mlog.fields(func="_cleanup_collection_version")
for version, data in collection_data.items():
release = changelog.changes.releases.get(str(version))
changes = (release or {}).get("changes")
if not changes:
flog.warning(
f"Trying to remove changelog entries from {collection_name} {version},"
" but found no release"
)
continue
for category, entries in data.changes.items():
if category not in changes:
flog.warning(
f"Trying to remove {category!r} changelog entries from"
f" {collection_name} {version}, but found no entries"
)
continue
for entry in entries:
try:
changes[category].remove(entry)
except ValueError:
flog.warning(
f"Cannot find {category!r} changelog entry for"
f" {collection_name} {version}: {entry!r}"
)


def _cleanup_collection_changelogs(
ansible_changelog: ChangelogData,
collection_collectors: list[CollectionChangelogCollector],
) -> None:
flog = mlog.fields(func="_populate_ansible_changelog")
rcce = ansible_changelog.remove_collection_changelog_entries
if not rcce:
return

for collection_collector in collection_collectors:
collection_data = rcce.get(collection_collector.collection)
if not collection_data:
continue
changelog = collection_collector.changelog
if not changelog:
flog.warning(
f"Trying to remove changelog entries from {collection_collector.collection},"
" but found no changelog"
)
continue
_cleanup_collection_version(
collection_collector.collection, collection_data, changelog
)


def get_changelog(
ansible_version: PypiVer,
deps_dir: str | None,
Expand Down Expand Up @@ -821,6 +940,8 @@ def get_changelog(
collect_changelogs(collectors, core_collector, collection_cache, galaxy_context)
)

_cleanup_collection_changelogs(ansible_changelog, collectors)

changelog = []

sorted_versions = collect_versions(versions, ansible_changelog.config)
Expand Down