Index: gclient.py |
diff --git a/gclient.py b/gclient.py |
index fd4075610dbeef40cdc2b754418c1f597e332113..8f9c3181f994abe05803f5b55782cd72d6a63aa3 100644 |
--- a/gclient.py |
+++ b/gclient.py |
@@ -569,7 +569,8 @@ class Dependency(gclient_utils.WorkItem, DependencySettings): |
# Arguments number differs from overridden method |
# pylint: disable=W0221 |
- def run(self, revision_overrides, command, args, work_queue, options): |
+ def run(self, revision_overrides, command, args, work_queue, options, |
+ **kwargs): |
"""Runs |command| then parse the DEPS file.""" |
logging.info('Dependency(%s).run()' % self.name) |
assert self._file_list == [] |
@@ -618,7 +619,18 @@ class Dependency(gclient_utils.WorkItem, DependencySettings): |
# Strip any leading path separators. |
while file_list[i].startswith(('\\', '/')): |
file_list[i] = file_list[i][1:] |
- elif command == 'recurse': |
+ |
+ # Always parse the DEPS file. |
+ self.ParseDepsFile() |
+ |
+ self._run_is_done(file_list, parsed_url) |
+ |
+ if self.recursion_limit: |
+ # Parse the dependencies of this dependency. |
+ for s in self.dependencies: |
+ work_queue.enqueue(s) |
+ |
+ if command == 'recurse': |
if not isinstance(parsed_url, self.FileImpl): |
# Skip file only checkout. |
scm = gclient_scm.GetScmName(parsed_url) |
@@ -630,25 +642,37 @@ class Dependency(gclient_utils.WorkItem, DependencySettings): |
env['GCLIENT_SCM'] = scm |
if parsed_url: |
env['GCLIENT_URL'] = parsed_url |
+ print_stdout = True |
+ filter_fn = None |
+ if options.prepend_dir: |
+ print_stdout = False |
+ def filter_fn(line): |
+ items = line.split('\0') |
M-A Ruel
2012/11/08 19:44:13
if '\0' in line:
...
else:
items = line.split('\
Isaac (away)
2012/11/09 03:58:52
I'm confused -- what are you suggesting here? Wil
|
+ if len(items) == 1: |
+ match = re.match('Binary file (.*) matches$', line) |
+ if match: |
+ print 'Binary file %s matches' % os.path.join( |
+ self.name, match.group(1)) |
+ else: |
+ print line |
+ elif len(items) == 2 and items[1]: |
+ print '%s : %s' % (os.path.join(self.name, items[0]), items[1]) |
+ else: |
+ # Multiple null bytes or a single trailing null byte indicate |
+ # git is likely displaying filenames only (such as with -l) |
+ print '\n'.join(os.path.join(self.name, f) for f in items if f) |
if os.path.isdir(cwd): |
try: |
gclient_utils.CheckCallAndFilter( |
- args, cwd=cwd, env=env, print_stdout=True) |
+ args, cwd=cwd, env=env, print_stdout=print_stdout, |
+ filter_fn=filter_fn, |
+ ) |
except subprocess2.CalledProcessError: |
if not options.ignore: |
raise |
else: |
print >> sys.stderr, 'Skipped missing %s' % cwd |
- # Always parse the DEPS file. |
- self.ParseDepsFile() |
- |
- self._run_is_done(file_list, parsed_url) |
- |
- if self.recursion_limit: |
- # Parse the dependencies of this dependency. |
- for s in self.dependencies: |
- work_queue.enqueue(s) |
@gclient_utils.lockedmethod |
def _run_is_done(self, file_list, parsed_url): |
@@ -1050,7 +1074,7 @@ solutions = [ |
print('Using safesync_url revision: %s.\n' % safe_rev) |
self._options.revisions.append('%s@%s' % (dep.name, safe_rev)) |
- def RunOnDeps(self, command, args): |
+ def RunOnDeps(self, command, args, ignore_requirements=False): |
"""Runs a command on each dependency in a client and its dependencies. |
Args: |
@@ -1074,7 +1098,8 @@ solutions = [ |
work_queue = gclient_utils.ExecutionQueue(self._options.jobs, pm) |
for s in self.dependencies: |
work_queue.enqueue(s) |
- work_queue.flush(revision_overrides, command, args, options=self._options) |
+ work_queue.flush(revision_overrides, command, args, options=self._options, |
+ ignore_requirements=ignore_requirements) |
# Once all the dependencies have been processed, it's now safe to run the |
# hooks. |
@@ -1234,9 +1259,11 @@ def CMDrecurse(parser, args): |
# Stop parsing at the first non-arg so that these go through to the command |
parser.disable_interspersed_args() |
parser.add_option('-s', '--scm', action='append', default=[], |
- help='choose scm types to operate upon') |
+ help='Choose scm types to operate upon.') |
parser.add_option('-i', '--ignore', action='store_true', |
- help='continue processing in case of non zero return code') |
+ help='Ignore non-zero return codes from subcommands.') |
+ parser.add_option('--prepend-dir', action='store_true', |
+ help='Prepend relative dir for use with git <cmd> --null.') |
options, args = parser.parse_args(args) |
if not args: |
print >> sys.stderr, 'Need to supply a command!' |
@@ -1256,7 +1283,7 @@ def CMDrecurse(parser, args): |
options.nohooks = True |
client = GClient.LoadCurrentConfig(options) |
- return client.RunOnDeps('recurse', args) |
+ return client.RunOnDeps('recurse', args, ignore_requirements=True) |
@attr('usage', '[args ...]') |
@@ -1266,8 +1293,38 @@ def CMDfetch(parser, args): |
Completely git-specific. Simply runs 'git fetch [args ...]' for each module. |
""" |
(options, args) = parser.parse_args(args) |
- args = ['-j%d' % options.jobs, '-s', 'git', 'git', 'fetch'] + args |
- return CMDrecurse(parser, args) |
+ return CMDrecurse(Parser(), [ |
+ '--jobs=%d' % options.jobs, '--scm=git', 'git', 'fetch'] + args) |
+ |
+ |
+def CMDgrep(parser, args): |
+ """Greps through git repos managed by gclient. |
+ |
+Runs 'git grep [args...]' for each module. |
+""" |
+ |
+ # We can't use optparse because it will try to parse arguments sent |
+ # to git grep and throw an error. :-( |
+ if not args or re.match('(-h|--help)$', args[0]): |
+ print >> sys.stderr, ( |
+ 'Usage: gclient grep [-j <N>] git-grep-args...\n\n' |
+ 'Example: "gclient grep -j10 -A2 RefCountedBase" runs\n"git grep ' |
+ '-A2 RefCountedBase" on each of gclient\'s git\nrepos with up to ' |
+ '10 jobs.\n\nBonus: page output by appending "|& less -FRSX" to the' |
+ ' end of your query.' |
+ ) |
+ return 1 |
+ |
+ jobs_arg = ['--jobs=1'] |
+ if re.match(r'(-j|--jobs=)\d+$', args[0]): |
+ jobs_arg, args = args[:1], args[1:] |
+ elif re.match(r'(-j|--jobs)$', args[0]): |
+ jobs_arg, args = args[:2], args[2:] |
+ |
+ return CMDrecurse( |
+ parser, |
+ jobs_arg + ['--ignore', '--prepend-dir', '--scm=git', 'git', 'grep', |
+ '--null', '--color=Always'] + args) |
@attr('usage', '[url] [safesync url]') |