blob: 3588fb449b401da83e59d2d94a87a73b11c6ec01 [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
Antoine Pitrou0ae29cf2009-03-13 22:33:17 +00005import errno
Guido van Rossuma9e20242007-03-08 00:43:48 +00006import unittest
7from array import array
8from weakref import proxy
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +00009from functools import wraps
Guido van Rossuma9e20242007-03-08 00:43:48 +000010
Georg Brandl1b37e872010-03-14 10:45:50 +000011from test.support import TESTFN, check_warnings, run_unittest, make_bad_fd
Guido van Rossuma9e20242007-03-08 00:43:48 +000012
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +000013from _io import FileIO as _FileIO
Guido van Rossuma9e20242007-03-08 00:43:48 +000014
15class AutoFileTests(unittest.TestCase):
16 # file tests for which a test file is automatically set up
17
18 def setUp(self):
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +000019 self.f = _FileIO(TESTFN, 'w')
Guido van Rossuma9e20242007-03-08 00:43:48 +000020
21 def tearDown(self):
22 if self.f:
23 self.f.close()
24 os.remove(TESTFN)
25
26 def testWeakRefs(self):
27 # verify weak references
28 p = proxy(self.f)
29 p.write(bytes(range(10)))
Ezio Melottib3aedd42010-11-20 19:04:17 +000030 self.assertEqual(self.f.tell(), p.tell())
Guido van Rossuma9e20242007-03-08 00:43:48 +000031 self.f.close()
32 self.f = None
33 self.assertRaises(ReferenceError, getattr, p, 'tell')
34
35 def testSeekTell(self):
36 self.f.write(bytes(range(20)))
Ezio Melottib3aedd42010-11-20 19:04:17 +000037 self.assertEqual(self.f.tell(), 20)
Guido van Rossuma9e20242007-03-08 00:43:48 +000038 self.f.seek(0)
Ezio Melottib3aedd42010-11-20 19:04:17 +000039 self.assertEqual(self.f.tell(), 0)
Guido van Rossuma9e20242007-03-08 00:43:48 +000040 self.f.seek(10)
Ezio Melottib3aedd42010-11-20 19:04:17 +000041 self.assertEqual(self.f.tell(), 10)
Guido van Rossuma9e20242007-03-08 00:43:48 +000042 self.f.seek(5, 1)
Ezio Melottib3aedd42010-11-20 19:04:17 +000043 self.assertEqual(self.f.tell(), 15)
Guido van Rossuma9e20242007-03-08 00:43:48 +000044 self.f.seek(-5, 1)
Ezio Melottib3aedd42010-11-20 19:04:17 +000045 self.assertEqual(self.f.tell(), 10)
Guido van Rossuma9e20242007-03-08 00:43:48 +000046 self.f.seek(-5, 2)
Ezio Melottib3aedd42010-11-20 19:04:17 +000047 self.assertEqual(self.f.tell(), 15)
Guido van Rossuma9e20242007-03-08 00:43:48 +000048
49 def testAttributes(self):
50 # verify expected attributes exist
51 f = self.f
Guido van Rossuma9e20242007-03-08 00:43:48 +000052
Ezio Melottib3aedd42010-11-20 19:04:17 +000053 self.assertEqual(f.mode, "wb")
54 self.assertEqual(f.closed, False)
Guido van Rossum31c3a572007-04-12 14:51:49 +000055
56 # verify the attributes are readonly
57 for attr in 'mode', 'closed':
58 self.assertRaises((AttributeError, TypeError),
59 setattr, f, attr, 'oops')
Guido van Rossuma9e20242007-03-08 00:43:48 +000060
61 def testReadinto(self):
62 # verify readinto
63 self.f.write(bytes([1, 2]))
64 self.f.close()
Guido van Rossuma45ea582007-05-15 21:25:12 +000065 a = array('b', b'x'*10)
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +000066 self.f = _FileIO(TESTFN, 'r')
Guido van Rossuma9e20242007-03-08 00:43:48 +000067 n = self.f.readinto(a)
Ezio Melottib3aedd42010-11-20 19:04:17 +000068 self.assertEqual(array('b', [1, 2]), a[:n])
Guido van Rossuma9e20242007-03-08 00:43:48 +000069
Benjamin Petersonbf5ff762009-12-13 19:25:34 +000070 def test_none_args(self):
71 self.f.write(b"hi\nbye\nabc")
72 self.f.close()
73 self.f = _FileIO(TESTFN, 'r')
74 self.assertEqual(self.f.read(None), b"hi\nbye\nabc")
75 self.f.seek(0)
76 self.assertEqual(self.f.readline(None), b"hi\n")
77 self.assertEqual(self.f.readlines(None), [b"bye\n", b"abc"])
78
Benjamin Peterson255058f2010-01-27 01:47:14 +000079 def test_reject(self):
80 self.assertRaises(TypeError, self.f.write, "Hello!")
81
Guido van Rossuma9e20242007-03-08 00:43:48 +000082 def testRepr(self):
Ezio Melottib3aedd42010-11-20 19:04:17 +000083 self.assertEqual(repr(self.f), "<_io.FileIO name=%r mode=%r>"
Antoine Pitrou716c4442009-05-23 19:04:03 +000084 % (self.f.name, self.f.mode))
85 del self.f.name
Ezio Melottib3aedd42010-11-20 19:04:17 +000086 self.assertEqual(repr(self.f), "<_io.FileIO fd=%r mode=%r>"
Antoine Pitrou716c4442009-05-23 19:04:03 +000087 % (self.f.fileno(), self.f.mode))
88 self.f.close()
Ezio Melottib3aedd42010-11-20 19:04:17 +000089 self.assertEqual(repr(self.f), "<_io.FileIO [closed]>")
Guido van Rossuma9e20242007-03-08 00:43:48 +000090
91 def testErrors(self):
92 f = self.f
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000093 self.assertTrue(not f.isatty())
94 self.assertTrue(not f.closed)
Ezio Melottib3aedd42010-11-20 19:04:17 +000095 #self.assertEqual(f.name, TESTFN)
Guido van Rossum31c3a572007-04-12 14:51:49 +000096 self.assertRaises(ValueError, f.read, 10) # Open for reading
Guido van Rossuma9e20242007-03-08 00:43:48 +000097 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000098 self.assertTrue(f.closed)
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +000099 f = _FileIO(TESTFN, 'r')
Guido van Rossum31c3a572007-04-12 14:51:49 +0000100 self.assertRaises(TypeError, f.readinto, "")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000101 self.assertTrue(not f.closed)
Guido van Rossum31c3a572007-04-12 14:51:49 +0000102 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000103 self.assertTrue(f.closed)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000104
105 def testMethods(self):
106 methods = ['fileno', 'isatty', 'read', 'readinto',
107 'seek', 'tell', 'truncate', 'write', 'seekable',
108 'readable', 'writable']
Guido van Rossuma9e20242007-03-08 00:43:48 +0000109
Guido van Rossum31c3a572007-04-12 14:51:49 +0000110 self.f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000111 self.assertTrue(self.f.closed)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000112
113 for methodname in methods:
114 method = getattr(self.f, methodname)
115 # should raise on closed file
116 self.assertRaises(ValueError, method)
117
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000118 def testOpendir(self):
119 # Issue 3703: opening a directory should fill the errno
120 # Windows always returns "[Errno 13]: Permission denied
121 # Unix calls dircheck() and returns "[Errno 21]: Is a directory"
122 try:
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000123 _FileIO('.', 'r')
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000124 except IOError as e:
125 self.assertNotEqual(e.errno, 0)
Benjamin Peterson1efc23c2008-12-29 18:02:28 +0000126 self.assertEqual(e.filename, ".")
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000127 else:
128 self.fail("Should have raised IOError")
129
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000130 #A set of functions testing that we get expected behaviour if someone has
131 #manually closed the internal file descriptor. First, a decorator:
132 def ClosedFD(func):
133 @wraps(func)
134 def wrapper(self):
135 #forcibly close the fd before invoking the problem function
136 f = self.f
137 os.close(f.fileno())
138 try:
139 func(self, f)
140 finally:
141 try:
142 self.f.close()
143 except IOError:
144 pass
145 return wrapper
Antoine Pitrou0ae29cf2009-03-13 22:33:17 +0000146
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000147 def ClosedFDRaises(func):
148 @wraps(func)
149 def wrapper(self):
150 #forcibly close the fd before invoking the problem function
151 f = self.f
152 os.close(f.fileno())
153 try:
154 func(self, f)
155 except IOError as e:
156 self.assertEqual(e.errno, errno.EBADF)
157 else:
158 self.fail("Should have raised IOError")
159 finally:
160 try:
161 self.f.close()
162 except IOError:
163 pass
164 return wrapper
165
166 @ClosedFDRaises
167 def testErrnoOnClose(self, f):
168 f.close()
169
170 @ClosedFDRaises
171 def testErrnoOnClosedWrite(self, f):
Benjamin Peterson255058f2010-01-27 01:47:14 +0000172 f.write(b'a')
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000173
174 @ClosedFDRaises
175 def testErrnoOnClosedSeek(self, f):
176 f.seek(0)
177
178 @ClosedFDRaises
179 def testErrnoOnClosedTell(self, f):
180 f.tell()
181
182 @ClosedFDRaises
183 def testErrnoOnClosedTruncate(self, f):
184 f.truncate(0)
185
186 @ClosedFD
187 def testErrnoOnClosedSeekable(self, f):
188 f.seekable()
189
190 @ClosedFD
191 def testErrnoOnClosedReadable(self, f):
192 f.readable()
193
194 @ClosedFD
195 def testErrnoOnClosedWritable(self, f):
196 f.writable()
197
198 @ClosedFD
199 def testErrnoOnClosedFileno(self, f):
200 f.fileno()
201
202 @ClosedFD
203 def testErrnoOnClosedIsatty(self, f):
204 self.assertEqual(f.isatty(), False)
205
206 def ReopenForRead(self):
207 try:
208 self.f.close()
209 except IOError:
210 pass
211 self.f = _FileIO(TESTFN, 'r')
212 os.close(self.f.fileno())
213 return self.f
214
215 @ClosedFDRaises
216 def testErrnoOnClosedRead(self, f):
217 f = self.ReopenForRead()
218 f.read(1)
219
220 @ClosedFDRaises
221 def testErrnoOnClosedReadall(self, f):
222 f = self.ReopenForRead()
223 f.readall()
224
225 @ClosedFDRaises
226 def testErrnoOnClosedReadinto(self, f):
227 f = self.ReopenForRead()
228 a = array('b', b'x'*10)
229 f.readinto(a)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000230
231class OtherFileTests(unittest.TestCase):
232
233 def testAbles(self):
234 try:
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000235 f = _FileIO(TESTFN, "w")
Ezio Melottib3aedd42010-11-20 19:04:17 +0000236 self.assertEqual(f.readable(), False)
237 self.assertEqual(f.writable(), True)
238 self.assertEqual(f.seekable(), True)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000239 f.close()
Guido van Rossumd8faa362007-04-27 19:54:29 +0000240
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000241 f = _FileIO(TESTFN, "r")
Ezio Melottib3aedd42010-11-20 19:04:17 +0000242 self.assertEqual(f.readable(), True)
243 self.assertEqual(f.writable(), False)
244 self.assertEqual(f.seekable(), True)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000245 f.close()
Guido van Rossumd8faa362007-04-27 19:54:29 +0000246
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000247 f = _FileIO(TESTFN, "a+")
Ezio Melottib3aedd42010-11-20 19:04:17 +0000248 self.assertEqual(f.readable(), True)
249 self.assertEqual(f.writable(), True)
250 self.assertEqual(f.seekable(), True)
251 self.assertEqual(f.isatty(), False)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000252 f.close()
Guido van Rossum682faf82007-04-12 14:56:58 +0000253
Thomas Hellerb0f48ab2007-07-12 11:29:02 +0000254 if sys.platform != "win32":
255 try:
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000256 f = _FileIO("/dev/tty", "a")
Thomas Hellerb0f48ab2007-07-12 11:29:02 +0000257 except EnvironmentError:
258 # When run in a cron job there just aren't any
259 # ttys, so skip the test. This also handles other
260 # OS'es that don't support /dev/tty.
261 pass
262 else:
Ezio Melottib3aedd42010-11-20 19:04:17 +0000263 self.assertEqual(f.readable(), False)
264 self.assertEqual(f.writable(), True)
Hye-Shik Changcaf871a2007-08-13 13:21:33 +0000265 if sys.platform != "darwin" and \
Antoine Pitroud7b30462009-05-23 16:34:50 +0000266 'bsd' not in sys.platform and \
Benjamin Peterson94fe10f2008-07-17 23:27:26 +0000267 not sys.platform.startswith('sunos'):
Hye-Shik Changcaf871a2007-08-13 13:21:33 +0000268 # Somehow /dev/tty appears seekable on some BSDs
Ezio Melottib3aedd42010-11-20 19:04:17 +0000269 self.assertEqual(f.seekable(), False)
270 self.assertEqual(f.isatty(), True)
Thomas Hellerb0f48ab2007-07-12 11:29:02 +0000271 f.close()
Guido van Rossuma9e20242007-03-08 00:43:48 +0000272 finally:
273 os.unlink(TESTFN)
274
275 def testModeStrings(self):
276 # check invalid mode strings
Benjamin Peterson44309e62008-11-22 00:41:45 +0000277 for mode in ("", "aU", "wU+", "rw", "rt"):
Guido van Rossuma9e20242007-03-08 00:43:48 +0000278 try:
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000279 f = _FileIO(TESTFN, mode)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000280 except ValueError:
281 pass
282 else:
283 f.close()
284 self.fail('%r is an invalid file mode' % mode)
285
Guido van Rossuma9e20242007-03-08 00:43:48 +0000286 def testUnicodeOpen(self):
287 # verify repr works for unicode too
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000288 f = _FileIO(str(TESTFN), "w")
Guido van Rossuma9e20242007-03-08 00:43:48 +0000289 f.close()
290 os.unlink(TESTFN)
291
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000292 def testBytesOpen(self):
293 # Opening a bytes filename
294 try:
295 fn = TESTFN.encode("ascii")
296 except UnicodeEncodeError:
297 # Skip test
298 return
299 f = _FileIO(fn, "w")
300 try:
301 f.write(b"abc")
302 f.close()
303 with open(TESTFN, "rb") as f:
Ezio Melottib3aedd42010-11-20 19:04:17 +0000304 self.assertEqual(f.read(), b"abc")
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000305 finally:
306 os.unlink(TESTFN)
307
Antoine Pitrou13348842012-01-29 18:36:34 +0100308 def testConstructorHandlesNULChars(self):
309 fn_with_NUL = 'foo\0bar'
310 self.assertRaises(TypeError, _FileIO, fn_with_NUL, 'w')
311 self.assertRaises(TypeError, _FileIO, bytes(fn_with_NUL, 'ascii'), 'w')
312
Benjamin Peterson806d4022009-01-19 15:11:51 +0000313 def testInvalidFd(self):
Antoine Pitrou7fb111b2009-03-04 11:14:01 +0000314 self.assertRaises(ValueError, _FileIO, -10)
315 self.assertRaises(OSError, _FileIO, make_bad_fd())
Antoine Pitrou00492492010-09-04 20:53:29 +0000316 if sys.platform == 'win32':
317 import msvcrt
318 self.assertRaises(IOError, msvcrt.get_osfhandle, make_bad_fd())
Benjamin Peterson806d4022009-01-19 15:11:51 +0000319
Guido van Rossuma9e20242007-03-08 00:43:48 +0000320 def testBadModeArgument(self):
321 # verify that we get a sensible error message for bad mode argument
322 bad_mode = "qwerty"
323 try:
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000324 f = _FileIO(TESTFN, bad_mode)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000325 except ValueError as msg:
Guido van Rossum360e4b82007-05-14 22:51:27 +0000326 if msg.args[0] != 0:
Guido van Rossuma9e20242007-03-08 00:43:48 +0000327 s = str(msg)
Ezio Melotti7fb4da72010-03-18 12:29:13 +0000328 if TESTFN in s or bad_mode not in s:
Guido van Rossuma9e20242007-03-08 00:43:48 +0000329 self.fail("bad error message for invalid mode: %s" % s)
Georg Brandl50da60c2008-01-06 21:38:54 +0000330 # if msg.args[0] == 0, we're probably on Windows where there may be
Guido van Rossuma9e20242007-03-08 00:43:48 +0000331 # no obvious way to discover why open() failed.
332 else:
333 f.close()
334 self.fail("no error for invalid mode: %s" % bad_mode)
335
Antoine Pitrou905a2ff2010-01-31 22:47:27 +0000336 def testTruncate(self):
337 f = _FileIO(TESTFN, 'w')
338 f.write(bytes(bytearray(range(10))))
339 self.assertEqual(f.tell(), 10)
340 f.truncate(5)
341 self.assertEqual(f.tell(), 10)
342 self.assertEqual(f.seek(0, os.SEEK_END), 5)
343 f.truncate(15)
344 self.assertEqual(f.tell(), 5)
345 self.assertEqual(f.seek(0, os.SEEK_END), 15)
Antoine Pitrou8d2b51b2010-10-30 16:19:14 +0000346 f.close()
Antoine Pitrou905a2ff2010-01-31 22:47:27 +0000347
Guido van Rossuma9e20242007-03-08 00:43:48 +0000348 def testTruncateOnWindows(self):
349 def bug801631():
350 # SF bug <http://www.python.org/sf/801631>
351 # "file.truncate fault on windows"
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000352 f = _FileIO(TESTFN, 'w')
Guido van Rossuma9e20242007-03-08 00:43:48 +0000353 f.write(bytes(range(11)))
354 f.close()
355
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000356 f = _FileIO(TESTFN,'r+')
Guido van Rossuma9e20242007-03-08 00:43:48 +0000357 data = f.read(5)
358 if data != bytes(range(5)):
359 self.fail("Read on file opened for update failed %r" % data)
360 if f.tell() != 5:
361 self.fail("File pos after read wrong %d" % f.tell())
362
363 f.truncate()
364 if f.tell() != 5:
365 self.fail("File pos after ftruncate wrong %d" % f.tell())
366
367 f.close()
368 size = os.path.getsize(TESTFN)
369 if size != 5:
370 self.fail("File size after ftruncate wrong %d" % size)
371
372 try:
373 bug801631()
374 finally:
375 os.unlink(TESTFN)
376
Walter Dörwald3a77c7a2007-06-06 16:31:14 +0000377 def testAppend(self):
378 try:
379 f = open(TESTFN, 'wb')
380 f.write(b'spam')
381 f.close()
382 f = open(TESTFN, 'ab')
383 f.write(b'eggs')
384 f.close()
385 f = open(TESTFN, 'rb')
386 d = f.read()
387 f.close()
388 self.assertEqual(d, b'spameggs')
389 finally:
390 try:
391 os.unlink(TESTFN)
392 except:
393 pass
394
Neal Norwitz6e0e0e62008-08-24 22:07:28 +0000395 def testInvalidInit(self):
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000396 self.assertRaises(TypeError, _FileIO, "1", 0, 0)
Neal Norwitz6e0e0e62008-08-24 22:07:28 +0000397
Benjamin Peterson65676e42008-11-05 21:42:45 +0000398 def testWarnings(self):
Florent Xiclunab14930c2010-03-13 15:26:44 +0000399 with check_warnings(quiet=True) as w:
Benjamin Peterson65676e42008-11-05 21:42:45 +0000400 self.assertEqual(w.warnings, [])
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000401 self.assertRaises(TypeError, _FileIO, [])
Benjamin Peterson65676e42008-11-05 21:42:45 +0000402 self.assertEqual(w.warnings, [])
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000403 self.assertRaises(ValueError, _FileIO, "/some/invalid/name", "rt")
Benjamin Peterson65676e42008-11-05 21:42:45 +0000404 self.assertEqual(w.warnings, [])
405
Neal Norwitz6e0e0e62008-08-24 22:07:28 +0000406
Guido van Rossuma9e20242007-03-08 00:43:48 +0000407def test_main():
408 # Historically, these tests have been sloppy about removing TESTFN.
409 # So get rid of it no matter what.
410 try:
411 run_unittest(AutoFileTests, OtherFileTests)
412 finally:
413 if os.path.exists(TESTFN):
414 os.unlink(TESTFN)
415
416if __name__ == '__main__':
417 test_main()