SimpleXMLRPCServer.py 25.1 KB
Newer Older
1 2 3 4
"""Simple XML-RPC Server.

This module can be used to create simple XML-RPC servers
by creating a server and either installing functions, a
5
class instance, or by extending the SimpleXMLRPCServer
6 7
class.

8 9 10
It can also be used to handle XML-RPC requests in a CGI
environment using CGIXMLRPCRequestHandler.

11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27
A list of possible usage patterns follows:

1. Install functions:

server = SimpleXMLRPCServer(("localhost", 8000))
server.register_function(pow)
server.register_function(lambda x,y: x+y, 'add')
server.serve_forever()

2. Install an instance:

class MyFuncs:
    def __init__(self):
        # make all of the string functions available through
        # string.func_name
        import string
        self.string = string
28 29 30 31 32
    def _listMethods(self):
        # implement this method so that system.listMethods
        # knows to advertise the strings methods
        return list_public_methods(self) + \
                ['string.' + method for method in list_public_methods(self.string)]
33 34
    def pow(self, x, y): return pow(x, y)
    def add(self, x, y) : return x + y
Tim Peters's avatar
Tim Peters committed
35

36
server = SimpleXMLRPCServer(("localhost", 8000))
37
server.register_introspection_functions()
38 39 40 41 42 43
server.register_instance(MyFuncs())
server.serve_forever()

3. Install an instance with custom dispatch method:

class Math:
44 45 46 47 48 49 50 51 52 53 54 55 56 57 58
    def _listMethods(self):
        # this method must be present for system.listMethods
        # to work
        return ['add', 'pow']
    def _methodHelp(self, method):
        # this method must be present for system.methodHelp
        # to work
        if method == 'add':
            return "add(2,3) => 5"
        elif method == 'pow':
            return "pow(x, y[, z]) => number"
        else:
            # By convention, return empty
            # string if no help is available
            return ""
59 60
    def _dispatch(self, method, params):
        if method == 'pow':
61
            return pow(*params)
62 63 64 65
        elif method == 'add':
            return params[0] + params[1]
        else:
            raise 'bad method'
66

67
server = SimpleXMLRPCServer(("localhost", 8000))
68
server.register_introspection_functions()
69 70 71
server.register_instance(Math())
server.serve_forever()

72
4. Subclass SimpleXMLRPCServer:
73

74
class MathServer(SimpleXMLRPCServer):
75 76 77 78 79 80 81 82 83
    def _dispatch(self, method, params):
        try:
            # We are forcing the 'export_' prefix on methods that are
            # callable through XML-RPC to prevent potential security
            # problems
            func = getattr(self, 'export_' + method)
        except AttributeError:
            raise Exception('method "%s" is not supported' % method)
        else:
84
            return func(*params)
85 86 87 88

    def export_add(self, x, y):
        return x + y

89
server = MathServer(("localhost", 8000))
90
server.serve_forever()
91 92 93 94 95 96

5. CGI script:

server = CGIXMLRPCRequestHandler()
server.register_function(pow)
server.handle_request()
97 98 99 100 101 102
"""

# Written by Brian Quinlan (brian@sweetapp.com).
# Based on code written by Fredrik Lundh.

import xmlrpclib
103
from xmlrpclib import Fault
104
import SocketServer
105 106
import BaseHTTPServer
import sys
107
import os
108
import traceback
109
import re
110 111 112 113
try:
    import fcntl
except ImportError:
    fcntl = None
114

115
def resolve_dotted_attribute(obj, attr, allow_dotted_names=True):
116
    """resolve_dotted_attribute(a, 'b.c.d') => a.b.c.d
117

118 119
    Resolves a dotted attribute name to an object.  Raises
    an AttributeError if any attribute in the chain starts with a '_'.
120 121 122

    If the optional allow_dotted_names argument is false, dots are not
    supported and this function operates similar to getattr(obj, attr).
123 124
    """

