Rework gclient 'recurse' command to use a WorkQueue.

Support --jobs in 'fetch' and 'recurse' commands. 

BUG=115840
TEST=

Review URL: http://codereview.chromium.org/9560010

git-svn-id: svn://svn.chromium.org/chrome/trunk/tools/depot_tools@124657 0039d316-1c4b-4281-b951-d872f2087c98
parent 12f944e3
...@@ -525,7 +525,7 @@ class Dependency(gclient_utils.WorkItem, DependencySettings): ...@@ -525,7 +525,7 @@ class Dependency(gclient_utils.WorkItem, DependencySettings):
# When running runhooks, there's no need to consult the SCM. # When running runhooks, there's no need to consult the SCM.
# All known hooks are expected to run unconditionally regardless of working # All known hooks are expected to run unconditionally regardless of working
# copy state, so skip the SCM status check. # copy state, so skip the SCM status check.
run_scm = command not in ('runhooks', None) run_scm = command not in ('runhooks', 'recurse', None)
parsed_url = self.LateOverride(self.url) parsed_url = self.LateOverride(self.url)
file_list = [] file_list = []
if run_scm and parsed_url: if run_scm and parsed_url:
...@@ -565,6 +565,23 @@ class Dependency(gclient_utils.WorkItem, DependencySettings): ...@@ -565,6 +565,23 @@ class Dependency(gclient_utils.WorkItem, DependencySettings):
# Strip any leading path separators. # Strip any leading path separators.
while file_list[i].startswith(('\\', '/')): while file_list[i].startswith(('\\', '/')):
file_list[i] = file_list[i][1:] file_list[i] = file_list[i][1:]
elif command is 'recurse':
if not isinstance(parsed_url, self.FileImpl):
# Skip file only checkout.
scm = gclient_scm.GetScmName(parsed_url)
if not options.scm or scm in options.scm:
cwd = os.path.normpath(os.path.join(self.root.root_dir, self.name))
# Pass in the SCM type as an env variable
env = os.environ.copy()
if scm:
env['GCLIENT_SCM'] = scm
if parsed_url:
env['GCLIENT_URL'] = parsed_url
if os.path.isdir(cwd):
gclient_utils.CheckCallAndFilter(
args, cwd=cwd, env=env, print_stdout=True)
else:
print >> sys.stderr, 'Skipped missing %s' % cwd
# Always parse the DEPS file. # Always parse the DEPS file.
self.ParseDepsFile() self.ParseDepsFile()
...@@ -965,12 +982,18 @@ solutions = [ ...@@ -965,12 +982,18 @@ solutions = [
""" """
if not self.dependencies: if not self.dependencies:
raise gclient_utils.Error('No solution specified') raise gclient_utils.Error('No solution specified')
revision_overrides = self._EnforceRevisions() revision_overrides = {}
# It's unnecessary to check for revision overrides for 'recurse'.
# Save a few seconds by not calling _EnforceRevisions() in that case.
if command is not 'recurse':
revision_overrides = self._EnforceRevisions()
pm = None pm = None
# Disable progress for non-tty stdout. # Disable progress for non-tty stdout.
if (command in ('update', 'revert') and sys.stdout.isatty() and not if (sys.stdout.isatty() and not self._options.verbose):
self._options.verbose): if command in ('update', 'revert'):
pm = Progress('Syncing projects', 1) pm = Progress('Syncing projects', 1)
elif command is 'recurse':
pm = Progress(' '.join(args), 1)
work_queue = gclient_utils.ExecutionQueue(self._options.jobs, pm) work_queue = gclient_utils.ExecutionQueue(self._options.jobs, pm)
for s in self.dependencies: for s in self.dependencies:
work_queue.enqueue(s) work_queue.enqueue(s)
...@@ -1131,7 +1154,6 @@ def CMDrecurse(parser, args): ...@@ -1131,7 +1154,6 @@ def CMDrecurse(parser, args):
parser.disable_interspersed_args() parser.disable_interspersed_args()
parser.add_option('-s', '--scm', action='append', default=[], parser.add_option('-s', '--scm', action='append', default=[],
help='choose scm types to operate upon') help='choose scm types to operate upon')
parser.remove_option('--jobs')
options, args = parser.parse_args(args) options, args = parser.parse_args(args)
if not args: if not args:
print >> sys.stderr, 'Need to supply a command!' print >> sys.stderr, 'Need to supply a command!'
...@@ -1142,28 +1164,16 @@ def CMDrecurse(parser, args): ...@@ -1142,28 +1164,16 @@ def CMDrecurse(parser, args):
'You need to run gclient sync at least once to use \'recurse\'.\n' 'You need to run gclient sync at least once to use \'recurse\'.\n'
'This is because .gclient_entries needs to exist and be up to date.') 'This is because .gclient_entries needs to exist and be up to date.')
return 1 return 1
root, entries = root_and_entries
# Normalize options.scm to a set()
scm_set = set() scm_set = set()
for scm in options.scm: for scm in options.scm:
scm_set.update(scm.split(',')) scm_set.update(scm.split(','))
options.scm = scm_set
# Pass in the SCM type as an env variable options.nohooks = True
env = os.environ.copy() client = GClient.LoadCurrentConfig(options)
return client.RunOnDeps('recurse', args)
for path, url in entries.iteritems():
scm = gclient_scm.GetScmName(url)
if scm_set and scm not in scm_set:
continue
cwd = os.path.normpath(os.path.join(root, path))
if scm:
env['GCLIENT_SCM'] = scm
if url:
env['GCLIENT_URL'] = url
if os.path.isdir(cwd):
subprocess2.call(args, cwd=cwd, env=env)
else:
print >> sys.stderr, 'Skipped missing %s' % cwd
return 0
@attr('usage', '[args ...]') @attr('usage', '[args ...]')
...@@ -1172,8 +1182,8 @@ def CMDfetch(parser, args): ...@@ -1172,8 +1182,8 @@ def CMDfetch(parser, args):
Completely git-specific. Simply runs 'git fetch [args ...]' for each module. Completely git-specific. Simply runs 'git fetch [args ...]' for each module.
""" """
(_, args) = parser.parse_args(args) (options, args) = parser.parse_args(args)
args = ['-s', 'git', 'git', 'fetch'] + args args = ['-j%d' % options.jobs, '-s', 'git', 'git', 'fetch'] + args
return CMDrecurse(parser, args) return CMDrecurse(parser, args)
......
#!/usr/bin/env python #!/usr/bin/env python
# Copyright (c) 2011 The Chromium Authors. All rights reserved. # Copyright (c) 2012 The Chromium Authors. All rights reserved.
# Use of this source code is governed by a BSD-style license that can be # Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file. # found in the LICENSE file.
...@@ -1159,7 +1159,7 @@ class GClientSmokeBoth(GClientSmokeBase): ...@@ -1159,7 +1159,7 @@ class GClientSmokeBoth(GClientSmokeBase):
'{"name": "src-git",' '{"name": "src-git",'
'"url": "' + self.git_base + 'repo_1"}]']) '"url": "' + self.git_base + 'repo_1"}]'])
self.gclient(['sync', '--deps', 'mac']) self.gclient(['sync', '--deps', 'mac'])
results = self.gclient(['recurse', 'sh', '-c', results = self.gclient(['recurse', '-j1', 'sh', '-c',
'echo $GCLIENT_SCM,$GCLIENT_URL,`pwd`']) 'echo $GCLIENT_SCM,$GCLIENT_URL,`pwd`'])
entries = [tuple(line.split(',')) entries = [tuple(line.split(','))
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment