blob: b45d79b164550724ad3ce46f46ece25cb8e32d94 [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
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000013
Georg Brandla4f46e12010-02-07 17:03:15 +000014from test.test_support import TESTFN, check_warnings, run_unittest, make_bad_fd
Serhiy Storchaka76249ea2014-02-07 10:06:05 +020015from test.test_support import py3k_bytes as bytes, cpython_only
Antoine Pitrou7c587bf2010-04-19 18:52:43 +000016from test.script_helper import run_python
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000017
Antoine Pitrou19690592009-06-12 20:14:08 +000018from _io import FileIO as _FileIO
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000019
20class AutoFileTests(unittest.TestCase):
21 # file tests for which a test file is automatically set up
22
23 def setUp(self):
Antoine Pitrou19690592009-06-12 20:14:08 +000024 self.f = _FileIO(TESTFN, 'w')
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000025
26 def tearDown(self):
27 if self.f:
28 self.f.close()
29 os.remove(TESTFN)
30
31 def testWeakRefs(self):
32 # verify weak references
33 p = proxy(self.f)
34 p.write(bytes(range(10)))
Ezio Melotti2623a372010-11-21 13:34:58 +000035 self.assertEqual(self.f.tell(), p.tell())
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000036 self.f.close()
37 self.f = None
38 self.assertRaises(ReferenceError, getattr, p, 'tell')
39
40 def testSeekTell(self):
Antoine Pitrou19690592009-06-12 20:14:08 +000041 self.f.write(bytes(range(20)))
Ezio Melotti2623a372010-11-21 13:34:58 +000042 self.assertEqual(self.f.tell(), 20)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000043 self.f.seek(0)
Ezio Melotti2623a372010-11-21 13:34:58 +000044 self.assertEqual(self.f.tell(), 0)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000045 self.f.seek(10)
Ezio Melotti2623a372010-11-21 13:34:58 +000046 self.assertEqual(self.f.tell(), 10)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000047 self.f.seek(5, 1)
Ezio Melotti2623a372010-11-21 13:34:58 +000048 self.assertEqual(self.f.tell(), 15)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000049 self.f.seek(-5, 1)
Ezio Melotti2623a372010-11-21 13:34:58 +000050 self.assertEqual(self.f.tell(), 10)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000051 self.f.seek(-5, 2)
Ezio Melotti2623a372010-11-21 13:34:58 +000052 self.assertEqual(self.f.tell(), 15)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000053
54 def testAttributes(self):
55 # verify expected attributes exist
56 f = self.f
57
Ezio Melotti2623a372010-11-21 13:34:58 +000058 self.assertEqual(f.mode, "wb")
59 self.assertEqual(f.closed, False)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000060
61 # verify the attributes are readonly
62 for attr in 'mode', 'closed':
63 self.assertRaises((AttributeError, TypeError),
64 setattr, f, attr, 'oops')
65
66 def testReadinto(self):
67 # verify readinto
Antoine Pitrou19690592009-06-12 20:14:08 +000068 self.f.write(b"\x01\x02")
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000069 self.f.close()
Antoine Pitrou19690592009-06-12 20:14:08 +000070 a = array(b'b', b'x'*10)
71 self.f = _FileIO(TESTFN, 'r')
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000072 n = self.f.readinto(a)
Ezio Melotti2623a372010-11-21 13:34:58 +000073 self.assertEqual(array(b'b', [1, 2]), a[:n])
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +000074
Antoine Pitrou78e761e2012-10-16 22:57:11 +020075 def testWritelinesList(self):
76 l = [b'123', b'456']
77 self.f.writelines(l)
78 self.f.close()
79 self.f = _FileIO(TESTFN, 'rb')
80 buf = self.f.read()
81 self.assertEqual(buf, b'123456')
82
83 def testWritelinesUserList(self):
84 l = UserList([b'123', b'456'])
85 self.f.writelines(l)
86 self.f.close()
87 self.f = _FileIO(TESTFN, 'rb')
88 buf = self.f.read()
89 self.assertEqual(buf, b'123456')
90
91 def testWritelinesError(self):
92 self.assertRaises(TypeError, self.f.writelines, [1, 2, 3])
93 self.assertRaises(TypeError, self.f.writelines, None)
94
Benjamin Petersonddd392c2009-12-13 19:19:07 +000095 def test_none_args(self):
96 self.f.write(b"hi\nbye\nabc")
97 self.f.close()
98 self.f = _FileIO(TESTFN, 'r')
99 self.assertEqual(self.f.read(None), b"hi\nbye\nabc")
100 self.f.seek(0)
101 self.assertEqual(self.f.readline(None), b"hi\n")
102 self.assertEqual(self.f.readlines(None), [b"bye\n", b"abc"])
103
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000104 def testRepr(self):
Ezio Melotti2623a372010-11-21 13:34:58 +0000105 self.assertEqual(repr(self.f), "<_io.FileIO name=%r mode='%s'>"
106 % (self.f.name, self.f.mode))
Antoine Pitrou19690592009-06-12 20:14:08 +0000107 del self.f.name
Ezio Melotti2623a372010-11-21 13:34:58 +0000108 self.assertEqual(repr(self.f), "<_io.FileIO fd=%r mode='%s'>"
109 % (self.f.fileno(), self.f.mode))
Antoine Pitrou19690592009-06-12 20:14:08 +0000110 self.f.close()
Ezio Melotti2623a372010-11-21 13:34:58 +0000111 self.assertEqual(repr(self.f), "<_io.FileIO [closed]>")
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000112
113 def testErrors(self):
114 f = self.f
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000115 self.assertTrue(not f.isatty())
116 self.assertTrue(not f.closed)
Ezio Melotti2623a372010-11-21 13:34:58 +0000117 #self.assertEqual(f.name, TESTFN)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000118 self.assertRaises(ValueError, f.read, 10) # Open for reading
119 f.close()
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000120 self.assertTrue(f.closed)
Antoine Pitrou19690592009-06-12 20:14:08 +0000121 f = _FileIO(TESTFN, 'r')
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000122 self.assertRaises(TypeError, f.readinto, "")
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000123 self.assertTrue(not f.closed)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000124 f.close()
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000125 self.assertTrue(f.closed)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000126
127 def testMethods(self):
128 methods = ['fileno', 'isatty', 'read', 'readinto',
129 'seek', 'tell', 'truncate', 'write', 'seekable',
130 'readable', 'writable']
131 if sys.platform.startswith('atheos'):
132 methods.remove('truncate')
133
134 self.f.close()
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000135 self.assertTrue(self.f.closed)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000136
137 for methodname in methods:
138 method = getattr(self.f, methodname)
139 # should raise on closed file
140 self.assertRaises(ValueError, method)
141
Benjamin Petersonf22c26e2008-09-01 14:13:43 +0000142 def testOpendir(self):
143 # Issue 3703: opening a directory should fill the errno
144 # Windows always returns "[Errno 13]: Permission denied
145 # Unix calls dircheck() and returns "[Errno 21]: Is a directory"
146 try:
Antoine Pitrou19690592009-06-12 20:14:08 +0000147 _FileIO('.', 'r')
Benjamin Petersonf22c26e2008-09-01 14:13:43 +0000148 except IOError as e:
149 self.assertNotEqual(e.errno, 0)
Benjamin Peterson7af65562008-12-29 17:56:58 +0000150 self.assertEqual(e.filename, ".")
Benjamin Petersonf22c26e2008-09-01 14:13:43 +0000151 else:
152 self.fail("Should have raised IOError")
153
Antoine Pitrouc2ec9922012-07-06 18:48:24 +0200154 @unittest.skipIf(os.name == 'nt', "test only works on a POSIX-like system")
155 def testOpenDirFD(self):
156 fd = os.open('.', os.O_RDONLY)
157 with self.assertRaises(IOError) as cm:
158 _FileIO(fd, 'r')
159 os.close(fd)
160 self.assertEqual(cm.exception.errno, errno.EISDIR)
161
Antoine Pitrou19690592009-06-12 20:14:08 +0000162 #A set of functions testing that we get expected behaviour if someone has
163 #manually closed the internal file descriptor. First, a decorator:
164 def ClosedFD(func):
165 @wraps(func)
166 def wrapper(self):
167 #forcibly close the fd before invoking the problem function
168 f = self.f
169 os.close(f.fileno())
170 try:
171 func(self, f)
172 finally:
173 try:
174 self.f.close()
175 except IOError:
176 pass
177 return wrapper
178
179 def ClosedFDRaises(func):
180 @wraps(func)
181 def wrapper(self):
182 #forcibly close the fd before invoking the problem function
183 f = self.f
184 os.close(f.fileno())
185 try:
186 func(self, f)
187 except IOError as e:
188 self.assertEqual(e.errno, errno.EBADF)
189 else:
190 self.fail("Should have raised IOError")
191 finally:
192 try:
193 self.f.close()
194 except IOError:
195 pass
196 return wrapper
197
198 @ClosedFDRaises
199 def testErrnoOnClose(self, f):
200 f.close()
201
202 @ClosedFDRaises
203 def testErrnoOnClosedWrite(self, f):
204 f.write('a')
205
206 @ClosedFDRaises
207 def testErrnoOnClosedSeek(self, f):
208 f.seek(0)
209
210 @ClosedFDRaises
211 def testErrnoOnClosedTell(self, f):
212 f.tell()
213
214 @ClosedFDRaises
215 def testErrnoOnClosedTruncate(self, f):
216 f.truncate(0)
217
218 @ClosedFD
219 def testErrnoOnClosedSeekable(self, f):
220 f.seekable()
221
222 @ClosedFD
223 def testErrnoOnClosedReadable(self, f):
224 f.readable()
225
226 @ClosedFD
227 def testErrnoOnClosedWritable(self, f):
228 f.writable()
229
230 @ClosedFD
231 def testErrnoOnClosedFileno(self, f):
232 f.fileno()
233
234 @ClosedFD
235 def testErrnoOnClosedIsatty(self, f):
236 self.assertEqual(f.isatty(), False)
237
238 def ReopenForRead(self):
239 try:
240 self.f.close()
241 except IOError:
242 pass
243 self.f = _FileIO(TESTFN, 'r')
244 os.close(self.f.fileno())
245 return self.f
246
247 @ClosedFDRaises
248 def testErrnoOnClosedRead(self, f):
249 f = self.ReopenForRead()
250 f.read(1)
251
252 @ClosedFDRaises
253 def testErrnoOnClosedReadall(self, f):
254 f = self.ReopenForRead()
255 f.readall()
256
257 @ClosedFDRaises
258 def testErrnoOnClosedReadinto(self, f):
259 f = self.ReopenForRead()
260 a = array(b'b', b'x'*10)
261 f.readinto(a)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000262
263class OtherFileTests(unittest.TestCase):
264
265 def testAbles(self):
266 try:
Antoine Pitrou19690592009-06-12 20:14:08 +0000267 f = _FileIO(TESTFN, "w")
Ezio Melotti2623a372010-11-21 13:34:58 +0000268 self.assertEqual(f.readable(), False)
269 self.assertEqual(f.writable(), True)
270 self.assertEqual(f.seekable(), True)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000271 f.close()
272
Antoine Pitrou19690592009-06-12 20:14:08 +0000273 f = _FileIO(TESTFN, "r")
Ezio Melotti2623a372010-11-21 13:34:58 +0000274 self.assertEqual(f.readable(), True)
275 self.assertEqual(f.writable(), False)
276 self.assertEqual(f.seekable(), True)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000277 f.close()
278
Antoine Pitrou19690592009-06-12 20:14:08 +0000279 f = _FileIO(TESTFN, "a+")
Ezio Melotti2623a372010-11-21 13:34:58 +0000280 self.assertEqual(f.readable(), True)
281 self.assertEqual(f.writable(), True)
282 self.assertEqual(f.seekable(), True)
283 self.assertEqual(f.isatty(), False)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000284 f.close()
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000285 finally:
286 os.unlink(TESTFN)
287
Zachary Ware1f702212013-12-10 14:09:20 -0600288 @unittest.skipIf(sys.platform == 'win32', 'no ttys on Windows')
289 def testAblesOnTTY(self):
290 try:
291 f = _FileIO("/dev/tty", "a")
292 except EnvironmentError:
293 # When run in a cron job there just aren't any
294 # ttys, so skip the test. This also handles other
295 # OS'es that don't support /dev/tty.
296 self.skipTest('need /dev/tty')
297 else:
298 self.assertEqual(f.readable(), False)
299 self.assertEqual(f.writable(), True)
300 if sys.platform != "darwin" and \
301 'bsd' not in sys.platform and \
302 not sys.platform.startswith('sunos'):
303 # Somehow /dev/tty appears seekable on some BSDs
304 self.assertEqual(f.seekable(), False)
305 self.assertEqual(f.isatty(), True)
306 f.close()
307
Antoine Pitrou213fec42013-09-04 20:46:33 +0200308 def testInvalidModeStrings(self):
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000309 # check invalid mode strings
Benjamin Petersonbfc51562008-11-22 01:59:15 +0000310 for mode in ("", "aU", "wU+", "rw", "rt"):
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000311 try:
Antoine Pitrou19690592009-06-12 20:14:08 +0000312 f = _FileIO(TESTFN, mode)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000313 except ValueError:
314 pass
315 else:
316 f.close()
317 self.fail('%r is an invalid file mode' % mode)
318
Antoine Pitrou213fec42013-09-04 20:46:33 +0200319 def testModeStrings(self):
320 # test that the mode attribute is correct for various mode strings
321 # given as init args
322 try:
323 for modes in [('w', 'wb'), ('wb', 'wb'), ('wb+', 'rb+'),
324 ('w+b', 'rb+'), ('a', 'ab'), ('ab', 'ab'),
325 ('ab+', 'ab+'), ('a+b', 'ab+'), ('r', 'rb'),
326 ('rb', 'rb'), ('rb+', 'rb+'), ('r+b', 'rb+')]:
327 # read modes are last so that TESTFN will exist first
328 with _FileIO(TESTFN, modes[0]) as f:
329 self.assertEqual(f.mode, modes[1])
330 finally:
331 if os.path.exists(TESTFN):
332 os.unlink(TESTFN)
333
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000334 def testUnicodeOpen(self):
335 # verify repr works for unicode too
Antoine Pitrou19690592009-06-12 20:14:08 +0000336 f = _FileIO(str(TESTFN), "w")
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000337 f.close()
338 os.unlink(TESTFN)
339
Antoine Pitrou19690592009-06-12 20:14:08 +0000340 def testBytesOpen(self):
341 # Opening a bytes filename
342 try:
343 fn = TESTFN.encode("ascii")
344 except UnicodeEncodeError:
Zachary Ware1f702212013-12-10 14:09:20 -0600345 self.skipTest('could not encode %r to ascii' % TESTFN)
Antoine Pitrou19690592009-06-12 20:14:08 +0000346 f = _FileIO(fn, "w")
347 try:
348 f.write(b"abc")
349 f.close()
350 with open(TESTFN, "rb") as f:
Ezio Melotti2623a372010-11-21 13:34:58 +0000351 self.assertEqual(f.read(), b"abc")
Antoine Pitrou19690592009-06-12 20:14:08 +0000352 finally:
353 os.unlink(TESTFN)
354
Benjamin Peterson5848d1f2009-01-19 00:08:08 +0000355 def testInvalidFd(self):
Antoine Pitrou19690592009-06-12 20:14:08 +0000356 self.assertRaises(ValueError, _FileIO, -10)
357 self.assertRaises(OSError, _FileIO, make_bad_fd())
Antoine Pitroue6ebcda2010-09-04 21:24:42 +0000358 if sys.platform == 'win32':
359 import msvcrt
360 self.assertRaises(IOError, msvcrt.get_osfhandle, make_bad_fd())
Serhiy Storchaka76249ea2014-02-07 10:06:05 +0200361
362 @cpython_only
363 def testInvalidFd_overflow(self):
Serhiy Storchaka74f49ab2013-01-19 12:55:39 +0200364 # Issue 15989
Serhiy Storchaka76249ea2014-02-07 10:06:05 +0200365 import _testcapi
Serhiy Storchaka74f49ab2013-01-19 12:55:39 +0200366 self.assertRaises(TypeError, _FileIO, _testcapi.INT_MAX + 1)
367 self.assertRaises(TypeError, _FileIO, _testcapi.INT_MIN - 1)
Benjamin Peterson5848d1f2009-01-19 00:08:08 +0000368
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000369 def testBadModeArgument(self):
370 # verify that we get a sensible error message for bad mode argument
371 bad_mode = "qwerty"
372 try:
Antoine Pitrou19690592009-06-12 20:14:08 +0000373 f = _FileIO(TESTFN, bad_mode)
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000374 except ValueError as msg:
375 if msg.args[0] != 0:
376 s = str(msg)
Ezio Melotti187f93d2010-03-17 14:22:34 +0000377 if TESTFN in s or bad_mode not in s:
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000378 self.fail("bad error message for invalid mode: %s" % s)
379 # if msg.args[0] == 0, we're probably on Windows where there may be
380 # no obvious way to discover why open() failed.
381 else:
382 f.close()
383 self.fail("no error for invalid mode: %s" % bad_mode)
384
Antoine Pitrouf3fa0742010-01-31 22:26:04 +0000385 def testTruncate(self):
386 f = _FileIO(TESTFN, 'w')
387 f.write(bytes(bytearray(range(10))))
388 self.assertEqual(f.tell(), 10)
389 f.truncate(5)
390 self.assertEqual(f.tell(), 10)
391 self.assertEqual(f.seek(0, os.SEEK_END), 5)
392 f.truncate(15)
393 self.assertEqual(f.tell(), 5)
394 self.assertEqual(f.seek(0, os.SEEK_END), 15)
Antoine Pitrou594a0462010-10-31 13:05:48 +0000395 f.close()
Antoine Pitrouf3fa0742010-01-31 22:26:04 +0000396
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000397 def testTruncateOnWindows(self):
398 def bug801631():
399 # SF bug <http://www.python.org/sf/801631>
400 # "file.truncate fault on windows"
Antoine Pitrou19690592009-06-12 20:14:08 +0000401 f = _FileIO(TESTFN, 'w')
402 f.write(bytes(range(11)))
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000403 f.close()
404
Antoine Pitrou19690592009-06-12 20:14:08 +0000405 f = _FileIO(TESTFN,'r+')
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000406 data = f.read(5)
Antoine Pitrou19690592009-06-12 20:14:08 +0000407 if data != bytes(range(5)):
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000408 self.fail("Read on file opened for update failed %r" % data)
409 if f.tell() != 5:
410 self.fail("File pos after read wrong %d" % f.tell())
411
412 f.truncate()
413 if f.tell() != 5:
414 self.fail("File pos after ftruncate wrong %d" % f.tell())
415
416 f.close()
417 size = os.path.getsize(TESTFN)
418 if size != 5:
419 self.fail("File size after ftruncate wrong %d" % size)
420
421 try:
422 bug801631()
423 finally:
424 os.unlink(TESTFN)
425
426 def testAppend(self):
427 try:
428 f = open(TESTFN, 'wb')
429 f.write(b'spam')
430 f.close()
431 f = open(TESTFN, 'ab')
432 f.write(b'eggs')
433 f.close()
434 f = open(TESTFN, 'rb')
435 d = f.read()
436 f.close()
437 self.assertEqual(d, b'spameggs')
438 finally:
439 try:
440 os.unlink(TESTFN)
441 except:
442 pass
443
Neal Norwitz901e4712008-08-24 22:03:05 +0000444 def testInvalidInit(self):
Antoine Pitrou19690592009-06-12 20:14:08 +0000445 self.assertRaises(TypeError, _FileIO, "1", 0, 0)
Neal Norwitz901e4712008-08-24 22:03:05 +0000446
Benjamin Peterson81e92e52008-11-03 22:34:57 +0000447 def testWarnings(self):
Florent Xicluna6de9e932010-03-07 12:18:33 +0000448 with check_warnings(quiet=True) as w:
Benjamin Peterson81e92e52008-11-03 22:34:57 +0000449 self.assertEqual(w.warnings, [])
Antoine Pitrou19690592009-06-12 20:14:08 +0000450 self.assertRaises(TypeError, _FileIO, [])
Benjamin Peterson81e92e52008-11-03 22:34:57 +0000451 self.assertEqual(w.warnings, [])
Antoine Pitrou19690592009-06-12 20:14:08 +0000452 self.assertRaises(ValueError, _FileIO, "/some/invalid/name", "rt")
Benjamin Peterson81e92e52008-11-03 22:34:57 +0000453 self.assertEqual(w.warnings, [])
454
Antoine Pitrou7c587bf2010-04-19 18:52:43 +0000455 def test_surrogates(self):
456 # Issue #8438: try to open a filename containing surrogates.
457 # It should either fail because the file doesn't exist or the filename
458 # can't be represented using the filesystem encoding, but not because
459 # of a LookupError for the error handler "surrogateescape".
460 filename = u'\udc80.txt'
461 try:
462 with _FileIO(filename):
463 pass
464 except (UnicodeEncodeError, IOError):
465 pass
466 # Spawn a separate Python process with a different "file system
467 # default encoding", to exercise this further.
468 env = dict(os.environ)
469 env[b'LC_CTYPE'] = b'C'
470 _, out = run_python('-c', 'import _io; _io.FileIO(%r)' % filename, env=env)
Ned Deilyaa1e1a22013-02-11 22:10:59 -0800471 if ('UnicodeEncodeError' not in out and not
472 ( ('IOError: [Errno 2] No such file or directory' in out) or
473 ('IOError: [Errno 22] Invalid argument' in out) ) ):
Antoine Pitrou7c587bf2010-04-19 18:52:43 +0000474 self.fail('Bad output: %r' % out)
Neal Norwitz901e4712008-08-24 22:03:05 +0000475
Hynek Schlawack9bd4bf22012-06-21 19:45:19 +0200476 def testUnclosedFDOnException(self):
477 class MyException(Exception): pass
478 class MyFileIO(_FileIO):
479 def __setattr__(self, name, value):
480 if name == "name":
481 raise MyException("blocked setting name")
482 return super(MyFileIO, self).__setattr__(name, value)
483 fd = os.open(__file__, os.O_RDONLY)
484 self.assertRaises(MyException, MyFileIO, fd)
485 os.close(fd) # should not raise OSError(EBADF)
486
Benjamin Peterson0e3a6cf2008-07-18 14:14:41 +0000487def test_main():
488 # Historically, these tests have been sloppy about removing TESTFN.
489 # So get rid of it no matter what.
490 try:
491 run_unittest(AutoFileTests, OtherFileTests)
492 finally:
493 if os.path.exists(TESTFN):
494 os.unlink(TESTFN)
495
496if __name__ == '__main__':
497 test_main()