blob: 26e4500ae8cfcd763d96989c25682a9a672b1d8b [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
Steve Dowereacee982017-02-04 14:38:11 -080012from test.support import (TESTFN, TESTFN_UNICODE, check_warnings, run_unittest,
Serhiy Storchakaa5af6e12017-03-19 19:25:29 +020013 make_bad_fd, cpython_only, swap_attr)
Antoine Pitrou131a4892012-10-16 22:57:11 +020014from collections import UserList
Guido van Rossuma9e20242007-03-08 00:43:48 +000015
Serhiy Storchaka71fd2242015-04-10 16:16:16 +030016import _io # C implementation of io
17import _pyio # Python implementation of io
Guido van Rossuma9e20242007-03-08 00:43:48 +000018
Serhiy Storchaka71fd2242015-04-10 16:16:16 +030019
20class AutoFileTests:
Guido van Rossuma9e20242007-03-08 00:43:48 +000021 # file tests for which a test file is automatically set up
22
23 def setUp(self):
Serhiy Storchaka71fd2242015-04-10 16:16:16 +030024 self.f = self.FileIO(TESTFN, 'w')
Guido van Rossuma9e20242007-03-08 00:43:48 +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 Melottib3aedd42010-11-20 19:04:17 +000035 self.assertEqual(self.f.tell(), p.tell())
Guido van Rossuma9e20242007-03-08 00:43:48 +000036 self.f.close()
37 self.f = None
38 self.assertRaises(ReferenceError, getattr, p, 'tell')
39
40 def testSeekTell(self):
41 self.f.write(bytes(range(20)))
Ezio Melottib3aedd42010-11-20 19:04:17 +000042 self.assertEqual(self.f.tell(), 20)
Guido van Rossuma9e20242007-03-08 00:43:48 +000043 self.f.seek(0)
Ezio Melottib3aedd42010-11-20 19:04:17 +000044 self.assertEqual(self.f.tell(), 0)
Guido van Rossuma9e20242007-03-08 00:43:48 +000045 self.f.seek(10)
Ezio Melottib3aedd42010-11-20 19:04:17 +000046 self.assertEqual(self.f.tell(), 10)
Guido van Rossuma9e20242007-03-08 00:43:48 +000047 self.f.seek(5, 1)
Ezio Melottib3aedd42010-11-20 19:04:17 +000048 self.assertEqual(self.f.tell(), 15)
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(), 10)
Guido van Rossuma9e20242007-03-08 00:43:48 +000051 self.f.seek(-5, 2)
Ezio Melottib3aedd42010-11-20 19:04:17 +000052 self.assertEqual(self.f.tell(), 15)
Guido van Rossuma9e20242007-03-08 00:43:48 +000053
54 def testAttributes(self):
55 # verify expected attributes exist
56 f = self.f
Guido van Rossuma9e20242007-03-08 00:43:48 +000057
Ezio Melottib3aedd42010-11-20 19:04:17 +000058 self.assertEqual(f.mode, "wb")
59 self.assertEqual(f.closed, False)
Guido van Rossum31c3a572007-04-12 14:51:49 +000060
61 # verify the attributes are readonly
62 for attr in 'mode', 'closed':
63 self.assertRaises((AttributeError, TypeError),
64 setattr, f, attr, 'oops')
Guido van Rossuma9e20242007-03-08 00:43:48 +000065
Antoine Pitroude687222014-06-29 20:07:28 -040066 def testBlksize(self):
67 # test private _blksize attribute
68 blksize = io.DEFAULT_BUFFER_SIZE
69 # try to get preferred blksize from stat.st_blksize, if available
70 if hasattr(os, 'fstat'):
71 fst = os.fstat(self.f.fileno())
72 blksize = getattr(fst, 'st_blksize', blksize)
73 self.assertEqual(self.f._blksize, blksize)
74
Serhiy Storchaka71fd2242015-04-10 16:16:16 +030075 # verify readinto
76 def testReadintoByteArray(self):
77 self.f.write(bytes([1, 2, 0, 255]))
Guido van Rossuma9e20242007-03-08 00:43:48 +000078 self.f.close()
Serhiy Storchaka71fd2242015-04-10 16:16:16 +030079
80 ba = bytearray(b'abcdefgh')
81 with self.FileIO(TESTFN, 'r') as f:
82 n = f.readinto(ba)
83 self.assertEqual(ba, b'\x01\x02\x00\xffefgh')
84 self.assertEqual(n, 4)
85
86 def _testReadintoMemoryview(self):
87 self.f.write(bytes([1, 2, 0, 255]))
88 self.f.close()
89
90 m = memoryview(bytearray(b'abcdefgh'))
91 with self.FileIO(TESTFN, 'r') as f:
92 n = f.readinto(m)
93 self.assertEqual(m, b'\x01\x02\x00\xffefgh')
94 self.assertEqual(n, 4)
95
96 m = memoryview(bytearray(b'abcdefgh')).cast('H', shape=[2, 2])
97 with self.FileIO(TESTFN, 'r') as f:
98 n = f.readinto(m)
99 self.assertEqual(bytes(m), b'\x01\x02\x00\xffefgh')
100 self.assertEqual(n, 4)
101
102 def _testReadintoArray(self):
103 self.f.write(bytes([1, 2, 0, 255]))
104 self.f.close()
105
106 a = array('B', b'abcdefgh')
107 with self.FileIO(TESTFN, 'r') as f:
108 n = f.readinto(a)
109 self.assertEqual(a, array('B', [1, 2, 0, 255, 101, 102, 103, 104]))
110 self.assertEqual(n, 4)
111
112 a = array('b', b'abcdefgh')
113 with self.FileIO(TESTFN, 'r') as f:
114 n = f.readinto(a)
115 self.assertEqual(a, array('b', [1, 2, 0, -1, 101, 102, 103, 104]))
116 self.assertEqual(n, 4)
117
118 a = array('I', b'abcdefgh')
119 with self.FileIO(TESTFN, 'r') as f:
120 n = f.readinto(a)
121 self.assertEqual(a, array('I', b'\x01\x02\x00\xffefgh'))
122 self.assertEqual(n, 4)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000123
Antoine Pitrou131a4892012-10-16 22:57:11 +0200124 def testWritelinesList(self):
125 l = [b'123', b'456']
126 self.f.writelines(l)
127 self.f.close()
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300128 self.f = self.FileIO(TESTFN, 'rb')
Antoine Pitrou131a4892012-10-16 22:57:11 +0200129 buf = self.f.read()
130 self.assertEqual(buf, b'123456')
131
132 def testWritelinesUserList(self):
133 l = UserList([b'123', b'456'])
134 self.f.writelines(l)
135 self.f.close()
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300136 self.f = self.FileIO(TESTFN, 'rb')
Antoine Pitrou131a4892012-10-16 22:57:11 +0200137 buf = self.f.read()
138 self.assertEqual(buf, b'123456')
139
140 def testWritelinesError(self):
141 self.assertRaises(TypeError, self.f.writelines, [1, 2, 3])
142 self.assertRaises(TypeError, self.f.writelines, None)
143 self.assertRaises(TypeError, self.f.writelines, "abc")
144
Benjamin Petersonbf5ff762009-12-13 19:25:34 +0000145 def test_none_args(self):
146 self.f.write(b"hi\nbye\nabc")
147 self.f.close()
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300148 self.f = self.FileIO(TESTFN, 'r')
Benjamin Petersonbf5ff762009-12-13 19:25:34 +0000149 self.assertEqual(self.f.read(None), b"hi\nbye\nabc")
150 self.f.seek(0)
151 self.assertEqual(self.f.readline(None), b"hi\n")
152 self.assertEqual(self.f.readlines(None), [b"bye\n", b"abc"])
153
Benjamin Peterson255058f2010-01-27 01:47:14 +0000154 def test_reject(self):
155 self.assertRaises(TypeError, self.f.write, "Hello!")
156
Guido van Rossuma9e20242007-03-08 00:43:48 +0000157 def testRepr(self):
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300158 self.assertEqual(repr(self.f),
159 "<%s.FileIO name=%r mode=%r closefd=True>" %
160 (self.modulename, self.f.name, self.f.mode))
Antoine Pitrou716c4442009-05-23 19:04:03 +0000161 del self.f.name
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300162 self.assertEqual(repr(self.f),
163 "<%s.FileIO fd=%r mode=%r closefd=True>" %
164 (self.modulename, self.f.fileno(), self.f.mode))
Antoine Pitrou716c4442009-05-23 19:04:03 +0000165 self.f.close()
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300166 self.assertEqual(repr(self.f),
167 "<%s.FileIO [closed]>" % (self.modulename,))
Guido van Rossuma9e20242007-03-08 00:43:48 +0000168
Serhiy Storchaka4954f9f2014-12-02 23:39:56 +0200169 def testReprNoCloseFD(self):
170 fd = os.open(TESTFN, os.O_RDONLY)
171 try:
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300172 with self.FileIO(fd, 'r', closefd=False) as f:
Serhiy Storchaka4954f9f2014-12-02 23:39:56 +0200173 self.assertEqual(repr(f),
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300174 "<%s.FileIO name=%r mode=%r closefd=False>" %
175 (self.modulename, f.name, f.mode))
Serhiy Storchaka4954f9f2014-12-02 23:39:56 +0200176 finally:
177 os.close(fd)
178
Serhiy Storchakaa5af6e12017-03-19 19:25:29 +0200179 def testRecursiveRepr(self):
180 # Issue #25455
181 with swap_attr(self.f, 'name', self.f):
182 with self.assertRaises(RuntimeError):
183 repr(self.f) # Should not crash
184
Guido van Rossuma9e20242007-03-08 00:43:48 +0000185 def testErrors(self):
186 f = self.f
Serhiy Storchaka0dcd80a2015-08-02 15:17:49 +0300187 self.assertFalse(f.isatty())
188 self.assertFalse(f.closed)
Ezio Melottib3aedd42010-11-20 19:04:17 +0000189 #self.assertEqual(f.name, TESTFN)
Guido van Rossum31c3a572007-04-12 14:51:49 +0000190 self.assertRaises(ValueError, f.read, 10) # Open for reading
Guido van Rossuma9e20242007-03-08 00:43:48 +0000191 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000192 self.assertTrue(f.closed)
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300193 f = self.FileIO(TESTFN, 'r')
Guido van Rossum31c3a572007-04-12 14:51:49 +0000194 self.assertRaises(TypeError, f.readinto, "")
Serhiy Storchaka0dcd80a2015-08-02 15:17:49 +0300195 self.assertFalse(f.closed)
Guido van Rossum31c3a572007-04-12 14:51:49 +0000196 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000197 self.assertTrue(f.closed)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000198
199 def testMethods(self):
Serhiy Storchakaf24131f2015-04-16 11:19:43 +0300200 methods = ['fileno', 'isatty', 'seekable', 'readable', 'writable',
201 'read', 'readall', 'readline', 'readlines',
202 'tell', 'truncate', 'flush']
Guido van Rossuma9e20242007-03-08 00:43:48 +0000203
Guido van Rossum31c3a572007-04-12 14:51:49 +0000204 self.f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000205 self.assertTrue(self.f.closed)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000206
207 for methodname in methods:
208 method = getattr(self.f, methodname)
209 # should raise on closed file
210 self.assertRaises(ValueError, method)
Serhiy Storchakaf24131f2015-04-16 11:19:43 +0300211
212 self.assertRaises(TypeError, self.f.readinto)
213 self.assertRaises(ValueError, self.f.readinto, bytearray(1))
214 self.assertRaises(TypeError, self.f.seek)
215 self.assertRaises(ValueError, self.f.seek, 0)
216 self.assertRaises(TypeError, self.f.write)
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300217 self.assertRaises(ValueError, self.f.write, b'')
Serhiy Storchakaf24131f2015-04-16 11:19:43 +0300218 self.assertRaises(TypeError, self.f.writelines)
219 self.assertRaises(ValueError, self.f.writelines, b'')
Guido van Rossuma9e20242007-03-08 00:43:48 +0000220
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000221 def testOpendir(self):
222 # Issue 3703: opening a directory should fill the errno
223 # Windows always returns "[Errno 13]: Permission denied
Antoine Pitroude687222014-06-29 20:07:28 -0400224 # Unix uses fstat and returns "[Errno 21]: Is a directory"
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000225 try:
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300226 self.FileIO('.', 'r')
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200227 except OSError as e:
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000228 self.assertNotEqual(e.errno, 0)
Benjamin Peterson1efc23c2008-12-29 18:02:28 +0000229 self.assertEqual(e.filename, ".")
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000230 else:
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200231 self.fail("Should have raised OSError")
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000232
Antoine Pitrou9235b252012-07-06 18:48:24 +0200233 @unittest.skipIf(os.name == 'nt', "test only works on a POSIX-like system")
234 def testOpenDirFD(self):
235 fd = os.open('.', os.O_RDONLY)
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200236 with self.assertRaises(OSError) as cm:
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300237 self.FileIO(fd, 'r')
Antoine Pitrou9235b252012-07-06 18:48:24 +0200238 os.close(fd)
239 self.assertEqual(cm.exception.errno, errno.EISDIR)
240
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000241 #A set of functions testing that we get expected behaviour if someone has
242 #manually closed the internal file descriptor. First, a decorator:
243 def ClosedFD(func):
244 @wraps(func)
245 def wrapper(self):
246 #forcibly close the fd before invoking the problem function
247 f = self.f
248 os.close(f.fileno())
249 try:
250 func(self, f)
251 finally:
252 try:
253 self.f.close()
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200254 except OSError:
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000255 pass
256 return wrapper
Antoine Pitrou0ae29cf2009-03-13 22:33:17 +0000257
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000258 def ClosedFDRaises(func):
259 @wraps(func)
260 def wrapper(self):
261 #forcibly close the fd before invoking the problem function
262 f = self.f
263 os.close(f.fileno())
264 try:
265 func(self, f)
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200266 except OSError as e:
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000267 self.assertEqual(e.errno, errno.EBADF)
268 else:
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200269 self.fail("Should have raised OSError")
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000270 finally:
271 try:
272 self.f.close()
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200273 except OSError:
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000274 pass
275 return wrapper
276
277 @ClosedFDRaises
278 def testErrnoOnClose(self, f):
279 f.close()
280
281 @ClosedFDRaises
282 def testErrnoOnClosedWrite(self, f):
Benjamin Peterson255058f2010-01-27 01:47:14 +0000283 f.write(b'a')
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000284
285 @ClosedFDRaises
286 def testErrnoOnClosedSeek(self, f):
287 f.seek(0)
288
289 @ClosedFDRaises
290 def testErrnoOnClosedTell(self, f):
291 f.tell()
292
293 @ClosedFDRaises
294 def testErrnoOnClosedTruncate(self, f):
295 f.truncate(0)
296
297 @ClosedFD
298 def testErrnoOnClosedSeekable(self, f):
299 f.seekable()
300
301 @ClosedFD
302 def testErrnoOnClosedReadable(self, f):
303 f.readable()
304
305 @ClosedFD
306 def testErrnoOnClosedWritable(self, f):
307 f.writable()
308
309 @ClosedFD
310 def testErrnoOnClosedFileno(self, f):
311 f.fileno()
312
313 @ClosedFD
314 def testErrnoOnClosedIsatty(self, f):
315 self.assertEqual(f.isatty(), False)
316
317 def ReopenForRead(self):
318 try:
319 self.f.close()
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200320 except OSError:
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000321 pass
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300322 self.f = self.FileIO(TESTFN, 'r')
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000323 os.close(self.f.fileno())
324 return self.f
325
326 @ClosedFDRaises
327 def testErrnoOnClosedRead(self, f):
328 f = self.ReopenForRead()
329 f.read(1)
330
331 @ClosedFDRaises
332 def testErrnoOnClosedReadall(self, f):
333 f = self.ReopenForRead()
334 f.readall()
335
336 @ClosedFDRaises
337 def testErrnoOnClosedReadinto(self, f):
338 f = self.ReopenForRead()
339 a = array('b', b'x'*10)
340 f.readinto(a)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000341
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300342class CAutoFileTests(AutoFileTests, unittest.TestCase):
343 FileIO = _io.FileIO
344 modulename = '_io'
345
346class PyAutoFileTests(AutoFileTests, unittest.TestCase):
347 FileIO = _pyio.FileIO
348 modulename = '_pyio'
349
350
351class OtherFileTests:
Guido van Rossuma9e20242007-03-08 00:43:48 +0000352
353 def testAbles(self):
354 try:
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300355 f = self.FileIO(TESTFN, "w")
Ezio Melottib3aedd42010-11-20 19:04:17 +0000356 self.assertEqual(f.readable(), False)
357 self.assertEqual(f.writable(), True)
358 self.assertEqual(f.seekable(), True)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000359 f.close()
Guido van Rossumd8faa362007-04-27 19:54:29 +0000360
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300361 f = self.FileIO(TESTFN, "r")
Ezio Melottib3aedd42010-11-20 19:04:17 +0000362 self.assertEqual(f.readable(), True)
363 self.assertEqual(f.writable(), False)
364 self.assertEqual(f.seekable(), True)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000365 f.close()
Guido van Rossumd8faa362007-04-27 19:54:29 +0000366
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300367 f = self.FileIO(TESTFN, "a+")
Ezio Melottib3aedd42010-11-20 19:04:17 +0000368 self.assertEqual(f.readable(), True)
369 self.assertEqual(f.writable(), True)
370 self.assertEqual(f.seekable(), True)
371 self.assertEqual(f.isatty(), False)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000372 f.close()
Guido van Rossum682faf82007-04-12 14:56:58 +0000373
Thomas Hellerb0f48ab2007-07-12 11:29:02 +0000374 if sys.platform != "win32":
375 try:
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300376 f = self.FileIO("/dev/tty", "a")
Andrew Svetlov3438fa42012-12-17 23:35:18 +0200377 except OSError:
Thomas Hellerb0f48ab2007-07-12 11:29:02 +0000378 # When run in a cron job there just aren't any
379 # ttys, so skip the test. This also handles other
380 # OS'es that don't support /dev/tty.
381 pass
382 else:
Ezio Melottib3aedd42010-11-20 19:04:17 +0000383 self.assertEqual(f.readable(), False)
384 self.assertEqual(f.writable(), True)
Hye-Shik Changcaf871a2007-08-13 13:21:33 +0000385 if sys.platform != "darwin" and \
Antoine Pitroud7b30462009-05-23 16:34:50 +0000386 'bsd' not in sys.platform and \
Martin Panter9544a362016-11-14 01:58:57 +0000387 not sys.platform.startswith(('sunos', 'aix')):
Hye-Shik Changcaf871a2007-08-13 13:21:33 +0000388 # Somehow /dev/tty appears seekable on some BSDs
Ezio Melottib3aedd42010-11-20 19:04:17 +0000389 self.assertEqual(f.seekable(), False)
390 self.assertEqual(f.isatty(), True)
Thomas Hellerb0f48ab2007-07-12 11:29:02 +0000391 f.close()
Guido van Rossuma9e20242007-03-08 00:43:48 +0000392 finally:
393 os.unlink(TESTFN)
394
Antoine Pitroue93b63b2013-09-04 20:46:33 +0200395 def testInvalidModeStrings(self):
Guido van Rossuma9e20242007-03-08 00:43:48 +0000396 # check invalid mode strings
Benjamin Peterson44309e62008-11-22 00:41:45 +0000397 for mode in ("", "aU", "wU+", "rw", "rt"):
Guido van Rossuma9e20242007-03-08 00:43:48 +0000398 try:
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300399 f = self.FileIO(TESTFN, mode)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000400 except ValueError:
401 pass
402 else:
403 f.close()
404 self.fail('%r is an invalid file mode' % mode)
405
Antoine Pitroue93b63b2013-09-04 20:46:33 +0200406 def testModeStrings(self):
407 # test that the mode attribute is correct for various mode strings
408 # given as init args
409 try:
410 for modes in [('w', 'wb'), ('wb', 'wb'), ('wb+', 'rb+'),
411 ('w+b', 'rb+'), ('a', 'ab'), ('ab', 'ab'),
412 ('ab+', 'ab+'), ('a+b', 'ab+'), ('r', 'rb'),
413 ('rb', 'rb'), ('rb+', 'rb+'), ('r+b', 'rb+')]:
414 # read modes are last so that TESTFN will exist first
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300415 with self.FileIO(TESTFN, modes[0]) as f:
Antoine Pitroue93b63b2013-09-04 20:46:33 +0200416 self.assertEqual(f.mode, modes[1])
417 finally:
418 if os.path.exists(TESTFN):
419 os.unlink(TESTFN)
420
Guido van Rossuma9e20242007-03-08 00:43:48 +0000421 def testUnicodeOpen(self):
422 # verify repr works for unicode too
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300423 f = self.FileIO(str(TESTFN), "w")
Guido van Rossuma9e20242007-03-08 00:43:48 +0000424 f.close()
425 os.unlink(TESTFN)
426
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000427 def testBytesOpen(self):
428 # Opening a bytes filename
429 try:
430 fn = TESTFN.encode("ascii")
431 except UnicodeEncodeError:
Zachary Ware9fe6d862013-12-08 00:20:35 -0600432 self.skipTest('could not encode %r to ascii' % TESTFN)
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300433 f = self.FileIO(fn, "w")
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000434 try:
435 f.write(b"abc")
436 f.close()
437 with open(TESTFN, "rb") as f:
Ezio Melottib3aedd42010-11-20 19:04:17 +0000438 self.assertEqual(f.read(), b"abc")
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000439 finally:
440 os.unlink(TESTFN)
441
Steve Dowereacee982017-02-04 14:38:11 -0800442 @unittest.skipIf(sys.getfilesystemencoding() != 'utf-8',
443 "test only works for utf-8 filesystems")
444 def testUtf8BytesOpen(self):
445 # Opening a UTF-8 bytes filename
446 try:
447 fn = TESTFN_UNICODE.encode("utf-8")
448 except UnicodeEncodeError:
449 self.skipTest('could not encode %r to utf-8' % TESTFN_UNICODE)
450 f = self.FileIO(fn, "w")
451 try:
452 f.write(b"abc")
453 f.close()
454 with open(TESTFN_UNICODE, "rb") as f:
455 self.assertEqual(f.read(), b"abc")
456 finally:
457 os.unlink(TESTFN_UNICODE)
458
Antoine Pitrou13348842012-01-29 18:36:34 +0100459 def testConstructorHandlesNULChars(self):
460 fn_with_NUL = 'foo\0bar'
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300461 self.assertRaises(ValueError, self.FileIO, fn_with_NUL, 'w')
462 self.assertRaises(ValueError, self.FileIO, bytes(fn_with_NUL, 'ascii'), 'w')
Antoine Pitrou13348842012-01-29 18:36:34 +0100463
Benjamin Peterson806d4022009-01-19 15:11:51 +0000464 def testInvalidFd(self):
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300465 self.assertRaises(ValueError, self.FileIO, -10)
466 self.assertRaises(OSError, self.FileIO, make_bad_fd())
Antoine Pitrou00492492010-09-04 20:53:29 +0000467 if sys.platform == 'win32':
468 import msvcrt
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200469 self.assertRaises(OSError, msvcrt.get_osfhandle, make_bad_fd())
Serhiy Storchaka5cfc79d2014-02-07 10:06:39 +0200470
Guido van Rossuma9e20242007-03-08 00:43:48 +0000471 def testBadModeArgument(self):
472 # verify that we get a sensible error message for bad mode argument
473 bad_mode = "qwerty"
474 try:
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300475 f = self.FileIO(TESTFN, bad_mode)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000476 except ValueError as msg:
Guido van Rossum360e4b82007-05-14 22:51:27 +0000477 if msg.args[0] != 0:
Guido van Rossuma9e20242007-03-08 00:43:48 +0000478 s = str(msg)
Ezio Melotti7fb4da72010-03-18 12:29:13 +0000479 if TESTFN in s or bad_mode not in s:
Guido van Rossuma9e20242007-03-08 00:43:48 +0000480 self.fail("bad error message for invalid mode: %s" % s)
Georg Brandl50da60c2008-01-06 21:38:54 +0000481 # if msg.args[0] == 0, we're probably on Windows where there may be
Guido van Rossuma9e20242007-03-08 00:43:48 +0000482 # no obvious way to discover why open() failed.
483 else:
484 f.close()
485 self.fail("no error for invalid mode: %s" % bad_mode)
486
Antoine Pitrou905a2ff2010-01-31 22:47:27 +0000487 def testTruncate(self):
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300488 f = self.FileIO(TESTFN, 'w')
Antoine Pitrou905a2ff2010-01-31 22:47:27 +0000489 f.write(bytes(bytearray(range(10))))
490 self.assertEqual(f.tell(), 10)
491 f.truncate(5)
492 self.assertEqual(f.tell(), 10)
Eli Bendersky74c503b2012-01-03 06:26:13 +0200493 self.assertEqual(f.seek(0, io.SEEK_END), 5)
Antoine Pitrou905a2ff2010-01-31 22:47:27 +0000494 f.truncate(15)
495 self.assertEqual(f.tell(), 5)
Eli Bendersky74c503b2012-01-03 06:26:13 +0200496 self.assertEqual(f.seek(0, io.SEEK_END), 15)
Antoine Pitrou8d2b51b2010-10-30 16:19:14 +0000497 f.close()
Antoine Pitrou905a2ff2010-01-31 22:47:27 +0000498
Guido van Rossuma9e20242007-03-08 00:43:48 +0000499 def testTruncateOnWindows(self):
500 def bug801631():
501 # SF bug <http://www.python.org/sf/801631>
502 # "file.truncate fault on windows"
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300503 f = self.FileIO(TESTFN, 'w')
Guido van Rossuma9e20242007-03-08 00:43:48 +0000504 f.write(bytes(range(11)))
505 f.close()
506
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300507 f = self.FileIO(TESTFN,'r+')
Guido van Rossuma9e20242007-03-08 00:43:48 +0000508 data = f.read(5)
509 if data != bytes(range(5)):
510 self.fail("Read on file opened for update failed %r" % data)
511 if f.tell() != 5:
512 self.fail("File pos after read wrong %d" % f.tell())
513
514 f.truncate()
515 if f.tell() != 5:
516 self.fail("File pos after ftruncate wrong %d" % f.tell())
517
518 f.close()
519 size = os.path.getsize(TESTFN)
520 if size != 5:
521 self.fail("File size after ftruncate wrong %d" % size)
522
523 try:
524 bug801631()
525 finally:
526 os.unlink(TESTFN)
527
Walter Dörwald3a77c7a2007-06-06 16:31:14 +0000528 def testAppend(self):
529 try:
530 f = open(TESTFN, 'wb')
531 f.write(b'spam')
532 f.close()
533 f = open(TESTFN, 'ab')
534 f.write(b'eggs')
535 f.close()
536 f = open(TESTFN, 'rb')
537 d = f.read()
538 f.close()
539 self.assertEqual(d, b'spameggs')
540 finally:
541 try:
542 os.unlink(TESTFN)
543 except:
544 pass
545
Neal Norwitz6e0e0e62008-08-24 22:07:28 +0000546 def testInvalidInit(self):
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300547 self.assertRaises(TypeError, self.FileIO, "1", 0, 0)
Neal Norwitz6e0e0e62008-08-24 22:07:28 +0000548
Benjamin Peterson65676e42008-11-05 21:42:45 +0000549 def testWarnings(self):
Florent Xiclunab14930c2010-03-13 15:26:44 +0000550 with check_warnings(quiet=True) as w:
Benjamin Peterson65676e42008-11-05 21:42:45 +0000551 self.assertEqual(w.warnings, [])
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300552 self.assertRaises(TypeError, self.FileIO, [])
Benjamin Peterson65676e42008-11-05 21:42:45 +0000553 self.assertEqual(w.warnings, [])
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300554 self.assertRaises(ValueError, self.FileIO, "/some/invalid/name", "rt")
Benjamin Peterson65676e42008-11-05 21:42:45 +0000555 self.assertEqual(w.warnings, [])
556
Hynek Schlawack9ed8b4e2012-06-21 20:20:25 +0200557 def testUnclosedFDOnException(self):
558 class MyException(Exception): pass
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300559 class MyFileIO(self.FileIO):
Hynek Schlawack9ed8b4e2012-06-21 20:20:25 +0200560 def __setattr__(self, name, value):
561 if name == "name":
562 raise MyException("blocked setting name")
563 return super(MyFileIO, self).__setattr__(name, value)
564 fd = os.open(__file__, os.O_RDONLY)
565 self.assertRaises(MyException, MyFileIO, fd)
566 os.close(fd) # should not raise OSError(EBADF)
567
Steve Dowerb82e17e2019-05-23 08:45:22 -0700568
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300569class COtherFileTests(OtherFileTests, unittest.TestCase):
570 FileIO = _io.FileIO
571 modulename = '_io'
572
573 @cpython_only
574 def testInvalidFd_overflow(self):
575 # Issue 15989
576 import _testcapi
577 self.assertRaises(TypeError, self.FileIO, _testcapi.INT_MAX + 1)
578 self.assertRaises(TypeError, self.FileIO, _testcapi.INT_MIN - 1)
579
Steve Dowerb82e17e2019-05-23 08:45:22 -0700580 def test_open_code(self):
581 # Check that the default behaviour of open_code matches
582 # open("rb")
583 with self.FileIO(__file__, "rb") as f:
584 expected = f.read()
585 with _io.open_code(__file__) as f:
586 actual = f.read()
587 self.assertEqual(expected, actual)
588
589
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300590class PyOtherFileTests(OtherFileTests, unittest.TestCase):
591 FileIO = _pyio.FileIO
592 modulename = '_pyio'
593
Steve Dowerb82e17e2019-05-23 08:45:22 -0700594 def test_open_code(self):
595 # Check that the default behaviour of open_code matches
596 # open("rb")
597 with self.FileIO(__file__, "rb") as f:
598 expected = f.read()
599 with check_warnings(quiet=True) as w:
600 # Always test _open_code_with_warning
601 with _pyio._open_code_with_warning(__file__) as f:
602 actual = f.read()
603 self.assertEqual(expected, actual)
604 self.assertNotEqual(w.warnings, [])
605
Neal Norwitz6e0e0e62008-08-24 22:07:28 +0000606
Guido van Rossuma9e20242007-03-08 00:43:48 +0000607def test_main():
608 # Historically, these tests have been sloppy about removing TESTFN.
609 # So get rid of it no matter what.
610 try:
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300611 run_unittest(CAutoFileTests, PyAutoFileTests,
612 COtherFileTests, PyOtherFileTests)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000613 finally:
614 if os.path.exists(TESTFN):
615 os.unlink(TESTFN)
616
617if __name__ == '__main__':
618 test_main()