| # Licensed to the Apache Software Foundation (ASF) under one |
| # or more contributor license agreements. See the NOTICE file |
| # distributed with this work for additional information |
| # regarding copyright ownership. The ASF licenses this file |
| # to you under the Apache License, Version 2.0 (the |
| # "License"); you may not use this file except in compliance |
| # with the License. You may obtain a copy of the License at |
| # |
| # http://www.apache.org/licenses/LICENSE-2.0 |
| # |
| # Unless required by applicable law or agreed to in writing, |
| # software distributed under the License 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. |
| |
| """Github utils for release changelog.""" |
| |
| from typing import Dict, List |
| |
| |
| class Changelog: |
| """Generate changelog according specific pull requests list. |
| |
| Each pull requests will only once in final result. If pull requests have more than one label we need, |
| will classify to high priority label type, currently priority is |
| `dsip > feature > bug > improvement > document > chore`. pr will into feature section if it with both `feature`, |
| `improvement`, `document` label. |
| |
| :param prs: pull requests list. |
| """ |
| |
| key_number = "number" |
| key_labels = "labels" |
| key_name = "name" |
| |
| label_dsip = "dsip" |
| label_feature = "feature" |
| label_bug = "bug" |
| label_improvement = "improvement" |
| label_document = "document" |
| label_chore = "chore" |
| |
| changelog_prefix = "\n\n<details><summary>Click to expand</summary>\n\n" |
| changelog_suffix = "\n\n</details>\n" |
| |
| def __init__(self, prs: List[Dict]): |
| self.prs = prs |
| self.dsips = [] |
| self.features = [] |
| self.bugfixs = [] |
| self.improvements = [] |
| self.documents = [] |
| self.chores = [] |
| self.others = [] |
| |
| def generate(self) -> str: |
| """Generate changelog.""" |
| self.classify() |
| final = [] |
| if self.dsips: |
| detail = f"## DSIP{self.changelog_prefix}{self._convert(self.dsips)}{self.changelog_suffix}" |
| final.append(detail) |
| if self.features: |
| detail = f"## Feature{self.changelog_prefix}{self._convert(self.features)}{self.changelog_suffix}" |
| final.append(detail) |
| if self.improvements: |
| detail = ( |
| f"## Improvement{self.changelog_prefix}" |
| f"{self._convert(self.improvements)}{self.changelog_suffix}" |
| ) |
| final.append(detail) |
| if self.bugfixs: |
| detail = f"## Bugfix{self.changelog_prefix}{self._convert(self.bugfixs)}{self.changelog_suffix}" |
| final.append(detail) |
| if self.documents: |
| detail = ( |
| f"## Document{self.changelog_prefix}" |
| f"{self._convert(self.documents)}{self.changelog_suffix}" |
| ) |
| final.append(detail) |
| if self.chores: |
| detail = f"## Chore{self.changelog_prefix}{self._convert(self.chores)}{self.changelog_suffix}" |
| final.append(detail) |
| if self.others: |
| detail = f"## Others{self.changelog_prefix}{self._convert(self.others)}{self.changelog_suffix}" |
| final.append(detail) |
| return "\n".join(final) |
| |
| @staticmethod |
| def _convert(prs: List[Dict]) -> str: |
| """Convert pull requests into changelog item text.""" |
| return "\n".join( |
| [f"- {pr['title']} (#{pr['number']}) @{pr['user']['login']}" for pr in prs] |
| ) |
| |
| def classify(self) -> None: |
| """Classify pull requests different kinds of section in changelog. |
| |
| Each pull requests only belongs to one single classification. |
| """ |
| for pr in self.prs: |
| if self.key_labels not in pr: |
| raise KeyError("PR %s do not have labels", pr[self.key_number]) |
| if self._is_dsip(pr): |
| self.dsips.append(pr) |
| elif self._is_feature(pr): |
| self.features.append(pr) |
| elif self._is_bugfix(pr): |
| self.bugfixs.append(pr) |
| elif self._is_improvement(pr): |
| self.improvements.append(pr) |
| elif self._is_document(pr): |
| self.documents.append(pr) |
| elif self._is_chore(pr): |
| self.chores.append(pr) |
| else: |
| self.others.append(pr) |
| |
| def _is_dsip(self, pr: Dict) -> bool: |
| """Belong to dsip pull requests.""" |
| return any( |
| [ |
| label[self.key_name].lower() == self.label_dsip |
| for label in pr[self.key_labels] |
| ] |
| ) |
| |
| def _is_feature(self, pr: Dict) -> bool: |
| """Belong to feature pull requests.""" |
| return any( |
| [ |
| label[self.key_name] == self.label_feature |
| for label in pr[self.key_labels] |
| ] |
| ) |
| |
| def _is_bugfix(self, pr: Dict) -> bool: |
| """Belong to bugfix pull requests.""" |
| return any( |
| [label[self.key_name] == self.label_bug for label in pr[self.key_labels]] |
| ) |
| |
| def _is_improvement(self, pr: Dict) -> bool: |
| """Belong to improvement pull requests.""" |
| return any( |
| [ |
| label[self.key_name] == self.label_improvement |
| for label in pr[self.key_labels] |
| ] |
| ) |
| |
| def _is_document(self, pr: Dict) -> bool: |
| """Belong to document pull requests.""" |
| return any( |
| [ |
| label[self.key_name] == self.label_document |
| for label in pr[self.key_labels] |
| ] |
| ) |
| |
| def _is_chore(self, pr: Dict) -> bool: |
| """Belong to chore pull requests.""" |
| return any( |
| [label[self.key_name] == self.label_chore for label in pr[self.key_labels]] |
| ) |