blob: e9281e33fa288174bfe9c220f60ed74e4d602ffd [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
Benjamin Peterson7522c742009-01-19 21:00:09 +000011from test.support import (TESTFN, findfile, check_warnings, run_unittest,
12 make_bad_fd)
Raymond Hettinger53dbe392008-02-12 20:03:09 +000013from collections import UserList
Guido van Rossuma9e20242007-03-08 00:43:48 +000014
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +000015from _io import FileIO as _FileIO
Guido van Rossuma9e20242007-03-08 00:43:48 +000016
17class AutoFileTests(unittest.TestCase):
18 # file tests for which a test file is automatically set up
19
20 def setUp(self):
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +000021 self.f = _FileIO(TESTFN, 'w')
Guido van Rossuma9e20242007-03-08 00:43:48 +000022
23 def tearDown(self):
24 if self.f:
25 self.f.close()
26 os.remove(TESTFN)
27
28 def testWeakRefs(self):
29 # verify weak references
30 p = proxy(self.f)
31 p.write(bytes(range(10)))
32 self.assertEquals(self.f.tell(), p.tell())
33 self.f.close()
34 self.f = None
35 self.assertRaises(ReferenceError, getattr, p, 'tell')
36
37 def testSeekTell(self):
38 self.f.write(bytes(range(20)))
39 self.assertEquals(self.f.tell(), 20)
40 self.f.seek(0)
41 self.assertEquals(self.f.tell(), 0)
42 self.f.seek(10)
43 self.assertEquals(self.f.tell(), 10)
44 self.f.seek(5, 1)
45 self.assertEquals(self.f.tell(), 15)
46 self.f.seek(-5, 1)
47 self.assertEquals(self.f.tell(), 10)
48 self.f.seek(-5, 2)
49 self.assertEquals(self.f.tell(), 15)
50
51 def testAttributes(self):
52 # verify expected attributes exist
53 f = self.f
Guido van Rossuma9e20242007-03-08 00:43:48 +000054
Benjamin Peterson44309e62008-11-22 00:41:45 +000055 self.assertEquals(f.mode, "wb")
Guido van Rossum31c3a572007-04-12 14:51:49 +000056 self.assertEquals(f.closed, False)
57
58 # verify the attributes are readonly
59 for attr in 'mode', 'closed':
60 self.assertRaises((AttributeError, TypeError),
61 setattr, f, attr, 'oops')
Guido van Rossuma9e20242007-03-08 00:43:48 +000062
63 def testReadinto(self):
64 # verify readinto
65 self.f.write(bytes([1, 2]))
66 self.f.close()
Guido van Rossuma45ea582007-05-15 21:25:12 +000067 a = array('b', b'x'*10)
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +000068 self.f = _FileIO(TESTFN, 'r')
Guido van Rossuma9e20242007-03-08 00:43:48 +000069 n = self.f.readinto(a)
70 self.assertEquals(array('b', [1, 2]), a[:n])
71
Benjamin Peterson6b59f772009-12-13 19:30:15 +000072 def test_none_args(self):
73 self.f.write(b"hi\nbye\nabc")
74 self.f.close()
75 self.f = _FileIO(TESTFN, 'r')
76 self.assertEqual(self.f.read(None), b"hi\nbye\nabc")
77 self.f.seek(0)
78 self.assertEqual(self.f.readline(None), b"hi\n")
79 self.assertEqual(self.f.readlines(None), [b"bye\n", b"abc"])
80
Benjamin Peterson371c3072010-01-27 01:56:08 +000081 def test_reject(self):
82 self.assertRaises(TypeError, self.f.write, "Hello!")
83
Guido van Rossuma9e20242007-03-08 00:43:48 +000084 def testRepr(self):
Antoine Pitrou716c4442009-05-23 19:04:03 +000085 self.assertEquals(repr(self.f), "<_io.FileIO name=%r mode=%r>"
86 % (self.f.name, self.f.mode))
87 del self.f.name
88 self.assertEquals(repr(self.f), "<_io.FileIO fd=%r mode=%r>"
89 % (self.f.fileno(), self.f.mode))
90 self.f.close()
91 self.assertEquals(repr(self.f), "<_io.FileIO [closed]>")
Guido van Rossuma9e20242007-03-08 00:43:48 +000092
93 def testErrors(self):
94 f = self.f
Georg Brandlab91fde2009-08-13 08:51:18 +000095 self.assertTrue(not f.isatty())
96 self.assertTrue(not f.closed)
Guido van Rossuma9e20242007-03-08 00:43:48 +000097 #self.assertEquals(f.name, TESTFN)
Guido van Rossum31c3a572007-04-12 14:51:49 +000098 self.assertRaises(ValueError, f.read, 10) # Open for reading
Guido van Rossuma9e20242007-03-08 00:43:48 +000099 f.close()
Georg Brandlab91fde2009-08-13 08:51:18 +0000100 self.assertTrue(f.closed)
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000101 f = _FileIO(TESTFN, 'r')
Guido van Rossum31c3a572007-04-12 14:51:49 +0000102 self.assertRaises(TypeError, f.readinto, "")
Georg Brandlab91fde2009-08-13 08:51:18 +0000103 self.assertTrue(not f.closed)
Guido van Rossum31c3a572007-04-12 14:51:49 +0000104 f.close()
Georg Brandlab91fde2009-08-13 08:51:18 +0000105 self.assertTrue(f.closed)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000106
107 def testMethods(self):
108 methods = ['fileno', 'isatty', 'read', 'readinto',
109 'seek', 'tell', 'truncate', 'write', 'seekable',
110 'readable', 'writable']
111 if sys.platform.startswith('atheos'):
112 methods.remove('truncate')
113
Guido van Rossum31c3a572007-04-12 14:51:49 +0000114 self.f.close()
Georg Brandlab91fde2009-08-13 08:51:18 +0000115 self.assertTrue(self.f.closed)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000116
117 for methodname in methods:
118 method = getattr(self.f, methodname)
119 # should raise on closed file
120 self.assertRaises(ValueError, method)
121
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000122 def testOpendir(self):
123 # Issue 3703: opening a directory should fill the errno
124 # Windows always returns "[Errno 13]: Permission denied
125 # Unix calls dircheck() and returns "[Errno 21]: Is a directory"
126 try:
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000127 _FileIO('.', 'r')
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000128 except IOError as e:
129 self.assertNotEqual(e.errno, 0)
Benjamin Peterson1efc23c2008-12-29 18:02:28 +0000130 self.assertEqual(e.filename, ".")
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000131 else:
132 self.fail("Should have raised IOError")
133
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000134 #A set of functions testing that we get expected behaviour if someone has
135 #manually closed the internal file descriptor. First, a decorator:
136 def ClosedFD(func):
137 @wraps(func)
138 def wrapper(self):
139 #forcibly close the fd before invoking the problem function
140 f = self.f
141 os.close(f.fileno())
142 try:
143 func(self, f)
144 finally:
145 try:
146 self.f.close()
147 except IOError:
148 pass
149 return wrapper
Antoine Pitrou0ae29cf2009-03-13 22:33:17 +0000150
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000151 def ClosedFDRaises(func):
152 @wraps(func)
153 def wrapper(self):
154 #forcibly close the fd before invoking the problem function
155 f = self.f
156 os.close(f.fileno())
157 try:
158 func(self, f)
159 except IOError as e:
160 self.assertEqual(e.errno, errno.EBADF)
161 else:
162 self.fail("Should have raised IOError")
163 finally:
164 try:
165 self.f.close()
166 except IOError:
167 pass
168 return wrapper
169
170 @ClosedFDRaises
171 def testErrnoOnClose(self, f):
172 f.close()
173
174 @ClosedFDRaises
175 def testErrnoOnClosedWrite(self, f):
Benjamin Peterson371c3072010-01-27 01:56:08 +0000176 f.write(b'a')
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000177
178 @ClosedFDRaises
179 def testErrnoOnClosedSeek(self, f):
180 f.seek(0)
181
182 @ClosedFDRaises
183 def testErrnoOnClosedTell(self, f):
184 f.tell()
185
186 @ClosedFDRaises
187 def testErrnoOnClosedTruncate(self, f):
188 f.truncate(0)
189
190 @ClosedFD
191 def testErrnoOnClosedSeekable(self, f):
192 f.seekable()
193
194 @ClosedFD
195 def testErrnoOnClosedReadable(self, f):
196 f.readable()
197
198 @ClosedFD
199 def testErrnoOnClosedWritable(self, f):
200 f.writable()
201
202 @ClosedFD
203 def testErrnoOnClosedFileno(self, f):
204 f.fileno()
205
206 @ClosedFD
207 def testErrnoOnClosedIsatty(self, f):
208 self.assertEqual(f.isatty(), False)
209
210 def ReopenForRead(self):
211 try:
212 self.f.close()
213 except IOError:
214 pass
215 self.f = _FileIO(TESTFN, 'r')
216 os.close(self.f.fileno())
217 return self.f
218
219 @ClosedFDRaises
220 def testErrnoOnClosedRead(self, f):
221 f = self.ReopenForRead()
222 f.read(1)
223
224 @ClosedFDRaises
225 def testErrnoOnClosedReadall(self, f):
226 f = self.ReopenForRead()
227 f.readall()
228
229 @ClosedFDRaises
230 def testErrnoOnClosedReadinto(self, f):
231 f = self.ReopenForRead()
232 a = array('b', b'x'*10)
233 f.readinto(a)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000234
235class OtherFileTests(unittest.TestCase):
236
237 def testAbles(self):
238 try:
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000239 f = _FileIO(TESTFN, "w")
Guido van Rossuma9e20242007-03-08 00:43:48 +0000240 self.assertEquals(f.readable(), False)
241 self.assertEquals(f.writable(), True)
242 self.assertEquals(f.seekable(), True)
243 f.close()
Guido van Rossumd8faa362007-04-27 19:54:29 +0000244
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000245 f = _FileIO(TESTFN, "r")
Guido van Rossuma9e20242007-03-08 00:43:48 +0000246 self.assertEquals(f.readable(), True)
247 self.assertEquals(f.writable(), False)
248 self.assertEquals(f.seekable(), True)
249 f.close()
Guido van Rossumd8faa362007-04-27 19:54:29 +0000250
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000251 f = _FileIO(TESTFN, "a+")
Guido van Rossuma9e20242007-03-08 00:43:48 +0000252 self.assertEquals(f.readable(), True)
253 self.assertEquals(f.writable(), True)
254 self.assertEquals(f.seekable(), True)
255 self.assertEquals(f.isatty(), False)
256 f.close()
Guido van Rossum682faf82007-04-12 14:56:58 +0000257
Thomas Hellerb0f48ab2007-07-12 11:29:02 +0000258 if sys.platform != "win32":
259 try:
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000260 f = _FileIO("/dev/tty", "a")
Thomas Hellerb0f48ab2007-07-12 11:29:02 +0000261 except EnvironmentError:
262 # When run in a cron job there just aren't any
263 # ttys, so skip the test. This also handles other
264 # OS'es that don't support /dev/tty.
265 pass
266 else:
Thomas Hellerb0f48ab2007-07-12 11:29:02 +0000267 self.assertEquals(f.readable(), False)
268 self.assertEquals(f.writable(), True)
Hye-Shik Changcaf871a2007-08-13 13:21:33 +0000269 if sys.platform != "darwin" and \
Antoine Pitroud7b30462009-05-23 16:34:50 +0000270 'bsd' not in sys.platform and \
Benjamin Peterson94fe10f2008-07-17 23:27:26 +0000271 not sys.platform.startswith('sunos'):
Hye-Shik Changcaf871a2007-08-13 13:21:33 +0000272 # Somehow /dev/tty appears seekable on some BSDs
Thomas Hellerb0f48ab2007-07-12 11:29:02 +0000273 self.assertEquals(f.seekable(), False)
274 self.assertEquals(f.isatty(), True)
275 f.close()
Guido van Rossuma9e20242007-03-08 00:43:48 +0000276 finally:
277 os.unlink(TESTFN)
278
279 def testModeStrings(self):
280 # check invalid mode strings
Benjamin Peterson44309e62008-11-22 00:41:45 +0000281 for mode in ("", "aU", "wU+", "rw", "rt"):
Guido van Rossuma9e20242007-03-08 00:43:48 +0000282 try:
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000283 f = _FileIO(TESTFN, mode)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000284 except ValueError:
285 pass
286 else:
287 f.close()
288 self.fail('%r is an invalid file mode' % mode)
289
Guido van Rossuma9e20242007-03-08 00:43:48 +0000290 def testUnicodeOpen(self):
291 # verify repr works for unicode too
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000292 f = _FileIO(str(TESTFN), "w")
Guido van Rossuma9e20242007-03-08 00:43:48 +0000293 f.close()
294 os.unlink(TESTFN)
295
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000296 def testBytesOpen(self):
297 # Opening a bytes filename
298 try:
299 fn = TESTFN.encode("ascii")
300 except UnicodeEncodeError:
301 # Skip test
302 return
303 f = _FileIO(fn, "w")
304 try:
305 f.write(b"abc")
306 f.close()
307 with open(TESTFN, "rb") as f:
308 self.assertEquals(f.read(), b"abc")
309 finally:
310 os.unlink(TESTFN)
311
Benjamin Peterson806d4022009-01-19 15:11:51 +0000312 def testInvalidFd(self):
Antoine Pitrou7fb111b2009-03-04 11:14:01 +0000313 self.assertRaises(ValueError, _FileIO, -10)
314 self.assertRaises(OSError, _FileIO, make_bad_fd())
Antoine Pitrou0bb502d2010-09-04 21:02:41 +0000315 if sys.platform == 'win32':
316 import msvcrt
317 self.assertRaises(IOError, msvcrt.get_osfhandle, make_bad_fd())
Benjamin Peterson806d4022009-01-19 15:11:51 +0000318
Guido van Rossuma9e20242007-03-08 00:43:48 +0000319 def testBadModeArgument(self):
320 # verify that we get a sensible error message for bad mode argument
321 bad_mode = "qwerty"
322 try:
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000323 f = _FileIO(TESTFN, bad_mode)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000324 except ValueError as msg:
Guido van Rossum360e4b82007-05-14 22:51:27 +0000325 if msg.args[0] != 0:
Guido van Rossuma9e20242007-03-08 00:43:48 +0000326 s = str(msg)
327 if s.find(TESTFN) != -1 or s.find(bad_mode) == -1:
328 self.fail("bad error message for invalid mode: %s" % s)
Georg Brandl50da60c2008-01-06 21:38:54 +0000329 # if msg.args[0] == 0, we're probably on Windows where there may be
Guido van Rossuma9e20242007-03-08 00:43:48 +0000330 # no obvious way to discover why open() failed.
331 else:
332 f.close()
333 self.fail("no error for invalid mode: %s" % bad_mode)
334
Antoine Pitrou66f9fea2010-01-31 23:20:26 +0000335 def testTruncate(self):
336 f = _FileIO(TESTFN, 'w')
337 f.write(bytes(bytearray(range(10))))
338 self.assertEqual(f.tell(), 10)
339 f.truncate(5)
340 self.assertEqual(f.tell(), 10)
341 self.assertEqual(f.seek(0, os.SEEK_END), 5)
342 f.truncate(15)
343 self.assertEqual(f.tell(), 5)
344 self.assertEqual(f.seek(0, os.SEEK_END), 15)
Antoine Pitrou735e3b12010-10-31 13:05:21 +0000345 f.close()
Antoine Pitrou66f9fea2010-01-31 23:20:26 +0000346
Guido van Rossuma9e20242007-03-08 00:43:48 +0000347 def testTruncateOnWindows(self):
348 def bug801631():
349 # SF bug <http://www.python.org/sf/801631>
350 # "file.truncate fault on windows"
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000351 f = _FileIO(TESTFN, 'w')
Guido van Rossuma9e20242007-03-08 00:43:48 +0000352 f.write(bytes(range(11)))
353 f.close()
354
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000355 f = _FileIO(TESTFN,'r+')
Guido van Rossuma9e20242007-03-08 00:43:48 +0000356 data = f.read(5)
357 if data != bytes(range(5)):
358 self.fail("Read on file opened for update failed %r" % data)
359 if f.tell() != 5:
360 self.fail("File pos after read wrong %d" % f.tell())
361
362 f.truncate()
363 if f.tell() != 5:
364 self.fail("File pos after ftruncate wrong %d" % f.tell())
365
366 f.close()
367 size = os.path.getsize(TESTFN)
368 if size != 5:
369 self.fail("File size after ftruncate wrong %d" % size)
370
371 try:
372 bug801631()
373 finally:
374 os.unlink(TESTFN)
375
Walter Dörwald3a77c7a2007-06-06 16:31:14 +0000376 def testAppend(self):
377 try:
378 f = open(TESTFN, 'wb')
379 f.write(b'spam')
380 f.close()
381 f = open(TESTFN, 'ab')
382 f.write(b'eggs')
383 f.close()
384 f = open(TESTFN, 'rb')
385 d = f.read()
386 f.close()
387 self.assertEqual(d, b'spameggs')
388 finally:
389 try:
390 os.unlink(TESTFN)
391 except:
392 pass
393
Neal Norwitz6e0e0e62008-08-24 22:07:28 +0000394 def testInvalidInit(self):
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000395 self.assertRaises(TypeError, _FileIO, "1", 0, 0)
Neal Norwitz6e0e0e62008-08-24 22:07:28 +0000396
Benjamin Peterson65676e42008-11-05 21:42:45 +0000397 def testWarnings(self):
398 with check_warnings() as w:
399 self.assertEqual(w.warnings, [])
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000400 self.assertRaises(TypeError, _FileIO, [])
Benjamin Peterson65676e42008-11-05 21:42:45 +0000401 self.assertEqual(w.warnings, [])
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000402 self.assertRaises(ValueError, _FileIO, "/some/invalid/name", "rt")
Benjamin Peterson65676e42008-11-05 21:42:45 +0000403 self.assertEqual(w.warnings, [])
404
Neal Norwitz6e0e0e62008-08-24 22:07:28 +0000405
Guido van Rossuma9e20242007-03-08 00:43:48 +0000406def test_main():
407 # Historically, these tests have been sloppy about removing TESTFN.
408 # So get rid of it no matter what.
409 try:
410 run_unittest(AutoFileTests, OtherFileTests)
411 finally:
412 if os.path.exists(TESTFN):
413 os.unlink(TESTFN)
414
415if __name__ == '__main__':
416 test_main()