blob: 2e187ac64ee23ae7d5de378a06f0356cc79a3bd8 [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
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000164 self.assertAlmostEqual(float(Foo1()), 42.)
165 self.assertAlmostEqual(float(Foo2()), 42.)
166 self.assertAlmostEqual(float(Foo3(21)), 42.)
167 self.assertRaises(TypeError, float, Foo4(42))
Benjamin Peterson2808d3c2009-04-15 21:34:27 +0000168 self.assertAlmostEqual(float(FooStr('8')), 9.)
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000169
Benjamin Petersona9157232015-03-06 09:08:44 -0500170 class Foo5:
171 def __float__(self):
172 return ""
173 self.assertRaises(TypeError, time.sleep, Foo5())
174
Serhiy Storchaka15095802015-11-25 15:47:01 +0200175 # Issue #24731
176 class F:
177 def __float__(self):
178 return OtherFloatSubclass(42.)
179 self.assertAlmostEqual(float(F()), 42.)
180 self.assertIs(type(float(F())), OtherFloatSubclass)
181 self.assertAlmostEqual(FloatSubclass(F()), 42.)
182 self.assertIs(type(FloatSubclass(F())), FloatSubclass)
183
Benjamin Petersonb3b8ba62011-10-28 19:42:48 -0400184 def test_is_integer(self):
185 self.assertFalse((1.1).is_integer())
186 self.assertTrue((1.).is_integer())
187 self.assertFalse(float("nan").is_integer())
188 self.assertFalse(float("inf").is_integer())
189
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000190 def test_floatasratio(self):
191 for f, ratio in [
192 (0.875, (7, 8)),
193 (-0.875, (-7, 8)),
194 (0.0, (0, 1)),
195 (11.5, (23, 2)),
196 ]:
197 self.assertEqual(f.as_integer_ratio(), ratio)
198
199 for i in range(10000):
200 f = random.random()
201 f *= 10 ** random.randint(-100, 100)
202 n, d = f.as_integer_ratio()
203 self.assertEqual(float(n).__truediv__(d), f)
204
205 R = fractions.Fraction
206 self.assertEqual(R(0, 1),
207 R(*float(0.0).as_integer_ratio()))
208 self.assertEqual(R(5, 2),
209 R(*float(2.5).as_integer_ratio()))
210 self.assertEqual(R(1, 2),
211 R(*float(0.5).as_integer_ratio()))
212 self.assertEqual(R(4728779608739021, 2251799813685248),
213 R(*float(2.1).as_integer_ratio()))
214 self.assertEqual(R(-4728779608739021, 2251799813685248),
215 R(*float(-2.1).as_integer_ratio()))
216 self.assertEqual(R(-2100, 1),
217 R(*float(-2100.0).as_integer_ratio()))
218
219 self.assertRaises(OverflowError, float('inf').as_integer_ratio)
220 self.assertRaises(OverflowError, float('-inf').as_integer_ratio)
221 self.assertRaises(ValueError, float('nan').as_integer_ratio)
222
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000223 def test_float_containment(self):
224 floats = (INF, -INF, 0.0, 1.0, NAN)
225 for f in floats:
Benjamin Peterson577473f2010-01-19 00:09:57 +0000226 self.assertIn(f, [f])
Benjamin Peterson577473f2010-01-19 00:09:57 +0000227 self.assertIn(f, (f,))
Benjamin Peterson577473f2010-01-19 00:09:57 +0000228 self.assertIn(f, {f})
Benjamin Peterson577473f2010-01-19 00:09:57 +0000229 self.assertIn(f, {f: None})
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000230 self.assertEqual([f].count(f), 1, "[].count('%r') != 1" % f)
Benjamin Peterson577473f2010-01-19 00:09:57 +0000231 self.assertIn(f, floats)
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000232
233 for f in floats:
234 # nonidentical containers, same type, same contents
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000235 self.assertTrue([f] == [f], "[%r] != [%r]" % (f, f))
236 self.assertTrue((f,) == (f,), "(%r,) != (%r,)" % (f, f))
237 self.assertTrue({f} == {f}, "{%r} != {%r}" % (f, f))
238 self.assertTrue({f : None} == {f: None}, "{%r : None} != "
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000239 "{%r : None}" % (f, f))
240
241 # identical containers
242 l, t, s, d = [f], (f,), {f}, {f: None}
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000243 self.assertTrue(l == l, "[%r] not equal to itself" % f)
244 self.assertTrue(t == t, "(%r,) not equal to itself" % f)
245 self.assertTrue(s == s, "{%r} not equal to itself" % f)
246 self.assertTrue(d == d, "{%r : None} not equal to itself" % f)
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000247
Mark Dickinson9ab44b52009-12-30 16:22:49 +0000248 def assertEqualAndEqualSign(self, a, b):
249 # fail unless a == b and a and b have the same sign bit;
250 # the only difference from assertEqual is that this test
Ezio Melotti13925002011-03-16 11:05:33 +0200251 # distinguishes -0.0 and 0.0.
Mark Dickinson9ab44b52009-12-30 16:22:49 +0000252 self.assertEqual((a, copysign(1.0, a)), (b, copysign(1.0, b)))
253
Eric Smith3ab08ca2010-12-04 15:17:38 +0000254 @support.requires_IEEE_754
Mark Dickinsond2a9b202010-12-04 12:25:30 +0000255 def test_float_mod(self):
256 # Check behaviour of % operator for IEEE 754 special cases.
257 # In particular, check signs of zeros.
258 mod = operator.mod
259
260 self.assertEqualAndEqualSign(mod(-1.0, 1.0), 0.0)
261 self.assertEqualAndEqualSign(mod(-1e-100, 1.0), 1.0)
262 self.assertEqualAndEqualSign(mod(-0.0, 1.0), 0.0)
263 self.assertEqualAndEqualSign(mod(0.0, 1.0), 0.0)
264 self.assertEqualAndEqualSign(mod(1e-100, 1.0), 1e-100)
265 self.assertEqualAndEqualSign(mod(1.0, 1.0), 0.0)
266
267 self.assertEqualAndEqualSign(mod(-1.0, -1.0), -0.0)
268 self.assertEqualAndEqualSign(mod(-1e-100, -1.0), -1e-100)
269 self.assertEqualAndEqualSign(mod(-0.0, -1.0), -0.0)
270 self.assertEqualAndEqualSign(mod(0.0, -1.0), -0.0)
271 self.assertEqualAndEqualSign(mod(1e-100, -1.0), -1.0)
272 self.assertEqualAndEqualSign(mod(1.0, -1.0), -0.0)
273
Eric Smith3ab08ca2010-12-04 15:17:38 +0000274 @support.requires_IEEE_754
Mark Dickinson9ab44b52009-12-30 16:22:49 +0000275 def test_float_pow(self):
276 # test builtin pow and ** operator for IEEE 754 special cases.
277 # Special cases taken from section F.9.4.4 of the C99 specification
278
279 for pow_op in pow, operator.pow:
280 # x**NAN is NAN for any x except 1
281 self.assertTrue(isnan(pow_op(-INF, NAN)))
282 self.assertTrue(isnan(pow_op(-2.0, NAN)))
283 self.assertTrue(isnan(pow_op(-1.0, NAN)))
284 self.assertTrue(isnan(pow_op(-0.5, NAN)))
285 self.assertTrue(isnan(pow_op(-0.0, NAN)))
286 self.assertTrue(isnan(pow_op(0.0, NAN)))
287 self.assertTrue(isnan(pow_op(0.5, NAN)))
288 self.assertTrue(isnan(pow_op(2.0, NAN)))
289 self.assertTrue(isnan(pow_op(INF, NAN)))
290 self.assertTrue(isnan(pow_op(NAN, NAN)))
291
292 # NAN**y is NAN for any y except +-0
293 self.assertTrue(isnan(pow_op(NAN, -INF)))
294 self.assertTrue(isnan(pow_op(NAN, -2.0)))
295 self.assertTrue(isnan(pow_op(NAN, -1.0)))
296 self.assertTrue(isnan(pow_op(NAN, -0.5)))
297 self.assertTrue(isnan(pow_op(NAN, 0.5)))
298 self.assertTrue(isnan(pow_op(NAN, 1.0)))
299 self.assertTrue(isnan(pow_op(NAN, 2.0)))
300 self.assertTrue(isnan(pow_op(NAN, INF)))
301
302 # (+-0)**y raises ZeroDivisionError for y a negative odd integer
303 self.assertRaises(ZeroDivisionError, pow_op, -0.0, -1.0)
304 self.assertRaises(ZeroDivisionError, pow_op, 0.0, -1.0)
305
306 # (+-0)**y raises ZeroDivisionError for y finite and negative
307 # but not an odd integer
308 self.assertRaises(ZeroDivisionError, pow_op, -0.0, -2.0)
309 self.assertRaises(ZeroDivisionError, pow_op, -0.0, -0.5)
310 self.assertRaises(ZeroDivisionError, pow_op, 0.0, -2.0)
311 self.assertRaises(ZeroDivisionError, pow_op, 0.0, -0.5)
312
313 # (+-0)**y is +-0 for y a positive odd integer
314 self.assertEqualAndEqualSign(pow_op(-0.0, 1.0), -0.0)
315 self.assertEqualAndEqualSign(pow_op(0.0, 1.0), 0.0)
316
317 # (+-0)**y is 0 for y finite and positive but not an odd integer
318 self.assertEqualAndEqualSign(pow_op(-0.0, 0.5), 0.0)
319 self.assertEqualAndEqualSign(pow_op(-0.0, 2.0), 0.0)
320 self.assertEqualAndEqualSign(pow_op(0.0, 0.5), 0.0)
321 self.assertEqualAndEqualSign(pow_op(0.0, 2.0), 0.0)
322
323 # (-1)**+-inf is 1
324 self.assertEqualAndEqualSign(pow_op(-1.0, -INF), 1.0)
325 self.assertEqualAndEqualSign(pow_op(-1.0, INF), 1.0)
326
327 # 1**y is 1 for any y, even if y is an infinity or nan
328 self.assertEqualAndEqualSign(pow_op(1.0, -INF), 1.0)
329 self.assertEqualAndEqualSign(pow_op(1.0, -2.0), 1.0)
330 self.assertEqualAndEqualSign(pow_op(1.0, -1.0), 1.0)
331 self.assertEqualAndEqualSign(pow_op(1.0, -0.5), 1.0)
332 self.assertEqualAndEqualSign(pow_op(1.0, -0.0), 1.0)
333 self.assertEqualAndEqualSign(pow_op(1.0, 0.0), 1.0)
334 self.assertEqualAndEqualSign(pow_op(1.0, 0.5), 1.0)
335 self.assertEqualAndEqualSign(pow_op(1.0, 1.0), 1.0)
336 self.assertEqualAndEqualSign(pow_op(1.0, 2.0), 1.0)
337 self.assertEqualAndEqualSign(pow_op(1.0, INF), 1.0)
338 self.assertEqualAndEqualSign(pow_op(1.0, NAN), 1.0)
339
340 # x**+-0 is 1 for any x, even if x is a zero, infinity, or nan
341 self.assertEqualAndEqualSign(pow_op(-INF, 0.0), 1.0)
342 self.assertEqualAndEqualSign(pow_op(-2.0, 0.0), 1.0)
343 self.assertEqualAndEqualSign(pow_op(-1.0, 0.0), 1.0)
344 self.assertEqualAndEqualSign(pow_op(-0.5, 0.0), 1.0)
345 self.assertEqualAndEqualSign(pow_op(-0.0, 0.0), 1.0)
346 self.assertEqualAndEqualSign(pow_op(0.0, 0.0), 1.0)
347 self.assertEqualAndEqualSign(pow_op(0.5, 0.0), 1.0)
348 self.assertEqualAndEqualSign(pow_op(1.0, 0.0), 1.0)
349 self.assertEqualAndEqualSign(pow_op(2.0, 0.0), 1.0)
350 self.assertEqualAndEqualSign(pow_op(INF, 0.0), 1.0)
351 self.assertEqualAndEqualSign(pow_op(NAN, 0.0), 1.0)
352 self.assertEqualAndEqualSign(pow_op(-INF, -0.0), 1.0)
353 self.assertEqualAndEqualSign(pow_op(-2.0, -0.0), 1.0)
354 self.assertEqualAndEqualSign(pow_op(-1.0, -0.0), 1.0)
355 self.assertEqualAndEqualSign(pow_op(-0.5, -0.0), 1.0)
356 self.assertEqualAndEqualSign(pow_op(-0.0, -0.0), 1.0)
357 self.assertEqualAndEqualSign(pow_op(0.0, -0.0), 1.0)
358 self.assertEqualAndEqualSign(pow_op(0.5, -0.0), 1.0)
359 self.assertEqualAndEqualSign(pow_op(1.0, -0.0), 1.0)
360 self.assertEqualAndEqualSign(pow_op(2.0, -0.0), 1.0)
361 self.assertEqualAndEqualSign(pow_op(INF, -0.0), 1.0)
362 self.assertEqualAndEqualSign(pow_op(NAN, -0.0), 1.0)
363
364 # x**y defers to complex pow for finite negative x and
365 # non-integral y.
366 self.assertEqual(type(pow_op(-2.0, -0.5)), complex)
367 self.assertEqual(type(pow_op(-2.0, 0.5)), complex)
368 self.assertEqual(type(pow_op(-1.0, -0.5)), complex)
369 self.assertEqual(type(pow_op(-1.0, 0.5)), complex)
370 self.assertEqual(type(pow_op(-0.5, -0.5)), complex)
371 self.assertEqual(type(pow_op(-0.5, 0.5)), complex)
372
373 # x**-INF is INF for abs(x) < 1
374 self.assertEqualAndEqualSign(pow_op(-0.5, -INF), INF)
375 self.assertEqualAndEqualSign(pow_op(-0.0, -INF), INF)
376 self.assertEqualAndEqualSign(pow_op(0.0, -INF), INF)
377 self.assertEqualAndEqualSign(pow_op(0.5, -INF), INF)
378
379 # x**-INF is 0 for abs(x) > 1
380 self.assertEqualAndEqualSign(pow_op(-INF, -INF), 0.0)
381 self.assertEqualAndEqualSign(pow_op(-2.0, -INF), 0.0)
382 self.assertEqualAndEqualSign(pow_op(2.0, -INF), 0.0)
383 self.assertEqualAndEqualSign(pow_op(INF, -INF), 0.0)
384
385 # x**INF is 0 for abs(x) < 1
386 self.assertEqualAndEqualSign(pow_op(-0.5, INF), 0.0)
387 self.assertEqualAndEqualSign(pow_op(-0.0, INF), 0.0)
388 self.assertEqualAndEqualSign(pow_op(0.0, INF), 0.0)
389 self.assertEqualAndEqualSign(pow_op(0.5, INF), 0.0)
390
391 # x**INF is INF for abs(x) > 1
392 self.assertEqualAndEqualSign(pow_op(-INF, INF), INF)
393 self.assertEqualAndEqualSign(pow_op(-2.0, INF), INF)
394 self.assertEqualAndEqualSign(pow_op(2.0, INF), INF)
395 self.assertEqualAndEqualSign(pow_op(INF, INF), INF)
396
397 # (-INF)**y is -0.0 for y a negative odd integer
398 self.assertEqualAndEqualSign(pow_op(-INF, -1.0), -0.0)
399
400 # (-INF)**y is 0.0 for y negative but not an odd integer
401 self.assertEqualAndEqualSign(pow_op(-INF, -0.5), 0.0)
402 self.assertEqualAndEqualSign(pow_op(-INF, -2.0), 0.0)
403
404 # (-INF)**y is -INF for y a positive odd integer
405 self.assertEqualAndEqualSign(pow_op(-INF, 1.0), -INF)
406
407 # (-INF)**y is INF for y positive but not an odd integer
408 self.assertEqualAndEqualSign(pow_op(-INF, 0.5), INF)
409 self.assertEqualAndEqualSign(pow_op(-INF, 2.0), INF)
410
411 # INF**y is INF for y positive
412 self.assertEqualAndEqualSign(pow_op(INF, 0.5), INF)
413 self.assertEqualAndEqualSign(pow_op(INF, 1.0), INF)
414 self.assertEqualAndEqualSign(pow_op(INF, 2.0), INF)
415
416 # INF**y is 0.0 for y negative
417 self.assertEqualAndEqualSign(pow_op(INF, -2.0), 0.0)
418 self.assertEqualAndEqualSign(pow_op(INF, -1.0), 0.0)
419 self.assertEqualAndEqualSign(pow_op(INF, -0.5), 0.0)
420
421 # basic checks not covered by the special cases above
422 self.assertEqualAndEqualSign(pow_op(-2.0, -2.0), 0.25)
423 self.assertEqualAndEqualSign(pow_op(-2.0, -1.0), -0.5)
424 self.assertEqualAndEqualSign(pow_op(-2.0, -0.0), 1.0)
425 self.assertEqualAndEqualSign(pow_op(-2.0, 0.0), 1.0)
426 self.assertEqualAndEqualSign(pow_op(-2.0, 1.0), -2.0)
427 self.assertEqualAndEqualSign(pow_op(-2.0, 2.0), 4.0)
428 self.assertEqualAndEqualSign(pow_op(-1.0, -2.0), 1.0)
429 self.assertEqualAndEqualSign(pow_op(-1.0, -1.0), -1.0)
430 self.assertEqualAndEqualSign(pow_op(-1.0, -0.0), 1.0)
431 self.assertEqualAndEqualSign(pow_op(-1.0, 0.0), 1.0)
432 self.assertEqualAndEqualSign(pow_op(-1.0, 1.0), -1.0)
433 self.assertEqualAndEqualSign(pow_op(-1.0, 2.0), 1.0)
434 self.assertEqualAndEqualSign(pow_op(2.0, -2.0), 0.25)
435 self.assertEqualAndEqualSign(pow_op(2.0, -1.0), 0.5)
436 self.assertEqualAndEqualSign(pow_op(2.0, -0.0), 1.0)
437 self.assertEqualAndEqualSign(pow_op(2.0, 0.0), 1.0)
438 self.assertEqualAndEqualSign(pow_op(2.0, 1.0), 2.0)
439 self.assertEqualAndEqualSign(pow_op(2.0, 2.0), 4.0)
440
441 # 1 ** large and -1 ** large; some libms apparently
442 # have problems with these
443 self.assertEqualAndEqualSign(pow_op(1.0, -1e100), 1.0)
444 self.assertEqualAndEqualSign(pow_op(1.0, 1e100), 1.0)
445 self.assertEqualAndEqualSign(pow_op(-1.0, -1e100), 1.0)
446 self.assertEqualAndEqualSign(pow_op(-1.0, 1e100), 1.0)
447
448 # check sign for results that underflow to 0
449 self.assertEqualAndEqualSign(pow_op(-2.0, -2000.0), 0.0)
450 self.assertEqual(type(pow_op(-2.0, -2000.5)), complex)
451 self.assertEqualAndEqualSign(pow_op(-2.0, -2001.0), -0.0)
452 self.assertEqualAndEqualSign(pow_op(2.0, -2000.0), 0.0)
453 self.assertEqualAndEqualSign(pow_op(2.0, -2000.5), 0.0)
454 self.assertEqualAndEqualSign(pow_op(2.0, -2001.0), 0.0)
455 self.assertEqualAndEqualSign(pow_op(-0.5, 2000.0), 0.0)
456 self.assertEqual(type(pow_op(-0.5, 2000.5)), complex)
457 self.assertEqualAndEqualSign(pow_op(-0.5, 2001.0), -0.0)
458 self.assertEqualAndEqualSign(pow_op(0.5, 2000.0), 0.0)
459 self.assertEqualAndEqualSign(pow_op(0.5, 2000.5), 0.0)
460 self.assertEqualAndEqualSign(pow_op(0.5, 2001.0), 0.0)
461
462 # check we don't raise an exception for subnormal results,
463 # and validate signs. Tests currently disabled, since
464 # they fail on systems where a subnormal result from pow
465 # is flushed to zero (e.g. Debian/ia64.)
466 #self.assertTrue(0.0 < pow_op(0.5, 1048) < 1e-315)
467 #self.assertTrue(0.0 < pow_op(-0.5, 1048) < 1e-315)
468 #self.assertTrue(0.0 < pow_op(0.5, 1047) < 1e-315)
469 #self.assertTrue(0.0 > pow_op(-0.5, 1047) > -1e-315)
470 #self.assertTrue(0.0 < pow_op(2.0, -1048) < 1e-315)
471 #self.assertTrue(0.0 < pow_op(-2.0, -1048) < 1e-315)
472 #self.assertTrue(0.0 < pow_op(2.0, -1047) < 1e-315)
473 #self.assertTrue(0.0 > pow_op(-2.0, -1047) > -1e-315)
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000474
475
Benjamin Petersone401c682010-07-02 23:25:44 +0000476@requires_setformat
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000477class FormatFunctionsTestCase(unittest.TestCase):
478
479 def setUp(self):
480 self.save_formats = {'double':float.__getformat__('double'),
481 'float':float.__getformat__('float')}
482
483 def tearDown(self):
484 float.__setformat__('double', self.save_formats['double'])
485 float.__setformat__('float', self.save_formats['float'])
486
487 def test_getformat(self):
Ezio Melottib58e0bd2010-01-23 15:40:09 +0000488 self.assertIn(float.__getformat__('double'),
489 ['unknown', 'IEEE, big-endian', 'IEEE, little-endian'])
490 self.assertIn(float.__getformat__('float'),
491 ['unknown', 'IEEE, big-endian', 'IEEE, little-endian'])
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000492 self.assertRaises(ValueError, float.__getformat__, 'chicken')
493 self.assertRaises(TypeError, float.__getformat__, 1)
494
495 def test_setformat(self):
496 for t in 'double', 'float':
497 float.__setformat__(t, 'unknown')
498 if self.save_formats[t] == 'IEEE, big-endian':
499 self.assertRaises(ValueError, float.__setformat__,
500 t, 'IEEE, little-endian')
501 elif self.save_formats[t] == 'IEEE, little-endian':
502 self.assertRaises(ValueError, float.__setformat__,
503 t, 'IEEE, big-endian')
504 else:
505 self.assertRaises(ValueError, float.__setformat__,
506 t, 'IEEE, big-endian')
507 self.assertRaises(ValueError, float.__setformat__,
508 t, 'IEEE, little-endian')
509 self.assertRaises(ValueError, float.__setformat__,
510 t, 'chicken')
511 self.assertRaises(ValueError, float.__setformat__,
512 'chicken', 'unknown')
513
Guido van Rossum2be161d2007-05-15 20:43:51 +0000514BE_DOUBLE_INF = b'\x7f\xf0\x00\x00\x00\x00\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000515LE_DOUBLE_INF = bytes(reversed(BE_DOUBLE_INF))
Guido van Rossum2be161d2007-05-15 20:43:51 +0000516BE_DOUBLE_NAN = b'\x7f\xf8\x00\x00\x00\x00\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000517LE_DOUBLE_NAN = bytes(reversed(BE_DOUBLE_NAN))
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000518
Guido van Rossum2be161d2007-05-15 20:43:51 +0000519BE_FLOAT_INF = b'\x7f\x80\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000520LE_FLOAT_INF = bytes(reversed(BE_FLOAT_INF))
Guido van Rossum2be161d2007-05-15 20:43:51 +0000521BE_FLOAT_NAN = b'\x7f\xc0\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000522LE_FLOAT_NAN = bytes(reversed(BE_FLOAT_NAN))
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000523
524# on non-IEEE platforms, attempting to unpack a bit pattern
525# representing an infinity or a NaN should raise an exception.
526
Benjamin Petersone401c682010-07-02 23:25:44 +0000527@requires_setformat
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000528class UnknownFormatTestCase(unittest.TestCase):
529 def setUp(self):
530 self.save_formats = {'double':float.__getformat__('double'),
531 'float':float.__getformat__('float')}
532 float.__setformat__('double', 'unknown')
533 float.__setformat__('float', 'unknown')
Tim Peters5d36a552005-06-03 22:40:27 +0000534
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000535 def tearDown(self):
536 float.__setformat__('double', self.save_formats['double'])
537 float.__setformat__('float', self.save_formats['float'])
538
539 def test_double_specials_dont_unpack(self):
540 for fmt, data in [('>d', BE_DOUBLE_INF),
541 ('>d', BE_DOUBLE_NAN),
542 ('<d', LE_DOUBLE_INF),
543 ('<d', LE_DOUBLE_NAN)]:
544 self.assertRaises(ValueError, struct.unpack, fmt, data)
545
546 def test_float_specials_dont_unpack(self):
547 for fmt, data in [('>f', BE_FLOAT_INF),
548 ('>f', BE_FLOAT_NAN),
549 ('<f', LE_FLOAT_INF),
550 ('<f', LE_FLOAT_NAN)]:
551 self.assertRaises(ValueError, struct.unpack, fmt, data)
552
553
554# on an IEEE platform, all we guarantee is that bit patterns
555# representing infinities or NaNs do not raise an exception; all else
556# is accident (today).
Guido van Rossum04110fb2007-08-24 16:32:05 +0000557# let's also try to guarantee that -0.0 and 0.0 don't get confused.
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000558
559class IEEEFormatTestCase(unittest.TestCase):
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000560
Eric Smith3ab08ca2010-12-04 15:17:38 +0000561 @support.requires_IEEE_754
Benjamin Petersone401c682010-07-02 23:25:44 +0000562 def test_double_specials_do_unpack(self):
563 for fmt, data in [('>d', BE_DOUBLE_INF),
564 ('>d', BE_DOUBLE_NAN),
565 ('<d', LE_DOUBLE_INF),
566 ('<d', LE_DOUBLE_NAN)]:
567 struct.unpack(fmt, data)
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000568
Eric Smith3ab08ca2010-12-04 15:17:38 +0000569 @support.requires_IEEE_754
Benjamin Petersone401c682010-07-02 23:25:44 +0000570 def test_float_specials_do_unpack(self):
571 for fmt, data in [('>f', BE_FLOAT_INF),
572 ('>f', BE_FLOAT_NAN),
573 ('<f', LE_FLOAT_INF),
574 ('<f', LE_FLOAT_NAN)]:
575 struct.unpack(fmt, data)
Guido van Rossum04110fb2007-08-24 16:32:05 +0000576
Eric Smith8c663262007-08-25 02:26:07 +0000577class FormatTestCase(unittest.TestCase):
Benjamin Petersone401c682010-07-02 23:25:44 +0000578
Eric Smith11fe3e02007-08-31 01:33:06 +0000579 def test_format(self):
Eric Smith8c663262007-08-25 02:26:07 +0000580 # these should be rewritten to use both format(x, spec) and
581 # x.__format__(spec)
582
583 self.assertEqual(format(0.0, 'f'), '0.000000')
584
585 # the default is 'g', except for empty format spec
586 self.assertEqual(format(0.0, ''), '0.0')
587 self.assertEqual(format(0.01, ''), '0.01')
588 self.assertEqual(format(0.01, 'g'), '0.01')
589
Eric Smith63376222009-05-05 14:04:18 +0000590 # empty presentation type should format in the same way as str
591 # (issue 5920)
592 x = 100/7.
593 self.assertEqual(format(x, ''), str(x))
594 self.assertEqual(format(x, '-'), str(x))
595 self.assertEqual(format(x, '>'), str(x))
596 self.assertEqual(format(x, '2'), str(x))
Eric Smith8c663262007-08-25 02:26:07 +0000597
598 self.assertEqual(format(1.0, 'f'), '1.000000')
Eric Smith8c663262007-08-25 02:26:07 +0000599
600 self.assertEqual(format(-1.0, 'f'), '-1.000000')
Eric Smith8c663262007-08-25 02:26:07 +0000601
602 self.assertEqual(format( 1.0, ' f'), ' 1.000000')
603 self.assertEqual(format(-1.0, ' f'), '-1.000000')
604 self.assertEqual(format( 1.0, '+f'), '+1.000000')
605 self.assertEqual(format(-1.0, '+f'), '-1.000000')
606
607 # % formatting
608 self.assertEqual(format(-1.0, '%'), '-100.000000%')
609
610 # conversion to string should fail
611 self.assertRaises(ValueError, format, 3.0, "s")
612
Eric Smith7b69c6c2008-01-27 21:07:59 +0000613 # other format specifiers shouldn't work on floats,
614 # in particular int specifiers
615 for format_spec in ([chr(x) for x in range(ord('a'), ord('z')+1)] +
616 [chr(x) for x in range(ord('A'), ord('Z')+1)]):
617 if not format_spec in 'eEfFgGn%':
618 self.assertRaises(ValueError, format, 0.0, format_spec)
619 self.assertRaises(ValueError, format, 1.0, format_spec)
620 self.assertRaises(ValueError, format, -1.0, format_spec)
621 self.assertRaises(ValueError, format, 1e100, format_spec)
622 self.assertRaises(ValueError, format, -1e100, format_spec)
623 self.assertRaises(ValueError, format, 1e-100, format_spec)
624 self.assertRaises(ValueError, format, -1e-100, format_spec)
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000625
Eric Smith741191f2009-05-06 13:08:15 +0000626 # issue 3382
627 self.assertEqual(format(NAN, 'f'), 'nan')
628 self.assertEqual(format(NAN, 'F'), 'NAN')
629 self.assertEqual(format(INF, 'f'), 'inf')
630 self.assertEqual(format(INF, 'F'), 'INF')
631
Eric Smith3ab08ca2010-12-04 15:17:38 +0000632 @support.requires_IEEE_754
Eric Smith0923d1d2009-04-16 20:16:10 +0000633 def test_format_testfile(self):
Brian Curtin076623b2010-10-31 00:03:45 +0000634 with open(format_testfile) as testfile:
635 for line in testfile:
636 if line.startswith('--'):
637 continue
638 line = line.strip()
639 if not line:
640 continue
Eric Smith0923d1d2009-04-16 20:16:10 +0000641
Brian Curtin076623b2010-10-31 00:03:45 +0000642 lhs, rhs = map(str.strip, line.split('->'))
643 fmt, arg = lhs.split()
644 self.assertEqual(fmt % float(arg), rhs)
645 self.assertEqual(fmt % -float(arg), '-' + rhs)
Eric Smith0923d1d2009-04-16 20:16:10 +0000646
Mark Dickinsond3ca5572009-04-29 18:47:07 +0000647 def test_issue5864(self):
Ezio Melottib3aedd42010-11-20 19:04:17 +0000648 self.assertEqual(format(123.456, '.4'), '123.5')
649 self.assertEqual(format(1234.56, '.4'), '1.235e+03')
650 self.assertEqual(format(12345.6, '.4'), '1.235e+04')
Mark Dickinsond3ca5572009-04-29 18:47:07 +0000651
Mark Dickinson7efad9e2009-04-17 20:59:58 +0000652class ReprTestCase(unittest.TestCase):
653 def test_repr(self):
654 floats_file = open(os.path.join(os.path.split(__file__)[0],
655 'floating_points.txt'))
656 for line in floats_file:
657 line = line.strip()
658 if not line or line.startswith('#'):
659 continue
660 v = eval(line)
661 self.assertEqual(v, eval(repr(v)))
662 floats_file.close()
663
Eric Smith0923d1d2009-04-16 20:16:10 +0000664 @unittest.skipUnless(getattr(sys, 'float_repr_style', '') == 'short',
665 "applies only when using short float repr style")
666 def test_short_repr(self):
667 # test short float repr introduced in Python 3.1. One aspect
668 # of this repr is that we get some degree of str -> float ->
669 # str roundtripping. In particular, for any numeric string
670 # containing 15 or fewer significant digits, those exact same
671 # digits (modulo trailing zeros) should appear in the output.
672 # No more repr(0.03) -> "0.029999999999999999"!
673
674 test_strings = [
675 # output always includes *either* a decimal point and at
676 # least one digit after that point, or an exponent.
677 '0.0',
678 '1.0',
679 '0.01',
680 '0.02',
681 '0.03',
682 '0.04',
683 '0.05',
684 '1.23456789',
685 '10.0',
686 '100.0',
687 # values >= 1e16 get an exponent...
688 '1000000000000000.0',
689 '9999999999999990.0',
690 '1e+16',
691 '1e+17',
692 # ... and so do values < 1e-4
693 '0.001',
694 '0.001001',
695 '0.00010000000000001',
696 '0.0001',
697 '9.999999999999e-05',
698 '1e-05',
699 # values designed to provoke failure if the FPU rounding
700 # precision isn't set correctly
701 '8.72293771110361e+25',
702 '7.47005307342313e+26',
703 '2.86438000439698e+28',
704 '8.89142905246179e+28',
705 '3.08578087079232e+35',
706 ]
707
708 for s in test_strings:
709 negs = '-'+s
710 self.assertEqual(s, repr(float(s)))
711 self.assertEqual(negs, repr(float(negs)))
Mark Dickinson388122d2010-08-04 20:56:28 +0000712 # Since Python 3.2, repr and str are identical
713 self.assertEqual(repr(float(s)), str(float(s)))
714 self.assertEqual(repr(float(negs)), str(float(negs)))
Benjamin Petersone401c682010-07-02 23:25:44 +0000715
Eric Smith3ab08ca2010-12-04 15:17:38 +0000716@support.requires_IEEE_754
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000717class RoundTestCase(unittest.TestCase):
Benjamin Petersone401c682010-07-02 23:25:44 +0000718
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000719 def test_inf_nan(self):
720 self.assertRaises(OverflowError, round, INF)
721 self.assertRaises(OverflowError, round, -INF)
722 self.assertRaises(ValueError, round, NAN)
Mark Dickinson4ca33d12009-11-24 10:59:34 +0000723 self.assertRaises(TypeError, round, INF, 0.0)
724 self.assertRaises(TypeError, round, -INF, 1.0)
725 self.assertRaises(TypeError, round, NAN, "ceci n'est pas un integer")
726 self.assertRaises(TypeError, round, -0.0, 1j)
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000727
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000728 def test_large_n(self):
729 for n in [324, 325, 400, 2**31-1, 2**31, 2**32, 2**100]:
730 self.assertEqual(round(123.456, n), 123.456)
731 self.assertEqual(round(-123.456, n), -123.456)
732 self.assertEqual(round(1e300, n), 1e300)
733 self.assertEqual(round(1e-320, n), 1e-320)
734 self.assertEqual(round(1e150, 300), 1e150)
735 self.assertEqual(round(1e300, 307), 1e300)
736 self.assertEqual(round(-3.1415, 308), -3.1415)
737 self.assertEqual(round(1e150, 309), 1e150)
738 self.assertEqual(round(1.4e-315, 315), 1e-315)
739
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000740 def test_small_n(self):
741 for n in [-308, -309, -400, 1-2**31, -2**31, -2**31-1, -2**100]:
742 self.assertEqual(round(123.456, n), 0.0)
743 self.assertEqual(round(-123.456, n), -0.0)
744 self.assertEqual(round(1e300, n), 0.0)
745 self.assertEqual(round(1e-320, n), 0.0)
746
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000747 def test_overflow(self):
748 self.assertRaises(OverflowError, round, 1.6e308, -308)
749 self.assertRaises(OverflowError, round, -1.7e308, -308)
750
751 @unittest.skipUnless(getattr(sys, 'float_repr_style', '') == 'short',
752 "applies only when using short float repr style")
753 def test_previous_round_bugs(self):
754 # particular cases that have occurred in bug reports
755 self.assertEqual(round(562949953421312.5, 1),
756 562949953421312.5)
757 self.assertEqual(round(56294995342131.5, 3),
758 56294995342131.5)
759 # round-half-even
760 self.assertEqual(round(25.0, -1), 20.0)
761 self.assertEqual(round(35.0, -1), 40.0)
762 self.assertEqual(round(45.0, -1), 40.0)
763 self.assertEqual(round(55.0, -1), 60.0)
764 self.assertEqual(round(65.0, -1), 60.0)
765 self.assertEqual(round(75.0, -1), 80.0)
766 self.assertEqual(round(85.0, -1), 80.0)
767 self.assertEqual(round(95.0, -1), 100.0)
768
769 @unittest.skipUnless(getattr(sys, 'float_repr_style', '') == 'short',
770 "applies only when using short float repr style")
771 def test_matches_float_format(self):
772 # round should give the same results as float formatting
773 for i in range(500):
774 x = i/1000.
775 self.assertEqual(float(format(x, '.0f')), round(x, 0))
776 self.assertEqual(float(format(x, '.1f')), round(x, 1))
777 self.assertEqual(float(format(x, '.2f')), round(x, 2))
778 self.assertEqual(float(format(x, '.3f')), round(x, 3))
779
780 for i in range(5, 5000, 10):
781 x = i/1000.
782 self.assertEqual(float(format(x, '.0f')), round(x, 0))
783 self.assertEqual(float(format(x, '.1f')), round(x, 1))
784 self.assertEqual(float(format(x, '.2f')), round(x, 2))
785 self.assertEqual(float(format(x, '.3f')), round(x, 3))
786
787 for i in range(500):
788 x = random.random()
789 self.assertEqual(float(format(x, '.0f')), round(x, 0))
790 self.assertEqual(float(format(x, '.1f')), round(x, 1))
791 self.assertEqual(float(format(x, '.2f')), round(x, 2))
792 self.assertEqual(float(format(x, '.3f')), round(x, 3))
793
Eric Smith8a10ecc2009-12-02 17:58:24 +0000794 def test_format_specials(self):
795 # Test formatting of nans and infs.
796
797 def test(fmt, value, expected):
798 # Test with both % and format().
799 self.assertEqual(fmt % value, expected, fmt)
Eric Smith984bb582010-11-25 16:08:06 +0000800 fmt = fmt[1:] # strip off the %
801 self.assertEqual(format(value, fmt), expected, fmt)
Eric Smith8a10ecc2009-12-02 17:58:24 +0000802
803 for fmt in ['%e', '%f', '%g', '%.0e', '%.6f', '%.20g',
804 '%#e', '%#f', '%#g', '%#.20e', '%#.15f', '%#.3g']:
805 pfmt = '%+' + fmt[1:]
806 sfmt = '% ' + fmt[1:]
807 test(fmt, INF, 'inf')
808 test(fmt, -INF, '-inf')
809 test(fmt, NAN, 'nan')
810 test(fmt, -NAN, 'nan')
811 # When asking for a sign, it's always provided. nans are
812 # always positive.
813 test(pfmt, INF, '+inf')
814 test(pfmt, -INF, '-inf')
815 test(pfmt, NAN, '+nan')
816 test(pfmt, -NAN, '+nan')
817 # When using ' ' for a sign code, only infs can be negative.
818 # Others have a space.
819 test(sfmt, INF, ' inf')
820 test(sfmt, -INF, '-inf')
821 test(sfmt, NAN, ' nan')
822 test(sfmt, -NAN, ' nan')
823
Steve Dowercb39d1f2015-04-15 16:10:59 -0400824 def test_None_ndigits(self):
825 for x in round(1.23), round(1.23, None), round(1.23, ndigits=None):
826 self.assertEqual(x, 1)
827 self.assertIsInstance(x, int)
828 for x in round(1.78), round(1.78, None), round(1.78, ndigits=None):
829 self.assertEqual(x, 2)
830 self.assertIsInstance(x, int)
831
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000832
Christian Heimes99170a52007-12-19 02:07:34 +0000833# Beginning with Python 2.6 float has cross platform compatible
Georg Brandl2ee470f2008-07-16 12:55:28 +0000834# ways to create and represent inf and nan
Christian Heimes99170a52007-12-19 02:07:34 +0000835class InfNanTest(unittest.TestCase):
836 def test_inf_from_str(self):
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000837 self.assertTrue(isinf(float("inf")))
838 self.assertTrue(isinf(float("+inf")))
839 self.assertTrue(isinf(float("-inf")))
840 self.assertTrue(isinf(float("infinity")))
841 self.assertTrue(isinf(float("+infinity")))
842 self.assertTrue(isinf(float("-infinity")))
Christian Heimes99170a52007-12-19 02:07:34 +0000843
844 self.assertEqual(repr(float("inf")), "inf")
845 self.assertEqual(repr(float("+inf")), "inf")
846 self.assertEqual(repr(float("-inf")), "-inf")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000847 self.assertEqual(repr(float("infinity")), "inf")
848 self.assertEqual(repr(float("+infinity")), "inf")
849 self.assertEqual(repr(float("-infinity")), "-inf")
Christian Heimes99170a52007-12-19 02:07:34 +0000850
851 self.assertEqual(repr(float("INF")), "inf")
852 self.assertEqual(repr(float("+Inf")), "inf")
853 self.assertEqual(repr(float("-iNF")), "-inf")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000854 self.assertEqual(repr(float("Infinity")), "inf")
855 self.assertEqual(repr(float("+iNfInItY")), "inf")
856 self.assertEqual(repr(float("-INFINITY")), "-inf")
Christian Heimes99170a52007-12-19 02:07:34 +0000857
858 self.assertEqual(str(float("inf")), "inf")
859 self.assertEqual(str(float("+inf")), "inf")
860 self.assertEqual(str(float("-inf")), "-inf")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000861 self.assertEqual(str(float("infinity")), "inf")
862 self.assertEqual(str(float("+infinity")), "inf")
863 self.assertEqual(str(float("-infinity")), "-inf")
Christian Heimes99170a52007-12-19 02:07:34 +0000864
865 self.assertRaises(ValueError, float, "info")
866 self.assertRaises(ValueError, float, "+info")
867 self.assertRaises(ValueError, float, "-info")
868 self.assertRaises(ValueError, float, "in")
869 self.assertRaises(ValueError, float, "+in")
870 self.assertRaises(ValueError, float, "-in")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000871 self.assertRaises(ValueError, float, "infinit")
872 self.assertRaises(ValueError, float, "+Infin")
873 self.assertRaises(ValueError, float, "-INFI")
874 self.assertRaises(ValueError, float, "infinitys")
Christian Heimes99170a52007-12-19 02:07:34 +0000875
Mark Dickinsonbd16edd2009-05-20 22:05:25 +0000876 self.assertRaises(ValueError, float, "++Inf")
877 self.assertRaises(ValueError, float, "-+inf")
878 self.assertRaises(ValueError, float, "+-infinity")
879 self.assertRaises(ValueError, float, "--Infinity")
880
Christian Heimes99170a52007-12-19 02:07:34 +0000881 def test_inf_as_str(self):
882 self.assertEqual(repr(1e300 * 1e300), "inf")
883 self.assertEqual(repr(-1e300 * 1e300), "-inf")
884
885 self.assertEqual(str(1e300 * 1e300), "inf")
886 self.assertEqual(str(-1e300 * 1e300), "-inf")
887
888 def test_nan_from_str(self):
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000889 self.assertTrue(isnan(float("nan")))
890 self.assertTrue(isnan(float("+nan")))
891 self.assertTrue(isnan(float("-nan")))
Christian Heimes99170a52007-12-19 02:07:34 +0000892
893 self.assertEqual(repr(float("nan")), "nan")
894 self.assertEqual(repr(float("+nan")), "nan")
895 self.assertEqual(repr(float("-nan")), "nan")
896
897 self.assertEqual(repr(float("NAN")), "nan")
898 self.assertEqual(repr(float("+NAn")), "nan")
899 self.assertEqual(repr(float("-NaN")), "nan")
900
901 self.assertEqual(str(float("nan")), "nan")
902 self.assertEqual(str(float("+nan")), "nan")
903 self.assertEqual(str(float("-nan")), "nan")
904
905 self.assertRaises(ValueError, float, "nana")
906 self.assertRaises(ValueError, float, "+nana")
907 self.assertRaises(ValueError, float, "-nana")
908 self.assertRaises(ValueError, float, "na")
909 self.assertRaises(ValueError, float, "+na")
910 self.assertRaises(ValueError, float, "-na")
911
Mark Dickinsonbd16edd2009-05-20 22:05:25 +0000912 self.assertRaises(ValueError, float, "++nan")
913 self.assertRaises(ValueError, float, "-+NAN")
914 self.assertRaises(ValueError, float, "+-NaN")
915 self.assertRaises(ValueError, float, "--nAn")
916
Christian Heimes99170a52007-12-19 02:07:34 +0000917 def test_nan_as_str(self):
918 self.assertEqual(repr(1e300 * 1e300 * 0), "nan")
919 self.assertEqual(repr(-1e300 * 1e300 * 0), "nan")
920
921 self.assertEqual(str(1e300 * 1e300 * 0), "nan")
922 self.assertEqual(str(-1e300 * 1e300 * 0), "nan")
Christian Heimes827b35c2007-12-10 22:19:17 +0000923
Mark Dickinsone383e822012-04-29 15:31:56 +0100924 def test_inf_signs(self):
925 self.assertEqual(copysign(1.0, float('inf')), 1.0)
926 self.assertEqual(copysign(1.0, float('-inf')), -1.0)
927
928 @unittest.skipUnless(getattr(sys, 'float_repr_style', '') == 'short',
929 "applies only when using short float repr style")
930 def test_nan_signs(self):
931 # When using the dtoa.c code, the sign of float('nan') should
932 # be predictable.
933 self.assertEqual(copysign(1.0, float('nan')), 1.0)
934 self.assertEqual(copysign(1.0, float('-nan')), -1.0)
935
936
Mark Dickinson65fe25e2008-07-16 11:30:51 +0000937fromHex = float.fromhex
938toHex = float.hex
939class HexFloatTestCase(unittest.TestCase):
940 MAX = fromHex('0x.fffffffffffff8p+1024') # max normal
941 MIN = fromHex('0x1p-1022') # min normal
942 TINY = fromHex('0x0.0000000000001p-1022') # min subnormal
943 EPS = fromHex('0x0.0000000000001p0') # diff between 1.0 and next float up
944
945 def identical(self, x, y):
946 # check that floats x and y are identical, or that both
947 # are NaNs
948 if isnan(x) or isnan(y):
949 if isnan(x) == isnan(y):
950 return
951 elif x == y and (x != 0.0 or copysign(1.0, x) == copysign(1.0, y)):
952 return
953 self.fail('%r not identical to %r' % (x, y))
954
955 def test_ends(self):
Mark Dickinson38bbc482008-07-16 11:32:23 +0000956 self.identical(self.MIN, ldexp(1.0, -1022))
957 self.identical(self.TINY, ldexp(1.0, -1074))
958 self.identical(self.EPS, ldexp(1.0, -52))
959 self.identical(self.MAX, 2.*(ldexp(1.0, 1023) - ldexp(1.0, 970)))
Mark Dickinson65fe25e2008-07-16 11:30:51 +0000960
961 def test_invalid_inputs(self):
962 invalid_inputs = [
963 'infi', # misspelt infinities and nans
964 '-Infinit',
965 '++inf',
966 '-+Inf',
967 '--nan',
968 '+-NaN',
969 'snan',
970 'NaNs',
971 'nna',
Mark Dickinsond1ec8b22009-05-11 15:45:15 +0000972 'an',
973 'nf',
974 'nfinity',
975 'inity',
976 'iinity',
Mark Dickinson65fe25e2008-07-16 11:30:51 +0000977 '0xnan',
978 '',
979 ' ',
980 'x1.0p0',
981 '0xX1.0p0',
982 '+ 0x1.0p0', # internal whitespace
983 '- 0x1.0p0',
984 '0 x1.0p0',
985 '0x 1.0p0',
986 '0x1 2.0p0',
987 '+0x1 .0p0',
988 '0x1. 0p0',
989 '-0x1.0 1p0',
990 '-0x1.0 p0',
991 '+0x1.0p +0',
992 '0x1.0p -0',
993 '0x1.0p 0',
994 '+0x1.0p+ 0',
995 '-0x1.0p- 0',
996 '++0x1.0p-0', # double signs
997 '--0x1.0p0',
998 '+-0x1.0p+0',
999 '-+0x1.0p0',
1000 '0x1.0p++0',
1001 '+0x1.0p+-0',
1002 '-0x1.0p-+0',
1003 '0x1.0p--0',
1004 '0x1.0.p0',
1005 '0x.p0', # no hex digits before or after point
1006 '0x1,p0', # wrong decimal point character
1007 '0x1pa',
1008 '0x1p\uff10', # fullwidth Unicode digits
1009 '\uff10x1p0',
1010 '0x\uff11p0',
1011 '0x1.\uff10p0',
1012 '0x1p0 \n 0x2p0',
1013 '0x1p0\0 0x1p0', # embedded null byte is not end of string
1014 ]
1015 for x in invalid_inputs:
Mark Dickinson589b7952008-08-21 20:05:56 +00001016 try:
1017 result = fromHex(x)
1018 except ValueError:
1019 pass
1020 else:
1021 self.fail('Expected float.fromhex(%r) to raise ValueError; '
1022 'got %r instead' % (x, result))
Mark Dickinson65fe25e2008-07-16 11:30:51 +00001023
1024
Mark Dickinsond1ec8b22009-05-11 15:45:15 +00001025 def test_whitespace(self):
1026 value_pairs = [
1027 ('inf', INF),
1028 ('-Infinity', -INF),
1029 ('nan', NAN),
1030 ('1.0', 1.0),
1031 ('-0x.2', -0.125),
1032 ('-0.0', -0.0)
1033 ]
1034 whitespace = [
1035 '',
1036 ' ',
1037 '\t',
1038 '\n',
1039 '\n \t',
1040 '\f',
1041 '\v',
1042 '\r'
1043 ]
1044 for inp, expected in value_pairs:
1045 for lead in whitespace:
1046 for trail in whitespace:
1047 got = fromHex(lead + inp + trail)
1048 self.identical(got, expected)
1049
1050
Mark Dickinson65fe25e2008-07-16 11:30:51 +00001051 def test_from_hex(self):
1052 MIN = self.MIN;
1053 MAX = self.MAX;
1054 TINY = self.TINY;
1055 EPS = self.EPS;
1056
1057 # two spellings of infinity, with optional signs; case-insensitive
1058 self.identical(fromHex('inf'), INF)
1059 self.identical(fromHex('+Inf'), INF)
1060 self.identical(fromHex('-INF'), -INF)
1061 self.identical(fromHex('iNf'), INF)
1062 self.identical(fromHex('Infinity'), INF)
1063 self.identical(fromHex('+INFINITY'), INF)
1064 self.identical(fromHex('-infinity'), -INF)
1065 self.identical(fromHex('-iNFiNitY'), -INF)
1066
1067 # nans with optional sign; case insensitive
1068 self.identical(fromHex('nan'), NAN)
1069 self.identical(fromHex('+NaN'), NAN)
1070 self.identical(fromHex('-NaN'), NAN)
1071 self.identical(fromHex('-nAN'), NAN)
1072
1073 # variations in input format
1074 self.identical(fromHex('1'), 1.0)
1075 self.identical(fromHex('+1'), 1.0)
1076 self.identical(fromHex('1.'), 1.0)
1077 self.identical(fromHex('1.0'), 1.0)
1078 self.identical(fromHex('1.0p0'), 1.0)
1079 self.identical(fromHex('01'), 1.0)
1080 self.identical(fromHex('01.'), 1.0)
1081 self.identical(fromHex('0x1'), 1.0)
1082 self.identical(fromHex('0x1.'), 1.0)
1083 self.identical(fromHex('0x1.0'), 1.0)
1084 self.identical(fromHex('+0x1.0'), 1.0)
1085 self.identical(fromHex('0x1p0'), 1.0)
1086 self.identical(fromHex('0X1p0'), 1.0)
1087 self.identical(fromHex('0X1P0'), 1.0)
1088 self.identical(fromHex('0x1P0'), 1.0)
1089 self.identical(fromHex('0x1.p0'), 1.0)
1090 self.identical(fromHex('0x1.0p0'), 1.0)
1091 self.identical(fromHex('0x.1p4'), 1.0)
1092 self.identical(fromHex('0x.1p04'), 1.0)
1093 self.identical(fromHex('0x.1p004'), 1.0)
1094 self.identical(fromHex('0x1p+0'), 1.0)
1095 self.identical(fromHex('0x1P-0'), 1.0)
1096 self.identical(fromHex('+0x1p0'), 1.0)
1097 self.identical(fromHex('0x01p0'), 1.0)
1098 self.identical(fromHex('0x1p00'), 1.0)
1099 self.identical(fromHex(' 0x1p0 '), 1.0)
1100 self.identical(fromHex('\n 0x1p0'), 1.0)
1101 self.identical(fromHex('0x1p0 \t'), 1.0)
1102 self.identical(fromHex('0xap0'), 10.0)
1103 self.identical(fromHex('0xAp0'), 10.0)
1104 self.identical(fromHex('0xaP0'), 10.0)
1105 self.identical(fromHex('0xAP0'), 10.0)
1106 self.identical(fromHex('0xbep0'), 190.0)
1107 self.identical(fromHex('0xBep0'), 190.0)
1108 self.identical(fromHex('0xbEp0'), 190.0)
1109 self.identical(fromHex('0XBE0P-4'), 190.0)
1110 self.identical(fromHex('0xBEp0'), 190.0)
1111 self.identical(fromHex('0xB.Ep4'), 190.0)
1112 self.identical(fromHex('0x.BEp8'), 190.0)
1113 self.identical(fromHex('0x.0BEp12'), 190.0)
1114
1115 # moving the point around
1116 pi = fromHex('0x1.921fb54442d18p1')
1117 self.identical(fromHex('0x.006487ed5110b46p11'), pi)
1118 self.identical(fromHex('0x.00c90fdaa22168cp10'), pi)
1119 self.identical(fromHex('0x.01921fb54442d18p9'), pi)
1120 self.identical(fromHex('0x.03243f6a8885a3p8'), pi)
1121 self.identical(fromHex('0x.06487ed5110b46p7'), pi)
1122 self.identical(fromHex('0x.0c90fdaa22168cp6'), pi)
1123 self.identical(fromHex('0x.1921fb54442d18p5'), pi)
1124 self.identical(fromHex('0x.3243f6a8885a3p4'), pi)
1125 self.identical(fromHex('0x.6487ed5110b46p3'), pi)
1126 self.identical(fromHex('0x.c90fdaa22168cp2'), pi)
1127 self.identical(fromHex('0x1.921fb54442d18p1'), pi)
1128 self.identical(fromHex('0x3.243f6a8885a3p0'), pi)
1129 self.identical(fromHex('0x6.487ed5110b46p-1'), pi)
1130 self.identical(fromHex('0xc.90fdaa22168cp-2'), pi)
1131 self.identical(fromHex('0x19.21fb54442d18p-3'), pi)
1132 self.identical(fromHex('0x32.43f6a8885a3p-4'), pi)
1133 self.identical(fromHex('0x64.87ed5110b46p-5'), pi)
1134 self.identical(fromHex('0xc9.0fdaa22168cp-6'), pi)
1135 self.identical(fromHex('0x192.1fb54442d18p-7'), pi)
1136 self.identical(fromHex('0x324.3f6a8885a3p-8'), pi)
1137 self.identical(fromHex('0x648.7ed5110b46p-9'), pi)
1138 self.identical(fromHex('0xc90.fdaa22168cp-10'), pi)
1139 self.identical(fromHex('0x1921.fb54442d18p-11'), pi)
1140 # ...
1141 self.identical(fromHex('0x1921fb54442d1.8p-47'), pi)
1142 self.identical(fromHex('0x3243f6a8885a3p-48'), pi)
1143 self.identical(fromHex('0x6487ed5110b46p-49'), pi)
1144 self.identical(fromHex('0xc90fdaa22168cp-50'), pi)
1145 self.identical(fromHex('0x1921fb54442d18p-51'), pi)
1146 self.identical(fromHex('0x3243f6a8885a30p-52'), pi)
1147 self.identical(fromHex('0x6487ed5110b460p-53'), pi)
1148 self.identical(fromHex('0xc90fdaa22168c0p-54'), pi)
1149 self.identical(fromHex('0x1921fb54442d180p-55'), pi)
1150
1151
1152 # results that should overflow...
1153 self.assertRaises(OverflowError, fromHex, '-0x1p1024')
1154 self.assertRaises(OverflowError, fromHex, '0x1p+1025')
1155 self.assertRaises(OverflowError, fromHex, '+0X1p1030')
1156 self.assertRaises(OverflowError, fromHex, '-0x1p+1100')
1157 self.assertRaises(OverflowError, fromHex, '0X1p123456789123456789')
1158 self.assertRaises(OverflowError, fromHex, '+0X.8p+1025')
1159 self.assertRaises(OverflowError, fromHex, '+0x0.8p1025')
1160 self.assertRaises(OverflowError, fromHex, '-0x0.4p1026')
1161 self.assertRaises(OverflowError, fromHex, '0X2p+1023')
1162 self.assertRaises(OverflowError, fromHex, '0x2.p1023')
1163 self.assertRaises(OverflowError, fromHex, '-0x2.0p+1023')
1164 self.assertRaises(OverflowError, fromHex, '+0X4p+1022')
1165 self.assertRaises(OverflowError, fromHex, '0x1.ffffffffffffffp+1023')
1166 self.assertRaises(OverflowError, fromHex, '-0X1.fffffffffffff9p1023')
1167 self.assertRaises(OverflowError, fromHex, '0X1.fffffffffffff8p1023')
1168 self.assertRaises(OverflowError, fromHex, '+0x3.fffffffffffffp1022')
1169 self.assertRaises(OverflowError, fromHex, '0x3fffffffffffffp+970')
1170 self.assertRaises(OverflowError, fromHex, '0x10000000000000000p960')
1171 self.assertRaises(OverflowError, fromHex, '-0Xffffffffffffffffp960')
1172
1173 # ...and those that round to +-max float
1174 self.identical(fromHex('+0x1.fffffffffffffp+1023'), MAX)
1175 self.identical(fromHex('-0X1.fffffffffffff7p1023'), -MAX)
1176 self.identical(fromHex('0X1.fffffffffffff7fffffffffffffp1023'), MAX)
1177
1178 # zeros
1179 self.identical(fromHex('0x0p0'), 0.0)
1180 self.identical(fromHex('0x0p1000'), 0.0)
1181 self.identical(fromHex('-0x0p1023'), -0.0)
1182 self.identical(fromHex('0X0p1024'), 0.0)
1183 self.identical(fromHex('-0x0p1025'), -0.0)
1184 self.identical(fromHex('0X0p2000'), 0.0)
1185 self.identical(fromHex('0x0p123456789123456789'), 0.0)
1186 self.identical(fromHex('-0X0p-0'), -0.0)
1187 self.identical(fromHex('-0X0p-1000'), -0.0)
1188 self.identical(fromHex('0x0p-1023'), 0.0)
1189 self.identical(fromHex('-0X0p-1024'), -0.0)
1190 self.identical(fromHex('-0x0p-1025'), -0.0)
1191 self.identical(fromHex('-0x0p-1072'), -0.0)
1192 self.identical(fromHex('0X0p-1073'), 0.0)
1193 self.identical(fromHex('-0x0p-1074'), -0.0)
1194 self.identical(fromHex('0x0p-1075'), 0.0)
1195 self.identical(fromHex('0X0p-1076'), 0.0)
1196 self.identical(fromHex('-0X0p-2000'), -0.0)
1197 self.identical(fromHex('-0x0p-123456789123456789'), -0.0)
1198
1199 # values that should underflow to 0
1200 self.identical(fromHex('0X1p-1075'), 0.0)
1201 self.identical(fromHex('-0X1p-1075'), -0.0)
1202 self.identical(fromHex('-0x1p-123456789123456789'), -0.0)
1203 self.identical(fromHex('0x1.00000000000000001p-1075'), TINY)
1204 self.identical(fromHex('-0x1.1p-1075'), -TINY)
1205 self.identical(fromHex('0x1.fffffffffffffffffp-1075'), TINY)
1206
1207 # check round-half-even is working correctly near 0 ...
1208 self.identical(fromHex('0x1p-1076'), 0.0)
1209 self.identical(fromHex('0X2p-1076'), 0.0)
1210 self.identical(fromHex('0X3p-1076'), TINY)
1211 self.identical(fromHex('0x4p-1076'), TINY)
1212 self.identical(fromHex('0X5p-1076'), TINY)
1213 self.identical(fromHex('0X6p-1076'), 2*TINY)
1214 self.identical(fromHex('0x7p-1076'), 2*TINY)
1215 self.identical(fromHex('0X8p-1076'), 2*TINY)
1216 self.identical(fromHex('0X9p-1076'), 2*TINY)
1217 self.identical(fromHex('0xap-1076'), 2*TINY)
1218 self.identical(fromHex('0Xbp-1076'), 3*TINY)
1219 self.identical(fromHex('0xcp-1076'), 3*TINY)
1220 self.identical(fromHex('0Xdp-1076'), 3*TINY)
1221 self.identical(fromHex('0Xep-1076'), 4*TINY)
1222 self.identical(fromHex('0xfp-1076'), 4*TINY)
1223 self.identical(fromHex('0x10p-1076'), 4*TINY)
1224 self.identical(fromHex('-0x1p-1076'), -0.0)
1225 self.identical(fromHex('-0X2p-1076'), -0.0)
1226 self.identical(fromHex('-0x3p-1076'), -TINY)
1227 self.identical(fromHex('-0X4p-1076'), -TINY)
1228 self.identical(fromHex('-0x5p-1076'), -TINY)
1229 self.identical(fromHex('-0x6p-1076'), -2*TINY)
1230 self.identical(fromHex('-0X7p-1076'), -2*TINY)
1231 self.identical(fromHex('-0X8p-1076'), -2*TINY)
1232 self.identical(fromHex('-0X9p-1076'), -2*TINY)
1233 self.identical(fromHex('-0Xap-1076'), -2*TINY)
1234 self.identical(fromHex('-0xbp-1076'), -3*TINY)
1235 self.identical(fromHex('-0xcp-1076'), -3*TINY)
1236 self.identical(fromHex('-0Xdp-1076'), -3*TINY)
1237 self.identical(fromHex('-0xep-1076'), -4*TINY)
1238 self.identical(fromHex('-0Xfp-1076'), -4*TINY)
1239 self.identical(fromHex('-0X10p-1076'), -4*TINY)
1240
1241 # ... and near MIN ...
1242 self.identical(fromHex('0x0.ffffffffffffd6p-1022'), MIN-3*TINY)
1243 self.identical(fromHex('0x0.ffffffffffffd8p-1022'), MIN-2*TINY)
1244 self.identical(fromHex('0x0.ffffffffffffdap-1022'), MIN-2*TINY)
1245 self.identical(fromHex('0x0.ffffffffffffdcp-1022'), MIN-2*TINY)
1246 self.identical(fromHex('0x0.ffffffffffffdep-1022'), MIN-2*TINY)
1247 self.identical(fromHex('0x0.ffffffffffffe0p-1022'), MIN-2*TINY)
1248 self.identical(fromHex('0x0.ffffffffffffe2p-1022'), MIN-2*TINY)
1249 self.identical(fromHex('0x0.ffffffffffffe4p-1022'), MIN-2*TINY)
1250 self.identical(fromHex('0x0.ffffffffffffe6p-1022'), MIN-2*TINY)
1251 self.identical(fromHex('0x0.ffffffffffffe8p-1022'), MIN-2*TINY)
1252 self.identical(fromHex('0x0.ffffffffffffeap-1022'), MIN-TINY)
1253 self.identical(fromHex('0x0.ffffffffffffecp-1022'), MIN-TINY)
1254 self.identical(fromHex('0x0.ffffffffffffeep-1022'), MIN-TINY)
1255 self.identical(fromHex('0x0.fffffffffffff0p-1022'), MIN-TINY)
1256 self.identical(fromHex('0x0.fffffffffffff2p-1022'), MIN-TINY)
1257 self.identical(fromHex('0x0.fffffffffffff4p-1022'), MIN-TINY)
1258 self.identical(fromHex('0x0.fffffffffffff6p-1022'), MIN-TINY)
1259 self.identical(fromHex('0x0.fffffffffffff8p-1022'), MIN)
1260 self.identical(fromHex('0x0.fffffffffffffap-1022'), MIN)
1261 self.identical(fromHex('0x0.fffffffffffffcp-1022'), MIN)
1262 self.identical(fromHex('0x0.fffffffffffffep-1022'), MIN)
1263 self.identical(fromHex('0x1.00000000000000p-1022'), MIN)
1264 self.identical(fromHex('0x1.00000000000002p-1022'), MIN)
1265 self.identical(fromHex('0x1.00000000000004p-1022'), MIN)
1266 self.identical(fromHex('0x1.00000000000006p-1022'), MIN)
1267 self.identical(fromHex('0x1.00000000000008p-1022'), MIN)
1268 self.identical(fromHex('0x1.0000000000000ap-1022'), MIN+TINY)
1269 self.identical(fromHex('0x1.0000000000000cp-1022'), MIN+TINY)
1270 self.identical(fromHex('0x1.0000000000000ep-1022'), MIN+TINY)
1271 self.identical(fromHex('0x1.00000000000010p-1022'), MIN+TINY)
1272 self.identical(fromHex('0x1.00000000000012p-1022'), MIN+TINY)
1273 self.identical(fromHex('0x1.00000000000014p-1022'), MIN+TINY)
1274 self.identical(fromHex('0x1.00000000000016p-1022'), MIN+TINY)
1275 self.identical(fromHex('0x1.00000000000018p-1022'), MIN+2*TINY)
1276
1277 # ... and near 1.0.
1278 self.identical(fromHex('0x0.fffffffffffff0p0'), 1.0-EPS)
1279 self.identical(fromHex('0x0.fffffffffffff1p0'), 1.0-EPS)
1280 self.identical(fromHex('0X0.fffffffffffff2p0'), 1.0-EPS)
1281 self.identical(fromHex('0x0.fffffffffffff3p0'), 1.0-EPS)
1282 self.identical(fromHex('0X0.fffffffffffff4p0'), 1.0-EPS)
1283 self.identical(fromHex('0X0.fffffffffffff5p0'), 1.0-EPS/2)
1284 self.identical(fromHex('0X0.fffffffffffff6p0'), 1.0-EPS/2)
1285 self.identical(fromHex('0x0.fffffffffffff7p0'), 1.0-EPS/2)
1286 self.identical(fromHex('0x0.fffffffffffff8p0'), 1.0-EPS/2)
1287 self.identical(fromHex('0X0.fffffffffffff9p0'), 1.0-EPS/2)
1288 self.identical(fromHex('0X0.fffffffffffffap0'), 1.0-EPS/2)
1289 self.identical(fromHex('0x0.fffffffffffffbp0'), 1.0-EPS/2)
1290 self.identical(fromHex('0X0.fffffffffffffcp0'), 1.0)
1291 self.identical(fromHex('0x0.fffffffffffffdp0'), 1.0)
1292 self.identical(fromHex('0X0.fffffffffffffep0'), 1.0)
1293 self.identical(fromHex('0x0.ffffffffffffffp0'), 1.0)
1294 self.identical(fromHex('0X1.00000000000000p0'), 1.0)
1295 self.identical(fromHex('0X1.00000000000001p0'), 1.0)
1296 self.identical(fromHex('0x1.00000000000002p0'), 1.0)
1297 self.identical(fromHex('0X1.00000000000003p0'), 1.0)
1298 self.identical(fromHex('0x1.00000000000004p0'), 1.0)
1299 self.identical(fromHex('0X1.00000000000005p0'), 1.0)
1300 self.identical(fromHex('0X1.00000000000006p0'), 1.0)
1301 self.identical(fromHex('0X1.00000000000007p0'), 1.0)
1302 self.identical(fromHex('0x1.00000000000007ffffffffffffffffffffp0'),
1303 1.0)
1304 self.identical(fromHex('0x1.00000000000008p0'), 1.0)
1305 self.identical(fromHex('0x1.00000000000008000000000000000001p0'),
1306 1+EPS)
1307 self.identical(fromHex('0X1.00000000000009p0'), 1.0+EPS)
1308 self.identical(fromHex('0x1.0000000000000ap0'), 1.0+EPS)
1309 self.identical(fromHex('0x1.0000000000000bp0'), 1.0+EPS)
1310 self.identical(fromHex('0X1.0000000000000cp0'), 1.0+EPS)
1311 self.identical(fromHex('0x1.0000000000000dp0'), 1.0+EPS)
1312 self.identical(fromHex('0x1.0000000000000ep0'), 1.0+EPS)
1313 self.identical(fromHex('0X1.0000000000000fp0'), 1.0+EPS)
1314 self.identical(fromHex('0x1.00000000000010p0'), 1.0+EPS)
1315 self.identical(fromHex('0X1.00000000000011p0'), 1.0+EPS)
1316 self.identical(fromHex('0x1.00000000000012p0'), 1.0+EPS)
1317 self.identical(fromHex('0X1.00000000000013p0'), 1.0+EPS)
1318 self.identical(fromHex('0X1.00000000000014p0'), 1.0+EPS)
1319 self.identical(fromHex('0x1.00000000000015p0'), 1.0+EPS)
1320 self.identical(fromHex('0x1.00000000000016p0'), 1.0+EPS)
1321 self.identical(fromHex('0X1.00000000000017p0'), 1.0+EPS)
1322 self.identical(fromHex('0x1.00000000000017ffffffffffffffffffffp0'),
1323 1.0+EPS)
1324 self.identical(fromHex('0x1.00000000000018p0'), 1.0+2*EPS)
1325 self.identical(fromHex('0X1.00000000000018000000000000000001p0'),
1326 1.0+2*EPS)
1327 self.identical(fromHex('0x1.00000000000019p0'), 1.0+2*EPS)
1328 self.identical(fromHex('0X1.0000000000001ap0'), 1.0+2*EPS)
1329 self.identical(fromHex('0X1.0000000000001bp0'), 1.0+2*EPS)
1330 self.identical(fromHex('0x1.0000000000001cp0'), 1.0+2*EPS)
1331 self.identical(fromHex('0x1.0000000000001dp0'), 1.0+2*EPS)
1332 self.identical(fromHex('0x1.0000000000001ep0'), 1.0+2*EPS)
1333 self.identical(fromHex('0X1.0000000000001fp0'), 1.0+2*EPS)
1334 self.identical(fromHex('0x1.00000000000020p0'), 1.0+2*EPS)
1335
1336 def test_roundtrip(self):
1337 def roundtrip(x):
1338 return fromHex(toHex(x))
1339
1340 for x in [NAN, INF, self.MAX, self.MIN, self.MIN-self.TINY, self.TINY, 0.0]:
1341 self.identical(x, roundtrip(x))
1342 self.identical(-x, roundtrip(-x))
1343
1344 # fromHex(toHex(x)) should exactly recover x, for any non-NaN float x.
1345 import random
1346 for i in range(10000):
1347 e = random.randrange(-1200, 1200)
1348 m = random.random()
1349 s = random.choice([1.0, -1.0])
1350 try:
1351 x = s*ldexp(m, e)
1352 except OverflowError:
1353 pass
1354 else:
1355 self.identical(x, fromHex(toHex(x)))
1356
Christian Heimes53876d92008-04-19 00:31:39 +00001357
Michael W. Hudsonba283e22005-05-27 15:23:20 +00001358if __name__ == '__main__':
Zachary Ware38c707e2015-04-13 15:00:43 -05001359 unittest.main()