blob: 06ea90c207f56c98c0b0bd6c5389b0ca5f25c72f [file] [log] [blame]
Benjamin Peterson101447b2015-03-09 10:37:50 -04001import fractions
Christian Heimes53876d92008-04-19 00:31:39 +00002import operator
Benjamin Peterson101447b2015-03-09 10:37:50 -04003import os
4import random
5import sys
6import struct
Benjamin Petersona9157232015-03-06 09:08:44 -05007import time
Benjamin Peterson101447b2015-03-09 10:37:50 -04008import unittest
9
10from test import support
Brett Cannona721aba2016-09-09 14:57:09 -070011from test.test_grammar import (VALID_UNDERSCORE_LITERALS,
12 INVALID_UNDERSCORE_LITERALS)
Benjamin Peterson101447b2015-03-09 10:37:50 -040013from math import isinf, isnan, copysign, ldexp
Michael W. Hudsonba283e22005-05-27 15:23:20 +000014
Christian Heimes53876d92008-04-19 00:31:39 +000015INF = float("inf")
16NAN = float("nan")
Christian Heimes99170a52007-12-19 02:07:34 +000017
Benjamin Petersone401c682010-07-02 23:25:44 +000018have_getformat = hasattr(float, "__getformat__")
19requires_getformat = unittest.skipUnless(have_getformat,
20 "requires __getformat__")
21requires_setformat = unittest.skipUnless(hasattr(float, "__setformat__"),
22 "requires __setformat__")
Mark Dickinson9ab44b52009-12-30 16:22:49 +000023
Eric Smith0923d1d2009-04-16 20:16:10 +000024#locate file with float format test values
25test_dir = os.path.dirname(__file__) or os.curdir
26format_testfile = os.path.join(test_dir, 'formatfloat_testcases.txt')
27
Serhiy Storchaka15095802015-11-25 15:47:01 +020028class FloatSubclass(float):
29 pass
30
31class OtherFloatSubclass(float):
32 pass
33
Christian Heimes81ee3ef2008-05-04 22:42:01 +000034class GeneralFloatCases(unittest.TestCase):
35
36 def test_float(self):
37 self.assertEqual(float(3.14), 3.14)
38 self.assertEqual(float(314), 314.0)
39 self.assertEqual(float(" 3.14 "), 3.14)
40 self.assertRaises(ValueError, float, " 0x3.1 ")
41 self.assertRaises(ValueError, float, " -0x3.p-1 ")
42 self.assertRaises(ValueError, float, " +0x3.p-1 ")
43 self.assertRaises(ValueError, float, "++3.14")
44 self.assertRaises(ValueError, float, "+-3.14")
45 self.assertRaises(ValueError, float, "-+3.14")
46 self.assertRaises(ValueError, float, "--3.14")
Eric Smith0923d1d2009-04-16 20:16:10 +000047 self.assertRaises(ValueError, float, ".nan")
48 self.assertRaises(ValueError, float, "+.inf")
49 self.assertRaises(ValueError, float, ".")
50 self.assertRaises(ValueError, float, "-.")
Alexander Belopolsky942af5a2010-12-04 03:38:46 +000051 self.assertRaises(TypeError, float, {})
Ezio Melottia5b95992013-11-07 19:18:34 +020052 self.assertRaisesRegex(TypeError, "not 'dict'", float, {})
Alexander Belopolsky942af5a2010-12-04 03:38:46 +000053 # Lone surrogate
Serhiy Storchaka9b6c60c2017-11-13 21:23:48 +020054 self.assertRaises(ValueError, float, '\uD8F0')
Mark Dickinsonc2d86892010-02-12 21:18:34 +000055 # check that we don't accept alternate exponent markers
56 self.assertRaises(ValueError, float, "-1.7d29")
57 self.assertRaises(ValueError, float, "3D-14")
Alexander Belopolsky942af5a2010-12-04 03:38:46 +000058 self.assertEqual(float(" \u0663.\u0661\u0664 "), 3.14)
59 self.assertEqual(float("\N{EM SPACE}3.14\N{EN SPACE}"), 3.14)
Mark Dickinsona9023be2009-10-27 22:12:20 +000060 # extra long strings should not be a problem
61 float(b'.' + b'1'*1000)
62 float('.' + '1'*1000)
Miss Islington (bot)c7214722018-07-13 20:58:12 -070063 # Invalid unicode string
64 # See bpo-34087
65 self.assertRaises(ValueError, float, '\u3053\u3093\u306b\u3061\u306f')
Christian Heimes81ee3ef2008-05-04 22:42:01 +000066
Brett Cannona721aba2016-09-09 14:57:09 -070067 def test_underscores(self):
68 for lit in VALID_UNDERSCORE_LITERALS:
69 if not any(ch in lit for ch in 'jJxXoObB'):
70 self.assertEqual(float(lit), eval(lit))
71 self.assertEqual(float(lit), float(lit.replace('_', '')))
72 for lit in INVALID_UNDERSCORE_LITERALS:
73 if lit in ('0_7', '09_99'): # octals are not recognized here
74 continue
75 if not any(ch in lit for ch in 'jJxXoObB'):
76 self.assertRaises(ValueError, float, lit)
77 # Additional test cases; nan and inf are never valid as literals,
78 # only in the float() constructor, but we don't allow underscores
79 # in or around them.
80 self.assertRaises(ValueError, float, '_NaN')
81 self.assertRaises(ValueError, float, 'Na_N')
82 self.assertRaises(ValueError, float, 'IN_F')
83 self.assertRaises(ValueError, float, '-_INF')
84 self.assertRaises(ValueError, float, '-INF_')
85 # Check that we handle bytes values correctly.
86 self.assertRaises(ValueError, float, b'0_.\xff9')
87
Martin Pantereeb896c2015-11-07 02:32:21 +000088 def test_non_numeric_input_types(self):
89 # Test possible non-numeric types for the argument x, including
90 # subclasses of the explicitly documented accepted types.
91 class CustomStr(str): pass
92 class CustomBytes(bytes): pass
93 class CustomByteArray(bytearray): pass
94
95 factories = [
96 bytes,
97 bytearray,
98 lambda b: CustomStr(b.decode()),
99 CustomBytes,
100 CustomByteArray,
101 memoryview,
102 ]
103 try:
104 from array import array
105 except ImportError:
106 pass
107 else:
108 factories.append(lambda b: array('B', b))
109
110 for f in factories:
111 x = f(b" 3.14 ")
112 with self.subTest(type(x)):
113 self.assertEqual(float(x), 3.14)
114 with self.assertRaisesRegex(ValueError, "could not convert"):
115 float(f(b'A' * 0x10))
116
117 def test_float_memoryview(self):
118 self.assertEqual(float(memoryview(b'12.3')[1:4]), 2.3)
119 self.assertEqual(float(memoryview(b'12.3\x00')[1:4]), 2.3)
120 self.assertEqual(float(memoryview(b'12.3 ')[1:4]), 2.3)
121 self.assertEqual(float(memoryview(b'12.3A')[1:4]), 2.3)
122 self.assertEqual(float(memoryview(b'12.34')[1:4]), 2.3)
123
Alexander Belopolsky942af5a2010-12-04 03:38:46 +0000124 def test_error_message(self):
Serhiy Storchaka9e6ac832017-03-09 20:07:58 +0200125 def check(s):
126 with self.assertRaises(ValueError, msg='float(%r)' % (s,)) as cm:
Alexander Belopolsky942af5a2010-12-04 03:38:46 +0000127 float(s)
Serhiy Storchaka9e6ac832017-03-09 20:07:58 +0200128 self.assertEqual(str(cm.exception),
129 'could not convert string to float: %r' % (s,))
Alexander Belopolsky942af5a2010-12-04 03:38:46 +0000130
Serhiy Storchaka9e6ac832017-03-09 20:07:58 +0200131 check('\xbd')
132 check('123\xbd')
133 check(' 123 456 ')
134 check(b' 123 456 ')
135
136 # non-ascii digits (error came from non-digit '!')
137 check('\u0663\u0661\u0664!')
138 # embedded NUL
139 check('123\x00')
140 check('123\x00 245')
141 check('123\x00245')
142 # byte string with embedded NUL
143 check(b'123\x00')
144 # non-UTF-8 byte string
145 check(b'123\xa0')
Alexander Belopolsky942af5a2010-12-04 03:38:46 +0000146
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000147 @support.run_with_locale('LC_NUMERIC', 'fr_FR', 'de_DE')
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000148 def test_float_with_comma(self):
149 # set locale to something that doesn't use '.' for the decimal point
150 # float must not accept the locale specific decimal point but
Ezio Melotti13925002011-03-16 11:05:33 +0200151 # it still has to accept the normal python syntax
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000152 import locale
153 if not locale.localeconv()['decimal_point'] == ',':
Zachary Ware9fe6d862013-12-08 00:20:35 -0600154 self.skipTest('decimal_point is not ","')
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000155
156 self.assertEqual(float(" 3.14 "), 3.14)
157 self.assertEqual(float("+3.14 "), 3.14)
158 self.assertEqual(float("-3.14 "), -3.14)
159 self.assertEqual(float(".14 "), .14)
160 self.assertEqual(float("3. "), 3.0)
161 self.assertEqual(float("3.e3 "), 3000.0)
162 self.assertEqual(float("3.2e3 "), 3200.0)
163 self.assertEqual(float("2.5e-1 "), 0.25)
164 self.assertEqual(float("5e-1"), 0.5)
165 self.assertRaises(ValueError, float, " 3,14 ")
166 self.assertRaises(ValueError, float, " +3,14 ")
167 self.assertRaises(ValueError, float, " -3,14 ")
168 self.assertRaises(ValueError, float, " 0x3.1 ")
169 self.assertRaises(ValueError, float, " -0x3.p-1 ")
170 self.assertRaises(ValueError, float, " +0x3.p-1 ")
171 self.assertEqual(float(" 25.e-1 "), 2.5)
Eli Bendersky67ebabd2011-02-25 10:14:17 +0000172 self.assertAlmostEqual(float(" .25e-1 "), .025)
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000173
174 def test_floatconversion(self):
175 # Make sure that calls to __float__() work properly
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000176 class Foo1(object):
177 def __float__(self):
178 return 42.
179
180 class Foo2(float):
181 def __float__(self):
182 return 42.
183
184 class Foo3(float):
185 def __new__(cls, value=0.):
186 return float.__new__(cls, 2*value)
187
188 def __float__(self):
189 return self
190
191 class Foo4(float):
192 def __float__(self):
193 return 42
194
Benjamin Peterson2808d3c2009-04-15 21:34:27 +0000195 # Issue 5759: __float__ not called on str subclasses (though it is on
196 # unicode subclasses).
197 class FooStr(str):
198 def __float__(self):
199 return float(str(self)) + 1
200
Serhiy Storchaka16931c32016-06-03 21:42:55 +0300201 self.assertEqual(float(Foo1()), 42.)
202 self.assertEqual(float(Foo2()), 42.)
203 with self.assertWarns(DeprecationWarning):
204 self.assertEqual(float(Foo3(21)), 42.)
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000205 self.assertRaises(TypeError, float, Foo4(42))
Serhiy Storchaka16931c32016-06-03 21:42:55 +0300206 self.assertEqual(float(FooStr('8')), 9.)
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000207
Benjamin Petersona9157232015-03-06 09:08:44 -0500208 class Foo5:
209 def __float__(self):
210 return ""
211 self.assertRaises(TypeError, time.sleep, Foo5())
212
Serhiy Storchaka15095802015-11-25 15:47:01 +0200213 # Issue #24731
214 class F:
215 def __float__(self):
216 return OtherFloatSubclass(42.)
Serhiy Storchaka16931c32016-06-03 21:42:55 +0300217 with self.assertWarns(DeprecationWarning):
218 self.assertEqual(float(F()), 42.)
219 with self.assertWarns(DeprecationWarning):
220 self.assertIs(type(float(F())), float)
221 with self.assertWarns(DeprecationWarning):
222 self.assertEqual(FloatSubclass(F()), 42.)
223 with self.assertWarns(DeprecationWarning):
224 self.assertIs(type(FloatSubclass(F())), FloatSubclass)
Serhiy Storchaka15095802015-11-25 15:47:01 +0200225
Serhiy Storchaka58d23e62017-03-06 00:53:39 +0200226 def test_keyword_args(self):
Serhiy Storchaka2e564242017-03-06 17:01:06 +0200227 with self.assertRaisesRegex(TypeError, 'keyword argument'):
228 float(x='3.14')
Serhiy Storchaka58d23e62017-03-06 00:53:39 +0200229
Benjamin Petersonb3b8ba62011-10-28 19:42:48 -0400230 def test_is_integer(self):
231 self.assertFalse((1.1).is_integer())
232 self.assertTrue((1.).is_integer())
233 self.assertFalse(float("nan").is_integer())
234 self.assertFalse(float("inf").is_integer())
235
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000236 def test_floatasratio(self):
237 for f, ratio in [
238 (0.875, (7, 8)),
239 (-0.875, (-7, 8)),
240 (0.0, (0, 1)),
241 (11.5, (23, 2)),
242 ]:
243 self.assertEqual(f.as_integer_ratio(), ratio)
244
245 for i in range(10000):
246 f = random.random()
247 f *= 10 ** random.randint(-100, 100)
248 n, d = f.as_integer_ratio()
249 self.assertEqual(float(n).__truediv__(d), f)
250
251 R = fractions.Fraction
252 self.assertEqual(R(0, 1),
253 R(*float(0.0).as_integer_ratio()))
254 self.assertEqual(R(5, 2),
255 R(*float(2.5).as_integer_ratio()))
256 self.assertEqual(R(1, 2),
257 R(*float(0.5).as_integer_ratio()))
258 self.assertEqual(R(4728779608739021, 2251799813685248),
259 R(*float(2.1).as_integer_ratio()))
260 self.assertEqual(R(-4728779608739021, 2251799813685248),
261 R(*float(-2.1).as_integer_ratio()))
262 self.assertEqual(R(-2100, 1),
263 R(*float(-2100.0).as_integer_ratio()))
264
265 self.assertRaises(OverflowError, float('inf').as_integer_ratio)
266 self.assertRaises(OverflowError, float('-inf').as_integer_ratio)
267 self.assertRaises(ValueError, float('nan').as_integer_ratio)
268
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000269 def test_float_containment(self):
270 floats = (INF, -INF, 0.0, 1.0, NAN)
271 for f in floats:
Benjamin Peterson577473f2010-01-19 00:09:57 +0000272 self.assertIn(f, [f])
Benjamin Peterson577473f2010-01-19 00:09:57 +0000273 self.assertIn(f, (f,))
Benjamin Peterson577473f2010-01-19 00:09:57 +0000274 self.assertIn(f, {f})
Benjamin Peterson577473f2010-01-19 00:09:57 +0000275 self.assertIn(f, {f: None})
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000276 self.assertEqual([f].count(f), 1, "[].count('%r') != 1" % f)
Benjamin Peterson577473f2010-01-19 00:09:57 +0000277 self.assertIn(f, floats)
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000278
279 for f in floats:
280 # nonidentical containers, same type, same contents
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000281 self.assertTrue([f] == [f], "[%r] != [%r]" % (f, f))
282 self.assertTrue((f,) == (f,), "(%r,) != (%r,)" % (f, f))
283 self.assertTrue({f} == {f}, "{%r} != {%r}" % (f, f))
284 self.assertTrue({f : None} == {f: None}, "{%r : None} != "
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000285 "{%r : None}" % (f, f))
286
287 # identical containers
288 l, t, s, d = [f], (f,), {f}, {f: None}
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000289 self.assertTrue(l == l, "[%r] not equal to itself" % f)
290 self.assertTrue(t == t, "(%r,) not equal to itself" % f)
291 self.assertTrue(s == s, "{%r} not equal to itself" % f)
292 self.assertTrue(d == d, "{%r : None} not equal to itself" % f)
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000293
Mark Dickinson9ab44b52009-12-30 16:22:49 +0000294 def assertEqualAndEqualSign(self, a, b):
295 # fail unless a == b and a and b have the same sign bit;
296 # the only difference from assertEqual is that this test
Ezio Melotti13925002011-03-16 11:05:33 +0200297 # distinguishes -0.0 and 0.0.
Mark Dickinson9ab44b52009-12-30 16:22:49 +0000298 self.assertEqual((a, copysign(1.0, a)), (b, copysign(1.0, b)))
299
Eric Smith3ab08ca2010-12-04 15:17:38 +0000300 @support.requires_IEEE_754
Mark Dickinsond2a9b202010-12-04 12:25:30 +0000301 def test_float_mod(self):
302 # Check behaviour of % operator for IEEE 754 special cases.
303 # In particular, check signs of zeros.
304 mod = operator.mod
305
306 self.assertEqualAndEqualSign(mod(-1.0, 1.0), 0.0)
307 self.assertEqualAndEqualSign(mod(-1e-100, 1.0), 1.0)
308 self.assertEqualAndEqualSign(mod(-0.0, 1.0), 0.0)
309 self.assertEqualAndEqualSign(mod(0.0, 1.0), 0.0)
310 self.assertEqualAndEqualSign(mod(1e-100, 1.0), 1e-100)
311 self.assertEqualAndEqualSign(mod(1.0, 1.0), 0.0)
312
313 self.assertEqualAndEqualSign(mod(-1.0, -1.0), -0.0)
314 self.assertEqualAndEqualSign(mod(-1e-100, -1.0), -1e-100)
315 self.assertEqualAndEqualSign(mod(-0.0, -1.0), -0.0)
316 self.assertEqualAndEqualSign(mod(0.0, -1.0), -0.0)
317 self.assertEqualAndEqualSign(mod(1e-100, -1.0), -1.0)
318 self.assertEqualAndEqualSign(mod(1.0, -1.0), -0.0)
319
Eric Smith3ab08ca2010-12-04 15:17:38 +0000320 @support.requires_IEEE_754
Mark Dickinson9ab44b52009-12-30 16:22:49 +0000321 def test_float_pow(self):
322 # test builtin pow and ** operator for IEEE 754 special cases.
323 # Special cases taken from section F.9.4.4 of the C99 specification
324
325 for pow_op in pow, operator.pow:
326 # x**NAN is NAN for any x except 1
327 self.assertTrue(isnan(pow_op(-INF, NAN)))
328 self.assertTrue(isnan(pow_op(-2.0, NAN)))
329 self.assertTrue(isnan(pow_op(-1.0, NAN)))
330 self.assertTrue(isnan(pow_op(-0.5, NAN)))
331 self.assertTrue(isnan(pow_op(-0.0, NAN)))
332 self.assertTrue(isnan(pow_op(0.0, NAN)))
333 self.assertTrue(isnan(pow_op(0.5, NAN)))
334 self.assertTrue(isnan(pow_op(2.0, NAN)))
335 self.assertTrue(isnan(pow_op(INF, NAN)))
336 self.assertTrue(isnan(pow_op(NAN, NAN)))
337
338 # NAN**y is NAN for any y except +-0
339 self.assertTrue(isnan(pow_op(NAN, -INF)))
340 self.assertTrue(isnan(pow_op(NAN, -2.0)))
341 self.assertTrue(isnan(pow_op(NAN, -1.0)))
342 self.assertTrue(isnan(pow_op(NAN, -0.5)))
343 self.assertTrue(isnan(pow_op(NAN, 0.5)))
344 self.assertTrue(isnan(pow_op(NAN, 1.0)))
345 self.assertTrue(isnan(pow_op(NAN, 2.0)))
346 self.assertTrue(isnan(pow_op(NAN, INF)))
347
348 # (+-0)**y raises ZeroDivisionError for y a negative odd integer
349 self.assertRaises(ZeroDivisionError, pow_op, -0.0, -1.0)
350 self.assertRaises(ZeroDivisionError, pow_op, 0.0, -1.0)
351
352 # (+-0)**y raises ZeroDivisionError for y finite and negative
353 # but not an odd integer
354 self.assertRaises(ZeroDivisionError, pow_op, -0.0, -2.0)
355 self.assertRaises(ZeroDivisionError, pow_op, -0.0, -0.5)
356 self.assertRaises(ZeroDivisionError, pow_op, 0.0, -2.0)
357 self.assertRaises(ZeroDivisionError, pow_op, 0.0, -0.5)
358
359 # (+-0)**y is +-0 for y a positive odd integer
360 self.assertEqualAndEqualSign(pow_op(-0.0, 1.0), -0.0)
361 self.assertEqualAndEqualSign(pow_op(0.0, 1.0), 0.0)
362
363 # (+-0)**y is 0 for y finite and positive but not an odd integer
364 self.assertEqualAndEqualSign(pow_op(-0.0, 0.5), 0.0)
365 self.assertEqualAndEqualSign(pow_op(-0.0, 2.0), 0.0)
366 self.assertEqualAndEqualSign(pow_op(0.0, 0.5), 0.0)
367 self.assertEqualAndEqualSign(pow_op(0.0, 2.0), 0.0)
368
369 # (-1)**+-inf is 1
370 self.assertEqualAndEqualSign(pow_op(-1.0, -INF), 1.0)
371 self.assertEqualAndEqualSign(pow_op(-1.0, INF), 1.0)
372
373 # 1**y is 1 for any y, even if y is an infinity or nan
374 self.assertEqualAndEqualSign(pow_op(1.0, -INF), 1.0)
375 self.assertEqualAndEqualSign(pow_op(1.0, -2.0), 1.0)
376 self.assertEqualAndEqualSign(pow_op(1.0, -1.0), 1.0)
377 self.assertEqualAndEqualSign(pow_op(1.0, -0.5), 1.0)
378 self.assertEqualAndEqualSign(pow_op(1.0, -0.0), 1.0)
379 self.assertEqualAndEqualSign(pow_op(1.0, 0.0), 1.0)
380 self.assertEqualAndEqualSign(pow_op(1.0, 0.5), 1.0)
381 self.assertEqualAndEqualSign(pow_op(1.0, 1.0), 1.0)
382 self.assertEqualAndEqualSign(pow_op(1.0, 2.0), 1.0)
383 self.assertEqualAndEqualSign(pow_op(1.0, INF), 1.0)
384 self.assertEqualAndEqualSign(pow_op(1.0, NAN), 1.0)
385
386 # x**+-0 is 1 for any x, even if x is a zero, infinity, or nan
387 self.assertEqualAndEqualSign(pow_op(-INF, 0.0), 1.0)
388 self.assertEqualAndEqualSign(pow_op(-2.0, 0.0), 1.0)
389 self.assertEqualAndEqualSign(pow_op(-1.0, 0.0), 1.0)
390 self.assertEqualAndEqualSign(pow_op(-0.5, 0.0), 1.0)
391 self.assertEqualAndEqualSign(pow_op(-0.0, 0.0), 1.0)
392 self.assertEqualAndEqualSign(pow_op(0.0, 0.0), 1.0)
393 self.assertEqualAndEqualSign(pow_op(0.5, 0.0), 1.0)
394 self.assertEqualAndEqualSign(pow_op(1.0, 0.0), 1.0)
395 self.assertEqualAndEqualSign(pow_op(2.0, 0.0), 1.0)
396 self.assertEqualAndEqualSign(pow_op(INF, 0.0), 1.0)
397 self.assertEqualAndEqualSign(pow_op(NAN, 0.0), 1.0)
398 self.assertEqualAndEqualSign(pow_op(-INF, -0.0), 1.0)
399 self.assertEqualAndEqualSign(pow_op(-2.0, -0.0), 1.0)
400 self.assertEqualAndEqualSign(pow_op(-1.0, -0.0), 1.0)
401 self.assertEqualAndEqualSign(pow_op(-0.5, -0.0), 1.0)
402 self.assertEqualAndEqualSign(pow_op(-0.0, -0.0), 1.0)
403 self.assertEqualAndEqualSign(pow_op(0.0, -0.0), 1.0)
404 self.assertEqualAndEqualSign(pow_op(0.5, -0.0), 1.0)
405 self.assertEqualAndEqualSign(pow_op(1.0, -0.0), 1.0)
406 self.assertEqualAndEqualSign(pow_op(2.0, -0.0), 1.0)
407 self.assertEqualAndEqualSign(pow_op(INF, -0.0), 1.0)
408 self.assertEqualAndEqualSign(pow_op(NAN, -0.0), 1.0)
409
410 # x**y defers to complex pow for finite negative x and
411 # non-integral y.
412 self.assertEqual(type(pow_op(-2.0, -0.5)), complex)
413 self.assertEqual(type(pow_op(-2.0, 0.5)), complex)
414 self.assertEqual(type(pow_op(-1.0, -0.5)), complex)
415 self.assertEqual(type(pow_op(-1.0, 0.5)), complex)
416 self.assertEqual(type(pow_op(-0.5, -0.5)), complex)
417 self.assertEqual(type(pow_op(-0.5, 0.5)), complex)
418
419 # x**-INF is INF for abs(x) < 1
420 self.assertEqualAndEqualSign(pow_op(-0.5, -INF), INF)
421 self.assertEqualAndEqualSign(pow_op(-0.0, -INF), INF)
422 self.assertEqualAndEqualSign(pow_op(0.0, -INF), INF)
423 self.assertEqualAndEqualSign(pow_op(0.5, -INF), INF)
424
425 # x**-INF is 0 for abs(x) > 1
426 self.assertEqualAndEqualSign(pow_op(-INF, -INF), 0.0)
427 self.assertEqualAndEqualSign(pow_op(-2.0, -INF), 0.0)
428 self.assertEqualAndEqualSign(pow_op(2.0, -INF), 0.0)
429 self.assertEqualAndEqualSign(pow_op(INF, -INF), 0.0)
430
431 # x**INF is 0 for abs(x) < 1
432 self.assertEqualAndEqualSign(pow_op(-0.5, INF), 0.0)
433 self.assertEqualAndEqualSign(pow_op(-0.0, INF), 0.0)
434 self.assertEqualAndEqualSign(pow_op(0.0, INF), 0.0)
435 self.assertEqualAndEqualSign(pow_op(0.5, INF), 0.0)
436
437 # x**INF is INF for abs(x) > 1
438 self.assertEqualAndEqualSign(pow_op(-INF, INF), INF)
439 self.assertEqualAndEqualSign(pow_op(-2.0, INF), INF)
440 self.assertEqualAndEqualSign(pow_op(2.0, INF), INF)
441 self.assertEqualAndEqualSign(pow_op(INF, INF), INF)
442
443 # (-INF)**y is -0.0 for y a negative odd integer
444 self.assertEqualAndEqualSign(pow_op(-INF, -1.0), -0.0)
445
446 # (-INF)**y is 0.0 for y negative but not an odd integer
447 self.assertEqualAndEqualSign(pow_op(-INF, -0.5), 0.0)
448 self.assertEqualAndEqualSign(pow_op(-INF, -2.0), 0.0)
449
450 # (-INF)**y is -INF for y a positive odd integer
451 self.assertEqualAndEqualSign(pow_op(-INF, 1.0), -INF)
452
453 # (-INF)**y is INF for y positive but not an odd integer
454 self.assertEqualAndEqualSign(pow_op(-INF, 0.5), INF)
455 self.assertEqualAndEqualSign(pow_op(-INF, 2.0), INF)
456
457 # INF**y is INF for y positive
458 self.assertEqualAndEqualSign(pow_op(INF, 0.5), INF)
459 self.assertEqualAndEqualSign(pow_op(INF, 1.0), INF)
460 self.assertEqualAndEqualSign(pow_op(INF, 2.0), INF)
461
462 # INF**y is 0.0 for y negative
463 self.assertEqualAndEqualSign(pow_op(INF, -2.0), 0.0)
464 self.assertEqualAndEqualSign(pow_op(INF, -1.0), 0.0)
465 self.assertEqualAndEqualSign(pow_op(INF, -0.5), 0.0)
466
467 # basic checks not covered by the special cases above
468 self.assertEqualAndEqualSign(pow_op(-2.0, -2.0), 0.25)
469 self.assertEqualAndEqualSign(pow_op(-2.0, -1.0), -0.5)
470 self.assertEqualAndEqualSign(pow_op(-2.0, -0.0), 1.0)
471 self.assertEqualAndEqualSign(pow_op(-2.0, 0.0), 1.0)
472 self.assertEqualAndEqualSign(pow_op(-2.0, 1.0), -2.0)
473 self.assertEqualAndEqualSign(pow_op(-2.0, 2.0), 4.0)
474 self.assertEqualAndEqualSign(pow_op(-1.0, -2.0), 1.0)
475 self.assertEqualAndEqualSign(pow_op(-1.0, -1.0), -1.0)
476 self.assertEqualAndEqualSign(pow_op(-1.0, -0.0), 1.0)
477 self.assertEqualAndEqualSign(pow_op(-1.0, 0.0), 1.0)
478 self.assertEqualAndEqualSign(pow_op(-1.0, 1.0), -1.0)
479 self.assertEqualAndEqualSign(pow_op(-1.0, 2.0), 1.0)
480 self.assertEqualAndEqualSign(pow_op(2.0, -2.0), 0.25)
481 self.assertEqualAndEqualSign(pow_op(2.0, -1.0), 0.5)
482 self.assertEqualAndEqualSign(pow_op(2.0, -0.0), 1.0)
483 self.assertEqualAndEqualSign(pow_op(2.0, 0.0), 1.0)
484 self.assertEqualAndEqualSign(pow_op(2.0, 1.0), 2.0)
485 self.assertEqualAndEqualSign(pow_op(2.0, 2.0), 4.0)
486
487 # 1 ** large and -1 ** large; some libms apparently
488 # have problems with these
489 self.assertEqualAndEqualSign(pow_op(1.0, -1e100), 1.0)
490 self.assertEqualAndEqualSign(pow_op(1.0, 1e100), 1.0)
491 self.assertEqualAndEqualSign(pow_op(-1.0, -1e100), 1.0)
492 self.assertEqualAndEqualSign(pow_op(-1.0, 1e100), 1.0)
493
494 # check sign for results that underflow to 0
495 self.assertEqualAndEqualSign(pow_op(-2.0, -2000.0), 0.0)
496 self.assertEqual(type(pow_op(-2.0, -2000.5)), complex)
497 self.assertEqualAndEqualSign(pow_op(-2.0, -2001.0), -0.0)
498 self.assertEqualAndEqualSign(pow_op(2.0, -2000.0), 0.0)
499 self.assertEqualAndEqualSign(pow_op(2.0, -2000.5), 0.0)
500 self.assertEqualAndEqualSign(pow_op(2.0, -2001.0), 0.0)
501 self.assertEqualAndEqualSign(pow_op(-0.5, 2000.0), 0.0)
502 self.assertEqual(type(pow_op(-0.5, 2000.5)), complex)
503 self.assertEqualAndEqualSign(pow_op(-0.5, 2001.0), -0.0)
504 self.assertEqualAndEqualSign(pow_op(0.5, 2000.0), 0.0)
505 self.assertEqualAndEqualSign(pow_op(0.5, 2000.5), 0.0)
506 self.assertEqualAndEqualSign(pow_op(0.5, 2001.0), 0.0)
507
508 # check we don't raise an exception for subnormal results,
509 # and validate signs. Tests currently disabled, since
510 # they fail on systems where a subnormal result from pow
511 # is flushed to zero (e.g. Debian/ia64.)
512 #self.assertTrue(0.0 < pow_op(0.5, 1048) < 1e-315)
513 #self.assertTrue(0.0 < pow_op(-0.5, 1048) < 1e-315)
514 #self.assertTrue(0.0 < pow_op(0.5, 1047) < 1e-315)
515 #self.assertTrue(0.0 > pow_op(-0.5, 1047) > -1e-315)
516 #self.assertTrue(0.0 < pow_op(2.0, -1048) < 1e-315)
517 #self.assertTrue(0.0 < pow_op(-2.0, -1048) < 1e-315)
518 #self.assertTrue(0.0 < pow_op(2.0, -1047) < 1e-315)
519 #self.assertTrue(0.0 > pow_op(-2.0, -1047) > -1e-315)
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000520
521
Benjamin Petersone401c682010-07-02 23:25:44 +0000522@requires_setformat
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000523class FormatFunctionsTestCase(unittest.TestCase):
524
525 def setUp(self):
526 self.save_formats = {'double':float.__getformat__('double'),
527 'float':float.__getformat__('float')}
528
529 def tearDown(self):
530 float.__setformat__('double', self.save_formats['double'])
531 float.__setformat__('float', self.save_formats['float'])
532
533 def test_getformat(self):
Ezio Melottib58e0bd2010-01-23 15:40:09 +0000534 self.assertIn(float.__getformat__('double'),
535 ['unknown', 'IEEE, big-endian', 'IEEE, little-endian'])
536 self.assertIn(float.__getformat__('float'),
537 ['unknown', 'IEEE, big-endian', 'IEEE, little-endian'])
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000538 self.assertRaises(ValueError, float.__getformat__, 'chicken')
539 self.assertRaises(TypeError, float.__getformat__, 1)
540
541 def test_setformat(self):
542 for t in 'double', 'float':
543 float.__setformat__(t, 'unknown')
544 if self.save_formats[t] == 'IEEE, big-endian':
545 self.assertRaises(ValueError, float.__setformat__,
546 t, 'IEEE, little-endian')
547 elif self.save_formats[t] == 'IEEE, little-endian':
548 self.assertRaises(ValueError, float.__setformat__,
549 t, 'IEEE, big-endian')
550 else:
551 self.assertRaises(ValueError, float.__setformat__,
552 t, 'IEEE, big-endian')
553 self.assertRaises(ValueError, float.__setformat__,
554 t, 'IEEE, little-endian')
555 self.assertRaises(ValueError, float.__setformat__,
556 t, 'chicken')
557 self.assertRaises(ValueError, float.__setformat__,
558 'chicken', 'unknown')
559
Guido van Rossum2be161d2007-05-15 20:43:51 +0000560BE_DOUBLE_INF = b'\x7f\xf0\x00\x00\x00\x00\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000561LE_DOUBLE_INF = bytes(reversed(BE_DOUBLE_INF))
Guido van Rossum2be161d2007-05-15 20:43:51 +0000562BE_DOUBLE_NAN = b'\x7f\xf8\x00\x00\x00\x00\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000563LE_DOUBLE_NAN = bytes(reversed(BE_DOUBLE_NAN))
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000564
Guido van Rossum2be161d2007-05-15 20:43:51 +0000565BE_FLOAT_INF = b'\x7f\x80\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000566LE_FLOAT_INF = bytes(reversed(BE_FLOAT_INF))
Guido van Rossum2be161d2007-05-15 20:43:51 +0000567BE_FLOAT_NAN = b'\x7f\xc0\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000568LE_FLOAT_NAN = bytes(reversed(BE_FLOAT_NAN))
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000569
570# on non-IEEE platforms, attempting to unpack a bit pattern
571# representing an infinity or a NaN should raise an exception.
572
Benjamin Petersone401c682010-07-02 23:25:44 +0000573@requires_setformat
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000574class UnknownFormatTestCase(unittest.TestCase):
575 def setUp(self):
576 self.save_formats = {'double':float.__getformat__('double'),
577 'float':float.__getformat__('float')}
578 float.__setformat__('double', 'unknown')
579 float.__setformat__('float', 'unknown')
Tim Peters5d36a552005-06-03 22:40:27 +0000580
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000581 def tearDown(self):
582 float.__setformat__('double', self.save_formats['double'])
583 float.__setformat__('float', self.save_formats['float'])
584
585 def test_double_specials_dont_unpack(self):
586 for fmt, data in [('>d', BE_DOUBLE_INF),
587 ('>d', BE_DOUBLE_NAN),
588 ('<d', LE_DOUBLE_INF),
589 ('<d', LE_DOUBLE_NAN)]:
590 self.assertRaises(ValueError, struct.unpack, fmt, data)
591
592 def test_float_specials_dont_unpack(self):
593 for fmt, data in [('>f', BE_FLOAT_INF),
594 ('>f', BE_FLOAT_NAN),
595 ('<f', LE_FLOAT_INF),
596 ('<f', LE_FLOAT_NAN)]:
597 self.assertRaises(ValueError, struct.unpack, fmt, data)
598
599
600# on an IEEE platform, all we guarantee is that bit patterns
601# representing infinities or NaNs do not raise an exception; all else
602# is accident (today).
Guido van Rossum04110fb2007-08-24 16:32:05 +0000603# let's also try to guarantee that -0.0 and 0.0 don't get confused.
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000604
605class IEEEFormatTestCase(unittest.TestCase):
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000606
Eric Smith3ab08ca2010-12-04 15:17:38 +0000607 @support.requires_IEEE_754
Benjamin Petersone401c682010-07-02 23:25:44 +0000608 def test_double_specials_do_unpack(self):
609 for fmt, data in [('>d', BE_DOUBLE_INF),
610 ('>d', BE_DOUBLE_NAN),
611 ('<d', LE_DOUBLE_INF),
612 ('<d', LE_DOUBLE_NAN)]:
613 struct.unpack(fmt, data)
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000614
Eric Smith3ab08ca2010-12-04 15:17:38 +0000615 @support.requires_IEEE_754
Benjamin Petersone401c682010-07-02 23:25:44 +0000616 def test_float_specials_do_unpack(self):
617 for fmt, data in [('>f', BE_FLOAT_INF),
618 ('>f', BE_FLOAT_NAN),
619 ('<f', LE_FLOAT_INF),
620 ('<f', LE_FLOAT_NAN)]:
621 struct.unpack(fmt, data)
Guido van Rossum04110fb2007-08-24 16:32:05 +0000622
Benjamin Peterson2bb69a52017-09-10 23:50:46 -0700623 @support.requires_IEEE_754
624 def test_serialized_float_rounding(self):
625 from _testcapi import FLT_MAX
626 self.assertEqual(struct.pack("<f", 3.40282356e38), struct.pack("<f", FLT_MAX))
627 self.assertEqual(struct.pack("<f", -3.40282356e38), struct.pack("<f", -FLT_MAX))
628
Eric Smith8c663262007-08-25 02:26:07 +0000629class FormatTestCase(unittest.TestCase):
Benjamin Petersone401c682010-07-02 23:25:44 +0000630
Eric Smith11fe3e02007-08-31 01:33:06 +0000631 def test_format(self):
Eric Smith8c663262007-08-25 02:26:07 +0000632 # these should be rewritten to use both format(x, spec) and
633 # x.__format__(spec)
634
635 self.assertEqual(format(0.0, 'f'), '0.000000')
636
637 # the default is 'g', except for empty format spec
638 self.assertEqual(format(0.0, ''), '0.0')
639 self.assertEqual(format(0.01, ''), '0.01')
640 self.assertEqual(format(0.01, 'g'), '0.01')
641
Eric Smith63376222009-05-05 14:04:18 +0000642 # empty presentation type should format in the same way as str
643 # (issue 5920)
644 x = 100/7.
645 self.assertEqual(format(x, ''), str(x))
646 self.assertEqual(format(x, '-'), str(x))
647 self.assertEqual(format(x, '>'), str(x))
648 self.assertEqual(format(x, '2'), str(x))
Eric Smith8c663262007-08-25 02:26:07 +0000649
650 self.assertEqual(format(1.0, 'f'), '1.000000')
Eric Smith8c663262007-08-25 02:26:07 +0000651
652 self.assertEqual(format(-1.0, 'f'), '-1.000000')
Eric Smith8c663262007-08-25 02:26:07 +0000653
654 self.assertEqual(format( 1.0, ' f'), ' 1.000000')
655 self.assertEqual(format(-1.0, ' f'), '-1.000000')
656 self.assertEqual(format( 1.0, '+f'), '+1.000000')
657 self.assertEqual(format(-1.0, '+f'), '-1.000000')
658
659 # % formatting
660 self.assertEqual(format(-1.0, '%'), '-100.000000%')
661
662 # conversion to string should fail
663 self.assertRaises(ValueError, format, 3.0, "s")
664
Eric Smith7b69c6c2008-01-27 21:07:59 +0000665 # other format specifiers shouldn't work on floats,
666 # in particular int specifiers
667 for format_spec in ([chr(x) for x in range(ord('a'), ord('z')+1)] +
668 [chr(x) for x in range(ord('A'), ord('Z')+1)]):
669 if not format_spec in 'eEfFgGn%':
670 self.assertRaises(ValueError, format, 0.0, format_spec)
671 self.assertRaises(ValueError, format, 1.0, format_spec)
672 self.assertRaises(ValueError, format, -1.0, format_spec)
673 self.assertRaises(ValueError, format, 1e100, format_spec)
674 self.assertRaises(ValueError, format, -1e100, format_spec)
675 self.assertRaises(ValueError, format, 1e-100, format_spec)
676 self.assertRaises(ValueError, format, -1e-100, format_spec)
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000677
Eric Smith741191f2009-05-06 13:08:15 +0000678 # issue 3382
679 self.assertEqual(format(NAN, 'f'), 'nan')
680 self.assertEqual(format(NAN, 'F'), 'NAN')
681 self.assertEqual(format(INF, 'f'), 'inf')
682 self.assertEqual(format(INF, 'F'), 'INF')
683
Eric Smith3ab08ca2010-12-04 15:17:38 +0000684 @support.requires_IEEE_754
Eric Smith0923d1d2009-04-16 20:16:10 +0000685 def test_format_testfile(self):
Brian Curtin076623b2010-10-31 00:03:45 +0000686 with open(format_testfile) as testfile:
687 for line in testfile:
688 if line.startswith('--'):
689 continue
690 line = line.strip()
691 if not line:
692 continue
Eric Smith0923d1d2009-04-16 20:16:10 +0000693
Brian Curtin076623b2010-10-31 00:03:45 +0000694 lhs, rhs = map(str.strip, line.split('->'))
695 fmt, arg = lhs.split()
696 self.assertEqual(fmt % float(arg), rhs)
697 self.assertEqual(fmt % -float(arg), '-' + rhs)
Eric Smith0923d1d2009-04-16 20:16:10 +0000698
Mark Dickinsond3ca5572009-04-29 18:47:07 +0000699 def test_issue5864(self):
Ezio Melottib3aedd42010-11-20 19:04:17 +0000700 self.assertEqual(format(123.456, '.4'), '123.5')
701 self.assertEqual(format(1234.56, '.4'), '1.235e+03')
702 self.assertEqual(format(12345.6, '.4'), '1.235e+04')
Mark Dickinsond3ca5572009-04-29 18:47:07 +0000703
Mark Dickinson7efad9e2009-04-17 20:59:58 +0000704class ReprTestCase(unittest.TestCase):
705 def test_repr(self):
706 floats_file = open(os.path.join(os.path.split(__file__)[0],
707 'floating_points.txt'))
708 for line in floats_file:
709 line = line.strip()
710 if not line or line.startswith('#'):
711 continue
712 v = eval(line)
713 self.assertEqual(v, eval(repr(v)))
714 floats_file.close()
715
Eric Smith0923d1d2009-04-16 20:16:10 +0000716 @unittest.skipUnless(getattr(sys, 'float_repr_style', '') == 'short',
717 "applies only when using short float repr style")
718 def test_short_repr(self):
719 # test short float repr introduced in Python 3.1. One aspect
720 # of this repr is that we get some degree of str -> float ->
721 # str roundtripping. In particular, for any numeric string
722 # containing 15 or fewer significant digits, those exact same
723 # digits (modulo trailing zeros) should appear in the output.
724 # No more repr(0.03) -> "0.029999999999999999"!
725
726 test_strings = [
727 # output always includes *either* a decimal point and at
728 # least one digit after that point, or an exponent.
729 '0.0',
730 '1.0',
731 '0.01',
732 '0.02',
733 '0.03',
734 '0.04',
735 '0.05',
736 '1.23456789',
737 '10.0',
738 '100.0',
739 # values >= 1e16 get an exponent...
740 '1000000000000000.0',
741 '9999999999999990.0',
742 '1e+16',
743 '1e+17',
744 # ... and so do values < 1e-4
745 '0.001',
746 '0.001001',
747 '0.00010000000000001',
748 '0.0001',
749 '9.999999999999e-05',
750 '1e-05',
751 # values designed to provoke failure if the FPU rounding
752 # precision isn't set correctly
753 '8.72293771110361e+25',
754 '7.47005307342313e+26',
755 '2.86438000439698e+28',
756 '8.89142905246179e+28',
757 '3.08578087079232e+35',
758 ]
759
760 for s in test_strings:
761 negs = '-'+s
762 self.assertEqual(s, repr(float(s)))
763 self.assertEqual(negs, repr(float(negs)))
Mark Dickinson388122d2010-08-04 20:56:28 +0000764 # Since Python 3.2, repr and str are identical
765 self.assertEqual(repr(float(s)), str(float(s)))
766 self.assertEqual(repr(float(negs)), str(float(negs)))
Benjamin Petersone401c682010-07-02 23:25:44 +0000767
Eric Smith3ab08ca2010-12-04 15:17:38 +0000768@support.requires_IEEE_754
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000769class RoundTestCase(unittest.TestCase):
Benjamin Petersone401c682010-07-02 23:25:44 +0000770
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000771 def test_inf_nan(self):
772 self.assertRaises(OverflowError, round, INF)
773 self.assertRaises(OverflowError, round, -INF)
774 self.assertRaises(ValueError, round, NAN)
Mark Dickinson4ca33d12009-11-24 10:59:34 +0000775 self.assertRaises(TypeError, round, INF, 0.0)
776 self.assertRaises(TypeError, round, -INF, 1.0)
777 self.assertRaises(TypeError, round, NAN, "ceci n'est pas un integer")
778 self.assertRaises(TypeError, round, -0.0, 1j)
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000779
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000780 def test_large_n(self):
781 for n in [324, 325, 400, 2**31-1, 2**31, 2**32, 2**100]:
782 self.assertEqual(round(123.456, n), 123.456)
783 self.assertEqual(round(-123.456, n), -123.456)
784 self.assertEqual(round(1e300, n), 1e300)
785 self.assertEqual(round(1e-320, n), 1e-320)
786 self.assertEqual(round(1e150, 300), 1e150)
787 self.assertEqual(round(1e300, 307), 1e300)
788 self.assertEqual(round(-3.1415, 308), -3.1415)
789 self.assertEqual(round(1e150, 309), 1e150)
790 self.assertEqual(round(1.4e-315, 315), 1e-315)
791
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000792 def test_small_n(self):
793 for n in [-308, -309, -400, 1-2**31, -2**31, -2**31-1, -2**100]:
794 self.assertEqual(round(123.456, n), 0.0)
795 self.assertEqual(round(-123.456, n), -0.0)
796 self.assertEqual(round(1e300, n), 0.0)
797 self.assertEqual(round(1e-320, n), 0.0)
798
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000799 def test_overflow(self):
800 self.assertRaises(OverflowError, round, 1.6e308, -308)
801 self.assertRaises(OverflowError, round, -1.7e308, -308)
802
803 @unittest.skipUnless(getattr(sys, 'float_repr_style', '') == 'short',
804 "applies only when using short float repr style")
805 def test_previous_round_bugs(self):
806 # particular cases that have occurred in bug reports
807 self.assertEqual(round(562949953421312.5, 1),
808 562949953421312.5)
809 self.assertEqual(round(56294995342131.5, 3),
810 56294995342131.5)
811 # round-half-even
812 self.assertEqual(round(25.0, -1), 20.0)
813 self.assertEqual(round(35.0, -1), 40.0)
814 self.assertEqual(round(45.0, -1), 40.0)
815 self.assertEqual(round(55.0, -1), 60.0)
816 self.assertEqual(round(65.0, -1), 60.0)
817 self.assertEqual(round(75.0, -1), 80.0)
818 self.assertEqual(round(85.0, -1), 80.0)
819 self.assertEqual(round(95.0, -1), 100.0)
820
821 @unittest.skipUnless(getattr(sys, 'float_repr_style', '') == 'short',
822 "applies only when using short float repr style")
823 def test_matches_float_format(self):
824 # round should give the same results as float formatting
825 for i in range(500):
826 x = i/1000.
827 self.assertEqual(float(format(x, '.0f')), round(x, 0))
828 self.assertEqual(float(format(x, '.1f')), round(x, 1))
829 self.assertEqual(float(format(x, '.2f')), round(x, 2))
830 self.assertEqual(float(format(x, '.3f')), round(x, 3))
831
832 for i in range(5, 5000, 10):
833 x = i/1000.
834 self.assertEqual(float(format(x, '.0f')), round(x, 0))
835 self.assertEqual(float(format(x, '.1f')), round(x, 1))
836 self.assertEqual(float(format(x, '.2f')), round(x, 2))
837 self.assertEqual(float(format(x, '.3f')), round(x, 3))
838
839 for i in range(500):
840 x = random.random()
841 self.assertEqual(float(format(x, '.0f')), round(x, 0))
842 self.assertEqual(float(format(x, '.1f')), round(x, 1))
843 self.assertEqual(float(format(x, '.2f')), round(x, 2))
844 self.assertEqual(float(format(x, '.3f')), round(x, 3))
845
Eric Smith8a10ecc2009-12-02 17:58:24 +0000846 def test_format_specials(self):
847 # Test formatting of nans and infs.
848
849 def test(fmt, value, expected):
850 # Test with both % and format().
851 self.assertEqual(fmt % value, expected, fmt)
Eric Smith984bb582010-11-25 16:08:06 +0000852 fmt = fmt[1:] # strip off the %
853 self.assertEqual(format(value, fmt), expected, fmt)
Eric Smith8a10ecc2009-12-02 17:58:24 +0000854
855 for fmt in ['%e', '%f', '%g', '%.0e', '%.6f', '%.20g',
856 '%#e', '%#f', '%#g', '%#.20e', '%#.15f', '%#.3g']:
857 pfmt = '%+' + fmt[1:]
858 sfmt = '% ' + fmt[1:]
859 test(fmt, INF, 'inf')
860 test(fmt, -INF, '-inf')
861 test(fmt, NAN, 'nan')
862 test(fmt, -NAN, 'nan')
863 # When asking for a sign, it's always provided. nans are
864 # always positive.
865 test(pfmt, INF, '+inf')
866 test(pfmt, -INF, '-inf')
867 test(pfmt, NAN, '+nan')
868 test(pfmt, -NAN, '+nan')
869 # When using ' ' for a sign code, only infs can be negative.
870 # Others have a space.
871 test(sfmt, INF, ' inf')
872 test(sfmt, -INF, '-inf')
873 test(sfmt, NAN, ' nan')
874 test(sfmt, -NAN, ' nan')
875
Steve Dowercb39d1f2015-04-15 16:10:59 -0400876 def test_None_ndigits(self):
877 for x in round(1.23), round(1.23, None), round(1.23, ndigits=None):
878 self.assertEqual(x, 1)
879 self.assertIsInstance(x, int)
880 for x in round(1.78), round(1.78, None), round(1.78, ndigits=None):
881 self.assertEqual(x, 2)
882 self.assertIsInstance(x, int)
883
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000884
Christian Heimes99170a52007-12-19 02:07:34 +0000885# Beginning with Python 2.6 float has cross platform compatible
Georg Brandl2ee470f2008-07-16 12:55:28 +0000886# ways to create and represent inf and nan
Christian Heimes99170a52007-12-19 02:07:34 +0000887class InfNanTest(unittest.TestCase):
888 def test_inf_from_str(self):
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000889 self.assertTrue(isinf(float("inf")))
890 self.assertTrue(isinf(float("+inf")))
891 self.assertTrue(isinf(float("-inf")))
892 self.assertTrue(isinf(float("infinity")))
893 self.assertTrue(isinf(float("+infinity")))
894 self.assertTrue(isinf(float("-infinity")))
Christian Heimes99170a52007-12-19 02:07:34 +0000895
896 self.assertEqual(repr(float("inf")), "inf")
897 self.assertEqual(repr(float("+inf")), "inf")
898 self.assertEqual(repr(float("-inf")), "-inf")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000899 self.assertEqual(repr(float("infinity")), "inf")
900 self.assertEqual(repr(float("+infinity")), "inf")
901 self.assertEqual(repr(float("-infinity")), "-inf")
Christian Heimes99170a52007-12-19 02:07:34 +0000902
903 self.assertEqual(repr(float("INF")), "inf")
904 self.assertEqual(repr(float("+Inf")), "inf")
905 self.assertEqual(repr(float("-iNF")), "-inf")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000906 self.assertEqual(repr(float("Infinity")), "inf")
907 self.assertEqual(repr(float("+iNfInItY")), "inf")
908 self.assertEqual(repr(float("-INFINITY")), "-inf")
Christian Heimes99170a52007-12-19 02:07:34 +0000909
910 self.assertEqual(str(float("inf")), "inf")
911 self.assertEqual(str(float("+inf")), "inf")
912 self.assertEqual(str(float("-inf")), "-inf")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000913 self.assertEqual(str(float("infinity")), "inf")
914 self.assertEqual(str(float("+infinity")), "inf")
915 self.assertEqual(str(float("-infinity")), "-inf")
Christian Heimes99170a52007-12-19 02:07:34 +0000916
917 self.assertRaises(ValueError, float, "info")
918 self.assertRaises(ValueError, float, "+info")
919 self.assertRaises(ValueError, float, "-info")
920 self.assertRaises(ValueError, float, "in")
921 self.assertRaises(ValueError, float, "+in")
922 self.assertRaises(ValueError, float, "-in")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000923 self.assertRaises(ValueError, float, "infinit")
924 self.assertRaises(ValueError, float, "+Infin")
925 self.assertRaises(ValueError, float, "-INFI")
926 self.assertRaises(ValueError, float, "infinitys")
Christian Heimes99170a52007-12-19 02:07:34 +0000927
Mark Dickinsonbd16edd2009-05-20 22:05:25 +0000928 self.assertRaises(ValueError, float, "++Inf")
929 self.assertRaises(ValueError, float, "-+inf")
930 self.assertRaises(ValueError, float, "+-infinity")
931 self.assertRaises(ValueError, float, "--Infinity")
932
Christian Heimes99170a52007-12-19 02:07:34 +0000933 def test_inf_as_str(self):
934 self.assertEqual(repr(1e300 * 1e300), "inf")
935 self.assertEqual(repr(-1e300 * 1e300), "-inf")
936
937 self.assertEqual(str(1e300 * 1e300), "inf")
938 self.assertEqual(str(-1e300 * 1e300), "-inf")
939
940 def test_nan_from_str(self):
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000941 self.assertTrue(isnan(float("nan")))
942 self.assertTrue(isnan(float("+nan")))
943 self.assertTrue(isnan(float("-nan")))
Christian Heimes99170a52007-12-19 02:07:34 +0000944
945 self.assertEqual(repr(float("nan")), "nan")
946 self.assertEqual(repr(float("+nan")), "nan")
947 self.assertEqual(repr(float("-nan")), "nan")
948
949 self.assertEqual(repr(float("NAN")), "nan")
950 self.assertEqual(repr(float("+NAn")), "nan")
951 self.assertEqual(repr(float("-NaN")), "nan")
952
953 self.assertEqual(str(float("nan")), "nan")
954 self.assertEqual(str(float("+nan")), "nan")
955 self.assertEqual(str(float("-nan")), "nan")
956
957 self.assertRaises(ValueError, float, "nana")
958 self.assertRaises(ValueError, float, "+nana")
959 self.assertRaises(ValueError, float, "-nana")
960 self.assertRaises(ValueError, float, "na")
961 self.assertRaises(ValueError, float, "+na")
962 self.assertRaises(ValueError, float, "-na")
963
Mark Dickinsonbd16edd2009-05-20 22:05:25 +0000964 self.assertRaises(ValueError, float, "++nan")
965 self.assertRaises(ValueError, float, "-+NAN")
966 self.assertRaises(ValueError, float, "+-NaN")
967 self.assertRaises(ValueError, float, "--nAn")
968
Christian Heimes99170a52007-12-19 02:07:34 +0000969 def test_nan_as_str(self):
970 self.assertEqual(repr(1e300 * 1e300 * 0), "nan")
971 self.assertEqual(repr(-1e300 * 1e300 * 0), "nan")
972
973 self.assertEqual(str(1e300 * 1e300 * 0), "nan")
974 self.assertEqual(str(-1e300 * 1e300 * 0), "nan")
Christian Heimes827b35c2007-12-10 22:19:17 +0000975
Mark Dickinsone383e822012-04-29 15:31:56 +0100976 def test_inf_signs(self):
977 self.assertEqual(copysign(1.0, float('inf')), 1.0)
978 self.assertEqual(copysign(1.0, float('-inf')), -1.0)
979
980 @unittest.skipUnless(getattr(sys, 'float_repr_style', '') == 'short',
981 "applies only when using short float repr style")
982 def test_nan_signs(self):
983 # When using the dtoa.c code, the sign of float('nan') should
984 # be predictable.
985 self.assertEqual(copysign(1.0, float('nan')), 1.0)
986 self.assertEqual(copysign(1.0, float('-nan')), -1.0)
987
988
Mark Dickinson65fe25e2008-07-16 11:30:51 +0000989fromHex = float.fromhex
990toHex = float.hex
991class HexFloatTestCase(unittest.TestCase):
992 MAX = fromHex('0x.fffffffffffff8p+1024') # max normal
993 MIN = fromHex('0x1p-1022') # min normal
994 TINY = fromHex('0x0.0000000000001p-1022') # min subnormal
995 EPS = fromHex('0x0.0000000000001p0') # diff between 1.0 and next float up
996
997 def identical(self, x, y):
998 # check that floats x and y are identical, or that both
999 # are NaNs
1000 if isnan(x) or isnan(y):
1001 if isnan(x) == isnan(y):
1002 return
1003 elif x == y and (x != 0.0 or copysign(1.0, x) == copysign(1.0, y)):
1004 return
1005 self.fail('%r not identical to %r' % (x, y))
1006
1007 def test_ends(self):
Mark Dickinson38bbc482008-07-16 11:32:23 +00001008 self.identical(self.MIN, ldexp(1.0, -1022))
1009 self.identical(self.TINY, ldexp(1.0, -1074))
1010 self.identical(self.EPS, ldexp(1.0, -52))
1011 self.identical(self.MAX, 2.*(ldexp(1.0, 1023) - ldexp(1.0, 970)))
Mark Dickinson65fe25e2008-07-16 11:30:51 +00001012
1013 def test_invalid_inputs(self):
1014 invalid_inputs = [
1015 'infi', # misspelt infinities and nans
1016 '-Infinit',
1017 '++inf',
1018 '-+Inf',
1019 '--nan',
1020 '+-NaN',
1021 'snan',
1022 'NaNs',
1023 'nna',
Mark Dickinsond1ec8b22009-05-11 15:45:15 +00001024 'an',
1025 'nf',
1026 'nfinity',
1027 'inity',
1028 'iinity',
Mark Dickinson65fe25e2008-07-16 11:30:51 +00001029 '0xnan',
1030 '',
1031 ' ',
1032 'x1.0p0',
1033 '0xX1.0p0',
1034 '+ 0x1.0p0', # internal whitespace
1035 '- 0x1.0p0',
1036 '0 x1.0p0',
1037 '0x 1.0p0',
1038 '0x1 2.0p0',
1039 '+0x1 .0p0',
1040 '0x1. 0p0',
1041 '-0x1.0 1p0',
1042 '-0x1.0 p0',
1043 '+0x1.0p +0',
1044 '0x1.0p -0',
1045 '0x1.0p 0',
1046 '+0x1.0p+ 0',
1047 '-0x1.0p- 0',
1048 '++0x1.0p-0', # double signs
1049 '--0x1.0p0',
1050 '+-0x1.0p+0',
1051 '-+0x1.0p0',
1052 '0x1.0p++0',
1053 '+0x1.0p+-0',
1054 '-0x1.0p-+0',
1055 '0x1.0p--0',
1056 '0x1.0.p0',
1057 '0x.p0', # no hex digits before or after point
1058 '0x1,p0', # wrong decimal point character
1059 '0x1pa',
1060 '0x1p\uff10', # fullwidth Unicode digits
1061 '\uff10x1p0',
1062 '0x\uff11p0',
1063 '0x1.\uff10p0',
1064 '0x1p0 \n 0x2p0',
1065 '0x1p0\0 0x1p0', # embedded null byte is not end of string
1066 ]
1067 for x in invalid_inputs:
Mark Dickinson589b7952008-08-21 20:05:56 +00001068 try:
1069 result = fromHex(x)
1070 except ValueError:
1071 pass
1072 else:
1073 self.fail('Expected float.fromhex(%r) to raise ValueError; '
1074 'got %r instead' % (x, result))
Mark Dickinson65fe25e2008-07-16 11:30:51 +00001075
1076
Mark Dickinsond1ec8b22009-05-11 15:45:15 +00001077 def test_whitespace(self):
1078 value_pairs = [
1079 ('inf', INF),
1080 ('-Infinity', -INF),
1081 ('nan', NAN),
1082 ('1.0', 1.0),
1083 ('-0x.2', -0.125),
1084 ('-0.0', -0.0)
1085 ]
1086 whitespace = [
1087 '',
1088 ' ',
1089 '\t',
1090 '\n',
1091 '\n \t',
1092 '\f',
1093 '\v',
1094 '\r'
1095 ]
1096 for inp, expected in value_pairs:
1097 for lead in whitespace:
1098 for trail in whitespace:
1099 got = fromHex(lead + inp + trail)
1100 self.identical(got, expected)
1101
1102
Mark Dickinson65fe25e2008-07-16 11:30:51 +00001103 def test_from_hex(self):
1104 MIN = self.MIN;
1105 MAX = self.MAX;
1106 TINY = self.TINY;
1107 EPS = self.EPS;
1108
1109 # two spellings of infinity, with optional signs; case-insensitive
1110 self.identical(fromHex('inf'), INF)
1111 self.identical(fromHex('+Inf'), INF)
1112 self.identical(fromHex('-INF'), -INF)
1113 self.identical(fromHex('iNf'), INF)
1114 self.identical(fromHex('Infinity'), INF)
1115 self.identical(fromHex('+INFINITY'), INF)
1116 self.identical(fromHex('-infinity'), -INF)
1117 self.identical(fromHex('-iNFiNitY'), -INF)
1118
1119 # nans with optional sign; case insensitive
1120 self.identical(fromHex('nan'), NAN)
1121 self.identical(fromHex('+NaN'), NAN)
1122 self.identical(fromHex('-NaN'), NAN)
1123 self.identical(fromHex('-nAN'), NAN)
1124
1125 # variations in input format
1126 self.identical(fromHex('1'), 1.0)
1127 self.identical(fromHex('+1'), 1.0)
1128 self.identical(fromHex('1.'), 1.0)
1129 self.identical(fromHex('1.0'), 1.0)
1130 self.identical(fromHex('1.0p0'), 1.0)
1131 self.identical(fromHex('01'), 1.0)
1132 self.identical(fromHex('01.'), 1.0)
1133 self.identical(fromHex('0x1'), 1.0)
1134 self.identical(fromHex('0x1.'), 1.0)
1135 self.identical(fromHex('0x1.0'), 1.0)
1136 self.identical(fromHex('+0x1.0'), 1.0)
1137 self.identical(fromHex('0x1p0'), 1.0)
1138 self.identical(fromHex('0X1p0'), 1.0)
1139 self.identical(fromHex('0X1P0'), 1.0)
1140 self.identical(fromHex('0x1P0'), 1.0)
1141 self.identical(fromHex('0x1.p0'), 1.0)
1142 self.identical(fromHex('0x1.0p0'), 1.0)
1143 self.identical(fromHex('0x.1p4'), 1.0)
1144 self.identical(fromHex('0x.1p04'), 1.0)
1145 self.identical(fromHex('0x.1p004'), 1.0)
1146 self.identical(fromHex('0x1p+0'), 1.0)
1147 self.identical(fromHex('0x1P-0'), 1.0)
1148 self.identical(fromHex('+0x1p0'), 1.0)
1149 self.identical(fromHex('0x01p0'), 1.0)
1150 self.identical(fromHex('0x1p00'), 1.0)
1151 self.identical(fromHex(' 0x1p0 '), 1.0)
1152 self.identical(fromHex('\n 0x1p0'), 1.0)
1153 self.identical(fromHex('0x1p0 \t'), 1.0)
1154 self.identical(fromHex('0xap0'), 10.0)
1155 self.identical(fromHex('0xAp0'), 10.0)
1156 self.identical(fromHex('0xaP0'), 10.0)
1157 self.identical(fromHex('0xAP0'), 10.0)
1158 self.identical(fromHex('0xbep0'), 190.0)
1159 self.identical(fromHex('0xBep0'), 190.0)
1160 self.identical(fromHex('0xbEp0'), 190.0)
1161 self.identical(fromHex('0XBE0P-4'), 190.0)
1162 self.identical(fromHex('0xBEp0'), 190.0)
1163 self.identical(fromHex('0xB.Ep4'), 190.0)
1164 self.identical(fromHex('0x.BEp8'), 190.0)
1165 self.identical(fromHex('0x.0BEp12'), 190.0)
1166
1167 # moving the point around
1168 pi = fromHex('0x1.921fb54442d18p1')
1169 self.identical(fromHex('0x.006487ed5110b46p11'), pi)
1170 self.identical(fromHex('0x.00c90fdaa22168cp10'), pi)
1171 self.identical(fromHex('0x.01921fb54442d18p9'), pi)
1172 self.identical(fromHex('0x.03243f6a8885a3p8'), pi)
1173 self.identical(fromHex('0x.06487ed5110b46p7'), pi)
1174 self.identical(fromHex('0x.0c90fdaa22168cp6'), pi)
1175 self.identical(fromHex('0x.1921fb54442d18p5'), pi)
1176 self.identical(fromHex('0x.3243f6a8885a3p4'), pi)
1177 self.identical(fromHex('0x.6487ed5110b46p3'), pi)
1178 self.identical(fromHex('0x.c90fdaa22168cp2'), pi)
1179 self.identical(fromHex('0x1.921fb54442d18p1'), pi)
1180 self.identical(fromHex('0x3.243f6a8885a3p0'), pi)
1181 self.identical(fromHex('0x6.487ed5110b46p-1'), pi)
1182 self.identical(fromHex('0xc.90fdaa22168cp-2'), pi)
1183 self.identical(fromHex('0x19.21fb54442d18p-3'), pi)
1184 self.identical(fromHex('0x32.43f6a8885a3p-4'), pi)
1185 self.identical(fromHex('0x64.87ed5110b46p-5'), pi)
1186 self.identical(fromHex('0xc9.0fdaa22168cp-6'), pi)
1187 self.identical(fromHex('0x192.1fb54442d18p-7'), pi)
1188 self.identical(fromHex('0x324.3f6a8885a3p-8'), pi)
1189 self.identical(fromHex('0x648.7ed5110b46p-9'), pi)
1190 self.identical(fromHex('0xc90.fdaa22168cp-10'), pi)
1191 self.identical(fromHex('0x1921.fb54442d18p-11'), pi)
1192 # ...
1193 self.identical(fromHex('0x1921fb54442d1.8p-47'), pi)
1194 self.identical(fromHex('0x3243f6a8885a3p-48'), pi)
1195 self.identical(fromHex('0x6487ed5110b46p-49'), pi)
1196 self.identical(fromHex('0xc90fdaa22168cp-50'), pi)
1197 self.identical(fromHex('0x1921fb54442d18p-51'), pi)
1198 self.identical(fromHex('0x3243f6a8885a30p-52'), pi)
1199 self.identical(fromHex('0x6487ed5110b460p-53'), pi)
1200 self.identical(fromHex('0xc90fdaa22168c0p-54'), pi)
1201 self.identical(fromHex('0x1921fb54442d180p-55'), pi)
1202
1203
1204 # results that should overflow...
1205 self.assertRaises(OverflowError, fromHex, '-0x1p1024')
1206 self.assertRaises(OverflowError, fromHex, '0x1p+1025')
1207 self.assertRaises(OverflowError, fromHex, '+0X1p1030')
1208 self.assertRaises(OverflowError, fromHex, '-0x1p+1100')
1209 self.assertRaises(OverflowError, fromHex, '0X1p123456789123456789')
1210 self.assertRaises(OverflowError, fromHex, '+0X.8p+1025')
1211 self.assertRaises(OverflowError, fromHex, '+0x0.8p1025')
1212 self.assertRaises(OverflowError, fromHex, '-0x0.4p1026')
1213 self.assertRaises(OverflowError, fromHex, '0X2p+1023')
1214 self.assertRaises(OverflowError, fromHex, '0x2.p1023')
1215 self.assertRaises(OverflowError, fromHex, '-0x2.0p+1023')
1216 self.assertRaises(OverflowError, fromHex, '+0X4p+1022')
1217 self.assertRaises(OverflowError, fromHex, '0x1.ffffffffffffffp+1023')
1218 self.assertRaises(OverflowError, fromHex, '-0X1.fffffffffffff9p1023')
1219 self.assertRaises(OverflowError, fromHex, '0X1.fffffffffffff8p1023')
1220 self.assertRaises(OverflowError, fromHex, '+0x3.fffffffffffffp1022')
1221 self.assertRaises(OverflowError, fromHex, '0x3fffffffffffffp+970')
1222 self.assertRaises(OverflowError, fromHex, '0x10000000000000000p960')
1223 self.assertRaises(OverflowError, fromHex, '-0Xffffffffffffffffp960')
1224
1225 # ...and those that round to +-max float
1226 self.identical(fromHex('+0x1.fffffffffffffp+1023'), MAX)
1227 self.identical(fromHex('-0X1.fffffffffffff7p1023'), -MAX)
1228 self.identical(fromHex('0X1.fffffffffffff7fffffffffffffp1023'), MAX)
1229
1230 # zeros
1231 self.identical(fromHex('0x0p0'), 0.0)
1232 self.identical(fromHex('0x0p1000'), 0.0)
1233 self.identical(fromHex('-0x0p1023'), -0.0)
1234 self.identical(fromHex('0X0p1024'), 0.0)
1235 self.identical(fromHex('-0x0p1025'), -0.0)
1236 self.identical(fromHex('0X0p2000'), 0.0)
1237 self.identical(fromHex('0x0p123456789123456789'), 0.0)
1238 self.identical(fromHex('-0X0p-0'), -0.0)
1239 self.identical(fromHex('-0X0p-1000'), -0.0)
1240 self.identical(fromHex('0x0p-1023'), 0.0)
1241 self.identical(fromHex('-0X0p-1024'), -0.0)
1242 self.identical(fromHex('-0x0p-1025'), -0.0)
1243 self.identical(fromHex('-0x0p-1072'), -0.0)
1244 self.identical(fromHex('0X0p-1073'), 0.0)
1245 self.identical(fromHex('-0x0p-1074'), -0.0)
1246 self.identical(fromHex('0x0p-1075'), 0.0)
1247 self.identical(fromHex('0X0p-1076'), 0.0)
1248 self.identical(fromHex('-0X0p-2000'), -0.0)
1249 self.identical(fromHex('-0x0p-123456789123456789'), -0.0)
1250
1251 # values that should underflow to 0
1252 self.identical(fromHex('0X1p-1075'), 0.0)
1253 self.identical(fromHex('-0X1p-1075'), -0.0)
1254 self.identical(fromHex('-0x1p-123456789123456789'), -0.0)
1255 self.identical(fromHex('0x1.00000000000000001p-1075'), TINY)
1256 self.identical(fromHex('-0x1.1p-1075'), -TINY)
1257 self.identical(fromHex('0x1.fffffffffffffffffp-1075'), TINY)
1258
1259 # check round-half-even is working correctly near 0 ...
1260 self.identical(fromHex('0x1p-1076'), 0.0)
1261 self.identical(fromHex('0X2p-1076'), 0.0)
1262 self.identical(fromHex('0X3p-1076'), TINY)
1263 self.identical(fromHex('0x4p-1076'), TINY)
1264 self.identical(fromHex('0X5p-1076'), TINY)
1265 self.identical(fromHex('0X6p-1076'), 2*TINY)
1266 self.identical(fromHex('0x7p-1076'), 2*TINY)
1267 self.identical(fromHex('0X8p-1076'), 2*TINY)
1268 self.identical(fromHex('0X9p-1076'), 2*TINY)
1269 self.identical(fromHex('0xap-1076'), 2*TINY)
1270 self.identical(fromHex('0Xbp-1076'), 3*TINY)
1271 self.identical(fromHex('0xcp-1076'), 3*TINY)
1272 self.identical(fromHex('0Xdp-1076'), 3*TINY)
1273 self.identical(fromHex('0Xep-1076'), 4*TINY)
1274 self.identical(fromHex('0xfp-1076'), 4*TINY)
1275 self.identical(fromHex('0x10p-1076'), 4*TINY)
1276 self.identical(fromHex('-0x1p-1076'), -0.0)
1277 self.identical(fromHex('-0X2p-1076'), -0.0)
1278 self.identical(fromHex('-0x3p-1076'), -TINY)
1279 self.identical(fromHex('-0X4p-1076'), -TINY)
1280 self.identical(fromHex('-0x5p-1076'), -TINY)
1281 self.identical(fromHex('-0x6p-1076'), -2*TINY)
1282 self.identical(fromHex('-0X7p-1076'), -2*TINY)
1283 self.identical(fromHex('-0X8p-1076'), -2*TINY)
1284 self.identical(fromHex('-0X9p-1076'), -2*TINY)
1285 self.identical(fromHex('-0Xap-1076'), -2*TINY)
1286 self.identical(fromHex('-0xbp-1076'), -3*TINY)
1287 self.identical(fromHex('-0xcp-1076'), -3*TINY)
1288 self.identical(fromHex('-0Xdp-1076'), -3*TINY)
1289 self.identical(fromHex('-0xep-1076'), -4*TINY)
1290 self.identical(fromHex('-0Xfp-1076'), -4*TINY)
1291 self.identical(fromHex('-0X10p-1076'), -4*TINY)
1292
1293 # ... and near MIN ...
1294 self.identical(fromHex('0x0.ffffffffffffd6p-1022'), MIN-3*TINY)
1295 self.identical(fromHex('0x0.ffffffffffffd8p-1022'), MIN-2*TINY)
1296 self.identical(fromHex('0x0.ffffffffffffdap-1022'), MIN-2*TINY)
1297 self.identical(fromHex('0x0.ffffffffffffdcp-1022'), MIN-2*TINY)
1298 self.identical(fromHex('0x0.ffffffffffffdep-1022'), MIN-2*TINY)
1299 self.identical(fromHex('0x0.ffffffffffffe0p-1022'), MIN-2*TINY)
1300 self.identical(fromHex('0x0.ffffffffffffe2p-1022'), MIN-2*TINY)
1301 self.identical(fromHex('0x0.ffffffffffffe4p-1022'), MIN-2*TINY)
1302 self.identical(fromHex('0x0.ffffffffffffe6p-1022'), MIN-2*TINY)
1303 self.identical(fromHex('0x0.ffffffffffffe8p-1022'), MIN-2*TINY)
1304 self.identical(fromHex('0x0.ffffffffffffeap-1022'), MIN-TINY)
1305 self.identical(fromHex('0x0.ffffffffffffecp-1022'), MIN-TINY)
1306 self.identical(fromHex('0x0.ffffffffffffeep-1022'), MIN-TINY)
1307 self.identical(fromHex('0x0.fffffffffffff0p-1022'), MIN-TINY)
1308 self.identical(fromHex('0x0.fffffffffffff2p-1022'), MIN-TINY)
1309 self.identical(fromHex('0x0.fffffffffffff4p-1022'), MIN-TINY)
1310 self.identical(fromHex('0x0.fffffffffffff6p-1022'), MIN-TINY)
1311 self.identical(fromHex('0x0.fffffffffffff8p-1022'), MIN)
1312 self.identical(fromHex('0x0.fffffffffffffap-1022'), MIN)
1313 self.identical(fromHex('0x0.fffffffffffffcp-1022'), MIN)
1314 self.identical(fromHex('0x0.fffffffffffffep-1022'), MIN)
1315 self.identical(fromHex('0x1.00000000000000p-1022'), MIN)
1316 self.identical(fromHex('0x1.00000000000002p-1022'), MIN)
1317 self.identical(fromHex('0x1.00000000000004p-1022'), MIN)
1318 self.identical(fromHex('0x1.00000000000006p-1022'), MIN)
1319 self.identical(fromHex('0x1.00000000000008p-1022'), MIN)
1320 self.identical(fromHex('0x1.0000000000000ap-1022'), MIN+TINY)
1321 self.identical(fromHex('0x1.0000000000000cp-1022'), MIN+TINY)
1322 self.identical(fromHex('0x1.0000000000000ep-1022'), MIN+TINY)
1323 self.identical(fromHex('0x1.00000000000010p-1022'), MIN+TINY)
1324 self.identical(fromHex('0x1.00000000000012p-1022'), MIN+TINY)
1325 self.identical(fromHex('0x1.00000000000014p-1022'), MIN+TINY)
1326 self.identical(fromHex('0x1.00000000000016p-1022'), MIN+TINY)
1327 self.identical(fromHex('0x1.00000000000018p-1022'), MIN+2*TINY)
1328
1329 # ... and near 1.0.
1330 self.identical(fromHex('0x0.fffffffffffff0p0'), 1.0-EPS)
1331 self.identical(fromHex('0x0.fffffffffffff1p0'), 1.0-EPS)
1332 self.identical(fromHex('0X0.fffffffffffff2p0'), 1.0-EPS)
1333 self.identical(fromHex('0x0.fffffffffffff3p0'), 1.0-EPS)
1334 self.identical(fromHex('0X0.fffffffffffff4p0'), 1.0-EPS)
1335 self.identical(fromHex('0X0.fffffffffffff5p0'), 1.0-EPS/2)
1336 self.identical(fromHex('0X0.fffffffffffff6p0'), 1.0-EPS/2)
1337 self.identical(fromHex('0x0.fffffffffffff7p0'), 1.0-EPS/2)
1338 self.identical(fromHex('0x0.fffffffffffff8p0'), 1.0-EPS/2)
1339 self.identical(fromHex('0X0.fffffffffffff9p0'), 1.0-EPS/2)
1340 self.identical(fromHex('0X0.fffffffffffffap0'), 1.0-EPS/2)
1341 self.identical(fromHex('0x0.fffffffffffffbp0'), 1.0-EPS/2)
1342 self.identical(fromHex('0X0.fffffffffffffcp0'), 1.0)
1343 self.identical(fromHex('0x0.fffffffffffffdp0'), 1.0)
1344 self.identical(fromHex('0X0.fffffffffffffep0'), 1.0)
1345 self.identical(fromHex('0x0.ffffffffffffffp0'), 1.0)
1346 self.identical(fromHex('0X1.00000000000000p0'), 1.0)
1347 self.identical(fromHex('0X1.00000000000001p0'), 1.0)
1348 self.identical(fromHex('0x1.00000000000002p0'), 1.0)
1349 self.identical(fromHex('0X1.00000000000003p0'), 1.0)
1350 self.identical(fromHex('0x1.00000000000004p0'), 1.0)
1351 self.identical(fromHex('0X1.00000000000005p0'), 1.0)
1352 self.identical(fromHex('0X1.00000000000006p0'), 1.0)
1353 self.identical(fromHex('0X1.00000000000007p0'), 1.0)
1354 self.identical(fromHex('0x1.00000000000007ffffffffffffffffffffp0'),
1355 1.0)
1356 self.identical(fromHex('0x1.00000000000008p0'), 1.0)
1357 self.identical(fromHex('0x1.00000000000008000000000000000001p0'),
1358 1+EPS)
1359 self.identical(fromHex('0X1.00000000000009p0'), 1.0+EPS)
1360 self.identical(fromHex('0x1.0000000000000ap0'), 1.0+EPS)
1361 self.identical(fromHex('0x1.0000000000000bp0'), 1.0+EPS)
1362 self.identical(fromHex('0X1.0000000000000cp0'), 1.0+EPS)
1363 self.identical(fromHex('0x1.0000000000000dp0'), 1.0+EPS)
1364 self.identical(fromHex('0x1.0000000000000ep0'), 1.0+EPS)
1365 self.identical(fromHex('0X1.0000000000000fp0'), 1.0+EPS)
1366 self.identical(fromHex('0x1.00000000000010p0'), 1.0+EPS)
1367 self.identical(fromHex('0X1.00000000000011p0'), 1.0+EPS)
1368 self.identical(fromHex('0x1.00000000000012p0'), 1.0+EPS)
1369 self.identical(fromHex('0X1.00000000000013p0'), 1.0+EPS)
1370 self.identical(fromHex('0X1.00000000000014p0'), 1.0+EPS)
1371 self.identical(fromHex('0x1.00000000000015p0'), 1.0+EPS)
1372 self.identical(fromHex('0x1.00000000000016p0'), 1.0+EPS)
1373 self.identical(fromHex('0X1.00000000000017p0'), 1.0+EPS)
1374 self.identical(fromHex('0x1.00000000000017ffffffffffffffffffffp0'),
1375 1.0+EPS)
1376 self.identical(fromHex('0x1.00000000000018p0'), 1.0+2*EPS)
1377 self.identical(fromHex('0X1.00000000000018000000000000000001p0'),
1378 1.0+2*EPS)
1379 self.identical(fromHex('0x1.00000000000019p0'), 1.0+2*EPS)
1380 self.identical(fromHex('0X1.0000000000001ap0'), 1.0+2*EPS)
1381 self.identical(fromHex('0X1.0000000000001bp0'), 1.0+2*EPS)
1382 self.identical(fromHex('0x1.0000000000001cp0'), 1.0+2*EPS)
1383 self.identical(fromHex('0x1.0000000000001dp0'), 1.0+2*EPS)
1384 self.identical(fromHex('0x1.0000000000001ep0'), 1.0+2*EPS)
1385 self.identical(fromHex('0X1.0000000000001fp0'), 1.0+2*EPS)
1386 self.identical(fromHex('0x1.00000000000020p0'), 1.0+2*EPS)
1387
1388 def test_roundtrip(self):
1389 def roundtrip(x):
1390 return fromHex(toHex(x))
1391
1392 for x in [NAN, INF, self.MAX, self.MIN, self.MIN-self.TINY, self.TINY, 0.0]:
1393 self.identical(x, roundtrip(x))
1394 self.identical(-x, roundtrip(-x))
1395
1396 # fromHex(toHex(x)) should exactly recover x, for any non-NaN float x.
1397 import random
1398 for i in range(10000):
1399 e = random.randrange(-1200, 1200)
1400 m = random.random()
1401 s = random.choice([1.0, -1.0])
1402 try:
1403 x = s*ldexp(m, e)
1404 except OverflowError:
1405 pass
1406 else:
1407 self.identical(x, fromHex(toHex(x)))
1408
Serhiy Storchakaea36c942016-05-12 10:37:58 +03001409 def test_subclass(self):
1410 class F(float):
1411 def __new__(cls, value):
1412 return float.__new__(cls, value + 1)
1413
1414 f = F.fromhex((1.5).hex())
1415 self.assertIs(type(f), F)
1416 self.assertEqual(f, 2.5)
1417
1418 class F2(float):
1419 def __init__(self, value):
1420 self.foo = 'bar'
1421
1422 f = F2.fromhex((1.5).hex())
1423 self.assertIs(type(f), F2)
1424 self.assertEqual(f, 1.5)
1425 self.assertEqual(getattr(f, 'foo', 'none'), 'bar')
1426
Christian Heimes53876d92008-04-19 00:31:39 +00001427
Michael W. Hudsonba283e22005-05-27 15:23:20 +00001428if __name__ == '__main__':
Zachary Ware38c707e2015-04-13 15:00:43 -05001429 unittest.main()