blob: 21b015330646f3a12c23672a10de65261679f6eb [file] [log] [blame]
Fred Drake99e87f92001-07-23 16:08:36 +00001import glob
2import os
Johannes Gijsbersa3beee12005-01-08 13:28:54 +00003import shutil
Antoine Pitrou3d068b22012-12-16 13:49:37 +01004import sys
Hynek Schlawacke26568f2012-12-27 10:10:11 +01005import unittest
6
Serhiy Storchakac2edcdd2014-09-11 12:17:37 +03007from test.support import (TESTFN, skip_unless_symlink,
Serhiy Storchakaa1b16ba2014-09-11 14:33:02 +03008 can_symlink, create_empty_file, change_cwd)
Fred Drake99e87f92001-07-23 16:08:36 +00009
Hynek Schlawackdec59ec2012-12-16 12:39:11 +010010
Fred Drake99e87f92001-07-23 16:08:36 +000011class GlobTests(unittest.TestCase):
12
13 def norm(self, *parts):
14 return os.path.normpath(os.path.join(self.tempdir, *parts))
15
Serhiy Storchakac2edcdd2014-09-11 12:17:37 +030016 def joins(self, *tuples):
17 return [os.path.join(self.tempdir, *parts) for parts in tuples]
18
Fred Drake99e87f92001-07-23 16:08:36 +000019 def mktemp(self, *parts):
Johannes Gijsbersa3beee12005-01-08 13:28:54 +000020 filename = self.norm(*parts)
21 base, file = os.path.split(filename)
22 if not os.path.exists(base):
23 os.makedirs(base)
Victor Stinnerbf816222011-06-30 23:25:47 +020024 create_empty_file(filename)
Fred Drake99e87f92001-07-23 16:08:36 +000025
26 def setUp(self):
Hynek Schlawackdec59ec2012-12-16 12:39:11 +010027 self.tempdir = TESTFN + "_dir"
Fred Drake99e87f92001-07-23 16:08:36 +000028 self.mktemp('a', 'D')
29 self.mktemp('aab', 'F')
Hynek Schlawackdec59ec2012-12-16 12:39:11 +010030 self.mktemp('.aa', 'G')
31 self.mktemp('.bb', 'H')
Fred Drake99e87f92001-07-23 16:08:36 +000032 self.mktemp('aaa', 'zzzF')
33 self.mktemp('ZZZ')
34 self.mktemp('a', 'bcd', 'EF')
35 self.mktemp('a', 'bcd', 'efg', 'ha')
Brian Curtin3b4499c2010-12-28 14:31:47 +000036 if can_symlink():
Johannes Gijsbersae882f72004-08-30 10:19:56 +000037 os.symlink(self.norm('broken'), self.norm('sym1'))
Hynek Schlawacke26568f2012-12-27 10:10:11 +010038 os.symlink('broken', self.norm('sym2'))
39 os.symlink(os.path.join('a', 'bcd'), self.norm('sym3'))
Fred Drake99e87f92001-07-23 16:08:36 +000040
41 def tearDown(self):
Johannes Gijsbersa3beee12005-01-08 13:28:54 +000042 shutil.rmtree(self.tempdir)
Fred Drake99e87f92001-07-23 16:08:36 +000043
Serhiy Storchakac2edcdd2014-09-11 12:17:37 +030044 def glob(self, *parts, **kwargs):
Fred Drake99e87f92001-07-23 16:08:36 +000045 if len(parts) == 1:
46 pattern = parts[0]
47 else:
48 pattern = os.path.join(*parts)
49 p = os.path.join(self.tempdir, pattern)
Serhiy Storchakac2edcdd2014-09-11 12:17:37 +030050 res = glob.glob(p, **kwargs)
51 self.assertEqual(list(glob.iglob(p, **kwargs)), res)
Hynek Schlawacke26568f2012-12-27 10:10:11 +010052 bres = [os.fsencode(x) for x in res]
Serhiy Storchakac2edcdd2014-09-11 12:17:37 +030053 self.assertEqual(glob.glob(os.fsencode(p), **kwargs), bres)
54 self.assertEqual(list(glob.iglob(os.fsencode(p), **kwargs)), bres)
Johannes Gijsbers836f5432005-01-08 13:13:19 +000055 return res
Fred Drake99e87f92001-07-23 16:08:36 +000056
57 def assertSequencesEqual_noorder(self, l1, l2):
Hynek Schlawacke26568f2012-12-27 10:10:11 +010058 l1 = list(l1)
59 l2 = list(l2)
Raymond Hettingera690a992003-11-16 16:17:49 +000060 self.assertEqual(set(l1), set(l2))
Hynek Schlawacke26568f2012-12-27 10:10:11 +010061 self.assertEqual(sorted(l1), sorted(l2))
Fred Drake99e87f92001-07-23 16:08:36 +000062
63 def test_glob_literal(self):
64 eq = self.assertSequencesEqual_noorder
Fred Drake99e87f92001-07-23 16:08:36 +000065 eq(self.glob('a'), [self.norm('a')])
66 eq(self.glob('a', 'D'), [self.norm('a', 'D')])
67 eq(self.glob('aab'), [self.norm('aab')])
68 eq(self.glob('zymurgy'), [])
69
Hynek Schlawacke26568f2012-12-27 10:10:11 +010070 res = glob.glob('*')
71 self.assertEqual({type(r) for r in res}, {str})
72 res = glob.glob(os.path.join(os.curdir, '*'))
73 self.assertEqual({type(r) for r in res}, {str})
74
75 res = glob.glob(b'*')
76 self.assertEqual({type(r) for r in res}, {bytes})
77 res = glob.glob(os.path.join(os.fsencode(os.curdir), b'*'))
78 self.assertEqual({type(r) for r in res}, {bytes})
Guido van Rossumd8faa362007-04-27 19:54:29 +000079
Fred Drake99e87f92001-07-23 16:08:36 +000080 def test_glob_one_directory(self):
81 eq = self.assertSequencesEqual_noorder
Fred Drake99e87f92001-07-23 16:08:36 +000082 eq(self.glob('a*'), map(self.norm, ['a', 'aab', 'aaa']))
83 eq(self.glob('*a'), map(self.norm, ['a', 'aaa']))
Hynek Schlawackdec59ec2012-12-16 12:39:11 +010084 eq(self.glob('.*'), map(self.norm, ['.aa', '.bb']))
85 eq(self.glob('?aa'), map(self.norm, ['aaa']))
Fred Drake99e87f92001-07-23 16:08:36 +000086 eq(self.glob('aa?'), map(self.norm, ['aaa', 'aab']))
87 eq(self.glob('aa[ab]'), map(self.norm, ['aaa', 'aab']))
88 eq(self.glob('*q'), [])
89
90 def test_glob_nested_directory(self):
91 eq = self.assertSequencesEqual_noorder
Fred Drake99e87f92001-07-23 16:08:36 +000092 if os.path.normcase("abCD") == "abCD":
93 # case-sensitive filesystem
94 eq(self.glob('a', 'bcd', 'E*'), [self.norm('a', 'bcd', 'EF')])
95 else:
96 # case insensitive filesystem
97 eq(self.glob('a', 'bcd', 'E*'), [self.norm('a', 'bcd', 'EF'),
98 self.norm('a', 'bcd', 'efg')])
99 eq(self.glob('a', 'bcd', '*g'), [self.norm('a', 'bcd', 'efg')])
100
101 def test_glob_directory_names(self):
102 eq = self.assertSequencesEqual_noorder
Fred Drake99e87f92001-07-23 16:08:36 +0000103 eq(self.glob('*', 'D'), [self.norm('a', 'D')])
104 eq(self.glob('*', '*a'), [])
105 eq(self.glob('a', '*', '*', '*a'),
106 [self.norm('a', 'bcd', 'efg', 'ha')])
Hynek Schlawacke26568f2012-12-27 10:10:11 +0100107 eq(self.glob('?a?', '*F'), [self.norm('aaa', 'zzzF'),
108 self.norm('aab', 'F')])
Fred Drake99e87f92001-07-23 16:08:36 +0000109
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000110 def test_glob_directory_with_trailing_slash(self):
Antoine Pitrou54615582012-12-16 16:03:01 +0100111 # Patterns ending with a slash shouldn't match non-dirs
Hynek Schlawacke26568f2012-12-27 10:10:11 +0100112 res = glob.glob(self.norm('Z*Z') + os.sep)
Antoine Pitrou54615582012-12-16 16:03:01 +0100113 self.assertEqual(res, [])
Hynek Schlawacke26568f2012-12-27 10:10:11 +0100114 res = glob.glob(self.norm('ZZZ') + os.sep)
Antoine Pitrou54615582012-12-16 16:03:01 +0100115 self.assertEqual(res, [])
Hynek Schlawacke26568f2012-12-27 10:10:11 +0100116 # When there is a wildcard pattern which ends with os.sep, glob()
Antoine Pitrou54615582012-12-16 16:03:01 +0100117 # doesn't blow up.
Hynek Schlawacke26568f2012-12-27 10:10:11 +0100118 res = glob.glob(self.norm('aa*') + os.sep)
Antoine Pitrou54615582012-12-16 16:03:01 +0100119 self.assertEqual(len(res), 2)
Hynek Schlawacke26568f2012-12-27 10:10:11 +0100120 # either of these results is reasonable
Antoine Pitrou54615582012-12-16 16:03:01 +0100121 self.assertIn(set(res), [
122 {self.norm('aaa'), self.norm('aab')},
123 {self.norm('aaa') + os.sep, self.norm('aab') + os.sep},
124 ])
125
126 def test_glob_bytes_directory_with_trailing_slash(self):
127 # Same as test_glob_directory_with_trailing_slash, but with a
128 # bytes argument.
Hynek Schlawacke26568f2012-12-27 10:10:11 +0100129 res = glob.glob(os.fsencode(self.norm('Z*Z') + os.sep))
Antoine Pitrou54615582012-12-16 16:03:01 +0100130 self.assertEqual(res, [])
Hynek Schlawacke26568f2012-12-27 10:10:11 +0100131 res = glob.glob(os.fsencode(self.norm('ZZZ') + os.sep))
Antoine Pitrou54615582012-12-16 16:03:01 +0100132 self.assertEqual(res, [])
Hynek Schlawacke26568f2012-12-27 10:10:11 +0100133 res = glob.glob(os.fsencode(self.norm('aa*') + os.sep))
Antoine Pitrou54615582012-12-16 16:03:01 +0100134 self.assertEqual(len(res), 2)
Hynek Schlawacke26568f2012-12-27 10:10:11 +0100135 # either of these results is reasonable
136 self.assertIn(set(res), [
137 {os.fsencode(self.norm('aaa')),
138 os.fsencode(self.norm('aab'))},
139 {os.fsencode(self.norm('aaa') + os.sep),
140 os.fsencode(self.norm('aab') + os.sep)},
Antoine Pitrou54615582012-12-16 16:03:01 +0100141 ])
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000142
Brian Curtin3b4499c2010-12-28 14:31:47 +0000143 @skip_unless_symlink
Hynek Schlawacke26568f2012-12-27 10:10:11 +0100144 def test_glob_symlinks(self):
145 eq = self.assertSequencesEqual_noorder
146 eq(self.glob('sym3'), [self.norm('sym3')])
147 eq(self.glob('sym3', '*'), [self.norm('sym3', 'EF'),
148 self.norm('sym3', 'efg')])
149 self.assertIn(self.glob('sym3' + os.sep),
150 [[self.norm('sym3')], [self.norm('sym3') + os.sep]])
151 eq(self.glob('*', '*F'),
152 [self.norm('aaa', 'zzzF'),
153 self.norm('aab', 'F'), self.norm('sym3', 'EF')])
154
155 @skip_unless_symlink
Johannes Gijsbersae882f72004-08-30 10:19:56 +0000156 def test_glob_broken_symlinks(self):
Brian Curtind40e6f72010-07-08 21:39:08 +0000157 eq = self.assertSequencesEqual_noorder
Hynek Schlawacke26568f2012-12-27 10:10:11 +0100158 eq(self.glob('sym*'), [self.norm('sym1'), self.norm('sym2'),
159 self.norm('sym3')])
Brian Curtind40e6f72010-07-08 21:39:08 +0000160 eq(self.glob('sym1'), [self.norm('sym1')])
161 eq(self.glob('sym2'), [self.norm('sym2')])
Johannes Gijsbersae882f72004-08-30 10:19:56 +0000162
Antoine Pitrou3d068b22012-12-16 13:49:37 +0100163 @unittest.skipUnless(sys.platform == "win32", "Win32 specific test")
164 def test_glob_magic_in_drive(self):
165 eq = self.assertSequencesEqual_noorder
166 eq(glob.glob('*:'), [])
167 eq(glob.glob(b'*:'), [])
168 eq(glob.glob('?:'), [])
169 eq(glob.glob(b'?:'), [])
170 eq(glob.glob('\\\\?\\c:\\'), ['\\\\?\\c:\\'])
171 eq(glob.glob(b'\\\\?\\c:\\'), [b'\\\\?\\c:\\'])
172 eq(glob.glob('\\\\*\\*\\'), [])
173 eq(glob.glob(b'\\\\*\\*\\'), [])
174
Serhiy Storchakafd32fff2013-11-18 13:06:43 +0200175 def check_escape(self, arg, expected):
176 self.assertEqual(glob.escape(arg), expected)
177 self.assertEqual(glob.escape(os.fsencode(arg)), os.fsencode(expected))
178
179 def test_escape(self):
180 check = self.check_escape
181 check('abc', 'abc')
182 check('[', '[[]')
183 check('?', '[?]')
184 check('*', '[*]')
185 check('[[_/*?*/_]]', '[[][[]_/[*][?][*]/_]]')
186 check('/[[_/*?*/_]]/', '/[[][[]_/[*][?][*]/_]]/')
187
188 @unittest.skipUnless(sys.platform == "win32", "Win32 specific test")
189 def test_escape_windows(self):
190 check = self.check_escape
191 check('?:?', '?:[?]')
192 check('*:*', '*:[*]')
193 check(r'\\?\c:\?', r'\\?\c:\[?]')
194 check(r'\\*\*\*', r'\\*\*\[*]')
195 check('//?/c:/?', '//?/c:/[?]')
196 check('//*/*/*', '//*/*/[*]')
Fred Drake2e2be372001-09-20 21:33:42 +0000197
Serhiy Storchakac2edcdd2014-09-11 12:17:37 +0300198 def rglob(self, *parts, **kwargs):
199 return self.glob(*parts, recursive=True, **kwargs)
200
201 def test_recursive_glob(self):
202 eq = self.assertSequencesEqual_noorder
203 full = [('ZZZ',),
204 ('a',), ('a', 'D'),
205 ('a', 'bcd'),
206 ('a', 'bcd', 'EF'),
207 ('a', 'bcd', 'efg'),
208 ('a', 'bcd', 'efg', 'ha'),
209 ('aaa',), ('aaa', 'zzzF'),
210 ('aab',), ('aab', 'F'),
211 ]
212 if can_symlink():
213 full += [('sym1',), ('sym2',),
214 ('sym3',),
215 ('sym3', 'EF'),
216 ('sym3', 'efg'),
217 ('sym3', 'efg', 'ha'),
218 ]
219 eq(self.rglob('**'), self.joins(('',), *full))
220 eq(self.rglob('.', '**'), self.joins(('.',''),
221 *(('.',) + i for i in full)))
222 dirs = [('a', ''), ('a', 'bcd', ''), ('a', 'bcd', 'efg', ''),
223 ('aaa', ''), ('aab', '')]
224 if can_symlink():
225 dirs += [('sym3', ''), ('sym3', 'efg', '')]
226 eq(self.rglob('**', ''), self.joins(('',), *dirs))
227
228 eq(self.rglob('a', '**'), self.joins(
229 ('a', ''), ('a', 'D'), ('a', 'bcd'), ('a', 'bcd', 'EF'),
230 ('a', 'bcd', 'efg'), ('a', 'bcd', 'efg', 'ha')))
231 eq(self.rglob('a**'), self.joins(('a',), ('aaa',), ('aab',)))
232 expect = [('a', 'bcd', 'EF')]
233 if can_symlink():
234 expect += [('sym3', 'EF')]
235 eq(self.rglob('**', 'EF'), self.joins(*expect))
236 expect = [('a', 'bcd', 'EF'), ('aaa', 'zzzF'), ('aab', 'F')]
237 if can_symlink():
238 expect += [('sym3', 'EF')]
239 eq(self.rglob('**', '*F'), self.joins(*expect))
240 eq(self.rglob('**', '*F', ''), [])
241 eq(self.rglob('**', 'bcd', '*'), self.joins(
242 ('a', 'bcd', 'EF'), ('a', 'bcd', 'efg')))
243 eq(self.rglob('a', '**', 'bcd'), self.joins(('a', 'bcd')))
244
245 predir = os.path.abspath(os.curdir)
246 try:
247 os.chdir(self.tempdir)
248 join = os.path.join
249 eq(glob.glob('**', recursive=True), [join(*i) for i in full])
250 eq(glob.glob(join('**', ''), recursive=True),
251 [join(*i) for i in dirs])
252 eq(glob.glob(join('**','zz*F'), recursive=True),
253 [join('aaa', 'zzzF')])
254 eq(glob.glob('**zz*F', recursive=True), [])
255 expect = [join('a', 'bcd', 'EF')]
256 if can_symlink():
257 expect += [join('sym3', 'EF')]
258 eq(glob.glob(join('**', 'EF'), recursive=True), expect)
259 finally:
260 os.chdir(predir)
261
262
263@skip_unless_symlink
264class SymlinkLoopGlobTests(unittest.TestCase):
265
266 def test_selflink(self):
267 tempdir = TESTFN + "_dir"
268 os.makedirs(tempdir)
Serhiy Storchakac2edcdd2014-09-11 12:17:37 +0300269 self.addCleanup(shutil.rmtree, tempdir)
Serhiy Storchakaa1b16ba2014-09-11 14:33:02 +0300270 with change_cwd(tempdir):
271 os.makedirs('dir')
272 create_empty_file(os.path.join('dir', 'file'))
273 os.symlink(os.curdir, os.path.join('dir', 'link'))
Serhiy Storchakac2edcdd2014-09-11 12:17:37 +0300274
Serhiy Storchakaa1b16ba2014-09-11 14:33:02 +0300275 results = glob.glob('**', recursive=True)
276 self.assertEqual(len(results), len(set(results)))
277 results = set(results)
278 depth = 0
279 while results:
280 path = os.path.join(*(['dir'] + ['link'] * depth))
281 self.assertIn(path, results)
282 results.remove(path)
283 if not results:
284 break
285 path = os.path.join(path, 'file')
286 self.assertIn(path, results)
287 results.remove(path)
288 depth += 1
Serhiy Storchakac2edcdd2014-09-11 12:17:37 +0300289
Serhiy Storchakaa1b16ba2014-09-11 14:33:02 +0300290 results = glob.glob(os.path.join('**', 'file'), recursive=True)
291 self.assertEqual(len(results), len(set(results)))
292 results = set(results)
293 depth = 0
294 while results:
295 path = os.path.join(*(['dir'] + ['link'] * depth + ['file']))
296 self.assertIn(path, results)
297 results.remove(path)
298 depth += 1
Serhiy Storchakac2edcdd2014-09-11 12:17:37 +0300299
Serhiy Storchakaa1b16ba2014-09-11 14:33:02 +0300300 results = glob.glob(os.path.join('**', ''), recursive=True)
301 self.assertEqual(len(results), len(set(results)))
302 results = set(results)
303 depth = 0
304 while results:
305 path = os.path.join(*(['dir'] + ['link'] * depth + ['']))
306 self.assertIn(path, results)
307 results.remove(path)
308 depth += 1
Fred Drake2e2be372001-09-20 21:33:42 +0000309
310
311if __name__ == "__main__":
Serhiy Storchakac2edcdd2014-09-11 12:17:37 +0300312 unittest.main()