blob: c6863d6e7c1e6e2753ce0f757f0843e0340d2c1f [file] [log] [blame]
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +00001# Adapted from test_file.py by Daniel Stutzbach
Antoine Pitrou19690592009-06-12 20:14:08 +00002
3from __future__ import unicode_literals
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +00004
5import sys
6import os
Antoine Pitrou19690592009-06-12 20:14:08 +00007import errno
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +00008import unittest
9from array import array
10from weakref import proxy
Antoine Pitrou19690592009-06-12 20:14:08 +000011from functools import wraps
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000012
Georg Brandla4f46e12010-02-07 17:03:15 +000013from test.test_support import TESTFN, check_warnings, run_unittest, make_bad_fd
Antoine Pitrou19690592009-06-12 20:14:08 +000014from test.test_support import py3k_bytes as bytes
Antoine Pitrou7c587bf2010-04-19 18:52:43 +000015from test.script_helper import run_python
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000016
Antoine Pitrou19690592009-06-12 20:14:08 +000017from _io import FileIO as _FileIO
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000018
19class AutoFileTests(unittest.TestCase):
20 # file tests for which a test file is automatically set up
21
22 def setUp(self):
Antoine Pitrou19690592009-06-12 20:14:08 +000023 self.f = _FileIO(TESTFN, 'w')
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000024
25 def tearDown(self):
26 if self.f:
27 self.f.close()
28 os.remove(TESTFN)
29
30 def testWeakRefs(self):
31 # verify weak references
32 p = proxy(self.f)
33 p.write(bytes(range(10)))
Ezio Melotti2623a372010-11-21 13:34:58 +000034 self.assertEqual(self.f.tell(), p.tell())
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000035 self.f.close()
36 self.f = None
37 self.assertRaises(ReferenceError, getattr, p, 'tell')
38
39 def testSeekTell(self):
Antoine Pitrou19690592009-06-12 20:14:08 +000040 self.f.write(bytes(range(20)))
Ezio Melotti2623a372010-11-21 13:34:58 +000041 self.assertEqual(self.f.tell(), 20)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000042 self.f.seek(0)
Ezio Melotti2623a372010-11-21 13:34:58 +000043 self.assertEqual(self.f.tell(), 0)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000044 self.f.seek(10)
Ezio Melotti2623a372010-11-21 13:34:58 +000045 self.assertEqual(self.f.tell(), 10)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000046 self.f.seek(5, 1)
Ezio Melotti2623a372010-11-21 13:34:58 +000047 self.assertEqual(self.f.tell(), 15)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000048 self.f.seek(-5, 1)
Ezio Melotti2623a372010-11-21 13:34:58 +000049 self.assertEqual(self.f.tell(), 10)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000050 self.f.seek(-5, 2)
Ezio Melotti2623a372010-11-21 13:34:58 +000051 self.assertEqual(self.f.tell(), 15)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000052
53 def testAttributes(self):
54 # verify expected attributes exist
55 f = self.f
56
Ezio Melotti2623a372010-11-21 13:34:58 +000057 self.assertEqual(f.mode, "wb")
58 self.assertEqual(f.closed, False)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000059
60 # verify the attributes are readonly
61 for attr in 'mode', 'closed':
62 self.assertRaises((AttributeError, TypeError),
63 setattr, f, attr, 'oops')
64
65 def testReadinto(self):
66 # verify readinto
Antoine Pitrou19690592009-06-12 20:14:08 +000067 self.f.write(b"\x01\x02")
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000068 self.f.close()
Antoine Pitrou19690592009-06-12 20:14:08 +000069 a = array(b'b', b'x'*10)
70 self.f = _FileIO(TESTFN, 'r')
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000071 n = self.f.readinto(a)
Ezio Melotti2623a372010-11-21 13:34:58 +000072 self.assertEqual(array(b'b', [1, 2]), a[:n])
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000073
Benjamin Petersonddd392c2009-12-13 19:19:07 +000074 def test_none_args(self):
75 self.f.write(b"hi\nbye\nabc")
76 self.f.close()
77 self.f = _FileIO(TESTFN, 'r')
78 self.assertEqual(self.f.read(None), b"hi\nbye\nabc")
79 self.f.seek(0)
80 self.assertEqual(self.f.readline(None), b"hi\n")
81 self.assertEqual(self.f.readlines(None), [b"bye\n", b"abc"])
82
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000083 def testRepr(self):
Ezio Melotti2623a372010-11-21 13:34:58 +000084 self.assertEqual(repr(self.f), "<_io.FileIO name=%r mode='%s'>"
85 % (self.f.name, self.f.mode))
Antoine Pitrou19690592009-06-12 20:14:08 +000086 del self.f.name
Ezio Melotti2623a372010-11-21 13:34:58 +000087 self.assertEqual(repr(self.f), "<_io.FileIO fd=%r mode='%s'>"
88 % (self.f.fileno(), self.f.mode))
Antoine Pitrou19690592009-06-12 20:14:08 +000089 self.f.close()
Ezio Melotti2623a372010-11-21 13:34:58 +000090 self.assertEqual(repr(self.f), "<_io.FileIO [closed]>")
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000091
92 def testErrors(self):
93 f = self.f
Benjamin Peterson5c8da862009-06-30 22:57:08 +000094 self.assertTrue(not f.isatty())
95 self.assertTrue(not f.closed)
Ezio Melotti2623a372010-11-21 13:34:58 +000096 #self.assertEqual(f.name, TESTFN)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000097 self.assertRaises(ValueError, f.read, 10) # Open for reading
98 f.close()
Benjamin Peterson5c8da862009-06-30 22:57:08 +000099 self.assertTrue(f.closed)
Antoine Pitrou19690592009-06-12 20:14:08 +0000100 f = _FileIO(TESTFN, 'r')
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000101 self.assertRaises(TypeError, f.readinto, "")
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000102 self.assertTrue(not f.closed)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000103 f.close()
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000104 self.assertTrue(f.closed)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000105
106 def testMethods(self):
107 methods = ['fileno', 'isatty', 'read', 'readinto',
108 'seek', 'tell', 'truncate', 'write', 'seekable',
109 'readable', 'writable']
110 if sys.platform.startswith('atheos'):
111 methods.remove('truncate')
112
113 self.f.close()
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000114 self.assertTrue(self.f.closed)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000115
116 for methodname in methods:
117 method = getattr(self.f, methodname)
118 # should raise on closed file
119 self.assertRaises(ValueError, method)
120
Benjamin Petersonf22c26e2008-09-01 14:13:43 +0000121 def testOpendir(self):
122 # Issue 3703: opening a directory should fill the errno
123 # Windows always returns "[Errno 13]: Permission denied
124 # Unix calls dircheck() and returns "[Errno 21]: Is a directory"
125 try:
Antoine Pitrou19690592009-06-12 20:14:08 +0000126 _FileIO('.', 'r')
Benjamin Petersonf22c26e2008-09-01 14:13:43 +0000127 except IOError as e:
128 self.assertNotEqual(e.errno, 0)
Benjamin Peterson7af65562008-12-29 17:56:58 +0000129 self.assertEqual(e.filename, ".")
Benjamin Petersonf22c26e2008-09-01 14:13:43 +0000130 else:
131 self.fail("Should have raised IOError")
132
Antoine Pitrouc2ec9922012-07-06 18:48:24 +0200133 @unittest.skipIf(os.name == 'nt', "test only works on a POSIX-like system")
134 def testOpenDirFD(self):
135 fd = os.open('.', os.O_RDONLY)
136 with self.assertRaises(IOError) as cm:
137 _FileIO(fd, 'r')
138 os.close(fd)
139 self.assertEqual(cm.exception.errno, errno.EISDIR)
140
Antoine Pitrou19690592009-06-12 20:14:08 +0000141 #A set of functions testing that we get expected behaviour if someone has
142 #manually closed the internal file descriptor. First, a decorator:
143 def ClosedFD(func):
144 @wraps(func)
145 def wrapper(self):
146 #forcibly close the fd before invoking the problem function
147 f = self.f
148 os.close(f.fileno())
149 try:
150 func(self, f)
151 finally:
152 try:
153 self.f.close()
154 except IOError:
155 pass
156 return wrapper
157
158 def ClosedFDRaises(func):
159 @wraps(func)
160 def wrapper(self):
161 #forcibly close the fd before invoking the problem function
162 f = self.f
163 os.close(f.fileno())
164 try:
165 func(self, f)
166 except IOError as e:
167 self.assertEqual(e.errno, errno.EBADF)
168 else:
169 self.fail("Should have raised IOError")
170 finally:
171 try:
172 self.f.close()
173 except IOError:
174 pass
175 return wrapper
176
177 @ClosedFDRaises
178 def testErrnoOnClose(self, f):
179 f.close()
180
181 @ClosedFDRaises
182 def testErrnoOnClosedWrite(self, f):
183 f.write('a')
184
185 @ClosedFDRaises
186 def testErrnoOnClosedSeek(self, f):
187 f.seek(0)
188
189 @ClosedFDRaises
190 def testErrnoOnClosedTell(self, f):
191 f.tell()
192
193 @ClosedFDRaises
194 def testErrnoOnClosedTruncate(self, f):
195 f.truncate(0)
196
197 @ClosedFD
198 def testErrnoOnClosedSeekable(self, f):
199 f.seekable()
200
201 @ClosedFD
202 def testErrnoOnClosedReadable(self, f):
203 f.readable()
204
205 @ClosedFD
206 def testErrnoOnClosedWritable(self, f):
207 f.writable()
208
209 @ClosedFD
210 def testErrnoOnClosedFileno(self, f):
211 f.fileno()
212
213 @ClosedFD
214 def testErrnoOnClosedIsatty(self, f):
215 self.assertEqual(f.isatty(), False)
216
217 def ReopenForRead(self):
218 try:
219 self.f.close()
220 except IOError:
221 pass
222 self.f = _FileIO(TESTFN, 'r')
223 os.close(self.f.fileno())
224 return self.f
225
226 @ClosedFDRaises
227 def testErrnoOnClosedRead(self, f):
228 f = self.ReopenForRead()
229 f.read(1)
230
231 @ClosedFDRaises
232 def testErrnoOnClosedReadall(self, f):
233 f = self.ReopenForRead()
234 f.readall()
235
236 @ClosedFDRaises
237 def testErrnoOnClosedReadinto(self, f):
238 f = self.ReopenForRead()
239 a = array(b'b', b'x'*10)
240 f.readinto(a)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000241
242class OtherFileTests(unittest.TestCase):
243
244 def testAbles(self):
245 try:
Antoine Pitrou19690592009-06-12 20:14:08 +0000246 f = _FileIO(TESTFN, "w")
Ezio Melotti2623a372010-11-21 13:34:58 +0000247 self.assertEqual(f.readable(), False)
248 self.assertEqual(f.writable(), True)
249 self.assertEqual(f.seekable(), True)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000250 f.close()
251
Antoine Pitrou19690592009-06-12 20:14:08 +0000252 f = _FileIO(TESTFN, "r")
Ezio Melotti2623a372010-11-21 13:34:58 +0000253 self.assertEqual(f.readable(), True)
254 self.assertEqual(f.writable(), False)
255 self.assertEqual(f.seekable(), True)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000256 f.close()
257
Antoine Pitrou19690592009-06-12 20:14:08 +0000258 f = _FileIO(TESTFN, "a+")
Ezio Melotti2623a372010-11-21 13:34:58 +0000259 self.assertEqual(f.readable(), True)
260 self.assertEqual(f.writable(), True)
261 self.assertEqual(f.seekable(), True)
262 self.assertEqual(f.isatty(), False)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000263 f.close()
264
265 if sys.platform != "win32":
266 try:
Antoine Pitrou19690592009-06-12 20:14:08 +0000267 f = _FileIO("/dev/tty", "a")
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000268 except EnvironmentError:
269 # When run in a cron job there just aren't any
270 # ttys, so skip the test. This also handles other
271 # OS'es that don't support /dev/tty.
272 pass
273 else:
Ezio Melotti2623a372010-11-21 13:34:58 +0000274 self.assertEqual(f.readable(), False)
275 self.assertEqual(f.writable(), True)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000276 if sys.platform != "darwin" and \
Antoine Pitrouf3bd6872009-05-23 16:32:32 +0000277 'bsd' not in sys.platform and \
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000278 not sys.platform.startswith('sunos'):
279 # Somehow /dev/tty appears seekable on some BSDs
Ezio Melotti2623a372010-11-21 13:34:58 +0000280 self.assertEqual(f.seekable(), False)
281 self.assertEqual(f.isatty(), True)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000282 f.close()
283 finally:
284 os.unlink(TESTFN)
285
286 def testModeStrings(self):
287 # check invalid mode strings
Benjamin Petersonbfc51562008-11-22 01:59:15 +0000288 for mode in ("", "aU", "wU+", "rw", "rt"):
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000289 try:
Antoine Pitrou19690592009-06-12 20:14:08 +0000290 f = _FileIO(TESTFN, mode)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000291 except ValueError:
292 pass
293 else:
294 f.close()
295 self.fail('%r is an invalid file mode' % mode)
296
297 def testUnicodeOpen(self):
298 # verify repr works for unicode too
Antoine Pitrou19690592009-06-12 20:14:08 +0000299 f = _FileIO(str(TESTFN), "w")
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000300 f.close()
301 os.unlink(TESTFN)
302
Antoine Pitrou19690592009-06-12 20:14:08 +0000303 def testBytesOpen(self):
304 # Opening a bytes filename
305 try:
306 fn = TESTFN.encode("ascii")
307 except UnicodeEncodeError:
308 # Skip test
309 return
310 f = _FileIO(fn, "w")
311 try:
312 f.write(b"abc")
313 f.close()
314 with open(TESTFN, "rb") as f:
Ezio Melotti2623a372010-11-21 13:34:58 +0000315 self.assertEqual(f.read(), b"abc")
Antoine Pitrou19690592009-06-12 20:14:08 +0000316 finally:
317 os.unlink(TESTFN)
318
Benjamin Peterson5848d1f2009-01-19 00:08:08 +0000319 def testInvalidFd(self):
Antoine Pitrou19690592009-06-12 20:14:08 +0000320 self.assertRaises(ValueError, _FileIO, -10)
321 self.assertRaises(OSError, _FileIO, make_bad_fd())
Antoine Pitroue6ebcda2010-09-04 21:24:42 +0000322 if sys.platform == 'win32':
323 import msvcrt
324 self.assertRaises(IOError, msvcrt.get_osfhandle, make_bad_fd())
Benjamin Peterson5848d1f2009-01-19 00:08:08 +0000325
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000326 def testBadModeArgument(self):
327 # verify that we get a sensible error message for bad mode argument
328 bad_mode = "qwerty"
329 try:
Antoine Pitrou19690592009-06-12 20:14:08 +0000330 f = _FileIO(TESTFN, bad_mode)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000331 except ValueError as msg:
332 if msg.args[0] != 0:
333 s = str(msg)
Ezio Melotti187f93d2010-03-17 14:22:34 +0000334 if TESTFN in s or bad_mode not in s:
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000335 self.fail("bad error message for invalid mode: %s" % s)
336 # if msg.args[0] == 0, we're probably on Windows where there may be
337 # no obvious way to discover why open() failed.
338 else:
339 f.close()
340 self.fail("no error for invalid mode: %s" % bad_mode)
341
Antoine Pitrouf3fa0742010-01-31 22:26:04 +0000342 def testTruncate(self):
343 f = _FileIO(TESTFN, 'w')
344 f.write(bytes(bytearray(range(10))))
345 self.assertEqual(f.tell(), 10)
346 f.truncate(5)
347 self.assertEqual(f.tell(), 10)
348 self.assertEqual(f.seek(0, os.SEEK_END), 5)
349 f.truncate(15)
350 self.assertEqual(f.tell(), 5)
351 self.assertEqual(f.seek(0, os.SEEK_END), 15)
Antoine Pitrou594a0462010-10-31 13:05:48 +0000352 f.close()
Antoine Pitrouf3fa0742010-01-31 22:26:04 +0000353
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000354 def testTruncateOnWindows(self):
355 def bug801631():
356 # SF bug <http://www.python.org/sf/801631>
357 # "file.truncate fault on windows"
Antoine Pitrou19690592009-06-12 20:14:08 +0000358 f = _FileIO(TESTFN, 'w')
359 f.write(bytes(range(11)))
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000360 f.close()
361
Antoine Pitrou19690592009-06-12 20:14:08 +0000362 f = _FileIO(TESTFN,'r+')
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000363 data = f.read(5)
Antoine Pitrou19690592009-06-12 20:14:08 +0000364 if data != bytes(range(5)):
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000365 self.fail("Read on file opened for update failed %r" % data)
366 if f.tell() != 5:
367 self.fail("File pos after read wrong %d" % f.tell())
368
369 f.truncate()
370 if f.tell() != 5:
371 self.fail("File pos after ftruncate wrong %d" % f.tell())
372
373 f.close()
374 size = os.path.getsize(TESTFN)
375 if size != 5:
376 self.fail("File size after ftruncate wrong %d" % size)
377
378 try:
379 bug801631()
380 finally:
381 os.unlink(TESTFN)
382
383 def testAppend(self):
384 try:
385 f = open(TESTFN, 'wb')
386 f.write(b'spam')
387 f.close()
388 f = open(TESTFN, 'ab')
389 f.write(b'eggs')
390 f.close()
391 f = open(TESTFN, 'rb')
392 d = f.read()
393 f.close()
394 self.assertEqual(d, b'spameggs')
395 finally:
396 try:
397 os.unlink(TESTFN)
398 except:
399 pass
400
Neal Norwitz901e4712008-08-24 22:03:05 +0000401 def testInvalidInit(self):
Antoine Pitrou19690592009-06-12 20:14:08 +0000402 self.assertRaises(TypeError, _FileIO, "1", 0, 0)
Neal Norwitz901e4712008-08-24 22:03:05 +0000403
Benjamin Peterson81e92e52008-11-03 22:34:57 +0000404 def testWarnings(self):
Florent Xicluna6de9e932010-03-07 12:18:33 +0000405 with check_warnings(quiet=True) as w:
Benjamin Peterson81e92e52008-11-03 22:34:57 +0000406 self.assertEqual(w.warnings, [])
Antoine Pitrou19690592009-06-12 20:14:08 +0000407 self.assertRaises(TypeError, _FileIO, [])
Benjamin Peterson81e92e52008-11-03 22:34:57 +0000408 self.assertEqual(w.warnings, [])
Antoine Pitrou19690592009-06-12 20:14:08 +0000409 self.assertRaises(ValueError, _FileIO, "/some/invalid/name", "rt")
Benjamin Peterson81e92e52008-11-03 22:34:57 +0000410 self.assertEqual(w.warnings, [])
411
Antoine Pitrou7c587bf2010-04-19 18:52:43 +0000412 def test_surrogates(self):
413 # Issue #8438: try to open a filename containing surrogates.
414 # It should either fail because the file doesn't exist or the filename
415 # can't be represented using the filesystem encoding, but not because
416 # of a LookupError for the error handler "surrogateescape".
417 filename = u'\udc80.txt'
418 try:
419 with _FileIO(filename):
420 pass
421 except (UnicodeEncodeError, IOError):
422 pass
423 # Spawn a separate Python process with a different "file system
424 # default encoding", to exercise this further.
425 env = dict(os.environ)
426 env[b'LC_CTYPE'] = b'C'
427 _, out = run_python('-c', 'import _io; _io.FileIO(%r)' % filename, env=env)
428 if ('UnicodeEncodeError' not in out and
429 'IOError: [Errno 2] No such file or directory' not in out):
430 self.fail('Bad output: %r' % out)
Neal Norwitz901e4712008-08-24 22:03:05 +0000431
Hynek Schlawack9bd4bf22012-06-21 19:45:19 +0200432 def testUnclosedFDOnException(self):
433 class MyException(Exception): pass
434 class MyFileIO(_FileIO):
435 def __setattr__(self, name, value):
436 if name == "name":
437 raise MyException("blocked setting name")
438 return super(MyFileIO, self).__setattr__(name, value)
439 fd = os.open(__file__, os.O_RDONLY)
440 self.assertRaises(MyException, MyFileIO, fd)
441 os.close(fd) # should not raise OSError(EBADF)
442
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000443def test_main():
444 # Historically, these tests have been sloppy about removing TESTFN.
445 # So get rid of it no matter what.
446 try:
447 run_unittest(AutoFileTests, OtherFileTests)
448 finally:
449 if os.path.exists(TESTFN):
450 os.unlink(TESTFN)
451
452if __name__ == '__main__':
453 test_main()