blob: cdc31237189eac26ffa58fc43f27f0eaac676543 [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
Benjamin Peterson5539c782009-01-19 17:37:42 +000013from test.test_support import (TESTFN, findfile, check_warnings, run_unittest,
14 make_bad_fd)
Antoine Pitrou19690592009-06-12 20:14:08 +000015from test.test_support import py3k_bytes as bytes
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)))
34 self.assertEquals(self.f.tell(), p.tell())
35 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)))
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000041 self.assertEquals(self.f.tell(), 20)
42 self.f.seek(0)
43 self.assertEquals(self.f.tell(), 0)
44 self.f.seek(10)
45 self.assertEquals(self.f.tell(), 10)
46 self.f.seek(5, 1)
47 self.assertEquals(self.f.tell(), 15)
48 self.f.seek(-5, 1)
49 self.assertEquals(self.f.tell(), 10)
50 self.f.seek(-5, 2)
51 self.assertEquals(self.f.tell(), 15)
52
53 def testAttributes(self):
54 # verify expected attributes exist
55 f = self.f
56
Benjamin Petersonbfc51562008-11-22 01:59:15 +000057 self.assertEquals(f.mode, "wb")
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000058 self.assertEquals(f.closed, False)
59
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)
Antoine Pitrou19690592009-06-12 20:14:08 +000072 self.assertEquals(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):
Antoine Pitrou19690592009-06-12 20:14:08 +000084 self.assertEquals(repr(self.f), "<_io.FileIO name=%r mode='%s'>"
85 % (self.f.name, self.f.mode))
86 del self.f.name
87 self.assertEquals(repr(self.f), "<_io.FileIO fd=%r mode='%s'>"
88 % (self.f.fileno(), self.f.mode))
89 self.f.close()
90 self.assertEquals(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)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000096 #self.assertEquals(f.name, TESTFN)
97 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 Pitrou19690592009-06-12 20:14:08 +0000133 #A set of functions testing that we get expected behaviour if someone has
134 #manually closed the internal file descriptor. First, a decorator:
135 def ClosedFD(func):
136 @wraps(func)
137 def wrapper(self):
138 #forcibly close the fd before invoking the problem function
139 f = self.f
140 os.close(f.fileno())
141 try:
142 func(self, f)
143 finally:
144 try:
145 self.f.close()
146 except IOError:
147 pass
148 return wrapper
149
150 def ClosedFDRaises(func):
151 @wraps(func)
152 def wrapper(self):
153 #forcibly close the fd before invoking the problem function
154 f = self.f
155 os.close(f.fileno())
156 try:
157 func(self, f)
158 except IOError as e:
159 self.assertEqual(e.errno, errno.EBADF)
160 else:
161 self.fail("Should have raised IOError")
162 finally:
163 try:
164 self.f.close()
165 except IOError:
166 pass
167 return wrapper
168
169 @ClosedFDRaises
170 def testErrnoOnClose(self, f):
171 f.close()
172
173 @ClosedFDRaises
174 def testErrnoOnClosedWrite(self, f):
175 f.write('a')
176
177 @ClosedFDRaises
178 def testErrnoOnClosedSeek(self, f):
179 f.seek(0)
180
181 @ClosedFDRaises
182 def testErrnoOnClosedTell(self, f):
183 f.tell()
184
185 @ClosedFDRaises
186 def testErrnoOnClosedTruncate(self, f):
187 f.truncate(0)
188
189 @ClosedFD
190 def testErrnoOnClosedSeekable(self, f):
191 f.seekable()
192
193 @ClosedFD
194 def testErrnoOnClosedReadable(self, f):
195 f.readable()
196
197 @ClosedFD
198 def testErrnoOnClosedWritable(self, f):
199 f.writable()
200
201 @ClosedFD
202 def testErrnoOnClosedFileno(self, f):
203 f.fileno()
204
205 @ClosedFD
206 def testErrnoOnClosedIsatty(self, f):
207 self.assertEqual(f.isatty(), False)
208
209 def ReopenForRead(self):
210 try:
211 self.f.close()
212 except IOError:
213 pass
214 self.f = _FileIO(TESTFN, 'r')
215 os.close(self.f.fileno())
216 return self.f
217
218 @ClosedFDRaises
219 def testErrnoOnClosedRead(self, f):
220 f = self.ReopenForRead()
221 f.read(1)
222
223 @ClosedFDRaises
224 def testErrnoOnClosedReadall(self, f):
225 f = self.ReopenForRead()
226 f.readall()
227
228 @ClosedFDRaises
229 def testErrnoOnClosedReadinto(self, f):
230 f = self.ReopenForRead()
231 a = array(b'b', b'x'*10)
232 f.readinto(a)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000233
234class OtherFileTests(unittest.TestCase):
235
236 def testAbles(self):
237 try:
Antoine Pitrou19690592009-06-12 20:14:08 +0000238 f = _FileIO(TESTFN, "w")
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000239 self.assertEquals(f.readable(), False)
240 self.assertEquals(f.writable(), True)
241 self.assertEquals(f.seekable(), True)
242 f.close()
243
Antoine Pitrou19690592009-06-12 20:14:08 +0000244 f = _FileIO(TESTFN, "r")
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000245 self.assertEquals(f.readable(), True)
246 self.assertEquals(f.writable(), False)
247 self.assertEquals(f.seekable(), True)
248 f.close()
249
Antoine Pitrou19690592009-06-12 20:14:08 +0000250 f = _FileIO(TESTFN, "a+")
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000251 self.assertEquals(f.readable(), True)
252 self.assertEquals(f.writable(), True)
253 self.assertEquals(f.seekable(), True)
254 self.assertEquals(f.isatty(), False)
255 f.close()
256
257 if sys.platform != "win32":
258 try:
Antoine Pitrou19690592009-06-12 20:14:08 +0000259 f = _FileIO("/dev/tty", "a")
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000260 except EnvironmentError:
261 # When run in a cron job there just aren't any
262 # ttys, so skip the test. This also handles other
263 # OS'es that don't support /dev/tty.
264 pass
265 else:
Antoine Pitrou19690592009-06-12 20:14:08 +0000266 f = _FileIO("/dev/tty", "a")
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000267 self.assertEquals(f.readable(), False)
268 self.assertEquals(f.writable(), True)
269 if sys.platform != "darwin" and \
Antoine Pitrouf3bd6872009-05-23 16:32:32 +0000270 'bsd' not in sys.platform and \
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000271 not sys.platform.startswith('sunos'):
272 # Somehow /dev/tty appears seekable on some BSDs
273 self.assertEquals(f.seekable(), False)
274 self.assertEquals(f.isatty(), True)
275 f.close()
276 finally:
277 os.unlink(TESTFN)
278
279 def testModeStrings(self):
280 # check invalid mode strings
Benjamin Petersonbfc51562008-11-22 01:59:15 +0000281 for mode in ("", "aU", "wU+", "rw", "rt"):
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000282 try:
Antoine Pitrou19690592009-06-12 20:14:08 +0000283 f = _FileIO(TESTFN, mode)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000284 except ValueError:
285 pass
286 else:
287 f.close()
288 self.fail('%r is an invalid file mode' % mode)
289
290 def testUnicodeOpen(self):
291 # verify repr works for unicode too
Antoine Pitrou19690592009-06-12 20:14:08 +0000292 f = _FileIO(str(TESTFN), "w")
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000293 f.close()
294 os.unlink(TESTFN)
295
Antoine Pitrou19690592009-06-12 20:14:08 +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 Peterson5848d1f2009-01-19 00:08:08 +0000312 def testInvalidFd(self):
Antoine Pitrou19690592009-06-12 20:14:08 +0000313 self.assertRaises(ValueError, _FileIO, -10)
314 self.assertRaises(OSError, _FileIO, make_bad_fd())
Benjamin Peterson5848d1f2009-01-19 00:08:08 +0000315
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000316 def testBadModeArgument(self):
317 # verify that we get a sensible error message for bad mode argument
318 bad_mode = "qwerty"
319 try:
Antoine Pitrou19690592009-06-12 20:14:08 +0000320 f = _FileIO(TESTFN, bad_mode)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000321 except ValueError as msg:
322 if msg.args[0] != 0:
323 s = str(msg)
324 if s.find(TESTFN) != -1 or s.find(bad_mode) == -1:
325 self.fail("bad error message for invalid mode: %s" % s)
326 # if msg.args[0] == 0, we're probably on Windows where there may be
327 # no obvious way to discover why open() failed.
328 else:
329 f.close()
330 self.fail("no error for invalid mode: %s" % bad_mode)
331
Antoine Pitrouf3fa0742010-01-31 22:26:04 +0000332 def testTruncate(self):
333 f = _FileIO(TESTFN, 'w')
334 f.write(bytes(bytearray(range(10))))
335 self.assertEqual(f.tell(), 10)
336 f.truncate(5)
337 self.assertEqual(f.tell(), 10)
338 self.assertEqual(f.seek(0, os.SEEK_END), 5)
339 f.truncate(15)
340 self.assertEqual(f.tell(), 5)
341 self.assertEqual(f.seek(0, os.SEEK_END), 15)
342
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000343 def testTruncateOnWindows(self):
344 def bug801631():
345 # SF bug <http://www.python.org/sf/801631>
346 # "file.truncate fault on windows"
Antoine Pitrou19690592009-06-12 20:14:08 +0000347 f = _FileIO(TESTFN, 'w')
348 f.write(bytes(range(11)))
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000349 f.close()
350
Antoine Pitrou19690592009-06-12 20:14:08 +0000351 f = _FileIO(TESTFN,'r+')
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000352 data = f.read(5)
Antoine Pitrou19690592009-06-12 20:14:08 +0000353 if data != bytes(range(5)):
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000354 self.fail("Read on file opened for update failed %r" % data)
355 if f.tell() != 5:
356 self.fail("File pos after read wrong %d" % f.tell())
357
358 f.truncate()
359 if f.tell() != 5:
360 self.fail("File pos after ftruncate wrong %d" % f.tell())
361
362 f.close()
363 size = os.path.getsize(TESTFN)
364 if size != 5:
365 self.fail("File size after ftruncate wrong %d" % size)
366
367 try:
368 bug801631()
369 finally:
370 os.unlink(TESTFN)
371
372 def testAppend(self):
373 try:
374 f = open(TESTFN, 'wb')
375 f.write(b'spam')
376 f.close()
377 f = open(TESTFN, 'ab')
378 f.write(b'eggs')
379 f.close()
380 f = open(TESTFN, 'rb')
381 d = f.read()
382 f.close()
383 self.assertEqual(d, b'spameggs')
384 finally:
385 try:
386 os.unlink(TESTFN)
387 except:
388 pass
389
Neal Norwitz901e4712008-08-24 22:03:05 +0000390 def testInvalidInit(self):
Antoine Pitrou19690592009-06-12 20:14:08 +0000391 self.assertRaises(TypeError, _FileIO, "1", 0, 0)
Neal Norwitz901e4712008-08-24 22:03:05 +0000392
Benjamin Peterson81e92e52008-11-03 22:34:57 +0000393 def testWarnings(self):
394 with check_warnings() as w:
395 self.assertEqual(w.warnings, [])
Antoine Pitrou19690592009-06-12 20:14:08 +0000396 self.assertRaises(TypeError, _FileIO, [])
Benjamin Peterson81e92e52008-11-03 22:34:57 +0000397 self.assertEqual(w.warnings, [])
Antoine Pitrou19690592009-06-12 20:14:08 +0000398 self.assertRaises(ValueError, _FileIO, "/some/invalid/name", "rt")
Benjamin Peterson81e92e52008-11-03 22:34:57 +0000399 self.assertEqual(w.warnings, [])
400
Neal Norwitz901e4712008-08-24 22:03:05 +0000401
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000402def test_main():
403 # Historically, these tests have been sloppy about removing TESTFN.
404 # So get rid of it no matter what.
405 try:
406 run_unittest(AutoFileTests, OtherFileTests)
407 finally:
408 if os.path.exists(TESTFN):
409 os.unlink(TESTFN)
410
411if __name__ == '__main__':
412 test_main()