2017-02-21 17:46:31 +01:00
|
|
|
import unittest
|
2018-01-20 04:05:24 -06:00
|
|
|
try:
|
|
|
|
|
import builtins
|
|
|
|
|
except ImportError:
|
|
|
|
|
import __builtin__ as builtins
|
2017-02-21 17:46:31 +01:00
|
|
|
|
|
|
|
|
from grep import grep
|
|
|
|
|
|
2017-04-04 14:01:56 +02:00
|
|
|
|
2018-11-05 16:05:18 +01:00
|
|
|
# Tests adapted from `problem-specifications//canonical-data.json` @ v1.2.0
|
2017-04-04 14:01:56 +02:00
|
|
|
|
2017-02-21 17:46:31 +01:00
|
|
|
ILIADFILENAME = 'iliad.txt'
|
|
|
|
|
ILIADCONTENTS = '''Achilles sing, O Goddess! Peleus' son;
|
|
|
|
|
His wrath pernicious, who ten thousand woes
|
|
|
|
|
Caused to Achaia's host, sent many a soul
|
|
|
|
|
Illustrious into Ades premature,
|
|
|
|
|
And Heroes gave (so stood the will of Jove)
|
|
|
|
|
To dogs and to all ravening fowls a prey,
|
|
|
|
|
When fierce dispute had separated once
|
|
|
|
|
The noble Chief Achilles from the son
|
2018-01-20 04:05:24 -06:00
|
|
|
Of Atreus, Agamemnon, King of men.'''
|
2017-02-21 17:46:31 +01:00
|
|
|
|
|
|
|
|
MIDSUMMERNIGHTFILENAME = 'midsummer-night.txt'
|
|
|
|
|
MIDSUMMERNIGHTCONTENTS = '''I do entreat your grace to pardon me.
|
|
|
|
|
I know not by what power I am made bold,
|
|
|
|
|
Nor how it may concern my modesty,
|
|
|
|
|
In such a presence here to plead my thoughts;
|
|
|
|
|
But I beseech your grace that I may know
|
|
|
|
|
The worst that may befall me in this case,
|
2018-01-20 04:05:24 -06:00
|
|
|
If I refuse to wed Demetrius.'''
|
2017-02-21 17:46:31 +01:00
|
|
|
|
|
|
|
|
PARADISELOSTFILENAME = 'paradise-lost.txt'
|
|
|
|
|
PARADISELOSTCONTENTS = '''Of Mans First Disobedience, and the Fruit
|
|
|
|
|
Of that Forbidden Tree, whose mortal tast
|
|
|
|
|
Brought Death into the World, and all our woe,
|
|
|
|
|
With loss of Eden, till one greater Man
|
|
|
|
|
Restore us, and regain the blissful Seat,
|
|
|
|
|
Sing Heav'nly Muse, that on the secret top
|
|
|
|
|
Of Oreb, or of Sinai, didst inspire
|
2018-01-20 04:05:24 -06:00
|
|
|
That Shepherd, who first taught the chosen Seed'''
|
|
|
|
|
FILENAMES = [
|
|
|
|
|
ILIADFILENAME,
|
|
|
|
|
MIDSUMMERNIGHTFILENAME,
|
|
|
|
|
PARADISELOSTFILENAME,
|
|
|
|
|
]
|
|
|
|
|
FILES = {}
|
2017-02-21 17:46:31 +01:00
|
|
|
|
|
|
|
|
|
2019-11-07 08:58:16 -05:00
|
|
|
class File:
|
2018-01-20 04:05:24 -06:00
|
|
|
def __init__(self, name=''):
|
|
|
|
|
self.name = name
|
|
|
|
|
self.contents = ''
|
|
|
|
|
|
|
|
|
|
def read(self):
|
|
|
|
|
return self.contents
|
|
|
|
|
|
|
|
|
|
def readlines(self):
|
|
|
|
|
return [line + '\n' for line in self.read().split('\n') if line]
|
|
|
|
|
|
|
|
|
|
def write(self, data):
|
|
|
|
|
self.contents += data
|
|
|
|
|
|
|
|
|
|
def __enter__(self):
|
|
|
|
|
return self
|
|
|
|
|
|
|
|
|
|
def __exit__(self, *args):
|
2017-02-21 17:46:31 +01:00
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
2018-01-20 04:05:24 -06:00
|
|
|
# Store builtin definition of open()
|
|
|
|
|
builtins.oldopen = builtins.open
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def open(name, mode='r', *args, **kwargs):
|
|
|
|
|
# if name is a mocked file name, lookup corresponding mocked file
|
|
|
|
|
if name in FILENAMES:
|
|
|
|
|
if mode == 'w' or name not in FILES:
|
|
|
|
|
FILES[name] = File(name)
|
|
|
|
|
return FILES[name]
|
|
|
|
|
# else call builtin open()
|
|
|
|
|
else:
|
|
|
|
|
return builtins.oldopen(name, mode, *args, **kwargs)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
builtins.open = open
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# remove mocked file contents
|
|
|
|
|
def remove_file(file_name):
|
|
|
|
|
del FILES[file_name]
|
|
|
|
|
|
|
|
|
|
|
2017-02-21 17:46:31 +01:00
|
|
|
def create_file(name, contents):
|
|
|
|
|
with open(name, 'w') as f:
|
|
|
|
|
f.write(contents)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class GrepTest(unittest.TestCase):
|
|
|
|
|
@classmethod
|
|
|
|
|
def setUpClass(self):
|
2018-01-20 04:05:24 -06:00
|
|
|
# Override builtin open() with mock-file-enabled one
|
|
|
|
|
builtins.open = open
|
2017-02-21 17:46:31 +01:00
|
|
|
create_file(ILIADFILENAME, ILIADCONTENTS)
|
|
|
|
|
create_file(MIDSUMMERNIGHTFILENAME, MIDSUMMERNIGHTCONTENTS)
|
|
|
|
|
create_file(PARADISELOSTFILENAME, PARADISELOSTCONTENTS)
|
2018-11-05 16:05:18 +01:00
|
|
|
self.maxDiff = None
|
2017-02-21 17:46:31 +01:00
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
|
def tearDownClass(self):
|
|
|
|
|
remove_file(ILIADFILENAME)
|
|
|
|
|
remove_file(MIDSUMMERNIGHTFILENAME)
|
|
|
|
|
remove_file(PARADISELOSTFILENAME)
|
2018-01-20 04:05:24 -06:00
|
|
|
# Restore builtin open()
|
|
|
|
|
builtins.open = builtins.oldopen
|
2017-02-21 17:46:31 +01:00
|
|
|
|
|
|
|
|
def test_one_file_one_match_no_flags(self):
|
|
|
|
|
self.assertMultiLineEqual(
|
2019-05-28 06:37:11 -07:00
|
|
|
grep("Agamemnon", "", [ILIADFILENAME]),
|
2018-01-20 04:05:24 -06:00
|
|
|
"Of Atreus, Agamemnon, King of men.\n"
|
|
|
|
|
)
|
2017-02-21 17:46:31 +01:00
|
|
|
|
|
|
|
|
def test_one_file_one_match_print_line_numbers_flag(self):
|
|
|
|
|
self.assertMultiLineEqual(
|
2019-05-28 06:37:11 -07:00
|
|
|
grep("Forbidden", "-n", [PARADISELOSTFILENAME]),
|
2018-01-20 04:05:24 -06:00
|
|
|
"2:Of that Forbidden Tree, whose mortal tast\n"
|
|
|
|
|
)
|
2017-02-21 17:46:31 +01:00
|
|
|
|
|
|
|
|
def test_one_file_one_match_case_insensitive_flag(self):
|
|
|
|
|
self.assertMultiLineEqual(
|
2019-05-28 06:37:11 -07:00
|
|
|
grep("FORBIDDEN", "-i", [PARADISELOSTFILENAME]),
|
2018-01-20 04:05:24 -06:00
|
|
|
"Of that Forbidden Tree, whose mortal tast\n"
|
|
|
|
|
)
|
2017-02-21 17:46:31 +01:00
|
|
|
|
|
|
|
|
def test_one_file_one_match_print_file_names_flag(self):
|
|
|
|
|
self.assertMultiLineEqual(
|
2019-05-28 06:37:11 -07:00
|
|
|
grep("Forbidden", "-l", [PARADISELOSTFILENAME]),
|
|
|
|
|
PARADISELOSTFILENAME + '\n')
|
2017-02-21 17:46:31 +01:00
|
|
|
|
|
|
|
|
def test_one_file_one_match_match_entire_lines_flag(self):
|
|
|
|
|
self.assertMultiLineEqual(
|
2017-03-23 13:37:20 +01:00
|
|
|
grep("With loss of Eden, till one greater Man",
|
2019-05-28 06:37:11 -07:00
|
|
|
"-x", [PARADISELOSTFILENAME]),
|
|
|
|
|
"With loss of Eden, till one greater Man\n")
|
2017-02-21 17:46:31 +01:00
|
|
|
|
|
|
|
|
def test_one_file_one_match_multiple_flags(self):
|
|
|
|
|
self.assertMultiLineEqual(
|
2019-05-28 06:37:11 -07:00
|
|
|
grep("OF ATREUS, Agamemnon, KIng of MEN.",
|
|
|
|
|
"-n -i -x", [ILIADFILENAME]),
|
|
|
|
|
"9:Of Atreus, Agamemnon, King of men.\n")
|
2017-02-21 17:46:31 +01:00
|
|
|
|
|
|
|
|
def test_one_file_several_matches_no_flags(self):
|
|
|
|
|
self.assertMultiLineEqual(
|
2019-05-28 06:37:11 -07:00
|
|
|
grep("may", "", [MIDSUMMERNIGHTFILENAME]),
|
2018-01-20 04:05:24 -06:00
|
|
|
"Nor how it may concern my modesty,\n"
|
|
|
|
|
"But I beseech your grace that I may know\n"
|
2019-05-28 06:37:11 -07:00
|
|
|
"The worst that may befall me in this case,\n")
|
2017-02-21 17:46:31 +01:00
|
|
|
|
|
|
|
|
def test_one_file_several_matches_print_line_numbers_flag(self):
|
|
|
|
|
self.assertMultiLineEqual(
|
2019-05-28 06:37:11 -07:00
|
|
|
grep("may", "-n", [MIDSUMMERNIGHTFILENAME]),
|
2018-01-20 04:05:24 -06:00
|
|
|
"3:Nor how it may concern my modesty,\n"
|
|
|
|
|
"5:But I beseech your grace that I may know\n"
|
2019-05-28 06:37:11 -07:00
|
|
|
"6:The worst that may befall me in this case,\n")
|
2017-02-21 17:46:31 +01:00
|
|
|
|
|
|
|
|
def test_one_file_several_matches_match_entire_lines_flag(self):
|
|
|
|
|
self.assertMultiLineEqual(
|
2019-05-28 06:37:11 -07:00
|
|
|
grep("may", "-x", [MIDSUMMERNIGHTFILENAME]),
|
|
|
|
|
"")
|
2017-02-21 17:46:31 +01:00
|
|
|
|
|
|
|
|
def test_one_file_several_matches_case_insensitive_flag(self):
|
|
|
|
|
self.assertMultiLineEqual(
|
2019-05-28 06:37:11 -07:00
|
|
|
grep("ACHILLES", "-i", [ILIADFILENAME]),
|
2018-01-20 04:05:24 -06:00
|
|
|
"Achilles sing, O Goddess! Peleus' son;\n"
|
2019-05-28 06:37:11 -07:00
|
|
|
"The noble Chief Achilles from the son\n")
|
2017-02-21 17:46:31 +01:00
|
|
|
|
|
|
|
|
def test_one_file_several_matches_inverted_flag(self):
|
|
|
|
|
self.assertMultiLineEqual(
|
2019-05-28 06:37:11 -07:00
|
|
|
grep("Of", "-v", [PARADISELOSTFILENAME]),
|
2018-01-20 04:05:24 -06:00
|
|
|
"Brought Death into the World, and all our woe,\n"
|
|
|
|
|
"With loss of Eden, till one greater Man\n"
|
|
|
|
|
"Restore us, and regain the blissful Seat,\n"
|
|
|
|
|
"Sing Heav'nly Muse, that on the secret top\n"
|
2019-05-28 06:37:11 -07:00
|
|
|
"That Shepherd, who first taught the chosen Seed\n")
|
2017-02-21 17:46:31 +01:00
|
|
|
|
2018-11-05 16:05:18 +01:00
|
|
|
def test_one_file_one_match_file_flag_takes_precedence_over_line(self):
|
|
|
|
|
self.assertMultiLineEqual(
|
2019-05-28 06:37:11 -07:00
|
|
|
grep("ten", "-n -l", [ILIADFILENAME]),
|
|
|
|
|
ILIADFILENAME + '\n')
|
2018-11-05 16:05:18 +01:00
|
|
|
|
2017-02-21 17:46:31 +01:00
|
|
|
def test_one_file_no_matches_various_flags(self):
|
|
|
|
|
self.assertMultiLineEqual(
|
2019-05-28 06:37:11 -07:00
|
|
|
grep("Gandalf", "-n -l -x -i", [ILIADFILENAME]),
|
|
|
|
|
"")
|
2017-02-21 17:46:31 +01:00
|
|
|
|
|
|
|
|
def test_multiple_files_one_match_no_flags(self):
|
|
|
|
|
self.assertMultiLineEqual(
|
2019-05-28 06:37:11 -07:00
|
|
|
grep("Agamemnon", "", FILENAMES),
|
|
|
|
|
"iliad.txt:Of Atreus, Agamemnon, King of men.\n")
|
2017-02-21 17:46:31 +01:00
|
|
|
|
|
|
|
|
def test_multiple_files_several_matches_no_flags(self):
|
|
|
|
|
self.assertMultiLineEqual(
|
2019-05-28 06:37:11 -07:00
|
|
|
grep("may", "", FILENAMES),
|
2018-01-20 04:05:24 -06:00
|
|
|
"midsummer-night.txt:Nor how it may concern my modesty,\n"
|
|
|
|
|
"midsummer-night.txt:But I beseech your grace that I may know\n"
|
2019-05-28 06:37:11 -07:00
|
|
|
"midsummer-night.txt:The worst that may befall me in this case,\n")
|
2017-02-21 17:46:31 +01:00
|
|
|
|
|
|
|
|
def test_multiple_files_several_matches_print_line_numbers_flag(self):
|
2018-01-20 04:05:24 -06:00
|
|
|
expected = (
|
|
|
|
|
"midsummer-night.txt:5:But I beseech your grace that I may know\n"
|
|
|
|
|
"midsummer-night.txt:6:The worst that may befall me in this case,"
|
|
|
|
|
"\nparadise-lost.txt:2:Of that Forbidden Tree, whose mortal tast\n"
|
2019-05-28 06:37:11 -07:00
|
|
|
"paradise-lost.txt:6:Sing Heav'nly Muse, that on the secret top\n")
|
|
|
|
|
self.assertMultiLineEqual(grep("that", "-n", FILENAMES), expected)
|
2017-02-21 17:46:31 +01:00
|
|
|
|
|
|
|
|
def test_multiple_files_one_match_print_file_names_flag(self):
|
|
|
|
|
self.assertMultiLineEqual(
|
2019-05-28 06:37:11 -07:00
|
|
|
grep("who", "-l", FILENAMES),
|
2018-01-20 04:05:24 -06:00
|
|
|
ILIADFILENAME + '\n' + PARADISELOSTFILENAME + '\n')
|
2017-02-21 17:46:31 +01:00
|
|
|
|
|
|
|
|
def test_multiple_files_several_matches_case_insensitive_flag(self):
|
2018-01-20 04:05:24 -06:00
|
|
|
expected = (
|
|
|
|
|
"iliad.txt:Caused to Achaia's host, sent many a soul\n"
|
|
|
|
|
"iliad.txt:Illustrious into Ades premature,\n"
|
|
|
|
|
"iliad.txt:And Heroes gave (so stood the will of Jove)\n"
|
|
|
|
|
"iliad.txt:To dogs and to all ravening fowls a prey,\n"
|
|
|
|
|
"midsummer-night.txt:I do entreat your grace to pardon me.\n"
|
|
|
|
|
"midsummer-night.txt:In such a presence here to plead my thoughts;"
|
|
|
|
|
"\nmidsummer-night.txt:If I refuse to wed Demetrius.\n"
|
|
|
|
|
"paradise-lost.txt:Brought Death into the World, and all our woe,"
|
|
|
|
|
"\nparadise-lost.txt:Restore us, and regain the blissful Seat,\n"
|
2019-05-28 06:37:11 -07:00
|
|
|
"paradise-lost.txt:Sing Heav'nly Muse, that on the secret top\n")
|
|
|
|
|
self.assertMultiLineEqual(grep("TO", "-i", FILENAMES), expected)
|
2017-02-21 17:46:31 +01:00
|
|
|
|
|
|
|
|
def test_multiple_files_several_matches_inverted_flag(self):
|
|
|
|
|
self.assertMultiLineEqual(
|
2019-05-28 06:37:11 -07:00
|
|
|
grep("a", "-v", FILENAMES),
|
2018-01-20 04:05:24 -06:00
|
|
|
"iliad.txt:Achilles sing, O Goddess! Peleus' son;\n"
|
|
|
|
|
"iliad.txt:The noble Chief Achilles from the son\n"
|
|
|
|
|
"midsummer-night.txt:If I refuse to wed Demetrius.\n"
|
|
|
|
|
)
|
2017-02-21 17:46:31 +01:00
|
|
|
|
|
|
|
|
def test_multiple_files_one_match_match_entire_lines_flag(self):
|
|
|
|
|
self.assertMultiLineEqual(
|
2019-05-28 06:37:11 -07:00
|
|
|
grep("But I beseech your grace that I may know", "-x", FILENAMES),
|
2017-03-23 13:37:20 +01:00
|
|
|
"midsummer-night.txt:But I beseech your grace that I may know\n")
|
2017-02-21 17:46:31 +01:00
|
|
|
|
|
|
|
|
def test_multiple_files_one_match_multiple_flags(self):
|
|
|
|
|
self.assertMultiLineEqual(
|
2019-05-28 06:37:11 -07:00
|
|
|
grep("WITH LOSS OF EDEN, TILL ONE GREATER MAN", "-n -i -x",
|
|
|
|
|
FILENAMES),
|
2017-03-23 13:37:20 +01:00
|
|
|
"paradise-lost.txt:4:With loss of Eden, till one greater Man\n")
|
2017-02-21 17:46:31 +01:00
|
|
|
|
|
|
|
|
def test_multiple_files_no_matches_various_flags(self):
|
|
|
|
|
self.assertMultiLineEqual(
|
2019-05-28 06:37:11 -07:00
|
|
|
grep("Frodo", "-n -l -x -i", FILENAMES),
|
2018-01-20 04:05:24 -06:00
|
|
|
""
|
|
|
|
|
)
|
2017-02-21 17:46:31 +01:00
|
|
|
|
2018-11-05 16:05:18 +01:00
|
|
|
def test_multiple_files_several_matches_file_flag_takes_precedence(self):
|
|
|
|
|
self.assertMultiLineEqual(
|
2019-05-28 06:37:11 -07:00
|
|
|
grep("who", "-n -l", FILENAMES),
|
|
|
|
|
ILIADFILENAME + '\n' + PARADISELOSTFILENAME + '\n')
|
2018-11-05 16:05:18 +01:00
|
|
|
|
|
|
|
|
def test_multiple_files_several_matches_inverted_match_entire_lines(self):
|
|
|
|
|
expected = (
|
|
|
|
|
"iliad.txt:Achilles sing, O Goddess! Peleus' son;\n"
|
|
|
|
|
"iliad.txt:His wrath pernicious, who ten thousand woes\n"
|
|
|
|
|
"iliad.txt:Caused to Achaia's host, sent many a soul\n"
|
|
|
|
|
"iliad.txt:And Heroes gave (so stood the will of Jove)\n"
|
|
|
|
|
"iliad.txt:To dogs and to all ravening fowls a prey,\n"
|
|
|
|
|
"iliad.txt:When fierce dispute had separated once\n"
|
|
|
|
|
"iliad.txt:The noble Chief Achilles from the son\n"
|
|
|
|
|
"iliad.txt:Of Atreus, Agamemnon, King of men.\n"
|
|
|
|
|
"midsummer-night.txt:I do entreat your grace to pardon me.\n"
|
|
|
|
|
"midsummer-night.txt:I know not by what power I am made bold,\n"
|
|
|
|
|
"midsummer-night.txt:Nor how it may concern my modesty,\n"
|
|
|
|
|
"midsummer-night.txt:In such a presence here to plead my thoughts;"
|
|
|
|
|
"\nmidsummer-night.txt:But I beseech your grace that I may know\n"
|
|
|
|
|
"midsummer-night.txt:The worst that may befall me in this case,\n"
|
|
|
|
|
"midsummer-night.txt:If I refuse to wed Demetrius.\n"
|
|
|
|
|
"paradise-lost.txt:Of Mans First Disobedience, and the Fruit\n"
|
|
|
|
|
"paradise-lost.txt:Of that Forbidden Tree, whose mortal tast\n"
|
|
|
|
|
"paradise-lost.txt:Brought Death into the World, and all our woe,"
|
|
|
|
|
"\nparadise-lost.txt:With loss of Eden, till one greater Man\n"
|
|
|
|
|
"paradise-lost.txt:Restore us, and regain the blissful Seat,\n"
|
|
|
|
|
"paradise-lost.txt:Sing Heav'nly Muse, that on the secret top\n"
|
|
|
|
|
"paradise-lost.txt:Of Oreb, or of Sinai, didst inspire\n"
|
|
|
|
|
"paradise-lost.txt:That Shepherd, who first taught the chosen Seed"
|
|
|
|
|
"\n"
|
|
|
|
|
)
|
|
|
|
|
self.assertMultiLineEqual(
|
2019-05-28 06:37:11 -07:00
|
|
|
grep("Illustrious into Ades premature,", "-x -v", FILENAMES),
|
2018-11-05 16:05:18 +01:00
|
|
|
expected
|
|
|
|
|
)
|
|
|
|
|
|
2017-02-21 17:46:31 +01:00
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
|
unittest.main()
|