125 126 127 128 129 130
    if allow_dotted_names:
        attrs = attr.split('.')
    else:
        attrs = [attr]

    for i in attrs:
131 132 133 134 135 136 137
        if i.startswith('_'):
            raise AttributeError(
                'attempt to access private attribute "%s"' % i
                )
        else:
            obj = getattr(obj,i)
    return obj
138

139 140 141 142 143 144
def list_public_methods(obj):
    """Returns a list of attribute strings, found in the specified
    object, which represent callable attributes"""

    return [member for member in dir(obj)
                if not member.startswith('_') and
145
                    hasattr(getattr(obj, member), '__call__')]
146 147 148 149 150 151 152

def remove_duplicates(lst):
    """remove_duplicates([2,2,2,1,3,3]) => [3,1,2]

    Returns a copy of a list without duplicates. Every list
    item must be hashable and the order of the items in the
    resulting list is not defined.
Tim Peters's avatar
Tim Peters committed
153
    """
154 155 156 157 158 159 160 161 162 163
    u = {}
    for x in lst:
        u[x] = 1

    return u.keys()

class SimpleXMLRPCDispatcher:
    """Mix-in class that dispatches XML-RPC requests.

    This class is used to register XML-RPC method handlers
Kristján Valur Jónsson's avatar
Kristján Valur Jónsson committed
164 165
    and then to dispatch them. This class doesn't need to be
    instanced directly when used by SimpleXMLRPCServer but it
Georg Brandl's avatar
Georg Brandl committed
166
    can be instanced when used by the MultiPathXMLRPCServer.
167
    """
Tim Peters's avatar
Tim Peters committed
168

169
    def __init__(self, allow_none=False, encoding=None):
170 171
        self.funcs = {}
        self.instance = None
172
        self.allow_none = allow_none
173
        self.encoding = encoding
174

175
    def register_instance(self, instance, allow_dotted_names=False):
176
        """Registers an instance to respond to XML-RPC requests.
177

178 179 180 181
        Only one instance can be installed at a time.

        If the registered instance has a _dispatch method then that
        method will be called with the name of the XML-RPC method and
182
        its parameters as a tuple
183 184 185 186 187 188 189 190 191 192
        e.g. instance._dispatch('add',(2,3))

        If the registered instance does not have a _dispatch method
        then the instance will be searched to find a matching method
        and, if found, will be called. Methods beginning with an '_'
        are considered private and will not be called by
        SimpleXMLRPCServer.

        If a registered function matches a XML-RPC request, then it
        will be called instead of the registered instance.
193 194 195 196 197 198 199 200 201 202 203 204 205

        If the optional allow_dotted_names argument is true and the
        instance does not have a _dispatch method, method names
        containing dots are supported and resolved, as long as none of
        the name segments start with an '_'.

            *** SECURITY WARNING: ***

            Enabling the allow_dotted_names options allows intruders
            to access your module's global variables and may allow
            intruders to execute arbitrary code on your machine.  Only
            use this option on a secure, closed network.

206 207
        """

208
        self.instance = instance
209
        self.allow_dotted_names = allow_dotted_names
210

211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227
    def register_function(self, function, name = None):
        """Registers a function to respond to XML-RPC requests.

        The optional name argument can be used to set a Unicode name
        for the function.
        """

        if name is None:
            name = function.__name__
        self.funcs[name] = function

    def register_introspection_functions(self):
        """Registers the XML-RPC introspection methods in the system
        namespace.

        see http://xmlrpc.usefulinc.com/doc/reserved.html
        """
Tim Peters's avatar
Tim Peters committed
228

229 230 231 232 233 234 235 236 237
        self.funcs.update({'system.listMethods' : self.system_listMethods,
                      'system.methodSignature' : self.system_methodSignature,
                      'system.methodHelp' : self.system_methodHelp})

    def register_multicall_functions(self):
        """Registers the XML-RPC multicall method in the system
        namespace.

        see http://www.xmlrpc.com/discuss/msgReader$1208"""
