blob: 17174dd295dfcc8b2c015dc5aaac45187a64d936 [file] [log] [blame]
Benjamin Peterson101447b2015-03-09 10:37:50 -04001import fractions
Christian Heimes53876d92008-04-19 00:31:39 +00002import operator
Benjamin Peterson101447b2015-03-09 10:37:50 -04003import os
4import random
5import sys
6import struct
Benjamin Petersona9157232015-03-06 09:08:44 -05007import time
Benjamin Peterson101447b2015-03-09 10:37:50 -04008import unittest
9
10from test import support
Brett Cannona721aba2016-09-09 14:57:09 -070011from test.test_grammar import (VALID_UNDERSCORE_LITERALS,
12 INVALID_UNDERSCORE_LITERALS)
Benjamin Peterson101447b2015-03-09 10:37:50 -040013from math import isinf, isnan, copysign, ldexp
Michael W. Hudsonba283e22005-05-27 15:23:20 +000014
Christian Heimes53876d92008-04-19 00:31:39 +000015INF = float("inf")
16NAN = float("nan")
Christian Heimes99170a52007-12-19 02:07:34 +000017
Benjamin Petersone401c682010-07-02 23:25:44 +000018have_getformat = hasattr(float, "__getformat__")
19requires_getformat = unittest.skipUnless(have_getformat,
20 "requires __getformat__")
21requires_setformat = unittest.skipUnless(hasattr(float, "__setformat__"),
22 "requires __setformat__")
Mark Dickinson9ab44b52009-12-30 16:22:49 +000023
Eric Smith0923d1d2009-04-16 20:16:10 +000024#locate file with float format test values
25test_dir = os.path.dirname(__file__) or os.curdir
26format_testfile = os.path.join(test_dir, 'formatfloat_testcases.txt')
27
Serhiy Storchaka15095802015-11-25 15:47:01 +020028class FloatSubclass(float):
29 pass
30
31class OtherFloatSubclass(float):
32 pass
33
Christian Heimes81ee3ef2008-05-04 22:42:01 +000034class GeneralFloatCases(unittest.TestCase):
35
36 def test_float(self):
37 self.assertEqual(float(3.14), 3.14)
38 self.assertEqual(float(314), 314.0)
39 self.assertEqual(float(" 3.14 "), 3.14)
40 self.assertRaises(ValueError, float, " 0x3.1 ")
41 self.assertRaises(ValueError, float, " -0x3.p-1 ")
42 self.assertRaises(ValueError, float, " +0x3.p-1 ")
43 self.assertRaises(ValueError, float, "++3.14")
44 self.assertRaises(ValueError, float, "+-3.14")
45 self.assertRaises(ValueError, float, "-+3.14")
46 self.assertRaises(ValueError, float, "--3.14")
Eric Smith0923d1d2009-04-16 20:16:10 +000047 self.assertRaises(ValueError, float, ".nan")
48 self.assertRaises(ValueError, float, "+.inf")
49 self.assertRaises(ValueError, float, ".")
50 self.assertRaises(ValueError, float, "-.")
Alexander Belopolsky942af5a2010-12-04 03:38:46 +000051 self.assertRaises(TypeError, float, {})
Ezio Melottia5b95992013-11-07 19:18:34 +020052 self.assertRaisesRegex(TypeError, "not 'dict'", float, {})
Alexander Belopolsky942af5a2010-12-04 03:38:46 +000053 # Lone surrogate
Serhiy Storchaka9b6c60c2017-11-13 21:23:48 +020054 self.assertRaises(ValueError, float, '\uD8F0')
Mark Dickinsonc2d86892010-02-12 21:18:34 +000055 # check that we don't accept alternate exponent markers
56 self.assertRaises(ValueError, float, "-1.7d29")
57 self.assertRaises(ValueError, float, "3D-14")
Alexander Belopolsky942af5a2010-12-04 03:38:46 +000058 self.assertEqual(float(" \u0663.\u0661\u0664 "), 3.14)
59 self.assertEqual(float("\N{EM SPACE}3.14\N{EN SPACE}"), 3.14)
Mark Dickinsona9023be2009-10-27 22:12:20 +000060 # extra long strings should not be a problem
61 float(b'.' + b'1'*1000)
62 float('.' + '1'*1000)
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):
Serhiy Storchaka9e6ac832017-03-09 20:07:58 +0200122 def check(s):
123 with self.assertRaises(ValueError, msg='float(%r)' % (s,)) as cm:
Alexander Belopolsky942af5a2010-12-04 03:38:46 +0000124 float(s)
Serhiy Storchaka9e6ac832017-03-09 20:07:58 +0200125 self.assertEqual(str(cm.exception),
126 'could not convert string to float: %r' % (s,))
Alexander Belopolsky942af5a2010-12-04 03:38:46 +0000127
Serhiy Storchaka9e6ac832017-03-09 20:07:58 +0200128 check('\xbd')
129 check('123\xbd')
130 check(' 123 456 ')
131 check(b' 123 456 ')
132
133 # non-ascii digits (error came from non-digit '!')
134 check('\u0663\u0661\u0664!')
135 # embedded NUL
136 check('123\x00')
137 check('123\x00 245')
138 check('123\x00245')
139 # byte string with embedded NUL
140 check(b'123\x00')
141 # non-UTF-8 byte string
142 check(b'123\xa0')
Alexander Belopolsky942af5a2010-12-04 03:38:46 +0000143
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000144 @support.run_with_locale('LC_NUMERIC', 'fr_FR', 'de_DE')
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000145 def test_float_with_comma(self):
146 # set locale to something that doesn't use '.' for the decimal point
147 # float must not accept the locale specific decimal point but
Ezio Melotti13925002011-03-16 11:05:33 +0200148 # it still has to accept the normal python syntax
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000149 import locale
150 if not locale.localeconv()['decimal_point'] == ',':
Zachary Ware9fe6d862013-12-08 00:20:35 -0600151 self.skipTest('decimal_point is not ","')
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000152
153 self.assertEqual(float(" 3.14 "), 3.14)
154 self.assertEqual(float("+3.14 "), 3.14)
155 self.assertEqual(float("-3.14 "), -3.14)
156 self.assertEqual(float(".14 "), .14)
157 self.assertEqual(float("3. "), 3.0)
158 self.assertEqual(float("3.e3 "), 3000.0)
159 self.assertEqual(float("3.2e3 "), 3200.0)
160 self.assertEqual(float("2.5e-1 "), 0.25)
161 self.assertEqual(float("5e-1"), 0.5)
162 self.assertRaises(ValueError, float, " 3,14 ")
163 self.assertRaises(ValueError, float, " +3,14 ")
164 self.assertRaises(ValueError, float, " -3,14 ")
165 self.assertRaises(ValueError, float, " 0x3.1 ")
166 self.assertRaises(ValueError, float, " -0x3.p-1 ")
167 self.assertRaises(ValueError, float, " +0x3.p-1 ")
168 self.assertEqual(float(" 25.e-1 "), 2.5)
Eli Bendersky67ebabd2011-02-25 10:14:17 +0000169 self.assertAlmostEqual(float(" .25e-1 "), .025)
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000170
171 def test_floatconversion(self):
172 # Make sure that calls to __float__() work properly
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000173 class Foo1(object):
174 def __float__(self):
175 return 42.
176
177 class Foo2(float):
178 def __float__(self):
179 return 42.
180
181 class Foo3(float):
182 def __new__(cls, value=0.):
183 return float.__new__(cls, 2*value)
184
185 def __float__(self):
186 return self
187
188 class Foo4(float):
189 def __float__(self):
190 return 42
191
Benjamin Peterson2808d3c2009-04-15 21:34:27 +0000192 # Issue 5759: __float__ not called on str subclasses (though it is on
193 # unicode subclasses).
194 class FooStr(str):
195 def __float__(self):
196 return float(str(self)) + 1
197
Serhiy Storchaka16931c32016-06-03 21:42:55 +0300198 self.assertEqual(float(Foo1()), 42.)
199 self.assertEqual(float(Foo2()), 42.)
200 with self.assertWarns(DeprecationWarning):
201 self.assertEqual(float(Foo3(21)), 42.)
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000202 self.assertRaises(TypeError, float, Foo4(42))
Serhiy Storchaka16931c32016-06-03 21:42:55 +0300203 self.assertEqual(float(FooStr('8')), 9.)
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000204
Benjamin Petersona9157232015-03-06 09:08:44 -0500205 class Foo5:
206 def __float__(self):
207 return ""
208 self.assertRaises(TypeError, time.sleep, Foo5())
209
Serhiy Storchaka15095802015-11-25 15:47:01 +0200210 # Issue #24731
211 class F:
212 def __float__(self):
213 return OtherFloatSubclass(42.)
Serhiy Storchaka16931c32016-06-03 21:42:55 +0300214 with self.assertWarns(DeprecationWarning):
215 self.assertEqual(float(F()), 42.)
216 with self.assertWarns(DeprecationWarning):
217 self.assertIs(type(float(F())), float)
218 with self.assertWarns(DeprecationWarning):
219 self.assertEqual(FloatSubclass(F()), 42.)
220 with self.assertWarns(DeprecationWarning):
221 self.assertIs(type(FloatSubclass(F())), FloatSubclass)
Serhiy Storchaka15095802015-11-25 15:47:01 +0200222
Serhiy Storchaka58d23e62017-03-06 00:53:39 +0200223 def test_keyword_args(self):
Serhiy Storchaka2e564242017-03-06 17:01:06 +0200224 with self.assertRaisesRegex(TypeError, 'keyword argument'):
225 float(x='3.14')
Serhiy Storchaka58d23e62017-03-06 00:53:39 +0200226
Benjamin Petersonb3b8ba62011-10-28 19:42:48 -0400227 def test_is_integer(self):
228 self.assertFalse((1.1).is_integer())
229 self.assertTrue((1.).is_integer())
230 self.assertFalse(float("nan").is_integer())
231 self.assertFalse(float("inf").is_integer())
232
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000233 def test_floatasratio(self):
234 for f, ratio in [
235 (0.875, (7, 8)),
236 (-0.875, (-7, 8)),
237 (0.0, (0, 1)),
238 (11.5, (23, 2)),
239 ]:
240 self.assertEqual(f.as_integer_ratio(), ratio)
241
242 for i in range(10000):
243 f = random.random()
244 f *= 10 ** random.randint(-100, 100)
245 n, d = f.as_integer_ratio()
246 self.assertEqual(float(n).__truediv__(d), f)
247
248 R = fractions.Fraction
249 self.assertEqual(R(0, 1),
250 R(*float(0.0).as_integer_ratio()))
251 self.assertEqual(R(5, 2),
252 R(*float(2.5).as_integer_ratio()))
253 self.assertEqual(R(1, 2),
254 R(*float(0.5).as_integer_ratio()))
255 self.assertEqual(R(4728779608739021, 2251799813685248),
256 R(*float(2.1).as_integer_ratio()))
257 self.assertEqual(R(-4728779608739021, 2251799813685248),
258 R(*float(-2.1).as_integer_ratio()))
259 self.assertEqual(R(-2100, 1),
260 R(*float(-2100.0).as_integer_ratio()))
261
262 self.assertRaises(OverflowError, float('inf').as_integer_ratio)
263 self.assertRaises(OverflowError, float('-inf').as_integer_ratio)
264 self.assertRaises(ValueError, float('nan').as_integer_ratio)
265
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000266 def test_float_containment(self):
267 floats = (INF, -INF, 0.0, 1.0, NAN)
268 for f in floats:
Benjamin Peterson577473f2010-01-19 00:09:57 +0000269 self.assertIn(f, [f])
Benjamin Peterson577473f2010-01-19 00:09:57 +0000270 self.assertIn(f, (f,))
Benjamin Peterson577473f2010-01-19 00:09:57 +0000271 self.assertIn(f, {f})
Benjamin Peterson577473f2010-01-19 00:09:57 +0000272 self.assertIn(f, {f: None})
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000273 self.assertEqual([f].count(f), 1, "[].count('%r') != 1" % f)
Benjamin Peterson577473f2010-01-19 00:09:57 +0000274 self.assertIn(f, floats)
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000275
276 for f in floats:
277 # nonidentical containers, same type, same contents
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000278 self.assertTrue([f] == [f], "[%r] != [%r]" % (f, f))
279 self.assertTrue((f,) == (f,), "(%r,) != (%r,)" % (f, f))
280 self.assertTrue({f} == {f}, "{%r} != {%r}" % (f, f))
281 self.assertTrue({f : None} == {f: None}, "{%r : None} != "
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000282 "{%r : None}" % (f, f))
283
284 # identical containers
285 l, t, s, d = [f], (f,), {f}, {f: None}
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000286 self.assertTrue(l == l, "[%r] not equal to itself" % f)
287 self.assertTrue(t == t, "(%r,) not equal to itself" % f)
288 self.assertTrue(s == s, "{%r} not equal to itself" % f)
289 self.assertTrue(d == d, "{%r : None} not equal to itself" % f)
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000290
Mark Dickinson9ab44b52009-12-30 16:22:49 +0000291 def assertEqualAndEqualSign(self, a, b):
292 # fail unless a == b and a and b have the same sign bit;
293 # the only difference from assertEqual is that this test
Ezio Melotti13925002011-03-16 11:05:33 +0200294 # distinguishes -0.0 and 0.0.
Mark Dickinson9ab44b52009-12-30 16:22:49 +0000295 self.assertEqual((a, copysign(1.0, a)), (b, copysign(1.0, b)))
296
Eric Smith3ab08ca2010-12-04 15:17:38 +0000297 @support.requires_IEEE_754
Mark Dickinsond2a9b202010-12-04 12:25:30 +0000298 def test_float_mod(self):
299 # Check behaviour of % operator for IEEE 754 special cases.
300 # In particular, check signs of zeros.
301 mod = operator.mod
302
303 self.assertEqualAndEqualSign(mod(-1.0, 1.0), 0.0)
304 self.assertEqualAndEqualSign(mod(-1e-100, 1.0), 1.0)
305 self.assertEqualAndEqualSign(mod(-0.0, 1.0), 0.0)
306 self.assertEqualAndEqualSign(mod(0.0, 1.0), 0.0)
307 self.assertEqualAndEqualSign(mod(1e-100, 1.0), 1e-100)
308 self.assertEqualAndEqualSign(mod(1.0, 1.0), 0.0)
309
310 self.assertEqualAndEqualSign(mod(-1.0, -1.0), -0.0)
311 self.assertEqualAndEqualSign(mod(-1e-100, -1.0), -1e-100)
312 self.assertEqualAndEqualSign(mod(-0.0, -1.0), -0.0)
313 self.assertEqualAndEqualSign(mod(0.0, -1.0), -0.0)
314 self.assertEqualAndEqualSign(mod(1e-100, -1.0), -1.0)
315 self.assertEqualAndEqualSign(mod(1.0, -1.0), -0.0)
316
Eric Smith3ab08ca2010-12-04 15:17:38 +0000317 @support.requires_IEEE_754
Mark Dickinson9ab44b52009-12-30 16:22:49 +0000318 def test_float_pow(self):
319 # test builtin pow and ** operator for IEEE 754 special cases.
320 # Special cases taken from section F.9.4.4 of the C99 specification
321
322 for pow_op in pow, operator.pow:
323 # x**NAN is NAN for any x except 1
324 self.assertTrue(isnan(pow_op(-INF, NAN)))
325 self.assertTrue(isnan(pow_op(-2.0, NAN)))
326 self.assertTrue(isnan(pow_op(-1.0, NAN)))
327 self.assertTrue(isnan(pow_op(-0.5, NAN)))
328 self.assertTrue(isnan(pow_op(-0.0, NAN)))
329 self.assertTrue(isnan(pow_op(0.0, NAN)))
330 self.assertTrue(isnan(pow_op(0.5, NAN)))
331 self.assertTrue(isnan(pow_op(2.0, NAN)))
332 self.assertTrue(isnan(pow_op(INF, NAN)))
333 self.assertTrue(isnan(pow_op(NAN, NAN)))
334
335 # NAN**y is NAN for any y except +-0
336 self.assertTrue(isnan(pow_op(NAN, -INF)))
337 self.assertTrue(isnan(pow_op(NAN, -2.0)))
338 self.assertTrue(isnan(pow_op(NAN, -1.0)))
339 self.assertTrue(isnan(pow_op(NAN, -0.5)))
340 self.assertTrue(isnan(pow_op(NAN, 0.5)))
341 self.assertTrue(isnan(pow_op(NAN, 1.0)))
342 self.assertTrue(isnan(pow_op(NAN, 2.0)))
343 self.assertTrue(isnan(pow_op(NAN, INF)))
344
345 # (+-0)**y raises ZeroDivisionError for y a negative odd integer
346 self.assertRaises(ZeroDivisionError, pow_op, -0.0, -1.0)
347 self.assertRaises(ZeroDivisionError, pow_op, 0.0, -1.0)
348
349 # (+-0)**y raises ZeroDivisionError for y finite and negative
350 # but not an odd integer
351 self.assertRaises(ZeroDivisionError, pow_op, -0.0, -2.0)
352 self.assertRaises(ZeroDivisionError, pow_op, -0.0, -0.5)
353 self.assertRaises(ZeroDivisionError, pow_op, 0.0, -2.0)
354 self.assertRaises(ZeroDivisionError, pow_op, 0.0, -0.5)
355
356 # (+-0)**y is +-0 for y a positive odd integer
357 self.assertEqualAndEqualSign(pow_op(-0.0, 1.0), -0.0)
358 self.assertEqualAndEqualSign(pow_op(0.0, 1.0), 0.0)
359
360 # (+-0)**y is 0 for y finite and positive but not an odd integer
361 self.assertEqualAndEqualSign(pow_op(-0.0, 0.5), 0.0)
362 self.assertEqualAndEqualSign(pow_op(-0.0, 2.0), 0.0)
363 self.assertEqualAndEqualSign(pow_op(0.0, 0.5), 0.0)
364 self.assertEqualAndEqualSign(pow_op(0.0, 2.0), 0.0)
365
366 # (-1)**+-inf is 1
367 self.assertEqualAndEqualSign(pow_op(-1.0, -INF), 1.0)
368 self.assertEqualAndEqualSign(pow_op(-1.0, INF), 1.0)
369
370 # 1**y is 1 for any y, even if y is an infinity or nan
371 self.assertEqualAndEqualSign(pow_op(1.0, -INF), 1.0)
372 self.assertEqualAndEqualSign(pow_op(1.0, -2.0), 1.0)
373 self.assertEqualAndEqualSign(pow_op(1.0, -1.0), 1.0)
374 self.assertEqualAndEqualSign(pow_op(1.0, -0.5), 1.0)
375 self.assertEqualAndEqualSign(pow_op(1.0, -0.0), 1.0)
376 self.assertEqualAndEqualSign(pow_op(1.0, 0.0), 1.0)
377 self.assertEqualAndEqualSign(pow_op(1.0, 0.5), 1.0)
378 self.assertEqualAndEqualSign(pow_op(1.0, 1.0), 1.0)
379 self.assertEqualAndEqualSign(pow_op(1.0, 2.0), 1.0)
380 self.assertEqualAndEqualSign(pow_op(1.0, INF), 1.0)
381 self.assertEqualAndEqualSign(pow_op(1.0, NAN), 1.0)
382
383 # x**+-0 is 1 for any x, even if x is a zero, infinity, or nan
384 self.assertEqualAndEqualSign(pow_op(-INF, 0.0), 1.0)
385 self.assertEqualAndEqualSign(pow_op(-2.0, 0.0), 1.0)
386 self.assertEqualAndEqualSign(pow_op(-1.0, 0.0), 1.0)
387 self.assertEqualAndEqualSign(pow_op(-0.5, 0.0), 1.0)
388 self.assertEqualAndEqualSign(pow_op(-0.0, 0.0), 1.0)
389 self.assertEqualAndEqualSign(pow_op(0.0, 0.0), 1.0)
390 self.assertEqualAndEqualSign(pow_op(0.5, 0.0), 1.0)
391 self.assertEqualAndEqualSign(pow_op(1.0, 0.0), 1.0)
392 self.assertEqualAndEqualSign(pow_op(2.0, 0.0), 1.0)
393 self.assertEqualAndEqualSign(pow_op(INF, 0.0), 1.0)
394 self.assertEqualAndEqualSign(pow_op(NAN, 0.0), 1.0)
395 self.assertEqualAndEqualSign(pow_op(-INF, -0.0), 1.0)
396 self.assertEqualAndEqualSign(pow_op(-2.0, -0.0), 1.0)
397 self.assertEqualAndEqualSign(pow_op(-1.0, -0.0), 1.0)
398 self.assertEqualAndEqualSign(pow_op(-0.5, -0.0), 1.0)
399 self.assertEqualAndEqualSign(pow_op(-0.0, -0.0), 1.0)
400 self.assertEqualAndEqualSign(pow_op(0.0, -0.0), 1.0)
401 self.assertEqualAndEqualSign(pow_op(0.5, -0.0), 1.0)
402 self.assertEqualAndEqualSign(pow_op(1.0, -0.0), 1.0)
403 self.assertEqualAndEqualSign(pow_op(2.0, -0.0), 1.0)
404 self.assertEqualAndEqualSign(pow_op(INF, -0.0), 1.0)
405 self.assertEqualAndEqualSign(pow_op(NAN, -0.0), 1.0)
406
407 # x**y defers to complex pow for finite negative x and
408 # non-integral y.
409 self.assertEqual(type(pow_op(-2.0, -0.5)), complex)
410 self.assertEqual(type(pow_op(-2.0, 0.5)), complex)
411 self.assertEqual(type(pow_op(-1.0, -0.5)), complex)
412 self.assertEqual(type(pow_op(-1.0, 0.5)), complex)
413 self.assertEqual(type(pow_op(-0.5, -0.5)), complex)
414 self.assertEqual(type(pow_op(-0.5, 0.5)), complex)
415
416 # x**-INF is INF for abs(x) < 1
417 self.assertEqualAndEqualSign(pow_op(-0.5, -INF), INF)
418 self.assertEqualAndEqualSign(pow_op(-0.0, -INF), INF)
419 self.assertEqualAndEqualSign(pow_op(0.0, -INF), INF)
420 self.assertEqualAndEqualSign(pow_op(0.5, -INF), INF)
421
422 # x**-INF is 0 for abs(x) > 1
423 self.assertEqualAndEqualSign(pow_op(-INF, -INF), 0.0)
424 self.assertEqualAndEqualSign(pow_op(-2.0, -INF), 0.0)
425 self.assertEqualAndEqualSign(pow_op(2.0, -INF), 0.0)
426 self.assertEqualAndEqualSign(pow_op(INF, -INF), 0.0)
427
428 # x**INF is 0 for abs(x) < 1
429 self.assertEqualAndEqualSign(pow_op(-0.5, INF), 0.0)
430 self.assertEqualAndEqualSign(pow_op(-0.0, INF), 0.0)
431 self.assertEqualAndEqualSign(pow_op(0.0, INF), 0.0)
432 self.assertEqualAndEqualSign(pow_op(0.5, INF), 0.0)
433
434 # x**INF is INF for abs(x) > 1
435 self.assertEqualAndEqualSign(pow_op(-INF, INF), INF)
436 self.assertEqualAndEqualSign(pow_op(-2.0, INF), INF)
437 self.assertEqualAndEqualSign(pow_op(2.0, INF), INF)
438 self.assertEqualAndEqualSign(pow_op(INF, INF), INF)
439
440 # (-INF)**y is -0.0 for y a negative odd integer
441 self.assertEqualAndEqualSign(pow_op(-INF, -1.0), -0.0)
442
443 # (-INF)**y is 0.0 for y negative but not an odd integer
444 self.assertEqualAndEqualSign(pow_op(-INF, -0.5), 0.0)
445 self.assertEqualAndEqualSign(pow_op(-INF, -2.0), 0.0)
446
447 # (-INF)**y is -INF for y a positive odd integer
448 self.assertEqualAndEqualSign(pow_op(-INF, 1.0), -INF)
449
450 # (-INF)**y is INF for y positive but not an odd integer
451 self.assertEqualAndEqualSign(pow_op(-INF, 0.5), INF)
452 self.assertEqualAndEqualSign(pow_op(-INF, 2.0), INF)
453
454 # INF**y is INF for y positive
455 self.assertEqualAndEqualSign(pow_op(INF, 0.5), INF)
456 self.assertEqualAndEqualSign(pow_op(INF, 1.0), INF)
457 self.assertEqualAndEqualSign(pow_op(INF, 2.0), INF)
458
459 # INF**y is 0.0 for y negative
460 self.assertEqualAndEqualSign(pow_op(INF, -2.0), 0.0)
461 self.assertEqualAndEqualSign(pow_op(INF, -1.0), 0.0)
462 self.assertEqualAndEqualSign(pow_op(INF, -0.5), 0.0)
463
464 # basic checks not covered by the special cases above
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 self.assertEqualAndEqualSign(pow_op(-1.0, -2.0), 1.0)
472 self.assertEqualAndEqualSign(pow_op(-1.0, -1.0), -1.0)
473 self.assertEqualAndEqualSign(pow_op(-1.0, -0.0), 1.0)
474 self.assertEqualAndEqualSign(pow_op(-1.0, 0.0), 1.0)
475 self.assertEqualAndEqualSign(pow_op(-1.0, 1.0), -1.0)
476 self.assertEqualAndEqualSign(pow_op(-1.0, 2.0), 1.0)
477 self.assertEqualAndEqualSign(pow_op(2.0, -2.0), 0.25)
478 self.assertEqualAndEqualSign(pow_op(2.0, -1.0), 0.5)
479 self.assertEqualAndEqualSign(pow_op(2.0, -0.0), 1.0)
480 self.assertEqualAndEqualSign(pow_op(2.0, 0.0), 1.0)
481 self.assertEqualAndEqualSign(pow_op(2.0, 1.0), 2.0)
482 self.assertEqualAndEqualSign(pow_op(2.0, 2.0), 4.0)
483
484 # 1 ** large and -1 ** large; some libms apparently
485 # have problems with these
486 self.assertEqualAndEqualSign(pow_op(1.0, -1e100), 1.0)
487 self.assertEqualAndEqualSign(pow_op(1.0, 1e100), 1.0)
488 self.assertEqualAndEqualSign(pow_op(-1.0, -1e100), 1.0)
489 self.assertEqualAndEqualSign(pow_op(-1.0, 1e100), 1.0)
490
491 # check sign for results that underflow to 0
492 self.assertEqualAndEqualSign(pow_op(-2.0, -2000.0), 0.0)
493 self.assertEqual(type(pow_op(-2.0, -2000.5)), complex)
494 self.assertEqualAndEqualSign(pow_op(-2.0, -2001.0), -0.0)
495 self.assertEqualAndEqualSign(pow_op(2.0, -2000.0), 0.0)
496 self.assertEqualAndEqualSign(pow_op(2.0, -2000.5), 0.0)
497 self.assertEqualAndEqualSign(pow_op(2.0, -2001.0), 0.0)
498 self.assertEqualAndEqualSign(pow_op(-0.5, 2000.0), 0.0)
499 self.assertEqual(type(pow_op(-0.5, 2000.5)), complex)
500 self.assertEqualAndEqualSign(pow_op(-0.5, 2001.0), -0.0)
501 self.assertEqualAndEqualSign(pow_op(0.5, 2000.0), 0.0)
502 self.assertEqualAndEqualSign(pow_op(0.5, 2000.5), 0.0)
503 self.assertEqualAndEqualSign(pow_op(0.5, 2001.0), 0.0)
504
505 # check we don't raise an exception for subnormal results,
506 # and validate signs. Tests currently disabled, since
507 # they fail on systems where a subnormal result from pow
508 # is flushed to zero (e.g. Debian/ia64.)
509 #self.assertTrue(0.0 < pow_op(0.5, 1048) < 1e-315)
510 #self.assertTrue(0.0 < pow_op(-0.5, 1048) < 1e-315)
511 #self.assertTrue(0.0 < pow_op(0.5, 1047) < 1e-315)
512 #self.assertTrue(0.0 > pow_op(-0.5, 1047) > -1e-315)
513 #self.assertTrue(0.0 < pow_op(2.0, -1048) < 1e-315)
514 #self.assertTrue(0.0 < pow_op(-2.0, -1048) < 1e-315)
515 #self.assertTrue(0.0 < pow_op(2.0, -1047) < 1e-315)
516 #self.assertTrue(0.0 > pow_op(-2.0, -1047) > -1e-315)
Mark Dickinson4a1f5932008-11-12 23:23:36 +0000517
518
Benjamin Petersone401c682010-07-02 23:25:44 +0000519@requires_setformat
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000520class FormatFunctionsTestCase(unittest.TestCase):
521
522 def setUp(self):
523 self.save_formats = {'double':float.__getformat__('double'),
524 'float':float.__getformat__('float')}
525
526 def tearDown(self):
527 float.__setformat__('double', self.save_formats['double'])
528 float.__setformat__('float', self.save_formats['float'])
529
530 def test_getformat(self):
Ezio Melottib58e0bd2010-01-23 15:40:09 +0000531 self.assertIn(float.__getformat__('double'),
532 ['unknown', 'IEEE, big-endian', 'IEEE, little-endian'])
533 self.assertIn(float.__getformat__('float'),
534 ['unknown', 'IEEE, big-endian', 'IEEE, little-endian'])
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000535 self.assertRaises(ValueError, float.__getformat__, 'chicken')
536 self.assertRaises(TypeError, float.__getformat__, 1)
537
538 def test_setformat(self):
539 for t in 'double', 'float':
540 float.__setformat__(t, 'unknown')
541 if self.save_formats[t] == 'IEEE, big-endian':
542 self.assertRaises(ValueError, float.__setformat__,
543 t, 'IEEE, little-endian')
544 elif self.save_formats[t] == 'IEEE, little-endian':
545 self.assertRaises(ValueError, float.__setformat__,
546 t, 'IEEE, big-endian')
547 else:
548 self.assertRaises(ValueError, float.__setformat__,
549 t, 'IEEE, big-endian')
550 self.assertRaises(ValueError, float.__setformat__,
551 t, 'IEEE, little-endian')
552 self.assertRaises(ValueError, float.__setformat__,
553 t, 'chicken')
554 self.assertRaises(ValueError, float.__setformat__,
555 'chicken', 'unknown')
556
Guido van Rossum2be161d2007-05-15 20:43:51 +0000557BE_DOUBLE_INF = b'\x7f\xf0\x00\x00\x00\x00\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000558LE_DOUBLE_INF = bytes(reversed(BE_DOUBLE_INF))
Guido van Rossum2be161d2007-05-15 20:43:51 +0000559BE_DOUBLE_NAN = b'\x7f\xf8\x00\x00\x00\x00\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000560LE_DOUBLE_NAN = bytes(reversed(BE_DOUBLE_NAN))
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000561
Guido van Rossum2be161d2007-05-15 20:43:51 +0000562BE_FLOAT_INF = b'\x7f\x80\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000563LE_FLOAT_INF = bytes(reversed(BE_FLOAT_INF))
Guido van Rossum2be161d2007-05-15 20:43:51 +0000564BE_FLOAT_NAN = b'\x7f\xc0\x00\x00'
Guido van Rossum254348e2007-11-21 19:29:53 +0000565LE_FLOAT_NAN = bytes(reversed(BE_FLOAT_NAN))
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000566
567# on non-IEEE platforms, attempting to unpack a bit pattern
568# representing an infinity or a NaN should raise an exception.
569
Benjamin Petersone401c682010-07-02 23:25:44 +0000570@requires_setformat
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000571class UnknownFormatTestCase(unittest.TestCase):
572 def setUp(self):
573 self.save_formats = {'double':float.__getformat__('double'),
574 'float':float.__getformat__('float')}
575 float.__setformat__('double', 'unknown')
576 float.__setformat__('float', 'unknown')
Tim Peters5d36a552005-06-03 22:40:27 +0000577
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000578 def tearDown(self):
579 float.__setformat__('double', self.save_formats['double'])
580 float.__setformat__('float', self.save_formats['float'])
581
582 def test_double_specials_dont_unpack(self):
583 for fmt, data in [('>d', BE_DOUBLE_INF),
584 ('>d', BE_DOUBLE_NAN),
585 ('<d', LE_DOUBLE_INF),
586 ('<d', LE_DOUBLE_NAN)]:
587 self.assertRaises(ValueError, struct.unpack, fmt, data)
588
589 def test_float_specials_dont_unpack(self):
590 for fmt, data in [('>f', BE_FLOAT_INF),
591 ('>f', BE_FLOAT_NAN),
592 ('<f', LE_FLOAT_INF),
593 ('<f', LE_FLOAT_NAN)]:
594 self.assertRaises(ValueError, struct.unpack, fmt, data)
595
596
597# on an IEEE platform, all we guarantee is that bit patterns
598# representing infinities or NaNs do not raise an exception; all else
599# is accident (today).
Guido van Rossum04110fb2007-08-24 16:32:05 +0000600# let's also try to guarantee that -0.0 and 0.0 don't get confused.
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000601
602class IEEEFormatTestCase(unittest.TestCase):
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000603
Eric Smith3ab08ca2010-12-04 15:17:38 +0000604 @support.requires_IEEE_754
Benjamin Petersone401c682010-07-02 23:25:44 +0000605 def test_double_specials_do_unpack(self):
606 for fmt, data in [('>d', BE_DOUBLE_INF),
607 ('>d', BE_DOUBLE_NAN),
608 ('<d', LE_DOUBLE_INF),
609 ('<d', LE_DOUBLE_NAN)]:
610 struct.unpack(fmt, data)
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000611
Eric Smith3ab08ca2010-12-04 15:17:38 +0000612 @support.requires_IEEE_754
Benjamin Petersone401c682010-07-02 23:25:44 +0000613 def test_float_specials_do_unpack(self):
614 for fmt, data in [('>f', BE_FLOAT_INF),
615 ('>f', BE_FLOAT_NAN),
616 ('<f', LE_FLOAT_INF),
617 ('<f', LE_FLOAT_NAN)]:
618 struct.unpack(fmt, data)
Guido van Rossum04110fb2007-08-24 16:32:05 +0000619
Benjamin Peterson2bb69a52017-09-10 23:50:46 -0700620 @support.requires_IEEE_754
621 def test_serialized_float_rounding(self):
622 from _testcapi import FLT_MAX
623 self.assertEqual(struct.pack("<f", 3.40282356e38), struct.pack("<f", FLT_MAX))
624 self.assertEqual(struct.pack("<f", -3.40282356e38), struct.pack("<f", -FLT_MAX))
625
Eric Smith8c663262007-08-25 02:26:07 +0000626class FormatTestCase(unittest.TestCase):
Benjamin Petersone401c682010-07-02 23:25:44 +0000627
Eric Smith11fe3e02007-08-31 01:33:06 +0000628 def test_format(self):
Eric Smith8c663262007-08-25 02:26:07 +0000629 # these should be rewritten to use both format(x, spec) and
630 # x.__format__(spec)
631
632 self.assertEqual(format(0.0, 'f'), '0.000000')
633
634 # the default is 'g', except for empty format spec
635 self.assertEqual(format(0.0, ''), '0.0')
636 self.assertEqual(format(0.01, ''), '0.01')
637 self.assertEqual(format(0.01, 'g'), '0.01')
638
Eric Smith63376222009-05-05 14:04:18 +0000639 # empty presentation type should format in the same way as str
640 # (issue 5920)
641 x = 100/7.
642 self.assertEqual(format(x, ''), str(x))
643 self.assertEqual(format(x, '-'), str(x))
644 self.assertEqual(format(x, '>'), str(x))
645 self.assertEqual(format(x, '2'), str(x))
Eric Smith8c663262007-08-25 02:26:07 +0000646
647 self.assertEqual(format(1.0, 'f'), '1.000000')
Eric Smith8c663262007-08-25 02:26:07 +0000648
649 self.assertEqual(format(-1.0, 'f'), '-1.000000')
Eric Smith8c663262007-08-25 02:26:07 +0000650
651 self.assertEqual(format( 1.0, ' f'), ' 1.000000')
652 self.assertEqual(format(-1.0, ' f'), '-1.000000')
653 self.assertEqual(format( 1.0, '+f'), '+1.000000')
654 self.assertEqual(format(-1.0, '+f'), '-1.000000')
655
656 # % formatting
657 self.assertEqual(format(-1.0, '%'), '-100.000000%')
658
659 # conversion to string should fail
660 self.assertRaises(ValueError, format, 3.0, "s")
661
Eric Smith7b69c6c2008-01-27 21:07:59 +0000662 # other format specifiers shouldn't work on floats,
663 # in particular int specifiers
664 for format_spec in ([chr(x) for x in range(ord('a'), ord('z')+1)] +
665 [chr(x) for x in range(ord('A'), ord('Z')+1)]):
666 if not format_spec in 'eEfFgGn%':
667 self.assertRaises(ValueError, format, 0.0, format_spec)
668 self.assertRaises(ValueError, format, 1.0, format_spec)
669 self.assertRaises(ValueError, format, -1.0, format_spec)
670 self.assertRaises(ValueError, format, 1e100, format_spec)
671 self.assertRaises(ValueError, format, -1e100, format_spec)
672 self.assertRaises(ValueError, format, 1e-100, format_spec)
673 self.assertRaises(ValueError, format, -1e-100, format_spec)
Michael W. Hudsonba283e22005-05-27 15:23:20 +0000674
Eric Smith741191f2009-05-06 13:08:15 +0000675 # issue 3382
676 self.assertEqual(format(NAN, 'f'), 'nan')
677 self.assertEqual(format(NAN, 'F'), 'NAN')
678 self.assertEqual(format(INF, 'f'), 'inf')
679 self.assertEqual(format(INF, 'F'), 'INF')
680
Eric Smith3ab08ca2010-12-04 15:17:38 +0000681 @support.requires_IEEE_754
Eric Smith0923d1d2009-04-16 20:16:10 +0000682 def test_format_testfile(self):
Brian Curtin076623b2010-10-31 00:03:45 +0000683 with open(format_testfile) as testfile:
684 for line in testfile:
685 if line.startswith('--'):
686 continue
687 line = line.strip()
688 if not line:
689 continue
Eric Smith0923d1d2009-04-16 20:16:10 +0000690
Brian Curtin076623b2010-10-31 00:03:45 +0000691 lhs, rhs = map(str.strip, line.split('->'))
692 fmt, arg = lhs.split()
693 self.assertEqual(fmt % float(arg), rhs)
694 self.assertEqual(fmt % -float(arg), '-' + rhs)
Eric Smith0923d1d2009-04-16 20:16:10 +0000695
Mark Dickinsond3ca5572009-04-29 18:47:07 +0000696 def test_issue5864(self):
Ezio Melottib3aedd42010-11-20 19:04:17 +0000697 self.assertEqual(format(123.456, '.4'), '123.5')
698 self.assertEqual(format(1234.56, '.4'), '1.235e+03')
699 self.assertEqual(format(12345.6, '.4'), '1.235e+04')
Mark Dickinsond3ca5572009-04-29 18:47:07 +0000700
Mark Dickinson7efad9e2009-04-17 20:59:58 +0000701class ReprTestCase(unittest.TestCase):
702 def test_repr(self):
703 floats_file = open(os.path.join(os.path.split(__file__)[0],
704 'floating_points.txt'))
705 for line in floats_file:
706 line = line.strip()
707 if not line or line.startswith('#'):
708 continue
709 v = eval(line)
710 self.assertEqual(v, eval(repr(v)))
711 floats_file.close()
712
Eric Smith0923d1d2009-04-16 20:16:10 +0000713 @unittest.skipUnless(getattr(sys, 'float_repr_style', '') == 'short',
714 "applies only when using short float repr style")
715 def test_short_repr(self):
716 # test short float repr introduced in Python 3.1. One aspect
717 # of this repr is that we get some degree of str -> float ->
718 # str roundtripping. In particular, for any numeric string
719 # containing 15 or fewer significant digits, those exact same
720 # digits (modulo trailing zeros) should appear in the output.
721 # No more repr(0.03) -> "0.029999999999999999"!
722
723 test_strings = [
724 # output always includes *either* a decimal point and at
725 # least one digit after that point, or an exponent.
726 '0.0',
727 '1.0',
728 '0.01',
729 '0.02',
730 '0.03',
731 '0.04',
732 '0.05',
733 '1.23456789',
734 '10.0',
735 '100.0',
736 # values >= 1e16 get an exponent...
737 '1000000000000000.0',
738 '9999999999999990.0',
739 '1e+16',
740 '1e+17',
741 # ... and so do values < 1e-4
742 '0.001',
743 '0.001001',
744 '0.00010000000000001',
745 '0.0001',
746 '9.999999999999e-05',
747 '1e-05',
748 # values designed to provoke failure if the FPU rounding
749 # precision isn't set correctly
750 '8.72293771110361e+25',
751 '7.47005307342313e+26',
752 '2.86438000439698e+28',
753 '8.89142905246179e+28',
754 '3.08578087079232e+35',
755 ]
756
757 for s in test_strings:
758 negs = '-'+s
759 self.assertEqual(s, repr(float(s)))
760 self.assertEqual(negs, repr(float(negs)))
Mark Dickinson388122d2010-08-04 20:56:28 +0000761 # Since Python 3.2, repr and str are identical
762 self.assertEqual(repr(float(s)), str(float(s)))
763 self.assertEqual(repr(float(negs)), str(float(negs)))
Benjamin Petersone401c682010-07-02 23:25:44 +0000764
Eric Smith3ab08ca2010-12-04 15:17:38 +0000765@support.requires_IEEE_754
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000766class RoundTestCase(unittest.TestCase):
Benjamin Petersone401c682010-07-02 23:25:44 +0000767
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000768 def test_inf_nan(self):
769 self.assertRaises(OverflowError, round, INF)
770 self.assertRaises(OverflowError, round, -INF)
771 self.assertRaises(ValueError, round, NAN)
Mark Dickinson4ca33d12009-11-24 10:59:34 +0000772 self.assertRaises(TypeError, round, INF, 0.0)
773 self.assertRaises(TypeError, round, -INF, 1.0)
774 self.assertRaises(TypeError, round, NAN, "ceci n'est pas un integer")
775 self.assertRaises(TypeError, round, -0.0, 1j)
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000776
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000777 def test_large_n(self):
778 for n in [324, 325, 400, 2**31-1, 2**31, 2**32, 2**100]:
779 self.assertEqual(round(123.456, n), 123.456)
780 self.assertEqual(round(-123.456, n), -123.456)
781 self.assertEqual(round(1e300, n), 1e300)
782 self.assertEqual(round(1e-320, n), 1e-320)
783 self.assertEqual(round(1e150, 300), 1e150)
784 self.assertEqual(round(1e300, 307), 1e300)
785 self.assertEqual(round(-3.1415, 308), -3.1415)
786 self.assertEqual(round(1e150, 309), 1e150)
787 self.assertEqual(round(1.4e-315, 315), 1e-315)
788
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000789 def test_small_n(self):
790 for n in [-308, -309, -400, 1-2**31, -2**31, -2**31-1, -2**100]:
791 self.assertEqual(round(123.456, n), 0.0)
792 self.assertEqual(round(-123.456, n), -0.0)
793 self.assertEqual(round(1e300, n), 0.0)
794 self.assertEqual(round(1e-320, n), 0.0)
795
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000796 def test_overflow(self):
797 self.assertRaises(OverflowError, round, 1.6e308, -308)
798 self.assertRaises(OverflowError, round, -1.7e308, -308)
799
800 @unittest.skipUnless(getattr(sys, 'float_repr_style', '') == 'short',
801 "applies only when using short float repr style")
802 def test_previous_round_bugs(self):
803 # particular cases that have occurred in bug reports
804 self.assertEqual(round(562949953421312.5, 1),
805 562949953421312.5)
806 self.assertEqual(round(56294995342131.5, 3),
807 56294995342131.5)
808 # round-half-even
809 self.assertEqual(round(25.0, -1), 20.0)
810 self.assertEqual(round(35.0, -1), 40.0)
811 self.assertEqual(round(45.0, -1), 40.0)
812 self.assertEqual(round(55.0, -1), 60.0)
813 self.assertEqual(round(65.0, -1), 60.0)
814 self.assertEqual(round(75.0, -1), 80.0)
815 self.assertEqual(round(85.0, -1), 80.0)
816 self.assertEqual(round(95.0, -1), 100.0)
817
818 @unittest.skipUnless(getattr(sys, 'float_repr_style', '') == 'short',
819 "applies only when using short float repr style")
820 def test_matches_float_format(self):
821 # round should give the same results as float formatting
822 for i in range(500):
823 x = i/1000.
824 self.assertEqual(float(format(x, '.0f')), round(x, 0))
825 self.assertEqual(float(format(x, '.1f')), round(x, 1))
826 self.assertEqual(float(format(x, '.2f')), round(x, 2))
827 self.assertEqual(float(format(x, '.3f')), round(x, 3))
828
829 for i in range(5, 5000, 10):
830 x = i/1000.
831 self.assertEqual(float(format(x, '.0f')), round(x, 0))
832 self.assertEqual(float(format(x, '.1f')), round(x, 1))
833 self.assertEqual(float(format(x, '.2f')), round(x, 2))
834 self.assertEqual(float(format(x, '.3f')), round(x, 3))
835
836 for i in range(500):
837 x = random.random()
838 self.assertEqual(float(format(x, '.0f')), round(x, 0))
839 self.assertEqual(float(format(x, '.1f')), round(x, 1))
840 self.assertEqual(float(format(x, '.2f')), round(x, 2))
841 self.assertEqual(float(format(x, '.3f')), round(x, 3))
842
Eric Smith8a10ecc2009-12-02 17:58:24 +0000843 def test_format_specials(self):
844 # Test formatting of nans and infs.
845
846 def test(fmt, value, expected):
847 # Test with both % and format().
848 self.assertEqual(fmt % value, expected, fmt)
Eric Smith984bb582010-11-25 16:08:06 +0000849 fmt = fmt[1:] # strip off the %
850 self.assertEqual(format(value, fmt), expected, fmt)
Eric Smith8a10ecc2009-12-02 17:58:24 +0000851
852 for fmt in ['%e', '%f', '%g', '%.0e', '%.6f', '%.20g',
853 '%#e', '%#f', '%#g', '%#.20e', '%#.15f', '%#.3g']:
854 pfmt = '%+' + fmt[1:]
855 sfmt = '% ' + fmt[1:]
856 test(fmt, INF, 'inf')
857 test(fmt, -INF, '-inf')
858 test(fmt, NAN, 'nan')
859 test(fmt, -NAN, 'nan')
860 # When asking for a sign, it's always provided. nans are
861 # always positive.
862 test(pfmt, INF, '+inf')
863 test(pfmt, -INF, '-inf')
864 test(pfmt, NAN, '+nan')
865 test(pfmt, -NAN, '+nan')
866 # When using ' ' for a sign code, only infs can be negative.
867 # Others have a space.
868 test(sfmt, INF, ' inf')
869 test(sfmt, -INF, '-inf')
870 test(sfmt, NAN, ' nan')
871 test(sfmt, -NAN, ' nan')
872
Steve Dowercb39d1f2015-04-15 16:10:59 -0400873 def test_None_ndigits(self):
874 for x in round(1.23), round(1.23, None), round(1.23, ndigits=None):
875 self.assertEqual(x, 1)
876 self.assertIsInstance(x, int)
877 for x in round(1.78), round(1.78, None), round(1.78, ndigits=None):
878 self.assertEqual(x, 2)
879 self.assertIsInstance(x, int)
880
Mark Dickinsone6a076d2009-04-18 11:48:33 +0000881
Christian Heimes99170a52007-12-19 02:07:34 +0000882# Beginning with Python 2.6 float has cross platform compatible
Georg Brandl2ee470f2008-07-16 12:55:28 +0000883# ways to create and represent inf and nan
Christian Heimes99170a52007-12-19 02:07:34 +0000884class InfNanTest(unittest.TestCase):
885 def test_inf_from_str(self):
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000886 self.assertTrue(isinf(float("inf")))
887 self.assertTrue(isinf(float("+inf")))
888 self.assertTrue(isinf(float("-inf")))
889 self.assertTrue(isinf(float("infinity")))
890 self.assertTrue(isinf(float("+infinity")))
891 self.assertTrue(isinf(float("-infinity")))
Christian Heimes99170a52007-12-19 02:07:34 +0000892
893 self.assertEqual(repr(float("inf")), "inf")
894 self.assertEqual(repr(float("+inf")), "inf")
895 self.assertEqual(repr(float("-inf")), "-inf")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000896 self.assertEqual(repr(float("infinity")), "inf")
897 self.assertEqual(repr(float("+infinity")), "inf")
898 self.assertEqual(repr(float("-infinity")), "-inf")
Christian Heimes99170a52007-12-19 02:07:34 +0000899
900 self.assertEqual(repr(float("INF")), "inf")
901 self.assertEqual(repr(float("+Inf")), "inf")
902 self.assertEqual(repr(float("-iNF")), "-inf")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000903 self.assertEqual(repr(float("Infinity")), "inf")
904 self.assertEqual(repr(float("+iNfInItY")), "inf")
905 self.assertEqual(repr(float("-INFINITY")), "-inf")
Christian Heimes99170a52007-12-19 02:07:34 +0000906
907 self.assertEqual(str(float("inf")), "inf")
908 self.assertEqual(str(float("+inf")), "inf")
909 self.assertEqual(str(float("-inf")), "-inf")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000910 self.assertEqual(str(float("infinity")), "inf")
911 self.assertEqual(str(float("+infinity")), "inf")
912 self.assertEqual(str(float("-infinity")), "-inf")
Christian Heimes99170a52007-12-19 02:07:34 +0000913
914 self.assertRaises(ValueError, float, "info")
915 self.assertRaises(ValueError, float, "+info")
916 self.assertRaises(ValueError, float, "-info")
917 self.assertRaises(ValueError, float, "in")
918 self.assertRaises(ValueError, float, "+in")
919 self.assertRaises(ValueError, float, "-in")
Georg Brandl2ee470f2008-07-16 12:55:28 +0000920 self.assertRaises(ValueError, float, "infinit")
921 self.assertRaises(ValueError, float, "+Infin")
922 self.assertRaises(ValueError, float, "-INFI")
923 self.assertRaises(ValueError, float, "infinitys")
Christian Heimes99170a52007-12-19 02:07:34 +0000924
Mark Dickinsonbd16edd2009-05-20 22:05:25 +0000925 self.assertRaises(ValueError, float, "++Inf")
926 self.assertRaises(ValueError, float, "-+inf")
927 self.assertRaises(ValueError, float, "+-infinity")
928 self.assertRaises(ValueError, float, "--Infinity")
929
Christian Heimes99170a52007-12-19 02:07:34 +0000930 def test_inf_as_str(self):
931 self.assertEqual(repr(1e300 * 1e300), "inf")
932 self.assertEqual(repr(-1e300 * 1e300), "-inf")
933
934 self.assertEqual(str(1e300 * 1e300), "inf")
935 self.assertEqual(str(-1e300 * 1e300), "-inf")
936
937 def test_nan_from_str(self):
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000938 self.assertTrue(isnan(float("nan")))
939 self.assertTrue(isnan(float("+nan")))
940 self.assertTrue(isnan(float("-nan")))
Christian Heimes99170a52007-12-19 02:07:34 +0000941
942 self.assertEqual(repr(float("nan")), "nan")
943 self.assertEqual(repr(float("+nan")), "nan")
944 self.assertEqual(repr(float("-nan")), "nan")
945
946 self.assertEqual(repr(float("NAN")), "nan")
947 self.assertEqual(repr(float("+NAn")), "nan")
948 self.assertEqual(repr(float("-NaN")), "nan")
949
950 self.assertEqual(str(float("nan")), "nan")
951 self.assertEqual(str(float("+nan")), "nan")
952 self.assertEqual(str(float("-nan")), "nan")
953
954 self.assertRaises(ValueError, float, "nana")
955 self.assertRaises(ValueError, float, "+nana")
956 self.assertRaises(ValueError, float, "-nana")
957 self.assertRaises(ValueError, float, "na")
958 self.assertRaises(ValueError, float, "+na")
959 self.assertRaises(ValueError, float, "-na")
960
Mark Dickinsonbd16edd2009-05-20 22:05:25 +0000961 self.assertRaises(ValueError, float, "++nan")
962 self.assertRaises(ValueError, float, "-+NAN")
963 self.assertRaises(ValueError, float, "+-NaN")
964 self.assertRaises(ValueError, float, "--nAn")
965
Christian Heimes99170a52007-12-19 02:07:34 +0000966 def test_nan_as_str(self):
967 self.assertEqual(repr(1e300 * 1e300 * 0), "nan")
968 self.assertEqual(repr(-1e300 * 1e300 * 0), "nan")
969
970 self.assertEqual(str(1e300 * 1e300 * 0), "nan")
971 self.assertEqual(str(-1e300 * 1e300 * 0), "nan")
Christian Heimes827b35c2007-12-10 22:19:17 +0000972
Mark Dickinsone383e822012-04-29 15:31:56 +0100973 def test_inf_signs(self):
974 self.assertEqual(copysign(1.0, float('inf')), 1.0)
975 self.assertEqual(copysign(1.0, float('-inf')), -1.0)
976
977 @unittest.skipUnless(getattr(sys, 'float_repr_style', '') == 'short',
978 "applies only when using short float repr style")
979 def test_nan_signs(self):
980 # When using the dtoa.c code, the sign of float('nan') should
981 # be predictable.
982 self.assertEqual(copysign(1.0, float('nan')), 1.0)
983 self.assertEqual(copysign(1.0, float('-nan')), -1.0)
984
985
Mark Dickinson65fe25e2008-07-16 11:30:51 +0000986fromHex = float.fromhex
987toHex = float.hex
988class HexFloatTestCase(unittest.TestCase):
989 MAX = fromHex('0x.fffffffffffff8p+1024') # max normal
990 MIN = fromHex('0x1p-1022') # min normal
991 TINY = fromHex('0x0.0000000000001p-1022') # min subnormal
992 EPS = fromHex('0x0.0000000000001p0') # diff between 1.0 and next float up
993
994 def identical(self, x, y):
995 # check that floats x and y are identical, or that both
996 # are NaNs
997 if isnan(x) or isnan(y):
998 if isnan(x) == isnan(y):
999 return
1000 elif x == y and (x != 0.0 or copysign(1.0, x) == copysign(1.0, y)):
1001 return
1002 self.fail('%r not identical to %r' % (x, y))
1003
1004 def test_ends(self):
Mark Dickinson38bbc482008-07-16 11:32:23 +00001005 self.identical(self.MIN, ldexp(1.0, -1022))
1006 self.identical(self.TINY, ldexp(1.0, -1074))
1007 self.identical(self.EPS, ldexp(1.0, -52))
1008 self.identical(self.MAX, 2.*(ldexp(1.0, 1023) - ldexp(1.0, 970)))
Mark Dickinson65fe25e2008-07-16 11:30:51 +00001009
1010 def test_invalid_inputs(self):
1011 invalid_inputs = [
1012 'infi', # misspelt infinities and nans
1013 '-Infinit',
1014 '++inf',
1015 '-+Inf',
1016 '--nan',
1017 '+-NaN',
1018 'snan',
1019 'NaNs',
1020 'nna',
Mark Dickinsond1ec8b22009-05-11 15:45:15 +00001021 'an',
1022 'nf',
1023 'nfinity',
1024 'inity',
1025 'iinity',
Mark Dickinson65fe25e2008-07-16 11:30:51 +00001026 '0xnan',
1027 '',
1028 ' ',
1029 'x1.0p0',
1030 '0xX1.0p0',
1031 '+ 0x1.0p0', # internal whitespace
1032 '- 0x1.0p0',
1033 '0 x1.0p0',
1034 '0x 1.0p0',
1035 '0x1 2.0p0',
1036 '+0x1 .0p0',
1037 '0x1. 0p0',
1038 '-0x1.0 1p0',
1039 '-0x1.0 p0',
1040 '+0x1.0p +0',
1041 '0x1.0p -0',
1042 '0x1.0p 0',
1043 '+0x1.0p+ 0',
1044 '-0x1.0p- 0',
1045 '++0x1.0p-0', # double signs
1046 '--0x1.0p0',
1047 '+-0x1.0p+0',
1048 '-+0x1.0p0',
1049 '0x1.0p++0',
1050 '+0x1.0p+-0',
1051 '-0x1.0p-+0',
1052 '0x1.0p--0',
1053 '0x1.0.p0',
1054 '0x.p0', # no hex digits before or after point
1055 '0x1,p0', # wrong decimal point character
1056 '0x1pa',
1057 '0x1p\uff10', # fullwidth Unicode digits
1058 '\uff10x1p0',
1059 '0x\uff11p0',
1060 '0x1.\uff10p0',
1061 '0x1p0 \n 0x2p0',
1062 '0x1p0\0 0x1p0', # embedded null byte is not end of string
1063 ]
1064 for x in invalid_inputs:
Mark Dickinson589b7952008-08-21 20:05:56 +00001065 try:
1066 result = fromHex(x)
1067 except ValueError:
1068 pass
1069 else:
1070 self.fail('Expected float.fromhex(%r) to raise ValueError; '
1071 'got %r instead' % (x, result))
Mark Dickinson65fe25e2008-07-16 11:30:51 +00001072
1073
Mark Dickinsond1ec8b22009-05-11 15:45:15 +00001074 def test_whitespace(self):
1075 value_pairs = [
1076 ('inf', INF),
1077 ('-Infinity', -INF),
1078 ('nan', NAN),
1079 ('1.0', 1.0),
1080 ('-0x.2', -0.125),
1081 ('-0.0', -0.0)
1082 ]
1083 whitespace = [
1084 '',
1085 ' ',
1086 '\t',
1087 '\n',
1088 '\n \t',
1089 '\f',
1090 '\v',
1091 '\r'
1092 ]
1093 for inp, expected in value_pairs:
1094 for lead in whitespace:
1095 for trail in whitespace:
1096 got = fromHex(lead + inp + trail)
1097 self.identical(got, expected)
1098
1099
Mark Dickinson65fe25e2008-07-16 11:30:51 +00001100 def test_from_hex(self):
1101 MIN = self.MIN;
1102 MAX = self.MAX;
1103 TINY = self.TINY;
1104 EPS = self.EPS;
1105
1106 # two spellings of infinity, with optional signs; case-insensitive
1107 self.identical(fromHex('inf'), INF)
1108 self.identical(fromHex('+Inf'), INF)
1109 self.identical(fromHex('-INF'), -INF)
1110 self.identical(fromHex('iNf'), INF)
1111 self.identical(fromHex('Infinity'), INF)
1112 self.identical(fromHex('+INFINITY'), INF)
1113 self.identical(fromHex('-infinity'), -INF)
1114 self.identical(fromHex('-iNFiNitY'), -INF)
1115
1116 # nans with optional sign; case insensitive
1117 self.identical(fromHex('nan'), NAN)
1118 self.identical(fromHex('+NaN'), NAN)
1119 self.identical(fromHex('-NaN'), NAN)
1120 self.identical(fromHex('-nAN'), NAN)
1121
1122 # variations in input format
1123 self.identical(fromHex('1'), 1.0)
1124 self.identical(fromHex('+1'), 1.0)
1125 self.identical(fromHex('1.'), 1.0)
1126 self.identical(fromHex('1.0'), 1.0)
1127 self.identical(fromHex('1.0p0'), 1.0)
1128 self.identical(fromHex('01'), 1.0)
1129 self.identical(fromHex('01.'), 1.0)
1130 self.identical(fromHex('0x1'), 1.0)
1131 self.identical(fromHex('0x1.'), 1.0)
1132 self.identical(fromHex('0x1.0'), 1.0)
1133 self.identical(fromHex('+0x1.0'), 1.0)
1134 self.identical(fromHex('0x1p0'), 1.0)
1135 self.identical(fromHex('0X1p0'), 1.0)
1136 self.identical(fromHex('0X1P0'), 1.0)
1137 self.identical(fromHex('0x1P0'), 1.0)
1138 self.identical(fromHex('0x1.p0'), 1.0)
1139 self.identical(fromHex('0x1.0p0'), 1.0)
1140 self.identical(fromHex('0x.1p4'), 1.0)
1141 self.identical(fromHex('0x.1p04'), 1.0)
1142 self.identical(fromHex('0x.1p004'), 1.0)
1143 self.identical(fromHex('0x1p+0'), 1.0)
1144 self.identical(fromHex('0x1P-0'), 1.0)
1145 self.identical(fromHex('+0x1p0'), 1.0)
1146 self.identical(fromHex('0x01p0'), 1.0)
1147 self.identical(fromHex('0x1p00'), 1.0)
1148 self.identical(fromHex(' 0x1p0 '), 1.0)
1149 self.identical(fromHex('\n 0x1p0'), 1.0)
1150 self.identical(fromHex('0x1p0 \t'), 1.0)
1151 self.identical(fromHex('0xap0'), 10.0)
1152 self.identical(fromHex('0xAp0'), 10.0)
1153 self.identical(fromHex('0xaP0'), 10.0)
1154 self.identical(fromHex('0xAP0'), 10.0)
1155 self.identical(fromHex('0xbep0'), 190.0)
1156 self.identical(fromHex('0xBep0'), 190.0)
1157 self.identical(fromHex('0xbEp0'), 190.0)
1158 self.identical(fromHex('0XBE0P-4'), 190.0)
1159 self.identical(fromHex('0xBEp0'), 190.0)
1160 self.identical(fromHex('0xB.Ep4'), 190.0)
1161 self.identical(fromHex('0x.BEp8'), 190.0)
1162 self.identical(fromHex('0x.0BEp12'), 190.0)
1163
1164 # moving the point around
1165 pi = fromHex('0x1.921fb54442d18p1')
1166 self.identical(fromHex('0x.006487ed5110b46p11'), pi)
1167 self.identical(fromHex('0x.00c90fdaa22168cp10'), pi)
1168 self.identical(fromHex('0x.01921fb54442d18p9'), pi)
1169 self.identical(fromHex('0x.03243f6a8885a3p8'), pi)
1170 self.identical(fromHex('0x.06487ed5110b46p7'), pi)
1171 self.identical(fromHex('0x.0c90fdaa22168cp6'), pi)
1172 self.identical(fromHex('0x.1921fb54442d18p5'), pi)
1173 self.identical(fromHex('0x.3243f6a8885a3p4'), pi)
1174 self.identical(fromHex('0x.6487ed5110b46p3'), pi)
1175 self.identical(fromHex('0x.c90fdaa22168cp2'), pi)
1176 self.identical(fromHex('0x1.921fb54442d18p1'), pi)
1177 self.identical(fromHex('0x3.243f6a8885a3p0'), pi)
1178 self.identical(fromHex('0x6.487ed5110b46p-1'), pi)
1179 self.identical(fromHex('0xc.90fdaa22168cp-2'), pi)
1180 self.identical(fromHex('0x19.21fb54442d18p-3'), pi)
1181 self.identical(fromHex('0x32.43f6a8885a3p-4'), pi)
1182 self.identical(fromHex('0x64.87ed5110b46p-5'), pi)
1183 self.identical(fromHex('0xc9.0fdaa22168cp-6'), pi)
1184 self.identical(fromHex('0x192.1fb54442d18p-7'), pi)
1185 self.identical(fromHex('0x324.3f6a8885a3p-8'), pi)
1186 self.identical(fromHex('0x648.7ed5110b46p-9'), pi)
1187 self.identical(fromHex('0xc90.fdaa22168cp-10'), pi)
1188 self.identical(fromHex('0x1921.fb54442d18p-11'), pi)
1189 # ...
1190 self.identical(fromHex('0x1921fb54442d1.8p-47'), pi)
1191 self.identical(fromHex('0x3243f6a8885a3p-48'), pi)
1192 self.identical(fromHex('0x6487ed5110b46p-49'), pi)
1193 self.identical(fromHex('0xc90fdaa22168cp-50'), pi)
1194 self.identical(fromHex('0x1921fb54442d18p-51'), pi)
1195 self.identical(fromHex('0x3243f6a8885a30p-52'), pi)
1196 self.identical(fromHex('0x6487ed5110b460p-53'), pi)
1197 self.identical(fromHex('0xc90fdaa22168c0p-54'), pi)
1198 self.identical(fromHex('0x1921fb54442d180p-55'), pi)
1199
1200
1201 # results that should overflow...
1202 self.assertRaises(OverflowError, fromHex, '-0x1p1024')
1203 self.assertRaises(OverflowError, fromHex, '0x1p+1025')
1204 self.assertRaises(OverflowError, fromHex, '+0X1p1030')
1205 self.assertRaises(OverflowError, fromHex, '-0x1p+1100')
1206 self.assertRaises(OverflowError, fromHex, '0X1p123456789123456789')
1207 self.assertRaises(OverflowError, fromHex, '+0X.8p+1025')
1208 self.assertRaises(OverflowError, fromHex, '+0x0.8p1025')
1209 self.assertRaises(OverflowError, fromHex, '-0x0.4p1026')
1210 self.assertRaises(OverflowError, fromHex, '0X2p+1023')
1211 self.assertRaises(OverflowError, fromHex, '0x2.p1023')
1212 self.assertRaises(OverflowError, fromHex, '-0x2.0p+1023')
1213 self.assertRaises(OverflowError, fromHex, '+0X4p+1022')
1214 self.assertRaises(OverflowError, fromHex, '0x1.ffffffffffffffp+1023')
1215 self.assertRaises(OverflowError, fromHex, '-0X1.fffffffffffff9p1023')
1216 self.assertRaises(OverflowError, fromHex, '0X1.fffffffffffff8p1023')
1217 self.assertRaises(OverflowError, fromHex, '+0x3.fffffffffffffp1022')
1218 self.assertRaises(OverflowError, fromHex, '0x3fffffffffffffp+970')
1219 self.assertRaises(OverflowError, fromHex, '0x10000000000000000p960')
1220 self.assertRaises(OverflowError, fromHex, '-0Xffffffffffffffffp960')
1221
1222 # ...and those that round to +-max float
1223 self.identical(fromHex('+0x1.fffffffffffffp+1023'), MAX)
1224 self.identical(fromHex('-0X1.fffffffffffff7p1023'), -MAX)
1225 self.identical(fromHex('0X1.fffffffffffff7fffffffffffffp1023'), MAX)
1226
1227 # zeros
1228 self.identical(fromHex('0x0p0'), 0.0)
1229 self.identical(fromHex('0x0p1000'), 0.0)
1230 self.identical(fromHex('-0x0p1023'), -0.0)
1231 self.identical(fromHex('0X0p1024'), 0.0)
1232 self.identical(fromHex('-0x0p1025'), -0.0)
1233 self.identical(fromHex('0X0p2000'), 0.0)
1234 self.identical(fromHex('0x0p123456789123456789'), 0.0)
1235 self.identical(fromHex('-0X0p-0'), -0.0)
1236 self.identical(fromHex('-0X0p-1000'), -0.0)
1237 self.identical(fromHex('0x0p-1023'), 0.0)
1238 self.identical(fromHex('-0X0p-1024'), -0.0)
1239 self.identical(fromHex('-0x0p-1025'), -0.0)
1240 self.identical(fromHex('-0x0p-1072'), -0.0)
1241 self.identical(fromHex('0X0p-1073'), 0.0)
1242 self.identical(fromHex('-0x0p-1074'), -0.0)
1243 self.identical(fromHex('0x0p-1075'), 0.0)
1244 self.identical(fromHex('0X0p-1076'), 0.0)
1245 self.identical(fromHex('-0X0p-2000'), -0.0)
1246 self.identical(fromHex('-0x0p-123456789123456789'), -0.0)
1247
1248 # values that should underflow to 0
1249 self.identical(fromHex('0X1p-1075'), 0.0)
1250 self.identical(fromHex('-0X1p-1075'), -0.0)
1251 self.identical(fromHex('-0x1p-123456789123456789'), -0.0)
1252 self.identical(fromHex('0x1.00000000000000001p-1075'), TINY)
1253 self.identical(fromHex('-0x1.1p-1075'), -TINY)
1254 self.identical(fromHex('0x1.fffffffffffffffffp-1075'), TINY)
1255
1256 # check round-half-even is working correctly near 0 ...
1257 self.identical(fromHex('0x1p-1076'), 0.0)
1258 self.identical(fromHex('0X2p-1076'), 0.0)
1259 self.identical(fromHex('0X3p-1076'), TINY)
1260 self.identical(fromHex('0x4p-1076'), TINY)
1261 self.identical(fromHex('0X5p-1076'), TINY)
1262 self.identical(fromHex('0X6p-1076'), 2*TINY)
1263 self.identical(fromHex('0x7p-1076'), 2*TINY)
1264 self.identical(fromHex('0X8p-1076'), 2*TINY)
1265 self.identical(fromHex('0X9p-1076'), 2*TINY)
1266 self.identical(fromHex('0xap-1076'), 2*TINY)
1267 self.identical(fromHex('0Xbp-1076'), 3*TINY)
1268 self.identical(fromHex('0xcp-1076'), 3*TINY)
1269 self.identical(fromHex('0Xdp-1076'), 3*TINY)
1270 self.identical(fromHex('0Xep-1076'), 4*TINY)
1271 self.identical(fromHex('0xfp-1076'), 4*TINY)
1272 self.identical(fromHex('0x10p-1076'), 4*TINY)
1273 self.identical(fromHex('-0x1p-1076'), -0.0)
1274 self.identical(fromHex('-0X2p-1076'), -0.0)
1275 self.identical(fromHex('-0x3p-1076'), -TINY)
1276 self.identical(fromHex('-0X4p-1076'), -TINY)
1277 self.identical(fromHex('-0x5p-1076'), -TINY)
1278 self.identical(fromHex('-0x6p-1076'), -2*TINY)
1279 self.identical(fromHex('-0X7p-1076'), -2*TINY)
1280 self.identical(fromHex('-0X8p-1076'), -2*TINY)
1281 self.identical(fromHex('-0X9p-1076'), -2*TINY)
1282 self.identical(fromHex('-0Xap-1076'), -2*TINY)
1283 self.identical(fromHex('-0xbp-1076'), -3*TINY)
1284 self.identical(fromHex('-0xcp-1076'), -3*TINY)
1285 self.identical(fromHex('-0Xdp-1076'), -3*TINY)
1286 self.identical(fromHex('-0xep-1076'), -4*TINY)
1287 self.identical(fromHex('-0Xfp-1076'), -4*TINY)
1288 self.identical(fromHex('-0X10p-1076'), -4*TINY)
1289
1290 # ... and near MIN ...
1291 self.identical(fromHex('0x0.ffffffffffffd6p-1022'), MIN-3*TINY)
1292 self.identical(fromHex('0x0.ffffffffffffd8p-1022'), MIN-2*TINY)
1293 self.identical(fromHex('0x0.ffffffffffffdap-1022'), MIN-2*TINY)
1294 self.identical(fromHex('0x0.ffffffffffffdcp-1022'), MIN-2*TINY)
1295 self.identical(fromHex('0x0.ffffffffffffdep-1022'), MIN-2*TINY)
1296 self.identical(fromHex('0x0.ffffffffffffe0p-1022'), MIN-2*TINY)
1297 self.identical(fromHex('0x0.ffffffffffffe2p-1022'), MIN-2*TINY)
1298 self.identical(fromHex('0x0.ffffffffffffe4p-1022'), MIN-2*TINY)
1299 self.identical(fromHex('0x0.ffffffffffffe6p-1022'), MIN-2*TINY)
1300 self.identical(fromHex('0x0.ffffffffffffe8p-1022'), MIN-2*TINY)
1301 self.identical(fromHex('0x0.ffffffffffffeap-1022'), MIN-TINY)
1302 self.identical(fromHex('0x0.ffffffffffffecp-1022'), MIN-TINY)
1303 self.identical(fromHex('0x0.ffffffffffffeep-1022'), MIN-TINY)
1304 self.identical(fromHex('0x0.fffffffffffff0p-1022'), MIN-TINY)
1305 self.identical(fromHex('0x0.fffffffffffff2p-1022'), MIN-TINY)
1306 self.identical(fromHex('0x0.fffffffffffff4p-1022'), MIN-TINY)
1307 self.identical(fromHex('0x0.fffffffffffff6p-1022'), MIN-TINY)
1308 self.identical(fromHex('0x0.fffffffffffff8p-1022'), MIN)
1309 self.identical(fromHex('0x0.fffffffffffffap-1022'), MIN)
1310 self.identical(fromHex('0x0.fffffffffffffcp-1022'), MIN)
1311 self.identical(fromHex('0x0.fffffffffffffep-1022'), MIN)
1312 self.identical(fromHex('0x1.00000000000000p-1022'), MIN)
1313 self.identical(fromHex('0x1.00000000000002p-1022'), MIN)
1314 self.identical(fromHex('0x1.00000000000004p-1022'), MIN)
1315 self.identical(fromHex('0x1.00000000000006p-1022'), MIN)
1316 self.identical(fromHex('0x1.00000000000008p-1022'), MIN)
1317 self.identical(fromHex('0x1.0000000000000ap-1022'), MIN+TINY)
1318 self.identical(fromHex('0x1.0000000000000cp-1022'), MIN+TINY)
1319 self.identical(fromHex('0x1.0000000000000ep-1022'), MIN+TINY)
1320 self.identical(fromHex('0x1.00000000000010p-1022'), MIN+TINY)
1321 self.identical(fromHex('0x1.00000000000012p-1022'), MIN+TINY)
1322 self.identical(fromHex('0x1.00000000000014p-1022'), MIN+TINY)
1323 self.identical(fromHex('0x1.00000000000016p-1022'), MIN+TINY)
1324 self.identical(fromHex('0x1.00000000000018p-1022'), MIN+2*TINY)
1325
1326 # ... and near 1.0.
1327 self.identical(fromHex('0x0.fffffffffffff0p0'), 1.0-EPS)
1328 self.identical(fromHex('0x0.fffffffffffff1p0'), 1.0-EPS)
1329 self.identical(fromHex('0X0.fffffffffffff2p0'), 1.0-EPS)
1330 self.identical(fromHex('0x0.fffffffffffff3p0'), 1.0-EPS)
1331 self.identical(fromHex('0X0.fffffffffffff4p0'), 1.0-EPS)
1332 self.identical(fromHex('0X0.fffffffffffff5p0'), 1.0-EPS/2)
1333 self.identical(fromHex('0X0.fffffffffffff6p0'), 1.0-EPS/2)
1334 self.identical(fromHex('0x0.fffffffffffff7p0'), 1.0-EPS/2)
1335 self.identical(fromHex('0x0.fffffffffffff8p0'), 1.0-EPS/2)
1336 self.identical(fromHex('0X0.fffffffffffff9p0'), 1.0-EPS/2)
1337 self.identical(fromHex('0X0.fffffffffffffap0'), 1.0-EPS/2)
1338 self.identical(fromHex('0x0.fffffffffffffbp0'), 1.0-EPS/2)
1339 self.identical(fromHex('0X0.fffffffffffffcp0'), 1.0)
1340 self.identical(fromHex('0x0.fffffffffffffdp0'), 1.0)
1341 self.identical(fromHex('0X0.fffffffffffffep0'), 1.0)
1342 self.identical(fromHex('0x0.ffffffffffffffp0'), 1.0)
1343 self.identical(fromHex('0X1.00000000000000p0'), 1.0)
1344 self.identical(fromHex('0X1.00000000000001p0'), 1.0)
1345 self.identical(fromHex('0x1.00000000000002p0'), 1.0)
1346 self.identical(fromHex('0X1.00000000000003p0'), 1.0)
1347 self.identical(fromHex('0x1.00000000000004p0'), 1.0)
1348 self.identical(fromHex('0X1.00000000000005p0'), 1.0)
1349 self.identical(fromHex('0X1.00000000000006p0'), 1.0)
1350 self.identical(fromHex('0X1.00000000000007p0'), 1.0)
1351 self.identical(fromHex('0x1.00000000000007ffffffffffffffffffffp0'),
1352 1.0)
1353 self.identical(fromHex('0x1.00000000000008p0'), 1.0)
1354 self.identical(fromHex('0x1.00000000000008000000000000000001p0'),
1355 1+EPS)
1356 self.identical(fromHex('0X1.00000000000009p0'), 1.0+EPS)
1357 self.identical(fromHex('0x1.0000000000000ap0'), 1.0+EPS)
1358 self.identical(fromHex('0x1.0000000000000bp0'), 1.0+EPS)
1359 self.identical(fromHex('0X1.0000000000000cp0'), 1.0+EPS)
1360 self.identical(fromHex('0x1.0000000000000dp0'), 1.0+EPS)
1361 self.identical(fromHex('0x1.0000000000000ep0'), 1.0+EPS)
1362 self.identical(fromHex('0X1.0000000000000fp0'), 1.0+EPS)
1363 self.identical(fromHex('0x1.00000000000010p0'), 1.0+EPS)
1364 self.identical(fromHex('0X1.00000000000011p0'), 1.0+EPS)
1365 self.identical(fromHex('0x1.00000000000012p0'), 1.0+EPS)
1366 self.identical(fromHex('0X1.00000000000013p0'), 1.0+EPS)
1367 self.identical(fromHex('0X1.00000000000014p0'), 1.0+EPS)
1368 self.identical(fromHex('0x1.00000000000015p0'), 1.0+EPS)
1369 self.identical(fromHex('0x1.00000000000016p0'), 1.0+EPS)
1370 self.identical(fromHex('0X1.00000000000017p0'), 1.0+EPS)
1371 self.identical(fromHex('0x1.00000000000017ffffffffffffffffffffp0'),
1372 1.0+EPS)
1373 self.identical(fromHex('0x1.00000000000018p0'), 1.0+2*EPS)
1374 self.identical(fromHex('0X1.00000000000018000000000000000001p0'),
1375 1.0+2*EPS)
1376 self.identical(fromHex('0x1.00000000000019p0'), 1.0+2*EPS)
1377 self.identical(fromHex('0X1.0000000000001ap0'), 1.0+2*EPS)
1378 self.identical(fromHex('0X1.0000000000001bp0'), 1.0+2*EPS)
1379 self.identical(fromHex('0x1.0000000000001cp0'), 1.0+2*EPS)
1380 self.identical(fromHex('0x1.0000000000001dp0'), 1.0+2*EPS)
1381 self.identical(fromHex('0x1.0000000000001ep0'), 1.0+2*EPS)
1382 self.identical(fromHex('0X1.0000000000001fp0'), 1.0+2*EPS)
1383 self.identical(fromHex('0x1.00000000000020p0'), 1.0+2*EPS)
1384
1385 def test_roundtrip(self):
1386 def roundtrip(x):
1387 return fromHex(toHex(x))
1388
1389 for x in [NAN, INF, self.MAX, self.MIN, self.MIN-self.TINY, self.TINY, 0.0]:
1390 self.identical(x, roundtrip(x))
1391 self.identical(-x, roundtrip(-x))
1392
1393 # fromHex(toHex(x)) should exactly recover x, for any non-NaN float x.
1394 import random
1395 for i in range(10000):
1396 e = random.randrange(-1200, 1200)
1397 m = random.random()
1398 s = random.choice([1.0, -1.0])
1399 try:
1400 x = s*ldexp(m, e)
1401 except OverflowError:
1402 pass
1403 else:
1404 self.identical(x, fromHex(toHex(x)))
1405
Serhiy Storchakaea36c942016-05-12 10:37:58 +03001406 def test_subclass(self):
1407 class F(float):
1408 def __new__(cls, value):
1409 return float.__new__(cls, value + 1)
1410
1411 f = F.fromhex((1.5).hex())
1412 self.assertIs(type(f), F)
1413 self.assertEqual(f, 2.5)
1414
1415 class F2(float):
1416 def __init__(self, value):
1417 self.foo = 'bar'
1418
1419 f = F2.fromhex((1.5).hex())
1420 self.assertIs(type(f), F2)
1421 self.assertEqual(f, 1.5)
1422 self.assertEqual(getattr(f, 'foo', 'none'), 'bar')
1423
Christian Heimes53876d92008-04-19 00:31:39 +00001424
Michael W. Hudsonba283e22005-05-27 15:23:20 +00001425if __name__ == '__main__':
Zachary Ware38c707e2015-04-13 15:00:43 -05001426 unittest.main()