OLD | NEW |
---|---|
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 Loading... | |
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 Loading... | |
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 Loading... | |
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 Loading... | |
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 |
OLD | NEW |