blob: c2a63ad504c8666dae7b7dd24be1725dc8fbee3c [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
Antoine Pitrou78e761e2012-10-16 22:57:11 +020012from UserList import UserList
Serhiy Storchaka74f49ab2013-01-19 12:55:39 +020013import _testcapi
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000014
Georg Brandla4f46e12010-02-07 17:03:15 +000015from test.test_support import TESTFN, check_warnings, run_unittest, make_bad_fd
Antoine Pitrou19690592009-06-12 20:14:08 +000016from test.test_support import py3k_bytes as bytes
Antoine Pitrou7c587bf2010-04-19 18:52:43 +000017from test.script_helper import run_python
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000018
Antoine Pitrou19690592009-06-12 20:14:08 +000019from _io import FileIO as _FileIO
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000020
21class AutoFileTests(unittest.TestCase):
22 # file tests for which a test file is automatically set up
23
24 def setUp(self):
Antoine Pitrou19690592009-06-12 20:14:08 +000025 self.f = _FileIO(TESTFN, 'w')
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000026
27 def tearDown(self):
28 if self.f:
29 self.f.close()
30 os.remove(TESTFN)
31
32 def testWeakRefs(self):
33 # verify weak references
34 p = proxy(self.f)
35 p.write(bytes(range(10)))
Ezio Melotti2623a372010-11-21 13:34:58 +000036 self.assertEqual(self.f.tell(), p.tell())
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000037 self.f.close()
38 self.f = None
39 self.assertRaises(ReferenceError, getattr, p, 'tell')
40
41 def testSeekTell(self):
Antoine Pitrou19690592009-06-12 20:14:08 +000042 self.f.write(bytes(range(20)))
Ezio Melotti2623a372010-11-21 13:34:58 +000043 self.assertEqual(self.f.tell(), 20)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000044 self.f.seek(0)
Ezio Melotti2623a372010-11-21 13:34:58 +000045 self.assertEqual(self.f.tell(), 0)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000046 self.f.seek(10)
Ezio Melotti2623a372010-11-21 13:34:58 +000047 self.assertEqual(self.f.tell(), 10)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000048 self.f.seek(5, 1)
Ezio Melotti2623a372010-11-21 13:34:58 +000049 self.assertEqual(self.f.tell(), 15)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000050 self.f.seek(-5, 1)
Ezio Melotti2623a372010-11-21 13:34:58 +000051 self.assertEqual(self.f.tell(), 10)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000052 self.f.seek(-5, 2)
Ezio Melotti2623a372010-11-21 13:34:58 +000053 self.assertEqual(self.f.tell(), 15)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000054
55 def testAttributes(self):
56 # verify expected attributes exist
57 f = self.f
58
Ezio Melotti2623a372010-11-21 13:34:58 +000059 self.assertEqual(f.mode, "wb")
60 self.assertEqual(f.closed, False)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000061
62 # verify the attributes are readonly
63 for attr in 'mode', 'closed':
64 self.assertRaises((AttributeError, TypeError),
65 setattr, f, attr, 'oops')
66
67 def testReadinto(self):
68 # verify readinto
Antoine Pitrou19690592009-06-12 20:14:08 +000069 self.f.write(b"\x01\x02")
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000070 self.f.close()
Antoine Pitrou19690592009-06-12 20:14:08 +000071 a = array(b'b', b'x'*10)
72 self.f = _FileIO(TESTFN, 'r')
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000073 n = self.f.readinto(a)
Ezio Melotti2623a372010-11-21 13:34:58 +000074 self.assertEqual(array(b'b', [1, 2]), a[:n])
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000075
Antoine Pitrou78e761e2012-10-16 22:57:11 +020076 def testWritelinesList(self):
77 l = [b'123', b'456']
78 self.f.writelines(l)
79 self.f.close()
80 self.f = _FileIO(TESTFN, 'rb')
81 buf = self.f.read()
82 self.assertEqual(buf, b'123456')
83
84 def testWritelinesUserList(self):
85 l = UserList([b'123', b'456'])
86 self.f.writelines(l)
87 self.f.close()
88 self.f = _FileIO(TESTFN, 'rb')
89 buf = self.f.read()
90 self.assertEqual(buf, b'123456')
91
92 def testWritelinesError(self):
93 self.assertRaises(TypeError, self.f.writelines, [1, 2, 3])
94 self.assertRaises(TypeError, self.f.writelines, None)
95
Benjamin Petersonddd392c2009-12-13 19:19:07 +000096 def test_none_args(self):
97 self.f.write(b"hi\nbye\nabc")
98 self.f.close()
99 self.f = _FileIO(TESTFN, 'r')
100 self.assertEqual(self.f.read(None), b"hi\nbye\nabc")
101 self.f.seek(0)
102 self.assertEqual(self.f.readline(None), b"hi\n")
103 self.assertEqual(self.f.readlines(None), [b"bye\n", b"abc"])
104
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000105 def testRepr(self):
Ezio Melotti2623a372010-11-21 13:34:58 +0000106 self.assertEqual(repr(self.f), "<_io.FileIO name=%r mode='%s'>"
107 % (self.f.name, self.f.mode))
Antoine Pitrou19690592009-06-12 20:14:08 +0000108 del self.f.name
Ezio Melotti2623a372010-11-21 13:34:58 +0000109 self.assertEqual(repr(self.f), "<_io.FileIO fd=%r mode='%s'>"
110 % (self.f.fileno(), self.f.mode))
Antoine Pitrou19690592009-06-12 20:14:08 +0000111 self.f.close()
Ezio Melotti2623a372010-11-21 13:34:58 +0000112 self.assertEqual(repr(self.f), "<_io.FileIO [closed]>")
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000113
114 def testErrors(self):
115 f = self.f
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000116 self.assertTrue(not f.isatty())
117 self.assertTrue(not f.closed)
Ezio Melotti2623a372010-11-21 13:34:58 +0000118 #self.assertEqual(f.name, TESTFN)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000119 self.assertRaises(ValueError, f.read, 10) # Open for reading
120 f.close()
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000121 self.assertTrue(f.closed)
Antoine Pitrou19690592009-06-12 20:14:08 +0000122 f = _FileIO(TESTFN, 'r')
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000123 self.assertRaises(TypeError, f.readinto, "")
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000124 self.assertTrue(not f.closed)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000125 f.close()
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000126 self.assertTrue(f.closed)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000127
128 def testMethods(self):
129 methods = ['fileno', 'isatty', 'read', 'readinto',
130 'seek', 'tell', 'truncate', 'write', 'seekable',
131 'readable', 'writable']
132 if sys.platform.startswith('atheos'):
133 methods.remove('truncate')
134
135 self.f.close()
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000136 self.assertTrue(self.f.closed)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000137
138 for methodname in methods:
139 method = getattr(self.f, methodname)
140 # should raise on closed file
141 self.assertRaises(ValueError, method)
142
Benjamin Petersonf22c26e2008-09-01 14:13:43 +0000143 def testOpendir(self):
144 # Issue 3703: opening a directory should fill the errno
145 # Windows always returns "[Errno 13]: Permission denied
146 # Unix calls dircheck() and returns "[Errno 21]: Is a directory"
147 try:
Antoine Pitrou19690592009-06-12 20:14:08 +0000148 _FileIO('.', 'r')
Benjamin Petersonf22c26e2008-09-01 14:13:43 +0000149 except IOError as e:
150 self.assertNotEqual(e.errno, 0)
Benjamin Peterson7af65562008-12-29 17:56:58 +0000151 self.assertEqual(e.filename, ".")
Benjamin Petersonf22c26e2008-09-01 14:13:43 +0000152 else:
153 self.fail("Should have raised IOError")
154
Antoine Pitrouc2ec9922012-07-06 18:48:24 +0200155 @unittest.skipIf(os.name == 'nt', "test only works on a POSIX-like system")
156 def testOpenDirFD(self):
157 fd = os.open('.', os.O_RDONLY)
158 with self.assertRaises(IOError) as cm:
159 _FileIO(fd, 'r')
160 os.close(fd)
161 self.assertEqual(cm.exception.errno, errno.EISDIR)
162
Antoine Pitrou19690592009-06-12 20:14:08 +0000163 #A set of functions testing that we get expected behaviour if someone has
164 #manually closed the internal file descriptor. First, a decorator:
165 def ClosedFD(func):
166 @wraps(func)
167 def wrapper(self):
168 #forcibly close the fd before invoking the problem function
169 f = self.f
170 os.close(f.fileno())
171 try:
172 func(self, f)
173 finally:
174 try:
175 self.f.close()
176 except IOError:
177 pass
178 return wrapper
179
180 def ClosedFDRaises(func):
181 @wraps(func)
182 def wrapper(self):
183 #forcibly close the fd before invoking the problem function
184 f = self.f
185 os.close(f.fileno())
186 try:
187 func(self, f)
188 except IOError as e:
189 self.assertEqual(e.errno, errno.EBADF)
190 else:
191 self.fail("Should have raised IOError")
192 finally:
193 try:
194 self.f.close()
195 except IOError:
196 pass
197 return wrapper
198
199 @ClosedFDRaises
200 def testErrnoOnClose(self, f):
201 f.close()
202
203 @ClosedFDRaises
204 def testErrnoOnClosedWrite(self, f):
205 f.write('a')
206
207 @ClosedFDRaises
208 def testErrnoOnClosedSeek(self, f):
209 f.seek(0)
210
211 @ClosedFDRaises
212 def testErrnoOnClosedTell(self, f):
213 f.tell()
214
215 @ClosedFDRaises
216 def testErrnoOnClosedTruncate(self, f):
217 f.truncate(0)
218
219 @ClosedFD
220 def testErrnoOnClosedSeekable(self, f):
221 f.seekable()
222
223 @ClosedFD
224 def testErrnoOnClosedReadable(self, f):
225 f.readable()
226
227 @ClosedFD
228 def testErrnoOnClosedWritable(self, f):
229 f.writable()
230
231 @ClosedFD
232 def testErrnoOnClosedFileno(self, f):
233 f.fileno()
234
235 @ClosedFD
236 def testErrnoOnClosedIsatty(self, f):
237 self.assertEqual(f.isatty(), False)
238
239 def ReopenForRead(self):
240 try:
241 self.f.close()
242 except IOError:
243 pass
244 self.f = _FileIO(TESTFN, 'r')
245 os.close(self.f.fileno())
246 return self.f
247
248 @ClosedFDRaises
249 def testErrnoOnClosedRead(self, f):
250 f = self.ReopenForRead()
251 f.read(1)
252
253 @ClosedFDRaises
254 def testErrnoOnClosedReadall(self, f):
255 f = self.ReopenForRead()
256 f.readall()
257
258 @ClosedFDRaises
259 def testErrnoOnClosedReadinto(self, f):
260 f = self.ReopenForRead()
261 a = array(b'b', b'x'*10)
262 f.readinto(a)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000263
264class OtherFileTests(unittest.TestCase):
265
266 def testAbles(self):
267 try:
Antoine Pitrou19690592009-06-12 20:14:08 +0000268 f = _FileIO(TESTFN, "w")
Ezio Melotti2623a372010-11-21 13:34:58 +0000269 self.assertEqual(f.readable(), False)
270 self.assertEqual(f.writable(), True)
271 self.assertEqual(f.seekable(), True)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000272 f.close()
273
Antoine Pitrou19690592009-06-12 20:14:08 +0000274 f = _FileIO(TESTFN, "r")
Ezio Melotti2623a372010-11-21 13:34:58 +0000275 self.assertEqual(f.readable(), True)
276 self.assertEqual(f.writable(), False)
277 self.assertEqual(f.seekable(), True)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000278 f.close()
279
Antoine Pitrou19690592009-06-12 20:14:08 +0000280 f = _FileIO(TESTFN, "a+")
Ezio Melotti2623a372010-11-21 13:34:58 +0000281 self.assertEqual(f.readable(), True)
282 self.assertEqual(f.writable(), True)
283 self.assertEqual(f.seekable(), True)
284 self.assertEqual(f.isatty(), False)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000285 f.close()
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000286 finally:
287 os.unlink(TESTFN)
288
Zachary Ware1f702212013-12-10 14:09:20 -0600289 @unittest.skipIf(sys.platform == 'win32', 'no ttys on Windows')
290 def testAblesOnTTY(self):
291 try:
292 f = _FileIO("/dev/tty", "a")
293 except EnvironmentError:
294 # When run in a cron job there just aren't any
295 # ttys, so skip the test. This also handles other
296 # OS'es that don't support /dev/tty.
297 self.skipTest('need /dev/tty')
298 else:
299 self.assertEqual(f.readable(), False)
300 self.assertEqual(f.writable(), True)
301 if sys.platform != "darwin" and \
302 'bsd' not in sys.platform and \
303 not sys.platform.startswith('sunos'):
304 # Somehow /dev/tty appears seekable on some BSDs
305 self.assertEqual(f.seekable(), False)
306 self.assertEqual(f.isatty(), True)
307 f.close()
308
Antoine Pitrou213fec42013-09-04 20:46:33 +0200309 def testInvalidModeStrings(self):
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000310 # check invalid mode strings
Benjamin Petersonbfc51562008-11-22 01:59:15 +0000311 for mode in ("", "aU", "wU+", "rw", "rt"):
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000312 try:
Antoine Pitrou19690592009-06-12 20:14:08 +0000313 f = _FileIO(TESTFN, mode)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000314 except ValueError:
315 pass
316 else:
317 f.close()
318 self.fail('%r is an invalid file mode' % mode)
319
Antoine Pitrou213fec42013-09-04 20:46:33 +0200320 def testModeStrings(self):
321 # test that the mode attribute is correct for various mode strings
322 # given as init args
323 try:
324 for modes in [('w', 'wb'), ('wb', 'wb'), ('wb+', 'rb+'),
325 ('w+b', 'rb+'), ('a', 'ab'), ('ab', 'ab'),
326 ('ab+', 'ab+'), ('a+b', 'ab+'), ('r', 'rb'),
327 ('rb', 'rb'), ('rb+', 'rb+'), ('r+b', 'rb+')]:
328 # read modes are last so that TESTFN will exist first
329 with _FileIO(TESTFN, modes[0]) as f:
330 self.assertEqual(f.mode, modes[1])
331 finally:
332 if os.path.exists(TESTFN):
333 os.unlink(TESTFN)
334
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000335 def testUnicodeOpen(self):
336 # verify repr works for unicode too
Antoine Pitrou19690592009-06-12 20:14:08 +0000337 f = _FileIO(str(TESTFN), "w")
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000338 f.close()
339 os.unlink(TESTFN)
340
Antoine Pitrou19690592009-06-12 20:14:08 +0000341 def testBytesOpen(self):
342 # Opening a bytes filename
343 try:
344 fn = TESTFN.encode("ascii")
345 except UnicodeEncodeError:
Zachary Ware1f702212013-12-10 14:09:20 -0600346 self.skipTest('could not encode %r to ascii' % TESTFN)
Antoine Pitrou19690592009-06-12 20:14:08 +0000347 f = _FileIO(fn, "w")
348 try:
349 f.write(b"abc")
350 f.close()
351 with open(TESTFN, "rb") as f:
Ezio Melotti2623a372010-11-21 13:34:58 +0000352 self.assertEqual(f.read(), b"abc")
Antoine Pitrou19690592009-06-12 20:14:08 +0000353 finally:
354 os.unlink(TESTFN)
355
Benjamin Peterson5848d1f2009-01-19 00:08:08 +0000356 def testInvalidFd(self):
Antoine Pitrou19690592009-06-12 20:14:08 +0000357 self.assertRaises(ValueError, _FileIO, -10)
358 self.assertRaises(OSError, _FileIO, make_bad_fd())
Antoine Pitroue6ebcda2010-09-04 21:24:42 +0000359 if sys.platform == 'win32':
360 import msvcrt
361 self.assertRaises(IOError, msvcrt.get_osfhandle, make_bad_fd())
Serhiy Storchaka74f49ab2013-01-19 12:55:39 +0200362 # Issue 15989
363 self.assertRaises(TypeError, _FileIO, _testcapi.INT_MAX + 1)
364 self.assertRaises(TypeError, _FileIO, _testcapi.INT_MIN - 1)
Benjamin Peterson5848d1f2009-01-19 00:08:08 +0000365
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000366 def testBadModeArgument(self):
367 # verify that we get a sensible error message for bad mode argument
368 bad_mode = "qwerty"
369 try:
Antoine Pitrou19690592009-06-12 20:14:08 +0000370 f = _FileIO(TESTFN, bad_mode)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000371 except ValueError as msg:
372 if msg.args[0] != 0:
373 s = str(msg)
Ezio Melotti187f93d2010-03-17 14:22:34 +0000374 if TESTFN in s or bad_mode not in s:
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000375 self.fail("bad error message for invalid mode: %s" % s)
376 # if msg.args[0] == 0, we're probably on Windows where there may be
377 # no obvious way to discover why open() failed.
378 else:
379 f.close()
380 self.fail("no error for invalid mode: %s" % bad_mode)
381
Antoine Pitrouf3fa0742010-01-31 22:26:04 +0000382 def testTruncate(self):
383 f = _FileIO(TESTFN, 'w')
384 f.write(bytes(bytearray(range(10))))
385 self.assertEqual(f.tell(), 10)
386 f.truncate(5)
387 self.assertEqual(f.tell(), 10)
388 self.assertEqual(f.seek(0, os.SEEK_END), 5)
389 f.truncate(15)
390 self.assertEqual(f.tell(), 5)
391 self.assertEqual(f.seek(0, os.SEEK_END), 15)
Antoine Pitrou594a0462010-10-31 13:05:48 +0000392 f.close()
Antoine Pitrouf3fa0742010-01-31 22:26:04 +0000393
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000394 def testTruncateOnWindows(self):
395 def bug801631():
396 # SF bug <http://www.python.org/sf/801631>
397 # "file.truncate fault on windows"
Antoine Pitrou19690592009-06-12 20:14:08 +0000398 f = _FileIO(TESTFN, 'w')
399 f.write(bytes(range(11)))
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000400 f.close()
401
Antoine Pitrou19690592009-06-12 20:14:08 +0000402 f = _FileIO(TESTFN,'r+')
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000403 data = f.read(5)
Antoine Pitrou19690592009-06-12 20:14:08 +0000404 if data != bytes(range(5)):
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000405 self.fail("Read on file opened for update failed %r" % data)
406 if f.tell() != 5:
407 self.fail("File pos after read wrong %d" % f.tell())
408
409 f.truncate()
410 if f.tell() != 5:
411 self.fail("File pos after ftruncate wrong %d" % f.tell())
412
413 f.close()
414 size = os.path.getsize(TESTFN)
415 if size != 5:
416 self.fail("File size after ftruncate wrong %d" % size)
417
418 try:
419 bug801631()
420 finally:
421 os.unlink(TESTFN)
422
423 def testAppend(self):
424 try:
425 f = open(TESTFN, 'wb')
426 f.write(b'spam')
427 f.close()
428 f = open(TESTFN, 'ab')
429 f.write(b'eggs')
430 f.close()
431 f = open(TESTFN, 'rb')
432 d = f.read()
433 f.close()
434 self.assertEqual(d, b'spameggs')
435 finally:
436 try:
437 os.unlink(TESTFN)
438 except:
439 pass
440
Neal Norwitz901e4712008-08-24 22:03:05 +0000441 def testInvalidInit(self):
Antoine Pitrou19690592009-06-12 20:14:08 +0000442 self.assertRaises(TypeError, _FileIO, "1", 0, 0)
Neal Norwitz901e4712008-08-24 22:03:05 +0000443
Benjamin Peterson81e92e52008-11-03 22:34:57 +0000444 def testWarnings(self):
Florent Xicluna6de9e932010-03-07 12:18:33 +0000445 with check_warnings(quiet=True) as w:
Benjamin Peterson81e92e52008-11-03 22:34:57 +0000446 self.assertEqual(w.warnings, [])
Antoine Pitrou19690592009-06-12 20:14:08 +0000447 self.assertRaises(TypeError, _FileIO, [])
Benjamin Peterson81e92e52008-11-03 22:34:57 +0000448 self.assertEqual(w.warnings, [])
Antoine Pitrou19690592009-06-12 20:14:08 +0000449 self.assertRaises(ValueError, _FileIO, "/some/invalid/name", "rt")
Benjamin Peterson81e92e52008-11-03 22:34:57 +0000450 self.assertEqual(w.warnings, [])
451
Antoine Pitrou7c587bf2010-04-19 18:52:43 +0000452 def test_surrogates(self):
453 # Issue #8438: try to open a filename containing surrogates.
454 # It should either fail because the file doesn't exist or the filename
455 # can't be represented using the filesystem encoding, but not because
456 # of a LookupError for the error handler "surrogateescape".
457 filename = u'\udc80.txt'
458 try:
459 with _FileIO(filename):
460 pass
461 except (UnicodeEncodeError, IOError):
462 pass
463 # Spawn a separate Python process with a different "file system
464 # default encoding", to exercise this further.
465 env = dict(os.environ)
466 env[b'LC_CTYPE'] = b'C'
467 _, out = run_python('-c', 'import _io; _io.FileIO(%r)' % filename, env=env)
Ned Deilyaa1e1a22013-02-11 22:10:59 -0800468 if ('UnicodeEncodeError' not in out and not
469 ( ('IOError: [Errno 2] No such file or directory' in out) or
470 ('IOError: [Errno 22] Invalid argument' in out) ) ):
Antoine Pitrou7c587bf2010-04-19 18:52:43 +0000471 self.fail('Bad output: %r' % out)
Neal Norwitz901e4712008-08-24 22:03:05 +0000472
Hynek Schlawack9bd4bf22012-06-21 19:45:19 +0200473 def testUnclosedFDOnException(self):
474 class MyException(Exception): pass
475 class MyFileIO(_FileIO):
476 def __setattr__(self, name, value):
477 if name == "name":
478 raise MyException("blocked setting name")
479 return super(MyFileIO, self).__setattr__(name, value)
480 fd = os.open(__file__, os.O_RDONLY)
481 self.assertRaises(MyException, MyFileIO, fd)
482 os.close(fd) # should not raise OSError(EBADF)
483
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000484def test_main():
485 # Historically, these tests have been sloppy about removing TESTFN.
486 # So get rid of it no matter what.
487 try:
488 run_unittest(AutoFileTests, OtherFileTests)
489 finally:
490 if os.path.exists(TESTFN):
491 os.unlink(TESTFN)
492
493if __name__ == '__main__':
494 test_main()