blob: 4ce3746b36d6a14c0b481ff780d18811829ae284 [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
72 def testRepr(self):
Antoine Pitrou716c4442009-05-23 19:04:03 +000073 self.assertEquals(repr(self.f), "<_io.FileIO name=%r mode=%r>"
74 % (self.f.name, self.f.mode))
75 del self.f.name
76 self.assertEquals(repr(self.f), "<_io.FileIO fd=%r mode=%r>"
77 % (self.f.fileno(), self.f.mode))
78 self.f.close()
79 self.assertEquals(repr(self.f), "<_io.FileIO [closed]>")
Guido van Rossuma9e20242007-03-08 00:43:48 +000080
81 def testErrors(self):
82 f = self.f
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000083 self.assertTrue(not f.isatty())
84 self.assertTrue(not f.closed)
Guido van Rossuma9e20242007-03-08 00:43:48 +000085 #self.assertEquals(f.name, TESTFN)
Guido van Rossum31c3a572007-04-12 14:51:49 +000086 self.assertRaises(ValueError, f.read, 10) # Open for reading
Guido van Rossuma9e20242007-03-08 00:43:48 +000087 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000088 self.assertTrue(f.closed)
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +000089 f = _FileIO(TESTFN, 'r')
Guido van Rossum31c3a572007-04-12 14:51:49 +000090 self.assertRaises(TypeError, f.readinto, "")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000091 self.assertTrue(not f.closed)
Guido van Rossum31c3a572007-04-12 14:51:49 +000092 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000093 self.assertTrue(f.closed)
Guido van Rossuma9e20242007-03-08 00:43:48 +000094
95 def testMethods(self):
96 methods = ['fileno', 'isatty', 'read', 'readinto',
97 'seek', 'tell', 'truncate', 'write', 'seekable',
98 'readable', 'writable']
Guido van Rossuma9e20242007-03-08 00:43:48 +000099
Guido van Rossum31c3a572007-04-12 14:51:49 +0000100 self.f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000101 self.assertTrue(self.f.closed)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000102
103 for methodname in methods:
104 method = getattr(self.f, methodname)
105 # should raise on closed file
106 self.assertRaises(ValueError, method)
107
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000108 def testOpendir(self):
109 # Issue 3703: opening a directory should fill the errno
110 # Windows always returns "[Errno 13]: Permission denied
111 # Unix calls dircheck() and returns "[Errno 21]: Is a directory"
112 try:
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000113 _FileIO('.', 'r')
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000114 except IOError as e:
115 self.assertNotEqual(e.errno, 0)
Benjamin Peterson1efc23c2008-12-29 18:02:28 +0000116 self.assertEqual(e.filename, ".")
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000117 else:
118 self.fail("Should have raised IOError")
119
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000120 #A set of functions testing that we get expected behaviour if someone has
121 #manually closed the internal file descriptor. First, a decorator:
122 def ClosedFD(func):
123 @wraps(func)
124 def wrapper(self):
125 #forcibly close the fd before invoking the problem function
126 f = self.f
127 os.close(f.fileno())
128 try:
129 func(self, f)
130 finally:
131 try:
132 self.f.close()
133 except IOError:
134 pass
135 return wrapper
Antoine Pitrou0ae29cf2009-03-13 22:33:17 +0000136
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000137 def ClosedFDRaises(func):
138 @wraps(func)
139 def wrapper(self):
140 #forcibly close the fd before invoking the problem function
141 f = self.f
142 os.close(f.fileno())
143 try:
144 func(self, f)
145 except IOError as e:
146 self.assertEqual(e.errno, errno.EBADF)
147 else:
148 self.fail("Should have raised IOError")
149 finally:
150 try:
151 self.f.close()
152 except IOError:
153 pass
154 return wrapper
155
156 @ClosedFDRaises
157 def testErrnoOnClose(self, f):
158 f.close()
159
160 @ClosedFDRaises
161 def testErrnoOnClosedWrite(self, f):
162 f.write('a')
163
164 @ClosedFDRaises
165 def testErrnoOnClosedSeek(self, f):
166 f.seek(0)
167
168 @ClosedFDRaises
169 def testErrnoOnClosedTell(self, f):
170 f.tell()
171
172 @ClosedFDRaises
173 def testErrnoOnClosedTruncate(self, f):
174 f.truncate(0)
175
176 @ClosedFD
177 def testErrnoOnClosedSeekable(self, f):
178 f.seekable()
179
180 @ClosedFD
181 def testErrnoOnClosedReadable(self, f):
182 f.readable()
183
184 @ClosedFD
185 def testErrnoOnClosedWritable(self, f):
186 f.writable()
187
188 @ClosedFD
189 def testErrnoOnClosedFileno(self, f):
190 f.fileno()
191
192 @ClosedFD
193 def testErrnoOnClosedIsatty(self, f):
194 self.assertEqual(f.isatty(), False)
195
196 def ReopenForRead(self):
197 try:
198 self.f.close()
199 except IOError:
200 pass
201 self.f = _FileIO(TESTFN, 'r')
202 os.close(self.f.fileno())
203 return self.f
204
205 @ClosedFDRaises
206 def testErrnoOnClosedRead(self, f):
207 f = self.ReopenForRead()
208 f.read(1)
209
210 @ClosedFDRaises
211 def testErrnoOnClosedReadall(self, f):
212 f = self.ReopenForRead()
213 f.readall()
214
215 @ClosedFDRaises
216 def testErrnoOnClosedReadinto(self, f):
217 f = self.ReopenForRead()
218 a = array('b', b'x'*10)
219 f.readinto(a)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000220
221class OtherFileTests(unittest.TestCase):
222
223 def testAbles(self):
224 try:
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000225 f = _FileIO(TESTFN, "w")
Guido van Rossuma9e20242007-03-08 00:43:48 +0000226 self.assertEquals(f.readable(), False)
227 self.assertEquals(f.writable(), True)
228 self.assertEquals(f.seekable(), True)
229 f.close()
Guido van Rossumd8faa362007-04-27 19:54:29 +0000230
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000231 f = _FileIO(TESTFN, "r")
Guido van Rossuma9e20242007-03-08 00:43:48 +0000232 self.assertEquals(f.readable(), True)
233 self.assertEquals(f.writable(), False)
234 self.assertEquals(f.seekable(), True)
235 f.close()
Guido van Rossumd8faa362007-04-27 19:54:29 +0000236
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000237 f = _FileIO(TESTFN, "a+")
Guido van Rossuma9e20242007-03-08 00:43:48 +0000238 self.assertEquals(f.readable(), True)
239 self.assertEquals(f.writable(), True)
240 self.assertEquals(f.seekable(), True)
241 self.assertEquals(f.isatty(), False)
242 f.close()
Guido van Rossum682faf82007-04-12 14:56:58 +0000243
Thomas Hellerb0f48ab2007-07-12 11:29:02 +0000244 if sys.platform != "win32":
245 try:
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000246 f = _FileIO("/dev/tty", "a")
Thomas Hellerb0f48ab2007-07-12 11:29:02 +0000247 except EnvironmentError:
248 # When run in a cron job there just aren't any
249 # ttys, so skip the test. This also handles other
250 # OS'es that don't support /dev/tty.
251 pass
252 else:
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000253 f = _FileIO("/dev/tty", "a")
Thomas Hellerb0f48ab2007-07-12 11:29:02 +0000254 self.assertEquals(f.readable(), False)
255 self.assertEquals(f.writable(), True)
Hye-Shik Changcaf871a2007-08-13 13:21:33 +0000256 if sys.platform != "darwin" and \
Antoine Pitroud7b30462009-05-23 16:34:50 +0000257 'bsd' not in sys.platform and \
Benjamin Peterson94fe10f2008-07-17 23:27:26 +0000258 not sys.platform.startswith('sunos'):
Hye-Shik Changcaf871a2007-08-13 13:21:33 +0000259 # Somehow /dev/tty appears seekable on some BSDs
Thomas Hellerb0f48ab2007-07-12 11:29:02 +0000260 self.assertEquals(f.seekable(), False)
261 self.assertEquals(f.isatty(), True)
262 f.close()
Guido van Rossuma9e20242007-03-08 00:43:48 +0000263 finally:
264 os.unlink(TESTFN)
265
266 def testModeStrings(self):
267 # check invalid mode strings
Benjamin Peterson44309e62008-11-22 00:41:45 +0000268 for mode in ("", "aU", "wU+", "rw", "rt"):
Guido van Rossuma9e20242007-03-08 00:43:48 +0000269 try:
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000270 f = _FileIO(TESTFN, mode)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000271 except ValueError:
272 pass
273 else:
274 f.close()
275 self.fail('%r is an invalid file mode' % mode)
276
Guido van Rossuma9e20242007-03-08 00:43:48 +0000277 def testUnicodeOpen(self):
278 # verify repr works for unicode too
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000279 f = _FileIO(str(TESTFN), "w")
Guido van Rossuma9e20242007-03-08 00:43:48 +0000280 f.close()
281 os.unlink(TESTFN)
282
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000283 def testBytesOpen(self):
284 # Opening a bytes filename
285 try:
286 fn = TESTFN.encode("ascii")
287 except UnicodeEncodeError:
288 # Skip test
289 return
290 f = _FileIO(fn, "w")
291 try:
292 f.write(b"abc")
293 f.close()
294 with open(TESTFN, "rb") as f:
295 self.assertEquals(f.read(), b"abc")
296 finally:
297 os.unlink(TESTFN)
298
Benjamin Peterson806d4022009-01-19 15:11:51 +0000299 def testInvalidFd(self):
Antoine Pitrou7fb111b2009-03-04 11:14:01 +0000300 self.assertRaises(ValueError, _FileIO, -10)
301 self.assertRaises(OSError, _FileIO, make_bad_fd())
Benjamin Peterson806d4022009-01-19 15:11:51 +0000302
Guido van Rossuma9e20242007-03-08 00:43:48 +0000303 def testBadModeArgument(self):
304 # verify that we get a sensible error message for bad mode argument
305 bad_mode = "qwerty"
306 try:
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000307 f = _FileIO(TESTFN, bad_mode)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000308 except ValueError as msg:
Guido van Rossum360e4b82007-05-14 22:51:27 +0000309 if msg.args[0] != 0:
Guido van Rossuma9e20242007-03-08 00:43:48 +0000310 s = str(msg)
311 if s.find(TESTFN) != -1 or s.find(bad_mode) == -1:
312 self.fail("bad error message for invalid mode: %s" % s)
Georg Brandl50da60c2008-01-06 21:38:54 +0000313 # if msg.args[0] == 0, we're probably on Windows where there may be
Guido van Rossuma9e20242007-03-08 00:43:48 +0000314 # no obvious way to discover why open() failed.
315 else:
316 f.close()
317 self.fail("no error for invalid mode: %s" % bad_mode)
318
319 def testTruncateOnWindows(self):
320 def bug801631():
321 # SF bug <http://www.python.org/sf/801631>
322 # "file.truncate fault on windows"
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000323 f = _FileIO(TESTFN, 'w')
Guido van Rossuma9e20242007-03-08 00:43:48 +0000324 f.write(bytes(range(11)))
325 f.close()
326
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000327 f = _FileIO(TESTFN,'r+')
Guido van Rossuma9e20242007-03-08 00:43:48 +0000328 data = f.read(5)
329 if data != bytes(range(5)):
330 self.fail("Read on file opened for update failed %r" % data)
331 if f.tell() != 5:
332 self.fail("File pos after read wrong %d" % f.tell())
333
334 f.truncate()
335 if f.tell() != 5:
336 self.fail("File pos after ftruncate wrong %d" % f.tell())
337
338 f.close()
339 size = os.path.getsize(TESTFN)
340 if size != 5:
341 self.fail("File size after ftruncate wrong %d" % size)
342
343 try:
344 bug801631()
345 finally:
346 os.unlink(TESTFN)
347
Walter Dörwald3a77c7a2007-06-06 16:31:14 +0000348 def testAppend(self):
349 try:
350 f = open(TESTFN, 'wb')
351 f.write(b'spam')
352 f.close()
353 f = open(TESTFN, 'ab')
354 f.write(b'eggs')
355 f.close()
356 f = open(TESTFN, 'rb')
357 d = f.read()
358 f.close()
359 self.assertEqual(d, b'spameggs')
360 finally:
361 try:
362 os.unlink(TESTFN)
363 except:
364 pass
365
Neal Norwitz6e0e0e62008-08-24 22:07:28 +0000366 def testInvalidInit(self):
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000367 self.assertRaises(TypeError, _FileIO, "1", 0, 0)
Neal Norwitz6e0e0e62008-08-24 22:07:28 +0000368
Benjamin Peterson65676e42008-11-05 21:42:45 +0000369 def testWarnings(self):
370 with check_warnings() as w:
371 self.assertEqual(w.warnings, [])
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000372 self.assertRaises(TypeError, _FileIO, [])
Benjamin Peterson65676e42008-11-05 21:42:45 +0000373 self.assertEqual(w.warnings, [])
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000374 self.assertRaises(ValueError, _FileIO, "/some/invalid/name", "rt")
Benjamin Peterson65676e42008-11-05 21:42:45 +0000375 self.assertEqual(w.warnings, [])
376
Neal Norwitz6e0e0e62008-08-24 22:07:28 +0000377
Guido van Rossuma9e20242007-03-08 00:43:48 +0000378def test_main():
379 # Historically, these tests have been sloppy about removing TESTFN.
380 # So get rid of it no matter what.
381 try:
382 run_unittest(AutoFileTests, OtherFileTests)
383 finally:
384 if os.path.exists(TESTFN):
385 os.unlink(TESTFN)
386
387if __name__ == '__main__':
388 test_main()