Tim Peters's avatar
Tim Peters committed
238

239
        self.funcs.update({'system.multicall' : self.system_multicall})
Tim Peters's avatar
Tim Peters committed
240

Kristján Valur Jónsson's avatar
Kristján Valur Jónsson committed
241
    def _marshaled_dispatch(self, data, dispatch_method = None, path = None):
242
        """Dispatches an XML-RPC method from marshalled (XML) data.
Tim Peters's avatar
Tim Peters committed
243

244 245 246
        XML-RPC methods are dispatched from the marshalled (XML) data
        using the _dispatch method and the result is returned as
        marshalled data. For backwards compatibility, a dispatch
Tim Peters's avatar
Tim Peters committed
247
        function can be provided as an argument (see comment in
248 249 250 251
        SimpleXMLRPCRequestHandler.do_POST) but overriding the
        existing method through subclassing is the prefered means
        of changing method dispatch behavior.
        """
Tim Peters's avatar
Tim Peters committed
252

253
        try:
254 255 256
            params, method = xmlrpclib.loads(data)

            # generate response
257 258
            if dispatch_method is not None:
                response = dispatch_method(method, params)
Tim Peters's avatar
Tim Peters committed
259
            else:
260
                response = self._dispatch(method, params)
261 262
            # wrap response in a singleton tuple
            response = (response,)
Tim Peters's avatar
Tim Peters committed
263
            response = xmlrpclib.dumps(response, methodresponse=1,
264
                                       allow_none=self.allow_none, encoding=self.encoding)
265
        except Fault, fault:
Tim Peters's avatar
Tim Peters committed
266
            response = xmlrpclib.dumps(fault, allow_none=self.allow_none,
267
                                       encoding=self.encoding)
268
        except:
269
            # report exception back to server
270
            exc_type, exc_value, exc_tb = sys.exc_info()
271
            response = xmlrpclib.dumps(
272
                xmlrpclib.Fault(1, "%s:%s" % (exc_type, exc_value)),
273
                encoding=self.encoding, allow_none=self.allow_none,
274 275 276 277 278 279 280 281
                )

        return response

    def system_listMethods(self):
        """system.listMethods() => ['add', 'subtract', 'multiple']

        Returns a list of the methods supported by the server."""
Tim Peters's avatar
Tim Peters committed
282

283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299
        methods = self.funcs.keys()
        if self.instance is not None:
            # Instance can implement _listMethod to return a list of
            # methods
            if hasattr(self.instance, '_listMethods'):
                methods = remove_duplicates(
                        methods + self.instance._listMethods()
                    )
            # if the instance has a _dispatch method then we
            # don't have enough information to provide a list
            # of methods
            elif not hasattr(self.instance, '_dispatch'):
                methods = remove_duplicates(
                        methods + list_public_methods(self.instance)
                    )
        methods.sort()
        return methods
Tim Peters's avatar
Tim Peters committed
300

301 302 303
    def system_methodSignature(self, method_name):
        """system.methodSignature('add') => [double, int, int]

304
        Returns a list describing the signature of the method. In the
305 306 307 308 309 310
        above example, the add method takes two integers as arguments
        and returns a double result.

        This server does NOT support system.methodSignature."""

        # See http://xmlrpc.usefulinc.com/doc/sysmethodsig.html
Tim Peters's avatar
Tim Peters committed
311

312 313 314 315 316 317
        return 'signatures not supported'

    def system_methodHelp(self, method_name):
        """system.methodHelp('add') => "Adds two integers together"

        Returns a string containing documentation for the specified method."""
Tim Peters's avatar
Tim Peters committed
318

319
        method = None
320
        if method_name in self.funcs:
