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

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/ comments Created 6 years, 5 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
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
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 """Parses a ref into a 'GitRefish' instance."""
M-A Ruel 2014/06/25 20:54:57 I'd like a small description of each argument, not
dnj 2014/06/25 21:48:31 Done.
271 # Use default set of remotes
272 remote = remote or cls._DEFAULT_REMOTE
273 remotes = {remote}
274 if other_remotes:
275 remotes.update(other_remotes)
276
277 ref = ref.strip()
M-A Ruel 2014/06/25 20:54:57 optional: Another option is to assert ref == ref.s
dnj 2014/06/25 21:48:31 I like this better. Done.
278 # Treat 'ref' as a '/'-delimited set of items; analyze their contents
279 ref_split = local_ref = remote_ref = tuple(ref.split('/'))
280 is_branch = True
281 if len(ref_split) > 1 and ref_split[:1] == ('refs',):
282 if len(ref_split) >= 3 and ref_split[1] == 'heads':
283 # refs/heads/foo/bar
284 #
285 # Local Ref: foo/bar
286 # Remote Ref: foo/bar
287 local_ref = remote_ref = ref_split[2:]
288 elif len(ref_split) >= 4 and ref_split[1] == 'remotes':
289 # refs/remotes/<REMOTE>/foo/bar
290 # This is a bad assumption, and this logic can be improved, but it
291 # is consistent with traditional 'gclient' logic.
292 #
293 # Local Ref: refs/remotes/<REMOTE>/foo/bar
294 # Remote: <REMOTE>
295 # Remote Ref: foo/bar
296 remote = ref_split[2]
297 remote_ref = ref_split[3:]
M-A Ruel 2014/06/25 20:54:57 Why not? remote_ref = '/'.join(ref_split[3:])
dnj 2014/06/25 21:48:31 I'm keeping the refs as tuples for easier handling
298 else:
299 # If it's not part of the standard branch set, assume it's a remote
300 # branch.
301 #
302 # refs/foo/bar
303 #
304 # Local Ref: refs/foo/bar
305 # Remote ref: refs/foo/bar
306 pass
307 elif len(ref_split) >= 2 and ref_split[0] in remotes:
308 # origin/master
309 #
310 # Local Ref: refs/remotes/origin/master
311 # Remote Ref: master
312 remote = ref_split[0]
313 remote_ref = ref_split[1:]
314 local_ref = ('refs', 'remotes') + ref_split
315 else:
316 # It could be a hash, a short-hash, or a tag
317 is_branch = False
M-A Ruel 2014/06/25 20:54:57 What about? git checkout -b foo/bar origin/master
dnj 2014/06/25 21:48:31 The current implementation intended to reflect the
318
319 # Determine how the ref should be referenced remotely
320 if is_branch:
321 # foo/bar/baz => origin/foo/bar/baz
322 remote_refspec = (remote,) + remote_ref
323 else:
324 # Refer to the hash/tag directly. This is actually not allowed in
325 # 'fetch', although it oftentimes works regardless.
326 #
327 # <HASH/TAG> => <HASH/TAG>
328 remote_refspec = (ref,)
329
330 # Calculate the upstream branch
331 if is_branch:
332 # Convert '/refs/heads/...' to 'refs/remotes/REMOTE/...'
333 if ref_split[:2] == ('refs', 'heads'):
334 upstream_branch = ('refs', 'remotes', remote) + ref_split[2:]
335 else:
336 upstream_branch = ref_split
337 else:
338 upstream_branch = None
339
340 def compose(ref_tuple):
341 return '/'.join(ref_tuple) if ref_tuple else ref_tuple
342
343 return cls(
344 source=ref,
345 is_branch=is_branch,
346 local_ref=compose(local_ref),
347 remote=remote,
348 remote_ref=compose(remote_ref),
349 remote_refspec=compose(remote_refspec),
350 upstream_branch=compose(upstream_branch),
351 )
352
353
233 class GitWrapper(SCMWrapper): 354 class GitWrapper(SCMWrapper):
234 """Wrapper for Git""" 355 """Wrapper for Git"""
235 name = 'git' 356 name = 'git'
236 remote = 'origin' 357 remote = 'origin'
237 358
238 cache_dir = None 359 cache_dir = None
239 360
240 def __init__(self, url=None, *args): 361 def __init__(self, url=None, *args):
241 """Removes 'git+' fake prefix from git URL.""" 362 """Removes 'git+' fake prefix from git URL."""
242 if url.startswith('git+http://') or url.startswith('git+https://'): 363 if url.startswith('git+http://') or url.startswith('git+https://'):
243 url = url[4:] 364 url = url[4:]
244 SCMWrapper.__init__(self, url, *args) 365 SCMWrapper.__init__(self, url, *args)
245 filter_kwargs = { 'time_throttle': 1, 'out_fh': self.out_fh } 366 filter_kwargs = { 'time_throttle': 1, 'out_fh': self.out_fh }
246 if self.out_cb: 367 if self.out_cb:
247 filter_kwargs['predicate'] = self.out_cb 368 filter_kwargs['predicate'] = self.out_cb
248 self.filter = gclient_utils.GitFilter(**filter_kwargs) 369 self.filter = gclient_utils.GitFilter(**filter_kwargs)
249 370
250 @staticmethod 371 @staticmethod
251 def BinaryExists(): 372 def BinaryExists():
252 """Returns true if the command exists.""" 373 """Returns true if the command exists."""
253 try: 374 try:
254 # We assume git is newer than 1.7. See: crbug.com/114483 375 # We assume git is newer than 1.7. See: crbug.com/114483
255 result, version = scm.GIT.AssertVersion('1.7') 376 result, version = scm.GIT.AssertVersion('1.7')
256 if not result: 377 if not result:
257 raise gclient_utils.Error('Git version is older than 1.7: %s' % version) 378 raise gclient_utils.Error('Git version is older than 1.7: %s' % version)
258 return result 379 return result
259 except OSError: 380 except OSError:
260 return False 381 return False
261 382
383 def ParseRefish(self, value, **kwargs):
384 kwargs.setdefault('remote', self.remote)
385 return GitRefish.Parse(value, **kwargs)
386
262 def GetCheckoutRoot(self): 387 def GetCheckoutRoot(self):
263 return scm.GIT.GetCheckoutRoot(self.checkout_path) 388 return scm.GIT.GetCheckoutRoot(self.checkout_path)
264 389
265 def GetRevisionDate(self, _revision): 390 def GetRevisionDate(self, _revision):
266 """Returns the given revision's date in ISO-8601 format (which contains the 391 """Returns the given revision's date in ISO-8601 format (which contains the
267 time zone).""" 392 time zone)."""
268 # TODO(floitsch): get the time-stamp of the given revision and not just the 393 # TODO(floitsch): get the time-stamp of the given revision and not just the
269 # time-stamp of the currently checked out revision. 394 # time-stamp of the currently checked out revision.
270 return self._Capture(['log', '-n', '1', '--format=%ai']) 395 return self._Capture(['log', '-n', '1', '--format=%ai'])
271 396
(...skipping 14 matching lines...) Expand all
286 411
287 The patch file is generated from a diff of the merge base of HEAD and 412 The patch file is generated from a diff of the merge base of HEAD and
288 its upstream branch. 413 its upstream branch.
289 """ 414 """
290 merge_base = self._Capture(['merge-base', 'HEAD', self.remote]) 415 merge_base = self._Capture(['merge-base', 'HEAD', self.remote])
291 gclient_utils.CheckCallAndFilter( 416 gclient_utils.CheckCallAndFilter(
292 ['git', 'diff', merge_base], 417 ['git', 'diff', merge_base],
293 cwd=self.checkout_path, 418 cwd=self.checkout_path,
294 filter_fn=GitDiffFilterer(self.relpath).Filter, print_func=self.Print) 419 filter_fn=GitDiffFilterer(self.relpath).Filter, print_func=self.Print)
295 420
296 def _FetchAndReset(self, revision, file_list, options): 421 def _FetchAndReset(self, refish, file_list, options):
297 """Equivalent to git fetch; git reset.""" 422 """Equivalent to git fetch; git reset."""
298 quiet = [] 423 quiet = []
299 if not options.verbose: 424 if not options.verbose:
300 quiet = ['--quiet'] 425 quiet = ['--quiet']
301 self._UpdateBranchHeads(options, fetch=False) 426 self._UpdateBranchHeads(options, fetch=False)
302 427
303 self._Fetch(options, prune=True, quiet=options.verbose) 428 self._Fetch(options, prune=True, quiet=options.verbose)
304 self._Run(['reset', '--hard', revision] + quiet, options) 429 self._Run(['reset', '--hard', refish.local_ref] + quiet, options)
305 if file_list is not None: 430 if file_list is not None:
306 files = self._Capture(['ls-files']).splitlines() 431 files = self._Capture(['ls-files']).splitlines()
307 file_list.extend([os.path.join(self.checkout_path, f) for f in files]) 432 file_list.extend([os.path.join(self.checkout_path, f) for f in files])
308 433
309 def _DisableHooks(self): 434 def _DisableHooks(self):
310 hook_dir = os.path.join(self.checkout_path, '.git', 'hooks') 435 hook_dir = os.path.join(self.checkout_path, '.git', 'hooks')
311 if not os.path.isdir(hook_dir): 436 if not os.path.isdir(hook_dir):
312 return 437 return
313 for f in os.listdir(hook_dir): 438 for f in os.listdir(hook_dir):
314 if not f.endswith('.sample') and not f.endswith('.disabled'): 439 if not f.endswith('.sample') and not f.endswith('.disabled'):
315 os.rename(os.path.join(hook_dir, f), 440 os.rename(os.path.join(hook_dir, f),
316 os.path.join(hook_dir, f + '.disabled')) 441 os.path.join(hook_dir, f + '.disabled'))
317 442
318 def update(self, options, args, file_list): 443 def update(self, options, args, file_list):
319 """Runs git to update or transparently checkout the working copy. 444 """Runs git to update or transparently checkout the working copy.
320 445
321 All updated files will be appended to file_list. 446 All updated files will be appended to file_list.
322 447
323 Raises: 448 Raises:
324 Error: if can't get URL for relative path. 449 Error: if can't get URL for relative path.
325 """ 450 """
326 if args: 451 if args:
327 raise gclient_utils.Error("Unsupported argument(s): %s" % ",".join(args)) 452 raise gclient_utils.Error("Unsupported argument(s): %s" % ",".join(args))
328 453
329 self._CheckMinVersion("1.6.6") 454 self._CheckMinVersion("1.6.6")
330 455
331 # If a dependency is not pinned, track the default remote branch. 456 # If a dependency is not pinned, track the default remote branch.
332 default_rev = 'refs/remotes/%s/master' % self.remote 457 default_rev = 'refs/remotes/%s/master' % (self.remote,)
333 url, deps_revision = gclient_utils.SplitUrlRevision(self.url) 458 url, deps_revision = gclient_utils.SplitUrlRevision(self.url)
334 rev_str = "" 459 rev_str = ""
335 revision = deps_revision 460 revision = deps_revision
336 managed = True 461 managed = True
337 if options.revision: 462 if options.revision:
338 # Override the revision number. 463 # Override the revision number.
339 revision = str(options.revision) 464 revision = str(options.revision)
340 if revision == 'unmanaged': 465 if revision == 'unmanaged':
341 # Check again for a revision in case an initial ref was specified 466 # Check again for a revision in case an initial ref was specified
342 # in the url, for example bla.git@refs/heads/custombranch 467 # in the url, for example bla.git@refs/heads/custombranch
(...skipping 26 matching lines...) Expand all
369 if revision.startswith('refs/'): 494 if revision.startswith('refs/'):
370 rev_type = "branch" 495 rev_type = "branch"
371 elif revision.startswith(self.remote + '/'): 496 elif revision.startswith(self.remote + '/'):
372 # Rewrite remote refs to their local equivalents. 497 # Rewrite remote refs to their local equivalents.
373 revision = 'refs/remotes/' + revision 498 revision = 'refs/remotes/' + revision
374 rev_type = "branch" 499 rev_type = "branch"
375 else: 500 else:
376 # hash is also a tag, only make a distinction at checkout 501 # hash is also a tag, only make a distinction at checkout
377 rev_type = "hash" 502 rev_type = "hash"
378 503
504 refish = self.ParseRefish(revision)
379 mirror = self._GetMirror(url, options) 505 mirror = self._GetMirror(url, options)
380 if mirror: 506 if mirror:
381 url = mirror.mirror_path 507 url = mirror.mirror_path
382 508
383 if (not os.path.exists(self.checkout_path) or 509 if (not os.path.exists(self.checkout_path) or
384 (os.path.isdir(self.checkout_path) and 510 (os.path.isdir(self.checkout_path) and
385 not os.path.exists(os.path.join(self.checkout_path, '.git')))): 511 not os.path.exists(os.path.join(self.checkout_path, '.git')))):
386 if mirror: 512 if mirror:
387 self._UpdateMirror(mirror, options) 513 self._UpdateMirror(mirror, options)
388 try: 514 try:
389 self._Clone(revision, url, options) 515 self._Clone(refish.local_ref, url, options)
390 except subprocess2.CalledProcessError: 516 except subprocess2.CalledProcessError:
391 self._DeleteOrMove(options.force) 517 self._DeleteOrMove(options.force)
392 self._Clone(revision, url, options) 518 self._Clone(refish.local_ref, url, options)
393 if deps_revision and deps_revision.startswith('branch-heads/'): 519 if deps_revision and deps_revision.startswith('branch-heads/'):
394 deps_branch = deps_revision.replace('branch-heads/', '') 520 deps_branch = deps_revision.replace('branch-heads/', '')
395 self._Capture(['branch', deps_branch, deps_revision]) 521 self._Capture(['branch', deps_branch, deps_revision])
396 self._Checkout(options, deps_branch, quiet=True) 522 self._Checkout(options, deps_branch, quiet=True)
397 if file_list is not None: 523 if file_list is not None:
398 files = self._Capture(['ls-files']).splitlines() 524 files = self._Capture(['ls-files']).splitlines()
399 file_list.extend([os.path.join(self.checkout_path, f) for f in files]) 525 file_list.extend([os.path.join(self.checkout_path, f) for f in files])
400 if not verbose: 526 if not verbose:
401 # Make the output a little prettier. It's nice to have some whitespace 527 # Make the output a little prettier. It's nice to have some whitespace
402 # between projects when cloning. 528 # between projects when cloning.
(...skipping 20 matching lines...) Expand all
423 if (current_url.rstrip('/') != url.rstrip('/') and 549 if (current_url.rstrip('/') != url.rstrip('/') and
424 url != 'git://foo' and 550 url != 'git://foo' and
425 subprocess2.capture( 551 subprocess2.capture(
426 ['git', 'config', 'remote.%s.gclient-auto-fix-url' % self.remote], 552 ['git', 'config', 'remote.%s.gclient-auto-fix-url' % self.remote],
427 cwd=self.checkout_path).strip() != 'False'): 553 cwd=self.checkout_path).strip() != 'False'):
428 self.Print('_____ switching %s to a new upstream' % self.relpath) 554 self.Print('_____ switching %s to a new upstream' % self.relpath)
429 # Make sure it's clean 555 # Make sure it's clean
430 self._CheckClean(rev_str) 556 self._CheckClean(rev_str)
431 # Switch over to the new upstream 557 # Switch over to the new upstream
432 self._Run(['remote', 'set-url', self.remote, url], options) 558 self._Run(['remote', 'set-url', self.remote, url], options)
433 self._FetchAndReset(revision, file_list, options) 559 self._FetchAndReset(refish, file_list, options)
434 return_early = True 560 return_early = True
435 561
436 if return_early: 562 if return_early:
437 return self._Capture(['rev-parse', '--verify', 'HEAD']) 563 return self._Capture(['rev-parse', '--verify', 'HEAD'])
438 564
439 cur_branch = self._GetCurrentBranch() 565 cur_branch = self._GetCurrentBranch()
440 566
441 # Cases: 567 # Cases:
442 # 0) HEAD is detached. Probably from our initial clone. 568 # 0) HEAD is detached. Probably from our initial clone.
443 # - make sure HEAD is contained by a named ref, then update. 569 # - 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) 592 upstream_branch = scm.GIT.GetUpstreamBranch(self.checkout_path)
467 if not upstream_branch or not upstream_branch.startswith('refs/remotes'): 593 if not upstream_branch or not upstream_branch.startswith('refs/remotes'):
468 current_type = "hash" 594 current_type = "hash"
469 logging.debug("Current branch is not tracking an upstream (remote)" 595 logging.debug("Current branch is not tracking an upstream (remote)"
470 " branch.") 596 " branch.")
471 elif upstream_branch.startswith('refs/remotes'): 597 elif upstream_branch.startswith('refs/remotes'):
472 current_type = "branch" 598 current_type = "branch"
473 else: 599 else:
474 raise gclient_utils.Error('Invalid Upstream: %s' % upstream_branch) 600 raise gclient_utils.Error('Invalid Upstream: %s' % upstream_branch)
475 601
476 if not scm.GIT.IsValidRevision(self.checkout_path, revision, sha_only=True): 602 if not scm.GIT.IsValidRevision(self.checkout_path, refish.local_ref,
603 sha_only=True):
477 # Update the remotes first so we have all the refs. 604 # Update the remotes first so we have all the refs.
478 remote_output = scm.GIT.Capture(['remote'] + verbose + ['update'], 605 remote_output = scm.GIT.Capture(['remote'] + verbose + ['update'],
479 cwd=self.checkout_path) 606 cwd=self.checkout_path)
480 if verbose: 607 if verbose:
481 self.Print(remote_output) 608 self.Print(remote_output)
482 609
483 self._UpdateBranchHeads(options, fetch=True) 610 self._UpdateBranchHeads(options, fetch=True)
484 611
485 # This is a big hammer, debatable if it should even be here... 612 # This is a big hammer, debatable if it should even be here...
486 if options.force or options.reset: 613 if options.force or options.reset:
487 target = 'HEAD' 614 target = 'HEAD'
488 if options.upstream and upstream_branch: 615 if options.upstream and upstream_branch:
489 target = upstream_branch 616 target = upstream_branch
490 self._Run(['reset', '--hard', target], options) 617 self._Run(['reset', '--hard', target], options)
491 618
492 if current_type == 'detached': 619 if current_type == 'detached':
493 # case 0 620 # case 0
494 self._CheckClean(rev_str) 621 self._CheckClean(rev_str)
495 self._CheckDetachedHead(rev_str, options) 622 self._CheckDetachedHead(rev_str, options)
496 if self._Capture(['rev-list', '-n', '1', 'HEAD']) == revision: 623 if self._Capture(['rev-list', '-n', '1', 'HEAD']) == refish.local_ref:
497 self.Print('Up-to-date; skipping checkout.') 624 self.Print('Up-to-date; skipping checkout.')
498 else: 625 else:
499 # 'git checkout' may need to overwrite existing untracked files. Allow 626 # 'git checkout' may need to overwrite existing untracked files. Allow
500 # it only when nuclear options are enabled. 627 # it only when nuclear options are enabled.
501 self._Checkout( 628 self._Checkout(
502 options, 629 options,
503 revision, 630 revision,
504 force=(options.force and options.delete_unversioned_trees), 631 force=(options.force and options.delete_unversioned_trees),
505 quiet=True, 632 quiet=True,
506 ) 633 )
507 if not printed_path: 634 if not printed_path:
508 self.Print('_____ %s%s' % (self.relpath, rev_str), timestamp=False) 635 self.Print('_____ %s%s' % (self.relpath, rev_str), timestamp=False)
509 elif current_type == 'hash': 636 elif current_type == 'hash':
510 # case 1 637 # case 1
511 if scm.GIT.IsGitSvn(self.checkout_path) and upstream_branch is not None: 638 if scm.GIT.IsGitSvn(self.checkout_path) and upstream_branch is not None:
512 # Our git-svn branch (upstream_branch) is our upstream 639 # Our git-svn branch (upstream_branch) is our upstream
513 self._AttemptRebase(upstream_branch, files, options, 640 self._AttemptRebase(upstream_branch, files, options,
514 newbase=revision, printed_path=printed_path, 641 newbase=refish.local_ref,
515 merge=options.merge) 642 printed_path=printed_path, merge=options.merge)
516 printed_path = True 643 printed_path = True
517 else: 644 else:
518 # Can't find a merge-base since we don't know our upstream. That makes 645 # 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 646 # 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. 647 # assume origin is our upstream since that's what the old behavior was.
521 upstream_branch = self.remote 648 upstream_branch = self.remote
522 if options.revision or deps_revision: 649 if options.revision or deps_revision:
523 upstream_branch = revision 650 upstream_branch = refish.local_ref
524 self._AttemptRebase(upstream_branch, files, options, 651 self._AttemptRebase(upstream_branch, files, options,
525 printed_path=printed_path, merge=options.merge) 652 printed_path=printed_path, merge=options.merge)
526 printed_path = True 653 printed_path = True
527 elif rev_type == 'hash': 654 elif not refish.is_branch:
528 # case 2 655 # case 2
529 self._AttemptRebase(upstream_branch, files, options, 656 self._AttemptRebase(upstream_branch, files, options,
530 newbase=revision, printed_path=printed_path, 657 newbase=refish.local_ref, printed_path=printed_path,
531 merge=options.merge) 658 merge=options.merge)
532 printed_path = True 659 printed_path = True
533 elif revision.replace('heads', 'remotes/' + self.remote) != upstream_branch: 660 elif refish.upstream_branch != upstream_branch:
534 # case 4 661 # case 4
535 new_base = revision.replace('heads', 'remotes/' + self.remote) 662 new_base = refish.upstream_branch
536 if not printed_path: 663 if not printed_path:
537 self.Print('_____ %s%s' % (self.relpath, rev_str), timestamp=False) 664 self.Print('_____ %s%s' % (self.relpath, rev_str), timestamp=False)
538 switch_error = ("Switching upstream branch from %s to %s\n" 665 switch_error = ("Switching upstream branch from %s to %s\n"
539 % (upstream_branch, new_base) + 666 % (upstream_branch, new_base) +
540 "Please merge or rebase manually:\n" + 667 "Please merge or rebase manually:\n" +
541 "cd %s; git rebase %s\n" % (self.checkout_path, new_base) + 668 "cd %s; git rebase %s\n" % (self.checkout_path, new_base) +
542 "OR git checkout -b <some new branch> %s" % new_base) 669 "OR git checkout -b <some new branch> %s" % new_base)
543 raise gclient_utils.Error(switch_error) 670 raise gclient_utils.Error(switch_error)
544 else: 671 else:
545 # case 3 - the default case 672 # case 3 - the default case
(...skipping 110 matching lines...) Expand 10 before | Expand all | Expand 10 after
656 return self.update(options, [], file_list) 783 return self.update(options, [], file_list)
657 784
658 default_rev = "refs/heads/master" 785 default_rev = "refs/heads/master"
659 if options.upstream: 786 if options.upstream:
660 if self._GetCurrentBranch(): 787 if self._GetCurrentBranch():
661 upstream_branch = scm.GIT.GetUpstreamBranch(self.checkout_path) 788 upstream_branch = scm.GIT.GetUpstreamBranch(self.checkout_path)
662 default_rev = upstream_branch or default_rev 789 default_rev = upstream_branch or default_rev
663 _, deps_revision = gclient_utils.SplitUrlRevision(self.url) 790 _, deps_revision = gclient_utils.SplitUrlRevision(self.url)
664 if not deps_revision: 791 if not deps_revision:
665 deps_revision = default_rev 792 deps_revision = default_rev
666 if deps_revision.startswith('refs/heads/'): 793 refish = self.ParseRefish(deps_revision)
667 deps_revision = deps_revision.replace('refs/heads/', self.remote + '/') 794 deps_revision = self.GetUsableRev(refish.remote_refspec, options)
668 deps_revision = self.GetUsableRev(deps_revision, options)
669 795
670 if file_list is not None: 796 if file_list is not None:
671 files = self._Capture(['diff', deps_revision, '--name-only']).split() 797 files = self._Capture(['diff', deps_revision, '--name-only']).split()
672 798
673 self._Run(['reset', '--hard', deps_revision], options) 799 self._Run(['reset', '--hard', deps_revision], options)
674 self._Run(['clean', '-f', '-d'], options) 800 self._Run(['clean', '-f', '-d'], options)
675 801
676 if file_list is not None: 802 if file_list is not None:
677 file_list.extend([os.path.join(self.checkout_path, f) for f in files]) 803 file_list.extend([os.path.join(self.checkout_path, f) for f in files])
678 804
(...skipping 156 matching lines...) Expand 10 before | Expand all | Expand 10 after
835 gclient_utils.safe_rename(os.path.join(tmp_dir, '.git'), 961 gclient_utils.safe_rename(os.path.join(tmp_dir, '.git'),
836 os.path.join(self.checkout_path, '.git')) 962 os.path.join(self.checkout_path, '.git'))
837 except: 963 except:
838 traceback.print_exc(file=self.out_fh) 964 traceback.print_exc(file=self.out_fh)
839 raise 965 raise
840 finally: 966 finally:
841 if os.listdir(tmp_dir): 967 if os.listdir(tmp_dir):
842 self.Print('_____ removing non-empty tmp dir %s' % tmp_dir) 968 self.Print('_____ removing non-empty tmp dir %s' % tmp_dir)
843 gclient_utils.rmtree(tmp_dir) 969 gclient_utils.rmtree(tmp_dir)
844 self._UpdateBranchHeads(options, fetch=True) 970 self._UpdateBranchHeads(options, fetch=True)
845 self._Checkout(options, revision.replace('refs/heads/', ''), quiet=True) 971
972 refish = self.ParseRefish(revision)
973 self._Checkout(options, refish.local_ref, quiet=True)
846 if self._GetCurrentBranch() is None: 974 if self._GetCurrentBranch() is None:
847 # Squelch git's very verbose detached HEAD warning and use our own 975 # Squelch git's very verbose detached HEAD warning and use our own
848 self.Print( 976 self.Print(
849 ('Checked out %s to a detached HEAD. Before making any commits\n' 977 "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' 978 "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' 979 "an existing branch or use 'git checkout %s -b <branch>' to\n"
852 'create a new branch for your work.') % (revision, self.remote)) 980 "create a new branch for your work." % (revision, self.remote))
853 981
854 def _AskForData(self, prompt, options): 982 def _AskForData(self, prompt, options):
855 if options.jobs > 1: 983 if options.jobs > 1:
856 self.Print(prompt) 984 self.Print(prompt)
857 raise gclient_utils.Error("Background task requires input. Rerun " 985 raise gclient_utils.Error("Background task requires input. Rerun "
858 "gclient with --jobs=1 so that\n" 986 "gclient with --jobs=1 so that\n"
859 "interaction is possible.") 987 "interaction is possible.")
860 try: 988 try:
861 return raw_input(prompt) 989 return raw_input(prompt)
862 except KeyboardInterrupt: 990 except KeyboardInterrupt:
(...skipping 676 matching lines...) Expand 10 before | Expand all | Expand 10 after
1539 new_command.append('--force') 1667 new_command.append('--force')
1540 if command[0] != 'checkout' and scm.SVN.AssertVersion('1.6')[0]: 1668 if command[0] != 'checkout' and scm.SVN.AssertVersion('1.6')[0]:
1541 new_command.extend(('--accept', 'theirs-conflict')) 1669 new_command.extend(('--accept', 'theirs-conflict'))
1542 elif options.manually_grab_svn_rev: 1670 elif options.manually_grab_svn_rev:
1543 new_command.append('--force') 1671 new_command.append('--force')
1544 if command[0] != 'checkout' and scm.SVN.AssertVersion('1.6')[0]: 1672 if command[0] != 'checkout' and scm.SVN.AssertVersion('1.6')[0]:
1545 new_command.extend(('--accept', 'postpone')) 1673 new_command.extend(('--accept', 'postpone'))
1546 elif command[0] != 'checkout' and scm.SVN.AssertVersion('1.6')[0]: 1674 elif command[0] != 'checkout' and scm.SVN.AssertVersion('1.6')[0]:
1547 new_command.extend(('--accept', 'postpone')) 1675 new_command.extend(('--accept', 'postpone'))
1548 return new_command 1676 return new_command
OLDNEW
« no previous file with comments | « gclient.py ('k') | tests/gclient_scm_test.py » ('j') | tests/gclient_scm_test.py » ('J')

Powered by Google App Engine
This is Rietveld 408576698