blob: 83ba3a73e86213b490e0684195494a5c43bebf38 [file] [log] [blame]
#!/usr/bin/env python3
# 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.
from __future__ import annotations
import textwrap
from pathlib import Path
import rich_click as click
from github import Github, Issue
from rich.console import Console
console = Console(width=400, color_system="standard")
MY_DIR_PATH = Path(__file__).parent.resolve()
SOURCE_DIR_PATH = MY_DIR_PATH.parents[1].resolve()
# !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
# NOTE! GitHub has secondary rate limits for issue creation, and you might be
# temporarily blocked from creating issues and PRs if you update too many
# issues in a short time
# !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
option_github_token = click.option(
"--github-token",
type=str,
help=textwrap.dedent(
"""
Github token used to authenticate.
You can omit it if you have the ``GITHUB_TOKEN`` env variable set
Can be generated with:
https://github.com/settings/tokens/new?description=Write%20issues&scopes=repo"""
),
envvar="GITHUB_TOKEN",
)
option_verbose = click.option(
"--verbose",
is_flag=True,
help="Print verbose information about performed steps",
)
option_dry_run = click.option(
"--dry-run",
is_flag=True,
help="Do not create issues, just print the issues to be created",
)
option_repository = click.option(
"--repository",
type=str,
default="apache/airflow",
help="Repo to use",
)
option_labels = click.option(
"--labels",
type=str,
default="AIP-47",
help="Label to filter the issues on (coma-separated)",
)
option_max_issues = click.option("--max-issues", type=int, help="Maximum number of issues to create")
option_start_from = click.option(
"--start-from",
type=int,
default=0,
help="Start from issue number N (useful if you are blocked by secondary rate limit)",
)
def process_paths_from_body(body: str, dry_run: bool, verbose: bool) -> tuple[str, int, int, int, int]:
count_re_added = 0
count_completed = 0
count_done = 0
count_all = 0
new_body = []
for line in body.splitlines(keepends=True):
if line.startswith("- ["):
if verbose:
console.print(line)
path = SOURCE_DIR_PATH / line[len("- [ ] ") :].strip().split(" ")[0]
if path.exists():
count_all += 1
prefix = ""
if line.startswith("- [x]"):
if dry_run:
prefix = "(changed) "
count_re_added += 1
new_body.append(prefix + line.replace("- [x]", "- [ ]"))
else:
count_done += 1
count_all += 1
prefix = ""
if line.startswith("- [ ]"):
if dry_run:
prefix = "(changed) "
count_completed += 1
new_body.append(prefix + line.replace("- [ ]", "- [x]"))
else:
if not dry_run:
new_body.append(line)
return "".join(new_body), count_re_added, count_completed, count_done, count_all
@option_repository
@option_labels
@option_dry_run
@option_github_token
@option_verbose
@option_max_issues
@option_start_from
@click.command()
def update_issue_status(
github_token: str,
max_issues: int | None,
dry_run: bool,
repository: str,
start_from: int,
verbose: bool,
labels: str,
):
"""Update status of the issues regarding the AIP-47 migration."""
g = Github(github_token)
repo = g.get_repo(repository)
issues = repo.get_issues(labels=labels.split(","), state="all")
max_issues = max_issues if max_issues is not None else issues.totalCount
total_re_added = 0
total_completed = 0
total_count_done = 0
total_count_all = 0
num_issues = 0
completed_open_issues: list[Issue.Issue] = []
completed_closed_issues: list[Issue.Issue] = []
not_completed_closed_issues: list[Issue.Issue] = []
not_completed_opened_issues: list[Issue.Issue] = []
per_issue_num_done: dict[int, int] = {}
per_issue_num_all: dict[int, int] = {}
for issue in issues[start_from : start_from + max_issues]:
console.print(f"[blue] {issue.id}: {issue.title}")
new_body, count_re_added, count_completed, count_done, count_all = process_paths_from_body(
issue.body, dry_run=dry_run, verbose=verbose
)
if count_all == 0:
continue
if count_re_added != 0 or count_completed != 0:
if dry_run:
print(new_body)
else:
issue.edit(body=new_body)
console.print()
console.print(f"[blue]Summary of performed actions: for {issue.title}[/]")
console.print(f" Re-added file number (still there): {count_re_added}")
console.print(f" Completed file number: {count_completed}")
console.print(f" Done {count_done}/{count_all} = {count_done / count_all:.2%}")
console.print()
total_re_added += count_re_added
total_completed += count_completed
total_count_done += count_done
total_count_all += count_all
per_issue_num_all[issue.id] = count_all
per_issue_num_done[issue.id] = count_done
if count_done == count_all:
if issue.state == "closed":
completed_closed_issues.append(issue)
else:
completed_open_issues.append(issue)
else:
if issue.state == "closed":
not_completed_closed_issues.append(issue)
else:
not_completed_opened_issues.append(issue)
num_issues += 1
console.print(f"[green]Summary of ALL actions: for {num_issues} issues[/]")
console.print(f" Re-added file number: {total_re_added}")
console.print(f" Completed file number: {total_completed}")
console.print()
console.print()
console.print(f"[green]Summary of ALL issues: for {num_issues} issues[/]")
console.print(
f" Completed and closed issues: {len(completed_closed_issues)}/{num_issues}: "
f"{len(completed_closed_issues) / num_issues:.2%}"
)
console.print(
f" Completed files {total_count_done}/{total_count_all} = "
f"{total_count_done / total_count_all:.2%}"
)
console.print()
if not_completed_closed_issues:
console.print("[yellow] Issues that are not completed and should be opened:[/]\n")
for issue in not_completed_closed_issues:
all = per_issue_num_all[issue.id]
done = per_issue_num_done[issue.id]
console.print(f" * [[yellow]{issue.title}[/]]({issue.html_url}): {done}/{all} : {done / all:.2%}")
console.print()
if completed_open_issues:
console.print("[yellow] Issues that are completed and should be closed:[/]\n")
for issue in completed_open_issues:
console.print(rf" * [[yellow]{issue.title}[/]]({issue.html_url})")
console.print()
if not_completed_opened_issues:
console.print("[yellow] Issues that are not completed and are still opened:[/]\n")
for issue in not_completed_opened_issues:
all = per_issue_num_all[issue.id]
done = per_issue_num_done[issue.id]
console.print(f" * [[yellow]{issue.title}[/]]({issue.html_url}): {done}/{all} : {done / all:.2%}")
console.print()
if completed_closed_issues:
console.print("[green] Issues that are completed and are already closed:[/]\n")
for issue in completed_closed_issues:
console.print(rf" * [[green]{issue.title}[/]]({issue.html_url})")
console.print()
console.print()
if __name__ == "__main__":
update_issue_status()