321 322 323 324 325 326 327 328 329 330 331
            method = self.funcs[method_name]
        elif self.instance is not None:
            # Instance can implement _methodHelp to return help for a method
            if hasattr(self.instance, '_methodHelp'):
                return self.instance._methodHelp(method_name)
            # if the instance has a _dispatch method then we
            # don't have enough information to provide help
            elif not hasattr(self.instance, '_dispatch'):
                try:
                    method = resolve_dotted_attribute(
                                self.instance,
332 333
                                method_name,
                                self.allow_dotted_names
334 335 336 337 338 339 340 341
                                )
                except AttributeError:
                    pass

        # Note that we aren't checking that the method actually
        # be a callable object of some kind
        if method is None:
            return ""
342
        else:
343
            import pydoc
Neal Norwitz's avatar
Neal Norwitz committed
344
            return pydoc.getdoc(method)
345

346 347 348 349 350 351
    def system_multicall(self, call_list):
        """system.multicall([{'methodName': 'add', 'params': [2, 2]}, ...]) => \
[[4], ...]

        Allows the caller to package multiple XML-RPC calls into a single
        request.
352

Tim Peters's avatar
Tim Peters committed
353
        See http://www.xmlrpc.com/discuss/msgReader$1208
354
        """
Tim Peters's avatar
Tim Peters committed
355

356 357 358 359 360 361 362 363 364 365 366 367 368 369 370
        results = []
        for call in call_list:
            method_name = call['methodName']
            params = call['params']

            try:
                # XXX A marshalling error in any response will fail the entire
                # multicall. If someone cares they should fix this.
                results.append([self._dispatch(method_name, params)])
            except Fault, fault:
                results.append(
                    {'faultCode' : fault.faultCode,
                     'faultString' : fault.faultString}
                    )
            except:
371
                exc_type, exc_value, exc_tb = sys.exc_info()
372 373
                results.append(
                    {'faultCode' : 1,
374
                     'faultString' : "%s:%s" % (exc_type, exc_value)}
375 376
                    )
        return results
Tim Peters's avatar
Tim Peters committed
377

378 379 380 381 382 383 384 385 386 387
    def _dispatch(self, method, params):
        """Dispatches the XML-RPC method.

        XML-RPC calls are forwarded to a registered function that
        matches the called XML-RPC method name. If no such function
        exists then the call is forwarded to the registered instance,
        if available.

        If the registered instance has a _dispatch method then that
        method will be called with the name of the XML-RPC method and
388
        its parameters as a tuple
389 390 391 392 393 394 395
        e.g. instance._dispatch('add',(2,3))

        If the registered instance does not have a _dispatch method
        then the instance will be searched to find a matching method
        and, if found, will be called.

        Methods beginning with an '_' are considered private and will
396
        not be called.
397 398 399 400 401
        """

        func = None
        try:
            # check to see if a matching function has been registered
402
            func = self.funcs[method]
403
        except KeyError:
404
            if self.instance is not None:
405
                # check for a _dispatch method
406 407
                if hasattr(self.instance, '_dispatch'):
                    return self.instance._dispatch(method, params)
408 409 410
                else:
                    # call instance method directly
                    try:
411 412
                        func = resolve_dotted_attribute(
                            self.instance,
413 414
                            method,
                            self.allow_dotted_names
415 416 417 418 419
                            )
                    except AttributeError:
                        pass

        if func is not None:
420
            return func(*params)
421 422
        else:
            raise Exception('method "%s" is not supported' % method)
Tim Peters's avatar
Tim Peters committed
423

424 425
class SimpleXMLRPCRequestHandler(BaseHTTPServer.BaseHTTPRequestHandler):
    """Simple XML-RPC request handler class.
426

427 428 429
    Handles all HTTP POST requests and attempts to decode them as
    XML-RPC requests.
    """
430

Andrew M. Kuchling's avatar
Andrew M. Kuchling committed
431 432 433 434
    # Class attribute listing the accessible path components;
    # paths not on this list will result in a 404 error.
    rpc_paths = ('/', '/RPC2')

