blob: c77dc5522a421d8f6e08d6a6202705b00c60c80e [file] [log] [blame]
Benjamin Petersone549ead2009-03-28 21:42:05 +00001from test.support import run_unittest, verbose
Antoine Pitrou83d6a872008-07-25 21:45:08 +00002import unittest
Martin v. Löwis88ad12a2001-04-13 08:09:50 +00003import locale
Guido van Rossumfc349862001-04-15 13:15:56 +00004import sys
Antoine Pitrou83d6a872008-07-25 21:45:08 +00005import codecs
Martin v. Löwis88ad12a2001-04-13 08:09:50 +00006
Antoine Pitrou13856ea2008-07-26 21:02:53 +00007enUS_locale = None
8
9def get_enUS_locale():
10 global enUS_locale
11 if sys.platform == 'darwin':
Benjamin Petersone549ead2009-03-28 21:42:05 +000012 raise unittest.SkipTest("Locale support on MacOSX is minimal")
Antoine Pitrou13856ea2008-07-26 21:02:53 +000013 if sys.platform.startswith("win"):
14 tlocs = ("En", "English")
15 else:
16 tlocs = ("en_US.UTF-8", "en_US.US-ASCII", "en_US")
17 oldlocale = locale.setlocale(locale.LC_NUMERIC)
18 for tloc in tlocs:
19 try:
20 locale.setlocale(locale.LC_NUMERIC, tloc)
21 except locale.Error:
22 continue
23 break
24 else:
Benjamin Petersone549ead2009-03-28 21:42:05 +000025 raise unittest.SkipTest(
Antoine Pitrou13856ea2008-07-26 21:02:53 +000026 "Test locale not supported (tried %s)" % (', '.join(tlocs)))
27 enUS_locale = tloc
28 locale.setlocale(locale.LC_NUMERIC, oldlocale)
29
30
Antoine Pitrou83d6a872008-07-25 21:45:08 +000031class BaseLocalizedTest(unittest.TestCase):
32 #
33 # Base class for tests using a real locale
34 #
Martin v. Löwis88ad12a2001-04-13 08:09:50 +000035
Antoine Pitrou83d6a872008-07-25 21:45:08 +000036 def setUp(self):
Antoine Pitrou83d6a872008-07-25 21:45:08 +000037 self.oldlocale = locale.setlocale(self.locale_type)
Antoine Pitrou13856ea2008-07-26 21:02:53 +000038 locale.setlocale(self.locale_type, enUS_locale)
Martin v. Löwis88ad12a2001-04-13 08:09:50 +000039 if verbose:
Antoine Pitrou13856ea2008-07-26 21:02:53 +000040 print("testing with \"%s\"..." % enUS_locale, end=' ')
Martin v. Löwis88ad12a2001-04-13 08:09:50 +000041
Antoine Pitrou83d6a872008-07-25 21:45:08 +000042 def tearDown(self):
43 locale.setlocale(self.locale_type, self.oldlocale)
Thomas Wouters477c8d52006-05-27 19:21:47 +000044
Thomas Wouters477c8d52006-05-27 19:21:47 +000045
Antoine Pitrou83d6a872008-07-25 21:45:08 +000046class BaseCookedTest(unittest.TestCase):
47 #
48 # Base class for tests using cooked localeconv() values
49 #
Georg Brandl3dbca812008-07-23 16:10:53 +000050
Antoine Pitrou83d6a872008-07-25 21:45:08 +000051 def setUp(self):
52 locale._override_localeconv = self.cooked_values
53
54 def tearDown(self):
55 locale._override_localeconv = {}
56
57class CCookedTest(BaseCookedTest):
58 # A cooked "C" locale
59
60 cooked_values = {
61 'currency_symbol': '',
62 'decimal_point': '.',
63 'frac_digits': 127,
64 'grouping': [],
65 'int_curr_symbol': '',
66 'int_frac_digits': 127,
67 'mon_decimal_point': '',
68 'mon_grouping': [],
69 'mon_thousands_sep': '',
70 'n_cs_precedes': 127,
71 'n_sep_by_space': 127,
72 'n_sign_posn': 127,
73 'negative_sign': '',
74 'p_cs_precedes': 127,
75 'p_sep_by_space': 127,
76 'p_sign_posn': 127,
77 'positive_sign': '',
78 'thousands_sep': ''
79 }
80
81class EnUSCookedTest(BaseCookedTest):
82 # A cooked "en_US" locale
83
84 cooked_values = {
85 'currency_symbol': '$',
86 'decimal_point': '.',
87 'frac_digits': 2,
88 'grouping': [3, 3, 0],
89 'int_curr_symbol': 'USD ',
90 'int_frac_digits': 2,
91 'mon_decimal_point': '.',
92 'mon_grouping': [3, 3, 0],
93 'mon_thousands_sep': ',',
94 'n_cs_precedes': 1,
95 'n_sep_by_space': 0,
96 'n_sign_posn': 1,
97 'negative_sign': '-',
98 'p_cs_precedes': 1,
99 'p_sep_by_space': 0,
100 'p_sign_posn': 1,
101 'positive_sign': '',
102 'thousands_sep': ','
103 }
104
105
Antoine Pitrou350370c2009-03-14 00:13:13 +0000106class FrFRCookedTest(BaseCookedTest):
107 # A cooked "fr_FR" locale with a space character as decimal separator
108 # and a non-ASCII currency symbol.
109
110 cooked_values = {
111 'currency_symbol': '\u20ac',
112 'decimal_point': ',',
113 'frac_digits': 2,
114 'grouping': [3, 3, 0],
115 'int_curr_symbol': 'EUR ',
116 'int_frac_digits': 2,
117 'mon_decimal_point': ',',
118 'mon_grouping': [3, 3, 0],
119 'mon_thousands_sep': ' ',
120 'n_cs_precedes': 0,
121 'n_sep_by_space': 1,
122 'n_sign_posn': 1,
123 'negative_sign': '-',
124 'p_cs_precedes': 0,
125 'p_sep_by_space': 1,
126 'p_sign_posn': 1,
127 'positive_sign': '',
128 'thousands_sep': ' '
129 }
130
131
Antoine Pitrou83d6a872008-07-25 21:45:08 +0000132class BaseFormattingTest(object):
133 #
134 # Utility functions for formatting tests
135 #
136
137 def _test_formatfunc(self, format, value, out, func, **format_opts):
138 self.assertEqual(
139 func(format, value, **format_opts), out)
140
141 def _test_format(self, format, value, out, **format_opts):
142 self._test_formatfunc(format, value, out,
143 func=locale.format, **format_opts)
144
145 def _test_format_string(self, format, value, out, **format_opts):
146 self._test_formatfunc(format, value, out,
147 func=locale.format_string, **format_opts)
148
149 def _test_currency(self, value, out, **format_opts):
150 self.assertEqual(locale.currency(value, **format_opts), out)
151
152
153class EnUSNumberFormatting(BaseFormattingTest):
Antoine Pitrou13856ea2008-07-26 21:02:53 +0000154 # XXX there is a grouping + padding bug when the thousands separator
155 # is empty but the grouping array contains values (e.g. Solaris 10)
Antoine Pitrou83d6a872008-07-25 21:45:08 +0000156
157 def setUp(self):
Antoine Pitrou83d6a872008-07-25 21:45:08 +0000158 self.sep = locale.localeconv()['thousands_sep']
159
160 def test_grouping(self):
161 self._test_format("%f", 1024, grouping=1, out='1%s024.000000' % self.sep)
162 self._test_format("%f", 102, grouping=1, out='102.000000')
163 self._test_format("%f", -42, grouping=1, out='-42.000000')
164 self._test_format("%+f", -42, grouping=1, out='-42.000000')
165
166 def test_grouping_and_padding(self):
167 self._test_format("%20.f", -42, grouping=1, out='-42'.rjust(20))
Antoine Pitrou13856ea2008-07-26 21:02:53 +0000168 if self.sep:
169 self._test_format("%+10.f", -4200, grouping=1,
170 out=('-4%s200' % self.sep).rjust(10))
171 self._test_format("%-10.f", -4200, grouping=1,
172 out=('-4%s200' % self.sep).ljust(10))
Antoine Pitrou83d6a872008-07-25 21:45:08 +0000173
174 def test_integer_grouping(self):
175 self._test_format("%d", 4200, grouping=True, out='4%s200' % self.sep)
176 self._test_format("%+d", 4200, grouping=True, out='+4%s200' % self.sep)
177 self._test_format("%+d", -4200, grouping=True, out='-4%s200' % self.sep)
178
Antoine Pitrou350370c2009-03-14 00:13:13 +0000179 def test_integer_grouping_and_padding(self):
180 self._test_format("%10d", 4200, grouping=True,
181 out=('4%s200' % self.sep).rjust(10))
182 self._test_format("%-10d", -4200, grouping=True,
183 out=('-4%s200' % self.sep).ljust(10))
184
Antoine Pitrou83d6a872008-07-25 21:45:08 +0000185 def test_simple(self):
186 self._test_format("%f", 1024, grouping=0, out='1024.000000')
187 self._test_format("%f", 102, grouping=0, out='102.000000')
188 self._test_format("%f", -42, grouping=0, out='-42.000000')
189 self._test_format("%+f", -42, grouping=0, out='-42.000000')
190
191 def test_padding(self):
192 self._test_format("%20.f", -42, grouping=0, out='-42'.rjust(20))
193 self._test_format("%+10.f", -4200, grouping=0, out='-4200'.rjust(10))
194 self._test_format("%-10.f", 4200, grouping=0, out='4200'.ljust(10))
195
196 def test_complex_formatting(self):
197 # Spaces in formatting string
198 self._test_format_string("One million is %i", 1000000, grouping=1,
199 out='One million is 1%s000%s000' % (self.sep, self.sep))
200 self._test_format_string("One million is %i", 1000000, grouping=1,
201 out='One million is 1%s000%s000' % (self.sep, self.sep))
202 # Dots in formatting string
203 self._test_format_string(".%f.", 1000.0, out='.1000.000000.')
204 # Padding
Antoine Pitrou13856ea2008-07-26 21:02:53 +0000205 if self.sep:
206 self._test_format_string("--> %10.2f", 4200, grouping=1,
207 out='--> ' + ('4%s200.00' % self.sep).rjust(10))
Antoine Pitrou83d6a872008-07-25 21:45:08 +0000208 # Asterisk formats
209 self._test_format_string("%10.*f", (2, 1000), grouping=0,
210 out='1000.00'.rjust(10))
Antoine Pitrou13856ea2008-07-26 21:02:53 +0000211 if self.sep:
212 self._test_format_string("%*.*f", (10, 2, 1000), grouping=1,
213 out=('1%s000.00' % self.sep).rjust(10))
Antoine Pitrou83d6a872008-07-25 21:45:08 +0000214 # Test more-in-one
Antoine Pitrou13856ea2008-07-26 21:02:53 +0000215 if self.sep:
216 self._test_format_string("int %i float %.2f str %s",
217 (1000, 1000.0, 'str'), grouping=1,
218 out='int 1%s000 float 1%s000.00 str str' %
219 (self.sep, self.sep))
Antoine Pitrou83d6a872008-07-25 21:45:08 +0000220
221
222class TestNumberFormatting(BaseLocalizedTest, EnUSNumberFormatting):
223 # Test number formatting with a real English locale.
224
225 locale_type = locale.LC_NUMERIC
226
227 def setUp(self):
228 BaseLocalizedTest.setUp(self)
229 EnUSNumberFormatting.setUp(self)
230
231
232class TestEnUSNumberFormatting(EnUSCookedTest, EnUSNumberFormatting):
233 # Test number formatting with a cooked "en_US" locale.
234
235 def setUp(self):
236 EnUSCookedTest.setUp(self)
237 EnUSNumberFormatting.setUp(self)
238
239 def test_currency(self):
240 self._test_currency(50000, "$50000.00")
241 self._test_currency(50000, "$50,000.00", grouping=True)
242 self._test_currency(50000, "USD 50,000.00",
243 grouping=True, international=True)
244
245
246class TestCNumberFormatting(CCookedTest, BaseFormattingTest):
247 # Test number formatting with a cooked "C" locale.
248
249 def test_grouping(self):
250 self._test_format("%.2f", 12345.67, grouping=True, out='12345.67')
251
252 def test_grouping_and_padding(self):
253 self._test_format("%9.2f", 12345.67, grouping=True, out=' 12345.67')
254
255
Antoine Pitrou350370c2009-03-14 00:13:13 +0000256class TestFrFRNumberFormatting(FrFRCookedTest, BaseFormattingTest):
257 # Test number formatting with a cooked "fr_FR" locale.
258
259 def test_decimal_point(self):
260 self._test_format("%.2f", 12345.67, out='12345,67')
261
262 def test_grouping(self):
263 self._test_format("%.2f", 345.67, grouping=True, out='345,67')
264 self._test_format("%.2f", 12345.67, grouping=True, out='12 345,67')
265
266 def test_grouping_and_padding(self):
267 self._test_format("%6.2f", 345.67, grouping=True, out='345,67')
268 self._test_format("%7.2f", 345.67, grouping=True, out=' 345,67')
269 self._test_format("%8.2f", 12345.67, grouping=True, out='12 345,67')
270 self._test_format("%9.2f", 12345.67, grouping=True, out='12 345,67')
271 self._test_format("%10.2f", 12345.67, grouping=True, out=' 12 345,67')
272 self._test_format("%-6.2f", 345.67, grouping=True, out='345,67')
273 self._test_format("%-7.2f", 345.67, grouping=True, out='345,67 ')
274 self._test_format("%-8.2f", 12345.67, grouping=True, out='12 345,67')
275 self._test_format("%-9.2f", 12345.67, grouping=True, out='12 345,67')
276 self._test_format("%-10.2f", 12345.67, grouping=True, out='12 345,67 ')
277
278 def test_integer_grouping(self):
279 self._test_format("%d", 200, grouping=True, out='200')
280 self._test_format("%d", 4200, grouping=True, out='4 200')
281
282 def test_integer_grouping_and_padding(self):
283 self._test_format("%4d", 4200, grouping=True, out='4 200')
284 self._test_format("%5d", 4200, grouping=True, out='4 200')
285 self._test_format("%10d", 4200, grouping=True, out='4 200'.rjust(10))
286 self._test_format("%-4d", 4200, grouping=True, out='4 200')
287 self._test_format("%-5d", 4200, grouping=True, out='4 200')
288 self._test_format("%-10d", 4200, grouping=True, out='4 200'.ljust(10))
289
290 def test_currency(self):
291 euro = '\u20ac'
292 self._test_currency(50000, "50000,00 " + euro)
293 self._test_currency(50000, "50 000,00 " + euro, grouping=True)
294 # XXX is the trailing space a bug?
295 self._test_currency(50000, "50 000,00 EUR ",
296 grouping=True, international=True)
297
298
Antoine Pitrou83d6a872008-07-25 21:45:08 +0000299class TestMiscellaneous(unittest.TestCase):
300 def test_getpreferredencoding(self):
301 # Invoke getpreferredencoding to make sure it does not cause exceptions.
302 enc = locale.getpreferredencoding()
303 if enc:
304 # If encoding non-empty, make sure it is valid
305 codecs.lookup(enc)
306
307 if hasattr(locale, "strcoll"):
308 def test_strcoll_3303(self):
309 # test crasher from bug #3303
310 self.assertRaises(TypeError, locale.strcoll, "a", None)
311 self.assertRaises(TypeError, locale.strcoll, b"a", None)
312
313
314def test_main():
Antoine Pitrou13856ea2008-07-26 21:02:53 +0000315 tests = [
316 TestMiscellaneous,
317 TestEnUSNumberFormatting,
Antoine Pitrou350370c2009-03-14 00:13:13 +0000318 TestCNumberFormatting,
319 TestFrFRNumberFormatting,
Antoine Pitrou13856ea2008-07-26 21:02:53 +0000320 ]
Benjamin Petersone549ead2009-03-28 21:42:05 +0000321 # SkipTest can't be raised inside unittests, handle it manually instead
Antoine Pitrou13856ea2008-07-26 21:02:53 +0000322 try:
323 get_enUS_locale()
Benjamin Petersone549ead2009-03-28 21:42:05 +0000324 except unittest.SkipTest as e:
Antoine Pitrou13856ea2008-07-26 21:02:53 +0000325 if verbose:
326 print("Some tests will be disabled: %s" % e)
327 else:
328 tests += [TestNumberFormatting]
329 run_unittest(*tests)
Antoine Pitrou83d6a872008-07-25 21:45:08 +0000330
331if __name__ == '__main__':
332 test_main()