blob: 5589d050b090fe3f9a1af178fad6903eab50925e [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
Georg Brandla4f46e12010-02-07 17:03:15 +000013from test.test_support import TESTFN, check_warnings, run_unittest, make_bad_fd
Antoine Pitrou19690592009-06-12 20:14:08 +000014from test.test_support import py3k_bytes as bytes
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000015
Antoine Pitrou19690592009-06-12 20:14:08 +000016from _io import FileIO as _FileIO
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000017
18class AutoFileTests(unittest.TestCase):
19 # file tests for which a test file is automatically set up
20
21 def setUp(self):
Antoine Pitrou19690592009-06-12 20:14:08 +000022 self.f = _FileIO(TESTFN, 'w')
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000023
24 def tearDown(self):
25 if self.f:
26 self.f.close()
27 os.remove(TESTFN)
28
29 def testWeakRefs(self):
30 # verify weak references
31 p = proxy(self.f)
32 p.write(bytes(range(10)))
33 self.assertEquals(self.f.tell(), p.tell())
34 self.f.close()
35 self.f = None
36 self.assertRaises(ReferenceError, getattr, p, 'tell')
37
38 def testSeekTell(self):
Antoine Pitrou19690592009-06-12 20:14:08 +000039 self.f.write(bytes(range(20)))
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000040 self.assertEquals(self.f.tell(), 20)
41 self.f.seek(0)
42 self.assertEquals(self.f.tell(), 0)
43 self.f.seek(10)
44 self.assertEquals(self.f.tell(), 10)
45 self.f.seek(5, 1)
46 self.assertEquals(self.f.tell(), 15)
47 self.f.seek(-5, 1)
48 self.assertEquals(self.f.tell(), 10)
49 self.f.seek(-5, 2)
50 self.assertEquals(self.f.tell(), 15)
51
52 def testAttributes(self):
53 # verify expected attributes exist
54 f = self.f
55
Benjamin Petersonbfc51562008-11-22 01:59:15 +000056 self.assertEquals(f.mode, "wb")
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000057 self.assertEquals(f.closed, False)
58
59 # verify the attributes are readonly
60 for attr in 'mode', 'closed':
61 self.assertRaises((AttributeError, TypeError),
62 setattr, f, attr, 'oops')
63
64 def testReadinto(self):
65 # verify readinto
Antoine Pitrou19690592009-06-12 20:14:08 +000066 self.f.write(b"\x01\x02")
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000067 self.f.close()
Antoine Pitrou19690592009-06-12 20:14:08 +000068 a = array(b'b', b'x'*10)
69 self.f = _FileIO(TESTFN, 'r')
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000070 n = self.f.readinto(a)
Antoine Pitrou19690592009-06-12 20:14:08 +000071 self.assertEquals(array(b'b', [1, 2]), a[:n])
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000072
Benjamin Petersonddd392c2009-12-13 19:19:07 +000073 def test_none_args(self):
74 self.f.write(b"hi\nbye\nabc")
75 self.f.close()
76 self.f = _FileIO(TESTFN, 'r')
77 self.assertEqual(self.f.read(None), b"hi\nbye\nabc")
78 self.f.seek(0)
79 self.assertEqual(self.f.readline(None), b"hi\n")
80 self.assertEqual(self.f.readlines(None), [b"bye\n", b"abc"])
81
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000082 def testRepr(self):
Antoine Pitrou19690592009-06-12 20:14:08 +000083 self.assertEquals(repr(self.f), "<_io.FileIO name=%r mode='%s'>"
84 % (self.f.name, self.f.mode))
85 del self.f.name
86 self.assertEquals(repr(self.f), "<_io.FileIO fd=%r mode='%s'>"
87 % (self.f.fileno(), self.f.mode))
88 self.f.close()
89 self.assertEquals(repr(self.f), "<_io.FileIO [closed]>")
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000090
91 def testErrors(self):
92 f = self.f
Benjamin Peterson5c8da862009-06-30 22:57:08 +000093 self.assertTrue(not f.isatty())
94 self.assertTrue(not f.closed)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000095 #self.assertEquals(f.name, TESTFN)
96 self.assertRaises(ValueError, f.read, 10) # Open for reading
97 f.close()
Benjamin Peterson5c8da862009-06-30 22:57:08 +000098 self.assertTrue(f.closed)
Antoine Pitrou19690592009-06-12 20:14:08 +000099 f = _FileIO(TESTFN, 'r')
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000100 self.assertRaises(TypeError, f.readinto, "")
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000101 self.assertTrue(not f.closed)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000102 f.close()
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000103 self.assertTrue(f.closed)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000104
105 def testMethods(self):
106 methods = ['fileno', 'isatty', 'read', 'readinto',
107 'seek', 'tell', 'truncate', 'write', 'seekable',
108 'readable', 'writable']
109 if sys.platform.startswith('atheos'):
110 methods.remove('truncate')
111
112 self.f.close()
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000113 self.assertTrue(self.f.closed)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000114
115 for methodname in methods:
116 method = getattr(self.f, methodname)
117 # should raise on closed file
118 self.assertRaises(ValueError, method)
119
Benjamin Petersonf22c26e2008-09-01 14:13:43 +0000120 def testOpendir(self):
121 # Issue 3703: opening a directory should fill the errno
122 # Windows always returns "[Errno 13]: Permission denied
123 # Unix calls dircheck() and returns "[Errno 21]: Is a directory"
124 try:
Antoine Pitrou19690592009-06-12 20:14:08 +0000125 _FileIO('.', 'r')
Benjamin Petersonf22c26e2008-09-01 14:13:43 +0000126 except IOError as e:
127 self.assertNotEqual(e.errno, 0)
Benjamin Peterson7af65562008-12-29 17:56:58 +0000128 self.assertEqual(e.filename, ".")
Benjamin Petersonf22c26e2008-09-01 14:13:43 +0000129 else:
130 self.fail("Should have raised IOError")
131
Antoine Pitrou19690592009-06-12 20:14:08 +0000132 #A set of functions testing that we get expected behaviour if someone has
133 #manually closed the internal file descriptor. First, a decorator:
134 def ClosedFD(func):
135 @wraps(func)
136 def wrapper(self):
137 #forcibly close the fd before invoking the problem function
138 f = self.f
139 os.close(f.fileno())
140 try:
141 func(self, f)
142 finally:
143 try:
144 self.f.close()
145 except IOError:
146 pass
147 return wrapper
148
149 def ClosedFDRaises(func):
150 @wraps(func)
151 def wrapper(self):
152 #forcibly close the fd before invoking the problem function
153 f = self.f
154 os.close(f.fileno())
155 try:
156 func(self, f)
157 except IOError as e:
158 self.assertEqual(e.errno, errno.EBADF)
159 else:
160 self.fail("Should have raised IOError")
161 finally:
162 try:
163 self.f.close()
164 except IOError:
165 pass
166 return wrapper
167
168 @ClosedFDRaises
169 def testErrnoOnClose(self, f):
170 f.close()
171
172 @ClosedFDRaises
173 def testErrnoOnClosedWrite(self, f):
174 f.write('a')
175
176 @ClosedFDRaises
177 def testErrnoOnClosedSeek(self, f):
178 f.seek(0)
179
180 @ClosedFDRaises
181 def testErrnoOnClosedTell(self, f):
182 f.tell()
183
184 @ClosedFDRaises
185 def testErrnoOnClosedTruncate(self, f):
186 f.truncate(0)
187
188 @ClosedFD
189 def testErrnoOnClosedSeekable(self, f):
190 f.seekable()
191
192 @ClosedFD
193 def testErrnoOnClosedReadable(self, f):
194 f.readable()
195
196 @ClosedFD
197 def testErrnoOnClosedWritable(self, f):
198 f.writable()
199
200 @ClosedFD
201 def testErrnoOnClosedFileno(self, f):
202 f.fileno()
203
204 @ClosedFD
205 def testErrnoOnClosedIsatty(self, f):
206 self.assertEqual(f.isatty(), False)
207
208 def ReopenForRead(self):
209 try:
210 self.f.close()
211 except IOError:
212 pass
213 self.f = _FileIO(TESTFN, 'r')
214 os.close(self.f.fileno())
215 return self.f
216
217 @ClosedFDRaises
218 def testErrnoOnClosedRead(self, f):
219 f = self.ReopenForRead()
220 f.read(1)
221
222 @ClosedFDRaises
223 def testErrnoOnClosedReadall(self, f):
224 f = self.ReopenForRead()
225 f.readall()
226
227 @ClosedFDRaises
228 def testErrnoOnClosedReadinto(self, f):
229 f = self.ReopenForRead()
230 a = array(b'b', b'x'*10)
231 f.readinto(a)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000232
233class OtherFileTests(unittest.TestCase):
234
235 def testAbles(self):
236 try:
Antoine Pitrou19690592009-06-12 20:14:08 +0000237 f = _FileIO(TESTFN, "w")
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000238 self.assertEquals(f.readable(), False)
239 self.assertEquals(f.writable(), True)
240 self.assertEquals(f.seekable(), True)
241 f.close()
242
Antoine Pitrou19690592009-06-12 20:14:08 +0000243 f = _FileIO(TESTFN, "r")
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000244 self.assertEquals(f.readable(), True)
245 self.assertEquals(f.writable(), False)
246 self.assertEquals(f.seekable(), True)
247 f.close()
248
Antoine Pitrou19690592009-06-12 20:14:08 +0000249 f = _FileIO(TESTFN, "a+")
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000250 self.assertEquals(f.readable(), True)
251 self.assertEquals(f.writable(), True)
252 self.assertEquals(f.seekable(), True)
253 self.assertEquals(f.isatty(), False)
254 f.close()
255
256 if sys.platform != "win32":
257 try:
Antoine Pitrou19690592009-06-12 20:14:08 +0000258 f = _FileIO("/dev/tty", "a")
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000259 except EnvironmentError:
260 # When run in a cron job there just aren't any
261 # ttys, so skip the test. This also handles other
262 # OS'es that don't support /dev/tty.
263 pass
264 else:
Antoine Pitrou19690592009-06-12 20:14:08 +0000265 f = _FileIO("/dev/tty", "a")
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000266 self.assertEquals(f.readable(), False)
267 self.assertEquals(f.writable(), True)
268 if sys.platform != "darwin" and \
Antoine Pitrouf3bd6872009-05-23 16:32:32 +0000269 'bsd' not in sys.platform and \
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000270 not sys.platform.startswith('sunos'):
271 # Somehow /dev/tty appears seekable on some BSDs
272 self.assertEquals(f.seekable(), False)
273 self.assertEquals(f.isatty(), True)
274 f.close()
275 finally:
276 os.unlink(TESTFN)
277
278 def testModeStrings(self):
279 # check invalid mode strings
Benjamin Petersonbfc51562008-11-22 01:59:15 +0000280 for mode in ("", "aU", "wU+", "rw", "rt"):
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000281 try:
Antoine Pitrou19690592009-06-12 20:14:08 +0000282 f = _FileIO(TESTFN, mode)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000283 except ValueError:
284 pass
285 else:
286 f.close()
287 self.fail('%r is an invalid file mode' % mode)
288
289 def testUnicodeOpen(self):
290 # verify repr works for unicode too
Antoine Pitrou19690592009-06-12 20:14:08 +0000291 f = _FileIO(str(TESTFN), "w")
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000292 f.close()
293 os.unlink(TESTFN)
294
Antoine Pitrou19690592009-06-12 20:14:08 +0000295 def testBytesOpen(self):
296 # Opening a bytes filename
297 try:
298 fn = TESTFN.encode("ascii")
299 except UnicodeEncodeError:
300 # Skip test
301 return
302 f = _FileIO(fn, "w")
303 try:
304 f.write(b"abc")
305 f.close()
306 with open(TESTFN, "rb") as f:
307 self.assertEquals(f.read(), b"abc")
308 finally:
309 os.unlink(TESTFN)
310
Benjamin Peterson5848d1f2009-01-19 00:08:08 +0000311 def testInvalidFd(self):
Antoine Pitrou19690592009-06-12 20:14:08 +0000312 self.assertRaises(ValueError, _FileIO, -10)
313 self.assertRaises(OSError, _FileIO, make_bad_fd())
Benjamin Peterson5848d1f2009-01-19 00:08:08 +0000314
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000315 def testBadModeArgument(self):
316 # verify that we get a sensible error message for bad mode argument
317 bad_mode = "qwerty"
318 try:
Antoine Pitrou19690592009-06-12 20:14:08 +0000319 f = _FileIO(TESTFN, bad_mode)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000320 except ValueError as msg:
321 if msg.args[0] != 0:
322 s = str(msg)
323 if s.find(TESTFN) != -1 or s.find(bad_mode) == -1:
324 self.fail("bad error message for invalid mode: %s" % s)
325 # if msg.args[0] == 0, we're probably on Windows where there may be
326 # no obvious way to discover why open() failed.
327 else:
328 f.close()
329 self.fail("no error for invalid mode: %s" % bad_mode)
330
Antoine Pitrouf3fa0742010-01-31 22:26:04 +0000331 def testTruncate(self):
332 f = _FileIO(TESTFN, 'w')
333 f.write(bytes(bytearray(range(10))))
334 self.assertEqual(f.tell(), 10)
335 f.truncate(5)
336 self.assertEqual(f.tell(), 10)
337 self.assertEqual(f.seek(0, os.SEEK_END), 5)
338 f.truncate(15)
339 self.assertEqual(f.tell(), 5)
340 self.assertEqual(f.seek(0, os.SEEK_END), 15)
341
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000342 def testTruncateOnWindows(self):
343 def bug801631():
344 # SF bug <http://www.python.org/sf/801631>
345 # "file.truncate fault on windows"
Antoine Pitrou19690592009-06-12 20:14:08 +0000346 f = _FileIO(TESTFN, 'w')
347 f.write(bytes(range(11)))
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000348 f.close()
349
Antoine Pitrou19690592009-06-12 20:14:08 +0000350 f = _FileIO(TESTFN,'r+')
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000351 data = f.read(5)
Antoine Pitrou19690592009-06-12 20:14:08 +0000352 if data != bytes(range(5)):
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000353 self.fail("Read on file opened for update failed %r" % data)
354 if f.tell() != 5:
355 self.fail("File pos after read wrong %d" % f.tell())
356
357 f.truncate()
358 if f.tell() != 5:
359 self.fail("File pos after ftruncate wrong %d" % f.tell())
360
361 f.close()
362 size = os.path.getsize(TESTFN)
363 if size != 5:
364 self.fail("File size after ftruncate wrong %d" % size)
365
366 try:
367 bug801631()
368 finally:
369 os.unlink(TESTFN)
370
371 def testAppend(self):
372 try:
373 f = open(TESTFN, 'wb')
374 f.write(b'spam')
375 f.close()
376 f = open(TESTFN, 'ab')
377 f.write(b'eggs')
378 f.close()
379 f = open(TESTFN, 'rb')
380 d = f.read()
381 f.close()
382 self.assertEqual(d, b'spameggs')
383 finally:
384 try:
385 os.unlink(TESTFN)
386 except:
387 pass
388
Neal Norwitz901e4712008-08-24 22:03:05 +0000389 def testInvalidInit(self):
Antoine Pitrou19690592009-06-12 20:14:08 +0000390 self.assertRaises(TypeError, _FileIO, "1", 0, 0)
Neal Norwitz901e4712008-08-24 22:03:05 +0000391
Benjamin Peterson81e92e52008-11-03 22:34:57 +0000392 def testWarnings(self):
393 with check_warnings() as w:
394 self.assertEqual(w.warnings, [])
Antoine Pitrou19690592009-06-12 20:14:08 +0000395 self.assertRaises(TypeError, _FileIO, [])
Benjamin Peterson81e92e52008-11-03 22:34:57 +0000396 self.assertEqual(w.warnings, [])
Antoine Pitrou19690592009-06-12 20:14:08 +0000397 self.assertRaises(ValueError, _FileIO, "/some/invalid/name", "rt")
Benjamin Peterson81e92e52008-11-03 22:34:57 +0000398 self.assertEqual(w.warnings, [])
399
Neal Norwitz901e4712008-08-24 22:03:05 +0000400
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000401def test_main():
402 # Historically, these tests have been sloppy about removing TESTFN.
403 # So get rid of it no matter what.
404 try:
405 run_unittest(AutoFileTests, OtherFileTests)
406 finally:
407 if os.path.exists(TESTFN):
408 os.unlink(TESTFN)
409
410if __name__ == '__main__':
411 test_main()