blob: 68b212e1959e8b21d11ba62303b255d2d2b4b09a [file] [log] [blame]
Michael W. Hudsonba283e22005-05-27 15:23:20 +00001
Benjamin Peterson101447b2015-03-09 10:37:50 -04002import fractions
Christian Heimes53876d92008-04-19 00:31:39 +00003import operator
Benjamin Peterson101447b2015-03-09 10:37:50 -04004import os
5import random
6import sys
7import struct
Benjamin Petersona9157232015-03-06 09:08:44 -05008import time
Benjamin Peterson101447b2015-03-09 10:37:50 -04009import unittest
10
11from test import support
12from math import isinf, isnan, copysign, ldexp
Michael W. Hudsonba283e22005-05-27 15:23:20 +000013
Christian Heimes53876d92008-04-19 00:31:39 +000014INF = float("inf")
15NAN = float("nan")
Christian Heimes99170a52007-12-19 02:07:34 +000016
Benjamin Petersone401c682010-07-02 23:25:44 +000017have_getformat = hasattr(float, "__getformat__")
18requires_getformat = unittest.skipUnless(have_getformat,
19 "requires __getformat__")
20requires_setformat = unittest.skipUnless(hasattr(float, "__setformat__"),
21 "requires __setformat__")
Mark Dickinson9ab44b52009-12-30 16:22:49 +000022
Eric Smith0923d1d2009-04-16 20:16:10 +000023#locate file with float format test values
24test_dir = os.path.dirname(__file__) or os.curdir
25format_testfile = os.path.join(test_dir, 'formatfloat_testcases.txt')
26
Serhiy Storchaka15095802015-11-25 15:47:01 +020027class FloatSubclass(float):
28 pass
29
30class OtherFloatSubclass(float):
31 pass
32
Christian Heimes81ee3ef2008-05-04 22:42:01 +000033class GeneralFloatCases(unittest.TestCase):
34
35 def test_float(self):
36 self.assertEqual(float(3.14), 3.14)
37 self.assertEqual(float(314), 314.0)
38 self.assertEqual(float(" 3.14 "), 3.14)
39 self.assertRaises(ValueError, float, " 0x3.1 ")
40 self.assertRaises(ValueError, float, " -0x3.p-1 ")
41 self.assertRaises(ValueError, float, " +0x3.p-1 ")
42 self.assertRaises(ValueError, float, "++3.14")
43 self.assertRaises(ValueError, float, "+-3.14")
44 self.assertRaises(ValueError, float, "-+3.14")
45 self.assertRaises(ValueError, float, "--3.14")
Eric Smith0923d1d2009-04-16 20:16:10 +000046 self.assertRaises(ValueError, float, ".nan")
47 self.assertRaises(ValueError, float, "+.inf")
48 self.assertRaises(ValueError, float, ".")
49 self.assertRaises(ValueError, float, "-.")
Alexander Belopolsky942af5a2010-12-04 03:38:46 +000050 self.assertRaises(TypeError, float, {})
Ezio Melottia5b95992013-11-07 19:18:34 +020051 self.assertRaisesRegex(TypeError, "not 'dict'", float, {})
Alexander Belopolsky942af5a2010-12-04 03:38:46 +000052 # Lone surrogate
53 self.assertRaises(UnicodeEncodeError, float, '\uD8F0')
Mark Dickinsonc2d86892010-02-12 21:18:34 +000054 # check that we don't accept alternate exponent markers
55 self.assertRaises(ValueError, float, "-1.7d29")
56 self.assertRaises(ValueError, float, "3D-14")
Alexander Belopolsky942af5a2010-12-04 03:38:46 +000057 self.assertEqual(float(" \u0663.\u0661\u0664 "), 3.14)
58 self.assertEqual(float("\N{EM SPACE}3.14\N{EN SPACE}"), 3.14)
Mark Dickinsona9023be2009-10-27 22:12:20 +000059 # extra long strings should not be a problem
60 float(b'.' + b'1'*1000)
61 float('.' + '1'*1000)
Christian Heimes81ee3ef2008-05-04 22:42:01 +000062
Martin Pantereeb896c2015-11-07 02:32:21 +000063 def test_non_numeric_input_types(self):
64 # Test possible non-numeric types for the argument x, including
65 # subclasses of the explicitly documented accepted types.
66 class CustomStr(str): pass
67 class CustomBytes(bytes): pass
68 class CustomByteArray(bytearray): pass
69
70 factories = [
71 bytes,
72 bytearray,
73 lambda b: CustomStr(b.decode()),
74 CustomBytes,
75 CustomByteArray,
76 memoryview,
77 ]
78 try:
79 from array import array
80 except ImportError:
81 pass
82 else:
83 factories.append(lambda b: array('B', b))
84
85 for f in factories:
86 x = f(b" 3.14 ")
87 with self.subTest(type(x)):
88 self.assertEqual(float(x), 3.14)
89 with self.assertRaisesRegex(ValueError, "could not convert"):
90 float(f(b'A' * 0x10))
91
92 def test_float_memoryview(self):
93 self.assertEqual(float(memoryview(b'12.3')[1:4]), 2.3)
94 self.assertEqual(float(memoryview(b'12.3\x00')[1:4]), 2.3)
95 self.assertEqual(float(memoryview(b'12.3 ')[1:4]), 2.3)
96 self.assertEqual(float(memoryview(b'12.3A')[1:4]), 2.3)
97 self.assertEqual(float(memoryview(b'12.34')[1:4]), 2.3)
98
Alexander Belopolsky942af5a2010-12-04 03:38:46 +000099 def test_error_message(self):
100 testlist = ('\xbd', '123\xbd', ' 123 456 ')
101 for s in testlist:
102 try:
103 float(s)
104 except ValueError as e:
105 self.assertIn(s.strip(), e.args[0])
106 else:
107 self.fail("Expected int(%r) to raise a ValueError", s)
108
109
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000110 @support.run_with_locale('LC_NUMERIC', 'fr_FR', 'de_DE')
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000111 def test_float_with_comma(self):
112 # set locale to something that doesn't use '.' for the decimal point
113 # float must not accept the locale specific decimal point but
Ezio Melotti13925002011-03-16 11:05:33 +0200114 # it still has to accept the normal python syntax
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000115 import locale
116 if not locale.localeconv()['decimal_point'] == ',':
Zachary Ware9fe6d862013-12-08 00:20:35 -0600117 self.skipTest('decimal_point is not ","')
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000118
119 self.assertEqual(float(" 3.14 "), 3.14)
120 self.assertEqual(float("+3.14 "), 3.14)
121 self.assertEqual(float("-3.14 "), -3.14)
122 self.assertEqual(float(".14 "), .14)
123 self.assertEqual(float("3. "), 3.0)
124 self.assertEqual(float("3.e3 "), 3000.0)
125 self.assertEqual(float("3.2e3 "), 3200.0)
126 self.assertEqual(float("2.5e-1 "), 0.25)
127 self.assertEqual(float("5e-1"), 0.5)
128 self.assertRaises(ValueError, float, " 3,14 ")
129 self.assertRaises(ValueError, float, " +3,14 ")
130 self.assertRaises(ValueError, float, " -3,14 ")
131 self.assertRaises(ValueError, float, " 0x3.1 ")
132 self.assertRaises(ValueError, float, " -0x3.p-1 ")
133 self.assertRaises(ValueError, float, " +0x3.p-1 ")
134 self.assertEqual(float(" 25.e-1 "), 2.5)
Eli Bendersky67ebabd2011-02-25 10:14:17 +0000135 self.assertAlmostEqual(float(" .25e-1 "), .025)
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000136
137 def test_floatconversion(self):
138 # Make sure that calls to __float__() work properly
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000139 class Foo1(object):
140 def __float__(self):
141 return 42.
142
143 class Foo2(float):
144 def __float__(self):
145 return 42.
146
147 class Foo3(float):
148 def __new__(cls, value=0.):
149 return float.__new__(cls, 2*value)
150
151 def __float__(self):
152 return self
153
154 class Foo4(float):
155 def __float__(self):
156 return 42
157
Benjamin Peterson2808d3c2009-04-15 21:34:27 +0000158 # Issue 5759: __float__ not called on str subclasses (though it is on
159 # unicode subclasses).
160 class FooStr(str):
161 def __float__(self):
162 return float(str(self)) + 1
163
Serhiy Storchaka16931c32016-06-03 21:42:55 +0300164 self.assertEqual(float(Foo1()), 42.)
165 self.assertEqual(float(Foo2()), 42.)
166 with self.assertWarns(DeprecationWarning):
167 self.assertEqual(float(Foo3(21)), 42.)
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000168 self.assertRaises(TypeError, float, Foo4(42))
Serhiy Storchaka16931c32016-06-03 21:42:55 +0300169 self.assertEqual(float(FooStr('8')), 9.)
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000170
Benjamin Petersona9157232015-03-06 09:08:44 -0500171 class Foo5:
172 def __float__(self):
173 return ""
174 self.assertRaises(TypeError, time.sleep, Foo5())
175
Serhiy Storchaka15095802015-11-25 15:47:01 +0200176 # Issue #24731
177 class F:
178 def __float__(self):
179 return OtherFloatSubclass(42.)
Serhiy Storchaka16931c32016-06-03 21:42:55 +0300180 with self.assertWarns(DeprecationWarning):
181 self.assertEqual(float(F()), 42.)
182 with self.assertWarns(DeprecationWarning):
183 self.assertIs(type(float(F())), float)
184 with self.assertWarns(DeprecationWarning):
185 self.assertEqual(FloatSubclass(F()), 42.)
186 with self.assertWarns(DeprecationWarning):
187 self.assertIs(type(FloatSubclass(F())), FloatSubclass)
Serhiy Storchaka15095802015-11-25 15:47:01 +0200188
Benjamin Petersonb3b8ba62011-10-28 19:42:48 -0400189 def test_is_integer(self):
190 self.assertFalse((1.1).is_integer())
191 self.assertTrue((1.).is_integer())
192 self.assertFalse(float("nan").is_integer())
193 self.assertFalse(float("inf").is_integer())
194
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000195 def test_floatasratio(self):
196 for f, ratio in [
197 (0.875, (7, 8)),
198 (-0.875, (-7, 8)),
199 (0.0, (0, 1)),
200 (11.5, (23, 2)),
201 ]:
202 self.assertEqual(f.as_integer_ratio(), ratio)
203
204 for i in range(10000):
205 f = random.random()
206 f *= 10 ** random.randint(-100, 100)
207 n, d = f.as_integer_ratio()
208 self.assertEqual(float(n).__truediv__(d), f)
209
210 R = fractions.Fraction
211 self.assertEqual(R(0, 1),
212 R(*float(0.0).as_integer_ratio()))
213 self.assertEqual(R(5, 2),
214 R(*float(2.5).as_integer_ratio()))
215 self.assertEqual(R(1, 2),
216 R(*float(0.5).as_integer_ratio()))
217 self.assertEqual(R(4728779608739021, 2251799813685248),
218 R(*float(2.1).as_integer_ratio()))
219 self.assertEqual(R(-4728779608739021, 2251799813685248),
220 R(*float(-2.1).as_integer_ratio()))
221 self.assertEqual(R(-2100, 1),
222 R(*float(-2100.0).as_integer_ratio()))
223
224 self.assertRaises(OverflowError, float('inf').as_integer_ratio)
225 self.assertRaises(OverflowError, float('-inf').as_integer_ratio)
226 self.assertRaises(ValueError, float('nan').as_integer_ratio)
227
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000228 def test_float_containment(self):
229 floats = (INF, -INF, 0.0, 1.0, NAN)
230 for f in floats:
Benjamin Peterson577473f2010-01-19 00:09:57 +0000231 self.assertIn(f, [f])
Benjamin Peterson577473f2010-01-19 00:09:57 +0000232 self.assertIn(f, (f,))
Benjamin Peterson577473f2010-01-19 00:09:57 +0000233 self.assertIn(f, {f})
Benjamin Peterson577473f2010-01-19 00:09:57 +0000234 self.assertIn(f, {f: None})
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000235 self.assertEqual([f].count(f), 1, "[].count('%r') != 1" % f)
Benjamin Peterson577473f2010-01-19 00:09:57 +0000236 self.assertIn(f, floats)
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000237
238 for f in floats:
239 # nonidentical containers, same type, same contents
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000240 self.assertTrue([f] == [f], "[%r] != [%r]" % (f, f))
241 self.assertTrue((f,) == (f,), "(%r,) != (%r,)" % (f, f))
242 self.assertTrue({f} == {f}, "{%r} != {%r}" % (f, f))
243 self.assertTrue({f : None} == {f: None}, "{%r : None} != "
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000244 "{%r : None}" % (f, f))
245
246 # identical containers
247 l, t, s, d = [f], (f,), {f}, {f: None}
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000248 self.assertTrue(l == l, "[%r] not equal to itself" % f)
249 self.assertTrue(t == t, "(%r,) not equal to itself" % f)
250 self.assertTrue(s == s, "{%r} not equal to itself" % f)
251 self.assertTrue(d == d, "{%r : None} not equal to itself" % f)
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000252
Mark Dickinson9ab44b52009-12-30 16:22:49 +0000253 def assertEqualAndEqualSign(self, a, b):
254 # fail unless a == b and a and b have the same sign bit;
255 # the only difference from assertEqual is that this test
Ezio Melotti13925002011-03-16 11:05:33 +0200256 # distinguishes -0.0 and 0.0.
Mark Dickinson9ab44b52009-12-30 16:22:49 +0000257 self.assertEqual((a, copysign(1.0, a)), (b, copysign(1.0, b)))
258
Eric Smith3ab08ca2010-12-04 15:17:38 +0000259 @support.requires_IEEE_754
Mark Dickinsond2a9b202010-12-04 12:25:30 +0000260 def test_float_mod(self):
261 # Check behaviour of % operator for IEEE 754 special cases.
262 # In particular, check signs of zeros.
263 mod = operator.mod
264
265 self.assertEqualAndEqualSign(mod(-1.0, 1.0), 0.0)
266 self.assertEqualAndEqualSign(mod(-1e-100, 1.0), 1.0)
267 self.assertEqualAndEqualSign(mod(-0.0, 1.0), 0.0)
268 self.assertEqualAndEqualSign(mod(0.0, 1.0), 0.0)
269 self.assertEqualAndEqualSign(mod(1e-100, 1.0), 1e-100)
270 self.assertEqualAndEqualSign(mod(1.0, 1.0), 0.0)
271
272 self.assertEqualAndEqualSign(mod(-1.0, -1.0), -0.0)
273 self.assertEqualAndEqualSign(mod(-1e-100, -1.0), -1e-100)
274 self.assertEqualAndEqualSign(mod(-0.0, -1.0), -0.0)
275 self.assertEqualAndEqualSign(mod(0.0, -1.0), -0.0)
276 self.assertEqualAndEqualSign(mod(1e-100, -1.0), -1.0)
277 self.assertEqualAndEqualSign(mod(1.0, -1.0), -0.0)
278
Eric Smith3ab08ca2010-12-04 15:17:38 +0000279 @support.requires_IEEE_754
Mark Dickinson9ab44b52009-12-30 16:22:49 +0000280 def test_float_pow(self):
281 # test builtin pow and ** operator for IEEE 754 special cases.
282 # Special cases taken from section F.9.4.4 of the C99 specification
283
284 for pow_op in pow, operator.pow:
285 # x**NAN is NAN for any x except 1
286 self.assertTrue(isnan(pow_op(-INF, NAN)))
287 self.assertTrue(isnan(pow_op(-2.0, NAN)))
288 self.assertTrue(isnan(pow_op(-1.0, NAN)))
289 self.assertTrue(isnan(pow_op(-0.5, NAN)))
290 self.assertTrue(isnan(pow_op(-0.0, NAN)))
291 self.assertTrue(isnan(pow_op(0.0, NAN)))
292 self.assertTrue(isnan(pow_op(0.5, NAN)))
293 self.assertTrue(isnan(pow_op(2.0, NAN)))
294 self.assertTrue(isnan(pow_op(INF, NAN)))
295 self.assertTrue(isnan(pow_op(NAN, NAN)))
296
297 # NAN**y is NAN for any y except +-0
298 self.assertTrue(isnan(pow_op(NAN, -INF)))
299 self.assertTrue(isnan(pow_op(NAN, -2.0)))
300 self.assertTrue(isnan(pow_op(NAN, -1.0)))
301 self.assertTrue(isnan(pow_op(NAN, -0.5)))
302 self.assertTrue(isnan(pow_op(NAN, 0.5)))
303 self.assertTrue(isnan(pow_op(NAN, 1.0)))
304 self.assertTrue(isnan(pow_op(NAN, 2.0)))
305 self.assertTrue(isnan(pow_op(NAN, INF)))
306
307 # (+-0)**y raises ZeroDivisionError for y a negative odd integer
308 self.assertRaises(ZeroDivisionError, pow_op, -0.0, -1.0)
309 self.assertRaises(ZeroDivisionError, pow_op, 0.0, -1.0)
310
311 # (+-0)**y raises ZeroDivisionError for y finite and negative
312 # but not an odd integer
313 self.assertRaises(ZeroDivisionError, pow_op, -0.0, -2.0)
314 self.assertRaises(ZeroDivisionError, pow_op, -0.0, -0.5)
315 self.assertRaises(ZeroDivisionError, pow_op, 0.0, -2.0)
316 self.assertRaises(ZeroDivisionError, pow_op, 0.0, -0.5)
317
318 # (+-0)**y is +-0 for y a positive odd integer
319 self.assertEqualAndEqualSign(pow_op(-0.0, 1.0), -0.0)
320 self.assertEqualAndEqualSign(pow_op(0.0, 1.0), 0.0)
321
322 # (+-0)**y is 0 for y finite and positive but not an odd integer
323 self.assertEqualAndEqualSign(pow_op(-0.0, 0.5), 0.0)
324 self.assertEqualAndEqualSign(pow_op(-0.0, 2.0), 0.0)
325 self.assertEqualAndEqualSign(pow_op(0.0, 0.5), 0.0)
326 self.assertEqualAndEqualSign(pow_op(0.0, 2.0), 0.0)
327
328 # (-1)**+-inf is 1
329 self.assertEqualAndEqualSign(pow_op(-1.0, -INF), 1.0)
330 self.assertEqualAndEqualSign(pow_op(-1.0, INF), 1.0)
331
332 # 1**y is 1 for any y, even if y is an infinity or nan
333 self.assertEqualAndEqualSign(pow_op(1.0, -INF), 1.0)
334 self.assertEqualAndEqualSign(pow_op(1.0, -2.0), 1.0)
335 self.assertEqualAndEqualSign(pow_op(1.0, -1.0), 1.0)
336 self.assertEqualAndEqualSign(pow_op(1.0, -0.5), 1.0)
337 self.assertEqualAndEqualSign(pow_op(1.0, -0.0), 1.0)
338 self.assertEqualAndEqualSign(pow_op(1.0, 0.0), 1.0)
339 self.assertEqualAndEqualSign(pow_op(1.0, 0.5), 1.0)
340 self.assertEqualAndEqualSign(pow_op(1.0, 1.0), 1.0)
341 self.assertEqualAndEqualSign(pow_op(1.0, 2.0), 1.0)
342 self.assertEqualAndEqualSign(pow_op(1.0, INF), 1.0)
343 self.assertEqualAndEqualSign(pow_op(1.0, NAN), 1.0)
344
345 # x**+-0 is 1 for any x, even if x is a zero, infinity, or nan
346 self.assertEqualAndEqualSign(pow_op(-INF, 0.0), 1.0)
347 self.assertEqualAndEqualSign(pow_op(-2.0, 0.0), 1.0)
348 self.assertEqualAndEqualSign(pow_op(-1.0, 0.0), 1.0)
349 self.assertEqualAndEqualSign(pow_op(-0.5, 0.0), 1.0)
350 self.assertEqualAndEqualSign(pow_op(-0.0, 0.0), 1.0)
351 self.assertEqualAndEqualSign(pow_op(0.0, 0.0), 1.0)
352 self.assertEqualAndEqualSign(pow_op(0.5, 0.0), 1.0)
353 self.assertEqualAndEqualSign(pow_op(1.0, 0.0), 1.0)
354 self.assertEqualAndEqualSign(pow_op(2.0, 0.0), 1.0)
355 self.assertEqualAndEqualSign(pow_op(INF, 0.0), 1.0)
356 self.assertEqualAndEqualSign(pow_op(NAN, 0.0), 1.0)
357 self.assertEqualAndEqualSign(pow_op(-INF, -0.0), 1.0)
358 self.assertEqualAndEqualSign(pow_op(-2.0, -0.0), 1.0)
359 self.assertEqualAndEqualSign(pow_op(-1.0, -0.0), 1.0)
360 self.assertEqualAndEqualSign(pow_op(-0.5, -0.0), 1.0)
361 self.assertEqualAndEqualSign(pow_op(-0.0, -0.0), 1.0)
362 self.assertEqualAndEqualSign(pow_op(0.0, -0.0), 1.0)
363 self.assertEqualAndEqualSign(pow_op(0.5, -0.0), 1.0)
364 self.assertEqualAndEqualSign(pow_op(1.0, -0.0), 1.0)
365 self.assertEqualAndEqualSign(pow_op(2.0, -0.0), 1.0)
366 self.assertEqualAndEqualSign(pow_op(INF, -0.0), 1.0)
367 self.assertEqualAndEqualSign(pow_op(NAN, -0.0), 1.0)
368
369 # x**y defers to complex pow for finite negative x and
370 # non-integral y.
371 self.assertEqual(type(pow_op(-2.0, -0.5)), complex)
372 self.assertEqual(type(pow_op(-2.0, 0.5)), complex)
373 self.assertEqual(type(pow_op(-1.0, -0.5)), complex)
374 self.assertEqual(type(pow_op(-1.0, 0.5)), complex)
375 self.assertEqual(type(pow_op(-0.5, -0.5)), complex)
376 self.assertEqual(type(pow_op(-0.5, 0.5)), complex)
377
378 # x**-INF is INF for abs(x) < 1
379 self.assertEqualAndEqualSign(pow_op(-0.5, -INF), INF)
380 self.assertEqualAndEqualSign(pow_op(-0.0, -INF), INF)
381 self.assertEqualAndEqualSign(pow_op(0.0, -INF), INF)
382 self.assertEqualAndEqualSign(pow_op(0.5, -INF), INF)
383
384 # x**-INF is 0 for abs(x) > 1
385 self.assertEqualAndEqualSign(pow_op(-INF, -INF), 0.0)
386 self.assertEqualAndEqualSign(pow_op(-2.0, -INF), 0.0)
387 self.assertEqualAndEqualSign(pow_op(2.0, -INF), 0.0)
388 self.assertEqualAndEqualSign(pow_op(INF, -INF), 0.0)
389
390 # x**INF is 0 for abs(x) < 1
391 self.assertEqualAndEqualSign(pow_op(-0.5, INF), 0.0)
392 self.assertEqualAndEqualSign(pow_op(-0.0, INF), 0.0)
393 self.assertEqualAndEqualSign(pow_op(0.0, INF), 0.0)
394 self.assertEqualAndEqualSign(pow_op(0.5, INF), 0.0)
395
396 # x**INF is INF for abs(x) > 1
397 self.assertEqualAndEqualSign(pow_op(-INF, INF), INF)
398 self.assertEqualAndEqualSign(pow_op(-2.0, INF), INF)
399 self.assertEqualAndEqualSign(pow_op(2.0, INF), INF)
400 self.assertEqualAndEqualSign(pow_op(INF, INF), INF)
401
402 # (-INF)**y is -0.0 for y a negative odd integer
403 self.assertEqualAndEqualSign(pow_op(-INF, -1.0), -0.0)
404
405 # (-INF)**y is 0.0 for y negative but not an odd integer
406 self.assertEqualAndEqualSign(pow_op(-INF, -0.5), 0.0)
407 self.assertEqualAndEqualSign(pow_op(-INF, -2.0), 0.0)
408
409 # (-INF)**y is -INF for y a positive odd integer
410 self.assertEqualAndEqualSign(pow_op(-INF, 1.0), -INF)
411
412 # (-INF)**y is INF for y positive but not an odd integer
413 self.assertEqualAndEqualSign(pow_op(-INF, 0.5), INF)
414 self.assertEqualAndEqualSign(pow_op(-INF, 2.0), INF)
415
416 # INF**y is INF for y positive
417 self.assertEqualAndEqualSign(pow_op(INF, 0.5), INF)
418 self.assertEqualAndEqualSign(pow_op(INF, 1.0), INF)
419 self.assertEqualAndEqualSign(pow_op(INF, 2.0), INF)
420
421 # INF**y is 0.0 for y negative
422 self.assertEqualAndEqualSign(pow_op(INF, -2.0), 0.0)
423 self.assertEqualAndEqualSign(pow_op(INF, -1.0), 0.0)
424 self.assertEqualAndEqualSign(pow_op(INF, -0.5), 0.0)
425
426 # basic checks not covered by the special cases above
427 self.assertEqualAndEqualSign(pow_op(-2.0, -2.0), 0.25)
428 self.assertEqualAndEqualSign(pow_op(-2.0, -1.0), -0.5)
429 self.assertEqualAndEqualSign(pow_op(-2.0, -0.0), 1.0)
430 self.assertEqualAndEqualSign(pow_op(-2.0, 0.0), 1.0)
431 self.assertEqualAndEqualSign(pow_op(-2.0, 1.0), -2.0)
432 self.assertEqualAndEqualSign(pow_op(-2.0, 2.0), 4.0)
433 self.assertEqualAndEqualSign(pow_op(-1.0, -2.0), 1.0)
434 self.assertEqualAndEqualSign(pow_op(-1.0, -1.0), -1.0)
435 self.assertEqualAndEqualSign(pow_op(-1.0, -0.0), 1.0)
436 self.assertEqualAndEqualSign(pow_op(-1.0, 0.0), 1.0)
437 self.assertEqualAndEqualSign(pow_op(-1.0, 1.0), -1.0)
438 self.assertEqualAndEqualSign(pow_op(-1.0, 2.0), 1.0)
439 self.assertEqualAndEqualSign(pow_op(2.0, -2.0), 0.25)
440 self.assertEqualAndEqualSign(pow_op(2.0, -1.0), 0.5)
441 self.assertEqualAndEqualSign(pow_op(2.0, -0.0), 1.0)
442 self.assertEqualAndEqualSign(pow_op(2.0, 0.0), 1.0)
443 self.assertEqualAndEqualSign(pow_op(2.0, 1.0), 2.0)
444 self.assertEqualAndEqualSign(pow_op(2.0, 2.0), 4.0)
445
446 # 1 ** large and -1 ** large; some libms apparently
447 # have problems with these
448 self.assertEqualAndEqualSign(pow_op(1.0, -1e100), 1.0)
449 self.assertEqualAndEqualSign(pow_op(1.0, 1e100), 1.0)
450 self.assertEqualAndEqualSign(pow_op(-1.0, -1e100), 1.0)
451 self.assertEqualAndEqualSign(pow_op(-1.0, 1e100), 1.0)
452
453 # check sign for results that underflow to 0
454 self.assertEqualAndEqualSign(pow_op(-2.0, -2000.0), 0.0)
455 self.assertEqual(type(pow_op(-2.0, -2000.5)), complex)
456 self.assertEqualAndEqualSign(pow_op(-2.0, -2001.0), -0.0)
457 self.assertEqualAndEqualSign(pow_op(2.0, -2000.0), 0.0)
458 self.assertEqualAndEqualSign(pow_op(2.0, -2000.5), 0.0)
459 self.assertEqualAndEqualSign(pow_op(2.0, -2001.0), 0.0)
460 self.assertEqualAndEqualSign(pow_op(-0.5, 2000.0), 0.0)
461 self.assertEqual(type(pow_op(-0.5, 2000.5)), complex)
462 self.assertEqualAndEqualSign(pow_op(-0.5, 2001.0), -0.0)
463 self.assertEqualAndEqualSign(pow_op(0.5, 2000.0), 0.0)
464 self.assertEqualAndEqualSign(pow_op(0.5, 2000.5), 0.0)
465 self.assertEqualAndEqualSign(pow_op(0.5, 2001.0), 0.0)
466
467 # check we don't raise an exception for subnormal results,
468 # and validate signs. Tests currently disabled, since
469 # they fail on systems where a subnormal result from pow
470 # is flushed to zero (e.g. Debian/ia64.)
471 #self.assertTrue(0.0 < pow_op(0.5, 1048) < 1e-315)
472 #self.assertTrue(0.0 < pow_op(-0.5, 1048) < 1e-315)
473 #self.assertTrue(0.0 < pow_op(0.5, 1047) < 1e-315)
474 #self.assertTrue(0.0 > pow_op(-0.5, 1047) > -1e-315)
475 #self.assertTrue(0.0 < pow_op(2.0, -1048) < 1e-315)
476 #self.assertTrue(0.0 < pow_op(-2.0, -1048) < 1e-315)
477 #self.assertTrue(0.0 < pow_op(2.0, -1047) < 1e-315)
478 #self.assertTrue(0.0 > pow_op(-2.0, -1047) > -1e-315)
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000479
480
Benjamin Petersone401c682010-07-02 23:25:44 +0000481@requires_setformat
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000482class FormatFunctionsTestCase(unittest.TestCase):
483
484 def setUp(self):
485 self.save_formats = {'double':float.__getformat__('double'),
486 'float':float.__getformat__('float')}
487
488 def tearDown(self):
489 float.__setformat__('double', self.save_formats['double'])
490 float.__setformat__('float', self.save_formats['float'])
491
492 def test_getformat(self):
Ezio Melottib58e0bd2010-01-23 15:40:09 +0000493 self.assertIn(float.__getformat__('double'),
494 ['unknown', 'IEEE, big-endian', 'IEEE, little-endian'])
495 self.assertIn(float.__getformat__('float'),
496 ['unknown', 'IEEE, big-endian', 'IEEE, little-endian'])
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000497 self.assertRaises(ValueError, float.__getformat__, 'chicken')
498 self.assertRaises(TypeError, float.__getformat__, 1)
499
500 def test_setformat(self):
501 for t in 'double', 'float':
502 float.__setformat__(t, 'unknown')
503 if self.save_formats[t] == 'IEEE, big-endian':
504 self.assertRaises(ValueError, float.__setformat__,
505 t, 'IEEE, little-endian')
506 elif self.save_formats[t] == 'IEEE, little-endian':
507 self.assertRaises(ValueError, float.__setformat__,
508 t, 'IEEE, big-endian')
509 else:
510 self.assertRaises(ValueError, float.__setformat__,
511 t, 'IEEE, big-endian')
512 self.assertRaises(ValueError, float.__setformat__,
513 t, 'IEEE, little-endian')
514 self.assertRaises(ValueError, float.__setformat__,
515 t, 'chicken')
516 self.assertRaises(ValueError, float.__setformat__,
517 'chicken', 'unknown')
518
Guido van Rossum2be161d2007-05-15 20:43:51 +0000519BE_DOUBLE_INF = b'\x7f\xf0\x00\x00\x00\x00\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000520LE_DOUBLE_INF = bytes(reversed(BE_DOUBLE_INF))
Guido van Rossum2be161d2007-05-15 20:43:51 +0000521BE_DOUBLE_NAN = b'\x7f\xf8\x00\x00\x00\x00\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000522LE_DOUBLE_NAN = bytes(reversed(BE_DOUBLE_NAN))
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000523
Guido van Rossum2be161d2007-05-15 20:43:51 +0000524BE_FLOAT_INF = b'\x7f\x80\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000525LE_FLOAT_INF = bytes(reversed(BE_FLOAT_INF))
Guido van Rossum2be161d2007-05-15 20:43:51 +0000526BE_FLOAT_NAN = b'\x7f\xc0\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000527LE_FLOAT_NAN = bytes(reversed(BE_FLOAT_NAN))
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000528
529# on non-IEEE platforms, attempting to unpack a bit pattern
530# representing an infinity or a NaN should raise an exception.
531
Benjamin Petersone401c682010-07-02 23:25:44 +0000532@requires_setformat
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000533class UnknownFormatTestCase(unittest.TestCase):
534 def setUp(self):
535 self.save_formats = {'double':float.__getformat__('double'),
536 'float':float.__getformat__('float')}
537 float.__setformat__('double', 'unknown')
538 float.__setformat__('float', 'unknown')
Tim Peters5d36a552005-06-03 22:40:27 +0000539
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000540 def tearDown(self):
541 float.__setformat__('double', self.save_formats['double'])
542 float.__setformat__('float', self.save_formats['float'])
543
544 def test_double_specials_dont_unpack(self):
545 for fmt, data in [('>d', BE_DOUBLE_INF),
546 ('>d', BE_DOUBLE_NAN),
547 ('<d', LE_DOUBLE_INF),
548 ('<d', LE_DOUBLE_NAN)]:
549 self.assertRaises(ValueError, struct.unpack, fmt, data)
550
551 def test_float_specials_dont_unpack(self):
552 for fmt, data in [('>f', BE_FLOAT_INF),
553 ('>f', BE_FLOAT_NAN),
554 ('<f', LE_FLOAT_INF),
555 ('<f', LE_FLOAT_NAN)]:
556 self.assertRaises(ValueError, struct.unpack, fmt, data)
557
558
559# on an IEEE platform, all we guarantee is that bit patterns
560# representing infinities or NaNs do not raise an exception; all else
561# is accident (today).
Guido van Rossum04110fb2007-08-24 16:32:05 +0000562# let's also try to guarantee that -0.0 and 0.0 don't get confused.
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000563
564class IEEEFormatTestCase(unittest.TestCase):
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000565
Eric Smith3ab08ca2010-12-04 15:17:38 +0000566 @support.requires_IEEE_754
Benjamin Petersone401c682010-07-02 23:25:44 +0000567 def test_double_specials_do_unpack(self):
568 for fmt, data in [('>d', BE_DOUBLE_INF),
569 ('>d', BE_DOUBLE_NAN),
570 ('<d', LE_DOUBLE_INF),
571 ('<d', LE_DOUBLE_NAN)]:
572 struct.unpack(fmt, data)
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000573
Eric Smith3ab08ca2010-12-04 15:17:38 +0000574 @support.requires_IEEE_754
Benjamin Petersone401c682010-07-02 23:25:44 +0000575 def test_float_specials_do_unpack(self):
576 for fmt, data in [('>f', BE_FLOAT_INF),
577 ('>f', BE_FLOAT_NAN),
578 ('<f', LE_FLOAT_INF),
579 ('<f', LE_FLOAT_NAN)]:
580 struct.unpack(fmt, data)
Guido van Rossum04110fb2007-08-24 16:32:05 +0000581
Eric Smith8c663262007-08-25 02:26:07 +0000582class FormatTestCase(unittest.TestCase):
Benjamin Petersone401c682010-07-02 23:25:44 +0000583
Eric Smith11fe3e02007-08-31 01:33:06 +0000584 def test_format(self):
Eric Smith8c663262007-08-25 02:26:07 +0000585 # these should be rewritten to use both format(x, spec) and
586 # x.__format__(spec)
587
588 self.assertEqual(format(0.0, 'f'), '0.000000')
589
590 # the default is 'g', except for empty format spec
591 self.assertEqual(format(0.0, ''), '0.0')
592 self.assertEqual(format(0.01, ''), '0.01')
593 self.assertEqual(format(0.01, 'g'), '0.01')
594
Eric Smith63376222009-05-05 14:04:18 +0000595 # empty presentation type should format in the same way as str
596 # (issue 5920)
597 x = 100/7.
598 self.assertEqual(format(x, ''), str(x))
599 self.assertEqual(format(x, '-'), str(x))
600 self.assertEqual(format(x, '>'), str(x))
601 self.assertEqual(format(x, '2'), str(x))
Eric Smith8c663262007-08-25 02:26:07 +0000602
603 self.assertEqual(format(1.0, 'f'), '1.000000')
Eric Smith8c663262007-08-25 02:26:07 +0000604
605 self.assertEqual(format(-1.0, 'f'), '-1.000000')
Eric Smith8c663262007-08-25 02:26:07 +0000606
607 self.assertEqual(format( 1.0, ' f'), ' 1.000000')
608 self.assertEqual(format(-1.0, ' f'), '-1.000000')
609 self.assertEqual(format( 1.0, '+f'), '+1.000000')
610 self.assertEqual(format(-1.0, '+f'), '-1.000000')
611
612 # % formatting
613 self.assertEqual(format(-1.0, '%'), '-100.000000%')
614
615 # conversion to string should fail
616 self.assertRaises(ValueError, format, 3.0, "s")
617
Eric Smith7b69c6c2008-01-27 21:07:59 +0000618 # other format specifiers shouldn't work on floats,
619 # in particular int specifiers
620 for format_spec in ([chr(x) for x in range(ord('a'), ord('z')+1)] +
621 [chr(x) for x in range(ord('A'), ord('Z')+1)]):
622 if not format_spec in 'eEfFgGn%':
623 self.assertRaises(ValueError, format, 0.0, format_spec)
624 self.assertRaises(ValueError, format, 1.0, format_spec)
625 self.assertRaises(ValueError, format, -1.0, format_spec)
626 self.assertRaises(ValueError, format, 1e100, format_spec)
627 self.assertRaises(ValueError, format, -1e100, format_spec)
628 self.assertRaises(ValueError, format, 1e-100, format_spec)
629 self.assertRaises(ValueError, format, -1e-100, format_spec)
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000630
Eric Smith741191f2009-05-06 13:08:15 +0000631 # issue 3382
632 self.assertEqual(format(NAN, 'f'), 'nan')
633 self.assertEqual(format(NAN, 'F'), 'NAN')
634 self.assertEqual(format(INF, 'f'), 'inf')
635 self.assertEqual(format(INF, 'F'), 'INF')
636
Eric Smith3ab08ca2010-12-04 15:17:38 +0000637 @support.requires_IEEE_754
Eric Smith0923d1d2009-04-16 20:16:10 +0000638 def test_format_testfile(self):
Brian Curtin076623b2010-10-31 00:03:45 +0000639 with open(format_testfile) as testfile:
640 for line in testfile:
641 if line.startswith('--'):
642 continue
643 line = line.strip()
644 if not line:
645 continue
Eric Smith0923d1d2009-04-16 20:16:10 +0000646
Brian Curtin076623b2010-10-31 00:03:45 +0000647 lhs, rhs = map(str.strip, line.split('->'))
648 fmt, arg = lhs.split()
649 self.assertEqual(fmt % float(arg), rhs)
650 self.assertEqual(fmt % -float(arg), '-' + rhs)
Eric Smith0923d1d2009-04-16 20:16:10 +0000651
Mark Dickinsond3ca5572009-04-29 18:47:07 +0000652 def test_issue5864(self):
Ezio Melottib3aedd42010-11-20 19:04:17 +0000653 self.assertEqual(format(123.456, '.4'), '123.5')
654 self.assertEqual(format(1234.56, '.4'), '1.235e+03')
655 self.assertEqual(format(12345.6, '.4'), '1.235e+04')
Mark Dickinsond3ca5572009-04-29 18:47:07 +0000656
Mark Dickinson7efad9e2009-04-17 20:59:58 +0000657class ReprTestCase(unittest.TestCase):
658 def test_repr(self):
659 floats_file = open(os.path.join(os.path.split(__file__)[0],
660 'floating_points.txt'))
661 for line in floats_file:
662 line = line.strip()
663 if not line or line.startswith('#'):
664 continue
665 v = eval(line)
666 self.assertEqual(v, eval(repr(v)))
667 floats_file.close()
668
Eric Smith0923d1d2009-04-16 20:16:10 +0000669 @unittest.skipUnless(getattr(sys, 'float_repr_style', '') == 'short',
670 "applies only when using short float repr style")
671 def test_short_repr(self):
672 # test short float repr introduced in Python 3.1. One aspect
673 # of this repr is that we get some degree of str -> float ->
674 # str roundtripping. In particular, for any numeric string
675 # containing 15 or fewer significant digits, those exact same
676 # digits (modulo trailing zeros) should appear in the output.
677 # No more repr(0.03) -> "0.029999999999999999"!
678
679 test_strings = [
680 # output always includes *either* a decimal point and at
681 # least one digit after that point, or an exponent.
682 '0.0',
683 '1.0',
684 '0.01',
685 '0.02',
686 '0.03',
687 '0.04',
688 '0.05',
689 '1.23456789',
690 '10.0',
691 '100.0',
692 # values >= 1e16 get an exponent...
693 '1000000000000000.0',
694 '9999999999999990.0',
695 '1e+16',
696 '1e+17',
697 # ... and so do values < 1e-4
698 '0.001',
699 '0.001001',
700 '0.00010000000000001',
701 '0.0001',
702 '9.999999999999e-05',
703 '1e-05',
704 # values designed to provoke failure if the FPU rounding
705 # precision isn't set correctly
706 '8.72293771110361e+25',
707 '7.47005307342313e+26',
708 '2.86438000439698e+28',
709 '8.89142905246179e+28',
710 '3.08578087079232e+35',
711 ]
712
713 for s in test_strings:
714 negs = '-'+s
715 self.assertEqual(s, repr(float(s)))
716 self.assertEqual(negs, repr(float(negs)))
Mark Dickinson388122d2010-08-04 20:56:28 +0000717 # Since Python 3.2, repr and str are identical
718 self.assertEqual(repr(float(s)), str(float(s)))
719 self.assertEqual(repr(float(negs)), str(float(negs)))
Benjamin Petersone401c682010-07-02 23:25:44 +0000720
Eric Smith3ab08ca2010-12-04 15:17:38 +0000721@support.requires_IEEE_754
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000722class RoundTestCase(unittest.TestCase):
Benjamin Petersone401c682010-07-02 23:25:44 +0000723
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000724 def test_inf_nan(self):
725 self.assertRaises(OverflowError, round, INF)
726 self.assertRaises(OverflowError, round, -INF)
727 self.assertRaises(ValueError, round, NAN)
Mark Dickinson4ca33d12009-11-24 10:59:34 +0000728 self.assertRaises(TypeError, round, INF, 0.0)
729 self.assertRaises(TypeError, round, -INF, 1.0)
730 self.assertRaises(TypeError, round, NAN, "ceci n'est pas un integer")
731 self.assertRaises(TypeError, round, -0.0, 1j)
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000732
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000733 def test_large_n(self):
734 for n in [324, 325, 400, 2**31-1, 2**31, 2**32, 2**100]:
735 self.assertEqual(round(123.456, n), 123.456)
736 self.assertEqual(round(-123.456, n), -123.456)
737 self.assertEqual(round(1e300, n), 1e300)
738 self.assertEqual(round(1e-320, n), 1e-320)
739 self.assertEqual(round(1e150, 300), 1e150)
740 self.assertEqual(round(1e300, 307), 1e300)
741 self.assertEqual(round(-3.1415, 308), -3.1415)
742 self.assertEqual(round(1e150, 309), 1e150)
743 self.assertEqual(round(1.4e-315, 315), 1e-315)
744
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000745 def test_small_n(self):
746 for n in [-308, -309, -400, 1-2**31, -2**31, -2**31-1, -2**100]:
747 self.assertEqual(round(123.456, n), 0.0)
748 self.assertEqual(round(-123.456, n), -0.0)
749 self.assertEqual(round(1e300, n), 0.0)
750 self.assertEqual(round(1e-320, n), 0.0)
751
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000752 def test_overflow(self):
753 self.assertRaises(OverflowError, round, 1.6e308, -308)
754 self.assertRaises(OverflowError, round, -1.7e308, -308)
755
756 @unittest.skipUnless(getattr(sys, 'float_repr_style', '') == 'short',
757 "applies only when using short float repr style")
758 def test_previous_round_bugs(self):
759 # particular cases that have occurred in bug reports
760 self.assertEqual(round(562949953421312.5, 1),
761 562949953421312.5)
762 self.assertEqual(round(56294995342131.5, 3),
763 56294995342131.5)
764 # round-half-even
765 self.assertEqual(round(25.0, -1), 20.0)
766 self.assertEqual(round(35.0, -1), 40.0)
767 self.assertEqual(round(45.0, -1), 40.0)
768 self.assertEqual(round(55.0, -1), 60.0)
769 self.assertEqual(round(65.0, -1), 60.0)
770 self.assertEqual(round(75.0, -1), 80.0)
771 self.assertEqual(round(85.0, -1), 80.0)
772 self.assertEqual(round(95.0, -1), 100.0)
773
774 @unittest.skipUnless(getattr(sys, 'float_repr_style', '') == 'short',
775 "applies only when using short float repr style")
776 def test_matches_float_format(self):
777 # round should give the same results as float formatting
778 for i in range(500):
779 x = i/1000.
780 self.assertEqual(float(format(x, '.0f')), round(x, 0))
781 self.assertEqual(float(format(x, '.1f')), round(x, 1))
782 self.assertEqual(float(format(x, '.2f')), round(x, 2))
783 self.assertEqual(float(format(x, '.3f')), round(x, 3))
784
785 for i in range(5, 5000, 10):
786 x = i/1000.
787 self.assertEqual(float(format(x, '.0f')), round(x, 0))
788 self.assertEqual(float(format(x, '.1f')), round(x, 1))
789 self.assertEqual(float(format(x, '.2f')), round(x, 2))
790 self.assertEqual(float(format(x, '.3f')), round(x, 3))
791
792 for i in range(500):
793 x = random.random()
794 self.assertEqual(float(format(x, '.0f')), round(x, 0))
795 self.assertEqual(float(format(x, '.1f')), round(x, 1))
796 self.assertEqual(float(format(x, '.2f')), round(x, 2))
797 self.assertEqual(float(format(x, '.3f')), round(x, 3))
798
Eric Smith8a10ecc2009-12-02 17:58:24 +0000799 def test_format_specials(self):
800 # Test formatting of nans and infs.
801
802 def test(fmt, value, expected):
803 # Test with both % and format().
804 self.assertEqual(fmt % value, expected, fmt)
Eric Smith984bb582010-11-25 16:08:06 +0000805 fmt = fmt[1:] # strip off the %
806 self.assertEqual(format(value, fmt), expected, fmt)
Eric Smith8a10ecc2009-12-02 17:58:24 +0000807
808 for fmt in ['%e', '%f', '%g', '%.0e', '%.6f', '%.20g',
809 '%#e', '%#f', '%#g', '%#.20e', '%#.15f', '%#.3g']:
810 pfmt = '%+' + fmt[1:]
811 sfmt = '% ' + fmt[1:]
812 test(fmt, INF, 'inf')
813 test(fmt, -INF, '-inf')
814 test(fmt, NAN, 'nan')
815 test(fmt, -NAN, 'nan')
816 # When asking for a sign, it's always provided. nans are
817 # always positive.
818 test(pfmt, INF, '+inf')
819 test(pfmt, -INF, '-inf')
820 test(pfmt, NAN, '+nan')
821 test(pfmt, -NAN, '+nan')
822 # When using ' ' for a sign code, only infs can be negative.
823 # Others have a space.
824 test(sfmt, INF, ' inf')
825 test(sfmt, -INF, '-inf')
826 test(sfmt, NAN, ' nan')
827 test(sfmt, -NAN, ' nan')
828
Steve Dowercb39d1f2015-04-15 16:10:59 -0400829 def test_None_ndigits(self):
830 for x in round(1.23), round(1.23, None), round(1.23, ndigits=None):
831 self.assertEqual(x, 1)
832 self.assertIsInstance(x, int)
833 for x in round(1.78), round(1.78, None), round(1.78, ndigits=None):
834 self.assertEqual(x, 2)
835 self.assertIsInstance(x, int)
836
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000837
Christian Heimes99170a52007-12-19 02:07:34 +0000838# Beginning with Python 2.6 float has cross platform compatible
Georg Brandl2ee470f2008-07-16 12:55:28 +0000839# ways to create and represent inf and nan
Christian Heimes99170a52007-12-19 02:07:34 +0000840class InfNanTest(unittest.TestCase):
841 def test_inf_from_str(self):
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000842 self.assertTrue(isinf(float("inf")))
843 self.assertTrue(isinf(float("+inf")))
844 self.assertTrue(isinf(float("-inf")))
845 self.assertTrue(isinf(float("infinity")))
846 self.assertTrue(isinf(float("+infinity")))
847 self.assertTrue(isinf(float("-infinity")))
Christian Heimes99170a52007-12-19 02:07:34 +0000848
849 self.assertEqual(repr(float("inf")), "inf")
850 self.assertEqual(repr(float("+inf")), "inf")
851 self.assertEqual(repr(float("-inf")), "-inf")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000852 self.assertEqual(repr(float("infinity")), "inf")
853 self.assertEqual(repr(float("+infinity")), "inf")
854 self.assertEqual(repr(float("-infinity")), "-inf")
Christian Heimes99170a52007-12-19 02:07:34 +0000855
856 self.assertEqual(repr(float("INF")), "inf")
857 self.assertEqual(repr(float("+Inf")), "inf")
858 self.assertEqual(repr(float("-iNF")), "-inf")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000859 self.assertEqual(repr(float("Infinity")), "inf")
860 self.assertEqual(repr(float("+iNfInItY")), "inf")
861 self.assertEqual(repr(float("-INFINITY")), "-inf")
Christian Heimes99170a52007-12-19 02:07:34 +0000862
863 self.assertEqual(str(float("inf")), "inf")
864 self.assertEqual(str(float("+inf")), "inf")
865 self.assertEqual(str(float("-inf")), "-inf")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000866 self.assertEqual(str(float("infinity")), "inf")
867 self.assertEqual(str(float("+infinity")), "inf")
868 self.assertEqual(str(float("-infinity")), "-inf")
Christian Heimes99170a52007-12-19 02:07:34 +0000869
870 self.assertRaises(ValueError, float, "info")
871 self.assertRaises(ValueError, float, "+info")
872 self.assertRaises(ValueError, float, "-info")
873 self.assertRaises(ValueError, float, "in")
874 self.assertRaises(ValueError, float, "+in")
875 self.assertRaises(ValueError, float, "-in")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000876 self.assertRaises(ValueError, float, "infinit")
877 self.assertRaises(ValueError, float, "+Infin")
878 self.assertRaises(ValueError, float, "-INFI")
879 self.assertRaises(ValueError, float, "infinitys")
Christian Heimes99170a52007-12-19 02:07:34 +0000880
Mark Dickinsonbd16edd2009-05-20 22:05:25 +0000881 self.assertRaises(ValueError, float, "++Inf")
882 self.assertRaises(ValueError, float, "-+inf")
883 self.assertRaises(ValueError, float, "+-infinity")
884 self.assertRaises(ValueError, float, "--Infinity")
885
Christian Heimes99170a52007-12-19 02:07:34 +0000886 def test_inf_as_str(self):
887 self.assertEqual(repr(1e300 * 1e300), "inf")
888 self.assertEqual(repr(-1e300 * 1e300), "-inf")
889
890 self.assertEqual(str(1e300 * 1e300), "inf")
891 self.assertEqual(str(-1e300 * 1e300), "-inf")
892
893 def test_nan_from_str(self):
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000894 self.assertTrue(isnan(float("nan")))
895 self.assertTrue(isnan(float("+nan")))
896 self.assertTrue(isnan(float("-nan")))
Christian Heimes99170a52007-12-19 02:07:34 +0000897
898 self.assertEqual(repr(float("nan")), "nan")
899 self.assertEqual(repr(float("+nan")), "nan")
900 self.assertEqual(repr(float("-nan")), "nan")
901
902 self.assertEqual(repr(float("NAN")), "nan")
903 self.assertEqual(repr(float("+NAn")), "nan")
904 self.assertEqual(repr(float("-NaN")), "nan")
905
906 self.assertEqual(str(float("nan")), "nan")
907 self.assertEqual(str(float("+nan")), "nan")
908 self.assertEqual(str(float("-nan")), "nan")
909
910 self.assertRaises(ValueError, float, "nana")
911 self.assertRaises(ValueError, float, "+nana")
912 self.assertRaises(ValueError, float, "-nana")
913 self.assertRaises(ValueError, float, "na")
914 self.assertRaises(ValueError, float, "+na")
915 self.assertRaises(ValueError, float, "-na")
916
Mark Dickinsonbd16edd2009-05-20 22:05:25 +0000917 self.assertRaises(ValueError, float, "++nan")
918 self.assertRaises(ValueError, float, "-+NAN")
919 self.assertRaises(ValueError, float, "+-NaN")
920 self.assertRaises(ValueError, float, "--nAn")
921
Christian Heimes99170a52007-12-19 02:07:34 +0000922 def test_nan_as_str(self):
923 self.assertEqual(repr(1e300 * 1e300 * 0), "nan")
924 self.assertEqual(repr(-1e300 * 1e300 * 0), "nan")
925
926 self.assertEqual(str(1e300 * 1e300 * 0), "nan")
927 self.assertEqual(str(-1e300 * 1e300 * 0), "nan")
Christian Heimes827b35c2007-12-10 22:19:17 +0000928
Mark Dickinsone383e822012-04-29 15:31:56 +0100929 def test_inf_signs(self):
930 self.assertEqual(copysign(1.0, float('inf')), 1.0)
931 self.assertEqual(copysign(1.0, float('-inf')), -1.0)
932
933 @unittest.skipUnless(getattr(sys, 'float_repr_style', '') == 'short',
934 "applies only when using short float repr style")
935 def test_nan_signs(self):
936 # When using the dtoa.c code, the sign of float('nan') should
937 # be predictable.
938 self.assertEqual(copysign(1.0, float('nan')), 1.0)
939 self.assertEqual(copysign(1.0, float('-nan')), -1.0)
940
941
Mark Dickinson65fe25e2008-07-16 11:30:51 +0000942fromHex = float.fromhex
943toHex = float.hex
944class HexFloatTestCase(unittest.TestCase):
945 MAX = fromHex('0x.fffffffffffff8p+1024') # max normal
946 MIN = fromHex('0x1p-1022') # min normal
947 TINY = fromHex('0x0.0000000000001p-1022') # min subnormal
948 EPS = fromHex('0x0.0000000000001p0') # diff between 1.0 and next float up
949
950 def identical(self, x, y):
951 # check that floats x and y are identical, or that both
952 # are NaNs
953 if isnan(x) or isnan(y):
954 if isnan(x) == isnan(y):
955 return
956 elif x == y and (x != 0.0 or copysign(1.0, x) == copysign(1.0, y)):
957 return
958 self.fail('%r not identical to %r' % (x, y))
959
960 def test_ends(self):
Mark Dickinson38bbc482008-07-16 11:32:23 +0000961 self.identical(self.MIN, ldexp(1.0, -1022))
962 self.identical(self.TINY, ldexp(1.0, -1074))
963 self.identical(self.EPS, ldexp(1.0, -52))
964 self.identical(self.MAX, 2.*(ldexp(1.0, 1023) - ldexp(1.0, 970)))
Mark Dickinson65fe25e2008-07-16 11:30:51 +0000965
966 def test_invalid_inputs(self):
967 invalid_inputs = [
968 'infi', # misspelt infinities and nans
969 '-Infinit',
970 '++inf',
971 '-+Inf',
972 '--nan',
973 '+-NaN',
974 'snan',
975 'NaNs',
976 'nna',
Mark Dickinsond1ec8b22009-05-11 15:45:15 +0000977 'an',
978 'nf',
979 'nfinity',
980 'inity',
981 'iinity',
Mark Dickinson65fe25e2008-07-16 11:30:51 +0000982 '0xnan',
983 '',
984 ' ',
985 'x1.0p0',
986 '0xX1.0p0',
987 '+ 0x1.0p0', # internal whitespace
988 '- 0x1.0p0',
989 '0 x1.0p0',
990 '0x 1.0p0',
991 '0x1 2.0p0',
992 '+0x1 .0p0',
993 '0x1. 0p0',
994 '-0x1.0 1p0',
995 '-0x1.0 p0',
996 '+0x1.0p +0',
997 '0x1.0p -0',
998 '0x1.0p 0',
999 '+0x1.0p+ 0',
1000 '-0x1.0p- 0',
1001 '++0x1.0p-0', # double signs
1002 '--0x1.0p0',
1003 '+-0x1.0p+0',
1004 '-+0x1.0p0',
1005 '0x1.0p++0',
1006 '+0x1.0p+-0',
1007 '-0x1.0p-+0',
1008 '0x1.0p--0',
1009 '0x1.0.p0',
1010 '0x.p0', # no hex digits before or after point
1011 '0x1,p0', # wrong decimal point character
1012 '0x1pa',
1013 '0x1p\uff10', # fullwidth Unicode digits
1014 '\uff10x1p0',
1015 '0x\uff11p0',
1016 '0x1.\uff10p0',
1017 '0x1p0 \n 0x2p0',
1018 '0x1p0\0 0x1p0', # embedded null byte is not end of string
1019 ]
1020 for x in invalid_inputs:
Mark Dickinson589b7952008-08-21 20:05:56 +00001021 try:
1022 result = fromHex(x)
1023 except ValueError:
1024 pass
1025 else:
1026 self.fail('Expected float.fromhex(%r) to raise ValueError; '
1027 'got %r instead' % (x, result))
Mark Dickinson65fe25e2008-07-16 11:30:51 +00001028
1029
Mark Dickinsond1ec8b22009-05-11 15:45:15 +00001030 def test_whitespace(self):
1031 value_pairs = [
1032 ('inf', INF),
1033 ('-Infinity', -INF),
1034 ('nan', NAN),
1035 ('1.0', 1.0),
1036 ('-0x.2', -0.125),
1037 ('-0.0', -0.0)
1038 ]
1039 whitespace = [
1040 '',
1041 ' ',
1042 '\t',
1043 '\n',
1044 '\n \t',
1045 '\f',
1046 '\v',
1047 '\r'
1048 ]
1049 for inp, expected in value_pairs:
1050 for lead in whitespace:
1051 for trail in whitespace:
1052 got = fromHex(lead + inp + trail)
1053 self.identical(got, expected)
1054
1055
Mark Dickinson65fe25e2008-07-16 11:30:51 +00001056 def test_from_hex(self):
1057 MIN = self.MIN;
1058 MAX = self.MAX;
1059 TINY = self.TINY;
1060 EPS = self.EPS;
1061
1062 # two spellings of infinity, with optional signs; case-insensitive
1063 self.identical(fromHex('inf'), INF)
1064 self.identical(fromHex('+Inf'), INF)
1065 self.identical(fromHex('-INF'), -INF)
1066 self.identical(fromHex('iNf'), INF)
1067 self.identical(fromHex('Infinity'), INF)
1068 self.identical(fromHex('+INFINITY'), INF)
1069 self.identical(fromHex('-infinity'), -INF)
1070 self.identical(fromHex('-iNFiNitY'), -INF)
1071
1072 # nans with optional sign; case insensitive
1073 self.identical(fromHex('nan'), NAN)
1074 self.identical(fromHex('+NaN'), NAN)
1075 self.identical(fromHex('-NaN'), NAN)
1076 self.identical(fromHex('-nAN'), NAN)
1077
1078 # variations in input format
1079 self.identical(fromHex('1'), 1.0)
1080 self.identical(fromHex('+1'), 1.0)
1081 self.identical(fromHex('1.'), 1.0)
1082 self.identical(fromHex('1.0'), 1.0)
1083 self.identical(fromHex('1.0p0'), 1.0)
1084 self.identical(fromHex('01'), 1.0)
1085 self.identical(fromHex('01.'), 1.0)
1086 self.identical(fromHex('0x1'), 1.0)
1087 self.identical(fromHex('0x1.'), 1.0)
1088 self.identical(fromHex('0x1.0'), 1.0)
1089 self.identical(fromHex('+0x1.0'), 1.0)
1090 self.identical(fromHex('0x1p0'), 1.0)
1091 self.identical(fromHex('0X1p0'), 1.0)
1092 self.identical(fromHex('0X1P0'), 1.0)
1093 self.identical(fromHex('0x1P0'), 1.0)
1094 self.identical(fromHex('0x1.p0'), 1.0)
1095 self.identical(fromHex('0x1.0p0'), 1.0)
1096 self.identical(fromHex('0x.1p4'), 1.0)
1097 self.identical(fromHex('0x.1p04'), 1.0)
1098 self.identical(fromHex('0x.1p004'), 1.0)
1099 self.identical(fromHex('0x1p+0'), 1.0)
1100 self.identical(fromHex('0x1P-0'), 1.0)
1101 self.identical(fromHex('+0x1p0'), 1.0)
1102 self.identical(fromHex('0x01p0'), 1.0)
1103 self.identical(fromHex('0x1p00'), 1.0)
1104 self.identical(fromHex(' 0x1p0 '), 1.0)
1105 self.identical(fromHex('\n 0x1p0'), 1.0)
1106 self.identical(fromHex('0x1p0 \t'), 1.0)
1107 self.identical(fromHex('0xap0'), 10.0)
1108 self.identical(fromHex('0xAp0'), 10.0)
1109 self.identical(fromHex('0xaP0'), 10.0)
1110 self.identical(fromHex('0xAP0'), 10.0)
1111 self.identical(fromHex('0xbep0'), 190.0)
1112 self.identical(fromHex('0xBep0'), 190.0)
1113 self.identical(fromHex('0xbEp0'), 190.0)
1114 self.identical(fromHex('0XBE0P-4'), 190.0)
1115 self.identical(fromHex('0xBEp0'), 190.0)
1116 self.identical(fromHex('0xB.Ep4'), 190.0)
1117 self.identical(fromHex('0x.BEp8'), 190.0)
1118 self.identical(fromHex('0x.0BEp12'), 190.0)
1119
1120 # moving the point around
1121 pi = fromHex('0x1.921fb54442d18p1')
1122 self.identical(fromHex('0x.006487ed5110b46p11'), pi)
1123 self.identical(fromHex('0x.00c90fdaa22168cp10'), pi)
1124 self.identical(fromHex('0x.01921fb54442d18p9'), pi)
1125 self.identical(fromHex('0x.03243f6a8885a3p8'), pi)
1126 self.identical(fromHex('0x.06487ed5110b46p7'), pi)
1127 self.identical(fromHex('0x.0c90fdaa22168cp6'), pi)
1128 self.identical(fromHex('0x.1921fb54442d18p5'), pi)
1129 self.identical(fromHex('0x.3243f6a8885a3p4'), pi)
1130 self.identical(fromHex('0x.6487ed5110b46p3'), pi)
1131 self.identical(fromHex('0x.c90fdaa22168cp2'), pi)
1132 self.identical(fromHex('0x1.921fb54442d18p1'), pi)
1133 self.identical(fromHex('0x3.243f6a8885a3p0'), pi)
1134 self.identical(fromHex('0x6.487ed5110b46p-1'), pi)
1135 self.identical(fromHex('0xc.90fdaa22168cp-2'), pi)
1136 self.identical(fromHex('0x19.21fb54442d18p-3'), pi)
1137 self.identical(fromHex('0x32.43f6a8885a3p-4'), pi)
1138 self.identical(fromHex('0x64.87ed5110b46p-5'), pi)
1139 self.identical(fromHex('0xc9.0fdaa22168cp-6'), pi)
1140 self.identical(fromHex('0x192.1fb54442d18p-7'), pi)
1141 self.identical(fromHex('0x324.3f6a8885a3p-8'), pi)
1142 self.identical(fromHex('0x648.7ed5110b46p-9'), pi)
1143 self.identical(fromHex('0xc90.fdaa22168cp-10'), pi)
1144 self.identical(fromHex('0x1921.fb54442d18p-11'), pi)
1145 # ...
1146 self.identical(fromHex('0x1921fb54442d1.8p-47'), pi)
1147 self.identical(fromHex('0x3243f6a8885a3p-48'), pi)
1148 self.identical(fromHex('0x6487ed5110b46p-49'), pi)
1149 self.identical(fromHex('0xc90fdaa22168cp-50'), pi)
1150 self.identical(fromHex('0x1921fb54442d18p-51'), pi)
1151 self.identical(fromHex('0x3243f6a8885a30p-52'), pi)
1152 self.identical(fromHex('0x6487ed5110b460p-53'), pi)
1153 self.identical(fromHex('0xc90fdaa22168c0p-54'), pi)
1154 self.identical(fromHex('0x1921fb54442d180p-55'), pi)
1155
1156
1157 # results that should overflow...
1158 self.assertRaises(OverflowError, fromHex, '-0x1p1024')
1159 self.assertRaises(OverflowError, fromHex, '0x1p+1025')
1160 self.assertRaises(OverflowError, fromHex, '+0X1p1030')
1161 self.assertRaises(OverflowError, fromHex, '-0x1p+1100')
1162 self.assertRaises(OverflowError, fromHex, '0X1p123456789123456789')
1163 self.assertRaises(OverflowError, fromHex, '+0X.8p+1025')
1164 self.assertRaises(OverflowError, fromHex, '+0x0.8p1025')
1165 self.assertRaises(OverflowError, fromHex, '-0x0.4p1026')
1166 self.assertRaises(OverflowError, fromHex, '0X2p+1023')
1167 self.assertRaises(OverflowError, fromHex, '0x2.p1023')
1168 self.assertRaises(OverflowError, fromHex, '-0x2.0p+1023')
1169 self.assertRaises(OverflowError, fromHex, '+0X4p+1022')
1170 self.assertRaises(OverflowError, fromHex, '0x1.ffffffffffffffp+1023')
1171 self.assertRaises(OverflowError, fromHex, '-0X1.fffffffffffff9p1023')
1172 self.assertRaises(OverflowError, fromHex, '0X1.fffffffffffff8p1023')
1173 self.assertRaises(OverflowError, fromHex, '+0x3.fffffffffffffp1022')
1174 self.assertRaises(OverflowError, fromHex, '0x3fffffffffffffp+970')
1175 self.assertRaises(OverflowError, fromHex, '0x10000000000000000p960')
1176 self.assertRaises(OverflowError, fromHex, '-0Xffffffffffffffffp960')
1177
1178 # ...and those that round to +-max float
1179 self.identical(fromHex('+0x1.fffffffffffffp+1023'), MAX)
1180 self.identical(fromHex('-0X1.fffffffffffff7p1023'), -MAX)
1181 self.identical(fromHex('0X1.fffffffffffff7fffffffffffffp1023'), MAX)
1182
1183 # zeros
1184 self.identical(fromHex('0x0p0'), 0.0)
1185 self.identical(fromHex('0x0p1000'), 0.0)
1186 self.identical(fromHex('-0x0p1023'), -0.0)
1187 self.identical(fromHex('0X0p1024'), 0.0)
1188 self.identical(fromHex('-0x0p1025'), -0.0)
1189 self.identical(fromHex('0X0p2000'), 0.0)
1190 self.identical(fromHex('0x0p123456789123456789'), 0.0)
1191 self.identical(fromHex('-0X0p-0'), -0.0)
1192 self.identical(fromHex('-0X0p-1000'), -0.0)
1193 self.identical(fromHex('0x0p-1023'), 0.0)
1194 self.identical(fromHex('-0X0p-1024'), -0.0)
1195 self.identical(fromHex('-0x0p-1025'), -0.0)
1196 self.identical(fromHex('-0x0p-1072'), -0.0)
1197 self.identical(fromHex('0X0p-1073'), 0.0)
1198 self.identical(fromHex('-0x0p-1074'), -0.0)
1199 self.identical(fromHex('0x0p-1075'), 0.0)
1200 self.identical(fromHex('0X0p-1076'), 0.0)
1201 self.identical(fromHex('-0X0p-2000'), -0.0)
1202 self.identical(fromHex('-0x0p-123456789123456789'), -0.0)
1203
1204 # values that should underflow to 0
1205 self.identical(fromHex('0X1p-1075'), 0.0)
1206 self.identical(fromHex('-0X1p-1075'), -0.0)
1207 self.identical(fromHex('-0x1p-123456789123456789'), -0.0)
1208 self.identical(fromHex('0x1.00000000000000001p-1075'), TINY)
1209 self.identical(fromHex('-0x1.1p-1075'), -TINY)
1210 self.identical(fromHex('0x1.fffffffffffffffffp-1075'), TINY)
1211
1212 # check round-half-even is working correctly near 0 ...
1213 self.identical(fromHex('0x1p-1076'), 0.0)
1214 self.identical(fromHex('0X2p-1076'), 0.0)
1215 self.identical(fromHex('0X3p-1076'), TINY)
1216 self.identical(fromHex('0x4p-1076'), TINY)
1217 self.identical(fromHex('0X5p-1076'), TINY)
1218 self.identical(fromHex('0X6p-1076'), 2*TINY)
1219 self.identical(fromHex('0x7p-1076'), 2*TINY)
1220 self.identical(fromHex('0X8p-1076'), 2*TINY)
1221 self.identical(fromHex('0X9p-1076'), 2*TINY)
1222 self.identical(fromHex('0xap-1076'), 2*TINY)
1223 self.identical(fromHex('0Xbp-1076'), 3*TINY)
1224 self.identical(fromHex('0xcp-1076'), 3*TINY)
1225 self.identical(fromHex('0Xdp-1076'), 3*TINY)
1226 self.identical(fromHex('0Xep-1076'), 4*TINY)
1227 self.identical(fromHex('0xfp-1076'), 4*TINY)
1228 self.identical(fromHex('0x10p-1076'), 4*TINY)
1229 self.identical(fromHex('-0x1p-1076'), -0.0)
1230 self.identical(fromHex('-0X2p-1076'), -0.0)
1231 self.identical(fromHex('-0x3p-1076'), -TINY)
1232 self.identical(fromHex('-0X4p-1076'), -TINY)
1233 self.identical(fromHex('-0x5p-1076'), -TINY)
1234 self.identical(fromHex('-0x6p-1076'), -2*TINY)
1235 self.identical(fromHex('-0X7p-1076'), -2*TINY)
1236 self.identical(fromHex('-0X8p-1076'), -2*TINY)
1237 self.identical(fromHex('-0X9p-1076'), -2*TINY)
1238 self.identical(fromHex('-0Xap-1076'), -2*TINY)
1239 self.identical(fromHex('-0xbp-1076'), -3*TINY)
1240 self.identical(fromHex('-0xcp-1076'), -3*TINY)
1241 self.identical(fromHex('-0Xdp-1076'), -3*TINY)
1242 self.identical(fromHex('-0xep-1076'), -4*TINY)
1243 self.identical(fromHex('-0Xfp-1076'), -4*TINY)
1244 self.identical(fromHex('-0X10p-1076'), -4*TINY)
1245
1246 # ... and near MIN ...
1247 self.identical(fromHex('0x0.ffffffffffffd6p-1022'), MIN-3*TINY)
1248 self.identical(fromHex('0x0.ffffffffffffd8p-1022'), MIN-2*TINY)
1249 self.identical(fromHex('0x0.ffffffffffffdap-1022'), MIN-2*TINY)
1250 self.identical(fromHex('0x0.ffffffffffffdcp-1022'), MIN-2*TINY)
1251 self.identical(fromHex('0x0.ffffffffffffdep-1022'), MIN-2*TINY)
1252 self.identical(fromHex('0x0.ffffffffffffe0p-1022'), MIN-2*TINY)
1253 self.identical(fromHex('0x0.ffffffffffffe2p-1022'), MIN-2*TINY)
1254 self.identical(fromHex('0x0.ffffffffffffe4p-1022'), MIN-2*TINY)
1255 self.identical(fromHex('0x0.ffffffffffffe6p-1022'), MIN-2*TINY)
1256 self.identical(fromHex('0x0.ffffffffffffe8p-1022'), MIN-2*TINY)
1257 self.identical(fromHex('0x0.ffffffffffffeap-1022'), MIN-TINY)
1258 self.identical(fromHex('0x0.ffffffffffffecp-1022'), MIN-TINY)
1259 self.identical(fromHex('0x0.ffffffffffffeep-1022'), MIN-TINY)
1260 self.identical(fromHex('0x0.fffffffffffff0p-1022'), MIN-TINY)
1261 self.identical(fromHex('0x0.fffffffffffff2p-1022'), MIN-TINY)
1262 self.identical(fromHex('0x0.fffffffffffff4p-1022'), MIN-TINY)
1263 self.identical(fromHex('0x0.fffffffffffff6p-1022'), MIN-TINY)
1264 self.identical(fromHex('0x0.fffffffffffff8p-1022'), MIN)
1265 self.identical(fromHex('0x0.fffffffffffffap-1022'), MIN)
1266 self.identical(fromHex('0x0.fffffffffffffcp-1022'), MIN)
1267 self.identical(fromHex('0x0.fffffffffffffep-1022'), MIN)
1268 self.identical(fromHex('0x1.00000000000000p-1022'), MIN)
1269 self.identical(fromHex('0x1.00000000000002p-1022'), MIN)
1270 self.identical(fromHex('0x1.00000000000004p-1022'), MIN)
1271 self.identical(fromHex('0x1.00000000000006p-1022'), MIN)
1272 self.identical(fromHex('0x1.00000000000008p-1022'), MIN)
1273 self.identical(fromHex('0x1.0000000000000ap-1022'), MIN+TINY)
1274 self.identical(fromHex('0x1.0000000000000cp-1022'), MIN+TINY)
1275 self.identical(fromHex('0x1.0000000000000ep-1022'), MIN+TINY)
1276 self.identical(fromHex('0x1.00000000000010p-1022'), MIN+TINY)
1277 self.identical(fromHex('0x1.00000000000012p-1022'), MIN+TINY)
1278 self.identical(fromHex('0x1.00000000000014p-1022'), MIN+TINY)
1279 self.identical(fromHex('0x1.00000000000016p-1022'), MIN+TINY)
1280 self.identical(fromHex('0x1.00000000000018p-1022'), MIN+2*TINY)
1281
1282 # ... and near 1.0.
1283 self.identical(fromHex('0x0.fffffffffffff0p0'), 1.0-EPS)
1284 self.identical(fromHex('0x0.fffffffffffff1p0'), 1.0-EPS)
1285 self.identical(fromHex('0X0.fffffffffffff2p0'), 1.0-EPS)
1286 self.identical(fromHex('0x0.fffffffffffff3p0'), 1.0-EPS)
1287 self.identical(fromHex('0X0.fffffffffffff4p0'), 1.0-EPS)
1288 self.identical(fromHex('0X0.fffffffffffff5p0'), 1.0-EPS/2)
1289 self.identical(fromHex('0X0.fffffffffffff6p0'), 1.0-EPS/2)
1290 self.identical(fromHex('0x0.fffffffffffff7p0'), 1.0-EPS/2)
1291 self.identical(fromHex('0x0.fffffffffffff8p0'), 1.0-EPS/2)
1292 self.identical(fromHex('0X0.fffffffffffff9p0'), 1.0-EPS/2)
1293 self.identical(fromHex('0X0.fffffffffffffap0'), 1.0-EPS/2)
1294 self.identical(fromHex('0x0.fffffffffffffbp0'), 1.0-EPS/2)
1295 self.identical(fromHex('0X0.fffffffffffffcp0'), 1.0)
1296 self.identical(fromHex('0x0.fffffffffffffdp0'), 1.0)
1297 self.identical(fromHex('0X0.fffffffffffffep0'), 1.0)
1298 self.identical(fromHex('0x0.ffffffffffffffp0'), 1.0)
1299 self.identical(fromHex('0X1.00000000000000p0'), 1.0)
1300 self.identical(fromHex('0X1.00000000000001p0'), 1.0)
1301 self.identical(fromHex('0x1.00000000000002p0'), 1.0)
1302 self.identical(fromHex('0X1.00000000000003p0'), 1.0)
1303 self.identical(fromHex('0x1.00000000000004p0'), 1.0)
1304 self.identical(fromHex('0X1.00000000000005p0'), 1.0)
1305 self.identical(fromHex('0X1.00000000000006p0'), 1.0)
1306 self.identical(fromHex('0X1.00000000000007p0'), 1.0)
1307 self.identical(fromHex('0x1.00000000000007ffffffffffffffffffffp0'),
1308 1.0)
1309 self.identical(fromHex('0x1.00000000000008p0'), 1.0)
1310 self.identical(fromHex('0x1.00000000000008000000000000000001p0'),
1311 1+EPS)
1312 self.identical(fromHex('0X1.00000000000009p0'), 1.0+EPS)
1313 self.identical(fromHex('0x1.0000000000000ap0'), 1.0+EPS)
1314 self.identical(fromHex('0x1.0000000000000bp0'), 1.0+EPS)
1315 self.identical(fromHex('0X1.0000000000000cp0'), 1.0+EPS)
1316 self.identical(fromHex('0x1.0000000000000dp0'), 1.0+EPS)
1317 self.identical(fromHex('0x1.0000000000000ep0'), 1.0+EPS)
1318 self.identical(fromHex('0X1.0000000000000fp0'), 1.0+EPS)
1319 self.identical(fromHex('0x1.00000000000010p0'), 1.0+EPS)
1320 self.identical(fromHex('0X1.00000000000011p0'), 1.0+EPS)
1321 self.identical(fromHex('0x1.00000000000012p0'), 1.0+EPS)
1322 self.identical(fromHex('0X1.00000000000013p0'), 1.0+EPS)
1323 self.identical(fromHex('0X1.00000000000014p0'), 1.0+EPS)
1324 self.identical(fromHex('0x1.00000000000015p0'), 1.0+EPS)
1325 self.identical(fromHex('0x1.00000000000016p0'), 1.0+EPS)
1326 self.identical(fromHex('0X1.00000000000017p0'), 1.0+EPS)
1327 self.identical(fromHex('0x1.00000000000017ffffffffffffffffffffp0'),
1328 1.0+EPS)
1329 self.identical(fromHex('0x1.00000000000018p0'), 1.0+2*EPS)
1330 self.identical(fromHex('0X1.00000000000018000000000000000001p0'),
1331 1.0+2*EPS)
1332 self.identical(fromHex('0x1.00000000000019p0'), 1.0+2*EPS)
1333 self.identical(fromHex('0X1.0000000000001ap0'), 1.0+2*EPS)
1334 self.identical(fromHex('0X1.0000000000001bp0'), 1.0+2*EPS)
1335 self.identical(fromHex('0x1.0000000000001cp0'), 1.0+2*EPS)
1336 self.identical(fromHex('0x1.0000000000001dp0'), 1.0+2*EPS)
1337 self.identical(fromHex('0x1.0000000000001ep0'), 1.0+2*EPS)
1338 self.identical(fromHex('0X1.0000000000001fp0'), 1.0+2*EPS)
1339 self.identical(fromHex('0x1.00000000000020p0'), 1.0+2*EPS)
1340
1341 def test_roundtrip(self):
1342 def roundtrip(x):
1343 return fromHex(toHex(x))
1344
1345 for x in [NAN, INF, self.MAX, self.MIN, self.MIN-self.TINY, self.TINY, 0.0]:
1346 self.identical(x, roundtrip(x))
1347 self.identical(-x, roundtrip(-x))
1348
1349 # fromHex(toHex(x)) should exactly recover x, for any non-NaN float x.
1350 import random
1351 for i in range(10000):
1352 e = random.randrange(-1200, 1200)
1353 m = random.random()
1354 s = random.choice([1.0, -1.0])
1355 try:
1356 x = s*ldexp(m, e)
1357 except OverflowError:
1358 pass
1359 else:
1360 self.identical(x, fromHex(toHex(x)))
1361
Serhiy Storchakaea36c942016-05-12 10:37:58 +03001362 def test_subclass(self):
1363 class F(float):
1364 def __new__(cls, value):
1365 return float.__new__(cls, value + 1)
1366
1367 f = F.fromhex((1.5).hex())
1368 self.assertIs(type(f), F)
1369 self.assertEqual(f, 2.5)
1370
1371 class F2(float):
1372 def __init__(self, value):
1373 self.foo = 'bar'
1374
1375 f = F2.fromhex((1.5).hex())
1376 self.assertIs(type(f), F2)
1377 self.assertEqual(f, 1.5)
1378 self.assertEqual(getattr(f, 'foo', 'none'), 'bar')
1379
Christian Heimes53876d92008-04-19 00:31:39 +00001380
Michael W. Hudsonba283e22005-05-27 15:23:20 +00001381if __name__ == '__main__':
Zachary Ware38c707e2015-04-13 15:00:43 -05001382 unittest.main()