2019-11-22 22:19:59 +01:00
|
|
|
import re
|
|
|
|
import time
|
|
|
|
from collections import defaultdict
|
|
|
|
|
|
|
|
from lib import Client
|
|
|
|
|
|
|
|
|
|
|
|
class Checker():
|
|
|
|
def __init__(self, gerrit_bot_phid, project_patch_for_review_phid, client):
|
|
|
|
self.gerrit_bot_phid = gerrit_bot_phid
|
|
|
|
self.project_patch_for_review_phid = project_patch_for_review_phid
|
|
|
|
self.client = client
|
|
|
|
|
|
|
|
def check(self, t_id):
|
2023-01-19 19:10:16 +01:00
|
|
|
"""
|
|
|
|
Returns true if the Patch-For-Review project should be removed from the Phabricator
|
|
|
|
task identified 't_id'.
|
|
|
|
"""
|
2019-11-22 22:19:59 +01:00
|
|
|
phid = self.client.lookupPhid(t_id)
|
|
|
|
return self.phid_check(phid)
|
|
|
|
|
2023-01-19 19:10:16 +01:00
|
|
|
def phid_check(self, phid) -> bool:
|
|
|
|
"""
|
|
|
|
Returns true if the Patch-For-Review project should be removed from the Phabricator
|
|
|
|
task identified by 'phid'.
|
|
|
|
"""
|
2019-11-22 22:19:59 +01:00
|
|
|
gerrit_bot_actions = []
|
2023-01-19 19:10:16 +01:00
|
|
|
|
|
|
|
# Note that transactions are returned in reverse chronological order (most recent first).
|
2019-11-22 22:19:59 +01:00
|
|
|
for transaction in self.client.getTransactions(phid):
|
2020-09-29 15:03:19 +02:00
|
|
|
if re.findall(re.escape('https://github.com/') + r'.+?/pull', str(transaction)):
|
2019-11-22 22:19:59 +01:00
|
|
|
return False
|
|
|
|
if transaction['authorPHID'] == self.gerrit_bot_phid:
|
|
|
|
gerrit_bot_actions.append(transaction)
|
|
|
|
else:
|
2023-01-19 19:10:16 +01:00
|
|
|
# If someone other than GerritBot adds the Patch-For-Review project, don't
|
|
|
|
# auto-remove it.
|
2019-11-22 22:19:59 +01:00
|
|
|
if transaction['type'] == 'projects':
|
|
|
|
check = self.project_patch_for_review_phid in str(
|
|
|
|
transaction['fields'])
|
|
|
|
add_check = "'add'" in str(transaction['fields'])
|
|
|
|
if check and add_check:
|
|
|
|
return False
|
|
|
|
|
|
|
|
gerrit_patch_status = defaultdict(list)
|
|
|
|
for case in gerrit_bot_actions:
|
|
|
|
if case['type'] != 'comment':
|
|
|
|
continue
|
|
|
|
|
|
|
|
if len(case['comments']) != 1:
|
|
|
|
return False
|
|
|
|
raw_comment = case['comments'][0]['content']['raw']
|
|
|
|
gerrit_patch_id = re.findall(
|
2020-07-01 20:50:02 +02:00
|
|
|
r'https://gerrit(?:-test|)\.wikimedia\.org/r/.*(\d+)(?:$|\]\])', raw_comment)[0]
|
2019-11-22 22:19:59 +01:00
|
|
|
merged = re.findall(
|
2021-03-12 10:50:20 +01:00
|
|
|
r'Change \d+ \*\*(?:merged|abandoned)\*\* by ',
|
2019-11-22 22:19:59 +01:00
|
|
|
raw_comment)
|
|
|
|
|
2023-01-19 19:10:16 +01:00
|
|
|
# Append True or False depending on whether the action was to
|
|
|
|
# open a patch (True) or merge a patch (False)
|
2019-11-22 22:19:59 +01:00
|
|
|
gerrit_patch_status[gerrit_patch_id].append(not(bool(merged)))
|
|
|
|
|
|
|
|
for patch in gerrit_patch_status:
|
2023-01-19 19:10:16 +01:00
|
|
|
# The normal sequence of GerritBot transactions for a Gerrit change is "Change
|
|
|
|
# \d+ had a related patch set uploaded" (indicated by True in
|
|
|
|
# gerrit_patch_status) eventually followed by "Change \d+ (merged|abandoned)
|
|
|
|
# by whoever" (indicated by False in gerrit_patch_status). The transactions
|
|
|
|
# are returned in reverse order so the opened/merged pattern will appear as
|
|
|
|
# the reverse of [True, False], which is [False, True].
|
2019-11-22 22:19:59 +01:00
|
|
|
if gerrit_patch_status[patch] != [False, True]:
|
|
|
|
return False
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
client = Client.newFromCreds()
|
|
|
|
|
2023-01-19 19:10:16 +01:00
|
|
|
gerrit_bot_phid = 'PHID-USER-idceizaw6elwiwm5xshb'
|
2019-11-22 22:19:59 +01:00
|
|
|
project_patch_for_review_phid = 'PHID-PROJ-onnxucoedheq3jevknyr'
|
|
|
|
checker = Checker(
|
2023-01-19 19:10:16 +01:00
|
|
|
gerrit_bot_phid,
|
2019-11-22 22:19:59 +01:00
|
|
|
project_patch_for_review_phid,
|
|
|
|
client)
|
|
|
|
gen = client.getTasksWithProject(project_patch_for_review_phid)
|
|
|
|
for phid in gen:
|
|
|
|
if checker.phid_check(phid):
|
|
|
|
print(client.taskDetails(phid)['id'])
|
|
|
|
try:
|
|
|
|
client.removeProjectByPhid(project_patch_for_review_phid, phid)
|
|
|
|
except BaseException:
|
|
|
|
continue
|
|
|
|
time.sleep(10)
|