blob: 7ea46bb52b89e20d86549f409f170c7e195b60a8 [file] [log] [blame]
Neal Norwitz7dbd2a32007-06-09 03:36:34 +00001
Walter Dörwald0fd583c2003-02-21 12:53:50 +00002import unittest
Neal Norwitzba965de2007-06-11 02:14:39 +00003import struct
Neal Norwitz7dbd2a32007-06-09 03:36:34 +00004import sys
Walter Dörwald0fd583c2003-02-21 12:53:50 +00005from test import test_support, string_tests
6
7
8class StrTest(
9 string_tests.CommonTest,
10 string_tests.MixinStrUnicodeUserStringTest,
Walter Dörwald57d88e52004-08-26 16:53:04 +000011 string_tests.MixinStrUserStringTest,
12 string_tests.MixinStrUnicodeTest,
Walter Dörwald0fd583c2003-02-21 12:53:50 +000013 ):
14
15 type2test = str
16
17 # We don't need to propagate to str
18 def fixtype(self, obj):
19 return obj
20
Walter Dörwald43440a62003-03-31 18:07:50 +000021 def test_formatting(self):
22 string_tests.MixinStrUnicodeUserStringTest.test_formatting(self)
23 self.assertRaises(OverflowError, '%c'.__mod__, 0x1234)
24
Brett Cannonc3647ac2005-04-26 03:45:26 +000025 def test_conversion(self):
26 # Make sure __str__() behaves properly
27 class Foo0:
28 def __unicode__(self):
29 return u"foo"
30
31 class Foo1:
32 def __str__(self):
33 return "foo"
34
35 class Foo2(object):
36 def __str__(self):
37 return "foo"
38
39 class Foo3(object):
40 def __str__(self):
41 return u"foo"
42
43 class Foo4(unicode):
44 def __str__(self):
45 return u"foo"
46
47 class Foo5(str):
48 def __str__(self):
49 return u"foo"
50
51 class Foo6(str):
52 def __str__(self):
53 return "foos"
54
55 def __unicode__(self):
56 return u"foou"
57
58 class Foo7(unicode):
59 def __str__(self):
60 return "foos"
61 def __unicode__(self):
62 return u"foou"
63
64 class Foo8(str):
65 def __new__(cls, content=""):
66 return str.__new__(cls, 2*content)
67 def __str__(self):
68 return self
69
70 class Foo9(str):
71 def __str__(self):
72 return "string"
73 def __unicode__(self):
74 return "not unicode"
75
76 self.assert_(str(Foo0()).startswith("<")) # this is different from __unicode__
77 self.assertEqual(str(Foo1()), "foo")
78 self.assertEqual(str(Foo2()), "foo")
79 self.assertEqual(str(Foo3()), "foo")
80 self.assertEqual(str(Foo4("bar")), "foo")
81 self.assertEqual(str(Foo5("bar")), "foo")
82 self.assertEqual(str(Foo6("bar")), "foos")
83 self.assertEqual(str(Foo7("bar")), "foos")
84 self.assertEqual(str(Foo8("foo")), "foofoo")
85 self.assertEqual(str(Foo9("foo")), "string")
86 self.assertEqual(unicode(Foo9("foo")), u"not unicode")
87
Neal Norwitz7dbd2a32007-06-09 03:36:34 +000088 def test_expandtabs_overflows_gracefully(self):
89 # This test only affects 32-bit platforms because expandtabs can only take
90 # an int as the max value, not a 64-bit C long. If expandtabs is changed
91 # to take a 64-bit long, this test should apply to all platforms.
Neal Norwitzba965de2007-06-11 02:14:39 +000092 if sys.maxint > (1 << 32) or struct.calcsize('P') != 4:
Neal Norwitz7dbd2a32007-06-09 03:36:34 +000093 return
94 self.assertRaises(OverflowError, 't\tt\t'.expandtabs, sys.maxint)
95
Eric Smitha9f7d622008-02-17 19:46:49 +000096 def test__format__(self):
97 def test(value, format, expected):
98 # test both with and without the trailing 's'
99 self.assertEqual(value.__format__(format), expected)
100 self.assertEqual(value.__format__(format + 's'), expected)
101
102 test('', '', '')
103 test('abc', '', 'abc')
104 test('abc', '.3', 'abc')
105 test('ab', '.3', 'ab')
106 test('abcdef', '.3', 'abc')
107 test('abcdef', '.0', '')
108 test('abc', '3.3', 'abc')
109 test('abc', '2.3', 'abc')
110 test('abc', '2.2', 'ab')
111 test('abc', '3.2', 'ab ')
112 test('result', 'x<0', 'result')
113 test('result', 'x<5', 'result')
114 test('result', 'x<6', 'result')
115 test('result', 'x<7', 'resultx')
116 test('result', 'x<8', 'resultxx')
117 test('result', ' <7', 'result ')
118 test('result', '<7', 'result ')
119 test('result', '>7', ' result')
120 test('result', '>8', ' result')
121 test('result', '^8', ' result ')
122 test('result', '^9', ' result ')
123 test('result', '^10', ' result ')
124 test('a', '10000', 'a' + ' ' * 9999)
125 test('', '10000', ' ' * 10000)
126 test('', '10000000', ' ' * 10000000)
127
128 def test_format(self):
129 self.assertEqual(''.format(), '')
130 self.assertEqual('a'.format(), 'a')
131 self.assertEqual('ab'.format(), 'ab')
132 self.assertEqual('a{{'.format(), 'a{')
133 self.assertEqual('a}}'.format(), 'a}')
134 self.assertEqual('{{b'.format(), '{b')
135 self.assertEqual('}}b'.format(), '}b')
136 self.assertEqual('a{{b'.format(), 'a{b')
137
138 # examples from the PEP:
139 import datetime
140 self.assertEqual("My name is {0}".format('Fred'), "My name is Fred")
141 self.assertEqual("My name is {0[name]}".format(dict(name='Fred')),
142 "My name is Fred")
143 self.assertEqual("My name is {0} :-{{}}".format('Fred'),
144 "My name is Fred :-{}")
145
146 d = datetime.date(2007, 8, 18)
147 self.assertEqual("The year is {0.year}".format(d),
148 "The year is 2007")
149
150 # classes we'll use for testing
151 class C:
152 def __init__(self, x=100):
153 self._x = x
154 def __format__(self, spec):
155 return spec
156
157 class D:
158 def __init__(self, x):
159 self.x = x
160 def __format__(self, spec):
161 return str(self.x)
162
163 # class with __str__, but no __format__
164 class E:
165 def __init__(self, x):
166 self.x = x
167 def __str__(self):
168 return 'E(' + self.x + ')'
169
170 # class with __repr__, but no __format__ or __str__
171 class F:
172 def __init__(self, x):
173 self.x = x
174 def __repr__(self):
175 return 'F(' + self.x + ')'
176
177 # class with __format__ that forwards to string, for some format_spec's
178 class G:
179 def __init__(self, x):
180 self.x = x
181 def __str__(self):
182 return "string is " + self.x
183 def __format__(self, format_spec):
184 if format_spec == 'd':
185 return 'G(' + self.x + ')'
186 return object.__format__(self, format_spec)
187
188 # class that returns a bad type from __format__
189 class H:
190 def __format__(self, format_spec):
191 return 1.0
192
193 class I(datetime.date):
194 def __format__(self, format_spec):
195 return self.strftime(format_spec)
196
197 class J(int):
198 def __format__(self, format_spec):
199 return int.__format__(self * 2, format_spec)
200
201
202 self.assertEqual(''.format(), '')
203 self.assertEqual('abc'.format(), 'abc')
204 self.assertEqual('{0}'.format('abc'), 'abc')
205 self.assertEqual('{0:}'.format('abc'), 'abc')
206 self.assertEqual('X{0}'.format('abc'), 'Xabc')
207 self.assertEqual('{0}X'.format('abc'), 'abcX')
208 self.assertEqual('X{0}Y'.format('abc'), 'XabcY')
209 self.assertEqual('{1}'.format(1, 'abc'), 'abc')
210 self.assertEqual('X{1}'.format(1, 'abc'), 'Xabc')
211 self.assertEqual('{1}X'.format(1, 'abc'), 'abcX')
212 self.assertEqual('X{1}Y'.format(1, 'abc'), 'XabcY')
213 self.assertEqual('{0}'.format(-15), '-15')
214 self.assertEqual('{0}{1}'.format(-15, 'abc'), '-15abc')
215 self.assertEqual('{0}X{1}'.format(-15, 'abc'), '-15Xabc')
216 self.assertEqual('{{'.format(), '{')
217 self.assertEqual('}}'.format(), '}')
218 self.assertEqual('{{}}'.format(), '{}')
219 self.assertEqual('{{x}}'.format(), '{x}')
220 self.assertEqual('{{{0}}}'.format(123), '{123}')
221 self.assertEqual('{{{{0}}}}'.format(), '{{0}}')
222 self.assertEqual('}}{{'.format(), '}{')
223 self.assertEqual('}}x{{'.format(), '}x{')
224
225 # weird field names
226 self.assertEqual("{0[foo-bar]}".format({'foo-bar':'baz'}), 'baz')
227 self.assertEqual("{0[foo bar]}".format({'foo bar':'baz'}), 'baz')
228 self.assertEqual("{0[ ]}".format({' ':3}), '3')
229
230 self.assertEqual('{foo._x}'.format(foo=C(20)), '20')
231 self.assertEqual('{1}{0}'.format(D(10), D(20)), '2010')
232 self.assertEqual('{0._x.x}'.format(C(D('abc'))), 'abc')
233 self.assertEqual('{0[0]}'.format(['abc', 'def']), 'abc')
234 self.assertEqual('{0[1]}'.format(['abc', 'def']), 'def')
235 self.assertEqual('{0[1][0]}'.format(['abc', ['def']]), 'def')
236 self.assertEqual('{0[1][0].x}'.format(['abc', [D('def')]]), 'def')
237
238 # strings
239 self.assertEqual('{0:.3s}'.format('abc'), 'abc')
240 self.assertEqual('{0:.3s}'.format('ab'), 'ab')
241 self.assertEqual('{0:.3s}'.format('abcdef'), 'abc')
242 self.assertEqual('{0:.0s}'.format('abcdef'), '')
243 self.assertEqual('{0:3.3s}'.format('abc'), 'abc')
244 self.assertEqual('{0:2.3s}'.format('abc'), 'abc')
245 self.assertEqual('{0:2.2s}'.format('abc'), 'ab')
246 self.assertEqual('{0:3.2s}'.format('abc'), 'ab ')
247 self.assertEqual('{0:x<0s}'.format('result'), 'result')
248 self.assertEqual('{0:x<5s}'.format('result'), 'result')
249 self.assertEqual('{0:x<6s}'.format('result'), 'result')
250 self.assertEqual('{0:x<7s}'.format('result'), 'resultx')
251 self.assertEqual('{0:x<8s}'.format('result'), 'resultxx')
252 self.assertEqual('{0: <7s}'.format('result'), 'result ')
253 self.assertEqual('{0:<7s}'.format('result'), 'result ')
254 self.assertEqual('{0:>7s}'.format('result'), ' result')
255 self.assertEqual('{0:>8s}'.format('result'), ' result')
256 self.assertEqual('{0:^8s}'.format('result'), ' result ')
257 self.assertEqual('{0:^9s}'.format('result'), ' result ')
258 self.assertEqual('{0:^10s}'.format('result'), ' result ')
259 self.assertEqual('{0:10000}'.format('a'), 'a' + ' ' * 9999)
260 self.assertEqual('{0:10000}'.format(''), ' ' * 10000)
261 self.assertEqual('{0:10000000}'.format(''), ' ' * 10000000)
262
263 # format specifiers for user defined type
264 self.assertEqual('{0:abc}'.format(C()), 'abc')
265
266 # !r and !s coersions
267 self.assertEqual('{0!s}'.format('Hello'), 'Hello')
268 self.assertEqual('{0!s:}'.format('Hello'), 'Hello')
269 self.assertEqual('{0!s:15}'.format('Hello'), 'Hello ')
270 self.assertEqual('{0!s:15s}'.format('Hello'), 'Hello ')
271 self.assertEqual('{0!r}'.format('Hello'), "'Hello'")
272 self.assertEqual('{0!r:}'.format('Hello'), "'Hello'")
273 self.assertEqual('{0!r}'.format(F('Hello')), 'F(Hello)')
274
275 # test fallback to object.__format__
276 self.assertEqual('{0}'.format({}), '{}')
277 self.assertEqual('{0}'.format([]), '[]')
278 self.assertEqual('{0}'.format([1]), '[1]')
279 self.assertEqual('{0}'.format(E('data')), 'E(data)')
280 self.assertEqual('{0:^10}'.format(E('data')), ' E(data) ')
281 self.assertEqual('{0:^10s}'.format(E('data')), ' E(data) ')
282 self.assertEqual('{0:d}'.format(G('data')), 'G(data)')
283 self.assertEqual('{0:>15s}'.format(G('data')), ' string is data')
284 self.assertEqual('{0!s}'.format(G('data')), 'string is data')
285
286 self.assertEqual("{0:date: %Y-%m-%d}".format(I(year=2007,
287 month=8,
288 day=27)),
289 "date: 2007-08-27")
290
291 # test deriving from a builtin type and overriding __format__
292 self.assertEqual("{0}".format(J(10)), "20")
293
294
295 # string format specifiers
296 self.assertEqual('{0:}'.format('a'), 'a')
297
298 # computed format specifiers
299 self.assertEqual("{0:.{1}}".format('hello world', 5), 'hello')
300 self.assertEqual("{0:.{1}s}".format('hello world', 5), 'hello')
301 self.assertEqual("{0:.{precision}s}".format('hello world', precision=5), 'hello')
302 self.assertEqual("{0:{width}.{precision}s}".format('hello world', width=10, precision=5), 'hello ')
303 self.assertEqual("{0:{width}.{precision}s}".format('hello world', width='10', precision='5'), 'hello ')
304
305 # test various errors
306 self.assertRaises(ValueError, '{'.format)
307 self.assertRaises(ValueError, '}'.format)
308 self.assertRaises(ValueError, 'a{'.format)
309 self.assertRaises(ValueError, 'a}'.format)
310 self.assertRaises(ValueError, '{a'.format)
311 self.assertRaises(ValueError, '}a'.format)
312 self.assertRaises(IndexError, '{0}'.format)
313 self.assertRaises(IndexError, '{1}'.format, 'abc')
314 self.assertRaises(KeyError, '{x}'.format)
315 self.assertRaises(ValueError, "}{".format)
316 self.assertRaises(ValueError, "{".format)
317 self.assertRaises(ValueError, "}".format)
318 self.assertRaises(ValueError, "abc{0:{}".format)
319 self.assertRaises(ValueError, "{0".format)
320 self.assertRaises(IndexError, "{0.}".format)
321 self.assertRaises(ValueError, "{0.}".format, 0)
322 self.assertRaises(IndexError, "{0[}".format)
323 self.assertRaises(ValueError, "{0[}".format, [])
324 self.assertRaises(KeyError, "{0]}".format)
325 self.assertRaises(ValueError, "{0.[]}".format, 0)
326 self.assertRaises(ValueError, "{0..foo}".format, 0)
327 self.assertRaises(ValueError, "{0[0}".format, 0)
328 self.assertRaises(ValueError, "{0[0:foo}".format, 0)
329 self.assertRaises(KeyError, "{c]}".format)
330 self.assertRaises(ValueError, "{{ {{{0}}".format, 0)
331 self.assertRaises(ValueError, "{0}}".format, 0)
332 self.assertRaises(KeyError, "{foo}".format, bar=3)
333 self.assertRaises(ValueError, "{0!x}".format, 3)
334 self.assertRaises(ValueError, "{0!}".format, 0)
335 self.assertRaises(ValueError, "{0!rs}".format, 0)
336 self.assertRaises(ValueError, "{!}".format)
337 self.assertRaises(ValueError, "{:}".format)
338 self.assertRaises(ValueError, "{:s}".format)
339 self.assertRaises(ValueError, "{}".format)
340
341 # can't have a replacement on the field name portion
342 self.assertRaises(TypeError, '{0[{1}]}'.format, 'abcdefg', 4)
343
344 # exceed maximum recursion depth
345 self.assertRaises(ValueError, "{0:{1:{2}}}".format, 'abc', 's', '')
346 self.assertRaises(ValueError, "{0:{1:{2:{3:{4:{5:{6}}}}}}}".format,
347 0, 1, 2, 3, 4, 5, 6, 7)
348
349 # string format spec errors
350 self.assertRaises(ValueError, "{0:-s}".format, '')
351 self.assertRaises(ValueError, format, "", "-")
352 self.assertRaises(ValueError, "{0:=s}".format, '')
353
Neal Norwitz7dbd2a32007-06-09 03:36:34 +0000354
Walter Dörwald0fd583c2003-02-21 12:53:50 +0000355def test_main():
Walter Dörwald21d3a322003-05-01 17:45:56 +0000356 test_support.run_unittest(StrTest)
Walter Dörwald0fd583c2003-02-21 12:53:50 +0000357
358if __name__ == "__main__":
359 test_main()