diff options
Diffstat (limited to 'subcmds/grep.py')
| -rw-r--r-- | subcmds/grep.py | 202 |
1 files changed, 110 insertions, 92 deletions
diff --git a/subcmds/grep.py b/subcmds/grep.py index 4dd85d57..8ac4ba14 100644 --- a/subcmds/grep.py +++ b/subcmds/grep.py | |||
| @@ -1,5 +1,3 @@ | |||
| 1 | # -*- coding:utf-8 -*- | ||
| 2 | # | ||
| 3 | # Copyright (C) 2009 The Android Open Source Project | 1 | # Copyright (C) 2009 The Android Open Source Project |
| 4 | # | 2 | # |
| 5 | # Licensed under the Apache License, Version 2.0 (the "License"); | 3 | # Licensed under the Apache License, Version 2.0 (the "License"); |
| @@ -14,14 +12,14 @@ | |||
| 14 | # See the License for the specific language governing permissions and | 12 | # See the License for the specific language governing permissions and |
| 15 | # limitations under the License. | 13 | # limitations under the License. |
| 16 | 14 | ||
| 17 | from __future__ import print_function | 15 | import functools |
| 18 | |||
| 19 | import sys | 16 | import sys |
| 20 | 17 | ||
| 21 | from color import Coloring | 18 | from color import Coloring |
| 22 | from command import PagedCommand | 19 | from command import DEFAULT_LOCAL_JOBS, PagedCommand |
| 23 | from error import GitError | 20 | from error import GitError |
| 24 | from git_command import git_require, GitCommand | 21 | from git_command import GitCommand |
| 22 | |||
| 25 | 23 | ||
| 26 | class GrepColoring(Coloring): | 24 | class GrepColoring(Coloring): |
| 27 | def __init__(self, config): | 25 | def __init__(self, config): |
| @@ -29,8 +27,9 @@ class GrepColoring(Coloring): | |||
| 29 | self.project = self.printer('project', attr='bold') | 27 | self.project = self.printer('project', attr='bold') |
| 30 | self.fail = self.printer('fail', fg='red') | 28 | self.fail = self.printer('fail', fg='red') |
| 31 | 29 | ||
| 30 | |||
| 32 | class Grep(PagedCommand): | 31 | class Grep(PagedCommand): |
| 33 | common = True | 32 | COMMON = True |
| 34 | helpSummary = "Print lines matching a pattern" | 33 | helpSummary = "Print lines matching a pattern" |
| 35 | helpUsage = """ | 34 | helpUsage = """ |
| 36 | %prog {pattern | -e pattern} [<project>...] | 35 | %prog {pattern | -e pattern} [<project>...] |
| @@ -63,30 +62,33 @@ contain a line that matches both expressions: | |||
| 63 | repo grep --all-match -e NODE -e Unexpected | 62 | repo grep --all-match -e NODE -e Unexpected |
| 64 | 63 | ||
| 65 | """ | 64 | """ |
| 65 | PARALLEL_JOBS = DEFAULT_LOCAL_JOBS | ||
| 66 | |||
| 67 | @staticmethod | ||
| 68 | def _carry_option(_option, opt_str, value, parser): | ||
| 69 | pt = getattr(parser.values, 'cmd_argv', None) | ||
| 70 | if pt is None: | ||
| 71 | pt = [] | ||
| 72 | setattr(parser.values, 'cmd_argv', pt) | ||
| 73 | |||
| 74 | if opt_str == '-(': | ||
| 75 | pt.append('(') | ||
| 76 | elif opt_str == '-)': | ||
| 77 | pt.append(')') | ||
| 78 | else: | ||
| 79 | pt.append(opt_str) | ||
| 66 | 80 | ||
| 67 | def _Options(self, p): | 81 | if value is not None: |
| 68 | def carry(option, | 82 | pt.append(value) |
| 69 | opt_str, | ||
| 70 | value, | ||
| 71 | parser): | ||
| 72 | pt = getattr(parser.values, 'cmd_argv', None) | ||
| 73 | if pt is None: | ||
| 74 | pt = [] | ||
| 75 | setattr(parser.values, 'cmd_argv', pt) | ||
| 76 | |||
| 77 | if opt_str == '-(': | ||
| 78 | pt.append('(') | ||
| 79 | elif opt_str == '-)': | ||
| 80 | pt.append(')') | ||
| 81 | else: | ||
| 82 | pt.append(opt_str) | ||
| 83 | 83 | ||
| 84 | if value is not None: | 84 | def _CommonOptions(self, p): |
| 85 | pt.append(value) | 85 | """Override common options slightly.""" |
| 86 | super()._CommonOptions(p, opt_v=False) | ||
| 86 | 87 | ||
| 88 | def _Options(self, p): | ||
| 87 | g = p.add_option_group('Sources') | 89 | g = p.add_option_group('Sources') |
| 88 | g.add_option('--cached', | 90 | g.add_option('--cached', |
| 89 | action='callback', callback=carry, | 91 | action='callback', callback=self._carry_option, |
| 90 | help='Search the index, instead of the work tree') | 92 | help='Search the index, instead of the work tree') |
| 91 | g.add_option('-r', '--revision', | 93 | g.add_option('-r', '--revision', |
| 92 | dest='revision', action='append', metavar='TREEish', | 94 | dest='revision', action='append', metavar='TREEish', |
| @@ -94,136 +96,111 @@ contain a line that matches both expressions: | |||
| 94 | 96 | ||
| 95 | g = p.add_option_group('Pattern') | 97 | g = p.add_option_group('Pattern') |
| 96 | g.add_option('-e', | 98 | g.add_option('-e', |
| 97 | action='callback', callback=carry, | 99 | action='callback', callback=self._carry_option, |
| 98 | metavar='PATTERN', type='str', | 100 | metavar='PATTERN', type='str', |
| 99 | help='Pattern to search for') | 101 | help='Pattern to search for') |
| 100 | g.add_option('-i', '--ignore-case', | 102 | g.add_option('-i', '--ignore-case', |
| 101 | action='callback', callback=carry, | 103 | action='callback', callback=self._carry_option, |
| 102 | help='Ignore case differences') | 104 | help='Ignore case differences') |
| 103 | g.add_option('-a', '--text', | 105 | g.add_option('-a', '--text', |
| 104 | action='callback', callback=carry, | 106 | action='callback', callback=self._carry_option, |
| 105 | help="Process binary files as if they were text") | 107 | help="Process binary files as if they were text") |
| 106 | g.add_option('-I', | 108 | g.add_option('-I', |
| 107 | action='callback', callback=carry, | 109 | action='callback', callback=self._carry_option, |
| 108 | help="Don't match the pattern in binary files") | 110 | help="Don't match the pattern in binary files") |
| 109 | g.add_option('-w', '--word-regexp', | 111 | g.add_option('-w', '--word-regexp', |
| 110 | action='callback', callback=carry, | 112 | action='callback', callback=self._carry_option, |
| 111 | help='Match the pattern only at word boundaries') | 113 | help='Match the pattern only at word boundaries') |
| 112 | g.add_option('-v', '--invert-match', | 114 | g.add_option('-v', '--invert-match', |
| 113 | action='callback', callback=carry, | 115 | action='callback', callback=self._carry_option, |
| 114 | help='Select non-matching lines') | 116 | help='Select non-matching lines') |
| 115 | g.add_option('-G', '--basic-regexp', | 117 | g.add_option('-G', '--basic-regexp', |
| 116 | action='callback', callback=carry, | 118 | action='callback', callback=self._carry_option, |
| 117 | help='Use POSIX basic regexp for patterns (default)') | 119 | help='Use POSIX basic regexp for patterns (default)') |
| 118 | g.add_option('-E', '--extended-regexp', | 120 | g.add_option('-E', '--extended-regexp', |
| 119 | action='callback', callback=carry, | 121 | action='callback', callback=self._carry_option, |
| 120 | help='Use POSIX extended regexp for patterns') | 122 | help='Use POSIX extended regexp for patterns') |
| 121 | g.add_option('-F', '--fixed-strings', | 123 | g.add_option('-F', '--fixed-strings', |
| 122 | action='callback', callback=carry, | 124 | action='callback', callback=self._carry_option, |
| 123 | help='Use fixed strings (not regexp) for pattern') | 125 | help='Use fixed strings (not regexp) for pattern') |
| 124 | 126 | ||
| 125 | g = p.add_option_group('Pattern Grouping') | 127 | g = p.add_option_group('Pattern Grouping') |
| 126 | g.add_option('--all-match', | 128 | g.add_option('--all-match', |
| 127 | action='callback', callback=carry, | 129 | action='callback', callback=self._carry_option, |
| 128 | help='Limit match to lines that have all patterns') | 130 | help='Limit match to lines that have all patterns') |
| 129 | g.add_option('--and', '--or', '--not', | 131 | g.add_option('--and', '--or', '--not', |
| 130 | action='callback', callback=carry, | 132 | action='callback', callback=self._carry_option, |
| 131 | help='Boolean operators to combine patterns') | 133 | help='Boolean operators to combine patterns') |
| 132 | g.add_option('-(', '-)', | 134 | g.add_option('-(', '-)', |
| 133 | action='callback', callback=carry, | 135 | action='callback', callback=self._carry_option, |
| 134 | help='Boolean operator grouping') | 136 | help='Boolean operator grouping') |
| 135 | 137 | ||
| 136 | g = p.add_option_group('Output') | 138 | g = p.add_option_group('Output') |
| 137 | g.add_option('-n', | 139 | g.add_option('-n', |
| 138 | action='callback', callback=carry, | 140 | action='callback', callback=self._carry_option, |
| 139 | help='Prefix the line number to matching lines') | 141 | help='Prefix the line number to matching lines') |
| 140 | g.add_option('-C', | 142 | g.add_option('-C', |
| 141 | action='callback', callback=carry, | 143 | action='callback', callback=self._carry_option, |
| 142 | metavar='CONTEXT', type='str', | 144 | metavar='CONTEXT', type='str', |
| 143 | help='Show CONTEXT lines around match') | 145 | help='Show CONTEXT lines around match') |
| 144 | g.add_option('-B', | 146 | g.add_option('-B', |
| 145 | action='callback', callback=carry, | 147 | action='callback', callback=self._carry_option, |
| 146 | metavar='CONTEXT', type='str', | 148 | metavar='CONTEXT', type='str', |
| 147 | help='Show CONTEXT lines before match') | 149 | help='Show CONTEXT lines before match') |
| 148 | g.add_option('-A', | 150 | g.add_option('-A', |
| 149 | action='callback', callback=carry, | 151 | action='callback', callback=self._carry_option, |
| 150 | metavar='CONTEXT', type='str', | 152 | metavar='CONTEXT', type='str', |
| 151 | help='Show CONTEXT lines after match') | 153 | help='Show CONTEXT lines after match') |
| 152 | g.add_option('-l', '--name-only', '--files-with-matches', | 154 | g.add_option('-l', '--name-only', '--files-with-matches', |
| 153 | action='callback', callback=carry, | 155 | action='callback', callback=self._carry_option, |
| 154 | help='Show only file names containing matching lines') | 156 | help='Show only file names containing matching lines') |
| 155 | g.add_option('-L', '--files-without-match', | 157 | g.add_option('-L', '--files-without-match', |
| 156 | action='callback', callback=carry, | 158 | action='callback', callback=self._carry_option, |
| 157 | help='Show only file names not containing matching lines') | 159 | help='Show only file names not containing matching lines') |
| 158 | 160 | ||
| 159 | 161 | def _ExecuteOne(self, cmd_argv, project): | |
| 160 | def Execute(self, opt, args): | 162 | """Process one project.""" |
| 161 | out = GrepColoring(self.manifest.manifestProject.config) | 163 | try: |
| 162 | 164 | p = GitCommand(project, | |
| 163 | cmd_argv = ['grep'] | 165 | cmd_argv, |
| 164 | if out.is_on and git_require((1, 6, 3)): | 166 | bare=False, |
| 165 | cmd_argv.append('--color') | 167 | capture_stdout=True, |
| 166 | cmd_argv.extend(getattr(opt, 'cmd_argv', [])) | 168 | capture_stderr=True) |
| 167 | 169 | except GitError as e: | |
| 168 | if '-e' not in cmd_argv: | 170 | return (project, -1, None, str(e)) |
| 169 | if not args: | 171 | |
| 170 | self.Usage() | 172 | return (project, p.Wait(), p.stdout, p.stderr) |
| 171 | cmd_argv.append('-e') | 173 | |
| 172 | cmd_argv.append(args[0]) | 174 | @staticmethod |
| 173 | args = args[1:] | 175 | def _ProcessResults(full_name, have_rev, _pool, out, results): |
| 174 | |||
| 175 | projects = self.GetProjects(args) | ||
| 176 | |||
| 177 | full_name = False | ||
| 178 | if len(projects) > 1: | ||
| 179 | cmd_argv.append('--full-name') | ||
| 180 | full_name = True | ||
| 181 | |||
| 182 | have_rev = False | ||
| 183 | if opt.revision: | ||
| 184 | if '--cached' in cmd_argv: | ||
| 185 | print('fatal: cannot combine --cached and --revision', file=sys.stderr) | ||
| 186 | sys.exit(1) | ||
| 187 | have_rev = True | ||
| 188 | cmd_argv.extend(opt.revision) | ||
| 189 | cmd_argv.append('--') | ||
| 190 | |||
| 191 | git_failed = False | 176 | git_failed = False |
| 192 | bad_rev = False | 177 | bad_rev = False |
| 193 | have_match = False | 178 | have_match = False |
| 194 | 179 | ||
| 195 | for project in projects: | 180 | for project, rc, stdout, stderr in results: |
| 196 | try: | 181 | if rc < 0: |
| 197 | p = GitCommand(project, | ||
| 198 | cmd_argv, | ||
| 199 | bare=False, | ||
| 200 | capture_stdout=True, | ||
| 201 | capture_stderr=True) | ||
| 202 | except GitError as e: | ||
| 203 | git_failed = True | 182 | git_failed = True |
| 204 | out.project('--- project %s ---' % project.relpath) | 183 | out.project('--- project %s ---' % project.relpath) |
| 205 | out.nl() | 184 | out.nl() |
| 206 | out.fail('%s', str(e)) | 185 | out.fail('%s', stderr) |
| 207 | out.nl() | 186 | out.nl() |
| 208 | continue | 187 | continue |
| 209 | 188 | ||
| 210 | if p.Wait() != 0: | 189 | if rc: |
| 211 | # no results | 190 | # no results |
| 212 | # | 191 | if stderr: |
| 213 | if p.stderr: | 192 | if have_rev and 'fatal: ambiguous argument' in stderr: |
| 214 | if have_rev and 'fatal: ambiguous argument' in p.stderr: | ||
| 215 | bad_rev = True | 193 | bad_rev = True |
| 216 | else: | 194 | else: |
| 217 | out.project('--- project %s ---' % project.relpath) | 195 | out.project('--- project %s ---' % project.relpath) |
| 218 | out.nl() | 196 | out.nl() |
| 219 | out.fail('%s', p.stderr.strip()) | 197 | out.fail('%s', stderr.strip()) |
| 220 | out.nl() | 198 | out.nl() |
| 221 | continue | 199 | continue |
| 222 | have_match = True | 200 | have_match = True |
| 223 | 201 | ||
| 224 | # We cut the last element, to avoid a blank line. | 202 | # We cut the last element, to avoid a blank line. |
| 225 | # | 203 | r = stdout.split('\n') |
| 226 | r = p.stdout.split('\n') | ||
| 227 | r = r[0:-1] | 204 | r = r[0:-1] |
| 228 | 205 | ||
| 229 | if have_rev and full_name: | 206 | if have_rev and full_name: |
| @@ -245,6 +222,47 @@ contain a line that matches both expressions: | |||
| 245 | for line in r: | 222 | for line in r: |
| 246 | print(line) | 223 | print(line) |
| 247 | 224 | ||
| 225 | return (git_failed, bad_rev, have_match) | ||
| 226 | |||
| 227 | def Execute(self, opt, args): | ||
| 228 | out = GrepColoring(self.manifest.manifestProject.config) | ||
| 229 | |||
| 230 | cmd_argv = ['grep'] | ||
| 231 | if out.is_on: | ||
| 232 | cmd_argv.append('--color') | ||
| 233 | cmd_argv.extend(getattr(opt, 'cmd_argv', [])) | ||
| 234 | |||
| 235 | if '-e' not in cmd_argv: | ||
| 236 | if not args: | ||
| 237 | self.Usage() | ||
| 238 | cmd_argv.append('-e') | ||
| 239 | cmd_argv.append(args[0]) | ||
| 240 | args = args[1:] | ||
| 241 | |||
| 242 | projects = self.GetProjects(args) | ||
| 243 | |||
| 244 | full_name = False | ||
| 245 | if len(projects) > 1: | ||
| 246 | cmd_argv.append('--full-name') | ||
| 247 | full_name = True | ||
| 248 | |||
| 249 | have_rev = False | ||
| 250 | if opt.revision: | ||
| 251 | if '--cached' in cmd_argv: | ||
| 252 | print('fatal: cannot combine --cached and --revision', file=sys.stderr) | ||
| 253 | sys.exit(1) | ||
| 254 | have_rev = True | ||
| 255 | cmd_argv.extend(opt.revision) | ||
| 256 | cmd_argv.append('--') | ||
| 257 | |||
| 258 | git_failed, bad_rev, have_match = self.ExecuteInParallel( | ||
| 259 | opt.jobs, | ||
| 260 | functools.partial(self._ExecuteOne, cmd_argv), | ||
| 261 | projects, | ||
| 262 | callback=functools.partial(self._ProcessResults, full_name, have_rev), | ||
| 263 | output=out, | ||
| 264 | ordered=True) | ||
| 265 | |||
| 248 | if git_failed: | 266 | if git_failed: |
| 249 | sys.exit(1) | 267 | sys.exit(1) |
| 250 | elif have_match: | 268 | elif have_match: |
