blob: c39bf93aad3bb235b962175a9a80a7c69b48b577 [file] [log] [blame]
Benjamin Peterson90f5ba52010-03-11 22:53:45 +00001#! /usr/bin/env python3
Guido van Rossumee60eb11998-08-12 17:47:52 +00002
3"""Reverse grep.
4
5Usage: rgrep [-i] pattern file
6"""
7
8import sys
9import re
Guido van Rossumee60eb11998-08-12 17:47:52 +000010import getopt
11
Florent Xiclunae4a33802010-08-09 12:24:20 +000012
Guido van Rossumee60eb11998-08-12 17:47:52 +000013def main():
Florent Xiclunae4a33802010-08-09 12:24:20 +000014 bufsize = 64 * 1024
Guido van Rossumee60eb11998-08-12 17:47:52 +000015 reflags = 0
16 opts, args = getopt.getopt(sys.argv[1:], "i")
17 for o, a in opts:
18 if o == '-i':
19 reflags = reflags | re.IGNORECASE
20 if len(args) < 2:
21 usage("not enough arguments")
22 if len(args) > 2:
23 usage("exactly one file argument required")
24 pattern, filename = args
25 try:
26 prog = re.compile(pattern, reflags)
Guido van Rossumb940e112007-01-10 16:19:56 +000027 except re.error as msg:
Florent Xiclunae4a33802010-08-09 12:24:20 +000028 usage("error in regular expression: %s" % msg)
Guido van Rossumee60eb11998-08-12 17:47:52 +000029 try:
30 f = open(filename)
Guido van Rossumb940e112007-01-10 16:19:56 +000031 except IOError as msg:
Florent Xiclunae4a33802010-08-09 12:24:20 +000032 usage("can't open %r: %s" % (filename, msg), 1)
Serhiy Storchaka172bb392019-03-30 08:33:02 +020033 with f:
34 f.seek(0, 2)
35 pos = f.tell()
36 leftover = None
37 while pos > 0:
38 size = min(pos, bufsize)
39 pos = pos - size
40 f.seek(pos)
41 buffer = f.read(size)
42 lines = buffer.split("\n")
43 del buffer
44 if leftover is None:
45 if not lines[-1]:
46 del lines[-1]
47 else:
48 lines[-1] = lines[-1] + leftover
49 if pos > 0:
50 leftover = lines[0]
51 del lines[0]
52 else:
53 leftover = None
54 for line in reversed(lines):
55 if prog.search(line):
56 print(line)
Guido van Rossumee60eb11998-08-12 17:47:52 +000057
Florent Xiclunae4a33802010-08-09 12:24:20 +000058
Guido van Rossumee60eb11998-08-12 17:47:52 +000059def usage(msg, code=2):
60 sys.stdout = sys.stderr
Collin Winter6afaeb72007-08-03 17:06:41 +000061 print(msg)
62 print(__doc__)
Guido van Rossumee60eb11998-08-12 17:47:52 +000063 sys.exit(code)
64
Florent Xiclunae4a33802010-08-09 12:24:20 +000065
Guido van Rossumee60eb11998-08-12 17:47:52 +000066if __name__ == '__main__':
67 main()