Skip to content

Commit a0658c0

Browse files
committed
add --negate flag to pygrep
1 parent 6ba50f3 commit a0658c0

File tree

2 files changed

+99
-4
lines changed

2 files changed

+99
-4
lines changed

pre_commit/languages/pygrep.py

Lines changed: 44 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
import argparse
22
import re
33
import sys
4+
from typing import NamedTuple
45
from typing import Optional
56
from typing import Pattern
67
from typing import Sequence
@@ -45,6 +46,46 @@ def _process_filename_at_once(pattern: Pattern[bytes], filename: str) -> int:
4546
return retv
4647

4748

49+
def _process_filename_by_line_negated(
50+
pattern: Pattern[bytes],
51+
filename: str,
52+
) -> int:
53+
with open(filename, 'rb') as f:
54+
for line in f:
55+
if pattern.search(line):
56+
return 0
57+
else:
58+
output.write_line(filename)
59+
return 1
60+
61+
62+
def _process_filename_at_once_negated(
63+
pattern: Pattern[bytes],
64+
filename: str,
65+
) -> int:
66+
with open(filename, 'rb') as f:
67+
contents = f.read()
68+
match = pattern.search(contents)
69+
if match:
70+
return 0
71+
else:
72+
output.write_line(filename)
73+
return 1
74+
75+
76+
class Choice(NamedTuple):
77+
multiline: bool
78+
negate: bool
79+
80+
81+
FNS = {
82+
Choice(multiline=True, negate=True): _process_filename_at_once_negated,
83+
Choice(multiline=True, negate=False): _process_filename_at_once,
84+
Choice(multiline=False, negate=True): _process_filename_by_line_negated,
85+
Choice(multiline=False, negate=False): _process_filename_by_line,
86+
}
87+
88+
4889
def run_hook(
4990
hook: Hook,
5091
file_args: Sequence[str],
@@ -64,6 +105,7 @@ def main(argv: Optional[Sequence[str]] = None) -> int:
64105
)
65106
parser.add_argument('-i', '--ignore-case', action='store_true')
66107
parser.add_argument('--multiline', action='store_true')
108+
parser.add_argument('--negate', action='store_true')
67109
parser.add_argument('pattern', help='python regex pattern.')
68110
parser.add_argument('filenames', nargs='*')
69111
args = parser.parse_args(argv)
@@ -75,11 +117,9 @@ def main(argv: Optional[Sequence[str]] = None) -> int:
75117
pattern = re.compile(args.pattern.encode(), flags)
76118

77119
retv = 0
120+
process_fn = FNS[Choice(multiline=args.multiline, negate=args.negate)]
78121
for filename in args.filenames:
79-
if args.multiline:
80-
retv |= _process_filename_at_once(pattern, filename)
81-
else:
82-
retv |= _process_filename_by_line(pattern, filename)
122+
retv |= process_fn(pattern, filename)
83123
return retv
84124

85125

tests/languages/pygrep_test.py

Lines changed: 55 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,9 @@ def some_files(tmpdir):
88
tmpdir.join('f1').write_binary(b'foo\nbar\n')
99
tmpdir.join('f2').write_binary(b'[INFO] hi\n')
1010
tmpdir.join('f3').write_binary(b"with'quotes\n")
11+
tmpdir.join('f4').write_binary(b'foo\npattern\nbar\n')
12+
tmpdir.join('f5').write_binary(b'[INFO] hi\npattern\nbar')
13+
tmpdir.join('f6').write_binary(b"pattern\nbarwith'foo\n")
1114
with tmpdir.as_cwd():
1215
yield
1316

@@ -30,6 +33,58 @@ def test_main(cap_out, pattern, expected_retcode, expected_out):
3033
assert out == expected_out
3134

3235

36+
@pytest.mark.usefixtures('some_files')
37+
def test_negate_by_line_no_match(cap_out):
38+
ret = pygrep.main(('pattern\nbar', 'f4', 'f5', 'f6', '--negate'))
39+
out = cap_out.get()
40+
assert ret == 1
41+
assert out == 'f4\nf5\nf6\n'
42+
43+
44+
@pytest.mark.usefixtures('some_files')
45+
def test_negate_by_line_two_match(cap_out):
46+
ret = pygrep.main(('foo', 'f4', 'f5', 'f6', '--negate'))
47+
out = cap_out.get()
48+
assert ret == 1
49+
assert out == 'f5\n'
50+
51+
52+
@pytest.mark.usefixtures('some_files')
53+
def test_negate_by_line_all_match(cap_out):
54+
ret = pygrep.main(('pattern', 'f4', 'f5', 'f6', '--negate'))
55+
out = cap_out.get()
56+
assert ret == 0
57+
assert out == ''
58+
59+
60+
@pytest.mark.usefixtures('some_files')
61+
def test_negate_by_file_no_match(cap_out):
62+
ret = pygrep.main(('baz', 'f4', 'f5', 'f6', '--negate', '--multiline'))
63+
out = cap_out.get()
64+
assert ret == 1
65+
assert out == 'f4\nf5\nf6\n'
66+
67+
68+
@pytest.mark.usefixtures('some_files')
69+
def test_negate_by_file_one_match(cap_out):
70+
ret = pygrep.main(
71+
('foo\npattern', 'f4', 'f5', 'f6', '--negate', '--multiline'),
72+
)
73+
out = cap_out.get()
74+
assert ret == 1
75+
assert out == 'f5\nf6\n'
76+
77+
78+
@pytest.mark.usefixtures('some_files')
79+
def test_negate_by_file_all_match(cap_out):
80+
ret = pygrep.main(
81+
('pattern\nbar', 'f4', 'f5', 'f6', '--negate', '--multiline'),
82+
)
83+
out = cap_out.get()
84+
assert ret == 0
85+
assert out == ''
86+
87+
3388
@pytest.mark.usefixtures('some_files')
3489
def test_ignore_case(cap_out):
3590
ret = pygrep.main(('--ignore-case', 'info', 'f1', 'f2', 'f3'))

0 commit comments

Comments
 (0)