| 
									
										
										
										
											2001-03-21 18:29:25 +00:00
										 |  |  | """Test cases for the fnmatch module.""" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-21 08:57:00 +03:00
										 |  |  | import os | 
					
						
							| 
									
										
										
										
											2022-06-05 11:46:29 +03:00
										 |  |  | import string | 
					
						
							| 
									
										
										
										
											2025-04-08 12:11:25 +02:00
										 |  |  | import unittest | 
					
						
							| 
									
										
										
										
											2018-02-09 13:30:19 +02:00
										 |  |  | import warnings | 
					
						
							| 
									
										
										
										
											2025-04-08 12:11:25 +02:00
										 |  |  | from fnmatch import fnmatch, fnmatchcase, translate, filter, filterfalse | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | IGNORECASE = os.path.normcase('P') == os.path.normcase('p') | 
					
						
							|  |  |  | NORMSEP = os.path.normcase('\\') == os.path.normcase('/') | 
					
						
							| 
									
										
										
										
											2001-03-21 18:29:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class FnmatchTestCase(unittest.TestCase): | 
					
						
							| 
									
										
										
										
											2010-07-10 13:52:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-21 08:57:00 +03:00
										 |  |  |     def check_match(self, filename, pattern, should_match=True, fn=fnmatch): | 
					
						
							| 
									
										
										
										
											2001-03-21 18:29:25 +00:00
										 |  |  |         if should_match: | 
					
						
							| 
									
										
											  
											
												Merged revisions 78101,78115,78117,78182,78188,78245,78386,78496 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/trunk
........
  r78101 | georg.brandl | 2010-02-08 01:04:54 +0100 (Mo, 08 Feb 2010) | 1 line
  Fix test_fnmatch.
........
  r78115 | georg.brandl | 2010-02-08 23:40:51 +0100 (Mo, 08 Feb 2010) | 1 line
  Fix missing string formatting placeholder.
........
  r78117 | georg.brandl | 2010-02-08 23:48:37 +0100 (Mo, 08 Feb 2010) | 1 line
  Convert test failure from output-producing to self.fail().
........
  r78182 | georg.brandl | 2010-02-14 09:18:23 +0100 (So, 14 Feb 2010) | 1 line
  #7926: fix stray parens.
........
  r78188 | georg.brandl | 2010-02-14 14:38:12 +0100 (So, 14 Feb 2010) | 1 line
  #7926: fix-up wording.
........
  r78245 | georg.brandl | 2010-02-19 20:36:08 +0100 (Fr, 19 Feb 2010) | 1 line
  #7967: PyXML is no more.
........
  r78386 | georg.brandl | 2010-02-23 22:48:57 +0100 (Di, 23 Feb 2010) | 1 line
  #6544: fix refleak in kqueue, occurring in certain error conditions.
........
  r78496 | georg.brandl | 2010-02-27 15:58:08 +0100 (Sa, 27 Feb 2010) | 1 line
  Link to http://www.python.org/dev/workflow/ from bugs page.
