blob: cdca5a8599655be99f57cc15a983f55feaed10ed [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
Eli Bendersky74c503b2012-01-03 06:26:13 +02005import io
Antoine Pitrou0ae29cf2009-03-13 22:33:17 +00006import errno
Guido van Rossuma9e20242007-03-08 00:43:48 +00007import unittest
8from array import array
9from weakref import proxy
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +000010from functools import wraps
Guido van Rossuma9e20242007-03-08 00:43:48 +000011
Serhiy Storchaka462c1f02021-09-08 18:08:57 +030012from test.support import run_unittest, cpython_only, swap_attr, gc_collect
Hai Shi883bc632020-07-06 17:12:49 +080013from test.support.os_helper import (TESTFN, TESTFN_UNICODE, make_bad_fd)
14from test.support.warnings_helper import check_warnings
Antoine Pitrou131a4892012-10-16 22:57:11 +020015from collections import UserList
Guido van Rossuma9e20242007-03-08 00:43:48 +000016
Serhiy Storchaka71fd2242015-04-10 16:16:16 +030017import _io # C implementation of io
18import _pyio # Python implementation of io
Guido van Rossuma9e20242007-03-08 00:43:48 +000019
Serhiy Storchaka71fd2242015-04-10 16:16:16 +030020
21class AutoFileTests:
Guido van Rossuma9e20242007-03-08 00:43:48 +000022 # file tests for which a test file is automatically set up
23
24 def setUp(self):
Serhiy Storchaka71fd2242015-04-10 16:16:16 +030025 self.f = self.FileIO(TESTFN, 'w')
Guido van Rossuma9e20242007-03-08 00:43:48 +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 Melottib3aedd42010-11-20 19:04:17 +000036 self.assertEqual(self.f.tell(), p.tell())
Guido van Rossuma9e20242007-03-08 00:43:48 +000037 self.f.close()
38 self.f = None
Serhiy Storchaka462c1f02021-09-08 18:08:57 +030039 gc_collect() # For PyPy or other GCs.
Guido van Rossuma9e20242007-03-08 00:43:48 +000040 self.assertRaises(ReferenceError, getattr, p, 'tell')
41
42 def testSeekTell(self):
43 self.f.write(bytes(range(20)))
Ezio Melottib3aedd42010-11-20 19:04:17 +000044 self.assertEqual(self.f.tell(), 20)
Guido van Rossuma9e20242007-03-08 00:43:48 +000045 self.f.seek(0)
Ezio Melottib3aedd42010-11-20 19:04:17 +000046 self.assertEqual(self.f.tell(), 0)
Guido van Rossuma9e20242007-03-08 00:43:48 +000047 self.f.seek(10)
Ezio Melottib3aedd42010-11-20 19:04:17 +000048 self.assertEqual(self.f.tell(), 10)
Guido van Rossuma9e20242007-03-08 00:43:48 +000049 self.f.seek(5, 1)
Ezio Melottib3aedd42010-11-20 19:04:17 +000050 self.assertEqual(self.f.tell(), 15)
Guido van Rossuma9e20242007-03-08 00:43:48 +000051 self.f.seek(-5, 1)
Ezio Melottib3aedd42010-11-20 19:04:17 +000052 self.assertEqual(self.f.tell(), 10)
Guido van Rossuma9e20242007-03-08 00:43:48 +000053 self.f.seek(-5, 2)
Ezio Melottib3aedd42010-11-20 19:04:17 +000054 self.assertEqual(self.f.tell(), 15)
Guido van Rossuma9e20242007-03-08 00:43:48 +000055
56 def testAttributes(self):
57 # verify expected attributes exist
58 f = self.f
Guido van Rossuma9e20242007-03-08 00:43:48 +000059
Ezio Melottib3aedd42010-11-20 19:04:17 +000060 self.assertEqual(f.mode, "wb")
61 self.assertEqual(f.closed, False)
Guido van Rossum31c3a572007-04-12 14:51:49 +000062
63 # verify the attributes are readonly
64 for attr in 'mode', 'closed':
65 self.assertRaises((AttributeError, TypeError),
66 setattr, f, attr, 'oops')
Guido van Rossuma9e20242007-03-08 00:43:48 +000067
Antoine Pitroude687222014-06-29 20:07:28 -040068 def testBlksize(self):
69 # test private _blksize attribute
70 blksize = io.DEFAULT_BUFFER_SIZE
71 # try to get preferred blksize from stat.st_blksize, if available
72 if hasattr(os, 'fstat'):
73 fst = os.fstat(self.f.fileno())
74 blksize = getattr(fst, 'st_blksize', blksize)
75 self.assertEqual(self.f._blksize, blksize)
76
Serhiy Storchaka71fd2242015-04-10 16:16:16 +030077 # verify readinto
78 def testReadintoByteArray(self):
79 self.f.write(bytes([1, 2, 0, 255]))
Guido van Rossuma9e20242007-03-08 00:43:48 +000080 self.f.close()
Serhiy Storchaka71fd2242015-04-10 16:16:16 +030081
82 ba = bytearray(b'abcdefgh')
83 with self.FileIO(TESTFN, 'r') as f:
84 n = f.readinto(ba)
85 self.assertEqual(ba, b'\x01\x02\x00\xffefgh')
86 self.assertEqual(n, 4)
87
88 def _testReadintoMemoryview(self):
89 self.f.write(bytes([1, 2, 0, 255]))
90 self.f.close()
91
92 m = memoryview(bytearray(b'abcdefgh'))
93 with self.FileIO(TESTFN, 'r') as f:
94 n = f.readinto(m)
95 self.assertEqual(m, b'\x01\x02\x00\xffefgh')
96 self.assertEqual(n, 4)
97
98 m = memoryview(bytearray(b'abcdefgh')).cast('H', shape=[2, 2])
99 with self.FileIO(TESTFN, 'r') as f:
100 n = f.readinto(m)
101 self.assertEqual(bytes(m), b'\x01\x02\x00\xffefgh')
102 self.assertEqual(n, 4)
103
104 def _testReadintoArray(self):
105 self.f.write(bytes([1, 2, 0, 255]))
106 self.f.close()
107
108 a = array('B', b'abcdefgh')
109 with self.FileIO(TESTFN, 'r') as f:
110 n = f.readinto(a)
111 self.assertEqual(a, array('B', [1, 2, 0, 255, 101, 102, 103, 104]))
112 self.assertEqual(n, 4)
113
114 a = array('b', b'abcdefgh')
115 with self.FileIO(TESTFN, 'r') as f:
116 n = f.readinto(a)
117 self.assertEqual(a, array('b', [1, 2, 0, -1, 101, 102, 103, 104]))
118 self.assertEqual(n, 4)
119
120 a = array('I', b'abcdefgh')
121 with self.FileIO(TESTFN, 'r') as f:
122 n = f.readinto(a)
123 self.assertEqual(a, array('I', b'\x01\x02\x00\xffefgh'))
124 self.assertEqual(n, 4)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000125
Antoine Pitrou131a4892012-10-16 22:57:11 +0200126 def testWritelinesList(self):
127 l = [b'123', b'456']
128 self.f.writelines(l)
129 self.f.close()
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300130 self.f = self.FileIO(TESTFN, 'rb')
Antoine Pitrou131a4892012-10-16 22:57:11 +0200131 buf = self.f.read()
132 self.assertEqual(buf, b'123456')
133
134 def testWritelinesUserList(self):
135 l = UserList([b'123', b'456'])
136 self.f.writelines(l)
137 self.f.close()
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300138 self.f = self.FileIO(TESTFN, 'rb')
Antoine Pitrou131a4892012-10-16 22:57:11 +0200139 buf = self.f.read()
140 self.assertEqual(buf, b'123456')
141
142 def testWritelinesError(self):
143 self.assertRaises(TypeError, self.f.writelines, [1, 2, 3])
144 self.assertRaises(TypeError, self.f.writelines, None)
145 self.assertRaises(TypeError, self.f.writelines, "abc")
146
Benjamin Petersonbf5ff762009-12-13 19:25:34 +0000147 def test_none_args(self):
148 self.f.write(b"hi\nbye\nabc")
149 self.f.close()
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300150 self.f = self.FileIO(TESTFN, 'r')
Benjamin Petersonbf5ff762009-12-13 19:25:34 +0000151 self.assertEqual(self.f.read(None), b"hi\nbye\nabc")
152 self.f.seek(0)
153 self.assertEqual(self.f.readline(None), b"hi\n")
154 self.assertEqual(self.f.readlines(None), [b"bye\n", b"abc"])
155
Benjamin Peterson255058f2010-01-27 01:47:14 +0000156 def test_reject(self):
157 self.assertRaises(TypeError, self.f.write, "Hello!")
158
Guido van Rossuma9e20242007-03-08 00:43:48 +0000159 def testRepr(self):
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300160 self.assertEqual(repr(self.f),
161 "<%s.FileIO name=%r mode=%r closefd=True>" %
162 (self.modulename, self.f.name, self.f.mode))
Antoine Pitrou716c4442009-05-23 19:04:03 +0000163 del self.f.name
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300164 self.assertEqual(repr(self.f),
165 "<%s.FileIO fd=%r mode=%r closefd=True>" %
166 (self.modulename, self.f.fileno(), self.f.mode))
Antoine Pitrou716c4442009-05-23 19:04:03 +0000167 self.f.close()
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300168 self.assertEqual(repr(self.f),
169 "<%s.FileIO [closed]>" % (self.modulename,))
Guido van Rossuma9e20242007-03-08 00:43:48 +0000170
Serhiy Storchaka4954f9f2014-12-02 23:39:56 +0200171 def testReprNoCloseFD(self):
172 fd = os.open(TESTFN, os.O_RDONLY)
173 try:
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300174 with self.FileIO(fd, 'r', closefd=False) as f:
Serhiy Storchaka4954f9f2014-12-02 23:39:56 +0200175 self.assertEqual(repr(f),
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300176 "<%s.FileIO name=%r mode=%r closefd=False>" %
177 (self.modulename, f.name, f.mode))
Serhiy Storchaka4954f9f2014-12-02 23:39:56 +0200178 finally:
179 os.close(fd)
180
Serhiy Storchakaa5af6e12017-03-19 19:25:29 +0200181 def testRecursiveRepr(self):
182 # Issue #25455
183 with swap_attr(self.f, 'name', self.f):
184 with self.assertRaises(RuntimeError):
185 repr(self.f) # Should not crash
186
Guido van Rossuma9e20242007-03-08 00:43:48 +0000187 def testErrors(self):
188 f = self.f
Serhiy Storchaka0dcd80a2015-08-02 15:17:49 +0300189 self.assertFalse(f.isatty())
190 self.assertFalse(f.closed)
Ezio Melottib3aedd42010-11-20 19:04:17 +0000191 #self.assertEqual(f.name, TESTFN)
Guido van Rossum31c3a572007-04-12 14:51:49 +0000192 self.assertRaises(ValueError, f.read, 10) # Open for reading
Guido van Rossuma9e20242007-03-08 00:43:48 +0000193 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000194 self.assertTrue(f.closed)
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300195 f = self.FileIO(TESTFN, 'r')
Guido van Rossum31c3a572007-04-12 14:51:49 +0000196 self.assertRaises(TypeError, f.readinto, "")
Serhiy Storchaka0dcd80a2015-08-02 15:17:49 +0300197 self.assertFalse(f.closed)
Guido van Rossum31c3a572007-04-12 14:51:49 +0000198 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000199 self.assertTrue(f.closed)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000200
201 def testMethods(self):
Serhiy Storchakaf24131f2015-04-16 11:19:43 +0300202 methods = ['fileno', 'isatty', 'seekable', 'readable', 'writable',
203 'read', 'readall', 'readline', 'readlines',
204 'tell', 'truncate', 'flush']
Guido van Rossuma9e20242007-03-08 00:43:48 +0000205
Guido van Rossum31c3a572007-04-12 14:51:49 +0000206 self.f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000207 self.assertTrue(self.f.closed)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000208
209 for methodname in methods:
210 method = getattr(self.f, methodname)
211 # should raise on closed file
212 self.assertRaises(ValueError, method)
Serhiy Storchakaf24131f2015-04-16 11:19:43 +0300213
214 self.assertRaises(TypeError, self.f.readinto)
215 self.assertRaises(ValueError, self.f.readinto, bytearray(1))
216 self.assertRaises(TypeError, self.f.seek)
217 self.assertRaises(ValueError, self.f.seek, 0)
218 self.assertRaises(TypeError, self.f.write)
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300219 self.assertRaises(ValueError, self.f.write, b'')
Serhiy Storchakaf24131f2015-04-16 11:19:43 +0300220 self.assertRaises(TypeError, self.f.writelines)
221 self.assertRaises(ValueError, self.f.writelines, b'')
Guido van Rossuma9e20242007-03-08 00:43:48 +0000222
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000223 def testOpendir(self):
224 # Issue 3703: opening a directory should fill the errno
225 # Windows always returns "[Errno 13]: Permission denied
Antoine Pitroude687222014-06-29 20:07:28 -0400226 # Unix uses fstat and returns "[Errno 21]: Is a directory"
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000227 try:
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300228 self.FileIO('.', 'r')
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200229 except OSError as e:
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000230 self.assertNotEqual(e.errno, 0)
Benjamin Peterson1efc23c2008-12-29 18:02:28 +0000231 self.assertEqual(e.filename, ".")
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000232 else:
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200233 self.fail("Should have raised OSError")
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000234
Antoine Pitrou9235b252012-07-06 18:48:24 +0200235 @unittest.skipIf(os.name == 'nt', "test only works on a POSIX-like system")
236 def testOpenDirFD(self):
237 fd = os.open('.', os.O_RDONLY)
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200238 with self.assertRaises(OSError) as cm:
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300239 self.FileIO(fd, 'r')
Antoine Pitrou9235b252012-07-06 18:48:24 +0200240 os.close(fd)
241 self.assertEqual(cm.exception.errno, errno.EISDIR)
242
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000243 #A set of functions testing that we get expected behaviour if someone has
244 #manually closed the internal file descriptor. First, a decorator:
245 def ClosedFD(func):
246 @wraps(func)
247 def wrapper(self):
248 #forcibly close the fd before invoking the problem function
249 f = self.f
250 os.close(f.fileno())
251 try:
252 func(self, f)
253 finally:
254 try:
255 self.f.close()
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200256 except OSError:
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000257 pass
258 return wrapper
Antoine Pitrou0ae29cf2009-03-13 22:33:17 +0000259
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000260 def ClosedFDRaises(func):
261 @wraps(func)
262 def wrapper(self):
263 #forcibly close the fd before invoking the problem function
264 f = self.f
265 os.close(f.fileno())
266 try:
267 func(self, f)
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200268 except OSError as e:
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000269 self.assertEqual(e.errno, errno.EBADF)
270 else:
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200271 self.fail("Should have raised OSError")
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000272 finally:
273 try:
274 self.f.close()
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200275 except OSError:
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000276 pass
277 return wrapper
278
279 @ClosedFDRaises
280 def testErrnoOnClose(self, f):
281 f.close()
282
283 @ClosedFDRaises
284 def testErrnoOnClosedWrite(self, f):
Benjamin Peterson255058f2010-01-27 01:47:14 +0000285 f.write(b'a')
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000286
287 @ClosedFDRaises
288 def testErrnoOnClosedSeek(self, f):
289 f.seek(0)
290
291 @ClosedFDRaises
292 def testErrnoOnClosedTell(self, f):
293 f.tell()
294
295 @ClosedFDRaises
296 def testErrnoOnClosedTruncate(self, f):
297 f.truncate(0)
298
299 @ClosedFD
300 def testErrnoOnClosedSeekable(self, f):
301 f.seekable()
302
303 @ClosedFD
304 def testErrnoOnClosedReadable(self, f):
305 f.readable()
306
307 @ClosedFD
308 def testErrnoOnClosedWritable(self, f):
309 f.writable()
310
311 @ClosedFD
312 def testErrnoOnClosedFileno(self, f):
313 f.fileno()
314
315 @ClosedFD
316 def testErrnoOnClosedIsatty(self, f):
317 self.assertEqual(f.isatty(), False)
318
319 def ReopenForRead(self):
320 try:
321 self.f.close()
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200322 except OSError:
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000323 pass
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300324 self.f = self.FileIO(TESTFN, 'r')
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000325 os.close(self.f.fileno())
326 return self.f
327
328 @ClosedFDRaises
329 def testErrnoOnClosedRead(self, f):
330 f = self.ReopenForRead()
331 f.read(1)
332
333 @ClosedFDRaises
334 def testErrnoOnClosedReadall(self, f):
335 f = self.ReopenForRead()
336 f.readall()
337
338 @ClosedFDRaises
339 def testErrnoOnClosedReadinto(self, f):
340 f = self.ReopenForRead()
341 a = array('b', b'x'*10)
342 f.readinto(a)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000343
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300344class CAutoFileTests(AutoFileTests, unittest.TestCase):
345 FileIO = _io.FileIO
346 modulename = '_io'
347
348class PyAutoFileTests(AutoFileTests, unittest.TestCase):
349 FileIO = _pyio.FileIO
350 modulename = '_pyio'
351
352
353class OtherFileTests:
Guido van Rossuma9e20242007-03-08 00:43:48 +0000354
355 def testAbles(self):
356 try:
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300357 f = self.FileIO(TESTFN, "w")
Ezio Melottib3aedd42010-11-20 19:04:17 +0000358 self.assertEqual(f.readable(), False)
359 self.assertEqual(f.writable(), True)
360 self.assertEqual(f.seekable(), True)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000361 f.close()
Guido van Rossumd8faa362007-04-27 19:54:29 +0000362
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300363 f = self.FileIO(TESTFN, "r")
Ezio Melottib3aedd42010-11-20 19:04:17 +0000364 self.assertEqual(f.readable(), True)
365 self.assertEqual(f.writable(), False)
366 self.assertEqual(f.seekable(), True)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000367 f.close()
Guido van Rossumd8faa362007-04-27 19:54:29 +0000368
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300369 f = self.FileIO(TESTFN, "a+")
Ezio Melottib3aedd42010-11-20 19:04:17 +0000370 self.assertEqual(f.readable(), True)
371 self.assertEqual(f.writable(), True)
372 self.assertEqual(f.seekable(), True)
373 self.assertEqual(f.isatty(), False)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000374 f.close()
Guido van Rossum682faf82007-04-12 14:56:58 +0000375
Thomas Hellerb0f48ab2007-07-12 11:29:02 +0000376 if sys.platform != "win32":
377 try:
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300378 f = self.FileIO("/dev/tty", "a")
Andrew Svetlov3438fa42012-12-17 23:35:18 +0200379 except OSError:
Thomas Hellerb0f48ab2007-07-12 11:29:02 +0000380 # When run in a cron job there just aren't any
381 # ttys, so skip the test. This also handles other
382 # OS'es that don't support /dev/tty.
383 pass
384 else:
Ezio Melottib3aedd42010-11-20 19:04:17 +0000385 self.assertEqual(f.readable(), False)
386 self.assertEqual(f.writable(), True)
Hye-Shik Changcaf871a2007-08-13 13:21:33 +0000387 if sys.platform != "darwin" and \
Antoine Pitroud7b30462009-05-23 16:34:50 +0000388 'bsd' not in sys.platform and \
Martin Panter9544a362016-11-14 01:58:57 +0000389 not sys.platform.startswith(('sunos', 'aix')):
Hye-Shik Changcaf871a2007-08-13 13:21:33 +0000390 # Somehow /dev/tty appears seekable on some BSDs
Ezio Melottib3aedd42010-11-20 19:04:17 +0000391 self.assertEqual(f.seekable(), False)
392 self.assertEqual(f.isatty(), True)
Thomas Hellerb0f48ab2007-07-12 11:29:02 +0000393 f.close()
Guido van Rossuma9e20242007-03-08 00:43:48 +0000394 finally:
395 os.unlink(TESTFN)
396
Antoine Pitroue93b63b2013-09-04 20:46:33 +0200397 def testInvalidModeStrings(self):
Guido van Rossuma9e20242007-03-08 00:43:48 +0000398 # check invalid mode strings
Benjamin Peterson44309e62008-11-22 00:41:45 +0000399 for mode in ("", "aU", "wU+", "rw", "rt"):
Guido van Rossuma9e20242007-03-08 00:43:48 +0000400 try:
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300401 f = self.FileIO(TESTFN, mode)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000402 except ValueError:
403 pass
404 else:
405 f.close()
406 self.fail('%r is an invalid file mode' % mode)
407
Antoine Pitroue93b63b2013-09-04 20:46:33 +0200408 def testModeStrings(self):
409 # test that the mode attribute is correct for various mode strings
410 # given as init args
411 try:
412 for modes in [('w', 'wb'), ('wb', 'wb'), ('wb+', 'rb+'),
413 ('w+b', 'rb+'), ('a', 'ab'), ('ab', 'ab'),
414 ('ab+', 'ab+'), ('a+b', 'ab+'), ('r', 'rb'),
415 ('rb', 'rb'), ('rb+', 'rb+'), ('r+b', 'rb+')]:
416 # read modes are last so that TESTFN will exist first
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300417 with self.FileIO(TESTFN, modes[0]) as f:
Antoine Pitroue93b63b2013-09-04 20:46:33 +0200418 self.assertEqual(f.mode, modes[1])
419 finally:
420 if os.path.exists(TESTFN):
421 os.unlink(TESTFN)
422
Guido van Rossuma9e20242007-03-08 00:43:48 +0000423 def testUnicodeOpen(self):
424 # verify repr works for unicode too
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300425 f = self.FileIO(str(TESTFN), "w")
Guido van Rossuma9e20242007-03-08 00:43:48 +0000426 f.close()
427 os.unlink(TESTFN)
428
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000429 def testBytesOpen(self):
430 # Opening a bytes filename
431 try:
432 fn = TESTFN.encode("ascii")
433 except UnicodeEncodeError:
Zachary Ware9fe6d862013-12-08 00:20:35 -0600434 self.skipTest('could not encode %r to ascii' % TESTFN)
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300435 f = self.FileIO(fn, "w")
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000436 try:
437 f.write(b"abc")
438 f.close()
439 with open(TESTFN, "rb") as f:
Ezio Melottib3aedd42010-11-20 19:04:17 +0000440 self.assertEqual(f.read(), b"abc")
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000441 finally:
442 os.unlink(TESTFN)
443
Steve Dowereacee982017-02-04 14:38:11 -0800444 @unittest.skipIf(sys.getfilesystemencoding() != 'utf-8',
445 "test only works for utf-8 filesystems")
446 def testUtf8BytesOpen(self):
447 # Opening a UTF-8 bytes filename
448 try:
449 fn = TESTFN_UNICODE.encode("utf-8")
450 except UnicodeEncodeError:
451 self.skipTest('could not encode %r to utf-8' % TESTFN_UNICODE)
452 f = self.FileIO(fn, "w")
453 try:
454 f.write(b"abc")
455 f.close()
456 with open(TESTFN_UNICODE, "rb") as f:
457 self.assertEqual(f.read(), b"abc")
458 finally:
459 os.unlink(TESTFN_UNICODE)
460
Antoine Pitrou13348842012-01-29 18:36:34 +0100461 def testConstructorHandlesNULChars(self):
462 fn_with_NUL = 'foo\0bar'
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300463 self.assertRaises(ValueError, self.FileIO, fn_with_NUL, 'w')
464 self.assertRaises(ValueError, self.FileIO, bytes(fn_with_NUL, 'ascii'), 'w')
Antoine Pitrou13348842012-01-29 18:36:34 +0100465
Benjamin Peterson806d4022009-01-19 15:11:51 +0000466 def testInvalidFd(self):
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300467 self.assertRaises(ValueError, self.FileIO, -10)
468 self.assertRaises(OSError, self.FileIO, make_bad_fd())
Antoine Pitrou00492492010-09-04 20:53:29 +0000469 if sys.platform == 'win32':
470 import msvcrt
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200471 self.assertRaises(OSError, msvcrt.get_osfhandle, make_bad_fd())
Serhiy Storchaka5cfc79d2014-02-07 10:06:39 +0200472
Guido van Rossuma9e20242007-03-08 00:43:48 +0000473 def testBadModeArgument(self):
474 # verify that we get a sensible error message for bad mode argument
475 bad_mode = "qwerty"
476 try:
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300477 f = self.FileIO(TESTFN, bad_mode)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000478 except ValueError as msg:
Guido van Rossum360e4b82007-05-14 22:51:27 +0000479 if msg.args[0] != 0:
Guido van Rossuma9e20242007-03-08 00:43:48 +0000480 s = str(msg)
Ezio Melotti7fb4da72010-03-18 12:29:13 +0000481 if TESTFN in s or bad_mode not in s:
Guido van Rossuma9e20242007-03-08 00:43:48 +0000482 self.fail("bad error message for invalid mode: %s" % s)
Georg Brandl50da60c2008-01-06 21:38:54 +0000483 # if msg.args[0] == 0, we're probably on Windows where there may be
Guido van Rossuma9e20242007-03-08 00:43:48 +0000484 # no obvious way to discover why open() failed.
485 else:
486 f.close()
487 self.fail("no error for invalid mode: %s" % bad_mode)
488
Antoine Pitrou905a2ff2010-01-31 22:47:27 +0000489 def testTruncate(self):
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300490 f = self.FileIO(TESTFN, 'w')
Antoine Pitrou905a2ff2010-01-31 22:47:27 +0000491 f.write(bytes(bytearray(range(10))))
492 self.assertEqual(f.tell(), 10)
493 f.truncate(5)
494 self.assertEqual(f.tell(), 10)
Eli Bendersky74c503b2012-01-03 06:26:13 +0200495 self.assertEqual(f.seek(0, io.SEEK_END), 5)
Antoine Pitrou905a2ff2010-01-31 22:47:27 +0000496 f.truncate(15)
497 self.assertEqual(f.tell(), 5)
Eli Bendersky74c503b2012-01-03 06:26:13 +0200498 self.assertEqual(f.seek(0, io.SEEK_END), 15)
Antoine Pitrou8d2b51b2010-10-30 16:19:14 +0000499 f.close()
Antoine Pitrou905a2ff2010-01-31 22:47:27 +0000500
Guido van Rossuma9e20242007-03-08 00:43:48 +0000501 def testTruncateOnWindows(self):
502 def bug801631():
503 # SF bug <http://www.python.org/sf/801631>
504 # "file.truncate fault on windows"
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300505 f = self.FileIO(TESTFN, 'w')
Guido van Rossuma9e20242007-03-08 00:43:48 +0000506 f.write(bytes(range(11)))
507 f.close()
508
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300509 f = self.FileIO(TESTFN,'r+')
Guido van Rossuma9e20242007-03-08 00:43:48 +0000510 data = f.read(5)
511 if data != bytes(range(5)):
512 self.fail("Read on file opened for update failed %r" % data)
513 if f.tell() != 5:
514 self.fail("File pos after read wrong %d" % f.tell())
515
516 f.truncate()
517 if f.tell() != 5:
518 self.fail("File pos after ftruncate wrong %d" % f.tell())
519
520 f.close()
521 size = os.path.getsize(TESTFN)
522 if size != 5:
523 self.fail("File size after ftruncate wrong %d" % size)
524
525 try:
526 bug801631()
527 finally:
528 os.unlink(TESTFN)
529
Walter Dörwald3a77c7a2007-06-06 16:31:14 +0000530 def testAppend(self):
531 try:
532 f = open(TESTFN, 'wb')
533 f.write(b'spam')
534 f.close()
535 f = open(TESTFN, 'ab')
536 f.write(b'eggs')
537 f.close()
538 f = open(TESTFN, 'rb')
539 d = f.read()
540 f.close()
541 self.assertEqual(d, b'spameggs')
542 finally:
543 try:
544 os.unlink(TESTFN)
545 except:
546 pass
547
Neal Norwitz6e0e0e62008-08-24 22:07:28 +0000548 def testInvalidInit(self):
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300549 self.assertRaises(TypeError, self.FileIO, "1", 0, 0)
Neal Norwitz6e0e0e62008-08-24 22:07:28 +0000550
Benjamin Peterson65676e42008-11-05 21:42:45 +0000551 def testWarnings(self):
Florent Xiclunab14930c2010-03-13 15:26:44 +0000552 with check_warnings(quiet=True) as w:
Benjamin Peterson65676e42008-11-05 21:42:45 +0000553 self.assertEqual(w.warnings, [])
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300554 self.assertRaises(TypeError, self.FileIO, [])
Benjamin Peterson65676e42008-11-05 21:42:45 +0000555 self.assertEqual(w.warnings, [])
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300556 self.assertRaises(ValueError, self.FileIO, "/some/invalid/name", "rt")
Benjamin Peterson65676e42008-11-05 21:42:45 +0000557 self.assertEqual(w.warnings, [])
558
Hynek Schlawack9ed8b4e2012-06-21 20:20:25 +0200559 def testUnclosedFDOnException(self):
560 class MyException(Exception): pass
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300561 class MyFileIO(self.FileIO):
Hynek Schlawack9ed8b4e2012-06-21 20:20:25 +0200562 def __setattr__(self, name, value):
563 if name == "name":
564 raise MyException("blocked setting name")
565 return super(MyFileIO, self).__setattr__(name, value)
566 fd = os.open(__file__, os.O_RDONLY)
567 self.assertRaises(MyException, MyFileIO, fd)
568 os.close(fd) # should not raise OSError(EBADF)
569
Steve Dowerb82e17e2019-05-23 08:45:22 -0700570
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300571class COtherFileTests(OtherFileTests, unittest.TestCase):
572 FileIO = _io.FileIO
573 modulename = '_io'
574
575 @cpython_only
576 def testInvalidFd_overflow(self):
577 # Issue 15989
578 import _testcapi
579 self.assertRaises(TypeError, self.FileIO, _testcapi.INT_MAX + 1)
580 self.assertRaises(TypeError, self.FileIO, _testcapi.INT_MIN - 1)
581
Steve Dowerb82e17e2019-05-23 08:45:22 -0700582 def test_open_code(self):
583 # Check that the default behaviour of open_code matches
584 # open("rb")
585 with self.FileIO(__file__, "rb") as f:
586 expected = f.read()
587 with _io.open_code(__file__) as f:
588 actual = f.read()
589 self.assertEqual(expected, actual)
590
591
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300592class PyOtherFileTests(OtherFileTests, unittest.TestCase):
593 FileIO = _pyio.FileIO
594 modulename = '_pyio'
595
Steve Dowerb82e17e2019-05-23 08:45:22 -0700596 def test_open_code(self):
597 # Check that the default behaviour of open_code matches
598 # open("rb")
599 with self.FileIO(__file__, "rb") as f:
600 expected = f.read()
601 with check_warnings(quiet=True) as w:
602 # Always test _open_code_with_warning
603 with _pyio._open_code_with_warning(__file__) as f:
604 actual = f.read()
605 self.assertEqual(expected, actual)
606 self.assertNotEqual(w.warnings, [])
607
Neal Norwitz6e0e0e62008-08-24 22:07:28 +0000608
Guido van Rossuma9e20242007-03-08 00:43:48 +0000609def test_main():
610 # Historically, these tests have been sloppy about removing TESTFN.
611 # So get rid of it no matter what.
612 try:
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300613 run_unittest(CAutoFileTests, PyAutoFileTests,
614 COtherFileTests, PyOtherFileTests)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000615 finally:
616 if os.path.exists(TESTFN):
617 os.unlink(TESTFN)
618
619if __name__ == '__main__':
620 test_main()