Fred Drake | 9175114 | 2001-03-21 18:29:25 +0000 | [diff] [blame] | 1 | """Test cases for the fnmatch module.""" |
| 2 | |
Fred Drake | 9175114 | 2001-03-21 18:29:25 +0000 | [diff] [blame] | 3 | import unittest |
Serhiy Storchaka | 8175547 | 2017-05-21 08:57:00 +0300 | [diff] [blame] | 4 | import os |
Serhiy Storchaka | 23cdbfa | 2018-02-09 13:30:19 +0200 | [diff] [blame] | 5 | import warnings |
Fred Drake | 9175114 | 2001-03-21 18:29:25 +0000 | [diff] [blame] | 6 | |
Antoine Pitrou | 6fdb74f | 2010-08-13 16:26:40 +0000 | [diff] [blame] | 7 | from fnmatch import fnmatch, fnmatchcase, translate, filter |
Fred Drake | 9175114 | 2001-03-21 18:29:25 +0000 | [diff] [blame] | 8 | |
| 9 | class FnmatchTestCase(unittest.TestCase): |
R. David Murray | 0425a8e | 2010-07-10 13:52:13 +0000 | [diff] [blame] | 10 | |
Serhiy Storchaka | 8175547 | 2017-05-21 08:57:00 +0300 | [diff] [blame] | 11 | def check_match(self, filename, pattern, should_match=True, fn=fnmatch): |
Fred Drake | 9175114 | 2001-03-21 18:29:25 +0000 | [diff] [blame] | 12 | if should_match: |
Georg Brandl | c0e22b7 | 2010-03-14 10:51:01 +0000 | [diff] [blame] | 13 | self.assertTrue(fn(filename, pattern), |
Fred Drake | 9175114 | 2001-03-21 18:29:25 +0000 | [diff] [blame] | 14 | "expected %r to match pattern %r" |
| 15 | % (filename, pattern)) |
| 16 | else: |
Serhiy Storchaka | 8175547 | 2017-05-21 08:57:00 +0300 | [diff] [blame] | 17 | self.assertFalse(fn(filename, pattern), |
Fred Drake | 9175114 | 2001-03-21 18:29:25 +0000 | [diff] [blame] | 18 | "expected %r not to match pattern %r" |
| 19 | % (filename, pattern)) |
| 20 | |
| 21 | def test_fnmatch(self): |
| 22 | check = self.check_match |
| 23 | check('abc', 'abc') |
| 24 | check('abc', '?*?') |
| 25 | check('abc', '???*') |
| 26 | check('abc', '*???') |
| 27 | check('abc', '???') |
| 28 | check('abc', '*') |
| 29 | check('abc', 'ab[cd]') |
| 30 | check('abc', 'ab[!de]') |
Serhiy Storchaka | 8175547 | 2017-05-21 08:57:00 +0300 | [diff] [blame] | 31 | check('abc', 'ab[de]', False) |
| 32 | check('a', '??', False) |
| 33 | check('a', 'b', False) |
Fred Drake | 9175114 | 2001-03-21 18:29:25 +0000 | [diff] [blame] | 34 | |
| 35 | # these test that '\' is handled correctly in character sets; |
Gregory P. Smith | 0109970 | 2009-08-16 18:58:46 +0000 | [diff] [blame] | 36 | # see SF bug #409651 |
Fred Drake | 9175114 | 2001-03-21 18:29:25 +0000 | [diff] [blame] | 37 | check('\\', r'[\]') |
| 38 | check('a', r'[!\]') |
Serhiy Storchaka | 8175547 | 2017-05-21 08:57:00 +0300 | [diff] [blame] | 39 | check('\\', r'[!\]', False) |
Fred Drake | 9175114 | 2001-03-21 18:29:25 +0000 | [diff] [blame] | 40 | |
Gregory P. Smith | 0109970 | 2009-08-16 18:58:46 +0000 | [diff] [blame] | 41 | # test that filenames with newlines in them are handled correctly. |
| 42 | # http://bugs.python.org/issue6665 |
| 43 | check('foo\nbar', 'foo*') |
| 44 | check('foo\nbar\n', 'foo*') |
| 45 | check('\nfoo', 'foo*', False) |
| 46 | check('\n', '*') |
| 47 | |
Tim Peters | b9c46a2 | 2020-05-05 21:28:24 -0500 | [diff] [blame] | 48 | def test_slow_fnmatch(self): |
| 49 | check = self.check_match |
| 50 | check('a' * 50, '*a*a*a*a*a*a*a*a*a*a') |
| 51 | # The next "takes forever" if the regexp translation is |
| 52 | # straightforward. See bpo-40480. |
| 53 | check('a' * 50 + 'b', '*a*a*a*a*a*a*a*a*a*a', False) |
| 54 | |
Guido van Rossum | f0af3e3 | 2008-10-02 18:55:37 +0000 | [diff] [blame] | 55 | def test_mix_bytes_str(self): |
| 56 | self.assertRaises(TypeError, fnmatch, 'test', b'*') |
| 57 | self.assertRaises(TypeError, fnmatch, b'test', '*') |
| 58 | self.assertRaises(TypeError, fnmatchcase, 'test', b'*') |
| 59 | self.assertRaises(TypeError, fnmatchcase, b'test', '*') |
| 60 | |
Georg Brandl | 89fad14 | 2010-03-14 10:23:39 +0000 | [diff] [blame] | 61 | def test_fnmatchcase(self): |
| 62 | check = self.check_match |
Serhiy Storchaka | 8175547 | 2017-05-21 08:57:00 +0300 | [diff] [blame] | 63 | check('abc', 'abc', True, fnmatchcase) |
| 64 | check('AbC', 'abc', False, fnmatchcase) |
| 65 | check('abc', 'AbC', False, fnmatchcase) |
| 66 | check('AbC', 'AbC', True, fnmatchcase) |
| 67 | |
| 68 | check('usr/bin', 'usr/bin', True, fnmatchcase) |
| 69 | check('usr\\bin', 'usr/bin', False, fnmatchcase) |
| 70 | check('usr/bin', 'usr\\bin', False, fnmatchcase) |
| 71 | check('usr\\bin', 'usr\\bin', True, fnmatchcase) |
Georg Brandl | 89fad14 | 2010-03-14 10:23:39 +0000 | [diff] [blame] | 72 | |
Guido van Rossum | f0af3e3 | 2008-10-02 18:55:37 +0000 | [diff] [blame] | 73 | def test_bytes(self): |
| 74 | self.check_match(b'test', b'te*') |
| 75 | self.check_match(b'test\xff', b'te*\xff') |
Gregory P. Smith | 0109970 | 2009-08-16 18:58:46 +0000 | [diff] [blame] | 76 | self.check_match(b'foo\nbar', b'foo*') |
| 77 | |
Serhiy Storchaka | 8175547 | 2017-05-21 08:57:00 +0300 | [diff] [blame] | 78 | def test_case(self): |
| 79 | ignorecase = os.path.normcase('ABC') == os.path.normcase('abc') |
| 80 | check = self.check_match |
| 81 | check('abc', 'abc') |
| 82 | check('AbC', 'abc', ignorecase) |
| 83 | check('abc', 'AbC', ignorecase) |
| 84 | check('AbC', 'AbC') |
| 85 | |
| 86 | def test_sep(self): |
| 87 | normsep = os.path.normcase('\\') == os.path.normcase('/') |
| 88 | check = self.check_match |
| 89 | check('usr/bin', 'usr/bin') |
| 90 | check('usr\\bin', 'usr/bin', normsep) |
| 91 | check('usr/bin', 'usr\\bin', normsep) |
| 92 | check('usr\\bin', 'usr\\bin') |
| 93 | |
Serhiy Storchaka | 23cdbfa | 2018-02-09 13:30:19 +0200 | [diff] [blame] | 94 | def test_warnings(self): |
| 95 | with warnings.catch_warnings(): |
| 96 | warnings.simplefilter('error', Warning) |
| 97 | check = self.check_match |
| 98 | check('[', '[[]') |
| 99 | check('&', '[a&&b]') |
| 100 | check('|', '[a||b]') |
| 101 | check('~', '[a~~b]') |
| 102 | check(',', '[a-z+--A-Z]') |
| 103 | check('.', '[a-z--/A-Z]') |
| 104 | |
Serhiy Storchaka | 8175547 | 2017-05-21 08:57:00 +0300 | [diff] [blame] | 105 | |
Brett Cannon | 4b16e13 | 2010-07-23 16:23:13 +0000 | [diff] [blame] | 106 | class TranslateTestCase(unittest.TestCase): |
| 107 | |
| 108 | def test_translate(self): |
Tim Peters | b1b4c79 | 2020-05-11 21:19:20 -0500 | [diff] [blame] | 109 | import re |
Serhiy Storchaka | bd48d27 | 2016-09-11 12:50:02 +0300 | [diff] [blame] | 110 | self.assertEqual(translate('*'), r'(?s:.*)\Z') |
| 111 | self.assertEqual(translate('?'), r'(?s:.)\Z') |
| 112 | self.assertEqual(translate('a?b*'), r'(?s:a.b.*)\Z') |
| 113 | self.assertEqual(translate('[abc]'), r'(?s:[abc])\Z') |
| 114 | self.assertEqual(translate('[]]'), r'(?s:[]])\Z') |
| 115 | self.assertEqual(translate('[!x]'), r'(?s:[^x])\Z') |
| 116 | self.assertEqual(translate('[^x]'), r'(?s:[\^x])\Z') |
| 117 | self.assertEqual(translate('[x'), r'(?s:\[x)\Z') |
Tim Peters | b9c46a2 | 2020-05-05 21:28:24 -0500 | [diff] [blame] | 118 | # from the docs |
| 119 | self.assertEqual(translate('*.txt'), r'(?s:.*\.txt)\Z') |
| 120 | # squash consecutive stars |
| 121 | self.assertEqual(translate('*********'), r'(?s:.*)\Z') |
| 122 | self.assertEqual(translate('A*********'), r'(?s:A.*)\Z') |
| 123 | self.assertEqual(translate('*********A'), r'(?s:.*A)\Z') |
| 124 | self.assertEqual(translate('A*********?[?]?'), r'(?s:A.*.[?].)\Z') |
| 125 | # fancy translation to prevent exponential-time match failure |
Tim Peters | b1b4c79 | 2020-05-11 21:19:20 -0500 | [diff] [blame] | 126 | t = translate('**a*a****a') |
| 127 | digits = re.findall(r'\d+', t) |
| 128 | self.assertEqual(len(digits), 4) |
| 129 | self.assertEqual(digits[0], digits[1]) |
| 130 | self.assertEqual(digits[2], digits[3]) |
| 131 | g1 = f"g{digits[0]}" # e.g., group name "g4" |
| 132 | g2 = f"g{digits[2]}" # e.g., group name "g5" |
| 133 | self.assertEqual(t, |
| 134 | fr'(?s:(?=(?P<{g1}>.*?a))(?P={g1})(?=(?P<{g2}>.*?a))(?P={g2}).*a)\Z') |
| 135 | # and try pasting multiple translate results - it's an undocumented |
| 136 | # feature that this works; all the pain of generating unique group |
| 137 | # names across calls exists to support this |
| 138 | r1 = translate('**a**a**a*') |
| 139 | r2 = translate('**b**b**b*') |
| 140 | r3 = translate('*c*c*c*') |
| 141 | fatre = "|".join([r1, r2, r3]) |
| 142 | self.assertTrue(re.match(fatre, 'abaccad')) |
| 143 | self.assertTrue(re.match(fatre, 'abxbcab')) |
| 144 | self.assertTrue(re.match(fatre, 'cbabcaxc')) |
| 145 | self.assertFalse(re.match(fatre, 'dabccbad')) |
Brett Cannon | 4b16e13 | 2010-07-23 16:23:13 +0000 | [diff] [blame] | 146 | |
| 147 | class FilterTestCase(unittest.TestCase): |
| 148 | |
| 149 | def test_filter(self): |
Serhiy Storchaka | 8175547 | 2017-05-21 08:57:00 +0300 | [diff] [blame] | 150 | self.assertEqual(filter(['Python', 'Ruby', 'Perl', 'Tcl'], 'P*'), |
| 151 | ['Python', 'Perl']) |
| 152 | self.assertEqual(filter([b'Python', b'Ruby', b'Perl', b'Tcl'], b'P*'), |
| 153 | [b'Python', b'Perl']) |
| 154 | |
| 155 | def test_mix_bytes_str(self): |
| 156 | self.assertRaises(TypeError, filter, ['test'], b'*') |
| 157 | self.assertRaises(TypeError, filter, [b'test'], '*') |
| 158 | |
| 159 | def test_case(self): |
| 160 | ignorecase = os.path.normcase('P') == os.path.normcase('p') |
| 161 | self.assertEqual(filter(['Test.py', 'Test.rb', 'Test.PL'], '*.p*'), |
| 162 | ['Test.py', 'Test.PL'] if ignorecase else ['Test.py']) |
| 163 | self.assertEqual(filter(['Test.py', 'Test.rb', 'Test.PL'], '*.P*'), |
| 164 | ['Test.py', 'Test.PL'] if ignorecase else ['Test.PL']) |
| 165 | |
| 166 | def test_sep(self): |
| 167 | normsep = os.path.normcase('\\') == os.path.normcase('/') |
| 168 | self.assertEqual(filter(['usr/bin', 'usr', 'usr\\lib'], 'usr/*'), |
| 169 | ['usr/bin', 'usr\\lib'] if normsep else ['usr/bin']) |
| 170 | self.assertEqual(filter(['usr/bin', 'usr', 'usr\\lib'], 'usr\\*'), |
| 171 | ['usr/bin', 'usr\\lib'] if normsep else ['usr\\lib']) |
Brett Cannon | 4b16e13 | 2010-07-23 16:23:13 +0000 | [diff] [blame] | 172 | |
| 173 | |
Fred Drake | 2e2be37 | 2001-09-20 21:33:42 +0000 | [diff] [blame] | 174 | if __name__ == "__main__": |
Zachary Ware | 38c707e | 2015-04-13 15:00:43 -0500 | [diff] [blame] | 175 | unittest.main() |