mirror of
https://github.com/cookiecutter/cookiecutter-django.git
synced 2025-10-24 04:31:02 +03:00
311 lines
12 KiB
Python
311 lines
12 KiB
Python
"""
|
|
Creates an issue that generates a table for dependency checking whether
|
|
all packages support the latest Django version. "Latest" does not include
|
|
patches, only comparing major and minor version numbers.
|
|
|
|
This script handles when there are multiple Django versions that need
|
|
to keep up to date.
|
|
"""
|
|
|
|
from __future__ import annotations
|
|
|
|
import os
|
|
import re
|
|
import sys
|
|
from pathlib import Path
|
|
from typing import TYPE_CHECKING
|
|
from typing import Any
|
|
from typing import NamedTuple
|
|
|
|
import requests
|
|
from github import Github
|
|
|
|
if TYPE_CHECKING:
|
|
from collections.abc import Iterable
|
|
|
|
from github.Issue import Issue
|
|
|
|
CURRENT_FILE = Path(__file__)
|
|
ROOT = CURRENT_FILE.parents[1]
|
|
REQUIREMENTS_DIR = ROOT / "{{cookiecutter.project_slug}}" / "requirements"
|
|
GITHUB_TOKEN = os.getenv("GITHUB_TOKEN", None)
|
|
GITHUB_REPO = os.getenv("GITHUB_REPOSITORY", None)
|
|
|
|
|
|
class DjVersion(NamedTuple):
|
|
"""
|
|
Wrapper to parse, compare and render Django versions.
|
|
|
|
Only keeps track on (major, minor) versions, excluding patches and pre-releases.
|
|
"""
|
|
|
|
major: int
|
|
minor: int
|
|
|
|
def __str__(self) -> str:
|
|
"""To render as string."""
|
|
return f"{self.major}.{self.minor}"
|
|
|
|
@classmethod
|
|
def parse(cls, version_str: str) -> DjVersion:
|
|
"""Parse interesting values from the version string."""
|
|
major, minor, *_ = version_str.split(".")
|
|
return cls(major=int(major), minor=int(minor))
|
|
|
|
@classmethod
|
|
def parse_to_tuple(cls, version_str: str):
|
|
version = cls.parse(version_str=version_str)
|
|
return version.major, version.minor
|
|
|
|
|
|
def get_package_info(package: str) -> dict:
|
|
"""Get package metadata using PyPI API."""
|
|
# "django" converts to "Django" on redirect
|
|
r = requests.get(f"https://pypi.org/pypi/{package}/json", allow_redirects=True) # noqa: S113
|
|
if not r.ok:
|
|
print(f"Couldn't find package: {package}")
|
|
sys.exit(1)
|
|
return r.json()
|
|
|
|
|
|
def get_django_versions() -> Iterable[DjVersion]:
|
|
"""List all django versions."""
|
|
django_package_info: dict[str, Any] = get_package_info("django")
|
|
releases = django_package_info["releases"].keys()
|
|
for release_str in releases:
|
|
if release_str.replace(".", "").isdigit():
|
|
# Exclude pre-releases with non-numeric characters in version
|
|
yield DjVersion.parse(release_str)
|
|
|
|
|
|
def get_name_and_version(requirements_line: str) -> tuple[str, ...]:
|
|
"""Get the name a version of a package from a line in the requirement file."""
|
|
full_name, version = requirements_line.split(" ", 1)[0].split("==")
|
|
name_without_extras = full_name.split("[", 1)[0]
|
|
return name_without_extras, version
|
|
|
|
|
|
def get_all_latest_django_versions(
|
|
django_max_version: tuple[DjVersion] | None = None,
|
|
) -> tuple[DjVersion, list[DjVersion]]:
|
|
"""
|
|
Grabs all Django versions that are worthy of a GitHub issue.
|
|
Depends on Django versions having higher major version or minor version.
|
|
"""
|
|
_django_max_version = (99, 99)
|
|
if django_max_version:
|
|
_django_max_version = django_max_version
|
|
|
|
print("Fetching all Django versions from PyPI")
|
|
base_txt = REQUIREMENTS_DIR / "base.txt"
|
|
with base_txt.open() as f:
|
|
for line in f.readlines():
|
|
if "django==" in line.lower():
|
|
break
|
|
else:
|
|
print(f"django not found in {base_txt}") # Huh...?
|
|
sys.exit(1)
|
|
|
|
# Begin parsing and verification
|
|
_, current_version_str = get_name_and_version(line)
|
|
# Get a tuple of (major, minor) - ignoring patch version
|
|
current_minor_version = DjVersion.parse(current_version_str)
|
|
newer_versions: set[DjVersion] = set()
|
|
for django_version in get_django_versions():
|
|
if current_minor_version < django_version <= _django_max_version:
|
|
newer_versions.add(django_version)
|
|
|
|
return current_minor_version, sorted(newer_versions, reverse=True)
|
|
|
|
|
|
_TABLE_HEADER = """
|
|
|
|
## {file}.txt
|
|
|
|
| Name | Version in Master | {dj_version} Compatible Version | OK |
|
|
| ---- | :---------------: | :-----------------------------: | :-: |
|
|
"""
|
|
VITAL_BUT_UNKNOWN = [
|
|
"django-environ", # not updated often
|
|
]
|
|
|
|
|
|
class GitHubManager:
|
|
def __init__(self, base_dj_version: DjVersion, needed_dj_versions: list[DjVersion]):
|
|
self.github = Github(GITHUB_TOKEN)
|
|
self.repo = self.github.get_repo(GITHUB_REPO)
|
|
|
|
self.base_dj_version = base_dj_version
|
|
self.needed_dj_versions = needed_dj_versions
|
|
# (major+minor) Version and description
|
|
self.existing_issues: dict[DjVersion, Issue] = {}
|
|
|
|
# Load all requirements from our requirements files and preload their
|
|
# package information like a cache:
|
|
self.requirements_files = ["base", "local", "production"]
|
|
# Format:
|
|
# requirement file name: {package name: (master_version, package_info)}
|
|
self.requirements: dict[str, dict[str, tuple[str, dict]]] = {x: {} for x in self.requirements_files}
|
|
|
|
def setup(self) -> None:
|
|
self.load_requirements()
|
|
self.load_existing_issues()
|
|
|
|
def load_requirements(self):
|
|
print("Reading requirements")
|
|
for requirements_file in self.requirements_files:
|
|
with (REQUIREMENTS_DIR / f"{requirements_file}.txt").open() as f:
|
|
for line in f.readlines():
|
|
if (
|
|
"==" in line
|
|
and not line.startswith("{%")
|
|
and not line.startswith(" #")
|
|
and not line.startswith("#")
|
|
and not line.startswith(" ")
|
|
):
|
|
name, version = get_name_and_version(line)
|
|
self.requirements[requirements_file][name] = (
|
|
version,
|
|
get_package_info(name),
|
|
)
|
|
|
|
def load_existing_issues(self):
|
|
"""Closes the issue if the base Django version is greater than needed"""
|
|
print("Load existing issues from GitHub")
|
|
qualifiers = {
|
|
"repo": GITHUB_REPO,
|
|
"author": "app/github-actions",
|
|
"state": "open",
|
|
"is": "issue",
|
|
"in": "title",
|
|
}
|
|
issues = list(self.github.search_issues("[Django Update]", "created", "desc", **qualifiers))
|
|
print(f"Found {len(issues)} issues matching search")
|
|
for issue in issues:
|
|
matches = re.match(r"\[Update Django] Django (\d+.\d+)$", issue.title)
|
|
if not matches:
|
|
continue
|
|
issue_version = DjVersion.parse(matches.group(1))
|
|
if self.base_dj_version >= issue_version:
|
|
self.close_issue(issue)
|
|
else:
|
|
self.existing_issues[issue_version] = issue
|
|
|
|
def get_compatibility(self, package_name: str, package_info: dict, needed_dj_version: DjVersion):
|
|
"""
|
|
Verify compatibility via setup.py classifiers. If Django is not in the
|
|
classifiers, then default compatibility is n/a and OK is ✅.
|
|
|
|
If it's a package that's vital but known to not be updated often, we give it
|
|
a ❓. If a package has ❓ or 🕒, then we allow manual update. Automatic updates
|
|
only include ❌ and ✅.
|
|
"""
|
|
# If issue previously existed, find package and skip any gtg, manually
|
|
# updated packages, or known releases that will happen but haven't yet
|
|
if issue := self.existing_issues.get(needed_dj_version):
|
|
if index := issue.body.find(package_name):
|
|
_name, _current, prev_compat, ok = (s.strip() for s in issue.body[index:].split("|", 4)[:4])
|
|
if ok in ("✅", "❓", "🕒"):
|
|
return prev_compat, ok
|
|
|
|
if package_name in VITAL_BUT_UNKNOWN:
|
|
return "", "❓"
|
|
|
|
# Check classifiers if it includes Django
|
|
supported_dj_versions: list[DjVersion] = []
|
|
for classifier in package_info["info"]["classifiers"]:
|
|
# Usually in the form of "Framework :: Django :: 3.2"
|
|
tokens = classifier.split(" ")
|
|
if len(tokens) >= 5 and tokens[2].lower() == "django" and "." in tokens[4]: # noqa: PLR2004
|
|
version = DjVersion.parse(tokens[4])
|
|
if len(version) == 2: # noqa: PLR2004
|
|
supported_dj_versions.append(version)
|
|
|
|
if supported_dj_versions:
|
|
if any(v >= needed_dj_version for v in supported_dj_versions):
|
|
return package_info["info"]["version"], "✅"
|
|
return "", "❌"
|
|
|
|
# Django classifier DNE; assume it isn't a Django lib
|
|
# Great exceptions include pylint-django, where we need to do this manually...
|
|
return "n/a", "✅"
|
|
|
|
HOME_PAGE_URL_KEYS = [
|
|
"home_page",
|
|
"project_url",
|
|
"docs_url",
|
|
"package_url",
|
|
"release_url",
|
|
"bugtrack_url",
|
|
]
|
|
|
|
def _get_md_home_page_url(self, package_info: dict):
|
|
urls = [package_info["info"].get(url_key) for url_key in self.HOME_PAGE_URL_KEYS]
|
|
try:
|
|
return f"[{{}}]({next(item for item in urls if item)})"
|
|
except StopIteration:
|
|
return "{}"
|
|
|
|
def generate_markdown(self, needed_dj_version: DjVersion):
|
|
requirements = f"{needed_dj_version} requirements tables\n\n"
|
|
for _file in self.requirements_files:
|
|
requirements += _TABLE_HEADER.format_map({"file": _file, "dj_version": needed_dj_version})
|
|
for package_name, (version, info) in self.requirements[_file].items():
|
|
compat_version, icon = self.get_compatibility(package_name, info, needed_dj_version)
|
|
requirements += (
|
|
f"| {self._get_md_home_page_url(info).format(package_name)} "
|
|
f"| {version.strip()} "
|
|
f"| {compat_version.strip()} "
|
|
f"| {icon} "
|
|
f"|\n"
|
|
)
|
|
|
|
return requirements
|
|
|
|
def create_or_edit_issue(self, needed_dj_version: DjVersion, description: str):
|
|
if issue := self.existing_issues.get(needed_dj_version):
|
|
print(f"Editing issue #{issue.number} for Django {needed_dj_version}")
|
|
issue.edit(body=description)
|
|
else:
|
|
print(f"Creating new issue for Django {needed_dj_version}")
|
|
issue = self.repo.create_issue(f"[Update Django] Django {needed_dj_version}", description)
|
|
issue.add_to_labels(f"django{needed_dj_version}")
|
|
|
|
@staticmethod
|
|
def close_issue(issue: Issue):
|
|
issue.edit(state="closed")
|
|
print(f"Closed issue {issue.title} (ID: [{issue.id}]({issue.url}))")
|
|
|
|
def generate(self):
|
|
for version in self.needed_dj_versions:
|
|
print(f"Handling GitHub issue for Django {version}")
|
|
md_content = self.generate_markdown(version)
|
|
print(f"Generated markdown:\n\n{md_content}")
|
|
self.create_or_edit_issue(version, md_content)
|
|
|
|
|
|
def main(django_max_version=None) -> None:
|
|
# Check if there are any djs
|
|
current_dj, latest_djs = get_all_latest_django_versions(django_max_version=django_max_version)
|
|
|
|
# Run the setup, which might close old issues
|
|
manager = GitHubManager(current_dj, latest_djs)
|
|
manager.setup()
|
|
|
|
if not latest_djs:
|
|
print("No new Django versions to update. Exiting...")
|
|
sys.exit(0)
|
|
|
|
manager.generate()
|
|
|
|
|
|
if __name__ == "__main__":
|
|
if GITHUB_REPO is None:
|
|
raise RuntimeError("No github repo, please set the environment variable GITHUB_REPOSITORY")
|
|
max_version = None
|
|
last_arg = sys.argv[-1]
|
|
if CURRENT_FILE.name not in last_arg:
|
|
max_version = DjVersion.parse_to_tuple(version_str=last_arg)
|
|
|
|
main(django_max_version=max_version)
|