........
											
										 
											2010-03-14 10:51:01 +00:00
										 |  |  |             self.assertTrue(fn(filename, pattern), | 
					
						
							| 
									
										
										
										
											2001-03-21 18:29:25 +00:00
										 |  |  |                          "expected %r to match pattern %r" | 
					
						
							|  |  |  |                          % (filename, pattern)) | 
					
						
							|  |  |  |         else: | 
					
						
							| 
									
										
										
										
											2017-05-21 08:57:00 +03:00
										 |  |  |             self.assertFalse(fn(filename, pattern), | 
					
						
							| 
									
										
										
										
											2001-03-21 18:29:25 +00:00
										 |  |  |                          "expected %r not to match pattern %r" | 
					
						
							|  |  |  |                          % (filename, pattern)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_fnmatch(self): | 
					
						
							|  |  |  |         check = self.check_match | 
					
						
							|  |  |  |         check('abc', 'abc') | 
					
						
							|  |  |  |         check('abc', '?*?') | 
					
						
							|  |  |  |         check('abc', '???*') | 
					
						
							|  |  |  |         check('abc', '*???') | 
					
						
							|  |  |  |         check('abc', '???') | 
					
						
							|  |  |  |         check('abc', '*') | 
					
						
							|  |  |  |         check('abc', 'ab[cd]') | 
					
						
							|  |  |  |         check('abc', 'ab[!de]') | 
					
						
							| 
									
										
										
										
											2017-05-21 08:57:00 +03:00
										 |  |  |         check('abc', 'ab[de]', False) | 
					
						
							|  |  |  |         check('a', '??', False) | 
					
						
							|  |  |  |         check('a', 'b', False) | 
					
						
							| 
									
										
										
										
											2001-03-21 18:29:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # these test that '\' is handled correctly in character sets; | 
					
						
							| 
									
										
										
										
											2009-08-16 18:58:46 +00:00
										 |  |  |         # see SF bug #409651 | 
					
						
							| 
									
										
										
										
											2001-03-21 18:29:25 +00:00
										 |  |  |         check('\\', r'[\]') | 
					
						
							|  |  |  |         check('a', r'[!\]') | 
					
						
							| 
									
										
										
										
											2017-05-21 08:57:00 +03:00
										 |  |  |         check('\\', r'[!\]', False) | 
					
						
							| 
									
										
										
										
											2001-03-21 18:29:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-16 18:58:46 +00:00
										 |  |  |         # test that filenames with newlines in them are handled correctly. | 
					
						
							|  |  |  |         # http://bugs.python.org/issue6665 | 
					
						
							|  |  |  |         check('foo\nbar', 'foo*') | 
					
						
							|  |  |  |         check('foo\nbar\n', 'foo*') | 
					
						
							|  |  |  |         check('\nfoo', 'foo*', False) | 
					
						
							|  |  |  |         check('\n', '*') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-05 21:28:24 -05:00
										 |  |  |     def test_slow_fnmatch(self): | 
					
						
							|  |  |  |         check = self.check_match | 
					
						
							|  |  |  |         check('a' * 50, '*a*a*a*a*a*a*a*a*a*a') | 
					
						
							|  |  |  |         # The next "takes forever" if the regexp translation is | 
					
						
							|  |  |  |         # straightforward.  See bpo-40480. | 
					
						
							|  |  |  |         check('a' * 50 + 'b', '*a*a*a*a*a*a*a*a*a*a', False) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-10-02 18:55:37 +00:00
										 |  |  |     def test_mix_bytes_str(self): | 
					
						
							|  |  |  |         self.assertRaises(TypeError, fnmatch, 'test', b'*') | 
					
						
							|  |  |  |         self.assertRaises(TypeError, fnmatch, b'test', '*') | 
					
						
							|  |  |  |         self.assertRaises(TypeError, fnmatchcase, 'test', b'*') | 
					
						
							|  |  |  |         self.assertRaises(TypeError, fnmatchcase, b'test', '*') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Merged revisions 78018,78035-78040,78042-78043,78046,78048-78052,78054,78059,78075-78080 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/trunk
........
  r78018 | georg.brandl | 2010-02-06 11:08:21 +0100 (Sa, 06 Feb 2010) | 1 line
  #7864: make deprecation notices a bit clearer.
........
  r78035 | georg.brandl | 2010-02-06 23:44:17 +0100 (Sa, 06 Feb 2010) | 1 line
  Fix duplicate import.
........
  r78036 | georg.brandl | 2010-02-06 23:49:47 +0100 (Sa, 06 Feb 2010) | 1 line
  Remove unused import.
........
  r78037 | georg.brandl | 2010-02-06 23:59:15 +0100 (Sa, 06 Feb 2010) | 1 line
  No need to assign the results of expressions used only for side effects.
........
  r78038 | georg.brandl | 2010-02-07 00:02:29 +0100 (So, 07 Feb 2010) | 1 line
  Add a missing import.
........
  r78039 | georg.brandl | 2010-02-07 00:06:24 +0100 (So, 07 Feb 2010) | 1 line
  Add missing imports.
........
  r78040 | georg.brandl | 2010-02-07 00:08:00 +0100 (So, 07 Feb 2010) | 1 line
  Fix a few UnboundLocalErrors in test_long.
........
  r78042 | georg.brandl | 2010-02-07 00:12:12 +0100 (So, 07 Feb 2010) | 1 line
  Add missing import.
........
  r78043 | georg.brandl | 2010-02-07 00:12:19 +0100 (So, 07 Feb 2010) | 1 line
  Remove duplicate test method.
........
  r78046 | georg.brandl | 2010-02-07 00:18:00 +0100 (So, 07 Feb 2010) | 1 line
  Fix various missing import/unbound name errors.
........
  r78048 | georg.brandl | 2010-02-07 00:23:45 +0100 (So, 07 Feb 2010) | 1 line
  We heard you like test failures so we put unbound locals in your test so that you can fail while you fail.
........
  r78049 | georg.brandl | 2010-02-07 00:33:33 +0100 (So, 07 Feb 2010) | 1 line
  Fix import/access for some identifiers.  _TestSharedCTypes does not seem to be executed?
........
  r78050 | georg.brandl | 2010-02-07 00:34:10 +0100 (So, 07 Feb 2010) | 1 line
  Fix more unbound locals in code paths that do not seem to be used.
........
  r78051 | georg.brandl | 2010-02-07 00:53:52 +0100 (So, 07 Feb 2010) | 1 line
  Add missing import when running these tests standalone.
........
  r78052 | georg.brandl | 2010-02-07 00:54:04 +0100 (So, 07 Feb 2010) | 1 line
  Add missing import when running these tests standalone.
........
  r78054 | georg.brandl | 2010-02-07 00:58:25 +0100 (So, 07 Feb 2010) | 1 line
  Add missing import.
........
  r78059 | georg.brandl | 2010-02-07 12:34:15 +0100 (So, 07 Feb 2010) | 1 line
  Use "regexp" consistently.
........
  r78075 | georg.brandl | 2010-02-07 13:16:12 +0100 (So, 07 Feb 2010) | 1 line
  Fix another duplicated test method.
........
  r78076 | georg.brandl | 2010-02-07 13:19:43 +0100 (So, 07 Feb 2010) | 1 line
  Fix wrong usage of "except X, Y:".
........
  r78077 | georg.brandl | 2010-02-07 13:25:50 +0100 (So, 07 Feb 2010) | 1 line
  Fix two redefined test methods.
........
  r78078 | georg.brandl | 2010-02-07 13:27:06 +0100 (So, 07 Feb 2010) | 1 line
  Fix a redefined test method.
........
  r78079 | georg.brandl | 2010-02-07 13:34:26 +0100 (So, 07 Feb 2010) | 1 line
  Add a minimal test for fnmatchcase().
........
  r78080 | georg.brandl | 2010-02-07 13:55:12 +0100 (So, 07 Feb 2010) | 1 line
  Remove duplicate test method.
........
											
										 
											2010-03-14 10:23:39 +00:00
										 |  |  |     def test_fnmatchcase(self): | 
					
						
							|  |  |  |         check = self.check_match | 
					
						
							| 
									
										
										
										
											2017-05-21 08:57:00 +03:00
										 |  |  |         check('abc', 'abc', True, fnmatchcase) | 
					
						
							|  |  |  |         check('AbC', 'abc', False, fnmatchcase) | 
					
						
							|  |  |  |         check('abc', 'AbC', False, fnmatchcase) | 
					
						
							|  |  |  |         check('AbC', 'AbC', True, fnmatchcase) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         check('usr/bin', 'usr/bin', True, fnmatchcase) | 
					
						
							|  |  |  |         check('usr\\bin', 'usr/bin', False, fnmatchcase) | 
					
						
							|  |  |  |         check('usr/bin', 'usr\\bin', False, fnmatchcase) | 
					
						
							|  |  |  |         check('usr\\bin', 'usr\\bin', True, fnmatchcase) | 
					
						
							| 
									
										
											  
											
												Merged revisions 78018,78035-78040,78042-78043,78046,78048-78052,78054,78059,78075-78080 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/trunk
........
  r78018 | georg.brandl | 2010-02-06 11:08:21 +0100 (Sa, 06 Feb 2010) | 1 line
  #7864: make deprecation notices a bit clearer.
........
  r78035 | georg.brandl | 2010-02-06 23:44:17 +0100 (Sa, 06 Feb 2010) | 1 line
  Fix duplicate import.
........
  r78036 | georg.brandl | 2010-02-06 23:49:47 +0100 (Sa, 06 Feb 2010) | 1 line
  Remove unused import.
........
  r78037 | georg.brandl | 2010-02-06 23:59:15 +0100 (Sa, 06 Feb 2010) | 1 line
  No need to assign the results of expressions used only for side effects.
........
  r78038 | georg.brandl | 2010-02-07 00:02:29 +0100 (So, 07 Feb 2010) | 1 line
  Add a missing import.
........
  r78039 | georg.brandl | 2010-02-07 00:06:24 +0100 (So, 07 Feb 2010) | 1 line
  Add missing imports.
........
  r78040 | georg.brandl | 2010-02-07 00:08:00 +0100 (So, 07 Feb 2010) | 1 line
  Fix a few UnboundLocalErrors in test_long.
........
  r78042 | georg.brandl | 2010-02-07 00:12:12 +0100 (So, 07 Feb 2010) | 1 line
  Add missing import.
........
  r78043 | georg.brandl | 2010-02-07 00:12:19 +0100 (So, 07 Feb 2010) | 1 line
  Remove duplicate test method.
........
  r78046 | georg.brandl | 2010-02-07 00:18:00 +0100 (So, 07 Feb 2010) | 1 line
  Fix various missing import/unbound name errors.
........
  r78048 | georg.brandl | 2010-02-07 00:23:45 +0100 (So, 07 Feb 2010) | 1 line
  We heard you like test failures so we put unbound locals in your test so that you can fail while you fail.
........
  r78049 | georg.brandl | 2010-02-07 00:33:33 +0100 (So, 07 Feb 2010) | 1 line
  Fix import/access for some identifiers.  _TestSharedCTypes does not seem to be executed?
........
  r78050 | georg.brandl | 2010-02-07 00:34:10 +0100 (So, 07 Feb 2010) | 1 line
  Fix more unbound locals in code paths that do not seem to be used.
........
  r78051 | georg.brandl | 2010-02-07 00:53:52 +0100 (So, 07 Feb 2010) | 1 line
  Add missing import when running these tests standalone.
........
  r78052 | georg.brandl | 2010-02-07 00:54:04 +0100 (So, 07 Feb 2010) | 1 line
  Add missing import when running these tests standalone.
........
  r78054 | georg.brandl | 2010-02-07 00:58:25 +0100 (So, 07 Feb 2010) | 1 line
  Add missing import.
........
  r78059 | georg.brandl | 2010-02-07 12:34:15 +0100 (So, 07 Feb 2010) | 1 line
  Use "regexp" consistently.
........
  r78075 | georg.brandl | 2010-02-07 13:16:12 +0100 (So, 07 Feb 2010) | 1 line
  Fix another duplicated test method.
........
  r78076 | georg.brandl | 2010-02-07 13:19:43 +0100 (So, 07 Feb 2010) | 1 line
  Fix wrong usage of "except X, Y:".
........
  r78077 | georg.brandl | 2010-02-07 13:25:50 +0100 (So, 07 Feb 2010) | 1 line
  Fix two redefined test methods.
........
  r78078 | georg.brandl | 2010-02-07 13:27:06 +0100 (So, 07 Feb 2010) | 1 line
  Fix a redefined test method.
........
  r78079 | georg.brandl | 2010-02-07 13:34:26 +0100 (So, 07 Feb 2010) | 1 line
  Add a minimal test for fnmatchcase().
........
  r78080 | georg.brandl | 2010-02-07 13:55:12 +0100 (So, 07 Feb 2010) | 1 line
  Remove duplicate test method.
........
											
										 
											2010-03-14 10:23:39 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-10-02 18:55:37 +00:00
										 |  |  |     def test_bytes(self): | 
					
						
							|  |  |  |         self.check_match(b'test', b'te*') | 
					
						
							|  |  |  |         self.check_match(b'test\xff', b'te*\xff') | 
					
						
							| 
									
										
										
										
											2009-08-16 18:58:46 +00:00
										 |  |  |         self.check_match(b'foo\nbar', b'foo*') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-21 08:57:00 +03:00
										 |  |  |     def test_case(self): | 
					
						
							|  |  |  |         check = self.check_match | 
					
						
							|  |  |  |         check('abc', 'abc') | 
					
						
							| 
									
										
										
										
											2025-04-08 12:11:25 +02:00
										 |  |  |         check('AbC', 'abc', IGNORECASE) | 
					
						
							|  |  |  |         check('abc', 'AbC', IGNORECASE) | 
					
						
							| 
									
										
										
										
											2017-05-21 08:57:00 +03:00
										 |  |  |         check('AbC', 'AbC') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_sep(self): | 
					
						
							|  |  |  |         check = self.check_match | 
					
						
							|  |  |  |         check('usr/bin', 'usr/bin') | 
					
						
							| 
									
										
										
										
											2025-04-08 12:11:25 +02:00
										 |  |  |         check('usr\\bin', 'usr/bin', NORMSEP) | 
					
						
							|  |  |  |         check('usr/bin', 'usr\\bin', NORMSEP) | 
					
						
							| 
									
										
										
										
											2017-05-21 08:57:00 +03:00
										 |  |  |         check('usr\\bin', 'usr\\bin') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-05 11:46:29 +03:00
										 |  |  |     def test_char_set(self): | 
					
						
							|  |  |  |         check = self.check_match | 
					
						
							|  |  |  |         tescases = string.ascii_lowercase + string.digits + string.punctuation | 
					
						
							|  |  |  |         for c in tescases: | 
					
						
							|  |  |  |             check(c, '[az]', c in 'az') | 
					
						
							|  |  |  |             check(c, '[!az]', c not in 'az') | 
					
						
							|  |  |  |         # Case insensitive. | 
					
						
							|  |  |  |         for c in tescases: | 
					
						
							| 
									
										
										
										
											2025-04-08 12:11:25 +02:00
										 |  |  |             check(c, '[AZ]', (c in 'az') and IGNORECASE) | 
					
						
							|  |  |  |             check(c, '[!AZ]', (c not in 'az') or not IGNORECASE) | 
					
						
							| 
									
										
										
										
											2022-06-05 11:46:29 +03:00
										 |  |  |         for c in string.ascii_uppercase: | 
					
						
							| 
									
										
										
										
											2025-04-08 12:11:25 +02:00
										 |  |  |             check(c, '[az]', (c in 'AZ') and IGNORECASE) | 
					
						
							|  |  |  |             check(c, '[!az]', (c not in 'AZ') or not IGNORECASE) | 
					
						
							| 
									
										
										
										
											2022-06-05 11:46:29 +03:00
										 |  |  |         # Repeated same character. | 
					
						
							|  |  |  |         for c in tescases: | 
					
						
							|  |  |  |             check(c, '[aa]', c == 'a') | 
					
						
							|  |  |  |         # Special cases. | 
					
						
							|  |  |  |         for c in tescases: | 
					
						
							|  |  |  |             check(c, '[^az]', c in '^az') | 
					
						
							|  |  |  |             check(c, '[[az]', c in '[az') | 
					
						
							|  |  |  |             check(c, r'[!]]', c != ']') | 
					
						
							|  |  |  |         check('[', '[') | 
					
						
							|  |  |  |         check('[]', '[]') | 
					
						
							|  |  |  |         check('[!', '[!') | 
					
						
							|  |  |  |         check('[!]', '[!]') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_range(self): | 
					
						
							|  |  |  |         check = self.check_match | 
					
						
							|  |  |  |         tescases = string.ascii_lowercase + string.digits + string.punctuation | 
					
						
							|  |  |  |         for c in tescases: | 
					
						
							|  |  |  |             check(c, '[b-d]', c in 'bcd') | 
					
						
							|  |  |  |             check(c, '[!b-d]', c not in 'bcd') | 
					
						
							|  |  |  |             check(c, '[b-dx-z]', c in 'bcdxyz') | 
					
						
							|  |  |  |             check(c, '[!b-dx-z]', c not in 'bcdxyz') | 
					
						
							|  |  |  |         # Case insensitive. | 
					
						
							|  |  |  |         for c in tescases: | 
					
						
							| 
									
										
										
										
											2025-04-08 12:11:25 +02:00
										 |  |  |             check(c, '[B-D]', (c in 'bcd') and IGNORECASE) | 
					
						
							|  |  |  |             check(c, '[!B-D]', (c not in 'bcd') or not IGNORECASE) | 
					
						
							| 
									
										
										
										
											2022-06-05 11:46:29 +03:00
										 |  |  |         for c in string.ascii_uppercase: | 
					
						
							| 
									
										
										
										
											2025-04-08 12:11:25 +02:00
										 |  |  |             check(c, '[b-d]', (c in 'BCD') and IGNORECASE) | 
					
						
							|  |  |  |             check(c, '[!b-d]', (c not in 'BCD') or not IGNORECASE) | 
					
						
							| 
									
										
										
										
											2022-06-05 11:46:29 +03:00
										 |  |  |         # Upper bound == lower bound. | 
					
						
							|  |  |  |         for c in tescases: | 
					
						
							|  |  |  |             check(c, '[b-b]', c == 'b') | 
					
						
							|  |  |  |         # Special cases. | 
					
						
							|  |  |  |         for c in tescases: | 
					
						
							|  |  |  |             check(c, '[!-#]', c not in '-#') | 
					
						
							|  |  |  |             check(c, '[!--.]', c not in '-.') | 
					
						
							|  |  |  |             check(c, '[^-`]', c in '^_`') | 
					
						
							| 
									
										
										
										
											2025-04-08 12:11:25 +02:00
										 |  |  |             if not (NORMSEP and c == '/'): | 
					
						
							| 
									
										
										
										
											2022-06-05 11:46:29 +03:00
										 |  |  |                 check(c, '[[-^]', c in r'[\]^') | 
					
						
							|  |  |  |                 check(c, r'[\-^]', c in r'\]^') | 
					
						
							|  |  |  |             check(c, '[b-]', c in '-b') | 
					
						
							|  |  |  |             check(c, '[!b-]', c not in '-b') | 
					
						
							|  |  |  |             check(c, '[-b]', c in '-b') | 
					
						
							|  |  |  |             check(c, '[!-b]', c not in '-b') | 
					
						
							|  |  |  |             check(c, '[-]', c in '-') | 
					
						
							|  |  |  |             check(c, '[!-]', c not in '-') | 
					
						
							|  |  |  |         # Upper bound is less that lower bound: error in RE. | 
					
						
							|  |  |  |         for c in tescases: | 
					
						
							|  |  |  |             check(c, '[d-b]', False) | 
					
						
							|  |  |  |             check(c, '[!d-b]', True) | 
					
						
							|  |  |  |             check(c, '[d-bx-z]', c in 'xyz') | 
					
						
							|  |  |  |             check(c, '[!d-bx-z]', c not in 'xyz') | 
					
						
							|  |  |  |             check(c, '[d-b^-`]', c in '^_`') | 
					
						
							| 
									
										
										
										
											2025-04-08 12:11:25 +02:00
										 |  |  |             if not (NORMSEP and c == '/'): | 
					
						
							| 
									
										
										
										
											2022-06-05 11:46:29 +03:00
										 |  |  |                 check(c, '[d-b[-^]', c in r'[\]^') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_sep_in_char_set(self): | 
					
						
							|  |  |  |         check = self.check_match | 
					
						
							|  |  |  |         check('/', r'[/]') | 
					
						
							|  |  |  |         check('\\', r'[\]') | 
					
						
							| 
									
										
										
										
											2025-04-08 12:11:25 +02:00
										 |  |  |         check('/', r'[\]', NORMSEP) | 
					
						
							|  |  |  |         check('\\', r'[/]', NORMSEP) | 
					
						
							| 
									
										
										
										
											2022-06-05 11:46:29 +03:00
										 |  |  |         check('[/]', r'[/]', False) | 
					
						
							|  |  |  |         check(r'[\\]', r'[/]', False) | 
					
						
							|  |  |  |         check('\\', r'[\t]') | 
					
						
							| 
									
										
										
										
											2025-04-08 12:11:25 +02:00
										 |  |  |         check('/', r'[\t]', NORMSEP) | 
					
						
							| 
									
										
										
										
											2022-06-05 11:46:29 +03:00
										 |  |  |         check('t', r'[\t]') | 
					
						
							|  |  |  |         check('\t', r'[\t]', False) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_sep_in_range(self): | 
					
						
							|  |  |  |         check = self.check_match | 
					
						
							| 
									
										
										
										
											2025-04-08 12:11:25 +02:00
										 |  |  |         check('a/b', 'a[.-0]b', not NORMSEP) | 
					
						
							| 
									
										
										
										
											2022-06-05 11:46:29 +03:00
										 |  |  |         check('a\\b', 'a[.-0]b', False) | 
					
						
							| 
									
										
										
										
											2025-04-08 12:11:25 +02:00
										 |  |  |         check('a\\b', 'a[Z-^]b', not NORMSEP) | 
					
						
							| 
									
										
										
										
											2022-06-05 11:46:29 +03:00
										 |  |  |         check('a/b', 'a[Z-^]b', False) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-08 12:11:25 +02:00
										 |  |  |         check('a/b', 'a[/-0]b', not NORMSEP) | 
					
						
							| 
									
										
										
										
											2022-06-05 11:46:29 +03:00
										 |  |  |         check(r'a\b', 'a[/-0]b', False) | 
					
						
							|  |  |  |         check('a[/-0]b', 'a[/-0]b', False) | 
					
						
							|  |  |  |         check(r'a[\-0]b', 'a[/-0]b', False) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         check('a/b', 'a[.-/]b') | 
					
						
							| 
									
										
										
										
											2025-04-08 12:11:25 +02:00
										 |  |  |         check(r'a\b', 'a[.-/]b', NORMSEP) | 
					
						
							| 
									
										
										
										
											2022-06-05 11:46:29 +03:00
										 |  |  |         check('a[.-/]b', 'a[.-/]b', False) | 
					
						
							|  |  |  |         check(r'a[.-\]b', 'a[.-/]b', False) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         check(r'a\b', r'a[\-^]b') | 
					
						
							| 
									
										
										
										
											2025-04-08 12:11:25 +02:00
										 |  |  |         check('a/b', r'a[\-^]b', NORMSEP) | 
					
						
							| 
									
										
										
										
											2022-06-05 11:46:29 +03:00
										 |  |  |         check(r'a[\-^]b', r'a[\-^]b', False) | 
					
						
							|  |  |  |         check('a[/-^]b', r'a[\-^]b', False) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-08 12:11:25 +02:00
										 |  |  |         check(r'a\b', r'a[Z-\]b', not NORMSEP) | 
					
						
							| 
									
										
										
										
											2022-06-05 11:46:29 +03:00
										 |  |  |         check('a/b', r'a[Z-\]b', False) | 
					
						
							|  |  |  |         check(r'a[Z-\]b', r'a[Z-\]b', False) | 
					
						
							|  |  |  |         check('a[Z-/]b', r'a[Z-\]b', False) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-09 13:30:19 +02:00
										 |  |  |     def test_warnings(self): | 
					
						
							|  |  |  |         with warnings.catch_warnings(): | 
					
						
							|  |  |  |             warnings.simplefilter('error', Warning) | 
					
						
							|  |  |  |             check = self.check_match | 
					
						
							|  |  |  |             check('[', '[[]') | 
					
						
							|  |  |  |             check('&', '[a&&b]') | 
					
						
							|  |  |  |             check('|', '[a||b]') | 
					
						
							|  |  |  |             check('~', '[a~~b]') | 
					
						
							|  |  |  |             check(',', '[a-z+--A-Z]') | 
					
						
							|  |  |  |             check('.', '[a-z--/A-Z]') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-21 08:57:00 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-23 16:23:13 +00:00
										 |  |  | class TranslateTestCase(unittest.TestCase): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_translate(self): | 
					
						
							| 
									
										
										
										
											2020-05-11 21:19:20 -05:00
										 |  |  |         import re | 
					
						
							| 
									
										
										
										
											2016-09-11 12:50:02 +03:00
										 |  |  |         self.assertEqual(translate('*'), r'(?s:.*)\Z') | 
					
						
							|  |  |  |         self.assertEqual(translate('?'), r'(?s:.)\Z') | 
					
						
							|  |  |  |         self.assertEqual(translate('a?b*'), r'(?s:a.b.*)\Z') | 
					
						
							|  |  |  |         self.assertEqual(translate('[abc]'), r'(?s:[abc])\Z') | 
					
						
							|  |  |  |         self.assertEqual(translate('[]]'), r'(?s:[]])\Z') | 
					
						
							|  |  |  |         self.assertEqual(translate('[!x]'), r'(?s:[^x])\Z') | 
					
						
							|  |  |  |         self.assertEqual(translate('[^x]'), r'(?s:[\^x])\Z') | 
					
						
							|  |  |  |         self.assertEqual(translate('[x'), r'(?s:\[x)\Z') | 
					
						
							| 
									
										
										
										
											2020-05-05 21:28:24 -05:00
										 |  |  |         # from the docs | 
					
						
							|  |  |  |         self.assertEqual(translate('*.txt'), r'(?s:.*\.txt)\Z') | 
					
						
							|  |  |  |         # squash consecutive stars | 
					
						
							|  |  |  |         self.assertEqual(translate('*********'), r'(?s:.*)\Z') | 
					
						
							|  |  |  |         self.assertEqual(translate('A*********'), r'(?s:A.*)\Z') | 
					
						
							|  |  |  |         self.assertEqual(translate('*********A'), r'(?s:.*A)\Z') | 
					
						
							|  |  |  |         self.assertEqual(translate('A*********?[?]?'), r'(?s:A.*.[?].)\Z') | 
					
						
							|  |  |  |         # fancy translation to prevent exponential-time match failure | 
					
						
							| 
									
										
										
										
											2020-05-11 21:19:20 -05:00
										 |  |  |         t = translate('**a*a****a') | 
					
						
							| 
									
										
										
										
											2022-03-21 12:49:43 -05:00
										 |  |  |         self.assertEqual(t, r'(?s:(?>.*?a)(?>.*?a).*a)\Z') | 
					
						
							| 
									
										
										
										
											2020-05-11 21:19:20 -05:00
										 |  |  |         # and try pasting multiple translate results - it's an undocumented | 
					
						
							| 
									
										
										
										
											2022-03-21 12:49:43 -05:00
										 |  |  |         # feature that this works | 
					
						
							| 
									
										
										
										
											2020-05-11 21:19:20 -05:00
										 |  |  |         r1 = translate('**a**a**a*') | 
					
						
							|  |  |  |         r2 = translate('**b**b**b*') | 
					
						
							|  |  |  |         r3 = translate('*c*c*c*') | 
					
						
							|  |  |  |         fatre = "|".join([r1, r2, r3]) | 
					
						
							|  |  |  |         self.assertTrue(re.match(fatre, 'abaccad')) | 
					
						
							|  |  |  |         self.assertTrue(re.match(fatre, 'abxbcab')) | 
					
						
							|  |  |  |         self.assertTrue(re.match(fatre, 'cbabcaxc')) | 
					
						
							|  |  |  |         self.assertFalse(re.match(fatre, 'dabccbad')) | 
					
						
							| 
									
										
										
										
											2010-07-23 16:23:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-27 17:42:45 +01:00
										 |  |  |     def test_translate_wildcards(self): | 
					
						
							|  |  |  |         for pattern, expect in [ | 
					
						
							|  |  |  |             ('ab*', r'(?s:ab.*)\Z'), | 
					
						
							|  |  |  |             ('ab*cd', r'(?s:ab.*cd)\Z'), | 
					
						
							|  |  |  |             ('ab*cd*', r'(?s:ab(?>.*?cd).*)\Z'), | 
					
						
							|  |  |  |             ('ab*cd*12', r'(?s:ab(?>.*?cd).*12)\Z'), | 
					
						
							|  |  |  |             ('ab*cd*12*', r'(?s:ab(?>.*?cd)(?>.*?12).*)\Z'), | 
					
						
							|  |  |  |             ('ab*cd*12*34', r'(?s:ab(?>.*?cd)(?>.*?12).*34)\Z'), | 
					
						
							|  |  |  |             ('ab*cd*12*34*', r'(?s:ab(?>.*?cd)(?>.*?12)(?>.*?34).*)\Z'), | 
					
						
							|  |  |  |         ]: | 
					
						
							|  |  |  |             with self.subTest(pattern): | 
					
						
							|  |  |  |                 translated = translate(pattern) | 
					
						
							|  |  |  |                 self.assertEqual(translated, expect, pattern) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for pattern, expect in [ | 
					
						
							|  |  |  |             ('*ab', r'(?s:.*ab)\Z'), | 
					
						
							|  |  |  |             ('*ab*', r'(?s:(?>.*?ab).*)\Z'), | 
					
						
							|  |  |  |             ('*ab*cd', r'(?s:(?>.*?ab).*cd)\Z'), | 
					
						
							|  |  |  |             ('*ab*cd*', r'(?s:(?>.*?ab)(?>.*?cd).*)\Z'), | 
					
						
							|  |  |  |             ('*ab*cd*12', r'(?s:(?>.*?ab)(?>.*?cd).*12)\Z'), | 
					
						
							|  |  |  |             ('*ab*cd*12*', r'(?s:(?>.*?ab)(?>.*?cd)(?>.*?12).*)\Z'), | 
					
						
							|  |  |  |             ('*ab*cd*12*34', r'(?s:(?>.*?ab)(?>.*?cd)(?>.*?12).*34)\Z'), | 
					
						
							|  |  |  |             ('*ab*cd*12*34*', r'(?s:(?>.*?ab)(?>.*?cd)(?>.*?12)(?>.*?34).*)\Z'), | 
					
						
							|  |  |  |         ]: | 
					
						
							|  |  |  |             with self.subTest(pattern): | 
					
						
							|  |  |  |                 translated = translate(pattern) | 
					
						
							|  |  |  |                 self.assertEqual(translated, expect, pattern) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_translate_expressions(self): | 
					
						
							|  |  |  |         for pattern, expect in [ | 
					
						
							|  |  |  |             ('[', r'(?s:\[)\Z'), | 
					
						
							|  |  |  |             ('[!', r'(?s:\[!)\Z'), | 
					
						
							|  |  |  |             ('[]', r'(?s:\[\])\Z'), | 
					
						
							|  |  |  |             ('[abc', r'(?s:\[abc)\Z'), | 
					
						
							|  |  |  |             ('[!abc', r'(?s:\[!abc)\Z'), | 
					
						
							|  |  |  |             ('[abc]', r'(?s:[abc])\Z'), | 
					
						
							|  |  |  |             ('[!abc]', r'(?s:[^abc])\Z'), | 
					
						
							|  |  |  |             ('[!abc][!def]', r'(?s:[^abc][^def])\Z'), | 
					
						
							|  |  |  |             # with [[ | 
					
						
							|  |  |  |             ('[[', r'(?s:\[\[)\Z'), | 
					
						
							|  |  |  |             ('[[a', r'(?s:\[\[a)\Z'), | 
					
						
							|  |  |  |             ('[[]', r'(?s:[\[])\Z'), | 
					
						
							|  |  |  |             ('[[]a', r'(?s:[\[]a)\Z'), | 
					
						
							|  |  |  |             ('[[]]', r'(?s:[\[]\])\Z'), | 
					
						
							|  |  |  |             ('[[]a]', r'(?s:[\[]a\])\Z'), | 
					
						
							|  |  |  |             ('[[a]', r'(?s:[\[a])\Z'), | 
					
						
							|  |  |  |             ('[[a]]', r'(?s:[\[a]\])\Z'), | 
					
						
							|  |  |  |             ('[[a]b', r'(?s:[\[a]b)\Z'), | 
					
						
							|  |  |  |             # backslashes | 
					
						
							|  |  |  |             ('[\\', r'(?s:\[\\)\Z'), | 
					
						
							|  |  |  |             (r'[\]', r'(?s:[\\])\Z'), | 
					
						
							|  |  |  |             (r'[\\]', r'(?s:[\\\\])\Z'), | 
					
						
							|  |  |  |         ]: | 
					
						
							|  |  |  |             with self.subTest(pattern): | 
					
						
							|  |  |  |                 translated = translate(pattern) | 
					
						
							|  |  |  |                 self.assertEqual(translated, expect, pattern) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_star_indices_locations(self): | 
					
						
							|  |  |  |         from fnmatch import _translate | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         blocks = ['a^b', '***', '?', '?', '[a-z]', '[1-9]', '*', '++', '[[a'] | 
					
						
							|  |  |  |         parts, star_indices = _translate(''.join(blocks), '*', '.') | 
					
						
							|  |  |  |         expect_parts = ['a', r'\^', 'b', '*', | 
					
						
							|  |  |  |                         '.', '.', '[a-z]', '[1-9]', '*', | 
					
						
							|  |  |  |                         r'\+', r'\+', r'\[', r'\[', 'a'] | 
					
						
							|  |  |  |         self.assertListEqual(parts, expect_parts) | 
					
						
							|  |  |  |         self.assertListEqual(star_indices, [3, 8]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-23 16:23:13 +00:00
										 |  |  | class FilterTestCase(unittest.TestCase): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_filter(self): | 
					
						
							| 
									
										
										
										
											2017-05-21 08:57:00 +03:00
										 |  |  |         self.assertEqual(filter(['Python', 'Ruby', 'Perl', 'Tcl'], 'P*'), | 
					
						
							|  |  |  |                          ['Python', 'Perl']) | 
					
						
							|  |  |  |         self.assertEqual(filter([b'Python', b'Ruby', b'Perl', b'Tcl'], b'P*'), | 
					
						
							|  |  |  |                          [b'Python', b'Perl']) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_mix_bytes_str(self): | 
					
						
							|  |  |  |         self.assertRaises(TypeError, filter, ['test'], b'*') | 
					
						
							|  |  |  |         self.assertRaises(TypeError, filter, [b'test'], '*') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_case(self): | 
					
						
							|  |  |  |         self.assertEqual(filter(['Test.py', 'Test.rb', 'Test.PL'], '*.p*'), | 
					
						
							| 
									
										
										
										
											2025-04-08 12:11:25 +02:00
										 |  |  |                          ['Test.py', 'Test.PL'] if IGNORECASE else ['Test.py']) | 
					
						
							| 
									
										
										
										
											2017-05-21 08:57:00 +03:00
										 |  |  |         self.assertEqual(filter(['Test.py', 'Test.rb', 'Test.PL'], '*.P*'), | 
					
						
							| 
									
										
										
										
											2025-04-08 12:11:25 +02:00
										 |  |  |                          ['Test.py', 'Test.PL'] if IGNORECASE else ['Test.PL']) | 
					
						
							| 
									
										
										
										
											2017-05-21 08:57:00 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_sep(self): | 
					
						
							|  |  |  |         self.assertEqual(filter(['usr/bin', 'usr', 'usr\\lib'], 'usr/*'), | 
					
						
							| 
									
										
										
										
											2025-04-08 12:11:25 +02:00
										 |  |  |                          ['usr/bin', 'usr\\lib'] if NORMSEP else ['usr/bin']) | 
					
						
							| 
									
										
										
										
											2017-05-21 08:57:00 +03:00
										 |  |  |         self.assertEqual(filter(['usr/bin', 'usr', 'usr\\lib'], 'usr\\*'), | 
					
						
							| 
									
										
										
										
											2025-04-08 12:11:25 +02:00
										 |  |  |                          ['usr/bin', 'usr\\lib'] if NORMSEP else ['usr\\lib']) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class FilterFalseTestCase(unittest.TestCase): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_filterfalse(self): | 
					
						
							|  |  |  |         actual = filterfalse(['Python', 'Ruby', 'Perl', 'Tcl'], 'P*') | 
					
						
							|  |  |  |         self.assertListEqual(actual, ['Ruby', 'Tcl']) | 
					
						
							|  |  |  |         actual = filterfalse([b'Python', b'Ruby', b'Perl', b'Tcl'], b'P*') | 
					
						
							|  |  |  |         self.assertListEqual(actual, [b'Ruby', b'Tcl']) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_mix_bytes_str(self): | 
					
						
							|  |  |  |         self.assertRaises(TypeError, filterfalse, ['test'], b'*') | 
					
						
							|  |  |  |         self.assertRaises(TypeError, filterfalse, [b'test'], '*') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_case(self): | 
					
						
							|  |  |  |         self.assertEqual(filterfalse(['Test.py', 'Test.rb', 'Test.PL'], '*.p*'), | 
					
						
							|  |  |  |                          ['Test.rb'] if IGNORECASE else ['Test.rb', 'Test.PL']) | 
					
						
							|  |  |  |         self.assertEqual(filterfalse(['Test.py', 'Test.rb', 'Test.PL'], '*.P*'), | 
					
						
							|  |  |  |                          ['Test.rb'] if IGNORECASE else ['Test.py', 'Test.rb',]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_sep(self): | 
					
						
							|  |  |  |         self.assertEqual(filterfalse(['usr/bin', 'usr', 'usr\\lib'], 'usr/*'), | 
					
						
							|  |  |  |                          ['usr'] if NORMSEP else ['usr', 'usr\\lib']) | 
					
						
							|  |  |  |         self.assertEqual(filterfalse(['usr/bin', 'usr', 'usr\\lib'], 'usr\\*'), | 
					
						
							|  |  |  |                          ['usr'] if NORMSEP else ['usr/bin', 'usr']) | 
					
						
							| 
									
										
										
										
											2010-07-23 16:23:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-09-20 21:33:42 +00:00
										 |  |  | if __name__ == "__main__": | 
					
						
							| 
									
										
										
										
											2015-04-13 15:00:43 -05:00
										 |  |  |     unittest.main() |