Benjamin Peterson | 90f5ba5 | 2010-03-11 22:53:45 +0000 | [diff] [blame] | 1 | #! /usr/bin/env python3 |
Guido van Rossum | ee60eb1 | 1998-08-12 17:47:52 +0000 | [diff] [blame] | 2 | |
| 3 | """Reverse grep. |
| 4 | |
| 5 | Usage: rgrep [-i] pattern file |
| 6 | """ |
| 7 | |
| 8 | import sys |
| 9 | import re |
Guido van Rossum | ee60eb1 | 1998-08-12 17:47:52 +0000 | [diff] [blame] | 10 | import getopt |
| 11 | |
Florent Xicluna | e4a3380 | 2010-08-09 12:24:20 +0000 | [diff] [blame^] | 12 | |
Guido van Rossum | ee60eb1 | 1998-08-12 17:47:52 +0000 | [diff] [blame] | 13 | def main(): |
Florent Xicluna | e4a3380 | 2010-08-09 12:24:20 +0000 | [diff] [blame^] | 14 | bufsize = 64 * 1024 |
Guido van Rossum | ee60eb1 | 1998-08-12 17:47:52 +0000 | [diff] [blame] | 15 | 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 Rossum | b940e11 | 2007-01-10 16:19:56 +0000 | [diff] [blame] | 27 | except re.error as msg: |
Florent Xicluna | e4a3380 | 2010-08-09 12:24:20 +0000 | [diff] [blame^] | 28 | usage("error in regular expression: %s" % msg) |
Guido van Rossum | ee60eb1 | 1998-08-12 17:47:52 +0000 | [diff] [blame] | 29 | try: |
| 30 | f = open(filename) |
Guido van Rossum | b940e11 | 2007-01-10 16:19:56 +0000 | [diff] [blame] | 31 | except IOError as msg: |
Florent Xicluna | e4a3380 | 2010-08-09 12:24:20 +0000 | [diff] [blame^] | 32 | usage("can't open %r: %s" % (filename, msg), 1) |
Guido van Rossum | ee60eb1 | 1998-08-12 17:47:52 +0000 | [diff] [blame] | 33 | f.seek(0, 2) |
| 34 | pos = f.tell() |
| 35 | leftover = None |
| 36 | while pos > 0: |
| 37 | size = min(pos, bufsize) |
| 38 | pos = pos - size |
| 39 | f.seek(pos) |
| 40 | buffer = f.read(size) |
Walter Dörwald | aaab30e | 2002-09-11 20:36:02 +0000 | [diff] [blame] | 41 | lines = buffer.split("\n") |
Guido van Rossum | ee60eb1 | 1998-08-12 17:47:52 +0000 | [diff] [blame] | 42 | del buffer |
| 43 | if leftover is None: |
| 44 | if not lines[-1]: |
| 45 | del lines[-1] |
| 46 | else: |
| 47 | lines[-1] = lines[-1] + leftover |
| 48 | if pos > 0: |
| 49 | leftover = lines[0] |
| 50 | del lines[0] |
| 51 | else: |
| 52 | leftover = None |
Florent Xicluna | e4a3380 | 2010-08-09 12:24:20 +0000 | [diff] [blame^] | 53 | for line in reversed(lines): |
Guido van Rossum | ee60eb1 | 1998-08-12 17:47:52 +0000 | [diff] [blame] | 54 | if prog.search(line): |
Collin Winter | 6afaeb7 | 2007-08-03 17:06:41 +0000 | [diff] [blame] | 55 | print(line) |
Guido van Rossum | ee60eb1 | 1998-08-12 17:47:52 +0000 | [diff] [blame] | 56 | |
Florent Xicluna | e4a3380 | 2010-08-09 12:24:20 +0000 | [diff] [blame^] | 57 | |
Guido van Rossum | ee60eb1 | 1998-08-12 17:47:52 +0000 | [diff] [blame] | 58 | def usage(msg, code=2): |
| 59 | sys.stdout = sys.stderr |
Collin Winter | 6afaeb7 | 2007-08-03 17:06:41 +0000 | [diff] [blame] | 60 | print(msg) |
| 61 | print(__doc__) |
Guido van Rossum | ee60eb1 | 1998-08-12 17:47:52 +0000 | [diff] [blame] | 62 | sys.exit(code) |
| 63 | |
Florent Xicluna | e4a3380 | 2010-08-09 12:24:20 +0000 | [diff] [blame^] | 64 | |
Guido van Rossum | ee60eb1 | 1998-08-12 17:47:52 +0000 | [diff] [blame] | 65 | if __name__ == '__main__': |
| 66 | main() |