435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459
    #if not None, encode responses larger than this, if possible
    encode_threshold = 1400 #a common MTU

    #Override form StreamRequestHandler: full buffering of output
    #and no Nagle.
    wbufsize = -1
    disable_nagle_algorithm = True

    # a re to match a gzip Accept-Encoding
    aepattern = re.compile(r"""
                            \s* ([^\s;]+) \s*            #content-coding
                            (;\s* q \s*=\s* ([0-9\.]+))? #q
                            """, re.VERBOSE | re.IGNORECASE)

    def accept_encodings(self):
        r = {}
        ae = self.headers.get("Accept-Encoding", "")
        for e in ae.split(","):
            match = self.aepattern.match(e)
            if match:
                v = match.group(3)
                v = float(v) if v else 1.0
                r[match.group(1)] = v
        return r

Andrew M. Kuchling's avatar
Andrew M. Kuchling committed
460 461 462 463 464 465 466
    def is_rpc_path_valid(self):
        if self.rpc_paths:
            return self.path in self.rpc_paths
        else:
            # If .rpc_paths is empty, just assume all paths are legal
            return True

467 468
    def do_POST(self):
        """Handles the HTTP POST request.
469

470 471 472
        Attempts to interpret all HTTP POST requests as XML-RPC calls,
        which are forwarded to the server's _dispatch method for handling.
        """
Tim Peters's avatar
Tim Peters committed
473

Andrew M. Kuchling's avatar
Andrew M. Kuchling committed
474 475 476 477
        # Check that the path is legal
        if not self.is_rpc_path_valid():
            self.report_404()
            return
Tim Peters's avatar
Tim Peters committed
478

479
        try:
Tim Peters's avatar
Tim Peters committed
480 481
            # Get arguments by reading body of request.
            # We read this in chunks to avoid straining
482 483 484 485 486 487 488 489 490 491 492
            # socket.read(); around the 10 or 15Mb mark, some platforms
            # begin to have problems (bug #792570).
            max_chunk_size = 10*1024*1024
            size_remaining = int(self.headers["content-length"])
            L = []
            while size_remaining:
                chunk_size = min(size_remaining, max_chunk_size)
                L.append(self.rfile.read(chunk_size))
                size_remaining -= len(L[-1])
            data = ''.join(L)

493 494 495 496
            data = self.decode_request_content(data)
            if data is None:
                return #response has been sent

497 498 499 500 501 502
            # In previous versions of SimpleXMLRPCServer, _dispatch
            # could be overridden in this class, instead of in
            # SimpleXMLRPCDispatcher. To maintain backwards compatibility,
            # check to see if a subclass implements _dispatch and dispatch
            # using that method if present.
            response = self.server._marshaled_dispatch(
Kristján Valur Jónsson's avatar
Kristján Valur Jónsson committed
503
                    data, getattr(self, '_dispatch', None), self.path
504
                )
505
        except Exception, e: # This should only happen if the module is buggy
506 507
            # internal error, report as HTTP server error
            self.send_response(500)
508 509 510 511 512 513 514

            # Send information about the exception if requested
            if hasattr(self.server, '_send_traceback_header') and \
                    self.server._send_traceback_header:
                self.send_header("X-exception", str(e))
                self.send_header("X-traceback", traceback.format_exc())

515
            self.send_header("Content-length", "0")
516
            self.end_headers()
517
        else:
518 519 520
            # got a valid XML RPC response
            self.send_response(200)
            self.send_header("Content-type", "text/xml")
521 522 523 524
            if self.encode_threshold is not None:
                if len(response) > self.encode_threshold:
                    q = self.accept_encodings().get("gzip", 0)
                    if q:
525 526 527 528 529
                        try:
                            response = xmlrpclib.gzip_encode(response)
                            self.send_header("Content-Encoding", "gzip")
                        except NotImplementedError:
                            pass
530 531 532
            self.send_header("Content-length", str(len(response)))
            self.end_headers()
            self.wfile.write(response)
533

