Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(606)

Side by Side Diff: gclient_scm.py

Issue 328843005: Consolidated 'git' refish parsing into a class (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/tools/depot_tools
Patch Set: Updated w/ nits, added new case(s) to refish parsing Created 6 years, 6 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
« no previous file with comments | « gclient.py ('k') | tests/gclient_scm_test.py » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 # Copyright (c) 2012 The Chromium Authors. All rights reserved. 1 # Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 # Use of this source code is governed by a BSD-style license that can be 2 # Use of this source code is governed by a BSD-style license that can be
3 # found in the LICENSE file. 3 # found in the LICENSE file.
4 4
5 """Gclient-specific SCM-specific operations.""" 5 """Gclient-specific SCM-specific operations."""
6 6
7 from __future__ import print_function 7 from __future__ import print_function
M-A Ruel 2014/07/07 16:47:07 We ought to remove them eventually.
8 8
9 import collections
9 import errno 10 import errno
10 import logging 11 import logging
11 import os 12 import os
12 import posixpath 13 import posixpath
13 import re 14 import re
14 import shlex 15 import shlex
15 import sys 16 import sys
16 import tempfile 17 import tempfile
17 import traceback 18 import traceback
18 import urlparse 19 import urlparse
(...skipping 133 matching lines...) Expand 10 before | Expand all | Expand 10 after
152 153
153 if not command in commands: 154 if not command in commands:
154 raise gclient_utils.Error('Unknown command %s' % command) 155 raise gclient_utils.Error('Unknown command %s' % command)
155 156
156 if not command in dir(self): 157 if not command in dir(self):
157 raise gclient_utils.Error('Command %s not implemented in %s wrapper' % ( 158 raise gclient_utils.Error('Command %s not implemented in %s wrapper' % (
158 command, self.__class__.__name__)) 159 command, self.__class__.__name__))
159 160
160 return getattr(self, command)(options, args, file_list) 161 return getattr(self, command)(options, args, file_list)
161 162
162 def GetActualRemoteURL(self, options): 163 def GetActualRemoteURL(self):
163 """Attempt to determine the remote URL for this SCMWrapper.""" 164 """Attempt to determine the remote URL for this SCMWrapper."""
164 # Git
165 if os.path.exists(os.path.join(self.checkout_path, '.git')): 165 if os.path.exists(os.path.join(self.checkout_path, '.git')):
166 actual_remote_url = shlex.split(scm.GIT.Capture( 166 actual_remote_url = shlex.split(scm.GIT.Capture(
167 ['config', '--local', '--get-regexp', r'remote.*.url'], 167 ['config', '--local', '--get-regexp', r'remote.*.url'],
168 cwd=self.checkout_path))[1] 168 cwd=self.checkout_path))[1]
169 169
170 # If a cache_dir is used, obtain the actual remote URL from the cache. 170 # If a cache_dir is used, obtain the actual remote URL from the cache.
171 if getattr(self, 'cache_dir', None): 171 if getattr(self, 'cache_dir', None):
172 url, _ = gclient_utils.SplitUrlRevision(self.url) 172 url, _ = gclient_utils.SplitUrlRevision(self.url)
173 mirror = git_cache.Mirror(url) 173 mirror = git_cache.Mirror(url)
174 if (mirror.exists() and mirror.mirror_path.replace('\\', '/') == 174 if (mirror.exists() and mirror.mirror_path.replace('\\', '/') ==
175 actual_remote_url.replace('\\', '/')): 175 actual_remote_url.replace('\\', '/')):
176 actual_remote_url = shlex.split(scm.GIT.Capture( 176 actual_remote_url = shlex.split(scm.GIT.Capture(
177 ['config', '--local', '--get-regexp', r'remote.*.url'], 177 ['config', '--local', '--get-regexp', r'remote.*.url'],
178 cwd=mirror.mirror_path))[1] 178 cwd=mirror.mirror_path))[1]
179 return actual_remote_url 179 return actual_remote_url
180 180
181 # Svn 181 # Svn
182 if os.path.exists(os.path.join(self.checkout_path, '.svn')): 182 if os.path.exists(os.path.join(self.checkout_path, '.svn')):
183 return scm.SVN.CaptureLocalInfo([], self.checkout_path)['URL'] 183 return scm.SVN.CaptureLocalInfo([], self.checkout_path)['URL']
184 return None 184 return None
185 185
186 def DoesRemoteURLMatch(self, options): 186 def DoesRemoteURLMatch(self, options):
187 """Determine whether the remote URL of this checkout is the expected URL.""" 187 """Determine whether the remote URL of this checkout is the expected URL."""
188 if not os.path.exists(self.checkout_path): 188 if not os.path.exists(self.checkout_path):
189 # A checkout which doesn't exist can't be broken. 189 # A checkout which doesn't exist can't be broken.
190 return True 190 return True
191 191
192 actual_remote_url = self.GetActualRemoteURL(options) 192 actual_remote_url = self.GetActualRemoteURL()
193 if actual_remote_url: 193 if actual_remote_url:
194 return (gclient_utils.SplitUrlRevision(actual_remote_url)[0].rstrip('/') 194 return (gclient_utils.SplitUrlRevision(actual_remote_url)[0].rstrip('/')
195 == gclient_utils.SplitUrlRevision(self.url)[0].rstrip('/')) 195 == gclient_utils.SplitUrlRevision(self.url)[0].rstrip('/'))
196 else: 196 else:
197 # This may occur if the self.checkout_path exists but does not contain a 197 # This may occur if the self.checkout_path exists but does not contain a
198 # valid git or svn checkout. 198 # valid git or svn checkout.
199 return False 199 return False
200 200
201 def _DeleteOrMove(self, force): 201 def _DeleteOrMove(self, force):
202 """Delete the checkout directory or move it out of the way. 202 """Delete the checkout directory or move it out of the way.
(...skipping 20 matching lines...) Expand all
223 dest_path = tempfile.mkdtemp( 223 dest_path = tempfile.mkdtemp(
224 prefix=os.path.basename(self.relpath), 224 prefix=os.path.basename(self.relpath),
225 dir=bad_scm_dir) 225 dir=bad_scm_dir)
226 self.Print('_____ Conflicting directory found in %s. Moving to %s.' 226 self.Print('_____ Conflicting directory found in %s. Moving to %s.'
227 % (self.checkout_path, dest_path)) 227 % (self.checkout_path, dest_path))
228 gclient_utils.AddWarning('Conflicting directory %s moved to %s.' 228 gclient_utils.AddWarning('Conflicting directory %s moved to %s.'
229 % (self.checkout_path, dest_path)) 229 % (self.checkout_path, dest_path))
230 shutil.move(self.checkout_path, dest_path) 230 shutil.move(self.checkout_path, dest_path)
231 231
232 232
233 _GitRefishBase = collections.namedtuple('GitRef', (
234 # The initial string that the parsed Refish came from
235 'source',
236 # Is this ref a branch?
237 'is_branch',
238 # The name of this ref when accessed through the local repository
239 'local_ref',
240 # The name of the remote that this refish resides on
241 'remote',
242 # The path of this refish on the remote (e.g., 'master')
243 'remote_ref',
244 # The remote-qualified path to this refish (e.g., 'origin/master')
245 'remote_refspec',
246 # If a branch, the expected name of the upstream branch that it should
247 # track; otherwise, 'None'
248 'upstream_branch',
249 ))
250
251
252 class GitRefish(_GitRefishBase):
253 # Disable 'no __init__' warning | pylint: disable=W0232
254 """Implements refish parsing and properties.
255
256 This class is designed to concentrate and codify the assumptions made by
257 'gclient' code about refs and revisions.
258 """
259
260 _DEFAULT_REMOTE = 'origin'
261
262 _GIT_SHA1_RE = re.compile('[0-9a-f]{40}', re.IGNORECASE)
263 _GIT_SHA1_SHORT_RE = re.compile('[0-9a-f]{4,40}', re.IGNORECASE)
264
265 def __str__(self):
266 return self.source
267
268 @classmethod
269 def Parse(cls, ref, remote=None, other_remotes=None):
270 """
M-A Ruel 2014/07/07 16:47:07 """Parses ...
271 Parses a ref into a 'GitRefish' instance.
272
273 Args:
274 ref: (str) The refish (branch, tag, hash, etc.) to parse
275 remote: (None/str) If supplied, the name of the primary remote. In
276 addtion to being recognized as a remote string, the primary remote
277 will also be used (as needed) when generating the remote refspec. If
278 omitted, the default remote name ('origin') will be used.
279 other_remotes: (None/iterable) If supplied, a list of strings to
280 recognize as remotes in addition to 'remote'.
281 """
282 assert ref == ref.strip()
283
284 # Use default set of remotes
285 remote = remote or cls._DEFAULT_REMOTE
286 remotes = {remote}
287 if other_remotes:
288 remotes.update(other_remotes)
289
290 # Treat 'ref' as a '/'-delimited set of items; analyze their contents
291 ref_split = local_ref = remote_ref = tuple(ref.split('/'))
292 is_branch = True
293 if len(ref_split) > 1:
294 if ref_split[0] == 'refs':
295 if len(ref_split) >= 3 and ref_split[1] == 'heads':
296 # refs/heads/foo/bar
297 #
298 # Local Ref: foo/bar
299 # Remote Ref: foo/bar
300 local_ref = remote_ref = ref_split[2:]
301 elif len(ref_split) >= 4 and ref_split[1] == 'remotes':
302 # refs/remotes/<REMOTE>/foo/bar
303 # This is a bad assumption, and this logic can be improved, but it
304 # is consistent with traditional 'gclient' logic.
305 #
306 # Local Ref: refs/remotes/<REMOTE>/foo/bar
307 # Remote: <REMOTE>
308 # Remote Ref: foo/bar
309 remote = ref_split[2]
310 remote_ref = ref_split[3:]
311 elif len(ref_split) >= 2 and ref_split[0] in remotes:
312 # origin/master
313 #
314 # Local Ref: refs/remotes/origin/master
315 # Remote Ref: master
316 remote = ref_split[0]
317 remote_ref = ref_split[1:]
318 local_ref = ('refs', 'remotes') + ref_split
319
320 # (Default) The refish has multiple paths. Assume at least that it's a
321 # branch name.
322 #
323 # foo/bar
324 # refs/foo/bar
325 #
326 # Local Ref: foo/bar
327 # Remote ref: foo/bar
328 else:
329 # It could be a hash, a short-hash, or a tag
330 is_branch = False
331
332 # Determine how the ref should be referenced remotely
333 if is_branch:
334 # foo/bar/baz => origin/foo/bar/baz
335 remote_refspec = (remote,) + remote_ref
336 else:
337 # Refer to the hash/tag directly. This is actually not allowed in
338 # 'fetch', although it oftentimes works regardless.
339 #
340 # <HASH/TAG> => <HASH/TAG>
341 remote_refspec = (ref,)
342
343 # Calculate the upstream branch
344 if is_branch:
345 # Convert '/refs/heads/...' to 'refs/remotes/REMOTE/...'
346 if ref_split[:2] == ('refs', 'heads'):
347 upstream_branch = ('refs', 'remotes', remote) + ref_split[2:]
348 else:
349 upstream_branch = ref_split
350 else:
351 upstream_branch = None
352
353 def compose(ref_tuple):
354 return '/'.join(ref_tuple) if ref_tuple else ref_tuple
355
356 return cls(
357 source=ref,
358 is_branch=is_branch,
359 local_ref=compose(local_ref),
360 remote=remote,
361 remote_ref=compose(remote_ref),
362 remote_refspec=compose(remote_refspec),
363 upstream_branch=compose(upstream_branch),
364 )
365
366
233 class GitWrapper(SCMWrapper): 367 class GitWrapper(SCMWrapper):
234 """Wrapper for Git""" 368 """Wrapper for Git"""
235 name = 'git' 369 name = 'git'
236 remote = 'origin' 370 remote = 'origin'
237 371
238 cache_dir = None 372 cache_dir = None
239 373
240 def __init__(self, url=None, *args): 374 def __init__(self, url=None, *args):
241 """Removes 'git+' fake prefix from git URL.""" 375 """Removes 'git+' fake prefix from git URL."""
242 if url.startswith('git+http://') or url.startswith('git+https://'): 376 if url.startswith('git+http://') or url.startswith('git+https://'):
243 url = url[4:] 377 url = url[4:]
244 SCMWrapper.__init__(self, url, *args) 378 SCMWrapper.__init__(self, url, *args)
245 filter_kwargs = { 'time_throttle': 1, 'out_fh': self.out_fh } 379 filter_kwargs = { 'time_throttle': 1, 'out_fh': self.out_fh }
246 if self.out_cb: 380 if self.out_cb:
247 filter_kwargs['predicate'] = self.out_cb 381 filter_kwargs['predicate'] = self.out_cb
248 self.filter = gclient_utils.GitFilter(**filter_kwargs) 382 self.filter = gclient_utils.GitFilter(**filter_kwargs)
249 383
250 @staticmethod 384 @staticmethod
251 def BinaryExists(): 385 def BinaryExists():
252 """Returns true if the command exists.""" 386 """Returns true if the command exists."""
253 try: 387 try:
254 # We assume git is newer than 1.7. See: crbug.com/114483 388 # We assume git is newer than 1.7. See: crbug.com/114483
255 result, version = scm.GIT.AssertVersion('1.7') 389 result, version = scm.GIT.AssertVersion('1.7')
256 if not result: 390 if not result:
257 raise gclient_utils.Error('Git version is older than 1.7: %s' % version) 391 raise gclient_utils.Error('Git version is older than 1.7: %s' % version)
258 return result 392 return result
259 except OSError: 393 except OSError:
260 return False 394 return False
261 395
396 def ParseRefish(self, value, **kwargs):
397 kwargs.setdefault('remote', self.remote)
398 return GitRefish.Parse(value, **kwargs)
399
262 def GetCheckoutRoot(self): 400 def GetCheckoutRoot(self):
263 return scm.GIT.GetCheckoutRoot(self.checkout_path) 401 return scm.GIT.GetCheckoutRoot(self.checkout_path)
264 402
265 def GetRevisionDate(self, _revision): 403 def GetRevisionDate(self, _revision):
266 """Returns the given revision's date in ISO-8601 format (which contains the 404 """Returns the given revision's date in ISO-8601 format (which contains the
267 time zone).""" 405 time zone)."""
268 # TODO(floitsch): get the time-stamp of the given revision and not just the 406 # TODO(floitsch): get the time-stamp of the given revision and not just the
269 # time-stamp of the currently checked out revision. 407 # time-stamp of the currently checked out revision.
270 return self._Capture(['log', '-n', '1', '--format=%ai']) 408 return self._Capture(['log', '-n', '1', '--format=%ai'])
271 409
(...skipping 14 matching lines...) Expand all
286 424
287 The patch file is generated from a diff of the merge base of HEAD and 425 The patch file is generated from a diff of the merge base of HEAD and
288 its upstream branch. 426 its upstream branch.
289 """ 427 """
290 merge_base = self._Capture(['merge-base', 'HEAD', self.remote]) 428 merge_base = self._Capture(['merge-base', 'HEAD', self.remote])
291 gclient_utils.CheckCallAndFilter( 429 gclient_utils.CheckCallAndFilter(
292 ['git', 'diff', merge_base], 430 ['git', 'diff', merge_base],
293 cwd=self.checkout_path, 431 cwd=self.checkout_path,
294 filter_fn=GitDiffFilterer(self.relpath).Filter, print_func=self.Print) 432 filter_fn=GitDiffFilterer(self.relpath).Filter, print_func=self.Print)
295 433
296 def _FetchAndReset(self, revision, file_list, options): 434 def _FetchAndReset(self, refish, file_list, options):
297 """Equivalent to git fetch; git reset.""" 435 """Equivalent to git fetch; git reset."""
298 quiet = [] 436 quiet = []
299 if not options.verbose: 437 if not options.verbose:
300 quiet = ['--quiet'] 438 quiet = ['--quiet']
301 self._UpdateBranchHeads(options, fetch=False) 439 self._UpdateBranchHeads(options, fetch=False)
302 440
303 self._Fetch(options, prune=True, quiet=options.verbose) 441 self._Fetch(options, prune=True, quiet=options.verbose)
304 self._Run(['reset', '--hard', revision] + quiet, options) 442 self._Run(['reset', '--hard', refish.local_ref] + quiet, options)
305 if file_list is not None: 443 if file_list is not None:
306 files = self._Capture(['ls-files']).splitlines() 444 files = self._Capture(['ls-files']).splitlines()
307 file_list.extend([os.path.join(self.checkout_path, f) for f in files]) 445 file_list.extend([os.path.join(self.checkout_path, f) for f in files])
308 446
309 def _DisableHooks(self): 447 def _DisableHooks(self):
310 hook_dir = os.path.join(self.checkout_path, '.git', 'hooks') 448 hook_dir = os.path.join(self.checkout_path, '.git', 'hooks')
311 if not os.path.isdir(hook_dir): 449 if not os.path.isdir(hook_dir):
312 return 450 return
313 for f in os.listdir(hook_dir): 451 for f in os.listdir(hook_dir):
314 if not f.endswith('.sample') and not f.endswith('.disabled'): 452 if not f.endswith('.sample') and not f.endswith('.disabled'):
315 os.rename(os.path.join(hook_dir, f), 453 os.rename(os.path.join(hook_dir, f),
316 os.path.join(hook_dir, f + '.disabled')) 454 os.path.join(hook_dir, f + '.disabled'))
317 455
318 def update(self, options, args, file_list): 456 def update(self, options, args, file_list):
319 """Runs git to update or transparently checkout the working copy. 457 """Runs git to update or transparently checkout the working copy.
320 458
321 All updated files will be appended to file_list. 459 All updated files will be appended to file_list.
322 460
323 Raises: 461 Raises:
324 Error: if can't get URL for relative path. 462 Error: if can't get URL for relative path.
325 """ 463 """
326 if args: 464 if args:
327 raise gclient_utils.Error("Unsupported argument(s): %s" % ",".join(args)) 465 raise gclient_utils.Error("Unsupported argument(s): %s" % ",".join(args))
328 466
329 self._CheckMinVersion("1.6.6") 467 self._CheckMinVersion("1.6.6")
330 468
331 # If a dependency is not pinned, track the default remote branch. 469 # If a dependency is not pinned, track the default remote branch.
332 default_rev = 'refs/remotes/%s/master' % self.remote 470 default_rev = 'refs/remotes/%s/master' % (self.remote,)
333 url, deps_revision = gclient_utils.SplitUrlRevision(self.url) 471 url, deps_revision = gclient_utils.SplitUrlRevision(self.url)
334 rev_str = "" 472 rev_str = ""
335 revision = deps_revision 473 revision = deps_revision
336 managed = True 474 managed = True
337 if options.revision: 475 if options.revision:
338 # Override the revision number. 476 # Override the revision number.
339 revision = str(options.revision) 477 revision = str(options.revision)
340 if revision == 'unmanaged': 478 if revision == 'unmanaged':
341 # Check again for a revision in case an initial ref was specified 479 # Check again for a revision in case an initial ref was specified
342 # in the url, for example bla.git@refs/heads/custombranch 480 # in the url, for example bla.git@refs/heads/custombranch
(...skipping 26 matching lines...) Expand all
369 if revision.startswith('refs/'): 507 if revision.startswith('refs/'):
370 rev_type = "branch" 508 rev_type = "branch"
371 elif revision.startswith(self.remote + '/'): 509 elif revision.startswith(self.remote + '/'):
372 # Rewrite remote refs to their local equivalents. 510 # Rewrite remote refs to their local equivalents.
373 revision = 'refs/remotes/' + revision 511 revision = 'refs/remotes/' + revision
374 rev_type = "branch" 512 rev_type = "branch"
375 else: 513 else:
376 # hash is also a tag, only make a distinction at checkout 514 # hash is also a tag, only make a distinction at checkout
377 rev_type = "hash" 515 rev_type = "hash"
378 516
517 refish = self.ParseRefish(revision)
379 mirror = self._GetMirror(url, options) 518 mirror = self._GetMirror(url, options)
380 if mirror: 519 if mirror:
381 url = mirror.mirror_path 520 url = mirror.mirror_path
382 521
383 if (not os.path.exists(self.checkout_path) or 522 if (not os.path.exists(self.checkout_path) or
384 (os.path.isdir(self.checkout_path) and 523 (os.path.isdir(self.checkout_path) and
385 not os.path.exists(os.path.join(self.checkout_path, '.git')))): 524 not os.path.exists(os.path.join(self.checkout_path, '.git')))):
386 if mirror: 525 if mirror:
387 self._UpdateMirror(mirror, options) 526 self._UpdateMirror(mirror, options)
388 try: 527 try:
389 self._Clone(revision, url, options) 528 self._Clone(refish.local_ref, url, options)
390 except subprocess2.CalledProcessError: 529 except subprocess2.CalledProcessError:
391 self._DeleteOrMove(options.force) 530 self._DeleteOrMove(options.force)
392 self._Clone(revision, url, options) 531 self._Clone(refish.local_ref, url, options)
393 if deps_revision and deps_revision.startswith('branch-heads/'): 532 if deps_revision and deps_revision.startswith('branch-heads/'):
394 deps_branch = deps_revision.replace('branch-heads/', '') 533 deps_branch = deps_revision.replace('branch-heads/', '')
395 self._Capture(['branch', deps_branch, deps_revision]) 534 self._Capture(['branch', deps_branch, deps_revision])
396 self._Checkout(options, deps_branch, quiet=True) 535 self._Checkout(options, deps_branch, quiet=True)
397 if file_list is not None: 536 if file_list is not None:
398 files = self._Capture(['ls-files']).splitlines() 537 files = self._Capture(['ls-files']).splitlines()
399 file_list.extend([os.path.join(self.checkout_path, f) for f in files]) 538 file_list.extend([os.path.join(self.checkout_path, f) for f in files])
400 if not verbose: 539 if not verbose:
401 # Make the output a little prettier. It's nice to have some whitespace 540 # Make the output a little prettier. It's nice to have some whitespace
402 # between projects when cloning. 541 # between projects when cloning.
(...skipping 20 matching lines...) Expand all
423 if (current_url.rstrip('/') != url.rstrip('/') and 562 if (current_url.rstrip('/') != url.rstrip('/') and
424 url != 'git://foo' and 563 url != 'git://foo' and
425 subprocess2.capture( 564 subprocess2.capture(
426 ['git', 'config', 'remote.%s.gclient-auto-fix-url' % self.remote], 565 ['git', 'config', 'remote.%s.gclient-auto-fix-url' % self.remote],
427 cwd=self.checkout_path).strip() != 'False'): 566 cwd=self.checkout_path).strip() != 'False'):
428 self.Print('_____ switching %s to a new upstream' % self.relpath) 567 self.Print('_____ switching %s to a new upstream' % self.relpath)
429 # Make sure it's clean 568 # Make sure it's clean
430 self._CheckClean(rev_str) 569 self._CheckClean(rev_str)
431 # Switch over to the new upstream 570 # Switch over to the new upstream
432 self._Run(['remote', 'set-url', self.remote, url], options) 571 self._Run(['remote', 'set-url', self.remote, url], options)
433 self._FetchAndReset(revision, file_list, options) 572 self._FetchAndReset(refish, file_list, options)
434 return_early = True 573 return_early = True
435 574
436 if return_early: 575 if return_early:
437 return self._Capture(['rev-parse', '--verify', 'HEAD']) 576 return self._Capture(['rev-parse', '--verify', 'HEAD'])
438 577
439 cur_branch = self._GetCurrentBranch() 578 cur_branch = self._GetCurrentBranch()
440 579
441 # Cases: 580 # Cases:
442 # 0) HEAD is detached. Probably from our initial clone. 581 # 0) HEAD is detached. Probably from our initial clone.
443 # - make sure HEAD is contained by a named ref, then update. 582 # - make sure HEAD is contained by a named ref, then update.
(...skipping 22 matching lines...) Expand all
466 upstream_branch = scm.GIT.GetUpstreamBranch(self.checkout_path) 605 upstream_branch = scm.GIT.GetUpstreamBranch(self.checkout_path)
467 if not upstream_branch or not upstream_branch.startswith('refs/remotes'): 606 if not upstream_branch or not upstream_branch.startswith('refs/remotes'):
468 current_type = "hash" 607 current_type = "hash"
469 logging.debug("Current branch is not tracking an upstream (remote)" 608 logging.debug("Current branch is not tracking an upstream (remote)"
470 " branch.") 609 " branch.")
471 elif upstream_branch.startswith('refs/remotes'): 610 elif upstream_branch.startswith('refs/remotes'):
472 current_type = "branch" 611 current_type = "branch"
473 else: 612 else:
474 raise gclient_utils.Error('Invalid Upstream: %s' % upstream_branch) 613 raise gclient_utils.Error('Invalid Upstream: %s' % upstream_branch)
475 614
476 if not scm.GIT.IsValidRevision(self.checkout_path, revision, sha_only=True): 615 if not scm.GIT.IsValidRevision(self.checkout_path, refish.local_ref,
616 sha_only=True):
477 # Update the remotes first so we have all the refs. 617 # Update the remotes first so we have all the refs.
478 remote_output = scm.GIT.Capture(['remote'] + verbose + ['update'], 618 remote_output = scm.GIT.Capture(['remote'] + verbose + ['update'],
479 cwd=self.checkout_path) 619 cwd=self.checkout_path)
480 if verbose: 620 if verbose:
481 self.Print(remote_output) 621 self.Print(remote_output)
482 622
483 self._UpdateBranchHeads(options, fetch=True) 623 self._UpdateBranchHeads(options, fetch=True)
484 624
485 # This is a big hammer, debatable if it should even be here... 625 # This is a big hammer, debatable if it should even be here...
486 if options.force or options.reset: 626 if options.force or options.reset:
487 target = 'HEAD' 627 target = 'HEAD'
488 if options.upstream and upstream_branch: 628 if options.upstream and upstream_branch:
489 target = upstream_branch 629 target = upstream_branch
490 self._Run(['reset', '--hard', target], options) 630 self._Run(['reset', '--hard', target], options)
491 631
492 if current_type == 'detached': 632 if current_type == 'detached':
493 # case 0 633 # case 0
494 self._CheckClean(rev_str) 634 self._CheckClean(rev_str)
495 self._CheckDetachedHead(rev_str, options) 635 self._CheckDetachedHead(rev_str, options)
496 if self._Capture(['rev-list', '-n', '1', 'HEAD']) == revision: 636 if self._Capture(['rev-list', '-n', '1', 'HEAD']) == refish.local_ref:
497 self.Print('Up-to-date; skipping checkout.') 637 self.Print('Up-to-date; skipping checkout.')
498 else: 638 else:
499 # 'git checkout' may need to overwrite existing untracked files. Allow 639 # 'git checkout' may need to overwrite existing untracked files. Allow
500 # it only when nuclear options are enabled. 640 # it only when nuclear options are enabled.
501 self._Checkout( 641 self._Checkout(
502 options, 642 options,
503 revision, 643 revision,
504 force=(options.force and options.delete_unversioned_trees), 644 force=(options.force and options.delete_unversioned_trees),
505 quiet=True, 645 quiet=True,
506 ) 646 )
507 if not printed_path: 647 if not printed_path:
508 self.Print('_____ %s%s' % (self.relpath, rev_str), timestamp=False) 648 self.Print('_____ %s%s' % (self.relpath, rev_str), timestamp=False)
509 elif current_type == 'hash': 649 elif current_type == 'hash':
510 # case 1 650 # case 1
511 if scm.GIT.IsGitSvn(self.checkout_path) and upstream_branch is not None: 651 if scm.GIT.IsGitSvn(self.checkout_path) and upstream_branch is not None:
512 # Our git-svn branch (upstream_branch) is our upstream 652 # Our git-svn branch (upstream_branch) is our upstream
513 self._AttemptRebase(upstream_branch, files, options, 653 self._AttemptRebase(upstream_branch, files, options,
514 newbase=revision, printed_path=printed_path, 654 newbase=refish.local_ref,
515 merge=options.merge) 655 printed_path=printed_path, merge=options.merge)
516 printed_path = True 656 printed_path = True
517 else: 657 else:
518 # Can't find a merge-base since we don't know our upstream. That makes 658 # Can't find a merge-base since we don't know our upstream. That makes
519 # this command VERY likely to produce a rebase failure. For now we 659 # this command VERY likely to produce a rebase failure. For now we
520 # assume origin is our upstream since that's what the old behavior was. 660 # assume origin is our upstream since that's what the old behavior was.
521 upstream_branch = self.remote 661 upstream_branch = self.remote
522 if options.revision or deps_revision: 662 if options.revision or deps_revision:
523 upstream_branch = revision 663 upstream_branch = refish.local_ref
524 self._AttemptRebase(upstream_branch, files, options, 664 self._AttemptRebase(upstream_branch, files, options,
525 printed_path=printed_path, merge=options.merge) 665 printed_path=printed_path, merge=options.merge)
526 printed_path = True 666 printed_path = True
527 elif rev_type == 'hash': 667 elif not refish.is_branch:
528 # case 2 668 # case 2
529 self._AttemptRebase(upstream_branch, files, options, 669 self._AttemptRebase(upstream_branch, files, options,
530 newbase=revision, printed_path=printed_path, 670 newbase=refish.local_ref, printed_path=printed_path,
531 merge=options.merge) 671 merge=options.merge)
532 printed_path = True 672 printed_path = True
533 elif revision.replace('heads', 'remotes/' + self.remote) != upstream_branch: 673 elif refish.upstream_branch != upstream_branch:
534 # case 4 674 # case 4
535 new_base = revision.replace('heads', 'remotes/' + self.remote) 675 new_base = refish.upstream_branch
536 if not printed_path: 676 if not printed_path:
537 self.Print('_____ %s%s' % (self.relpath, rev_str), timestamp=False) 677 self.Print('_____ %s%s' % (self.relpath, rev_str), timestamp=False)
538 switch_error = ("Switching upstream branch from %s to %s\n" 678 switch_error = ("Switching upstream branch from %s to %s\n"
539 % (upstream_branch, new_base) + 679 % (upstream_branch, new_base) +
540 "Please merge or rebase manually:\n" + 680 "Please merge or rebase manually:\n" +
541 "cd %s; git rebase %s\n" % (self.checkout_path, new_base) + 681 "cd %s; git rebase %s\n" % (self.checkout_path, new_base) +
542 "OR git checkout -b <some new branch> %s" % new_base) 682 "OR git checkout -b <some new branch> %s" % new_base)
543 raise gclient_utils.Error(switch_error) 683 raise gclient_utils.Error(switch_error)
544 else: 684 else:
545 # case 3 - the default case 685 # case 3 - the default case
(...skipping 110 matching lines...) Expand 10 before | Expand all | Expand 10 after
656 return self.update(options, [], file_list) 796 return self.update(options, [], file_list)
657 797
658 default_rev = "refs/heads/master" 798 default_rev = "refs/heads/master"
659 if options.upstream: 799 if options.upstream:
660 if self._GetCurrentBranch(): 800 if self._GetCurrentBranch():
661 upstream_branch = scm.GIT.GetUpstreamBranch(self.checkout_path) 801 upstream_branch = scm.GIT.GetUpstreamBranch(self.checkout_path)
662 default_rev = upstream_branch or default_rev 802 default_rev = upstream_branch or default_rev
663 _, deps_revision = gclient_utils.SplitUrlRevision(self.url) 803 _, deps_revision = gclient_utils.SplitUrlRevision(self.url)
664 if not deps_revision: 804 if not deps_revision:
665 deps_revision = default_rev 805 deps_revision = default_rev
666 if deps_revision.startswith('refs/heads/'): 806 refish = self.ParseRefish(deps_revision)
667 deps_revision = deps_revision.replace('refs/heads/', self.remote + '/') 807 deps_revision = self.GetUsableRev(refish.remote_refspec, options)
668 deps_revision = self.GetUsableRev(deps_revision, options)
669 808
670 if file_list is not None: 809 if file_list is not None:
671 files = self._Capture(['diff', deps_revision, '--name-only']).split() 810 files = self._Capture(['diff', deps_revision, '--name-only']).split()
672 811
673 self._Run(['reset', '--hard', deps_revision], options) 812 self._Run(['reset', '--hard', deps_revision], options)
674 self._Run(['clean', '-f', '-d'], options) 813 self._Run(['clean', '-f', '-d'], options)
675 814
676 if file_list is not None: 815 if file_list is not None:
677 file_list.extend([os.path.join(self.checkout_path, f) for f in files]) 816 file_list.extend([os.path.join(self.checkout_path, f) for f in files])
678 817
(...skipping 156 matching lines...) Expand 10 before | Expand all | Expand 10 after
835 gclient_utils.safe_rename(os.path.join(tmp_dir, '.git'), 974 gclient_utils.safe_rename(os.path.join(tmp_dir, '.git'),
836 os.path.join(self.checkout_path, '.git')) 975 os.path.join(self.checkout_path, '.git'))
837 except: 976 except:
838 traceback.print_exc(file=self.out_fh) 977 traceback.print_exc(file=self.out_fh)
839 raise 978 raise
840 finally: 979 finally:
841 if os.listdir(tmp_dir): 980 if os.listdir(tmp_dir):
842 self.Print('_____ removing non-empty tmp dir %s' % tmp_dir) 981 self.Print('_____ removing non-empty tmp dir %s' % tmp_dir)
843 gclient_utils.rmtree(tmp_dir) 982 gclient_utils.rmtree(tmp_dir)
844 self._UpdateBranchHeads(options, fetch=True) 983 self._UpdateBranchHeads(options, fetch=True)
845 self._Checkout(options, revision.replace('refs/heads/', ''), quiet=True) 984
985 refish = self.ParseRefish(revision)
986 self._Checkout(options, refish.local_ref, quiet=True)
846 if self._GetCurrentBranch() is None: 987 if self._GetCurrentBranch() is None:
847 # Squelch git's very verbose detached HEAD warning and use our own 988 # Squelch git's very verbose detached HEAD warning and use our own
848 self.Print( 989 self.Print(
849 ('Checked out %s to a detached HEAD. Before making any commits\n' 990 "Checked out %s to a detached HEAD. Before making any commits\n"
850 'in this repo, you should use \'git checkout <branch>\' to switch to\n' 991 "in this repo, you should use 'git checkout <branch>' to switch to\n"
851 'an existing branch or use \'git checkout %s -b <branch>\' to\n' 992 "an existing branch or use 'git checkout %s -b <branch>' to\n"
852 'create a new branch for your work.') % (revision, self.remote)) 993 "create a new branch for your work." % (revision, self.remote))
853 994
854 def _AskForData(self, prompt, options): 995 def _AskForData(self, prompt, options):
855 if options.jobs > 1: 996 if options.jobs > 1:
856 self.Print(prompt) 997 self.Print(prompt)
857 raise gclient_utils.Error("Background task requires input. Rerun " 998 raise gclient_utils.Error("Background task requires input. Rerun "
858 "gclient with --jobs=1 so that\n" 999 "gclient with --jobs=1 so that\n"
859 "interaction is possible.") 1000 "interaction is possible.")
860 try: 1001 try:
861 return raw_input(prompt) 1002 return raw_input(prompt)
862 except KeyboardInterrupt: 1003 except KeyboardInterrupt:
(...skipping 676 matching lines...) Expand 10 before | Expand all | Expand 10 after
1539 new_command.append('--force') 1680 new_command.append('--force')
1540 if command[0] != 'checkout' and scm.SVN.AssertVersion('1.6')[0]: 1681 if command[0] != 'checkout' and scm.SVN.AssertVersion('1.6')[0]:
1541 new_command.extend(('--accept', 'theirs-conflict')) 1682 new_command.extend(('--accept', 'theirs-conflict'))
1542 elif options.manually_grab_svn_rev: 1683 elif options.manually_grab_svn_rev:
1543 new_command.append('--force') 1684 new_command.append('--force')
1544 if command[0] != 'checkout' and scm.SVN.AssertVersion('1.6')[0]: 1685 if command[0] != 'checkout' and scm.SVN.AssertVersion('1.6')[0]:
1545 new_command.extend(('--accept', 'postpone')) 1686 new_command.extend(('--accept', 'postpone'))
1546 elif command[0] != 'checkout' and scm.SVN.AssertVersion('1.6')[0]: 1687 elif command[0] != 'checkout' and scm.SVN.AssertVersion('1.6')[0]:
1547 new_command.extend(('--accept', 'postpone')) 1688 new_command.extend(('--accept', 'postpone'))
1548 return new_command 1689 return new_command
OLDNEW
« no previous file with comments | « gclient.py ('k') | tests/gclient_scm_test.py » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698