Loading...
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 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 | #!/usr/bin/env python3 # Copyright (c) 2022 Intel Corp. # SPDX-License-Identifier: Apache-2.0 import argparse import sys import os import time import datetime from github import Github, GithubException from github.GithubException import UnknownObjectException from collections import defaultdict TOP_DIR = os.path.join(os.path.dirname(__file__)) sys.path.insert(0, os.path.join(TOP_DIR, "scripts")) from get_maintainer import Maintainers def log(s): if args.verbose > 0: print(s, file=sys.stdout) def parse_args(): global args parser = argparse.ArgumentParser( description=__doc__, formatter_class=argparse.RawDescriptionHelpFormatter) parser.add_argument("-M", "--maintainer-file", required=False, default="MAINTAINERS.yml", help="Maintainer file to be used.") parser.add_argument("-P", "--pull_request", required=False, default=None, type=int, help="Operate on one pull-request only.") parser.add_argument("-s", "--since", required=False, help="Process pull-requests since date.") parser.add_argument("-y", "--dry-run", action="store_true", default=False, help="Dry run only.") parser.add_argument("-o", "--org", default="zephyrproject-rtos", help="Github organisation") parser.add_argument("-r", "--repo", default="zephyr", help="Github repository") parser.add_argument("-v", "--verbose", action="count", default=0, help="Verbose Output") args = parser.parse_args() def process_pr(gh, maintainer_file, number): gh_repo = gh.get_repo(f"{args.org}/{args.repo}") pr = gh_repo.get_pull(number) log(f"working on https://github.com/{args.org}/{args.repo}/pull/{pr.number} : {pr.title}") labels = set() collab = set() area_counter = defaultdict(int) maint = defaultdict(int) num_files = 0 all_areas = set() fn = list(pr.get_files()) if len(fn) > 500: log(f"Too many files changed ({len(fn)}), skipping....") return for f in pr.get_files(): num_files += 1 log(f"file: {f.filename}") areas = maintainer_file.path2areas(f.filename) if areas: all_areas.update(areas) for a in areas: area_counter[a.name] += 1 labels.update(a.labels) collab.update(a.collaborators) collab.update(a.maintainers) for p in a.maintainers: maint[p] += 1 ac = dict(sorted(area_counter.items(), key=lambda item: item[1], reverse=True)) log(f"Area matches: {ac}") log(f"labels: {labels}") log(f"collab: {collab}") if len(labels) > 10: log(f"Too many labels to be applied") return sm = dict(sorted(maint.items(), key=lambda item: item[1], reverse=True)) log(f"Submitted by: {pr.user.login}") log(f"candidate maintainers: {sm}") prop = 0 if sm: maintainer = list(sm.keys())[0] if len(ac) > 1 and list(ac.values())[0] == list(ac.values())[1]: log("++ Platform/Drivers takes precedence over subsystem...") for aa in ac: if 'Documentation' in aa: log("++ With multiple areas of same weight including docs, take something else other than Documentation as the maintainer") for a in all_areas: if a.name == aa and a.maintainers[0] == maintainer: maintainer = list(sm.keys())[1] elif 'Platform' in aa: log(f"Set maintainer of area {aa}") for a in all_areas: if a.name == aa: if a.maintainers: maintainer = a.maintainers[0] break # if the submitter is the same as the maintainer, check if we have # multiple maintainers if pr.user.login == maintainer: log("Submitter is same as Assignee, trying to find another assignee...") aff = list(ac.keys())[0] for a in all_areas: if a.name == aff: if len(a.maintainers) > 1: maintainer = a.maintainers[1] else: log(f"This area has only one maintainer, keeping assignee as {maintainer}") prop = (maint[maintainer] / num_files) * 100 if prop < 20: maintainer = "None" else: maintainer = "None" log(f"Picked maintainer: {maintainer} ({prop:.2f}% ownership)") log("+++++++++++++++++++++++++") # Set labels if labels and len(labels) < 10: for l in labels: log(f"adding label {l}...") if not args.dry_run: pr.add_to_labels(l) if collab: reviewers = [] existing_reviewers = set() revs = pr.get_reviews() for review in revs: existing_reviewers.add(review.user) rl = pr.get_review_requests() page = 0 for r in rl: existing_reviewers |= set(r.get_page(page)) page += 1 for c in collab: try: u = gh.get_user(c) if pr.user != u and gh_repo.has_in_collaborators(u): if u not in existing_reviewers: reviewers.append(c) except UnknownObjectException as e: log(f"Can't get user '{c}', account does not exist anymore? ({e})") if reviewers: try: log(f"adding reviewers {reviewers}...") if not args.dry_run: pr.create_review_request(reviewers=reviewers) except GithubException: log("cant add reviewer") ms = [] # assignees if maintainer != 'None' and not pr.assignee: try: u = gh.get_user(maintainer) ms.append(u) except GithubException: log(f"Error: Unknown user") for mm in ms: log(f"Adding assignee {mm}...") if not args.dry_run: pr.add_to_assignees(mm) else: log("not setting assignee") time.sleep(1) def main(): parse_args() token = os.environ.get('GITHUB_TOKEN', None) if not token: sys.exit('Github token not set in environment, please set the ' 'GITHUB_TOKEN environment variable and retry.') gh = Github(token) maintainer_file = Maintainers(args.maintainer_file) if args.pull_request: process_pr(gh, maintainer_file, args.pull_request) else: if args.since: since = args.since else: today = datetime.date.today() since = today - datetime.timedelta(days=1) common_prs = f'repo:{args.org}/{args.repo} is:open is:pr base:main -is:draft no:assignee created:>{since}' pulls = gh.search_issues(query=f'{common_prs}') for issue in pulls: process_pr(gh, maintainer_file, issue.number) if __name__ == "__main__": main() |