2020-01-15 17:40:32 +01:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
# Copyright 2019 Google LLC
|
|
|
|
#
|
|
|
|
# Licensed under the the Apache License v2.0 with LLVM Exceptions (the "License");
|
|
|
|
# you may not use this file except in compliance with the License.
|
|
|
|
# You may obtain a copy of the License at
|
|
|
|
#
|
|
|
|
# https://llvm.org/LICENSE.txt
|
|
|
|
#
|
|
|
|
# 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.
|
|
|
|
import argparse
|
|
|
|
import json
|
|
|
|
import os
|
|
|
|
import re
|
|
|
|
import subprocess
|
|
|
|
import sys
|
|
|
|
from typing import List, Optional, Tuple
|
|
|
|
|
|
|
|
from phabricator import Phabricator
|
2020-01-23 09:47:55 +01:00
|
|
|
from git import Repo, GitCommandError
|
2020-01-15 17:40:32 +01:00
|
|
|
|
|
|
|
class ApplyPatch:
|
|
|
|
"""Apply a diff from Phabricator on local working copy.
|
|
|
|
|
|
|
|
This script is a rewrite of `arc patch` to accomodate for dependencies
|
|
|
|
that have already landed, but could not be identified by `arc patch`.
|
|
|
|
|
|
|
|
For a given diff_id, this class will get the dependencies listed on Phabricator.
|
|
|
|
For each dependency D it wil check, if D has already landed by looking
|
|
|
|
for the id of D in the gif history. If D has not landed, it will download
|
|
|
|
the patch for D and try to apply it locally. Once this class has applied all
|
|
|
|
dependencies, it will apply the diff itself.
|
|
|
|
|
|
|
|
|
|
|
|
This script must be called from the root folder of a local checkout of
|
|
|
|
https://github.com/llvm/llvm-project
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self, diff_id:str, comment_file_path: str, token: str, url: str, store_json_diff: str):
|
|
|
|
# TODO: turn os.environ parameter into command line arguments
|
|
|
|
# this would be much clearer and easier for testing
|
|
|
|
self.comment_file_path = comment_file_path
|
|
|
|
self.conduit_token = token # type: Optional[str]
|
|
|
|
self.host = url # type: Optional[str]
|
|
|
|
self._load_arcrc()
|
|
|
|
self.diff_id = diff_id # type: str
|
|
|
|
self.diff_json_path = store_json_diff # type: str
|
|
|
|
if not self.host.endswith('/api/'):
|
|
|
|
self.host += '/api/'
|
2020-01-20 12:06:57 +01:00
|
|
|
self.phab = self._create_phab()
|
2020-01-15 17:40:32 +01:00
|
|
|
self.git_hash = None # type: Optional[str]
|
|
|
|
self.msg = [] # type: List[str]
|
|
|
|
self.repo = Repo(os.getcwd()) # type: Repo
|
|
|
|
|
|
|
|
def _load_arcrc(self):
|
|
|
|
"""Load arc configuration from file if not set."""
|
|
|
|
if self.conduit_token is not None or self.host is not None:
|
|
|
|
return
|
|
|
|
print('Loading configuration from ~/.arcrc file')
|
|
|
|
with open(os.path.expanduser('~/.arcrc'), 'r') as arcrc_file:
|
|
|
|
arcrc = json.load(arcrc_file)
|
|
|
|
# use the first host configured in the file
|
|
|
|
self.host = next(iter(arcrc['hosts']))
|
|
|
|
self.conduit_token = arcrc['hosts'][self.host]['token']
|
|
|
|
|
|
|
|
def run(self):
|
|
|
|
"""try to apply the patch from phabricator
|
|
|
|
|
|
|
|
Write to `self.comment_file` for showing error messages on Phabricator.
|
|
|
|
"""
|
|
|
|
|
|
|
|
try:
|
2020-01-22 11:39:10 +01:00
|
|
|
revision_id, dependencies, base_revision = self._get_dependencies()
|
|
|
|
print('Checking out {}...'.format(base_revision))
|
2020-01-23 09:47:55 +01:00
|
|
|
try:
|
|
|
|
self.repo.git.checkout(base_revision)
|
|
|
|
except GitCommandError:
|
|
|
|
print('ERROR checking out revision {}. It`s not in the '
|
|
|
|
'repository. Using master instead.'.format(base_revision))
|
|
|
|
self.repo.git.checkout('master')
|
2020-01-22 11:39:10 +01:00
|
|
|
print('Revision is {}'.format(self.repo.head.commit.hexsha))
|
2020-01-22 13:19:31 +01:00
|
|
|
print('git reset, git cleanup...')
|
2020-01-22 12:08:51 +00:00
|
|
|
self.repo.git.reset('--hard')
|
2020-01-20 13:04:09 +00:00
|
|
|
self.repo.git.clean('-fdx')
|
2020-01-15 17:40:32 +01:00
|
|
|
print('Analyzing {}'.format(diff_to_str(revision_id)))
|
|
|
|
if len(dependencies) > 0:
|
|
|
|
print('This diff depends on: {}'.format(diff_list_to_str(dependencies)))
|
|
|
|
missing, landed = self._get_missing_landed_dependencies(dependencies)
|
|
|
|
print(' These have already landed: {}'.format(diff_list_to_str(landed)))
|
|
|
|
print(' These are missing on master: {}'.format(diff_list_to_str(missing)))
|
|
|
|
for revision in missing:
|
|
|
|
self._apply_revision(revision)
|
|
|
|
self._apply_diff(self.diff_id, revision_id)
|
2020-01-20 13:16:45 +00:00
|
|
|
print('done.')
|
2020-01-15 17:40:32 +01:00
|
|
|
finally:
|
|
|
|
self._write_error_message()
|
|
|
|
|
2020-01-20 12:37:49 +01:00
|
|
|
def _create_phab(self):
|
2020-01-20 12:06:57 +01:00
|
|
|
phab = Phabricator(token=self.conduit_token, host=self.host)
|
2020-01-20 12:45:07 +01:00
|
|
|
phab.update_interfaces()
|
2020-01-20 12:06:57 +01:00
|
|
|
return phab
|
|
|
|
|
2020-01-15 17:40:32 +01:00
|
|
|
def _get_diff(self, diff_id: str):
|
|
|
|
"""Get a diff from Phabricator based on it's diff id."""
|
|
|
|
return self.phab.differential.getdiff(diff_id=diff_id)
|
|
|
|
|
|
|
|
def _get_revision(self, revision_id: int):
|
|
|
|
"""Get a revision from Phabricator based on its revision id."""
|
|
|
|
return self.phab.differential.query(ids=[revision_id])[0]
|
|
|
|
|
|
|
|
def _get_revisions(self, *, phids: str = None):
|
|
|
|
"""Get a list of revisions from Phabricator based on their PH-IDs."""
|
2020-01-20 13:29:58 +01:00
|
|
|
if phids is None:
|
|
|
|
raise InputError('no arguments given')
|
|
|
|
if phids == []:
|
|
|
|
# Handle an empty query locally. Otherwise the connection
|
|
|
|
# will time out.
|
|
|
|
return []
|
|
|
|
return self.phab.differential.query(phids=phids)
|
2020-01-15 17:40:32 +01:00
|
|
|
|
|
|
|
|
2020-01-22 11:39:10 +01:00
|
|
|
def _get_dependencies(self) -> Tuple[int,List[int],str]:
|
2020-01-15 17:40:32 +01:00
|
|
|
"""Get all dependencies for the diff."""
|
2020-01-22 11:39:10 +01:00
|
|
|
diff = self._get_diff(self.diff_id)
|
|
|
|
revision_id = int(diff.revisionID)
|
2020-01-15 17:40:32 +01:00
|
|
|
revision = self._get_revision(revision_id)
|
2020-01-22 11:39:10 +01:00
|
|
|
base_revision = diff['sourceControlBaseRevision']
|
|
|
|
if base_revision is None or len(base_revision) == 0:
|
|
|
|
base_revision = 'master'
|
2020-01-15 17:40:32 +01:00
|
|
|
dependency_ids = revision['auxiliary']['phabricator:depends-on']
|
|
|
|
revisions = self._get_revisions(phids=dependency_ids)
|
|
|
|
diff_ids = [int(rev['id']) for rev in revisions]
|
|
|
|
# It seems Phabricator lists the dependencies in the opposite order,
|
|
|
|
# so we reverse the order before returning the list, so that they
|
|
|
|
# can be applied in this order
|
|
|
|
diff_ids.reverse()
|
2020-01-22 11:39:10 +01:00
|
|
|
return revision_id, diff_ids, base_revision
|
2020-01-15 17:40:32 +01:00
|
|
|
|
|
|
|
def _apply_diff(self, diff_id: int, revision_id: int):
|
|
|
|
"""Download and apply a diff to the local working copy."""
|
|
|
|
print('Applying diff {} for revision {}...'.format(diff_id, diff_to_str(revision_id)))
|
|
|
|
diff = self.phab.differential.getrawdiff(diffID=diff_id).response
|
2020-01-22 13:19:31 +01:00
|
|
|
proc = subprocess.run('patch -p1', input=diff, shell=True, text=True,
|
2020-01-15 17:40:32 +01:00
|
|
|
stdout=subprocess.PIPE, stderr=subprocess.PIPE)
|
|
|
|
if proc.returncode != 0:
|
|
|
|
raise Exception('Applying patch failed:\n{}'.format(proc.stdout + proc.stderr))
|
|
|
|
|
|
|
|
def _apply_revision(self, revision_id: int):
|
|
|
|
"""Download and apply the latest diff of a revision to the local working copy."""
|
|
|
|
revision = self._get_revision(revision_id)
|
|
|
|
# take the diff_id with the highest number, this should be latest one
|
|
|
|
diff_id = max(revision['diffs'])
|
|
|
|
self._apply_diff(diff_id, revision_id)
|
|
|
|
|
|
|
|
def _write_error_message(self):
|
|
|
|
"""Write the log message to a file."""
|
|
|
|
if self.comment_file_path is None:
|
|
|
|
return
|
|
|
|
|
|
|
|
if len(self.msg) == 0:
|
|
|
|
return
|
|
|
|
print('writing error message to {}'.format(self.comment_file_path))
|
|
|
|
with open(self.comment_file_path, 'a') as comment_file:
|
|
|
|
text = '\n\n'.join(self.msg)
|
|
|
|
comment_file.write(text)
|
|
|
|
|
|
|
|
def _get_landed_revisions(self, limit: int = 1000):
|
|
|
|
"""Get list of landed revisions from current git branch."""
|
|
|
|
diff_regex = re.compile(r'^Differential Revision: https:\/\/reviews\.llvm\.org\/(.*)$', re.MULTILINE)
|
|
|
|
for commit in self.repo.iter_commits("master", max_count=limit):
|
|
|
|
result = diff_regex.search(commit.message)
|
|
|
|
if result is not None:
|
|
|
|
yield result.group(1)
|
|
|
|
return
|
|
|
|
|
|
|
|
def _get_missing_landed_dependencies(self, dependencies: List[int]) -> Tuple[List[int], List[int]]:
|
|
|
|
"""Check which of the dependencies have already landed on the current branch."""
|
|
|
|
landed_deps = []
|
|
|
|
missing_deps = []
|
|
|
|
for dependency in dependencies:
|
|
|
|
if diff_to_str(dependency) in self._get_landed_revisions():
|
|
|
|
landed_deps.append(dependency)
|
|
|
|
else:
|
|
|
|
missing_deps.append(dependency)
|
|
|
|
return missing_deps, landed_deps
|
|
|
|
|
|
|
|
|
|
|
|
def diff_to_str(diff: int) -> str:
|
|
|
|
"""Convert a diff id to a string with leading "D"."""
|
|
|
|
return 'D{}'.format(diff)
|
|
|
|
|
|
|
|
|
|
|
|
def diff_list_to_str(diffs: List[int]) -> str:
|
|
|
|
"""Convert list of diff ids to a comma separated list, prefixed with "D"."""
|
|
|
|
return ', '.join([diff_to_str(d) for d in diffs])
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
parser = argparse.ArgumentParser(description='Apply Phabricator patch to working directory.')
|
|
|
|
parser.add_argument('diff_id', type=str)
|
|
|
|
parser.add_argument('--comment-file', type=str, dest='comment_file_path', default=None)
|
|
|
|
parser.add_argument('--token', type=str, default=None)
|
|
|
|
parser.add_argument('--url', type=str, default=None)
|
|
|
|
parser.add_argument('--store-json-diff', dest='store_json_diff', type=str, default=None)
|
|
|
|
args = parser.parse_args()
|
|
|
|
patcher = ApplyPatch(args.diff_id, args.comment_file_path, args.token, args.url, args.store_json_diff)
|
|
|
|
patcher.run()
|
|
|
|
|