Commit 4c5c8ab6 authored by Edward Lemur's avatar Edward Lemur Committed by Commit Bot

Reland "Reland "gclient: Require a target ref when applying patches.""

This is a reland of 822dbc51

Skia has fixed the test.

Original change's description:
> Reland "gclient: Require a target ref when applying patches."
>
> This is a reland of 1d6478a5
>
> Original change's description:
> > gclient: Require a target ref when applying patches.
> >
> > Bug: 956807
> > Change-Id: Icfffe965f9f4651f22e8ba32c60133a5620bb350
> > Reviewed-on: https://chromium-review.googlesource.com/c/chromium/tools/depot_tools/+/1616804
> > Reviewed-by: Andrii Shyshkalov <tandrii@chromium.org>
> > Commit-Queue: Edward Lesmes <ehmaldonado@chromium.org>
>
> Bug: 956807
> Change-Id: I3de8475a091ce6a2a14ff7dcfb92507a205ef78c
> Reviewed-on: https://chromium-review.googlesource.com/c/chromium/tools/depot_tools/+/1623594
> Reviewed-by: Andrii Shyshkalov <tandrii@chromium.org>
> Commit-Queue: Edward Lesmes <ehmaldonado@chromium.org>

TBR=tandrii@chromium.org

Bug: 956807
Change-Id: Ie273dafb921206f6346678c1861320129ad5b6fd
Reviewed-on: https://chromium-review.googlesource.com/c/chromium/tools/depot_tools/+/1649695Reviewed-by: 's avatarEdward Lesmes <ehmaldonado@chromium.org>
Commit-Queue: Edward Lesmes <ehmaldonado@chromium.org>
parent 073b8ac2
...@@ -1574,13 +1574,12 @@ it or fix the checkout. ...@@ -1574,13 +1574,12 @@ it or fix the checkout.
return patch_refs, target_branches return patch_refs, target_branches
for given_patch_ref in self._options.patch_refs: for given_patch_ref in self._options.patch_refs:
patch_repo, _, patch_ref = given_patch_ref.partition('@') patch_repo, _, patch_ref = given_patch_ref.partition('@')
if not patch_repo or not patch_ref: if not patch_repo or not patch_ref or ':' not in patch_ref:
raise gclient_utils.Error( raise gclient_utils.Error(
'Wrong revision format: %s should be of the form ' 'Wrong revision format: %s should be of the form '
'patch_repo@[target_branch:]patch_ref.' % given_patch_ref) 'patch_repo@target_branch:patch_ref.' % given_patch_ref)
if ':' in patch_ref: target_branch, _, patch_ref = patch_ref.partition(':')
target_branch, _, patch_ref = patch_ref.partition(':') target_branches[patch_repo] = target_branch
target_branches[patch_repo] = target_branch
patch_refs[patch_repo] = patch_ref patch_refs[patch_repo] = patch_ref
return patch_refs, target_branches return patch_refs, target_branches
...@@ -2598,7 +2597,7 @@ def CMDsync(parser, args): ...@@ -2598,7 +2597,7 @@ def CMDsync(parser, args):
parser.add_option('--patch-ref', action='append', parser.add_option('--patch-ref', action='append',
dest='patch_refs', metavar='GERRIT_REF', default=[], dest='patch_refs', metavar='GERRIT_REF', default=[],
help='Patches the given reference with the format ' help='Patches the given reference with the format '
'dep@[target-ref:]patch-ref. ' 'dep@target-ref:patch-ref. '
'For |dep|, you can specify URLs as well as paths, ' 'For |dep|, you can specify URLs as well as paths, '
'with URLs taking preference. ' 'with URLs taking preference. '
'|patch-ref| will be applied to |dep|, rebased on top ' '|patch-ref| will be applied to |dep|, rebased on top '
...@@ -2608,10 +2607,7 @@ def CMDsync(parser, args): ...@@ -2608,10 +2607,7 @@ def CMDsync(parser, args):
'|target-ref| is the target branch against which a ' '|target-ref| is the target branch against which a '
'patch was created, it is used to determine which ' 'patch was created, it is used to determine which '
'commits from the |patch-ref| actually constitute a ' 'commits from the |patch-ref| actually constitute a '
'patch. If not given, we will iterate over all remote ' 'patch.')
'branches and select one that contains the revision '
'|dep| is synced at. '
'WARNING: |target-ref| will be mandatory soon.')
parser.add_option('--with_branch_heads', action='store_true', parser.add_option('--with_branch_heads', action='store_true',
help='Clone git "branch_heads" refspecs in addition to ' help='Clone git "branch_heads" refspecs in addition to '
'the default refspecs. This adds about 1/2GB to a ' 'the default refspecs. This adds about 1/2GB to a '
......
...@@ -351,28 +351,6 @@ class GitWrapper(SCMWrapper): ...@@ -351,28 +351,6 @@ class GitWrapper(SCMWrapper):
self.Print('FAILED to break lock: %s: %s' % (to_break, ex)) self.Print('FAILED to break lock: %s: %s' % (to_break, ex))
raise raise
# TODO(ehmaldonado): Remove after bot_update is modified to pass the patch's
# branch.
def _GetTargetBranchForCommit(self, commit):
"""Get the remote branch a commit is part of."""
_WELL_KNOWN_BRANCHES = [
'refs/remotes/origin/master',
'refs/remotes/origin/infra/config',
'refs/remotes/origin/lkgr',
]
for branch in _WELL_KNOWN_BRANCHES:
if scm.GIT.IsAncestor(self.checkout_path, commit, branch):
return branch
remote_refs = self._Capture(
['for-each-ref', 'refs/remotes/%s' % self.remote,
'--format=%(refname)']).splitlines()
for ref in sorted(remote_refs, reverse=True):
if scm.GIT.IsAncestor(self.checkout_path, commit, ref):
return ref
self.Print('Failed to find a remote ref that contains %s. '
'Candidate refs were %s.' % (commit, remote_refs))
return None
def apply_patch_ref(self, patch_repo, patch_rev, target_rev, options, def apply_patch_ref(self, patch_repo, patch_rev, target_rev, options,
file_list): file_list):
"""Apply a patch on top of the revision we're synced at. """Apply a patch on top of the revision we're synced at.
...@@ -419,8 +397,7 @@ class GitWrapper(SCMWrapper): ...@@ -419,8 +397,7 @@ class GitWrapper(SCMWrapper):
base_rev = self._Capture(['rev-parse', 'HEAD']) base_rev = self._Capture(['rev-parse', 'HEAD'])
if not target_rev: if not target_rev:
# TODO(ehmaldonado): Raise an error once |target_rev| is mandatory. raise gclient_utils.Error('A target revision for the patch must be given')
target_rev = self._GetTargetBranchForCommit(base_rev) or base_rev
elif target_rev.startswith('refs/heads/'): elif target_rev.startswith('refs/heads/'):
# If |target_rev| is in refs/heads/**, try first to find the corresponding # If |target_rev| is in refs/heads/**, try first to find the corresponding
# remote ref for it, since |target_rev| might point to a local ref which # remote ref for it, since |target_rev| might point to a local ref which
......
...@@ -1168,8 +1168,9 @@ class GerritChangesTest(fake_repos.FakeReposTestBase): ...@@ -1168,8 +1168,9 @@ class GerritChangesTest(fake_repos.FakeReposTestBase):
scm.update(self.options, None, file_list) scm.update(self.options, None, file_list)
self.assertEqual(self.githash('repo_1', 4), self.gitrevparse(self.root_dir)) self.assertEqual(self.githash('repo_1', 4), self.gitrevparse(self.root_dir))
scm.apply_patch_ref(self.url, 'refs/changes/35/1235/1', None, self.options, scm.apply_patch_ref(
file_list) self.url, 'refs/changes/35/1235/1', 'refs/heads/master', self.options,
file_list)
self.assertCommits([1, 2, 3, 4, 5, 6]) self.assertCommits([1, 2, 3, 4, 5, 6])
self.assertEqual(self.githash('repo_1', 4), self.gitrevparse(self.root_dir)) self.assertEqual(self.githash('repo_1', 4), self.gitrevparse(self.root_dir))
...@@ -1190,8 +1191,9 @@ class GerritChangesTest(fake_repos.FakeReposTestBase): ...@@ -1190,8 +1191,9 @@ class GerritChangesTest(fake_repos.FakeReposTestBase):
self.assertEqual(self.githash('repo_1', 1), self.gitrevparse(self.root_dir)) self.assertEqual(self.githash('repo_1', 1), self.gitrevparse(self.root_dir))
# Apply the change on top of that. # Apply the change on top of that.
scm.apply_patch_ref(self.url, 'refs/changes/35/1235/1', None, self.options, scm.apply_patch_ref(
file_list) self.url, 'refs/changes/35/1235/1', 'refs/heads/master', self.options,
file_list)
self.assertCommits([1, 5, 6]) self.assertCommits([1, 5, 6])
self.assertEqual(self.githash('repo_1', 1), self.gitrevparse(self.root_dir)) self.assertEqual(self.githash('repo_1', 1), self.gitrevparse(self.root_dir))
...@@ -1207,8 +1209,9 @@ class GerritChangesTest(fake_repos.FakeReposTestBase): ...@@ -1207,8 +1209,9 @@ class GerritChangesTest(fake_repos.FakeReposTestBase):
self.assertEqual(self.githash('repo_1', 9), self.gitrevparse(self.root_dir)) self.assertEqual(self.githash('repo_1', 9), self.gitrevparse(self.root_dir))
# Apply the change on top of that. # Apply the change on top of that.
scm.apply_patch_ref(self.url, 'refs/changes/36/1236/1', None, self.options, scm.apply_patch_ref(
file_list) self.url, 'refs/changes/36/1236/1', 'refs/heads/feature', self.options,
file_list)
self.assertCommits([1, 2, 7, 8, 9, 10]) self.assertCommits([1, 2, 7, 8, 9, 10])
self.assertEqual(self.githash('repo_1', 9), self.gitrevparse(self.root_dir)) self.assertEqual(self.githash('repo_1', 9), self.gitrevparse(self.root_dir))
...@@ -1225,8 +1228,9 @@ class GerritChangesTest(fake_repos.FakeReposTestBase): ...@@ -1225,8 +1228,9 @@ class GerritChangesTest(fake_repos.FakeReposTestBase):
self.assertEqual(self.githash('repo_1', 7), self.gitrevparse(self.root_dir)) self.assertEqual(self.githash('repo_1', 7), self.gitrevparse(self.root_dir))
# Apply the change on top of that. # Apply the change on top of that.
scm.apply_patch_ref(self.url, 'refs/changes/36/1236/1', None, self.options, scm.apply_patch_ref(
file_list) self.url, 'refs/changes/36/1236/1', 'refs/heads/feature', self.options,
file_list)
# We shouldn't have rebased on top of 2 (which is the merge base between # We shouldn't have rebased on top of 2 (which is the merge base between
# remote's master branch and the change) but on top of 7 (which is the # remote's master branch and the change) but on top of 7 (which is the
...@@ -1245,8 +1249,9 @@ class GerritChangesTest(fake_repos.FakeReposTestBase): ...@@ -1245,8 +1249,9 @@ class GerritChangesTest(fake_repos.FakeReposTestBase):
# Apply refs/changes/34/1234/1, created for remote's master branch on top of # Apply refs/changes/34/1234/1, created for remote's master branch on top of
# remote's feature branch. # remote's feature branch.
scm.apply_patch_ref(self.url, 'refs/changes/35/1235/1', 'refs/heads/master', scm.apply_patch_ref(
self.options, file_list) self.url, 'refs/changes/35/1235/1', 'refs/heads/master', self.options,
file_list)
# Commits 5 and 6 are part of the patch, and commits 1, 2, 7, 8 and 9 are # Commits 5 and 6 are part of the patch, and commits 1, 2, 7, 8 and 9 are
# part of remote's feature branch. # part of remote's feature branch.
...@@ -1264,8 +1269,9 @@ class GerritChangesTest(fake_repos.FakeReposTestBase): ...@@ -1264,8 +1269,9 @@ class GerritChangesTest(fake_repos.FakeReposTestBase):
self.assertEqual(self.githash('repo_1', 4), self.gitrevparse(self.root_dir)) self.assertEqual(self.githash('repo_1', 4), self.gitrevparse(self.root_dir))
# Apply the change on top of that. # Apply the change on top of that.
scm.apply_patch_ref(self.url, 'refs/changes/35/1235/1', None, self.options, scm.apply_patch_ref(
file_list) self.url, 'refs/changes/35/1235/1', 'refs/heads/master', self.options,
file_list)
self.assertCommits([1, 2, 3, 5, 6]) self.assertCommits([1, 2, 3, 5, 6])
self.assertEqual(self.githash('repo_1', 4), self.gitrevparse(self.root_dir)) self.assertEqual(self.githash('repo_1', 4), self.gitrevparse(self.root_dir))
...@@ -1283,8 +1289,9 @@ class GerritChangesTest(fake_repos.FakeReposTestBase): ...@@ -1283,8 +1289,9 @@ class GerritChangesTest(fake_repos.FakeReposTestBase):
self.assertEqual(self.githash('repo_1', 1), self.gitrevparse(self.root_dir)) self.assertEqual(self.githash('repo_1', 1), self.gitrevparse(self.root_dir))
# Apply the change on top of that. # Apply the change on top of that.
scm.apply_patch_ref(self.url, 'refs/changes/35/1235/1', None, self.options, scm.apply_patch_ref(
file_list) self.url, 'refs/changes/35/1235/1', 'refs/heads/master', self.options,
file_list)
self.assertCommits([1, 2, 3, 5, 6]) self.assertCommits([1, 2, 3, 5, 6])
self.assertEqual(self.githash('repo_1', 1), self.gitrevparse(self.root_dir)) self.assertEqual(self.githash('repo_1', 1), self.gitrevparse(self.root_dir))
...@@ -1298,8 +1305,9 @@ class GerritChangesTest(fake_repos.FakeReposTestBase): ...@@ -1298,8 +1305,9 @@ class GerritChangesTest(fake_repos.FakeReposTestBase):
scm.update(self.options, None, file_list) scm.update(self.options, None, file_list)
self.assertEqual(self.githash('repo_1', 4), self.gitrevparse(self.root_dir)) self.assertEqual(self.githash('repo_1', 4), self.gitrevparse(self.root_dir))
scm.apply_patch_ref(self.url, 'refs/changes/35/1235/1', None, self.options, scm.apply_patch_ref(
file_list) self.url, 'refs/changes/35/1235/1', 'refs/heads/master', self.options,
file_list)
self.assertCommits([1, 2, 3, 4, 5, 6]) self.assertCommits([1, 2, 3, 4, 5, 6])
# The commit hash after cherry-picking is not known, but it must be # The commit hash after cherry-picking is not known, but it must be
...@@ -1318,15 +1326,17 @@ class GerritChangesTest(fake_repos.FakeReposTestBase): ...@@ -1318,15 +1326,17 @@ class GerritChangesTest(fake_repos.FakeReposTestBase):
# Checkout 'refs/changes/34/1234/1' modifies the 'change' file, so trying to # Checkout 'refs/changes/34/1234/1' modifies the 'change' file, so trying to
# patch 'refs/changes/36/1236/1' creates a patch failure. # patch 'refs/changes/36/1236/1' creates a patch failure.
with self.assertRaises(subprocess2.CalledProcessError) as cm: with self.assertRaises(subprocess2.CalledProcessError) as cm:
scm.apply_patch_ref(self.url, 'refs/changes/36/1236/1', None, scm.apply_patch_ref(
self.options, file_list) self.url, 'refs/changes/36/1236/1', 'refs/heads/master', self.options,
file_list)
self.assertEqual(cm.exception.cmd[:2], ['git', 'cherry-pick']) self.assertEqual(cm.exception.cmd[:2], ['git', 'cherry-pick'])
self.assertIn('error: could not apply', cm.exception.stderr) self.assertIn('error: could not apply', cm.exception.stderr)
# Try to apply 'refs/changes/35/1235/1', which doesn't have a merge # Try to apply 'refs/changes/35/1235/1', which doesn't have a merge
# conflict. # conflict.
scm.apply_patch_ref(self.url, 'refs/changes/35/1235/1', None, self.options, scm.apply_patch_ref(
file_list) self.url, 'refs/changes/35/1235/1', 'refs/heads/master', self.options,
file_list)
self.assertCommits([1, 2, 3, 5, 6]) self.assertCommits([1, 2, 3, 5, 6])
self.assertEqual(self.githash('repo_1', 5), self.gitrevparse(self.root_dir)) self.assertEqual(self.githash('repo_1', 5), self.gitrevparse(self.root_dir))
...@@ -1343,8 +1353,9 @@ class GerritChangesTest(fake_repos.FakeReposTestBase): ...@@ -1343,8 +1353,9 @@ class GerritChangesTest(fake_repos.FakeReposTestBase):
# 'refs/changes/34/1234/1' will be an empty commit, since the changes were # 'refs/changes/34/1234/1' will be an empty commit, since the changes were
# already present in the tree as commit 11. # already present in the tree as commit 11.
# Make sure we deal with this gracefully. # Make sure we deal with this gracefully.
scm.apply_patch_ref(self.url, 'refs/changes/35/1235/1', None, self.options, scm.apply_patch_ref(
file_list) self.url, 'refs/changes/35/1235/1', 'refs/heads/feature', self.options,
file_list)
self.assertCommits([1, 2, 3, 5, 6, 12]) self.assertCommits([1, 2, 3, 5, 6, 12])
self.assertEqual(self.githash('repo_1', 12), self.assertEqual(self.githash('repo_1', 12),
self.gitrevparse(self.root_dir)) self.gitrevparse(self.root_dir))
...@@ -1366,8 +1377,9 @@ class GerritChangesTest(fake_repos.FakeReposTestBase): ...@@ -1366,8 +1377,9 @@ class GerritChangesTest(fake_repos.FakeReposTestBase):
# Try to apply 'refs/changes/35/1235/1', which doesn't have a merge # Try to apply 'refs/changes/35/1235/1', which doesn't have a merge
# conflict. # conflict.
scm.apply_patch_ref(self.url, 'refs/changes/35/1235/1', None, self.options, scm.apply_patch_ref(
file_list) self.url, 'refs/changes/35/1235/1', 'refs/heads/master', self.options,
file_list)
self.assertCommits([1, 2, 3, 5, 6]) self.assertCommits([1, 2, 3, 5, 6])
self.assertEqual(self.githash('repo_1', 5), self.gitrevparse(self.root_dir)) self.assertEqual(self.githash('repo_1', 5), self.gitrevparse(self.root_dir))
......
...@@ -589,29 +589,6 @@ class GClientSmokeGIT(GClientSmokeBase): ...@@ -589,29 +589,6 @@ class GClientSmokeGIT(GClientSmokeBase):
self.assertTree(tree) self.assertTree(tree)
def testSyncPatchRef(self): def testSyncPatchRef(self):
if not self.enabled:
return
self.gclient(['config', self.git_base + 'repo_1', '--name', 'src'])
self.gclient([
'sync', '-v', '-v', '-v',
'--revision', 'src/repo2@%s' % self.githash('repo_2', 1),
'--patch-ref',
'%srepo_2@%s' % (self.git_base, self.githash('repo_2', 2)),
])
# Assert that repo_2 files coincide with revision @2 (the patch ref)
tree = self.mangle_git_tree(('repo_1@2', 'src'),
('repo_2@2', 'src/repo2'),
('repo_3@2', 'src/repo2/repo_renamed'))
tree['src/git_hooked1'] = 'git_hooked1'
tree['src/git_hooked2'] = 'git_hooked2'
self.assertTree(tree)
# Assert that HEAD revision of repo_2 is @1 (the base we synced to) since we
# should have done a soft reset.
self.assertEqual(
self.githash('repo_2', 1),
self.gitrevparse(os.path.join(self.root_dir, 'src/repo2')))
def testSyncPatchRefBranch(self):
if not self.enabled: if not self.enabled:
return return
self.gclient(['config', self.git_base + 'repo_1', '--name', 'src']) self.gclient(['config', self.git_base + 'repo_1', '--name', 'src'])
...@@ -643,7 +620,8 @@ class GClientSmokeGIT(GClientSmokeBase): ...@@ -643,7 +620,8 @@ class GClientSmokeGIT(GClientSmokeBase):
'sync', '-v', '-v', '-v', 'sync', '-v', '-v', '-v',
'--revision', 'src/repo2@%s' % self.githash('repo_2', 1), '--revision', 'src/repo2@%s' % self.githash('repo_2', 1),
'--patch-ref', '--patch-ref',
'%srepo_2@%s' % (self.git_base, self.githash('repo_2', 2)), '%srepo_2@refs/heads/master:%s' % (
self.git_base, self.githash('repo_2', 2)),
'--nohooks', '--nohooks',
]) ])
# Assert that repo_2 files coincide with revision @2 (the patch ref) # Assert that repo_2 files coincide with revision @2 (the patch ref)
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment