blob: b32e0e0d71a4e91ffeaba72638f92322d5a4aea7 [file] [log] [blame]
Tim Peters3230d5c2001-07-11 22:21:17 +00001'''
2Tests for fileinput module.
3Nick Mathewson
4'''
5
Guido van Rossumd8faa362007-04-27 19:54:29 +00006import unittest
Benjamin Petersonee8712c2008-05-20 21:35:26 +00007from test.support import verbose, TESTFN, run_unittest
8from test.support import unlink as safe_unlink
briancurtin906f0c42011-03-15 10:29:41 -04009import os, sys, re
Guido van Rossum34d19282007-08-09 01:03:29 +000010from io import StringIO
Georg Brandlc98eeed2006-02-19 14:57:47 +000011from fileinput import FileInput, hook_encoded
briancurtin906f0c42011-03-15 10:29:41 -040012import fileinput
13import collections
14import gzip, bz2
15import types
16import codecs
Tim Peters3230d5c2001-07-11 22:21:17 +000017
18# The fileinput module has 2 interfaces: the FileInput class which does
19# all the work, and a few functions (input, etc.) that use a global _state
briancurtin906f0c42011-03-15 10:29:41 -040020# variable.
Tim Peters3230d5c2001-07-11 22:21:17 +000021
22# Write lines (a list of lines) to temp file number i, and return the
23# temp file's name.
Tim Peters4d7cad12006-02-19 21:22:10 +000024def writeTmp(i, lines, mode='w'): # opening in text mode is the default
Tim Peters3230d5c2001-07-11 22:21:17 +000025 name = TESTFN + str(i)
Tim Peters4d7cad12006-02-19 21:22:10 +000026 f = open(name, mode)
Guido van Rossumc43e79f2007-06-18 18:26:36 +000027 for line in lines:
28 f.write(line)
Tim Peters3230d5c2001-07-11 22:21:17 +000029 f.close()
30 return name
31
Tim Peters3230d5c2001-07-11 22:21:17 +000032def remove_tempfiles(*names):
33 for name in names:
Guido van Rossume22905a2007-08-27 23:09:25 +000034 if name:
35 safe_unlink(name)
Tim Peters3230d5c2001-07-11 22:21:17 +000036
Guido van Rossumd8faa362007-04-27 19:54:29 +000037class BufferSizesTests(unittest.TestCase):
38 def test_buffer_sizes(self):
39 # First, run the tests with default and teeny buffer size.
40 for round, bs in (0, 0), (1, 30):
Neal Norwitz2595e762008-03-24 06:10:13 +000041 t1 = t2 = t3 = t4 = None
Guido van Rossumd8faa362007-04-27 19:54:29 +000042 try:
43 t1 = writeTmp(1, ["Line %s of file 1\n" % (i+1) for i in range(15)])
44 t2 = writeTmp(2, ["Line %s of file 2\n" % (i+1) for i in range(10)])
45 t3 = writeTmp(3, ["Line %s of file 3\n" % (i+1) for i in range(5)])
46 t4 = writeTmp(4, ["Line %s of file 4\n" % (i+1) for i in range(1)])
47 self.buffer_size_test(t1, t2, t3, t4, bs, round)
48 finally:
49 remove_tempfiles(t1, t2, t3, t4)
Tim Peters3230d5c2001-07-11 22:21:17 +000050
Guido van Rossumd8faa362007-04-27 19:54:29 +000051 def buffer_size_test(self, t1, t2, t3, t4, bs=0, round=0):
52 pat = re.compile(r'LINE (\d+) OF FILE (\d+)')
Tim Peters3230d5c2001-07-11 22:21:17 +000053
Guido van Rossumd8faa362007-04-27 19:54:29 +000054 start = 1 + round*6
55 if verbose:
56 print('%s. Simple iteration (bs=%s)' % (start+0, bs))
57 fi = FileInput(files=(t1, t2, t3, t4), bufsize=bs)
Tim Peters3230d5c2001-07-11 22:21:17 +000058 lines = list(fi)
Tim Peters3230d5c2001-07-11 22:21:17 +000059 fi.close()
Guido van Rossumd8faa362007-04-27 19:54:29 +000060 self.assertEqual(len(lines), 31)
61 self.assertEqual(lines[4], 'Line 5 of file 1\n')
62 self.assertEqual(lines[30], 'Line 1 of file 4\n')
63 self.assertEqual(fi.lineno(), 31)
64 self.assertEqual(fi.filename(), t4)
Tim Peters3230d5c2001-07-11 22:21:17 +000065
Guido van Rossumd8faa362007-04-27 19:54:29 +000066 if verbose:
67 print('%s. Status variables (bs=%s)' % (start+1, bs))
68 fi = FileInput(files=(t1, t2, t3, t4), bufsize=bs)
69 s = "x"
70 while s and s != 'Line 6 of file 2\n':
71 s = fi.readline()
72 self.assertEqual(fi.filename(), t2)
73 self.assertEqual(fi.lineno(), 21)
74 self.assertEqual(fi.filelineno(), 6)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000075 self.assertFalse(fi.isfirstline())
76 self.assertFalse(fi.isstdin())
Tim Peters3230d5c2001-07-11 22:21:17 +000077
Guido van Rossumd8faa362007-04-27 19:54:29 +000078 if verbose:
79 print('%s. Nextfile (bs=%s)' % (start+2, bs))
80 fi.nextfile()
81 self.assertEqual(fi.readline(), 'Line 1 of file 3\n')
82 self.assertEqual(fi.lineno(), 22)
83 fi.close()
Tim Peters3230d5c2001-07-11 22:21:17 +000084
Guido van Rossumd8faa362007-04-27 19:54:29 +000085 if verbose:
86 print('%s. Stdin (bs=%s)' % (start+3, bs))
87 fi = FileInput(files=(t1, t2, t3, t4, '-'), bufsize=bs)
88 savestdin = sys.stdin
89 try:
90 sys.stdin = StringIO("Line 1 of stdin\nLine 2 of stdin\n")
91 lines = list(fi)
92 self.assertEqual(len(lines), 33)
93 self.assertEqual(lines[32], 'Line 2 of stdin\n')
94 self.assertEqual(fi.filename(), '<stdin>')
95 fi.nextfile()
96 finally:
97 sys.stdin = savestdin
Tim Peters3230d5c2001-07-11 22:21:17 +000098
Guido van Rossumd8faa362007-04-27 19:54:29 +000099 if verbose:
100 print('%s. Boundary conditions (bs=%s)' % (start+4, bs))
101 fi = FileInput(files=(t1, t2, t3, t4), bufsize=bs)
102 self.assertEqual(fi.lineno(), 0)
103 self.assertEqual(fi.filename(), None)
104 fi.nextfile()
105 self.assertEqual(fi.lineno(), 0)
106 self.assertEqual(fi.filename(), None)
Tim Peters3230d5c2001-07-11 22:21:17 +0000107
Guido van Rossumd8faa362007-04-27 19:54:29 +0000108 if verbose:
109 print('%s. Inplace (bs=%s)' % (start+5, bs))
110 savestdout = sys.stdout
111 try:
112 fi = FileInput(files=(t1, t2, t3, t4), inplace=1, bufsize=bs)
113 for line in fi:
114 line = line[:-1].upper()
115 print(line)
116 fi.close()
117 finally:
118 sys.stdout = savestdout
Tim Peters3230d5c2001-07-11 22:21:17 +0000119
Guido van Rossumd8faa362007-04-27 19:54:29 +0000120 fi = FileInput(files=(t1, t2, t3, t4), bufsize=bs)
121 for line in fi:
122 self.assertEqual(line[-1], '\n')
123 m = pat.match(line[:-1])
124 self.assertNotEqual(m, None)
125 self.assertEqual(int(m.group(1)), fi.filelineno())
126 fi.close()
Georg Brandle4662172006-02-19 09:51:27 +0000127
briancurtin906f0c42011-03-15 10:29:41 -0400128class UnconditionallyRaise:
129 def __init__(self, exception_type):
130 self.exception_type = exception_type
131 self.invoked = False
132 def __call__(self, *args, **kwargs):
133 self.invoked = True
134 raise self.exception_type()
135
Guido van Rossumd8faa362007-04-27 19:54:29 +0000136class FileInputTests(unittest.TestCase):
briancurtin906f0c42011-03-15 10:29:41 -0400137
Guido van Rossumd8faa362007-04-27 19:54:29 +0000138 def test_zero_byte_files(self):
Neal Norwitz2595e762008-03-24 06:10:13 +0000139 t1 = t2 = t3 = t4 = None
Guido van Rossumd8faa362007-04-27 19:54:29 +0000140 try:
141 t1 = writeTmp(1, [""])
142 t2 = writeTmp(2, [""])
143 t3 = writeTmp(3, ["The only line there is.\n"])
144 t4 = writeTmp(4, [""])
145 fi = FileInput(files=(t1, t2, t3, t4))
Georg Brandl67e9fb92006-02-19 13:56:17 +0000146
Guido van Rossumd8faa362007-04-27 19:54:29 +0000147 line = fi.readline()
148 self.assertEqual(line, 'The only line there is.\n')
149 self.assertEqual(fi.lineno(), 1)
150 self.assertEqual(fi.filelineno(), 1)
151 self.assertEqual(fi.filename(), t3)
Georg Brandlc029f872006-02-19 14:12:34 +0000152
Guido van Rossumd8faa362007-04-27 19:54:29 +0000153 line = fi.readline()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000154 self.assertFalse(line)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000155 self.assertEqual(fi.lineno(), 1)
156 self.assertEqual(fi.filelineno(), 0)
157 self.assertEqual(fi.filename(), t4)
158 fi.close()
159 finally:
160 remove_tempfiles(t1, t2, t3, t4)
Georg Brandlc98eeed2006-02-19 14:57:47 +0000161
Guido van Rossumd8faa362007-04-27 19:54:29 +0000162 def test_files_that_dont_end_with_newline(self):
Neal Norwitz2595e762008-03-24 06:10:13 +0000163 t1 = t2 = None
Guido van Rossumd8faa362007-04-27 19:54:29 +0000164 try:
165 t1 = writeTmp(1, ["A\nB\nC"])
166 t2 = writeTmp(2, ["D\nE\nF"])
167 fi = FileInput(files=(t1, t2))
168 lines = list(fi)
169 self.assertEqual(lines, ["A\n", "B\n", "C", "D\n", "E\n", "F"])
170 self.assertEqual(fi.filelineno(), 3)
171 self.assertEqual(fi.lineno(), 6)
172 finally:
173 remove_tempfiles(t1, t2)
174
Guido van Rossumc43e79f2007-06-18 18:26:36 +0000175## def test_unicode_filenames(self):
176## # XXX A unicode string is always returned by writeTmp.
177## # So is this needed?
178## try:
179## t1 = writeTmp(1, ["A\nB"])
180## encoding = sys.getfilesystemencoding()
181## if encoding is None:
182## encoding = 'ascii'
183## fi = FileInput(files=str(t1, encoding))
184## lines = list(fi)
185## self.assertEqual(lines, ["A\n", "B"])
186## finally:
187## remove_tempfiles(t1)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000188
189 def test_fileno(self):
Neal Norwitz2595e762008-03-24 06:10:13 +0000190 t1 = t2 = None
Guido van Rossumd8faa362007-04-27 19:54:29 +0000191 try:
192 t1 = writeTmp(1, ["A\nB"])
193 t2 = writeTmp(2, ["C\nD"])
194 fi = FileInput(files=(t1, t2))
195 self.assertEqual(fi.fileno(), -1)
196 line =next( fi)
197 self.assertNotEqual(fi.fileno(), -1)
198 fi.nextfile()
199 self.assertEqual(fi.fileno(), -1)
200 line = list(fi)
201 self.assertEqual(fi.fileno(), -1)
202 finally:
203 remove_tempfiles(t1, t2)
204
205 def test_opening_mode(self):
206 try:
207 # invalid mode, should raise ValueError
208 fi = FileInput(mode="w")
209 self.fail("FileInput should reject invalid mode argument")
210 except ValueError:
211 pass
Guido van Rossume22905a2007-08-27 23:09:25 +0000212 t1 = None
Guido van Rossumd8faa362007-04-27 19:54:29 +0000213 try:
214 # try opening in universal newline mode
Guido van Rossume22905a2007-08-27 23:09:25 +0000215 t1 = writeTmp(1, [b"A\nB\r\nC\rD"], mode="wb")
Guido van Rossumd8faa362007-04-27 19:54:29 +0000216 fi = FileInput(files=t1, mode="U")
217 lines = list(fi)
218 self.assertEqual(lines, ["A\n", "B\n", "C\n", "D"])
219 finally:
220 remove_tempfiles(t1)
221
Guido van Rossume22905a2007-08-27 23:09:25 +0000222 def test_file_opening_hook(self):
223 try:
224 # cannot use openhook and inplace mode
225 fi = FileInput(inplace=1, openhook=lambda f, m: None)
226 self.fail("FileInput should raise if both inplace "
227 "and openhook arguments are given")
228 except ValueError:
229 pass
230 try:
231 fi = FileInput(openhook=1)
232 self.fail("FileInput should check openhook for being callable")
233 except ValueError:
234 pass
briancurtin906f0c42011-03-15 10:29:41 -0400235
236 class CustomOpenHook:
237 def __init__(self):
238 self.invoked = False
239 def __call__(self, *args):
240 self.invoked = True
241 return open(*args)
242
243 t = writeTmp(1, ["\n"])
244 self.addCleanup(remove_tempfiles, t)
245 custom_open_hook = CustomOpenHook()
246 with FileInput([t], openhook=custom_open_hook) as fi:
247 fi.readline()
248 self.assertTrue(custom_open_hook.invoked, "openhook not invoked")
Guido van Rossumd8faa362007-04-27 19:54:29 +0000249
Georg Brandl6cb7b652010-07-31 20:08:15 +0000250 def test_context_manager(self):
251 try:
252 t1 = writeTmp(1, ["A\nB\nC"])
253 t2 = writeTmp(2, ["D\nE\nF"])
254 with FileInput(files=(t1, t2)) as fi:
255 lines = list(fi)
256 self.assertEqual(lines, ["A\n", "B\n", "C", "D\n", "E\n", "F"])
257 self.assertEqual(fi.filelineno(), 3)
258 self.assertEqual(fi.lineno(), 6)
259 self.assertEqual(fi._files, ())
260 finally:
261 remove_tempfiles(t1, t2)
262
263 def test_close_on_exception(self):
264 try:
265 t1 = writeTmp(1, [""])
266 with FileInput(files=t1) as fi:
267 raise IOError
268 except IOError:
269 self.assertEqual(fi._files, ())
270 finally:
271 remove_tempfiles(t1)
272
briancurtin906f0c42011-03-15 10:29:41 -0400273 def test_empty_files_list_specified_to_constructor(self):
274 with FileInput(files=[]) as fi:
275 self.assertEquals(fi._files, ('-',))
276
277 def test__getitem__(self):
278 """Tests invoking FileInput.__getitem__() with the current
279 line number"""
280 t = writeTmp(1, ["line1\n", "line2\n"])
281 self.addCleanup(remove_tempfiles, t)
282 with FileInput(files=[t]) as fi:
283 retval1 = fi[0]
284 self.assertEqual(retval1, "line1\n")
285 retval2 = fi[1]
286 self.assertEqual(retval2, "line2\n")
287
288 def test__getitem__invalid_key(self):
289 """Tests invoking FileInput.__getitem__() with an index unequal to
290 the line number"""
291 t = writeTmp(1, ["line1\n", "line2\n"])
292 self.addCleanup(remove_tempfiles, t)
293 with FileInput(files=[t]) as fi:
294 with self.assertRaises(RuntimeError) as cm:
295 fi[1]
296 self.assertEquals(cm.exception.args, ("accessing lines out of order",))
297
298 def test__getitem__eof(self):
299 """Tests invoking FileInput.__getitem__() with the line number but at
300 end-of-input"""
301 t = writeTmp(1, [])
302 self.addCleanup(remove_tempfiles, t)
303 with FileInput(files=[t]) as fi:
304 with self.assertRaises(IndexError) as cm:
305 fi[0]
306 self.assertEquals(cm.exception.args, ("end of input reached",))
307
308 def test_nextfile_oserror_deleting_backup(self):
309 """Tests invoking FileInput.nextfile() when the attempt to delete
310 the backup file would raise OSError. This error is expected to be
311 silently ignored"""
312
313 os_unlink_orig = os.unlink
314 os_unlink_replacement = UnconditionallyRaise(OSError)
315 try:
316 t = writeTmp(1, ["\n"])
317 self.addCleanup(remove_tempfiles, t)
318 with FileInput(files=[t], inplace=True) as fi:
319 next(fi) # make sure the file is opened
320 os.unlink = os_unlink_replacement
321 fi.nextfile()
322 finally:
323 os.unlink = os_unlink_orig
324
325 # sanity check to make sure that our test scenario was actually hit
326 self.assertTrue(os_unlink_replacement.invoked,
327 "os.unlink() was not invoked")
328
329 def test_readline_os_fstat_raises_OSError(self):
330 """Tests invoking FileInput.readline() when os.fstat() raises OSError.
331 This exception should be silently discarded."""
332
333 os_fstat_orig = os.fstat
334 os_fstat_replacement = UnconditionallyRaise(OSError)
335 try:
336 t = writeTmp(1, ["\n"])
337 self.addCleanup(remove_tempfiles, t)
338 with FileInput(files=[t], inplace=True) as fi:
339 os.fstat = os_fstat_replacement
340 fi.readline()
341 finally:
342 os.fstat = os_fstat_orig
343
344 # sanity check to make sure that our test scenario was actually hit
345 self.assertTrue(os_fstat_replacement.invoked,
346 "os.fstat() was not invoked")
347
348 @unittest.skipIf(not hasattr(os, "chmod"), "os.chmod does not exist")
349 def test_readline_os_chmod_raises_OSError(self):
350 """Tests invoking FileInput.readline() when os.chmod() raises OSError.
351 This exception should be silently discarded."""
352
353 os_chmod_orig = os.chmod
354 os_chmod_replacement = UnconditionallyRaise(OSError)
355 try:
356 t = writeTmp(1, ["\n"])
357 self.addCleanup(remove_tempfiles, t)
358 with FileInput(files=[t], inplace=True) as fi:
359 os.chmod = os_chmod_replacement
360 fi.readline()
361 finally:
362 os.chmod = os_chmod_orig
363
364 # sanity check to make sure that our test scenario was actually hit
365 self.assertTrue(os_chmod_replacement.invoked,
366 "os.fstat() was not invoked")
367
368 def test_fileno_when_ValueError_raised(self):
369 class FilenoRaisesValueError(UnconditionallyRaise):
370 def __init__(self):
371 UnconditionallyRaise.__init__(self, ValueError)
372 def fileno(self):
373 self.__call__()
374
375 unconditionally_raise_ValueError = FilenoRaisesValueError()
376 t = writeTmp(1, ["\n"])
377 self.addCleanup(remove_tempfiles, t)
378 with FileInput(files=[t]) as fi:
379 file_backup = fi._file
380 try:
381 fi._file = unconditionally_raise_ValueError
382 result = fi.fileno()
383 finally:
384 fi._file = file_backup # make sure the file gets cleaned up
385
386 # sanity check to make sure that our test scenario was actually hit
387 self.assertTrue(unconditionally_raise_ValueError.invoked,
388 "_file.fileno() was not invoked")
389
390 self.assertEqual(result, -1, "fileno() should return -1")
391
392class MockFileInput:
393 """A class that mocks out fileinput.FileInput for use during unit tests"""
394
395 def __init__(self, files=None, inplace=False, backup="", bufsize=0,
396 mode="r", openhook=None):
397 self.files = files
398 self.inplace = inplace
399 self.backup = backup
400 self.bufsize = bufsize
401 self.mode = mode
402 self.openhook = openhook
403 self._file = None
404 self.invocation_counts = collections.defaultdict(lambda: 0)
405 self.return_values = {}
406
407 def close(self):
408 self.invocation_counts["close"] += 1
409
410 def nextfile(self):
411 self.invocation_counts["nextfile"] += 1
412 return self.return_values["nextfile"]
413
414 def filename(self):
415 self.invocation_counts["filename"] += 1
416 return self.return_values["filename"]
417
418 def lineno(self):
419 self.invocation_counts["lineno"] += 1
420 return self.return_values["lineno"]
421
422 def filelineno(self):
423 self.invocation_counts["filelineno"] += 1
424 return self.return_values["filelineno"]
425
426 def fileno(self):
427 self.invocation_counts["fileno"] += 1
428 return self.return_values["fileno"]
429
430 def isfirstline(self):
431 self.invocation_counts["isfirstline"] += 1
432 return self.return_values["isfirstline"]
433
434 def isstdin(self):
435 self.invocation_counts["isstdin"] += 1
436 return self.return_values["isstdin"]
437
438class BaseFileInputGlobalMethodsTest(unittest.TestCase):
439 """Base class for unit tests for the global function of
440 the fileinput module."""
441
442 def setUp(self):
443 self._orig_state = fileinput._state
444 self._orig_FileInput = fileinput.FileInput
445 fileinput.FileInput = MockFileInput
446
447 def tearDown(self):
448 fileinput.FileInput = self._orig_FileInput
449 fileinput._state = self._orig_state
450
451 def assertExactlyOneInvocation(self, mock_file_input, method_name):
452 # assert that the method with the given name was invoked once
453 actual_count = mock_file_input.invocation_counts[method_name]
454 self.assertEqual(actual_count, 1, method_name)
455 # assert that no other unexpected methods were invoked
456 actual_total_count = len(mock_file_input.invocation_counts)
457 self.assertEqual(actual_total_count, 1)
458
459class Test_fileinput_input(BaseFileInputGlobalMethodsTest):
460 """Unit tests for fileinput.input()"""
461
462 def test_state_is_not_None_and_state_file_is_not_None(self):
463 """Tests invoking fileinput.input() when fileinput._state is not None
464 and its _file attribute is also not None. Expect RuntimeError to
465 be raised with a meaningful error message and for fileinput._state
466 to *not* be modified."""
467 instance = MockFileInput()
468 instance._file = object()
469 fileinput._state = instance
470 with self.assertRaises(RuntimeError) as cm:
471 fileinput.input()
472 self.assertEqual(("input() already active",), cm.exception.args)
473 self.assertIs(instance, fileinput._state, "fileinput._state")
474
475 def test_state_is_not_None_and_state_file_is_None(self):
476 """Tests invoking fileinput.input() when fileinput._state is not None
477 but its _file attribute *is* None. Expect it to create and return
478 a new fileinput.FileInput object with all method parameters passed
479 explicitly to the __init__() method; also ensure that
480 fileinput._state is set to the returned instance."""
481 instance = MockFileInput()
482 instance._file = None
483 fileinput._state = instance
484 self.do_test_call_input()
485
486 def test_state_is_None(self):
487 """Tests invoking fileinput.input() when fileinput._state is None
488 Expect it to create and return a new fileinput.FileInput object
489 with all method parameters passed explicitly to the __init__()
490 method; also ensure that fileinput._state is set to the returned
491 instance."""
492 fileinput._state = None
493 self.do_test_call_input()
494
495 def do_test_call_input(self):
496 """Tests that fileinput.input() creates a new fileinput.FileInput
497 object, passing the given parameters unmodified to
498 fileinput.FileInput.__init__(). Note that this test depends on the
499 monkey patching of fileinput.FileInput done by setUp()."""
500 files = object()
501 inplace = object()
502 backup = object()
503 bufsize = object()
504 mode = object()
505 openhook = object()
506
507 # call fileinput.input() with different values for each argument
508 result = fileinput.input(files=files, inplace=inplace, backup=backup,
509 bufsize=bufsize,
510 mode=mode, openhook=openhook)
511
512 # ensure fileinput._state was set to the returned object
513 self.assertIs(result, fileinput._state, "fileinput._state")
514
515 # ensure the parameters to fileinput.input() were passed directly
516 # to FileInput.__init__()
517 self.assertIs(files, result.files, "files")
518 self.assertIs(inplace, result.inplace, "inplace")
519 self.assertIs(backup, result.backup, "backup")
520 self.assertIs(bufsize, result.bufsize, "bufsize")
521 self.assertIs(mode, result.mode, "mode")
522 self.assertIs(openhook, result.openhook, "openhook")
523
524class Test_fileinput_close(BaseFileInputGlobalMethodsTest):
525 """Unit tests for fileinput.close()"""
526
527 def test_state_is_None(self):
528 """Tests that fileinput.close() does nothing if fileinput._state
529 is None"""
530 fileinput._state = None
531 fileinput.close()
532 self.assertIsNone(fileinput._state)
533
534 def test_state_is_not_None(self):
535 """Tests that fileinput.close() invokes close() on fileinput._state
536 and sets _state=None"""
537 instance = MockFileInput()
538 fileinput._state = instance
539 fileinput.close()
540 self.assertExactlyOneInvocation(instance, "close")
541 self.assertIsNone(fileinput._state)
542
543class Test_fileinput_nextfile(BaseFileInputGlobalMethodsTest):
544 """Unit tests for fileinput.nextfile()"""
545
546 def test_state_is_None(self):
547 """Tests fileinput.nextfile() when fileinput._state is None.
548 Ensure that it raises RuntimeError with a meaningful error message
549 and does not modify fileinput._state"""
550 fileinput._state = None
551 with self.assertRaises(RuntimeError) as cm:
552 fileinput.nextfile()
553 self.assertEqual(("no active input()",), cm.exception.args)
554 self.assertIsNone(fileinput._state)
555
556 def test_state_is_not_None(self):
557 """Tests fileinput.nextfile() when fileinput._state is not None.
558 Ensure that it invokes fileinput._state.nextfile() exactly once,
559 returns whatever it returns, and does not modify fileinput._state
560 to point to a different object."""
561 nextfile_retval = object()
562 instance = MockFileInput()
563 instance.return_values["nextfile"] = nextfile_retval
564 fileinput._state = instance
565 retval = fileinput.nextfile()
566 self.assertExactlyOneInvocation(instance, "nextfile")
567 self.assertIs(retval, nextfile_retval)
568 self.assertIs(fileinput._state, instance)
569
570class Test_fileinput_filename(BaseFileInputGlobalMethodsTest):
571 """Unit tests for fileinput.filename()"""
572
573 def test_state_is_None(self):
574 """Tests fileinput.filename() when fileinput._state is None.
575 Ensure that it raises RuntimeError with a meaningful error message
576 and does not modify fileinput._state"""
577 fileinput._state = None
578 with self.assertRaises(RuntimeError) as cm:
579 fileinput.filename()
580 self.assertEqual(("no active input()",), cm.exception.args)
581 self.assertIsNone(fileinput._state)
582
583 def test_state_is_not_None(self):
584 """Tests fileinput.filename() when fileinput._state is not None.
585 Ensure that it invokes fileinput._state.filename() exactly once,
586 returns whatever it returns, and does not modify fileinput._state
587 to point to a different object."""
588 filename_retval = object()
589 instance = MockFileInput()
590 instance.return_values["filename"] = filename_retval
591 fileinput._state = instance
592 retval = fileinput.filename()
593 self.assertExactlyOneInvocation(instance, "filename")
594 self.assertIs(retval, filename_retval)
595 self.assertIs(fileinput._state, instance)
596
597class Test_fileinput_lineno(BaseFileInputGlobalMethodsTest):
598 """Unit tests for fileinput.lineno()"""
599
600 def test_state_is_None(self):
601 """Tests fileinput.lineno() when fileinput._state is None.
602 Ensure that it raises RuntimeError with a meaningful error message
603 and does not modify fileinput._state"""
604 fileinput._state = None
605 with self.assertRaises(RuntimeError) as cm:
606 fileinput.lineno()
607 self.assertEqual(("no active input()",), cm.exception.args)
608 self.assertIsNone(fileinput._state)
609
610 def test_state_is_not_None(self):
611 """Tests fileinput.lineno() when fileinput._state is not None.
612 Ensure that it invokes fileinput._state.lineno() exactly once,
613 returns whatever it returns, and does not modify fileinput._state
614 to point to a different object."""
615 lineno_retval = object()
616 instance = MockFileInput()
617 instance.return_values["lineno"] = lineno_retval
618 fileinput._state = instance
619 retval = fileinput.lineno()
620 self.assertExactlyOneInvocation(instance, "lineno")
621 self.assertIs(retval, lineno_retval)
622 self.assertIs(fileinput._state, instance)
623
624class Test_fileinput_filelineno(BaseFileInputGlobalMethodsTest):
625 """Unit tests for fileinput.filelineno()"""
626
627 def test_state_is_None(self):
628 """Tests fileinput.filelineno() when fileinput._state is None.
629 Ensure that it raises RuntimeError with a meaningful error message
630 and does not modify fileinput._state"""
631 fileinput._state = None
632 with self.assertRaises(RuntimeError) as cm:
633 fileinput.filelineno()
634 self.assertEqual(("no active input()",), cm.exception.args)
635 self.assertIsNone(fileinput._state)
636
637 def test_state_is_not_None(self):
638 """Tests fileinput.filelineno() when fileinput._state is not None.
639 Ensure that it invokes fileinput._state.filelineno() exactly once,
640 returns whatever it returns, and does not modify fileinput._state
641 to point to a different object."""
642 filelineno_retval = object()
643 instance = MockFileInput()
644 instance.return_values["filelineno"] = filelineno_retval
645 fileinput._state = instance
646 retval = fileinput.filelineno()
647 self.assertExactlyOneInvocation(instance, "filelineno")
648 self.assertIs(retval, filelineno_retval)
649 self.assertIs(fileinput._state, instance)
650
651class Test_fileinput_fileno(BaseFileInputGlobalMethodsTest):
652 """Unit tests for fileinput.fileno()"""
653
654 def test_state_is_None(self):
655 """Tests fileinput.fileno() when fileinput._state is None.
656 Ensure that it raises RuntimeError with a meaningful error message
657 and does not modify fileinput._state"""
658 fileinput._state = None
659 with self.assertRaises(RuntimeError) as cm:
660 fileinput.fileno()
661 self.assertEqual(("no active input()",), cm.exception.args)
662 self.assertIsNone(fileinput._state)
663
664 def test_state_is_not_None(self):
665 """Tests fileinput.fileno() when fileinput._state is not None.
666 Ensure that it invokes fileinput._state.fileno() exactly once,
667 returns whatever it returns, and does not modify fileinput._state
668 to point to a different object."""
669 fileno_retval = object()
670 instance = MockFileInput()
671 instance.return_values["fileno"] = fileno_retval
672 instance.fileno_retval = fileno_retval
673 fileinput._state = instance
674 retval = fileinput.fileno()
675 self.assertExactlyOneInvocation(instance, "fileno")
676 self.assertIs(retval, fileno_retval)
677 self.assertIs(fileinput._state, instance)
678
679class Test_fileinput_isfirstline(BaseFileInputGlobalMethodsTest):
680 """Unit tests for fileinput.isfirstline()"""
681
682 def test_state_is_None(self):
683 """Tests fileinput.isfirstline() when fileinput._state is None.
684 Ensure that it raises RuntimeError with a meaningful error message
685 and does not modify fileinput._state"""
686 fileinput._state = None
687 with self.assertRaises(RuntimeError) as cm:
688 fileinput.isfirstline()
689 self.assertEqual(("no active input()",), cm.exception.args)
690 self.assertIsNone(fileinput._state)
691
692 def test_state_is_not_None(self):
693 """Tests fileinput.isfirstline() when fileinput._state is not None.
694 Ensure that it invokes fileinput._state.isfirstline() exactly once,
695 returns whatever it returns, and does not modify fileinput._state
696 to point to a different object."""
697 isfirstline_retval = object()
698 instance = MockFileInput()
699 instance.return_values["isfirstline"] = isfirstline_retval
700 fileinput._state = instance
701 retval = fileinput.isfirstline()
702 self.assertExactlyOneInvocation(instance, "isfirstline")
703 self.assertIs(retval, isfirstline_retval)
704 self.assertIs(fileinput._state, instance)
705
706class Test_fileinput_isstdin(BaseFileInputGlobalMethodsTest):
707 """Unit tests for fileinput.isstdin()"""
708
709 def test_state_is_None(self):
710 """Tests fileinput.isstdin() when fileinput._state is None.
711 Ensure that it raises RuntimeError with a meaningful error message
712 and does not modify fileinput._state"""
713 fileinput._state = None
714 with self.assertRaises(RuntimeError) as cm:
715 fileinput.isstdin()
716 self.assertEqual(("no active input()",), cm.exception.args)
717 self.assertIsNone(fileinput._state)
718
719 def test_state_is_not_None(self):
720 """Tests fileinput.isstdin() when fileinput._state is not None.
721 Ensure that it invokes fileinput._state.isstdin() exactly once,
722 returns whatever it returns, and does not modify fileinput._state
723 to point to a different object."""
724 isstdin_retval = object()
725 instance = MockFileInput()
726 instance.return_values["isstdin"] = isstdin_retval
727 fileinput._state = instance
728 retval = fileinput.isstdin()
729 self.assertExactlyOneInvocation(instance, "isstdin")
730 self.assertIs(retval, isstdin_retval)
731 self.assertIs(fileinput._state, instance)
732
733class InvocationRecorder:
734 def __init__(self):
735 self.invocation_count = 0
736 def __call__(self, *args, **kwargs):
737 self.invocation_count += 1
738 self.last_invocation = (args, kwargs)
739
740class Test_hook_compressed(unittest.TestCase):
741 """Unit tests for fileinput.hook_compressed()"""
742
743 def setUp(self):
744 self.fake_open = InvocationRecorder()
745
746 def test_empty_string(self):
747 self.do_test_use_builtin_open("", 1)
748
749 def test_no_ext(self):
750 self.do_test_use_builtin_open("abcd", 2)
751
752 def test_gz_ext(self):
753 original_open = gzip.open
754 gzip.open = self.fake_open
755 try:
756 result = fileinput.hook_compressed("test.gz", 3)
757 finally:
758 gzip.open = original_open
759
760 self.assertEqual(self.fake_open.invocation_count, 1)
761 self.assertEqual(self.fake_open.last_invocation, (("test.gz", 3), {}))
762
763 def test_bz2_ext(self):
764 original_open = bz2.BZ2File
765 bz2.BZ2File = self.fake_open
766 try:
767 result = fileinput.hook_compressed("test.bz2", 4)
768 finally:
769 bz2.BZ2File = original_open
770
771 self.assertEqual(self.fake_open.invocation_count, 1)
772 self.assertEqual(self.fake_open.last_invocation, (("test.bz2", 4), {}))
773
774 def test_blah_ext(self):
775 self.do_test_use_builtin_open("abcd.blah", 5)
776
777 def test_Gz_ext(self):
778 self.do_test_use_builtin_open("abcd.Gz", 6)
779
780 def test_Bz2_ext(self):
781 self.do_test_use_builtin_open("abcd.Bz2", 7)
782
783 def do_test_use_builtin_open(self, filename, mode):
784 original_open = self.replace_builtin_open(self.fake_open)
785 try:
786 result = fileinput.hook_compressed(filename, mode)
787 finally:
788 self.replace_builtin_open(original_open)
789
790 self.assertEqual(self.fake_open.invocation_count, 1)
791 self.assertEqual(self.fake_open.last_invocation,
792 ((filename, mode), {}))
793
794 @staticmethod
795 def replace_builtin_open(new_open_func):
796 builtins_type = type(__builtins__)
797 if builtins_type is dict:
798 original_open = __builtins__["open"]
799 __builtins__["open"] = new_open_func
800 elif builtins_type is types.ModuleType:
801 original_open = __builtins__.open
802 __builtins__.open = new_open_func
803 else:
804 raise RuntimeError(
805 "unknown __builtins__ type: %r (unable to replace open)" %
806 builtins_type)
807
808 return original_open
809
810class Test_hook_encoded(unittest.TestCase):
811 """Unit tests for fileinput.hook_encoded()"""
812
813 def test(self):
814 encoding = object()
815 result = fileinput.hook_encoded(encoding)
816
817 fake_open = InvocationRecorder()
818 original_open = codecs.open
819 codecs.open = fake_open
820 try:
821 filename = object()
822 mode = object()
823 open_result = result(filename, mode)
824 finally:
825 codecs.open = original_open
826
827 self.assertEqual(fake_open.invocation_count, 1)
828
829 args = fake_open.last_invocation[0]
830 self.assertIs(args[0], filename)
831 self.assertIs(args[1], mode)
832 self.assertIs(args[2], encoding)
Georg Brandl6cb7b652010-07-31 20:08:15 +0000833
Guido van Rossumd8faa362007-04-27 19:54:29 +0000834def test_main():
briancurtin906f0c42011-03-15 10:29:41 -0400835 run_unittest(
836 BufferSizesTests,
837 FileInputTests,
838 Test_fileinput_input,
839 Test_fileinput_close,
840 Test_fileinput_nextfile,
841 Test_fileinput_filename,
842 Test_fileinput_lineno,
843 Test_fileinput_filelineno,
844 Test_fileinput_fileno,
845 Test_fileinput_isfirstline,
846 Test_fileinput_isstdin,
847 Test_hook_compressed,
848 Test_hook_encoded,
849 )
Guido van Rossumd8faa362007-04-27 19:54:29 +0000850
851if __name__ == "__main__":
852 test_main()