Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(1494)

Unified Diff: gclient.py

Issue 11312116: Add gclient grep for git repos (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/tools/depot_tools
Patch Set: Created 8 years, 1 month ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « no previous file | gclient_utils.py » ('j') | gclient_utils.py » ('J')
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
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]')
« no previous file with comments | « no previous file | gclient_utils.py » ('j') | gclient_utils.py » ('J')

Powered by Google App Engine
This is Rietveld 408576698