1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88
|
import subprocess
from dataclasses import dataclass
from datetime import datetime
from typing import List, Set
import requests # type: ignore
@dataclass
class PullRequest:
number: int
title: str
user: str
user_url: str
url: str
class ChangelogGenerator:
def __init__(
self,
username: str,
repository: str,
current_version: str,
new_version: str,
):
self.username = username
self.repository = repository
self.base_url = f"https://api.github.com/repos/{username}/{repository}"
self.current_version = current_version
self.new_version = new_version
self.commits = self.get_commits_after_tag(current_version)
self.prs = self.get_prs_for_commits(self.commits)
def get_commits_after_tag(self, tag: str) -> List[str]:
result = subprocess.run(
["git", "log", f"{tag}..HEAD", "--pretty=format:%H"],
stdout=subprocess.PIPE,
text=True,
)
return result.stdout.split()
def get_pr_for_commit(self, commit_sha: str) -> PullRequest:
url = f"{self.base_url}/commits/{commit_sha}/pulls"
response = requests.get(url)
response.raise_for_status()
pr_data = response.json()[0]
return PullRequest(
number=pr_data["number"],
title=pr_data["title"],
user=pr_data["user"]["login"],
user_url=pr_data["user"]["html_url"],
url=pr_data["html_url"],
)
def get_prs_for_commits(self, commit_shas: List[str]) -> List[PullRequest]:
prs: List[PullRequest] = []
unique_prs: Set[int] = set()
for commit_sha in commit_shas:
pr = self.get_pr_for_commit(commit_sha)
pr_id = pr.number
if pr_id not in unique_prs:
unique_prs.add(pr_id)
prs.append(pr)
prs.sort(key=lambda pr: pr.number, reverse=True)
return prs
def generate_changelog(self) -> str:
markdown = f"\n## [{self.new_version}] - {datetime.now().strftime('%Y-%m-%d')}\n"
for pr in self.prs:
markdown += (
f"### {pr.title.capitalize()}\n"
f"- Author - [{pr.user}]({pr.user_url})\n"
f"- PR <{pr.url}>\n"
)
markdown += f"\n[{self.new_version}]: https://pypi.org/project/{self.repository}/{self.new_version}\n"
return markdown
if __name__ == "__main__":
generator = ChangelogGenerator(
username="BeanieODM",
repository="beanie",
current_version="1.30.0",
new_version="2.0.0",
)
changelog = generator.generate_changelog()
print(changelog)
|