534 535 536 537 538 539 540 541
    def decode_request_content(self, data):
        #support gzip encoding of request
        encoding = self.headers.get("content-encoding", "identity").lower()
        if encoding == "identity":
            return data
        if encoding == "gzip":
            try:
                return xmlrpclib.gzip_decode(data)
542 543
            except NotImplementedError:
                self.send_response(501, "encoding %r not supported" % encoding)
544 545 546 547 548 549
            except ValueError:
                self.send_response(400, "error decoding gzip content")
        else:
            self.send_response(501, "encoding %r not supported" % encoding)
        self.send_header("Content-length", "0")
        self.end_headers()
Tim Peters's avatar
Tim Peters committed
550

Andrew M. Kuchling's avatar
Andrew M. Kuchling committed
551 552
    def report_404 (self):
            # Report a 404 error
Tim Peters's avatar
Tim Peters committed
553 554 555 556 557 558
        self.send_response(404)
        response = 'No such page'
        self.send_header("Content-type", "text/plain")
        self.send_header("Content-length", str(len(response)))
        self.end_headers()
        self.wfile.write(response)
Andrew M. Kuchling's avatar
Andrew M. Kuchling committed
559

560 561
    def log_request(self, code='-', size='-'):
        """Selectively log an accepted request."""
562

563 564 565
        if self.server.logRequests:
            BaseHTTPServer.BaseHTTPRequestHandler.log_request(self, code, size)

566
class SimpleXMLRPCServer(SocketServer.TCPServer,
567
                         SimpleXMLRPCDispatcher):
568 569 570
    """Simple XML-RPC server.

    Simple XML-RPC server that allows functions and a single instance
571 572 573 574
    to be installed to handle requests. The default implementation
    attempts to dispatch XML-RPC calls to the functions or instance
    installed in the server. Override the _dispatch method inhereted
    from SimpleXMLRPCDispatcher to change this behavior.
575 576
    """

577 578
    allow_reuse_address = True

579 580 581 582 583 584
    # Warning: this is for debugging purposes only! Never set this to True in
    # production code, as will be sending out sensitive information (exception
    # and stack trace details) when exceptions are raised inside
    # SimpleXMLRPCRequestHandler.do_POST
    _send_traceback_header = False

585
    def __init__(self, addr, requestHandler=SimpleXMLRPCRequestHandler,
586
                 logRequests=True, allow_none=False, encoding=None, bind_and_activate=True):
587
        self.logRequests = logRequests
Tim Peters's avatar
Tim Peters committed
588

589
        SimpleXMLRPCDispatcher.__init__(self, allow_none, encoding)
590
        SocketServer.TCPServer.__init__(self, addr, requestHandler, bind_and_activate)
Tim Peters's avatar
Tim Peters committed
591

Tim Peters's avatar
Tim Peters committed
592 593
        # [Bug #1222790] If possible, set close-on-exec flag; if a
        # method spawns a subprocess, the subprocess shouldn't have
594
        # the listening socket open.
595
        if fcntl is not None and hasattr(fcntl, 'FD_CLOEXEC'):
596 597 598 599
            flags = fcntl.fcntl(self.fileno(), fcntl.F_GETFD)
            flags |= fcntl.FD_CLOEXEC
            fcntl.fcntl(self.fileno(), fcntl.F_SETFD, flags)

