From 89ece429fbb3047eabe9e8eb3df3076d74926eb3 Mon Sep 17 00:00:00 2001 From: David James <davidjames@google.com> Date: Thu, 9 Jan 2014 18:51:58 -0800 Subject: [PATCH] Clean up duplicate logic in subcmds/sync.py. The fetch logic is now shared between the jobs == 1 and jobs > 1 cases. This refactoring also fixes a bug where opts.force_broken was not honored when jobs > 1. Change-Id: Ic886f3c3c00f3d8fc73a65366328fed3c44dc3be --- subcmds/sync.py | 81 ++++++++++++++++++++----------------------------- 1 file changed, 33 insertions(+), 48 deletions(-) diff --git a/subcmds/sync.py b/subcmds/sync.py index 27c8c728..a0a68960 100644 --- a/subcmds/sync.py +++ b/subcmds/sync.py @@ -219,7 +219,7 @@ later is required to fix a server side protocol bug. dest='repo_upgraded', action='store_true', help=SUPPRESS_HELP) - def _FetchProjectList(self, opt, projects, *args): + def _FetchProjectList(self, opt, projects, *args, **kwargs): """Main function of the fetch threads when jobs are > 1. Delegates most of the work to _FetchHelper. @@ -227,11 +227,11 @@ later is required to fix a server side protocol bug. Args: opt: Program options returned from optparse. See _Options(). projects: Projects to fetch. - *args: Remaining arguments to pass to _FetchHelper. See the + *args, **kwargs: Remaining arguments to pass to _FetchHelper. See the _FetchHelper docstring for details. """ for project in projects: - success = self._FetchHelper(opt, project, *args) + success = self._FetchHelper(opt, project, *args, **kwargs) if not success and not opt.force_broken: break @@ -304,62 +304,47 @@ later is required to fix a server side protocol bug. def _Fetch(self, projects, opt): fetched = set() + lock = _threading.Lock() pm = Progress('Fetching projects', len(projects)) - if self.jobs == 1: - for project in projects: - pm.update() - if not opt.quiet: - print('Fetching project %s' % project.name) - if project.Sync_NetworkHalf( - quiet=opt.quiet, - current_branch_only=opt.current_branch_only, - clone_bundle=not opt.no_clone_bundle, - no_tags=opt.no_tags, - archive=self.manifest.IsArchive): - fetched.add(project.gitdir) - else: - print('error: Cannot fetch %s' % project.name, file=sys.stderr) - if opt.force_broken: - print('warn: --force-broken, continuing to sync', file=sys.stderr) - else: - sys.exit(1) - else: - objdir_project_map = dict() - for project in projects: - objdir_project_map.setdefault(project.objdir, []).append(project) + objdir_project_map = dict() + for project in projects: + objdir_project_map.setdefault(project.objdir, []).append(project) - threads = set() - lock = _threading.Lock() - sem = _threading.Semaphore(self.jobs) - err_event = _threading.Event() - for project_list in objdir_project_map.values(): - # Check for any errors before starting any new threads. - # ...we'll let existing threads finish, though. - if err_event.isSet(): - break + threads = set() + sem = _threading.Semaphore(self.jobs) + err_event = _threading.Event() + for project_list in objdir_project_map.values(): + # Check for any errors before running any more tasks. + # ...we'll let existing threads finish, though. + if err_event.isSet() and not opt.force_broken: + break - sem.acquire() + sem.acquire() + kwargs = dict(opt=opt, + projects=project_list, + lock=lock, + fetched=fetched, + pm=pm, + sem=sem, + err_event=err_event) + if self.jobs > 1: t = _threading.Thread(target = self._FetchProjectList, - args = (opt, - project_list, - lock, - fetched, - pm, - sem, - err_event)) + kwargs = kwargs) # Ensure that Ctrl-C will not freeze the repo process. t.daemon = True threads.add(t) t.start() + else: + self._FetchProjectList(**kwargs) - for t in threads: - t.join() + for t in threads: + t.join() - # If we saw an error, exit with code 1 so that other scripts can check. - if err_event.isSet(): - print('\nerror: Exited sync due to fetch errors', file=sys.stderr) - sys.exit(1) + # If we saw an error, exit with code 1 so that other scripts can check. + if err_event.isSet(): + print('\nerror: Exited sync due to fetch errors', file=sys.stderr) + sys.exit(1) pm.end() self._fetch_times.Save()