blob: 0963bcc0ef9be78e97387f0887f3eb33eab9bb72 [file] [log] [blame]
Walter Dörwald5edd7852003-06-18 14:26:18 +00001import unittest
2from test import test_support
3
4import warnings
5warnings.filterwarnings(
6 "ignore",
7 category=DeprecationWarning,
8 message=".*complex divmod.*are deprecated"
9)
10
Tim Peters0f336042001-03-18 08:21:57 +000011from random import random
12
Walter Dörwald5edd7852003-06-18 14:26:18 +000013# These tests ensure that complex math does the right thing
Fred Drake68773e72001-12-13 19:57:53 +000014
Walter Dörwald5edd7852003-06-18 14:26:18 +000015class ComplexTest(unittest.TestCase):
Tim Peters0f336042001-03-18 08:21:57 +000016
Walter Dörwald5edd7852003-06-18 14:26:18 +000017 def assertAlmostEqual(self, a, b):
18 if isinstance(a, complex):
19 if isinstance(b, complex):
20 unittest.TestCase.assertAlmostEqual(self, a.real, b.real)
21 unittest.TestCase.assertAlmostEqual(self, a.imag, b.imag)
22 else:
23 unittest.TestCase.assertAlmostEqual(self, a.real, b)
24 unittest.TestCase.assertAlmostEqual(self, a.imag, 0.)
25 else:
26 if isinstance(b, complex):
27 unittest.TestCase.assertAlmostEqual(self, a, b.real)
28 unittest.TestCase.assertAlmostEqual(self, 0., b.imag)
29 else:
30 unittest.TestCase.assertAlmostEqual(self, a, b)
Tim Peters0f336042001-03-18 08:21:57 +000031
Walter Dörwald5edd7852003-06-18 14:26:18 +000032 def assertCloseAbs(self, x, y, eps=1e-9):
33 """Return true iff floats x and y "are close\""""
34 # put the one with larger magnitude second
35 if abs(x) > abs(y):
36 x, y = y, x
37 if y == 0:
38 return abs(x) < eps
39 if x == 0:
40 return abs(y) < eps
41 # check that relative difference < eps
42 self.assert_(abs((x-y)/y) < eps)
Tim Petersc5b235c2001-09-06 23:00:21 +000043
Walter Dörwald5edd7852003-06-18 14:26:18 +000044 def assertClose(self, x, y, eps=1e-9):
45 """Return true iff complexes x and y "are close\""""
46 self.assertCloseAbs(x.real, y.real, eps)
47 self.assertCloseAbs(x.imag, y.imag, eps)
Tim Peters0f336042001-03-18 08:21:57 +000048
Walter Dörwald5edd7852003-06-18 14:26:18 +000049 def assertIs(self, a, b):
50 self.assert_(a is b)
Tim Peters0f336042001-03-18 08:21:57 +000051
Walter Dörwald5edd7852003-06-18 14:26:18 +000052 def check_div(self, x, y):
53 """Compute complex z=x*y, and check that z/x==y and z/y==x."""
54 z = x * y
55 if x != 0:
56 q = z / x
57 self.assertClose(q, y)
58 if y != 0:
59 q = z / y
60 self.assertClose(q, x)
Tim Peters0f336042001-03-18 08:21:57 +000061
Walter Dörwald5edd7852003-06-18 14:26:18 +000062 def test_div(self):
63 simple_real = [float(i) for i in xrange(-5, 6)]
64 simple_complex = [complex(x, y) for x in simple_real for y in simple_real]
65 for x in simple_complex:
66 for y in simple_complex:
67 self.check_div(x, y)
Tim Peters0f336042001-03-18 08:21:57 +000068
Walter Dörwald5edd7852003-06-18 14:26:18 +000069 # A naive complex division algorithm (such as in 2.0) is very prone to
70 # nonsense errors for these (overflows and underflows).
71 self.check_div(complex(1e200, 1e200), 1+0j)
72 self.check_div(complex(1e-200, 1e-200), 1+0j)
Tim Peters0f336042001-03-18 08:21:57 +000073
Walter Dörwald5edd7852003-06-18 14:26:18 +000074 # Just for fun.
75 for i in xrange(100):
76 self.check_div(complex(random(), random()),
77 complex(random(), random()))
Neal Norwitzfc37af82001-12-29 01:02:21 +000078
Walter Dörwald5edd7852003-06-18 14:26:18 +000079 self.assertRaises(ZeroDivisionError, complex.__div__, 1+1j, 0+0j)
80 # FIXME: The following currently crashes on Alpha
81 # self.assertRaises(OverflowError, pow, 1e200+1j, 1e200+1j)
Neal Norwitzfc37af82001-12-29 01:02:21 +000082
Walter Dörwald5edd7852003-06-18 14:26:18 +000083 def test_truediv(self):
84 self.assertAlmostEqual(complex.__truediv__(2+0j, 1+1j), 1-1j)
85 self.assertRaises(ZeroDivisionError, complex.__truediv__, 1+1j, 0+0j)
Neal Norwitz5a0f0102001-12-29 14:31:46 +000086
Walter Dörwald5edd7852003-06-18 14:26:18 +000087 def test_floordiv(self):
88 self.assertAlmostEqual(complex.__floordiv__(3+0j, 1.5+0j), 2)
89 self.assertRaises(ZeroDivisionError, complex.__floordiv__, 3+0j, 0+0j)
Neal Norwitzfc37af82001-12-29 01:02:21 +000090
Walter Dörwald5edd7852003-06-18 14:26:18 +000091 def test_coerce(self):
92 self.assertRaises(OverflowError, complex.__coerce__, 1+1j, 1L<<10000)
Neal Norwitzfc37af82001-12-29 01:02:21 +000093
Walter Dörwald5edd7852003-06-18 14:26:18 +000094 def test_richcompare(self):
95 self.assertRaises(OverflowError, complex.__eq__, 1+1j, 1L<<10000)
96 self.assertEqual(complex.__lt__(1+1j, None), NotImplemented)
97 self.assertIs(complex.__eq__(1+1j, 1+1j), True)
98 self.assertIs(complex.__eq__(1+1j, 2+2j), False)
99 self.assertIs(complex.__ne__(1+1j, 1+1j), False)
100 self.assertIs(complex.__ne__(1+1j, 2+2j), True)
101 self.assertRaises(TypeError, complex.__lt__, 1+1j, 2+2j)
102 self.assertRaises(TypeError, complex.__le__, 1+1j, 2+2j)
103 self.assertRaises(TypeError, complex.__gt__, 1+1j, 2+2j)
104 self.assertRaises(TypeError, complex.__ge__, 1+1j, 2+2j)
Tim Peters0f336042001-03-18 08:21:57 +0000105
Walter Dörwald5edd7852003-06-18 14:26:18 +0000106 def test_mod(self):
107 self.assertRaises(ZeroDivisionError, (1+1j).__mod__, 0+0j)
108
109 def test_divmod(self):
110 self.assertRaises(ZeroDivisionError, divmod, 1+1j, 0+0j)
111
112 def test_pow(self):
113 self.assertAlmostEqual(pow(1+1j, 0+0j), 1.0)
114 self.assertAlmostEqual(pow(0+0j, 2+0j), 0.0)
115 self.assertRaises(ZeroDivisionError, pow, 0+0j, 1j)
116 self.assertAlmostEqual(pow(1j, -1), 1/1j)
117 self.assertAlmostEqual(pow(1j, 200), 1)
118 self.assertRaises(ValueError, pow, 1+1j, 1+1j, 1+1j)
119
120 def test_boolcontext(self):
121 for i in xrange(100):
122 self.assert_(complex(random() + 1e-6, random() + 1e-6))
123 self.assert_(not complex(0.0, 0.0))
124
125 def test_conjugate(self):
126 self.assertClose(complex(5.3, 9.8).conjugate(), 5.3-9.8j)
127
128 def test_constructor(self):
129 class OS:
130 def __init__(self, value): self.value = value
131 def __complex__(self): return self.value
132 class NS(object):
133 def __init__(self, value): self.value = value
134 def __complex__(self): return self.value
135 self.assertEqual(complex(OS(1+10j)), 1+10j)
136 self.assertEqual(complex(NS(1+10j)), 1+10j)
137 self.assertRaises(TypeError, complex, OS(None))
138 self.assertRaises(TypeError, complex, NS(None))
139
140 self.assertAlmostEqual(complex("1+10j"), 1+10j)
141 self.assertAlmostEqual(complex(10), 10+0j)
142 self.assertAlmostEqual(complex(10.0), 10+0j)
143 self.assertAlmostEqual(complex(10L), 10+0j)
144 self.assertAlmostEqual(complex(10+0j), 10+0j)
145 self.assertAlmostEqual(complex(1,10), 1+10j)
146 self.assertAlmostEqual(complex(1,10L), 1+10j)
147 self.assertAlmostEqual(complex(1,10.0), 1+10j)
148 self.assertAlmostEqual(complex(1L,10), 1+10j)
149 self.assertAlmostEqual(complex(1L,10L), 1+10j)
150 self.assertAlmostEqual(complex(1L,10.0), 1+10j)
151 self.assertAlmostEqual(complex(1.0,10), 1+10j)
152 self.assertAlmostEqual(complex(1.0,10L), 1+10j)
153 self.assertAlmostEqual(complex(1.0,10.0), 1+10j)
154 self.assertAlmostEqual(complex(3.14+0j), 3.14+0j)
155 self.assertAlmostEqual(complex(3.14), 3.14+0j)
156 self.assertAlmostEqual(complex(314), 314.0+0j)
157 self.assertAlmostEqual(complex(314L), 314.0+0j)
158 self.assertAlmostEqual(complex(3.14+0j, 0j), 3.14+0j)
159 self.assertAlmostEqual(complex(3.14, 0.0), 3.14+0j)
160 self.assertAlmostEqual(complex(314, 0), 314.0+0j)
161 self.assertAlmostEqual(complex(314L, 0L), 314.0+0j)
162 self.assertAlmostEqual(complex(0j, 3.14j), -3.14+0j)
163 self.assertAlmostEqual(complex(0.0, 3.14j), -3.14+0j)
164 self.assertAlmostEqual(complex(0j, 3.14), 3.14j)
165 self.assertAlmostEqual(complex(0.0, 3.14), 3.14j)
166 self.assertAlmostEqual(complex("1"), 1+0j)
167 self.assertAlmostEqual(complex("1j"), 1j)
168 self.assertAlmostEqual(complex(), 0)
169 self.assertAlmostEqual(complex("-1"), -1)
170 self.assertAlmostEqual(complex("+1"), +1)
171
172 class complex2(complex): pass
173 self.assertAlmostEqual(complex(complex2(1+1j)), 1+1j)
174 self.assertAlmostEqual(complex(real=17, imag=23), 17+23j)
175 self.assertAlmostEqual(complex(real=17+23j), 17+23j)
176 self.assertAlmostEqual(complex(real=17+23j, imag=23), 17+46j)
177 self.assertAlmostEqual(complex(real=1+2j, imag=3+4j), -3+5j)
178
179 c = 3.14 + 1j
180 self.assert_(complex(c) is c)
181 del c
182
183 self.assertRaises(TypeError, complex, "1", "1")
184 self.assertRaises(TypeError, complex, 1, "1")
185
186 self.assertEqual(complex(" 3.14+J "), 3.14+1j)
187 if test_support.have_unicode:
188 self.assertEqual(complex(unicode(" 3.14+J ")), 3.14+1j)
189
190 # SF bug 543840: complex(string) accepts strings with \0
191 # Fixed in 2.3.
192 self.assertRaises(ValueError, complex, '1+1j\0j')
193
194 self.assertRaises(TypeError, int, 5+3j)
195 self.assertRaises(TypeError, long, 5+3j)
196 self.assertRaises(TypeError, float, 5+3j)
197 self.assertRaises(ValueError, complex, "")
198 self.assertRaises(TypeError, complex, None)
199 self.assertRaises(ValueError, complex, "\0")
200 self.assertRaises(TypeError, complex, "1", "2")
201 self.assertRaises(TypeError, complex, "1", 42)
202 self.assertRaises(TypeError, complex, 1, "2")
203 self.assertRaises(ValueError, complex, "1+")
204 self.assertRaises(ValueError, complex, "1+1j+1j")
205 self.assertRaises(ValueError, complex, "--")
206 if test_support.have_unicode:
207 self.assertRaises(ValueError, complex, unicode("1"*500))
208 self.assertRaises(ValueError, complex, unicode("x"))
209
210 class EvilExc(Exception):
211 pass
212
213 class evilcomplex:
214 def __complex__(self):
215 raise EvilExc
216
217 self.assertRaises(EvilExc, complex, evilcomplex())
218
219 class float2:
220 def __init__(self, value):
221 self.value = value
222 def __float__(self):
223 return self.value
224
225 self.assertAlmostEqual(complex(float2(42.)), 42)
226 self.assertAlmostEqual(complex(real=float2(17.), imag=float2(23.)), 17+23j)
227 self.assertRaises(TypeError, complex, float2(None))
228
229 def test_hash(self):
230 for x in xrange(-30, 30):
231 self.assertEqual(hash(x), hash(complex(x, 0)))
232 x /= 3.0 # now check against floating point
233 self.assertEqual(hash(x), hash(complex(x, 0.)))
234
235 def test_abs(self):
236 nums = [complex(x/3., y/7.) for x in xrange(-9,9) for y in xrange(-9,9)]
237 for num in nums:
238 self.assertAlmostEqual((num.real**2 + num.imag**2) ** 0.5, abs(num))
239
240 def test_repr(self):
241 self.assertEqual(repr(1+6j), '(1+6j)')
242
243 def test_neg(self):
244 self.assertEqual(-(1+6j), -1-6j)
245
246
247def test_main():
248 test_support.run_unittest(ComplexTest)
249
250if __name__ == "__main__":
251 test_main()