blob: 91e3b7ab49d4d27d0fec421021417ea09ef9edd5 [file] [log] [blame]
Benjamin Peterson101447b2015-03-09 10:37:50 -04001import fractions
Christian Heimes53876d92008-04-19 00:31:39 +00002import operator
Benjamin Peterson101447b2015-03-09 10:37:50 -04003import os
4import random
5import sys
6import struct
Benjamin Petersona9157232015-03-06 09:08:44 -05007import time
Benjamin Peterson101447b2015-03-09 10:37:50 -04008import unittest
9
10from test import support
Brett Cannona721aba2016-09-09 14:57:09 -070011from test.test_grammar import (VALID_UNDERSCORE_LITERALS,
12 INVALID_UNDERSCORE_LITERALS)
Benjamin Peterson101447b2015-03-09 10:37:50 -040013from math import isinf, isnan, copysign, ldexp
Michael W. Hudsonba283e22005-05-27 15:23:20 +000014
Christian Heimes53876d92008-04-19 00:31:39 +000015INF = float("inf")
16NAN = float("nan")
Christian Heimes99170a52007-12-19 02:07:34 +000017
Benjamin Petersone401c682010-07-02 23:25:44 +000018have_getformat = hasattr(float, "__getformat__")
19requires_getformat = unittest.skipUnless(have_getformat,
20 "requires __getformat__")
21requires_setformat = unittest.skipUnless(hasattr(float, "__setformat__"),
22 "requires __setformat__")
Mark Dickinson9ab44b52009-12-30 16:22:49 +000023
Eric Smith0923d1d2009-04-16 20:16:10 +000024#locate file with float format test values
25test_dir = os.path.dirname(__file__) or os.curdir
26format_testfile = os.path.join(test_dir, 'formatfloat_testcases.txt')
27
Serhiy Storchaka15095802015-11-25 15:47:01 +020028class FloatSubclass(float):
29 pass
30
31class OtherFloatSubclass(float):
32 pass
33
Christian Heimes81ee3ef2008-05-04 22:42:01 +000034class GeneralFloatCases(unittest.TestCase):
35
36 def test_float(self):
37 self.assertEqual(float(3.14), 3.14)
38 self.assertEqual(float(314), 314.0)
39 self.assertEqual(float(" 3.14 "), 3.14)
40 self.assertRaises(ValueError, float, " 0x3.1 ")
41 self.assertRaises(ValueError, float, " -0x3.p-1 ")
42 self.assertRaises(ValueError, float, " +0x3.p-1 ")
43 self.assertRaises(ValueError, float, "++3.14")
44 self.assertRaises(ValueError, float, "+-3.14")
45 self.assertRaises(ValueError, float, "-+3.14")
46 self.assertRaises(ValueError, float, "--3.14")
Eric Smith0923d1d2009-04-16 20:16:10 +000047 self.assertRaises(ValueError, float, ".nan")
48 self.assertRaises(ValueError, float, "+.inf")
49 self.assertRaises(ValueError, float, ".")
50 self.assertRaises(ValueError, float, "-.")
Alexander Belopolsky942af5a2010-12-04 03:38:46 +000051 self.assertRaises(TypeError, float, {})
Ezio Melottia5b95992013-11-07 19:18:34 +020052 self.assertRaisesRegex(TypeError, "not 'dict'", float, {})
Alexander Belopolsky942af5a2010-12-04 03:38:46 +000053 # Lone surrogate
54 self.assertRaises(UnicodeEncodeError, float, '\uD8F0')
Mark Dickinsonc2d86892010-02-12 21:18:34 +000055 # check that we don't accept alternate exponent markers
56 self.assertRaises(ValueError, float, "-1.7d29")
57 self.assertRaises(ValueError, float, "3D-14")
Alexander Belopolsky942af5a2010-12-04 03:38:46 +000058 self.assertEqual(float(" \u0663.\u0661\u0664 "), 3.14)
59 self.assertEqual(float("\N{EM SPACE}3.14\N{EN SPACE}"), 3.14)
Mark Dickinsona9023be2009-10-27 22:12:20 +000060 # extra long strings should not be a problem
61 float(b'.' + b'1'*1000)
62 float('.' + '1'*1000)
Christian Heimes81ee3ef2008-05-04 22:42:01 +000063
Brett Cannona721aba2016-09-09 14:57:09 -070064 def test_underscores(self):
65 for lit in VALID_UNDERSCORE_LITERALS:
66 if not any(ch in lit for ch in 'jJxXoObB'):
67 self.assertEqual(float(lit), eval(lit))
68 self.assertEqual(float(lit), float(lit.replace('_', '')))
69 for lit in INVALID_UNDERSCORE_LITERALS:
70 if lit in ('0_7', '09_99'): # octals are not recognized here
71 continue
72 if not any(ch in lit for ch in 'jJxXoObB'):
73 self.assertRaises(ValueError, float, lit)
74 # Additional test cases; nan and inf are never valid as literals,
75 # only in the float() constructor, but we don't allow underscores
76 # in or around them.
77 self.assertRaises(ValueError, float, '_NaN')
78 self.assertRaises(ValueError, float, 'Na_N')
79 self.assertRaises(ValueError, float, 'IN_F')
80 self.assertRaises(ValueError, float, '-_INF')
81 self.assertRaises(ValueError, float, '-INF_')
82 # Check that we handle bytes values correctly.
83 self.assertRaises(ValueError, float, b'0_.\xff9')
84
Martin Pantereeb896c2015-11-07 02:32:21 +000085 def test_non_numeric_input_types(self):
86 # Test possible non-numeric types for the argument x, including
87 # subclasses of the explicitly documented accepted types.
88 class CustomStr(str): pass
89 class CustomBytes(bytes): pass
90 class CustomByteArray(bytearray): pass
91
92 factories = [
93 bytes,
94 bytearray,
95 lambda b: CustomStr(b.decode()),
96 CustomBytes,
97 CustomByteArray,
98 memoryview,
99 ]
100 try:
101 from array import array
102 except ImportError:
103 pass
104 else:
105 factories.append(lambda b: array('B', b))
106
107 for f in factories:
108 x = f(b" 3.14 ")
109 with self.subTest(type(x)):
110 self.assertEqual(float(x), 3.14)
111 with self.assertRaisesRegex(ValueError, "could not convert"):
112 float(f(b'A' * 0x10))
113
114 def test_float_memoryview(self):
115 self.assertEqual(float(memoryview(b'12.3')[1:4]), 2.3)
116 self.assertEqual(float(memoryview(b'12.3\x00')[1:4]), 2.3)
117 self.assertEqual(float(memoryview(b'12.3 ')[1:4]), 2.3)
118 self.assertEqual(float(memoryview(b'12.3A')[1:4]), 2.3)
119 self.assertEqual(float(memoryview(b'12.34')[1:4]), 2.3)
120
Alexander Belopolsky942af5a2010-12-04 03:38:46 +0000121 def test_error_message(self):
122 testlist = ('\xbd', '123\xbd', ' 123 456 ')
123 for s in testlist:
124 try:
125 float(s)
126 except ValueError as e:
127 self.assertIn(s.strip(), e.args[0])
128 else:
129 self.fail("Expected int(%r) to raise a ValueError", s)
130
131
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000132 @support.run_with_locale('LC_NUMERIC', 'fr_FR', 'de_DE')
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000133 def test_float_with_comma(self):
134 # set locale to something that doesn't use '.' for the decimal point
135 # float must not accept the locale specific decimal point but
Ezio Melotti13925002011-03-16 11:05:33 +0200136 # it still has to accept the normal python syntax
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000137 import locale
138 if not locale.localeconv()['decimal_point'] == ',':
Zachary Ware9fe6d862013-12-08 00:20:35 -0600139 self.skipTest('decimal_point is not ","')
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000140
141 self.assertEqual(float(" 3.14 "), 3.14)
142 self.assertEqual(float("+3.14 "), 3.14)
143 self.assertEqual(float("-3.14 "), -3.14)
144 self.assertEqual(float(".14 "), .14)
145 self.assertEqual(float("3. "), 3.0)
146 self.assertEqual(float("3.e3 "), 3000.0)
147 self.assertEqual(float("3.2e3 "), 3200.0)
148 self.assertEqual(float("2.5e-1 "), 0.25)
149 self.assertEqual(float("5e-1"), 0.5)
150 self.assertRaises(ValueError, float, " 3,14 ")
151 self.assertRaises(ValueError, float, " +3,14 ")
152 self.assertRaises(ValueError, float, " -3,14 ")
153 self.assertRaises(ValueError, float, " 0x3.1 ")
154 self.assertRaises(ValueError, float, " -0x3.p-1 ")
155 self.assertRaises(ValueError, float, " +0x3.p-1 ")
156 self.assertEqual(float(" 25.e-1 "), 2.5)
Eli Bendersky67ebabd2011-02-25 10:14:17 +0000157 self.assertAlmostEqual(float(" .25e-1 "), .025)
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000158
159 def test_floatconversion(self):
160 # Make sure that calls to __float__() work properly
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000161 class Foo1(object):
162 def __float__(self):
163 return 42.
164
165 class Foo2(float):
166 def __float__(self):
167 return 42.
168
169 class Foo3(float):
170 def __new__(cls, value=0.):
171 return float.__new__(cls, 2*value)
172
173 def __float__(self):
174 return self
175
176 class Foo4(float):
177 def __float__(self):
178 return 42
179
Benjamin Peterson2808d3c2009-04-15 21:34:27 +0000180 # Issue 5759: __float__ not called on str subclasses (though it is on
181 # unicode subclasses).
182 class FooStr(str):
183 def __float__(self):
184 return float(str(self)) + 1
185
Serhiy Storchaka16931c32016-06-03 21:42:55 +0300186 self.assertEqual(float(Foo1()), 42.)
187 self.assertEqual(float(Foo2()), 42.)
188 with self.assertWarns(DeprecationWarning):
189 self.assertEqual(float(Foo3(21)), 42.)
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000190 self.assertRaises(TypeError, float, Foo4(42))
Serhiy Storchaka16931c32016-06-03 21:42:55 +0300191 self.assertEqual(float(FooStr('8')), 9.)
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000192
Benjamin Petersona9157232015-03-06 09:08:44 -0500193 class Foo5:
194 def __float__(self):
195 return ""
196 self.assertRaises(TypeError, time.sleep, Foo5())
197
Serhiy Storchaka15095802015-11-25 15:47:01 +0200198 # Issue #24731
199 class F:
200 def __float__(self):
201 return OtherFloatSubclass(42.)
Serhiy Storchaka16931c32016-06-03 21:42:55 +0300202 with self.assertWarns(DeprecationWarning):
203 self.assertEqual(float(F()), 42.)
204 with self.assertWarns(DeprecationWarning):
205 self.assertIs(type(float(F())), float)
206 with self.assertWarns(DeprecationWarning):
207 self.assertEqual(FloatSubclass(F()), 42.)
208 with self.assertWarns(DeprecationWarning):
209 self.assertIs(type(FloatSubclass(F())), FloatSubclass)
Serhiy Storchaka15095802015-11-25 15:47:01 +0200210
Serhiy Storchaka58d23e62017-03-06 00:53:39 +0200211 def test_keyword_args(self):
212 with self.assertWarns(DeprecationWarning):
213 self.assertEqual(float(x='3.14'), 3.14)
214
Benjamin Petersonb3b8ba62011-10-28 19:42:48 -0400215 def test_is_integer(self):
216 self.assertFalse((1.1).is_integer())
217 self.assertTrue((1.).is_integer())
218 self.assertFalse(float("nan").is_integer())
219 self.assertFalse(float("inf").is_integer())
220
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000221 def test_floatasratio(self):
222 for f, ratio in [
223 (0.875, (7, 8)),
224 (-0.875, (-7, 8)),
225 (0.0, (0, 1)),
226 (11.5, (23, 2)),
227 ]:
228 self.assertEqual(f.as_integer_ratio(), ratio)
229
230 for i in range(10000):
231 f = random.random()
232 f *= 10 ** random.randint(-100, 100)
233 n, d = f.as_integer_ratio()
234 self.assertEqual(float(n).__truediv__(d), f)
235
236 R = fractions.Fraction
237 self.assertEqual(R(0, 1),
238 R(*float(0.0).as_integer_ratio()))
239 self.assertEqual(R(5, 2),
240 R(*float(2.5).as_integer_ratio()))
241 self.assertEqual(R(1, 2),
242 R(*float(0.5).as_integer_ratio()))
243 self.assertEqual(R(4728779608739021, 2251799813685248),
244 R(*float(2.1).as_integer_ratio()))
245 self.assertEqual(R(-4728779608739021, 2251799813685248),
246 R(*float(-2.1).as_integer_ratio()))
247 self.assertEqual(R(-2100, 1),
248 R(*float(-2100.0).as_integer_ratio()))
249
250 self.assertRaises(OverflowError, float('inf').as_integer_ratio)
251 self.assertRaises(OverflowError, float('-inf').as_integer_ratio)
252 self.assertRaises(ValueError, float('nan').as_integer_ratio)
253
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000254 def test_float_containment(self):
255 floats = (INF, -INF, 0.0, 1.0, NAN)
256 for f in floats:
Benjamin Peterson577473f2010-01-19 00:09:57 +0000257 self.assertIn(f, [f])
Benjamin Peterson577473f2010-01-19 00:09:57 +0000258 self.assertIn(f, (f,))
Benjamin Peterson577473f2010-01-19 00:09:57 +0000259 self.assertIn(f, {f})
Benjamin Peterson577473f2010-01-19 00:09:57 +0000260 self.assertIn(f, {f: None})
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000261 self.assertEqual([f].count(f), 1, "[].count('%r') != 1" % f)
Benjamin Peterson577473f2010-01-19 00:09:57 +0000262 self.assertIn(f, floats)
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000263
264 for f in floats:
265 # nonidentical containers, same type, same contents
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000266 self.assertTrue([f] == [f], "[%r] != [%r]" % (f, f))
267 self.assertTrue((f,) == (f,), "(%r,) != (%r,)" % (f, f))
268 self.assertTrue({f} == {f}, "{%r} != {%r}" % (f, f))
269 self.assertTrue({f : None} == {f: None}, "{%r : None} != "
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000270 "{%r : None}" % (f, f))
271
272 # identical containers
273 l, t, s, d = [f], (f,), {f}, {f: None}
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000274 self.assertTrue(l == l, "[%r] not equal to itself" % f)
275 self.assertTrue(t == t, "(%r,) not equal to itself" % f)
276 self.assertTrue(s == s, "{%r} not equal to itself" % f)
277 self.assertTrue(d == d, "{%r : None} not equal to itself" % f)
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000278
Mark Dickinson9ab44b52009-12-30 16:22:49 +0000279 def assertEqualAndEqualSign(self, a, b):
280 # fail unless a == b and a and b have the same sign bit;
281 # the only difference from assertEqual is that this test
Ezio Melotti13925002011-03-16 11:05:33 +0200282 # distinguishes -0.0 and 0.0.
Mark Dickinson9ab44b52009-12-30 16:22:49 +0000283 self.assertEqual((a, copysign(1.0, a)), (b, copysign(1.0, b)))
284
Eric Smith3ab08ca2010-12-04 15:17:38 +0000285 @support.requires_IEEE_754
Mark Dickinsond2a9b202010-12-04 12:25:30 +0000286 def test_float_mod(self):
287 # Check behaviour of % operator for IEEE 754 special cases.
288 # In particular, check signs of zeros.
289 mod = operator.mod
290
291 self.assertEqualAndEqualSign(mod(-1.0, 1.0), 0.0)
292 self.assertEqualAndEqualSign(mod(-1e-100, 1.0), 1.0)
293 self.assertEqualAndEqualSign(mod(-0.0, 1.0), 0.0)
294 self.assertEqualAndEqualSign(mod(0.0, 1.0), 0.0)
295 self.assertEqualAndEqualSign(mod(1e-100, 1.0), 1e-100)
296 self.assertEqualAndEqualSign(mod(1.0, 1.0), 0.0)
297
298 self.assertEqualAndEqualSign(mod(-1.0, -1.0), -0.0)
299 self.assertEqualAndEqualSign(mod(-1e-100, -1.0), -1e-100)
300 self.assertEqualAndEqualSign(mod(-0.0, -1.0), -0.0)
301 self.assertEqualAndEqualSign(mod(0.0, -1.0), -0.0)
302 self.assertEqualAndEqualSign(mod(1e-100, -1.0), -1.0)
303 self.assertEqualAndEqualSign(mod(1.0, -1.0), -0.0)
304
Eric Smith3ab08ca2010-12-04 15:17:38 +0000305 @support.requires_IEEE_754
Mark Dickinson9ab44b52009-12-30 16:22:49 +0000306 def test_float_pow(self):
307 # test builtin pow and ** operator for IEEE 754 special cases.
308 # Special cases taken from section F.9.4.4 of the C99 specification
309
310 for pow_op in pow, operator.pow:
311 # x**NAN is NAN for any x except 1
312 self.assertTrue(isnan(pow_op(-INF, NAN)))
313 self.assertTrue(isnan(pow_op(-2.0, NAN)))
314 self.assertTrue(isnan(pow_op(-1.0, NAN)))
315 self.assertTrue(isnan(pow_op(-0.5, NAN)))
316 self.assertTrue(isnan(pow_op(-0.0, NAN)))
317 self.assertTrue(isnan(pow_op(0.0, NAN)))
318 self.assertTrue(isnan(pow_op(0.5, NAN)))
319 self.assertTrue(isnan(pow_op(2.0, NAN)))
320 self.assertTrue(isnan(pow_op(INF, NAN)))
321 self.assertTrue(isnan(pow_op(NAN, NAN)))
322
323 # NAN**y is NAN for any y except +-0
324 self.assertTrue(isnan(pow_op(NAN, -INF)))
325 self.assertTrue(isnan(pow_op(NAN, -2.0)))
326 self.assertTrue(isnan(pow_op(NAN, -1.0)))
327 self.assertTrue(isnan(pow_op(NAN, -0.5)))
328 self.assertTrue(isnan(pow_op(NAN, 0.5)))
329 self.assertTrue(isnan(pow_op(NAN, 1.0)))
330 self.assertTrue(isnan(pow_op(NAN, 2.0)))
331 self.assertTrue(isnan(pow_op(NAN, INF)))
332
333 # (+-0)**y raises ZeroDivisionError for y a negative odd integer
334 self.assertRaises(ZeroDivisionError, pow_op, -0.0, -1.0)
335 self.assertRaises(ZeroDivisionError, pow_op, 0.0, -1.0)
336
337 # (+-0)**y raises ZeroDivisionError for y finite and negative
338 # but not an odd integer
339 self.assertRaises(ZeroDivisionError, pow_op, -0.0, -2.0)
340 self.assertRaises(ZeroDivisionError, pow_op, -0.0, -0.5)
341 self.assertRaises(ZeroDivisionError, pow_op, 0.0, -2.0)
342 self.assertRaises(ZeroDivisionError, pow_op, 0.0, -0.5)
343
344 # (+-0)**y is +-0 for y a positive odd integer
345 self.assertEqualAndEqualSign(pow_op(-0.0, 1.0), -0.0)
346 self.assertEqualAndEqualSign(pow_op(0.0, 1.0), 0.0)
347
348 # (+-0)**y is 0 for y finite and positive but not an odd integer
349 self.assertEqualAndEqualSign(pow_op(-0.0, 0.5), 0.0)
350 self.assertEqualAndEqualSign(pow_op(-0.0, 2.0), 0.0)
351 self.assertEqualAndEqualSign(pow_op(0.0, 0.5), 0.0)
352 self.assertEqualAndEqualSign(pow_op(0.0, 2.0), 0.0)
353
354 # (-1)**+-inf is 1
355 self.assertEqualAndEqualSign(pow_op(-1.0, -INF), 1.0)
356 self.assertEqualAndEqualSign(pow_op(-1.0, INF), 1.0)
357
358 # 1**y is 1 for any y, even if y is an infinity or nan
359 self.assertEqualAndEqualSign(pow_op(1.0, -INF), 1.0)
360 self.assertEqualAndEqualSign(pow_op(1.0, -2.0), 1.0)
361 self.assertEqualAndEqualSign(pow_op(1.0, -1.0), 1.0)
362 self.assertEqualAndEqualSign(pow_op(1.0, -0.5), 1.0)
363 self.assertEqualAndEqualSign(pow_op(1.0, -0.0), 1.0)
364 self.assertEqualAndEqualSign(pow_op(1.0, 0.0), 1.0)
365 self.assertEqualAndEqualSign(pow_op(1.0, 0.5), 1.0)
366 self.assertEqualAndEqualSign(pow_op(1.0, 1.0), 1.0)
367 self.assertEqualAndEqualSign(pow_op(1.0, 2.0), 1.0)
368 self.assertEqualAndEqualSign(pow_op(1.0, INF), 1.0)
369 self.assertEqualAndEqualSign(pow_op(1.0, NAN), 1.0)
370
371 # x**+-0 is 1 for any x, even if x is a zero, infinity, or nan
372 self.assertEqualAndEqualSign(pow_op(-INF, 0.0), 1.0)
373 self.assertEqualAndEqualSign(pow_op(-2.0, 0.0), 1.0)
374 self.assertEqualAndEqualSign(pow_op(-1.0, 0.0), 1.0)
375 self.assertEqualAndEqualSign(pow_op(-0.5, 0.0), 1.0)
376 self.assertEqualAndEqualSign(pow_op(-0.0, 0.0), 1.0)
377 self.assertEqualAndEqualSign(pow_op(0.0, 0.0), 1.0)
378 self.assertEqualAndEqualSign(pow_op(0.5, 0.0), 1.0)
379 self.assertEqualAndEqualSign(pow_op(1.0, 0.0), 1.0)
380 self.assertEqualAndEqualSign(pow_op(2.0, 0.0), 1.0)
381 self.assertEqualAndEqualSign(pow_op(INF, 0.0), 1.0)
382 self.assertEqualAndEqualSign(pow_op(NAN, 0.0), 1.0)
383 self.assertEqualAndEqualSign(pow_op(-INF, -0.0), 1.0)
384 self.assertEqualAndEqualSign(pow_op(-2.0, -0.0), 1.0)
385 self.assertEqualAndEqualSign(pow_op(-1.0, -0.0), 1.0)
386 self.assertEqualAndEqualSign(pow_op(-0.5, -0.0), 1.0)
387 self.assertEqualAndEqualSign(pow_op(-0.0, -0.0), 1.0)
388 self.assertEqualAndEqualSign(pow_op(0.0, -0.0), 1.0)
389 self.assertEqualAndEqualSign(pow_op(0.5, -0.0), 1.0)
390 self.assertEqualAndEqualSign(pow_op(1.0, -0.0), 1.0)
391 self.assertEqualAndEqualSign(pow_op(2.0, -0.0), 1.0)
392 self.assertEqualAndEqualSign(pow_op(INF, -0.0), 1.0)
393 self.assertEqualAndEqualSign(pow_op(NAN, -0.0), 1.0)
394
395 # x**y defers to complex pow for finite negative x and
396 # non-integral y.
397 self.assertEqual(type(pow_op(-2.0, -0.5)), complex)
398 self.assertEqual(type(pow_op(-2.0, 0.5)), complex)
399 self.assertEqual(type(pow_op(-1.0, -0.5)), complex)
400 self.assertEqual(type(pow_op(-1.0, 0.5)), complex)
401 self.assertEqual(type(pow_op(-0.5, -0.5)), complex)
402 self.assertEqual(type(pow_op(-0.5, 0.5)), complex)
403
404 # x**-INF is INF for abs(x) < 1
405 self.assertEqualAndEqualSign(pow_op(-0.5, -INF), INF)
406 self.assertEqualAndEqualSign(pow_op(-0.0, -INF), INF)
407 self.assertEqualAndEqualSign(pow_op(0.0, -INF), INF)
408 self.assertEqualAndEqualSign(pow_op(0.5, -INF), INF)
409
410 # x**-INF is 0 for abs(x) > 1
411 self.assertEqualAndEqualSign(pow_op(-INF, -INF), 0.0)
412 self.assertEqualAndEqualSign(pow_op(-2.0, -INF), 0.0)
413 self.assertEqualAndEqualSign(pow_op(2.0, -INF), 0.0)
414 self.assertEqualAndEqualSign(pow_op(INF, -INF), 0.0)
415
416 # x**INF is 0 for abs(x) < 1
417 self.assertEqualAndEqualSign(pow_op(-0.5, INF), 0.0)
418 self.assertEqualAndEqualSign(pow_op(-0.0, INF), 0.0)
419 self.assertEqualAndEqualSign(pow_op(0.0, INF), 0.0)
420 self.assertEqualAndEqualSign(pow_op(0.5, INF), 0.0)
421
422 # x**INF is INF for abs(x) > 1
423 self.assertEqualAndEqualSign(pow_op(-INF, INF), INF)
424 self.assertEqualAndEqualSign(pow_op(-2.0, INF), INF)
425 self.assertEqualAndEqualSign(pow_op(2.0, INF), INF)
426 self.assertEqualAndEqualSign(pow_op(INF, INF), INF)
427
428 # (-INF)**y is -0.0 for y a negative odd integer
429 self.assertEqualAndEqualSign(pow_op(-INF, -1.0), -0.0)
430
431 # (-INF)**y is 0.0 for y negative but not an odd integer
432 self.assertEqualAndEqualSign(pow_op(-INF, -0.5), 0.0)
433 self.assertEqualAndEqualSign(pow_op(-INF, -2.0), 0.0)
434
435 # (-INF)**y is -INF for y a positive odd integer
436 self.assertEqualAndEqualSign(pow_op(-INF, 1.0), -INF)
437
438 # (-INF)**y is INF for y positive but not an odd integer
439 self.assertEqualAndEqualSign(pow_op(-INF, 0.5), INF)
440 self.assertEqualAndEqualSign(pow_op(-INF, 2.0), INF)
441
442 # INF**y is INF for y positive
443 self.assertEqualAndEqualSign(pow_op(INF, 0.5), INF)
444 self.assertEqualAndEqualSign(pow_op(INF, 1.0), INF)
445 self.assertEqualAndEqualSign(pow_op(INF, 2.0), INF)
446
447 # INF**y is 0.0 for y negative
448 self.assertEqualAndEqualSign(pow_op(INF, -2.0), 0.0)
449 self.assertEqualAndEqualSign(pow_op(INF, -1.0), 0.0)
450 self.assertEqualAndEqualSign(pow_op(INF, -0.5), 0.0)
451
452 # basic checks not covered by the special cases above
453 self.assertEqualAndEqualSign(pow_op(-2.0, -2.0), 0.25)
454 self.assertEqualAndEqualSign(pow_op(-2.0, -1.0), -0.5)
455 self.assertEqualAndEqualSign(pow_op(-2.0, -0.0), 1.0)
456 self.assertEqualAndEqualSign(pow_op(-2.0, 0.0), 1.0)
457 self.assertEqualAndEqualSign(pow_op(-2.0, 1.0), -2.0)
458 self.assertEqualAndEqualSign(pow_op(-2.0, 2.0), 4.0)
459 self.assertEqualAndEqualSign(pow_op(-1.0, -2.0), 1.0)
460 self.assertEqualAndEqualSign(pow_op(-1.0, -1.0), -1.0)
461 self.assertEqualAndEqualSign(pow_op(-1.0, -0.0), 1.0)
462 self.assertEqualAndEqualSign(pow_op(-1.0, 0.0), 1.0)
463 self.assertEqualAndEqualSign(pow_op(-1.0, 1.0), -1.0)
464 self.assertEqualAndEqualSign(pow_op(-1.0, 2.0), 1.0)
465 self.assertEqualAndEqualSign(pow_op(2.0, -2.0), 0.25)
466 self.assertEqualAndEqualSign(pow_op(2.0, -1.0), 0.5)
467 self.assertEqualAndEqualSign(pow_op(2.0, -0.0), 1.0)
468 self.assertEqualAndEqualSign(pow_op(2.0, 0.0), 1.0)
469 self.assertEqualAndEqualSign(pow_op(2.0, 1.0), 2.0)
470 self.assertEqualAndEqualSign(pow_op(2.0, 2.0), 4.0)
471
472 # 1 ** large and -1 ** large; some libms apparently
473 # have problems with these
474 self.assertEqualAndEqualSign(pow_op(1.0, -1e100), 1.0)
475 self.assertEqualAndEqualSign(pow_op(1.0, 1e100), 1.0)
476 self.assertEqualAndEqualSign(pow_op(-1.0, -1e100), 1.0)
477 self.assertEqualAndEqualSign(pow_op(-1.0, 1e100), 1.0)
478
479 # check sign for results that underflow to 0
480 self.assertEqualAndEqualSign(pow_op(-2.0, -2000.0), 0.0)
481 self.assertEqual(type(pow_op(-2.0, -2000.5)), complex)
482 self.assertEqualAndEqualSign(pow_op(-2.0, -2001.0), -0.0)
483 self.assertEqualAndEqualSign(pow_op(2.0, -2000.0), 0.0)
484 self.assertEqualAndEqualSign(pow_op(2.0, -2000.5), 0.0)
485 self.assertEqualAndEqualSign(pow_op(2.0, -2001.0), 0.0)
486 self.assertEqualAndEqualSign(pow_op(-0.5, 2000.0), 0.0)
487 self.assertEqual(type(pow_op(-0.5, 2000.5)), complex)
488 self.assertEqualAndEqualSign(pow_op(-0.5, 2001.0), -0.0)
489 self.assertEqualAndEqualSign(pow_op(0.5, 2000.0), 0.0)
490 self.assertEqualAndEqualSign(pow_op(0.5, 2000.5), 0.0)
491 self.assertEqualAndEqualSign(pow_op(0.5, 2001.0), 0.0)
492
493 # check we don't raise an exception for subnormal results,
494 # and validate signs. Tests currently disabled, since
495 # they fail on systems where a subnormal result from pow
496 # is flushed to zero (e.g. Debian/ia64.)
497 #self.assertTrue(0.0 < pow_op(0.5, 1048) < 1e-315)
498 #self.assertTrue(0.0 < pow_op(-0.5, 1048) < 1e-315)
499 #self.assertTrue(0.0 < pow_op(0.5, 1047) < 1e-315)
500 #self.assertTrue(0.0 > pow_op(-0.5, 1047) > -1e-315)
501 #self.assertTrue(0.0 < pow_op(2.0, -1048) < 1e-315)
502 #self.assertTrue(0.0 < pow_op(-2.0, -1048) < 1e-315)
503 #self.assertTrue(0.0 < pow_op(2.0, -1047) < 1e-315)
504 #self.assertTrue(0.0 > pow_op(-2.0, -1047) > -1e-315)
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000505
506
Benjamin Petersone401c682010-07-02 23:25:44 +0000507@requires_setformat
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000508class FormatFunctionsTestCase(unittest.TestCase):
509
510 def setUp(self):
511 self.save_formats = {'double':float.__getformat__('double'),
512 'float':float.__getformat__('float')}
513
514 def tearDown(self):
515 float.__setformat__('double', self.save_formats['double'])
516 float.__setformat__('float', self.save_formats['float'])
517
518 def test_getformat(self):
Ezio Melottib58e0bd2010-01-23 15:40:09 +0000519 self.assertIn(float.__getformat__('double'),
520 ['unknown', 'IEEE, big-endian', 'IEEE, little-endian'])
521 self.assertIn(float.__getformat__('float'),
522 ['unknown', 'IEEE, big-endian', 'IEEE, little-endian'])
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000523 self.assertRaises(ValueError, float.__getformat__, 'chicken')
524 self.assertRaises(TypeError, float.__getformat__, 1)
525
526 def test_setformat(self):
527 for t in 'double', 'float':
528 float.__setformat__(t, 'unknown')
529 if self.save_formats[t] == 'IEEE, big-endian':
530 self.assertRaises(ValueError, float.__setformat__,
531 t, 'IEEE, little-endian')
532 elif self.save_formats[t] == 'IEEE, little-endian':
533 self.assertRaises(ValueError, float.__setformat__,
534 t, 'IEEE, big-endian')
535 else:
536 self.assertRaises(ValueError, float.__setformat__,
537 t, 'IEEE, big-endian')
538 self.assertRaises(ValueError, float.__setformat__,
539 t, 'IEEE, little-endian')
540 self.assertRaises(ValueError, float.__setformat__,
541 t, 'chicken')
542 self.assertRaises(ValueError, float.__setformat__,
543 'chicken', 'unknown')
544
Guido van Rossum2be161d2007-05-15 20:43:51 +0000545BE_DOUBLE_INF = b'\x7f\xf0\x00\x00\x00\x00\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000546LE_DOUBLE_INF = bytes(reversed(BE_DOUBLE_INF))
Guido van Rossum2be161d2007-05-15 20:43:51 +0000547BE_DOUBLE_NAN = b'\x7f\xf8\x00\x00\x00\x00\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000548LE_DOUBLE_NAN = bytes(reversed(BE_DOUBLE_NAN))
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000549
Guido van Rossum2be161d2007-05-15 20:43:51 +0000550BE_FLOAT_INF = b'\x7f\x80\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000551LE_FLOAT_INF = bytes(reversed(BE_FLOAT_INF))
Guido van Rossum2be161d2007-05-15 20:43:51 +0000552BE_FLOAT_NAN = b'\x7f\xc0\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000553LE_FLOAT_NAN = bytes(reversed(BE_FLOAT_NAN))
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000554
555# on non-IEEE platforms, attempting to unpack a bit pattern
556# representing an infinity or a NaN should raise an exception.
557
Benjamin Petersone401c682010-07-02 23:25:44 +0000558@requires_setformat
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000559class UnknownFormatTestCase(unittest.TestCase):
560 def setUp(self):
561 self.save_formats = {'double':float.__getformat__('double'),
562 'float':float.__getformat__('float')}
563 float.__setformat__('double', 'unknown')
564 float.__setformat__('float', 'unknown')
Tim Peters5d36a552005-06-03 22:40:27 +0000565
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000566 def tearDown(self):
567 float.__setformat__('double', self.save_formats['double'])
568 float.__setformat__('float', self.save_formats['float'])
569
570 def test_double_specials_dont_unpack(self):
571 for fmt, data in [('>d', BE_DOUBLE_INF),
572 ('>d', BE_DOUBLE_NAN),
573 ('<d', LE_DOUBLE_INF),
574 ('<d', LE_DOUBLE_NAN)]:
575 self.assertRaises(ValueError, struct.unpack, fmt, data)
576
577 def test_float_specials_dont_unpack(self):
578 for fmt, data in [('>f', BE_FLOAT_INF),
579 ('>f', BE_FLOAT_NAN),
580 ('<f', LE_FLOAT_INF),
581 ('<f', LE_FLOAT_NAN)]:
582 self.assertRaises(ValueError, struct.unpack, fmt, data)
583
584
585# on an IEEE platform, all we guarantee is that bit patterns
586# representing infinities or NaNs do not raise an exception; all else
587# is accident (today).
Guido van Rossum04110fb2007-08-24 16:32:05 +0000588# let's also try to guarantee that -0.0 and 0.0 don't get confused.
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000589
590class IEEEFormatTestCase(unittest.TestCase):
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000591
Eric Smith3ab08ca2010-12-04 15:17:38 +0000592 @support.requires_IEEE_754
Benjamin Petersone401c682010-07-02 23:25:44 +0000593 def test_double_specials_do_unpack(self):
594 for fmt, data in [('>d', BE_DOUBLE_INF),
595 ('>d', BE_DOUBLE_NAN),
596 ('<d', LE_DOUBLE_INF),
597 ('<d', LE_DOUBLE_NAN)]:
598 struct.unpack(fmt, data)
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000599
Eric Smith3ab08ca2010-12-04 15:17:38 +0000600 @support.requires_IEEE_754
Benjamin Petersone401c682010-07-02 23:25:44 +0000601 def test_float_specials_do_unpack(self):
602 for fmt, data in [('>f', BE_FLOAT_INF),
603 ('>f', BE_FLOAT_NAN),
604 ('<f', LE_FLOAT_INF),
605 ('<f', LE_FLOAT_NAN)]:
606 struct.unpack(fmt, data)
Guido van Rossum04110fb2007-08-24 16:32:05 +0000607
Eric Smith8c663262007-08-25 02:26:07 +0000608class FormatTestCase(unittest.TestCase):
Benjamin Petersone401c682010-07-02 23:25:44 +0000609
Eric Smith11fe3e02007-08-31 01:33:06 +0000610 def test_format(self):
Eric Smith8c663262007-08-25 02:26:07 +0000611 # these should be rewritten to use both format(x, spec) and
612 # x.__format__(spec)
613
614 self.assertEqual(format(0.0, 'f'), '0.000000')
615
616 # the default is 'g', except for empty format spec
617 self.assertEqual(format(0.0, ''), '0.0')
618 self.assertEqual(format(0.01, ''), '0.01')
619 self.assertEqual(format(0.01, 'g'), '0.01')
620
Eric Smith63376222009-05-05 14:04:18 +0000621 # empty presentation type should format in the same way as str
622 # (issue 5920)
623 x = 100/7.
624 self.assertEqual(format(x, ''), str(x))
625 self.assertEqual(format(x, '-'), str(x))
626 self.assertEqual(format(x, '>'), str(x))
627 self.assertEqual(format(x, '2'), str(x))
Eric Smith8c663262007-08-25 02:26:07 +0000628
629 self.assertEqual(format(1.0, 'f'), '1.000000')
Eric Smith8c663262007-08-25 02:26:07 +0000630
631 self.assertEqual(format(-1.0, 'f'), '-1.000000')
Eric Smith8c663262007-08-25 02:26:07 +0000632
633 self.assertEqual(format( 1.0, ' f'), ' 1.000000')
634 self.assertEqual(format(-1.0, ' f'), '-1.000000')
635 self.assertEqual(format( 1.0, '+f'), '+1.000000')
636 self.assertEqual(format(-1.0, '+f'), '-1.000000')
637
638 # % formatting
639 self.assertEqual(format(-1.0, '%'), '-100.000000%')
640
641 # conversion to string should fail
642 self.assertRaises(ValueError, format, 3.0, "s")
643
Eric Smith7b69c6c2008-01-27 21:07:59 +0000644 # other format specifiers shouldn't work on floats,
645 # in particular int specifiers
646 for format_spec in ([chr(x) for x in range(ord('a'), ord('z')+1)] +
647 [chr(x) for x in range(ord('A'), ord('Z')+1)]):
648 if not format_spec in 'eEfFgGn%':
649 self.assertRaises(ValueError, format, 0.0, format_spec)
650 self.assertRaises(ValueError, format, 1.0, format_spec)
651 self.assertRaises(ValueError, format, -1.0, format_spec)
652 self.assertRaises(ValueError, format, 1e100, format_spec)
653 self.assertRaises(ValueError, format, -1e100, format_spec)
654 self.assertRaises(ValueError, format, 1e-100, format_spec)
655 self.assertRaises(ValueError, format, -1e-100, format_spec)
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000656
Eric Smith741191f2009-05-06 13:08:15 +0000657 # issue 3382
658 self.assertEqual(format(NAN, 'f'), 'nan')
659 self.assertEqual(format(NAN, 'F'), 'NAN')
660 self.assertEqual(format(INF, 'f'), 'inf')
661 self.assertEqual(format(INF, 'F'), 'INF')
662
Eric Smith3ab08ca2010-12-04 15:17:38 +0000663 @support.requires_IEEE_754
Eric Smith0923d1d2009-04-16 20:16:10 +0000664 def test_format_testfile(self):
Brian Curtin076623b2010-10-31 00:03:45 +0000665 with open(format_testfile) as testfile:
666 for line in testfile:
667 if line.startswith('--'):
668 continue
669 line = line.strip()
670 if not line:
671 continue
Eric Smith0923d1d2009-04-16 20:16:10 +0000672
Brian Curtin076623b2010-10-31 00:03:45 +0000673 lhs, rhs = map(str.strip, line.split('->'))
674 fmt, arg = lhs.split()
675 self.assertEqual(fmt % float(arg), rhs)
676 self.assertEqual(fmt % -float(arg), '-' + rhs)
Eric Smith0923d1d2009-04-16 20:16:10 +0000677
Mark Dickinsond3ca5572009-04-29 18:47:07 +0000678 def test_issue5864(self):
Ezio Melottib3aedd42010-11-20 19:04:17 +0000679 self.assertEqual(format(123.456, '.4'), '123.5')
680 self.assertEqual(format(1234.56, '.4'), '1.235e+03')
681 self.assertEqual(format(12345.6, '.4'), '1.235e+04')
Mark Dickinsond3ca5572009-04-29 18:47:07 +0000682
Mark Dickinson7efad9e2009-04-17 20:59:58 +0000683class ReprTestCase(unittest.TestCase):
684 def test_repr(self):
685 floats_file = open(os.path.join(os.path.split(__file__)[0],
686 'floating_points.txt'))
687 for line in floats_file:
688 line = line.strip()
689 if not line or line.startswith('#'):
690 continue
691 v = eval(line)
692 self.assertEqual(v, eval(repr(v)))
693 floats_file.close()
694
Eric Smith0923d1d2009-04-16 20:16:10 +0000695 @unittest.skipUnless(getattr(sys, 'float_repr_style', '') == 'short',
696 "applies only when using short float repr style")
697 def test_short_repr(self):
698 # test short float repr introduced in Python 3.1. One aspect
699 # of this repr is that we get some degree of str -> float ->
700 # str roundtripping. In particular, for any numeric string
701 # containing 15 or fewer significant digits, those exact same
702 # digits (modulo trailing zeros) should appear in the output.
703 # No more repr(0.03) -> "0.029999999999999999"!
704
705 test_strings = [
706 # output always includes *either* a decimal point and at
707 # least one digit after that point, or an exponent.
708 '0.0',
709 '1.0',
710 '0.01',
711 '0.02',
712 '0.03',
713 '0.04',
714 '0.05',
715 '1.23456789',
716 '10.0',
717 '100.0',
718 # values >= 1e16 get an exponent...
719 '1000000000000000.0',
720 '9999999999999990.0',
721 '1e+16',
722 '1e+17',
723 # ... and so do values < 1e-4
724 '0.001',
725 '0.001001',
726 '0.00010000000000001',
727 '0.0001',
728 '9.999999999999e-05',
729 '1e-05',
730 # values designed to provoke failure if the FPU rounding
731 # precision isn't set correctly
732 '8.72293771110361e+25',
733 '7.47005307342313e+26',
734 '2.86438000439698e+28',
735 '8.89142905246179e+28',
736 '3.08578087079232e+35',
737 ]
738
739 for s in test_strings:
740 negs = '-'+s
741 self.assertEqual(s, repr(float(s)))
742 self.assertEqual(negs, repr(float(negs)))
Mark Dickinson388122d2010-08-04 20:56:28 +0000743 # Since Python 3.2, repr and str are identical
744 self.assertEqual(repr(float(s)), str(float(s)))
745 self.assertEqual(repr(float(negs)), str(float(negs)))
Benjamin Petersone401c682010-07-02 23:25:44 +0000746
Eric Smith3ab08ca2010-12-04 15:17:38 +0000747@support.requires_IEEE_754
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000748class RoundTestCase(unittest.TestCase):
Benjamin Petersone401c682010-07-02 23:25:44 +0000749
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000750 def test_inf_nan(self):
751 self.assertRaises(OverflowError, round, INF)
752 self.assertRaises(OverflowError, round, -INF)
753 self.assertRaises(ValueError, round, NAN)
Mark Dickinson4ca33d12009-11-24 10:59:34 +0000754 self.assertRaises(TypeError, round, INF, 0.0)
755 self.assertRaises(TypeError, round, -INF, 1.0)
756 self.assertRaises(TypeError, round, NAN, "ceci n'est pas un integer")
757 self.assertRaises(TypeError, round, -0.0, 1j)
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000758
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000759 def test_large_n(self):
760 for n in [324, 325, 400, 2**31-1, 2**31, 2**32, 2**100]:
761 self.assertEqual(round(123.456, n), 123.456)
762 self.assertEqual(round(-123.456, n), -123.456)
763 self.assertEqual(round(1e300, n), 1e300)
764 self.assertEqual(round(1e-320, n), 1e-320)
765 self.assertEqual(round(1e150, 300), 1e150)
766 self.assertEqual(round(1e300, 307), 1e300)
767 self.assertEqual(round(-3.1415, 308), -3.1415)
768 self.assertEqual(round(1e150, 309), 1e150)
769 self.assertEqual(round(1.4e-315, 315), 1e-315)
770
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000771 def test_small_n(self):
772 for n in [-308, -309, -400, 1-2**31, -2**31, -2**31-1, -2**100]:
773 self.assertEqual(round(123.456, n), 0.0)
774 self.assertEqual(round(-123.456, n), -0.0)
775 self.assertEqual(round(1e300, n), 0.0)
776 self.assertEqual(round(1e-320, n), 0.0)
777
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000778 def test_overflow(self):
779 self.assertRaises(OverflowError, round, 1.6e308, -308)
780 self.assertRaises(OverflowError, round, -1.7e308, -308)
781
782 @unittest.skipUnless(getattr(sys, 'float_repr_style', '') == 'short',
783 "applies only when using short float repr style")
784 def test_previous_round_bugs(self):
785 # particular cases that have occurred in bug reports
786 self.assertEqual(round(562949953421312.5, 1),
787 562949953421312.5)
788 self.assertEqual(round(56294995342131.5, 3),
789 56294995342131.5)
790 # round-half-even
791 self.assertEqual(round(25.0, -1), 20.0)
792 self.assertEqual(round(35.0, -1), 40.0)
793 self.assertEqual(round(45.0, -1), 40.0)
794 self.assertEqual(round(55.0, -1), 60.0)
795 self.assertEqual(round(65.0, -1), 60.0)
796 self.assertEqual(round(75.0, -1), 80.0)
797 self.assertEqual(round(85.0, -1), 80.0)
798 self.assertEqual(round(95.0, -1), 100.0)
799
800 @unittest.skipUnless(getattr(sys, 'float_repr_style', '') == 'short',
801 "applies only when using short float repr style")
802 def test_matches_float_format(self):
803 # round should give the same results as float formatting
804 for i in range(500):
805 x = i/1000.
806 self.assertEqual(float(format(x, '.0f')), round(x, 0))
807 self.assertEqual(float(format(x, '.1f')), round(x, 1))
808 self.assertEqual(float(format(x, '.2f')), round(x, 2))
809 self.assertEqual(float(format(x, '.3f')), round(x, 3))
810
811 for i in range(5, 5000, 10):
812 x = i/1000.
813 self.assertEqual(float(format(x, '.0f')), round(x, 0))
814 self.assertEqual(float(format(x, '.1f')), round(x, 1))
815 self.assertEqual(float(format(x, '.2f')), round(x, 2))
816 self.assertEqual(float(format(x, '.3f')), round(x, 3))
817
818 for i in range(500):
819 x = random.random()
820 self.assertEqual(float(format(x, '.0f')), round(x, 0))
821 self.assertEqual(float(format(x, '.1f')), round(x, 1))
822 self.assertEqual(float(format(x, '.2f')), round(x, 2))
823 self.assertEqual(float(format(x, '.3f')), round(x, 3))
824
Eric Smith8a10ecc2009-12-02 17:58:24 +0000825 def test_format_specials(self):
826 # Test formatting of nans and infs.
827
828 def test(fmt, value, expected):
829 # Test with both % and format().
830 self.assertEqual(fmt % value, expected, fmt)
Eric Smith984bb582010-11-25 16:08:06 +0000831 fmt = fmt[1:] # strip off the %
832 self.assertEqual(format(value, fmt), expected, fmt)
Eric Smith8a10ecc2009-12-02 17:58:24 +0000833
834 for fmt in ['%e', '%f', '%g', '%.0e', '%.6f', '%.20g',
835 '%#e', '%#f', '%#g', '%#.20e', '%#.15f', '%#.3g']:
836 pfmt = '%+' + fmt[1:]
837 sfmt = '% ' + fmt[1:]
838 test(fmt, INF, 'inf')
839 test(fmt, -INF, '-inf')
840 test(fmt, NAN, 'nan')
841 test(fmt, -NAN, 'nan')
842 # When asking for a sign, it's always provided. nans are
843 # always positive.
844 test(pfmt, INF, '+inf')
845 test(pfmt, -INF, '-inf')
846 test(pfmt, NAN, '+nan')
847 test(pfmt, -NAN, '+nan')
848 # When using ' ' for a sign code, only infs can be negative.
849 # Others have a space.
850 test(sfmt, INF, ' inf')
851 test(sfmt, -INF, '-inf')
852 test(sfmt, NAN, ' nan')
853 test(sfmt, -NAN, ' nan')
854
Steve Dowercb39d1f2015-04-15 16:10:59 -0400855 def test_None_ndigits(self):
856 for x in round(1.23), round(1.23, None), round(1.23, ndigits=None):
857 self.assertEqual(x, 1)
858 self.assertIsInstance(x, int)
859 for x in round(1.78), round(1.78, None), round(1.78, ndigits=None):
860 self.assertEqual(x, 2)
861 self.assertIsInstance(x, int)
862
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000863
Christian Heimes99170a52007-12-19 02:07:34 +0000864# Beginning with Python 2.6 float has cross platform compatible
Georg Brandl2ee470f2008-07-16 12:55:28 +0000865# ways to create and represent inf and nan
Christian Heimes99170a52007-12-19 02:07:34 +0000866class InfNanTest(unittest.TestCase):
867 def test_inf_from_str(self):
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000868 self.assertTrue(isinf(float("inf")))
869 self.assertTrue(isinf(float("+inf")))
870 self.assertTrue(isinf(float("-inf")))
871 self.assertTrue(isinf(float("infinity")))
872 self.assertTrue(isinf(float("+infinity")))
873 self.assertTrue(isinf(float("-infinity")))
Christian Heimes99170a52007-12-19 02:07:34 +0000874
875 self.assertEqual(repr(float("inf")), "inf")
876 self.assertEqual(repr(float("+inf")), "inf")
877 self.assertEqual(repr(float("-inf")), "-inf")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000878 self.assertEqual(repr(float("infinity")), "inf")
879 self.assertEqual(repr(float("+infinity")), "inf")
880 self.assertEqual(repr(float("-infinity")), "-inf")
Christian Heimes99170a52007-12-19 02:07:34 +0000881
882 self.assertEqual(repr(float("INF")), "inf")
883 self.assertEqual(repr(float("+Inf")), "inf")
884 self.assertEqual(repr(float("-iNF")), "-inf")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000885 self.assertEqual(repr(float("Infinity")), "inf")
886 self.assertEqual(repr(float("+iNfInItY")), "inf")
887 self.assertEqual(repr(float("-INFINITY")), "-inf")
Christian Heimes99170a52007-12-19 02:07:34 +0000888
889 self.assertEqual(str(float("inf")), "inf")
890 self.assertEqual(str(float("+inf")), "inf")
891 self.assertEqual(str(float("-inf")), "-inf")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000892 self.assertEqual(str(float("infinity")), "inf")
893 self.assertEqual(str(float("+infinity")), "inf")
894 self.assertEqual(str(float("-infinity")), "-inf")
Christian Heimes99170a52007-12-19 02:07:34 +0000895
896 self.assertRaises(ValueError, float, "info")
897 self.assertRaises(ValueError, float, "+info")
898 self.assertRaises(ValueError, float, "-info")
899 self.assertRaises(ValueError, float, "in")
900 self.assertRaises(ValueError, float, "+in")
901 self.assertRaises(ValueError, float, "-in")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000902 self.assertRaises(ValueError, float, "infinit")
903 self.assertRaises(ValueError, float, "+Infin")
904 self.assertRaises(ValueError, float, "-INFI")
905 self.assertRaises(ValueError, float, "infinitys")
Christian Heimes99170a52007-12-19 02:07:34 +0000906
Mark Dickinsonbd16edd2009-05-20 22:05:25 +0000907 self.assertRaises(ValueError, float, "++Inf")
908 self.assertRaises(ValueError, float, "-+inf")
909 self.assertRaises(ValueError, float, "+-infinity")
910 self.assertRaises(ValueError, float, "--Infinity")
911
Christian Heimes99170a52007-12-19 02:07:34 +0000912 def test_inf_as_str(self):
913 self.assertEqual(repr(1e300 * 1e300), "inf")
914 self.assertEqual(repr(-1e300 * 1e300), "-inf")
915
916 self.assertEqual(str(1e300 * 1e300), "inf")
917 self.assertEqual(str(-1e300 * 1e300), "-inf")
918
919 def test_nan_from_str(self):
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000920 self.assertTrue(isnan(float("nan")))
921 self.assertTrue(isnan(float("+nan")))
922 self.assertTrue(isnan(float("-nan")))
Christian Heimes99170a52007-12-19 02:07:34 +0000923
924 self.assertEqual(repr(float("nan")), "nan")
925 self.assertEqual(repr(float("+nan")), "nan")
926 self.assertEqual(repr(float("-nan")), "nan")
927
928 self.assertEqual(repr(float("NAN")), "nan")
929 self.assertEqual(repr(float("+NAn")), "nan")
930 self.assertEqual(repr(float("-NaN")), "nan")
931
932 self.assertEqual(str(float("nan")), "nan")
933 self.assertEqual(str(float("+nan")), "nan")
934 self.assertEqual(str(float("-nan")), "nan")
935
936 self.assertRaises(ValueError, float, "nana")
937 self.assertRaises(ValueError, float, "+nana")
938 self.assertRaises(ValueError, float, "-nana")
939 self.assertRaises(ValueError, float, "na")
940 self.assertRaises(ValueError, float, "+na")
941 self.assertRaises(ValueError, float, "-na")
942
Mark Dickinsonbd16edd2009-05-20 22:05:25 +0000943 self.assertRaises(ValueError, float, "++nan")
944 self.assertRaises(ValueError, float, "-+NAN")
945 self.assertRaises(ValueError, float, "+-NaN")
946 self.assertRaises(ValueError, float, "--nAn")
947
Christian Heimes99170a52007-12-19 02:07:34 +0000948 def test_nan_as_str(self):
949 self.assertEqual(repr(1e300 * 1e300 * 0), "nan")
950 self.assertEqual(repr(-1e300 * 1e300 * 0), "nan")
951
952 self.assertEqual(str(1e300 * 1e300 * 0), "nan")
953 self.assertEqual(str(-1e300 * 1e300 * 0), "nan")
Christian Heimes827b35c2007-12-10 22:19:17 +0000954
Mark Dickinsone383e822012-04-29 15:31:56 +0100955 def test_inf_signs(self):
956 self.assertEqual(copysign(1.0, float('inf')), 1.0)
957 self.assertEqual(copysign(1.0, float('-inf')), -1.0)
958
959 @unittest.skipUnless(getattr(sys, 'float_repr_style', '') == 'short',
960 "applies only when using short float repr style")
961 def test_nan_signs(self):
962 # When using the dtoa.c code, the sign of float('nan') should
963 # be predictable.
964 self.assertEqual(copysign(1.0, float('nan')), 1.0)
965 self.assertEqual(copysign(1.0, float('-nan')), -1.0)
966
967
Mark Dickinson65fe25e2008-07-16 11:30:51 +0000968fromHex = float.fromhex
969toHex = float.hex
970class HexFloatTestCase(unittest.TestCase):
971 MAX = fromHex('0x.fffffffffffff8p+1024') # max normal
972 MIN = fromHex('0x1p-1022') # min normal
973 TINY = fromHex('0x0.0000000000001p-1022') # min subnormal
974 EPS = fromHex('0x0.0000000000001p0') # diff between 1.0 and next float up
975
976 def identical(self, x, y):
977 # check that floats x and y are identical, or that both
978 # are NaNs
979 if isnan(x) or isnan(y):
980 if isnan(x) == isnan(y):
981 return
982 elif x == y and (x != 0.0 or copysign(1.0, x) == copysign(1.0, y)):
983 return
984 self.fail('%r not identical to %r' % (x, y))
985
986 def test_ends(self):
Mark Dickinson38bbc482008-07-16 11:32:23 +0000987 self.identical(self.MIN, ldexp(1.0, -1022))
988 self.identical(self.TINY, ldexp(1.0, -1074))
989 self.identical(self.EPS, ldexp(1.0, -52))
990 self.identical(self.MAX, 2.*(ldexp(1.0, 1023) - ldexp(1.0, 970)))
Mark Dickinson65fe25e2008-07-16 11:30:51 +0000991
992 def test_invalid_inputs(self):
993 invalid_inputs = [
994 'infi', # misspelt infinities and nans
995 '-Infinit',
996 '++inf',
997 '-+Inf',
998 '--nan',
999 '+-NaN',
1000 'snan',
1001 'NaNs',
1002 'nna',
Mark Dickinsond1ec8b22009-05-11 15:45:15 +00001003 'an',
1004 'nf',
1005 'nfinity',
1006 'inity',
1007 'iinity',
Mark Dickinson65fe25e2008-07-16 11:30:51 +00001008 '0xnan',
1009 '',
1010 ' ',
1011 'x1.0p0',
1012 '0xX1.0p0',
1013 '+ 0x1.0p0', # internal whitespace
1014 '- 0x1.0p0',
1015 '0 x1.0p0',
1016 '0x 1.0p0',
1017 '0x1 2.0p0',
1018 '+0x1 .0p0',
1019 '0x1. 0p0',
1020 '-0x1.0 1p0',
1021 '-0x1.0 p0',
1022 '+0x1.0p +0',
1023 '0x1.0p -0',
1024 '0x1.0p 0',
1025 '+0x1.0p+ 0',
1026 '-0x1.0p- 0',
1027 '++0x1.0p-0', # double signs
1028 '--0x1.0p0',
1029 '+-0x1.0p+0',
1030 '-+0x1.0p0',
1031 '0x1.0p++0',
1032 '+0x1.0p+-0',
1033 '-0x1.0p-+0',
1034 '0x1.0p--0',
1035 '0x1.0.p0',
1036 '0x.p0', # no hex digits before or after point
1037 '0x1,p0', # wrong decimal point character
1038 '0x1pa',
1039 '0x1p\uff10', # fullwidth Unicode digits
1040 '\uff10x1p0',
1041 '0x\uff11p0',
1042 '0x1.\uff10p0',
1043 '0x1p0 \n 0x2p0',
1044 '0x1p0\0 0x1p0', # embedded null byte is not end of string
1045 ]
1046 for x in invalid_inputs:
Mark Dickinson589b7952008-08-21 20:05:56 +00001047 try:
1048 result = fromHex(x)
1049 except ValueError:
1050 pass
1051 else:
1052 self.fail('Expected float.fromhex(%r) to raise ValueError; '
1053 'got %r instead' % (x, result))
Mark Dickinson65fe25e2008-07-16 11:30:51 +00001054
1055
Mark Dickinsond1ec8b22009-05-11 15:45:15 +00001056 def test_whitespace(self):
1057 value_pairs = [
1058 ('inf', INF),
1059 ('-Infinity', -INF),
1060 ('nan', NAN),
1061 ('1.0', 1.0),
1062 ('-0x.2', -0.125),
1063 ('-0.0', -0.0)
1064 ]
1065 whitespace = [
1066 '',
1067 ' ',
1068 '\t',
1069 '\n',
1070 '\n \t',
1071 '\f',
1072 '\v',
1073 '\r'
1074 ]
1075 for inp, expected in value_pairs:
1076 for lead in whitespace:
1077 for trail in whitespace:
1078 got = fromHex(lead + inp + trail)
1079 self.identical(got, expected)
1080
1081
Mark Dickinson65fe25e2008-07-16 11:30:51 +00001082 def test_from_hex(self):
1083 MIN = self.MIN;
1084 MAX = self.MAX;
1085 TINY = self.TINY;
1086 EPS = self.EPS;
1087
1088 # two spellings of infinity, with optional signs; case-insensitive
1089 self.identical(fromHex('inf'), INF)
1090 self.identical(fromHex('+Inf'), INF)
1091 self.identical(fromHex('-INF'), -INF)
1092 self.identical(fromHex('iNf'), INF)
1093 self.identical(fromHex('Infinity'), INF)
1094 self.identical(fromHex('+INFINITY'), INF)
1095 self.identical(fromHex('-infinity'), -INF)
1096 self.identical(fromHex('-iNFiNitY'), -INF)
1097
1098 # nans with optional sign; case insensitive
1099 self.identical(fromHex('nan'), NAN)
1100 self.identical(fromHex('+NaN'), NAN)
1101 self.identical(fromHex('-NaN'), NAN)
1102 self.identical(fromHex('-nAN'), NAN)
1103
1104 # variations in input format
1105 self.identical(fromHex('1'), 1.0)
1106 self.identical(fromHex('+1'), 1.0)
1107 self.identical(fromHex('1.'), 1.0)
1108 self.identical(fromHex('1.0'), 1.0)
1109 self.identical(fromHex('1.0p0'), 1.0)
1110 self.identical(fromHex('01'), 1.0)
1111 self.identical(fromHex('01.'), 1.0)
1112 self.identical(fromHex('0x1'), 1.0)
1113 self.identical(fromHex('0x1.'), 1.0)
1114 self.identical(fromHex('0x1.0'), 1.0)
1115 self.identical(fromHex('+0x1.0'), 1.0)
1116 self.identical(fromHex('0x1p0'), 1.0)
1117 self.identical(fromHex('0X1p0'), 1.0)
1118 self.identical(fromHex('0X1P0'), 1.0)
1119 self.identical(fromHex('0x1P0'), 1.0)
1120 self.identical(fromHex('0x1.p0'), 1.0)
1121 self.identical(fromHex('0x1.0p0'), 1.0)
1122 self.identical(fromHex('0x.1p4'), 1.0)
1123 self.identical(fromHex('0x.1p04'), 1.0)
1124 self.identical(fromHex('0x.1p004'), 1.0)
1125 self.identical(fromHex('0x1p+0'), 1.0)
1126 self.identical(fromHex('0x1P-0'), 1.0)
1127 self.identical(fromHex('+0x1p0'), 1.0)
1128 self.identical(fromHex('0x01p0'), 1.0)
1129 self.identical(fromHex('0x1p00'), 1.0)
1130 self.identical(fromHex(' 0x1p0 '), 1.0)
1131 self.identical(fromHex('\n 0x1p0'), 1.0)
1132 self.identical(fromHex('0x1p0 \t'), 1.0)
1133 self.identical(fromHex('0xap0'), 10.0)
1134 self.identical(fromHex('0xAp0'), 10.0)
1135 self.identical(fromHex('0xaP0'), 10.0)
1136 self.identical(fromHex('0xAP0'), 10.0)
1137 self.identical(fromHex('0xbep0'), 190.0)
1138 self.identical(fromHex('0xBep0'), 190.0)
1139 self.identical(fromHex('0xbEp0'), 190.0)
1140 self.identical(fromHex('0XBE0P-4'), 190.0)
1141 self.identical(fromHex('0xBEp0'), 190.0)
1142 self.identical(fromHex('0xB.Ep4'), 190.0)
1143 self.identical(fromHex('0x.BEp8'), 190.0)
1144 self.identical(fromHex('0x.0BEp12'), 190.0)
1145
1146 # moving the point around
1147 pi = fromHex('0x1.921fb54442d18p1')
1148 self.identical(fromHex('0x.006487ed5110b46p11'), pi)
1149 self.identical(fromHex('0x.00c90fdaa22168cp10'), pi)
1150 self.identical(fromHex('0x.01921fb54442d18p9'), pi)
1151 self.identical(fromHex('0x.03243f6a8885a3p8'), pi)
1152 self.identical(fromHex('0x.06487ed5110b46p7'), pi)
1153 self.identical(fromHex('0x.0c90fdaa22168cp6'), pi)
1154 self.identical(fromHex('0x.1921fb54442d18p5'), pi)
1155 self.identical(fromHex('0x.3243f6a8885a3p4'), pi)
1156 self.identical(fromHex('0x.6487ed5110b46p3'), pi)
1157 self.identical(fromHex('0x.c90fdaa22168cp2'), pi)
1158 self.identical(fromHex('0x1.921fb54442d18p1'), pi)
1159 self.identical(fromHex('0x3.243f6a8885a3p0'), pi)
1160 self.identical(fromHex('0x6.487ed5110b46p-1'), pi)
1161 self.identical(fromHex('0xc.90fdaa22168cp-2'), pi)
1162 self.identical(fromHex('0x19.21fb54442d18p-3'), pi)
1163 self.identical(fromHex('0x32.43f6a8885a3p-4'), pi)
1164 self.identical(fromHex('0x64.87ed5110b46p-5'), pi)
1165 self.identical(fromHex('0xc9.0fdaa22168cp-6'), pi)
1166 self.identical(fromHex('0x192.1fb54442d18p-7'), pi)
1167 self.identical(fromHex('0x324.3f6a8885a3p-8'), pi)
1168 self.identical(fromHex('0x648.7ed5110b46p-9'), pi)
1169 self.identical(fromHex('0xc90.fdaa22168cp-10'), pi)
1170 self.identical(fromHex('0x1921.fb54442d18p-11'), pi)
1171 # ...
1172 self.identical(fromHex('0x1921fb54442d1.8p-47'), pi)
1173 self.identical(fromHex('0x3243f6a8885a3p-48'), pi)
1174 self.identical(fromHex('0x6487ed5110b46p-49'), pi)
1175 self.identical(fromHex('0xc90fdaa22168cp-50'), pi)
1176 self.identical(fromHex('0x1921fb54442d18p-51'), pi)
1177 self.identical(fromHex('0x3243f6a8885a30p-52'), pi)
1178 self.identical(fromHex('0x6487ed5110b460p-53'), pi)
1179 self.identical(fromHex('0xc90fdaa22168c0p-54'), pi)
1180 self.identical(fromHex('0x1921fb54442d180p-55'), pi)
1181
1182
1183 # results that should overflow...
1184 self.assertRaises(OverflowError, fromHex, '-0x1p1024')
1185 self.assertRaises(OverflowError, fromHex, '0x1p+1025')
1186 self.assertRaises(OverflowError, fromHex, '+0X1p1030')
1187 self.assertRaises(OverflowError, fromHex, '-0x1p+1100')
1188 self.assertRaises(OverflowError, fromHex, '0X1p123456789123456789')
1189 self.assertRaises(OverflowError, fromHex, '+0X.8p+1025')
1190 self.assertRaises(OverflowError, fromHex, '+0x0.8p1025')
1191 self.assertRaises(OverflowError, fromHex, '-0x0.4p1026')
1192 self.assertRaises(OverflowError, fromHex, '0X2p+1023')
1193 self.assertRaises(OverflowError, fromHex, '0x2.p1023')
1194 self.assertRaises(OverflowError, fromHex, '-0x2.0p+1023')
1195 self.assertRaises(OverflowError, fromHex, '+0X4p+1022')
1196 self.assertRaises(OverflowError, fromHex, '0x1.ffffffffffffffp+1023')
1197 self.assertRaises(OverflowError, fromHex, '-0X1.fffffffffffff9p1023')
1198 self.assertRaises(OverflowError, fromHex, '0X1.fffffffffffff8p1023')
1199 self.assertRaises(OverflowError, fromHex, '+0x3.fffffffffffffp1022')
1200 self.assertRaises(OverflowError, fromHex, '0x3fffffffffffffp+970')
1201 self.assertRaises(OverflowError, fromHex, '0x10000000000000000p960')
1202 self.assertRaises(OverflowError, fromHex, '-0Xffffffffffffffffp960')
1203
1204 # ...and those that round to +-max float
1205 self.identical(fromHex('+0x1.fffffffffffffp+1023'), MAX)
1206 self.identical(fromHex('-0X1.fffffffffffff7p1023'), -MAX)
1207 self.identical(fromHex('0X1.fffffffffffff7fffffffffffffp1023'), MAX)
1208
1209 # zeros
1210 self.identical(fromHex('0x0p0'), 0.0)
1211 self.identical(fromHex('0x0p1000'), 0.0)
1212 self.identical(fromHex('-0x0p1023'), -0.0)
1213 self.identical(fromHex('0X0p1024'), 0.0)
1214 self.identical(fromHex('-0x0p1025'), -0.0)
1215 self.identical(fromHex('0X0p2000'), 0.0)
1216 self.identical(fromHex('0x0p123456789123456789'), 0.0)
1217 self.identical(fromHex('-0X0p-0'), -0.0)
1218 self.identical(fromHex('-0X0p-1000'), -0.0)
1219 self.identical(fromHex('0x0p-1023'), 0.0)
1220 self.identical(fromHex('-0X0p-1024'), -0.0)
1221 self.identical(fromHex('-0x0p-1025'), -0.0)
1222 self.identical(fromHex('-0x0p-1072'), -0.0)
1223 self.identical(fromHex('0X0p-1073'), 0.0)
1224 self.identical(fromHex('-0x0p-1074'), -0.0)
1225 self.identical(fromHex('0x0p-1075'), 0.0)
1226 self.identical(fromHex('0X0p-1076'), 0.0)
1227 self.identical(fromHex('-0X0p-2000'), -0.0)
1228 self.identical(fromHex('-0x0p-123456789123456789'), -0.0)
1229
1230 # values that should underflow to 0
1231 self.identical(fromHex('0X1p-1075'), 0.0)
1232 self.identical(fromHex('-0X1p-1075'), -0.0)
1233 self.identical(fromHex('-0x1p-123456789123456789'), -0.0)
1234 self.identical(fromHex('0x1.00000000000000001p-1075'), TINY)
1235 self.identical(fromHex('-0x1.1p-1075'), -TINY)
1236 self.identical(fromHex('0x1.fffffffffffffffffp-1075'), TINY)
1237
1238 # check round-half-even is working correctly near 0 ...
1239 self.identical(fromHex('0x1p-1076'), 0.0)
1240 self.identical(fromHex('0X2p-1076'), 0.0)
1241 self.identical(fromHex('0X3p-1076'), TINY)
1242 self.identical(fromHex('0x4p-1076'), TINY)
1243 self.identical(fromHex('0X5p-1076'), TINY)
1244 self.identical(fromHex('0X6p-1076'), 2*TINY)
1245 self.identical(fromHex('0x7p-1076'), 2*TINY)
1246 self.identical(fromHex('0X8p-1076'), 2*TINY)
1247 self.identical(fromHex('0X9p-1076'), 2*TINY)
1248 self.identical(fromHex('0xap-1076'), 2*TINY)
1249 self.identical(fromHex('0Xbp-1076'), 3*TINY)
1250 self.identical(fromHex('0xcp-1076'), 3*TINY)
1251 self.identical(fromHex('0Xdp-1076'), 3*TINY)
1252 self.identical(fromHex('0Xep-1076'), 4*TINY)
1253 self.identical(fromHex('0xfp-1076'), 4*TINY)
1254 self.identical(fromHex('0x10p-1076'), 4*TINY)
1255 self.identical(fromHex('-0x1p-1076'), -0.0)
1256 self.identical(fromHex('-0X2p-1076'), -0.0)
1257 self.identical(fromHex('-0x3p-1076'), -TINY)
1258 self.identical(fromHex('-0X4p-1076'), -TINY)
1259 self.identical(fromHex('-0x5p-1076'), -TINY)
1260 self.identical(fromHex('-0x6p-1076'), -2*TINY)
1261 self.identical(fromHex('-0X7p-1076'), -2*TINY)
1262 self.identical(fromHex('-0X8p-1076'), -2*TINY)
1263 self.identical(fromHex('-0X9p-1076'), -2*TINY)
1264 self.identical(fromHex('-0Xap-1076'), -2*TINY)
1265 self.identical(fromHex('-0xbp-1076'), -3*TINY)
1266 self.identical(fromHex('-0xcp-1076'), -3*TINY)
1267 self.identical(fromHex('-0Xdp-1076'), -3*TINY)
1268 self.identical(fromHex('-0xep-1076'), -4*TINY)
1269 self.identical(fromHex('-0Xfp-1076'), -4*TINY)
1270 self.identical(fromHex('-0X10p-1076'), -4*TINY)
1271
1272 # ... and near MIN ...
1273 self.identical(fromHex('0x0.ffffffffffffd6p-1022'), MIN-3*TINY)
1274 self.identical(fromHex('0x0.ffffffffffffd8p-1022'), MIN-2*TINY)
1275 self.identical(fromHex('0x0.ffffffffffffdap-1022'), MIN-2*TINY)
1276 self.identical(fromHex('0x0.ffffffffffffdcp-1022'), MIN-2*TINY)
1277 self.identical(fromHex('0x0.ffffffffffffdep-1022'), MIN-2*TINY)
1278 self.identical(fromHex('0x0.ffffffffffffe0p-1022'), MIN-2*TINY)
1279 self.identical(fromHex('0x0.ffffffffffffe2p-1022'), MIN-2*TINY)
1280 self.identical(fromHex('0x0.ffffffffffffe4p-1022'), MIN-2*TINY)
1281 self.identical(fromHex('0x0.ffffffffffffe6p-1022'), MIN-2*TINY)
1282 self.identical(fromHex('0x0.ffffffffffffe8p-1022'), MIN-2*TINY)
1283 self.identical(fromHex('0x0.ffffffffffffeap-1022'), MIN-TINY)
1284 self.identical(fromHex('0x0.ffffffffffffecp-1022'), MIN-TINY)
1285 self.identical(fromHex('0x0.ffffffffffffeep-1022'), MIN-TINY)
1286 self.identical(fromHex('0x0.fffffffffffff0p-1022'), MIN-TINY)
1287 self.identical(fromHex('0x0.fffffffffffff2p-1022'), MIN-TINY)
1288 self.identical(fromHex('0x0.fffffffffffff4p-1022'), MIN-TINY)
1289 self.identical(fromHex('0x0.fffffffffffff6p-1022'), MIN-TINY)
1290 self.identical(fromHex('0x0.fffffffffffff8p-1022'), MIN)
1291 self.identical(fromHex('0x0.fffffffffffffap-1022'), MIN)
1292 self.identical(fromHex('0x0.fffffffffffffcp-1022'), MIN)
1293 self.identical(fromHex('0x0.fffffffffffffep-1022'), MIN)
1294 self.identical(fromHex('0x1.00000000000000p-1022'), MIN)
1295 self.identical(fromHex('0x1.00000000000002p-1022'), MIN)
1296 self.identical(fromHex('0x1.00000000000004p-1022'), MIN)
1297 self.identical(fromHex('0x1.00000000000006p-1022'), MIN)
1298 self.identical(fromHex('0x1.00000000000008p-1022'), MIN)
1299 self.identical(fromHex('0x1.0000000000000ap-1022'), MIN+TINY)
1300 self.identical(fromHex('0x1.0000000000000cp-1022'), MIN+TINY)
1301 self.identical(fromHex('0x1.0000000000000ep-1022'), MIN+TINY)
1302 self.identical(fromHex('0x1.00000000000010p-1022'), MIN+TINY)
1303 self.identical(fromHex('0x1.00000000000012p-1022'), MIN+TINY)
1304 self.identical(fromHex('0x1.00000000000014p-1022'), MIN+TINY)
1305 self.identical(fromHex('0x1.00000000000016p-1022'), MIN+TINY)
1306 self.identical(fromHex('0x1.00000000000018p-1022'), MIN+2*TINY)
1307
1308 # ... and near 1.0.
1309 self.identical(fromHex('0x0.fffffffffffff0p0'), 1.0-EPS)
1310 self.identical(fromHex('0x0.fffffffffffff1p0'), 1.0-EPS)
1311 self.identical(fromHex('0X0.fffffffffffff2p0'), 1.0-EPS)
1312 self.identical(fromHex('0x0.fffffffffffff3p0'), 1.0-EPS)
1313 self.identical(fromHex('0X0.fffffffffffff4p0'), 1.0-EPS)
1314 self.identical(fromHex('0X0.fffffffffffff5p0'), 1.0-EPS/2)
1315 self.identical(fromHex('0X0.fffffffffffff6p0'), 1.0-EPS/2)
1316 self.identical(fromHex('0x0.fffffffffffff7p0'), 1.0-EPS/2)
1317 self.identical(fromHex('0x0.fffffffffffff8p0'), 1.0-EPS/2)
1318 self.identical(fromHex('0X0.fffffffffffff9p0'), 1.0-EPS/2)
1319 self.identical(fromHex('0X0.fffffffffffffap0'), 1.0-EPS/2)
1320 self.identical(fromHex('0x0.fffffffffffffbp0'), 1.0-EPS/2)
1321 self.identical(fromHex('0X0.fffffffffffffcp0'), 1.0)
1322 self.identical(fromHex('0x0.fffffffffffffdp0'), 1.0)
1323 self.identical(fromHex('0X0.fffffffffffffep0'), 1.0)
1324 self.identical(fromHex('0x0.ffffffffffffffp0'), 1.0)
1325 self.identical(fromHex('0X1.00000000000000p0'), 1.0)
1326 self.identical(fromHex('0X1.00000000000001p0'), 1.0)
1327 self.identical(fromHex('0x1.00000000000002p0'), 1.0)
1328 self.identical(fromHex('0X1.00000000000003p0'), 1.0)
1329 self.identical(fromHex('0x1.00000000000004p0'), 1.0)
1330 self.identical(fromHex('0X1.00000000000005p0'), 1.0)
1331 self.identical(fromHex('0X1.00000000000006p0'), 1.0)
1332 self.identical(fromHex('0X1.00000000000007p0'), 1.0)
1333 self.identical(fromHex('0x1.00000000000007ffffffffffffffffffffp0'),
1334 1.0)
1335 self.identical(fromHex('0x1.00000000000008p0'), 1.0)
1336 self.identical(fromHex('0x1.00000000000008000000000000000001p0'),
1337 1+EPS)
1338 self.identical(fromHex('0X1.00000000000009p0'), 1.0+EPS)
1339 self.identical(fromHex('0x1.0000000000000ap0'), 1.0+EPS)
1340 self.identical(fromHex('0x1.0000000000000bp0'), 1.0+EPS)
1341 self.identical(fromHex('0X1.0000000000000cp0'), 1.0+EPS)
1342 self.identical(fromHex('0x1.0000000000000dp0'), 1.0+EPS)
1343 self.identical(fromHex('0x1.0000000000000ep0'), 1.0+EPS)
1344 self.identical(fromHex('0X1.0000000000000fp0'), 1.0+EPS)
1345 self.identical(fromHex('0x1.00000000000010p0'), 1.0+EPS)
1346 self.identical(fromHex('0X1.00000000000011p0'), 1.0+EPS)
1347 self.identical(fromHex('0x1.00000000000012p0'), 1.0+EPS)
1348 self.identical(fromHex('0X1.00000000000013p0'), 1.0+EPS)
1349 self.identical(fromHex('0X1.00000000000014p0'), 1.0+EPS)
1350 self.identical(fromHex('0x1.00000000000015p0'), 1.0+EPS)
1351 self.identical(fromHex('0x1.00000000000016p0'), 1.0+EPS)
1352 self.identical(fromHex('0X1.00000000000017p0'), 1.0+EPS)
1353 self.identical(fromHex('0x1.00000000000017ffffffffffffffffffffp0'),
1354 1.0+EPS)
1355 self.identical(fromHex('0x1.00000000000018p0'), 1.0+2*EPS)
1356 self.identical(fromHex('0X1.00000000000018000000000000000001p0'),
1357 1.0+2*EPS)
1358 self.identical(fromHex('0x1.00000000000019p0'), 1.0+2*EPS)
1359 self.identical(fromHex('0X1.0000000000001ap0'), 1.0+2*EPS)
1360 self.identical(fromHex('0X1.0000000000001bp0'), 1.0+2*EPS)
1361 self.identical(fromHex('0x1.0000000000001cp0'), 1.0+2*EPS)
1362 self.identical(fromHex('0x1.0000000000001dp0'), 1.0+2*EPS)
1363 self.identical(fromHex('0x1.0000000000001ep0'), 1.0+2*EPS)
1364 self.identical(fromHex('0X1.0000000000001fp0'), 1.0+2*EPS)
1365 self.identical(fromHex('0x1.00000000000020p0'), 1.0+2*EPS)
1366
1367 def test_roundtrip(self):
1368 def roundtrip(x):
1369 return fromHex(toHex(x))
1370
1371 for x in [NAN, INF, self.MAX, self.MIN, self.MIN-self.TINY, self.TINY, 0.0]:
1372 self.identical(x, roundtrip(x))
1373 self.identical(-x, roundtrip(-x))
1374
1375 # fromHex(toHex(x)) should exactly recover x, for any non-NaN float x.
1376 import random
1377 for i in range(10000):
1378 e = random.randrange(-1200, 1200)
1379 m = random.random()
1380 s = random.choice([1.0, -1.0])
1381 try:
1382 x = s*ldexp(m, e)
1383 except OverflowError:
1384 pass
1385 else:
1386 self.identical(x, fromHex(toHex(x)))
1387
Serhiy Storchakaea36c942016-05-12 10:37:58 +03001388 def test_subclass(self):
1389 class F(float):
1390 def __new__(cls, value):
1391 return float.__new__(cls, value + 1)
1392
1393 f = F.fromhex((1.5).hex())
1394 self.assertIs(type(f), F)
1395 self.assertEqual(f, 2.5)
1396
1397 class F2(float):
1398 def __init__(self, value):
1399 self.foo = 'bar'
1400
1401 f = F2.fromhex((1.5).hex())
1402 self.assertIs(type(f), F2)
1403 self.assertEqual(f, 1.5)
1404 self.assertEqual(getattr(f, 'foo', 'none'), 'bar')
1405
Christian Heimes53876d92008-04-19 00:31:39 +00001406
Michael W. Hudsonba283e22005-05-27 15:23:20 +00001407if __name__ == '__main__':
Zachary Ware38c707e2015-04-13 15:00:43 -05001408 unittest.main()