From f4545126197781beb03bb0fd47e7f24ce5af6ca8 Mon Sep 17 00:00:00 2001 From: Mike Frysinger Date: Mon, 11 Nov 2019 04:34:16 -0500 Subject: sync: make .git init more robust Hitting Ctrl-C in the middle of this func will leave the .git in a bad state that requires manual recovery. The code tries to catch all exceptions and recover by deleting the incomplete .git dir, but it omits KeyboardInterrupt which Exception misses. We could add that to the recovery path, but we can make this more robust with a different approach: set up everything in .git.tmp/ and only move it to .git/ once we've fully initialized it. Change-Id: I0f5b97f2e19fc39cffc3e5e23993a2da7220f4e3 Reviewed-on: https://gerrit-review.googlesource.com/c/git-repo/+/244733 Reviewed-by: David Pursehouse Tested-by: Mike Frysinger --- project.py | 64 +++++++++++++++++++++++++++++++++----------------------------- 1 file changed, 34 insertions(+), 30 deletions(-) (limited to 'project.py') diff --git a/project.py b/project.py index 51160a94..a2a3adc8 100755 --- a/project.py +++ b/project.py @@ -2706,41 +2706,45 @@ class Project(object): raise def _InitWorkTree(self, force_sync=False, submodules=False): - dotgit = os.path.join(self.worktree, '.git') - init_dotgit = not os.path.exists(dotgit) + realdotgit = os.path.join(self.worktree, '.git') + tmpdotgit = realdotgit + '.tmp' + init_dotgit = not os.path.exists(realdotgit) + if init_dotgit: + dotgit = tmpdotgit + platform_utils.rmtree(tmpdotgit, ignore_errors=True) + os.makedirs(tmpdotgit) + self._ReferenceGitDir(self.gitdir, tmpdotgit, share_refs=True, + copy_all=False) + else: + dotgit = realdotgit + try: - if init_dotgit: - os.makedirs(dotgit) - self._ReferenceGitDir(self.gitdir, dotgit, share_refs=True, - copy_all=False) + self._CheckDirReference(self.gitdir, dotgit, share_refs=True) + except GitError as e: + if force_sync and not init_dotgit: + try: + platform_utils.rmtree(dotgit) + return self._InitWorkTree(force_sync=False, submodules=submodules) + except: + raise e + raise e - try: - self._CheckDirReference(self.gitdir, dotgit, share_refs=True) - except GitError as e: - if force_sync: - try: - platform_utils.rmtree(dotgit) - return self._InitWorkTree(force_sync=False, submodules=submodules) - except: - raise e - raise e + if init_dotgit: + _lwrite(os.path.join(tmpdotgit, HEAD), '%s\n' % self.GetRevisionId()) - if init_dotgit: - _lwrite(os.path.join(dotgit, HEAD), '%s\n' % self.GetRevisionId()) + # Now that the .git dir is fully set up, move it to its final home. + platform_utils.rename(tmpdotgit, realdotgit) - cmd = ['read-tree', '--reset', '-u'] - cmd.append('-v') - cmd.append(HEAD) - if GitCommand(self, cmd).Wait() != 0: - raise GitError("cannot initialize work tree for " + self.name) + # Finish checking out the worktree. + cmd = ['read-tree', '--reset', '-u'] + cmd.append('-v') + cmd.append(HEAD) + if GitCommand(self, cmd).Wait() != 0: + raise GitError('Cannot initialize work tree for ' + self.name) - if submodules: - self._SyncSubmodules(quiet=True) - self._CopyAndLinkFiles() - except Exception: - if init_dotgit: - platform_utils.rmtree(dotgit) - raise + if submodules: + self._SyncSubmodules(quiet=True) + self._CopyAndLinkFiles() def _get_symlink_error_message(self): if platform_utils.isWindows(): -- cgit v1.2.3-54-g00ecf