blob: 906952ccf259d3b4b7d23ac6b8319598ef4985a3 [file] [log] [blame]
Guido van Rossumde3bc7c2007-03-08 01:02:00 +00001# Adapted from test_file.py by Daniel Stutzbach
Guido van Rossuma9e20242007-03-08 00:43:48 +00002
3import sys
4import os
Eli Bendersky74c503b2012-01-03 06:26:13 +02005import io
Antoine Pitrou0ae29cf2009-03-13 22:33:17 +00006import errno
Guido van Rossuma9e20242007-03-08 00:43:48 +00007import unittest
8from array import array
9from weakref import proxy
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +000010from functools import wraps
Guido van Rossuma9e20242007-03-08 00:43:48 +000011
Georg Brandl1b37e872010-03-14 10:45:50 +000012from test.support import TESTFN, check_warnings, run_unittest, make_bad_fd
Guido van Rossuma9e20242007-03-08 00:43:48 +000013
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +000014from _io import FileIO as _FileIO
Guido van Rossuma9e20242007-03-08 00:43:48 +000015
16class AutoFileTests(unittest.TestCase):
17 # file tests for which a test file is automatically set up
18
19 def setUp(self):
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +000020 self.f = _FileIO(TESTFN, 'w')
Guido van Rossuma9e20242007-03-08 00:43:48 +000021
22 def tearDown(self):
23 if self.f:
24 self.f.close()
25 os.remove(TESTFN)
26
27 def testWeakRefs(self):
28 # verify weak references
29 p = proxy(self.f)
30 p.write(bytes(range(10)))
Ezio Melottib3aedd42010-11-20 19:04:17 +000031 self.assertEqual(self.f.tell(), p.tell())
Guido van Rossuma9e20242007-03-08 00:43:48 +000032 self.f.close()
33 self.f = None
34 self.assertRaises(ReferenceError, getattr, p, 'tell')
35
36 def testSeekTell(self):
37 self.f.write(bytes(range(20)))
Ezio Melottib3aedd42010-11-20 19:04:17 +000038 self.assertEqual(self.f.tell(), 20)
Guido van Rossuma9e20242007-03-08 00:43:48 +000039 self.f.seek(0)
Ezio Melottib3aedd42010-11-20 19:04:17 +000040 self.assertEqual(self.f.tell(), 0)
Guido van Rossuma9e20242007-03-08 00:43:48 +000041 self.f.seek(10)
Ezio Melottib3aedd42010-11-20 19:04:17 +000042 self.assertEqual(self.f.tell(), 10)
Guido van Rossuma9e20242007-03-08 00:43:48 +000043 self.f.seek(5, 1)
Ezio Melottib3aedd42010-11-20 19:04:17 +000044 self.assertEqual(self.f.tell(), 15)
Guido van Rossuma9e20242007-03-08 00:43:48 +000045 self.f.seek(-5, 1)
Ezio Melottib3aedd42010-11-20 19:04:17 +000046 self.assertEqual(self.f.tell(), 10)
Guido van Rossuma9e20242007-03-08 00:43:48 +000047 self.f.seek(-5, 2)
Ezio Melottib3aedd42010-11-20 19:04:17 +000048 self.assertEqual(self.f.tell(), 15)
Guido van Rossuma9e20242007-03-08 00:43:48 +000049
50 def testAttributes(self):
51 # verify expected attributes exist
52 f = self.f
Guido van Rossuma9e20242007-03-08 00:43:48 +000053
Ezio Melottib3aedd42010-11-20 19:04:17 +000054 self.assertEqual(f.mode, "wb")
55 self.assertEqual(f.closed, False)
Guido van Rossum31c3a572007-04-12 14:51:49 +000056
57 # verify the attributes are readonly
58 for attr in 'mode', 'closed':
59 self.assertRaises((AttributeError, TypeError),
60 setattr, f, attr, 'oops')
Guido van Rossuma9e20242007-03-08 00:43:48 +000061
62 def testReadinto(self):
63 # verify readinto
64 self.f.write(bytes([1, 2]))
65 self.f.close()
Guido van Rossuma45ea582007-05-15 21:25:12 +000066 a = array('b', b'x'*10)
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +000067 self.f = _FileIO(TESTFN, 'r')
Guido van Rossuma9e20242007-03-08 00:43:48 +000068 n = self.f.readinto(a)
Ezio Melottib3aedd42010-11-20 19:04:17 +000069 self.assertEqual(array('b', [1, 2]), a[:n])
Guido van Rossuma9e20242007-03-08 00:43:48 +000070
Benjamin Petersonbf5ff762009-12-13 19:25:34 +000071 def test_none_args(self):
72 self.f.write(b"hi\nbye\nabc")
73 self.f.close()
74 self.f = _FileIO(TESTFN, 'r')
75 self.assertEqual(self.f.read(None), b"hi\nbye\nabc")
76 self.f.seek(0)
77 self.assertEqual(self.f.readline(None), b"hi\n")
78 self.assertEqual(self.f.readlines(None), [b"bye\n", b"abc"])
79
Benjamin Peterson255058f2010-01-27 01:47:14 +000080 def test_reject(self):
81 self.assertRaises(TypeError, self.f.write, "Hello!")
82
Guido van Rossuma9e20242007-03-08 00:43:48 +000083 def testRepr(self):
Ezio Melottib3aedd42010-11-20 19:04:17 +000084 self.assertEqual(repr(self.f), "<_io.FileIO name=%r mode=%r>"
Antoine Pitrou716c4442009-05-23 19:04:03 +000085 % (self.f.name, self.f.mode))
86 del self.f.name
Ezio Melottib3aedd42010-11-20 19:04:17 +000087 self.assertEqual(repr(self.f), "<_io.FileIO fd=%r mode=%r>"
Antoine Pitrou716c4442009-05-23 19:04:03 +000088 % (self.f.fileno(), self.f.mode))
89 self.f.close()
Ezio Melottib3aedd42010-11-20 19:04:17 +000090 self.assertEqual(repr(self.f), "<_io.FileIO [closed]>")
Guido van Rossuma9e20242007-03-08 00:43:48 +000091
92 def testErrors(self):
93 f = self.f
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000094 self.assertTrue(not f.isatty())
95 self.assertTrue(not f.closed)
Ezio Melottib3aedd42010-11-20 19:04:17 +000096 #self.assertEqual(f.name, TESTFN)
Guido van Rossum31c3a572007-04-12 14:51:49 +000097 self.assertRaises(ValueError, f.read, 10) # Open for reading
Guido van Rossuma9e20242007-03-08 00:43:48 +000098 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000099 self.assertTrue(f.closed)
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000100 f = _FileIO(TESTFN, 'r')
Guido van Rossum31c3a572007-04-12 14:51:49 +0000101 self.assertRaises(TypeError, f.readinto, "")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000102 self.assertTrue(not f.closed)
Guido van Rossum31c3a572007-04-12 14:51:49 +0000103 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000104 self.assertTrue(f.closed)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000105
106 def testMethods(self):
107 methods = ['fileno', 'isatty', 'read', 'readinto',
108 'seek', 'tell', 'truncate', 'write', 'seekable',
109 'readable', 'writable']
Guido van Rossuma9e20242007-03-08 00:43:48 +0000110
Guido van Rossum31c3a572007-04-12 14:51:49 +0000111 self.f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000112 self.assertTrue(self.f.closed)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000113
114 for methodname in methods:
115 method = getattr(self.f, methodname)
116 # should raise on closed file
117 self.assertRaises(ValueError, method)
118
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000119 def testOpendir(self):
120 # Issue 3703: opening a directory should fill the errno
121 # Windows always returns "[Errno 13]: Permission denied
122 # Unix calls dircheck() and returns "[Errno 21]: Is a directory"
123 try:
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000124 _FileIO('.', 'r')
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000125 except IOError as e:
126 self.assertNotEqual(e.errno, 0)
Benjamin Peterson1efc23c2008-12-29 18:02:28 +0000127 self.assertEqual(e.filename, ".")
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000128 else:
129 self.fail("Should have raised IOError")
130
Antoine Pitrou9235b252012-07-06 18:48:24 +0200131 @unittest.skipIf(os.name == 'nt', "test only works on a POSIX-like system")
132 def testOpenDirFD(self):
133 fd = os.open('.', os.O_RDONLY)
134 with self.assertRaises(IOError) as cm:
135 _FileIO(fd, 'r')
136 os.close(fd)
137 self.assertEqual(cm.exception.errno, errno.EISDIR)
138
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000139 #A set of functions testing that we get expected behaviour if someone has
140 #manually closed the internal file descriptor. First, a decorator:
141 def ClosedFD(func):
142 @wraps(func)
143 def wrapper(self):
144 #forcibly close the fd before invoking the problem function
145 f = self.f
146 os.close(f.fileno())
147 try:
148 func(self, f)
149 finally:
150 try:
151 self.f.close()
152 except IOError:
153 pass
154 return wrapper
Antoine Pitrou0ae29cf2009-03-13 22:33:17 +0000155
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000156 def ClosedFDRaises(func):
157 @wraps(func)
158 def wrapper(self):
159 #forcibly close the fd before invoking the problem function
160 f = self.f
161 os.close(f.fileno())
162 try:
163 func(self, f)
164 except IOError as e:
165 self.assertEqual(e.errno, errno.EBADF)
166 else:
167 self.fail("Should have raised IOError")
168 finally:
169 try:
170 self.f.close()
171 except IOError:
172 pass
173 return wrapper
174
175 @ClosedFDRaises
176 def testErrnoOnClose(self, f):
177 f.close()
178
179 @ClosedFDRaises
180 def testErrnoOnClosedWrite(self, f):
Benjamin Peterson255058f2010-01-27 01:47:14 +0000181 f.write(b'a')
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000182
183 @ClosedFDRaises
184 def testErrnoOnClosedSeek(self, f):
185 f.seek(0)
186
187 @ClosedFDRaises
188 def testErrnoOnClosedTell(self, f):
189 f.tell()
190
191 @ClosedFDRaises
192 def testErrnoOnClosedTruncate(self, f):
193 f.truncate(0)
194
195 @ClosedFD
196 def testErrnoOnClosedSeekable(self, f):
197 f.seekable()
198
199 @ClosedFD
200 def testErrnoOnClosedReadable(self, f):
201 f.readable()
202
203 @ClosedFD
204 def testErrnoOnClosedWritable(self, f):
205 f.writable()
206
207 @ClosedFD
208 def testErrnoOnClosedFileno(self, f):
209 f.fileno()
210
211 @ClosedFD
212 def testErrnoOnClosedIsatty(self, f):
213 self.assertEqual(f.isatty(), False)
214
215 def ReopenForRead(self):
216 try:
217 self.f.close()
218 except IOError:
219 pass
220 self.f = _FileIO(TESTFN, 'r')
221 os.close(self.f.fileno())
222 return self.f
223
224 @ClosedFDRaises
225 def testErrnoOnClosedRead(self, f):
226 f = self.ReopenForRead()
227 f.read(1)
228
229 @ClosedFDRaises
230 def testErrnoOnClosedReadall(self, f):
231 f = self.ReopenForRead()
232 f.readall()
233
234 @ClosedFDRaises
235 def testErrnoOnClosedReadinto(self, f):
236 f = self.ReopenForRead()
237 a = array('b', b'x'*10)
238 f.readinto(a)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000239
240class OtherFileTests(unittest.TestCase):
241
242 def testAbles(self):
243 try:
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000244 f = _FileIO(TESTFN, "w")
Ezio Melottib3aedd42010-11-20 19:04:17 +0000245 self.assertEqual(f.readable(), False)
246 self.assertEqual(f.writable(), True)
247 self.assertEqual(f.seekable(), True)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000248 f.close()
Guido van Rossumd8faa362007-04-27 19:54:29 +0000249
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000250 f = _FileIO(TESTFN, "r")
Ezio Melottib3aedd42010-11-20 19:04:17 +0000251 self.assertEqual(f.readable(), True)
252 self.assertEqual(f.writable(), False)
253 self.assertEqual(f.seekable(), True)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000254 f.close()
Guido van Rossumd8faa362007-04-27 19:54:29 +0000255
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000256 f = _FileIO(TESTFN, "a+")
Ezio Melottib3aedd42010-11-20 19:04:17 +0000257 self.assertEqual(f.readable(), True)
258 self.assertEqual(f.writable(), True)
259 self.assertEqual(f.seekable(), True)
260 self.assertEqual(f.isatty(), False)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000261 f.close()
Guido van Rossum682faf82007-04-12 14:56:58 +0000262
Thomas Hellerb0f48ab2007-07-12 11:29:02 +0000263 if sys.platform != "win32":
264 try:
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000265 f = _FileIO("/dev/tty", "a")
Thomas Hellerb0f48ab2007-07-12 11:29:02 +0000266 except EnvironmentError:
267 # When run in a cron job there just aren't any
268 # ttys, so skip the test. This also handles other
269 # OS'es that don't support /dev/tty.
270 pass
271 else:
Ezio Melottib3aedd42010-11-20 19:04:17 +0000272 self.assertEqual(f.readable(), False)
273 self.assertEqual(f.writable(), True)
Hye-Shik Changcaf871a2007-08-13 13:21:33 +0000274 if sys.platform != "darwin" and \
Antoine Pitroud7b30462009-05-23 16:34:50 +0000275 'bsd' not in sys.platform and \
Benjamin Peterson94fe10f2008-07-17 23:27:26 +0000276 not sys.platform.startswith('sunos'):
Hye-Shik Changcaf871a2007-08-13 13:21:33 +0000277 # Somehow /dev/tty appears seekable on some BSDs
Ezio Melottib3aedd42010-11-20 19:04:17 +0000278 self.assertEqual(f.seekable(), False)
279 self.assertEqual(f.isatty(), True)
Thomas Hellerb0f48ab2007-07-12 11:29:02 +0000280 f.close()
Guido van Rossuma9e20242007-03-08 00:43:48 +0000281 finally:
282 os.unlink(TESTFN)
283
284 def testModeStrings(self):
285 # check invalid mode strings
Benjamin Peterson44309e62008-11-22 00:41:45 +0000286 for mode in ("", "aU", "wU+", "rw", "rt"):
Guido van Rossuma9e20242007-03-08 00:43:48 +0000287 try:
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000288 f = _FileIO(TESTFN, mode)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000289 except ValueError:
290 pass
291 else:
292 f.close()
293 self.fail('%r is an invalid file mode' % mode)
294
Guido van Rossuma9e20242007-03-08 00:43:48 +0000295 def testUnicodeOpen(self):
296 # verify repr works for unicode too
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000297 f = _FileIO(str(TESTFN), "w")
Guido van Rossuma9e20242007-03-08 00:43:48 +0000298 f.close()
299 os.unlink(TESTFN)
300
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000301 def testBytesOpen(self):
302 # Opening a bytes filename
303 try:
304 fn = TESTFN.encode("ascii")
305 except UnicodeEncodeError:
306 # Skip test
307 return
308 f = _FileIO(fn, "w")
309 try:
310 f.write(b"abc")
311 f.close()
312 with open(TESTFN, "rb") as f:
Ezio Melottib3aedd42010-11-20 19:04:17 +0000313 self.assertEqual(f.read(), b"abc")
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000314 finally:
315 os.unlink(TESTFN)
316
Antoine Pitrou13348842012-01-29 18:36:34 +0100317 def testConstructorHandlesNULChars(self):
318 fn_with_NUL = 'foo\0bar'
319 self.assertRaises(TypeError, _FileIO, fn_with_NUL, 'w')
320 self.assertRaises(TypeError, _FileIO, bytes(fn_with_NUL, 'ascii'), 'w')
321
Benjamin Peterson806d4022009-01-19 15:11:51 +0000322 def testInvalidFd(self):
Antoine Pitrou7fb111b2009-03-04 11:14:01 +0000323 self.assertRaises(ValueError, _FileIO, -10)
324 self.assertRaises(OSError, _FileIO, make_bad_fd())
Antoine Pitrou00492492010-09-04 20:53:29 +0000325 if sys.platform == 'win32':
326 import msvcrt
327 self.assertRaises(IOError, msvcrt.get_osfhandle, make_bad_fd())
Benjamin Peterson806d4022009-01-19 15:11:51 +0000328
Guido van Rossuma9e20242007-03-08 00:43:48 +0000329 def testBadModeArgument(self):
330 # verify that we get a sensible error message for bad mode argument
331 bad_mode = "qwerty"
332 try:
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000333 f = _FileIO(TESTFN, bad_mode)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000334 except ValueError as msg:
Guido van Rossum360e4b82007-05-14 22:51:27 +0000335 if msg.args[0] != 0:
Guido van Rossuma9e20242007-03-08 00:43:48 +0000336 s = str(msg)
Ezio Melotti7fb4da72010-03-18 12:29:13 +0000337 if TESTFN in s or bad_mode not in s:
Guido van Rossuma9e20242007-03-08 00:43:48 +0000338 self.fail("bad error message for invalid mode: %s" % s)
Georg Brandl50da60c2008-01-06 21:38:54 +0000339 # if msg.args[0] == 0, we're probably on Windows where there may be
Guido van Rossuma9e20242007-03-08 00:43:48 +0000340 # no obvious way to discover why open() failed.
341 else:
342 f.close()
343 self.fail("no error for invalid mode: %s" % bad_mode)
344
Antoine Pitrou905a2ff2010-01-31 22:47:27 +0000345 def testTruncate(self):
346 f = _FileIO(TESTFN, 'w')
347 f.write(bytes(bytearray(range(10))))
348 self.assertEqual(f.tell(), 10)
349 f.truncate(5)
350 self.assertEqual(f.tell(), 10)
Eli Bendersky74c503b2012-01-03 06:26:13 +0200351 self.assertEqual(f.seek(0, io.SEEK_END), 5)
Antoine Pitrou905a2ff2010-01-31 22:47:27 +0000352 f.truncate(15)
353 self.assertEqual(f.tell(), 5)
Eli Bendersky74c503b2012-01-03 06:26:13 +0200354 self.assertEqual(f.seek(0, io.SEEK_END), 15)
Antoine Pitrou8d2b51b2010-10-30 16:19:14 +0000355 f.close()
Antoine Pitrou905a2ff2010-01-31 22:47:27 +0000356
Guido van Rossuma9e20242007-03-08 00:43:48 +0000357 def testTruncateOnWindows(self):
358 def bug801631():
359 # SF bug <http://www.python.org/sf/801631>
360 # "file.truncate fault on windows"
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000361 f = _FileIO(TESTFN, 'w')
Guido van Rossuma9e20242007-03-08 00:43:48 +0000362 f.write(bytes(range(11)))
363 f.close()
364
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000365 f = _FileIO(TESTFN,'r+')
Guido van Rossuma9e20242007-03-08 00:43:48 +0000366 data = f.read(5)
367 if data != bytes(range(5)):
368 self.fail("Read on file opened for update failed %r" % data)
369 if f.tell() != 5:
370 self.fail("File pos after read wrong %d" % f.tell())
371
372 f.truncate()
373 if f.tell() != 5:
374 self.fail("File pos after ftruncate wrong %d" % f.tell())
375
376 f.close()
377 size = os.path.getsize(TESTFN)
378 if size != 5:
379 self.fail("File size after ftruncate wrong %d" % size)
380
381 try:
382 bug801631()
383 finally:
384 os.unlink(TESTFN)
385
Walter Dörwald3a77c7a2007-06-06 16:31:14 +0000386 def testAppend(self):
387 try:
388 f = open(TESTFN, 'wb')
389 f.write(b'spam')
390 f.close()
391 f = open(TESTFN, 'ab')
392 f.write(b'eggs')
393 f.close()
394 f = open(TESTFN, 'rb')
395 d = f.read()
396 f.close()
397 self.assertEqual(d, b'spameggs')
398 finally:
399 try:
400 os.unlink(TESTFN)
401 except:
402 pass
403
Neal Norwitz6e0e0e62008-08-24 22:07:28 +0000404 def testInvalidInit(self):
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000405 self.assertRaises(TypeError, _FileIO, "1", 0, 0)
Neal Norwitz6e0e0e62008-08-24 22:07:28 +0000406
Benjamin Peterson65676e42008-11-05 21:42:45 +0000407 def testWarnings(self):
Florent Xiclunab14930c2010-03-13 15:26:44 +0000408 with check_warnings(quiet=True) as w:
Benjamin Peterson65676e42008-11-05 21:42:45 +0000409 self.assertEqual(w.warnings, [])
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000410 self.assertRaises(TypeError, _FileIO, [])
Benjamin Peterson65676e42008-11-05 21:42:45 +0000411 self.assertEqual(w.warnings, [])
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000412 self.assertRaises(ValueError, _FileIO, "/some/invalid/name", "rt")
Benjamin Peterson65676e42008-11-05 21:42:45 +0000413 self.assertEqual(w.warnings, [])
414
Hynek Schlawack9ed8b4e2012-06-21 20:20:25 +0200415 def testUnclosedFDOnException(self):
416 class MyException(Exception): pass
417 class MyFileIO(_FileIO):
418 def __setattr__(self, name, value):
419 if name == "name":
420 raise MyException("blocked setting name")
421 return super(MyFileIO, self).__setattr__(name, value)
422 fd = os.open(__file__, os.O_RDONLY)
423 self.assertRaises(MyException, MyFileIO, fd)
424 os.close(fd) # should not raise OSError(EBADF)
425
Neal Norwitz6e0e0e62008-08-24 22:07:28 +0000426
Guido van Rossuma9e20242007-03-08 00:43:48 +0000427def test_main():
428 # Historically, these tests have been sloppy about removing TESTFN.
429 # So get rid of it no matter what.
430 try:
431 run_unittest(AutoFileTests, OtherFileTests)
432 finally:
433 if os.path.exists(TESTFN):
434 os.unlink(TESTFN)
435
436if __name__ == '__main__':
437 test_main()