Martin v. Löwis | b96e0e5 | 2000-09-19 16:35:39 +0000 | [diff] [blame] | 1 | # Tests StringIO and cStringIO |
| 2 | |
Barry Warsaw | 7f8ff47 | 2001-09-22 04:33:47 +0000 | [diff] [blame] | 3 | import unittest |
| 4 | import StringIO |
| 5 | import cStringIO |
| 6 | import types |
Antoine Pitrou | a5016cd | 2011-10-21 22:22:04 +0200 | [diff] [blame] | 7 | import array |
Serhiy Storchaka | 276f1d5 | 2013-02-09 13:47:43 +0200 | [diff] [blame] | 8 | import sys |
Barry Warsaw | 04f357c | 2002-07-23 19:04:11 +0000 | [diff] [blame] | 9 | from test import test_support |
Martin v. Löwis | b96e0e5 | 2000-09-19 16:35:39 +0000 | [diff] [blame] | 10 | |
Jeremy Hylton | cafd495 | 2001-02-09 23:44:22 +0000 | [diff] [blame] | 11 | |
Barry Warsaw | 7f8ff47 | 2001-09-22 04:33:47 +0000 | [diff] [blame] | 12 | class TestGenericStringIO(unittest.TestCase): |
| 13 | # use a class variable MODULE to define which module is being tested |
Jeremy Hylton | cafd495 | 2001-02-09 23:44:22 +0000 | [diff] [blame] | 14 | |
Marc-André Lemburg | e47df7a | 2001-09-24 17:34:52 +0000 | [diff] [blame] | 15 | # Line of data to test as string |
| 16 | _line = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ!' |
| 17 | |
| 18 | # Constructor to use for the test data (._line is passed to this |
| 19 | # constructor) |
| 20 | constructor = str |
| 21 | |
Barry Warsaw | 7f8ff47 | 2001-09-22 04:33:47 +0000 | [diff] [blame] | 22 | def setUp(self): |
Marc-André Lemburg | e47df7a | 2001-09-24 17:34:52 +0000 | [diff] [blame] | 23 | self._lines = self.constructor((self._line + '\n') * 5) |
Barry Warsaw | 7f8ff47 | 2001-09-22 04:33:47 +0000 | [diff] [blame] | 24 | self._fp = self.MODULE.StringIO(self._lines) |
Fred Drake | 9543833 | 2000-09-28 04:25:33 +0000 | [diff] [blame] | 25 | |
Barry Warsaw | 7f8ff47 | 2001-09-22 04:33:47 +0000 | [diff] [blame] | 26 | def test_reads(self): |
| 27 | eq = self.assertEqual |
Walter Dörwald | bb9c739 | 2004-11-01 17:10:19 +0000 | [diff] [blame] | 28 | self.assertRaises(TypeError, self._fp.seek) |
Marc-André Lemburg | e47df7a | 2001-09-24 17:34:52 +0000 | [diff] [blame] | 29 | eq(self._fp.read(10), self._line[:10]) |
Serhiy Storchaka | 8d7d6bc | 2013-02-13 12:26:58 +0200 | [diff] [blame] | 30 | eq(self._fp.read(0), '') |
| 31 | eq(self._fp.readline(0), '') |
Marc-André Lemburg | e47df7a | 2001-09-24 17:34:52 +0000 | [diff] [blame] | 32 | eq(self._fp.readline(), self._line[10:] + '\n') |
Barry Warsaw | 7f8ff47 | 2001-09-22 04:33:47 +0000 | [diff] [blame] | 33 | eq(len(self._fp.readlines(60)), 2) |
Benjamin Peterson | 4147216 | 2009-12-13 17:29:16 +0000 | [diff] [blame] | 34 | self._fp.seek(0) |
| 35 | eq(self._fp.readline(-1), self._line + '\n') |
Barry Warsaw | 7f8ff47 | 2001-09-22 04:33:47 +0000 | [diff] [blame] | 36 | |
| 37 | def test_writes(self): |
| 38 | f = self.MODULE.StringIO() |
Walter Dörwald | bb9c739 | 2004-11-01 17:10:19 +0000 | [diff] [blame] | 39 | self.assertRaises(TypeError, f.seek) |
Marc-André Lemburg | e47df7a | 2001-09-24 17:34:52 +0000 | [diff] [blame] | 40 | f.write(self._line[:6]) |
Barry Warsaw | 7f8ff47 | 2001-09-22 04:33:47 +0000 | [diff] [blame] | 41 | f.seek(3) |
Marc-André Lemburg | e47df7a | 2001-09-24 17:34:52 +0000 | [diff] [blame] | 42 | f.write(self._line[20:26]) |
| 43 | f.write(self._line[52]) |
Barry Warsaw | 7f8ff47 | 2001-09-22 04:33:47 +0000 | [diff] [blame] | 44 | self.assertEqual(f.getvalue(), 'abcuvwxyz!') |
| 45 | |
| 46 | def test_writelines(self): |
| 47 | f = self.MODULE.StringIO() |
Marc-André Lemburg | e47df7a | 2001-09-24 17:34:52 +0000 | [diff] [blame] | 48 | f.writelines([self._line[0], self._line[1], self._line[2]]) |
Barry Warsaw | 7f8ff47 | 2001-09-22 04:33:47 +0000 | [diff] [blame] | 49 | f.seek(0) |
| 50 | self.assertEqual(f.getvalue(), 'abc') |
| 51 | |
Michael W. Hudson | 10402a3 | 2005-09-22 09:19:01 +0000 | [diff] [blame] | 52 | def test_writelines_error(self): |
| 53 | def errorGen(): |
| 54 | yield 'a' |
| 55 | raise KeyboardInterrupt() |
| 56 | f = self.MODULE.StringIO() |
| 57 | self.assertRaises(KeyboardInterrupt, f.writelines, errorGen()) |
| 58 | |
Barry Warsaw | 7f8ff47 | 2001-09-22 04:33:47 +0000 | [diff] [blame] | 59 | def test_truncate(self): |
| 60 | eq = self.assertEqual |
| 61 | f = self.MODULE.StringIO() |
| 62 | f.write(self._lines) |
| 63 | f.seek(10) |
| 64 | f.truncate() |
| 65 | eq(f.getvalue(), 'abcdefghij') |
Barry Warsaw | 7f8ff47 | 2001-09-22 04:33:47 +0000 | [diff] [blame] | 66 | f.truncate(5) |
| 67 | eq(f.getvalue(), 'abcde') |
Tim Peters | 037b3ee | 2004-08-21 06:55:43 +0000 | [diff] [blame] | 68 | f.write('xyz') |
| 69 | eq(f.getvalue(), 'abcdexyz') |
Martin v. Löwis | cffcc8b | 2006-11-19 10:41:41 +0000 | [diff] [blame] | 70 | self.assertRaises(IOError, f.truncate, -1) |
Barry Warsaw | 7f8ff47 | 2001-09-22 04:33:47 +0000 | [diff] [blame] | 71 | f.close() |
| 72 | self.assertRaises(ValueError, f.write, 'frobnitz') |
| 73 | |
Raymond Hettinger | 5475f23 | 2003-08-08 12:20:03 +0000 | [diff] [blame] | 74 | def test_closed_flag(self): |
| 75 | f = self.MODULE.StringIO() |
| 76 | self.assertEqual(f.closed, False) |
| 77 | f.close() |
| 78 | self.assertEqual(f.closed, True) |
| 79 | f = self.MODULE.StringIO("abc") |
| 80 | self.assertEqual(f.closed, False) |
| 81 | f.close() |
| 82 | self.assertEqual(f.closed, True) |
| 83 | |
Walter Dörwald | 197e832 | 2006-03-15 22:13:13 +0000 | [diff] [blame] | 84 | def test_isatty(self): |
| 85 | f = self.MODULE.StringIO() |
| 86 | self.assertRaises(TypeError, f.isatty, None) |
| 87 | self.assertEqual(f.isatty(), False) |
| 88 | f.close() |
| 89 | self.assertRaises(ValueError, f.isatty) |
| 90 | |
Barry Warsaw | 7f8ff47 | 2001-09-22 04:33:47 +0000 | [diff] [blame] | 91 | def test_iterator(self): |
| 92 | eq = self.assertEqual |
Benjamin Peterson | 5c8da86 | 2009-06-30 22:57:08 +0000 | [diff] [blame] | 93 | unless = self.assertTrue |
Raymond Hettinger | 352f947 | 2003-04-24 15:50:11 +0000 | [diff] [blame] | 94 | eq(iter(self._fp), self._fp) |
Barry Warsaw | 4565350 | 2001-09-25 21:40:04 +0000 | [diff] [blame] | 95 | # Does this object support the iteration protocol? |
Raymond Hettinger | 352f947 | 2003-04-24 15:50:11 +0000 | [diff] [blame] | 96 | unless(hasattr(self._fp, '__iter__')) |
| 97 | unless(hasattr(self._fp, 'next')) |
Barry Warsaw | 7f8ff47 | 2001-09-22 04:33:47 +0000 | [diff] [blame] | 98 | i = 0 |
| 99 | for line in self._fp: |
| 100 | eq(line, self._line + '\n') |
| 101 | i += 1 |
| 102 | eq(i, 5) |
Walter Dörwald | 0af5d93 | 2006-03-15 08:23:53 +0000 | [diff] [blame] | 103 | self._fp.close() |
| 104 | self.assertRaises(ValueError, self._fp.next) |
Barry Warsaw | 7f8ff47 | 2001-09-22 04:33:47 +0000 | [diff] [blame] | 105 | |
Benjamin Peterson | 4fa68ab | 2011-05-26 09:56:41 -0500 | [diff] [blame] | 106 | def test_getvalue(self): |
| 107 | self._fp.close() |
| 108 | self.assertRaises(ValueError, self._fp.getvalue) |
| 109 | |
Serhiy Storchaka | 276f1d5 | 2013-02-09 13:47:43 +0200 | [diff] [blame] | 110 | @test_support.bigmemtest(test_support._2G + 2**26, memuse=2.001) |
| 111 | def test_reads_from_large_stream(self, size): |
| 112 | linesize = 2**26 # 64 MiB |
| 113 | lines = ['x' * (linesize - 1) + '\n'] * (size // linesize) + \ |
| 114 | ['y' * (size % linesize)] |
| 115 | f = self.MODULE.StringIO(''.join(lines)) |
| 116 | for i, expected in enumerate(lines): |
| 117 | line = f.read(len(expected)) |
| 118 | self.assertEqual(len(line), len(expected)) |
| 119 | self.assertEqual(line, expected) |
| 120 | self.assertEqual(f.read(), '') |
| 121 | f.seek(0) |
| 122 | for i, expected in enumerate(lines): |
| 123 | line = f.readline() |
| 124 | self.assertEqual(len(line), len(expected)) |
| 125 | self.assertEqual(line, expected) |
| 126 | self.assertEqual(f.readline(), '') |
| 127 | f.seek(0) |
| 128 | self.assertEqual(f.readlines(), lines) |
| 129 | self.assertEqual(f.readlines(), []) |
| 130 | f.seek(0) |
| 131 | self.assertEqual(f.readlines(size), lines) |
| 132 | self.assertEqual(f.readlines(), []) |
| 133 | |
| 134 | # In worst case cStringIO requires 2 + 1 + 1/2 + 1/2**2 + ... = 4 |
| 135 | # bytes per input character. |
| 136 | @test_support.bigmemtest(test_support._2G, memuse=4) |
| 137 | def test_writes_to_large_stream(self, size): |
| 138 | s = 'x' * 2**26 # 64 MiB |
| 139 | f = self.MODULE.StringIO() |
| 140 | n = size |
| 141 | while n > len(s): |
| 142 | f.write(s) |
| 143 | n -= len(s) |
| 144 | s = None |
| 145 | f.write('x' * n) |
| 146 | self.assertEqual(len(f.getvalue()), size) |
| 147 | |
| 148 | |
Barry Warsaw | 7f8ff47 | 2001-09-22 04:33:47 +0000 | [diff] [blame] | 149 | class TestStringIO(TestGenericStringIO): |
| 150 | MODULE = StringIO |
| 151 | |
Marc-André Lemburg | f853be9 | 2002-01-06 17:15:05 +0000 | [diff] [blame] | 152 | def test_unicode(self): |
| 153 | |
Michael W. Hudson | e1c67d1 | 2002-05-13 09:42:16 +0000 | [diff] [blame] | 154 | if not test_support.have_unicode: return |
| 155 | |
Marc-André Lemburg | f853be9 | 2002-01-06 17:15:05 +0000 | [diff] [blame] | 156 | # The StringIO module also supports concatenating Unicode |
| 157 | # snippets to larger Unicode strings. This is tested by this |
| 158 | # method. Note that cStringIO does not support this extension. |
Tim Peters | e441860 | 2002-02-16 07:34:19 +0000 | [diff] [blame] | 159 | |
Marc-André Lemburg | f853be9 | 2002-01-06 17:15:05 +0000 | [diff] [blame] | 160 | f = self.MODULE.StringIO() |
| 161 | f.write(self._line[:6]) |
| 162 | f.seek(3) |
| 163 | f.write(unicode(self._line[20:26])) |
| 164 | f.write(unicode(self._line[52])) |
| 165 | s = f.getvalue() |
| 166 | self.assertEqual(s, unicode('abcuvwxyz!')) |
| 167 | self.assertEqual(type(s), types.UnicodeType) |
| 168 | |
Barry Warsaw | 7f8ff47 | 2001-09-22 04:33:47 +0000 | [diff] [blame] | 169 | class TestcStringIO(TestGenericStringIO): |
| 170 | MODULE = cStringIO |
| 171 | |
Antoine Pitrou | a5016cd | 2011-10-21 22:22:04 +0200 | [diff] [blame] | 172 | def test_array_support(self): |
Antoine Pitrou | f678e82 | 2011-10-21 22:22:43 +0200 | [diff] [blame] | 173 | # Issue #1730114: cStringIO should accept array objects |
Antoine Pitrou | a5016cd | 2011-10-21 22:22:04 +0200 | [diff] [blame] | 174 | a = array.array('B', [0,1,2]) |
| 175 | f = self.MODULE.StringIO(a) |
| 176 | self.assertEqual(f.getvalue(), '\x00\x01\x02') |
| 177 | |
Antoine Pitrou | 5a77fe9 | 2011-10-22 21:26:01 +0200 | [diff] [blame] | 178 | def test_unicode(self): |
| 179 | |
| 180 | if not test_support.have_unicode: return |
| 181 | |
| 182 | # The cStringIO module converts Unicode strings to character |
| 183 | # strings when writing them to cStringIO objects. |
| 184 | # Check that this works. |
| 185 | |
| 186 | f = self.MODULE.StringIO() |
| 187 | f.write(u'abcde') |
| 188 | s = f.getvalue() |
| 189 | self.assertEqual(s, 'abcde') |
| 190 | self.assertEqual(type(s), str) |
| 191 | |
| 192 | f = self.MODULE.StringIO(u'abcde') |
| 193 | s = f.getvalue() |
| 194 | self.assertEqual(s, 'abcde') |
| 195 | self.assertEqual(type(s), str) |
| 196 | |
| 197 | self.assertRaises(UnicodeEncodeError, self.MODULE.StringIO, u'\xf4') |
| 198 | |
Antoine Pitrou | a5016cd | 2011-10-21 22:22:04 +0200 | [diff] [blame] | 199 | |
Finn Bock | 793ead5 | 2001-12-09 20:06:32 +0000 | [diff] [blame] | 200 | import sys |
| 201 | if sys.platform.startswith('java'): |
| 202 | # Jython doesn't have a buffer object, so we just do a useless |
| 203 | # fake of the buffer tests. |
| 204 | buffer = str |
| 205 | |
Marc-André Lemburg | e47df7a | 2001-09-24 17:34:52 +0000 | [diff] [blame] | 206 | class TestBufferStringIO(TestStringIO): |
| 207 | constructor = buffer |
| 208 | |
| 209 | class TestBuffercStringIO(TestcStringIO): |
| 210 | constructor = buffer |
| 211 | |
Kristján Valur Jónsson | 9795ca4 | 2013-03-19 17:17:47 -0700 | [diff] [blame] | 212 | class TestMemoryviewcStringIO(TestcStringIO): |
| 213 | constructor = memoryview |
| 214 | |
Barry Warsaw | 7f8ff47 | 2001-09-22 04:33:47 +0000 | [diff] [blame] | 215 | |
| 216 | def test_main(): |
Florent Xicluna | 0762788 | 2010-03-21 01:14:24 +0000 | [diff] [blame] | 217 | test_support.run_unittest(TestStringIO, TestcStringIO) |
| 218 | with test_support.check_py3k_warnings(("buffer.. not supported", |
| 219 | DeprecationWarning)): |
| 220 | test_support.run_unittest(TestBufferStringIO, TestBuffercStringIO) |
Kristján Valur Jónsson | 9795ca4 | 2013-03-19 17:17:47 -0700 | [diff] [blame] | 221 | test_support.run_unittest(TestMemoryviewcStringIO) |
Barry Warsaw | 7f8ff47 | 2001-09-22 04:33:47 +0000 | [diff] [blame] | 222 | |
| 223 | if __name__ == '__main__': |
| 224 | test_main() |