Kristján Valur Jónsson's avatar
Kristján Valur Jónsson committed
600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637
class MultiPathXMLRPCServer(SimpleXMLRPCServer):
    """Multipath XML-RPC Server
    This specialization of SimpleXMLRPCServer allows the user to create
    multiple Dispatcher instances and assign them to different
    HTTP request paths.  This makes it possible to run two or more
    'virtual XML-RPC servers' at the same port.
    Make sure that the requestHandler accepts the paths in question.
    """
    def __init__(self, addr, requestHandler=SimpleXMLRPCRequestHandler,
                 logRequests=True, allow_none=False, encoding=None, bind_and_activate=True):

        SimpleXMLRPCServer.__init__(self, addr, requestHandler, logRequests, allow_none,
                                    encoding, bind_and_activate)
        self.dispatchers = {}
        self.allow_none = allow_none
        self.encoding = encoding

    def add_dispatcher(self, path, dispatcher):
        self.dispatchers[path] = dispatcher
        return dispatcher

    def get_dispatcher(self, path):
        return self.dispatchers[path]

    def _marshaled_dispatch(self, data, dispatch_method = None, path = None):
        try:
            response = self.dispatchers[path]._marshaled_dispatch(
               data, dispatch_method, path)
        except:
            # report low level exception back to server
            # (each dispatcher should have handled their own
            # exceptions)
            exc_type, exc_value = sys.exc_info()[:2]
            response = xmlrpclib.dumps(
                xmlrpclib.Fault(1, "%s:%s" % (exc_type, exc_value)),
                encoding=self.encoding, allow_none=self.allow_none)
        return response

638 639
class CGIXMLRPCRequestHandler(SimpleXMLRPCDispatcher):
    """Simple handler for XML-RPC data passed through CGI."""
Tim Peters's avatar
Tim Peters committed
640

641 642
    def __init__(self, allow_none=False, encoding=None):
        SimpleXMLRPCDispatcher.__init__(self, allow_none, encoding)
643 644 645

    def handle_xmlrpc(self, request_text):
        """Handle a single XML-RPC request"""
Tim Peters's avatar
Tim Peters committed
646

647
        response = self._marshaled_dispatch(request_text)
Tim Peters's avatar
Tim Peters committed
648

649 650 651
        print 'Content-Type: text/xml'
        print 'Content-Length: %d' % len(response)
        print
652
        sys.stdout.write(response)
653 654 655 656 657 658

    def handle_get(self):
        """Handle a single HTTP GET request.

        Default implementation indicates an error because
        XML-RPC uses the POST method.
659 660
        """

661 662 663
        code = 400
        message, explain = \
                 BaseHTTPServer.BaseHTTPRequestHandler.responses[code]
Tim Peters's avatar
Tim Peters committed
664

665 666
        response = BaseHTTPServer.DEFAULT_ERROR_MESSAGE % \
            {
Tim Peters's avatar
Tim Peters committed
667 668
             'code' : code,
             'message' : message,
669 670 671
             'explain' : explain
            }
        print 'Status: %d %s' % (code, message)
672
        print 'Content-Type: %s' % BaseHTTPServer.DEFAULT_ERROR_CONTENT_TYPE
673 674
        print 'Content-Length: %d' % len(response)
        print
675
        sys.stdout.write(response)
Tim Peters's avatar
Tim Peters committed
676

677 678
    def handle_request(self, request_text = None):
        """Handle a single XML-RPC request passed through a CGI post method.
Tim Peters's avatar
Tim Peters committed
679

680 681 682
        If no XML data is given then it is read from stdin. The resulting
        XML-RPC response is printed to stdout along with the correct HTTP
        headers.
683
        """
Tim Peters's avatar
Tim Peters committed
684

685 686 687 688 689
        if request_text is None and \
            os.environ.get('REQUEST_METHOD', None) == 'GET':
            self.handle_get()
        else:
            # POST data is normally available through stdin
690 691
            try:
                length = int(os.environ.get('CONTENT_LENGTH', None))
692
            except (TypeError, ValueError):
693
                length = -1
694
            if request_text is None:
695
                request_text = sys.stdin.read(length)
696

697
            self.handle_xmlrpc(request_text)
Tim Peters's avatar
Tim Peters committed
698

699
if __name__ == '__main__':
700
    print 'Running XML-RPC server on port 8000'
701 702 703 704
    server = SimpleXMLRPCServer(("localhost", 8000))
    server.register_function(pow)
    server.register_function(lambda x,y: x+y, 'add')
    server.serve_forever()