blob: 7fc52e3872cc735946a9da701ef691ea4e191f45 [file] [log] [blame]
Guido van Rossumb5a755e2007-07-18 18:15:48 +00001import base64
Fred Drakeba613c32005-02-10 18:33:30 +00002import datetime
Skip Montanaro3e7bba92001-10-19 16:06:52 +00003import sys
Guido van Rossumb5a755e2007-07-18 18:15:48 +00004import time
Skip Montanaro419abda2001-10-01 17:47:44 +00005import unittest
Ezio Melotti31916322013-08-10 18:37:05 +03006from unittest import mock
Georg Brandl38eceaa2008-05-26 11:14:17 +00007import xmlrpc.client as xmlrpclib
8import xmlrpc.server
Georg Brandl24420152008-05-26 16:32:26 +00009import http.client
Christian Heimes255f53b2007-12-08 15:33:56 +000010import socket
11import os
Senthil Kumaranb3af08f2009-04-01 20:20:43 +000012import re
R. David Murray0548ce02009-10-26 08:24:14 +000013import io
14import contextlib
Benjamin Petersonee8712c2008-05-20 21:35:26 +000015from test import support
Skip Montanaro419abda2001-10-01 17:47:44 +000016
Victor Stinner45df8202010-04-28 22:31:17 +000017try:
Brett Cannon603dcf22013-06-12 20:04:19 -040018 import gzip
Brett Cannon260fbe82013-07-04 18:16:15 -040019except ImportError:
Brett Cannon603dcf22013-06-12 20:04:19 -040020 gzip = None
21try:
Victor Stinner45df8202010-04-28 22:31:17 +000022 import threading
Brett Cannon260fbe82013-07-04 18:16:15 -040023except ImportError:
Victor Stinner45df8202010-04-28 22:31:17 +000024 threading = None
25
Skip Montanaro419abda2001-10-01 17:47:44 +000026alist = [{'astring': 'foo@bar.baz.spam',
27 'afloat': 7283.43,
Skip Montanaro3e7bba92001-10-19 16:06:52 +000028 'anint': 2**20,
Guido van Rossume2a383d2007-01-15 16:59:06 +000029 'ashortlong': 2,
Skip Montanaro419abda2001-10-01 17:47:44 +000030 'anotherlist': ['.zyx.41'],
Guido van Rossum54a40cb2007-08-27 22:27:41 +000031 'abase64': xmlrpclib.Binary(b"my dog has fleas"),
Florent Xicluna61665192011-11-15 20:53:25 +010032 'b64bytes': b"my dog has fleas",
33 'b64bytearray': bytearray(b"my dog has fleas"),
Guido van Rossume7ba4952007-06-06 23:52:48 +000034 'boolean': False,
Guido van Rossumef87d6e2007-05-02 19:09:54 +000035 'unicode': '\u4000\u6000\u8000',
36 'ukey\u4000': 'regular value',
Fred Drakeba613c32005-02-10 18:33:30 +000037 'datetime1': xmlrpclib.DateTime('20050210T11:41:23'),
38 'datetime2': xmlrpclib.DateTime(
Guido van Rossumcd16bf62007-06-13 18:07:49 +000039 (2005, 2, 10, 11, 41, 23, 0, 1, -1)),
Fred Drakeba613c32005-02-10 18:33:30 +000040 'datetime3': xmlrpclib.DateTime(
Guido van Rossumcd16bf62007-06-13 18:07:49 +000041 datetime.datetime(2005, 2, 10, 11, 41, 23)),
Skip Montanaro419abda2001-10-01 17:47:44 +000042 }]
43
44class XMLRPCTestCase(unittest.TestCase):
45
46 def test_dump_load(self):
Guido van Rossum54a40cb2007-08-27 22:27:41 +000047 dump = xmlrpclib.dumps((alist,))
48 load = xmlrpclib.loads(dump)
Ezio Melottib3aedd42010-11-20 19:04:17 +000049 self.assertEqual(alist, load[0][0])
Skip Montanaro419abda2001-10-01 17:47:44 +000050
Fred Drakeba613c32005-02-10 18:33:30 +000051 def test_dump_bare_datetime(self):
Skip Montanaro174dd222005-05-14 20:54:16 +000052 # This checks that an unwrapped datetime.date object can be handled
53 # by the marshalling code. This can't be done via test_dump_load()
Florent Xicluna61665192011-11-15 20:53:25 +010054 # since with use_builtin_types set to 1 the unmarshaller would create
Skip Montanaro174dd222005-05-14 20:54:16 +000055 # datetime objects for the 'datetime[123]' keys as well
Guido van Rossumcd16bf62007-06-13 18:07:49 +000056 dt = datetime.datetime(2005, 2, 10, 11, 41, 23)
Florent Xicluna61665192011-11-15 20:53:25 +010057 self.assertEqual(dt, xmlrpclib.DateTime('20050210T11:41:23'))
Fred Drakeba613c32005-02-10 18:33:30 +000058 s = xmlrpclib.dumps((dt,))
Fred Drakeba613c32005-02-10 18:33:30 +000059
Florent Xicluna61665192011-11-15 20:53:25 +010060 result, m = xmlrpclib.loads(s, use_builtin_types=True)
61 (newdt,) = result
62 self.assertEqual(newdt, dt)
63 self.assertIs(type(newdt), datetime.datetime)
64 self.assertIsNone(m)
65
66 result, m = xmlrpclib.loads(s, use_builtin_types=False)
67 (newdt,) = result
68 self.assertEqual(newdt, dt)
69 self.assertIs(type(newdt), xmlrpclib.DateTime)
70 self.assertIsNone(m)
71
72 result, m = xmlrpclib.loads(s, use_datetime=True)
73 (newdt,) = result
74 self.assertEqual(newdt, dt)
75 self.assertIs(type(newdt), datetime.datetime)
76 self.assertIsNone(m)
77
78 result, m = xmlrpclib.loads(s, use_datetime=False)
79 (newdt,) = result
80 self.assertEqual(newdt, dt)
81 self.assertIs(type(newdt), xmlrpclib.DateTime)
82 self.assertIsNone(m)
83
Skip Montanaro174dd222005-05-14 20:54:16 +000084
Christian Heimesdae2a892008-04-19 00:55:37 +000085 def test_datetime_before_1900(self):
Christian Heimes81ee3ef2008-05-04 22:42:01 +000086 # same as before but with a date before 1900
Christian Heimesdae2a892008-04-19 00:55:37 +000087 dt = datetime.datetime(1, 2, 10, 11, 41, 23)
Florent Xicluna61665192011-11-15 20:53:25 +010088 self.assertEqual(dt, xmlrpclib.DateTime('00010210T11:41:23'))
Christian Heimesdae2a892008-04-19 00:55:37 +000089 s = xmlrpclib.dumps((dt,))
Christian Heimesdae2a892008-04-19 00:55:37 +000090
Florent Xicluna61665192011-11-15 20:53:25 +010091 result, m = xmlrpclib.loads(s, use_builtin_types=True)
92 (newdt,) = result
93 self.assertEqual(newdt, dt)
94 self.assertIs(type(newdt), datetime.datetime)
95 self.assertIsNone(m)
96
97 result, m = xmlrpclib.loads(s, use_builtin_types=False)
98 (newdt,) = result
99 self.assertEqual(newdt, dt)
100 self.assertIs(type(newdt), xmlrpclib.DateTime)
101 self.assertIsNone(m)
Christian Heimesdae2a892008-04-19 00:55:37 +0000102
Andrew M. Kuchlingbdb39012005-12-04 19:11:17 +0000103 def test_bug_1164912 (self):
104 d = xmlrpclib.DateTime()
Tim Peters536cf992005-12-25 23:18:31 +0000105 ((new_d,), dummy) = xmlrpclib.loads(xmlrpclib.dumps((d,),
Andrew M. Kuchlingbdb39012005-12-04 19:11:17 +0000106 methodresponse=True))
Ezio Melottie9615932010-01-24 19:26:24 +0000107 self.assertIsInstance(new_d.value, str)
Andrew M. Kuchlingbdb39012005-12-04 19:11:17 +0000108
109 # Check that the output of dumps() is still an 8-bit string
110 s = xmlrpclib.dumps((new_d,), methodresponse=True)
Ezio Melottie9615932010-01-24 19:26:24 +0000111 self.assertIsInstance(s, str)
Andrew M. Kuchlingbdb39012005-12-04 19:11:17 +0000112
Thomas Wouters89f507f2006-12-13 04:49:30 +0000113 def test_newstyle_class(self):
114 class T(object):
115 pass
116 t = T()
117 t.x = 100
118 t.y = "Hello"
119 ((t2,), dummy) = xmlrpclib.loads(xmlrpclib.dumps((t,)))
Ezio Melottib3aedd42010-11-20 19:04:17 +0000120 self.assertEqual(t2, t.__dict__)
Thomas Wouters89f507f2006-12-13 04:49:30 +0000121
Skip Montanaro3e7bba92001-10-19 16:06:52 +0000122 def test_dump_big_long(self):
Guido van Rossume2a383d2007-01-15 16:59:06 +0000123 self.assertRaises(OverflowError, xmlrpclib.dumps, (2**99,))
Skip Montanaro3e7bba92001-10-19 16:06:52 +0000124
125 def test_dump_bad_dict(self):
126 self.assertRaises(TypeError, xmlrpclib.dumps, ({(1,2,3): 1},))
127
Guido van Rossumb5a755e2007-07-18 18:15:48 +0000128 def test_dump_recursive_seq(self):
129 l = [1,2,3]
130 t = [3,4,5,l]
131 l.append(t)
132 self.assertRaises(TypeError, xmlrpclib.dumps, (l,))
133
134 def test_dump_recursive_dict(self):
135 d = {'1':1, '2':1}
136 t = {'3':3, 'd':d}
137 d['t'] = t
138 self.assertRaises(TypeError, xmlrpclib.dumps, (d,))
139
Skip Montanaro3e7bba92001-10-19 16:06:52 +0000140 def test_dump_big_int(self):
Christian Heimesa37d4c62007-12-04 23:02:19 +0000141 if sys.maxsize > 2**31-1:
Skip Montanaro3e7bba92001-10-19 16:06:52 +0000142 self.assertRaises(OverflowError, xmlrpclib.dumps,
Guido van Rossume2a383d2007-01-15 16:59:06 +0000143 (int(2**34),))
Skip Montanaro3e7bba92001-10-19 16:06:52 +0000144
Guido van Rossumb5a755e2007-07-18 18:15:48 +0000145 xmlrpclib.dumps((xmlrpclib.MAXINT, xmlrpclib.MININT))
Jeremy Hylton1afc1692008-06-18 20:49:58 +0000146 self.assertRaises(OverflowError, xmlrpclib.dumps,
147 (xmlrpclib.MAXINT+1,))
148 self.assertRaises(OverflowError, xmlrpclib.dumps,
149 (xmlrpclib.MININT-1,))
Guido van Rossumb5a755e2007-07-18 18:15:48 +0000150
151 def dummy_write(s):
152 pass
153
154 m = xmlrpclib.Marshaller()
155 m.dump_int(xmlrpclib.MAXINT, dummy_write)
156 m.dump_int(xmlrpclib.MININT, dummy_write)
Jeremy Hylton1afc1692008-06-18 20:49:58 +0000157 self.assertRaises(OverflowError, m.dump_int,
158 xmlrpclib.MAXINT+1, dummy_write)
159 self.assertRaises(OverflowError, m.dump_int,
160 xmlrpclib.MININT-1, dummy_write)
Guido van Rossumb5a755e2007-07-18 18:15:48 +0000161
Florent Xicluna1917ea92012-07-07 17:03:25 +0200162 def test_dump_double(self):
163 xmlrpclib.dumps((float(2 ** 34),))
164 xmlrpclib.dumps((float(xmlrpclib.MAXINT),
165 float(xmlrpclib.MININT)))
166 xmlrpclib.dumps((float(xmlrpclib.MAXINT + 42),
167 float(xmlrpclib.MININT - 42)))
168
169 def dummy_write(s):
170 pass
171
172 m = xmlrpclib.Marshaller()
173 m.dump_double(xmlrpclib.MAXINT, dummy_write)
174 m.dump_double(xmlrpclib.MININT, dummy_write)
175 m.dump_double(xmlrpclib.MAXINT + 42, dummy_write)
176 m.dump_double(xmlrpclib.MININT - 42, dummy_write)
177
Andrew M. Kuchling0b852032003-04-25 00:27:24 +0000178 def test_dump_none(self):
179 value = alist + [None]
180 arg1 = (alist + [None],)
181 strg = xmlrpclib.dumps(arg1, allow_none=True)
Ezio Melottib3aedd42010-11-20 19:04:17 +0000182 self.assertEqual(value,
Andrew M. Kuchling0b852032003-04-25 00:27:24 +0000183 xmlrpclib.loads(strg)[0][0])
184 self.assertRaises(TypeError, xmlrpclib.dumps, (arg1,))
185
Florent Xicluna61665192011-11-15 20:53:25 +0100186 def test_dump_bytes(self):
187 sample = b"my dog has fleas"
188 self.assertEqual(sample, xmlrpclib.Binary(sample))
189 for type_ in bytes, bytearray, xmlrpclib.Binary:
190 value = type_(sample)
191 s = xmlrpclib.dumps((value,))
192
193 result, m = xmlrpclib.loads(s, use_builtin_types=True)
194 (newvalue,) = result
195 self.assertEqual(newvalue, sample)
196 self.assertIs(type(newvalue), bytes)
197 self.assertIsNone(m)
198
199 result, m = xmlrpclib.loads(s, use_builtin_types=False)
200 (newvalue,) = result
201 self.assertEqual(newvalue, sample)
202 self.assertIs(type(newvalue), xmlrpclib.Binary)
203 self.assertIsNone(m)
204
Georg Brandlc8dcfb62009-02-13 10:50:01 +0000205 def test_get_host_info(self):
206 # see bug #3613, this raised a TypeError
207 transp = xmlrpc.client.Transport()
Ezio Melottib3aedd42010-11-20 19:04:17 +0000208 self.assertEqual(transp.get_host_info("user@host.tld"),
Georg Brandlc8dcfb62009-02-13 10:50:01 +0000209 ('host.tld',
210 [('Authorization', 'Basic dXNlcg==')], {}))
211
Senthil Kumaran8ce1f1f2010-11-18 15:00:53 +0000212 def test_ssl_presence(self):
Senthil Kumaran6a0b5c42010-11-18 17:08:48 +0000213 try:
214 import ssl
Antoine Pitrou98b644f2010-11-19 20:07:52 +0000215 except ImportError:
216 has_ssl = False
Senthil Kumaran8ce1f1f2010-11-18 15:00:53 +0000217 else:
Senthil Kumaran6a0b5c42010-11-18 17:08:48 +0000218 has_ssl = True
Senthil Kumaran8ce1f1f2010-11-18 15:00:53 +0000219 try:
220 xmlrpc.client.ServerProxy('https://localhost:9999').bad_function()
Senthil Kumaran6a0b5c42010-11-18 17:08:48 +0000221 except NotImplementedError:
222 self.assertFalse(has_ssl, "xmlrpc client's error with SSL support")
Andrew Svetlov0832af62012-12-18 23:10:48 +0200223 except OSError:
Senthil Kumaran6a0b5c42010-11-18 17:08:48 +0000224 self.assertTrue(has_ssl)
Ezio Melottie9615932010-01-24 19:26:24 +0000225
Guido van Rossumb5a755e2007-07-18 18:15:48 +0000226class HelperTestCase(unittest.TestCase):
227 def test_escape(self):
228 self.assertEqual(xmlrpclib.escape("a&b"), "a&b")
229 self.assertEqual(xmlrpclib.escape("a<b"), "a&lt;b")
230 self.assertEqual(xmlrpclib.escape("a>b"), "a&gt;b")
231
232class FaultTestCase(unittest.TestCase):
233 def test_repr(self):
234 f = xmlrpclib.Fault(42, 'Test Fault')
235 self.assertEqual(repr(f), "<Fault 42: 'Test Fault'>")
236 self.assertEqual(repr(f), str(f))
237
238 def test_dump_fault(self):
239 f = xmlrpclib.Fault(42, 'Test Fault')
240 s = xmlrpclib.dumps((f,))
241 (newf,), m = xmlrpclib.loads(s)
Ezio Melottib3aedd42010-11-20 19:04:17 +0000242 self.assertEqual(newf, {'faultCode': 42, 'faultString': 'Test Fault'})
243 self.assertEqual(m, None)
Guido van Rossumb5a755e2007-07-18 18:15:48 +0000244
245 s = xmlrpclib.Marshaller().dumps(f)
246 self.assertRaises(xmlrpclib.Fault, xmlrpclib.loads, s)
247
Christian Heimeseda9e2b2007-12-08 22:32:47 +0000248 def test_dotted_attribute(self):
Florent Xiclunac4fec932011-10-30 20:19:32 +0100249 # this will raise AttributeError because code don't want us to use
Christian Heimeseda9e2b2007-12-08 22:32:47 +0000250 # private methods
251 self.assertRaises(AttributeError,
Georg Brandl38eceaa2008-05-26 11:14:17 +0000252 xmlrpc.server.resolve_dotted_attribute, str, '__add')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000253 self.assertTrue(xmlrpc.server.resolve_dotted_attribute(str, 'title'))
Guido van Rossumb5a755e2007-07-18 18:15:48 +0000254
255class DateTimeTestCase(unittest.TestCase):
256 def test_default(self):
Ezio Melotti31916322013-08-10 18:37:05 +0300257 with mock.patch('time.localtime') as localtime_mock:
258 time_struct = time.struct_time(
259 [2013, 7, 15, 0, 24, 49, 0, 196, 0])
260 localtime_mock.return_value = time_struct
261 localtime = time.localtime()
262 t = xmlrpclib.DateTime()
263 self.assertEqual(str(t),
264 time.strftime("%Y%m%dT%H:%M:%S", localtime))
Guido van Rossumb5a755e2007-07-18 18:15:48 +0000265
266 def test_time(self):
267 d = 1181399930.036952
268 t = xmlrpclib.DateTime(d)
Jeremy Hylton1afc1692008-06-18 20:49:58 +0000269 self.assertEqual(str(t),
270 time.strftime("%Y%m%dT%H:%M:%S", time.localtime(d)))
Guido van Rossumb5a755e2007-07-18 18:15:48 +0000271
272 def test_time_tuple(self):
273 d = (2007,6,9,10,38,50,5,160,0)
274 t = xmlrpclib.DateTime(d)
275 self.assertEqual(str(t), '20070609T10:38:50')
276
277 def test_time_struct(self):
278 d = time.localtime(1181399930.036952)
279 t = xmlrpclib.DateTime(d)
Jeremy Hylton1afc1692008-06-18 20:49:58 +0000280 self.assertEqual(str(t), time.strftime("%Y%m%dT%H:%M:%S", d))
Guido van Rossumb5a755e2007-07-18 18:15:48 +0000281
282 def test_datetime_datetime(self):
283 d = datetime.datetime(2007,1,2,3,4,5)
284 t = xmlrpclib.DateTime(d)
285 self.assertEqual(str(t), '20070102T03:04:05')
286
Guido van Rossumb5a755e2007-07-18 18:15:48 +0000287 def test_repr(self):
288 d = datetime.datetime(2007,1,2,3,4,5)
289 t = xmlrpclib.DateTime(d)
290 val ="<DateTime '20070102T03:04:05' at %x>" % id(t)
291 self.assertEqual(repr(t), val)
292
293 def test_decode(self):
294 d = ' 20070908T07:11:13 '
295 t1 = xmlrpclib.DateTime()
296 t1.decode(d)
297 tref = xmlrpclib.DateTime(datetime.datetime(2007,9,8,7,11,13))
298 self.assertEqual(t1, tref)
299
300 t2 = xmlrpclib._datetime(d)
Ezio Melotti31916322013-08-10 18:37:05 +0300301 self.assertEqual(t2, tref)
Guido van Rossumb5a755e2007-07-18 18:15:48 +0000302
Florent Xicluna3fa29f72011-10-30 20:18:50 +0100303 def test_comparison(self):
304 now = datetime.datetime.now()
305 dtime = xmlrpclib.DateTime(now.timetuple())
306
307 # datetime vs. DateTime
308 self.assertTrue(dtime == now)
309 self.assertTrue(now == dtime)
310 then = now + datetime.timedelta(seconds=4)
311 self.assertTrue(then >= dtime)
312 self.assertTrue(dtime < then)
313
314 # str vs. DateTime
315 dstr = now.strftime("%Y%m%dT%H:%M:%S")
316 self.assertTrue(dtime == dstr)
317 self.assertTrue(dstr == dtime)
318 dtime_then = xmlrpclib.DateTime(then.timetuple())
319 self.assertTrue(dtime_then >= dstr)
320 self.assertTrue(dstr < dtime_then)
321
322 # some other types
323 dbytes = dstr.encode('ascii')
324 dtuple = now.timetuple()
325 with self.assertRaises(TypeError):
326 dtime == 1970
327 with self.assertRaises(TypeError):
328 dtime != dbytes
329 with self.assertRaises(TypeError):
330 dtime == bytearray(dbytes)
331 with self.assertRaises(TypeError):
332 dtime != dtuple
333 with self.assertRaises(TypeError):
334 dtime < float(1970)
335 with self.assertRaises(TypeError):
336 dtime > dbytes
337 with self.assertRaises(TypeError):
338 dtime <= bytearray(dbytes)
339 with self.assertRaises(TypeError):
340 dtime >= dtuple
341
Guido van Rossumb5a755e2007-07-18 18:15:48 +0000342class BinaryTestCase(unittest.TestCase):
Guido van Rossum54a40cb2007-08-27 22:27:41 +0000343
344 # XXX What should str(Binary(b"\xff")) return? I'm chosing "\xff"
345 # for now (i.e. interpreting the binary data as Latin-1-encoded
346 # text). But this feels very unsatisfactory. Perhaps we should
347 # only define repr(), and return r"Binary(b'\xff')" instead?
348
Guido van Rossumb5a755e2007-07-18 18:15:48 +0000349 def test_default(self):
350 t = xmlrpclib.Binary()
351 self.assertEqual(str(t), '')
352
353 def test_string(self):
Guido van Rossum54a40cb2007-08-27 22:27:41 +0000354 d = b'\x01\x02\x03abc123\xff\xfe'
Guido van Rossumb5a755e2007-07-18 18:15:48 +0000355 t = xmlrpclib.Binary(d)
Guido van Rossum54a40cb2007-08-27 22:27:41 +0000356 self.assertEqual(str(t), str(d, "latin-1"))
Guido van Rossumb5a755e2007-07-18 18:15:48 +0000357
358 def test_decode(self):
Guido van Rossum54a40cb2007-08-27 22:27:41 +0000359 d = b'\x01\x02\x03abc123\xff\xfe'
Georg Brandlb54d8012009-06-04 09:11:51 +0000360 de = base64.encodebytes(d)
Guido van Rossumb5a755e2007-07-18 18:15:48 +0000361 t1 = xmlrpclib.Binary()
362 t1.decode(de)
Guido van Rossum54a40cb2007-08-27 22:27:41 +0000363 self.assertEqual(str(t1), str(d, "latin-1"))
Guido van Rossumb5a755e2007-07-18 18:15:48 +0000364
365 t2 = xmlrpclib._binary(de)
Guido van Rossum54a40cb2007-08-27 22:27:41 +0000366 self.assertEqual(str(t2), str(d, "latin-1"))
Guido van Rossumb5a755e2007-07-18 18:15:48 +0000367
368
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000369ADDR = PORT = URL = None
Skip Montanaro419abda2001-10-01 17:47:44 +0000370
Christian Heimesaf98da12008-01-27 15:18:18 +0000371# The evt is set twice. First when the server is ready to serve.
372# Second when the server has been shutdown. The user must clear
373# the event after it has been set the first time to catch the second set.
Kristján Valur Jónsson985fc6a2009-07-01 10:01:31 +0000374def http_server(evt, numrequests, requestHandler=None):
Guido van Rossumaf554a02007-08-16 23:48:43 +0000375 class TestInstanceClass:
376 def div(self, x, y):
Guido van Rossumaf554a02007-08-16 23:48:43 +0000377 return x // y
378
Christian Heimes255f53b2007-12-08 15:33:56 +0000379 def _methodHelp(self, name):
380 if name == 'div':
381 return 'This is the div function'
382
383 def my_function():
384 '''This is my function'''
385 return True
386
Georg Brandl38eceaa2008-05-26 11:14:17 +0000387 class MyXMLRPCServer(xmlrpc.server.SimpleXMLRPCServer):
Christian Heimesbbe741d2008-03-28 10:53:29 +0000388 def get_request(self):
389 # Ensure the socket is always non-blocking. On Linux, socket
390 # attributes are not inherited like they are on *BSD and Windows.
391 s, port = self.socket.accept()
392 s.setblocking(True)
393 return s, port
394
Kristján Valur Jónsson985fc6a2009-07-01 10:01:31 +0000395 if not requestHandler:
396 requestHandler = xmlrpc.server.SimpleXMLRPCRequestHandler
397 serv = MyXMLRPCServer(("localhost", 0), requestHandler,
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000398 logRequests=False, bind_and_activate=False)
Guido van Rossumaf554a02007-08-16 23:48:43 +0000399 try:
Guido van Rossumaf554a02007-08-16 23:48:43 +0000400 serv.server_bind()
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000401 global ADDR, PORT, URL
402 ADDR, PORT = serv.socket.getsockname()
403 #connect to IP address directly. This avoids socket.create_connection()
Ezio Melottie130a522011-10-19 10:58:56 +0300404 #trying to connect to "localhost" using all address families, which
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000405 #causes slowdown e.g. on vista which supports AF_INET6. The server listens
406 #on AF_INET only.
407 URL = "http://%s:%d"%(ADDR, PORT)
Guido van Rossumaf554a02007-08-16 23:48:43 +0000408 serv.server_activate()
409 serv.register_introspection_functions()
410 serv.register_multicall_functions()
411 serv.register_function(pow)
412 serv.register_function(lambda x,y: x+y, 'add')
Christian Heimes255f53b2007-12-08 15:33:56 +0000413 serv.register_function(my_function)
Guido van Rossumaf554a02007-08-16 23:48:43 +0000414 serv.register_instance(TestInstanceClass())
Christian Heimesaf98da12008-01-27 15:18:18 +0000415 evt.set()
Guido van Rossumaf554a02007-08-16 23:48:43 +0000416
417 # handle up to 'numrequests' requests
418 while numrequests > 0:
419 serv.handle_request()
420 numrequests -= 1
421
422 except socket.timeout:
423 pass
424 finally:
425 serv.socket.close()
426 PORT = None
427 evt.set()
428
Kristján Valur Jónsson1f2a1ae2009-12-16 10:50:44 +0000429def http_multi_server(evt, numrequests, requestHandler=None):
430 class TestInstanceClass:
431 def div(self, x, y):
432 return x // y
433
434 def _methodHelp(self, name):
435 if name == 'div':
436 return 'This is the div function'
437
438 def my_function():
439 '''This is my function'''
440 return True
441
442 class MyXMLRPCServer(xmlrpc.server.MultiPathXMLRPCServer):
443 def get_request(self):
444 # Ensure the socket is always non-blocking. On Linux, socket
445 # attributes are not inherited like they are on *BSD and Windows.
446 s, port = self.socket.accept()
447 s.setblocking(True)
448 return s, port
449
450 if not requestHandler:
451 requestHandler = xmlrpc.server.SimpleXMLRPCRequestHandler
452 class MyRequestHandler(requestHandler):
453 rpc_paths = []
454
Florent Xicluna3fa29f72011-10-30 20:18:50 +0100455 class BrokenDispatcher:
456 def _marshaled_dispatch(self, data, dispatch_method=None, path=None):
457 raise RuntimeError("broken dispatcher")
458
Kristján Valur Jónsson1f2a1ae2009-12-16 10:50:44 +0000459 serv = MyXMLRPCServer(("localhost", 0), MyRequestHandler,
460 logRequests=False, bind_and_activate=False)
461 serv.socket.settimeout(3)
462 serv.server_bind()
463 try:
464 global ADDR, PORT, URL
465 ADDR, PORT = serv.socket.getsockname()
466 #connect to IP address directly. This avoids socket.create_connection()
Ezio Melottie130a522011-10-19 10:58:56 +0300467 #trying to connect to "localhost" using all address families, which
Kristján Valur Jónsson1f2a1ae2009-12-16 10:50:44 +0000468 #causes slowdown e.g. on vista which supports AF_INET6. The server listens
469 #on AF_INET only.
470 URL = "http://%s:%d"%(ADDR, PORT)
471 serv.server_activate()
472 paths = ["/foo", "/foo/bar"]
473 for path in paths:
474 d = serv.add_dispatcher(path, xmlrpc.server.SimpleXMLRPCDispatcher())
475 d.register_introspection_functions()
476 d.register_multicall_functions()
477 serv.get_dispatcher(paths[0]).register_function(pow)
478 serv.get_dispatcher(paths[1]).register_function(lambda x,y: x+y, 'add')
Florent Xicluna3fa29f72011-10-30 20:18:50 +0100479 serv.add_dispatcher("/is/broken", BrokenDispatcher())
Kristján Valur Jónsson1f2a1ae2009-12-16 10:50:44 +0000480 evt.set()
481
482 # handle up to 'numrequests' requests
483 while numrequests > 0:
484 serv.handle_request()
485 numrequests -= 1
486
487 except socket.timeout:
488 pass
489 finally:
490 serv.socket.close()
491 PORT = None
492 evt.set()
493
Christian Heimesaf98da12008-01-27 15:18:18 +0000494# This function prevents errors like:
495# <ProtocolError for localhost:57527/RPC2: 500 Internal Server Error>
496def is_unavailable_exception(e):
497 '''Returns True if the given ProtocolError is the product of a server-side
498 exception caused by the 'temporarily unavailable' response sometimes
499 given by operations on non-blocking sockets.'''
500
501 # sometimes we get a -1 error code and/or empty headers
Christian Heimes26855632008-01-27 23:50:43 +0000502 try:
503 if e.errcode == -1 or e.headers is None:
504 return True
505 exc_mess = e.headers.get('X-exception')
506 except AttributeError:
Andrew Svetlov0832af62012-12-18 23:10:48 +0200507 # Ignore OSErrors here.
Christian Heimes26855632008-01-27 23:50:43 +0000508 exc_mess = str(e)
Guido van Rossumaf554a02007-08-16 23:48:43 +0000509
Christian Heimes26855632008-01-27 23:50:43 +0000510 if exc_mess and 'temporarily unavailable' in exc_mess.lower():
511 return True
Thomas Woutersed03b412007-08-28 21:37:11 +0000512
R. David Murray378c0cf2010-02-24 01:46:21 +0000513def make_request_and_skipIf(condition, reason):
514 # If we skip the test, we have to make a request because the
515 # the server created in setUp blocks expecting one to come in.
516 if not condition:
517 return lambda func: func
518 def decorator(func):
519 def make_request_and_skip(self):
520 try:
521 xmlrpclib.ServerProxy(URL).my_function()
Andrew Svetlov0832af62012-12-18 23:10:48 +0200522 except (xmlrpclib.ProtocolError, OSError) as e:
R. David Murray378c0cf2010-02-24 01:46:21 +0000523 if not is_unavailable_exception(e):
524 raise
525 raise unittest.SkipTest(reason)
526 return make_request_and_skip
527 return decorator
528
Victor Stinner45df8202010-04-28 22:31:17 +0000529@unittest.skipUnless(threading, 'Threading required for this test.')
Kristján Valur Jónsson985fc6a2009-07-01 10:01:31 +0000530class BaseServerTestCase(unittest.TestCase):
531 requestHandler = None
Kristján Valur Jónssonf6087ca2009-07-12 22:45:18 +0000532 request_count = 1
Kristján Valur Jónsson1f2a1ae2009-12-16 10:50:44 +0000533 threadFunc = staticmethod(http_server)
Victor Stinner45df8202010-04-28 22:31:17 +0000534
Guido van Rossumaf554a02007-08-16 23:48:43 +0000535 def setUp(self):
Guido van Rossum61e21b52007-08-20 19:06:03 +0000536 # enable traceback reporting
Georg Brandl38eceaa2008-05-26 11:14:17 +0000537 xmlrpc.server.SimpleXMLRPCServer._send_traceback_header = True
Guido van Rossum61e21b52007-08-20 19:06:03 +0000538
Guido van Rossumaf554a02007-08-16 23:48:43 +0000539 self.evt = threading.Event()
Guido van Rossum61e21b52007-08-20 19:06:03 +0000540 # start server thread to handle requests
Kristján Valur Jónssonf6087ca2009-07-12 22:45:18 +0000541 serv_args = (self.evt, self.request_count, self.requestHandler)
Kristján Valur Jónsson1f2a1ae2009-12-16 10:50:44 +0000542 threading.Thread(target=self.threadFunc, args=serv_args).start()
Guido van Rossumaf554a02007-08-16 23:48:43 +0000543
Christian Heimesaf98da12008-01-27 15:18:18 +0000544 # wait for the server to be ready
545 self.evt.wait()
546 self.evt.clear()
Guido van Rossumaf554a02007-08-16 23:48:43 +0000547
548 def tearDown(self):
549 # wait on the server thread to terminate
Charles-François Natalicd96b4f2012-02-18 14:53:41 +0100550 self.evt.wait()
Guido van Rossumaf554a02007-08-16 23:48:43 +0000551
Guido van Rossum61e21b52007-08-20 19:06:03 +0000552 # disable traceback reporting
Georg Brandl38eceaa2008-05-26 11:14:17 +0000553 xmlrpc.server.SimpleXMLRPCServer._send_traceback_header = False
Guido van Rossum61e21b52007-08-20 19:06:03 +0000554
Kristján Valur Jónsson985fc6a2009-07-01 10:01:31 +0000555class SimpleServerTestCase(BaseServerTestCase):
Guido van Rossumaf554a02007-08-16 23:48:43 +0000556 def test_simple1(self):
Guido van Rossum04110fb2007-08-24 16:32:05 +0000557 try:
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000558 p = xmlrpclib.ServerProxy(URL)
Guido van Rossum04110fb2007-08-24 16:32:05 +0000559 self.assertEqual(p.pow(6,8), 6**8)
Andrew Svetlov0832af62012-12-18 23:10:48 +0200560 except (xmlrpclib.ProtocolError, OSError) as e:
Christian Heimes412dc9c2008-01-27 18:55:54 +0000561 # ignore failures due to non-blocking socket 'unavailable' errors
562 if not is_unavailable_exception(e):
563 # protocol error; provide additional information in test output
Christian Heimes26855632008-01-27 23:50:43 +0000564 self.fail("%s\n%s" % (e, getattr(e, "headers", "")))
Guido van Rossumaf554a02007-08-16 23:48:43 +0000565
Senthil Kumaranb3af08f2009-04-01 20:20:43 +0000566 def test_nonascii(self):
567 start_string = 'P\N{LATIN SMALL LETTER Y WITH CIRCUMFLEX}t'
568 end_string = 'h\N{LATIN SMALL LETTER O WITH HORN}n'
569 try:
570 p = xmlrpclib.ServerProxy(URL)
571 self.assertEqual(p.add(start_string, end_string),
572 start_string + end_string)
Andrew Svetlov0832af62012-12-18 23:10:48 +0200573 except (xmlrpclib.ProtocolError, OSError) as e:
Senthil Kumaranb3af08f2009-04-01 20:20:43 +0000574 # ignore failures due to non-blocking socket 'unavailable' errors
575 if not is_unavailable_exception(e):
576 # protocol error; provide additional information in test output
577 self.fail("%s\n%s" % (e, getattr(e, "headers", "")))
578
Georg Brandlfceab5a2008-01-19 20:08:23 +0000579 # [ch] The test 404 is causing lots of false alarms.
580 def XXXtest_404(self):
Georg Brandl24420152008-05-26 16:32:26 +0000581 # send POST with http.client, it should return 404 header and
Christian Heimes255f53b2007-12-08 15:33:56 +0000582 # 'Not Found' message.
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000583 conn = httplib.client.HTTPConnection(ADDR, PORT)
Christian Heimes99d73f22007-12-08 16:13:06 +0000584 conn.request('POST', '/this-is-not-valid')
Christian Heimes255f53b2007-12-08 15:33:56 +0000585 response = conn.getresponse()
586 conn.close()
587
588 self.assertEqual(response.status, 404)
589 self.assertEqual(response.reason, 'Not Found')
590
Guido van Rossumaf554a02007-08-16 23:48:43 +0000591 def test_introspection1(self):
Jeremy Hylton1afc1692008-06-18 20:49:58 +0000592 expected_methods = set(['pow', 'div', 'my_function', 'add',
593 'system.listMethods', 'system.methodHelp',
594 'system.methodSignature', 'system.multicall'])
Guido van Rossum04110fb2007-08-24 16:32:05 +0000595 try:
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000596 p = xmlrpclib.ServerProxy(URL)
Guido van Rossum04110fb2007-08-24 16:32:05 +0000597 meth = p.system.listMethods()
Guido van Rossum04110fb2007-08-24 16:32:05 +0000598 self.assertEqual(set(meth), expected_methods)
Andrew Svetlov0832af62012-12-18 23:10:48 +0200599 except (xmlrpclib.ProtocolError, OSError) as e:
Christian Heimes412dc9c2008-01-27 18:55:54 +0000600 # ignore failures due to non-blocking socket 'unavailable' errors
601 if not is_unavailable_exception(e):
602 # protocol error; provide additional information in test output
Christian Heimes26855632008-01-27 23:50:43 +0000603 self.fail("%s\n%s" % (e, getattr(e, "headers", "")))
Christian Heimes412dc9c2008-01-27 18:55:54 +0000604
Guido van Rossumaf554a02007-08-16 23:48:43 +0000605
606 def test_introspection2(self):
Guido van Rossum04110fb2007-08-24 16:32:05 +0000607 try:
Christian Heimes255f53b2007-12-08 15:33:56 +0000608 # test _methodHelp()
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000609 p = xmlrpclib.ServerProxy(URL)
Guido van Rossum04110fb2007-08-24 16:32:05 +0000610 divhelp = p.system.methodHelp('div')
611 self.assertEqual(divhelp, 'This is the div function')
Andrew Svetlov0832af62012-12-18 23:10:48 +0200612 except (xmlrpclib.ProtocolError, OSError) as e:
Christian Heimes412dc9c2008-01-27 18:55:54 +0000613 # ignore failures due to non-blocking socket 'unavailable' errors
614 if not is_unavailable_exception(e):
615 # protocol error; provide additional information in test output
Christian Heimes26855632008-01-27 23:50:43 +0000616 self.fail("%s\n%s" % (e, getattr(e, "headers", "")))
Guido van Rossumaf554a02007-08-16 23:48:43 +0000617
R. David Murray378c0cf2010-02-24 01:46:21 +0000618 @make_request_and_skipIf(sys.flags.optimize >= 2,
619 "Docstrings are omitted with -O2 and above")
Guido van Rossumaf554a02007-08-16 23:48:43 +0000620 def test_introspection3(self):
Christian Heimes255f53b2007-12-08 15:33:56 +0000621 try:
622 # test native doc
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000623 p = xmlrpclib.ServerProxy(URL)
Christian Heimes255f53b2007-12-08 15:33:56 +0000624 myfunction = p.system.methodHelp('my_function')
625 self.assertEqual(myfunction, 'This is my function')
Andrew Svetlov0832af62012-12-18 23:10:48 +0200626 except (xmlrpclib.ProtocolError, OSError) as e:
Christian Heimes255f53b2007-12-08 15:33:56 +0000627 # ignore failures due to non-blocking socket 'unavailable' errors
628 if not is_unavailable_exception(e):
629 # protocol error; provide additional information in test output
Christian Heimes26855632008-01-27 23:50:43 +0000630 self.fail("%s\n%s" % (e, getattr(e, "headers", "")))
Christian Heimes255f53b2007-12-08 15:33:56 +0000631
632 def test_introspection4(self):
Guido van Rossumaf554a02007-08-16 23:48:43 +0000633 # the SimpleXMLRPCServer doesn't support signatures, but
Thomas Woutersed03b412007-08-28 21:37:11 +0000634 # at least check that we can try making the call
Guido van Rossum04110fb2007-08-24 16:32:05 +0000635 try:
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000636 p = xmlrpclib.ServerProxy(URL)
Guido van Rossum04110fb2007-08-24 16:32:05 +0000637 divsig = p.system.methodSignature('div')
638 self.assertEqual(divsig, 'signatures not supported')
Andrew Svetlov0832af62012-12-18 23:10:48 +0200639 except (xmlrpclib.ProtocolError, OSError) as e:
Christian Heimes412dc9c2008-01-27 18:55:54 +0000640 # ignore failures due to non-blocking socket 'unavailable' errors
641 if not is_unavailable_exception(e):
642 # protocol error; provide additional information in test output
Christian Heimes26855632008-01-27 23:50:43 +0000643 self.fail("%s\n%s" % (e, getattr(e, "headers", "")))
Guido van Rossumaf554a02007-08-16 23:48:43 +0000644
645 def test_multicall(self):
Guido van Rossum04110fb2007-08-24 16:32:05 +0000646 try:
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000647 p = xmlrpclib.ServerProxy(URL)
Guido van Rossum04110fb2007-08-24 16:32:05 +0000648 multicall = xmlrpclib.MultiCall(p)
649 multicall.add(2,3)
650 multicall.pow(6,8)
651 multicall.div(127,42)
652 add_result, pow_result, div_result = multicall()
653 self.assertEqual(add_result, 2+3)
654 self.assertEqual(pow_result, 6**8)
655 self.assertEqual(div_result, 127//42)
Andrew Svetlov0832af62012-12-18 23:10:48 +0200656 except (xmlrpclib.ProtocolError, OSError) as e:
Christian Heimes412dc9c2008-01-27 18:55:54 +0000657 # ignore failures due to non-blocking socket 'unavailable' errors
658 if not is_unavailable_exception(e):
659 # protocol error; provide additional information in test output
Christian Heimes26855632008-01-27 23:50:43 +0000660 self.fail("%s\n%s" % (e, getattr(e, "headers", "")))
Guido van Rossumaf554a02007-08-16 23:48:43 +0000661
Christian Heimes255f53b2007-12-08 15:33:56 +0000662 def test_non_existing_multicall(self):
663 try:
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000664 p = xmlrpclib.ServerProxy(URL)
Christian Heimes255f53b2007-12-08 15:33:56 +0000665 multicall = xmlrpclib.MultiCall(p)
666 multicall.this_is_not_exists()
667 result = multicall()
668
669 # result.results contains;
Martin v. Löwis250ad612008-04-07 05:43:42 +0000670 # [{'faultCode': 1, 'faultString': '<class \'exceptions.Exception\'>:'
Christian Heimes255f53b2007-12-08 15:33:56 +0000671 # 'method "this_is_not_exists" is not supported'>}]
672
673 self.assertEqual(result.results[0]['faultCode'], 1)
674 self.assertEqual(result.results[0]['faultString'],
Martin v. Löwis250ad612008-04-07 05:43:42 +0000675 '<class \'Exception\'>:method "this_is_not_exists" '
Christian Heimes255f53b2007-12-08 15:33:56 +0000676 'is not supported')
Andrew Svetlov0832af62012-12-18 23:10:48 +0200677 except (xmlrpclib.ProtocolError, OSError) as e:
Christian Heimes255f53b2007-12-08 15:33:56 +0000678 # ignore failures due to non-blocking socket 'unavailable' errors
679 if not is_unavailable_exception(e):
680 # protocol error; provide additional information in test output
Christian Heimes26855632008-01-27 23:50:43 +0000681 self.fail("%s\n%s" % (e, getattr(e, "headers", "")))
Christian Heimes255f53b2007-12-08 15:33:56 +0000682
Christian Heimesfe337bf2008-03-23 21:54:12 +0000683 def test_dotted_attribute(self):
684 # Raises an AttributeError because private methods are not allowed.
685 self.assertRaises(AttributeError,
Georg Brandl38eceaa2008-05-26 11:14:17 +0000686 xmlrpc.server.resolve_dotted_attribute, str, '__add')
Christian Heimesfe337bf2008-03-23 21:54:12 +0000687
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000688 self.assertTrue(xmlrpc.server.resolve_dotted_attribute(str, 'title'))
Christian Heimesfe337bf2008-03-23 21:54:12 +0000689 # Get the test to run faster by sending a request with test_simple1.
690 # This avoids waiting for the socket timeout.
691 self.test_simple1()
692
Victor Stinnerda6eb532011-09-23 01:29:44 +0200693 def test_unicode_host(self):
694 server = xmlrpclib.ServerProxy("http://%s:%d/RPC2" % (ADDR, PORT))
695 self.assertEqual(server.add("a", "\xe9"), "a\xe9")
696
Charles-François Natalicd96b4f2012-02-18 14:53:41 +0100697 def test_partial_post(self):
698 # Check that a partial POST doesn't make the server loop: issue #14001.
699 conn = http.client.HTTPConnection(ADDR, PORT)
700 conn.request('POST', '/RPC2 HTTP/1.0\r\nContent-Length: 100\r\n\r\nbye')
701 conn.close()
702
703
Kristján Valur Jónsson1f2a1ae2009-12-16 10:50:44 +0000704class MultiPathServerTestCase(BaseServerTestCase):
705 threadFunc = staticmethod(http_multi_server)
706 request_count = 2
707 def test_path1(self):
708 p = xmlrpclib.ServerProxy(URL+"/foo")
709 self.assertEqual(p.pow(6,8), 6**8)
710 self.assertRaises(xmlrpclib.Fault, p.add, 6, 8)
Florent Xicluna3fa29f72011-10-30 20:18:50 +0100711
Kristján Valur Jónsson1f2a1ae2009-12-16 10:50:44 +0000712 def test_path2(self):
713 p = xmlrpclib.ServerProxy(URL+"/foo/bar")
714 self.assertEqual(p.add(6,8), 6+8)
715 self.assertRaises(xmlrpclib.Fault, p.pow, 6, 8)
716
Florent Xicluna3fa29f72011-10-30 20:18:50 +0100717 def test_path3(self):
718 p = xmlrpclib.ServerProxy(URL+"/is/broken")
719 self.assertRaises(xmlrpclib.Fault, p.add, 6, 8)
720
Kristján Valur Jónsson985fc6a2009-07-01 10:01:31 +0000721#A test case that verifies that a server using the HTTP/1.1 keep-alive mechanism
722#does indeed serve subsequent requests on the same connection
Kristján Valur Jónssonf6087ca2009-07-12 22:45:18 +0000723class BaseKeepaliveServerTestCase(BaseServerTestCase):
Kristján Valur Jónsson985fc6a2009-07-01 10:01:31 +0000724 #a request handler that supports keep-alive and logs requests into a
725 #class variable
726 class RequestHandler(xmlrpc.server.SimpleXMLRPCRequestHandler):
727 parentClass = xmlrpc.server.SimpleXMLRPCRequestHandler
728 protocol_version = 'HTTP/1.1'
729 myRequests = []
730 def handle(self):
731 self.myRequests.append([])
Kristján Valur Jónssonf6087ca2009-07-12 22:45:18 +0000732 self.reqidx = len(self.myRequests)-1
Kristján Valur Jónsson985fc6a2009-07-01 10:01:31 +0000733 return self.parentClass.handle(self)
734 def handle_one_request(self):
735 result = self.parentClass.handle_one_request(self)
Kristján Valur Jónssonf6087ca2009-07-12 22:45:18 +0000736 self.myRequests[self.reqidx].append(self.raw_requestline)
Kristján Valur Jónsson985fc6a2009-07-01 10:01:31 +0000737 return result
738
739 requestHandler = RequestHandler
740 def setUp(self):
741 #clear request log
742 self.RequestHandler.myRequests = []
743 return BaseServerTestCase.setUp(self)
744
Kristján Valur Jónssonf6087ca2009-07-12 22:45:18 +0000745#A test case that verifies that a server using the HTTP/1.1 keep-alive mechanism
746#does indeed serve subsequent requests on the same connection
747class KeepaliveServerTestCase1(BaseKeepaliveServerTestCase):
Kristján Valur Jónsson985fc6a2009-07-01 10:01:31 +0000748 def test_two(self):
749 p = xmlrpclib.ServerProxy(URL)
Kristján Valur Jónsson1bbb19a2009-07-11 21:57:16 +0000750 #do three requests.
Kristján Valur Jónsson985fc6a2009-07-01 10:01:31 +0000751 self.assertEqual(p.pow(6,8), 6**8)
752 self.assertEqual(p.pow(6,8), 6**8)
Kristján Valur Jónsson1bbb19a2009-07-11 21:57:16 +0000753 self.assertEqual(p.pow(6,8), 6**8)
Victor Stinnera935e8f2011-01-03 14:30:44 +0000754 p("close")()
Kristján Valur Jónsson1bbb19a2009-07-11 21:57:16 +0000755
756 #they should have all been handled by a single request handler
Kristján Valur Jónsson985fc6a2009-07-01 10:01:31 +0000757 self.assertEqual(len(self.RequestHandler.myRequests), 1)
Kristján Valur Jónsson1bbb19a2009-07-11 21:57:16 +0000758
759 #check that we did at least two (the third may be pending append
760 #due to thread scheduling)
761 self.assertGreaterEqual(len(self.RequestHandler.myRequests[-1]), 2)
Kristján Valur Jónsson985fc6a2009-07-01 10:01:31 +0000762
Victor Stinnera935e8f2011-01-03 14:30:44 +0000763
Kristján Valur Jónssonf6087ca2009-07-12 22:45:18 +0000764#test special attribute access on the serverproxy, through the __call__
765#function.
766class KeepaliveServerTestCase2(BaseKeepaliveServerTestCase):
767 #ask for two keepalive requests to be handled.
768 request_count=2
769
770 def test_close(self):
771 p = xmlrpclib.ServerProxy(URL)
772 #do some requests with close.
773 self.assertEqual(p.pow(6,8), 6**8)
774 self.assertEqual(p.pow(6,8), 6**8)
775 self.assertEqual(p.pow(6,8), 6**8)
776 p("close")() #this should trigger a new keep-alive request
777 self.assertEqual(p.pow(6,8), 6**8)
778 self.assertEqual(p.pow(6,8), 6**8)
779 self.assertEqual(p.pow(6,8), 6**8)
Victor Stinnera935e8f2011-01-03 14:30:44 +0000780 p("close")()
Kristján Valur Jónssonf6087ca2009-07-12 22:45:18 +0000781
782 #they should have all been two request handlers, each having logged at least
783 #two complete requests
784 self.assertEqual(len(self.RequestHandler.myRequests), 2)
785 self.assertGreaterEqual(len(self.RequestHandler.myRequests[-1]), 2)
786 self.assertGreaterEqual(len(self.RequestHandler.myRequests[-2]), 2)
787
Victor Stinnera935e8f2011-01-03 14:30:44 +0000788
Kristján Valur Jónssonf6087ca2009-07-12 22:45:18 +0000789 def test_transport(self):
790 p = xmlrpclib.ServerProxy(URL)
791 #do some requests with close.
792 self.assertEqual(p.pow(6,8), 6**8)
793 p("transport").close() #same as above, really.
794 self.assertEqual(p.pow(6,8), 6**8)
Victor Stinnera935e8f2011-01-03 14:30:44 +0000795 p("close")()
Kristján Valur Jónssonf6087ca2009-07-12 22:45:18 +0000796 self.assertEqual(len(self.RequestHandler.myRequests), 2)
797
Kristján Valur Jónsson985fc6a2009-07-01 10:01:31 +0000798#A test case that verifies that gzip encoding works in both directions
799#(for a request and the response)
Brett Cannon603dcf22013-06-12 20:04:19 -0400800@unittest.skipIf(gzip is None, 'requires gzip')
Kristján Valur Jónsson985fc6a2009-07-01 10:01:31 +0000801class GzipServerTestCase(BaseServerTestCase):
802 #a request handler that supports keep-alive and logs requests into a
803 #class variable
804 class RequestHandler(xmlrpc.server.SimpleXMLRPCRequestHandler):
805 parentClass = xmlrpc.server.SimpleXMLRPCRequestHandler
806 protocol_version = 'HTTP/1.1'
807
808 def do_POST(self):
809 #store content of last request in class
810 self.__class__.content_length = int(self.headers["content-length"])
811 return self.parentClass.do_POST(self)
812 requestHandler = RequestHandler
813
814 class Transport(xmlrpclib.Transport):
815 #custom transport, stores the response length for our perusal
816 fake_gzip = False
817 def parse_response(self, response):
818 self.response_length=int(response.getheader("content-length", 0))
819 return xmlrpclib.Transport.parse_response(self, response)
820
821 def send_content(self, connection, body):
822 if self.fake_gzip:
823 #add a lone gzip header to induce decode error remotely
824 connection.putheader("Content-Encoding", "gzip")
825 return xmlrpclib.Transport.send_content(self, connection, body)
826
Victor Stinner45df8202010-04-28 22:31:17 +0000827 def setUp(self):
828 BaseServerTestCase.setUp(self)
829
Kristján Valur Jónsson985fc6a2009-07-01 10:01:31 +0000830 def test_gzip_request(self):
831 t = self.Transport()
832 t.encode_threshold = None
833 p = xmlrpclib.ServerProxy(URL, transport=t)
834 self.assertEqual(p.pow(6,8), 6**8)
835 a = self.RequestHandler.content_length
836 t.encode_threshold = 0 #turn on request encoding
837 self.assertEqual(p.pow(6,8), 6**8)
838 b = self.RequestHandler.content_length
839 self.assertTrue(a>b)
Victor Stinnera935e8f2011-01-03 14:30:44 +0000840 p("close")()
Kristján Valur Jónsson985fc6a2009-07-01 10:01:31 +0000841
842 def test_bad_gzip_request(self):
843 t = self.Transport()
844 t.encode_threshold = None
845 t.fake_gzip = True
846 p = xmlrpclib.ServerProxy(URL, transport=t)
Ezio Melottied3a7d22010-12-01 02:32:32 +0000847 cm = self.assertRaisesRegex(xmlrpclib.ProtocolError,
848 re.compile(r"\b400\b"))
Kristján Valur Jónsson985fc6a2009-07-01 10:01:31 +0000849 with cm:
850 p.pow(6, 8)
Victor Stinnera935e8f2011-01-03 14:30:44 +0000851 p("close")()
Kristján Valur Jónsson985fc6a2009-07-01 10:01:31 +0000852
853 def test_gsip_response(self):
854 t = self.Transport()
855 p = xmlrpclib.ServerProxy(URL, transport=t)
856 old = self.requestHandler.encode_threshold
857 self.requestHandler.encode_threshold = None #no encoding
858 self.assertEqual(p.pow(6,8), 6**8)
859 a = t.response_length
860 self.requestHandler.encode_threshold = 0 #always encode
861 self.assertEqual(p.pow(6,8), 6**8)
Victor Stinnera935e8f2011-01-03 14:30:44 +0000862 p("close")()
Kristján Valur Jónsson985fc6a2009-07-01 10:01:31 +0000863 b = t.response_length
864 self.requestHandler.encode_threshold = old
865 self.assertTrue(a>b)
866
867#Test special attributes of the ServerProxy object
868class ServerProxyTestCase(unittest.TestCase):
Victor Stinner45df8202010-04-28 22:31:17 +0000869 def setUp(self):
870 unittest.TestCase.setUp(self)
871 if threading:
872 self.url = URL
873 else:
874 # Without threading, http_server() and http_multi_server() will not
875 # be executed and URL is still equal to None. 'http://' is a just
876 # enough to choose the scheme (HTTP)
877 self.url = 'http://'
878
Kristján Valur Jónsson985fc6a2009-07-01 10:01:31 +0000879 def test_close(self):
Victor Stinner45df8202010-04-28 22:31:17 +0000880 p = xmlrpclib.ServerProxy(self.url)
Kristján Valur Jónsson985fc6a2009-07-01 10:01:31 +0000881 self.assertEqual(p('close')(), None)
882
883 def test_transport(self):
884 t = xmlrpclib.Transport()
Victor Stinner45df8202010-04-28 22:31:17 +0000885 p = xmlrpclib.ServerProxy(self.url, transport=t)
Kristján Valur Jónsson985fc6a2009-07-01 10:01:31 +0000886 self.assertEqual(p('transport'), t)
887
Guido van Rossum61e21b52007-08-20 19:06:03 +0000888# This is a contrived way to make a failure occur on the server side
889# in order to test the _send_traceback_header flag on the server
Barry Warsaw820c1202008-06-12 04:06:45 +0000890class FailingMessageClass(http.client.HTTPMessage):
891 def get(self, key, failobj=None):
Guido van Rossum61e21b52007-08-20 19:06:03 +0000892 key = key.lower()
893 if key == 'content-length':
894 return 'I am broken'
Barry Warsaw820c1202008-06-12 04:06:45 +0000895 return super().get(key, failobj)
Guido van Rossum61e21b52007-08-20 19:06:03 +0000896
897
Victor Stinner45df8202010-04-28 22:31:17 +0000898@unittest.skipUnless(threading, 'Threading required for this test.')
Guido van Rossum61e21b52007-08-20 19:06:03 +0000899class FailingServerTestCase(unittest.TestCase):
900 def setUp(self):
901 self.evt = threading.Event()
902 # start server thread to handle requests
Guido van Rossum15c97462007-11-02 16:10:06 +0000903 serv_args = (self.evt, 1)
Guido van Rossum61e21b52007-08-20 19:06:03 +0000904 threading.Thread(target=http_server, args=serv_args).start()
905
Christian Heimesaf98da12008-01-27 15:18:18 +0000906 # wait for the server to be ready
907 self.evt.wait()
908 self.evt.clear()
Guido van Rossum61e21b52007-08-20 19:06:03 +0000909
910 def tearDown(self):
911 # wait on the server thread to terminate
912 self.evt.wait()
913 # reset flag
Georg Brandl38eceaa2008-05-26 11:14:17 +0000914 xmlrpc.server.SimpleXMLRPCServer._send_traceback_header = False
Guido van Rossum61e21b52007-08-20 19:06:03 +0000915 # reset message class
Barry Warsaw820c1202008-06-12 04:06:45 +0000916 default_class = http.client.HTTPMessage
917 xmlrpc.server.SimpleXMLRPCRequestHandler.MessageClass = default_class
Guido van Rossum61e21b52007-08-20 19:06:03 +0000918
919 def test_basic(self):
920 # check that flag is false by default
Georg Brandl38eceaa2008-05-26 11:14:17 +0000921 flagval = xmlrpc.server.SimpleXMLRPCServer._send_traceback_header
Guido van Rossum61e21b52007-08-20 19:06:03 +0000922 self.assertEqual(flagval, False)
923
Guido van Rossum04110fb2007-08-24 16:32:05 +0000924 # enable traceback reporting
Georg Brandl38eceaa2008-05-26 11:14:17 +0000925 xmlrpc.server.SimpleXMLRPCServer._send_traceback_header = True
Guido van Rossum04110fb2007-08-24 16:32:05 +0000926
927 # test a call that shouldn't fail just as a smoke test
928 try:
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000929 p = xmlrpclib.ServerProxy(URL)
Guido van Rossum04110fb2007-08-24 16:32:05 +0000930 self.assertEqual(p.pow(6,8), 6**8)
Andrew Svetlov0832af62012-12-18 23:10:48 +0200931 except (xmlrpclib.ProtocolError, OSError) as e:
Christian Heimes412dc9c2008-01-27 18:55:54 +0000932 # ignore failures due to non-blocking socket 'unavailable' errors
933 if not is_unavailable_exception(e):
934 # protocol error; provide additional information in test output
Christian Heimes26855632008-01-27 23:50:43 +0000935 self.fail("%s\n%s" % (e, getattr(e, "headers", "")))
Guido van Rossum61e21b52007-08-20 19:06:03 +0000936
937 def test_fail_no_info(self):
938 # use the broken message class
Georg Brandl38eceaa2008-05-26 11:14:17 +0000939 xmlrpc.server.SimpleXMLRPCRequestHandler.MessageClass = FailingMessageClass
Guido van Rossum61e21b52007-08-20 19:06:03 +0000940
941 try:
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000942 p = xmlrpclib.ServerProxy(URL)
Guido van Rossum61e21b52007-08-20 19:06:03 +0000943 p.pow(6,8)
Andrew Svetlov0832af62012-12-18 23:10:48 +0200944 except (xmlrpclib.ProtocolError, OSError) as e:
Christian Heimes412dc9c2008-01-27 18:55:54 +0000945 # ignore failures due to non-blocking socket 'unavailable' errors
Christian Heimes26855632008-01-27 23:50:43 +0000946 if not is_unavailable_exception(e) and hasattr(e, "headers"):
Christian Heimes412dc9c2008-01-27 18:55:54 +0000947 # The two server-side error headers shouldn't be sent back in this case
948 self.assertTrue(e.headers.get("X-exception") is None)
949 self.assertTrue(e.headers.get("X-traceback") is None)
Guido van Rossum61e21b52007-08-20 19:06:03 +0000950 else:
951 self.fail('ProtocolError not raised')
952
953 def test_fail_with_info(self):
954 # use the broken message class
Georg Brandl38eceaa2008-05-26 11:14:17 +0000955 xmlrpc.server.SimpleXMLRPCRequestHandler.MessageClass = FailingMessageClass
Guido van Rossum61e21b52007-08-20 19:06:03 +0000956
957 # Check that errors in the server send back exception/traceback
958 # info when flag is set
Georg Brandl38eceaa2008-05-26 11:14:17 +0000959 xmlrpc.server.SimpleXMLRPCServer._send_traceback_header = True
Guido van Rossum61e21b52007-08-20 19:06:03 +0000960
961 try:
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000962 p = xmlrpclib.ServerProxy(URL)
Guido van Rossum61e21b52007-08-20 19:06:03 +0000963 p.pow(6,8)
Andrew Svetlov0832af62012-12-18 23:10:48 +0200964 except (xmlrpclib.ProtocolError, OSError) as e:
Christian Heimes412dc9c2008-01-27 18:55:54 +0000965 # ignore failures due to non-blocking socket 'unavailable' errors
Christian Heimes26855632008-01-27 23:50:43 +0000966 if not is_unavailable_exception(e) and hasattr(e, "headers"):
Christian Heimes412dc9c2008-01-27 18:55:54 +0000967 # We should get error info in the response
968 expected_err = "invalid literal for int() with base 10: 'I am broken'"
Barry Warsaw820c1202008-06-12 04:06:45 +0000969 self.assertEqual(e.headers.get("X-exception"), expected_err)
970 self.assertTrue(e.headers.get("X-traceback") is not None)
Guido van Rossum61e21b52007-08-20 19:06:03 +0000971 else:
972 self.fail('ProtocolError not raised')
973
R. David Murray0548ce02009-10-26 08:24:14 +0000974
975@contextlib.contextmanager
976def captured_stdout(encoding='utf-8'):
977 """A variation on support.captured_stdout() which gives a text stream
978 having a `buffer` attribute.
979 """
980 import io
981 orig_stdout = sys.stdout
982 sys.stdout = io.TextIOWrapper(io.BytesIO(), encoding=encoding)
983 try:
984 yield sys.stdout
985 finally:
986 sys.stdout = orig_stdout
987
988
Christian Heimes255f53b2007-12-08 15:33:56 +0000989class CGIHandlerTestCase(unittest.TestCase):
990 def setUp(self):
Georg Brandl38eceaa2008-05-26 11:14:17 +0000991 self.cgi = xmlrpc.server.CGIXMLRPCRequestHandler()
Christian Heimes255f53b2007-12-08 15:33:56 +0000992
993 def tearDown(self):
994 self.cgi = None
995
996 def test_cgi_get(self):
Walter Dörwaldb525e182009-04-26 21:39:21 +0000997 with support.EnvironmentVarGuard() as env:
Walter Dörwald155374d2009-05-01 19:58:58 +0000998 env['REQUEST_METHOD'] = 'GET'
Walter Dörwaldb525e182009-04-26 21:39:21 +0000999 # if the method is GET and no request_text is given, it runs handle_get
1000 # get sysout output
R. David Murray0548ce02009-10-26 08:24:14 +00001001 with captured_stdout(encoding=self.cgi.encoding) as data_out:
Nick Coghlan6ead5522009-10-18 13:19:33 +00001002 self.cgi.handle_request()
Christian Heimes255f53b2007-12-08 15:33:56 +00001003
Walter Dörwaldb525e182009-04-26 21:39:21 +00001004 # parse Status header
Nick Coghlan6ead5522009-10-18 13:19:33 +00001005 data_out.seek(0)
1006 handle = data_out.read()
Walter Dörwaldb525e182009-04-26 21:39:21 +00001007 status = handle.split()[1]
1008 message = ' '.join(handle.split()[2:4])
Christian Heimes255f53b2007-12-08 15:33:56 +00001009
Walter Dörwaldb525e182009-04-26 21:39:21 +00001010 self.assertEqual(status, '400')
1011 self.assertEqual(message, 'Bad Request')
Christian Heimes255f53b2007-12-08 15:33:56 +00001012
Christian Heimes255f53b2007-12-08 15:33:56 +00001013
1014 def test_cgi_xmlrpc_response(self):
1015 data = """<?xml version='1.0'?>
Nick Coghlan6ead5522009-10-18 13:19:33 +00001016 <methodCall>
1017 <methodName>test_method</methodName>
1018 <params>
1019 <param>
1020 <value><string>foo</string></value>
1021 </param>
1022 <param>
1023 <value><string>bar</string></value>
1024 </param>
1025 </params>
1026 </methodCall>
1027 """
Christian Heimes255f53b2007-12-08 15:33:56 +00001028
Nick Coghlan6ead5522009-10-18 13:19:33 +00001029 with support.EnvironmentVarGuard() as env, \
R. David Murray0548ce02009-10-26 08:24:14 +00001030 captured_stdout(encoding=self.cgi.encoding) as data_out, \
Nick Coghlan6ead5522009-10-18 13:19:33 +00001031 support.captured_stdin() as data_in:
1032 data_in.write(data)
1033 data_in.seek(0)
Walter Dörwald155374d2009-05-01 19:58:58 +00001034 env['CONTENT_LENGTH'] = str(len(data))
Georg Brandlc7485062009-04-01 15:53:15 +00001035 self.cgi.handle_request()
Nick Coghlan6ead5522009-10-18 13:19:33 +00001036 data_out.seek(0)
Christian Heimes255f53b2007-12-08 15:33:56 +00001037
1038 # will respond exception, if so, our goal is achieved ;)
Nick Coghlan6ead5522009-10-18 13:19:33 +00001039 handle = data_out.read()
Christian Heimes255f53b2007-12-08 15:33:56 +00001040
Jeremy Hylton1afc1692008-06-18 20:49:58 +00001041 # start with 44th char so as not to get http header, we just
1042 # need only xml
Christian Heimes255f53b2007-12-08 15:33:56 +00001043 self.assertRaises(xmlrpclib.Fault, xmlrpclib.loads, handle[44:])
1044
Senthil Kumaranb3af08f2009-04-01 20:20:43 +00001045 # Also test the content-length returned by handle_request
1046 # Using the same test method inorder to avoid all the datapassing
1047 # boilerplate code.
1048 # Test for bug: http://bugs.python.org/issue5040
1049
1050 content = handle[handle.find("<?xml"):]
1051
Ezio Melottib3aedd42010-11-20 19:04:17 +00001052 self.assertEqual(
Senthil Kumaranb3af08f2009-04-01 20:20:43 +00001053 int(re.search('Content-Length: (\d+)', handle).group(1)),
1054 len(content))
1055
Senthil Kumaranb3af08f2009-04-01 20:20:43 +00001056
Florent Xicluna1b7458b2011-12-09 22:35:06 +01001057class UseBuiltinTypesTestCase(unittest.TestCase):
1058
1059 def test_use_builtin_types(self):
1060 # SimpleXMLRPCDispatcher.__init__ accepts use_builtin_types, which
1061 # makes all dispatch of binary data as bytes instances, and all
1062 # dispatch of datetime argument as datetime.datetime instances.
1063 self.log = []
1064 expected_bytes = b"my dog has fleas"
1065 expected_date = datetime.datetime(2008, 5, 26, 18, 25, 12)
1066 marshaled = xmlrpclib.dumps((expected_bytes, expected_date), 'foobar')
1067 def foobar(*args):
1068 self.log.extend(args)
1069 handler = xmlrpc.server.SimpleXMLRPCDispatcher(
1070 allow_none=True, encoding=None, use_builtin_types=True)
1071 handler.register_function(foobar)
1072 handler._marshaled_dispatch(marshaled)
1073 self.assertEqual(len(self.log), 2)
1074 mybytes, mydate = self.log
1075 self.assertEqual(self.log, [expected_bytes, expected_date])
1076 self.assertIs(type(mydate), datetime.datetime)
1077 self.assertIs(type(mybytes), bytes)
1078
1079 def test_cgihandler_has_use_builtin_types_flag(self):
1080 handler = xmlrpc.server.CGIXMLRPCRequestHandler(use_builtin_types=True)
1081 self.assertTrue(handler.use_builtin_types)
1082
1083 def test_xmlrpcserver_has_use_builtin_types_flag(self):
1084 server = xmlrpc.server.SimpleXMLRPCServer(("localhost", 0),
1085 use_builtin_types=True)
1086 server.server_close()
1087 self.assertTrue(server.use_builtin_types)
1088
1089
Brett Cannone382b582013-06-12 21:25:23 -04001090@support.reap_threads
1091def test_main():
1092 support.run_unittest(XMLRPCTestCase, HelperTestCase, DateTimeTestCase,
1093 BinaryTestCase, FaultTestCase, UseBuiltinTypesTestCase,
1094 SimpleServerTestCase, KeepaliveServerTestCase1,
1095 KeepaliveServerTestCase2, GzipServerTestCase,
1096 MultiPathServerTestCase, ServerProxyTestCase, FailingServerTestCase,
1097 CGIHandlerTestCase)
1098
1099
Skip Montanaro419abda2001-10-01 17:47:44 +00001100if __name__ == "__main__":
Brett Cannone382b582013-06-12 21:25:23 -04001101 test_main()