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

Side by Side Diff: gclient_scm.py

Issue 19359002: Allow gclient clone in non-empty directories (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/tools/depot_tools
Patch Set: Created 7 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
« no previous file with comments | « gclient.py ('k') | no next file » | 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 import collections 7 import collections
8 import logging 8 import logging
9 import os 9 import os
10 import posixpath 10 import posixpath
11 import re 11 import re
12 import sys 12 import sys
13 import tempfile
13 import threading 14 import threading
14 import time 15 import time
15 16
16 import gclient_utils 17 import gclient_utils
17 import scm 18 import scm
18 import subprocess2 19 import subprocess2
19 20
20 21
21 THIS_FILE_PATH = os.path.abspath(__file__) 22 THIS_FILE_PATH = os.path.abspath(__file__)
22 23
(...skipping 320 matching lines...) Expand 10 before | Expand all | Expand 10 after
343 if revision.startswith('refs/'): 344 if revision.startswith('refs/'):
344 rev_type = "branch" 345 rev_type = "branch"
345 elif revision.startswith('origin/'): 346 elif revision.startswith('origin/'):
346 # For compatability with old naming, translate 'origin' to 'refs/heads' 347 # For compatability with old naming, translate 'origin' to 'refs/heads'
347 revision = revision.replace('origin/', 'refs/heads/') 348 revision = revision.replace('origin/', 'refs/heads/')
348 rev_type = "branch" 349 rev_type = "branch"
349 else: 350 else:
350 # hash is also a tag, only make a distinction at checkout 351 # hash is also a tag, only make a distinction at checkout
351 rev_type = "hash" 352 rev_type = "hash"
352 353
353 if not os.path.exists(self.checkout_path) or ( 354 def PostCloneSteps():
354 os.path.isdir(self.checkout_path) and
355 not os.listdir(self.checkout_path)):
356 gclient_utils.safe_makedirs(os.path.dirname(self.checkout_path))
357 self._Clone(revision, url, options)
358 self.UpdateSubmoduleConfig() 355 self.UpdateSubmoduleConfig()
359 if file_list is not None: 356 if file_list is not None:
360 files = self._Capture(['ls-files']).splitlines() 357 files = self._Capture(['ls-files']).splitlines()
361 file_list.extend([os.path.join(self.checkout_path, f) for f in files]) 358 file_list.extend([os.path.join(self.checkout_path, f) for f in files])
362 if not verbose: 359 if not verbose:
363 # Make the output a little prettier. It's nice to have some whitespace 360 # Make the output a little prettier. It's nice to have some whitespace
364 # between projects when cloning. 361 # between projects when cloning.
365 print('') 362 print('')
366 return 363
364 if (os.path.exists(self.checkout_path) and
365 getattr(options, 'stage_clone', None) and
366 not os.path.exists(os.path.join(self.checkout_path, '.git'))):
367 parent_dir = os.path.dirname(self.checkout_path)
368 gclient_utils.safe_makedirs(parent_dir)
369 try:
370 tmp_dir = tempfile.mkdtemp(
371 suffix='_%s' % os.path.basename(self.checkout_path),
372 dir=parent_dir)
373 self._Clone(revision, url, options, clone_path=tmp_dir)
374 src_files = os.listdir(tmp_dir)
375 overlap = set(src_files).intersection(os.listdir(self.checkout_path))
376 if overlap:
377 raise gclient_utils.Error(
378 "%s clone failure. Files exist: %s" % (
379 self.checkout_path, sorted(overlap)))
380 print('\n_____ copying %s -> %s' % (tmp_dir, self.checkout_path))
381 for src_file in src_files:
382 os.rename(os.path.join(tmp_dir, src_file),
383 os.path.join(self.checkout_path, src_file))
384 return PostCloneSteps()
385 finally:
386 if os.path.exists(tmp_dir):
387 if os.listdir(tmp_dir):
388 print('\n_____ removing non-empty tmp dir %s' % tmp_dir)
389 gclient_utils.rmtree(tmp_dir)
390 elif not os.path.exists(self.checkout_path) or (
391 os.path.isdir(self.checkout_path) and
392 not os.listdir(self.checkout_path)):
393 gclient_utils.safe_makedirs(os.path.dirname(self.checkout_path))
394 self._Clone(revision, url, options)
395 return PostCloneSteps()
367 396
368 if not managed: 397 if not managed:
369 self._UpdateBranchHeads(options, fetch=False) 398 self._UpdateBranchHeads(options, fetch=False)
370 self.UpdateSubmoduleConfig() 399 self.UpdateSubmoduleConfig()
371 print ('________ unmanaged solution; skipping %s' % self.relpath) 400 print ('________ unmanaged solution; skipping %s' % self.relpath)
372 return 401 return
373 402
374 if not os.path.exists(os.path.join(self.checkout_path, '.git')): 403 if not os.path.exists(os.path.join(self.checkout_path, '.git')):
375 raise gclient_utils.Error('\n____ %s%s\n' 404 raise gclient_utils.Error('\n____ %s%s\n'
376 '\tPath is not a git repo. No .git dir.\n' 405 '\tPath is not a git repo. No .git dir.\n'
(...skipping 28 matching lines...) Expand all
405 print('_____ %s/.git is corrupted, rebuilding' % self.relpath) 434 print('_____ %s/.git is corrupted, rebuilding' % self.relpath)
406 self._Run(['init'], options) 435 self._Run(['init'], options)
407 self._Run(['remote', 'set-url', 'origin', url], options) 436 self._Run(['remote', 'set-url', 'origin', url], options)
408 437
409 if not self._HasHead(): 438 if not self._HasHead():
410 # Previous checkout was aborted before branches could be created in repo, 439 # Previous checkout was aborted before branches could be created in repo,
411 # so we need to reconstruct them here. 440 # so we need to reconstruct them here.
412 self._Run(['-c', 'core.deltaBaseCacheLimit=2g', 'pull', 'origin', 441 self._Run(['-c', 'core.deltaBaseCacheLimit=2g', 'pull', 'origin',
413 'master'], options) 442 'master'], options)
414 self._FetchAndReset(revision, file_list, options) 443 self._FetchAndReset(revision, file_list, options)
415 444
M-A Ruel 2013/07/16 13:46:32 .
416 cur_branch = self._GetCurrentBranch() 445 cur_branch = self._GetCurrentBranch()
417 446
418 # Cases: 447 # Cases:
419 # 0) HEAD is detached. Probably from our initial clone. 448 # 0) HEAD is detached. Probably from our initial clone.
420 # - make sure HEAD is contained by a named ref, then update. 449 # - make sure HEAD is contained by a named ref, then update.
421 # Cases 1-4. HEAD is a branch. 450 # Cases 1-4. HEAD is a branch.
422 # 1) current branch is not tracking a remote branch (could be git-svn) 451 # 1) current branch is not tracking a remote branch (could be git-svn)
423 # - try to rebase onto the new hash or branch 452 # - try to rebase onto the new hash or branch
424 # 2) current branch is tracking a remote branch with local committed 453 # 2) current branch is tracking a remote branch with local committed
425 # changes, but the DEPS file switched to point to a hash 454 # changes, but the DEPS file switched to point to a hash
(...skipping 341 matching lines...) Expand 10 before | Expand all | Expand 10 after
767 existing_url = self._Capture(['config', 'remote.origin.url'], 796 existing_url = self._Capture(['config', 'remote.origin.url'],
768 cwd=folder) 797 cwd=folder)
769 assert self._NormalizeGitURL(existing_url) == self._NormalizeGitURL(url) 798 assert self._NormalizeGitURL(existing_url) == self._NormalizeGitURL(url)
770 799
771 # Would normally use `git remote update`, but it doesn't support 800 # Would normally use `git remote update`, but it doesn't support
772 # --progress, so use fetch instead. 801 # --progress, so use fetch instead.
773 self._Run(['fetch'] + v + ['--multiple', '--progress', '--all'], 802 self._Run(['fetch'] + v + ['--multiple', '--progress', '--all'],
774 options, git_filter=True, filter_fn=filter_fn, cwd=folder) 803 options, git_filter=True, filter_fn=filter_fn, cwd=folder)
775 return folder 804 return folder
776 805
777 def _Clone(self, revision, url, options): 806 def _Clone(self, revision, url, options, clone_path=None):
778 """Clone a git repository from the given URL. 807 """Clone a git repository from the given URL.
779 808
780 Once we've cloned the repo, we checkout a working branch if the specified 809 Once we've cloned the repo, we checkout a working branch if the specified
781 revision is a branch head. If it is a tag or a specific commit, then we 810 revision is a branch head. If it is a tag or a specific commit, then we
782 leave HEAD detached as it makes future updates simpler -- in this case the 811 leave HEAD detached as it makes future updates simpler -- in this case the
783 user should first create a new branch or switch to an existing branch before 812 user should first create a new branch or switch to an existing branch before
784 making changes in the repo.""" 813 making changes in the repo."""
785 if not options.verbose: 814 if not options.verbose:
786 # git clone doesn't seem to insert a newline properly before printing 815 # git clone doesn't seem to insert a newline properly before printing
787 # to stdout 816 # to stdout
788 print('') 817 print('')
789 template_path = os.path.join( 818 template_path = os.path.join(
790 os.path.dirname(THIS_FILE_PATH), 'git-templates') 819 os.path.dirname(THIS_FILE_PATH), 'git-templates')
791 clone_cmd = ['-c', 'core.deltaBaseCacheLimit=2g', 'clone', '--progress', 820 clone_cmd = ['-c', 'core.deltaBaseCacheLimit=2g', 'clone', '--progress',
792 '--template=%s' % template_path] 821 '--template=%s' % template_path]
793 if self.cache_dir: 822 if self.cache_dir:
794 clone_cmd.append('--shared') 823 clone_cmd.append('--shared')
795 if revision.startswith('refs/heads/'): 824 if revision.startswith('refs/heads/'):
796 clone_cmd.extend(['-b', revision.replace('refs/heads/', '')]) 825 clone_cmd.extend(['-b', revision.replace('refs/heads/', '')])
797 detach_head = False 826 detach_head = False
798 else: 827 else:
799 detach_head = True 828 detach_head = True
800 if options.verbose: 829 if options.verbose:
801 clone_cmd.append('--verbose') 830 clone_cmd.append('--verbose')
802 clone_cmd.extend([url, self.checkout_path]) 831 clone_cmd.extend([url, clone_path or self.checkout_path])
803 832
804 # If the parent directory does not exist, Git clone on Windows will not 833 # If the parent directory does not exist, Git clone on Windows will not
805 # create it, so we need to do it manually. 834 # create it, so we need to do it manually.
806 parent_dir = os.path.dirname(self.checkout_path) 835 parent_dir = os.path.dirname(self.checkout_path)
807 if not os.path.exists(parent_dir): 836 if not os.path.exists(parent_dir):
808 gclient_utils.safe_makedirs(parent_dir) 837 gclient_utils.safe_makedirs(parent_dir)
809 838
810 for _ in range(3): 839 for _ in range(3):
811 try: 840 try:
812 self._Run(clone_cmd, options, cwd=self._root_dir, git_filter=True) 841 self._Run(clone_cmd, options, cwd=self._root_dir, git_filter=True)
(...skipping 622 matching lines...) Expand 10 before | Expand all | Expand 10 after
1435 new_command.append('--force') 1464 new_command.append('--force')
1436 if command[0] != 'checkout' and scm.SVN.AssertVersion('1.6')[0]: 1465 if command[0] != 'checkout' and scm.SVN.AssertVersion('1.6')[0]:
1437 new_command.extend(('--accept', 'theirs-conflict')) 1466 new_command.extend(('--accept', 'theirs-conflict'))
1438 elif options.manually_grab_svn_rev: 1467 elif options.manually_grab_svn_rev:
1439 new_command.append('--force') 1468 new_command.append('--force')
1440 if command[0] != 'checkout' and scm.SVN.AssertVersion('1.6')[0]: 1469 if command[0] != 'checkout' and scm.SVN.AssertVersion('1.6')[0]:
1441 new_command.extend(('--accept', 'postpone')) 1470 new_command.extend(('--accept', 'postpone'))
1442 elif command[0] != 'checkout' and scm.SVN.AssertVersion('1.6')[0]: 1471 elif command[0] != 'checkout' and scm.SVN.AssertVersion('1.6')[0]:
1443 new_command.extend(('--accept', 'postpone')) 1472 new_command.extend(('--accept', 'postpone'))
1444 return new_command 1473 return new_command
OLDNEW
« no previous file with comments | « gclient.py ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698