blob: 59cc38f6ca2dfa617aa63d96b1dccf1b898c782d [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 Storchaka5cfc79d2014-02-07 10:06:39 +020012from test.support import TESTFN, check_warnings, run_unittest, make_bad_fd, cpython_only
Antoine Pitrou131a4892012-10-16 22:57:11 +020013from collections import UserList
Guido van Rossuma9e20242007-03-08 00:43:48 +000014
Serhiy Storchaka71fd2242015-04-10 16:16:16 +030015import _io # C implementation of io
16import _pyio # Python implementation of io
Guido van Rossuma9e20242007-03-08 00:43:48 +000017
Serhiy Storchaka71fd2242015-04-10 16:16:16 +030018
19class AutoFileTests:
Guido van Rossuma9e20242007-03-08 00:43:48 +000020 # file tests for which a test file is automatically set up
21
22 def setUp(self):
Serhiy Storchaka71fd2242015-04-10 16:16:16 +030023 self.f = self.FileIO(TESTFN, 'w')
Guido van Rossuma9e20242007-03-08 00:43:48 +000024
25 def tearDown(self):
26 if self.f:
27 self.f.close()
28 os.remove(TESTFN)
29
30 def testWeakRefs(self):
31 # verify weak references
32 p = proxy(self.f)
33 p.write(bytes(range(10)))
Ezio Melottib3aedd42010-11-20 19:04:17 +000034 self.assertEqual(self.f.tell(), p.tell())
Guido van Rossuma9e20242007-03-08 00:43:48 +000035 self.f.close()
36 self.f = None
37 self.assertRaises(ReferenceError, getattr, p, 'tell')
38
39 def testSeekTell(self):
40 self.f.write(bytes(range(20)))
Ezio Melottib3aedd42010-11-20 19:04:17 +000041 self.assertEqual(self.f.tell(), 20)
Guido van Rossuma9e20242007-03-08 00:43:48 +000042 self.f.seek(0)
Ezio Melottib3aedd42010-11-20 19:04:17 +000043 self.assertEqual(self.f.tell(), 0)
Guido van Rossuma9e20242007-03-08 00:43:48 +000044 self.f.seek(10)
Ezio Melottib3aedd42010-11-20 19:04:17 +000045 self.assertEqual(self.f.tell(), 10)
Guido van Rossuma9e20242007-03-08 00:43:48 +000046 self.f.seek(5, 1)
Ezio Melottib3aedd42010-11-20 19:04:17 +000047 self.assertEqual(self.f.tell(), 15)
Guido van Rossuma9e20242007-03-08 00:43:48 +000048 self.f.seek(-5, 1)
Ezio Melottib3aedd42010-11-20 19:04:17 +000049 self.assertEqual(self.f.tell(), 10)
Guido van Rossuma9e20242007-03-08 00:43:48 +000050 self.f.seek(-5, 2)
Ezio Melottib3aedd42010-11-20 19:04:17 +000051 self.assertEqual(self.f.tell(), 15)
Guido van Rossuma9e20242007-03-08 00:43:48 +000052
53 def testAttributes(self):
54 # verify expected attributes exist
55 f = self.f
Guido van Rossuma9e20242007-03-08 00:43:48 +000056
Ezio Melottib3aedd42010-11-20 19:04:17 +000057 self.assertEqual(f.mode, "wb")
58 self.assertEqual(f.closed, False)
Guido van Rossum31c3a572007-04-12 14:51:49 +000059
60 # verify the attributes are readonly
61 for attr in 'mode', 'closed':
62 self.assertRaises((AttributeError, TypeError),
63 setattr, f, attr, 'oops')
Guido van Rossuma9e20242007-03-08 00:43:48 +000064
Antoine Pitroude687222014-06-29 20:07:28 -040065 def testBlksize(self):
66 # test private _blksize attribute
67 blksize = io.DEFAULT_BUFFER_SIZE
68 # try to get preferred blksize from stat.st_blksize, if available
69 if hasattr(os, 'fstat'):
70 fst = os.fstat(self.f.fileno())
71 blksize = getattr(fst, 'st_blksize', blksize)
72 self.assertEqual(self.f._blksize, blksize)
73
Serhiy Storchaka71fd2242015-04-10 16:16:16 +030074 # verify readinto
75 def testReadintoByteArray(self):
76 self.f.write(bytes([1, 2, 0, 255]))
Guido van Rossuma9e20242007-03-08 00:43:48 +000077 self.f.close()
Serhiy Storchaka71fd2242015-04-10 16:16:16 +030078
79 ba = bytearray(b'abcdefgh')
80 with self.FileIO(TESTFN, 'r') as f:
81 n = f.readinto(ba)
82 self.assertEqual(ba, b'\x01\x02\x00\xffefgh')
83 self.assertEqual(n, 4)
84
85 def _testReadintoMemoryview(self):
86 self.f.write(bytes([1, 2, 0, 255]))
87 self.f.close()
88
89 m = memoryview(bytearray(b'abcdefgh'))
90 with self.FileIO(TESTFN, 'r') as f:
91 n = f.readinto(m)
92 self.assertEqual(m, b'\x01\x02\x00\xffefgh')
93 self.assertEqual(n, 4)
94
95 m = memoryview(bytearray(b'abcdefgh')).cast('H', shape=[2, 2])
96 with self.FileIO(TESTFN, 'r') as f:
97 n = f.readinto(m)
98 self.assertEqual(bytes(m), b'\x01\x02\x00\xffefgh')
99 self.assertEqual(n, 4)
100
101 def _testReadintoArray(self):
102 self.f.write(bytes([1, 2, 0, 255]))
103 self.f.close()
104
105 a = array('B', b'abcdefgh')
106 with self.FileIO(TESTFN, 'r') as f:
107 n = f.readinto(a)
108 self.assertEqual(a, array('B', [1, 2, 0, 255, 101, 102, 103, 104]))
109 self.assertEqual(n, 4)
110
111 a = array('b', b'abcdefgh')
112 with self.FileIO(TESTFN, 'r') as f:
113 n = f.readinto(a)
114 self.assertEqual(a, array('b', [1, 2, 0, -1, 101, 102, 103, 104]))
115 self.assertEqual(n, 4)
116
117 a = array('I', b'abcdefgh')
118 with self.FileIO(TESTFN, 'r') as f:
119 n = f.readinto(a)
120 self.assertEqual(a, array('I', b'\x01\x02\x00\xffefgh'))
121 self.assertEqual(n, 4)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000122
Antoine Pitrou131a4892012-10-16 22:57:11 +0200123 def testWritelinesList(self):
124 l = [b'123', b'456']
125 self.f.writelines(l)
126 self.f.close()
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300127 self.f = self.FileIO(TESTFN, 'rb')
Antoine Pitrou131a4892012-10-16 22:57:11 +0200128 buf = self.f.read()
129 self.assertEqual(buf, b'123456')
130
131 def testWritelinesUserList(self):
132 l = UserList([b'123', b'456'])
133 self.f.writelines(l)
134 self.f.close()
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300135 self.f = self.FileIO(TESTFN, 'rb')
Antoine Pitrou131a4892012-10-16 22:57:11 +0200136 buf = self.f.read()
137 self.assertEqual(buf, b'123456')
138
139 def testWritelinesError(self):
140 self.assertRaises(TypeError, self.f.writelines, [1, 2, 3])
141 self.assertRaises(TypeError, self.f.writelines, None)
142 self.assertRaises(TypeError, self.f.writelines, "abc")
143
Benjamin Petersonbf5ff762009-12-13 19:25:34 +0000144 def test_none_args(self):
145 self.f.write(b"hi\nbye\nabc")
146 self.f.close()
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300147 self.f = self.FileIO(TESTFN, 'r')
Benjamin Petersonbf5ff762009-12-13 19:25:34 +0000148 self.assertEqual(self.f.read(None), b"hi\nbye\nabc")
149 self.f.seek(0)
150 self.assertEqual(self.f.readline(None), b"hi\n")
151 self.assertEqual(self.f.readlines(None), [b"bye\n", b"abc"])
152
Benjamin Peterson255058f2010-01-27 01:47:14 +0000153 def test_reject(self):
154 self.assertRaises(TypeError, self.f.write, "Hello!")
155
Guido van Rossuma9e20242007-03-08 00:43:48 +0000156 def testRepr(self):
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300157 self.assertEqual(repr(self.f),
158 "<%s.FileIO name=%r mode=%r closefd=True>" %
159 (self.modulename, self.f.name, self.f.mode))
Antoine Pitrou716c4442009-05-23 19:04:03 +0000160 del self.f.name
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300161 self.assertEqual(repr(self.f),
162 "<%s.FileIO fd=%r mode=%r closefd=True>" %
163 (self.modulename, self.f.fileno(), self.f.mode))
Antoine Pitrou716c4442009-05-23 19:04:03 +0000164 self.f.close()
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300165 self.assertEqual(repr(self.f),
166 "<%s.FileIO [closed]>" % (self.modulename,))
Guido van Rossuma9e20242007-03-08 00:43:48 +0000167
Serhiy Storchaka4954f9f2014-12-02 23:39:56 +0200168 def testReprNoCloseFD(self):
169 fd = os.open(TESTFN, os.O_RDONLY)
170 try:
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300171 with self.FileIO(fd, 'r', closefd=False) as f:
Serhiy Storchaka4954f9f2014-12-02 23:39:56 +0200172 self.assertEqual(repr(f),
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300173 "<%s.FileIO name=%r mode=%r closefd=False>" %
174 (self.modulename, f.name, f.mode))
Serhiy Storchaka4954f9f2014-12-02 23:39:56 +0200175 finally:
176 os.close(fd)
177
Guido van Rossuma9e20242007-03-08 00:43:48 +0000178 def testErrors(self):
179 f = self.f
Serhiy Storchaka0dcd80a2015-08-02 15:17:49 +0300180 self.assertFalse(f.isatty())
181 self.assertFalse(f.closed)
Ezio Melottib3aedd42010-11-20 19:04:17 +0000182 #self.assertEqual(f.name, TESTFN)
Guido van Rossum31c3a572007-04-12 14:51:49 +0000183 self.assertRaises(ValueError, f.read, 10) # Open for reading
Guido van Rossuma9e20242007-03-08 00:43:48 +0000184 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000185 self.assertTrue(f.closed)
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300186 f = self.FileIO(TESTFN, 'r')
Guido van Rossum31c3a572007-04-12 14:51:49 +0000187 self.assertRaises(TypeError, f.readinto, "")
Serhiy Storchaka0dcd80a2015-08-02 15:17:49 +0300188 self.assertFalse(f.closed)
Guido van Rossum31c3a572007-04-12 14:51:49 +0000189 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000190 self.assertTrue(f.closed)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000191
192 def testMethods(self):
Serhiy Storchakaf24131f2015-04-16 11:19:43 +0300193 methods = ['fileno', 'isatty', 'seekable', 'readable', 'writable',
194 'read', 'readall', 'readline', 'readlines',
195 'tell', 'truncate', 'flush']
Guido van Rossuma9e20242007-03-08 00:43:48 +0000196
Guido van Rossum31c3a572007-04-12 14:51:49 +0000197 self.f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000198 self.assertTrue(self.f.closed)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000199
200 for methodname in methods:
201 method = getattr(self.f, methodname)
202 # should raise on closed file
203 self.assertRaises(ValueError, method)
Serhiy Storchakaf24131f2015-04-16 11:19:43 +0300204
205 self.assertRaises(TypeError, self.f.readinto)
206 self.assertRaises(ValueError, self.f.readinto, bytearray(1))
207 self.assertRaises(TypeError, self.f.seek)
208 self.assertRaises(ValueError, self.f.seek, 0)
209 self.assertRaises(TypeError, self.f.write)
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300210 self.assertRaises(ValueError, self.f.write, b'')
Serhiy Storchakaf24131f2015-04-16 11:19:43 +0300211 self.assertRaises(TypeError, self.f.writelines)
212 self.assertRaises(ValueError, self.f.writelines, b'')
Guido van Rossuma9e20242007-03-08 00:43:48 +0000213
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000214 def testOpendir(self):
215 # Issue 3703: opening a directory should fill the errno
216 # Windows always returns "[Errno 13]: Permission denied
Antoine Pitroude687222014-06-29 20:07:28 -0400217 # Unix uses fstat and returns "[Errno 21]: Is a directory"
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000218 try:
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300219 self.FileIO('.', 'r')
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200220 except OSError as e:
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000221 self.assertNotEqual(e.errno, 0)
Benjamin Peterson1efc23c2008-12-29 18:02:28 +0000222 self.assertEqual(e.filename, ".")
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000223 else:
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200224 self.fail("Should have raised OSError")
Benjamin Peterson3e4f0552008-09-02 00:31:15 +0000225
Antoine Pitrou9235b252012-07-06 18:48:24 +0200226 @unittest.skipIf(os.name == 'nt', "test only works on a POSIX-like system")
227 def testOpenDirFD(self):
228 fd = os.open('.', os.O_RDONLY)
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200229 with self.assertRaises(OSError) as cm:
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300230 self.FileIO(fd, 'r')
Antoine Pitrou9235b252012-07-06 18:48:24 +0200231 os.close(fd)
232 self.assertEqual(cm.exception.errno, errno.EISDIR)
233
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000234 #A set of functions testing that we get expected behaviour if someone has
235 #manually closed the internal file descriptor. First, a decorator:
236 def ClosedFD(func):
237 @wraps(func)
238 def wrapper(self):
239 #forcibly close the fd before invoking the problem function
240 f = self.f
241 os.close(f.fileno())
242 try:
243 func(self, f)
244 finally:
245 try:
246 self.f.close()
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200247 except OSError:
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000248 pass
249 return wrapper
Antoine Pitrou0ae29cf2009-03-13 22:33:17 +0000250
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000251 def ClosedFDRaises(func):
252 @wraps(func)
253 def wrapper(self):
254 #forcibly close the fd before invoking the problem function
255 f = self.f
256 os.close(f.fileno())
257 try:
258 func(self, f)
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200259 except OSError as e:
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000260 self.assertEqual(e.errno, errno.EBADF)
261 else:
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200262 self.fail("Should have raised OSError")
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000263 finally:
264 try:
265 self.f.close()
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200266 except OSError:
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000267 pass
268 return wrapper
269
270 @ClosedFDRaises
271 def testErrnoOnClose(self, f):
272 f.close()
273
274 @ClosedFDRaises
275 def testErrnoOnClosedWrite(self, f):
Benjamin Peterson255058f2010-01-27 01:47:14 +0000276 f.write(b'a')
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000277
278 @ClosedFDRaises
279 def testErrnoOnClosedSeek(self, f):
280 f.seek(0)
281
282 @ClosedFDRaises
283 def testErrnoOnClosedTell(self, f):
284 f.tell()
285
286 @ClosedFDRaises
287 def testErrnoOnClosedTruncate(self, f):
288 f.truncate(0)
289
290 @ClosedFD
291 def testErrnoOnClosedSeekable(self, f):
292 f.seekable()
293
294 @ClosedFD
295 def testErrnoOnClosedReadable(self, f):
296 f.readable()
297
298 @ClosedFD
299 def testErrnoOnClosedWritable(self, f):
300 f.writable()
301
302 @ClosedFD
303 def testErrnoOnClosedFileno(self, f):
304 f.fileno()
305
306 @ClosedFD
307 def testErrnoOnClosedIsatty(self, f):
308 self.assertEqual(f.isatty(), False)
309
310 def ReopenForRead(self):
311 try:
312 self.f.close()
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200313 except OSError:
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000314 pass
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300315 self.f = self.FileIO(TESTFN, 'r')
Kristján Valur Jónssona8abe862009-03-24 15:27:42 +0000316 os.close(self.f.fileno())
317 return self.f
318
319 @ClosedFDRaises
320 def testErrnoOnClosedRead(self, f):
321 f = self.ReopenForRead()
322 f.read(1)
323
324 @ClosedFDRaises
325 def testErrnoOnClosedReadall(self, f):
326 f = self.ReopenForRead()
327 f.readall()
328
329 @ClosedFDRaises
330 def testErrnoOnClosedReadinto(self, f):
331 f = self.ReopenForRead()
332 a = array('b', b'x'*10)
333 f.readinto(a)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000334
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300335class CAutoFileTests(AutoFileTests, unittest.TestCase):
336 FileIO = _io.FileIO
337 modulename = '_io'
338
339class PyAutoFileTests(AutoFileTests, unittest.TestCase):
340 FileIO = _pyio.FileIO
341 modulename = '_pyio'
342
343
344class OtherFileTests:
Guido van Rossuma9e20242007-03-08 00:43:48 +0000345
346 def testAbles(self):
347 try:
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300348 f = self.FileIO(TESTFN, "w")
Ezio Melottib3aedd42010-11-20 19:04:17 +0000349 self.assertEqual(f.readable(), False)
350 self.assertEqual(f.writable(), True)
351 self.assertEqual(f.seekable(), True)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000352 f.close()
Guido van Rossumd8faa362007-04-27 19:54:29 +0000353
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300354 f = self.FileIO(TESTFN, "r")
Ezio Melottib3aedd42010-11-20 19:04:17 +0000355 self.assertEqual(f.readable(), True)
356 self.assertEqual(f.writable(), False)
357 self.assertEqual(f.seekable(), True)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000358 f.close()
Guido van Rossumd8faa362007-04-27 19:54:29 +0000359
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300360 f = self.FileIO(TESTFN, "a+")
Ezio Melottib3aedd42010-11-20 19:04:17 +0000361 self.assertEqual(f.readable(), True)
362 self.assertEqual(f.writable(), True)
363 self.assertEqual(f.seekable(), True)
364 self.assertEqual(f.isatty(), False)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000365 f.close()
Guido van Rossum682faf82007-04-12 14:56:58 +0000366
Thomas Hellerb0f48ab2007-07-12 11:29:02 +0000367 if sys.platform != "win32":
368 try:
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300369 f = self.FileIO("/dev/tty", "a")
Andrew Svetlov3438fa42012-12-17 23:35:18 +0200370 except OSError:
Thomas Hellerb0f48ab2007-07-12 11:29:02 +0000371 # When run in a cron job there just aren't any
372 # ttys, so skip the test. This also handles other
373 # OS'es that don't support /dev/tty.
374 pass
375 else:
Ezio Melottib3aedd42010-11-20 19:04:17 +0000376 self.assertEqual(f.readable(), False)
377 self.assertEqual(f.writable(), True)
Hye-Shik Changcaf871a2007-08-13 13:21:33 +0000378 if sys.platform != "darwin" and \
Antoine Pitroud7b30462009-05-23 16:34:50 +0000379 'bsd' not in sys.platform and \
Benjamin Peterson94fe10f2008-07-17 23:27:26 +0000380 not sys.platform.startswith('sunos'):
Hye-Shik Changcaf871a2007-08-13 13:21:33 +0000381 # Somehow /dev/tty appears seekable on some BSDs
Ezio Melottib3aedd42010-11-20 19:04:17 +0000382 self.assertEqual(f.seekable(), False)
383 self.assertEqual(f.isatty(), True)
Thomas Hellerb0f48ab2007-07-12 11:29:02 +0000384 f.close()
Guido van Rossuma9e20242007-03-08 00:43:48 +0000385 finally:
386 os.unlink(TESTFN)
387
Antoine Pitroue93b63b2013-09-04 20:46:33 +0200388 def testInvalidModeStrings(self):
Guido van Rossuma9e20242007-03-08 00:43:48 +0000389 # check invalid mode strings
Benjamin Peterson44309e62008-11-22 00:41:45 +0000390 for mode in ("", "aU", "wU+", "rw", "rt"):
Guido van Rossuma9e20242007-03-08 00:43:48 +0000391 try:
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300392 f = self.FileIO(TESTFN, mode)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000393 except ValueError:
394 pass
395 else:
396 f.close()
397 self.fail('%r is an invalid file mode' % mode)
398
Antoine Pitroue93b63b2013-09-04 20:46:33 +0200399 def testModeStrings(self):
400 # test that the mode attribute is correct for various mode strings
401 # given as init args
402 try:
403 for modes in [('w', 'wb'), ('wb', 'wb'), ('wb+', 'rb+'),
404 ('w+b', 'rb+'), ('a', 'ab'), ('ab', 'ab'),
405 ('ab+', 'ab+'), ('a+b', 'ab+'), ('r', 'rb'),
406 ('rb', 'rb'), ('rb+', 'rb+'), ('r+b', 'rb+')]:
407 # read modes are last so that TESTFN will exist first
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300408 with self.FileIO(TESTFN, modes[0]) as f:
Antoine Pitroue93b63b2013-09-04 20:46:33 +0200409 self.assertEqual(f.mode, modes[1])
410 finally:
411 if os.path.exists(TESTFN):
412 os.unlink(TESTFN)
413
Guido van Rossuma9e20242007-03-08 00:43:48 +0000414 def testUnicodeOpen(self):
415 # verify repr works for unicode too
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300416 f = self.FileIO(str(TESTFN), "w")
Guido van Rossuma9e20242007-03-08 00:43:48 +0000417 f.close()
418 os.unlink(TESTFN)
419
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000420 def testBytesOpen(self):
421 # Opening a bytes filename
422 try:
423 fn = TESTFN.encode("ascii")
424 except UnicodeEncodeError:
Zachary Ware9fe6d862013-12-08 00:20:35 -0600425 self.skipTest('could not encode %r to ascii' % TESTFN)
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300426 f = self.FileIO(fn, "w")
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000427 try:
428 f.write(b"abc")
429 f.close()
430 with open(TESTFN, "rb") as f:
Ezio Melottib3aedd42010-11-20 19:04:17 +0000431 self.assertEqual(f.read(), b"abc")
Benjamin Peterson4fa88fa2009-03-04 00:14:51 +0000432 finally:
433 os.unlink(TESTFN)
434
Antoine Pitrou13348842012-01-29 18:36:34 +0100435 def testConstructorHandlesNULChars(self):
436 fn_with_NUL = 'foo\0bar'
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300437 self.assertRaises(ValueError, self.FileIO, fn_with_NUL, 'w')
438 self.assertRaises(ValueError, self.FileIO, bytes(fn_with_NUL, 'ascii'), 'w')
Antoine Pitrou13348842012-01-29 18:36:34 +0100439
Benjamin Peterson806d4022009-01-19 15:11:51 +0000440 def testInvalidFd(self):
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300441 self.assertRaises(ValueError, self.FileIO, -10)
442 self.assertRaises(OSError, self.FileIO, make_bad_fd())
Antoine Pitrou00492492010-09-04 20:53:29 +0000443 if sys.platform == 'win32':
444 import msvcrt
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200445 self.assertRaises(OSError, msvcrt.get_osfhandle, make_bad_fd())
Serhiy Storchaka5cfc79d2014-02-07 10:06:39 +0200446
Guido van Rossuma9e20242007-03-08 00:43:48 +0000447 def testBadModeArgument(self):
448 # verify that we get a sensible error message for bad mode argument
449 bad_mode = "qwerty"
450 try:
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300451 f = self.FileIO(TESTFN, bad_mode)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000452 except ValueError as msg:
Guido van Rossum360e4b82007-05-14 22:51:27 +0000453 if msg.args[0] != 0:
Guido van Rossuma9e20242007-03-08 00:43:48 +0000454 s = str(msg)
Ezio Melotti7fb4da72010-03-18 12:29:13 +0000455 if TESTFN in s or bad_mode not in s:
Guido van Rossuma9e20242007-03-08 00:43:48 +0000456 self.fail("bad error message for invalid mode: %s" % s)
Georg Brandl50da60c2008-01-06 21:38:54 +0000457 # if msg.args[0] == 0, we're probably on Windows where there may be
Guido van Rossuma9e20242007-03-08 00:43:48 +0000458 # no obvious way to discover why open() failed.
459 else:
460 f.close()
461 self.fail("no error for invalid mode: %s" % bad_mode)
462
Antoine Pitrou905a2ff2010-01-31 22:47:27 +0000463 def testTruncate(self):
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300464 f = self.FileIO(TESTFN, 'w')
Antoine Pitrou905a2ff2010-01-31 22:47:27 +0000465 f.write(bytes(bytearray(range(10))))
466 self.assertEqual(f.tell(), 10)
467 f.truncate(5)
468 self.assertEqual(f.tell(), 10)
Eli Bendersky74c503b2012-01-03 06:26:13 +0200469 self.assertEqual(f.seek(0, io.SEEK_END), 5)
Antoine Pitrou905a2ff2010-01-31 22:47:27 +0000470 f.truncate(15)
471 self.assertEqual(f.tell(), 5)
Eli Bendersky74c503b2012-01-03 06:26:13 +0200472 self.assertEqual(f.seek(0, io.SEEK_END), 15)
Antoine Pitrou8d2b51b2010-10-30 16:19:14 +0000473 f.close()
Antoine Pitrou905a2ff2010-01-31 22:47:27 +0000474
Guido van Rossuma9e20242007-03-08 00:43:48 +0000475 def testTruncateOnWindows(self):
476 def bug801631():
477 # SF bug <http://www.python.org/sf/801631>
478 # "file.truncate fault on windows"
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300479 f = self.FileIO(TESTFN, 'w')
Guido van Rossuma9e20242007-03-08 00:43:48 +0000480 f.write(bytes(range(11)))
481 f.close()
482
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300483 f = self.FileIO(TESTFN,'r+')
Guido van Rossuma9e20242007-03-08 00:43:48 +0000484 data = f.read(5)
485 if data != bytes(range(5)):
486 self.fail("Read on file opened for update failed %r" % data)
487 if f.tell() != 5:
488 self.fail("File pos after read wrong %d" % f.tell())
489
490 f.truncate()
491 if f.tell() != 5:
492 self.fail("File pos after ftruncate wrong %d" % f.tell())
493
494 f.close()
495 size = os.path.getsize(TESTFN)
496 if size != 5:
497 self.fail("File size after ftruncate wrong %d" % size)
498
499 try:
500 bug801631()
501 finally:
502 os.unlink(TESTFN)
503
Walter Dörwald3a77c7a2007-06-06 16:31:14 +0000504 def testAppend(self):
505 try:
506 f = open(TESTFN, 'wb')
507 f.write(b'spam')
508 f.close()
509 f = open(TESTFN, 'ab')
510 f.write(b'eggs')
511 f.close()
512 f = open(TESTFN, 'rb')
513 d = f.read()
514 f.close()
515 self.assertEqual(d, b'spameggs')
516 finally:
517 try:
518 os.unlink(TESTFN)
519 except:
520 pass
521
Neal Norwitz6e0e0e62008-08-24 22:07:28 +0000522 def testInvalidInit(self):
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300523 self.assertRaises(TypeError, self.FileIO, "1", 0, 0)
Neal Norwitz6e0e0e62008-08-24 22:07:28 +0000524
Benjamin Peterson65676e42008-11-05 21:42:45 +0000525 def testWarnings(self):
Florent Xiclunab14930c2010-03-13 15:26:44 +0000526 with check_warnings(quiet=True) as w:
Benjamin Peterson65676e42008-11-05 21:42:45 +0000527 self.assertEqual(w.warnings, [])
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300528 self.assertRaises(TypeError, self.FileIO, [])
Benjamin Peterson65676e42008-11-05 21:42:45 +0000529 self.assertEqual(w.warnings, [])
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300530 self.assertRaises(ValueError, self.FileIO, "/some/invalid/name", "rt")
Benjamin Peterson65676e42008-11-05 21:42:45 +0000531 self.assertEqual(w.warnings, [])
532
Hynek Schlawack9ed8b4e2012-06-21 20:20:25 +0200533 def testUnclosedFDOnException(self):
534 class MyException(Exception): pass
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300535 class MyFileIO(self.FileIO):
Hynek Schlawack9ed8b4e2012-06-21 20:20:25 +0200536 def __setattr__(self, name, value):
537 if name == "name":
538 raise MyException("blocked setting name")
539 return super(MyFileIO, self).__setattr__(name, value)
540 fd = os.open(__file__, os.O_RDONLY)
541 self.assertRaises(MyException, MyFileIO, fd)
542 os.close(fd) # should not raise OSError(EBADF)
543
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300544class COtherFileTests(OtherFileTests, unittest.TestCase):
545 FileIO = _io.FileIO
546 modulename = '_io'
547
548 @cpython_only
549 def testInvalidFd_overflow(self):
550 # Issue 15989
551 import _testcapi
552 self.assertRaises(TypeError, self.FileIO, _testcapi.INT_MAX + 1)
553 self.assertRaises(TypeError, self.FileIO, _testcapi.INT_MIN - 1)
554
555class PyOtherFileTests(OtherFileTests, unittest.TestCase):
556 FileIO = _pyio.FileIO
557 modulename = '_pyio'
558
Neal Norwitz6e0e0e62008-08-24 22:07:28 +0000559
Guido van Rossuma9e20242007-03-08 00:43:48 +0000560def test_main():
561 # Historically, these tests have been sloppy about removing TESTFN.
562 # So get rid of it no matter what.
563 try:
Serhiy Storchaka71fd2242015-04-10 16:16:16 +0300564 run_unittest(CAutoFileTests, PyAutoFileTests,
565 COtherFileTests, PyOtherFileTests)
Guido van Rossuma9e20242007-03-08 00:43:48 +0000566 finally:
567 if os.path.exists(TESTFN):
568 os.unlink(TESTFN)
569
570if __name__ == '__main__':
571 test_main()